Merge pull request #2725 from libgit2/vmg/attr-null

Do not assume blob contents are NULL terminated
This commit is contained in:
Edward Thomson 2014-11-21 13:16:42 -05:00
commit e0482934e6
6 changed files with 23 additions and 19 deletions

View File

@ -38,6 +38,7 @@ static void parse_opts(struct opts *o, int argc, char *argv[]);
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
int i, line, break_on_null_hunk; int i, line, break_on_null_hunk;
size_t rawsize;
char spec[1024] = {0}; char spec[1024] = {0};
struct opts o = {0}; struct opts o = {0};
const char *rawdata; const char *rawdata;
@ -94,23 +95,24 @@ int main(int argc, char *argv[])
git_object_free(obj); git_object_free(obj);
rawdata = git_blob_rawcontent(blob); rawdata = git_blob_rawcontent(blob);
rawsize = git_blob_rawsize(blob);
/** Produce the output. */ /** Produce the output. */
line = 1; line = 1;
i = 0; i = 0;
break_on_null_hunk = 0; break_on_null_hunk = 0;
while (i < git_blob_rawsize(blob)) { while (i < rawsize) {
const char *eol = strchr(rawdata+i, '\n'); const char *eol = memchr(rawdata + i, '\n', rawsize - i);
char oid[10] = {0}; char oid[10] = {0};
const git_blame_hunk *hunk = git_blame_get_hunk_byline(blame, line); const git_blame_hunk *hunk = git_blame_get_hunk_byline(blame, line);
if (break_on_null_hunk && !hunk) break; if (break_on_null_hunk && !hunk)
break;
if (hunk) { if (hunk) {
char sig[128] = {0}; char sig[128] = {0};
break_on_null_hunk = 1; break_on_null_hunk = 1;
git_oid_tostr(oid, 10, &hunk->final_commit_id); git_oid_tostr(oid, 10, &hunk->final_commit_id);
snprintf(sig, 30, "%s <%s>", hunk->final_signature->name, hunk->final_signature->email); snprintf(sig, 30, "%s <%s>", hunk->final_signature->name, hunk->final_signature->email);
@ -118,8 +120,8 @@ int main(int argc, char *argv[])
oid, oid,
sig, sig,
line, line,
(int)(eol-rawdata-i), (int)(eol - rawdata - i),
rawdata+i); rawdata + i);
} }
i = (int)(eol - rawdata + 1); i = (int)(eol - rawdata + 1);

View File

@ -103,7 +103,6 @@ int git_attr_file__load(
int error = 0; int error = 0;
git_blob *blob = NULL; git_blob *blob = NULL;
git_buf content = GIT_BUF_INIT; git_buf content = GIT_BUF_INIT;
const char *data = NULL;
git_attr_file *file; git_attr_file *file;
struct stat st; struct stat st;
@ -120,7 +119,9 @@ int git_attr_file__load(
(error = git_blob_lookup(&blob, repo, &id)) < 0) (error = git_blob_lookup(&blob, repo, &id)) < 0)
return error; return error;
data = git_blob_rawcontent(blob); /* Do not assume that data straight from the ODB is NULL-terminated;
* copy the contents of a file to a buffer to work on */
git_buf_put(&content, git_blob_rawcontent(blob), git_blob_rawsize(blob));
break; break;
} }
case GIT_ATTR_FILE__FROM_FILE: { case GIT_ATTR_FILE__FROM_FILE: {
@ -143,7 +144,6 @@ int git_attr_file__load(
if (error < 0) if (error < 0)
return GIT_ENOTFOUND; return GIT_ENOTFOUND;
data = content.ptr;
break; break;
} }
default: default:
@ -154,7 +154,7 @@ int git_attr_file__load(
if ((error = git_attr_file__new(&file, entry, source)) < 0) if ((error = git_attr_file__new(&file, entry, source)) < 0)
goto cleanup; goto cleanup;
if (parser && (error = parser(repo, file, data)) < 0) { if (parser && (error = parser(repo, file, git_buf_cstr(&content))) < 0) {
git_attr_file__free(file); git_attr_file__free(file);
goto cleanup; goto cleanup;
} }

View File

@ -143,6 +143,7 @@ int git_buf_text_lf_to_crlf(git_buf *tgt, const git_buf *src)
tgt->ptr[tgt->size++] = '\n'; tgt->ptr[tgt->size++] = '\n';
} }
tgt->ptr[tgt->size] = '\0';
return git_buf_put(tgt, scan, end - scan); return git_buf_put(tgt, scan, end - scan);
} }

View File

@ -176,10 +176,13 @@ int git_buf_putcn(git_buf *buf, char c, size_t len)
int git_buf_put(git_buf *buf, const char *data, size_t len) int git_buf_put(git_buf *buf, const char *data, size_t len)
{ {
ENSURE_SIZE(buf, buf->size + len + 1); if (len) {
memmove(buf->ptr + buf->size, data, len); assert(data);
buf->size += len; ENSURE_SIZE(buf, buf->size + len + 1);
buf->ptr[buf->size] = '\0'; memmove(buf->ptr + buf->size, data, len);
buf->size += len;
buf->ptr[buf->size] = '\0';
}
return 0; return 0;
} }

View File

@ -323,11 +323,10 @@ static int note_new(
git_signature_dup(&note->committer, git_commit_committer(commit)) < 0) git_signature_dup(&note->committer, git_commit_committer(commit)) < 0)
return -1; return -1;
note->message = git__strdup((char *)git_blob_rawcontent(blob)); note->message = git__strndup(git_blob_rawcontent(blob), git_blob_rawsize(blob));
GITERR_CHECK_ALLOC(note->message); GITERR_CHECK_ALLOC(note->message);
*out = note; *out = note;
return 0; return 0;
} }

View File

@ -281,11 +281,10 @@ check_buf_append_abc(
/* more variations on append tests */ /* more variations on append tests */
void test_core_buffer__5(void) void test_core_buffer__5(void)
{ {
check_buf_append("", "", "", 0, 8); check_buf_append("", "", "", 0, 0);
check_buf_append("a", "", "a", 1, 8); check_buf_append("a", "", "a", 1, 0);
check_buf_append("", "a", "a", 1, 8); check_buf_append("", "a", "a", 1, 8);
check_buf_append("", "a", "a", 1, 8); check_buf_append("", "a", "a", 1, 8);
check_buf_append("a", "", "a", 1, 8);
check_buf_append("a", "b", "ab", 2, 8); check_buf_append("a", "b", "ab", 2, 8);
check_buf_append("", "abcdefgh", "abcdefgh", 8, 16); check_buf_append("", "abcdefgh", "abcdefgh", 8, 16);
check_buf_append("abcdefgh", "", "abcdefgh", 8, 16); check_buf_append("abcdefgh", "", "abcdefgh", 8, 16);