[clang] [StaticAnalyzer] Fix state update in VisitObjCForCollectionStmt (PR #124477)
Ziqing Luo via cfe-commits
cfe-commits at lists.llvm.org
Thu Jan 30 12:42:48 PST 2025
ziqingluo-90 wrote:
> I wanted to push to your branch but for some reason I could not. Here is what I would have proposed to unblock this change:
>
> ```diff
> commit ab9670b613be2bdd802342f031bd5e3d20680925
> Author: Balazs Benics <benicsbalazs at gmail.com>
> Date: 2025.01.29 13:02:16
>
> Add a unittest demonstrating that we no longer crash
>
> diff --git a/clang/unittests/StaticAnalyzer/CMakeLists.txt b/clang/unittests/StaticAnalyzer/CMakeLists.txt
> index f5da86e54560..7b574bdf7cbc 100644
> --- a/clang/unittests/StaticAnalyzer/CMakeLists.txt
> +++ b/clang/unittests/StaticAnalyzer/CMakeLists.txt
> @@ -15,6 +15,7 @@ add_clang_unittest(StaticAnalysisTests
> IsCLibraryFunctionTest.cpp
> MemRegionDescriptiveNameTest.cpp
> NoStateChangeFuncVisitorTest.cpp
> + ObjCTest.cpp
> ParamRegionTest.cpp
> RangeSetTest.cpp
> RegisterCustomCheckersTest.cpp
> diff --git a/clang/unittests/StaticAnalyzer/ObjCTest.cpp b/clang/unittests/StaticAnalyzer/ObjCTest.cpp
> new file mode 100644
> index 000000000000..16cab5336813
> --- /dev/null
> +++ b/clang/unittests/StaticAnalyzer/ObjCTest.cpp
> @@ -0,0 +1,62 @@
> +//===----------------------------------------------------------------------===//
> +//
> +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
> +// See https://llvm.org/LICENSE.txt for license information.
> +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
> +//
> +//===----------------------------------------------------------------------===//
> +
> +#include "CheckerRegistration.h"
> +#include "clang/StaticAnalyzer/Core/Checker.h"
> +#include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
> +#include "clang/StaticAnalyzer/Core/PathSensitive/ProgramStateTrait.h"
> +#include "clang/StaticAnalyzer/Frontend/AnalysisConsumer.h"
> +#include "clang/StaticAnalyzer/Frontend/CheckerRegistry.h"
> +#include "gtest/gtest.h"
> +
> +using namespace clang;
> +using namespace ento;
> +
> +// Some dummy trait that we can mutate back and forth to force a new State.
> +REGISTER_TRAIT_WITH_PROGRAMSTATE(Flag, bool)
> +
> +namespace {
> +class FlipFlagOnCheckLocation : public Checker<check::Location> {
> +public:
> + // We make sure we alter the State every time we model a checkLocation event.
> + void checkLocation(SVal l, bool isLoad, const Stmt *S,
> + CheckerContext &C) const {
> + ProgramStateRef State = C.getState();
> + State = State->set<Flag>(!State->get<Flag>());
> + C.addTransition(State);
> + }
> +};
> +
> +void addFlagFlipperChecker(AnalysisASTConsumer &AnalysisConsumer,
> + AnalyzerOptions &AnOpts) {
> + AnOpts.CheckersAndPackages = {{"test.FlipFlagOnCheckLocation", true}};
> + AnalysisConsumer.AddCheckerRegistrationFn([](CheckerRegistry &Registry) {
> + Registry.addChecker<FlipFlagOnCheckLocation>("test.FlipFlagOnCheckLocation",
> + "Description", "");
> + });
> +}
> +
> +TEST(ObjCTest, CheckLocationEventsShouldMaterializeInObjCForCollectionStmts) {
> + // Previously, the `ExprEngine::hasMoreIteration` may fired an assertion
> + // because we forgot to handle correctly the resulting nodes of the
> + // check::Location callback for the ObjCForCollectionStmts.
> + // This caused inconsistencies in the graph and triggering the assertion.
> + std::string Diags;
> + EXPECT_TRUE(runCheckerOnCodeWithArgs<addFlagFlipperChecker>(
> + R"(
> + @class NSArray, NSDictionary, NSString;
> + extern void NSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
> + void entrypoint(NSArray *bowl) {
> + for (NSString *fruit in bowl) { // no-crash
> + NSLog(@"Fruit: %@", fruit);
> + }
> + })",
> + {"-x", "objective-c"}, Diags));
> +}
> +
> +} // namespace
> ```
>
> Feel free to tweak it, especially the comment as you know the most of this context. I checked and without your fix the test would crash, and after your fix it would work.
I wish I could see your reply earlier! It took me a while to reduce the downstream reproducer and come up a minimal test. This way of testing looks great. Maybe I should add this test anyway. Thank you!
https://github.com/llvm/llvm-project/pull/124477
More information about the cfe-commits
mailing list