aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--lapi.c4
-rw-r--r--ldebug.c28
-rw-r--r--ldebug.h1
-rw-r--r--lgc.c16
-rw-r--r--lobject.c2
-rw-r--r--lobject.h18
-rw-r--r--lparser.c12
-rw-r--r--lstate.c2
-rw-r--r--lstate.h2
-rw-r--r--lstring.c11
-rw-r--r--lundump.c2
-rw-r--r--lundump.h3
-rw-r--r--lvm.c30
-rw-r--r--manual/manual.of4
-rw-r--r--testes/calls.lua2
-rw-r--r--testes/db.lua6
-rw-r--r--testes/errors.lua8
-rw-r--r--testes/files.lua8
-rw-r--r--testes/pm.lua56
-rw-r--r--testes/sort.lua2
-rw-r--r--testes/strings.lua3
-rw-r--r--testes/utf8.lua2
22 files changed, 134 insertions, 88 deletions
diff --git a/lapi.c b/lapi.c
index a1eb7dc6..767fa55e 100644
--- a/lapi.c
+++ b/lapi.c
@@ -417,9 +417,9 @@ LUA_API const char *lua_tolstring (lua_State *L, int idx, size_t *len) {
417 o = index2value(L, idx); /* previous call may reallocate the stack */ 417 o = index2value(L, idx); /* previous call may reallocate the stack */
418 } 418 }
419 if (len != NULL) 419 if (len != NULL)
420 *len = vslen(o); 420 *len = tsslen(tsvalue(o));
421 lua_unlock(L); 421 lua_unlock(L);
422 return svalue(o); 422 return getstr(tsvalue(o));
423} 423}
424 424
425 425
diff --git a/ldebug.c b/ldebug.c
index 28b1caab..690ac38f 100644
--- a/ldebug.c
+++ b/ldebug.c
@@ -426,7 +426,7 @@ static const char *getobjname (const Proto *p, int lastpc, int reg,
426*/ 426*/
427static void kname (const Proto *p, int c, const char **name) { 427static void kname (const Proto *p, int c, const char **name) {
428 TValue *kvalue = &p->k[c]; 428 TValue *kvalue = &p->k[c];
429 *name = (ttisstring(kvalue)) ? svalue(kvalue) : "?"; 429 *name = (ttisstring(kvalue)) ? getstr(tsvalue(kvalue)) : "?";
430} 430}
431 431
432 432
@@ -569,7 +569,7 @@ static const char *getobjname (const Proto *p, int lastpc, int reg,
569 int b = (op == OP_LOADK) ? GETARG_Bx(i) 569 int b = (op == OP_LOADK) ? GETARG_Bx(i)
570 : GETARG_Ax(p->code[pc + 1]); 570 : GETARG_Ax(p->code[pc + 1]);
571 if (ttisstring(&p->k[b])) { 571 if (ttisstring(&p->k[b])) {
572 *name = svalue(&p->k[b]); 572 *name = getstr(tsvalue(&p->k[b]));
573 return "constant"; 573 return "constant";
574 } 574 }
575 break; 575 break;
@@ -627,7 +627,7 @@ static const char *funcnamefromcode (lua_State *L, const Proto *p,
627 default: 627 default:
628 return NULL; /* cannot find a reasonable name */ 628 return NULL; /* cannot find a reasonable name */
629 } 629 }
630 *name = getstr(G(L)->tmname[tm]) + 2; 630 *name = getshrstr(G(L)->tmname[tm]) + 2;
631 return "metamethod"; 631 return "metamethod";
632} 632}
633 633
@@ -866,6 +866,28 @@ static int changedline (const Proto *p, int oldpc, int newpc) {
866 866
867 867
868/* 868/*
869** Traces Lua calls. If code is running the first instruction of a function,
870** and function is not vararg, and it is not coming from an yield,
871** calls 'luaD_hookcall'. (Vararg functions will call 'luaD_hookcall'
872** after adjusting its variable arguments; otherwise, they could call
873** a line/count hook before the call hook. Functions coming from
874** an yield already called 'luaD_hookcall' before yielding.)
875*/
876int luaG_tracecall (lua_State *L) {
877 CallInfo *ci = L->ci;
878 Proto *p = ci_func(ci)->p;
879 ci->u.l.trap = 1; /* ensure hooks will be checked */
880 if (ci->u.l.savedpc == p->code) { /* first instruction (not resuming)? */
881 if (p->is_vararg)
882 return 0; /* hooks will start at VARARGPREP instruction */
883 else if (!(ci->callstatus & CIST_HOOKYIELD)) /* not yieded? */
884 luaD_hookcall(L, ci); /* check 'call' hook */
885 }
886 return 1; /* keep 'trap' on */
887}
888
889
890/*
869** Traces the execution of a Lua function. Called before the execution 891** Traces the execution of a Lua function. Called before the execution
870** of each opcode, when debug is on. 'L->oldpc' stores the last 892** of each opcode, when debug is on. 'L->oldpc' stores the last
871** instruction traced, to detect line changes. When entering a new 893** instruction traced, to detect line changes. When entering a new
diff --git a/ldebug.h b/ldebug.h
index 2c3074c6..2bfce3cb 100644
--- a/ldebug.h
+++ b/ldebug.h
@@ -58,6 +58,7 @@ LUAI_FUNC const char *luaG_addinfo (lua_State *L, const char *msg,
58 TString *src, int line); 58 TString *src, int line);
59LUAI_FUNC l_noret luaG_errormsg (lua_State *L); 59LUAI_FUNC l_noret luaG_errormsg (lua_State *L);
60LUAI_FUNC int luaG_traceexec (lua_State *L, const Instruction *pc); 60LUAI_FUNC int luaG_traceexec (lua_State *L, const Instruction *pc);
61LUAI_FUNC int luaG_tracecall (lua_State *L);
61 62
62 63
63#endif 64#endif
diff --git a/lgc.c b/lgc.c
index f68c5af0..81682454 100644
--- a/lgc.c
+++ b/lgc.c
@@ -533,10 +533,12 @@ static void traversestrongtable (global_State *g, Table *h) {
533static void traversetable (global_State *g, Table *h) { 533static void traversetable (global_State *g, Table *h) {
534 const char *weakkey, *weakvalue; 534 const char *weakkey, *weakvalue;
535 const TValue *mode = gfasttm(g, h->metatable, TM_MODE); 535 const TValue *mode = gfasttm(g, h->metatable, TM_MODE);
536 TString *smode;
536 markobjectN(g, h->metatable); 537 markobjectN(g, h->metatable);
537 if (mode && ttisstring(mode) && /* is there a weak mode? */ 538 if (mode && ttisshrstring(mode) && /* is there a weak mode? */
538 (cast_void(weakkey = strchr(svalue(mode), 'k')), 539 (cast_void(smode = tsvalue(mode)),
539 cast_void(weakvalue = strchr(svalue(mode), 'v')), 540 cast_void(weakkey = strchr(getshrstr(smode), 'k')),
541 cast_void(weakvalue = strchr(getshrstr(smode), 'v')),
540 (weakkey || weakvalue))) { /* is really weak? */ 542 (weakkey || weakvalue))) { /* is really weak? */
541 if (!weakkey) /* strong keys? */ 543 if (!weakkey) /* strong keys? */
542 traverseweakvalue(g, h); 544 traverseweakvalue(g, h);
@@ -624,7 +626,9 @@ static void traversethread (global_State *g, lua_State *th) {
624 for (uv = th->openupval; uv != NULL; uv = uv->u.open.next) 626 for (uv = th->openupval; uv != NULL; uv = uv->u.open.next)
625 markobject(g, uv); /* open upvalues cannot be collected */ 627 markobject(g, uv); /* open upvalues cannot be collected */
626 if (g->gcstate == GCSatomic) { /* final traversal? */ 628 if (g->gcstate == GCSatomic) { /* final traversal? */
627 for (; o < th->stack_last.p + EXTRA_STACK; o++) 629 if (!g->gcemergency)
630 luaD_shrinkstack(th); /* do not change stack in emergency cycle */
631 for (o = th->top.p; o < th->stack_last.p + EXTRA_STACK; o++)
628 setnilvalue(s2v(o)); /* clear dead stack slice */ 632 setnilvalue(s2v(o)); /* clear dead stack slice */
629 /* 'remarkupvals' may have removed thread from 'twups' list */ 633 /* 'remarkupvals' may have removed thread from 'twups' list */
630 if (!isintwups(th) && th->openupval != NULL) { 634 if (!isintwups(th) && th->openupval != NULL) {
@@ -632,8 +636,6 @@ static void traversethread (global_State *g, lua_State *th) {
632 g->twups = th; 636 g->twups = th;
633 } 637 }
634 } 638 }
635 else if (!g->gcemergency)
636 luaD_shrinkstack(th); /* do not change stack in emergency cycle */
637} 639}
638 640
639 641
@@ -1644,6 +1646,8 @@ static void fullinc (lua_State *L, global_State *g) {
1644 entersweep(L); /* sweep everything to turn them back to white */ 1646 entersweep(L); /* sweep everything to turn them back to white */
1645 /* finish any pending sweep phase to start a new cycle */ 1647 /* finish any pending sweep phase to start a new cycle */
1646 luaC_runtilstate(L, bitmask(GCSpause)); 1648 luaC_runtilstate(L, bitmask(GCSpause));
1649 luaC_runtilstate(L, bitmask(GCSpropagate)); /* start new cycle */
1650 g->gcstate = GCSenteratomic; /* go straight to atomic phase ??? */
1647 luaC_runtilstate(L, bitmask(GCScallfin)); /* run up to finalizers */ 1651 luaC_runtilstate(L, bitmask(GCScallfin)); /* run up to finalizers */
1648 /* 'marked' must be correct after a full GC cycle */ 1652 /* 'marked' must be correct after a full GC cycle */
1649 lua_assert(g->marked == gettotalobjs(g)); 1653 lua_assert(g->marked == gettotalobjs(g));
diff --git a/lobject.c b/lobject.c
index f73ffc6d..9cfa5227 100644
--- a/lobject.c
+++ b/lobject.c
@@ -542,7 +542,7 @@ const char *luaO_pushvfstring (lua_State *L, const char *fmt, va_list argp) {
542 addstr2buff(&buff, fmt, strlen(fmt)); /* rest of 'fmt' */ 542 addstr2buff(&buff, fmt, strlen(fmt)); /* rest of 'fmt' */
543 clearbuff(&buff); /* empty buffer into the stack */ 543 clearbuff(&buff); /* empty buffer into the stack */
544 lua_assert(buff.pushed == 1); 544 lua_assert(buff.pushed == 1);
545 return svalue(s2v(L->top.p - 1)); 545 return getstr(tsvalue(s2v(L->top.p - 1)));
546} 546}
547 547
548 548
diff --git a/lobject.h b/lobject.h
index 1c300e86..79dc6b1c 100644
--- a/lobject.h
+++ b/lobject.h
@@ -386,7 +386,7 @@ typedef struct GCObject {
386typedef struct TString { 386typedef struct TString {
387 CommonHeader; 387 CommonHeader;
388 lu_byte extra; /* reserved words for short strings; "has hash" for longs */ 388 lu_byte extra; /* reserved words for short strings; "has hash" for longs */
389 lu_byte shrlen; /* length for short strings */ 389 lu_byte shrlen; /* length for short strings, 0xFF for long strings */
390 unsigned int hash; 390 unsigned int hash;
391 union { 391 union {
392 size_t lnglen; /* length for long strings */ 392 size_t lnglen; /* length for long strings */
@@ -398,19 +398,17 @@ typedef struct TString {
398 398
399 399
400/* 400/*
401** Get the actual string (array of bytes) from a 'TString'. 401** Get the actual string (array of bytes) from a 'TString'. (Generic
402** version and specialized versions for long and short strings.)
402*/ 403*/
403#define getstr(ts) ((ts)->contents) 404#define getstr(ts) ((ts)->contents)
405#define getlngstr(ts) check_exp((ts)->shrlen == 0xFF, (ts)->contents)
406#define getshrstr(ts) check_exp((ts)->shrlen != 0xFF, (ts)->contents)
404 407
405 408
406/* get the actual string (array of bytes) from a Lua value */
407#define svalue(o) getstr(tsvalue(o))
408
409/* get string length from 'TString *s' */ 409/* get string length from 'TString *s' */
410#define tsslen(s) ((s)->tt == LUA_VSHRSTR ? (s)->shrlen : (s)->u.lnglen) 410#define tsslen(s) \
411 411 ((s)->shrlen != 0xFF ? (s)->shrlen : (s)->u.lnglen)
412/* get string length from 'TValue *o' */
413#define vslen(o) tsslen(tsvalue(o))
414 412
415/* }================================================================== */ 413/* }================================================================== */
416 414
diff --git a/lparser.c b/lparser.c
index 2ac04160..81c6df22 100644
--- a/lparser.c
+++ b/lparser.c
@@ -1030,10 +1030,11 @@ static int explist (LexState *ls, expdesc *v) {
1030} 1030}
1031 1031
1032 1032
1033static void funcargs (LexState *ls, expdesc *f, int line) { 1033static void funcargs (LexState *ls, expdesc *f) {
1034 FuncState *fs = ls->fs; 1034 FuncState *fs = ls->fs;
1035 expdesc args; 1035 expdesc args;
1036 int base, nparams; 1036 int base, nparams;
1037 int line = ls->linenumber;
1037 switch (ls->t.token) { 1038 switch (ls->t.token) {
1038 case '(': { /* funcargs -> '(' [ explist ] ')' */ 1039 case '(': { /* funcargs -> '(' [ explist ] ')' */
1039 luaX_next(ls); 1040 luaX_next(ls);
@@ -1071,8 +1072,8 @@ static void funcargs (LexState *ls, expdesc *f, int line) {
1071 } 1072 }
1072 init_exp(f, VCALL, luaK_codeABC(fs, OP_CALL, base, nparams+1, 2)); 1073 init_exp(f, VCALL, luaK_codeABC(fs, OP_CALL, base, nparams+1, 2));
1073 luaK_fixline(fs, line); 1074 luaK_fixline(fs, line);
1074 fs->freereg = base+1; /* call remove function and arguments and leaves 1075 fs->freereg = base+1; /* call removes function and arguments and leaves
1075 (unless changed) one result */ 1076 one result (unless changed later) */
1076} 1077}
1077 1078
1078 1079
@@ -1111,7 +1112,6 @@ static void suffixedexp (LexState *ls, expdesc *v) {
1111 /* suffixedexp -> 1112 /* suffixedexp ->
1112 primaryexp { '.' NAME | '[' exp ']' | ':' NAME funcargs | funcargs } */ 1113 primaryexp { '.' NAME | '[' exp ']' | ':' NAME funcargs | funcargs } */
1113 FuncState *fs = ls->fs; 1114 FuncState *fs = ls->fs;
1114 int line = ls->linenumber;
1115 primaryexp(ls, v); 1115 primaryexp(ls, v);
1116 for (;;) { 1116 for (;;) {
1117 switch (ls->t.token) { 1117 switch (ls->t.token) {
@@ -1131,12 +1131,12 @@ static void suffixedexp (LexState *ls, expdesc *v) {
1131 luaX_next(ls); 1131 luaX_next(ls);
1132 codename(ls, &key); 1132 codename(ls, &key);
1133 luaK_self(fs, v, &key); 1133 luaK_self(fs, v, &key);
1134 funcargs(ls, v, line); 1134 funcargs(ls, v);
1135 break; 1135 break;
1136 } 1136 }
1137 case '(': case TK_STRING: case '{': { /* funcargs */ 1137 case '(': case TK_STRING: case '{': { /* funcargs */
1138 luaK_exp2nextreg(fs, v); 1138 luaK_exp2nextreg(fs, v);
1139 funcargs(ls, v, line); 1139 funcargs(ls, v);
1140 break; 1140 break;
1141 } 1141 }
1142 default: return; 1142 default: return;
diff --git a/lstate.c b/lstate.c
index 37301c3c..6f4c7e28 100644
--- a/lstate.c
+++ b/lstate.c
@@ -396,7 +396,7 @@ void luaE_warning (lua_State *L, const char *msg, int tocont) {
396void luaE_warnerror (lua_State *L, const char *where) { 396void luaE_warnerror (lua_State *L, const char *where) {
397 TValue *errobj = s2v(L->top.p - 1); /* error object */ 397 TValue *errobj = s2v(L->top.p - 1); /* error object */
398 const char *msg = (ttisstring(errobj)) 398 const char *msg = (ttisstring(errobj))
399 ? svalue(errobj) 399 ? getstr(tsvalue(errobj))
400 : "error object is not a string"; 400 : "error object is not a string";
401 /* produce warning "error in %s (%s)" (where, msg) */ 401 /* produce warning "error in %s (%s)" (where, msg) */
402 luaE_warning(L, "error in ", 1); 402 luaE_warning(L, "error in ", 1);
diff --git a/lstate.h b/lstate.h
index 488d2c5a..f42db35d 100644
--- a/lstate.h
+++ b/lstate.h
@@ -182,7 +182,7 @@ struct CallInfo {
182 union { 182 union {
183 struct { /* only for Lua functions */ 183 struct { /* only for Lua functions */
184 const Instruction *savedpc; 184 const Instruction *savedpc;
185 volatile l_signalT trap; 185 volatile l_signalT trap; /* function is tracing lines/counts */
186 int nextraargs; /* # of extra arguments in vararg functions */ 186 int nextraargs; /* # of extra arguments in vararg functions */
187 } l; 187 } l;
188 struct { /* only for C functions */ 188 struct { /* only for C functions */
diff --git a/lstring.c b/lstring.c
index 13dcaf42..1032ad86 100644
--- a/lstring.c
+++ b/lstring.c
@@ -36,7 +36,7 @@ int luaS_eqlngstr (TString *a, TString *b) {
36 lua_assert(a->tt == LUA_VLNGSTR && b->tt == LUA_VLNGSTR); 36 lua_assert(a->tt == LUA_VLNGSTR && b->tt == LUA_VLNGSTR);
37 return (a == b) || /* same instance or... */ 37 return (a == b) || /* same instance or... */
38 ((len == b->u.lnglen) && /* equal length and ... */ 38 ((len == b->u.lnglen) && /* equal length and ... */
39 (memcmp(getstr(a), getstr(b), len) == 0)); /* equal contents */ 39 (memcmp(getlngstr(a), getlngstr(b), len) == 0)); /* equal contents */
40} 40}
41 41
42 42
@@ -52,7 +52,7 @@ unsigned int luaS_hashlongstr (TString *ts) {
52 lua_assert(ts->tt == LUA_VLNGSTR); 52 lua_assert(ts->tt == LUA_VLNGSTR);
53 if (ts->extra == 0) { /* no hash? */ 53 if (ts->extra == 0) { /* no hash? */
54 size_t len = ts->u.lnglen; 54 size_t len = ts->u.lnglen;
55 ts->hash = luaS_hash(getstr(ts), len, ts->hash); 55 ts->hash = luaS_hash(getlngstr(ts), len, ts->hash);
56 ts->extra = 1; /* now it has its hash */ 56 ts->extra = 1; /* now it has its hash */
57 } 57 }
58 return ts->hash; 58 return ts->hash;
@@ -157,6 +157,7 @@ static TString *createstrobj (lua_State *L, size_t l, int tag, unsigned int h) {
157TString *luaS_createlngstrobj (lua_State *L, size_t l) { 157TString *luaS_createlngstrobj (lua_State *L, size_t l) {
158 TString *ts = createstrobj(L, l, LUA_VLNGSTR, G(L)->seed); 158 TString *ts = createstrobj(L, l, LUA_VLNGSTR, G(L)->seed);
159 ts->u.lnglen = l; 159 ts->u.lnglen = l;
160 ts->shrlen = 0xFF; /* signals that it is a long string */
160 return ts; 161 return ts;
161} 162}
162 163
@@ -193,7 +194,7 @@ static TString *internshrstr (lua_State *L, const char *str, size_t l) {
193 TString **list = &tb->hash[lmod(h, tb->size)]; 194 TString **list = &tb->hash[lmod(h, tb->size)];
194 lua_assert(str != NULL); /* otherwise 'memcmp'/'memcpy' are undefined */ 195 lua_assert(str != NULL); /* otherwise 'memcmp'/'memcpy' are undefined */
195 for (ts = *list; ts != NULL; ts = ts->u.hnext) { 196 for (ts = *list; ts != NULL; ts = ts->u.hnext) {
196 if (l == ts->shrlen && (memcmp(str, getstr(ts), l * sizeof(char)) == 0)) { 197 if (l == ts->shrlen && (memcmp(str, getshrstr(ts), l * sizeof(char)) == 0)) {
197 /* found! */ 198 /* found! */
198 if (isdead(g, ts)) /* dead (but not collected yet)? */ 199 if (isdead(g, ts)) /* dead (but not collected yet)? */
199 changewhite(ts); /* resurrect it */ 200 changewhite(ts); /* resurrect it */
@@ -206,7 +207,7 @@ static TString *internshrstr (lua_State *L, const char *str, size_t l) {
206 list = &tb->hash[lmod(h, tb->size)]; /* rehash with new size */ 207 list = &tb->hash[lmod(h, tb->size)]; /* rehash with new size */
207 } 208 }
208 ts = createstrobj(L, l, LUA_VSHRSTR, h); 209 ts = createstrobj(L, l, LUA_VSHRSTR, h);
209 memcpy(getstr(ts), str, l * sizeof(char)); 210 memcpy(getshrstr(ts), str, l * sizeof(char));
210 ts->shrlen = cast_byte(l); 211 ts->shrlen = cast_byte(l);
211 ts->u.hnext = *list; 212 ts->u.hnext = *list;
212 *list = ts; 213 *list = ts;
@@ -226,7 +227,7 @@ TString *luaS_newlstr (lua_State *L, const char *str, size_t l) {
226 if (l_unlikely(l >= (MAX_SIZE - sizeof(TString))/sizeof(char))) 227 if (l_unlikely(l >= (MAX_SIZE - sizeof(TString))/sizeof(char)))
227 luaM_toobig(L); 228 luaM_toobig(L);
228 ts = luaS_createlngstrobj(L, l); 229 ts = luaS_createlngstrobj(L, l);
229 memcpy(getstr(ts), str, l * sizeof(char)); 230 memcpy(getlngstr(ts), str, l * sizeof(char));
230 return ts; 231 return ts;
231 } 232 }
232} 233}
diff --git a/lundump.c b/lundump.c
index 1df15e53..0c93c995 100644
--- a/lundump.c
+++ b/lundump.c
@@ -131,7 +131,7 @@ static TString *loadStringN (LoadState *S, Proto *p) {
131 ts = luaS_createlngstrobj(L, size); /* create string */ 131 ts = luaS_createlngstrobj(L, size); /* create string */
132 setsvalue2s(L, L->top.p, ts); /* anchor it ('loadVector' can GC) */ 132 setsvalue2s(L, L->top.p, ts); /* anchor it ('loadVector' can GC) */
133 luaD_inctop(L); 133 luaD_inctop(L);
134 loadVector(S, getstr(ts), size); /* load directly in final place */ 134 loadVector(S, getlngstr(ts), size); /* load directly in final place */
135 L->top.p--; /* pop string */ 135 L->top.p--; /* pop string */
136 } 136 }
137 luaC_objbarrier(L, p, ts); 137 luaC_objbarrier(L, p, ts);
diff --git a/lundump.h b/lundump.h
index 7def905b..bfabaa63 100644
--- a/lundump.h
+++ b/lundump.h
@@ -21,8 +21,7 @@
21/* 21/*
22** Encode major-minor version in one byte, one nibble for each 22** Encode major-minor version in one byte, one nibble for each
23*/ 23*/
24#define MYINT(s) (s[0]-'0') /* assume one-digit numerals */ 24#define LUAC_VERSION (LUA_VERSION_MAJOR_N*16+LUA_VERSION_MINOR_N)
25#define LUAC_VERSION (MYINT(LUA_VERSION_MAJOR)*16+MYINT(LUA_VERSION_MINOR))
26 25
27#define LUAC_FORMAT 0 /* this is the official format */ 26#define LUAC_FORMAT 0 /* this is the official format */
28 27
diff --git a/lvm.c b/lvm.c
index a3128c3d..256d689f 100644
--- a/lvm.c
+++ b/lvm.c
@@ -91,8 +91,10 @@ static int l_strton (const TValue *obj, TValue *result) {
91 lua_assert(obj != result); 91 lua_assert(obj != result);
92 if (!cvt2num(obj)) /* is object not a string? */ 92 if (!cvt2num(obj)) /* is object not a string? */
93 return 0; 93 return 0;
94 else 94 else {
95 return (luaO_str2num(svalue(obj), result) == vslen(obj) + 1); 95 TString *st = tsvalue(obj);
96 return (luaO_str2num(getstr(st), result) == tsslen(st) + 1);
97 }
96} 98}
97 99
98 100
@@ -627,8 +629,9 @@ int luaV_equalobj (lua_State *L, const TValue *t1, const TValue *t2) {
627static void copy2buff (StkId top, int n, char *buff) { 629static void copy2buff (StkId top, int n, char *buff) {
628 size_t tl = 0; /* size already copied */ 630 size_t tl = 0; /* size already copied */
629 do { 631 do {
630 size_t l = vslen(s2v(top - n)); /* length of string being copied */ 632 TString *st = tsvalue(s2v(top - n));
631 memcpy(buff + tl, svalue(s2v(top - n)), l * sizeof(char)); 633 size_t l = tsslen(st); /* length of string being copied */
634 memcpy(buff + tl, getstr(st), l * sizeof(char));
632 tl += l; 635 tl += l;
633 } while (--n > 0); 636 } while (--n > 0);
634} 637}
@@ -654,11 +657,11 @@ void luaV_concat (lua_State *L, int total) {
654 } 657 }
655 else { 658 else {
656 /* at least two non-empty string values; get as many as possible */ 659 /* at least two non-empty string values; get as many as possible */
657 size_t tl = vslen(s2v(top - 1)); 660 size_t tl = tsslen(tsvalue(s2v(top - 1)));
658 TString *ts; 661 TString *ts;
659 /* collect total length and number of strings */ 662 /* collect total length and number of strings */
660 for (n = 1; n < total && tostring(L, s2v(top - n - 1)); n++) { 663 for (n = 1; n < total && tostring(L, s2v(top - n - 1)); n++) {
661 size_t l = vslen(s2v(top - n - 1)); 664 size_t l = tsslen(tsvalue(s2v(top - n - 1)));
662 if (l_unlikely(l >= (MAX_SIZE/sizeof(char)) - tl)) { 665 if (l_unlikely(l >= (MAX_SIZE/sizeof(char)) - tl)) {
663 L->top.p = top - total; /* pop strings to avoid wasting stack */ 666 L->top.p = top - total; /* pop strings to avoid wasting stack */
664 luaG_runerror(L, "string length overflow"); 667 luaG_runerror(L, "string length overflow");
@@ -672,7 +675,7 @@ void luaV_concat (lua_State *L, int total) {
672 } 675 }
673 else { /* long string; copy strings directly to final result */ 676 else { /* long string; copy strings directly to final result */
674 ts = luaS_createlngstrobj(L, tl); 677 ts = luaS_createlngstrobj(L, tl);
675 copy2buff(top, n, getstr(ts)); 678 copy2buff(top, n, getlngstr(ts));
676 } 679 }
677 setsvalue2s(L, top - n, ts); /* create result */ 680 setsvalue2s(L, top - n, ts); /* create result */
678 } 681 }
@@ -1158,18 +1161,11 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
1158 startfunc: 1161 startfunc:
1159 trap = L->hookmask; 1162 trap = L->hookmask;
1160 returning: /* trap already set */ 1163 returning: /* trap already set */
1161 cl = clLvalue(s2v(ci->func.p)); 1164 cl = ci_func(ci);
1162 k = cl->p->k; 1165 k = cl->p->k;
1163 pc = ci->u.l.savedpc; 1166 pc = ci->u.l.savedpc;
1164 if (l_unlikely(trap)) { 1167 if (l_unlikely(trap))
1165 if (pc == cl->p->code) { /* first instruction (not resuming)? */ 1168 trap = luaG_tracecall(L);
1166 if (cl->p->is_vararg)
1167 trap = 0; /* hooks will start after VARARGPREP instruction */
1168 else /* check 'call' hook */
1169 luaD_hookcall(L, ci);
1170 }
1171 ci->u.l.trap = 1; /* assume trap is on, for now */
1172 }
1173 base = ci->func.p + 1; 1169 base = ci->func.p + 1;
1174 /* main loop of interpreter */ 1170 /* main loop of interpreter */
1175 for (;;) { 1171 for (;;) {
diff --git a/manual/manual.of b/manual/manual.of
index 1e219f9a..c16039b4 100644
--- a/manual/manual.of
+++ b/manual/manual.of
@@ -9033,6 +9033,10 @@ Lua does not consult any environment variables.
9033In particular, 9033In particular,
9034the values of @Lid{package.path} and @Lid{package.cpath} 9034the values of @Lid{package.path} and @Lid{package.cpath}
9035are set with the default paths defined in @id{luaconf.h}. 9035are set with the default paths defined in @id{luaconf.h}.
9036To signal to the libraries that this option is on,
9037the stand-alone interpreter sets the field
9038@idx{"LUA_NOENV"} in the registry to a true value.
9039Other libraries may consult this field for the same purpose.
9036 9040
9037The options @T{-e}, @T{-l}, and @T{-W} are handled in 9041The options @T{-e}, @T{-l}, and @T{-W} are handled in
9038the order they appear. 9042the order they appear.
diff --git a/testes/calls.lua b/testes/calls.lua
index 559ca935..7468d4ab 100644
--- a/testes/calls.lua
+++ b/testes/calls.lua
@@ -342,7 +342,7 @@ do -- another bug (in 5.4.0)
342end 342end
343 343
344 344
345do -- another bug (since 5.2) 345if not _port then -- another bug (since 5.2)
346 -- corrupted binary dump: list of upvalue names is larger than number 346 -- corrupted binary dump: list of upvalue names is larger than number
347 -- of upvalues, overflowing the array of upvalues. 347 -- of upvalues, overflowing the array of upvalues.
348 local code = 348 local code =
diff --git a/testes/db.lua b/testes/db.lua
index 67b58934..d3758c41 100644
--- a/testes/db.lua
+++ b/testes/db.lua
@@ -345,7 +345,7 @@ function f(a,b)
345 local _, y = debug.getlocal(1, 2) 345 local _, y = debug.getlocal(1, 2)
346 assert(x == a and y == b) 346 assert(x == a and y == b)
347 assert(debug.setlocal(2, 3, "pera") == "AA".."AA") 347 assert(debug.setlocal(2, 3, "pera") == "AA".."AA")
348 assert(debug.setlocal(2, 4, "maçã") == "B") 348 assert(debug.setlocal(2, 4, "manga") == "B")
349 x = debug.getinfo(2) 349 x = debug.getinfo(2)
350 assert(x.func == g and x.what == "Lua" and x.name == 'g' and 350 assert(x.func == g and x.what == "Lua" and x.name == 'g' and
351 x.nups == 2 and string.find(x.source, "^@.*db%.lua$")) 351 x.nups == 2 and string.find(x.source, "^@.*db%.lua$"))
@@ -373,9 +373,9 @@ function g (...)
373 local arg = {...} 373 local arg = {...}
374 do local a,b,c; a=math.sin(40); end 374 do local a,b,c; a=math.sin(40); end
375 local feijao 375 local feijao
376 local AAAA,B = "xuxu", "mamão" 376 local AAAA,B = "xuxu", "abacate"
377 f(AAAA,B) 377 f(AAAA,B)
378 assert(AAAA == "pera" and B == "maçã") 378 assert(AAAA == "pera" and B == "manga")
379 do 379 do
380 local B = 13 380 local B = 13
381 local x,y = debug.getlocal(1,5) 381 local x,y = debug.getlocal(1,5)
diff --git a/testes/errors.lua b/testes/errors.lua
index bf6f389d..b777a329 100644
--- a/testes/errors.lua
+++ b/testes/errors.lua
@@ -392,19 +392,19 @@ lineerror("a\n=\n-\n\nprint\n;", 3)
392 392
393lineerror([[ 393lineerror([[
394a 394a
395( 395( -- <<
39623) 39623)
397]], 1) 397]], 2)
398 398
399lineerror([[ 399lineerror([[
400local a = {x = 13} 400local a = {x = 13}
401a 401a
402. 402.
403x 403x
404( 404( -- <<
40523 40523
406) 406)
407]], 2) 407]], 5)
408 408
409lineerror([[ 409lineerror([[
410local a = {x = 13} 410local a = {x = 13}
diff --git a/testes/files.lua b/testes/files.lua
index 149e9c76..2582406f 100644
--- a/testes/files.lua
+++ b/testes/files.lua
@@ -92,8 +92,8 @@ assert(io.output():seek("end") == string.len("alo joao"))
92 92
93assert(io.output():seek("set") == 0) 93assert(io.output():seek("set") == 0)
94 94
95assert(io.write('"álo"', "{a}\n", "second line\n", "third line \n")) 95assert(io.write('"alo"', "{a}\n", "second line\n", "third line \n"))
96assert(io.write('çfourth_line')) 96assert(io.write('Xfourth_line'))
97io.output(io.stdout) 97io.output(io.stdout)
98collectgarbage() -- file should be closed by GC 98collectgarbage() -- file should be closed by GC
99assert(io.input() == io.stdin and rawequal(io.output(), io.stdout)) 99assert(io.input() == io.stdin and rawequal(io.output(), io.stdout))
@@ -300,14 +300,14 @@ do -- test error returns
300end 300end
301checkerr("invalid format", io.read, "x") 301checkerr("invalid format", io.read, "x")
302assert(io.read(0) == "") -- not eof 302assert(io.read(0) == "") -- not eof
303assert(io.read(5, 'l') == '"álo"') 303assert(io.read(5, 'l') == '"alo"')
304assert(io.read(0) == "") 304assert(io.read(0) == "")
305assert(io.read() == "second line") 305assert(io.read() == "second line")
306local x = io.input():seek() 306local x = io.input():seek()
307assert(io.read() == "third line ") 307assert(io.read() == "third line ")
308assert(io.input():seek("set", x)) 308assert(io.input():seek("set", x))
309assert(io.read('L') == "third line \n") 309assert(io.read('L') == "third line \n")
310assert(io.read(1) == "ç") 310assert(io.read(1) == "X")
311assert(io.read(string.len"fourth_line") == "fourth_line") 311assert(io.read(string.len"fourth_line") == "fourth_line")
312assert(io.input():seek("cur", -string.len"fourth_line")) 312assert(io.input():seek("cur", -string.len"fourth_line"))
313assert(io.read() == "fourth_line") 313assert(io.read() == "fourth_line")
diff --git a/testes/pm.lua b/testes/pm.lua
index 795596d4..44454dff 100644
--- a/testes/pm.lua
+++ b/testes/pm.lua
@@ -1,6 +1,9 @@
1-- $Id: testes/pm.lua $ 1-- $Id: testes/pm.lua $
2-- See Copyright Notice in file all.lua 2-- See Copyright Notice in file all.lua
3 3
4-- UTF-8 file
5
6
4print('testing pattern matching') 7print('testing pattern matching')
5 8
6local function checkerror (msg, f, ...) 9local function checkerror (msg, f, ...)
@@ -50,6 +53,19 @@ assert(f('aLo_ALO', '%a*') == 'aLo')
50 53
51assert(f(" \n\r*&\n\r xuxu \n\n", "%g%g%g+") == "xuxu") 54assert(f(" \n\r*&\n\r xuxu \n\n", "%g%g%g+") == "xuxu")
52 55
56
57-- Adapt a pattern to UTF-8
58local function PU (p)
59 -- break '?' into each individual byte of a character
60 p = string.gsub(p, "(" .. utf8.charpattern .. ")%?", function (c)
61 return string.gsub(c, ".", "%0?")
62 end)
63 -- change '.' to utf-8 character patterns
64 p = string.gsub(p, "%.", utf8.charpattern)
65 return p
66end
67
68
53assert(f('aaab', 'a*') == 'aaa'); 69assert(f('aaab', 'a*') == 'aaa');
54assert(f('aaa', '^.*$') == 'aaa'); 70assert(f('aaa', '^.*$') == 'aaa');
55assert(f('aaa', 'b*') == ''); 71assert(f('aaa', 'b*') == '');
@@ -73,16 +89,16 @@ assert(f('aaa', '^.-$') == 'aaa')
73assert(f('aabaaabaaabaaaba', 'b.*b') == 'baaabaaabaaab') 89assert(f('aabaaabaaabaaaba', 'b.*b') == 'baaabaaabaaab')
74assert(f('aabaaabaaabaaaba', 'b.-b') == 'baaab') 90assert(f('aabaaabaaabaaaba', 'b.-b') == 'baaab')
75assert(f('alo xo', '.o$') == 'xo') 91assert(f('alo xo', '.o$') == 'xo')
76assert(f(' \n isto é assim', '%S%S*') == 'isto') 92assert(f(' \n isto é assim', '%S%S*') == 'isto')
77assert(f(' \n isto é assim', '%S*$') == 'assim') 93assert(f(' \n isto é assim', '%S*$') == 'assim')
78assert(f(' \n isto é assim', '[a-z]*$') == 'assim') 94assert(f(' \n isto é assim', '[a-z]*$') == 'assim')
79assert(f('um caracter ? extra', '[^%sa-z]') == '?') 95assert(f('um caracter ? extra', '[^%sa-z]') == '?')
80assert(f('', 'a?') == '') 96assert(f('', 'a?') == '')
81assert(f('á', 'á?') == 'á') 97assert(f('á', PU'á?') == 'á')
82assert(f('ábl', 'á?b?l?') == 'ábl') 98assert(f('ábl', PU'á?b?l?') == 'ábl')
83assert(f(' ábl', 'á?b?l?') == '') 99assert(f(' ábl', PU'á?b?l?') == '')
84assert(f('aa', '^aa?a?a') == 'aa') 100assert(f('aa', '^aa?a?a') == 'aa')
85assert(f(']]]áb', '[^]]') == 'á') 101assert(f(']]]áb', '[^]]+') == 'áb')
86assert(f("0alo alo", "%x*") == "0a") 102assert(f("0alo alo", "%x*") == "0a")
87assert(f("alo alo", "%C+") == "alo alo") 103assert(f("alo alo", "%C+") == "alo alo")
88print('+') 104print('+')
@@ -136,28 +152,28 @@ assert(string.match("alo xyzK", "(%w+)K") == "xyz")
136assert(string.match("254 K", "(%d*)K") == "") 152assert(string.match("254 K", "(%d*)K") == "")
137assert(string.match("alo ", "(%w*)$") == "") 153assert(string.match("alo ", "(%w*)$") == "")
138assert(not string.match("alo ", "(%w+)$")) 154assert(not string.match("alo ", "(%w+)$"))
139assert(string.find("(álo)", "%(á") == 1) 155assert(string.find("(álo)", "%(á") == 1)
140local a, b, c, d, e = string.match("âlo alo", "^(((.).).* (%w*))$") 156local a, b, c, d, e = string.match("âlo alo", PU"^(((.).). (%w*))$")
141assert(a == 'âlo alo' and b == 'âl' and c == 'â' and d == 'alo' and e == nil) 157assert(a == 'âlo alo' and b == 'âl' and c == 'â' and d == 'alo' and e == nil)
142a, b, c, d = string.match('0123456789', '(.+(.?)())') 158a, b, c, d = string.match('0123456789', '(.+(.?)())')
143assert(a == '0123456789' and b == '' and c == 11 and d == nil) 159assert(a == '0123456789' and b == '' and c == 11 and d == nil)
144print('+') 160print('+')
145 161
146assert(string.gsub('ülo ülo', 'ü', 'x') == 'xlo xlo') 162assert(string.gsub('ülo ülo', 'ü', 'x') == 'xlo xlo')
147assert(string.gsub('alo úlo ', ' +$', '') == 'alo úlo') -- trim 163assert(string.gsub('alo úlo ', ' +$', '') == 'alo úlo') -- trim
148assert(string.gsub(' alo alo ', '^%s*(.-)%s*$', '%1') == 'alo alo') -- double trim 164assert(string.gsub(' alo alo ', '^%s*(.-)%s*$', '%1') == 'alo alo') -- double trim
149assert(string.gsub('alo alo \n 123\n ', '%s+', ' ') == 'alo alo 123 ') 165assert(string.gsub('alo alo \n 123\n ', '%s+', ' ') == 'alo alo 123 ')
150local t = "abç d" 166local t = "abç d"
151a, b = string.gsub(t, '(.)', '%1@') 167a, b = string.gsub(t, PU'(.)', '%1@')
152assert('@'..a == string.gsub(t, '', '@') and b == 5) 168assert(a == "a@b@ç@ @d@" and b == 5)
153a, b = string.gsub('abçd', '(.)', '%0@', 2) 169a, b = string.gsub('abçd', PU'(.)', '%0@', 2)
154assert(a == 'a@b@çd' and b == 2) 170assert(a == 'a@b@çd' and b == 2)
155assert(string.gsub('alo alo', '()[al]', '%1') == '12o 56o') 171assert(string.gsub('alo alo', '()[al]', '%1') == '12o 56o')
156assert(string.gsub("abc=xyz", "(%w*)(%p)(%w+)", "%3%2%1-%0") == 172assert(string.gsub("abc=xyz", "(%w*)(%p)(%w+)", "%3%2%1-%0") ==
157 "xyz=abc-abc=xyz") 173 "xyz=abc-abc=xyz")
158assert(string.gsub("abc", "%w", "%1%0") == "aabbcc") 174assert(string.gsub("abc", "%w", "%1%0") == "aabbcc")
159assert(string.gsub("abc", "%w+", "%0%1") == "abcabc") 175assert(string.gsub("abc", "%w+", "%0%1") == "abcabc")
160assert(string.gsub('áéí', '$', '\0óú') == 'áéí\0óú') 176assert(string.gsub('áéí', '$', '\0óú') == 'áéí\0óú')
161assert(string.gsub('', '^', 'r') == 'r') 177assert(string.gsub('', '^', 'r') == 'r')
162assert(string.gsub('', '$', 'r') == 'r') 178assert(string.gsub('', '$', 'r') == 'r')
163print('+') 179print('+')
@@ -188,8 +204,8 @@ do
188end 204end
189 205
190function f(a,b) return string.gsub(a,'.',b) end 206function f(a,b) return string.gsub(a,'.',b) end
191assert(string.gsub("trocar tudo em |teste|b| é |beleza|al|", "|([^|]*)|([^|]*)|", f) == 207assert(string.gsub("trocar tudo em |teste|b| é |beleza|al|", "|([^|]*)|([^|]*)|", f) ==
192 "trocar tudo em bbbbb é alalalalalal") 208 "trocar tudo em bbbbb é alalalalalal")
193 209
194local function dostring (s) return load(s, "")() or "" end 210local function dostring (s) return load(s, "")() or "" end
195assert(string.gsub("alo $a='x'$ novamente $return a$", 211assert(string.gsub("alo $a='x'$ novamente $return a$",
diff --git a/testes/sort.lua b/testes/sort.lua
index 52919b8c..40bb2d8a 100644
--- a/testes/sort.lua
+++ b/testes/sort.lua
@@ -289,7 +289,7 @@ timesort(a, limit, function(x,y) return nil end, "equal")
289 289
290for i,v in pairs(a) do assert(v == false) end 290for i,v in pairs(a) do assert(v == false) end
291 291
292AA = {"álo", "\0first :-)", "alo", "then this one", "45", "and a new"} 292AA = {"\xE1lo", "\0first :-)", "alo", "then this one", "45", "and a new"}
293table.sort(AA) 293table.sort(AA)
294check(AA) 294check(AA)
295 295
diff --git a/testes/strings.lua b/testes/strings.lua
index b033c6ab..90983edd 100644
--- a/testes/strings.lua
+++ b/testes/strings.lua
@@ -1,6 +1,9 @@
1-- $Id: testes/strings.lua $ 1-- $Id: testes/strings.lua $
2-- See Copyright Notice in file all.lua 2-- See Copyright Notice in file all.lua
3 3
4-- ISO Latin encoding
5
6
4print('testing strings and string library') 7print('testing strings and string library')
5 8
6local maxi <const> = math.maxinteger 9local maxi <const> = math.maxinteger
diff --git a/testes/utf8.lua b/testes/utf8.lua
index c5a9dd3f..efadbd5c 100644
--- a/testes/utf8.lua
+++ b/testes/utf8.lua
@@ -1,6 +1,8 @@
1-- $Id: testes/utf8.lua $ 1-- $Id: testes/utf8.lua $
2-- See Copyright Notice in file all.lua 2-- See Copyright Notice in file all.lua
3 3
4-- UTF-8 file
5
4print "testing UTF-8 library" 6print "testing UTF-8 library"
5 7
6local utf8 = require'utf8' 8local utf8 = require'utf8'