sqstate.cpp 18 KB

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