sqclosure.h 6.0 KB

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