[analyzer] Fix false positive on introspection of a block's internal layout.
authorArtem Dergachev <artem.dergachev@gmail.com>
Fri, 6 Dec 2019 21:20:36 +0000 (13:20 -0800)
committerArtem Dergachev <artem.dergachev@gmail.com>
Fri, 6 Dec 2019 21:24:20 +0000 (13:24 -0800)
When implementation of the block runtime is available, we should not
warn that block layout fields are uninitialized simply because they're
on the stack.

clang/lib/StaticAnalyzer/Core/RegionStore.cpp
clang/test/Analysis/blocks.m

index 5d2ef59..4797f56 100644 (file)
@@ -1951,7 +1951,8 @@ RegionStoreManager::getBindingForFieldOrElementCommon(RegionBindingsConstRef B,
     if (hasSymbolicIndex)
       return UnknownVal();
 
-    if (!hasPartialLazyBinding)
+    // Additionally allow introspection of a block's internal layout.
+    if (!hasPartialLazyBinding && !isa<BlockDataRegion>(R->getBaseRegion()))
       return UndefinedVal();
   }
 
index 98d0f8a..a21a605 100644 (file)
@@ -47,6 +47,10 @@ typedef struct __aslmsg *aslmsg;
 aslclient asl_open(const char *ident, const char *facility, uint32_t opts);
 int asl_log(aslclient asl, aslmsg msg, int level, const char *format, ...) __attribute__((__format__ (__printf__, 4, 5)));
 
+struct Block_layout {
+  int flags;
+};
+
 //===----------------------------------------------------------------------===//
 // Begin actual test cases.
 //===----------------------------------------------------------------------===//
@@ -241,3 +245,8 @@ void call_block_with_fewer_arguments() {
   b(); // expected-warning {{Block taking 1 argument is called with fewer (0)}}
 }
 #endif
+
+int getBlockFlags() {
+  int x = 0;
+  return ((struct Block_layout *)^{ (void)x; })->flags; // no-warning
+}