mirror of
https://github.com/pocketpy/pocketpy
synced 2025-10-23 21:10:19 +00:00
Compare commits
No commits in common. "4b4351e3fa2ca1f9b521fcba8b40f4b7c69bfee4" and "381e2d5b9a2ac85470e93fc17a68ca5daed0993f" have entirely different histories.
4b4351e3fa
...
381e2d5b9a
@ -90,14 +90,11 @@ pk_FrameResult pk_VM__vectorcall(pk_VM* self, uint16_t argc, uint16_t kwargc, bo
|
|||||||
const char* pk_opname(Opcode op);
|
const char* pk_opname(Opcode op);
|
||||||
|
|
||||||
// type registration
|
// type registration
|
||||||
void pk_object__register();
|
|
||||||
void pk_number__register();
|
void pk_number__register();
|
||||||
py_Type pk_str__register();
|
py_Type pk_str__register();
|
||||||
py_Type pk_bytes__register();
|
py_Type pk_bytes__register();
|
||||||
py_Type pk_list__register();
|
py_Type pk_list__register();
|
||||||
|
|
||||||
py_TValue pk_builtins__register();
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
@ -208,14 +208,12 @@ bool py_binaryop(const py_Ref lhs, const py_Ref rhs, py_Name op, py_Name rop);
|
|||||||
void py_assign(py_Ref dst, const py_Ref src);
|
void py_assign(py_Ref dst, const py_Ref src);
|
||||||
|
|
||||||
/************* Stack Operations *************/
|
/************* Stack Operations *************/
|
||||||
/// Return a reference to the i-th object from the top of the stack.
|
/// Returns a reference to the i-th object from the top of the stack.
|
||||||
/// i should be negative, e.g. (-1) means TOS.
|
/// i should be negative, e.g. (-1) means TOS.
|
||||||
py_StackRef py_peek(int i);
|
py_StackRef py_peek(int i);
|
||||||
/// Push the object to the stack.
|
/// Pushes the object to the stack.
|
||||||
void py_push(const py_Ref src);
|
void py_push(const py_Ref src);
|
||||||
/// Push a nil object to the stack.
|
/// Pops an object from the stack.
|
||||||
void py_pushnil();
|
|
||||||
/// Pop an object from the stack.
|
|
||||||
void py_pop();
|
void py_pop();
|
||||||
/// Shrink the stack by n.
|
/// Shrink the stack by n.
|
||||||
void py_shrink(int n);
|
void py_shrink(int n);
|
||||||
@ -293,8 +291,8 @@ bool py_callmethod(py_Ref self, py_Name, int argc, py_Ref argv);
|
|||||||
/// The stack remains unchanged after the operation.
|
/// The stack remains unchanged after the operation.
|
||||||
bool py_callmagic(py_Name name, int argc, py_Ref argv);
|
bool py_callmagic(py_Name name, int argc, py_Ref argv);
|
||||||
|
|
||||||
bool py_str(py_Ref val);
|
#define py_repr(self) py_callmagic(__repr__, 1, self)
|
||||||
bool py_repr(py_Ref val);
|
#define py_str(self) py_callmagic(__str__, 1, self)
|
||||||
|
|
||||||
/// The return value of the most recent call.
|
/// The return value of the most recent call.
|
||||||
py_GlobalRef py_retval();
|
py_GlobalRef py_retval();
|
||||||
|
@ -9,7 +9,6 @@ MAGIC_METHOD(__iter__)
|
|||||||
MAGIC_METHOD(__next__)
|
MAGIC_METHOD(__next__)
|
||||||
MAGIC_METHOD(__neg__)
|
MAGIC_METHOD(__neg__)
|
||||||
MAGIC_METHOD(__invert__)
|
MAGIC_METHOD(__invert__)
|
||||||
MAGIC_METHOD(__bool__)
|
|
||||||
// logical operators
|
// logical operators
|
||||||
MAGIC_METHOD(__contains__)
|
MAGIC_METHOD(__contains__)
|
||||||
/////////////////////////////
|
/////////////////////////////
|
||||||
|
@ -272,29 +272,24 @@ ImagExpr* ImagExpr__new(int line, double value) {
|
|||||||
typedef struct LiteralExpr {
|
typedef struct LiteralExpr {
|
||||||
EXPR_COMMON_HEADER
|
EXPR_COMMON_HEADER
|
||||||
const TokenValue* value;
|
const TokenValue* value;
|
||||||
bool negated;
|
|
||||||
} LiteralExpr;
|
} LiteralExpr;
|
||||||
|
|
||||||
void LiteralExpr__emit_(Expr* self_, Ctx* ctx) {
|
void LiteralExpr__emit_(Expr* self_, Ctx* ctx) {
|
||||||
LiteralExpr* self = (LiteralExpr*)self_;
|
LiteralExpr* self = (LiteralExpr*)self_;
|
||||||
switch(self->value->index) {
|
switch(self->value->index) {
|
||||||
case TokenValue_I64: {
|
case TokenValue_I64: {
|
||||||
py_i64 val = self->value->_i64;
|
int64_t val = self->value->_i64;
|
||||||
if(self->negated) val = -val;
|
|
||||||
Ctx__emit_int(ctx, val, self->line);
|
Ctx__emit_int(ctx, val, self->line);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case TokenValue_F64: {
|
case TokenValue_F64: {
|
||||||
py_TValue value;
|
py_TValue value;
|
||||||
py_f64 val = self->value->_f64;
|
py_newfloat(&value, self->value->_f64);
|
||||||
if(self->negated) val = -val;
|
|
||||||
py_newfloat(&value, val);
|
|
||||||
int index = Ctx__add_const(ctx, &value);
|
int index = Ctx__add_const(ctx, &value);
|
||||||
Ctx__emit_(ctx, OP_LOAD_CONST, index, self->line);
|
Ctx__emit_(ctx, OP_LOAD_CONST, index, self->line);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case TokenValue_STR: {
|
case TokenValue_STR: {
|
||||||
assert(!self->negated);
|
|
||||||
c11_sv sv = c11_string__sv(self->value->_str);
|
c11_sv sv = c11_string__sv(self->value->_str);
|
||||||
int index = Ctx__add_const_string(ctx, sv);
|
int index = Ctx__add_const_string(ctx, sv);
|
||||||
Ctx__emit_(ctx, OP_LOAD_CONST, index, self->line);
|
Ctx__emit_(ctx, OP_LOAD_CONST, index, self->line);
|
||||||
@ -313,7 +308,6 @@ LiteralExpr* LiteralExpr__new(int line, const TokenValue* value) {
|
|||||||
self->vt = &Vt;
|
self->vt = &Vt;
|
||||||
self->line = line;
|
self->line = line;
|
||||||
self->value = value;
|
self->value = value;
|
||||||
self->negated = false;
|
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -336,7 +330,9 @@ void Literal0Expr__emit_(Expr* self_, Ctx* ctx) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Literal0Expr* Literal0Expr__new(int line, TokenIndex token) {
|
Literal0Expr* Literal0Expr__new(int line, TokenIndex token) {
|
||||||
const static ExprVt Vt = {.emit_ = Literal0Expr__emit_, .is_json_object = true};
|
const static ExprVt Vt = {.emit_ = Literal0Expr__emit_,
|
||||||
|
.is_literal = true,
|
||||||
|
.is_json_object = true};
|
||||||
static_assert_expr_size(Literal0Expr);
|
static_assert_expr_size(Literal0Expr);
|
||||||
Literal0Expr* self = PoolExpr_alloc();
|
Literal0Expr* self = PoolExpr_alloc();
|
||||||
self->vt = &Vt;
|
self->vt = &Vt;
|
||||||
@ -879,7 +875,7 @@ static void _emit_compare(BinaryExpr* self, Ctx* ctx, c11_vector* jmps) {
|
|||||||
Ctx__emit_(ctx, OP_ROT_THREE, BC_NOARG, self->line); // [b, a, b]
|
Ctx__emit_(ctx, OP_ROT_THREE, BC_NOARG, self->line); // [b, a, b]
|
||||||
Ctx__emit_(ctx, OP_BINARY_OP, cmp_token2name(self->op), self->line);
|
Ctx__emit_(ctx, OP_BINARY_OP, cmp_token2name(self->op), self->line);
|
||||||
// [b, RES]
|
// [b, RES]
|
||||||
int index = Ctx__emit_(ctx, OP_SHORTCUT_IF_FALSE_OR_POP, BC_NOARG, self->line);
|
int index = Ctx__emit_(ctx, OP_JUMP_IF_FALSE_OR_POP, BC_NOARG, self->line);
|
||||||
c11_vector__push(int, jmps, index);
|
c11_vector__push(int, jmps, index);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -953,13 +949,10 @@ static void BinaryExpr__emit_(Expr* self_, Ctx* ctx) {
|
|||||||
for(int i = 0; i < jmps.count; i++) {
|
for(int i = 0; i < jmps.count; i++) {
|
||||||
Ctx__patch_jump(ctx, c11__getitem(int, &jmps, i));
|
Ctx__patch_jump(ctx, c11__getitem(int, &jmps, i));
|
||||||
}
|
}
|
||||||
c11_vector__dtor(&jmps);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
BinaryExpr* BinaryExpr__new(int line, TokenIndex op, bool inplace) {
|
BinaryExpr* BinaryExpr__new(int line, TokenIndex op, bool inplace) {
|
||||||
const static ExprVt Vt = {.emit_ = BinaryExpr__emit_,
|
const static ExprVt Vt = {.emit_ = BinaryExpr__emit_, .dtor = BinaryExpr__dtor};
|
||||||
.dtor = BinaryExpr__dtor,
|
|
||||||
.is_binary = true};
|
|
||||||
static_assert_expr_size(BinaryExpr);
|
static_assert_expr_size(BinaryExpr);
|
||||||
BinaryExpr* self = PoolExpr_alloc();
|
BinaryExpr* self = PoolExpr_alloc();
|
||||||
self->vt = &Vt;
|
self->vt = &Vt;
|
||||||
@ -1778,19 +1771,7 @@ static Error* exprUnaryOp(Compiler* self) {
|
|||||||
check(parse_expression(self, PREC_UNARY + 1, false));
|
check(parse_expression(self, PREC_UNARY + 1, false));
|
||||||
Expr* e = Ctx__s_popx(ctx());
|
Expr* e = Ctx__s_popx(ctx());
|
||||||
switch(op) {
|
switch(op) {
|
||||||
case TK_SUB: {
|
case TK_SUB: Ctx__s_push(ctx(), (Expr*)UnaryExpr__new(line, e, OP_UNARY_NEGATIVE)); break;
|
||||||
// constant fold
|
|
||||||
if(e->vt->is_literal) {
|
|
||||||
LiteralExpr* le = (LiteralExpr*)e;
|
|
||||||
if(le->value->index == TokenValue_I64 || le->value->index == TokenValue_F64) {
|
|
||||||
le->negated = true;
|
|
||||||
}
|
|
||||||
Ctx__s_push(ctx(), e);
|
|
||||||
} else {
|
|
||||||
Ctx__s_push(ctx(), (Expr*)UnaryExpr__new(line, e, OP_UNARY_NEGATIVE));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case TK_INVERT: Ctx__s_push(ctx(), (Expr*)UnaryExpr__new(line, e, OP_UNARY_INVERT)); break;
|
case TK_INVERT: Ctx__s_push(ctx(), (Expr*)UnaryExpr__new(line, e, OP_UNARY_INVERT)); break;
|
||||||
case TK_MUL: Ctx__s_push(ctx(), (Expr*)StarredExpr__new(line, e, 1)); break;
|
case TK_MUL: Ctx__s_push(ctx(), (Expr*)StarredExpr__new(line, e, 1)); break;
|
||||||
case TK_POW: Ctx__s_push(ctx(), (Expr*)StarredExpr__new(line, e, 2)); break;
|
case TK_POW: Ctx__s_push(ctx(), (Expr*)StarredExpr__new(line, e, 2)); break;
|
||||||
|
@ -30,11 +30,7 @@ static bool stack_binaryop(pk_VM* self, py_Name op, py_Name rop);
|
|||||||
#define THIRD() (self->stack.sp - 3)
|
#define THIRD() (self->stack.sp - 3)
|
||||||
#define FOURTH() (self->stack.sp - 4)
|
#define FOURTH() (self->stack.sp - 4)
|
||||||
#define STACK_SHRINK(n) (self->stack.sp -= n)
|
#define STACK_SHRINK(n) (self->stack.sp -= n)
|
||||||
#define PUSH(v) \
|
#define PUSH(v) (*self->stack.sp++ = *v)
|
||||||
do { \
|
|
||||||
*self->stack.sp = *v; \
|
|
||||||
self->stack.sp++; \
|
|
||||||
} while(0)
|
|
||||||
#define POP() (--self->stack.sp)
|
#define POP() (--self->stack.sp)
|
||||||
#define POPX() (*--self->stack.sp)
|
#define POPX() (*--self->stack.sp)
|
||||||
#define SP() (self->stack.sp)
|
#define SP() (self->stack.sp)
|
||||||
@ -90,35 +86,11 @@ pk_FrameResult pk_VM__run_top_frame(pk_VM* self) {
|
|||||||
c11_sbuf buf;
|
c11_sbuf buf;
|
||||||
c11_sbuf__ctor(&buf);
|
c11_sbuf__ctor(&buf);
|
||||||
for(py_Ref p = self->stack.begin; p != SP(); p++) {
|
for(py_Ref p = self->stack.begin; p != SP(); p++) {
|
||||||
switch(p->type) {
|
c11_sbuf__write_cstr(&buf, py_tpname(p->type));
|
||||||
case 0: c11_sbuf__write_cstr(&buf, "nil"); break;
|
|
||||||
case tp_int: c11_sbuf__write_i64(&buf, p->_i64); break;
|
|
||||||
case tp_float: c11_sbuf__write_f64(&buf, p->_f64, -1); break;
|
|
||||||
case tp_bool: c11_sbuf__write_cstr(&buf, p->_bool ? "True" : "False"); break;
|
|
||||||
case tp_none_type: c11_sbuf__write_cstr(&buf, "None"); break;
|
|
||||||
case tp_type: {
|
|
||||||
pk_sprintf(&buf, "<class '%t'>", py_totype(p));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case tp_str: {
|
|
||||||
int size;
|
|
||||||
const char* data = py_tostrn(p, &size);
|
|
||||||
pk_sprintf(&buf, "%q", (c11_sv){data, size});
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
default: {
|
|
||||||
pk_sprintf(&buf, "(%t)", p->type);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if(p != TOP()) c11_sbuf__write_cstr(&buf, ", ");
|
if(p != TOP()) c11_sbuf__write_cstr(&buf, ", ");
|
||||||
}
|
}
|
||||||
c11_string* stack_str = c11_sbuf__submit(&buf);
|
c11_string* stack_str = c11_sbuf__submit(&buf);
|
||||||
printf("L%-3d: %-25s %-6d [%s]\n",
|
printf("L%-3d: %-25s %-6d [%s]\n", Frame__lineno(frame), pk_opname(byte.op), byte.arg, stack_str->data);
|
||||||
Frame__lineno(frame),
|
|
||||||
pk_opname(byte.op),
|
|
||||||
byte.arg,
|
|
||||||
stack_str->data);
|
|
||||||
c11_string__delete(stack_str);
|
c11_string__delete(stack_str);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -722,10 +694,17 @@ static bool stack_binaryop(pk_VM* self, py_Name op, py_Name rop) {
|
|||||||
// [a, b]
|
// [a, b]
|
||||||
py_Ref magic = py_tpfindmagic(SECOND()->type, op);
|
py_Ref magic = py_tpfindmagic(SECOND()->type, op);
|
||||||
if(magic) {
|
if(magic) {
|
||||||
|
if(magic->type == tp_nativefunc) {
|
||||||
|
bool ok = magic->_cfunc(2, SECOND());
|
||||||
|
if(!ok) return false;
|
||||||
|
if(self->last_retval.type != tp_not_implemented_type) return true;
|
||||||
|
} else {
|
||||||
|
// standard call
|
||||||
bool ok = py_call(magic, 2, SECOND());
|
bool ok = py_call(magic, 2, SECOND());
|
||||||
if(!ok) return false;
|
if(!ok) return false;
|
||||||
if(self->last_retval.type != tp_not_implemented_type) return true;
|
if(self->last_retval.type != tp_not_implemented_type) return true;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// try reverse operation
|
// try reverse operation
|
||||||
if(rop) {
|
if(rop) {
|
||||||
// [a, b] -> [b, a]
|
// [a, b] -> [b, a]
|
||||||
@ -734,11 +713,18 @@ static bool stack_binaryop(pk_VM* self, py_Name op, py_Name rop) {
|
|||||||
*SECOND() = tmp;
|
*SECOND() = tmp;
|
||||||
magic = py_tpfindmagic(SECOND()->type, rop);
|
magic = py_tpfindmagic(SECOND()->type, rop);
|
||||||
if(magic) {
|
if(magic) {
|
||||||
|
if(magic->type == tp_nativefunc) {
|
||||||
|
bool ok = magic->_cfunc(2, SECOND());
|
||||||
|
if(!ok) return false;
|
||||||
|
if(self->last_retval.type != tp_not_implemented_type) return true;
|
||||||
|
} else {
|
||||||
|
// standard call
|
||||||
bool ok = py_call(magic, 2, SECOND());
|
bool ok = py_call(magic, 2, SECOND());
|
||||||
if(!ok) return false;
|
if(!ok) return false;
|
||||||
if(self->last_retval.type != tp_not_implemented_type) return true;
|
if(self->last_retval.type != tp_not_implemented_type) return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// eq/ne op never fails due to object.__eq__
|
// eq/ne op never fails due to object.__eq__
|
||||||
return py_exception("TypeError", "unsupported operand type(s) for '%n'", op);
|
return py_exception("TypeError", "unsupported operand type(s) for '%n'", op);
|
||||||
}
|
}
|
||||||
|
@ -56,7 +56,12 @@ void pk_TypeInfo__ctor(pk_TypeInfo* self,
|
|||||||
|
|
||||||
void pk_TypeInfo__dtor(pk_TypeInfo* self) { c11_vector__dtor(&self->annotated_fields); }
|
void pk_TypeInfo__dtor(pk_TypeInfo* self) { c11_vector__dtor(&self->annotated_fields); }
|
||||||
|
|
||||||
|
static bool _py_object__new__(int argc, py_Ref argv) {
|
||||||
|
assert(argc >= 1);
|
||||||
|
py_Type cls = argv[0].type;
|
||||||
|
py_newobject(py_retval(), cls, 0, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
void pk_VM__ctor(pk_VM* self) {
|
void pk_VM__ctor(pk_VM* self) {
|
||||||
self->top_frame = NULL;
|
self->top_frame = NULL;
|
||||||
@ -90,7 +95,6 @@ void pk_VM__ctor(pk_VM* self) {
|
|||||||
|
|
||||||
validate(tp_object, pk_VM__new_type(self, "object", 0, NULL, true));
|
validate(tp_object, pk_VM__new_type(self, "object", 0, NULL, true));
|
||||||
validate(tp_type, pk_VM__new_type(self, "type", 1, NULL, false));
|
validate(tp_type, pk_VM__new_type(self, "type", 1, NULL, false));
|
||||||
pk_object__register();
|
|
||||||
|
|
||||||
validate(tp_int, pk_VM__new_type(self, "int", tp_object, NULL, false));
|
validate(tp_int, pk_VM__new_type(self, "int", tp_object, NULL, false));
|
||||||
validate(tp_float, pk_VM__new_type(self, "float", tp_object, NULL, false));
|
validate(tp_float, pk_VM__new_type(self, "float", tp_object, NULL, false));
|
||||||
@ -132,7 +136,7 @@ void pk_VM__ctor(pk_VM* self) {
|
|||||||
#undef validate
|
#undef validate
|
||||||
|
|
||||||
self->StopIteration = *py_tpobject(tp_stop_iteration);
|
self->StopIteration = *py_tpobject(tp_stop_iteration);
|
||||||
self->builtins = pk_builtins__register();
|
self->builtins = *py_newmodule("builtins", NULL);
|
||||||
|
|
||||||
/* Setup Public Builtin Types */
|
/* Setup Public Builtin Types */
|
||||||
py_Type public_types[] = {tp_object,
|
py_Type public_types[] = {tp_object,
|
||||||
@ -162,6 +166,10 @@ void pk_VM__ctor(pk_VM* self) {
|
|||||||
py_newnotimplemented(&tmp);
|
py_newnotimplemented(&tmp);
|
||||||
py_setdict(&self->builtins, py_name("NotImplemented"), &tmp);
|
py_setdict(&self->builtins, py_name("NotImplemented"), &tmp);
|
||||||
|
|
||||||
|
/* Do Buildin Bindings*/
|
||||||
|
// object.__new__
|
||||||
|
py_bindmagic(tp_object, __new__, _py_object__new__);
|
||||||
|
|
||||||
self->main = *py_newmodule("__main__", NULL);
|
self->main = *py_newmodule("__main__", NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,29 +50,3 @@ py_Ref py_newmodule(const char* name, const char* package) {
|
|||||||
py_poptmp(2);
|
py_poptmp(2);
|
||||||
return py_getmodule(name);
|
return py_getmodule(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////
|
|
||||||
|
|
||||||
static bool _py_builtins__repr(int argc, py_Ref argv){
|
|
||||||
PY_CHECK_ARGC(1);
|
|
||||||
return py_repr(argv);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool _py_builtins__exit(int argc, py_Ref argv){
|
|
||||||
int code = 0;
|
|
||||||
if(argc > 1) return TypeError("exit() takes at most 1 argument");
|
|
||||||
if(argc == 1){
|
|
||||||
PY_CHECK_ARG_TYPE(0, tp_int);
|
|
||||||
code = py_toint(argv);
|
|
||||||
}
|
|
||||||
// return py_exception("SystemExit", "%d", code);
|
|
||||||
exit(code);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
py_TValue pk_builtins__register(){
|
|
||||||
py_Ref builtins = py_newmodule("builtins", NULL);
|
|
||||||
py_bindnativefunc(builtins, "repr", _py_builtins__repr);
|
|
||||||
py_bindnativefunc(builtins, "exit", _py_builtins__exit);
|
|
||||||
return *builtins;
|
|
||||||
}
|
|
@ -7,12 +7,12 @@
|
|||||||
static bool _py_int##name(int argc, py_Ref argv) { \
|
static bool _py_int##name(int argc, py_Ref argv) { \
|
||||||
PY_CHECK_ARGC(2); \
|
PY_CHECK_ARGC(2); \
|
||||||
if(py_isint(&argv[1])) { \
|
if(py_isint(&argv[1])) { \
|
||||||
py_i64 lhs = py_toint(&argv[0]); \
|
int64_t lhs = py_toint(&argv[0]); \
|
||||||
py_i64 rhs = py_toint(&argv[1]); \
|
int64_t rhs = py_toint(&argv[1]); \
|
||||||
rint(py_retval(), lhs op rhs); \
|
rint(py_retval(), lhs op rhs); \
|
||||||
} else if(py_isfloat(&argv[1])) { \
|
} else if(py_isfloat(&argv[1])) { \
|
||||||
py_i64 lhs = py_toint(&argv[0]); \
|
int64_t lhs = py_toint(&argv[0]); \
|
||||||
py_f64 rhs = py_tofloat(&argv[1]); \
|
double rhs = py_tofloat(&argv[1]); \
|
||||||
rfloat(py_retval(), lhs op rhs); \
|
rfloat(py_retval(), lhs op rhs); \
|
||||||
} else { \
|
} else { \
|
||||||
py_newnotimplemented(py_retval()); \
|
py_newnotimplemented(py_retval()); \
|
||||||
@ -21,8 +21,8 @@
|
|||||||
} \
|
} \
|
||||||
static bool _py_float##name(int argc, py_Ref argv) { \
|
static bool _py_float##name(int argc, py_Ref argv) { \
|
||||||
PY_CHECK_ARGC(2); \
|
PY_CHECK_ARGC(2); \
|
||||||
py_f64 lhs = py_tofloat(&argv[0]); \
|
double lhs = py_tofloat(&argv[0]); \
|
||||||
py_f64 rhs; \
|
double rhs; \
|
||||||
if(py_castfloat(&argv[1], &rhs)) { \
|
if(py_castfloat(&argv[1], &rhs)) { \
|
||||||
rfloat(py_retval(), lhs op rhs); \
|
rfloat(py_retval(), lhs op rhs); \
|
||||||
} else { \
|
} else { \
|
||||||
@ -36,7 +36,7 @@ DEF_NUM_BINARY_OP(__sub__, -, py_newint, py_newfloat)
|
|||||||
DEF_NUM_BINARY_OP(__mul__, *, py_newint, py_newfloat)
|
DEF_NUM_BINARY_OP(__mul__, *, py_newint, py_newfloat)
|
||||||
|
|
||||||
DEF_NUM_BINARY_OP(__eq__, ==, py_newbool, py_newbool)
|
DEF_NUM_BINARY_OP(__eq__, ==, py_newbool, py_newbool)
|
||||||
DEF_NUM_BINARY_OP(__ne__, !=, py_newbool, py_newbool)
|
DEF_NUM_BINARY_OP(__ne__, ==, py_newbool, py_newbool)
|
||||||
DEF_NUM_BINARY_OP(__lt__, <, py_newbool, py_newbool)
|
DEF_NUM_BINARY_OP(__lt__, <, py_newbool, py_newbool)
|
||||||
DEF_NUM_BINARY_OP(__le__, <=, py_newbool, py_newbool)
|
DEF_NUM_BINARY_OP(__le__, <=, py_newbool, py_newbool)
|
||||||
DEF_NUM_BINARY_OP(__gt__, >, py_newbool, py_newbool)
|
DEF_NUM_BINARY_OP(__gt__, >, py_newbool, py_newbool)
|
||||||
@ -46,22 +46,22 @@ DEF_NUM_BINARY_OP(__ge__, >=, py_newbool, py_newbool)
|
|||||||
|
|
||||||
static bool _py_int__neg__(int argc, py_Ref argv) {
|
static bool _py_int__neg__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_i64 val = py_toint(&argv[0]);
|
int64_t val = py_toint(&argv[0]);
|
||||||
py_newint(py_retval(), -val);
|
py_newint(py_retval(), -val);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_float__neg__(int argc, py_Ref argv) {
|
static bool _py_float__neg__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_f64 val = py_tofloat(&argv[0]);
|
double val = py_tofloat(&argv[0]);
|
||||||
py_newfloat(py_retval(), -val);
|
py_newfloat(py_retval(), -val);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_int__truediv__(int argc, py_Ref argv) {
|
static bool _py_int__truediv__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
py_i64 lhs = py_toint(&argv[0]);
|
int64_t lhs = py_toint(&argv[0]);
|
||||||
py_f64 rhs;
|
double rhs;
|
||||||
if(py_castfloat(&argv[1], &rhs)) {
|
if(py_castfloat(&argv[1], &rhs)) {
|
||||||
py_newfloat(py_retval(), lhs / rhs);
|
py_newfloat(py_retval(), lhs / rhs);
|
||||||
} else {
|
} else {
|
||||||
@ -72,8 +72,8 @@ static bool _py_int__truediv__(int argc, py_Ref argv) {
|
|||||||
|
|
||||||
static bool _py_float__truediv__(int argc, py_Ref argv) {
|
static bool _py_float__truediv__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
py_f64 lhs = py_tofloat(&argv[0]);
|
double lhs = py_tofloat(&argv[0]);
|
||||||
py_f64 rhs;
|
double rhs;
|
||||||
if(py_castfloat(&argv[1], &rhs)) {
|
if(py_castfloat(&argv[1], &rhs)) {
|
||||||
py_newfloat(py_retval(), lhs / rhs);
|
py_newfloat(py_retval(), lhs / rhs);
|
||||||
} else {
|
} else {
|
||||||
@ -87,8 +87,8 @@ static bool _py_float__truediv__(int argc, py_Ref argv) {
|
|||||||
static bool _py_number__pow__(int argc, py_Ref argv) {
|
static bool _py_number__pow__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
if(py_isint(&argv[0]) && py_isint(&argv[1])) {
|
if(py_isint(&argv[0]) && py_isint(&argv[1])) {
|
||||||
py_i64 lhs = py_toint(&argv[0]);
|
int64_t lhs = py_toint(&argv[0]);
|
||||||
py_i64 rhs = py_toint(&argv[1]);
|
int64_t rhs = py_toint(&argv[1]);
|
||||||
if(rhs < 0) {
|
if(rhs < 0) {
|
||||||
if(lhs == 0) {
|
if(lhs == 0) {
|
||||||
return ZeroDivisionError("0.0 cannot be raised to a negative power");
|
return ZeroDivisionError("0.0 cannot be raised to a negative power");
|
||||||
@ -97,7 +97,7 @@ static bool _py_number__pow__(int argc, py_Ref argv) {
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// rhs >= 0
|
// rhs >= 0
|
||||||
py_i64 ret = 1;
|
int64_t ret = 1;
|
||||||
while(true) {
|
while(true) {
|
||||||
if(rhs & 1) ret *= lhs;
|
if(rhs & 1) ret *= lhs;
|
||||||
rhs >>= 1;
|
rhs >>= 1;
|
||||||
@ -107,7 +107,7 @@ static bool _py_number__pow__(int argc, py_Ref argv) {
|
|||||||
py_newint(py_retval(), ret);
|
py_newint(py_retval(), ret);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
py_f64 lhs, rhs;
|
double lhs, rhs;
|
||||||
py_castfloat(&argv[0], &lhs);
|
py_castfloat(&argv[0], &lhs);
|
||||||
if(py_castfloat(&argv[1], &rhs)) {
|
if(py_castfloat(&argv[1], &rhs)) {
|
||||||
py_newfloat(py_retval(), pow(lhs, rhs));
|
py_newfloat(py_retval(), pow(lhs, rhs));
|
||||||
@ -120,9 +120,9 @@ static bool _py_number__pow__(int argc, py_Ref argv) {
|
|||||||
|
|
||||||
static bool _py_int__floordiv__(int argc, py_Ref argv) {
|
static bool _py_int__floordiv__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
py_i64 lhs = py_toint(&argv[0]);
|
int64_t lhs = py_toint(&argv[0]);
|
||||||
if(py_isint(&argv[1])) {
|
if(py_isint(&argv[1])) {
|
||||||
py_i64 rhs = py_toint(&argv[1]);
|
int64_t rhs = py_toint(&argv[1]);
|
||||||
if(rhs == 0) return -1;
|
if(rhs == 0) return -1;
|
||||||
py_newint(py_retval(), lhs / rhs);
|
py_newint(py_retval(), lhs / rhs);
|
||||||
} else {
|
} else {
|
||||||
@ -133,9 +133,9 @@ static bool _py_int__floordiv__(int argc, py_Ref argv) {
|
|||||||
|
|
||||||
static bool _py_int__mod__(int argc, py_Ref argv) {
|
static bool _py_int__mod__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
py_i64 lhs = py_toint(&argv[0]);
|
int64_t lhs = py_toint(&argv[0]);
|
||||||
if(py_isint(&argv[1])) {
|
if(py_isint(&argv[1])) {
|
||||||
py_i64 rhs = py_toint(&argv[1]);
|
int64_t rhs = py_toint(&argv[1]);
|
||||||
if(rhs == 0) return ZeroDivisionError("integer division or modulo by zero");
|
if(rhs == 0) return ZeroDivisionError("integer division or modulo by zero");
|
||||||
py_newint(py_retval(), lhs % rhs);
|
py_newint(py_retval(), lhs % rhs);
|
||||||
} else {
|
} else {
|
||||||
@ -146,14 +146,14 @@ static bool _py_int__mod__(int argc, py_Ref argv) {
|
|||||||
|
|
||||||
static bool _py_int__invert__(int argc, py_Ref argv) {
|
static bool _py_int__invert__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_i64 val = py_toint(&argv[0]);
|
int64_t val = py_toint(&argv[0]);
|
||||||
py_newint(py_retval(), ~val);
|
py_newint(py_retval(), ~val);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_int__bit_length(int argc, py_Ref argv) {
|
static bool _py_int__bit_length(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_i64 x = py_toint(py_arg(0));
|
int64_t x = py_toint(py_arg(0));
|
||||||
if(x < 0) x = -x;
|
if(x < 0) x = -x;
|
||||||
int bits = 0;
|
int bits = 0;
|
||||||
while(x) {
|
while(x) {
|
||||||
@ -167,9 +167,9 @@ static bool _py_int__bit_length(int argc, py_Ref argv) {
|
|||||||
#define DEF_INT_BITWISE_OP(name, op) \
|
#define DEF_INT_BITWISE_OP(name, op) \
|
||||||
static bool _py_int##name(int argc, py_Ref argv) { \
|
static bool _py_int##name(int argc, py_Ref argv) { \
|
||||||
PY_CHECK_ARGC(2); \
|
PY_CHECK_ARGC(2); \
|
||||||
py_i64 lhs = py_toint(&argv[0]); \
|
int64_t lhs = py_toint(&argv[0]); \
|
||||||
if(py_isint(&argv[1])) { \
|
if(py_isint(&argv[1])) { \
|
||||||
py_i64 rhs = py_toint(&argv[1]); \
|
int64_t rhs = py_toint(&argv[1]); \
|
||||||
py_newint(py_retval(), lhs op rhs); \
|
py_newint(py_retval(), lhs op rhs); \
|
||||||
} else { \
|
} else { \
|
||||||
py_newnotimplemented(py_retval()); \
|
py_newnotimplemented(py_retval()); \
|
||||||
@ -187,7 +187,7 @@ DEF_INT_BITWISE_OP(__rshift__, >>)
|
|||||||
|
|
||||||
static bool _py_int__repr__(int argc, py_Ref argv) {
|
static bool _py_int__repr__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_i64 val = py_toint(&argv[0]);
|
int64_t val = py_toint(&argv[0]);
|
||||||
char buf[32];
|
char buf[32];
|
||||||
int size = snprintf(buf, sizeof(buf), "%lld", (long long)val);
|
int size = snprintf(buf, sizeof(buf), "%lld", (long long)val);
|
||||||
py_newstrn(py_retval(), buf, size);
|
py_newstrn(py_retval(), buf, size);
|
||||||
@ -196,7 +196,7 @@ static bool _py_int__repr__(int argc, py_Ref argv) {
|
|||||||
|
|
||||||
static bool _py_float__repr__(int argc, py_Ref argv) {
|
static bool _py_float__repr__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_f64 val = py_tofloat(&argv[0]);
|
double val = py_tofloat(&argv[0]);
|
||||||
char buf[32];
|
char buf[32];
|
||||||
int size = snprintf(buf, sizeof(buf), "%f", val);
|
int size = snprintf(buf, sizeof(buf), "%f", val);
|
||||||
py_newstrn(py_retval(), buf, size);
|
py_newstrn(py_retval(), buf, size);
|
||||||
@ -223,7 +223,7 @@ static py_i64 c11_8bytes__hash(union c11_8bytes u) {
|
|||||||
|
|
||||||
static bool _py_int__hash__(int argc, py_Ref argv) {
|
static bool _py_int__hash__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_i64 val = py_toint(&argv[0]);
|
int64_t val = py_toint(&argv[0]);
|
||||||
union c11_8bytes u = {._i64 = val};
|
union c11_8bytes u = {._i64 = val};
|
||||||
py_newint(py_retval(), c11_8bytes__hash(u));
|
py_newint(py_retval(), c11_8bytes__hash(u));
|
||||||
return true;
|
return true;
|
||||||
@ -231,7 +231,7 @@ static bool _py_int__hash__(int argc, py_Ref argv) {
|
|||||||
|
|
||||||
static bool _py_float__hash__(int argc, py_Ref argv) {
|
static bool _py_float__hash__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
py_f64 val = py_tofloat(&argv[0]);
|
double val = py_tofloat(&argv[0]);
|
||||||
union c11_8bytes u = {._f64 = val};
|
union c11_8bytes u = {._f64 = val};
|
||||||
py_newint(py_retval(), c11_8bytes__hash(u));
|
py_newint(py_retval(), c11_8bytes__hash(u));
|
||||||
return true;
|
return true;
|
||||||
@ -248,7 +248,7 @@ static bool _py_int__new__(int argc, py_Ref argv) {
|
|||||||
switch(argv[1].type) {
|
switch(argv[1].type) {
|
||||||
case tp_float: {
|
case tp_float: {
|
||||||
// int(1.1) == 1
|
// int(1.1) == 1
|
||||||
py_newint(py_retval(), (py_i64)py_tofloat(&argv[1]));
|
py_newint(py_retval(), (int64_t)py_tofloat(&argv[1]));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case tp_int: {
|
case tp_int: {
|
||||||
@ -258,7 +258,7 @@ static bool _py_int__new__(int argc, py_Ref argv) {
|
|||||||
}
|
}
|
||||||
case tp_bool: {
|
case tp_bool: {
|
||||||
// int(True) == 1
|
// int(True) == 1
|
||||||
py_newint(py_retval(), (py_i64)py_tobool(&argv[1]));
|
py_newint(py_retval(), (int64_t)py_tobool(&argv[1]));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case tp_str: break; // leave to the next block
|
case tp_str: break; // leave to the next block
|
||||||
@ -340,7 +340,7 @@ static bool _py_float__new__(int argc, py_Ref argv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// tp_bool
|
// tp_bool
|
||||||
static bool _py_bool__new__(int argc, py_Ref argv) {
|
static bool _py_bool__new__(int argc, py_Ref argv){
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
int res = py_bool(argv);
|
int res = py_bool(argv);
|
||||||
if(res == -1) return false;
|
if(res == -1) return false;
|
||||||
@ -348,24 +348,24 @@ static bool _py_bool__new__(int argc, py_Ref argv) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_bool__hash__(int argc, py_Ref argv) {
|
static bool _py_bool__hash__(int argc, py_Ref argv){
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
bool res = py_tobool(argv);
|
bool res = py_tobool(argv);
|
||||||
py_newint(py_retval(), res);
|
py_newint(py_retval(), res);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_bool__repr__(int argc, py_Ref argv) {
|
static bool _py_bool__repr__(int argc, py_Ref argv){
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
bool res = py_tobool(argv);
|
bool res = py_tobool(argv);
|
||||||
py_newstr(py_retval(), res ? "True" : "False");
|
py_newstr(py_retval(), res ? "True" : "False");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_bool__eq__(int argc, py_Ref argv) {
|
static bool _py_bool__eq__(int argc, py_Ref argv){
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
bool lhs = py_tobool(&argv[0]);
|
bool lhs = py_tobool(&argv[0]);
|
||||||
if(argv[1].type == tp_bool) {
|
if(argv[1].type == tp_bool){
|
||||||
bool rhs = py_tobool(&argv[1]);
|
bool rhs = py_tobool(&argv[1]);
|
||||||
py_newbool(py_retval(), lhs == rhs);
|
py_newbool(py_retval(), lhs == rhs);
|
||||||
} else {
|
} else {
|
||||||
@ -374,10 +374,10 @@ static bool _py_bool__eq__(int argc, py_Ref argv) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _py_bool__ne__(int argc, py_Ref argv) {
|
static bool _py_bool__ne__(int argc, py_Ref argv){
|
||||||
PY_CHECK_ARGC(2);
|
PY_CHECK_ARGC(2);
|
||||||
bool lhs = py_tobool(&argv[0]);
|
bool lhs = py_tobool(&argv[0]);
|
||||||
if(argv[1].type == tp_bool) {
|
if(argv[1].type == tp_bool){
|
||||||
bool rhs = py_tobool(&argv[1]);
|
bool rhs = py_tobool(&argv[1]);
|
||||||
py_newbool(py_retval(), lhs != rhs);
|
py_newbool(py_retval(), lhs != rhs);
|
||||||
} else {
|
} else {
|
||||||
|
@ -1,37 +0,0 @@
|
|||||||
#include "pocketpy/interpreter/vm.h"
|
|
||||||
#include "pocketpy/pocketpy.h"
|
|
||||||
|
|
||||||
static bool _py_object__new__(int argc, py_Ref argv) {
|
|
||||||
assert(argc >= 1);
|
|
||||||
py_Type cls = argv[0].type;
|
|
||||||
py_newobject(py_retval(), cls, 0, 0);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool _py_object__hash__(int argc, py_Ref argv) {
|
|
||||||
PY_CHECK_ARGC(1);
|
|
||||||
assert(argv->is_ptr);
|
|
||||||
py_newint(py_retval(), (py_i64)argv->_obj);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool _py_object__eq__(int argc, py_Ref argv) {
|
|
||||||
PY_CHECK_ARGC(2);
|
|
||||||
bool res = py_isidentical(py_arg(0), py_arg(1));
|
|
||||||
py_newbool(py_retval(), res);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool _py_object__ne__(int argc, py_Ref argv) {
|
|
||||||
PY_CHECK_ARGC(2);
|
|
||||||
bool res = py_isidentical(py_arg(0), py_arg(1));
|
|
||||||
py_newbool(py_retval(), !res);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void pk_object__register() {
|
|
||||||
py_bindmagic(tp_object, __new__, _py_object__new__);
|
|
||||||
py_bindmagic(tp_object, __hash__, _py_object__hash__);
|
|
||||||
py_bindmagic(tp_object, __eq__, _py_object__eq__);
|
|
||||||
py_bindmagic(tp_object, __ne__, _py_object__ne__);
|
|
||||||
}
|
|
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
bool py_isidentical(const py_Ref lhs, const py_Ref rhs) {
|
bool py_isidentical(const py_Ref lhs, const py_Ref rhs) {
|
||||||
if(lhs->type != rhs->type) return false;
|
if(lhs->type != rhs->type) return false;
|
||||||
switch(lhs->type) {
|
switch(lhs->type){
|
||||||
case tp_int: return lhs->_i64 == rhs->_i64;
|
case tp_int: return lhs->_i64 == rhs->_i64;
|
||||||
case tp_float: return lhs->_f64 == rhs->_f64;
|
case tp_float: return lhs->_f64 == rhs->_f64;
|
||||||
case tp_bool: return lhs->_bool == rhs->_bool;
|
case tp_bool: return lhs->_bool == rhs->_bool;
|
||||||
@ -17,48 +17,9 @@ bool py_isidentical(const py_Ref lhs, const py_Ref rhs) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int py_bool(const py_Ref val) {
|
int py_bool(const py_Ref val) { return 1; }
|
||||||
switch(val->type) {
|
|
||||||
case tp_bool: return val->_bool;
|
|
||||||
case tp_int: return val->_i64 != 0;
|
|
||||||
case tp_float: return val->_f64 != 0;
|
|
||||||
case tp_none_type: return 0;
|
|
||||||
default: {
|
|
||||||
py_Ref tmp = py_tpfindmagic(val->type, __bool__);
|
|
||||||
if(tmp) {
|
|
||||||
bool ok = py_call(tmp, 1, val);
|
|
||||||
if(!ok) return -1;
|
|
||||||
return py_tobool(py_retval());
|
|
||||||
} else {
|
|
||||||
tmp = py_tpfindmagic(val->type, __len__);
|
|
||||||
if(tmp) {
|
|
||||||
bool ok = py_call(tmp, 1, val);
|
|
||||||
if(!ok) return -1;
|
|
||||||
return py_toint(py_retval());
|
|
||||||
} else {
|
|
||||||
return 1; // True
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool py_hash(const py_Ref val, int64_t* out) {
|
bool py_hash(const py_Ref val, int64_t* out) { return 0; }
|
||||||
py_Type t = val->type;
|
|
||||||
pk_TypeInfo* types = (pk_TypeInfo*)pk_current_vm->types.data;
|
|
||||||
do {
|
|
||||||
py_Ref _hash = &types[t].magic[__hash__];
|
|
||||||
py_Ref _eq = &types[t].magic[__eq__];
|
|
||||||
if(!py_isnil(_hash) && !py_isnil(_eq)) {
|
|
||||||
bool ok = py_call(_hash, 1, val);
|
|
||||||
if(!ok) return false;
|
|
||||||
*out = py_toint(py_retval());
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
t = types[t].base;
|
|
||||||
} while(t);
|
|
||||||
return TypeError("unhashable type: '%t'", val->type);
|
|
||||||
}
|
|
||||||
|
|
||||||
int py_getattr(const py_Ref self, py_Name name, py_Ref out) { return -1; }
|
int py_getattr(const py_Ref self, py_Name name, py_Ref out) { return -1; }
|
||||||
|
|
||||||
|
@ -52,15 +52,8 @@ unsigned char* py_tobytes(const py_Ref self, int* size) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////
|
////////////////////////////////
|
||||||
static bool _py_str__new__(int argc, py_Ref argv) {
|
|
||||||
assert(argc >= 1);
|
static bool _py_str__new__(int argc, py_Ref argv) { return true; }
|
||||||
if(argc == 1) {
|
|
||||||
py_newstr(py_retval(), "");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if(argc > 2) return TypeError("str() takes at most 1 argument");
|
|
||||||
return py_str(py_arg(1));
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool _py_str__hash__(int argc, py_Ref argv) {
|
static bool _py_str__hash__(int argc, py_Ref argv) {
|
||||||
PY_CHECK_ARGC(1);
|
PY_CHECK_ARGC(1);
|
||||||
@ -164,7 +157,7 @@ static bool _py_str__getitem__(int argc, py_Ref argv) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define DEF_STR_CMP_OP(op, __f, __cond) \
|
#define DEF_STR_CMP_OP(op, f, condition) \
|
||||||
static bool _py_str##op(int argc, py_Ref argv) { \
|
static bool _py_str##op(int argc, py_Ref argv) { \
|
||||||
PY_CHECK_ARGC(2); \
|
PY_CHECK_ARGC(2); \
|
||||||
c11_string* self = py_touserdata(&argv[0]); \
|
c11_string* self = py_touserdata(&argv[0]); \
|
||||||
@ -172,8 +165,8 @@ static bool _py_str__getitem__(int argc, py_Ref argv) {
|
|||||||
py_newnotimplemented(py_retval()); \
|
py_newnotimplemented(py_retval()); \
|
||||||
} else { \
|
} else { \
|
||||||
c11_string* other = py_touserdata(&argv[1]); \
|
c11_string* other = py_touserdata(&argv[1]); \
|
||||||
int res = __f(c11_string__sv(self), c11_string__sv(other)); \
|
int res = c11_sv__cmp(c11_string__sv(self), c11_string__sv(other)); \
|
||||||
py_newbool(py_retval(), __cond); \
|
py_newbool(py_retval(), condition); \
|
||||||
} \
|
} \
|
||||||
return true; \
|
return true; \
|
||||||
}
|
}
|
||||||
@ -288,11 +281,3 @@ py_Type pk_bytes__register() {
|
|||||||
// no need to dtor because the memory is controlled by the object
|
// no need to dtor because the memory is controlled by the object
|
||||||
return type;
|
return type;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool py_str(py_Ref val) {
|
|
||||||
py_Ref tmp = py_tpfindmagic(val->type, __str__);
|
|
||||||
if(!tmp) return py_repr(val);
|
|
||||||
return py_call(tmp, 1, val);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool py_repr(py_Ref val) { return py_callmagic(__repr__, 1, val); }
|
|
@ -64,11 +64,6 @@ void py_push(const py_Ref src){
|
|||||||
*vm->stack.sp++ = *src;
|
*vm->stack.sp++ = *src;
|
||||||
}
|
}
|
||||||
|
|
||||||
void py_pushnil(){
|
|
||||||
pk_VM* vm = pk_current_vm;
|
|
||||||
py_newnil(vm->stack.sp++);
|
|
||||||
}
|
|
||||||
|
|
||||||
py_Ref py_pushtmp(){
|
py_Ref py_pushtmp(){
|
||||||
pk_VM* vm = pk_current_vm;
|
pk_VM* vm = pk_current_vm;
|
||||||
py_newnil(vm->stack.sp++);
|
py_newnil(vm->stack.sp++);
|
||||||
|
@ -184,22 +184,11 @@ bool py_exec(const char* source) { return pk_VM__exec(pk_current_vm, source, "<e
|
|||||||
|
|
||||||
bool py_eval(const char* source) { return pk_VM__exec(pk_current_vm, source, "<eval>", EVAL_MODE); }
|
bool py_eval(const char* source) { return pk_VM__exec(pk_current_vm, source, "<eval>", EVAL_MODE); }
|
||||||
|
|
||||||
bool py_exec2(const char* source, const char* filename, enum CompileMode mode) {
|
bool py_exec2(const char* source, const char* filename, enum CompileMode mode){
|
||||||
return pk_VM__exec(pk_current_vm, source, filename, mode);
|
return pk_VM__exec(pk_current_vm, source, filename, mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool py_call(py_Ref f, int argc, py_Ref argv) {
|
bool py_call(py_Ref f, int argc, py_Ref argv) { return -1; }
|
||||||
if(f->type == tp_nativefunc) {
|
|
||||||
return f->_cfunc(argc, argv);
|
|
||||||
} else {
|
|
||||||
pk_VM* vm = pk_current_vm;
|
|
||||||
py_push(f);
|
|
||||||
py_pushnil();
|
|
||||||
for(int i = 0; i < argc; i++)
|
|
||||||
py_push(py_offset(argv, i));
|
|
||||||
return pk_VM__vectorcall(vm, argc, 0, false) == RES_ERROR;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool py_callmethod(py_Ref self, py_Name name, int argc, py_Ref argv) { return -1; }
|
bool py_callmethod(py_Ref self, py_Name name, int argc, py_Ref argv) { return -1; }
|
||||||
|
|
||||||
@ -296,6 +285,9 @@ bool py_callmagic(py_Name name, int argc, py_Ref argv) {
|
|||||||
assert(argc >= 1);
|
assert(argc >= 1);
|
||||||
assert(py_ismagicname(name));
|
assert(py_ismagicname(name));
|
||||||
py_Ref tmp = py_tpfindmagic(argv->type, name);
|
py_Ref tmp = py_tpfindmagic(argv->type, name);
|
||||||
if(!tmp) return AttributeError(argv, name);
|
if(!tmp){
|
||||||
|
return AttributeError(argv, name);
|
||||||
|
}
|
||||||
|
if(tmp->type == tp_nativefunc) return tmp->_cfunc(argc, argv);
|
||||||
return py_call(tmp, argc, argv);
|
return py_call(tmp, argc, argv);
|
||||||
}
|
}
|
||||||
|
106
tests/01_int.py
106
tests/01_int.py
@ -103,49 +103,71 @@ assert ~0 == -1
|
|||||||
assert str(1) == '1'
|
assert str(1) == '1'
|
||||||
assert repr(1) == '1'
|
assert repr(1) == '1'
|
||||||
|
|
||||||
assert 1 < 2 < 3
|
try:
|
||||||
assert 4 > 3 >= 3
|
1 // 0
|
||||||
|
exit(1)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
1 % 0
|
||||||
|
exit(1)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
2**60 // 0
|
||||||
|
exit(1)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
2**60 % 0
|
||||||
|
exit(1)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
divmod(1, 0)
|
||||||
|
exit(1)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
divmod(2**60, 0)
|
||||||
|
exit(1)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
pass
|
||||||
|
|
||||||
assert not 1 < 2 > 3
|
assert not 1 < 2 > 3
|
||||||
|
|
||||||
# try:
|
try:
|
||||||
# 1 // 0
|
x = eval("231231312312312312312312312312312312314354657553423345632")
|
||||||
# exit(1)
|
print(f"eval should fail, but got {x!r}")
|
||||||
# except ZeroDivisionError:
|
exit(1)
|
||||||
# pass
|
except SyntaxError:
|
||||||
|
pass
|
||||||
|
|
||||||
# try:
|
assert int("-5") == -5
|
||||||
# 1 % 0
|
assert int("-4") == -4
|
||||||
# exit(1)
|
assert int("-3") == -3
|
||||||
# except ZeroDivisionError:
|
assert int("-2") == -2
|
||||||
# pass
|
assert int("-1") == -1
|
||||||
|
assert int("0") == 0
|
||||||
|
assert int("1") == 1
|
||||||
|
assert int("2") == 2
|
||||||
|
assert int("3") == 3
|
||||||
|
assert int("4") == 4
|
||||||
|
assert int("5") == 5
|
||||||
|
assert int("6") == 6
|
||||||
|
assert int("7") == 7
|
||||||
|
assert int("8") == 8
|
||||||
|
assert int("9") == 9
|
||||||
|
assert int("10") == 10
|
||||||
|
assert int("11") == 11
|
||||||
|
assert int("12") == 12
|
||||||
|
assert int("13") == 13
|
||||||
|
assert int("14") == 14
|
||||||
|
assert int("15") == 15
|
||||||
|
assert int("16") == 16
|
||||||
|
|
||||||
# try:
|
|
||||||
# 2**60 // 0
|
|
||||||
# exit(1)
|
|
||||||
# except ZeroDivisionError:
|
|
||||||
# pass
|
|
||||||
|
|
||||||
# try:
|
|
||||||
# 2**60 % 0
|
|
||||||
# exit(1)
|
|
||||||
# except ZeroDivisionError:
|
|
||||||
# pass
|
|
||||||
|
|
||||||
# try:
|
|
||||||
# divmod(1, 0)
|
|
||||||
# exit(1)
|
|
||||||
# except ZeroDivisionError:
|
|
||||||
# pass
|
|
||||||
|
|
||||||
# try:
|
|
||||||
# divmod(2**60, 0)
|
|
||||||
# exit(1)
|
|
||||||
# except ZeroDivisionError:
|
|
||||||
# pass
|
|
||||||
|
|
||||||
# try:
|
|
||||||
# x = eval("231231312312312312312312312312312312314354657553423345632")
|
|
||||||
# print(f"eval should fail, but got {x!r}")
|
|
||||||
# exit(1)
|
|
||||||
# except SyntaxError:
|
|
||||||
# pass
|
|
Loading…
x
Reference in New Issue
Block a user