c - 在不重启服务器的情况下重新加载模块

标签 c tarantool

美好的一天! 我有一个关于在 tarantool 中重新加载 c 模块的小问题 例如:我有一个公开方法的 c 模块:

int calculate(lua_State* L);

另外我声明了入口点:

extern "C"
{    
    LUA_API int luaopen_cuendemodule(lua_State *L);
}

现在,我在 tarantool 中加载这个模块(“testmodule.so”):

require('testmodule')
box.schema.func.create('testmodule.calculate')
box.schema.user.grant('user', 'execute', 'function', 'testmodule.calculate')

现在我从我的 C# 客户端调用这个方法:

await tarantoolClient.Call<TarantoolTuple<CalculateParameters>, CalculationResults>("testmodule.calculate", TarantoolTuple.Create(....));

它按预期工作 - 执行了方法计算并返回了结果

但如果我想更新我的模块,那么问题就开始了:在我替换 so 文件并调用计算方法后,我的 tarantool 重新启动,我可以在 dmesg 中看到类似“tarntool invalid opcode in testmodule.so”的内容

阅读文档后,我在函数定义中看到了如下附加参数:

box.schema.func.create('testmodule.calculate', {language = 'C'})

但在此之后,如果我从 C# 调用它,我会收到异常消息“无法动态加载函数 undefined symbol 计算”

我在 ubuntu 上使用 tarantool 1.7 我是用 gcc 8.1.0 编译的

最佳答案

没有很好的方法来做到这一点(意味着 - 一种可移植的方式)。我认为,最好的方法是使用类似 dlopen()[1] 的方法,该函数允许打开(一般管理共享对象)共享对象,但您必须对此非常谨慎,它也可能会使您的代码失败甚至你可以有 sigfault。

一个很好的例子是:https://github.com/tarantool/mqtt ,该模块不使用这些函数(func.create 等),但它也可以扩展。

所以重点是:如果你开发一个 C 模块,你必须考虑重新加载策略。 例如,*-unix 类系统有很多允许重新加载一些共享对象的功能,而且 tarantool 也有一些功能。

另外,我建议你开始将模块视为 Lua 的 C 模块,实际上是一样的。

附言

一些重新加载模块也可用:https://github.com/Mons/tnt-package-reload (我没有测试模块),https://github.com/tarantool/reload (我没有测试模块)

[1] http://man7.org/linux/man-pages/man3/dlopen.3.html

关于c - 在不重启服务器的情况下重新加载模块,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51038456/

相关文章:

c - 哪些系统调用可以返回 EINTR 或 EAGAIN 错误代码?

c - 'function' 的先前声明在 C 中

sql - 如何在 Tarantool 中更改 sql 表的格式

javascript - 无法通过Nodejs连接到Docker内部的Tarantool容器

tarantool - 如何在 tarantool 墨盒中执行热重载代码?

lua - 按二级索引从 Tarantool 中选择,并按另一个字段和限制/偏移量排序

c:找到两个简单列表之间的连接(公共(public)节点)

c - 输入文件中数组字符串的动态内存分配

c - 使用环形缓冲区和 pthread 的生产者 - 消费者