Commit e802d8cc by Vicent Marti

Implement internal methods to write on sources

The new 'git__source_printf' does an overflow-safe printf on a source
bfufer.

The new 'git__source_write' does an overflow-safe byte write on a source
buffer.

Signed-off-by: Vicent Marti <tanoku@gmail.com>
parent f49a2e49
......@@ -22,6 +22,7 @@
* the Free Software Foundation, 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include <stdarg.h>
#include "common.h"
#include "repository.h"
......@@ -91,42 +92,92 @@ void git_repository_free(git_repository *repo)
free(repo);
}
void git_object__source_prepare_write(git_object *object)
static int source_resize(git_odb_source *src)
{
size_t base_size = 512;
size_t write_offset, new_size;
void *new_data;
if (object->source.write_ptr != NULL || object->source.open)
git_object__source_close(object);
write_offset = src->write_ptr - src->raw.data;
/* TODO: proper size calculation */
object->source.raw.data = git__malloc(base_size);
object->source.raw.len = base_size;
new_size = src->raw.len * 2;
if ((new_data = git__malloc(new_size)) == NULL)
return GIT_ENOMEM;
object->source.write_ptr = object->source.raw.data;
object->source.written_bytes = 0;
memcpy(new_data, src->raw.data, src->written_bytes);
free(src->raw.data);
object->source.open = 1;
object->source.out_of_sync = 1;
src->raw.data = new_data;
src->raw.len = new_size;
src->write_ptr = new_data + write_offset;
return GIT_SUCCESS;
}
int git_object__source_write(git_object *object, const void *bytes, size_t len)
int git__source_printf(git_odb_source *source, const char *format, ...)
{
assert(object);
va_list arglist;
int len, did_resize = 0;
if (!object->source.open || object->source.write_ptr == NULL)
if (!source->open || source->write_ptr == NULL)
return GIT_ERROR;
/* TODO: resize buffer on overflow */
if (object->source.written_bytes + len >= object->source.raw.len)
va_start(arglist, format);
len = vsnprintf(source->write_ptr, source->raw.len - source->written_bytes, format, arglist);
while (source->written_bytes + len >= source->raw.len) {
if (source_resize(source) < 0)
return GIT_ENOMEM;
memcpy(object->source.write_ptr, bytes, len);
object->source.write_ptr += len;
object->source.written_bytes += len;
did_resize = 1;
}
if (did_resize)
vsnprintf(source->write_ptr, source->raw.len - source->written_bytes, format, arglist);
source->write_ptr += len;
source->written_bytes += len;
return GIT_SUCCESS;
}
int git__source_write(git_odb_source *source, const void *bytes, size_t len)
{
assert(source);
if (!source->open || source->write_ptr == NULL)
return GIT_ERROR;
while (source->written_bytes + len >= source->raw.len) {
if (source_resize(source) < 0)
return GIT_ENOMEM;
}
memcpy(source->write_ptr, bytes, len);
source->write_ptr += len;
source->written_bytes += len;
return GIT_SUCCESS;
}
void git_object__source_prepare_write(git_object *object)
{
const size_t base_size = 4096; /* 4Kb base size */
if (object->source.write_ptr != NULL || object->source.open)
git_object__source_close(object);
/* TODO: proper size calculation */
object->source.raw.data = git__malloc(base_size);
object->source.raw.len = base_size;
object->source.write_ptr = object->source.raw.data;
object->source.written_bytes = 0;
object->source.open = 1;
object->source.out_of_sync = 1;
}
int git_object__source_writeback(git_object *object)
{
int error;
......
......@@ -30,7 +30,9 @@ struct git_repository {
int git_object__source_open(git_object *object);
void git_object__source_close(git_object *object);
void git_object__source_prepare_write(git_object *object);
int git_object__source_write(git_object *object, const void *bytes, size_t len);
int git_object__source_writeback(git_object *object);
int git__source_printf(git_odb_source *source, const char *format, ...);
int git__source_write(git_odb_source *source, const void *bytes, size_t len);
#endif
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment