gpt4 book ai didi

c - 在lua中设置不同的内存分配函数

转载 作者:太空宇宙 更新时间:2023-11-03 23:28:35 25 4
gpt4 key购买 nike

我正在做 Roberto Ierusalimschy 的第三版“Programming in Lua”一书中的练习。我的练习 32.1 的解决方案中有一个错误。该语句在代码中作为注释提供。

/*
Exercise 32.1:
Write a library that allows a script to limit the total amount of memory
used by its Lua state. It may offer a single function, setlimit, to set that
limit.
The library should set its own allocation funciton. This function, before
calling the original allocator, checks the total memory in use and returns
NULL if the requested memory exeeds the limit.
(Hint: the library can use lua_gc to initialize its byte count when it
starts. It also can use the user data of the allocation function to keep its
state: the byte count, the current memory limit, etc.; remember to use the
original user data when calling the original allocation function.)
*/

#ifdef WIN32
#define LUA_EXPORT __declspec(dllexport)
#else
#define LUA_EXPORT
#endif

#include <lauxlib.h>

typedef struct MemLimitUData
{
size_t mem_limit;
size_t currently_used;
lua_Alloc original_alloc;
void *original_ud;
}
MemLimitUData;

static int l_setlimit(lua_State *L)
{
MemLimitUData *ud;
size_t mem_limit = luaL_checkinteger(L, 1);

lua_getallocf(L, &ud);
ud->mem_limit = mem_limit;

return 0;
}

static int l_getlimit(lua_State *L)
{
MemLimitUData *ud;

lua_getallocf(L, &ud);
lua_pushnumber(L, ud->mem_limit);

return 1;
}

static void *l_alloc(void *ud, void *ptr, size_t osize, size_t nsize)
{
MemLimitUData *udata = (MemLimitUData*)ud;

if (udata->mem_limit != 0 &&
udata->mem_limit < udata->currently_used - osize + nsize)
{
return NULL;
}

udata->currently_used += nsize - osize;
return udata->original_alloc(udata->original_ud, ptr, osize, nsize);
}

static const luaL_Reg memlimit[] =
{
{ "setlimit", l_setlimit },
{ "getlimit", l_getlimit },
{ NULL, NULL }
};

int LUA_EXPORT luaopen_memlimit(lua_State *L)
{
MemLimitUData *ud =
(MemLimitUData*)lua_newuserdata(L, sizeof(MemLimitUData));

ud->mem_limit = 0;
ud->currently_used =
lua_gc(L, LUA_GCCOUNT, 0) * 1024 + lua_gc(L, LUA_GCCOUNTB, 0);
ud->original_alloc = lua_getallocf(L, &ud->original_ud);

lua_setallocf(L, l_alloc, ud);
luaL_newlib(L, memlimit);

return 1;
}

当我将源构建为 memlimit.dll 并从 Lua 脚本中使用它时,

local memlimit = require"memlimit"
脚本结束时程序崩溃。当我使用调试器查找问题时,有问题的语句似乎在 Lua 内部。该文件是 lmem.c84:

newblock = (*g->frealloc)(g->ud, block, osize, nsize);

使用的Lua版本是5.2.3。

我做错了什么破坏了 Lua 内存管理?

最佳答案

我没有尝试过您的代码,但当我阅读它时,这里引起了我的注意:

luaopen_memlimit 中的 ud 被创建为用户数据,但没有锚定在 Lua 中。将其传递给 lua_getallocf 不算作锚定。当程序通过 lua_close 尝试使用您的 l_alloc 释放所有数据时,ud 可能正在被收集。您可能应该使用普通的 malloc 或原始的 allocf 来创建 ud

关于c - 在lua中设置不同的内存分配函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20709938/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com