aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRoberto Ierusalimschy <roberto@inf.puc-rio.br>2024-03-18 15:56:32 -0300
committerRoberto Ierusalimschy <roberto@inf.puc-rio.br>2024-03-18 15:56:32 -0300
commitce6f5502c99ce9a367e25b678e375db6f8164d73 (patch)
tree47f36dc2f6da96dfda325d7b587f3a20599e9317
parentba710603811c68fe3a69b3bb98e9038d37489a79 (diff)
downloadlua-ce6f5502c99ce9a367e25b678e375db6f8164d73.tar.gz
lua-ce6f5502c99ce9a367e25b678e375db6f8164d73.tar.bz2
lua-ce6f5502c99ce9a367e25b678e375db6f8164d73.zip
'luaH_get' functions return 'TValue'
Instead of receiving a parameter telling them where to put the result of the query, these functions return the TValue directly. (That is, they return a structure.)
Diffstat (limited to '')
-rw-r--r--lapi.c61
-rw-r--r--lcode.c5
-rw-r--r--ldump.c4
-rw-r--r--lobject.h19
-rw-r--r--ltable.c53
-rw-r--r--ltable.h29
-rw-r--r--lundump.c3
-rw-r--r--lvm.c54
-rw-r--r--lvm.h18
9 files changed, 124 insertions, 122 deletions
diff --git a/lapi.c b/lapi.c
index 7df63798..a6ef5663 100644
--- a/lapi.c
+++ b/lapi.c
@@ -666,47 +666,45 @@ LUA_API int lua_pushthread (lua_State *L) {
666 666
667 667
668static int auxgetstr (lua_State *L, const TValue *t, const char *k) { 668static int auxgetstr (lua_State *L, const TValue *t, const char *k) {
669 int hres; 669 TValue aux;
670 TString *str = luaS_new(L, k); 670 TString *str = luaS_new(L, k);
671 luaV_fastget(t, str, s2v(L->top.p), luaH_getstr, hres); 671 luaV_fastget(t, str, s2v(L->top.p), luaH_getstr, aux);
672 if (hres == HOK) { 672 if (!isemptyV(aux)) {
673 api_incr_top(L); 673 api_incr_top(L);
674 } 674 }
675 else { 675 else {
676 setsvalue2s(L, L->top.p, str); 676 setsvalue2s(L, L->top.p, str);
677 api_incr_top(L); 677 api_incr_top(L);
678 luaV_finishget(L, t, s2v(L->top.p - 1), L->top.p - 1, hres); 678 luaV_finishget(L, t, s2v(L->top.p - 1), L->top.p - 1, aux);
679 } 679 }
680 lua_unlock(L); 680 lua_unlock(L);
681 return ttype(s2v(L->top.p - 1)); 681 return ttype(s2v(L->top.p - 1));
682} 682}
683 683
684 684
685static void getGlobalTable (lua_State *L, TValue *gt) { 685static TValue getGlobalTable (lua_State *L) {
686 Table *registry = hvalue(&G(L)->l_registry); 686 Table *registry = hvalue(&G(L)->l_registry);
687 int hres = luaH_getint(registry, LUA_RIDX_GLOBALS, gt); 687 return luaH_getint(registry, LUA_RIDX_GLOBALS);
688 (void)hres; /* avoid warnings (not used) when checks are off */
689 api_check(L, hres == HOK, "global table must exist");
690} 688}
691 689
692 690
693LUA_API int lua_getglobal (lua_State *L, const char *name) { 691LUA_API int lua_getglobal (lua_State *L, const char *name) {
694 TValue gt; 692 TValue gt;
695 lua_lock(L); 693 lua_lock(L);
696 getGlobalTable(L, &gt); 694 gt = getGlobalTable(L);
697 return auxgetstr(L, &gt, name); 695 return auxgetstr(L, &gt, name);
698} 696}
699 697
700 698
701LUA_API int lua_gettable (lua_State *L, int idx) { 699LUA_API int lua_gettable (lua_State *L, int idx) {
702 int hres; 700 TValue aux;
703 TValue *t; 701 TValue *t;
704 lua_lock(L); 702 lua_lock(L);
705 api_checkpop(L, 1); 703 api_checkpop(L, 1);
706 t = index2value(L, idx); 704 t = index2value(L, idx);
707 luaV_fastget(t, s2v(L->top.p - 1), s2v(L->top.p - 1), luaH_get, hres); 705 luaV_fastget(t, s2v(L->top.p - 1), s2v(L->top.p - 1), luaH_get, aux);
708 if (hres != HOK) 706 if (isemptyV(aux))
709 luaV_finishget(L, t, s2v(L->top.p - 1), L->top.p - 1, hres); 707 luaV_finishget(L, t, s2v(L->top.p - 1), L->top.p - 1, aux);
710 lua_unlock(L); 708 lua_unlock(L);
711 return ttype(s2v(L->top.p - 1)); 709 return ttype(s2v(L->top.p - 1));
712} 710}
@@ -720,14 +718,14 @@ LUA_API int lua_getfield (lua_State *L, int idx, const char *k) {
720 718
721LUA_API int lua_geti (lua_State *L, int idx, lua_Integer n) { 719LUA_API int lua_geti (lua_State *L, int idx, lua_Integer n) {
722 TValue *t; 720 TValue *t;
723 int hres; 721 TValue aux;
724 lua_lock(L); 722 lua_lock(L);
725 t = index2value(L, idx); 723 t = index2value(L, idx);
726 luaV_fastgeti(t, n, s2v(L->top.p), hres); 724 luaV_fastgeti(t, n, s2v(L->top.p), aux);
727 if (hres != HOK) { 725 if (isemptyV(aux)) {
728 TValue key; 726 TValue key;
729 setivalue(&key, n); 727 setivalue(&key, n);
730 luaV_finishget(L, t, &key, L->top.p, hres); 728 luaV_finishget(L, t, &key, L->top.p, aux);
731 } 729 }
732 api_incr_top(L); 730 api_incr_top(L);
733 lua_unlock(L); 731 lua_unlock(L);
@@ -735,12 +733,14 @@ LUA_API int lua_geti (lua_State *L, int idx, lua_Integer n) {
735} 733}
736 734
737 735
738l_sinline int finishrawget (lua_State *L, int hres) { 736l_sinline int finishrawget (lua_State *L, TValue res) {
739 if (hres != HOK) /* avoid copying empty items to the stack */ 737 if (isemptyV(res)) /* avoid copying empty items to the stack */
740 setnilvalue(s2v(L->top.p)); 738 setnilvalue(s2v(L->top.p));
739 else
740 setobjV(L, s2v(L->top.p), res);
741 api_incr_top(L); 741 api_incr_top(L);
742 lua_unlock(L); 742 lua_unlock(L);
743 return ttype(s2v(L->top.p - 1)); 743 return ttypeV(res);
744} 744}
745 745
746 746
@@ -753,23 +753,23 @@ l_sinline Table *gettable (lua_State *L, int idx) {
753 753
754LUA_API int lua_rawget (lua_State *L, int idx) { 754LUA_API int lua_rawget (lua_State *L, int idx) {
755 Table *t; 755 Table *t;
756 TValue res;
756 lua_lock(L); 757 lua_lock(L);
757 api_checkpop(L, 1); 758 api_checkpop(L, 1);
758 t = gettable(L, idx); 759 t = gettable(L, idx);
759 if (luaH_get(t, s2v(L->top.p - 1), s2v(L->top.p - 1)) != HOK) 760 res = luaH_get(t, s2v(L->top.p - 1));
760 setnilvalue(s2v(L->top.p - 1)); 761 L->top.p--; /* pop key */
761 lua_unlock(L); 762 return finishrawget(L, res);
762 return ttype(s2v(L->top.p - 1));
763} 763}
764 764
765 765
766LUA_API int lua_rawgeti (lua_State *L, int idx, lua_Integer n) { 766LUA_API int lua_rawgeti (lua_State *L, int idx, lua_Integer n) {
767 Table *t; 767 Table *t;
768 int hres; 768 TValue aux;
769 lua_lock(L); 769 lua_lock(L);
770 t = gettable(L, idx); 770 t = gettable(L, idx);
771 luaH_fastgeti(t, n, s2v(L->top.p), hres); 771 luaH_fastgeti(t, n, s2v(L->top.p), aux);
772 return finishrawget(L, hres); 772 return finishrawget(L, aux);
773} 773}
774 774
775 775
@@ -779,7 +779,7 @@ LUA_API int lua_rawgetp (lua_State *L, int idx, const void *p) {
779 lua_lock(L); 779 lua_lock(L);
780 t = gettable(L, idx); 780 t = gettable(L, idx);
781 setpvalue(&k, cast_voidp(p)); 781 setpvalue(&k, cast_voidp(p));
782 return finishrawget(L, luaH_get(t, &k, s2v(L->top.p))); 782 return finishrawget(L, luaH_get(t, &k));
783} 783}
784 784
785 785
@@ -872,7 +872,7 @@ static void auxsetstr (lua_State *L, const TValue *t, const char *k) {
872LUA_API void lua_setglobal (lua_State *L, const char *name) { 872LUA_API void lua_setglobal (lua_State *L, const char *name) {
873 TValue gt; 873 TValue gt;
874 lua_lock(L); /* unlock done in 'auxsetstr' */ 874 lua_lock(L); /* unlock done in 'auxsetstr' */
875 getGlobalTable(L, &gt); 875 gt = getGlobalTable(L);
876 auxsetstr(L, &gt, name); 876 auxsetstr(L, &gt, name);
877} 877}
878 878
@@ -1122,8 +1122,7 @@ LUA_API int lua_load (lua_State *L, lua_Reader reader, void *data,
1122 LClosure *f = clLvalue(s2v(L->top.p - 1)); /* get new function */ 1122 LClosure *f = clLvalue(s2v(L->top.p - 1)); /* get new function */
1123 if (f->nupvalues >= 1) { /* does it have an upvalue? */ 1123 if (f->nupvalues >= 1) { /* does it have an upvalue? */
1124 /* get global table from registry */ 1124 /* get global table from registry */
1125 TValue gt; 1125 TValue gt = getGlobalTable(L);
1126 getGlobalTable(L, &gt);
1127 /* set global table as 1st upvalue of 'f' (may be LUA_ENV) */ 1126 /* set global table as 1st upvalue of 'f' (may be LUA_ENV) */
1128 setobj(L, f->upvals[0]->v.p, &gt); 1127 setobj(L, f->upvals[0]->v.p, &gt);
1129 luaC_barrier(L, f->upvals[0], &gt); 1128 luaC_barrier(L, f->upvals[0], &gt);
diff --git a/lcode.c b/lcode.c
index 0d888822..18bf9413 100644
--- a/lcode.c
+++ b/lcode.c
@@ -541,12 +541,11 @@ static void freeexps (FuncState *fs, expdesc *e1, expdesc *e2) {
541** a function can make some indices wrong. 541** a function can make some indices wrong.
542*/ 542*/
543static int addk (FuncState *fs, TValue *key, TValue *v) { 543static int addk (FuncState *fs, TValue *key, TValue *v) {
544 TValue val;
545 lua_State *L = fs->ls->L; 544 lua_State *L = fs->ls->L;
546 Proto *f = fs->f; 545 Proto *f = fs->f;
547 int aux = luaH_get(fs->ls->h, key, &val); /* query scanner table */ 546 TValue val = luaH_get(fs->ls->h, key); /* query scanner table */
548 int k, oldsize; 547 int k, oldsize;
549 if (aux == HOK && ttisinteger(&val)) { /* is there an index there? */ 548 if (ttisintegerV(val)) { /* is there an index there? */
550 k = cast_int(ivalue(&val)); 549 k = cast_int(ivalue(&val));
551 /* correct value? (warning: must distinguish floats from integers!) */ 550 /* correct value? (warning: must distinguish floats from integers!) */
552 if (k < fs->nk && ttypetag(&f->k[k]) == ttypetag(v) && 551 if (k < fs->nk && ttypetag(&f->k[k]) == ttypetag(v) &&
diff --git a/ldump.c b/ldump.c
index 0d20fb0a..34b63a8a 100644
--- a/ldump.c
+++ b/ldump.c
@@ -132,8 +132,8 @@ static void dumpString (DumpState *D, TString *ts) {
132 if (ts == NULL) 132 if (ts == NULL)
133 dumpSize(D, 0); 133 dumpSize(D, 0);
134 else { 134 else {
135 TValue idx; 135 TValue idx = luaH_getstr(D->h, ts);
136 if (luaH_getstr(D->h, ts, &idx) == HOK) { /* string already saved? */ 136 if (!isemptyV(idx)) { /* string already saved? */
137 dumpSize(D, 1); /* reuse a saved string */ 137 dumpSize(D, 1); /* reuse a saved string */
138 dumpSize(D, cast_sizet(ivalue(&idx))); /* index of saved string */ 138 dumpSize(D, cast_sizet(ivalue(&idx))); /* index of saved string */
139 } 139 }
diff --git a/lobject.h b/lobject.h
index 81dfd475..69fa6260 100644
--- a/lobject.h
+++ b/lobject.h
@@ -75,6 +75,7 @@ typedef struct TValue {
75 75
76/* raw type tag of a TValue */ 76/* raw type tag of a TValue */
77#define rawtt(o) ((o)->tt_) 77#define rawtt(o) ((o)->tt_)
78#define rawttV(o) ((o).tt_)
78 79
79/* tag with no variants (bits 0-3) */ 80/* tag with no variants (bits 0-3) */
80#define novariant(t) ((t) & 0x0F) 81#define novariant(t) ((t) & 0x0F)
@@ -82,14 +83,18 @@ typedef struct TValue {
82/* type tag of a TValue (bits 0-3 for tags + variant bits 4-5) */ 83/* type tag of a TValue (bits 0-3 for tags + variant bits 4-5) */
83#define withvariant(t) ((t) & 0x3F) 84#define withvariant(t) ((t) & 0x3F)
84#define ttypetag(o) withvariant(rawtt(o)) 85#define ttypetag(o) withvariant(rawtt(o))
86#define ttypetagV(o) withvariant(rawttV(o))
85 87
86/* type of a TValue */ 88/* type of a TValue */
87#define ttype(o) (novariant(rawtt(o))) 89#define ttype(o) (novariant(rawtt(o)))
90#define ttypeV(o) (novariant(rawttV(o)))
88 91
89 92
90/* Macros to test type */ 93/* Macros to test type */
91#define checktag(o,t) (rawtt(o) == (t)) 94#define checktag(o,t) (rawtt(o) == (t))
95#define checktagV(o,t) (rawttV(o) == (t))
92#define checktype(o,t) (ttype(o) == (t)) 96#define checktype(o,t) (ttype(o) == (t))
97#define checktypeV(o,t) (ttypeV(o) == (t))
93 98
94 99
95/* Macros for internal tests */ 100/* Macros for internal tests */
@@ -112,6 +117,7 @@ typedef struct TValue {
112 117
113/* set a value's tag */ 118/* set a value's tag */
114#define settt_(o,t) ((o)->tt_=(t)) 119#define settt_(o,t) ((o)->tt_=(t))
120#define setttV_(o,t) ((o).tt_=(t))
115 121
116 122
117/* main macro to copy values (from 'obj2' to 'obj1') */ 123/* main macro to copy values (from 'obj2' to 'obj1') */
@@ -120,6 +126,11 @@ typedef struct TValue {
120 io1->value_ = io2->value_; settt_(io1, io2->tt_); \ 126 io1->value_ = io2->value_; settt_(io1, io2->tt_); \
121 checkliveness(L,io1); lua_assert(!isnonstrictnil(io1)); } 127 checkliveness(L,io1); lua_assert(!isnonstrictnil(io1)); }
122 128
129#define setobjV(L,obj1,obj2) \
130 { TValue *io1=(obj1); const TValue io2=(obj2); \
131 io1->value_ = io2.value_; settt_(io1, io2.tt_); \
132 checkliveness(L,io1); lua_assert(!isnonstrictnil(io1)); }
133
123/* 134/*
124** Different types of assignments, according to source and destination. 135** Different types of assignments, according to source and destination.
125** (They are mostly equal now, but may be different in the future.) 136** (They are mostly equal now, but may be different in the future.)
@@ -188,9 +199,15 @@ typedef union {
188/* Value returned for a key not found in a table (absent key) */ 199/* Value returned for a key not found in a table (absent key) */
189#define LUA_VABSTKEY makevariant(LUA_TNIL, 2) 200#define LUA_VABSTKEY makevariant(LUA_TNIL, 2)
190 201
202/* Special "value" to signal that a fast get is accessing a non-table */
203#define LUA_VNOTABLE makevariant(LUA_TNIL, 3)
204
205#define setnotableV(obj) setttV_(obj, LUA_VNOTABLE)
206
191 207
192/* macro to test for (any kind of) nil */ 208/* macro to test for (any kind of) nil */
193#define ttisnil(v) checktype((v), LUA_TNIL) 209#define ttisnil(v) checktype((v), LUA_TNIL)
210#define ttisnilV(v) checktypeV((v), LUA_TNIL)
194 211
195#define tagisempty(tag) (novariant(tag) == LUA_TNIL) 212#define tagisempty(tag) (novariant(tag) == LUA_TNIL)
196 213
@@ -217,6 +234,7 @@ typedef union {
217** be accepted as empty.) 234** be accepted as empty.)
218*/ 235*/
219#define isempty(v) ttisnil(v) 236#define isempty(v) ttisnil(v)
237#define isemptyV(v) checktypeV((v), LUA_TNIL)
220 238
221 239
222/* macro defining a value corresponding to an absent key */ 240/* macro defining a value corresponding to an absent key */
@@ -328,6 +346,7 @@ typedef struct GCObject {
328#define ttisnumber(o) checktype((o), LUA_TNUMBER) 346#define ttisnumber(o) checktype((o), LUA_TNUMBER)
329#define ttisfloat(o) checktag((o), LUA_VNUMFLT) 347#define ttisfloat(o) checktag((o), LUA_VNUMFLT)
330#define ttisinteger(o) checktag((o), LUA_VNUMINT) 348#define ttisinteger(o) checktag((o), LUA_VNUMINT)
349#define ttisintegerV(o) checktagV((o), LUA_VNUMINT)
331 350
332#define nvalue(o) check_exp(ttisnumber(o), \ 351#define nvalue(o) check_exp(ttisnumber(o), \
333 (ttisinteger(o) ? cast_num(ivalue(o)) : fltvalue(o))) 352 (ttisinteger(o) ? cast_num(ivalue(o)) : fltvalue(o)))
diff --git a/ltable.c b/ltable.c
index c5f48716..f675f39b 100644
--- a/ltable.c
+++ b/ltable.c
@@ -904,28 +904,14 @@ static int hashkeyisempty (Table *t, lua_Integer key) {
904} 904}
905 905
906 906
907static int finishnodeget (const TValue *val, TValue *res) { 907TValue luaH_getint (Table *t, lua_Integer key) {
908 if (!ttisnil(val)) {
909 setobj(((lua_State*)NULL), res, val);
910 return HOK; /* success */
911 }
912 else
913 return HNOTFOUND; /* could not get value */
914}
915
916
917int luaH_getint (Table *t, lua_Integer key, TValue *res) {
918 if (keyinarray(t, key)) { 908 if (keyinarray(t, key)) {
919 int tag = *getArrTag(t, key - 1); 909 TValue res;
920 if (!tagisempty(tag)) { 910 arr2objV(t, key, res);
921 farr2val(t, key, tag, res); 911 return res;
922 return HOK; /* success */
923 }
924 else
925 return ~cast_int(key); /* empty slot in the array part */
926 } 912 }
927 else 913 else
928 return finishnodeget(getintfromhash(t, key), res); 914 return *getintfromhash(t, key);
929} 915}
930 916
931 917
@@ -948,8 +934,8 @@ const TValue *luaH_Hgetshortstr (Table *t, TString *key) {
948} 934}
949 935
950 936
951int luaH_getshortstr (Table *t, TString *key, TValue *res) { 937TValue luaH_getshortstr (Table *t, TString *key) {
952 return finishnodeget(luaH_Hgetshortstr(t, key), res); 938 return *luaH_Hgetshortstr(t, key);
953} 939}
954 940
955 941
@@ -964,8 +950,8 @@ static const TValue *Hgetstr (Table *t, TString *key) {
964} 950}
965 951
966 952
967int luaH_getstr (Table *t, TString *key, TValue *res) { 953TValue luaH_getstr (Table *t, TString *key) {
968 return finishnodeget(Hgetstr(t, key), res); 954 return *Hgetstr(t, key);
969} 955}
970 956
971 957
@@ -981,34 +967,31 @@ TString *luaH_getstrkey (Table *t, TString *key) {
981/* 967/*
982** main search function 968** main search function
983*/ 969*/
984int luaH_get (Table *t, const TValue *key, TValue *res) { 970TValue luaH_get (Table *t, const TValue *key) {
985 const TValue *slot;
986 switch (ttypetag(key)) { 971 switch (ttypetag(key)) {
987 case LUA_VSHRSTR: 972 case LUA_VSHRSTR:
988 slot = luaH_Hgetshortstr(t, tsvalue(key)); 973 return *luaH_Hgetshortstr(t, tsvalue(key));
989 break; 974 break;
990 case LUA_VNUMINT: 975 case LUA_VNUMINT:
991 return luaH_getint(t, ivalue(key), res); 976 return luaH_getint(t, ivalue(key));
992 case LUA_VNIL: 977 case LUA_VNIL:
993 slot = &absentkey; 978 return absentkey;
994 break; 979 break;
995 case LUA_VNUMFLT: { 980 case LUA_VNUMFLT: {
996 lua_Integer k; 981 lua_Integer k;
997 if (luaV_flttointeger(fltvalue(key), &k, F2Ieq)) /* integral index? */ 982 if (luaV_flttointeger(fltvalue(key), &k, F2Ieq)) /* integral index? */
998 return luaH_getint(t, k, res); /* use specialized version */ 983 return luaH_getint(t, k); /* use specialized version */
999 /* else... */ 984 /* else... */
1000 } /* FALLTHROUGH */ 985 } /* FALLTHROUGH */
1001 default: 986 default:
1002 slot = getgeneric(t, key, 0); 987 return *getgeneric(t, key, 0);
1003 break;
1004 } 988 }
1005 return finishnodeget(slot, res);
1006} 989}
1007 990
1008 991
1009static int finishnodeset (Table *t, const TValue *slot, TValue *val) { 992static int finishnodeset (Table *t, const TValue *slot, TValue *val) {
1010 if (!ttisnil(slot)) { 993 if (!ttisnil(slot)) {
1011 setobj(((lua_State*)NULL), cast(TValue*, slot), val); 994 setobj(cast(lua_State*, NULL), cast(TValue*, slot), val);
1012 return HOK; /* success */ 995 return HOK; /* success */
1013 } 996 }
1014 else if (isabstkey(slot)) 997 else if (isabstkey(slot))
@@ -1022,7 +1005,7 @@ static int rawfinishnodeset (const TValue *slot, TValue *val) {
1022 if (isabstkey(slot)) 1005 if (isabstkey(slot))
1023 return 0; /* no slot with that key */ 1006 return 0; /* no slot with that key */
1024 else { 1007 else {
1025 setobj(((lua_State*)NULL), cast(TValue*, slot), val); 1008 setobj(cast(lua_State*, NULL), cast(TValue*, slot), val);
1026 return 1; /* success */ 1009 return 1; /* success */
1027 } 1010 }
1028} 1011}
diff --git a/ltable.h b/ltable.h
index 8b0340b5..10dae5c7 100644
--- a/ltable.h
+++ b/ltable.h
@@ -46,13 +46,11 @@
46 46
47 47
48 48
49#define luaH_fastgeti(t,k,res,hres) \ 49#define luaH_fastgeti(t,k,res,aux) \
50 { Table *h = t; lua_Unsigned u = l_castS2U(k); \ 50 { Table *h = t; lua_Unsigned u = l_castS2U(k); \
51 if ((u - 1u < h->alimit)) { \ 51 if ((u - 1u < h->alimit)) arr2objV(h,u,aux); \
52 int tag = *getArrTag(h,(u)-1u); \ 52 else aux = luaH_getint(h, u); \
53 if (tagisempty(tag)) hres = HNOTFOUND; \ 53 if (!isemptyV(aux)) setobjV(cast(lua_State*, NULL), res, aux); }
54 else { farr2val(h, u, tag, res); hres = HOK; }} \
55 else { hres = luaH_getint(h, u, res); }}
56 54
57 55
58#define luaH_fastseti(t,k,val,hres) \ 56#define luaH_fastseti(t,k,val,hres) \
@@ -64,15 +62,13 @@
64 else { hres = luaH_psetint(h, u, val); }} 62 else { hres = luaH_psetint(h, u, val); }}
65 63
66 64
67/* results from get/pset */ 65/* results from pset */
68#define HOK 0 66#define HOK 0
69#define HNOTFOUND 1 67#define HNOTFOUND 1
70#define HNOTATABLE 2 68#define HNOTATABLE 2
71#define HFIRSTNODE 3 69#define HFIRSTNODE 3
72 70
73/* 71/*
74** 'luaH_get*' operations set 'res' and return HOK, unless the value is
75** absent. In that case, they set nothing and return HNOTFOUND.
76** The 'luaH_pset*' (pre-set) operations set the given value and return 72** The 'luaH_pset*' (pre-set) operations set the given value and return
77** HOK, unless the original value is absent. In that case, if the key 73** HOK, unless the original value is absent. In that case, if the key
78** is really absent, they return HNOTFOUND. Otherwise, if there is a 74** is really absent, they return HNOTFOUND. Otherwise, if there is a
@@ -109,8 +105,10 @@ struct ArrayCell {
109/* 105/*
110** Move TValues to/from arrays, using Lua indices 106** Move TValues to/from arrays, using Lua indices
111*/ 107*/
112#define arr2obj(h,k,val) \ 108#define arr2objV(h,k,val) \
113 ((val)->tt_ = *getArrTag(h,(k)-1u), (val)->value_ = *getArrVal(h,(k)-1u)) 109 ((val).tt_ = *getArrTag(h,(k)-1u), (val).value_ = *getArrVal(h,(k)-1u))
110
111#define arr2obj(h,k,val) arr2objV(h,k,*(val))
114 112
115#define obj2arr(h,k,val) \ 113#define obj2arr(h,k,val) \
116 (*getArrTag(h,(k)-1u) = (val)->tt_, *getArrVal(h,(k)-1u) = (val)->value_) 114 (*getArrTag(h,(k)-1u) = (val)->tt_, *getArrVal(h,(k)-1u) = (val)->value_)
@@ -128,12 +126,11 @@ struct ArrayCell {
128 (*tag = (val)->tt_, *getArrVal(h,(k)-1u) = (val)->value_) 126 (*tag = (val)->tt_, *getArrVal(h,(k)-1u) = (val)->value_)
129 127
130 128
131LUAI_FUNC int luaH_get (Table *t, const TValue *key, TValue *res); 129LUAI_FUNC TValue luaH_get (Table *t, const TValue *key);
132LUAI_FUNC int luaH_getshortstr (Table *t, TString *key, TValue *res); 130LUAI_FUNC TValue luaH_getshortstr (Table *t, TString *key);
133LUAI_FUNC int luaH_getstr (Table *t, TString *key, TValue *res); 131LUAI_FUNC TValue luaH_getstr (Table *t, TString *key);
134LUAI_FUNC int luaH_getint (Table *t, lua_Integer key, TValue *res); 132LUAI_FUNC TValue luaH_getint (Table *t, lua_Integer key);
135 133
136/* Special get for metamethods */
137LUAI_FUNC const TValue *luaH_Hgetshortstr (Table *t, TString *key); 134LUAI_FUNC const TValue *luaH_Hgetshortstr (Table *t, TString *key);
138 135
139LUAI_FUNC TString *luaH_getstrkey (Table *t, TString *key); 136LUAI_FUNC TString *luaH_getstrkey (Table *t, TString *key);
diff --git a/lundump.c b/lundump.c
index 51d5dc66..593a4951 100644
--- a/lundump.c
+++ b/lundump.c
@@ -149,8 +149,7 @@ static void loadString (LoadState *S, Proto *p, TString **sl) {
149 } 149 }
150 else if (size == 1) { /* previously saved string? */ 150 else if (size == 1) { /* previously saved string? */
151 lua_Integer idx = cast(lua_Integer, loadSize(S)); /* get its index */ 151 lua_Integer idx = cast(lua_Integer, loadSize(S)); /* get its index */
152 TValue stv; 152 TValue stv = luaH_getint(S->h, idx); /* get its value */
153 luaH_getint(S->h, idx, &stv); /* get its value */
154 *sl = ts = tsvalue(&stv); 153 *sl = ts = tsvalue(&stv);
155 luaC_objbarrier(L, p, ts); 154 luaC_objbarrier(L, p, ts);
156 return; /* do not save it again */ 155 return; /* do not save it again */
diff --git a/lvm.c b/lvm.c
index 78e39b71..a251f423 100644
--- a/lvm.c
+++ b/lvm.c
@@ -287,12 +287,13 @@ static int floatforloop (StkId ra) {
287/* 287/*
288** Finish the table access 'val = t[key]'. 288** Finish the table access 'val = t[key]'.
289*/ 289*/
290void luaV_finishget (lua_State *L, const TValue *t, TValue *key, StkId val, 290void luaV_finishget_ (lua_State *L, const TValue *t, TValue *key, StkId val,
291 int hres) { 291 int tag) {
292 int loop; /* counter to avoid infinite loops */ 292 int loop; /* counter to avoid infinite loops */
293 const TValue *tm; /* metamethod */ 293 const TValue *tm; /* metamethod */
294 TValue aux;
294 for (loop = 0; loop < MAXTAGLOOP; loop++) { 295 for (loop = 0; loop < MAXTAGLOOP; loop++) {
295 if (hres == HNOTATABLE) { /* 't' is not a table? */ 296 if (tag == LUA_VNOTABLE) { /* 't' is not a table? */
296 lua_assert(!ttistable(t)); 297 lua_assert(!ttistable(t));
297 tm = luaT_gettmbyobj(L, t, TM_INDEX); 298 tm = luaT_gettmbyobj(L, t, TM_INDEX);
298 if (l_unlikely(notm(tm))) 299 if (l_unlikely(notm(tm)))
@@ -312,10 +313,11 @@ void luaV_finishget (lua_State *L, const TValue *t, TValue *key, StkId val,
312 return; 313 return;
313 } 314 }
314 t = tm; /* else try to access 'tm[key]' */ 315 t = tm; /* else try to access 'tm[key]' */
315 luaV_fastget(t, key, s2v(val), luaH_get, hres); 316 luaV_fastget(t, key, s2v(val), luaH_get, aux);
316 if (hres == HOK) 317 if (!isemptyV(aux))
317 return; /* done */ 318 return; /* done */
318 /* else repeat (tail call 'luaV_finishget') */ 319 /* else repeat (tail call 'luaV_finishget') */
320 tag = ttypetagV(aux);
319 } 321 }
320 luaG_runerror(L, "'__index' chain too long; possible loop"); 322 luaG_runerror(L, "'__index' chain too long; possible loop");
321} 323}
@@ -1245,36 +1247,36 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
1245 TValue *upval = cl->upvals[GETARG_B(i)]->v.p; 1247 TValue *upval = cl->upvals[GETARG_B(i)]->v.p;
1246 TValue *rc = KC(i); 1248 TValue *rc = KC(i);
1247 TString *key = tsvalue(rc); /* key must be a short string */ 1249 TString *key = tsvalue(rc); /* key must be a short string */
1248 int hres; 1250 TValue aux;
1249 luaV_fastget(upval, key, s2v(ra), luaH_getshortstr, hres); 1251 luaV_fastget(upval, key, s2v(ra), luaH_getshortstr, aux);
1250 if (hres != HOK) 1252 if (isemptyV(aux))
1251 Protect(luaV_finishget(L, upval, rc, ra, hres)); 1253 Protect(luaV_finishget(L, upval, rc, ra, aux));
1252 vmbreak; 1254 vmbreak;
1253 } 1255 }
1254 vmcase(OP_GETTABLE) { 1256 vmcase(OP_GETTABLE) {
1255 StkId ra = RA(i); 1257 StkId ra = RA(i);
1256 TValue *rb = vRB(i); 1258 TValue *rb = vRB(i);
1257 TValue *rc = vRC(i); 1259 TValue *rc = vRC(i);
1258 int hres; 1260 TValue aux;
1259 if (ttisinteger(rc)) { /* fast track for integers? */ 1261 if (ttisinteger(rc)) { /* fast track for integers? */
1260 luaV_fastgeti(rb, ivalue(rc), s2v(ra), hres); 1262 luaV_fastgeti(rb, ivalue(rc), s2v(ra), aux);
1261 } 1263 }
1262 else 1264 else
1263 luaV_fastget(rb, rc, s2v(ra), luaH_get, hres); 1265 luaV_fastget(rb, rc, s2v(ra), luaH_get, aux);
1264 if (hres != HOK) /* fast track for integers? */ 1266 if (isemptyV(aux)) /* fast track for integers? */
1265 Protect(luaV_finishget(L, rb, rc, ra, hres)); 1267 Protect(luaV_finishget(L, rb, rc, ra, aux));
1266 vmbreak; 1268 vmbreak;
1267 } 1269 }
1268 vmcase(OP_GETI) { 1270 vmcase(OP_GETI) {
1269 StkId ra = RA(i); 1271 StkId ra = RA(i);
1270 TValue *rb = vRB(i); 1272 TValue *rb = vRB(i);
1271 int c = GETARG_C(i); 1273 int c = GETARG_C(i);
1272 int hres; 1274 TValue aux;
1273 luaV_fastgeti(rb, c, s2v(ra), hres); 1275 luaV_fastgeti(rb, c, s2v(ra), aux);
1274 if (hres != HOK) { 1276 if (isemptyV(aux)) {
1275 TValue key; 1277 TValue key;
1276 setivalue(&key, c); 1278 setivalue(&key, c);
1277 Protect(luaV_finishget(L, rb, &key, ra, hres)); 1279 Protect(luaV_finishget(L, rb, &key, ra, aux));
1278 } 1280 }
1279 vmbreak; 1281 vmbreak;
1280 } 1282 }
@@ -1283,10 +1285,10 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
1283 TValue *rb = vRB(i); 1285 TValue *rb = vRB(i);
1284 TValue *rc = KC(i); 1286 TValue *rc = KC(i);
1285 TString *key = tsvalue(rc); /* key must be a short string */ 1287 TString *key = tsvalue(rc); /* key must be a short string */
1286 int hres; 1288 TValue aux;
1287 luaV_fastget(rb, key, s2v(ra), luaH_getshortstr, hres); 1289 luaV_fastget(rb, key, s2v(ra), luaH_getshortstr, aux);
1288 if (hres != HOK) 1290 if (isemptyV(aux))
1289 Protect(luaV_finishget(L, rb, rc, ra, hres)); 1291 Protect(luaV_finishget(L, rb, rc, ra, aux));
1290 vmbreak; 1292 vmbreak;
1291 } 1293 }
1292 vmcase(OP_SETTABUP) { 1294 vmcase(OP_SETTABUP) {
@@ -1368,14 +1370,14 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
1368 } 1370 }
1369 vmcase(OP_SELF) { 1371 vmcase(OP_SELF) {
1370 StkId ra = RA(i); 1372 StkId ra = RA(i);
1371 int hres; 1373 TValue aux;
1372 TValue *rb = vRB(i); 1374 TValue *rb = vRB(i);
1373 TValue *rc = RKC(i); 1375 TValue *rc = RKC(i);
1374 TString *key = tsvalue(rc); /* key must be a string */ 1376 TString *key = tsvalue(rc); /* key must be a string */
1375 setobj2s(L, ra + 1, rb); 1377 setobj2s(L, ra + 1, rb);
1376 luaV_fastget(rb, key, s2v(ra), luaH_getstr, hres); 1378 luaV_fastget(rb, key, s2v(ra), luaH_getstr, aux);
1377 if (hres != HOK) 1379 if (isemptyV(aux))
1378 Protect(luaV_finishget(L, rb, rc, ra, hres)); 1380 Protect(luaV_finishget(L, rb, rc, ra, aux));
1379 vmbreak; 1381 vmbreak;
1380 } 1382 }
1381 vmcase(OP_ADDI) { 1383 vmcase(OP_ADDI) {
diff --git a/lvm.h b/lvm.h
index 54ee5dd7..3b11e789 100644
--- a/lvm.h
+++ b/lvm.h
@@ -78,17 +78,19 @@ typedef enum {
78/* 78/*
79** fast track for 'gettable' 79** fast track for 'gettable'
80*/ 80*/
81#define luaV_fastget(t,k,res,f, hres) \ 81#define luaV_fastget(t,k,res,f, aux) \
82 (hres = (!ttistable(t) ? HNOTATABLE : f(hvalue(t), k, res))) 82 {if (!ttistable(t)) setnotableV(aux); \
83 else { aux = f(hvalue(t), k); \
84 if (!isemptyV(aux)) { setobjV(cast(lua_State*, NULL), res, aux); } } }
83 85
84 86
85/* 87/*
86** Special case of 'luaV_fastget' for integers, inlining the fast case 88** Special case of 'luaV_fastget' for integers, inlining the fast case
87** of 'luaH_getint'. 89** of 'luaH_getint'.
88*/ 90*/
89#define luaV_fastgeti(t,k,res,hres) \ 91#define luaV_fastgeti(t,k,res,aux) \
90 if (!ttistable(t)) hres = HNOTATABLE; \ 92 { if (!ttistable(t)) setnotableV(aux); \
91 else { luaH_fastgeti(hvalue(t), k, res, hres); } 93 else { luaH_fastgeti(hvalue(t), k, res, aux); } }
92 94
93 95
94#define luaV_fastset(t,k,val,hres,f) \ 96#define luaV_fastset(t,k,val,hres,f) \
@@ -120,8 +122,10 @@ LUAI_FUNC int luaV_tointeger (const TValue *obj, lua_Integer *p, F2Imod mode);
120LUAI_FUNC int luaV_tointegerns (const TValue *obj, lua_Integer *p, 122LUAI_FUNC int luaV_tointegerns (const TValue *obj, lua_Integer *p,
121 F2Imod mode); 123 F2Imod mode);
122LUAI_FUNC int luaV_flttointeger (lua_Number n, lua_Integer *p, F2Imod mode); 124LUAI_FUNC int luaV_flttointeger (lua_Number n, lua_Integer *p, F2Imod mode);
123LUAI_FUNC void luaV_finishget (lua_State *L, const TValue *t, TValue *key, 125#define luaV_finishget(L,t,key,val,aux) \
124 StkId val, int aux); 126 luaV_finishget_(L,t,key,val,ttypetagV(aux))
127LUAI_FUNC void luaV_finishget_ (lua_State *L, const TValue *t, TValue *key,
128 StkId val, int tag);
125LUAI_FUNC void luaV_finishset (lua_State *L, const TValue *t, TValue *key, 129LUAI_FUNC void luaV_finishset (lua_State *L, const TValue *t, TValue *key,
126 TValue *val, int aux); 130 TValue *val, int aux);
127LUAI_FUNC void luaV_finishOp (lua_State *L); 131LUAI_FUNC void luaV_finishOp (lua_State *L);