| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394 |
- #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<RangeIter>(type);
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* _0){
- RangeIter& self = PK_OBJ_GET(RangeIter, _0);
- 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<ArrayIter>(type);
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* _0){
- ArrayIter& self = _CAST(ArrayIter&, _0);
- 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<StringIter>(type);
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyObject* _0){
- StringIter& self = _CAST(StringIter&, _0);
- 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<Generator>(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
|