在 SWIG 输入类型映射中释放指针数组的正确方法?
Proper way to free a pointer array in SWIG input typemap?
您好,我正在尝试使用 SWIG 包装以下函数。
static void readTable(int argc, t_atom *argv) { //accepts table in Lua e.g. readTable({"ab",3});
for (int i=0; i<argc; ++i) {
if (argv[i].a_type == A_FLOAT)
printf("FLOAT : %g\n", argv[i].a_w.w_float);
else if (argv[i].a_type == A_SYMBOL)
printf("SYMBOL : %s\n", argv[i].a_w.w_symbol->s_name);
}
}
这是我创建的类型图。
%include "exception.i"
%typemap(in) (int argc, t_atom *argv)
{
if (!lua_istable(L, 1)) {
SWIG_exception(SWIG_RuntimeError, "argument mismatch: table expected");
}
lua_len(L, 1);
= lua_tointeger(L, -1);
= (t_atom *)getbytes( * sizeof(t_atom)); //internally calls calloc()
for (int i=0; i<; ++i) {
lua_pushinteger(L, i+1);
lua_gettable(L, 1);
if(lua_isnumber(L, -1)) {
[i].a_type = A_FLOAT;
[i].a_w.w_float = lua_tonumber(L, -1);
}
else if(lua_isstring(L, -1)) {
[i].a_type = A_SYMBOL;
[i].a_w.w_symbol = gensym(lua_tostring(L, -1));
}
}
freebytes(, * sizeof(t_atom)); //internally calls free()
}
如您所见,我使用内部调用 calloc()
的 getbytes()
为数组分配内存,并使用内部调用 free()
的 freebytes()
释放内存。
令人惊讶的是,当我在 Lua 中调用 readTable()
函数时,它工作正常而没有崩溃。
例如,在 Lua 中调用 test.readTable({3, "abc"})
会打印以下结果。
FLOAT : 3
SYMBOL : abc
我的问题是
a) 当我释放 SWIG 界面中的内存时,readTable()
如何在不崩溃的情况下打印结果?也许内存没有被正确释放?
b) 如果 a) 没有意义或使用不安全,您是否建议在函数 readTable()
中调用 freebytes()
而不是像下面这样?
static void readTable(int argc, t_atom *argv) {
for (int i=0; i<argc; ++i) {
if (argv[i].a_type == A_FLOAT)
post("FLOAT : %g", argv[i].a_w.w_float);
else if (argv[i].a_type == A_SYMBOL)
post("SYMBOL : %s", argv[i].a_w.w_symbol->s_name);
}
freebytes(argv, argc * sizeof(t_atom));
}
这正是 %typemap(freearg)
所针对的情况。
%include "exception.i"
%typemap(in) (int argc, t_atom *argv) {
if (!lua_istable(L, $input)) {
SWIG_exception(SWIG_RuntimeError, "argument mismatch: table expected");
}
lua_len(L, $input);
= lua_tointeger(L, -1);
= (t_atom *)getbytes( * sizeof(t_atom)); // internally calls calloc()
for (int i = 0; i < ; ++i) {
lua_pushinteger(L, i + 1);
lua_gettable(L, $input);
if (lua_isnumber(L, -1)) {
[i].a_type = A_FLOAT;
[i].a_w.w_float = lua_tonumber(L, -1);
} else if (lua_isstring(L, -1)) {
[i].a_type = A_SYMBOL;
[i].a_w.w_symbol = gensym(lua_tostring(L, -1));
} else {
SWIG_exception(SWIG_RuntimeError, "unhandled argument type");
}
}
}
%typemap(freearg) (int argc, t_atom *argv) {
freebytes(, * sizeof(t_atom)); // internally calls free()
}
static void readTable(const std::string &name, int argc, t_atom *argv);
这是 SWIG 3.0 生成的代码
static int _wrap_readTable(lua_State* L) {
int SWIG_arg = 0;
std::string *arg1 = 0 ;
int arg2 ;
t_atom *arg3 = (t_atom *) 0 ;
SWIG_check_num_args("readTable",2,2)
if(!lua_isuserdata(L,1)) SWIG_fail_arg("readTable",1,"std::string const &");
if (!SWIG_IsOK(SWIG_ConvertPtr(L,1,(void**)&arg1,SWIGTYPE_p_std__string,0))){
SWIG_fail_ptr("readTable",1,SWIGTYPE_p_std__string);
}
{
if (!lua_istable(L, 2)) {
SWIG_exception(SWIG_RuntimeError, "argument mismatch: table expected");
}
lua_len(L, 2);
arg2 = lua_tointeger(L, -1);
arg3 = (t_atom *)getbytes(arg2 * sizeof(t_atom)); // internally calls calloc()
for (int i = 0; i < arg2; ++i) {
lua_pushinteger(L, i + 1);
lua_gettable(L, 2);
if (lua_isnumber(L, -1)) {
arg3[i].a_type = A_FLOAT;
arg3[i].a_w.w_float = lua_tonumber(L, -1);
} else if (lua_isstring(L, -1)) {
arg3[i].a_type = A_SYMBOL;
arg3[i].a_w.w_symbol = gensym(lua_tostring(L, -1));
} else {
SWIG_exception(SWIG_RuntimeError, "unhandled argument type");
}
}
}
readTable((std::string const &)*arg1,arg2,arg3);
{
freebytes(arg3, arg2 * sizeof(t_atom)); // internally calls free()
}
return SWIG_arg;
if(0) SWIG_fail;
fail:
{
freebytes(arg3, arg2 * sizeof(t_atom)); // internally calls free()
}
lua_error(L);
return SWIG_arg;
}
您好,我正在尝试使用 SWIG 包装以下函数。
static void readTable(int argc, t_atom *argv) { //accepts table in Lua e.g. readTable({"ab",3});
for (int i=0; i<argc; ++i) {
if (argv[i].a_type == A_FLOAT)
printf("FLOAT : %g\n", argv[i].a_w.w_float);
else if (argv[i].a_type == A_SYMBOL)
printf("SYMBOL : %s\n", argv[i].a_w.w_symbol->s_name);
}
}
这是我创建的类型图。
%include "exception.i"
%typemap(in) (int argc, t_atom *argv)
{
if (!lua_istable(L, 1)) {
SWIG_exception(SWIG_RuntimeError, "argument mismatch: table expected");
}
lua_len(L, 1);
= lua_tointeger(L, -1);
= (t_atom *)getbytes( * sizeof(t_atom)); //internally calls calloc()
for (int i=0; i<; ++i) {
lua_pushinteger(L, i+1);
lua_gettable(L, 1);
if(lua_isnumber(L, -1)) {
[i].a_type = A_FLOAT;
[i].a_w.w_float = lua_tonumber(L, -1);
}
else if(lua_isstring(L, -1)) {
[i].a_type = A_SYMBOL;
[i].a_w.w_symbol = gensym(lua_tostring(L, -1));
}
}
freebytes(, * sizeof(t_atom)); //internally calls free()
}
如您所见,我使用内部调用 calloc()
的 getbytes()
为数组分配内存,并使用内部调用 free()
的 freebytes()
释放内存。
令人惊讶的是,当我在 Lua 中调用 readTable()
函数时,它工作正常而没有崩溃。
例如,在 Lua 中调用 test.readTable({3, "abc"})
会打印以下结果。
FLOAT : 3
SYMBOL : abc
我的问题是
a) 当我释放 SWIG 界面中的内存时,readTable()
如何在不崩溃的情况下打印结果?也许内存没有被正确释放?
b) 如果 a) 没有意义或使用不安全,您是否建议在函数 readTable()
中调用 freebytes()
而不是像下面这样?
static void readTable(int argc, t_atom *argv) {
for (int i=0; i<argc; ++i) {
if (argv[i].a_type == A_FLOAT)
post("FLOAT : %g", argv[i].a_w.w_float);
else if (argv[i].a_type == A_SYMBOL)
post("SYMBOL : %s", argv[i].a_w.w_symbol->s_name);
}
freebytes(argv, argc * sizeof(t_atom));
}
这正是 %typemap(freearg)
所针对的情况。
%include "exception.i"
%typemap(in) (int argc, t_atom *argv) {
if (!lua_istable(L, $input)) {
SWIG_exception(SWIG_RuntimeError, "argument mismatch: table expected");
}
lua_len(L, $input);
= lua_tointeger(L, -1);
= (t_atom *)getbytes( * sizeof(t_atom)); // internally calls calloc()
for (int i = 0; i < ; ++i) {
lua_pushinteger(L, i + 1);
lua_gettable(L, $input);
if (lua_isnumber(L, -1)) {
[i].a_type = A_FLOAT;
[i].a_w.w_float = lua_tonumber(L, -1);
} else if (lua_isstring(L, -1)) {
[i].a_type = A_SYMBOL;
[i].a_w.w_symbol = gensym(lua_tostring(L, -1));
} else {
SWIG_exception(SWIG_RuntimeError, "unhandled argument type");
}
}
}
%typemap(freearg) (int argc, t_atom *argv) {
freebytes(, * sizeof(t_atom)); // internally calls free()
}
static void readTable(const std::string &name, int argc, t_atom *argv);
这是 SWIG 3.0 生成的代码
static int _wrap_readTable(lua_State* L) {
int SWIG_arg = 0;
std::string *arg1 = 0 ;
int arg2 ;
t_atom *arg3 = (t_atom *) 0 ;
SWIG_check_num_args("readTable",2,2)
if(!lua_isuserdata(L,1)) SWIG_fail_arg("readTable",1,"std::string const &");
if (!SWIG_IsOK(SWIG_ConvertPtr(L,1,(void**)&arg1,SWIGTYPE_p_std__string,0))){
SWIG_fail_ptr("readTable",1,SWIGTYPE_p_std__string);
}
{
if (!lua_istable(L, 2)) {
SWIG_exception(SWIG_RuntimeError, "argument mismatch: table expected");
}
lua_len(L, 2);
arg2 = lua_tointeger(L, -1);
arg3 = (t_atom *)getbytes(arg2 * sizeof(t_atom)); // internally calls calloc()
for (int i = 0; i < arg2; ++i) {
lua_pushinteger(L, i + 1);
lua_gettable(L, 2);
if (lua_isnumber(L, -1)) {
arg3[i].a_type = A_FLOAT;
arg3[i].a_w.w_float = lua_tonumber(L, -1);
} else if (lua_isstring(L, -1)) {
arg3[i].a_type = A_SYMBOL;
arg3[i].a_w.w_symbol = gensym(lua_tostring(L, -1));
} else {
SWIG_exception(SWIG_RuntimeError, "unhandled argument type");
}
}
}
readTable((std::string const &)*arg1,arg2,arg3);
{
freebytes(arg3, arg2 * sizeof(t_atom)); // internally calls free()
}
return SWIG_arg;
if(0) SWIG_fail;
fail:
{
freebytes(arg3, arg2 * sizeof(t_atom)); // internally calls free()
}
lua_error(L);
return SWIG_arg;
}