| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131 |
- #pragma once
- #include "cffi.h"
- #include "common.h"
- #include "frame.h"
- namespace pkpy{
- struct RangeIter{
- PY_CLASS(RangeIter, builtins, "_range_iterator")
- Range r;
- i64 current;
- RangeIter(Range r) : r(r), current(r.start) {}
- static void _register(VM* vm, PyObject* mod, PyObject* type){
- vm->_all_types[OBJ_GET(Type, type)].subclass_enabled = false;
- vm->bind_notimplemented_constructor<RangeIter>(type);
- vm->bind__iter__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
- vm->bind__next__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
- RangeIter& self = _CAST(RangeIter&, obj);
- bool has_next = self.r.step > 0 ? self.current < self.r.stop : self.current > self.r.stop;
- if(!has_next) return vm->StopIteration;
- self.current += self.r.step;
- return VAR(self.current - self.r.step);
- });
- }
- };
- struct ArrayIter{
- PY_CLASS(ArrayIter, builtins, "_array_iterator")
- PyObject* ref;
- PyObject** begin;
- PyObject** end;
- PyObject** current;
- ArrayIter(PyObject* ref, PyObject** begin, PyObject** end)
- : ref(ref), begin(begin), end(end), current(begin) {}
- void _gc_mark() const{ OBJ_MARK(ref); }
- static void _register(VM* vm, PyObject* mod, PyObject* type){
- vm->_all_types[OBJ_GET(Type, type)].subclass_enabled = false;
- vm->bind_notimplemented_constructor<ArrayIter>(type);
- vm->bind__iter__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
- vm->bind__next__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
- ArrayIter& self = _CAST(ArrayIter&, obj);
- if(self.current == self.end) return vm->StopIteration;
- return *self.current++;
- });
- }
- };
- struct StringIter{
- PY_CLASS(StringIter, builtins, "_string_iterator")
- PyObject* ref;
- Str* str;
- int index;
- StringIter(PyObject* ref) : ref(ref), str(&OBJ_GET(Str, ref)), index(0) {}
- void _gc_mark() const{ OBJ_MARK(ref); }
- static void _register(VM* vm, PyObject* mod, PyObject* type){
- vm->_all_types[OBJ_GET(Type, type)].subclass_enabled = false;
- vm->bind_notimplemented_constructor<StringIter>(type);
- vm->bind__iter__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
- vm->bind__next__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
- StringIter& self = _CAST(StringIter&, obj);
- // TODO: optimize this... operator[] is of O(n) complexity
- if(self.index == self.str->u8_length()) return vm->StopIteration;
- return VAR(self.str->u8_getitem(self.index++));
- });
- }
- };
- struct Generator{
- PY_CLASS(Generator, builtins, "_generator")
- Frame frame;
- int state; // 0,1,2
- List s_backup;
- Generator(Frame&& frame, ArgsView buffer): frame(std::move(frame)), state(0) {
- for(PyObject* obj: buffer) s_backup.push_back(obj);
- }
- void _gc_mark() const{
- frame._gc_mark();
- for(PyObject* obj: s_backup) OBJ_MARK(obj);
- }
- PyObject* next(VM* vm){
- if(state == 2) return vm->StopIteration;
- // reset frame._sp_base
- frame._sp_base = frame._s->_sp;
- frame._locals.a = frame._s->_sp;
- // restore the context
- for(PyObject* obj: s_backup) frame._s->push(obj);
- s_backup.clear();
- vm->callstack.push(std::move(frame));
- PyObject* ret = vm->_run_top_frame();
- if(ret == PY_OP_YIELD){
- // backup the context
- frame = std::move(vm->callstack.top());
- PyObject* ret = frame._s->popx();
- for(PyObject* obj: frame.stack_view()) s_backup.push_back(obj);
- vm->_pop_frame();
- state = 1;
- if(ret == vm->StopIteration) state = 2;
- return ret;
- }else{
- state = 2;
- return vm->StopIteration;
- }
- }
- static void _register(VM* vm, PyObject* mod, PyObject* type){
- vm->_all_types[OBJ_GET(Type, type)].subclass_enabled = false;
- vm->bind_notimplemented_constructor<Generator>(type);
- vm->bind__iter__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){ return obj; });
- vm->bind__next__(OBJ_GET(Type, type), [](VM* vm, PyObject* obj){
- Generator& self = _CAST(Generator&, obj);
- return self.next(vm);
- });
- }
- };
- inline PyObject* VM::_py_generator(Frame&& frame, ArgsView buffer){
- return VAR_T(Generator, std::move(frame), buffer);
- }
- } // namespace pkpy
|