iter.cpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. #include "pocketpy/iter.h"
  2. namespace pkpy{
  3. void RangeIter::_register(VM* vm, PyObject* mod, PyObject* type){
  4. vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false;
  5. vm->bind_notimplemented_constructor<RangeIter>(type);
  6. vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
  7. vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
  8. RangeIter& self = _CAST(RangeIter&, obj);
  9. bool has_next = self.r.step > 0 ? self.current < self.r.stop : self.current > self.r.stop;
  10. if(!has_next) return vm->StopIteration;
  11. self.current += self.r.step;
  12. return VAR(self.current - self.r.step);
  13. });
  14. }
  15. void ArrayIter::_register(VM* vm, PyObject* mod, PyObject* type){
  16. vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false;
  17. vm->bind_notimplemented_constructor<ArrayIter>(type);
  18. vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
  19. vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
  20. ArrayIter& self = _CAST(ArrayIter&, obj);
  21. if(self.current == self.end) return vm->StopIteration;
  22. return *self.current++;
  23. });
  24. }
  25. void StringIter::_register(VM* vm, PyObject* mod, PyObject* type){
  26. vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false;
  27. vm->bind_notimplemented_constructor<StringIter>(type);
  28. vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
  29. vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
  30. StringIter& self = _CAST(StringIter&, obj);
  31. if(self.index == self.str->size) return vm->StopIteration;
  32. int start = self.index;
  33. int len = utf8len(self.str->data[self.index]);
  34. self.index += len;
  35. return VAR(self.str->substr(start, len));
  36. });
  37. }
  38. PyObject* Generator::next(VM* vm){
  39. if(state == 2) return vm->StopIteration;
  40. // reset frame._sp_base
  41. frame._sp_base = vm->s_data._sp;
  42. frame._locals.a = vm->s_data._sp;
  43. // restore the context
  44. for(PyObject* obj: s_backup) vm->s_data.push(obj);
  45. s_backup.clear();
  46. vm->callstack.emplace(std::move(frame));
  47. PyObject* ret;
  48. try{
  49. ret = vm->_run_top_frame();
  50. }catch(...){
  51. state = 2; // end this generator immediately when an exception is thrown
  52. throw;
  53. }
  54. if(ret == PY_OP_YIELD){
  55. // backup the context
  56. frame = std::move(vm->callstack.top());
  57. ret = vm->s_data.popx();
  58. for(PyObject* obj: frame.stack_view(&vm->s_data)) s_backup.push_back(obj);
  59. vm->_pop_frame();
  60. state = 1;
  61. if(ret == vm->StopIteration) state = 2;
  62. return ret;
  63. }else{
  64. state = 2;
  65. return vm->StopIteration;
  66. }
  67. }
  68. void Generator::_register(VM* vm, PyObject* mod, PyObject* type){
  69. vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false;
  70. vm->bind_notimplemented_constructor<Generator>(type);
  71. vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
  72. vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
  73. Generator& self = _CAST(Generator&, obj);
  74. return self.next(vm);
  75. });
  76. }
  77. PyObject* VM::_py_generator(Frame&& frame, ArgsView buffer){
  78. return VAR_T(Generator, std::move(frame), buffer);
  79. }
  80. } // namespace pkpy