Compare commits

...

4 Commits

Author SHA1 Message Date
blueloveTH
79012a6b08 some fix 2024-07-07 00:56:02 +08:00
blueloveTH
18fe69d579 some update 2024-07-07 00:28:41 +08:00
blueloveTH
5c959e7274 support function 2024-07-07 00:06:21 +08:00
blueloveTH
803e7f1791 some fix 2024-07-06 23:07:44 +08:00
21 changed files with 613 additions and 383 deletions

View File

@ -34,6 +34,7 @@ c11_vector c11_vector__copy(const c11_vector* self);
void c11_vector__reserve(c11_vector* self, int capacity); void c11_vector__reserve(c11_vector* self, int capacity);
void c11_vector__clear(c11_vector* self); void c11_vector__clear(c11_vector* self);
void* c11_vector__emplace(c11_vector* self); void* c11_vector__emplace(c11_vector* self);
bool c11_vector__contains(const c11_vector* self, void* elem);
c11_array c11_vector__submit(c11_vector* self); c11_array c11_vector__submit(c11_vector* self);
#define c11__getitem(T, self, index) (((T*)(self)->data)[index]) #define c11__getitem(T, self, index) (((T*)(self)->data)[index])

View File

@ -47,7 +47,7 @@ typedef struct Frame {
} Frame; } Frame;
Frame* Frame__new(const CodeObject* co, Frame* Frame__new(const CodeObject* co,
const py_TValue* module, PyObject* module,
const py_TValue* function, const py_TValue* function,
py_TValue* p0, py_TValue* p0,
py_TValue* locals, py_TValue* locals,

View File

@ -95,6 +95,8 @@ 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_Type pk_function__register();
py_Type pk_nativefunc__register();
py_TValue pk_builtins__register(); py_TValue pk_builtins__register();

View File

@ -7,6 +7,7 @@
#include "pocketpy/common/smallmap.h" #include "pocketpy/common/smallmap.h"
#include "pocketpy/objects/base.h" #include "pocketpy/objects/base.h"
#include "pocketpy/objects/sourcedata.h" #include "pocketpy/objects/sourcedata.h"
#include "pocketpy/objects/namedict.h"
#include "pocketpy/pocketpy.h" #include "pocketpy/pocketpy.h"
#ifdef __cplusplus #ifdef __cplusplus
@ -20,7 +21,6 @@ typedef enum FuncType {
FuncType_UNSET, FuncType_UNSET,
FuncType_NORMAL, FuncType_NORMAL,
FuncType_SIMPLE, FuncType_SIMPLE,
FuncType_EMPTY,
FuncType_GENERATOR, FuncType_GENERATOR,
} FuncType; } FuncType;
@ -39,6 +39,7 @@ typedef enum CodeBlockType {
} CodeBlockType; } CodeBlockType;
typedef enum Opcode { typedef enum Opcode {
#define OPCODE(name) OP_##name, #define OPCODE(name) OP_##name,
#include "pocketpy/xmacros/opcodes.h" #include "pocketpy/xmacros/opcodes.h"
#undef OPCODE #undef OPCODE
@ -121,6 +122,17 @@ void FuncDecl__dtor(FuncDecl* self);
void FuncDecl__add_kwarg(FuncDecl* self, int index, uint16_t key, const py_TValue* value); void FuncDecl__add_kwarg(FuncDecl* self, int index, uint16_t key, const py_TValue* value);
void FuncDecl__gc_mark(const FuncDecl* self); void FuncDecl__gc_mark(const FuncDecl* self);
// runtime function
typedef struct Function {
FuncDecl_ decl;
PyObject* module; // weak ref
PyObject* clazz; // weak ref
pk_NameDict* closure; // strong ref
} Function;
void Function__ctor(Function* self, FuncDecl_ decl, PyObject* module);
void Function__dtor(Function* self);
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif #endif

View File

@ -38,13 +38,7 @@ enum BindType {
BindType_CLASSMETHOD, BindType_CLASSMETHOD,
}; };
enum CompileMode { enum CompileMode { EXEC_MODE, EVAL_MODE, REPL_MODE, JSON_MODE, CELL_MODE };
EXEC_MODE,
EVAL_MODE,
REPL_MODE,
JSON_MODE,
CELL_MODE
};
/************* Global VMs *************/ /************* Global VMs *************/
void py_initialize(); void py_initialize();
@ -294,7 +288,9 @@ bool py_callmethod(py_Ref self, py_Name, int argc, py_Ref argv);
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); bool py_str(py_Ref val);
bool py_repr(py_Ref val);
#define py_repr(val) py_callmagic(__repr__, 1, val)
#define py_len(val) py_callmagic(__len__, 1, val)
/// The return value of the most recent call. /// The return value of the most recent call.
py_GlobalRef py_retval(); py_GlobalRef py_retval();

View File

@ -63,6 +63,14 @@ void* c11_vector__emplace(c11_vector* self){
return p; return p;
} }
bool c11_vector__contains(const c11_vector *self, void *elem){
for(int i = 0; i < self->count; i++){
void* p = (char*)self->data + self->elem_size * i;
if(memcmp(p, elem, self->elem_size) == 0) return true;
}
return false;
}
c11_array c11_vector__submit(c11_vector* self){ c11_array c11_vector__submit(c11_vector* self){
c11_array retval = { c11_array retval = {
.data = self->data, .data = self->data,

View File

@ -4,6 +4,7 @@
#include "pocketpy/objects/sourcedata.h" #include "pocketpy/objects/sourcedata.h"
#include "pocketpy/objects/object.h" #include "pocketpy/objects/object.h"
#include "pocketpy/common/strname.h" #include "pocketpy/common/strname.h"
#include "pocketpy/common/sstream.h"
#include "pocketpy/common/config.h" #include "pocketpy/common/config.h"
#include "pocketpy/common/memorypool.h" #include "pocketpy/common/memorypool.h"
#include <ctype.h> #include <ctype.h>
@ -635,7 +636,6 @@ static void _load_simple_expr(Ctx* ctx, c11_sv expr, int line) {
// name or name.name // name or name.name
bool is_fastpath = false; bool is_fastpath = false;
if(is_identifier(expr)) { if(is_identifier(expr)) {
// ctx->emit_(OP_LOAD_NAME, py_Name(expr.sv()).index, line);
Ctx__emit_(ctx, OP_LOAD_NAME, py_name2(expr), line); Ctx__emit_(ctx, OP_LOAD_NAME, py_name2(expr), line);
is_fastpath = true; is_fastpath = true;
} else { } else {
@ -1639,16 +1639,10 @@ static Error* pop_context(Compiler* self) {
if(is_simple) { if(is_simple) {
func->type = FuncType_SIMPLE; func->type = FuncType_SIMPLE;
} else {
bool is_empty = false;
if(func->code.codes.count == 1) {
Bytecode bc = c11__getitem(Bytecode, &func->code.codes, 0);
if(bc.op == OP_RETURN_VALUE && bc.arg == 1) { is_empty = true; }
}
if(is_empty) func->type = FuncType_EMPTY;
} else
func->type = FuncType_NORMAL; func->type = FuncType_NORMAL;
} }
}
assert(func->type != FuncType_UNSET); assert(func->type != FuncType_UNSET);
} }
@ -2188,6 +2182,291 @@ Error* try_compile_assignment(Compiler* self, bool* is_assign) {
return NULL; return NULL;
} }
static FuncDecl_ push_f_context(Compiler* self, c11_sv name, int* out_index) {
FuncDecl_ decl = FuncDecl__rcnew(self->src, name);
decl->code.start_line = self->i == 0 ? 1 : prev()->line;
decl->nested = name_scope(self) == NAME_LOCAL;
// add_func_decl
Ctx* top_ctx = ctx();
c11_vector__push(FuncDecl_, &top_ctx->co->func_decls, decl);
*out_index = top_ctx->co->func_decls.count - 1;
// push new context
top_ctx = c11_vector__emplace(&self->contexts);
Ctx__ctor(top_ctx, &decl->code, decl, self->contexts.count);
return decl;
}
static Error* read_literal(Compiler* self, py_Ref out) {
Error* err;
advance();
const TokenValue* value = &prev()->value;
bool negated = false;
switch(prev()->type) {
case TK_SUB:
consume(TK_NUM);
value = &prev()->value;
negated = true;
case TK_NUM: {
if(value->index == TokenValue_I64) {
py_newint(out, negated ? -value->_i64 : value->_i64);
} else if(value->index == TokenValue_F64) {
py_newfloat(out, negated ? -value->_f64 : value->_f64);
} else {
return SyntaxError();
}
return NULL;
}
case TK_STR: py_newstr(out, value->_str->data); return NULL;
case TK_TRUE: py_newbool(out, true); return NULL;
case TK_FALSE: py_newbool(out, false); return NULL;
case TK_NONE: py_newnone(out); return NULL;
case TK_DOTDOTDOT: py_newellipsis(out); return NULL;
case TK_LPAREN: {
py_TValue cpnts[4];
int count = 0;
while(true) {
if(count == 4) return SyntaxError("default argument tuple exceeds 4 elements");
check(read_literal(self, &cpnts[count]));
count += 1;
if(curr()->type == TK_RPAREN) break;
consume(TK_COMMA);
if(curr()->type == TK_RPAREN) break;
}
consume(TK_RPAREN);
py_newtuple(out, count);
for(int i = 0; i < count; i++) {
py_tuple__setitem(out, i, &cpnts[i]);
}
return NULL;
}
default: *out = PY_NIL; return NULL;
}
}
static Error* _compile_f_args(Compiler* self, FuncDecl* decl, bool enable_type_hints) {
int state = 0; // 0 for args, 1 for *args, 2 for k=v, 3 for **kwargs
Error* err;
do {
if(state > 3) return SyntaxError();
if(state == 3) return SyntaxError("**kwargs should be the last argument");
match_newlines();
if(match(TK_MUL)) {
if(state < 1)
state = 1;
else
return SyntaxError("*args should be placed before **kwargs");
} else if(match(TK_POW)) {
state = 3;
}
consume(TK_ID);
py_Name name = py_name2(Token__sv(prev()));
// check duplicate argument name
py_Name tmp_name;
c11__foreach(int, &decl->args, j) {
tmp_name = c11__getitem(py_Name, &decl->args, *j);
if(tmp_name == name) return SyntaxError("duplicate argument name");
}
c11__foreach(FuncDeclKwArg, &decl->kwargs, kv) {
tmp_name = c11__getitem(py_Name, &decl->code.varnames, kv->index);
if(tmp_name == name) return SyntaxError("duplicate argument name");
}
if(decl->starred_arg != -1) {
tmp_name = c11__getitem(py_Name, &decl->code.varnames, decl->starred_arg);
if(tmp_name == name) return SyntaxError("duplicate argument name");
}
if(decl->starred_kwarg != -1) {
tmp_name = c11__getitem(py_Name, &decl->code.varnames, decl->starred_kwarg);
if(tmp_name == name) return SyntaxError("duplicate argument name");
}
// eat type hints
if(enable_type_hints && match(TK_COLON)) check(consume_type_hints(self));
if(state == 0 && curr()->type == TK_ASSIGN) state = 2;
int index = Ctx__add_varname(ctx(), name);
switch(state) {
case 0: c11_vector__push(int, &decl->args, index); break;
case 1:
decl->starred_arg = index;
state += 1;
break;
case 2: {
consume(TK_ASSIGN);
py_TValue value;
check(read_literal(self, &value));
if(py_isnil(&value)) return SyntaxError("default argument must be a literal");
FuncDecl__add_kwarg(decl, index, name, &value);
} break;
case 3:
decl->starred_kwarg = index;
state += 1;
break;
}
} while(match(TK_COMMA));
return NULL;
}
static Error* compile_function(Compiler* self, int decorators) {
Error* err;
consume(TK_ID);
c11_sv decl_name = Token__sv(prev());
int decl_index;
FuncDecl_ decl = push_f_context(self, decl_name, &decl_index);
consume(TK_LPAREN);
if(!match(TK_RPAREN)) {
check(_compile_f_args(self, decl, true));
consume(TK_RPAREN);
}
if(match(TK_ARROW)) check(consume_type_hints(self));
check(compile_block_body(self, compile_stmt));
check(pop_context(self));
if(decl->code.codes.count >= 2) {
Bytecode* codes = (Bytecode*)decl->code.codes.data;
if(codes[0].op == OP_LOAD_CONST && codes[1].op == OP_POP_TOP) {
// handle optional docstring
py_TValue* consts = decl->code.consts.data;
py_TValue* c = &consts[codes[0].arg];
if(py_isstr(c)) {
decl->docstring = py_tostr(c);
codes[0].op = OP_NO_OP;
codes[1].op = OP_NO_OP;
}
}
}
Ctx__emit_(ctx(), OP_LOAD_FUNCTION, decl_index, prev()->line);
Ctx__s_emit_decorators(ctx(), decorators);
if(ctx()->is_compiling_class) {
Ctx__emit_(ctx(), OP_STORE_CLASS_ATTR, py_name2(decl_name), prev()->line);
} else {
NameExpr* e = NameExpr__new(prev()->line, py_name2(decl_name), name_scope(self));
vtemit_store((Expr*)e, ctx());
vtdelete((Expr*)e);
}
return NULL;
}
static Error* compile_decorated(Compiler* self) {
Error* err;
int count = 0;
do {
check(EXPR(self));
count += 1;
if(!match_newlines()) return SyntaxError("expected a newline after '@'");
} while(match(TK_DECORATOR));
if(match(TK_CLASS)) {
// check(compile_class(count));
} else {
consume(TK_DEF);
check(compile_function(self, count));
}
return NULL;
}
// import a [as b]
// import a [as b], c [as d]
static Error* compile_normal_import(Compiler* self) {
do {
consume(TK_ID);
c11_sv name = Token__sv(prev());
int index = Ctx__add_const_string(ctx(), name);
Ctx__emit_(ctx(), OP_IMPORT_PATH, index, prev()->line);
if(match(TK_AS)) {
consume(TK_ID);
name = Token__sv(prev());
}
Ctx__emit_store_name(ctx(), name_scope(self), py_name2(name), prev()->line);
} while(match(TK_COMMA));
consume_end_stmt();
return NULL;
}
// from a import b [as c], d [as e]
// from a.b import c [as d]
// from . import a [as b]
// from .a import b [as c]
// from ..a import b [as c]
// from .a.b import c [as d]
// from xxx import *
static Error* compile_from_import(c11_sbuf* buf, Compiler* self) {
int dots = 0;
while(true) {
switch(curr()->type) {
case TK_DOT: dots += 1; break;
case TK_DOTDOT: dots += 2; break;
case TK_DOTDOTDOT: dots += 3; break;
default: goto __EAT_DOTS_END;
}
advance();
}
__EAT_DOTS_END:
for(int i = 0; i < dots; i++) {
c11_sbuf__write_char(buf, '.');
}
if(dots > 0) {
// @id is optional if dots > 0
if(match(TK_ID)) {
c11_sbuf__write_sv(buf, Token__sv(prev()));
while(match(TK_DOT)) {
consume(TK_ID);
c11_sbuf__write_char(buf, '.');
c11_sbuf__write_sv(buf, Token__sv(prev()));
}
}
} else {
// @id is required if dots == 0
consume(TK_ID);
c11_sbuf__write_sv(buf, Token__sv(prev()));
while(match(TK_DOT)) {
consume(TK_ID);
c11_sbuf__write_char(buf, '.');
c11_sbuf__write_sv(buf, Token__sv(prev()));
}
}
c11_string* path = c11_sbuf__submit(buf);
Ctx__emit_(ctx(),
OP_IMPORT_PATH,
Ctx__add_const_string(ctx(), c11_string__sv(path)),
prev()->line);
consume(TK_IMPORT);
if(match(TK_MUL)) {
if(name_scope(self) != NAME_GLOBAL)
return SyntaxError("from <module> import * can only be used in global scope");
// pop the module and import __all__
Ctx__emit_(ctx(), OP_POP_IMPORT_STAR, BC_NOARG, prev()->line);
consume_end_stmt();
return NULL;
}
do {
Ctx__emit_(ctx(), OP_DUP_TOP, BC_NOARG, BC_KEEPLINE);
consume(TK_ID);
c11_sv name = Token__sv(prev());
Ctx__emit_(ctx(), OP_LOAD_ATTR, py_name2(name), prev()->line);
if(match(TK_AS)) {
consume(TK_ID);
name = Token__sv(prev());
}
Ctx__emit_store_name(ctx(), name_scope(self), py_name2(name), prev()->line);
} while(match(TK_COMMA));
Ctx__emit_(ctx(), OP_POP_TOP, BC_NOARG, BC_KEEPLINE);
consume_end_stmt();
return NULL;
}
static Error* compile_try_except(Compiler* self) {
assert(false);
return NULL;
}
static Error* compile_stmt(Compiler* self) { static Error* compile_stmt(Compiler* self) {
Error* err; Error* err;
if(match(TK_CLASS)) { if(match(TK_CLASS)) {
@ -2242,11 +2521,18 @@ static Error* compile_stmt(Compiler* self) {
case TK_IF: check(compile_if_stmt(self)); break; case TK_IF: check(compile_if_stmt(self)); break;
case TK_WHILE: check(compile_while_loop(self)); break; case TK_WHILE: check(compile_while_loop(self)); break;
case TK_FOR: check(compile_for_loop(self)); break; case TK_FOR: check(compile_for_loop(self)); break;
// case TK_IMPORT: check(compile_normal_import()); break; case TK_IMPORT: check(compile_normal_import(self)); break;
// case TK_FROM: check(compile_from_import()); break; case TK_FROM: {
// case TK_DEF: check(compile_function()); break; c11_sbuf buf;
// case TK_DECORATOR: check(compile_decorated()); break; c11_sbuf__ctor(&buf);
// case TK_TRY: check(compile_try_except()); break; err = compile_from_import(&buf, self);
c11_sbuf__dtor(&buf);
if(err) return err;
break;
}
case TK_DEF: check(compile_function(self, 0)); break;
case TK_DECORATOR: check(compile_decorated(self)); break;
case TK_TRY: check(compile_try_except(self)); break;
case TK_PASS: consume_end_stmt(); break; case TK_PASS: consume_end_stmt(); break;
/*************************************************/ /*************************************************/
case TK_ASSERT: { case TK_ASSERT: {

View File

@ -51,10 +51,7 @@ static bool stack_binaryop(pk_VM* self, py_Name op, py_Name rop);
pk_FrameResult res = pk_VM__vectorcall(self, (argc), (kwargc), true); \ pk_FrameResult res = pk_VM__vectorcall(self, (argc), (kwargc), true); \
switch(res) { \ switch(res) { \
case RES_RETURN: PUSH(&self->last_retval); break; \ case RES_RETURN: PUSH(&self->last_retval); break; \
case RES_CALL: \ case RES_CALL: frame = self->top_frame; goto __NEXT_FRAME; \
frame = self->top_frame; \
PUSH(&self->last_retval); \
goto __NEXT_FRAME; \
case RES_ERROR: goto __ERROR; \ case RES_ERROR: goto __ERROR; \
default: c11__unreachedable(); \ default: c11__unreachedable(); \
} \ } \
@ -164,20 +161,17 @@ pk_FrameResult pk_VM__run_top_frame(pk_VM* self) {
/*****************************************/ /*****************************************/
case OP_LOAD_ELLIPSIS: py_newellipsis(SP()++); DISPATCH(); case OP_LOAD_ELLIPSIS: py_newellipsis(SP()++); DISPATCH();
case OP_LOAD_FUNCTION: { case OP_LOAD_FUNCTION: {
// FuncDecl_ decl = c11__getitem(FuncDecl_, &frame->co->func_decls, byte.arg); FuncDecl_ decl = c11__getitem(FuncDecl_, &frame->co->func_decls, byte.arg);
// py_TValue obj; Function* ud = py_newobject(SP(), tp_function, 0, sizeof(Function));
// if(decl->nested) { Function__ctor(ud, decl, frame->module);
// NameDict* captured = frame->_locals.to_namedict(); if(decl->nested) {
// obj = ud->closure = FastLocals__to_namedict(frame->locals, frame->locals_co);
// new_object<Function>(tp_function, decl, frame->_module, nullptr, py_Name name = py_name2(c11_string__sv(decl->code.name));
// captured); // capture itself to allow recursion
// uint16_t name = py_Name__map2(py_Str__sv(&decl->code->name)); pk_NameDict__set(ud->closure, name, *SP());
// captured->set(name, obj); }
// } else { SP()++;
// obj = new_object<Function>(tp_function, decl, frame->_module, nullptr, DISPATCH();
// nullptr);
// }
// PUSH(obj);DISPATCH();
} }
case OP_LOAD_NULL: case OP_LOAD_NULL:
py_newnil(SP()++); py_newnil(SP()++);
@ -699,6 +693,18 @@ pk_FrameResult pk_VM__run_top_frame(pk_VM* self) {
*TOP() = self->last_retval; *TOP() = self->last_retval;
DISPATCH(); DISPATCH();
} }
///////////
case OP_RAISE_ASSERT: {
if(byte.arg) {
if(!py_str(TOP())) goto __ERROR;
POP();
py_exception("AssertionError", "%s", py_tostr(py_retval()));
} else {
py_exception("AssertionError", "");
}
goto __ERROR;
}
default: c11__unreachedable(); default: c11__unreachedable();
} }

View File

@ -6,9 +6,7 @@ void ValueStack__ctor(ValueStack* self) {
self->end = self->begin + PK_VM_STACK_SIZE; self->end = self->begin + PK_VM_STACK_SIZE;
} }
void ValueStack__clear(ValueStack* self) { void ValueStack__clear(ValueStack* self) { self->sp = self->begin; }
self->sp = self->begin;
}
py_TValue* FastLocals__try_get_by_name(py_TValue* locals, const CodeObject* co, py_Name name) { py_TValue* FastLocals__try_get_by_name(py_TValue* locals, const CodeObject* co, py_Name name) {
int index = c11_smallmap_n2i__get(&co->varnames_inv, name, -1); int index = c11_smallmap_n2i__get(&co->varnames_inv, name, -1);
@ -20,9 +18,7 @@ pk_NameDict* FastLocals__to_namedict(py_TValue* locals, const CodeObject* co) {
pk_NameDict* dict = pk_NameDict__new(); pk_NameDict* dict = pk_NameDict__new();
c11__foreach(c11_smallmap_n2i_KV, &co->varnames_inv, entry) { c11__foreach(c11_smallmap_n2i_KV, &co->varnames_inv, entry) {
py_TValue value = locals[entry->value]; py_TValue value = locals[entry->value];
if(!py_isnil(&value)){ if(!py_isnil(&value)) { pk_NameDict__set(dict, entry->key, value); }
pk_NameDict__set(dict, entry->key, value);
}
} }
return dict; return dict;
} }
@ -35,17 +31,20 @@ UnwindTarget* UnwindTarget__new(UnwindTarget* next, int iblock, int offset){
return self; return self;
} }
void UnwindTarget__delete(UnwindTarget* self){ void UnwindTarget__delete(UnwindTarget* self) { free(self); }
free(self);
}
Frame* Frame__new(const CodeObject* co, const py_TValue* module, const py_TValue* function, py_TValue* p0, py_TValue* locals, const CodeObject* locals_co){ Frame* Frame__new(const CodeObject* co,
PyObject* module,
const py_TValue* function,
py_TValue* p0,
py_TValue* locals,
const CodeObject* locals_co) {
static_assert(sizeof(Frame) <= kPoolFrameBlockSize, "!(sizeof(Frame) <= kPoolFrameBlockSize)"); static_assert(sizeof(Frame) <= kPoolFrameBlockSize, "!(sizeof(Frame) <= kPoolFrameBlockSize)");
Frame* self = PoolFrame_alloc(); Frame* self = PoolFrame_alloc();
self->f_back = NULL; self->f_back = NULL;
self->ip = (Bytecode*)co->codes.data - 1; self->ip = (Bytecode*)co->codes.data - 1;
self->co = co; self->co = co;
self->module = module->_obj; self->module = module;
self->function = function ? function->_obj : NULL; self->function = function ? function->_obj : NULL;
self->p0 = p0; self->p0 = p0;
self->locals = locals; self->locals = locals;
@ -82,7 +81,8 @@ int Frame__prepare_jump_exception_handler(Frame* self, ValueStack* _s){
void Frame__prepare_jump_break(Frame* self, ValueStack* _s, int target) { void Frame__prepare_jump_break(Frame* self, ValueStack* _s, int target) {
int iblock = Frame__iblock(self); int iblock = Frame__iblock(self);
if(target >= self->co->codes.count) { if(target >= self->co->codes.count) {
while(iblock >= 0) iblock = Frame__exit_block(self, _s, iblock); while(iblock >= 0)
iblock = Frame__exit_block(self, _s, iblock);
} else { } else {
// BUG (solved) // BUG (solved)
// for i in range(4): // for i in range(4):

View File

@ -56,8 +56,6 @@ 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); }
void pk_VM__ctor(pk_VM* self) { void pk_VM__ctor(pk_VM* self) {
self->top_frame = NULL; self->top_frame = NULL;
@ -106,8 +104,8 @@ void pk_VM__ctor(pk_VM* self) {
validate(tp_range, pk_VM__new_type(self, "range", tp_object, NULL, false)); validate(tp_range, pk_VM__new_type(self, "range", tp_object, NULL, false));
validate(tp_module, pk_VM__new_type(self, "module", tp_object, NULL, false)); validate(tp_module, pk_VM__new_type(self, "module", tp_object, NULL, false));
validate(tp_function, pk_VM__new_type(self, "function", tp_object, NULL, false)); validate(tp_function, pk_function__register());
validate(tp_nativefunc, pk_VM__new_type(self, "nativefunc", tp_object, NULL, false)); validate(tp_nativefunc, pk_nativefunc__register());
validate(tp_bound_method, pk_VM__new_type(self, "bound_method", tp_object, NULL, false)); validate(tp_bound_method, pk_VM__new_type(self, "bound_method", tp_object, NULL, false));
validate(tp_super, pk_VM__new_type(self, "super", tp_object, NULL, false)); validate(tp_super, pk_VM__new_type(self, "super", tp_object, NULL, false));
@ -220,74 +218,64 @@ pk_FrameResult pk_VM__vectorcall(pk_VM* self, uint16_t argc, uint16_t kwargc, bo
// [unbound, self, args..., kwargs...] // [unbound, self, args..., kwargs...]
} }
// PyVar* _base = args.begin();
py_Ref argv = py_isnil(p0 + 1) ? p0 + 2 : p0 + 1; py_Ref argv = py_isnil(p0 + 1) ? p0 + 2 : p0 + 1;
int argc2 = argv - p0;
#if 0 if(p0->type == tp_function) {
if(callable_t == tp_function) {
/*****************_py_call*****************/ /*****************_py_call*****************/
// check stack overflow // check stack overflow
if(self->stack.sp > self->stack.end) { if(self->stack.sp > self->stack.end) {
StackOverflowError(); py_exception("StackOverflowError", "");
return RES_ERROR; return RES_ERROR;
} }
const Function& fn = PK_OBJ_GET(Function, callable); Function* fn = py_touserdata(p0);
const CodeObject* co = fn.decl->code; const CodeObject* co = &fn->decl->code;
switch(fn.decl->type) { switch(fn->decl->type) {
case FuncType_NORMAL: case FuncType_NORMAL:
__prepare_py_call(__vectorcall_buffer, args, kwargs, fn.decl); assert(false);
// copy buffer back to stack // __prepare_py_call(__vectorcall_buffer, args, kwargs, fn.decl);
s_data.reset(_base + co->nlocals); // // copy buffer back to stack
for(int j = 0; j < co->nlocals; j++) // self->stack.sp = argv + co->nlocals;
_base[j] = __vectorcall_buffer[j]; // for(int j = 0; j < co->nlocals; j++)
// argv[j] = self->__vectorcall_buffer[j];
break; break;
case FuncType_SIMPLE: case FuncType_SIMPLE:
if(args.size() != fn.decl->args.count) { if(argc2 != fn->decl->args.count) {
TypeError(pk_format("{} takes {} positional arguments but {} were given", const char* fmt = "%s() takes %d positional arguments but %d were given";
&co->name, TypeError(fmt, co->name, fn->decl->args.count, argc2);
fn.decl->args.count, return RES_ERROR;
args.size()));
} }
if(!kwargs.empty()) { if(kwargc) {
TypeError(pk_format("{} takes no keyword arguments", &co->name)); TypeError("%s() takes no keyword arguments", co->name->data);
return RES_ERROR;
} }
// [callable, <self>, args..., local_vars...] // [callable, <self>, args..., local_vars...]
// ^p0 ^p1 ^_sp // ^p0 ^p1 ^_sp
s_data.reset(_base + co->nlocals); self->stack.sp = argv + co->nlocals;
// initialize local variables to PY_NIL // initialize local variables to PY_NIL
std::memset(p1, 0, (char*)s_data._sp - (char*)p1); memset(p1, 0, (char*)self->stack.sp - (char*)p1);
break; break;
case FuncType_EMPTY:
if(args.size() != fn.decl->args.count) {
TypeError(pk_format("{} takes {} positional arguments but {} were given",
&co->name,
fn.decl->args.count,
args.size()));
}
if(!kwargs.empty()) {
TypeError(pk_format("{} takes no keyword arguments", &co->name));
}
s_data.reset(p0);
return None;
case FuncType_GENERATOR: case FuncType_GENERATOR:
__prepare_py_call(__vectorcall_buffer, args, kwargs, fn.decl); assert(false);
s_data.reset(p0); break;
callstack.emplace(nullptr, co, fn._module, callable.get(), nullptr); // __prepare_py_call(__vectorcall_buffer, args, kwargs, fn.decl);
return __py_generator( // s_data.reset(p0);
callstack.popx(), // callstack.emplace(nullptr, co, fn._module, callable.get(), nullptr);
ArgsView(__vectorcall_buffer, __vectorcall_buffer + co->nlocals)); // return __py_generator(
default: c11__unreachedable() // callstack.popx(),
// ArgsView(__vectorcall_buffer, __vectorcall_buffer + co->nlocals));
default: c11__unreachedable();
}; };
// simple or normal // simple or normal
callstack.emplace(p0, co, fn._module, callable.get(), args.begin()); Frame* frame = Frame__new(co, fn->module, p0, p0, argv, co);
if(op_call) return pkpy_OP_CALL; pk_VM__push_frame(self, frame);
return __run_top_frame(); if(opcall) return RES_CALL;
return pk_VM__run_top_frame(self);
/*****************_py_call*****************/ /*****************_py_call*****************/
} }
#endif
if(p0->type == tp_nativefunc) { if(p0->type == tp_nativefunc) {
// const auto& f = PK_OBJ_GET(NativeFunc, callable); // const auto& f = PK_OBJ_GET(NativeFunc, callable);

View File

@ -92,3 +92,16 @@ void CodeObject__dtor(CodeObject* self) {
} }
c11_vector__dtor(&self->func_decls); c11_vector__dtor(&self->func_decls);
} }
void Function__ctor(Function* self, FuncDecl_ decl, PyObject* module) {
PK_INCREF(decl);
self->decl = decl;
self->module = module;
self->clazz = NULL;
self->closure = NULL;
}
void Function__dtor(Function* self) {
PK_DECREF(self->decl);
if(self->closure) pk_NameDict__delete(self->closure);
}

View File

@ -70,9 +70,36 @@ static bool _py_builtins__exit(int argc, py_Ref argv){
return false; return false;
} }
static bool _py_builtins__len(int argc, py_Ref argv) {
PY_CHECK_ARGC(1);
return py_len(argv);
}
py_TValue pk_builtins__register() { py_TValue pk_builtins__register() {
py_Ref builtins = py_newmodule("builtins", NULL); py_Ref builtins = py_newmodule("builtins", NULL);
py_bindnativefunc(builtins, "repr", _py_builtins__repr); py_bindnativefunc(builtins, "repr", _py_builtins__repr);
py_bindnativefunc(builtins, "exit", _py_builtins__exit); py_bindnativefunc(builtins, "exit", _py_builtins__exit);
py_bindnativefunc(builtins, "len", _py_builtins__len);
return *builtins; return *builtins;
} }
py_Type pk_function__register() {
pk_VM* vm = pk_current_vm;
py_Type type = pk_VM__new_type(vm, "function", tp_object, NULL, false);
pk_TypeInfo* ti = c11__at(pk_TypeInfo, &vm->types, type);
ti->dtor = (void (*)(void*))Function__dtor;
return type;
}
static bool _py_nativefunc__repr(int argc, py_Ref argv) {
PY_CHECK_ARGC(1);
py_newstr(py_retval(), "<nativefunc object>");
return true;
}
py_Type pk_nativefunc__register() {
pk_VM* vm = pk_current_vm;
py_Type type = pk_VM__new_type(vm, "nativefunc", tp_object, NULL, false);
py_bindmagic(type, __repr__, _py_nativefunc__repr);
return type;
}

View File

@ -1,4 +1,5 @@
#include "pocketpy/interpreter/vm.h" #include "pocketpy/interpreter/vm.h"
#include "pocketpy/common/sstream.h"
#include "pocketpy/pocketpy.h" #include "pocketpy/pocketpy.h"
#include <math.h> #include <math.h>
@ -197,9 +198,12 @@ 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]); py_f64 val = py_tofloat(&argv[0]);
char buf[32]; c11_sbuf buf;
int size = snprintf(buf, sizeof(buf), "%f", val); c11_sbuf__ctor(&buf);
py_newstrn(py_retval(), buf, size); c11_sbuf__write_f64(&buf, val, -1);
c11_string* res = c11_sbuf__submit(&buf);
py_newstrn(py_retval(), res->data, res->size);
c11_string__delete(res);
return true; return true;
} }
@ -341,12 +345,19 @@ 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); assert(argc > 0);
int res = py_bool(argv); if(argc == 1){
py_newbool(py_retval(), false);
return true;
}
if(argc == 2){
int res = py_bool(py_arg(1));
if(res == -1) return false; if(res == -1) return false;
py_newbool(py_retval(), res); py_newbool(py_retval(), res);
return true; return true;
} }
return TypeError("bool() takes at most 1 argument");
}
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);

View File

@ -1,4 +1,5 @@
#include "pocketpy/interpreter/vm.h" #include "pocketpy/interpreter/vm.h"
#include "pocketpy/common/sstream.h"
#include "pocketpy/pocketpy.h" #include "pocketpy/pocketpy.h"
static bool _py_object__new__(int argc, py_Ref argv) { static bool _py_object__new__(int argc, py_Ref argv) {
@ -29,9 +30,22 @@ static bool _py_object__ne__(int argc, py_Ref argv) {
return true; return true;
} }
static bool _py_object__repr__(int argc, py_Ref argv) {
PY_CHECK_ARGC(1);
assert(argv->is_ptr);
c11_sbuf buf;
c11_sbuf__ctor(&buf);
pk_sprintf(&buf, "<%t object at %p>", argv->type, argv->_obj);
c11_string* res = c11_sbuf__submit(&buf);
py_newstrn(py_retval(), res->data, res->size);
c11_string__delete(res);
return true;
}
void pk_object__register() { void pk_object__register() {
py_bindmagic(tp_object, __new__, _py_object__new__); py_bindmagic(tp_object, __new__, _py_object__new__);
py_bindmagic(tp_object, __hash__, _py_object__hash__); py_bindmagic(tp_object, __hash__, _py_object__hash__);
py_bindmagic(tp_object, __eq__, _py_object__eq__); py_bindmagic(tp_object, __eq__, _py_object__eq__);
py_bindmagic(tp_object, __ne__, _py_object__ne__); py_bindmagic(tp_object, __ne__, _py_object__ne__);
py_bindmagic(tp_object, __repr__, _py_object__repr__);
} }

View File

@ -145,8 +145,15 @@ static bool _py_str__str__(int argc, py_Ref argv) {
static bool _py_str__repr__(int argc, py_Ref argv) { static bool _py_str__repr__(int argc, py_Ref argv) {
PY_CHECK_ARGC(1); PY_CHECK_ARGC(1);
assert(false); c11_sbuf buf;
return false; c11_sbuf__ctor(&buf);
int size;
const char* data = py_tostrn(&argv[0], &size);
c11_sbuf__write_quoted(&buf, (c11_sv){data, size}, '\'');
c11_string* res = c11_sbuf__submit(&buf);
py_newstrn(py_retval(), res->data, res->size);
c11_string__delete(res);
return true;
} }
static bool _py_str__iter__(int argc, py_Ref argv) { static bool _py_str__iter__(int argc, py_Ref argv) {
@ -294,5 +301,3 @@ bool py_str(py_Ref val) {
if(!tmp) return py_repr(val); if(!tmp) return py_repr(val);
return py_call(tmp, 1, val); return py_call(tmp, 1, val);
} }
bool py_repr(py_Ref val) { return py_callmagic(__repr__, 1, val); }

View File

@ -170,7 +170,7 @@ static bool
// disassemble(&co); // disassemble(&co);
Frame* frame = Frame__new(&co, &vm->main, NULL, vm->stack.sp, vm->stack.sp, &co); Frame* frame = Frame__new(&co, vm->main._obj, NULL, vm->stack.sp, vm->stack.sp, &co);
pk_VM__push_frame(vm, frame); pk_VM__push_frame(vm, frame);
pk_FrameResult res = pk_VM__run_top_frame(vm); pk_FrameResult res = pk_VM__run_top_frame(vm);
CodeObject__dtor(&co); CodeObject__dtor(&co);

View File

@ -36,9 +36,17 @@ int main(int argc, char** argv) {
if(argc == 1) { if(argc == 1) {
printf("pocketpy " PK_VERSION " (" __DATE__ ", " __TIME__ ") "); printf("pocketpy " PK_VERSION " (" __DATE__ ", " __TIME__ ") ");
printf("[%d bit] on %s" "\n", (int)(sizeof(void*) * 8), PY_SYS_PLATFORM_STRING); printf(
printf("https://github.com/pocketpy/pocketpy" "\n"); "[%d bit] on %s"
printf("Type \"exit()\" to exit." "\n"); "\n",
(int)(sizeof(void*) * 8),
PY_SYS_PLATFORM_STRING);
printf(
"https://github.com/pocketpy/pocketpy"
"\n");
printf(
"Type \"exit()\" to exit."
"\n");
while(true) { while(true) {
int size = py_replinput(buf); int size = py_replinput(buf);
@ -49,9 +57,11 @@ int main(int argc, char** argv) {
} }
} else { } else {
char* source = read_file(argv[1]); char* source = read_file(argv[1]);
if(source) {
if(!py_exec(source)) py_printexc(); if(!py_exec(source)) py_printexc();
free(source); free(source);
} }
}
py_finalize(); py_finalize();
return 0; return 0;

View File

@ -1,103 +1,4 @@
# test int literals def f(a, b):
assert 0xffff == 65535 return a+b
assert 0xAAFFFF == 11206655
assert 0x7fffffff == 2147483647
assert -0xffff == -65535
assert -0xAAFFFF == -11206655
assert -0x7fffffff == -2147483647
# test 64-bit
assert 2**60-1 + 546 - 0xfffffffffffff == 1148417904979477026
# test oct literals assert f(1, 2) == 3
assert 0o1234 == 668
assert 0o17777777777 == 2147483647
assert -0o1234 == -668
assert -0o17777777777 == -2147483647
# test binary literals
assert 0b10010 == 18
assert -0b10010 == -18
assert 0b11111111111111111111111111111111 == 4294967295
assert -0b11111 == -31
# test == != >= <= < >
assert -1 == -1
assert -1 != 1
assert -1 >= -1
assert -1 <= -1
assert -1 < 1
assert -1 > -2
# test + - * % ** //
assert -1 + 1 == 0
assert -1 - 1 == -2
assert 4 * -1 == -4
assert 5 % 2 == 1
assert 2 ** 3 == 8
assert 4 // 2 == 2
assert 5 // 2 == 2
# test += -= *= //=
x = 3
x += 1
assert x == 4
x -= 1
assert x == 3
x *= 2
assert x == 6
x //= 2
assert x == 3
# test bit_length
assert (1).bit_length() == 1
assert (2).bit_length() == 2
assert (3).bit_length() == 2
assert (-1).bit_length() == 1
assert (-2).bit_length() == 2
assert (-3).bit_length() == 2
assert (123123123123123).bit_length() == 47
assert (-3123123123).bit_length() == 32
# test int()
assert int() == 0
assert int(True) == 1
assert int(False) == 0
assert int(1) == 1
assert int(1.0) == 1
assert int(1.1) == 1
assert int(1.9) == 1
assert int(-1.9) == -1
assert int(1.5) == 1
assert int(-1.5) == -1
assert int("123") == 123
assert int("0x123", 16) == 291
assert int("0o123", 8) == 83
assert int("-0x123", 16) == -291
assert int("-0o123", 8) == -83
assert int("-123") == -123
assert int("+123") == 123
# test >> << & | ^
assert 12 >> 1 == 6
assert 12 << 1 == 24
assert 12 & 1 == 0
assert 12 | 1 == 13
assert 12 ^ 1 == 13
# test high precision int pow
assert 7**21 == 558545864083284007
assert 2**60 == 1152921504606846976
assert -2**60 == -1152921504606846976
assert 4**13 == 67108864
assert (-4)**13 == -67108864
assert ~3 == -4
assert ~-3 == 2
assert ~0 == -1
# tmp code
assert [1, 2].__len__() == 2

View File

@ -1,7 +1,3 @@
def eq(a, b):
dt = a - b
return dt > -0.001 and dt < 0.001
# test == != >= <= < > # test == != >= <= < >
assert 1.0 == 1.0 assert 1.0 == 1.0
assert 1.0 != 1.1 assert 1.0 != 1.1
@ -10,6 +6,10 @@ assert 1.0 <= 1.0
assert 1.0 < 1.1 assert 1.0 < 1.1
assert 1.1 > 1.0 assert 1.1 > 1.0
def eq(a, b):
dt = a - b
return dt > -0.001 and dt < 0.001
# test + - * ** / # test + - * ** /
assert eq(1.5 + 3, 4.5) assert eq(1.5 + 3, 4.5)
assert eq(1.5 + 3.9, 5.4) assert eq(1.5 + 3.9, 5.4)
@ -53,14 +53,11 @@ assert eq(float("123"), 123.0)
assert eq(float("123.456"), 123.456) assert eq(float("123.456"), 123.456)
import math
inf = float("inf") inf = float("inf")
assert 1/0 == inf assert 1/0 == inf
assert -1/0 == -inf assert -1/0 == -inf
assert 1/inf == 0 assert 1/inf == 0
assert -1/inf == 0 assert -1/inf == 0
assert math.isnan(0/0)
assert 2**-6000 == 0.0 assert 2**-6000 == 0.0
assert 2.0 ** 6000 == inf assert 2.0 ** 6000 == inf
@ -74,7 +71,6 @@ assert eq(2 * .5, 1.0)
assert eq(2 * (.5), 1.0) assert eq(2 * (.5), 1.0)
assert eq(2 * (.5 + 1), 3.0) assert eq(2 * (.5 + 1), 3.0)
assert 1e3 == 1000.0 assert 1e3 == 1000.0
assert 1e-3 == 0.001 assert 1e-3 == 0.001
assert -1e3 == -1000.0 assert -1e3 == -1000.0
@ -83,15 +79,18 @@ assert 1e0 == 1.0
assert 1e-0 == 1.0 assert 1e-0 == 1.0
assert 2e3 == 2000.0 assert 2e3 == 2000.0
assert 2e3j == 2000j
assert -2e-3 == -0.002 assert -2e-3 == -0.002
assert -2e-3j == -0.002j
assert 3.4e-3 == 0.0034 assert 3.4e-3 == 0.0034
assert 3.4e+3 == 3400.0 assert 3.4e+3 == 3400.0
try: # import math
float('-x13') # assert math.isnan(0/0)
exit(1)
except ValueError: # assert 2e3j == 2000j
pass # assert -2e-3j == -0.002j
# try:
# float('-x13')
# exit(1)
# except ValueError:
# pass

View File

@ -12,6 +12,7 @@ assert True or False
assert not False assert not False
assert not (not True) assert not (not True)
assert bool() == False
assert bool(0) == False assert bool(0) == False
assert bool(1) == True assert bool(1) == True
assert bool([]) == False assert bool([]) == False
@ -19,7 +20,14 @@ assert bool("abc") == True
assert bool([1,2]) == True assert bool([1,2]) == True
assert bool('') == False assert bool('') == False
# extra compare for None # is operator
assert None == None assert None == None
assert None is None
assert ... == ... assert ... == ...
assert ... is ...
assert NotImplemented == NotImplemented assert NotImplemented == NotImplemented
assert NotImplemented is NotImplemented
assert True is True
assert False is False

View File

@ -44,7 +44,6 @@ assert s.replace("foo","ball") == "balltball"
assert s.startswith('f') == True;assert s.endswith('o') == False assert s.startswith('f') == True;assert s.endswith('o') == False
assert t.startswith('this') == True; assert t.startswith('this') == True;
assert t.split('w') == ['this is string example....', 'o', '!!!'] assert t.split('w') == ['this is string example....', 'o', '!!!']
assert "a,b,c".split(',') == ['a', 'b', 'c'] assert "a,b,c".split(',') == ['a', 'b', 'c']
assert 'a,'.split(',') == ['a'] assert 'a,'.split(',') == ['a']
@ -110,29 +109,9 @@ num = 6
assert str(num) == '6' assert str(num) == '6'
# test Lo group names # test Lo group names
测试 = "test" 测试 = "test"
assert 测试 == "test" assert 测试 == "test"
assert "Hello, {}!".format("World") == "Hello, World!"
assert "{} {} {}".format("I", "love", "Python") == "I love Python"
assert "{0} {1} {2}".format("I", "love", "Python") == "I love Python"
assert "{2} {1} {0}".format("I", "love", "Python") == "Python love I"
assert "{0}{1}{0}".format("abra", "cad") == "abracadabra"
assert "{k}={v}".format(k="key", v="value") == "key=value"
assert "{k}={k}".format(k="key") == "key=key"
assert "{0}={1}".format('{0}', '{1}') == "{0}={1}"
assert "{{{0}}}".format(1) == "{1}"
assert "{0}{1}{1}".format(1, 2, 3) == "122"
try:
"{0}={1}}".format(1, 2)
exit(1)
except ValueError:
pass
assert "{{{}xxx{}x}}".format(1, 2) == "{1xxx2x}"
assert "{{abc}}".format() == "{abc}"
# 3rd slice # 3rd slice
a = "Hello, World!" a = "Hello, World!"
assert a[::-1] == "!dlroW ,olleH" assert a[::-1] == "!dlroW ,olleH"
@ -152,20 +131,35 @@ assert b[5:2:-2] == [',', 'l']
a = '123' a = '123'
assert a.rjust(5) == ' 123' assert a.rjust(5) == ' 123'
assert a.rjust(5, '0') == '00123' assert a.rjust(5, '0') == '00123'
try:
a.rjust(5, '00')
exit(1)
except TypeError:
pass
assert a.ljust(5) == '123 ' assert a.ljust(5) == '123 '
assert a.ljust(5, '0') == '12300' assert a.ljust(5, '0') == '12300'
try:
a.ljust(5, '00')
exit(1)
except TypeError:
pass
assert '\x30\x31\x32' == '012' assert '\x30\x31\x32' == '012'
assert '\b\b\b' == '\x08\x08\x08'
assert repr('\x1f\x1e\x1f') == '\'\\x1f\\x1e\\x1f\''
assert hex(-42) == '-0x2a'
assert hex(42) == '0x2a'
assert hex(0) == '0x0'
assert hex(1) == '0x1'
assert hex(15) == '0xf'
assert hex(16) == '0x10'
assert hex(255) == '0xff'
assert hex(256) == '0x100'
assert hex(257) == '0x101'
assert hex(17) == '0x11'
a = '123'
assert a.index('2') == 1
assert a.index('1') == 0
assert a.index('3') == 2
assert a.index('2', 1) == 1
assert a.index('1', 0) == 0
assert a.find('1') == 0
assert a.find('1', 1) == -1
a = 'abcd' a = 'abcd'
assert list(a) == ['a', 'b', 'c', 'd'] assert list(a) == ['a', 'b', 'c', 'd']
@ -184,78 +178,27 @@ assert list(a) == ['b']
a = '' a = ''
assert list(a) == [''] assert list(a) == ['']
assert '\b\b\b' == '\x08\x08\x08' # test format()
assert "Hello, {}!".format("World") == "Hello, World!"
assert "{} {} {}".format("I", "love", "Python") == "I love Python"
assert "{0} {1} {2}".format("I", "love", "Python") == "I love Python"
assert "{2} {1} {0}".format("I", "love", "Python") == "Python love I"
assert "{0}{1}{0}".format("abra", "cad") == "abracadabra"
assert "{k}={v}".format(k="key", v="value") == "key=value"
assert "{k}={k}".format(k="key") == "key=key"
assert "{0}={1}".format('{0}', '{1}') == "{0}={1}"
assert "{{{0}}}".format(1) == "{1}"
assert "{0}{1}{1}".format(1, 2, 3) == "122"
# try:
# "{0}={1}}".format(1, 2)
# exit(1)
# except ValueError:
# pass
assert "{{{}xxx{}x}}".format(1, 2) == "{1xxx2x}"
assert "{{abc}}".format() == "{abc}"
# test f-string
stack=[1,2,3,4]; assert f"{stack[2:]}" == '[3, 4]' stack=[1,2,3,4]; assert f"{stack[2:]}" == '[3, 4]'
assert repr('\x1f\x1e\x1f') == '\'\\x1f\\x1e\\x1f\''
assert hex(-42) == '-0x2a'
assert hex(42) == '0x2a'
assert hex(0) == '0x0'
assert hex(1) == '0x1'
assert hex(15) == '0xf'
assert hex(16) == '0x10'
assert hex(255) == '0xff'
assert hex(256) == '0x100'
assert hex(257) == '0x101'
assert hex(17) == '0x11'
import c
assert repr(c.NULL) == '<void* at 0x0>'
assert repr(c.void_p(1)) == '<void* at 0x1>'
assert repr(c.void_p(15)) == '<void* at 0xf>'
assert repr(c.void_p(16)) == '<void* at 0x10>'
assert repr(c.void_p(255)) == '<void* at 0xff>'
assert repr(c.void_p(256)) == '<void* at 0x100>'
assert repr(c.void_p(257)) == '<void* at 0x101>'
assert repr(c.void_p(17)) == '<void* at 0x11>'
# random hex test
import random
def test(__min, __max):
for _ in range(100):
num = random.randint(__min, __max)
hex_num = hex(num)
assert eval(hex_num) == num
if num >= 0:
assert repr(c.void_p(num)) == f'<void* at 0x{hex_num[2:]}>'
test(0, 100)
test(0, 100000)
test(-100, 100)
test(-100000, 100000)
test(-2**30, 2**30)
a = '123'
assert a.index('2') == 1
assert a.index('1') == 0
assert a.index('3') == 2
assert a.index('2', 1) == 1
assert a.index('1', 0) == 0
try:
a.index('1', 1)
exit(1)
except ValueError:
pass
try:
a.index('1', -1)
exit(1)
except ValueError:
pass
assert a.find('1') == 0
assert a.find('1', 1) == -1
try:
a.find('1', -1)
exit(1)
except ValueError:
pass