Reference.cpp 2.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  1. /**
  2. * Copyright (c) 2006-2017 LOVE Development Team
  3. *
  4. * This software is provided 'as-is', without any express or implied
  5. * warranty. In no event will the authors be held liable for any damages
  6. * arising from the use of this software.
  7. *
  8. * Permission is granted to anyone to use this software for any purpose,
  9. * including commercial applications, and to alter it and redistribute it
  10. * freely, subject to the following restrictions:
  11. *
  12. * 1. The origin of this software must not be misrepresented; you must not
  13. * claim that you wrote the original software. If you use this software
  14. * in a product, an acknowledgment in the product documentation would be
  15. * appreciated but is not required.
  16. * 2. Altered source versions must be plainly marked as such, and must not be
  17. * misrepresented as being the original software.
  18. * 3. This notice may not be removed or altered from any source distribution.
  19. **/
  20. #include "Reference.h"
  21. namespace love
  22. {
  23. const char REFERENCE_TABLE_NAME[] = "love-references";
  24. Reference::Reference()
  25. : pinnedL(nullptr)
  26. , idx(LUA_REFNIL)
  27. {
  28. }
  29. Reference::Reference(lua_State *L)
  30. : pinnedL(nullptr)
  31. , idx(LUA_REFNIL)
  32. {
  33. ref(L);
  34. }
  35. Reference::~Reference()
  36. {
  37. unref();
  38. }
  39. void Reference::ref(lua_State *L)
  40. {
  41. unref(); // Previously created reference needs to be cleared
  42. pinnedL = luax_getpinnedthread(L);
  43. luax_insist(L, LUA_REGISTRYINDEX, REFERENCE_TABLE_NAME);
  44. lua_insert(L, -2); // Move reference table behind value.
  45. idx = luaL_ref(L, -2);
  46. lua_pop(L, 1);
  47. }
  48. void Reference::unref()
  49. {
  50. if (idx != LUA_REFNIL)
  51. {
  52. // We use a pinned thread/coroutine for the Lua state because we know it
  53. // hasn't been garbage collected and is valid, as long as the whole lua
  54. // state is still open.
  55. luax_insist(pinnedL, LUA_REGISTRYINDEX, REFERENCE_TABLE_NAME);
  56. luaL_unref(pinnedL, -1, idx);
  57. lua_pop(pinnedL, 1);
  58. idx = LUA_REFNIL;
  59. }
  60. }
  61. void Reference::push(lua_State *L)
  62. {
  63. if (idx != LUA_REFNIL)
  64. {
  65. luax_insist(L, LUA_REGISTRYINDEX, REFERENCE_TABLE_NAME);
  66. lua_rawgeti(L, -1, idx);
  67. lua_remove(L, -2);
  68. }
  69. else
  70. lua_pushnil(L);
  71. }
  72. } // love