[compiler-rt] [TySan] Fix struct access with different bases (PR #120412)
via llvm-commits
llvm-commits at lists.llvm.org
Mon Jan 13 05:48:49 PST 2025
https://github.com/gbMattN updated https://github.com/llvm/llvm-project/pull/120412
>From 6682db92d3bb88d3d176b5de86d92fc2385f0461 Mon Sep 17 00:00:00 2001
From: gbMattN <matthew.nagy at sony.com>
Date: Mon, 13 Jan 2025 13:48:28 +0000
Subject: [PATCH] [TySan] Fix struct access with different bases
---
compiler-rt/lib/tysan/tysan.cpp | 11 +++++
.../tysan/struct-offset-different-base.cpp | 49 +++++++++++++++++++
2 files changed, 60 insertions(+)
create mode 100644 compiler-rt/test/tysan/struct-offset-different-base.cpp
diff --git a/compiler-rt/lib/tysan/tysan.cpp b/compiler-rt/lib/tysan/tysan.cpp
index 39d78e7c95e0cd..92cb3a5ba21126 100644
--- a/compiler-rt/lib/tysan/tysan.cpp
+++ b/compiler-rt/lib/tysan/tysan.cpp
@@ -131,6 +131,17 @@ static bool isAliasingLegalUp(tysan_type_descriptor *TDA,
break;
}
+ // This offset can't be negative. Therefore we must be accessing something
+ // before the current type (not legal) or partially inside the last type.
+ // In the latter case, we adjust Idx.
+ if (TDA->Struct.Members[Idx].Offset > OffsetA) {
+ // Trying to access something before the current type.
+ if (!Idx)
+ return false;
+
+ Idx -= 1;
+ }
+
OffsetA -= TDA->Struct.Members[Idx].Offset;
TDA = TDA->Struct.Members[Idx].Type;
} else {
diff --git a/compiler-rt/test/tysan/struct-offset-different-base.cpp b/compiler-rt/test/tysan/struct-offset-different-base.cpp
new file mode 100644
index 00000000000000..862595de8dc81f
--- /dev/null
+++ b/compiler-rt/test/tysan/struct-offset-different-base.cpp
@@ -0,0 +1,49 @@
+// RUN: %clangxx_tysan -O0 %s -o %t && %run %t >%t.out 2>&1
+// RUN: FileCheck %s --implicit-check-not ERROR < %t.out
+
+// Modified reproducer from https://github.com/llvm/llvm-project/issues/105960
+
+#include <stdio.h>
+
+struct inner1 {
+ char buffer;
+ int i;
+};
+
+struct inner2 {
+ char buffer;
+ int i;
+ float endBuffer;
+};
+
+void init_inner1(inner1 *iPtr) { iPtr->i = 200; }
+void init_inner2(inner2 *iPtr) {
+ iPtr->i = 400;
+ iPtr->endBuffer = 413.0f;
+}
+
+struct outer {
+ inner1 foo;
+ inner2 bar;
+ char buffer;
+};
+
+int main(void) {
+ outer *l = new outer();
+
+ init_inner1(&l->foo);
+ init_inner2(&l->bar);
+
+ int access = l->foo.i;
+ printf("Accessed value 1 is %d\n", access);
+ access = l->bar.i;
+ printf("Accessed value 2 is %d\n", access);
+ float fAccess = l->bar.endBuffer;
+ printf("Accessed value 3 is %f\n", fAccess);
+
+ return 0;
+}
+
+// CHECK: Accessed value 1 is 200
+// CHECK: Accessed value 2 is 400
+// CHECK: Accessed value 3 is 413.0
More information about the llvm-commits
mailing list