| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105 |
- #include "pocketpy/iter.h"
- namespace pkpy{
- void RangeIter::_register(VM* vm, PyVar mod, PyVar type){
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0) -> unsigned{
- RangeIter& self = PK_OBJ_GET(RangeIter, _0);
- if(self.r.step > 0){
- if(self.current >= self.r.stop) return 0;
- }else{
- if(self.current <= self.r.stop) return 0;
- }
- vm->s_data.push(VAR(self.current));
- self.current += self.r.step;
- return 1;
- });
- }
- void ArrayIter::_register(VM* vm, PyVar mod, PyVar type){
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0) -> unsigned{
- ArrayIter& self = _CAST(ArrayIter&, _0);
- if(self.current == self.end) return 0;
- vm->s_data.push(*self.current++);
- return 1;
- });
- }
- void StringIter::_register(VM* vm, PyVar mod, PyVar type){
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0) -> unsigned{
- StringIter& self = _CAST(StringIter&, _0);
- Str& s = PK_OBJ_GET(Str, self.ref);
- if(self.i == s.size) return 0;
- int start = self.i;
- int len = utf8len(s.data[self.i]);
- self.i += len;
- vm->s_data.push(VAR(s.substr(start, len)));
- return 1;
- });
- }
- PyVar 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(PyVar obj: s_backup) vm->s_data.push(obj);
- s_backup.clear();
- vm->callstack.emplace(std::move(frame));
- PyVar 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(PyVar 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, PyVar mod, PyVar type){
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0) -> unsigned{
- Generator& self = _CAST(Generator&, _0);
- PyVar retval = self.next(vm);
- if(retval == vm->StopIteration) return 0;
- vm->s_data.push(retval);
- return 1;
- });
- }
- void DictItemsIter::_register(VM *vm, PyVar mod, PyVar type){
- vm->bind__iter__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0){ return _0; });
- vm->bind__next__(PK_OBJ_GET(Type, type), [](VM* vm, PyVar _0) -> unsigned{
- DictItemsIter& self = _CAST(DictItemsIter&, _0);
- Dict& d = PK_OBJ_GET(Dict, self.ref);
- if(self.i == -1) return 0;
- vm->s_data.push(d._items[self.i].first);
- vm->s_data.push(d._items[self.i].second);
- self.i = d._nodes[self.i].next;
- return 2;
- });
- }
- PyVar VM::__py_generator(Frame&& frame, ArgsView buffer){
- return vm->new_user_object<Generator>(std::move(frame), buffer);
- }
- } // namespace pkpy
|