| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134 |
- #include "pocketpy/interpreter/iter.hpp"
- namespace pkpy {
- void RangeIter::_register(VM* vm, PyObject* mod, PyObject* type) {
- vm->bind__iter__(type->as<Type>(), [](VM* vm, PyVar _0) {
- return _0;
- });
- vm->bind__next__(type->as<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, PyObject* mod, PyObject* type) {
- vm->bind__iter__(type->as<Type>(), [](VM* vm, PyVar _0) {
- return _0;
- });
- vm->bind__next__(type->as<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, PyObject* mod, PyObject* type) {
- vm->bind__iter__(type->as<Type>(), [](VM* vm, PyVar _0) {
- return _0;
- });
- vm->bind__next__(type->as<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, PyObject* mod, PyObject* type) {
- vm->bind__iter__(type->as<Type>(), [](VM* vm, PyVar _0) {
- return _0;
- });
- vm->bind__next__(type->as<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 = c11__u8_header(s[self.i], false);
- self.i += len;
- vm->s_data.push(VAR(s.slice(start, self.i)));
- 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);
- 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, PyObject* mod, PyObject* type) {
- vm->bind__iter__(type->as<Type>(), [](VM* vm, PyVar _0) {
- return _0;
- });
- vm->bind__next__(type->as<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, PyObject* mod, PyObject* type) {
- vm->bind__iter__(type->as<Type>(), [](VM* vm, PyVar _0) {
- return _0;
- });
- vm->bind__next__(type->as<Type>(), [](VM* vm, PyVar _0) -> unsigned {
- DictItemsIter& self = _CAST(DictItemsIter&, _0);
- PyVar key, val;
- if (pkpy_DictIter__next(&self.it, (PyVar*)(&key), (PyVar*)(&val))) {
- vm->s_data.push(key);
- vm->s_data.push(val);
- return 2;
- }
- return 0;
- });
- }
- PyVar VM::__py_generator(LinkedFrame* frame, ArgsView buffer) {
- return vm->new_user_object<Generator>(std::move(frame), buffer);
- }
- } // namespace pkpy
|