| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 |
- #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.current >= self.r.stop) return 0;
- vm->s_data.emplace(VM::tp_int, self.current);
- self.current += self.r.step;
- return 1;
- });
- }
- void RangeIterR::_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{
- RangeIterR& self = PK_OBJ_GET(RangeIterR, _0);
- if(self.current <= self.r.stop) return 0;
- vm->s_data.emplace(VM::tp_int, 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
- lf->frame._sp_base = vm->s_data._sp;
- lf->frame._locals.a = vm->s_data._sp;
- // restore the context
- for(PyVar obj: s_backup) vm->s_data.push(obj);
- // relocate stack objects (their addresses become invalid)
- for(PyVar* p=lf->frame.actual_sp_base(); p!=vm->s_data.end(); p++){
- if(p->type == VM::tp_stack_memory){
- // TODO: refactor this
- int count = p->as<StackMemory>().count;
- if(count < 0){
- void* new_p = p + count;
- p[1]._1 = reinterpret_cast<i64>(new_p);
- }
- }
- }
- s_backup.clear();
- vm->callstack.pushx(lf);
- lf = nullptr;
- 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
- lf = vm->callstack.popx();
- ret = vm->s_data.popx();
- for(PyVar obj: lf->frame.stack_view(&vm->s_data)) s_backup.push_back(obj);
- vm->s_data.reset(lf->frame._sp_base);
- // TODO: should we add this snippet here?
- // #if PK_ENABLE_PROFILER
- // if(!_next_breakpoint.empty() && callstack.size()<_next_breakpoint.callstack_size){
- // _next_breakpoint = NextBreakpoint();
- // }
- // #endif
- 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(LinkedFrame* frame, ArgsView buffer){
- return vm->new_user_object<Generator>(std::move(frame), buffer);
- }
- } // namespace pkpy
|