Skip to content

Commit

Permalink
Merge branch 'jk/http-leakfixes' into next
Browse files Browse the repository at this point in the history
Leakfixes.

* jk/http-leakfixes: (28 commits)
  http-push: clean up local_refs at exit
  http-push: clean up loose request when falling back to packed
  http-push: clean up objects list
  http-push: free xml_ctx.cdata after use
  http-push: free remote_ls_ctx.dentry_name
  http-push: free transfer_request strbuf
  http-push: free transfer_request dest field
  http-push: free curl header lists
  http-push: free repo->url string
  http-push: clear refspecs before exiting
  http-walker: free fake packed_git list
  remote-curl: free HEAD ref with free_one_ref()
  http: stop leaking buffer in http_get_info_packs()
  http: call git_inflate_end() when releasing http_object_request
  http: fix leak of http_object_request struct
  http: fix leak when redacting cookies from curl trace
  transport-helper: fix leak of dummy refs_list
  fetch-pack: clear pack lockfiles list
  fetch: free "raw" string when shrinking refspec
  transport-helper: fix strbuf leak in push_refs_with_push()
  ...
  • Loading branch information
gitster committed Sep 27, 2024
2 parents 8c1aa7e + f4c768c commit c52b418
Show file tree
Hide file tree
Showing 29 changed files with 123 additions and 39 deletions.
14 changes: 13 additions & 1 deletion builtin/fetch-pack.c
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@ int cmd_fetch_pack(int argc,
struct ref *fetched_refs = NULL, *remote_refs = NULL;
const char *dest = NULL;
struct ref **sought = NULL;
struct ref **sought_to_free = NULL;
int nr_sought = 0, alloc_sought = 0;
int fd[2];
struct string_list pack_lockfiles = STRING_LIST_INIT_DUP;
Expand Down Expand Up @@ -243,6 +244,13 @@ int cmd_fetch_pack(int argc,
BUG("unknown protocol version");
}

/*
* Create a shallow copy of `sought` so that we can free all of its entries.
* This is because `fetch_pack()` will modify the array to evict some
* entries, but won't free those.
*/
DUP_ARRAY(sought_to_free, sought, nr_sought);

fetched_refs = fetch_pack(&args, fd, remote_refs, sought, nr_sought,
&shallow, pack_lockfiles_ptr, version);

Expand Down Expand Up @@ -280,9 +288,13 @@ int cmd_fetch_pack(int argc,
oid_to_hex(&ref->old_oid), ref->name);

for (size_t i = 0; i < nr_sought; i++)
free_one_ref(sought[i]);
free_one_ref(sought_to_free[i]);
free(sought_to_free);
free(sought);
free_refs(fetched_refs);
free_refs(remote_refs);
list_objects_filter_release(&args.filter_options);
oid_array_clear(&shallow);
string_list_clear(&pack_lockfiles, 0);
return ret;
}
1 change: 1 addition & 0 deletions builtin/fetch.c
Original file line number Diff line number Diff line change
Expand Up @@ -456,6 +456,7 @@ static void filter_prefetch_refspec(struct refspec *rs)

free(rs->items[i].src);
free(rs->items[i].dst);
free(rs->raw[i]);

for (j = i + 1; j < rs->nr; j++) {
rs->items[j - 1] = rs->items[j];
Expand Down
1 change: 1 addition & 0 deletions builtin/push.c
Original file line number Diff line number Diff line change
Expand Up @@ -669,6 +669,7 @@ int cmd_push(int argc,
rc = do_push(flags, push_options, remote);
string_list_clear(&push_options_cmdline, 0);
string_list_clear(&push_options_config, 0);
clear_cas_option(&cas);
if (rc == -1)
usage_with_options(push_usage, options);
else
Expand Down
2 changes: 2 additions & 0 deletions builtin/send-pack.c
Original file line number Diff line number Diff line change
Expand Up @@ -343,5 +343,7 @@ int cmd_send_pack(int argc,
free_refs(remote_refs);
free_refs(local_refs);
refspec_clear(&rs);
oid_array_clear(&shallow);
clear_cas_option(&cas);
return ret;
}
3 changes: 2 additions & 1 deletion commit.c
Original file line number Diff line number Diff line change
Expand Up @@ -595,7 +595,8 @@ int repo_parse_commit_internal(struct repository *r,
}

ret = parse_commit_buffer(r, item, buffer, size, 0);
if (save_commit_buffer && !ret) {
if (save_commit_buffer && !ret &&
!get_cached_commit_buffer(r, item, NULL)) {
set_commit_buffer(r, item, buffer, size);
return 0;
}
Expand Down
1 change: 1 addition & 0 deletions connect.c
Original file line number Diff line number Diff line change
Expand Up @@ -1485,6 +1485,7 @@ struct child_process *git_connect(int fd[2], const char *url,

free(hostandport);
free(path);
child_process_clear(conn);
free(conn);
strbuf_release(&cmd);
return NULL;
Expand Down
16 changes: 11 additions & 5 deletions http-fetch.c
Original file line number Diff line number Diff line change
Expand Up @@ -106,6 +106,7 @@ int cmd_main(int argc, const char **argv)
int nongit;
struct object_id packfile_hash;
struct strvec index_pack_args = STRVEC_INIT;
int ret;

setup_git_directory_gently(&nongit);

Expand Down Expand Up @@ -157,8 +158,8 @@ int cmd_main(int argc, const char **argv)

fetch_single_packfile(&packfile_hash, argv[arg],
index_pack_args.v);

return 0;
ret = 0;
goto out;
}

if (index_pack_args.nr)
Expand All @@ -170,7 +171,12 @@ int cmd_main(int argc, const char **argv)
commit_id = (char **) &argv[arg++];
commits = 1;
}
return fetch_using_walker(argv[arg], get_verbosely, get_recover,
commits, commit_id, write_ref,
commits_on_stdin);

ret = fetch_using_walker(argv[arg], get_verbosely, get_recover,
commits, commit_id, write_ref,
commits_on_stdin);

out:
strvec_clear(&index_pack_args);
return ret;
}
40 changes: 28 additions & 12 deletions http-push.c
Original file line number Diff line number Diff line change
Expand Up @@ -275,7 +275,7 @@ static void start_fetch_loose(struct transfer_request *request)
if (!start_active_slot(slot)) {
fprintf(stderr, "Unable to start GET request\n");
repo->can_update_info_refs = 0;
release_http_object_request(obj_req);
release_http_object_request(&obj_req);
release_request(request);
}
}
Expand Down Expand Up @@ -375,7 +375,7 @@ static void start_put(struct transfer_request *request)
/* Set it up */
git_deflate_init(&stream, zlib_compression_level);
size = git_deflate_bound(&stream, len + hdrlen);
strbuf_init(&request->buffer.buf, size);
strbuf_grow(&request->buffer.buf, size);
request->buffer.posn = 0;

/* Compress it */
Expand Down Expand Up @@ -437,9 +437,11 @@ static void start_move(struct transfer_request *request)
if (start_active_slot(slot)) {
request->slot = slot;
request->state = RUN_MOVE;
request->headers = dav_headers;
} else {
request->state = ABORTED;
FREE_AND_NULL(request->url);
curl_slist_free_all(dav_headers);
}
}

Expand Down Expand Up @@ -512,6 +514,8 @@ static void release_request(struct transfer_request *request)
}

free(request->url);
free(request->dest);
strbuf_release(&request->buffer.buf);
free(request);
}

Expand Down Expand Up @@ -578,9 +582,10 @@ static void finish_request(struct transfer_request *request)
if (obj_req->rename == 0)
request->obj->flags |= (LOCAL | REMOTE);

release_http_object_request(&obj_req);

/* Try fetching packed if necessary */
if (request->obj->flags & LOCAL) {
release_http_object_request(obj_req);
release_request(request);
} else
start_fetch_packed(request);
Expand Down Expand Up @@ -649,12 +654,10 @@ static void add_fetch_request(struct object *obj)
return;

obj->flags |= FETCHING;
request = xmalloc(sizeof(*request));
CALLOC_ARRAY(request, 1);
request->obj = obj;
request->url = NULL;
request->lock = NULL;
request->headers = NULL;
request->state = NEED_FETCH;
strbuf_init(&request->buffer.buf, 0);
request->next = request_queue_head;
request_queue_head = request;

Expand Down Expand Up @@ -685,12 +688,11 @@ static int add_send_request(struct object *obj, struct remote_lock *lock)
}

obj->flags |= PUSHING;
request = xmalloc(sizeof(*request));
CALLOC_ARRAY(request, 1);
request->obj = obj;
request->url = NULL;
request->lock = lock;
request->headers = NULL;
request->state = NEED_PUSH;
strbuf_init(&request->buffer.buf, 0);
request->next = request_queue_head;
request_queue_head = request;

Expand Down Expand Up @@ -912,6 +914,7 @@ static struct remote_lock *lock_remote(const char *path, long timeout)
result = XML_Parse(parser, in_buffer.buf,
in_buffer.len, 1);
free(ctx.name);
free(ctx.cdata);
if (result != XML_STATUS_OK) {
fprintf(stderr, "XML error: %s\n",
XML_ErrorString(
Expand Down Expand Up @@ -1169,6 +1172,7 @@ static void remote_ls(const char *path, int flags,
result = XML_Parse(parser, in_buffer.buf,
in_buffer.len, 1);
free(ctx.name);
free(ctx.cdata);

if (result != XML_STATUS_OK) {
fprintf(stderr, "XML error: %s\n",
Expand All @@ -1182,6 +1186,7 @@ static void remote_ls(const char *path, int flags,
}

free(ls.path);
free(ls.dentry_name);
free(url);
strbuf_release(&out_buffer.buf);
strbuf_release(&in_buffer);
Expand Down Expand Up @@ -1370,9 +1375,13 @@ static int get_delta(struct rev_info *revs, struct remote_lock *lock)
}

while (objects) {
struct object_list *next = objects->next;

if (!(objects->item->flags & UNINTERESTING))
count += add_send_request(objects->item, lock);
objects = objects->next;

free(objects);
objects = next;
}

return count;
Expand All @@ -1398,6 +1407,7 @@ static int update_remote(const struct object_id *oid, struct remote_lock *lock)
if (start_active_slot(slot)) {
run_active_slot(slot);
strbuf_release(&out_buffer.buf);
curl_slist_free_all(dav_headers);
if (results.curl_result != CURLE_OK) {
fprintf(stderr,
"PUT error: curl result=%d, HTTP code=%ld\n",
Expand All @@ -1407,6 +1417,7 @@ static int update_remote(const struct object_id *oid, struct remote_lock *lock)
}
} else {
strbuf_release(&out_buffer.buf);
curl_slist_free_all(dav_headers);
fprintf(stderr, "Unable to start PUT request\n");
return 0;
}
Expand Down Expand Up @@ -1516,6 +1527,7 @@ static void update_remote_info_refs(struct remote_lock *lock)
results.curl_result, results.http_code);
}
}
curl_slist_free_all(dav_headers);
}
strbuf_release(&buffer.buf);
}
Expand Down Expand Up @@ -1707,7 +1719,7 @@ int cmd_main(int argc, const char **argv)
int rc = 0;
int i;
int new_refs;
struct ref *ref, *local_refs;
struct ref *ref, *local_refs = NULL;

CALLOC_ARRAY(repo, 1);

Expand Down Expand Up @@ -1972,6 +1984,7 @@ int cmd_main(int argc, const char **argv)
cleanup:
if (info_ref_lock)
unlock_remote(info_ref_lock);
free(repo->url);
free(repo);

http_cleanup();
Expand All @@ -1983,5 +1996,8 @@ int cmd_main(int argc, const char **argv)
request = next_request;
}

refspec_clear(&rs);
free_refs(local_refs);

return rc;
}
18 changes: 14 additions & 4 deletions http-walker.c
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ static void start_object_request(struct object_request *obj_req)
obj_req->state = ACTIVE;
if (!start_active_slot(slot)) {
obj_req->state = ABORTED;
release_http_object_request(req);
release_http_object_request(&req);
return;
}
}
Expand Down Expand Up @@ -110,7 +110,7 @@ static void process_object_response(void *callback_data)
if (obj_req->repo->next) {
obj_req->repo =
obj_req->repo->next;
release_http_object_request(obj_req->req);
release_http_object_request(&obj_req->req);
start_object_request(obj_req);
return;
}
Expand Down Expand Up @@ -495,7 +495,7 @@ static int fetch_object(struct walker *walker, unsigned char *hash)

if (repo_has_object_file(the_repository, &obj_req->oid)) {
if (obj_req->req)
abort_http_object_request(obj_req->req);
abort_http_object_request(&obj_req->req);
abort_object_request(obj_req);
return 0;
}
Expand Down Expand Up @@ -543,7 +543,7 @@ static int fetch_object(struct walker *walker, unsigned char *hash)
strbuf_release(&buf);
}

release_http_object_request(req);
release_http_object_request(&obj_req->req);
release_object_request(obj_req);
return ret;
}
Expand Down Expand Up @@ -579,8 +579,18 @@ static void cleanup(struct walker *walker)
if (data) {
alt = data->alt;
while (alt) {
struct packed_git *pack;

alt_next = alt->next;

pack = alt->packs;
while (pack) {
struct packed_git *pack_next = pack->next;
close_pack(pack);
free(pack);
pack = pack_next;
}

free(alt->base);
free(alt);

Expand Down
Loading

0 comments on commit c52b418

Please sign in to comment.