blueloveTH 53d272ce2f ...
2023-07-08 19:43:17 +08:00

171 lines
5.7 KiB
C++

#pragma once
#include "common.h"
#include "vm.h"
namespace pkpy {
#define PY_CLASS(T, mod, name) \
static Type _type(VM* vm) { \
static const StrName __x0(#mod); \
static const StrName __x1(#name); \
return PK_OBJ_GET(Type, vm->_modules[__x0]->attr(__x1)); \
} \
static void _check_type(VM* vm, PyObject* val){ \
if(!vm->isinstance(val, T::_type(vm))){ \
vm->TypeError("expected '" #mod "." #name "', got " + OBJ_NAME(vm->_t(val)).escape()); \
} \
} \
static PyObject* register_class(VM* vm, PyObject* mod) { \
if(OBJ_NAME(mod) != #mod) { \
auto msg = fmt("register_class() failed: ", OBJ_NAME(mod), " != ", #mod); \
throw std::runtime_error(msg); \
} \
PyObject* type = vm->new_type_object(mod, #name, vm->tp_object); \
T::_register(vm, mod, type); \
type->attr()._try_perfect_rehash(); \
return type; \
}
#define VAR_T(T, ...) vm->heap.gcnew<T>(T::_type(vm), T(__VA_ARGS__))
int c99_sizeof(VM*, const Str&);
inline PyObject* py_var(VM* vm, void* p);
inline PyObject* py_var(VM* vm, char* p);
struct VoidP{
PY_CLASS(VoidP, c, void_p)
void* ptr;
int base_offset;
VoidP(void* ptr): ptr(ptr), base_offset(1){}
VoidP(): ptr(nullptr), base_offset(1){}
bool operator==(const VoidP& other) const {
return ptr == other.ptr && base_offset == other.base_offset;
}
bool operator!=(const VoidP& other) const {
return ptr != other.ptr || base_offset != other.base_offset;
}
bool operator<(const VoidP& other) const { return ptr < other.ptr; }
bool operator<=(const VoidP& other) const { return ptr <= other.ptr; }
bool operator>(const VoidP& other) const { return ptr > other.ptr; }
bool operator>=(const VoidP& other) const { return ptr >= other.ptr; }
Str hex() const{
std::stringstream ss;
ss << std::hex << reinterpret_cast<intptr_t>(ptr);
return "0x" + ss.str();
}
static void _register(VM* vm, PyObject* mod, PyObject* type);
};
struct C99Struct{
PY_CLASS(C99Struct, c, struct)
static constexpr int INLINE_SIZE = 24;
char _inlined[INLINE_SIZE];
char* p;
int size;
C99Struct(int new_size){
this->size = new_size;
if(size <= INLINE_SIZE){
p = _inlined;
}else{
p = (char*)malloc(size);
}
}
template<typename T>
C99Struct(std::monostate _, const T& data): C99Struct(sizeof(T)){
static_assert(std::is_pod_v<T>);
static_assert(!std::is_pointer_v<T>);
memcpy(p, &data, this->size);
}
C99Struct(void* p, int size): C99Struct(size){
if(p != nullptr) memcpy(this->p, p, size);
}
C99Struct(const C99Struct& other): C99Struct(other.p, other.size){}
~C99Struct(){ if(p!=_inlined) free(p); }
static void _register(VM* vm, PyObject* mod, PyObject* type);
};
struct ReflField{
std::string_view name;
int offset;
bool operator<(const ReflField& other) const{ return name < other.name; }
bool operator==(const ReflField& other) const{ return name == other.name; }
bool operator!=(const ReflField& other) const{ return name != other.name; }
bool operator<(std::string_view other) const{ return name < other; }
bool operator==(std::string_view other) const{ return name == other; }
bool operator!=(std::string_view other) const{ return name != other; }
};
struct ReflType{
std::string_view name;
size_t size;
std::vector<ReflField> fields;
};
inline static std::map<std::string_view, ReflType> _refl_types;
inline void add_refl_type(std::string_view name, size_t size, std::vector<ReflField> fields){
ReflType type{name, size, std::move(fields)};
std::sort(type.fields.begin(), type.fields.end());
_refl_types[name] = std::move(type);
}
struct C99ReflType final: ReflType{
PY_CLASS(C99ReflType, c, _refl)
C99ReflType(const ReflType& type){
this->name = type.name;
this->size = type.size;
this->fields = type.fields;
}
static void _register(VM* vm, PyObject* mod, PyObject* type);
};
static_assert(sizeof(Py_<C99Struct>) <= 64);
static_assert(sizeof(Py_<Tuple>) <= 64);
inline PyObject* py_var(VM* vm, void* p){
return VAR_T(VoidP, p);
}
inline PyObject* py_var(VM* vm, char* p){
return VAR_T(VoidP, p);
}
/***********************************************/
template<typename T>
T to_void_p(VM* vm, PyObject* var){
static_assert(std::is_pointer_v<T>);
if(var == vm->None) return nullptr; // None can be casted to any pointer implicitly
VoidP& p = CAST(VoidP&, var);
return reinterpret_cast<T>(p.ptr);
}
template<typename T>
T to_c99_struct(VM* vm, PyObject* var){
static_assert(std::is_pod_v<T>);
C99Struct& pod = CAST(C99Struct&, var);
return *reinterpret_cast<T*>(pod.p);
}
template<typename T>
std::enable_if_t<std::is_pod_v<T> && !std::is_pointer_v<T>, PyObject*> py_var(VM* vm, const T& data){
return VAR_T(C99Struct, std::monostate(), data);
}
/*****************************************************************/
void add_module_c(VM* vm);
} // namespace pkpy