blob: 5326d83dfdd79b61fff9b5c2de93141b476bd30a [file] [log] [blame]
// Check that we can patch and un-patch DSOs loaded with dlopen.
//
// RUN: split-file %s %t
// RUN: %clangxx_xray -g -fPIC -fxray-instrument -fxray-shared -shared -std=c++11 %t/testlib.cpp -o %t/testlib.so
// RUN: %clangxx_xray -g -fPIC -rdynamic -fxray-instrument -fxray-shared -std=c++11 %t/main.cpp -o %t/main.o
//
// RUN: env XRAY_OPTIONS="patch_premain=true" %run %t/main.o %t/testlib.so 2>&1 | FileCheck %s
// REQUIRES: target={{(aarch64|x86_64)-.*}}
//--- main.cpp
#include "xray/xray_interface.h"
#include <cstdio>
#include <dlfcn.h>
void test_handler(int32_t fid, XRayEntryType type) {
printf("called: %d, type=%d\n", fid, static_cast<int32_t>(type));
}
[[clang::xray_always_instrument]] void instrumented_in_executable() {
printf("instrumented_in_executable called\n");
}
typedef void (*dso_func_type)();
int main(int argc, char **argv) {
if (argc < 2) {
printf("Shared library argument missing\n");
// CHECK-NOT: Shared library argument missing
return 1;
}
const char *dso_path = argv[1];
void *dso_handle = dlopen(dso_path, RTLD_LAZY);
if (!dso_handle) {
printf("Failed to load shared library\n");
char *error = dlerror();
if (error) {
fprintf(stderr, "%s\n", error);
return 1;
}
return 1;
}
dso_func_type instrumented_in_dso =
(dso_func_type)dlsym(dso_handle, "_Z19instrumented_in_dsov");
if (!instrumented_in_dso) {
printf("Failed to find symbol\n");
char *error = dlerror();
if (error) {
fprintf(stderr, "%s\n", error);
return 1;
}
return 1;
}
__xray_set_handler(test_handler);
instrumented_in_executable();
// CHECK: called: {{.*}}, type=0
// CHECK-NEXT: instrumented_in_executable called
// CHECK-NEXT: called: {{.*}}, type=1
instrumented_in_dso();
// CHECK-NEXT: called: {{.*}}, type=0
// CHECK-NEXT: instrumented_in_dso called
// CHECK-NEXT: called: {{.*}}, type=1
auto status = __xray_unpatch();
printf("unpatching status: %d\n", static_cast<int32_t>(status));
// CHECK-NEXT: unpatching status: 1
instrumented_in_executable();
// CHECK-NEXT: instrumented_in_executable called
instrumented_in_dso();
// CHECK-NEXT: instrumented_in_dso called
status = __xray_patch();
printf("patching status: %d\n", static_cast<int32_t>(status));
// CHECK-NEXT: patching status: 1
instrumented_in_executable();
// CHECK-NEXT: called: {{.*}}, type=0
// CHECK-NEXT: instrumented_in_executable called
// CHECK-NEXT: called: {{.*}}, type=1
instrumented_in_dso();
// CHECK-NEXT: called: {{.*}}, type=0
// CHECK-NEXT: instrumented_in_dso called
// CHECK-NEXT: called: {{.*}}, type=1
dlclose(dso_handle);
status = __xray_unpatch();
printf("unpatching status: %d\n", static_cast<int32_t>(status));
// CHECK-NEXT: unpatching status: 1
}
//--- testlib.cpp
#include <cstdio>
[[clang::xray_always_instrument]] void instrumented_in_dso() {
printf("instrumented_in_dso called\n");
}