#include "pocketpy/iter.h" namespace pkpy{ void RangeIter::_register(VM* vm, PyObject* mod, PyObject* type){ vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false; vm->bind_notimplemented_constructor(type); vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; }); vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ RangeIter& self = PK_OBJ_GET(RangeIter, obj); if(self.r.step > 0){ if(self.current >= self.r.stop) return vm->StopIteration; }else{ if(self.current <= self.r.stop) return vm->StopIteration; } PyObject* ret = VAR(self.current); self.current += self.r.step; return ret; }); } void ArrayIter::_register(VM* vm, PyObject* mod, PyObject* type){ vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false; vm->bind_notimplemented_constructor(type); vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; }); vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ ArrayIter& self = _CAST(ArrayIter&, obj); if(self.current == self.end) return vm->StopIteration; return *self.current++; }); } void StringIter::_register(VM* vm, PyObject* mod, PyObject* type){ vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false; vm->bind_notimplemented_constructor(type); vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; }); vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ StringIter& self = _CAST(StringIter&, obj); if(self.index == self.str->size) return vm->StopIteration; int start = self.index; int len = utf8len(self.str->data[self.index]); self.index += len; return VAR(self.str->substr(start, len)); }); } PyObject* Generator::next(VM* vm){ if(state == 2) return vm->StopIteration; // reset frame._sp_base frame._sp_base = vm->s_data._sp; frame._locals.a = vm->s_data._sp; // restore the context for(PyObject* obj: s_backup) vm->s_data.push(obj); s_backup.clear(); vm->callstack.emplace(std::move(frame)); PyObject* ret; try{ ret = vm->_run_top_frame(); }catch(...){ state = 2; // end this generator immediately when an exception is thrown throw; } if(ret == PY_OP_YIELD){ // backup the context frame = std::move(vm->callstack.top()); ret = vm->s_data.popx(); for(PyObject* obj: frame.stack_view(&vm->s_data)) s_backup.push_back(obj); vm->_pop_frame(); state = 1; if(ret == vm->StopIteration) state = 2; return ret; }else{ state = 2; return vm->StopIteration; } } void Generator::_register(VM* vm, PyObject* mod, PyObject* type){ vm->_all_types[PK_OBJ_GET(Type, type)].subclass_enabled = false; vm->bind_notimplemented_constructor(type); vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; }); vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ Generator& self = _CAST(Generator&, obj); return self.next(vm); }); } PyObject* VM::_py_generator(Frame&& frame, ArgsView buffer){ return VAR_T(Generator, std::move(frame), buffer); } } // namespace pkpy