sqclosure.h 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. /* see copyright notice in squirrel.h */
  2. #ifndef _SQCLOSURE_H_
  3. #define _SQCLOSURE_H_
  4. #define _CALC_CLOSURE_SIZE(func) (sizeof(SQClosure) + (func->_noutervalues*sizeof(SQObjectPtr)) + (func->_ndefaultparams*sizeof(SQObjectPtr)))
  5. struct SQFunctionProto;
  6. struct SQClass;
  7. struct SQClosure : public CHAINABLE_OBJ
  8. {
  9. private:
  10. SQClosure(SQSharedState *ss,SQFunctionProto *func, SQWeakRef *root):
  11. _env(NULL),_root(root),_base(NULL),_function(func)
  12. { __ObjAddRef(_function); __ObjAddRef(_root); INIT_CHAIN();ADD_TO_CHAIN(&_ss(this)->_gc_chain,this); }
  13. public:
  14. static SQClosure *Create(SQSharedState *ss,SQFunctionProto *func,SQWeakRef *root){
  15. SQInteger size = _CALC_CLOSURE_SIZE(func);
  16. SQClosure *nc=(SQClosure*)SQ_MALLOC(size);
  17. new (nc) SQClosure(ss,func,root);
  18. nc->_outervalues = (SQObjectPtr *)(nc + 1);
  19. nc->_defaultparams = &nc->_outervalues[func->_noutervalues];
  20. _CONSTRUCT_VECTOR(SQObjectPtr,func->_noutervalues,nc->_outervalues);
  21. _CONSTRUCT_VECTOR(SQObjectPtr,func->_ndefaultparams,nc->_defaultparams);
  22. return nc;
  23. }
  24. void Release(){
  25. SQFunctionProto *f = _function;
  26. SQInteger size = _CALC_CLOSURE_SIZE(f);
  27. _DESTRUCT_VECTOR(SQObjectPtr,f->_noutervalues,_outervalues);
  28. _DESTRUCT_VECTOR(SQObjectPtr,f->_ndefaultparams,_defaultparams);
  29. __ObjRelease(_function);
  30. this->~SQClosure();
  31. sq_vm_free(this,size);
  32. }
  33. void SetRoot(SQWeakRef *r)
  34. {
  35. __ObjRelease(_root);
  36. _root = r;
  37. __ObjAddRef(_root);
  38. }
  39. SQClosure *Clone()
  40. {
  41. SQFunctionProto *f = _function;
  42. SQClosure * ret = SQClosure::Create(_opt_ss(this),f,_root);
  43. ret->_env = _env;
  44. if(ret->_env) __ObjAddRef(ret->_env);
  45. _COPY_VECTOR(ret->_outervalues,_outervalues,f->_noutervalues);
  46. _COPY_VECTOR(ret->_defaultparams,_defaultparams,f->_ndefaultparams);
  47. return ret;
  48. }
  49. ~SQClosure();
  50. bool SaveAsSource(SQVM *v,SQUserPointer up,SQWRITEFUNC write);
  51. bool Save(SQVM *v,SQUserPointer up,SQWRITEFUNC write);
  52. static bool Load(SQVM *v,SQUserPointer up,SQREADFUNC read,SQObjectPtr &ret);
  53. #ifndef NO_GARBAGE_COLLECTOR
  54. void Mark(SQCollectable **chain);
  55. void Finalize(){
  56. SQFunctionProto *f = _function;
  57. _NULL_SQOBJECT_VECTOR(_outervalues,f->_noutervalues);
  58. _NULL_SQOBJECT_VECTOR(_defaultparams,f->_ndefaultparams);
  59. }
  60. SQObjectType GetType() {return OT_CLOSURE;}
  61. #endif
  62. SQWeakRef *_env;
  63. SQWeakRef *_root;
  64. SQClass *_base;
  65. SQFunctionProto *_function;
  66. SQObjectPtr *_outervalues;
  67. SQObjectPtr *_defaultparams;
  68. };
  69. //////////////////////////////////////////////
  70. struct SQOuter : public CHAINABLE_OBJ
  71. {
  72. private:
  73. SQOuter(SQSharedState *ss, SQObjectPtr *outer){_valptr = outer; _next = NULL; INIT_CHAIN(); ADD_TO_CHAIN(&_ss(this)->_gc_chain,this); }
  74. public:
  75. static SQOuter *Create(SQSharedState *ss, SQObjectPtr *outer)
  76. {
  77. SQOuter *nc = (SQOuter*)SQ_MALLOC(sizeof(SQOuter));
  78. new (nc) SQOuter(ss, outer);
  79. return nc;
  80. }
  81. ~SQOuter() { REMOVE_FROM_CHAIN(&_ss(this)->_gc_chain,this); }
  82. void Release()
  83. {
  84. this->~SQOuter();
  85. sq_vm_free(this,sizeof(SQOuter));
  86. }
  87. #ifndef NO_GARBAGE_COLLECTOR
  88. void Mark(SQCollectable **chain);
  89. void Finalize() { _value.Null(); }
  90. SQObjectType GetType() {return OT_OUTER;}
  91. #endif
  92. SQObjectPtr *_valptr; /* pointer to value on stack, or _value below */
  93. SQInteger _idx; /* idx in stack array, for relocation */
  94. SQObjectPtr _value; /* value of outer after stack frame is closed */
  95. SQOuter *_next_outer; /* pointer to next outer when frame is open */
  96. };
  97. //////////////////////////////////////////////
  98. struct SQGenerator : public CHAINABLE_OBJ
  99. {
  100. enum SQGeneratorState{eRunning,eSuspended,eDead};
  101. private:
  102. SQGenerator(SQSharedState *ss,SQClosure *closure){_closure=closure;_state=eRunning;_ci._generator=NULL;INIT_CHAIN();ADD_TO_CHAIN(&_ss(this)->_gc_chain,this);}
  103. public:
  104. static SQGenerator *Create(SQSharedState *ss,SQClosure *closure){
  105. SQGenerator *nc=(SQGenerator*)SQ_MALLOC(sizeof(SQGenerator));
  106. new (nc) SQGenerator(ss,closure);
  107. return nc;
  108. }
  109. ~SQGenerator()
  110. {
  111. REMOVE_FROM_CHAIN(&_ss(this)->_gc_chain,this);
  112. }
  113. void Kill(){
  114. _state=eDead;
  115. _stack.resize(0);
  116. _closure.Null();}
  117. void Release(){
  118. sq_delete(this,SQGenerator);
  119. }
  120. bool Yield(SQVM *v,SQInteger target);
  121. bool Resume(SQVM *v,SQObjectPtr &dest);
  122. #ifndef NO_GARBAGE_COLLECTOR
  123. void Mark(SQCollectable **chain);
  124. void Finalize(){_stack.resize(0);_closure.Null();}
  125. SQObjectType GetType() {return OT_GENERATOR;}
  126. #endif
  127. SQObjectPtr _closure;
  128. SQObjectPtrVec _stack;
  129. SQVM::CallInfo _ci;
  130. ExceptionsTraps _etraps;
  131. SQGeneratorState _state;
  132. };
  133. #define _CALC_NATVIVECLOSURE_SIZE(noutervalues) (sizeof(SQNativeClosure) + (noutervalues*sizeof(SQObjectPtr)))
  134. struct SQNativeClosure : public CHAINABLE_OBJ
  135. {
  136. private:
  137. SQNativeClosure(SQSharedState *ss,SQFUNCTION func){_function=func;INIT_CHAIN();ADD_TO_CHAIN(&_ss(this)->_gc_chain,this); _env = NULL;}
  138. public:
  139. static SQNativeClosure *Create(SQSharedState *ss,SQFUNCTION func,SQInteger nouters)
  140. {
  141. SQInteger size = _CALC_NATVIVECLOSURE_SIZE(nouters);
  142. SQNativeClosure *nc=(SQNativeClosure*)SQ_MALLOC(size);
  143. new (nc) SQNativeClosure(ss,func);
  144. nc->_outervalues = (SQObjectPtr *)(nc + 1);
  145. nc->_noutervalues = nouters;
  146. _CONSTRUCT_VECTOR(SQObjectPtr,nc->_noutervalues,nc->_outervalues);
  147. return nc;
  148. }
  149. SQNativeClosure *Clone()
  150. {
  151. SQNativeClosure * ret = SQNativeClosure::Create(_opt_ss(this),_function,_noutervalues);
  152. ret->_env = _env;
  153. if(ret->_env) __ObjAddRef(ret->_env);
  154. ret->_name = _name;
  155. _COPY_VECTOR(ret->_outervalues,_outervalues,_noutervalues);
  156. ret->_typecheck.copy(_typecheck);
  157. ret->_nparamscheck = _nparamscheck;
  158. return ret;
  159. }
  160. ~SQNativeClosure()
  161. {
  162. __ObjRelease(_env);
  163. REMOVE_FROM_CHAIN(&_ss(this)->_gc_chain,this);
  164. }
  165. void Release(){
  166. SQInteger size = _CALC_NATVIVECLOSURE_SIZE(_noutervalues);
  167. _DESTRUCT_VECTOR(SQObjectPtr,_noutervalues,_outervalues);
  168. this->~SQNativeClosure();
  169. sq_free(this,size);
  170. }
  171. #ifndef NO_GARBAGE_COLLECTOR
  172. void Mark(SQCollectable **chain);
  173. void Finalize() { _NULL_SQOBJECT_VECTOR(_outervalues,_noutervalues); }
  174. SQObjectType GetType() {return OT_NATIVECLOSURE;}
  175. #endif
  176. SQInteger _nparamscheck;
  177. SQIntVec _typecheck;
  178. SQObjectPtr *_outervalues;
  179. SQUnsignedInteger _noutervalues;
  180. SQWeakRef *_env;
  181. SQFUNCTION _function;
  182. SQObjectPtr _name;
  183. };
  184. #endif //_SQCLOSURE_H_