#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <dlfcn.h>
#include <mach-o/dyld_priv.h>
#if __has_feature(ptrauth_calls)
#include <ptrauth.h>
#endif
#include "test_support.h"
extern void* __dso_handle;
extern void verifyDylib();
static const void* stripPointer(const void* ptr)
{
#if __has_feature(ptrauth_calls)
return __builtin_ptrauth_strip(ptr, ptrauth_key_asia);
#else
return ptr;
#endif
}
int bar()
{
return 2;
}
static int foo()
{
return 3;
}
__attribute__((visibility("hidden"))) int hide()
{
return 4;
}
static void verifybar()
{
Dl_info info;
if ( dladdr(&bar, &info) == 0 ) {
FAIL("dladdr(&bar, xx) failed");
}
if ( strcmp(info.dli_sname, "bar") != 0 ) {
FAIL("dladdr()->dli_sname is \"%s\" instead of \"bar\"", info.dli_sname);
}
if ( info.dli_saddr != stripPointer(&bar) ) {
FAIL("dladdr()->dli_saddr is not &bar");
}
if ( info.dli_fbase != &__dso_handle ) {
FAIL("dladdr()->dli_fbase is not image that contains &bar");
}
}
static void verifyfoo()
{
Dl_info info;
if ( dladdr(&foo, &info) == 0 ) {
FAIL("dladdr(&foo, xx) failed");
}
if ( strcmp(info.dli_sname, "foo") != 0 ) {
FAIL("dladdr()->dli_sname is \"%s\" instead of \"foo\"", info.dli_sname);
}
if ( info.dli_saddr != stripPointer(&foo) ) {
FAIL("dladdr()->dli_saddr is not &foo");
}
if ( info.dli_fbase != &__dso_handle ) {
FAIL("dladdr()->dli_fbase is not image that contains &foo");
}
}
static void verifyhide()
{
Dl_info info;
if ( dladdr(&hide, &info) == 0 ) {
FAIL("dladdr(&hide, xx) failed");
}
if ( strcmp(info.dli_sname, "hide") != 0 ) {
FAIL("dladdr()->dli_sname is \"%s\" instead of \"hide\"", info.dli_sname);
}
if ( info.dli_saddr != stripPointer(&hide) ) {
FAIL("dladdr()->dli_saddr is not &hide");
}
if ( info.dli_fbase != &__dso_handle ) {
FAIL("dladdr()->dli_fbase is not image that contains &hide");
}
}
static void verifymalloc()
{
Dl_info info;
if ( dladdr(&malloc, &info) == 0 ) {
FAIL("dladdr(&malloc, xx) failed");
}
if ( strcmp(info.dli_sname, "malloc") != 0 ) {
FAIL("dladdr()->dli_sname is \"%s\" instead of \"malloc\"", info.dli_sname);
}
if ( info.dli_saddr != stripPointer(&malloc) ) {
FAIL("dladdr()->dli_saddr is not &malloc");
}
if ( info.dli_fbase != dyld_image_header_containing_address(&malloc) ) {
FAIL("dladdr()->dli_fbase is not image that contains &malloc");
}
}
int main(int argc, const char* argv[], const char* envp[], const char* apple[]) {
verifybar();
verifyhide();
verifyfoo();
verifymalloc();
verifyDylib();
PASS("Success");
}