lj_clib.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  1. /*
  2. ** FFI C library loader.
  3. ** Copyright (C) 2005-2023 Mike Pall. See Copyright Notice in luajit.h
  4. */
  5. #include "lj_obj.h"
  6. #if LJ_HASFFI
  7. #include "lj_gc.h"
  8. #include "lj_err.h"
  9. #include "lj_tab.h"
  10. #include "lj_str.h"
  11. #include "lj_udata.h"
  12. #include "lj_ctype.h"
  13. #include "lj_cconv.h"
  14. #include "lj_cdata.h"
  15. #include "lj_clib.h"
  16. #include "lj_strfmt.h"
  17. /* -- OS-specific functions ----------------------------------------------- */
  18. #if LJ_TARGET_DLOPEN
  19. #include <dlfcn.h>
  20. #include <stdio.h>
  21. #if defined(RTLD_DEFAULT) && !defined(NO_RTLD_DEFAULT)
  22. #define CLIB_DEFHANDLE RTLD_DEFAULT
  23. #elif LJ_TARGET_OSX || LJ_TARGET_BSD
  24. #define CLIB_DEFHANDLE ((void *)(intptr_t)-2)
  25. #else
  26. #define CLIB_DEFHANDLE NULL
  27. #endif
  28. LJ_NORET LJ_NOINLINE static void clib_error_(lua_State *L)
  29. {
  30. lj_err_callermsg(L, dlerror());
  31. }
  32. #define clib_error(L, fmt, name) clib_error_(L)
  33. #if LJ_TARGET_CYGWIN
  34. #define CLIB_SOPREFIX "cyg"
  35. #else
  36. #define CLIB_SOPREFIX "lib"
  37. #endif
  38. #if LJ_TARGET_OSX
  39. #define CLIB_SOEXT "%s.dylib"
  40. #elif LJ_TARGET_CYGWIN
  41. #define CLIB_SOEXT "%s.dll"
  42. #else
  43. #define CLIB_SOEXT "%s.so"
  44. #endif
  45. static const char *clib_extname(lua_State *L, const char *name)
  46. {
  47. if (!strchr(name, '/')
  48. #if LJ_TARGET_CYGWIN
  49. && !strchr(name, '\\')
  50. #endif
  51. ) {
  52. if (!strchr(name, '.')) {
  53. name = lj_strfmt_pushf(L, CLIB_SOEXT, name);
  54. L->top--;
  55. #if LJ_TARGET_CYGWIN
  56. } else {
  57. return name;
  58. #endif
  59. }
  60. if (!(name[0] == CLIB_SOPREFIX[0] && name[1] == CLIB_SOPREFIX[1] &&
  61. name[2] == CLIB_SOPREFIX[2])) {
  62. name = lj_strfmt_pushf(L, CLIB_SOPREFIX "%s", name);
  63. L->top--;
  64. }
  65. }
  66. return name;
  67. }
  68. /* Check for a recognized ld script line. */
  69. static const char *clib_check_lds(lua_State *L, const char *buf)
  70. {
  71. char *p, *e;
  72. if ((!strncmp(buf, "GROUP", 5) || !strncmp(buf, "INPUT", 5)) &&
  73. (p = strchr(buf, '('))) {
  74. while (*++p == ' ') ;
  75. for (e = p; *e && *e != ' ' && *e != ')'; e++) ;
  76. return strdata(lj_str_new(L, p, e-p));
  77. }
  78. return NULL;
  79. }
  80. /* Quick and dirty solution to resolve shared library name from ld script. */
  81. static const char *clib_resolve_lds(lua_State *L, const char *name)
  82. {
  83. FILE *fp = fopen(name, "r");
  84. const char *p = NULL;
  85. if (fp) {
  86. char buf[256];
  87. if (fgets(buf, sizeof(buf), fp)) {
  88. if (!strncmp(buf, "/* GNU ld script", 16)) { /* ld script magic? */
  89. while (fgets(buf, sizeof(buf), fp)) { /* Check all lines. */
  90. p = clib_check_lds(L, buf);
  91. if (p) break;
  92. }
  93. } else { /* Otherwise check only the first line. */
  94. p = clib_check_lds(L, buf);
  95. }
  96. }
  97. fclose(fp);
  98. }
  99. return p;
  100. }
  101. static void *clib_loadlib(lua_State *L, const char *name, int global)
  102. {
  103. void *h = dlopen(clib_extname(L, name),
  104. RTLD_LAZY | (global?RTLD_GLOBAL:RTLD_LOCAL));
  105. if (!h) {
  106. const char *e, *err = dlerror();
  107. if (err && *err == '/' && (e = strchr(err, ':')) &&
  108. (name = clib_resolve_lds(L, strdata(lj_str_new(L, err, e-err))))) {
  109. h = dlopen(name, RTLD_LAZY | (global?RTLD_GLOBAL:RTLD_LOCAL));
  110. if (h) return h;
  111. err = dlerror();
  112. }
  113. if (!err) err = "dlopen failed";
  114. lj_err_callermsg(L, err);
  115. }
  116. return h;
  117. }
  118. static void clib_unloadlib(CLibrary *cl)
  119. {
  120. if (cl->handle && cl->handle != CLIB_DEFHANDLE)
  121. dlclose(cl->handle);
  122. }
  123. static void *clib_getsym(CLibrary *cl, const char *name)
  124. {
  125. void *p = dlsym(cl->handle, name);
  126. return p;
  127. }
  128. #elif LJ_TARGET_WINDOWS
  129. #define WIN32_LEAN_AND_MEAN
  130. #include <windows.h>
  131. #ifndef GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS
  132. #define GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS 4
  133. #define GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT 2
  134. BOOL WINAPI GetModuleHandleExA(DWORD, LPCSTR, HMODULE*);
  135. #endif
  136. #define CLIB_DEFHANDLE ((void *)-1)
  137. /* Default libraries. */
  138. enum {
  139. CLIB_HANDLE_EXE,
  140. #if !LJ_TARGET_UWP
  141. CLIB_HANDLE_DLL,
  142. CLIB_HANDLE_CRT,
  143. CLIB_HANDLE_KERNEL32,
  144. CLIB_HANDLE_USER32,
  145. CLIB_HANDLE_GDI32,
  146. #endif
  147. CLIB_HANDLE_MAX
  148. };
  149. static void *clib_def_handle[CLIB_HANDLE_MAX];
  150. LJ_NORET LJ_NOINLINE static void clib_error(lua_State *L, const char *fmt,
  151. const char *name)
  152. {
  153. DWORD err = GetLastError();
  154. #if LJ_TARGET_XBOXONE
  155. wchar_t wbuf[128];
  156. char buf[128*2];
  157. if (!FormatMessageW(FORMAT_MESSAGE_IGNORE_INSERTS|FORMAT_MESSAGE_FROM_SYSTEM,
  158. NULL, err, 0, wbuf, sizeof(wbuf)/sizeof(wchar_t), NULL) ||
  159. !WideCharToMultiByte(CP_ACP, 0, wbuf, 128, buf, 128*2, NULL, NULL))
  160. #else
  161. char buf[128];
  162. if (!FormatMessageA(FORMAT_MESSAGE_IGNORE_INSERTS|FORMAT_MESSAGE_FROM_SYSTEM,
  163. NULL, err, 0, buf, sizeof(buf), NULL))
  164. #endif
  165. buf[0] = '\0';
  166. lj_err_callermsg(L, lj_strfmt_pushf(L, fmt, name, buf));
  167. }
  168. static int clib_needext(const char *s)
  169. {
  170. while (*s) {
  171. if (*s == '/' || *s == '\\' || *s == '.') return 0;
  172. s++;
  173. }
  174. return 1;
  175. }
  176. static const char *clib_extname(lua_State *L, const char *name)
  177. {
  178. if (clib_needext(name)) {
  179. name = lj_strfmt_pushf(L, "%s.dll", name);
  180. L->top--;
  181. }
  182. return name;
  183. }
  184. static void *clib_loadlib(lua_State *L, const char *name, int global)
  185. {
  186. DWORD oldwerr = GetLastError();
  187. void *h = LJ_WIN_LOADLIBA(clib_extname(L, name));
  188. if (!h) clib_error(L, "cannot load module " LUA_QS ": %s", name);
  189. SetLastError(oldwerr);
  190. UNUSED(global);
  191. return h;
  192. }
  193. static void clib_unloadlib(CLibrary *cl)
  194. {
  195. if (cl->handle == CLIB_DEFHANDLE) {
  196. #if !LJ_TARGET_UWP
  197. MSize i;
  198. for (i = CLIB_HANDLE_KERNEL32; i < CLIB_HANDLE_MAX; i++) {
  199. void *h = clib_def_handle[i];
  200. if (h) {
  201. clib_def_handle[i] = NULL;
  202. FreeLibrary((HINSTANCE)h);
  203. }
  204. }
  205. #endif
  206. } else if (cl->handle) {
  207. FreeLibrary((HINSTANCE)cl->handle);
  208. }
  209. }
  210. #if LJ_TARGET_UWP
  211. EXTERN_C IMAGE_DOS_HEADER __ImageBase;
  212. #endif
  213. static void *clib_getsym(CLibrary *cl, const char *name)
  214. {
  215. void *p = NULL;
  216. if (cl->handle == CLIB_DEFHANDLE) { /* Search default libraries. */
  217. MSize i;
  218. for (i = 0; i < CLIB_HANDLE_MAX; i++) {
  219. HINSTANCE h = (HINSTANCE)clib_def_handle[i];
  220. if (!(void *)h) { /* Resolve default library handles (once). */
  221. #if LJ_TARGET_UWP
  222. h = (HINSTANCE)&__ImageBase;
  223. #else
  224. switch (i) {
  225. case CLIB_HANDLE_EXE: GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, NULL, &h); break;
  226. case CLIB_HANDLE_DLL:
  227. GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS|GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
  228. (const char *)clib_def_handle, &h);
  229. break;
  230. case CLIB_HANDLE_CRT:
  231. GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS|GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
  232. (const char *)&_fmode, &h);
  233. break;
  234. case CLIB_HANDLE_KERNEL32: h = LJ_WIN_LOADLIBA("kernel32.dll"); break;
  235. case CLIB_HANDLE_USER32: h = LJ_WIN_LOADLIBA("user32.dll"); break;
  236. case CLIB_HANDLE_GDI32: h = LJ_WIN_LOADLIBA("gdi32.dll"); break;
  237. }
  238. if (!h) continue;
  239. #endif
  240. clib_def_handle[i] = (void *)h;
  241. }
  242. p = (void *)GetProcAddress(h, name);
  243. if (p) break;
  244. }
  245. } else {
  246. p = (void *)GetProcAddress((HINSTANCE)cl->handle, name);
  247. }
  248. return p;
  249. }
  250. #else
  251. #define CLIB_DEFHANDLE NULL
  252. LJ_NORET LJ_NOINLINE static void clib_error(lua_State *L, const char *fmt,
  253. const char *name)
  254. {
  255. lj_err_callermsg(L, lj_strfmt_pushf(L, fmt, name, "no support for this OS"));
  256. }
  257. static void *clib_loadlib(lua_State *L, const char *name, int global)
  258. {
  259. lj_err_callermsg(L, "no support for loading dynamic libraries for this OS");
  260. UNUSED(name); UNUSED(global);
  261. return NULL;
  262. }
  263. static void clib_unloadlib(CLibrary *cl)
  264. {
  265. UNUSED(cl);
  266. }
  267. static void *clib_getsym(CLibrary *cl, const char *name)
  268. {
  269. UNUSED(cl); UNUSED(name);
  270. return NULL;
  271. }
  272. #endif
  273. /* -- C library indexing -------------------------------------------------- */
  274. #if LJ_TARGET_X86 && LJ_ABI_WIN
  275. /* Compute argument size for fastcall/stdcall functions. */
  276. static CTSize clib_func_argsize(CTState *cts, CType *ct)
  277. {
  278. CTSize n = 0;
  279. while (ct->sib) {
  280. CType *d;
  281. ct = ctype_get(cts, ct->sib);
  282. if (ctype_isfield(ct->info)) {
  283. d = ctype_rawchild(cts, ct);
  284. n += ((d->size + 3) & ~3);
  285. }
  286. }
  287. return n;
  288. }
  289. #endif
  290. /* Get redirected or mangled external symbol. */
  291. static const char *clib_extsym(CTState *cts, CType *ct, GCstr *name)
  292. {
  293. if (ct->sib) {
  294. CType *ctf = ctype_get(cts, ct->sib);
  295. if (ctype_isxattrib(ctf->info, CTA_REDIR))
  296. return strdata(gco2str(gcref(ctf->name)));
  297. }
  298. return strdata(name);
  299. }
  300. /* Index a C library by name. */
  301. TValue *lj_clib_index(lua_State *L, CLibrary *cl, GCstr *name)
  302. {
  303. TValue *tv = lj_tab_setstr(L, cl->cache, name);
  304. if (LJ_UNLIKELY(tvisnil(tv))) {
  305. CTState *cts = ctype_cts(L);
  306. CType *ct;
  307. CTypeID id = lj_ctype_getname(cts, &ct, name, CLNS_INDEX);
  308. if (!id)
  309. lj_err_callerv(L, LJ_ERR_FFI_NODECL, strdata(name));
  310. if (ctype_isconstval(ct->info)) {
  311. CType *ctt = ctype_child(cts, ct);
  312. lj_assertCTS(ctype_isinteger(ctt->info) && ctt->size <= 4,
  313. "only 32 bit const supported"); /* NYI */
  314. if ((ctt->info & CTF_UNSIGNED) && (int32_t)ct->size < 0)
  315. setnumV(tv, (lua_Number)(uint32_t)ct->size);
  316. else
  317. setintV(tv, (int32_t)ct->size);
  318. } else {
  319. const char *sym = clib_extsym(cts, ct, name);
  320. #if LJ_TARGET_WINDOWS
  321. DWORD oldwerr = GetLastError();
  322. #endif
  323. void *p = clib_getsym(cl, sym);
  324. GCcdata *cd;
  325. lj_assertCTS(ctype_isfunc(ct->info) || ctype_isextern(ct->info),
  326. "unexpected ctype %08x in clib", ct->info);
  327. #if LJ_TARGET_X86 && LJ_ABI_WIN
  328. /* Retry with decorated name for fastcall/stdcall functions. */
  329. if (!p && ctype_isfunc(ct->info)) {
  330. CTInfo cconv = ctype_cconv(ct->info);
  331. if (cconv == CTCC_FASTCALL || cconv == CTCC_STDCALL) {
  332. CTSize sz = clib_func_argsize(cts, ct);
  333. const char *symd = lj_strfmt_pushf(L,
  334. cconv == CTCC_FASTCALL ? "@%s@%d" : "_%s@%d",
  335. sym, sz);
  336. L->top--;
  337. p = clib_getsym(cl, symd);
  338. }
  339. }
  340. #endif
  341. if (!p)
  342. clib_error(L, "cannot resolve symbol " LUA_QS ": %s", sym);
  343. #if LJ_TARGET_WINDOWS
  344. SetLastError(oldwerr);
  345. #endif
  346. cd = lj_cdata_new(cts, id, CTSIZE_PTR);
  347. *(void **)cdataptr(cd) = p;
  348. setcdataV(L, tv, cd);
  349. lj_gc_anybarriert(L, cl->cache);
  350. }
  351. }
  352. return tv;
  353. }
  354. /* -- C library management ------------------------------------------------ */
  355. /* Create a new CLibrary object and push it on the stack. */
  356. static CLibrary *clib_new(lua_State *L, GCtab *mt)
  357. {
  358. GCtab *t = lj_tab_new(L, 0, 0);
  359. GCudata *ud = lj_udata_new(L, sizeof(CLibrary), t);
  360. CLibrary *cl = (CLibrary *)uddata(ud);
  361. cl->cache = t;
  362. ud->udtype = UDTYPE_FFI_CLIB;
  363. /* NOBARRIER: The GCudata is new (marked white). */
  364. setgcref(ud->metatable, obj2gco(mt));
  365. setudataV(L, L->top++, ud);
  366. return cl;
  367. }
  368. /* Load a C library. */
  369. void lj_clib_load(lua_State *L, GCtab *mt, GCstr *name, int global)
  370. {
  371. void *handle = clib_loadlib(L, strdata(name), global);
  372. CLibrary *cl = clib_new(L, mt);
  373. cl->handle = handle;
  374. }
  375. /* Unload a C library. */
  376. void lj_clib_unload(CLibrary *cl)
  377. {
  378. clib_unloadlib(cl);
  379. cl->handle = NULL;
  380. }
  381. /* Create the default C library object. */
  382. void lj_clib_default(lua_State *L, GCtab *mt)
  383. {
  384. CLibrary *cl = clib_new(L, mt);
  385. cl->handle = CLIB_DEFHANDLE;
  386. }
  387. #endif