pocketpy/src/iter.h
2023-04-09 22:19:13 +08:00

90 lines
2.0 KiB
C++

#pragma once
#include "ceval.h"
namespace pkpy{
class RangeIter : public BaseIter {
i64 current;
Range r; // copy by value, so we don't need to keep ref
public:
RangeIter(VM* vm, PyObject* ref) : BaseIter(vm) {
this->r = OBJ_GET(Range, ref);
this->current = r.start;
}
bool _has_next(){
return r.step > 0 ? current < r.stop : current > r.stop;
}
PyObject* next(){
if(!_has_next()) return nullptr;
current += r.step;
return VAR(current-r.step);
}
};
template <typename T>
class ArrayIter : public BaseIter {
PyObject* ref;
int index;
public:
ArrayIter(VM* vm, PyObject* ref) : BaseIter(vm), ref(ref), index(0) {}
PyObject* next() override{
const T* p = &OBJ_GET(T, ref);
if(index == p->size()) return nullptr;
return p->operator[](index++);
}
void _gc_mark() const override {
OBJ_MARK(ref);
}
};
class StringIter : public BaseIter {
PyObject* ref;
int index;
public:
StringIter(VM* vm, PyObject* ref) : BaseIter(vm), ref(ref), index(0) {}
PyObject* next() override{
// TODO: optimize this to use iterator
// operator[] is O(n) complexity
Str* str = &OBJ_GET(Str, ref);
if(index == str->u8_length()) return nullptr;
return VAR(str->u8_getitem(index++));
}
void _gc_mark() const override {
OBJ_MARK(ref);
}
};
inline PyObject* Generator::next(){
if(state == 2) return nullptr;
vm->_push_new_frame(std::move(frame));
PyObject* ret = vm->_run_top_frame();
if(ret == vm->_py_op_yield){
frame = std::move(vm->callstack.top());
vm->callstack.pop();
state = 1;
return frame.popx();
}else{
state = 2;
return nullptr;
}
}
inline void Generator::_gc_mark() const{
frame._gc_mark();
}
template<typename T>
void _gc_mark(T& t) {
if constexpr(std::is_base_of_v<BaseIter, T>){
t._gc_mark();
}
}
} // namespace pkpy