- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我的问题如下:
我有一个包含这样的函数指针的结构:
typedef void (CALLING_COVNENTION * functionType_t) (/*...*/);
typedef struct myFuncpointerStruc_s
{
/*...*/
functionType_t funcIdentifier;
/*...*/
}myFuncpointerStruc_t;
现在我想为它分配一个完全相同类型的函数指针。但遗憾的是 dlsym()
返回的是 void *
而不是 function_poitner_type
所以我的第一次尝试:
functionscope ()
{
myFuncpointerStruc_t structIdentifier;
structIdentifier.funcIdentifier= dlsym (hAndle, "calledName");
}
结束于警告:
WARNING: ISO C forbids assignment between function pointer and 'void *'
好吧,这让我很困惑……gcc 从来没有对我说那么严厉的话。但是好吧,让我们变得棘手吧,我认为可能会有任何后门来保持它的标准一致性。所以我尝试了这个:
functionscope ()
{
myFuncpointerStruc_t structIdentifier;
*(void **) &structIdentifier.funcIdentifier = dlsym (hAndle, "calledName");
}
嗯....显然...
warning: dereferencing type-punned pointer will break strict aliasing rules
因为我不知道 dlsym()
内部发生了什么,所以我不知道我会中断,因为我返回了一个真正的 void *
别名我的函数,或者完全不同的东西,我可以吗?
所以我不知道,现在使用这个函数指针是否会违反严格的别名规则?
但无论如何:对于我的公司来说,使用编译器标志 -fstrict-aliasing
是合法的。所以这对我来说也不是一个可能的解决方案,即使它符合要求。
所以我继续尝试。
我的下一个想法是:如何将 dlsym()
的 char *
版本解析为 memcopy?
结果:
functionscope ()
{
myFuncpointerStruc_t structIdentifier;
unsigned char *localTestPtr = ((unsigned char *) dlsym (hAndle, "calledName");
memcpy (g_interfaceSvCheck.pfnPFD_SVCHK_GET_VERSION, localTestPtr, sizeof (localTestPtr));
}
我现在收到的警告再次非常具体:
warning: ISO C forbids passing argument 1 of ´memcpy´ between function pointer and ´void *´
所以我的下一个想法是“让我们回到字节”,也许在那里我会找到一个可以继续工作的方法。(但慢慢地我的想法用完了......)
所以我做了:
functionscope ()
{
size_t testIndex;
myFuncpointerStruc_t structIdentifier;
unsigned char *localTestPtr = ((unsigned char *) dlsym (hAndle, "calledName");
for (testIndex = 0; testIndex < sizeof (localTestPtr); testIndex++)
{
((unsigned char *)(structIdentifier.funcIdentifier))[testIndex] = ((unsigned char *)&localTestPtr)[testIndex];
}
}
这以一个安静的不同警告结束:
WARNING: ISO C forbids conversion of function pointer to object pointer type
所以最后我最疯狂的尝试是:
functionscope ()
{
size_t testIndex;
myFuncpointerStruc_t structIdentifier;
unsigned char *localTestPtr = ((unsigned char *) dlsym (hAndle, "calledName");
unsigned char * FakeFunctionPointer;
FakeFunctionPointer =
(((unsigned char *)structIdentifier) + offsetof (myFuncpointerStruc_t , funcIdentifier));
for (testIndex = 0; testIndex < sizeof (localTestPtr); testIndex++)
{
FakeFunctionPointer[testIndex] = ((unsigned char *)&localTestPtr)[testIndex];
}
}
问题是,offsetoff
的第二个参数无法转换为 void *
,因为它是一个函数指针。
我被困住了,不知道如何更进一步,有人能帮我找到解决方案吗?
我只是通过使用 gcc 4.2 而不是 clang 收到这些警告。
objective-c 版本是 -std=c99
要避免的是 -std=POSIX
和 -std=gnu99
但是这个 gcc 警告对我来说并没有过时。我目前在 FreeBSD 上工作,我真的不知道如何解决这个问题。
最佳答案
这有点棘手:
#define CALLING_CONVENTION
typedef void (CALLING_CONVENTION * functionType_t) (/*...*/);
typedef struct myFuncpointerStruc_s
{
/*...*/
functionType_t funcIdentifier;
/*...*/
} myFuncpointerStruc_t;
int main()
{
myFuncpointerStruc_t structIdentifier = {0};
{
functionType_t * pfuncIdentifier = &structIdentifier.funcIdentifier;
*(void **) (pfuncIdentifier) = dlsym(handle, "calledName");
}
...
}
上述解决方案只是编译器的技巧。从 -O1
开始不不再工作......:-(
既然确实无解,唯一的办法就是让编译器冷静下来。 GNU 人员对此表现出了洞察力并发明了:__attribute__ ((__may_alias__))
这里我们可以这样使用:
#define CALLING_CONVENTION
typedef void * __attribute__ ((__may_alias__)) pvoid_may_alias_t;
typedef void (CALLING_CONVENTION * functionType_t) (/*...*/);
typedef struct myFuncpointerStruc_s
{
/*...*/
functionType_t funcIdentifier;
/*...*/
} myFuncpointerStruc_t;
int main()
{
myFuncpointerStruc_t structIdentifier = {0};
*(pvoid_may_alias_t *) (&structIdentifier.funcIdentifier) = dlsym(handle, "calledName");
}
...
}
顺便说一句:ldsym()
从何而来?我只知道dlsym()
。
关于c - 如何将 dlsym() 的返回值分配给函数类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19116487/
是否可以使用 dlsym 从框架加载已命名的未导出符号? ? 我尝试导入的符号有一个名称,在框架内引用它。这是我需要调用的函数。 我正在尝试照常进行 dlopen + dlsym方式,但是当我尝试加载
我收到此链接器错误: system/core/libacc/tests/main.cpp:42: error: undefined reference to 'dlsym' 您能告诉我 ubuntu
我使用 dlsym() 来调用我的 malloc 版本,而不是默认库 malloc: lt_malloc = (void*(*)(size_t))dlsym(RTLD_NEXT, "malloc")
1).直接打印函数地址: printf("strerror=%p, strerror_r=%p\n", strerror, strerror_r); strerror=0x8049ec0, strer
所以我一直试图从本地 dylib 获取符号,但每当我尝试做任何事情时,我都会遇到段错误,我不确定为什么会发生这种情况。我已经在其他程序中看到过这样做并且它们工作正常,所以我知道它有效但我似乎无法做到:
我真的很讨厌在这里提问。但我一直在看some of the other posts ,并且诸如此类的解决方案似乎不起作用。这可能是我对语法的误解。 我正在改进我的一些旧代码。问题中的函数循环通过一些加
作为 malloc 包装器,我使用了这段经典的代码片段: #define _GNU_SOURCE #include #include #include void* malloc(size_t s
我正在用 C 编写一个插件应用程序,我正在使用 dlopen/dlsym 动态加载某些功能的“实现”。例如,我有以下指向函数的指针 struct cti_t* (*create)() = 0; 然后我
我使用 dlopen 在我的程序中加载 pthread 库,并使用 dlsym 获取处理程序到 pthread_create 和 pthread_join >. 如何确保两个处理程序的版本兼容?我不在
man page dlsym() 列表 *(void **) (&cosine) = dlsym(handle, "cos"); 作为转换 dlsym() 返回值的解决方法。 这里的*(void
我在 C 中使用 dlsym() 我有一个问题 dlsym() 的返回值是否应该显式转换或者它是否被正确地隐式转换。这是函数: double (*(compile)(void))(double x,
我有一个动态加载库的应用程序,它动态加载库... 在 Windows 中,我能够遍历所有加载的模块以查找我感兴趣的符号。不知道如何在 Unix/Linux 环境中执行此操作。我知道我可以对前两个符号使
奇怪的是 dlsym 可以从剥离的二进制文件中导入函数。 谁能告诉我为什么/怎么做? === FILE: a.c === int a1() { return 1; } int a2() { retur
我正在尝试使用 Qt 5.3.0 在 Linux Mint 17 中编译 C++ 程序。我已将以下行添加到 .pro 文件中: QMAKE_LFLAGS += -ldl -fPIC 我的程序使用了 d
我正在使用 POSIX dlopen/dlsym API 在运行时加载动态库,然后按名称从这些库中调用函数。 存储 dlsym 的结果在性能方面是个好主意吗?某处?或者 dlsym 是否已经进行了自己
有没有跑路的希望dlopen(NULL, ...)并为静态编译的二进制文件获取符号? 例如,使用以下代码,如果程序是动态编译的并且我使用 -rdynamic,我可以获得符号。 . $ gcc -o f
我正在从共享对象 (a.so) 调用函数。我使用 dlopen 加载 a.so 并使用 dlsym 加载映射函数。 int (*funcPtr)() = reinterpret_cast(dl
我正在尝试使用 dlsym() 在 libcrypto.so 文件中查找 connect 符号,并希望用我的方法替换此 connect 。 void * lib_orig_tols = dlopen
我想从我的共享库 (.so) 中访问一个类符号这是我需要做的正确方法吗? MyClass* pMyClass= (MyClass*)dlsym(phandle, "MyClass"); 如果这是正确的
dlsym为枚举值返回 null NSLog(@"%s",dlsym(RTLD_DEFAULT, "NSTextAlignmentCenter")); 输出 2014-01-28 13:07:04.2
我是一名优秀的程序员,十分优秀!