Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a static analysis job to prevent assertions with side effects #1881

Closed
wants to merge 3 commits into from

Conversation

newren
Copy link

@newren newren commented Mar 13, 2025

We have several hundred assert() invocations in our code base. Some have suggested that we should add a recommendation in our CodingGuidelines to avoid their use, because there is a risk that someone might include something with a side-effect in their assertion, which can lead to a very difficult to debug problem. However, CodingGuidelines are going to be less effective at preventing that foot-gun than a CI job which can warn of assertions that possibly have side-effects. So, let's add a CI job instead.

While it is difficult to perfectly determine whether any expression has side effects, a simple compiler/linker hack can prove that all but 9 of our several hundred assert() calls are indeed free from them. While I believe the remaining 9 are also free of side effects, it's easier to just convert those 9 to a new macro (which will not be compiled out when NDEBUG is defined), and instruct any future assertion writers to likewise switch to that alternative macro if they have a slightly more involved assert() invocation.

See https://github.com/newren/git/actions/runs/13845548634/job/38743076293#step:4:1938 for an example of it running in CI and reporting possibly problematic assertions (sample output also included in the commit message of the middle commit in this series if you don't have access to view the link; I'm not sure what the rules on that are).

Changes since v1:

  • Tweaked commit message for patch 2
    Changes since v2:
  • Rename BUT_IF_NOT() -> ASSERT(). Didn't have a strong opinion on the set of alternatives Junio gave, so went with Taylor's small preference. If anyone has a strong preference here, I can pick a different alternative.
  • Fixed shell style issues (indentation, multi-line pipes, multiple lines with stderr redirects) in patch 2

cc: "brian m. carlson" [email protected]
cc: Elijah Newren [email protected]
cc: Taylor Blau [email protected]

@newren
Copy link
Author

newren commented Mar 14, 2025

/submit

Copy link

gitgitgadget bot commented Mar 14, 2025

Submitted as [email protected]

To fetch this version into FETCH_HEAD:

git fetch https://github.com/gitgitgadget/git/ pr-1881/newren/assertion-side-effects-v1

To fetch this version to local tag pr-1881/newren/assertion-side-effects-v1:

git fetch --no-tags https://github.com/gitgitgadget/git/ tag pr-1881/newren/assertion-side-effects-v1

Copy link

gitgitgadget bot commented Mar 14, 2025

User "brian m. carlson" <[email protected]> has been added to the cc: list.

Copy link

gitgitgadget bot commented Mar 14, 2025

User Elijah Newren <[email protected]> has been added to the cc: list.

@newren newren force-pushed the assertion-side-effects branch from 4c66803 to 20c763f Compare March 16, 2025 06:38
@newren
Copy link
Author

newren commented Mar 16, 2025

/submit

Copy link

gitgitgadget bot commented Mar 16, 2025

Submitted as [email protected]

To fetch this version into FETCH_HEAD:

git fetch https://github.com/gitgitgadget/git/ pr-1881/newren/assertion-side-effects-v2

To fetch this version to local tag pr-1881/newren/assertion-side-effects-v2:

git fetch --no-tags https://github.com/gitgitgadget/git/ tag pr-1881/newren/assertion-side-effects-v2

Copy link

gitgitgadget bot commented Mar 17, 2025

User Taylor Blau <[email protected]> has been added to the cc: list.

Copy link

gitgitgadget bot commented Mar 17, 2025

On the Git mailing list, Taylor Blau wrote (reply to this):

On Sun, Mar 16, 2025 at 06:41:59AM +0000, Elijah Newren via GitGitGadget wrote:
> Elijah Newren (3):
>   git-compat-util: introduce BUG_IF_NOT() macro
>   ci: add build checking for side-effects in assert() calls
>   treewide: replace assert() with BUG_IF_NOT() in special cases

Nice, this version looks great to me. I left a couple of notes
throughout, but they range from "could be done later" to idle
commentary. Thanks for working on this, and I'm sorry to have sent you
down such a rabbit hole ;-).

    Reviewed-by: Taylor Blau <[email protected]>

Thanks,
Taylor

newren added 3 commits March 17, 2025 23:50
Create a ASSERT() macro which is similar to assert(), but will not be
compiled out when NDEBUG is defined, and is thus safe to use even if its
argument has side-effects.

We will use this new macro in a subsequent commit to convert a few
existing assert() invocations to ASSERT().  In particular, we'll
convert the handful of invocations which cannot be proven to be free of
side effects with a simple compiler/linker hack.

Signed-off-by: Elijah Newren <[email protected]>
It is a big no-no to have side-effects in an assertion, because if the
assert() is compiled out, you don't get that side-effect, leading to the
code behaving differently.  That can be a large headache to debug.

We have roughly 566 assert() calls in our codebase (my grep might have
picked up things that aren't actually assert() calls, but most appeared
to be).  All but 9 of them can be determined by gcc to be free of side
effects with a clever redefine of assert() provided by Bruno De Fraine
(from
https://stackoverflow.com/questions/10593492/catching-assert-with-side-effects),
who upon request has graciously placed his two-liner into the public
domain without warranty of any kind.  The current 9 assert() calls
flagged by this clever redefinition of assert() appear to me to be free
of side effects as well, but are too complicated for a compiler/linker
to figure that since each assertion involves some kind of function call.
Add a CI job which will find and report these possibly problematic
assertions, and have the job suggest to the user that they replace these
with ASSERT() calls.

Example output from running:

```
ERROR: The compiler could not verify the following assert()
       calls are free of side-effects.  Please replace with
       ASSERT() calls.
/home/newren/floss/git/diffcore-rename.c:1409
	assert(!dir_rename_count || strmap_empty(dir_rename_count));
/home/newren/floss/git/merge-ort.c:1645
			assert(renames->deferred[side].trivial_merges_okay &&
			       !strset_contains(&renames->deferred[side].target_dirs,
						path));
/home/newren/floss/git/merge-ort.c:794
	assert(omittable_hint ==
	       (!starts_with(type_short_descriptions[type], "CONFLICT") &&
		!starts_with(type_short_descriptions[type], "ERROR")) ||
	       type == CONFLICT_DIR_RENAME_SUGGESTED);
/home/newren/floss/git/merge-recursive.c:1200
	assert(!merge_remote_util(commit));
/home/newren/floss/git/object-file.c:2709
	assert(would_convert_to_git_filter_fd(istate, path));
/home/newren/floss/git/parallel-checkout.c:280
	assert(is_eligible_for_parallel_checkout(pc_item->ce, &pc_item->ca));
/home/newren/floss/git/scalar.c:244
	assert(have_fsmonitor_support());
/home/newren/floss/git/scalar.c:254
	assert(have_fsmonitor_support());
/home/newren/floss/git/sequencer.c:4968
		assert(!(opts->signoff || opts->no_commit ||
			 opts->record_origin || should_edit(opts) ||
			 opts->committer_date_is_author_date ||
			 opts->ignore_date));
```

Note that if there are possibly problematic assertions, not necessarily
all of them will be shown in a single run, because the compiler errors
may include something like "ld: ... more undefined references to
`not_supposed_to_survive' follow" instead of listing each individually.
But in such cases, once you clean up a few that are shown in your first
run, subsequent runs will show (some of) the ones that remain, allowing
you to iteratively remove them all.

Helped-by: Bruno De Fraine <[email protected]>
Signed-off-by: Elijah Newren <[email protected]>
When the compiler/linker cannot verify that an assert() invocation is
free of side effects for us (e.g. because the assertion includes some
kind of function call), replace the use of assert() with ASSERT().

Signed-off-by: Elijah Newren <[email protected]>
@newren newren force-pushed the assertion-side-effects branch from 20c763f to 82b7344 Compare March 18, 2025 06:51
@newren
Copy link
Author

newren commented Mar 19, 2025

/submit

Copy link

gitgitgadget bot commented Mar 19, 2025

Submitted as [email protected]

To fetch this version into FETCH_HEAD:

git fetch https://github.com/gitgitgadget/git/ pr-1881/newren/assertion-side-effects-v3

To fetch this version to local tag pr-1881/newren/assertion-side-effects-v3:

git fetch --no-tags https://github.com/gitgitgadget/git/ tag pr-1881/newren/assertion-side-effects-v3

Copy link

gitgitgadget bot commented Mar 19, 2025

On the Git mailing list, Taylor Blau wrote (reply to this):

On Wed, Mar 19, 2025 at 04:22:55PM +0000, Elijah Newren via GitGitGadget wrote:
> Changes since v1:
>
>  * Tweaked commit message for patch 2 Changes since v2:
>  * Rename BUT_IF_NOT() -> ASSERT(). Didn't have a strong opinion on the set
>    of alternatives Junio gave, so went with Taylor's small preference. If
>    anyone has a strong preference here, I can pick a different alternative.
>  * Fixed shell style issues (indentation, multi-line pipes, multiple lines
>    with stderr redirects) in patch 2
>
> Elijah Newren (3):
>   git-compat-util: introduce ASSERT() macro
>   ci: add build checking for side-effects in assert() calls
>   treewide: replace assert() with ASSERT() in special cases
>
>  Makefile                      |  4 ++++
>  ci/check-unsafe-assertions.sh | 18 ++++++++++++++++++
>  ci/run-static-analysis.sh     |  2 ++
>  diffcore-rename.c             |  2 +-
>  git-compat-util.h             |  8 ++++++++
>  merge-ort.c                   |  4 ++--
>  merge-recursive.c             |  2 +-
>  object-file.c                 |  2 +-
>  parallel-checkout.c           |  2 +-
>  scalar.c                      |  4 ++--
>  sequencer.c                   |  2 +-
>  11 files changed, 41 insertions(+), 9 deletions(-)
>  create mode 100755 ci/check-unsafe-assertions.sh

Thanks, this version LGTM.

Thanks,
Taylor

Copy link

gitgitgadget bot commented Mar 21, 2025

This patch series was integrated into seen via git@d8e081e.

@gitgitgadget gitgitgadget bot added the seen label Mar 21, 2025
Copy link

gitgitgadget bot commented Mar 24, 2025

This branch is now known as en/assert-wo-side-effects.

Copy link

gitgitgadget bot commented Mar 24, 2025

This patch series was integrated into seen via git@2a069bb.

Copy link

gitgitgadget bot commented Mar 25, 2025

This patch series was integrated into seen via git@d733f65.

Copy link

gitgitgadget bot commented Mar 26, 2025

There was a status update in the "Cooking" section about the branch en/assert-wo-side-effects on the Git mailing list:

Ensure what we write in assert() does not have side effects,
and introduce ASSERT() macro to mark those that cannot be
mechanically checked for lack of side effects.

Will merge to 'next'.
source: <[email protected]>

Copy link

gitgitgadget bot commented Mar 28, 2025

This patch series was integrated into seen via git@d9993e8.

Copy link

gitgitgadget bot commented Mar 29, 2025

This patch series was integrated into seen via git@db1177d.

Copy link

gitgitgadget bot commented Mar 29, 2025

This patch series was integrated into next via git@de89bb3.

@gitgitgadget gitgitgadget bot added the next label Mar 29, 2025
Copy link

gitgitgadget bot commented Apr 7, 2025

There was a status update in the "Cooking" section about the branch en/assert-wo-side-effects on the Git mailing list:

Ensure what we write in assert() does not have side effects,
and introduce ASSERT() macro to mark those that cannot be
mechanically checked for lack of side effects.

Will merge to 'master'.
source: <[email protected]>

Copy link

gitgitgadget bot commented Apr 7, 2025

This patch series was integrated into seen via git@94c7f5e.

Copy link

gitgitgadget bot commented Apr 8, 2025

This patch series was integrated into seen via git@27a3607.

Copy link

gitgitgadget bot commented Apr 8, 2025

This patch series was integrated into seen via git@b97b360.

Copy link

gitgitgadget bot commented Apr 8, 2025

This patch series was integrated into master via git@b97b360.

Copy link

gitgitgadget bot commented Apr 8, 2025

This patch series was integrated into next via git@b97b360.

@gitgitgadget gitgitgadget bot added the master label Apr 8, 2025
@gitgitgadget gitgitgadget bot closed this Apr 8, 2025
Copy link

gitgitgadget bot commented Apr 8, 2025

Closed via b97b360.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant