On Thu, Jul 15 2021, Jeff King wrote: > On Wed, Jul 14, 2021 at 08:57:37PM +0200, Andrzej Hunt wrote: > >> > @@ -1331,8 +1336,10 @@ static int git_default_core_config(const char *var, const char *value, void *cb) >> > if (!strcmp(var, "core.attributesfile")) >> > return git_config_pathname(&git_attributes_file, var, value); >> > - if (!strcmp(var, "core.hookspath")) >> > + if (!strcmp(var, "core.hookspath")) { >> > + UNLEAK(git_hooks_path); >> > return git_config_pathname(&git_hooks_path, var, value); >> > + } >> >> Why is the UNLEAK necessary here? We generally want to limit use of UNLEAK >> to cmd_* functions or direct helpers. git_default_core_config() seems >> generic enough that it could be called from anywhere, and using UNLEAK here >> means we're potentially masking a real leak? >> >> IIUC the leak here happens because: >> - git_hooks_path is a global variable - hence it's unlikely we'd ever >> bother cleaning it up. >> - git_default_core_config() gets called a first time with >> core.hookspath, and we end up allocating new memory into >> git_hooks_path. >> - git_default_core_config() gets called again with core.hookspath, >> and we overwrite git_hooks_path with a new string which leaks >> the string that git_hooks_path used to point to. >> >> So I think the real fix is to free(git_hooks_path) instead of an UNLEAK? >> (Looking at the surrounding code, it looks like the same pattern of leak >> might be repeated for other similar globals - is it worth auditing those >> while we're here?) > > This is a common leak pattern in Git. We do something like: > > static const char *foo = "default"; > ... > int config_cb(const char *var, const char *value, void *) > { > if (!strcmp(var, "core.foo")) > foo = xstrdup(value); > } > > So we leak if the variable appears twice. But we can't just call > "free(foo)" here. In the first call, it's pointing to a string literal! > > In the case of git_hooks_path, it defaults to NULL, so this works out > OK. But it's setting up a trap for somebody later on, who assigns it a > default value (and the compiler won't help; it's a "const char *", so > the assignment is fine, and the free() would already be casting away the > constness). > > I see a few possible solutions: > > - instead of strdup'ing long-lived config values, strintern() them. > This is really leaking them, but in a way that we hold on to the old > values. This is actually more or less what UNLEAK() is doing under > the hood (saving a reference to the old buffer, even the variable is > overwritten). > > - find a way to tell when a string comes from the heap versus a > literal. I don't think you can do this portably without keeping your > own separate flag. We could abstract away some of the pain with a > struct like: > > struct def_string { > /* might point to heap memory; const because you must > * check flag before modifying */ > const char *value; > int from_heap; > } > > /* regular static initialization is OK if you don't want a default */ > #define DEF_STRING_INIT(str) { .value = str } > > static void def_string_set(struct def_string *ds, const char *value) > { > if (ds->from_heap) > free(ds->value); > ds->value = xstrdup(value); > ds->from_heap = 1; > } > > The annoying thing is all of the users need to refer to > git_hook_path.value instead of just git_hook_path. If you don't mind > a little macro hackery, we could get around that by declaring pairs > of variables. Like: > > #define DEF_STRING_DECLARE(name, value) \ > const char *name = value; \ > int name##_from_heap > > #define DEF_STRING_SET(name, value) do { \ > if (name##_from_heap) \ > free(name); \ > name = xstrdup(value); \ > name##_from_heap = 1; \ > } while(0) > > I can't say I _love_ any of that, but I think it would work (and > probably we'd adapt our helpers like git_config_pathname() to take a > def_string. Or I guess just have a def_string_free() which can be called > before writing into them). > > But maybe there's a better solution I'm missing. Instead of: "int from_heap" in your "def_string" I think we should just use "struct string_list_item". I.e. you want a void* here. Why? <Digression> I have an unsent series for handling some more common cases in the string-list API. I started writing it due to a very related problem, i.e. that we conflate "string init dup/nodup" with "do we want to free?". We (ab)use the "strdup_strings" in a few places to free that sort of thing at the end if we have heap-allocated strings, but ones we did not strdup ourselves, e.g. this in merge-ort.c (not picking on Elijah (CC'd) here, it's common in lots of places, and this one was pretty much lifted from merge-recursive). opti->paths_to_free.strdup_strings = 1; string_list_clear(&opti->paths_to_free, 0); opti->paths_to_free.strdup_strings = 0; So I improved the string-list and strmap free functions so you can instead do: string_list_clear_strings((&opti->paths_to_free, 0); And that along with some other changes allows you to clear (or not) any combination of the string, util, or have a callback function of your own run (but be ensured to run all of those before we get to any of the other freeing). </Digression> You must be thinking what any of this has to do with heap strings in C, well one common case you've not discussed is that we sometimes do the equivalent of, with string-list.h or not (somewhat pseudocode); void add_to_list(struct string_list *list, char *on_heap_now_we_own_it) { char *ptr = on_heap_now_we_own_it; char *mydup = xstrdup("foo"); ptr++; /* skip first byte */ string_list_append(list, ptr); string_list_append(list, mydup); } And: struct string_list list = STRING_LIST_INIT_NODUP; /* other stuff here, we get strings from somewhere etc. */ add_to_list(list, some_string); So now you're left with needing to free both at the end, but we since we did ptr++ there we can't free() that (we'd need to free(ptr - 1), but how to keep track of that?). Well, tying this back to my clear() improvements for string-list.h I thought a really neat solution to this was: string_list_append(list, ptr)->util = on_heap_now_we_own_it; string_list_append(list, mydup)->util = mydup; I.e. by convention we store the pointer we need to free (if any) in the "util" field. And then if you get a string not from the heap you just leave the "util" as NULL, and at the end you just free() all your "util" fields, and it just so happens that some of them are the same as the "string" field. We're not in the habit of passing loose "string_list_item" around now, but I don't see why we wouldn't (possibly with a change to extract that bit out, so we could use it in other places). The neat thing about doing this is also that you're not left with every API boundary needing to deal with your new "def_string", a lot of them use string_list already, and hardly need to change anything, to the extent that we do need to change anything having a "void *util" is a lot more generally usable. You end up getting memory management for free as you gain a feature to pass arbitrary data along with your items.