summaryrefslogtreecommitdiff
path: root/libunwind
diff options
context:
space:
mode:
authorMingming Liu <mingmingl@google.com>2025-09-10 15:25:31 -0700
committerGitHub <noreply@github.com>2025-09-10 15:25:31 -0700
commit1417dafa1db9cb1b2b09438aa9f53ea5ab6e36e2 (patch)
tree57f4b1f313c8cf74eed8819870f39c36ea263c68 /libunwind
parent898b813bc8a6d0276bf0f4769f5f2f64b34e632d (diff)
parentb8cefcb601ddaa18482555c4ff363c01a270c2fe (diff)
Merge branch 'main' into users/mingmingl-llvm/samplefdo-profile-formatusers/mingmingl-llvm/samplefdo-profile-format
Diffstat (limited to 'libunwind')
-rw-r--r--libunwind/src/DwarfParser.hpp2
-rw-r--r--libunwind/test/eh_frame_fde_pc_range.pass.cpp60
2 files changed, 61 insertions, 1 deletions
diff --git a/libunwind/src/DwarfParser.hpp b/libunwind/src/DwarfParser.hpp
index 7e85025dd054..25250e081098 100644
--- a/libunwind/src/DwarfParser.hpp
+++ b/libunwind/src/DwarfParser.hpp
@@ -273,7 +273,7 @@ bool CFI_Parser<A>::findFDE(A &addressSpace, pint_t pc, pint_t ehSectionStart,
pint_t pcRange = addressSpace.getEncodedP(
p, nextCFI, cieInfo->pointerEncoding & 0x0F);
// Test if pc is within the function this FDE covers.
- if ((pcStart < pc) && (pc <= pcStart + pcRange)) {
+ if ((pcStart <= pc) && (pc < pcStart + pcRange)) {
// parse rest of info
fdeInfo->lsda = 0;
// check for augmentation length
diff --git a/libunwind/test/eh_frame_fde_pc_range.pass.cpp b/libunwind/test/eh_frame_fde_pc_range.pass.cpp
new file mode 100644
index 000000000000..39c8e8066264
--- /dev/null
+++ b/libunwind/test/eh_frame_fde_pc_range.pass.cpp
@@ -0,0 +1,60 @@
+// -*- C++ -*-
+//===----------------------------------------------------------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+// Manually marking the .eh_frame_hdr as DW_EH_PE_omit to make libunwind to do
+// the linear search.
+// Assuming the begining of the function is at the start of the FDE range.
+
+// clang-format off
+
+// REQUIRES: target={{x86_64-.+-linux-gnu}}
+// aarch64,arm have a cross toolchain build(llvm-clang-win-x-aarch64, etc)
+// where objdump is not available.
+
+// TODO: Figure out why this fails with Memory Sanitizer.
+// XFAIL: msan
+
+// RUN: %{build}
+// RUN: objcopy --dump-section .eh_frame_hdr=%t_ehf_hdr.bin %t.exe
+// RUN: echo -ne '\xFF' | dd of=%t_ehf_hdr.bin bs=1 seek=2 count=2 conv=notrunc status=none
+// RUN: objcopy --update-section .eh_frame_hdr=%t_ehf_hdr.bin %t.exe
+// RUN: %{exec} %t.exe
+
+// clang-format on
+
+#include <assert.h>
+#include <libunwind.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <unwind.h>
+
+void f() {
+ printf("123\n");
+ void *pc = __builtin_return_address(0);
+ void *fpc = (void *)&f;
+ void *fpc1 = (void *)((uintptr_t)fpc + 1);
+
+ struct dwarf_eh_bases bases;
+ const void *fde_pc = _Unwind_Find_FDE(pc, &bases);
+ const void *fde_fpc = _Unwind_Find_FDE(fpc, &bases);
+ const void *fde_fpc1 = _Unwind_Find_FDE(fpc1, &bases);
+ printf("fde_pc = %p\n", fde_pc);
+ printf("fde_fpc = %p\n", fde_fpc);
+ printf("fde_fpc1 = %p\n", fde_fpc1);
+ fflush(stdout);
+ assert(fde_pc != NULL);
+ assert(fde_fpc != NULL);
+ assert(fde_fpc1 != NULL);
+ assert(fde_fpc == fde_fpc1);
+}
+
+int main() {
+ f();
+ return 0;
+}