Sid*_*gal 3 c++ symbols g++ shared-libraries redis
我在加载 Redis 模块时遇到一些问题。我只是从https://redis.io/topics/modules-intro复制示例,但我将其剥离了。
#include "redismodule.h"
#include <stdlib.h>
int RedisModule_OnLoad(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
if (RedisModule_Init(ctx,"avromodule",1,REDISMODULE_APIVER_1)
== REDISMODULE_ERR) return REDISMODULE_ERR;
return REDISMODULE_OK;
}
Run Code Online (Sandbox Code Playgroud)
这保存在 avromodule.cpp 中。我使用以下命令编译它:
g++ -shared -fPIC -o avromodule.so avromodule.cpp
Run Code Online (Sandbox Code Playgroud)
然后我转到 Redis CLI 并尝试加载模块。
10.XXX.XXX.XXX:7004> module load /path/to/module/avromodule.so
(error) ERR Error loading the extension. Please check the server logs.
Run Code Online (Sandbox Code Playgroud)
服务器日志给我以下错误:
159392:M 17 May 10:21:19.773 # Module /path/to/module/avromodule.so does not export RedisModule_OnLoad() symbol. Module not loaded.
Run Code Online (Sandbox Code Playgroud)
上述错误对我来说没有意义,因为我使用“nm”命令得到以下输出:
$ nm -CD avromodule.so | grep " T "
0000000000003622 T RedisModule_OnLoad(RedisModuleCtx*, RedisModuleString**, int)
000000000000366c T _fini
0000000000002878 T _init
Run Code Online (Sandbox Code Playgroud)
有谁知道这里可能出了什么问题吗?我知道我使用的是 C++,而不是推荐的 C,但这应该仍然有效。
发生这种情况是因为您的 C++ 编译器损坏RedisModule_OnLoad了名称。
RedisModule_OnLoad被 GCC 重命名为 to __Z18RedisModule_OnLoadP14RedisModuleCtxPP17RedisModuleStringi,因此 Redis 无法找到它正在寻找的导出符号。
$ nm avromodule.so | grep OnLoad
0000000000000970 T __Z18RedisModule_OnLoadP14RedisModuleCtxPP17RedisModuleStringi
Run Code Online (Sandbox Code Playgroud)
您可以使用该extern "C"指令来确保导出的符号保持未损坏。
#include "redismodule.h"
#include <stdlib.h>
#ifdef __cplusplus
extern "C" {
#endif
int RedisModule_OnLoad(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
if (RedisModule_Init(ctx,"avromodule",1,REDISMODULE_APIVER_1)
== REDISMODULE_ERR) return REDISMODULE_ERR;
return REDISMODULE_OK;
}
#ifdef __cplusplus
}
#endif
Run Code Online (Sandbox Code Playgroud)
这会导致导出未损坏的符号
nm avromodule.so | grep OnLoad
0000000000000970 T _RedisModule_OnLoad
Run Code Online (Sandbox Code Playgroud)