sqstate.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699
  1. /*
  2. see copyright notice in squirrel.h
  3. */
  4. #include "sqpcheader.h"
  5. #include "sqopcodes.h"
  6. #include "sqvm.h"
  7. #include "sqfuncproto.h"
  8. #include "sqclosure.h"
  9. #include "sqstring.h"
  10. #include "sqtable.h"
  11. #include "sqarray.h"
  12. #include "squserdata.h"
  13. #include "sqclass.h"
  14. //SQObjectPtr _null_;
  15. //SQObjectPtr _true_(true);
  16. //SQObjectPtr _false_(false);
  17. //SQObjectPtr _one_((SQInteger)1);
  18. //SQObjectPtr _minusone_((SQInteger)-1);
  19. SQSharedState::SQSharedState()
  20. {
  21. _compilererrorhandler = NULL;
  22. _printfunc = NULL;
  23. _errorfunc = NULL;
  24. _debuginfo = false;
  25. _notifyallexceptions = false;
  26. }
  27. #define newsysstring(s) { \
  28. _systemstrings->push_back(SQString::Create(this,s)); \
  29. }
  30. #define newmetamethod(s) { \
  31. _metamethods->push_back(SQString::Create(this,s)); \
  32. _table(_metamethodsmap)->NewSlot(_metamethods->back(),(SQInteger)(_metamethods->size()-1)); \
  33. }
  34. bool CompileTypemask(SQIntVec &res,const SQChar *typemask)
  35. {
  36. SQInteger i = 0;
  37. SQInteger mask = 0;
  38. while(typemask[i] != 0) {
  39. switch(typemask[i]){
  40. case 'o': mask |= _RT_NULL; break;
  41. case 'i': mask |= _RT_INTEGER; break;
  42. case 'f': mask |= _RT_FLOAT; break;
  43. case 'n': mask |= (_RT_FLOAT | _RT_INTEGER); break;
  44. case 's': mask |= _RT_STRING; break;
  45. case 't': mask |= _RT_TABLE; break;
  46. case 'a': mask |= _RT_ARRAY; break;
  47. case 'u': mask |= _RT_USERDATA; break;
  48. case 'c': mask |= (_RT_CLOSURE | _RT_NATIVECLOSURE); break;
  49. case 'b': mask |= _RT_BOOL; break;
  50. case 'g': mask |= _RT_GENERATOR; break;
  51. case 'p': mask |= _RT_USERPOINTER; break;
  52. case 'v': mask |= _RT_THREAD; break;
  53. case 'x': mask |= _RT_INSTANCE; break;
  54. case 'y': mask |= _RT_CLASS; break;
  55. case 'r': mask |= _RT_WEAKREF; break;
  56. case '.': mask = -1; res.push_back(mask); i++; mask = 0; continue;
  57. case ' ': i++; continue; //ignores spaces
  58. default:
  59. return false;
  60. }
  61. i++;
  62. if(typemask[i] == '|') {
  63. i++;
  64. if(typemask[i] == 0)
  65. return false;
  66. continue;
  67. }
  68. res.push_back(mask);
  69. mask = 0;
  70. }
  71. return true;
  72. }
  73. SQTable *CreateDefaultDelegate(SQSharedState *ss,SQRegFunction *funcz)
  74. {
  75. SQInteger i=0;
  76. SQTable *t=SQTable::Create(ss,0);
  77. while(funcz[i].name!=0){
  78. SQNativeClosure *nc = SQNativeClosure::Create(ss,funcz[i].f,0);
  79. nc->_nparamscheck = funcz[i].nparamscheck;
  80. nc->_name = SQString::Create(ss,funcz[i].name);
  81. if(funcz[i].typemask && !CompileTypemask(nc->_typecheck,funcz[i].typemask))
  82. return NULL;
  83. t->NewSlot(SQString::Create(ss,funcz[i].name),nc);
  84. i++;
  85. }
  86. return t;
  87. }
  88. void SQSharedState::Init()
  89. {
  90. _scratchpad=NULL;
  91. _scratchpadsize=0;
  92. #ifndef NO_GARBAGE_COLLECTOR
  93. _gc_chain=NULL;
  94. #endif
  95. _stringtable = (SQStringTable*)SQ_MALLOC(sizeof(SQStringTable));
  96. new (_stringtable) SQStringTable(this);
  97. sq_new(_metamethods,SQObjectPtrVec);
  98. sq_new(_systemstrings,SQObjectPtrVec);
  99. sq_new(_types,SQObjectPtrVec);
  100. _metamethodsmap = SQTable::Create(this,MT_LAST-1);
  101. //adding type strings to avoid memory trashing
  102. //types names
  103. newsysstring(_SC("null"));
  104. newsysstring(_SC("table"));
  105. newsysstring(_SC("array"));
  106. newsysstring(_SC("closure"));
  107. newsysstring(_SC("string"));
  108. newsysstring(_SC("userdata"));
  109. newsysstring(_SC("integer"));
  110. newsysstring(_SC("float"));
  111. newsysstring(_SC("userpointer"));
  112. newsysstring(_SC("function"));
  113. newsysstring(_SC("generator"));
  114. newsysstring(_SC("thread"));
  115. newsysstring(_SC("class"));
  116. newsysstring(_SC("instance"));
  117. newsysstring(_SC("bool"));
  118. //meta methods
  119. newmetamethod(MM_ADD);
  120. newmetamethod(MM_SUB);
  121. newmetamethod(MM_MUL);
  122. newmetamethod(MM_DIV);
  123. newmetamethod(MM_UNM);
  124. newmetamethod(MM_MODULO);
  125. newmetamethod(MM_SET);
  126. newmetamethod(MM_GET);
  127. newmetamethod(MM_TYPEOF);
  128. newmetamethod(MM_NEXTI);
  129. newmetamethod(MM_CMP);
  130. newmetamethod(MM_CALL);
  131. newmetamethod(MM_CLONED);
  132. newmetamethod(MM_NEWSLOT);
  133. newmetamethod(MM_DELSLOT);
  134. newmetamethod(MM_TOSTRING);
  135. newmetamethod(MM_NEWMEMBER);
  136. newmetamethod(MM_INHERITED);
  137. _constructoridx = SQString::Create(this,_SC("constructor"));
  138. _registry = SQTable::Create(this,0);
  139. _consts = SQTable::Create(this,0);
  140. _table_default_delegate = CreateDefaultDelegate(this,_table_default_delegate_funcz);
  141. _array_default_delegate = CreateDefaultDelegate(this,_array_default_delegate_funcz);
  142. _string_default_delegate = CreateDefaultDelegate(this,_string_default_delegate_funcz);
  143. _number_default_delegate = CreateDefaultDelegate(this,_number_default_delegate_funcz);
  144. _closure_default_delegate = CreateDefaultDelegate(this,_closure_default_delegate_funcz);
  145. _generator_default_delegate = CreateDefaultDelegate(this,_generator_default_delegate_funcz);
  146. _thread_default_delegate = CreateDefaultDelegate(this,_thread_default_delegate_funcz);
  147. _class_default_delegate = CreateDefaultDelegate(this,_class_default_delegate_funcz);
  148. _instance_default_delegate = CreateDefaultDelegate(this,_instance_default_delegate_funcz);
  149. _weakref_default_delegate = CreateDefaultDelegate(this,_weakref_default_delegate_funcz);
  150. }
  151. SQSharedState::~SQSharedState()
  152. {
  153. _constructoridx.Null();
  154. _table(_registry)->Finalize();
  155. _table(_consts)->Finalize();
  156. _table(_metamethodsmap)->Finalize();
  157. _registry.Null();
  158. _consts.Null();
  159. _metamethodsmap.Null();
  160. while(!_systemstrings->empty()) {
  161. _systemstrings->back().Null();
  162. _systemstrings->pop_back();
  163. }
  164. _thread(_root_vm)->Finalize();
  165. _root_vm.Null();
  166. _table_default_delegate.Null();
  167. _array_default_delegate.Null();
  168. _string_default_delegate.Null();
  169. _number_default_delegate.Null();
  170. _closure_default_delegate.Null();
  171. _generator_default_delegate.Null();
  172. _thread_default_delegate.Null();
  173. _class_default_delegate.Null();
  174. _instance_default_delegate.Null();
  175. _weakref_default_delegate.Null();
  176. _refs_table.Finalize();
  177. #ifndef NO_GARBAGE_COLLECTOR
  178. SQCollectable *t = _gc_chain;
  179. SQCollectable *nx = NULL;
  180. if(t) {
  181. t->_uiRef++;
  182. while(t) {
  183. t->Finalize();
  184. nx = t->_next;
  185. if(nx) nx->_uiRef++;
  186. if(--t->_uiRef == 0)
  187. t->Release();
  188. t = nx;
  189. }
  190. }
  191. assert(_gc_chain==NULL); //just to proove a theory
  192. while(_gc_chain){
  193. _gc_chain->_uiRef++;
  194. _gc_chain->Release();
  195. }
  196. #endif
  197. sq_delete(_types,SQObjectPtrVec);
  198. sq_delete(_systemstrings,SQObjectPtrVec);
  199. sq_delete(_metamethods,SQObjectPtrVec);
  200. sq_delete(_stringtable,SQStringTable);
  201. if(_scratchpad)SQ_FREE(_scratchpad,_scratchpadsize);
  202. }
  203. SQInteger SQSharedState::GetMetaMethodIdxByName(const SQObjectPtr &name)
  204. {
  205. if(type(name) != OT_STRING)
  206. return -1;
  207. SQObjectPtr ret;
  208. if(_table(_metamethodsmap)->Get(name,ret)) {
  209. return _integer(ret);
  210. }
  211. return -1;
  212. }
  213. void SQSharedState::CallDelayedReleaseHooks(SQVM *vm, int count)
  214. {
  215. if(_delayed_release_hook.size()){
  216. if(count == 0) count = _delayed_release_hook.size();
  217. for(SQInteger i=0; i < count; ++i){
  218. SQDelayedReleseHook &dh = _delayed_release_hook[i];
  219. dh.hook(dh.ptr, dh.size, vm);
  220. }
  221. _delayed_release_hook.removeFromBegining(count);
  222. }
  223. }
  224. #ifndef NO_GARBAGE_COLLECTOR
  225. void SQSharedState::MarkObject(SQObjectPtr &o,SQCollectable **chain)
  226. {
  227. switch(type(o)){
  228. case OT_TABLE:_table(o)->Mark(chain);break;
  229. case OT_ARRAY:_array(o)->Mark(chain);break;
  230. case OT_USERDATA:_userdata(o)->Mark(chain);break;
  231. case OT_CLOSURE:_closure(o)->Mark(chain);break;
  232. case OT_NATIVECLOSURE:_nativeclosure(o)->Mark(chain);break;
  233. case OT_GENERATOR:_generator(o)->Mark(chain);break;
  234. case OT_THREAD:_thread(o)->Mark(chain);break;
  235. case OT_CLASS:_class(o)->Mark(chain);break;
  236. case OT_INSTANCE:_instance(o)->Mark(chain);break;
  237. case OT_OUTER:_outer(o)->Mark(chain);break;
  238. case OT_FUNCPROTO:_funcproto(o)->Mark(chain);break;
  239. default: break; //shutup compiler
  240. }
  241. }
  242. void SQSharedState::RunMark(SQVM *vm,SQCollectable **tchain)
  243. {
  244. SQVM *vms = _thread(_root_vm);
  245. vms->Mark(tchain);
  246. _refs_table.Mark(tchain);
  247. MarkObject(_registry,tchain);
  248. MarkObject(_consts,tchain);
  249. MarkObject(_metamethodsmap,tchain);
  250. MarkObject(_table_default_delegate,tchain);
  251. MarkObject(_array_default_delegate,tchain);
  252. MarkObject(_string_default_delegate,tchain);
  253. MarkObject(_number_default_delegate,tchain);
  254. MarkObject(_generator_default_delegate,tchain);
  255. MarkObject(_thread_default_delegate,tchain);
  256. MarkObject(_closure_default_delegate,tchain);
  257. MarkObject(_class_default_delegate,tchain);
  258. MarkObject(_instance_default_delegate,tchain);
  259. MarkObject(_weakref_default_delegate,tchain);
  260. }
  261. SQInteger SQSharedState::ResurrectUnreachable(SQVM *vm)
  262. {
  263. SQInteger n=0;
  264. SQCollectable *tchain=NULL;
  265. RunMark(vm,&tchain);
  266. SQCollectable *resurrected = _gc_chain;
  267. SQCollectable *t = resurrected;
  268. //SQCollectable *nx = NULL;
  269. _gc_chain = tchain;
  270. SQArray *ret = NULL;
  271. if(resurrected) {
  272. ret = SQArray::Create(this,0);
  273. SQCollectable *rlast = NULL;
  274. while(t) {
  275. rlast = t;
  276. SQObjectType type = t->GetType();
  277. if(type != OT_FUNCPROTO && type != OT_OUTER) {
  278. SQObject sqo;
  279. sqo._type = type;
  280. sqo._unVal.pRefCounted = t;
  281. ret->Append(sqo);
  282. }
  283. t = t->_next;
  284. n++;
  285. }
  286. assert(rlast->_next == NULL);
  287. rlast->_next = _gc_chain;
  288. if(_gc_chain)
  289. {
  290. _gc_chain->_prev = rlast;
  291. }
  292. _gc_chain = resurrected;
  293. }
  294. t = _gc_chain;
  295. while(t) {
  296. t->UnMark();
  297. t = t->_next;
  298. }
  299. if(ret) {
  300. SQObjectPtr temp = ret;
  301. vm->Push(temp);
  302. }
  303. else {
  304. vm->PushNull();
  305. }
  306. return n;
  307. }
  308. SQInteger SQSharedState::CollectGarbage(SQVM *vm)
  309. {
  310. SQInteger n = 0;
  311. SQCollectable *tchain = NULL;
  312. RunMark(vm,&tchain);
  313. SQCollectable *t = _gc_chain;
  314. SQCollectable *nx = NULL;
  315. if(t) {
  316. t->_uiRef++;
  317. while(t) {
  318. t->Finalize();
  319. nx = t->_next;
  320. if(nx) nx->_uiRef++;
  321. if(--t->_uiRef == 0)
  322. t->Release();
  323. t = nx;
  324. n++;
  325. }
  326. }
  327. t = tchain;
  328. while(t) {
  329. t->UnMark();
  330. t = t->_next;
  331. }
  332. _gc_chain = tchain;
  333. return n;
  334. }
  335. #endif
  336. #ifndef NO_GARBAGE_COLLECTOR
  337. void SQCollectable::AddToChain(SQCollectable **chain,SQCollectable *c)
  338. {
  339. c->_prev = NULL;
  340. c->_next = *chain;
  341. if(*chain) (*chain)->_prev = c;
  342. *chain = c;
  343. }
  344. void SQCollectable::RemoveFromChain(SQCollectable **chain,SQCollectable *c)
  345. {
  346. if(c->_prev) c->_prev->_next = c->_next;
  347. else *chain = c->_next;
  348. if(c->_next)
  349. c->_next->_prev = c->_prev;
  350. c->_next = NULL;
  351. c->_prev = NULL;
  352. }
  353. #endif
  354. SQChar* SQSharedState::GetScratchPad(SQInteger size)
  355. {
  356. SQInteger newsize;
  357. if(size>0) {
  358. if(_scratchpadsize < size) {
  359. newsize = size + (size>>1);
  360. _scratchpad = (SQChar *)SQ_REALLOC(_scratchpad,_scratchpadsize,newsize);
  361. _scratchpadsize = newsize;
  362. }else if(_scratchpadsize >= (size<<5)) {
  363. newsize = _scratchpadsize >> 1;
  364. _scratchpad = (SQChar *)SQ_REALLOC(_scratchpad,_scratchpadsize,newsize);
  365. _scratchpadsize = newsize;
  366. }
  367. }
  368. return _scratchpad;
  369. }
  370. RefTable::RefTable()
  371. {
  372. AllocNodes(4);
  373. }
  374. void RefTable::Finalize()
  375. {
  376. RefNode *nodes = _nodes;
  377. for(SQUnsignedInteger n = 0; n < _numofslots; n++) {
  378. nodes->obj.Null();
  379. nodes++;
  380. }
  381. }
  382. RefTable::~RefTable()
  383. {
  384. SQ_FREE(_buckets,(_numofslots * sizeof(RefNode *)) + (_numofslots * sizeof(RefNode)));
  385. }
  386. #ifndef NO_GARBAGE_COLLECTOR
  387. void RefTable::Mark(SQCollectable **chain)
  388. {
  389. RefNode *nodes = (RefNode *)_nodes;
  390. for(SQUnsignedInteger n = 0; n < _numofslots; n++) {
  391. if(type(nodes->obj) != OT_NULL) {
  392. SQSharedState::MarkObject(nodes->obj,chain);
  393. }
  394. nodes++;
  395. }
  396. }
  397. #endif
  398. void RefTable::AddRef(SQObject &obj)
  399. {
  400. SQHash mainpos;
  401. RefNode *prev;
  402. RefNode *ref = Get(obj,mainpos,&prev,true);
  403. ref->refs++;
  404. }
  405. SQUnsignedInteger RefTable::GetRefCount(SQObject &obj)
  406. {
  407. SQHash mainpos;
  408. RefNode *prev;
  409. RefNode *ref = Get(obj,mainpos,&prev,true);
  410. return ref->refs;
  411. }
  412. SQBool RefTable::Release(SQObject &obj)
  413. {
  414. SQHash mainpos;
  415. RefNode *prev;
  416. RefNode *ref = Get(obj,mainpos,&prev,false);
  417. if(ref) {
  418. if(--ref->refs == 0) {
  419. SQObjectPtr o = ref->obj;
  420. if(prev) {
  421. prev->next = ref->next;
  422. }
  423. else {
  424. _buckets[mainpos] = ref->next;
  425. }
  426. ref->next = _freelist;
  427. _freelist = ref;
  428. _slotused--;
  429. ref->obj.Null();
  430. //<<FIXME>>test for shrink?
  431. return SQTrue;
  432. }
  433. }
  434. else {
  435. assert(0);
  436. }
  437. return SQFalse;
  438. }
  439. void RefTable::Resize(SQUnsignedInteger size)
  440. {
  441. RefNode **oldbucks = _buckets;
  442. RefNode *t = _nodes;
  443. SQUnsignedInteger oldnumofslots = _numofslots;
  444. AllocNodes(size);
  445. //rehash
  446. SQUnsignedInteger nfound = 0;
  447. for(SQUnsignedInteger n = 0; n < oldnumofslots; n++) {
  448. if(type(t->obj) != OT_NULL) {
  449. //add back;
  450. assert(t->refs != 0);
  451. RefNode *nn = Add(::HashObj(t->obj)&(_numofslots-1),t->obj);
  452. nn->refs = t->refs;
  453. t->obj.Null();
  454. nfound++;
  455. }
  456. t++;
  457. }
  458. assert(nfound == oldnumofslots);
  459. SQ_FREE(oldbucks,(oldnumofslots * sizeof(RefNode *)) + (oldnumofslots * sizeof(RefNode)));
  460. }
  461. RefTable::RefNode *RefTable::Add(SQHash mainpos,SQObject &obj)
  462. {
  463. RefNode *t = _buckets[mainpos];
  464. RefNode *newnode = _freelist;
  465. newnode->obj = obj;
  466. _buckets[mainpos] = newnode;
  467. _freelist = _freelist->next;
  468. newnode->next = t;
  469. assert(newnode->refs == 0);
  470. _slotused++;
  471. return newnode;
  472. }
  473. RefTable::RefNode *RefTable::Get(SQObject &obj,SQHash &mainpos,RefNode **prev,bool add)
  474. {
  475. RefNode *ref;
  476. mainpos = ::HashObj(obj)&(_numofslots-1);
  477. *prev = NULL;
  478. for (ref = _buckets[mainpos]; ref; ) {
  479. if(_rawval(ref->obj) == _rawval(obj) && type(ref->obj) == type(obj))
  480. break;
  481. *prev = ref;
  482. ref = ref->next;
  483. }
  484. if(ref == NULL && add) {
  485. if(_numofslots == _slotused) {
  486. assert(_freelist == 0);
  487. Resize(_numofslots*2);
  488. mainpos = ::HashObj(obj)&(_numofslots-1);
  489. }
  490. ref = Add(mainpos,obj);
  491. }
  492. return ref;
  493. }
  494. void RefTable::AllocNodes(SQUnsignedInteger size)
  495. {
  496. RefNode **bucks;
  497. RefNode *nodes;
  498. bucks = (RefNode **)SQ_MALLOC((size * sizeof(RefNode *)) + (size * sizeof(RefNode)));
  499. nodes = (RefNode *)&bucks[size];
  500. RefNode *temp = nodes;
  501. SQUnsignedInteger n;
  502. for(n = 0; n < size - 1; n++) {
  503. bucks[n] = NULL;
  504. temp->refs = 0;
  505. new (&temp->obj) SQObjectPtr;
  506. temp->next = temp+1;
  507. temp++;
  508. }
  509. bucks[n] = NULL;
  510. temp->refs = 0;
  511. new (&temp->obj) SQObjectPtr;
  512. temp->next = NULL;
  513. _freelist = nodes;
  514. _nodes = nodes;
  515. _buckets = bucks;
  516. _slotused = 0;
  517. _numofslots = size;
  518. }
  519. //////////////////////////////////////////////////////////////////////////
  520. //SQStringTable
  521. /*
  522. * The following code is based on Lua 4.0 (Copyright 1994-2002 Tecgraf, PUC-Rio.)
  523. * http://www.lua.org/copyright.html#4
  524. * http://www.lua.org/source/4.0.1/src_lstring.c.html
  525. */
  526. SQStringTable::SQStringTable(SQSharedState *ss)
  527. {
  528. _sharedstate = ss;
  529. AllocNodes(4);
  530. _slotused = 0;
  531. }
  532. SQStringTable::~SQStringTable()
  533. {
  534. SQ_FREE(_strings,sizeof(SQString*)*_numofslots);
  535. _strings = NULL;
  536. }
  537. void SQStringTable::AllocNodes(SQInteger size)
  538. {
  539. _numofslots = size;
  540. _strings = (SQString**)SQ_MALLOC(sizeof(SQString*)*_numofslots);
  541. memset(_strings,0,sizeof(SQString*)*_numofslots);
  542. }
  543. SQString *SQStringTable::Contains(const SQChar *news, SQInteger &len, SQHash &newhash, SQHash &h)
  544. {
  545. if(len<0)
  546. len = (SQInteger)scstrlen(news);
  547. newhash = ::_hashstr(news,len);
  548. h = newhash&(_numofslots-1);
  549. SQString *s;
  550. SQInteger calculated_len = rsl(len);
  551. for (s = _strings[h]; s; s = s->_next){
  552. if(s->_len == len && (!memcmp(news,s->_val,calculated_len)))
  553. return s; //found
  554. }
  555. return 0;
  556. }
  557. SQString *SQStringTable::Add(const SQChar *news,SQInteger len)
  558. {
  559. SQHash newhash, h;
  560. SQString *t = Contains(news, len, newhash, h);
  561. if(t) return t;
  562. t = NewStrBuf(len);
  563. memcpy(t->_val,news,rsl(len));
  564. return Add(t, newhash, h);
  565. }
  566. SQString *SQStringTable::Add(SQString *strBuf)
  567. {
  568. SQHash newhash, h;
  569. SQString *t = Contains(strBuf->_val, strBuf->_len, newhash, h);
  570. if(t) {
  571. SQ_FREE(strBuf,sizeof(SQString) + rsl(strBuf->_len));
  572. return t;
  573. }
  574. return Add(strBuf, newhash, h);
  575. }
  576. SQString *SQStringTable::Add(SQString *strBuf, SQHash newhash, SQHash h)
  577. {
  578. strBuf->_sharedstate = _sharedstate;
  579. strBuf->_val[strBuf->_len] = _SC('\0');
  580. strBuf->_hash = newhash;
  581. strBuf->_next = _strings[h];
  582. _strings[h] = strBuf;
  583. _slotused++;
  584. if (_slotused > _numofslots) /* too crowded? */
  585. Resize(_numofslots*2);
  586. return strBuf;
  587. }
  588. SQString *SQStringTable::NewStrBuf(SQInteger len)
  589. {
  590. SQInteger calculated_size = rsl(len)+sizeof(SQString);
  591. SQString *t = (SQString *)SQ_MALLOC(calculated_size);
  592. memset(t, 0, calculated_size);
  593. new (t) SQString;
  594. t->_len = len;
  595. return t;
  596. }
  597. void SQStringTable::Resize(SQInteger size)
  598. {
  599. SQInteger oldsize=_numofslots;
  600. SQString **oldtable=_strings;
  601. AllocNodes(size);
  602. for (SQInteger i=0; i<oldsize; i++){
  603. SQString *p = oldtable[i];
  604. while(p){
  605. SQString *next = p->_next;
  606. SQHash h = p->_hash&(_numofslots-1);
  607. p->_next = _strings[h];
  608. _strings[h] = p;
  609. p = next;
  610. }
  611. }
  612. SQ_FREE(oldtable,oldsize*sizeof(SQString*));
  613. }
  614. void SQStringTable::Remove(SQString *bs)
  615. {
  616. SQString *s;
  617. SQString *prev=NULL;
  618. SQHash h = bs->_hash&(_numofslots - 1);
  619. for (s = _strings[h]; s; ){
  620. if(s == bs){
  621. if(prev)
  622. prev->_next = s->_next;
  623. else
  624. _strings[h] = s->_next;
  625. _slotused--;
  626. SQInteger slen = s->_len;
  627. s->~SQString();
  628. SQ_FREE(s,sizeof(SQString) + rsl(slen));
  629. return;
  630. }
  631. prev = s;
  632. s = s->_next;
  633. }
  634. assert(0);//if this fail something is wrong
  635. }