123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336 |
- /**
- * Copyright (c) 2006-2010 LOVE Development Team
- *
- * This software is provided 'as-is', without any express or implied
- * warranty. In no event will the authors be held liable for any damages
- * arising from the use of this software.
- *
- * Permission is granted to anyone to use this software for any purpose,
- * including commercial applications, and to alter it and redistribute it
- * freely, subject to the following restrictions:
- *
- * 1. The origin of this software must not be misrepresented; you must not
- * claim that you wrote the original software. If you use this software
- * in a product, an acknowledgment in the product documentation would be
- * appreciated but is not required.
- * 2. Altered source versions must be plainly marked as such, and must not be
- * misrepresented as being the original software.
- * 3. This notice may not be removed or altered from any source distribution.
- **/
- #include "wrap_Thread.h"
- namespace love
- {
- namespace thread
- {
- namespace sdl
- {
- Thread *luax_checkthread(lua_State *L, int idx)
- {
- return luax_checktype<Thread>(L, idx, "Thread", THREAD_THREAD_T);
- }
- int w_Thread_start(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- t->start();
- return 0;
- }
- int w_Thread_kill(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- t->kill();
- return 0;
- }
- int w_Thread_wait(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- t->wait();
- return 0;
- }
- int w_Thread_getName(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- lua_pushstring(L, t->getName().c_str());
- return 1;
- }
- int w_Thread_receive(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- std::string name = luaL_checkstring(L, 2);
- ThreadVariant *v = t->receive(name);
- if (!v)
- {
- lua_pushnil(L);
- return 1;
- }
- v->retain();
- t->clear(name);
- switch(v->type)
- {
- case BOOLEAN:
- lua_pushboolean(L, v->data.boolean);
- break;
- case NUMBER:
- lua_pushnumber(L, v->data.number);
- break;
- case STRING:
- lua_pushstring(L, v->data.string);
- break;
- case LUSERDATA:
- lua_pushlightuserdata(L, v->data.userdata);
- break;
- case FUSERDATA:
- const char *name;
- love::types.find(v->udatatype, name);
- ((love::Object *) v->data.userdata)->retain();
- luax_newtype(L, name, v->flags, v->data.userdata);
- break;
- default:
- lua_pushnil(L);
- break;
- }
- v->release();
- return 1;
- }
- int w_Thread_demand(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- std::string name = luaL_checkstring(L, 2);
- ThreadVariant *v = t->demand(name);
- if (!v)
- {
- lua_pushnil(L);
- return 1;
- }
- v->retain();
- t->clear(name);
- switch(v->type)
- {
- case BOOLEAN:
- lua_pushboolean(L, v->data.boolean);
- break;
- case NUMBER:
- lua_pushnumber(L, v->data.number);
- break;
- case STRING:
- lua_pushstring(L, v->data.string);
- break;
- case LUSERDATA:
- lua_pushlightuserdata(L, v->data.userdata);
- break;
- case FUSERDATA:
- const char *name;
- types.find(v->udatatype, name);
- ((love::Object *) v->data.userdata)->retain();
- luax_newtype(L, name, v->flags, v->data.userdata);
- break;
- default:
- lua_pushnil(L);
- break;
- }
- v->release();
- return 1;
- }
- int w_Thread_peek(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- std::string name = luaL_checkstring(L, 2);
- ThreadVariant *v = t->receive(name);
- if (!v)
- {
- lua_pushnil(L);
- return 1;
- }
- v->retain();
- switch(v->type)
- {
- case BOOLEAN:
- lua_pushboolean(L, v->data.boolean);
- break;
- case NUMBER:
- lua_pushnumber(L, v->data.number);
- break;
- case STRING:
- lua_pushstring(L, v->data.string);
- break;
- case LUSERDATA:
- lua_pushlightuserdata(L, v->data.userdata);
- break;
- case FUSERDATA:
- const char *name;
- types.find(v->udatatype, name);
- ((love::Object *) v->data.userdata)->retain();
- luax_newtype(L, name, v->flags, v->data.userdata);
- break;
- default:
- lua_pushnil(L);
- break;
- }
- v->release();
- return 1;
- }
- Type extractudatatype(lua_State * L, int idx)
- {
- Type t = INVALID_ID;
- if (!lua_isuserdata(L, idx))
- return t;
- if (luaL_getmetafield (L, idx, "__tostring") == 0)
- return t;
- lua_pushvalue(L, idx);
- int result = lua_pcall(L, 1, 1, 0);
- if (result == 0)
- types.find(lua_tostring(L, -1), t);
- if (result == 0 || result == LUA_ERRRUN)
- lua_pop(L, 1);
- return t;
- }
- int w_Thread_send(lua_State *L)
- {
- Thread *t = luax_checkthread(L, 1);
- std::string name = luaL_checkstring(L, 2);
- ThreadVariant *v;
- if (lua_isboolean(L, 3))
- {
- v = new ThreadVariant((bool) lua_toboolean(L, 3));
- }
- else if (lua_isnumber(L, 3))
- {
- v = new ThreadVariant(lua_tonumber(L, 3));
- }
- else if (lua_isstring(L, 3))
- {
- v = new ThreadVariant(lua_tostring(L, 3));
- }
- else if (lua_islightuserdata(L, 3))
- {
- v = new ThreadVariant(lua_touserdata(L, 3));
- }
- else if (lua_isuserdata(L, 3))
- {
- v = new ThreadVariant(extractudatatype(L, 3), lua_touserdata(L, 3));
- }
- else
- {
- return luaL_error(L, "Expected boolean, number, string or userdata");
- }
- t->send(name, v);
- v->release();
- return 0;
- }
- static const luaL_Reg type_functions[] = {
- { "start", w_Thread_start },
- { "kill", w_Thread_kill },
- { "wait", w_Thread_wait },
- { "getName", w_Thread_getName },
- { "receive", w_Thread_receive },
- { "demand", w_Thread_demand },
- { "peek", w_Thread_peek },
- { "send", w_Thread_send },
- { 0, 0 }
- };
- int luaopen_thread(lua_State *L)
- {
- return luax_register_type(L, "Thread", type_functions);
- }
- static ThreadModule *instance;
- int w_newThread(lua_State *L)
- {
- std::string name = luaL_checkstring(L, 1);
- if (lua_isstring(L, 2))
- luax_convobj(L, 2, "filesystem", "read");
- love::Data *data = luax_checktype<love::Data>(L, 2, "Data", DATA_T);
- Thread *t = instance->newThread(name, data);
- luax_newtype(L, "Thread", THREAD_THREAD_T, (void*)t);
- return 1;
- }
- int w_getThreads(lua_State *L)
- {
- unsigned count = instance->getThreadCount();
- Thread **list = new Thread*[count];
- instance->getThreads(list);
- lua_newtable(L);
- for (unsigned int i = 0; i<count; i++)
- {
- luax_newtype(L, "Thread", THREAD_THREAD_T, (void*) list[i]);
- list[i]->lock();
- list[i]->retain();
- list[i]->unlock();
- lua_setfield(L, -2, list[i]->getName().c_str());
- }
- delete[] list;
- return 1;
- }
- int w_getThread(lua_State *L)
- {
- std::string name = luaL_checkstring(L, 1);
- Thread *t = instance->getThread(name);
- if (t)
- {
- luax_newtype(L, "Thread", THREAD_THREAD_T, (void*)t);
- t->lock();
- t->retain();
- t->unlock();
- }
- else
- lua_pushnil(L);
- return 1;
- }
- // List of functions to wrap.
- static const luaL_Reg module_functions[] = {
- { "newThread", w_newThread },
- { "getThread", w_getThread },
- { "getThreads", w_getThreads },
- { 0, 0 }
- };
- static const lua_CFunction types[] = {
- luaopen_thread,
- 0
- };
- int luaopen_love_thread(lua_State *L)
- {
- if(instance == 0)
- {
- try
- {
- instance = new ThreadModule();
- }
- catch(Exception & e)
- {
- return luaL_error(L, e.what());
- }
- }
- else
- instance->retain();
- WrappedModule w;
- w.module = instance;
- w.name = "thread";
- w.flags = MODULE_T;
- w.functions = module_functions;
- w.types = types;
- return luax_register_module(L, w);
- }
- }
- }
- }
|