2 ** $Id: lstring.c,v 2.26.1.1 2013/04/12 18:48:47 roberto Exp $
3 ** String table (keeps all strings handled by Lua)
4 ** See Copyright Notice in lua.h
11 #include <sys/lua/lua.h>
20 ** Lua will use at most ~(2^LUAI_HASHLIMIT) bytes from a string to
23 #if !defined(LUAI_HASHLIMIT)
24 #define LUAI_HASHLIMIT 5
29 ** equality for long strings
31 int luaS_eqlngstr (TString
*a
, TString
*b
) {
32 size_t len
= a
->tsv
.len
;
33 lua_assert(a
->tsv
.tt
== LUA_TLNGSTR
&& b
->tsv
.tt
== LUA_TLNGSTR
);
34 return (a
== b
) || /* same instance or... */
35 ((len
== b
->tsv
.len
) && /* equal length and ... */
36 (memcmp(getstr(a
), getstr(b
), len
) == 0)); /* equal contents */
41 ** equality for strings
43 int luaS_eqstr (TString
*a
, TString
*b
) {
44 return (a
->tsv
.tt
== b
->tsv
.tt
) &&
45 (a
->tsv
.tt
== LUA_TSHRSTR
? eqshrstr(a
, b
) : luaS_eqlngstr(a
, b
));
49 unsigned int luaS_hash (const char *str
, size_t l
, unsigned int seed
) {
50 unsigned int h
= seed
^ cast(unsigned int, l
);
52 size_t step
= (l
>> LUAI_HASHLIMIT
) + 1;
53 for (l1
= l
; l1
>= step
; l1
-= step
)
54 h
= h
^ ((h
<<5) + (h
>>2) + cast_byte(str
[l1
- 1]));
60 ** resizes the string table
62 void luaS_resize (lua_State
*L
, int newsize
) {
64 stringtable
*tb
= &G(L
)->strt
;
65 /* cannot resize while GC is traversing strings */
66 luaC_runtilstate(L
, ~bitmask(GCSsweepstring
));
67 if (newsize
> tb
->size
) {
68 luaM_reallocvector(L
, tb
->hash
, tb
->size
, newsize
, GCObject
*);
69 for (i
= tb
->size
; i
< newsize
; i
++) tb
->hash
[i
] = NULL
;
72 for (i
=0; i
<tb
->size
; i
++) {
73 GCObject
*p
= tb
->hash
[i
];
75 while (p
) { /* for each node in the list */
76 GCObject
*next
= gch(p
)->next
; /* save next */
77 unsigned int h
= lmod(gco2ts(p
)->hash
, newsize
); /* new position */
78 gch(p
)->next
= tb
->hash
[h
]; /* chain it */
80 resetoldbit(p
); /* see MOVE OLD rule */
84 if (newsize
< tb
->size
) {
85 /* shrinking slice must be empty */
86 lua_assert(tb
->hash
[newsize
] == NULL
&& tb
->hash
[tb
->size
- 1] == NULL
);
87 luaM_reallocvector(L
, tb
->hash
, tb
->size
, newsize
, GCObject
*);
94 ** creates a new string object
96 static TString
*createstrobj (lua_State
*L
, const char *str
, size_t l
,
97 int tag
, unsigned int h
, GCObject
**list
) {
100 size_t totalsize
; /* total size of TString object */
101 totalsize
= sizeof(TString
) + ((l
+ 1) * sizeof(char));
102 ts
= &luaC_newobj(L
, tag
, totalsize
, list
, 0)->ts
;
106 sbuf
= (char *)(TString
*)(ts
+ 1);
107 memcpy(sbuf
, str
, l
*sizeof(char));
108 sbuf
[l
] = '\0'; /* ending 0 */
114 ** creates a new short string, inserting it into string table
116 static TString
*newshrstr (lua_State
*L
, const char *str
, size_t l
,
118 GCObject
**list
; /* (pointer to) list where it will be inserted */
119 stringtable
*tb
= &G(L
)->strt
;
121 if (tb
->nuse
>= cast(lu_int32
, tb
->size
) && tb
->size
<= MAX_INT
/2)
122 luaS_resize(L
, tb
->size
*2); /* too crowded */
123 list
= &tb
->hash
[lmod(h
, tb
->size
)];
124 s
= createstrobj(L
, str
, l
, LUA_TSHRSTR
, h
, list
);
131 ** checks whether short string exists and reuses it or creates a new one
133 static TString
*internshrstr (lua_State
*L
, const char *str
, size_t l
) {
135 global_State
*g
= G(L
);
136 unsigned int h
= luaS_hash(str
, l
, g
->seed
);
137 for (o
= g
->strt
.hash
[lmod(h
, g
->strt
.size
)];
140 TString
*ts
= rawgco2ts(o
);
141 if (h
== ts
->tsv
.hash
&&
143 (memcmp(str
, getstr(ts
), l
* sizeof(char)) == 0)) {
144 if (isdead(G(L
), o
)) /* string is dead (but was not collected yet)? */
145 changewhite(o
); /* resurrect it */
149 return newshrstr(L
, str
, l
, h
); /* not found; create a new string */
154 ** new string (with explicit length)
156 TString
*luaS_newlstr (lua_State
*L
, const char *str
, size_t l
) {
157 if (l
<= LUAI_MAXSHORTLEN
) /* short string? */
158 return internshrstr(L
, str
, l
);
160 if (l
+ 1 > (MAX_SIZET
- sizeof(TString
))/sizeof(char))
162 return createstrobj(L
, str
, l
, LUA_TLNGSTR
, G(L
)->seed
, NULL
);
168 ** new zero-terminated string
170 TString
*luaS_new (lua_State
*L
, const char *str
) {
171 return luaS_newlstr(L
, str
, strlen(str
));
175 Udata
*luaS_newudata (lua_State
*L
, size_t s
, Table
*e
) {
177 if (s
> MAX_SIZET
- sizeof(Udata
))
179 u
= &luaC_newobj(L
, LUA_TUSERDATA
, sizeof(Udata
) + s
, NULL
, 0)->u
;
181 u
->uv
.metatable
= NULL
;