You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
217 lines
6.0 KiB
C
217 lines
6.0 KiB
C
/* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2021 Stefano Trettel
|
|
*
|
|
* Software repository: MoonCCD, https://github.com/stetre/moonccd
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
* copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*/
|
|
|
|
#include "internal.h"
|
|
|
|
ud_t *newuserdata(lua_State *L, void *handle, const char *mt, const char *tracename)
|
|
{
|
|
ud_t *ud;
|
|
/* we use handle as search key */
|
|
ud = (ud_t *)udata_new(L, sizeof(ud_t), (uint64_t)(uintptr_t)handle, mt);
|
|
memset(ud, 0, sizeof(ud_t));
|
|
ud->handle = handle;
|
|
MarkValid(ud);
|
|
if (trace_objects)
|
|
printf("create %s %p (%p)\n", tracename, (void *)ud, handle);
|
|
return ud;
|
|
}
|
|
|
|
int freeuserdata(lua_State *L, ud_t *ud, const char *tracename)
|
|
{
|
|
/* The 'Valid' mark prevents double calls when an object is explicitly destroyed,
|
|
* and subsequently deleted also by the GC (the ud sticks around until the GC
|
|
* collects it, so we mark it as invalid when the object is explicitly destroyed
|
|
* by the script, or implicitly destroyed because child of a destroyed object). */
|
|
int i;
|
|
if (!IsValid(ud))
|
|
return 0;
|
|
CancelValid(ud);
|
|
if (ud->info)
|
|
Free(L, ud->info);
|
|
for (i = 0; i < 6; i++)
|
|
if (ud->ref[i] != LUA_NOREF)
|
|
luaL_unref(L, LUA_REGISTRYINDEX, ud->ref[i]);
|
|
if (trace_objects)
|
|
printf("delete %s %p (%p)\n", tracename, (void *)ud, ud->handle);
|
|
udata_free(L, (uint64_t)(uintptr_t)ud->handle);
|
|
return 1;
|
|
}
|
|
|
|
static int freeifchild(lua_State *L, const void *mem, const char *mt, const void *parent_ud)
|
|
/* callback for udata_scan */
|
|
{
|
|
ud_t *ud = (ud_t *)mem;
|
|
(void)mt;
|
|
if (IsValid(ud) && (ud->parent_ud == parent_ud))
|
|
ud->destructor(L, ud);
|
|
return 0;
|
|
}
|
|
|
|
int freechildren(lua_State *L, const char *mt, ud_t *parent_ud)
|
|
/* calls the self destructor for all 'mt' objects that are children of the given parent_ud */
|
|
{
|
|
return udata_scan(L, mt, parent_ud, freeifchild);
|
|
}
|
|
|
|
int pushuserdata(lua_State *L, ud_t *ud)
|
|
{
|
|
if (!IsValid(ud))
|
|
return unexpected(L);
|
|
return udata_push(L, (uint64_t)(uintptr_t)ud->handle);
|
|
}
|
|
|
|
ud_t *userdata(const void *handle)
|
|
{
|
|
ud_t *ud = (ud_t *)udata_mem((uint64_t)(uintptr_t)handle);
|
|
if (ud && IsValid(ud))
|
|
return ud;
|
|
return NULL;
|
|
}
|
|
|
|
void *testxxx(lua_State *L, int arg, ud_t **udp, const char *mt)
|
|
{
|
|
ud_t *ud = (ud_t *)udata_test(L, arg, mt);
|
|
if (ud && IsValid(ud))
|
|
{
|
|
if (udp)
|
|
*udp = ud;
|
|
return ud->handle;
|
|
}
|
|
if (udp)
|
|
*udp = NULL;
|
|
return 0;
|
|
}
|
|
|
|
#if 0
|
|
void *testoneofxxx(lua_State *L, int arg, ud_t **udp, char **mtp)
|
|
{
|
|
void *handle = NULL;
|
|
int i = 0;
|
|
char *mt = NULL;
|
|
while((mt = mtp[i++]) != NULL)
|
|
{
|
|
handle = testxxx(L, arg, udp, mt);
|
|
if(handle) return handle;
|
|
}
|
|
if(udp) *udp = NULL;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
void *checkxxx(lua_State *L, int arg, ud_t **udp, const char *mt)
|
|
{
|
|
ud_t *ud = (ud_t *)udata_test(L, arg, mt);
|
|
if (ud && IsValid(ud))
|
|
{
|
|
if (udp)
|
|
*udp = ud;
|
|
return ud->handle;
|
|
}
|
|
lua_pushfstring(L, "not a %s", mt);
|
|
luaL_argerror(L, arg, lua_tostring(L, -1));
|
|
return 0;
|
|
}
|
|
|
|
void *optxxx(lua_State *L, int arg, ud_t **udp, const char *mt)
|
|
/* This differs from testxxx in that it fails in case the argument is
|
|
* something different than either none/nil or the expected userdata type
|
|
*/
|
|
{
|
|
if (lua_isnoneornil(L, arg))
|
|
{
|
|
if (udp)
|
|
*udp = NULL;
|
|
return 0;
|
|
}
|
|
return checkxxx(L, arg, udp, mt);
|
|
}
|
|
|
|
int pushxxx(lua_State *L, void *handle)
|
|
{
|
|
return udata_push(L, (uint64_t)(uintptr_t)handle);
|
|
}
|
|
|
|
void **checkxxxlist(lua_State *L, int arg, int *count, int *err, const char *mt)
|
|
/* xxx* checkxxxlist(lua_State *L, int arg, int *count, int *err)
|
|
* Checks if the variable at arg on the Lua stack is a list of xxx objects.
|
|
* On success, returns an array of xxx handles and sets its length in *count.
|
|
* The array s Malloc'd and must be released by the caller using Free(L, ...).
|
|
* On error, sets *err to ERR_XXX, *count to 0, and returns NULL.
|
|
*/
|
|
{
|
|
void **list;
|
|
int i;
|
|
|
|
*count = 0;
|
|
*err = 0;
|
|
if (lua_isnoneornil(L, arg))
|
|
{
|
|
*err = ERR_NOTPRESENT;
|
|
return NULL;
|
|
}
|
|
if (lua_type(L, arg) != LUA_TTABLE)
|
|
{
|
|
*err = ERR_TABLE;
|
|
return NULL;
|
|
}
|
|
*count = luaL_len(L, arg);
|
|
if (*count == 0)
|
|
{
|
|
*err = ERR_EMPTY;
|
|
return NULL;
|
|
}
|
|
list = (void **)MallocNoErr(L, sizeof(void *) * (*count));
|
|
|
|
if (!list)
|
|
{
|
|
*count = 0;
|
|
*err = ERR_MEMORY;
|
|
return NULL;
|
|
}
|
|
|
|
for (i = 0; i < *count; i++)
|
|
{
|
|
lua_rawgeti(L, arg, i + 1);
|
|
list[i] = (void *)(uintptr_t)testxxx(L, -1, NULL, mt);
|
|
if (!list[i])
|
|
{
|
|
Free(L, list);
|
|
*count = 0;
|
|
*err = ERR_TYPE;
|
|
return NULL;
|
|
}
|
|
lua_pop(L, 1);
|
|
}
|
|
return list;
|
|
}
|
|
|
|
int setmetatable(lua_State *L, const char *mt)
|
|
/* Sets the metatable of the table on top of the stack */
|
|
{
|
|
luaL_getmetatable(L, mt);
|
|
lua_setmetatable(L, -2);
|
|
return 0;
|
|
}
|