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

static analyzer with ctu and z3 crashes 15.0.1 #58118

Open
dilyanpalauzov opened this issue Oct 3, 2022 · 23 comments
Open

static analyzer with ctu and z3 crashes 15.0.1 #58118

dilyanpalauzov opened this issue Oct 3, 2022 · 23 comments
Labels
clang:static analyzer crash Prefer [crash-on-valid] or [crash-on-invalid]

Comments

@dilyanpalauzov
Copy link

dilyanpalauzov commented Oct 3, 2022

With CodeChecker 6.20.0, self-compiled clang 15.0.1, enabled Z3 solver 4.11.2 on openldap branch OPENLDAP_REL_ENG_2_6, calling

> CodeChecker analyze ./compile_commands.json --ctu --z3=on --output ./reports

prints

[INFO 2022-10-03 10:00] - [13/622] clangsa analyzed base64.c successfully.                                                              [ERROR 2022-10-03 10:00] - Analyzing encode.c with clangsa CTU failed!                                                                  [ERROR 2022-10-03 10:00] -                                                                                                              
                                                                                                                                        
[ERROR 2022-10-03 10:00] -                                                                                                              
CTU loaded AST file: ast/git/openldap/libraries/liblber/bprint.c.ast 
CTU loaded AST file: ast/git/openldap/libraries/liblber/io.c.ast
PLEASE submit a bug report to https://github.com/llvm/llvm-project/issues/ and include the crash backtrace, preprocessed source, and associated run script. 
Stack dump:                                                                                                                             
0.      Program arguments: /usr/local/bin/clang-15 --analyze -Qunused-arguments -Xclang -analyzer-opt-analyze-headers -Xclang -analyzer-
output=plist-multi-file -o /git/openldap/reports/encode.c_clangsa_1bc933992a4816fdd22aca0f6cff381d.plist -Xclang -analyzer-config -Xclang expand-macros=true -Xclang -analyzer-checker=alpha.security.cert.env.InvalidPtr,alpha.security.cert.pos.34c,core.CallAndMessage,core.D
ivideZero,core.NonNullParamChecker,core.NullDereference,core.StackAddressEscape,core.UndefinedBinaryOperatorResult,core.VLASize,core.uninitialized.ArraySubscript,core.uninitialized.Assign,core.uninitialized.Branch,core.uninitialized.CapturedBlockVariable,core.uninitialized.UndefReturn,cplusplus.InnerPointer,cplusplus.Move,cplusplus.NewDelete,cplusplus.NewDeleteLeaks,cplusplus.PlacementNew,cplusplus.PureVirtualCall,cplusplus.StringChecker,deadcode.DeadStores,nullability.NullPassedToNonnull,nullability.NullReturnedFromNonnull,optin.cplusplus.UninitializedObject,optin.cplusplus.VirtualCall,optin.portability.UnixAPI,security.FloatLoopCounter,security.insecureAPI.UncheckedReturn,security.insecureAPI.getpw,security.insecureAPI.gets,security.insecureAPI.mkstemp,security.insecureAPI.mktemp,security.insecureAPI.rand,security.insecureAPI.vfork,unix.API,unix.Malloc,unix.MallocSizeof,unix.MismatchedDeallocator,unix.Vfork,unix.cstring.BadSizeArg,unix.cstring.NullArg,valist.CopyToSelf,valist.Uninitialized,valist.Unterminated -Xclang -analyzer-config -Xclang aggressive-binary-operation-
simplification=true -Xclang -analyzer-constraints=z3 -Xclang -analyzer-config -Xclang experimental-enable-naive-ctu-analysis=true -Xclang -analyzer-config -Xclang ctu-dir=/git/openldap/reports/ctu-dir/x86_64 -Xclang -analyzer-config -Xclang display-ctu-progress=true -x c 
--target=x86_64-pc-linux-gnu -std=gnu17 -O2 -I/git/openldap/include -I/git/openldap/include -DLBER_LIBRARY -fPIC -DPIC -isystem /usr/local/include -isystem /usr/include /git/openldap/libraries/liblber/encode.c
1.      <eof> parser at end of file
2.      While analyzing stack: 
        #0 Calling ber_put_bitstring at line 582
        #1 Calling ber_printf
3.      /git/openldap/libraries/liblber/encode.c:291:15: Error evaluating statement
4.      /git/openldap/libraries/liblber/encode.c:291:15: Error evaluating statement

[ERROR 2022-10-03 10:00] - 


[ERROR 2022-10-03 10:00] - 
CTU loaded AST file: ast/git/openldap/libraries/liblber/bprint.c.ast 
CTU loaded AST file: ast/git/openldap/libraries/liblber/io.c.ast
PLEASE submit a bug report to https://github.com/llvm/llvm-project/issues/ and include the crash backtrace, preprocessed source, and associated run script.
Stack dump:
0.      Program arguments: /usr/local/bin/clang-15 --analyze -Qunused-arguments -Xclang -analyzer-opt-analyze-headers -Xclang -analyzer-output=plist-multi-file -o /git/openldap/reports/encode.c_clangsa_1bc933992a4816fdd22aca0f6cff381d.plist -Xclang -analyzer-config -Xclang expand-macros=true -Xclang -analyzer-checker=alpha.security.cert.env.InvalidPtr,alpha.security.cert.pos.34c,core.CallAndMessage,core.DivideZero,core.NonNullParamChecker,core.NullDereference,core.StackAddressEscape,core.UndefinedBinaryOperatorResult,core.VLASize,core.uninitialized.ArraySubscript,core.uninitialized.Assign,core.uninitialized.Branch,core.uninitialized.CapturedBlockVariable,core.uninitialized.UndefReturn,cplusplus.InnerPointer,cplusplus.Move,cplusplus.NewDelete,cplusplus.NewDeleteLeaks,cplusplus.PlacementNew,cplusplus.PureVirtualCall,cplusplus.StringChecker,deadcode.DeadStores,nullability.NullPassedToNonnull,nullability.NullReturnedFromNonnull,optin.cplusplus.UninitializedObject,optin.cplusplus.VirtualCall,optin.portability.UnixAPI,security.FloatLoopCounter,security.insecureAPI.UncheckedReturn,security.insecureAPI.getpw,security.insecureAPI.gets,security.insecureAPI.mkstemp,security.insecureAPI.mktemp,security.insecureAPI.rand,security.insecureAPI.vfork,unix.API,unix.Malloc,unix.MallocSizeof,unix.MismatchedDeallocator,unix.Vfork,unix.cstring.BadSizeArg,unix.cstring.NullArg,valist.CopyToSelf,valist.Uninitialized,valist.Unterminated -Xclang -analyzer-config -Xclang aggressive-binary-operation-
simplification=true -Xclang -analyzer-constraints=z3 -Xclang -analyzer-config -Xclang experimental-enable-naive-ctu-analysis=true -Xclang -analyzer-config -Xclang ctu-dir=/git/openldap/reports/ctu-dir/x86_64 -Xclang -analyzer-config -Xclang display-ctu-progress=true -x c 
--target=x86_64-pc-linux-gnu -std=gnu17 -O2 -I/git/openldap/include -I/git/openldap/include -DLBER_LIBRARY -fPIC -DPIC -isystem /usr/local/include -isystem /usr/include /git/openldap/libraries/liblber/encode.c
1.      <eof> parser at end of file
2.      While analyzing stack: 
        #0 Calling ber_put_bitstring at line 582
        #1 Calling ber_printf
3.      /git/openldap/libraries/liblber/encode.c:291:15: Error evaluating statement
4.      /git/openldap/libraries/liblber/encode.c:291:15: Error evaluating statement

/git/openldap/libraries/liblber/encode.c is https://git.openldap.org/openldap/openldap/-/blob/OPENLDAP_REL_ENG_2_6/libraries/liblber/encode.c

Full backtrace, without debug symbols:

#0  0x00007fd9a16338ee in x86_64_fallback_frame_state (context=0x7ffcc69bffe0, fs=0x7ffcc69c00d0) at ./md-unwind-support.h:63           
        pc = 0x800b7818 <error: Cannot access memory at address 0x800b7818>                                                             
        sc = <optimized out>                                                                                                            
        new_cfa = <optimized out>                                                                                                       
        pc = <optimized out>                                                                                                            
        sc = <optimized out>                                                                                                            
        new_cfa = <optimized out>                                                                                                       
        uc_ = <optimized out>                                                                                                           
#1  uw_frame_state_for (context=context@entry=0x7ffcc69bffe0, fs=fs@entry=0x7ffcc69c00d0) at ../../../libgcc/unwind-dw2.c:1271          
        fde = 0x0
        cie = <optimized out>
        aug = <optimized out>
        insn = <optimized out>
        end = <optimized out>
#2  0x00007fd9a163564b in _Unwind_Backtrace (trace=0x7fd9a1534310 <backtrace_helper>, trace_argument=0x7ffcc69c0290)          [148/1909]
    at ../../../libgcc/unwind.inc:303
        fs = {
          regs = {
            reg = {{
                loc = {
                  reg = 0,
                  offset = 0,
                  exp = 0x0
                },
                how = REG_UNSAVED
              } <repeats 18 times>},
            prev = 0x0,
            cfa_offset = 0,
            cfa_reg = 0,
            cfa_exp = 0x0,
            cfa_how = CFA_UNSET
          },
          pc = 0x0,
          personality = 0x0,
          data_align = 0,
          code_align = 0,
          retaddr_column = 0,
          fde_encoding = 0 '\000',
          lsda_encoding = 0 '\000',
          saw_z = 0 '\000',
          signal_frame = 0 '\000',
          eh_ptr = 0x0
        }
        context = {
          reg = {0x7ffcc69c0510, 0x7ffcc69c0508, 0x7ffcc69c0518, 0x7ffcc69c0500, 0x7ffcc69c04f0, 0x7ffcc69c04e8, 0x7ffcc69c04f8, 
            0x7ffcc69c0520, 0x7ffcc69c04a8, 0x7ffcc69c04b0, 0x7ffcc69c04b8, 0x7ffcc69c04c0, 0x7ffcc69c04c8, 0x7ffcc69c04d0, 
            0x7ffcc69c04d8, 0x7ffcc69c04e0, 0x7ffcc69c0528, 0x0},
          cfa = 0x7ffcc69c08d8,
          ra = 0x800b7818,
          lsda = 0x0,
          bases = {
            tbase = 0x0,
            dbase = 0x0,
            func = 0x7fd9a146128f
          },
          flags = 13835058055282163712,
          version = 0,
          args_size = 0,
          by_value = '\000' <repeats 17 times>
        }
        code = <optimized out>
#3  0x00007fd9a15343f0 in __GI___backtrace (array=<optimized out>, size=256) at backtrace.c:79
        arg = {
          array = 0x7fd9a1be9780 <llvm::sys::PrintStackTrace(llvm::raw_ostream&, int)::StackTrace>,
          unwind_link = 0x7fd9a16174c0 <global>,
          cfa = 140723640599680,
          cnt = 4,
          size = 256
        }
#4  0x00007fd9a1b467e4 in PrintStackTraceSignalHandler(void*) () from /usr/local/bin/../lib/libLLVMSupport.so.15
No symbol table info available.
#5  0x00007fd9a1b44524 in llvm::sys::CleanupOnSignal(unsigned long) () from /usr/local/bin/../lib/libLLVMSupport.so.15
No symbol table info available.
#6  0x00007fd9a1a2a4a8 in CrashRecoverySignalHandler(int) () from /usr/local/bin/../lib/libLLVMSupport.so.15
No symbol table info available.
#7  <signal handler called>
No locals.
#8  0x00000000800b7818 in ?? ()
No symbol table info available.
#9  0x00007fd99cf81f05 in clang::ento::SValBuilder::makeSymbolVal(clang::ento::SymExpr const*) ()
   from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#10 0x00007fd99cfccddf in clang::ento::SMTConstraintManager::canReasonAbout(clang::ento::SVal) const ()
   from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#11 0x00007fd99cfc378b in clang::ento::SimpleConstraintManager::assumeAux(llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>, cla
ng::ento::NonLoc, bool) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#12 0x00007fd99cfc4003 in clang::ento::SimpleConstraintManager::assumeInternal(llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>
, clang::ento::DefinedSVal, bool) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#13 0x00007fd99cec2ea8 in std::pair<llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>, llvm::IntrusiveRefCntPtr<clang::ento::Pro
gramState const> > clang::ento::ConstraintManager::assumeDualImpl<clang::ento::ConstraintManager::assumeDual(llvm::IntrusiveRefCntPtr<cl
ang::ento::ProgramState const>, clang::ento::DefinedSVal)::{lambda(bool)#1}>(llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>&,
 clang::ento::ConstraintManager::assumeDual(llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>, clang::ento::DefinedSVal)::{lambd
a(bool)#1}&) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#14 0x00007fd99cec32bd in clang::ento::ConstraintManager::assumeDual(llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>, clang::e
nto::DefinedSVal) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#15 0x00007fd99cfdd965 in clang::ento::SValBuilder::evalIntegralCast(llvm::IntrusiveRefCntPtr<clang::ento::ProgramState const>, clang::e
nto::SVal, clang::QualType, clang::QualType) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#16 0x00007fd99cf1a380 in clang::ento::ExprEngine::VisitCast(clang::CastExpr const*, clang::Expr const*, clang::ento::ExplodedNode*, cla
ng::ento::ExplodedNodeSet&) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#17 0x00007fd99cf09aab in clang::ento::ExprEngine::Visit(clang::Stmt const*, clang::ento::ExplodedNode*, clang::ento::ExplodedNodeSet&)
    () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#18 0x00007fd99cf0b986 in clang::ento::ExprEngine::ProcessStmt(clang::Stmt const*, clang::ento::ExplodedNode*) ()
   from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#19 0x00007fd99cf0bb92 in clang::ento::ExprEngine::processCFGElement(clang::CFGElement, clang::ento::ExplodedNode*, unsigned int, clang:
:ento::NodeBuilderContext*) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#20 0x00007fd99cec916f in clang::ento::CoreEngine::dispatchWorkItem(clang::ento::ExplodedNode*, clang::ProgramPoint, clang::ent[47/1909]
stUnit const&) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#21 0x00007fd99cec933e in clang::ento::CoreEngine::ExecuteWorkList(clang::LocationContext const*, unsigned int, llvm::IntrusiveRefCntPtr
<clang::ento::ProgramState const>) () from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15
No symbol table info available.
#22 0x00007fd9a07e8a07 in (anonymous namespace)::AnalysisConsumer::HandleCode(clang::Decl*, unsigned int, clang::ento::ExprEngine::Inlin
ingModes, llvm::DenseSet<clang::Decl const*, llvm::DenseMapInfo<clang::Decl const*, void> >*) ()
   from /usr/local/bin/../lib/../lib/libclangStaticAnalyzerFrontend.so.15
No symbol table info available.
#23 0x00007fd9a07fc025 in (anonymous namespace)::AnalysisConsumer::HandleTranslationUnit(clang::ASTContext&) ()
   from /usr/local/bin/../lib/../lib/libclangStaticAnalyzerFrontend.so.15
No symbol table info available.
#24 0x00007fd99f47c1e9 in clang::ParseAST(clang::Sema&, bool, bool) () from /usr/local/bin/../lib/../lib/libclangParse.so.15
No symbol table info available.
#25 0x00007fd9a3bbaa49 in clang::FrontendAction::Execute() () from /usr/local/bin/../lib/libclangFrontend.so.15
No symbol table info available.
#26 0x00007fd9a3b3abab in clang::CompilerInstance::ExecuteAction(clang::FrontendAction&) ()
   from /usr/local/bin/../lib/libclangFrontend.so.15
No symbol table info available.
#27 0x00007fd9a5dc7db3 in clang::ExecuteCompilerInvocation(clang::CompilerInstance*) ()
   from /usr/local/bin/../lib/libclangFrontendTool.so.15
No symbol table info available.
#28 0x000000000041540d in cc1_main(llvm::ArrayRef<char const*>, char const*, void*) ()
No symbol table info available.
#29 0x000000000040efac in ExecuteCC1Tool(llvm::SmallVectorImpl<char const*>&) ()                                                        
No symbol table info available.                                                                                                         
#30 0x00007fd9a38a4b25 in void llvm::function_ref<void ()>::callback_fn<clang::driver::CC1Command::Execute(llvm::ArrayRef<llvm::Optional
<llvm::StringRef> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*, bool*) const::{lambda()#1}>(long)
 () from /usr/local/bin/../lib/libclangDriver.so.15
No symbol table info available.
#31 0x00007fd9a1a2a5a3 in llvm::CrashRecoveryContext::RunSafely(llvm::function_ref<void ()>) ()
   from /usr/local/bin/../lib/libLLVMSupport.so.15
No symbol table info available.
#32 0x00007fd9a38a4eb9 in clang::driver::CC1Command::Execute(llvm::ArrayRef<llvm::Optional<llvm::StringRef> >, std::__cxx11::basic_strin
g<char, std::char_traits<char>, std::allocator<char> >*, bool*) const [clone .part.0] ()
   from /usr/local/bin/../lib/libclangDriver.so.15
No symbol table info available.
#33 0x00007fd9a386e776 in clang::driver::Compilation::ExecuteCommand(clang::driver::Command const&, clang::driver::Command const*&, bool
) const () from /usr/local/bin/../lib/libclangDriver.so.15
No symbol table info available.
#34 0x00007fd9a386f16d in clang::driver::Compilation::ExecuteJobs(clang::driver::JobList const&, llvm::SmallVectorImpl<std::pair<int, cl
ang::driver::Command const*> >&, bool) const () from /usr/local/bin/../lib/libclangDriver.so.15
No symbol table info available.
#35 0x00007fd9a3880f1c in clang::driver::Driver::ExecuteCompilation(clang::driver::Compilation&, llvm::SmallVectorImpl<std::pair<int, cl
ang::driver::Command const*> >&) () from /usr/local/bin/../lib/libclangDriver.so.15
No symbol table info available.
#36 0x000000000041334d in clang_main(int, char**) ()
No symbol table info available.
#37 0x00007fd9a144d1a7 in __libc_start_call_main (main=main@entry=0x40e4a0 <main>, argc=argc@entry=48, argv=argv@entry=0x7ffcc69c4238)
    at ../sysdeps/nptl/libc_start_call_main.h:58
        self = <optimized out>
        result = <optimized out>
        unwind_buf = {
          cancel_jmp_buf = {{
              jmp_buf = {0, -7990426904508217125, 140723640615480, 4252832, 140572830297152, 4343240, 7991004392999373019, 
                7975610616575314139},
              mask_was_saved = 0
            }},
          priv = {
            pad = {0x0, 0x0, 0x7fd9a99d6c40 <_rtld_local_ro>, 0x4245c8},
            data = {
              prev = 0x0,
              cleanup = 0x0,
              canceltype = -1449300928
            }
          }
        }
        not_first_call = <optimized out>
#38 0x00007fd9a144d258 in __libc_start_main_impl (main=0x40e4a0 <main>, argc=48, argv=0x7ffcc69c4238, init=<optimized out>, 
    fini=<optimized out>, rtld_fini=<optimized out>, stack_end=0x7ffcc69c4228) at ../csu/libc-start.c:409
No locals.
#39 0x000000000040e4d1 in _start () at ../sysdeps/x86_64/start.S:116
No locals.

bprint.c.ast.gz
io.c.ast.gz

@llvmbot
Copy link
Member

llvmbot commented Oct 3, 2022

@llvm/issue-subscribers-clang-static-analyzer

@steakhal
Copy link
Contributor

The z3 constraint solver is not supported.
z3 refutation is supported though.

Ask CodeChecker about the --z3=on flag. I think they should simply drop it.

Can I close this?

@dilyanpalauzov
Copy link
Author

dilyanpalauzov commented Oct 13, 2022

I filled Ericsson/codechecker#3757 about removing Z3 constraint solver from code checker.

The problem report is, that clang crashes. It is not codechecker’s fault, that clang crashes. The solution would be to fix clang, so that it does not crash.

@dilyanpalauzov
Copy link
Author

If the z3 constraint solver is not supported, and is activated by -analyze -analyzer-constraints=z3, why do the files clang/test/Analysis/analyzer_test.py, clang/test/Analysis/missing-z3-nocrash.c, clang/test/Analysis/z3/pretty-dump.c, clang/test/Analysis/z3/enabled.c, and clang/test/Analysis/z3/D83660.c in llvm-project-15.0.2.src contain -analyzer-constraints=z3?

@ryao
Copy link

ryao commented Oct 17, 2022

@mikhailramalho Is there any chance you could take a look?

@dilyanpalauzov
Copy link
Author

dilyanpalauzov commented Oct 17, 2022

llvm-project-15.0.2.src/clang/lib/Frontend/CompilerInvocation.cpp:ParseAnalyzerArgs() contains:

    if (Value == NumConstraints) {
      Diags.Report(diag::err_drv_invalid_value)
        << A->getAsString(Args) << Name;
    } else {
#ifndef LLVM_WITH_Z3
      if (Value == AnalysisConstraints::Z3ConstraintsModel) {
        Diags.Report(diag::err_analyzer_not_built_with_z3);
      }
#endif // LLVM_WITH_Z3                                                                                                                  
      Opts.AnalysisConstraintsOpt = Value;
    }

I expect, when I pass -analyzer-constraints=z3 to clang, not compiled with Z3 to get the above “not build with z3 error”.

With Clang 14.0.5 (I have not checked it source code, if the above function is different), when it is not compiled with Z3, calling:

$ clang --analyze -Xclang -analyzer-constraints=z3 a.c
fatal error: error in backend: LLVM was not compiled with Z3 support, rebuild with -DLLVM_ENABLE_Z3_SOLVER=ON
PLEASE submit a bug report to https://github.com/llvm/llvm-project/issues/ and include the crash backtrace, preprocessed source, and associated run script.                                                 
Stack dump:                                                                                                                             
0.      Program arguments: clang --analyze -Xclang -analyzer-constraints=z3 a.c              
1.      <eof> parser at end of file                                                                                                     
 #0 0x00007fc682c0c975 llvm::sys::PrintStackTrace(llvm::raw_ostream&, int) (/lib64/libLLVM-14.so+0xc0c975)             
 #1 0x00007fc682c0cbfb (/lib64/libLLVM-14.so+0xc0cbfb)                                                                                  
 #2 0x00007fc682c0a2b4 llvm::sys::RunSignalHandlers() (/lib64/libLLVM-14.so+0xc0a2b4)
 #3 0x00007fc682c0be8b llvm::sys::CleanupOnSignal(unsigned long) (/lib64/libLLVM-14.so+0xc0be8b)
 #4 0x00007fc682b1a439 (/lib64/libLLVM-14.so+0xb1a439)                                                                                  
 #5 0x00007fc682b1a342 (/lib64/libLLVM-14.so+0xb1a342)      
 #6 0x00007fc682c067be llvm::sys::Process::Exit(int, bool) (/lib64/libLLVM-14.so+0xc067be)                                               #7 0x00005561c5e92347 (/usr/bin/clang-14+0x15347)
 #8 0x00007fc682b2c1b1 llvm::report_fatal_error(llvm::Twine const&, bool) (/lib64/libLLVM-14.so+0xb2c1b1)                                #9 0x00007fc682b2c07a (/lib64/libLLVM-14.so+0xb2c07a)                                                                                  
#10 0x00007fc682bf4946 (/lib64/libLLVM-14.so+0xbf4946)                                                                                  #11 0x00007fc68abc512a clang::ento::CreateZ3ConstraintManager(clang::ento::ProgramStateManager&, clang::ento::ExprEngine*) (/lib64/libclang-cpp.so.14+0x25c512a)                       
#12 0x00007fc68ab73d1f clang::ento::ProgramStateManager::ProgramStateManager(clang::ASTContext&, std::unique_ptr<clang::ento::StoreManager, std::default_delete<clang::ento::StoreManager> > (*)(clang::ento::ProgramStateManager&), std::unique_ptr<clang::ento::ConstraintManager, std::default_delete<clang::ento::ConstraintManager> > (*)(clang::ento::ProgramStateManager&, clang::ento::ExprEngine*), llvm::BumpPtrAllocatorImpl<llvm::MallocAllocator, 4096ul, 4096ul, 128ul>&, clang::ento::ExprEngine*) (/lib64/libclang-cpp.so.14+0x2573d1f)
#13 0x00007fc68ab1df92 clang::ento::ExprEngine::ExprEngine(clang::cross_tu::CrossTranslationUnitContext&, clang::ento::AnalysisManager&, llvm::DenseSet<clang::Decl const*, llvm::DenseMapInfo<clang::Decl const*, void> >*, clang::ento::FunctionSummariesTy*, clang::ento::ExprEngine::InliningModes) (/lib64/libclang-cpp.so.14+0x251df92)
#14 0x00007fc68aedc1d4 (/lib64/libclang-cpp.so.14+0x28dc1d4)
#15 0x00007fc68aeeb2cf (/lib64/libclang-cpp.so.14+0x28eb2cf)
#16 0x00007fc68aeebd63 (/lib64/libclang-cpp.so.14+0x28ebd63)
#17 0x00007fc688ff26c9 clang::ParseAST(clang::Sema&, bool, bool) (/lib64/libclang-cpp.so.14+0x9f26c9)
#18 0x00007fc68a8e15f9 clang::FrontendAction::Execute() (/lib64/libclang-cpp.so.14+0x22e15f9)
#19 0x00007fc68a8718a9 clang::CompilerInstance::ExecuteAction(clang::FrontendAction&) (/lib64/libclang-cpp.so.14+0x22718a9)
#20 0x00007fc68a94de5c clang::ExecuteCompilerInvocation(clang::CompilerInstance*) (/lib64/libclang-cpp.so.14+0x234de5c)
#21 0x00005561c5e935ee cc1_main(llvm::ArrayRef<char const*>, char const*, void*) (/usr/bin/clang-14+0x165ee)
#22 0x00005561c5e90623 (/usr/bin/clang-14+0x13623)
#23 0x00007fc68a57d1c9 (/lib64/libclang-cpp.so.14+0x1f7d1c9)
#24 0x00007fc682b1a315 llvm::CrashRecoveryContext::RunSafely(llvm::function_ref<void ()>) (/lib64/libLLVM-14.so+0xb1a315)
#25 0x00007fc68a57da31 (/lib64/libclang-cpp.so.14+0x1f7da31)
#26 0x00007fc68a55199b clang::driver::Compilation::ExecuteCommand(clang::driver::Command const&, clang::driver::Command const*&) const (/lib64/libclang-cpp.so.14+0x1f5199b)
#27 0x00007fc68a55213e clang::driver::Compilation::ExecuteJobs(clang::driver::JobList const&, llvm::SmallVectorImpl<std::pair<int, clang::driver::Command const*> >&) const (/lib64/libclang-cpp.so.14+0x1f5213e)
#28 0x00007fc68a55c5c2 clang::driver::Driver::ExecuteCompilation(clang::driver::Compilation&, llvm::SmallVectorImpl<std::pair<int, clang::driver::Command const*> >&) (/lib64/libclang-cpp.so.14+0x1f5c5c2)
#29 0x00005561c5e8da02 main (/usr/bin/clang-14+0x10a02)
#30 0x00007fc681829550 __libc_start_call_main (/lib64/libc.so.6+0x29550)
#31 0x00007fc681829609 __libc_start_main@GLIBC_2.2.5 (/lib64/libc.so.6+0x29609)
#32 0x00005561c5e8faa5 _start (/usr/bin/clang-14+0x12aa5)

@dilyanpalauzov
Copy link
Author

Slide 27 of https://llvm.org/devmtg/2020-09/slides/Using_the_clang_static_ananalyzer_to_find_bugs.pdf contains:

Example of unhandled bitwise operations
▪ This program is safe, albeit brittle
1: unsigned int func(unsigned int a) {
2:   unsigned int *z = 0;
3:   if ((a & 1) && ((a & 1) ^1))
4:     return *z; // unreachable
5:   return 0;
6: }
$ clang --analyze test.cpp
test.cpp:5:16: warning: Dereference of null pointer (loaded from variable 'z') [core.NullDereference]
return *z;
^~
1 warning generated.
$ clang --analyze -Xclang -analyzer-config -Xclang crosscheck-with-z3=true test.cpp # Z3 Refutation, preferred

$ clang --analyze -Xclang -analyzer-constraints=z3 func.c # Z3 constraint manager, slower

When I try with b.c:

#include <stdio.h>

unsigned int func(unsigned int a) {
  unsigned int *z = 0;
  if ((a & 1) && ((a & 1) ^1))
    return *z; // unreachable
  return 0;
}

int main() {
  printf("%i\n", func(7));
}

all of

$ clang --analyzer-output text --analyze b.c
$ clang --analyzer-output text --analyze -Xclang -analyzer-config -Xclang crosscheck-with-z3=true b.c
$ clang --analyzer-output text --analyze -Xclang -analyzer-constraints=z3 b.c

produce no output with clang 15.0.1.

@balazs-benics-sonarsource

If the z3 constraint solver is not supported, and is activated by -analyze -analyzer-constraints=z3, why do the files clang/test/Analysis/analyzer_test.py, clang/test/Analysis/missing-z3-nocrash.c, clang/test/Analysis/z3/pretty-dump.c, clang/test/Analysis/z3/enabled.c, and clang/test/Analysis/z3/D83660.c in llvm-project-15.0.2.src contain -analyzer-constraints=z3?

I believe, at some point, there was a push for having this, but I don't know anyone maintaining it now.
There are no build bots running any of those tests, I believe.
One can play with the (huge number of) crashes on the testsuite by running ./build/bin/llvm-lit --param USE_Z3_SOLVER=1 --show-unsupported --show-xfail clang/test/Analysis. This will not only run the tests by the original solver, but also with the z3-based solver.

I expect, when I pass -analyzer-constraints=z3 to clang, not compiled with Z3 to get the above “not build with z3 error”.
With Clang 14.0.5 (I have not checked it source code, if the above function is different), when it is not compiled with Z3, calling:

Yes, previously the detection of the availability Z3 involved crashing clang. That was fixed by https://reviews.llvm.org/D120325

Slide 27 of https://llvm.org/devmtg/2020-09/slides/Using_the_clang_static_ananalyzer_to_find_bugs.pdf contains:
clang --analyze -Xclang -analyzer-constraints=z3 func.c

-analyzer-constraints=z3 is not only slower but also crashes a lot of times.

Could you elaborate on your last example? You said all of [...] produce no output with clang 15.0.1.
Well, the range-based solver is constantly being improved, making it smarter and smarter by the time.
It could be the case that in clang-15.0.1 the solver is smart enough to figure out that branch is unreachable.
However, by looking at the bitwise operations, I'm actually really surprised that we are not falling for it.

About the other two invocations:
Refutation is much cheaper than the z3-based solver, hence that brings value for production use. That is being actively maintained by the CodeChecker team AFAIK. So, I'm not surprised that we don't crash on that code, nor that we don't have a FP for it.
About the z3-based solver (-analyzer-constraints=z3), that is not maintained, and it could be the case that it happens to work for this example. Not crashing nor producing FP.

@dilyanpalauzov
Copy link
Author

Could you elaborate on your last example? You said all of [...] produce no output with clang 15.0.1.
Well, the range-based solver is constantly being improved, making it smarter and smarter by the time.
It could be the case that in clang-15.0.1 the solver is smart enough to figure out that branch is unreachable.
However, by looking at the bitwise operations, I'm actually really surprised that we are not falling for it.

I mean what I said. The slide 27 above says that for the above input clang shall generate error (return *z dereferences NULL), when invoked without any form of Z3, but when I try it there is no error (the code is always detected to be unreachable, or for another reason).

@ryao
Copy link

ryao commented Oct 26, 2022

That is odd. When I tried it last week, it was a false positive as reported in the slides. I believe that I tested against 15.0.2.

@balazs-benics-sonarsource
Copy link

balazs-benics-sonarsource commented Oct 26, 2022

I figured out what's going on.

  1. We have a bug that we miss the null pointer dereference of *z if that not used (e.g. by some subexpression or initializes some variable) You can read more about why here. We should treat return statements as 'reads', we know about this.
  2. Our range-based solver is now smart enough to deal with the bit-fiddling stuff. You can see that the value of the expressions is known to be constants, hence we can constant-fold them into a concrete decision that the branch should not be taken.
    https://godbolt.org/z/57qqs5sKx

This range-based solver improvement was likely a recent improvement, which at the time of the mentioned presentation was not yet done.

EDIT: my first point might not be accurate, it seems like in return statements we detect null-derefs, as we should. I must have mixed this issue with something else.

@dilyanpalauzov
Copy link
Author

Reading the previous comment, including the EDIT:, my understanding is that clangsa has a bug missing null pointer dereference of *z, and at the same time an improvement, detecting better the false messages.

Can you provide an example input, b.c (and possilbly other files), so that

$ clang --analyzer-output text --analyze b.c
$ clang --analyzer-output text --analyze -Xclang -analyzer-config -Xclang crosscheck-with-z3=true b.c
$ clang --analyzer-output text --analyze -Xclang -analyzer-constraints=z3 b.c

produces three different results?

@balazs-benics-sonarsource

Can you provide an example input, b.c (and possilbly other files), so that
$ clang --analyzer-output text --analyze b.c
$ clang --analyzer-output text --analyze -Xclang -analyzer-config -Xclang crosscheck-with-z3=true b.c
$ clang --analyzer-output text --analyze -Xclang -analyzer-constraints=z3 b.c
produces three different results?

It seems like the example is from clang/test/Analysis/z3-crosscheck.c:

int foo(int x) 
{
  int *z = 0;
  if ((x & 1) && ((x & 1) ^ 1))
#ifdef NO_CROSSCHECK
      return *z; // expected-warning {{Dereference of null pointer (loaded from variable 'z')}}
#else
      return *z; // no-warning
#endif
  return 0;
}

Any by reading that one might think that the range-based solver should report there a null dereference, while the z3 cross-checked invocation should not. Unfortunately, the tests with // REQUIRES: z3 are never run. The reason why they are never run, relates to the broken z3-based solver implementation. You can read more about that at D83677.
Since these tests are never run, they can get outdated; which probably happened with this one.

Coming back to your question, the difference between the three invocations.
Let's have a look at this example:

void func(int a, int b) {
  assert(a >= 1 && a <= 100);
  assert(b >= 0 && b <= 100);

  if (a*a == b*b && b == 0) {
    // should be unreachable
    clang_analyzer_warnIfReached();
  }
  clang_analyzer_warnIfReached();
}
  1. The range-based solver (right now) cannot conclude that the a*a == b*b && b == 0 is always false, hence it will reach the true branch.
  2. The range-based solver with z3 cross-checking, it would still visit the seemingly feasible branch, find and construct the bugreport for the true-branch, but prior emitting the issue to the user we collect the constraints and create a SAT model for it which we pass to Z3 to see if there are contradictions; If there are, that means that the bugpath is infeasible, hence we discard the bug report without emitting it. The important thing is that we did spend time analysing an infeasible program path.
  3. The z3-based solver could (and probably does) recognize that the true-branch is infeasible, so we simply don't spend time on analysing it - we just skip it. In theory, this could save analysis time, but in practice calling z3 for checking each condition is way to expensive; and this is why it runs so slow.

The bottom line is that, z3 crosscheck should have the benefit of the z3-based solver, without paying the price for calling z3 at each step during the analysis - only at the bug report construction.

@haoNoQ
Copy link
Collaborator

haoNoQ commented Oct 27, 2022

I filled Ericsson/codechecker#3757 about removing Z3 constraint solver from code checker.

The problem report is, that clang crashes. It is not codechecker’s fault, that clang crashes. The solution would be to fix clang, so that it does not crash.

If the z3 constraint solver is not supported, and is activated by -analyze -analyzer-constraints=z3, why do the files clang/test/Analysis/analyzer_test.py, clang/test/Analysis/missing-z3-nocrash.c, clang/test/Analysis/z3/pretty-dump.c, clang/test/Analysis/z3/enabled.c, and clang/test/Analysis/z3/D83660.c in llvm-project-15.0.2.src contain -analyzer-constraints=z3?

I want to chime in real quick about this confusion. I think your bug report is valuable, but the feature is indeed not supported, so we aren't likely to address it at the same rate as other crashes of this kind in supported use cases.

In case of LLVM, the word "supported" is not a synonym of "exists" or "implemented" or "available". The way LLVM is developed (which is very different from most open-source projects), every feature is supposed to be developed under a runtime flag. LLVM does not have feature branches. Developer policy explicitly forbids making feature branches. This leads to the situation where LLVM and Clang provide a lot of features that are available in the shipped binaries, covered with tests (because everything needs to be covered with tests), but not actually considered ready for everyday use aka "supported".

Static analysis with Z3 is one of such features. It was a valuable experiment which could have been successful in replacing the default constraint solver, it turned out to be unsuccessful (too slow), we're not removing it because it can be useful for gathering more experimental data for future research on constraint solvers (as this is still a difficult open problem), but we never claimed (or at least shouldn't have claimed) that we actually support it for any practical purposes.

When it comes to Clang, anything behind clang -cc1 or behind -Xclang isn't a supported interface. Only the Clang Driver is supported (alongside a few alternative drivers such as clang-cl).

With the static analyzer it's a bit of a mess because some actual features like -enable-checker don't have corresponding driver flags; they totally should have them. That said, such manual use of the static analyzer is also not recommended, the actual supported use case is with a GUI such as CodeChecker or scan-build or an IDE that integrates the static analyzer. This is because text output is inevitably terrible, you can't have basic quality of life in a non-visual tool.

So this presentation you're quoting, it just tells developers of the static analyzer how to access these guts of the implementation and experiment with them. This is absolutely not how users should use the static analyzer in their everyday life. I definitely think @vabridgers should have placed disclaimers about this all over the place! In other words, if you're not actively interested in addressing such crashes yourself, you're not supposed to touch these options.

So, this isn't about what you can or cannot do, this is about who's responsible for the outcome. In this sense, even though the compiler is crashing, this isn't a bug in the compiler. It's a bug in the interface that gives you access to a feature that doesn't exist. It becomes a bug in clang when we actually announce that it is supported, provide a proper user interface (CLI or GUI) to access it, or maybe make this the default behavior. But without all that, it's just a glorified never-shipped feature branch.

Yes, it's valuable to know that the feature has crashes on top of the slowdown. If somebody wants to use that feature, they have to worry not only about slowdowns, but also crashes, they'll have to recognize that the feature isn't fully implemented, and they'll have to work on its stability.

If this feature causes more confusion of this kind, we may choose to remove it entirely. Ideally, LLVM "branches" aren't supposed to be long-living. Maybe we should purge them more aggressively just to avoid confusion. For now it doesn't seem like it though, most of the time there's value in keeping that feature committed, especially in cases like this when it doesn't cost us anything, given that this feature mostly lives behind a compile-time cmake flag.

@whisperity
Copy link
Member

given that this feature mostly lives behind a compile-time cmake flag.

The problem here is that, to the best of my understanding, both the Z3 as-a-constraint-solver and Z3 as-refutation live behind the same CMake flag, namely the one that builds LLVM with Z3 as a DSO to be loaded. So if I compile Clang with Z3 support, I will get both the (in CodeChecker parlance) --z3-refutation and --z3(-as-a-constraint-solver) options.


Either way, I've merged a patch to CodeChecker which emphasises that Z3 (as-a-constraint-solver) is a massively experimental and crash-laden feature. But there could be the chance that there are vendors who supply a build of Clang that has certain checks that rely on Z3 (as-a-constraint-solver) being available and those vendors take care of making sure their downstream checks do not crash.

The default behaviour (from CodeChecker's perspective) is: range-based constraint solver + Z3 refutation (if available).

@ryao
Copy link

ryao commented Oct 27, 2022

The bottom line is that, z3 crosscheck should have the benefit of the z3-based solver, without paying the price for calling z3 at each step during the analysis - only at the bug report construction.

Are there no false negatives from the range based solver that z3 refution would have no chance to correct since it is only run on positive results?

@balazs-benics-sonarsource

The bottom line is that, z3 crosscheck should have the benefit of the z3-based solver, without paying the price for calling z3 at each step during the analysis - only at the bug report construction.

Are there no false negatives from the range based solver that z3 refution would have no chance to correct since it is only run on positive results?

There could be false-negatives with both the range-based and the z3-based solver. During the analysis we might have hard-coded models for functions, where if we make a typo like this, then we will mismodel it and potentially cause FPs and FNs like. But there could be other situations, like the modeling of casts, and its shortcomings which are well demonstrated by D103096.
Speaking of false-negatives, z3 refutation won't help with them, that's for sure.

@ryao
Copy link

ryao commented Oct 27, 2022

The bottom line is that, z3 crosscheck should have the benefit of the z3-based solver, without paying the price for calling z3 at each step during the analysis - only at the bug report construction.

Are there no false negatives from the range based solver that z3 refution would have no chance to correct since it is only run on positive results?

There could be false-negatives with both the range-based and the z3-based solver. During the analysis we might have hard-coded models for functions, where if we make a typo like this, then we will mismodel it and potentially cause FPs and FNs like. But there could be other situations, like the modeling of casts, and its shortcomings which are well demonstrated by D103096. Speaking of false-negatives, z3 refutation won't help with them, that's for sure.

z3 refution would not help with false negatives since it is restricted to positive results. However, i was under the impression that the z3-based solver could from my read of this:

Speed comes at the expense of precision -- symbolic analysis does not handle some arithmetic and bitwise operations. Z3 can compensate for some of these shortcoming.

https://llvm.org/devmtg/2020-09/slides/Using_the_clang_static_ananalyzer_to_find_bugs.pdf

Because of that, I have been looking into adopting z3-based solver scans in OpenZFS alongside the regular scans with z3 refution as a result of that. However, there are two main drawbacks:

  1. There are numerous crashes that reduce the number of reports that we receive from the z3 solver. Unfortunately, there is a segfault in the llvm-symbolizer that keeps me from getting good stack traces. I plan to investigate this some more as my time allows over the coming weeks (as while the crash issue is something others have too, they do not have the bad stack trace issue and I suspect that the way I build LLVM is responsible). The crash issue is something that I hope the wider LLVM community would be able to help solve with the aid of good bug reports, although if no one tackles it, I might end up trying to debug it myself at some point. That would be in the somewhat distant future since the OpenZFS project has me extremely busy already and this is just one of several approaches that I want to take in parallel to perform static analysis on the project.
  2. The computational requirements make this take days to run on my current development hardware, which is impractical since I cannot dedicate days of compute time to it on that. However, this is a solvable problem. The OpenZFS project is used in a number of large scale deployments and as a result, its contributors can obtain access to more beefy hardware to run the Z3-based solver just by asking the right user(s). I have already reached out to someone for access to more capable hardware.

By the way, @balazs-benics-sonarsource, I notice that you are related to sonarsource. While it is offtopic here, one of the things I would like to do is adopt sonarcloud scans for OpenZFS, but the requirement that I give access to SonarCloud to act on my behalf without any explanation of what this meant has blocked me from doing it. I tried emailing sonarsource to ask for more information, but Evelyn Martinez sent me back a boilerplate email that had nothing to do with my inquiry (I asked about the security implications of adopting SonarCloud in an OSS project and received an email telling me that I could not get an evaluation for SonarQube).

A number of people trust me to safeguard my github account against compromise and it only takes a single incident to destroy people's trust, so I hope that it is understandable that I am very cautious about granting third parties permission to act on my behalf. The issue could be somewhat mitigated by making another github account, but if I recall, that is against Github's ToS, so I cannot do that.

Would you be willing to put me in contact with someone who could explain the security implications of signing up with sonarcloud (e.g. to what extent can a bad actor cause harm to my github projects if sonarcloud is compromised) and how I can mitigate them as much as possible? If you are willing to help, you should be able to find my email in my github profile, so we could take this conversation to email to avoid derailing this issue.

@balazs-benics-sonarsource
  1. There are numerous crashes that reduce the number of reports that we receive from the z3 solver. Unfortunately, there is a segfault in the llvm-symbolizer that keeps me from getting good stack traces. I plan to investigate this some more as my time allows over the coming weeks (as while the crash issue is something others have too, they do not have the bad stack trace issue and I suspect that the way I build LLVM is responsible). The crash issue is something that I hope the wider LLVM community would be able to help solve with the aid of good bug reports, although if no one tackles it, I might end up trying to debug it myself at some point.

For bugreports, the best would be to have some catchy example. It could be semi-automatically generated by creduce under good hands. However, a stack-trace, platform, z3 version, llvm version (commit hash), and the preprocessed file should be enough to file and resolve any issues.

The best I can do for z3-based solver issues is to submit a test case to the repo so that we will have a reminder about the crashing case when we decide to resolve those crashes. Of course, for this, we would need a reduced reproducer. Feel free to contact me if you need help with this.

  1. The computational requirements make this take days to run on my current development hardware, which is impractical since I cannot dedicate days of compute time to it on that. However, this is a solvable problem. [...]

Interesting and motivating to see.

[...] you should be able to find my email in my github profile, so we could take this conversation to email to avoid derailing this issue.

Done.

@ryao
Copy link

ryao commented Dec 16, 2022

#9  0x00007fd99cf81f05 in clang::ento::SValBuilder::makeSymbolVal(clang::ento::SymExpr const*) ()
   from /usr/local/bin/../lib/../lib/../lib/libclangStaticAnalyzerCore.so.15

This calls another function that has some issue that causes a signal that triggers CrashRecoverySignalHandler(). Unfortunately, on my machine, things crash in a way that prevent me from seeing any symbols in backtraces, so this is very difficult to examine for me.

That said, this appears to be in clang/include/clang/StaticAnalyzer/Core/PathSensitive/SValBuilder.h:

  DefinedSVal makeSymbolVal(SymbolRef Sym) {
    if (Loc::isLocType(Sym->getType()))
      return makeLoc(Sym);
    return nonloc::SymbolVal(Sym);
  }

Presumably, a runtime issue happened in one of the functions called in that function body.

I tried looking through coverity reports to see if the issue was reported there, but nothing looked related. For my sanity, I had coverity only show me defect reports whose file system paths that matched */StaticAnalyzer/Core/*. If the issue really was caught by Coverity, but its file path does not match that, then I would have missed it.

Tangentially, defect report CID-1490697 looked very interesting. What is interesting about it is that it says SSE->getRHS() is passed to a function argument named lhs and SSE->getLHS() is passed to a function argument named rhs. That kind of thing is usually a bug, but in rare circumstances, it could be intentional. Usually, those involve test cases that intentionally write incorrect code to check how it is handled, but I am not sure that there is no reason to ever do it. I am not familiar enough with the code to be certain if the report describes an actual bug and since it seems unrelated to this, I did not look at it very long.

This code is also difficult for me to navigate. My usual tool for navigating codebases is cscope and it usually is okay on C++ code, but it cannot find a global definition for SymbolRef. Looking at all "C symbols", it finds a struct named SymbolRef, but C++ structures do not have member functions, so that cannot be it unless the C++ standard changed to allow them since I last accquinted myself with C++. Searching for an implementation for getType() predictably returns an absurd amount of matches. I imagine that I could find it by doing a build, finding a build command relevant , modifying it to get the preprocessed code and then doing grep, but that requires more time than I have to look at this right now.

I am a little more successful with the other functions:

  static bool isLocType(QualType T) {
    return T->isAnyPointerType() || T->isBlockPointerType() ||
           T->isReferenceType() || T->isNullPtrType();
  }
  SymbolVal(SymbolRef sym) : NonLoc(SymbolValKind, sym) {
    assert(sym);
    assert(!Loc::isLocType(sym->getType()));
  }
  loc::MemRegionVal makeLoc(SymbolRef sym) {
    return loc::MemRegionVal(MemMgr.getSymbolicRegion(sym));
  }

Nothing there stands out to me as possible causes. Finding some way to get Clang's static analyzer to generate a core dump instead of handling whatever went wrong internally would probably be more useful in figuring out the cause here, since then I could reproduce the issue locally to generate the core dump and load it into a debugger to see where the runtime issue actually is.

I guess another option would be to reduce this to a minimal test case and then attach gdb to the clang invocation so that it hopefully intercepts the signal before clang's signal handler. That would involve getting the preprocessed source output and trimming it until there is nothing left to trim without making it stop triggering this. I do not have time to volunteer to do that.

Given that the only people who use this tool are software developers, I suspect that other people affected by it could volunteer to track this down. It might even be easier for them since their systems are producing decent backtraces, unlike mine.

@dilyanpalauzov
Copy link
Author

Looking at all "C symbols", it finds a struct named SymbolRef, but C++ structures do not have member functions, so that cannot be it unless the C++ standard changed to allow them since I last accquinted myself with C++.

In C++ struct and class are almost the same. The only difference is that after struct { there is an implicit public:, while after class { there is an implicit private:.

@whisperity
Copy link
Member

@ryao

Tangentially, defect report CID-1490697 looked very interesting. What is interesting about it is that it says SSE->getRHS() is passed to a function argument named lhs and SSE->getLHS() is passed to a function argument named rhs. That kind of thing is usually a bug, but in rare circumstances, it could be intentional. Usually, those involve test cases that intentionally write incorrect code to check how it is handled, but I am not sure that there is no reason to ever do it. I am not familiar enough with the code to be certain if the report describes an actual bug and since it seems unrelated to this, I did not look at it very long.

(FYI: Clang-Tidy has a check named readability-suspicious-call-argument that hunts for such cases. IIRC, if the argument expression passed is a simple function call, it synthesises the "name" from the function name. Although the get might make Tidy itself not report this case. But know that we are working towards this in LLVM's own supplied tooling.)

This code is also difficult for me to navigate. My usual tool for navigating codebases is cscope and it usually is okay on C++ code, but it cannot find a global definition for SymbolRef.

LLVM's code is very complex because of all the template thingamajig that we are doing under the hood. However, I recommend trying clangd out. It integrates nicely into various editors that support LSP, both Vim and Emacs are covered. 😉

Looking at all "C symbols", it finds a struct named SymbolRef, but C++ structures do not have member functions, so that cannot be it unless the C++ standard changed to allow them since I last accquinted myself with C++.

This seems a bit tangential, and I am confused here. Do you mean the C standard changing? In C++, struct and class mean the exact same thing, they both introduce a record type, and record types can have member functions. (It is as it has always been, all the way back to the original C++98.) The only difference is that if you use class then all members and inherited bases are private by default, and if you use struct they are public by default.

Searching for an implementation for getType() predictably returns an absurd amount of matches. I imagine that I could find it by doing a build, finding a build command relevant , modifying it to get the preprocessed code and then doing grep, but that requires more time than I have to look at this right now.

(To navigate the code with Clangd, you do not even need to have a full build, only the compilation with CMake prepared. LLVM creates some generated code during the build which should be available so all the files are meaningfully preprocessed, but those codes are small, you only need to build the targets named tablegen or tblgen.)

Finding some way to get Clang's static analyzer to generate a core dump instead of handling whatever went wrong internally would probably be more useful in figuring out the cause here

Clang does the job of printing a stack trace and the invocation arguments when a deadly signal arrives, but does not invalidate the signal. That being said, a coredump should still be created once we returned from the signal handler. Did you set the ulimit appropriately? Clang's core dump will be huge...

void llvm::sys::PrintStackTrace(raw_ostream &OS, int Depth) {

@balazs-benics-sonarsource

@rayo I assume you are still trying to use the z3-based solver, and the mentioned crash relates to that. Please confirm.
I'll try to analyze OpenZFS to see if I have any crashes.
Let me know how you invoked CodeChecker. Which TU fails, etc.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
clang:static analyzer crash Prefer [crash-on-valid] or [crash-on-invalid]
Projects
None yet
Development

No branches or pull requests

8 participants