mirror of
https://github.com/LuaJIT/LuaJIT.git
synced 2025-04-21 14:23:26 +00:00
Compare commits
16 Commits
d86cd7f3a5
...
5bf05586f8
Author | SHA1 | Date | |
---|---|---|---|
![]() |
5bf05586f8 | ||
![]() |
e3c70a7d81 | ||
![]() |
7db2d1b12a | ||
![]() |
e0551670c9 | ||
![]() |
85c3f2fb6f | ||
![]() |
eee16efa77 | ||
![]() |
4219efae43 | ||
![]() |
0254770582 | ||
![]() |
f14556234c | ||
![]() |
d508715ab6 | ||
![]() |
e27ee68817 | ||
![]() |
55a42da36e | ||
![]() |
423ac2144b | ||
![]() |
54dc2fa5d7 | ||
![]() |
b1179ea5f7 | ||
![]() |
5eb9509468 |
2
Makefile
2
Makefile
@ -110,7 +110,7 @@ else
|
|||||||
endif
|
endif
|
||||||
TARGET_SYS?= $(HOST_SYS)
|
TARGET_SYS?= $(HOST_SYS)
|
||||||
|
|
||||||
ifeq (Darwin,$(TARGET_SYS))
|
ifneq (,$(filter $(TARGET_SYS),Darwin iOS))
|
||||||
INSTALL_SONAME= $(INSTALL_DYLIBNAME)
|
INSTALL_SONAME= $(INSTALL_DYLIBNAME)
|
||||||
INSTALL_SOSHORT1= $(INSTALL_DYLIBSHORT1)
|
INSTALL_SOSHORT1= $(INSTALL_DYLIBSHORT1)
|
||||||
INSTALL_SOSHORT2= $(INSTALL_DYLIBSHORT2)
|
INSTALL_SOSHORT2= $(INSTALL_DYLIBSHORT2)
|
||||||
|
@ -117,7 +117,7 @@ hold all user-configurable settings:
|
|||||||
<li><tt>Makefile</tt> has settings for <b>installing</b> LuaJIT (POSIX
|
<li><tt>Makefile</tt> has settings for <b>installing</b> LuaJIT (POSIX
|
||||||
only).</li>
|
only).</li>
|
||||||
<li><tt>src/Makefile</tt> has settings for <b>compiling</b> LuaJIT
|
<li><tt>src/Makefile</tt> has settings for <b>compiling</b> LuaJIT
|
||||||
under POSIX, MinGW or Cygwin.</li>
|
under POSIX or MinGW.</li>
|
||||||
<li><tt>src/msvcbuild.bat</tt> has settings for compiling LuaJIT with
|
<li><tt>src/msvcbuild.bat</tt> has settings for compiling LuaJIT with
|
||||||
MSVC (Visual Studio).</li>
|
MSVC (Visual Studio).</li>
|
||||||
</ul>
|
</ul>
|
||||||
@ -195,10 +195,8 @@ Obviously the prefixes given during build and installation need to be the same.
|
|||||||
<h2 id="windows">Windows Systems</h2>
|
<h2 id="windows">Windows Systems</h2>
|
||||||
<h3>Prerequisites</h3>
|
<h3>Prerequisites</h3>
|
||||||
<p>
|
<p>
|
||||||
Either install one of the open source SDKs
|
Either install the open source SDK <a href="http://mingw.org/"><span class="ext">»</span> MinGW</a>,
|
||||||
(<a href="http://mingw.org/"><span class="ext">»</span> MinGW</a> or
|
which comes with a modified GCC plus the required development headers.
|
||||||
<a href="https://www.cygwin.com/"><span class="ext">»</span> Cygwin</a>), which come with a modified
|
|
||||||
GCC plus the required development headers.
|
|
||||||
Or install Microsoft's Visual Studio (MSVC).
|
Or install Microsoft's Visual Studio (MSVC).
|
||||||
</p>
|
</p>
|
||||||
<h3>Building with MSVC</h3>
|
<h3>Building with MSVC</h3>
|
||||||
@ -217,9 +215,9 @@ Then follow the installation instructions below.
|
|||||||
<p>
|
<p>
|
||||||
For an x64 to ARM64 cross-build run this first: <tt>vcvarsall.bat x64_arm64</tt>
|
For an x64 to ARM64 cross-build run this first: <tt>vcvarsall.bat x64_arm64</tt>
|
||||||
</p>
|
</p>
|
||||||
<h3>Building with MinGW or Cygwin</h3>
|
<h3>Building with MinGW</h3>
|
||||||
<p>
|
<p>
|
||||||
Open a command prompt window and make sure the MinGW or Cygwin programs
|
Open a command prompt window and make sure the MinGW programs
|
||||||
are in your path. Then <tt>cd</tt> to the directory of the git repository.
|
are in your path. Then <tt>cd</tt> to the directory of the git repository.
|
||||||
Then run this command for MinGW:
|
Then run this command for MinGW:
|
||||||
</p>
|
</p>
|
||||||
@ -227,12 +225,6 @@ Then run this command for MinGW:
|
|||||||
mingw32-make
|
mingw32-make
|
||||||
</pre>
|
</pre>
|
||||||
<p>
|
<p>
|
||||||
Or this command for Cygwin:
|
|
||||||
</p>
|
|
||||||
<pre class="code">
|
|
||||||
make
|
|
||||||
</pre>
|
|
||||||
<p>
|
|
||||||
Then follow the installation instructions below.
|
Then follow the installation instructions below.
|
||||||
</p>
|
</p>
|
||||||
<h3>Installing LuaJIT</h3>
|
<h3>Installing LuaJIT</h3>
|
||||||
@ -249,6 +241,19 @@ absolute path names — all modules are loaded relative to the
|
|||||||
directory where <tt>luajit.exe</tt> is installed
|
directory where <tt>luajit.exe</tt> is installed
|
||||||
(see <tt>src/luaconf.h</tt>).
|
(see <tt>src/luaconf.h</tt>).
|
||||||
</p>
|
</p>
|
||||||
|
<p>
|
||||||
|
The final directory layout should look like this:
|
||||||
|
</p>
|
||||||
|
<pre class="code">
|
||||||
|
├── luajit.exe
|
||||||
|
├── lua51.dll
|
||||||
|
├── <- put your own classic Lua/C API modules (*.dll) here
|
||||||
|
└── lua
|
||||||
|
├── <- put your own Lua modules (*.lua) here
|
||||||
|
└── jit
|
||||||
|
├── bc.lua
|
||||||
|
└── (etc …)
|
||||||
|
</pre>
|
||||||
|
|
||||||
<h2 id="cross">Cross-compiling LuaJIT</h2>
|
<h2 id="cross">Cross-compiling LuaJIT</h2>
|
||||||
<p>
|
<p>
|
||||||
|
@ -25,7 +25,7 @@ lib_ffi.o: lib_ffi.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
|
|||||||
lib_init.o: lib_init.c lua.h luaconf.h lauxlib.h lualib.h lj_arch.h
|
lib_init.o: lib_init.c lua.h luaconf.h lauxlib.h lualib.h lj_arch.h
|
||||||
lib_io.o: lib_io.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
|
lib_io.o: lib_io.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
|
||||||
lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_state.h \
|
lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_state.h \
|
||||||
lj_strfmt.h lj_ff.h lj_ffdef.h lj_lib.h lj_libdef.h
|
lj_strfmt.h lj_ff.h lj_ffdef.h lj_lib.h lj_strscan.h lj_libdef.h
|
||||||
lib_jit.o: lib_jit.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
|
lib_jit.o: lib_jit.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
|
||||||
lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_debug.h lj_str.h lj_tab.h \
|
lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_debug.h lj_str.h lj_tab.h \
|
||||||
lj_state.h lj_bc.h lj_ctype.h lj_ir.h lj_jit.h lj_ircall.h lj_iropt.h \
|
lj_state.h lj_bc.h lj_ctype.h lj_ir.h lj_jit.h lj_ircall.h lj_iropt.h \
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
#include "lj_strfmt.h"
|
#include "lj_strfmt.h"
|
||||||
#include "lj_ff.h"
|
#include "lj_ff.h"
|
||||||
#include "lj_lib.h"
|
#include "lj_lib.h"
|
||||||
|
#include "lj_strscan.h"
|
||||||
|
|
||||||
/* Userdata payload for I/O file. */
|
/* Userdata payload for I/O file. */
|
||||||
typedef struct IOFileUD {
|
typedef struct IOFileUD {
|
||||||
@ -323,13 +324,14 @@ LJLIB_CF(io_method_seek)
|
|||||||
FILE *fp = io_tofile(L)->fp;
|
FILE *fp = io_tofile(L)->fp;
|
||||||
int opt = lj_lib_checkopt(L, 2, 1, "\3set\3cur\3end");
|
int opt = lj_lib_checkopt(L, 2, 1, "\3set\3cur\3end");
|
||||||
int64_t ofs = 0;
|
int64_t ofs = 0;
|
||||||
cTValue *o;
|
TValue *o;
|
||||||
int res;
|
int res;
|
||||||
if (opt == 0) opt = SEEK_SET;
|
if (opt == 0) opt = SEEK_SET;
|
||||||
else if (opt == 1) opt = SEEK_CUR;
|
else if (opt == 1) opt = SEEK_CUR;
|
||||||
else if (opt == 2) opt = SEEK_END;
|
else if (opt == 2) opt = SEEK_END;
|
||||||
o = L->base+2;
|
o = L->base+2;
|
||||||
if (o < L->top) {
|
if (o < L->top) {
|
||||||
|
if (tvisstr(o)) lj_strscan_num(strV(o), o);
|
||||||
if (tvisint(o))
|
if (tvisint(o))
|
||||||
ofs = (int64_t)intV(o);
|
ofs = (int64_t)intV(o);
|
||||||
else if (tvisnum(o))
|
else if (tvisnum(o))
|
||||||
|
@ -179,7 +179,7 @@ static const void *bcread_varinfo(GCproto *pt)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Read a single constant key/value of a template table. */
|
/* Read a single constant key/value of a template table. */
|
||||||
static void bcread_ktabk(LexState *ls, TValue *o)
|
static void bcread_ktabk(LexState *ls, TValue *o, GCtab *t)
|
||||||
{
|
{
|
||||||
MSize tp = bcread_uleb128(ls);
|
MSize tp = bcread_uleb128(ls);
|
||||||
if (tp >= BCDUMP_KTAB_STR) {
|
if (tp >= BCDUMP_KTAB_STR) {
|
||||||
@ -191,6 +191,8 @@ static void bcread_ktabk(LexState *ls, TValue *o)
|
|||||||
} else if (tp == BCDUMP_KTAB_NUM) {
|
} else if (tp == BCDUMP_KTAB_NUM) {
|
||||||
o->u32.lo = bcread_uleb128(ls);
|
o->u32.lo = bcread_uleb128(ls);
|
||||||
o->u32.hi = bcread_uleb128(ls);
|
o->u32.hi = bcread_uleb128(ls);
|
||||||
|
} else if (tp == BCDUMP_KTAB_NIL) { /* Restore nil value marker. */
|
||||||
|
settabV(ls->L, o, t);
|
||||||
} else {
|
} else {
|
||||||
lj_assertLS(tp <= BCDUMP_KTAB_TRUE, "bad constant type %d", tp);
|
lj_assertLS(tp <= BCDUMP_KTAB_TRUE, "bad constant type %d", tp);
|
||||||
setpriV(o, ~tp);
|
setpriV(o, ~tp);
|
||||||
@ -207,15 +209,15 @@ static GCtab *bcread_ktab(LexState *ls)
|
|||||||
MSize i;
|
MSize i;
|
||||||
TValue *o = tvref(t->array);
|
TValue *o = tvref(t->array);
|
||||||
for (i = 0; i < narray; i++, o++)
|
for (i = 0; i < narray; i++, o++)
|
||||||
bcread_ktabk(ls, o);
|
bcread_ktabk(ls, o, t);
|
||||||
}
|
}
|
||||||
if (nhash) { /* Read hash entries. */
|
if (nhash) { /* Read hash entries. */
|
||||||
MSize i;
|
MSize i;
|
||||||
for (i = 0; i < nhash; i++) {
|
for (i = 0; i < nhash; i++) {
|
||||||
TValue key;
|
TValue key;
|
||||||
bcread_ktabk(ls, &key);
|
bcread_ktabk(ls, &key, t);
|
||||||
lj_assertLS(!tvisnil(&key), "nil key");
|
lj_assertLS(!tvisnil(&key), "nil key");
|
||||||
bcread_ktabk(ls, lj_tab_set(ls->L, t, &key));
|
bcread_ktabk(ls, lj_tab_set(ls->L, t, &key), t);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return t;
|
return t;
|
||||||
|
@ -71,6 +71,8 @@ static void bcwrite_ktabk(BCWriteCtx *ctx, cTValue *o, int narrow)
|
|||||||
*p++ = BCDUMP_KTAB_NUM;
|
*p++ = BCDUMP_KTAB_NUM;
|
||||||
p = lj_strfmt_wuleb128(p, o->u32.lo);
|
p = lj_strfmt_wuleb128(p, o->u32.lo);
|
||||||
p = lj_strfmt_wuleb128(p, o->u32.hi);
|
p = lj_strfmt_wuleb128(p, o->u32.hi);
|
||||||
|
} else if (tvistab(o)) { /* Write the nil value marker as a nil. */
|
||||||
|
*p++ = BCDUMP_KTAB_NIL;
|
||||||
} else {
|
} else {
|
||||||
lj_assertBCW(tvispri(o), "unhandled type %d", itype(o));
|
lj_assertBCW(tvispri(o), "unhandled type %d", itype(o));
|
||||||
*p++ = BCDUMP_KTAB_NIL+~itype(o);
|
*p++ = BCDUMP_KTAB_NIL+~itype(o);
|
||||||
@ -133,7 +135,7 @@ static void bcwrite_ktab_sorted_hash(BCWriteCtx *ctx, Node *node, MSize nhash)
|
|||||||
TValue **heap = ctx->heap;
|
TValue **heap = ctx->heap;
|
||||||
MSize i = nhash;
|
MSize i = nhash;
|
||||||
for (;; node--) { /* Build heap. */
|
for (;; node--) { /* Build heap. */
|
||||||
if (!tvisnil(&node->key)) {
|
if (!tvisnil(&node->val)) {
|
||||||
bcwrite_ktabk_heap_insert(heap, --i, nhash, &node->key);
|
bcwrite_ktabk_heap_insert(heap, --i, nhash, &node->key);
|
||||||
if (i == 0) break;
|
if (i == 0) break;
|
||||||
}
|
}
|
||||||
@ -163,7 +165,7 @@ static void bcwrite_ktab(BCWriteCtx *ctx, char *p, const GCtab *t)
|
|||||||
MSize i, hmask = t->hmask;
|
MSize i, hmask = t->hmask;
|
||||||
Node *node = noderef(t->node);
|
Node *node = noderef(t->node);
|
||||||
for (i = 0; i <= hmask; i++)
|
for (i = 0; i <= hmask; i++)
|
||||||
nhash += !tvisnil(&node[i].key);
|
nhash += !tvisnil(&node[i].val);
|
||||||
}
|
}
|
||||||
/* Write number of array slots and hash slots. */
|
/* Write number of array slots and hash slots. */
|
||||||
p = lj_strfmt_wuleb128(p, narray);
|
p = lj_strfmt_wuleb128(p, narray);
|
||||||
@ -184,7 +186,7 @@ static void bcwrite_ktab(BCWriteCtx *ctx, char *p, const GCtab *t)
|
|||||||
} else {
|
} else {
|
||||||
MSize i = nhash;
|
MSize i = nhash;
|
||||||
for (;; node--)
|
for (;; node--)
|
||||||
if (!tvisnil(&node->key)) {
|
if (!tvisnil(&node->val)) {
|
||||||
bcwrite_ktabk(ctx, &node->key, 0);
|
bcwrite_ktabk(ctx, &node->key, 0);
|
||||||
bcwrite_ktabk(ctx, &node->val, 1);
|
bcwrite_ktabk(ctx, &node->val, 1);
|
||||||
if (--i == 0) break;
|
if (--i == 0) break;
|
||||||
|
@ -262,6 +262,14 @@ static void *callback_mcode_init(global_State *g, uint32_t *page)
|
|||||||
#define CCPROT_CREATE 0
|
#define CCPROT_CREATE 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Check for macOS hardened runtime. */
|
||||||
|
#if LUAJIT_SECURITY_MCODE != 0 && defined(MAP_JIT) && __ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__ >= 110000
|
||||||
|
#include <pthread.h>
|
||||||
|
#define CCMAP_CREATE MAP_JIT
|
||||||
|
#else
|
||||||
|
#define CCMAP_CREATE 0
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Allocate and initialize area for callback function pointers. */
|
/* Allocate and initialize area for callback function pointers. */
|
||||||
@ -276,10 +284,13 @@ static void callback_mcode_new(CTState *cts)
|
|||||||
if (!p)
|
if (!p)
|
||||||
lj_err_caller(cts->L, LJ_ERR_FFI_CBACKOV);
|
lj_err_caller(cts->L, LJ_ERR_FFI_CBACKOV);
|
||||||
#elif LJ_TARGET_POSIX
|
#elif LJ_TARGET_POSIX
|
||||||
p = mmap(NULL, sz, (PROT_READ|PROT_WRITE|CCPROT_CREATE), MAP_PRIVATE|MAP_ANONYMOUS,
|
p = mmap(NULL, sz, PROT_READ|PROT_WRITE|CCPROT_CREATE,
|
||||||
-1, 0);
|
MAP_PRIVATE|MAP_ANONYMOUS|CCMAP_CREATE, -1, 0);
|
||||||
if (p == MAP_FAILED)
|
if (p == MAP_FAILED)
|
||||||
lj_err_caller(cts->L, LJ_ERR_FFI_CBACKOV);
|
lj_err_caller(cts->L, LJ_ERR_FFI_CBACKOV);
|
||||||
|
#if CCMAP_CREATE
|
||||||
|
pthread_jit_write_protect_np(0);
|
||||||
|
#endif
|
||||||
#else
|
#else
|
||||||
/* Fallback allocator. Fails if memory is not executable by default. */
|
/* Fallback allocator. Fails if memory is not executable by default. */
|
||||||
p = lj_mem_new(cts->L, sz);
|
p = lj_mem_new(cts->L, sz);
|
||||||
@ -296,8 +307,12 @@ static void callback_mcode_new(CTState *cts)
|
|||||||
LJ_WIN_VPROTECT(p, sz, PAGE_EXECUTE_READ, &oprot);
|
LJ_WIN_VPROTECT(p, sz, PAGE_EXECUTE_READ, &oprot);
|
||||||
}
|
}
|
||||||
#elif LJ_TARGET_POSIX
|
#elif LJ_TARGET_POSIX
|
||||||
|
#if CCMAP_CREATE
|
||||||
|
pthread_jit_write_protect_np(1);
|
||||||
|
#else
|
||||||
mprotect(p, sz, (PROT_READ|PROT_EXEC));
|
mprotect(p, sz, (PROT_READ|PROT_EXEC));
|
||||||
#endif
|
#endif
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free area for callback function pointers. */
|
/* Free area for callback function pointers. */
|
||||||
|
@ -33,10 +33,12 @@
|
|||||||
_("int16_t", INT16) \
|
_("int16_t", INT16) \
|
||||||
_("int32_t", INT32) \
|
_("int32_t", INT32) \
|
||||||
_("int64_t", INT64) \
|
_("int64_t", INT64) \
|
||||||
|
_("int128_t", INT128) \
|
||||||
_("uint8_t", UINT8) \
|
_("uint8_t", UINT8) \
|
||||||
_("uint16_t", UINT16) \
|
_("uint16_t", UINT16) \
|
||||||
_("uint32_t", UINT32) \
|
_("uint32_t", UINT32) \
|
||||||
_("uint64_t", UINT64) \
|
_("uint64_t", UINT64) \
|
||||||
|
_("uint128_t", UINT128) \
|
||||||
_("intptr_t", INT_PSZ) \
|
_("intptr_t", INT_PSZ) \
|
||||||
_("uintptr_t", UINT_PSZ) \
|
_("uintptr_t", UINT_PSZ) \
|
||||||
/* From POSIX. */ \
|
/* From POSIX. */ \
|
||||||
@ -55,6 +57,7 @@
|
|||||||
_("__int16", 2, CTOK_INT) \
|
_("__int16", 2, CTOK_INT) \
|
||||||
_("__int32", 4, CTOK_INT) \
|
_("__int32", 4, CTOK_INT) \
|
||||||
_("__int64", 8, CTOK_INT) \
|
_("__int64", 8, CTOK_INT) \
|
||||||
|
_("__int128", 16, CTOK_INT) \
|
||||||
_("float", 4, CTOK_FP) \
|
_("float", 4, CTOK_FP) \
|
||||||
_("double", 8, CTOK_FP) \
|
_("double", 8, CTOK_FP) \
|
||||||
_("long", 0, CTOK_LONG) \
|
_("long", 0, CTOK_LONG) \
|
||||||
|
@ -292,6 +292,8 @@ typedef struct CTState {
|
|||||||
_(UINT32, 4, CT_NUM, CTF_UNSIGNED|CTALIGN(2)) \
|
_(UINT32, 4, CT_NUM, CTF_UNSIGNED|CTALIGN(2)) \
|
||||||
_(INT64, 8, CT_NUM, CTF_LONG_IF8|CTALIGN(3)) \
|
_(INT64, 8, CT_NUM, CTF_LONG_IF8|CTALIGN(3)) \
|
||||||
_(UINT64, 8, CT_NUM, CTF_UNSIGNED|CTF_LONG_IF8|CTALIGN(3)) \
|
_(UINT64, 8, CT_NUM, CTF_UNSIGNED|CTF_LONG_IF8|CTALIGN(3)) \
|
||||||
|
_(INT128, 16, CT_NUM, CTALIGN(4)) \
|
||||||
|
_(UINT128, 16, CT_NUM, CTF_UNSIGNED|CTALIGN(4)) \
|
||||||
_(FLOAT, 4, CT_NUM, CTF_FP|CTALIGN(2)) \
|
_(FLOAT, 4, CT_NUM, CTF_FP|CTALIGN(2)) \
|
||||||
_(DOUBLE, 8, CT_NUM, CTF_FP|CTALIGN(3)) \
|
_(DOUBLE, 8, CT_NUM, CTF_FP|CTALIGN(3)) \
|
||||||
_(COMPLEX_FLOAT, 8, CT_ARRAY, CTF_COMPLEX|CTALIGN(2)|CTID_FLOAT) \
|
_(COMPLEX_FLOAT, 8, CT_ARRAY, CTF_COMPLEX|CTALIGN(2)|CTID_FLOAT) \
|
||||||
|
@ -98,6 +98,14 @@ static int mcode_setprot(void *p, size_t sz, DWORD prot)
|
|||||||
#define MAP_ANONYMOUS MAP_ANON
|
#define MAP_ANONYMOUS MAP_ANON
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Check for macOS hardened runtime. */
|
||||||
|
#if LUAJIT_SECURITY_MCODE != 0 && defined(MAP_JIT) && __ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__ >= 110000
|
||||||
|
#include <pthread.h>
|
||||||
|
#define MCMAP_CREATE MAP_JIT
|
||||||
|
#else
|
||||||
|
#define MCMAP_CREATE 0
|
||||||
|
#endif
|
||||||
|
|
||||||
#define MCPROT_RW (PROT_READ|PROT_WRITE)
|
#define MCPROT_RW (PROT_READ|PROT_WRITE)
|
||||||
#define MCPROT_RX (PROT_READ|PROT_EXEC)
|
#define MCPROT_RX (PROT_READ|PROT_EXEC)
|
||||||
#define MCPROT_RWX (PROT_READ|PROT_WRITE|PROT_EXEC)
|
#define MCPROT_RWX (PROT_READ|PROT_WRITE|PROT_EXEC)
|
||||||
@ -109,10 +117,14 @@ static int mcode_setprot(void *p, size_t sz, DWORD prot)
|
|||||||
|
|
||||||
static void *mcode_alloc_at(jit_State *J, uintptr_t hint, size_t sz, int prot)
|
static void *mcode_alloc_at(jit_State *J, uintptr_t hint, size_t sz, int prot)
|
||||||
{
|
{
|
||||||
void *p = mmap((void *)hint, sz, prot|MCPROT_CREATE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0);
|
void *p = mmap((void *)hint, sz, prot|MCPROT_CREATE, MAP_PRIVATE|MAP_ANONYMOUS|MCMAP_CREATE, -1, 0);
|
||||||
if (p == MAP_FAILED) {
|
if (p == MAP_FAILED) {
|
||||||
if (!hint) lj_trace_err(J, LJ_TRERR_MCODEAL);
|
if (!hint) lj_trace_err(J, LJ_TRERR_MCODEAL);
|
||||||
p = NULL;
|
p = NULL;
|
||||||
|
#if MCMAP_CREATE
|
||||||
|
} else {
|
||||||
|
pthread_jit_write_protect_np(0);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
@ -125,7 +137,12 @@ static void mcode_free(jit_State *J, void *p, size_t sz)
|
|||||||
|
|
||||||
static int mcode_setprot(void *p, size_t sz, int prot)
|
static int mcode_setprot(void *p, size_t sz, int prot)
|
||||||
{
|
{
|
||||||
|
#if MCMAP_CREATE
|
||||||
|
pthread_jit_write_protect_np((prot & PROT_EXEC));
|
||||||
|
return 0;
|
||||||
|
#else
|
||||||
return mprotect(p, sz, prot);
|
return mprotect(p, sz, prot);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
@ -2217,9 +2217,11 @@ LJFOLD(HREF TDUP KNUM)
|
|||||||
LJFOLDF(fwd_href_tdup)
|
LJFOLDF(fwd_href_tdup)
|
||||||
{
|
{
|
||||||
TValue keyv;
|
TValue keyv;
|
||||||
|
cTValue *val;
|
||||||
lj_ir_kvalue(J->L, &keyv, fright);
|
lj_ir_kvalue(J->L, &keyv, fright);
|
||||||
if (lj_tab_get(J->L, ir_ktab(IR(fleft->op1)), &keyv) == niltvg(J2G(J)) &&
|
val = lj_tab_get(J->L, ir_ktab(IR(fleft->op1)), &keyv);
|
||||||
lj_opt_fwd_href_nokey(J))
|
/* Check for either nil or the nil value marker in the template table. */
|
||||||
|
if ((tvisnil(val) || tvistab(val)) && lj_opt_fwd_href_nokey(J))
|
||||||
return lj_ir_kkptr(J, niltvg(J2G(J)));
|
return lj_ir_kkptr(J, niltvg(J2G(J)));
|
||||||
return NEXTFOLD;
|
return NEXTFOLD;
|
||||||
}
|
}
|
||||||
|
@ -233,7 +233,9 @@ static TRef fwd_ahload(jit_State *J, IRRef xref)
|
|||||||
return lj_ir_knum_u64(J, tv->u64);
|
return lj_ir_knum_u64(J, tv->u64);
|
||||||
else if (tvisint(tv))
|
else if (tvisint(tv))
|
||||||
return lj_ir_kint(J, intV(tv));
|
return lj_ir_kint(J, intV(tv));
|
||||||
else if (tvisgcv(tv))
|
else if (tvistab(tv)) /* Template table nil value marker. */
|
||||||
|
return TREF_NIL;
|
||||||
|
else if (tvisstr(tv))
|
||||||
return lj_ir_kstr(J, strV(tv));
|
return lj_ir_kstr(J, strV(tv));
|
||||||
}
|
}
|
||||||
/* Othwerwise: don't intern as a constant. */
|
/* Othwerwise: don't intern as a constant. */
|
||||||
|
@ -1725,7 +1725,7 @@ static void expr_table(LexState *ls, ExpDesc *e)
|
|||||||
FuncState *fs = ls->fs;
|
FuncState *fs = ls->fs;
|
||||||
BCLine line = ls->linenumber;
|
BCLine line = ls->linenumber;
|
||||||
GCtab *t = NULL;
|
GCtab *t = NULL;
|
||||||
int vcall = 0, needarr = 0, fixt = 0;
|
int vcall = 0, needarr = 0;
|
||||||
uint32_t narr = 1; /* First array index. */
|
uint32_t narr = 1; /* First array index. */
|
||||||
uint32_t nhash = 0; /* Number of hash entries. */
|
uint32_t nhash = 0; /* Number of hash entries. */
|
||||||
BCReg freg = fs->freereg;
|
BCReg freg = fs->freereg;
|
||||||
@ -1769,9 +1769,10 @@ static void expr_table(LexState *ls, ExpDesc *e)
|
|||||||
lj_gc_anybarriert(fs->L, t);
|
lj_gc_anybarriert(fs->L, t);
|
||||||
if (expr_isk_nojump(&val)) { /* Add const key/value to template table. */
|
if (expr_isk_nojump(&val)) { /* Add const key/value to template table. */
|
||||||
expr_kvalue(fs, v, &val);
|
expr_kvalue(fs, v, &val);
|
||||||
} else { /* Otherwise create dummy string key (avoids lj_tab_newkey). */
|
/* Mark nil value with table value itself to preserve the key. */
|
||||||
settabV(fs->L, v, t); /* Preserve key with table itself as value. */
|
if (key.k == VKSTR && tvisnil(v)) settabV(fs->L, v, t);
|
||||||
fixt = 1; /* Fix this later, after all resizes. */
|
} else { /* Preserve the key for the following non-const store. */
|
||||||
|
settabV(fs->L, v, t);
|
||||||
goto nonconst;
|
goto nonconst;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -1813,17 +1814,6 @@ static void expr_table(LexState *ls, ExpDesc *e)
|
|||||||
} else {
|
} else {
|
||||||
if (needarr && t->asize < narr)
|
if (needarr && t->asize < narr)
|
||||||
lj_tab_reasize(fs->L, t, narr-1);
|
lj_tab_reasize(fs->L, t, narr-1);
|
||||||
if (fixt) { /* Fix value for dummy keys in template table. */
|
|
||||||
Node *node = noderef(t->node);
|
|
||||||
uint32_t i, hmask = t->hmask;
|
|
||||||
for (i = 0; i <= hmask; i++) {
|
|
||||||
Node *n = &node[i];
|
|
||||||
if (tvistab(&n->val)) {
|
|
||||||
lj_assertFS(tabV(&n->val) == t, "bad dummy key in template table");
|
|
||||||
setnilV(&n->val); /* Turn value into nil. */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
lj_gc_check(fs->L);
|
lj_gc_check(fs->L);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2079,6 +2079,7 @@ static TRef rec_tnew(jit_State *J, uint32_t ah)
|
|||||||
/* -- Concatenation ------------------------------------------------------- */
|
/* -- Concatenation ------------------------------------------------------- */
|
||||||
|
|
||||||
typedef struct RecCatDataCP {
|
typedef struct RecCatDataCP {
|
||||||
|
TValue savetv[5+LJ_FR2];
|
||||||
jit_State *J;
|
jit_State *J;
|
||||||
BCReg baseslot, topslot;
|
BCReg baseslot, topslot;
|
||||||
TRef tr;
|
TRef tr;
|
||||||
@ -2119,7 +2120,9 @@ static TValue *rec_mm_concat_cp(lua_State *L, lua_CFunction dummy, void *ud)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
/* Pass partial result. */
|
/* Pass partial result. */
|
||||||
topslot = J->maxslot--;
|
rcd->topslot = topslot = J->maxslot--;
|
||||||
|
/* Save updated range of slots. */
|
||||||
|
memcpy(rcd->savetv, &L->base[topslot-1], sizeof(rcd->savetv));
|
||||||
*xbase = tr;
|
*xbase = tr;
|
||||||
top = xbase;
|
top = xbase;
|
||||||
setstrV(J->L, &ix.keyv, &J2G(J)->strempty); /* Simulate string result. */
|
setstrV(J->L, &ix.keyv, &J2G(J)->strempty); /* Simulate string result. */
|
||||||
@ -2139,16 +2142,18 @@ static TRef rec_cat(jit_State *J, BCReg baseslot, BCReg topslot)
|
|||||||
{
|
{
|
||||||
lua_State *L = J->L;
|
lua_State *L = J->L;
|
||||||
ptrdiff_t delta = L->top - L->base;
|
ptrdiff_t delta = L->top - L->base;
|
||||||
TValue savetv[5+LJ_FR2], errobj;
|
TValue errobj;
|
||||||
RecCatDataCP rcd;
|
RecCatDataCP rcd;
|
||||||
int errcode;
|
int errcode;
|
||||||
rcd.J = J;
|
rcd.J = J;
|
||||||
rcd.baseslot = baseslot;
|
rcd.baseslot = baseslot;
|
||||||
rcd.topslot = topslot;
|
rcd.topslot = topslot;
|
||||||
memcpy(savetv, &L->base[topslot-1], sizeof(savetv)); /* Save slots. */
|
/* Save slots. */
|
||||||
|
memcpy(rcd.savetv, &L->base[topslot-1], sizeof(rcd.savetv));
|
||||||
errcode = lj_vm_cpcall(L, NULL, &rcd, rec_mm_concat_cp);
|
errcode = lj_vm_cpcall(L, NULL, &rcd, rec_mm_concat_cp);
|
||||||
if (errcode) copyTV(L, &errobj, L->top-1);
|
if (errcode) copyTV(L, &errobj, L->top-1);
|
||||||
memcpy(&L->base[topslot-1], savetv, sizeof(savetv)); /* Restore slots. */
|
/* Restore slots. */
|
||||||
|
memcpy(&L->base[rcd.topslot-1], rcd.savetv, sizeof(rcd.savetv));
|
||||||
if (errcode) {
|
if (errcode) {
|
||||||
L->top = L->base + delta;
|
L->top = L->base + delta;
|
||||||
copyTV(L, L->top++, &errobj);
|
copyTV(L, L->top++, &errobj);
|
||||||
|
@ -194,6 +194,7 @@ GCtab * LJ_FASTCALL lj_tab_dup(lua_State *L, const GCtab *kt)
|
|||||||
Node *next = nextnode(kn);
|
Node *next = nextnode(kn);
|
||||||
/* Don't use copyTV here, since it asserts on a copy of a dead key. */
|
/* Don't use copyTV here, since it asserts on a copy of a dead key. */
|
||||||
n->val = kn->val; n->key = kn->key;
|
n->val = kn->val; n->key = kn->key;
|
||||||
|
if (tvistab(&n->val)) setnilV(&n->val); /* Replace nil value marker. */
|
||||||
setmref(n->next, next == NULL? next : (Node *)((char *)next + d));
|
setmref(n->next, next == NULL? next : (Node *)((char *)next + d));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -222,14 +222,6 @@ static void trace_unpatch(jit_State *J, GCtrace *T)
|
|||||||
bc_isret(op), "bad original bytecode %d", op);
|
bc_isret(op), "bad original bytecode %d", op);
|
||||||
*pc = T->startins;
|
*pc = T->startins;
|
||||||
break;
|
break;
|
||||||
case BC_JMP:
|
|
||||||
lj_assertJ(op == BC_ITERL, "bad original bytecode %d", op);
|
|
||||||
pc += bc_j(*pc)+2;
|
|
||||||
if (bc_op(*pc) == BC_JITERL) {
|
|
||||||
lj_assertJ(traceref(J, bc_d(*pc)) == T, "JITERL references other trace");
|
|
||||||
*pc = T->startins;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case BC_JFUNCF:
|
case BC_JFUNCF:
|
||||||
lj_assertJ(op == BC_FUNCF, "bad original bytecode %d", op);
|
lj_assertJ(op == BC_FUNCF, "bad original bytecode %d", op);
|
||||||
*pc = T->startins;
|
*pc = T->startins;
|
||||||
@ -245,18 +237,19 @@ static void trace_flushroot(jit_State *J, GCtrace *T)
|
|||||||
GCproto *pt = &gcref(T->startpt)->pt;
|
GCproto *pt = &gcref(T->startpt)->pt;
|
||||||
lj_assertJ(T->root == 0, "not a root trace");
|
lj_assertJ(T->root == 0, "not a root trace");
|
||||||
lj_assertJ(pt != NULL, "trace has no prototype");
|
lj_assertJ(pt != NULL, "trace has no prototype");
|
||||||
/* First unpatch any modified bytecode. */
|
|
||||||
trace_unpatch(J, T);
|
|
||||||
/* Unlink root trace from chain anchored in prototype. */
|
/* Unlink root trace from chain anchored in prototype. */
|
||||||
if (pt->trace == T->traceno) { /* Trace is first in chain. Easy. */
|
if (pt->trace == T->traceno) { /* Trace is first in chain. Easy. */
|
||||||
pt->trace = T->nextroot;
|
pt->trace = T->nextroot;
|
||||||
|
unpatch:
|
||||||
|
/* Unpatch modified bytecode only if the trace has not been flushed. */
|
||||||
|
trace_unpatch(J, T);
|
||||||
} else if (pt->trace) { /* Otherwise search in chain of root traces. */
|
} else if (pt->trace) { /* Otherwise search in chain of root traces. */
|
||||||
GCtrace *T2 = traceref(J, pt->trace);
|
GCtrace *T2 = traceref(J, pt->trace);
|
||||||
if (T2) {
|
if (T2) {
|
||||||
for (; T2->nextroot; T2 = traceref(J, T2->nextroot))
|
for (; T2->nextroot; T2 = traceref(J, T2->nextroot))
|
||||||
if (T2->nextroot == T->traceno) {
|
if (T2->nextroot == T->traceno) {
|
||||||
T2->nextroot = T->nextroot; /* Unlink from chain. */
|
T2->nextroot = T->nextroot; /* Unlink from chain. */
|
||||||
break;
|
goto unpatch;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
23
src/luajit.c
23
src/luajit.c
@ -35,6 +35,21 @@
|
|||||||
|
|
||||||
#if !LJ_TARGET_CONSOLE
|
#if !LJ_TARGET_CONSOLE
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
|
||||||
|
#if LJ_TARGET_POSIX
|
||||||
|
/* Improve signal handling on POSIX. Try CTRL-C on: luajit -e 'io.read()' */
|
||||||
|
static void signal_set(int sig, void (*h)(int))
|
||||||
|
{
|
||||||
|
struct sigaction sa;
|
||||||
|
memset(&sa, 0, sizeof(sa));
|
||||||
|
sa.sa_handler = h;
|
||||||
|
sigemptyset(&sa.sa_mask);
|
||||||
|
sigaction(sig, &sa, NULL);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
#define signal_set signal
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static lua_State *globalL = NULL;
|
static lua_State *globalL = NULL;
|
||||||
@ -54,8 +69,8 @@ static void lstop(lua_State *L, lua_Debug *ar)
|
|||||||
|
|
||||||
static void laction(int i)
|
static void laction(int i)
|
||||||
{
|
{
|
||||||
signal(i, SIG_DFL); /* if another SIGINT happens before lstop,
|
/* Terminate process if another SIGINT happens (double CTRL-C). */
|
||||||
terminate process (default action) */
|
signal_set(i, SIG_DFL);
|
||||||
lua_sethook(globalL, lstop, LUA_MASKCALL | LUA_MASKRET | LUA_MASKCOUNT, 1);
|
lua_sethook(globalL, lstop, LUA_MASKCALL | LUA_MASKRET | LUA_MASKCOUNT, 1);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -117,11 +132,11 @@ static int docall(lua_State *L, int narg, int clear)
|
|||||||
lua_pushcfunction(L, traceback); /* push traceback function */
|
lua_pushcfunction(L, traceback); /* push traceback function */
|
||||||
lua_insert(L, base); /* put it under chunk and args */
|
lua_insert(L, base); /* put it under chunk and args */
|
||||||
#if !LJ_TARGET_CONSOLE
|
#if !LJ_TARGET_CONSOLE
|
||||||
signal(SIGINT, laction);
|
signal_set(SIGINT, laction);
|
||||||
#endif
|
#endif
|
||||||
status = lua_pcall(L, narg, (clear ? 0 : LUA_MULTRET), base);
|
status = lua_pcall(L, narg, (clear ? 0 : LUA_MULTRET), base);
|
||||||
#if !LJ_TARGET_CONSOLE
|
#if !LJ_TARGET_CONSOLE
|
||||||
signal(SIGINT, SIG_DFL);
|
signal_set(SIGINT, SIG_DFL);
|
||||||
#endif
|
#endif
|
||||||
lua_remove(L, base); /* remove traceback function */
|
lua_remove(L, base); /* remove traceback function */
|
||||||
/* force a complete garbage collection in case of errors */
|
/* force a complete garbage collection in case of errors */
|
||||||
|
@ -8,7 +8,8 @@
|
|||||||
@rem nogc64 disable LJ_GC64 mode for x64
|
@rem nogc64 disable LJ_GC64 mode for x64
|
||||||
@rem debug emit debug symbols
|
@rem debug emit debug symbols
|
||||||
@rem amalg amalgamated build
|
@rem amalg amalgamated build
|
||||||
@rem static static linkage
|
@rem static create static lib to statically link into your project
|
||||||
|
@rem mixed create static lib to build a DLL in your project
|
||||||
|
|
||||||
@if not defined INCLUDE goto :FAIL
|
@if not defined INCLUDE goto :FAIL
|
||||||
|
|
||||||
@ -106,12 +107,14 @@ buildvm -m folddef -o lj_folddef.h lj_opt_fold.c
|
|||||||
@if "%1"=="static" goto :STATIC
|
@if "%1"=="static" goto :STATIC
|
||||||
%LJCOMPILE% %LJDYNBUILD% lj_*.c lib_*.c
|
%LJCOMPILE% %LJDYNBUILD% lj_*.c lib_*.c
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
|
@if "%1"=="mixed" goto :STATICLIB
|
||||||
%LJLINK% /DLL /OUT:%LJDLLNAME% lj_*.obj lib_*.obj
|
%LJLINK% /DLL /OUT:%LJDLLNAME% lj_*.obj lib_*.obj
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
@goto :MTDLL
|
@goto :MTDLL
|
||||||
:STATIC
|
:STATIC
|
||||||
%LJCOMPILE% lj_*.c lib_*.c
|
%LJCOMPILE% lj_*.c lib_*.c
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
|
:STATICLIB
|
||||||
%LJLIB% /OUT:%LJLIBNAME% lj_*.obj lib_*.obj
|
%LJLIB% /OUT:%LJLIBNAME% lj_*.obj lib_*.obj
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
@goto :MTDLL
|
@goto :MTDLL
|
||||||
@ -119,13 +122,15 @@ buildvm -m folddef -o lj_folddef.h lj_opt_fold.c
|
|||||||
@if "%2"=="static" goto :AMALGSTATIC
|
@if "%2"=="static" goto :AMALGSTATIC
|
||||||
%LJCOMPILE% %LJDYNBUILD% ljamalg.c
|
%LJCOMPILE% %LJDYNBUILD% ljamalg.c
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
|
@if "%2"=="mixed" goto :AMALGSTATICLIB
|
||||||
%LJLINK% /DLL /OUT:%LJDLLNAME% ljamalg.obj lj_vm.obj
|
%LJLINK% /DLL /OUT:%LJDLLNAME% ljamalg.obj lj_vm.obj
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
@goto :MTDLL
|
@goto :MTDLL
|
||||||
:AMALGSTATIC
|
:AMALGSTATIC
|
||||||
%LJCOMPILE% ljamalg.c
|
%LJCOMPILE% ljamalg.c
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
%LJLINK% /OUT:%LJDLLNAME% ljamalg.obj lj_vm.obj
|
:AMALGSTATICLIB
|
||||||
|
%LJLIB% /OUT:%LJLIBNAME% ljamalg.obj lj_vm.obj
|
||||||
@if errorlevel 1 goto :BAD
|
@if errorlevel 1 goto :BAD
|
||||||
:MTDLL
|
:MTDLL
|
||||||
if exist %LJDLLNAME%.manifest^
|
if exist %LJDLLNAME%.manifest^
|
||||||
|
Loading…
Reference in New Issue
Block a user