Message ID | 80dcc2ba3aa0ef72abe18f8525d571ea39ac6382.1741911652.git.gitgitgadget@gmail.com (mailing list archive) |
---|---|
State | Superseded |
Headers | show |
Series | Add a static analysis job to prevent assertions with side effects | expand |
"Elijah Newren via GitGitGadget" <gitgitgadget@gmail.com> writes: > +make CHECK_ASSERTION_SIDE_EFFECTS=1 >compiler_output 2>compiler_error > +if test $? != 0 > +then > + echo "ERROR: The compiler could not verify the following assert()" >&2 > + echo " calls are free of side-effects. Please replace with" >&2 > + echo " BUG_IF_NOT() calls." >&2 > + grep undefined.reference.to..not_supposed_to_survive compiler_error \ > + | sed -e s/:[^:]*$// | sort | uniq | tr ':' ' ' \ > + | while read f l A few style guides: - doing multiple echo into the same descriptor is easier to read if you have redirection near the beginning, i.e. echo >&2 "message one" echo >&2 "message two that may way be longer than the previous" ehco >&2 "message three" - multi-line pipelines are easier to follow without backslash by ending the previous line with a '|', i.e. grep ... file | sed -e ... | while read file line do ... - I thought our "one indent one tab" standard extends to shell scripts as well? > diff --git a/git-compat-util.h b/git-compat-util.h > index c3415ad7e0a..0aefd763751 100644 > --- a/git-compat-util.h > +++ b/git-compat-util.h > @@ -1584,4 +1584,10 @@ static inline void *container_of_or_null_offset(void *ptr, size_t offset) > ((uintptr_t)&(ptr)->member - (uintptr_t)(ptr)) > #endif /* !__GNUC__ */ > > +#ifdef CHECK_ASSERTION_SIDE_EFFECTS > +#undef assert > +extern int not_supposed_to_survive; > +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) > +#endif /* CHECK_ASSERTION_SIDE_EFFECTS */ Cute. As this checking assert is in void context, the optimizing compiler knows that the entire thing can be optimized away ONLY IF it can somehow prove that (expr) has no side effect. And if it does not optimize it away, you will hit an error from the linker, saying that the undefined variable is being used. This requires a fairly good optimizing compiler that can peek into (as in "inline") what is in expr to notice, so it cannot be free of false positive, but at least when the optimization works as expected, it is provably (modulo optimizer bugs) side-effect free. Is this something we can use in our project? I am just double checking. Thanks.
On 2025-03-14 at 01:06:21, Junio C Hamano wrote: > "Elijah Newren via GitGitGadget" <gitgitgadget@gmail.com> writes: > > > diff --git a/git-compat-util.h b/git-compat-util.h > > index c3415ad7e0a..0aefd763751 100644 > > --- a/git-compat-util.h > > +++ b/git-compat-util.h > > @@ -1584,4 +1584,10 @@ static inline void *container_of_or_null_offset(void *ptr, size_t offset) > > ((uintptr_t)&(ptr)->member - (uintptr_t)(ptr)) > > #endif /* !__GNUC__ */ > > > > +#ifdef CHECK_ASSERTION_SIDE_EFFECTS > > +#undef assert > > +extern int not_supposed_to_survive; > > +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) > > +#endif /* CHECK_ASSERTION_SIDE_EFFECTS */ > > Cute. As this checking assert is in void context, the optimizing > compiler knows that the entire thing can be optimized away ONLY IF > it can somehow prove that (expr) has no side effect. And if it does > not optimize it away, you will hit an error from the linker, saying > that the undefined variable is being used. I agree this is very clever. > This requires a fairly good optimizing compiler that can peek into > (as in "inline") what is in expr to notice, so it cannot be free of > false positive, but at least when the optimization works as expected, > it is provably (modulo optimizer bugs) side-effect free. > > Is this something we can use in our project? I am just double > checking. I believe it's valid in C99. Certainly some compiler might be bad at optimizing, or a user may have compiled with -O0, but this is run in CI, where we have known good compilers and can control the optimization flags. I doubt GCC, Clang, or MSVC will have problems here, and since this is not on by default, users using something less capable (the Tiny C Compiler, maybe?) or a vendor compiler won't even see it. Was there some other case that you were concerned about?
"brian m. carlson" <sandals@crustytoothpaste.net> writes: >> Is this something we can use in our project? I am just double >> checking. > > I believe it's valid in C99. Certainly some compiler might be bad at > optimizing, or a user may have compiled with -O0, but this is run in CI, > where we have known good compilers and can control the optimization > flags. I doubt GCC, Clang, or MSVC will have problems here, and since > this is not on by default, users using something less capable (the Tiny > C Compiler, maybe?) or a vendor compiler won't even see it. > > Was there some other case that you were concerned about? Licensing, mostly, as clever things we see are not necessarily home grown. I know the patch came with DCO sign-off, but it does not hurt to double check.
On Thu, Mar 13, 2025 at 6:20 PM Junio C Hamano <gitster@pobox.com> wrote: > > "brian m. carlson" <sandals@crustytoothpaste.net> writes: > > >> Is this something we can use in our project? I am just double > >> checking. > > > > I believe it's valid in C99. Certainly some compiler might be bad at > > optimizing, or a user may have compiled with -O0, but this is run in CI, > > where we have known good compilers and can control the optimization > > flags. I doubt GCC, Clang, or MSVC will have problems here, and since > > this is not on by default, users using something less capable (the Tiny > > C Compiler, maybe?) or a vendor compiler won't even see it. > > > > Was there some other case that you were concerned about? > > Licensing, mostly, as clever things we see are not necessarily home > grown. I know the patch came with DCO sign-off, but it does not > hurt to double check. These two lines: > +extern int not_supposed_to_survive; > +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) , which serve as the core trick, I had used elsewhere before. Doing some searches, it looks like those likely came from https://stackoverflow.com/questions/10593492/catching-assert-with-side-effects. And it appears that StackOverflow is CC-BY-SA-3.0 or -4.0. Doh, sorry. Anyone got a clever alternative? The rest of the patch was all written by me.
Elijah Newren <newren@gmail.com> writes: >> Licensing, mostly, as clever things we see are not necessarily home >> grown. I know the patch came with DCO sign-off, but it does not >> hurt to double check. > > These two lines: > >> +extern int not_supposed_to_survive; >> +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) > > , which serve as the core trick, I had used elsewhere before. It may be arguable that it is too small to be copyrightable and there is no other way to express the idea behind that check, but in any case ... > Anyone got a clever alternative? ... as I cannot unsee your patch, I cannot be the one who comes up with a clever alternative, if we are worried about licensing with what you posted X-<.
On Fri, Mar 14, 2025 at 10:29 AM Junio C Hamano <gitster@pobox.com> wrote: > > Elijah Newren <newren@gmail.com> writes: > > >> Licensing, mostly, as clever things we see are not necessarily home > >> grown. I know the patch came with DCO sign-off, but it does not > >> hurt to double check. > > > > These two lines: > > > >> +extern int not_supposed_to_survive; > >> +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) > > > > , which serve as the core trick, I had used elsewhere before. > > It may be arguable that it is too small to be copyrightable and > there is no other way to express the idea behind that check, but > in any case ... That's what I had been assuming, but then you, brian, and Taylor all pointed out how clever it was making me think otherwise. > > Anyone got a clever alternative? > > ... as I cannot unsee your patch, I cannot be the one who comes up > with a clever alternative, if we are worried about licensing with > what you posted X-<. Turns out we don't need an alternative. I contacted the author, who responded and placed the two-liner into the public domain with no warranty of any kind. I'll send a re-roll with an updated commit message.
On Sat, Mar 15, 2025 at 11:38 PM Elijah Newren <newren@gmail.com> wrote: > > On Fri, Mar 14, 2025 at 10:29 AM Junio C Hamano <gitster@pobox.com> wrote: > > > > Elijah Newren <newren@gmail.com> writes: > > > > >> Licensing, mostly, as clever things we see are not necessarily home > > >> grown. I know the patch came with DCO sign-off, but it does not > > >> hurt to double check. > > > > > > These two lines: > > > > > >> +extern int not_supposed_to_survive; > > >> +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) > > > > > > , which serve as the core trick, I had used elsewhere before. > > > > It may be arguable that it is too small to be copyrightable and > > there is no other way to express the idea behind that check, but > > in any case ... > > That's what I had been assuming, but then you, brian, and Taylor all > pointed out how clever it was making me think otherwise. > > > > Anyone got a clever alternative? > > > > ... as I cannot unsee your patch, I cannot be the one who comes up > > with a clever alternative, if we are worried about licensing with > > what you posted X-<. > > Turns out we don't need an alternative. I contacted the author, who > responded and placed the two-liner into the public domain with no > warranty of any kind. I'll send a re-roll with an updated commit > message. And of course, in my excitement to send a re-roll addressing that issue, I totally spaced fixing the style issues you pointed out. I'll wait a couple days for any comments and then send a re-roll with those fixes.
Elijah Newren <newren@gmail.com> writes: >> >> +extern int not_supposed_to_survive; >> >> +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) >> > >> > , which serve as the core trick, I had used elsewhere before. >> >> It may be arguable that it is too small to be copyrightable and >> there is no other way to express the idea behind that check, but >> in any case ... > > That's what I had been assuming, but then you, brian, and Taylor all > pointed out how clever it was making me think otherwise. Heh, cleverness lies in the idea, not the expression, and copyright is about expression. > Turns out we don't need an alternative. I contacted the author, who > responded and placed the two-liner into the public domain with no > warranty of any kind. I'll send a re-roll with an updated commit > message. Wonderful. That is the best solution. Thanks.
diff --git a/Makefile b/Makefile index 7315507381e..57774912f18 100644 --- a/Makefile +++ b/Makefile @@ -2261,6 +2261,10 @@ ifdef WITH_BREAKING_CHANGES BASIC_CFLAGS += -DWITH_BREAKING_CHANGES endif +ifdef CHECK_ASSERTION_SIDE_EFFECTS + BASIC_CFLAGS += -DCHECK_ASSERTION_SIDE_EFFECTS +endif + ifdef INCLUDE_LIBGIT_RS # Enable symbol hiding in contrib/libgit-sys/libgitpub.a without making # us rebuild the whole tree every time we run a Rust build. diff --git a/ci/check-unsafe-assertions.sh b/ci/check-unsafe-assertions.sh new file mode 100755 index 00000000000..d66091efd22 --- /dev/null +++ b/ci/check-unsafe-assertions.sh @@ -0,0 +1,18 @@ +#!/bin/sh + +make CHECK_ASSERTION_SIDE_EFFECTS=1 >compiler_output 2>compiler_error +if test $? != 0 +then + echo "ERROR: The compiler could not verify the following assert()" >&2 + echo " calls are free of side-effects. Please replace with" >&2 + echo " BUG_IF_NOT() calls." >&2 + grep undefined.reference.to..not_supposed_to_survive compiler_error \ + | sed -e s/:[^:]*$// | sort | uniq | tr ':' ' ' \ + | while read f l + do + printf "${f}:${l}\n " + awk -v start="$l" 'NR >= start { print; if (/\);/) exit }' $f + done + exit 1 +fi +rm compiler_output compiler_error diff --git a/ci/run-static-analysis.sh b/ci/run-static-analysis.sh index 0d51e5ce0e7..ae714e020ae 100755 --- a/ci/run-static-analysis.sh +++ b/ci/run-static-analysis.sh @@ -31,4 +31,6 @@ exit 1 make check-pot +${0%/*}/check-unsafe-assertions.sh + save_good_tree diff --git a/git-compat-util.h b/git-compat-util.h index c3415ad7e0a..0aefd763751 100644 --- a/git-compat-util.h +++ b/git-compat-util.h @@ -1584,4 +1584,10 @@ static inline void *container_of_or_null_offset(void *ptr, size_t offset) ((uintptr_t)&(ptr)->member - (uintptr_t)(ptr)) #endif /* !__GNUC__ */ +#ifdef CHECK_ASSERTION_SIDE_EFFECTS +#undef assert +extern int not_supposed_to_survive; +#define assert(expr) ((void)(not_supposed_to_survive || (expr))) +#endif /* CHECK_ASSERTION_SIDE_EFFECTS */ + #endif