On Wed, Jul 13 2022, Phillip Wood wrote: > On 11/07/2022 11:02, Ævar Arnfjörð Bjarmason wrote: >> On Mon, Jul 11 2022, Phillip Wood wrote: > [...] >>> Thanks for showing this, it is really helpful to see a concrete >>> example. I was especially interested to see how you went about the >>> conversion without redefining standard library functions or >>> introducing non-namespaced identifiers in files that included >>> xdiff.h. Unfortunately you have not done that and so I don't think the >>> approach above is viable for a well behaved library. >> Why? Because there's some header where doing e.g.: >> #include "git2/something.h" >> Will directly include git-xdiff.h by proxy into the user's program? > > No because you're redefining malloc() and introducing ALLOC_GROW() etc > in > src/libgit2/{Blame_git.c,Diff_xdiff.c,Merge_file.c,Patch_generate.c,Checkout.c} > and > Test/libgit2/merge/files.c. It is not expected that including xdiff.h > will change a bunch of symbols in the file it is included in. ...which is why if you read to the sha1collisiondetection commit below you'd follow that up with including a header at the end of xdiff.h which would do: #undef malloc etc., so you wouldn't leak that macro beyond the code that needs it, and you wouldn't leak the xdl_* macros either, which are purely needed internally in that code. So even aside from my suggestion of doing it this way it seems to me the structure has macro hygene issues worth fixing, see e.g. how we have refs-internal.h v.s. refs.h in git.git for similar reasons. >> I admit I didn't check that, and assumed that these were only >> included >> by other *.c files in libgit2 itself. I.e. it would compile xdiff for >> its own use, but then expose its own API. >> Anyway, if it is directly included in some user-exposed *.h file >> then >> yes, you don't want to overwrite their "malloc", but that's a small >> matter of doing an "#undef malloc" at the end (which as the below >> linked-to patch shows we'd support by having macros like >> SHA1DC_CUSTOM_TRAILING_INCLUDE_UBC_CHECK_C). >> Aside from what I'm proposing here doing such "undef at the end" >> seems >> like a good idea in any case, because if there is any macro leakage here >> you're e.g. re-defining "XDL_BUG" and other things that aren't clearly >> in the libgit2 namespace now, no? >> >>>> Now, I think that's obviously worth adjusting, e.g. the series I've got >>>> here could make this easier for libgit2 by including st_mult() at least, >>>> I'm not sure what you want to do about regexec_buf(). >>>> For the monkeypatching you do now of creating a "git-xdiff.h" I >>>> think it >>>> would be very good to get a patch like what I managed to get >>>> sha1collisiondetection.git to accept for our own use-case, which allows >>>> us to use their upstream code as-is from a submodule: >>>> https://github.com/cr-marcstevens/sha1collisiondetection/commit/b45fcef >>> >>> Thanks for the link, That's a really good example where all the >>> identifiers are namespaced and the public include file does not >>> pollute the code that is including it. If you come up with something >>> like that where the client code can set up same #defines for malloc, >>> calloc, realloc and free that would be a good way forward. >> I don't need to come up with that, you've got the linker to do that. >> I.e. for almost any "normal" use of xdiff you'd simply compile it >> with >> its reference to malloc(), and then you either link that library that >> already links to libc into your program. >> So if you use a custom malloc the xdiff code might still use libc >> malloc, or you'd link the two together and link the resulting program >> with your own custom malloc. >> As explained in the previous & linked-to threads this is how almost >> everyone would include & use such a library, and indeed that's how git >> itself can use malloc() and free() in its sources, but have options to >> link to libc malloc, nedmalloc etc. >> But instead of using the linker to resolve "malloc" to git__malloc >> you'd >> like to effectively include the upstream *.[ch] files directly, and >> pretend as though the upstream source used git__malloc() or whatever to >> begin with. >> I don't really understand why you can't just do that at the linker >> level, especially as doing so guards you better against namespace >> pollution. But whatever, the suggestion(s) above assume you've got a >> good reason, but show that we don't need to prefix every standard >> function just to accommodate that. >> Instead we can just provide a knob to include a header of yours >> after >> all others have been included (which the libgit2 monkeypatches to xdiff >> already include), and have that header define "malloc" as "git__malloc", >> "BUG" as "GIT_ASSERT" etc. >> And yes, if you're in turn providing an interface where others will >> then >> include your header that's doing that you'll need to apply some >> namespacing paranoia, namely to "#undef malloc" etc. >> But none of that requires git to carry prefixed versions of standard >> functions you'd wish to replace, which the patches here show. >> >>> I do not think we should require projects to be defining there own >>> versions of [C]ALLOC_*() and BUG() just to use xdiff. >> No, I don't think so either. I.e. the idea with these patches is >> that we >> could bundle up xdiff/* and git-shared-util.h into one distributed >> libgit, which down the road we could even roll independent releases for >> (as upstream seems to have forever gone away). >> Whereas the proposals coming out of libgit2[1] for generalizing >> xdiff/ >> for general use seem to stop just short of the specific hacks we need to >> get it running for libgit2, but e.g. leave "xdl_malloc" defined as >> "xmalloc". >> That isn't a standard library function, and therefore the first >> thing >> you'd need to do when using it as a library is to find out how git.git >> uses that, and copy/paste it or define your own replacement. >> Whereas I think it should be the other way around, we should >> e.g. ship a >> shimmy BUG() that calls fprintf(stderr, ...) and abort(), but for >> advanced users such as libgit2 guard those with "ifndef" or whatever, so >> you can have it call GIT_ASSERT() and the like instead. >> 1. https://lore.kernel.org/git/20220209013354.GB7@abe733c6e288/ ^ I.e. this.