Merge branch 'ot/ref-filter-plug-leaks'

Plugging a handful of memory leaks in the ref-filter codepath.

* ot/ref-filter-plug-leaks:
  ref-filter: free item->value and item->value->s
  ls-remote: release memory instead of UNLEAK
  ref-filter: free memory from used_atom
This commit is contained in:
Junio C Hamano 2018-10-30 15:43:44 +09:00
commit 9d00100c38
2 changed files with 59 additions and 43 deletions

View File

@ -151,6 +151,6 @@ int cmd_ls_remote(int argc, const char **argv, const char *prefix)
} }
UNLEAK(sorting); UNLEAK(sorting);
UNLEAK(ref_array); ref_array_clear(&ref_array);
return status; return status;
} }

View File

@ -875,7 +875,7 @@ static void grab_common_values(struct atom_value *val, int deref, struct expand_
if (deref) if (deref)
name++; name++;
if (!strcmp(name, "objecttype")) if (!strcmp(name, "objecttype"))
v->s = type_name(oi->type); v->s = xstrdup(type_name(oi->type));
else if (!strcmp(name, "objectsize")) { else if (!strcmp(name, "objectsize")) {
v->value = oi->size; v->value = oi->size;
v->s = xstrfmt("%lu", oi->size); v->s = xstrfmt("%lu", oi->size);
@ -899,9 +899,9 @@ static void grab_tag_values(struct atom_value *val, int deref, struct object *ob
if (deref) if (deref)
name++; name++;
if (!strcmp(name, "tag")) if (!strcmp(name, "tag"))
v->s = tag->tag; v->s = xstrdup(tag->tag);
else if (!strcmp(name, "type") && tag->tagged) else if (!strcmp(name, "type") && tag->tagged)
v->s = type_name(tag->tagged->type); v->s = xstrdup(type_name(tag->tagged->type));
else if (!strcmp(name, "object") && tag->tagged) else if (!strcmp(name, "object") && tag->tagged)
v->s = xstrdup(oid_to_hex(&tag->tagged->oid)); v->s = xstrdup(oid_to_hex(&tag->tagged->oid));
} }
@ -1032,7 +1032,7 @@ static void grab_date(const char *buf, struct atom_value *v, const char *atomnam
v->value = timestamp; v->value = timestamp;
return; return;
bad: bad:
v->s = ""; v->s = xstrdup("");
v->value = 0; v->value = 0;
} }
@ -1227,7 +1227,7 @@ static void fill_missing_values(struct atom_value *val)
for (i = 0; i < used_atom_cnt; i++) { for (i = 0; i < used_atom_cnt; i++) {
struct atom_value *v = &val[i]; struct atom_value *v = &val[i];
if (v->s == NULL) if (v->s == NULL)
v->s = ""; v->s = xstrdup("");
} }
} }
@ -1273,7 +1273,8 @@ static inline char *copy_advance(char *dst, const char *src)
static const char *lstrip_ref_components(const char *refname, int len) static const char *lstrip_ref_components(const char *refname, int len)
{ {
long remaining = len; long remaining = len;
const char *start = refname; const char *start = xstrdup(refname);
const char *to_free = start;
if (len < 0) { if (len < 0) {
int i; int i;
@ -1294,20 +1295,24 @@ static const char *lstrip_ref_components(const char *refname, int len)
while (remaining > 0) { while (remaining > 0) {
switch (*start++) { switch (*start++) {
case '\0': case '\0':
return ""; free((char *)to_free);
return xstrdup("");
case '/': case '/':
remaining--; remaining--;
break; break;
} }
} }
start = xstrdup(start);
free((char *)to_free);
return start; return start;
} }
static const char *rstrip_ref_components(const char *refname, int len) static const char *rstrip_ref_components(const char *refname, int len)
{ {
long remaining = len; long remaining = len;
char *start = xstrdup(refname); const char *start = xstrdup(refname);
const char *to_free = start;
if (len < 0) { if (len < 0) {
int i; int i;
@ -1327,9 +1332,10 @@ static const char *rstrip_ref_components(const char *refname, int len)
while (remaining-- > 0) { while (remaining-- > 0) {
char *p = strrchr(start, '/'); char *p = strrchr(start, '/');
if (p == NULL) if (p == NULL) {
return ""; free((char *)to_free);
else return xstrdup("");
} else
p[0] = '\0'; p[0] = '\0';
} }
return start; return start;
@ -1344,7 +1350,7 @@ static const char *show_ref(struct refname_atom *atom, const char *refname)
else if (atom->option == R_RSTRIP) else if (atom->option == R_RSTRIP)
return rstrip_ref_components(refname, atom->rstrip); return rstrip_ref_components(refname, atom->rstrip);
else else
return refname; return xstrdup(refname);
} }
static void fill_remote_ref_details(struct used_atom *atom, const char *refname, static void fill_remote_ref_details(struct used_atom *atom, const char *refname,
@ -1358,7 +1364,7 @@ static void fill_remote_ref_details(struct used_atom *atom, const char *refname,
NULL, AHEAD_BEHIND_FULL) < 0) { NULL, AHEAD_BEHIND_FULL) < 0) {
*s = xstrdup(msgs.gone); *s = xstrdup(msgs.gone);
} else if (!num_ours && !num_theirs) } else if (!num_ours && !num_theirs)
*s = ""; *s = xstrdup("");
else if (!num_ours) else if (!num_ours)
*s = xstrfmt(msgs.behind, num_theirs); *s = xstrfmt(msgs.behind, num_theirs);
else if (!num_theirs) else if (!num_theirs)
@ -1373,36 +1379,31 @@ static void fill_remote_ref_details(struct used_atom *atom, const char *refname,
} }
} else if (atom->u.remote_ref.option == RR_TRACKSHORT) { } else if (atom->u.remote_ref.option == RR_TRACKSHORT) {
if (stat_tracking_info(branch, &num_ours, &num_theirs, if (stat_tracking_info(branch, &num_ours, &num_theirs,
NULL, AHEAD_BEHIND_FULL) < 0) NULL, AHEAD_BEHIND_FULL) < 0) {
*s = xstrdup("");
return; return;
}
if (!num_ours && !num_theirs) if (!num_ours && !num_theirs)
*s = "="; *s = xstrdup("=");
else if (!num_ours) else if (!num_ours)
*s = "<"; *s = xstrdup("<");
else if (!num_theirs) else if (!num_theirs)
*s = ">"; *s = xstrdup(">");
else else
*s = "<>"; *s = xstrdup("<>");
} else if (atom->u.remote_ref.option == RR_REMOTE_NAME) { } else if (atom->u.remote_ref.option == RR_REMOTE_NAME) {
int explicit; int explicit;
const char *remote = atom->u.remote_ref.push ? const char *remote = atom->u.remote_ref.push ?
pushremote_for_branch(branch, &explicit) : pushremote_for_branch(branch, &explicit) :
remote_for_branch(branch, &explicit); remote_for_branch(branch, &explicit);
if (explicit) *s = xstrdup(explicit ? remote : "");
*s = xstrdup(remote);
else
*s = "";
} else if (atom->u.remote_ref.option == RR_REMOTE_REF) { } else if (atom->u.remote_ref.option == RR_REMOTE_REF) {
int explicit; int explicit;
const char *merge; const char *merge;
merge = remote_ref_for_branch(branch, atom->u.remote_ref.push, merge = remote_ref_for_branch(branch, atom->u.remote_ref.push,
&explicit); &explicit);
if (explicit) *s = xstrdup(explicit ? merge : "");
*s = xstrdup(merge);
else
*s = "";
} else } else
BUG("unhandled RR_* enum"); BUG("unhandled RR_* enum");
} }
@ -1451,7 +1452,7 @@ char *get_head_description(void)
static const char *get_symref(struct used_atom *atom, struct ref_array_item *ref) static const char *get_symref(struct used_atom *atom, struct ref_array_item *ref)
{ {
if (!ref->symref) if (!ref->symref)
return ""; return xstrdup("");
else else
return show_ref(&atom->u.refname, ref->symref); return show_ref(&atom->u.refname, ref->symref);
} }
@ -1510,7 +1511,7 @@ static int populate_value(struct ref_array_item *ref, struct strbuf *err)
ref->symref = resolve_refdup(ref->refname, RESOLVE_REF_READING, ref->symref = resolve_refdup(ref->refname, RESOLVE_REF_READING,
NULL, NULL); NULL, NULL);
if (!ref->symref) if (!ref->symref)
ref->symref = ""; ref->symref = xstrdup("");
} }
/* Fill in specials first */ /* Fill in specials first */
@ -1536,20 +1537,23 @@ static int populate_value(struct ref_array_item *ref, struct strbuf *err)
refname = get_symref(atom, ref); refname = get_symref(atom, ref);
else if (starts_with(name, "upstream")) { else if (starts_with(name, "upstream")) {
const char *branch_name; const char *branch_name;
v->s = "";
/* only local branches may have an upstream */ /* only local branches may have an upstream */
if (!skip_prefix(ref->refname, "refs/heads/", if (!skip_prefix(ref->refname, "refs/heads/",
&branch_name)) &branch_name)) {
v->s = xstrdup("");
continue; continue;
}
branch = branch_get(branch_name); branch = branch_get(branch_name);
refname = branch_get_upstream(branch, NULL); refname = branch_get_upstream(branch, NULL);
if (refname) if (refname)
fill_remote_ref_details(atom, refname, branch, &v->s); fill_remote_ref_details(atom, refname, branch, &v->s);
else
v->s = xstrdup("");
continue; continue;
} else if (atom->u.remote_ref.push) { } else if (atom->u.remote_ref.push) {
const char *branch_name; const char *branch_name;
v->s = ""; v->s = xstrdup("");
if (!skip_prefix(ref->refname, "refs/heads/", if (!skip_prefix(ref->refname, "refs/heads/",
&branch_name)) &branch_name))
continue; continue;
@ -1562,10 +1566,12 @@ static int populate_value(struct ref_array_item *ref, struct strbuf *err)
if (!refname) if (!refname)
continue; continue;
} }
/* We will definitely re-init v->s on the next line. */
free((char *)v->s);
fill_remote_ref_details(atom, refname, branch, &v->s); fill_remote_ref_details(atom, refname, branch, &v->s);
continue; continue;
} else if (starts_with(name, "color:")) { } else if (starts_with(name, "color:")) {
v->s = atom->u.color; v->s = xstrdup(atom->u.color);
continue; continue;
} else if (!strcmp(name, "flag")) { } else if (!strcmp(name, "flag")) {
char buf[256], *cp = buf; char buf[256], *cp = buf;
@ -1574,7 +1580,7 @@ static int populate_value(struct ref_array_item *ref, struct strbuf *err)
if (ref->flag & REF_ISPACKED) if (ref->flag & REF_ISPACKED)
cp = copy_advance(cp, ",packed"); cp = copy_advance(cp, ",packed");
if (cp == buf) if (cp == buf)
v->s = ""; v->s = xstrdup("");
else { else {
*cp = '\0'; *cp = '\0';
v->s = xstrdup(buf + 1); v->s = xstrdup(buf + 1);
@ -1584,40 +1590,42 @@ static int populate_value(struct ref_array_item *ref, struct strbuf *err)
continue; continue;
} else if (!strcmp(name, "HEAD")) { } else if (!strcmp(name, "HEAD")) {
if (atom->u.head && !strcmp(ref->refname, atom->u.head)) if (atom->u.head && !strcmp(ref->refname, atom->u.head))
v->s = "*"; v->s = xstrdup("*");
else else
v->s = " "; v->s = xstrdup(" ");
continue; continue;
} else if (starts_with(name, "align")) { } else if (starts_with(name, "align")) {
v->handler = align_atom_handler; v->handler = align_atom_handler;
v->s = ""; v->s = xstrdup("");
continue; continue;
} else if (!strcmp(name, "end")) { } else if (!strcmp(name, "end")) {
v->handler = end_atom_handler; v->handler = end_atom_handler;
v->s = ""; v->s = xstrdup("");
continue; continue;
} else if (starts_with(name, "if")) { } else if (starts_with(name, "if")) {
const char *s; const char *s;
v->s = "";
if (skip_prefix(name, "if:", &s)) if (skip_prefix(name, "if:", &s))
v->s = xstrdup(s); v->s = xstrdup(s);
else
v->s = xstrdup("");
v->handler = if_atom_handler; v->handler = if_atom_handler;
continue; continue;
} else if (!strcmp(name, "then")) { } else if (!strcmp(name, "then")) {
v->handler = then_atom_handler; v->handler = then_atom_handler;
v->s = ""; v->s = xstrdup("");
continue; continue;
} else if (!strcmp(name, "else")) { } else if (!strcmp(name, "else")) {
v->handler = else_atom_handler; v->handler = else_atom_handler;
v->s = ""; v->s = xstrdup("");
continue; continue;
} else } else
continue; continue;
if (!deref) if (!deref)
v->s = refname; v->s = xstrdup(refname);
else else
v->s = xstrfmt("%s^{}", refname); v->s = xstrfmt("%s^{}", refname);
free((char *)refname);
} }
for (i = 0; i < used_atom_cnt; i++) { for (i = 0; i < used_atom_cnt; i++) {
@ -1988,6 +1996,10 @@ static int ref_filter_handler(const char *refname, const struct object_id *oid,
static void free_array_item(struct ref_array_item *item) static void free_array_item(struct ref_array_item *item)
{ {
free((char *)item->symref); free((char *)item->symref);
if (item->value) {
free((char *)item->value->s);
free(item->value);
}
free(item); free(item);
} }
@ -1996,6 +2008,10 @@ void ref_array_clear(struct ref_array *array)
{ {
int i; int i;
for (i = 0; i < used_atom_cnt; i++)
free((char *)used_atom[i].name);
FREE_AND_NULL(used_atom);
used_atom_cnt = 0;
for (i = 0; i < array->nr; i++) for (i = 0; i < array->nr; i++)
free_array_item(array->items[i]); free_array_item(array->items[i]);
FREE_AND_NULL(array->items); FREE_AND_NULL(array->items);