mirror of
https://github.com/pocketpy/pocketpy
synced 2025-10-24 13:30:18 +00:00
Compare commits
No commits in common. "8264f125d60ef7d4453bd0aea1e6512d4fa8c1e7" and "6a7da5a1d50c9a8516ff4c32aad5a54a19e8b885" have entirely different histories.
8264f125d6
...
6a7da5a1d5
@ -1,5 +1,3 @@
|
||||
set -e
|
||||
|
||||
# if no $1 default arm64-v8a
|
||||
if [ -z $1 ]; then
|
||||
$1=arm64-v8a
|
||||
|
@ -1,5 +1,3 @@
|
||||
set -e
|
||||
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
|
@ -1,10 +1,10 @@
|
||||
set -e
|
||||
|
||||
python prebuild.py
|
||||
|
||||
rm -rf web/lib
|
||||
mkdir web/lib
|
||||
|
||||
SRC=$(find src/ -name "*.c")
|
||||
SRC_C=$(find src/ -name "*.c")
|
||||
SRC_CPP=$(find src/ -name "*.cpp")
|
||||
SRC="$SRC_C $SRC_CPP"
|
||||
|
||||
emcc $SRC -Iinclude/ -s -Os -sEXPORTED_FUNCTIONS=_py_initialize,_py_finalize,_py_exec,_py_replinput -sEXPORTED_RUNTIME_METHODS=ccall -o web/lib/pocketpy.js
|
||||
em++ $SRC -Iinclude/ -fexceptions -frtti -s -Os -sEXPORTED_FUNCTIONS=_pkpy_new_repl,_pkpy_repl_input,_pkpy_new_vm -sEXPORTED_RUNTIME_METHODS=ccall -o web/lib/pocketpy.js
|
||||
|
@ -1,14 +1,20 @@
|
||||
#pragma once
|
||||
// generated by prebuild.py
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
const char* load_kPythonLib(const char* name);
|
||||
|
||||
extern const char kPythonLibs__enum[];
|
||||
extern const char kPythonLibs__long[];
|
||||
extern const char kPythonLibs__set[];
|
||||
extern const char kPythonLibs_bisect[];
|
||||
extern const char kPythonLibs_builtins[];
|
||||
extern const char kPythonLibs_cmath[];
|
||||
extern const char kPythonLibs_collections[];
|
||||
extern const char kPythonLibs_colorsys[];
|
||||
extern const char kPythonLibs_datetime[];
|
||||
extern const char kPythonLibs_functools[];
|
||||
extern const char kPythonLibs_heapq[];
|
||||
@ -17,3 +23,7 @@ extern const char kPythonLibs_operator[];
|
||||
extern const char kPythonLibs_pickle[];
|
||||
extern const char kPythonLibs_this[];
|
||||
extern const char kPythonLibs_typing[];
|
||||
|
||||
#ifdef __cplusplus
|
||||
} // extern "C"
|
||||
#endif
|
||||
|
@ -2,6 +2,10 @@
|
||||
|
||||
#include <stdbool.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#define c11__less(a, b) ((a) < (b))
|
||||
|
||||
#define c11__lower_bound(T, ptr, count, key, less, out_index) \
|
||||
@ -38,3 +42,8 @@ bool c11__stable_sort(void* ptr,
|
||||
int elem_size,
|
||||
int (*f_lt)(const void* a, const void* b, void* extra),
|
||||
void* extra);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
|
||||
#endif
|
@ -1,5 +1,9 @@
|
||||
#pragma once
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#define kPoolExprBlockSize 128
|
||||
#define kPoolFrameBlockSize 80
|
||||
#define kPoolObjectBlockSize 80
|
||||
@ -20,3 +24,8 @@ void PoolObject_dealloc(void* p);
|
||||
void PoolObject_shrink_to_fit();
|
||||
|
||||
void Pools_debug_info(char* buffer, int size);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
|
@ -4,6 +4,10 @@
|
||||
#include "pocketpy/common/str.h"
|
||||
#include <stdint.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#define SMALLMAP_T__HEADER
|
||||
#define K uint16_t
|
||||
#define V int
|
||||
@ -20,3 +24,7 @@
|
||||
#define equal(a, b) (c11_sv__cmp((a), (b)) == 0)
|
||||
#include "pocketpy/xmacros/smallmap.h"
|
||||
#undef SMALLMAP_T__HEADER
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -6,6 +6,10 @@
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct c11_sbuf {
|
||||
c11_vector data;
|
||||
} c11_sbuf;
|
||||
@ -30,3 +34,7 @@ void c11_sbuf__py_submit(c11_sbuf* self, py_Ref out);
|
||||
|
||||
void pk_vsprintf(c11_sbuf* ss, const char* fmt, va_list args);
|
||||
void pk_sprintf(c11_sbuf* ss, const char* fmt, ...);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -4,6 +4,10 @@
|
||||
#include "pocketpy/common/utils.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
/* string */
|
||||
typedef struct c11_string{
|
||||
// int size | char[] | '\0'
|
||||
@ -17,8 +21,6 @@ typedef struct c11_bytes{
|
||||
unsigned char data[]; // flexible array member
|
||||
} c11_bytes;
|
||||
|
||||
bool c11_bytes__eq(c11_bytes* self, c11_bytes* other);
|
||||
|
||||
int c11_sv__cmp(c11_sv self, c11_sv other);
|
||||
int c11_sv__cmp2(c11_sv self, const char* other);
|
||||
|
||||
@ -70,3 +72,7 @@ typedef enum IntParsingResult{
|
||||
} IntParsingResult;
|
||||
|
||||
IntParsingResult c11__parse_uint(c11_sv text, int64_t* out, int base);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -3,5 +3,13 @@
|
||||
#include <stdint.h>
|
||||
#include "pocketpy/common/str.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
void py_Name__initialize();
|
||||
void py_Name__finalize();
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -3,6 +3,10 @@
|
||||
#include "stdio.h"
|
||||
#include "stdlib.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#define PK_REGION(name) 1
|
||||
|
||||
#define PK_SLICE_LOOP(i, start, stop, step) \
|
||||
@ -48,3 +52,7 @@ typedef struct RefCounted {
|
||||
free(obj); \
|
||||
} \
|
||||
} while(0)
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -7,6 +7,10 @@
|
||||
#include <stdbool.h>
|
||||
#include <assert.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct c11_array {
|
||||
void* data;
|
||||
int count;
|
||||
@ -88,3 +92,7 @@ c11_array c11_vector__submit(c11_vector* self);
|
||||
// NOTE: here we do an extra NULL check for it to avoid UB
|
||||
#define c11__foreach(T, self, it) \
|
||||
for(T* it = (self)->data; it && it != (T*)(self)->data + (self)->count; it++)
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -5,4 +5,12 @@
|
||||
#include "pocketpy/objects/sourcedata.h"
|
||||
#include "pocketpy/objects/codeobject.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
Error* pk_compile(SourceData_ src, CodeObject* out);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -6,6 +6,10 @@
|
||||
#include "pocketpy/objects/error.h"
|
||||
#include <stdint.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
extern const char* TokenSymbols[];
|
||||
|
||||
typedef enum TokenIndex{
|
||||
@ -91,3 +95,7 @@ Error* Lexer__process_and_dump(SourceData_ src, c11_string** out_string);
|
||||
void TokenArray__dtor(TokenArray* self);
|
||||
|
||||
#define Token__sv(self) (c11_sv){(self)->start, (self)->length}
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -6,7 +6,10 @@
|
||||
#include "pocketpy/objects/object.h"
|
||||
#include "pocketpy/common/config.h"
|
||||
#include "pocketpy/common/strname.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
py_TValue* FastLocals__try_get_by_name(py_TValue* locals, const CodeObject* co, py_Name name);
|
||||
NameDict* FastLocals__to_namedict(py_TValue* locals, const CodeObject* co);
|
||||
@ -35,7 +38,7 @@ typedef struct Frame {
|
||||
struct Frame* f_back;
|
||||
const Bytecode* ip;
|
||||
const CodeObject* co;
|
||||
py_GlobalRef module;
|
||||
py_TValue module; // weak ref
|
||||
py_StackRef function; // a function object or NULL (global scope)
|
||||
py_StackRef p0; // unwinding base
|
||||
py_StackRef locals; // locals base
|
||||
@ -43,7 +46,7 @@ typedef struct Frame {
|
||||
} Frame;
|
||||
|
||||
Frame* Frame__new(const CodeObject* co,
|
||||
py_GlobalRef module,
|
||||
py_TValue* module,
|
||||
py_StackRef function,
|
||||
py_StackRef p0,
|
||||
py_StackRef locals);
|
||||
@ -63,3 +66,7 @@ int Frame__exit_block(Frame* self, ValueStack*, int);
|
||||
void Frame__gc_mark(Frame* self);
|
||||
UnwindTarget* Frame__find_unwind_target(Frame* self, int iblock);
|
||||
void Frame__set_unwind_target(Frame* self, py_TValue* sp);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -1,6 +1,10 @@
|
||||
#include "pocketpy/objects/object.h"
|
||||
#include "pocketpy/common/config.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct ManagedHeap{
|
||||
c11_vector no_gc;
|
||||
c11_vector gen;
|
||||
@ -24,3 +28,7 @@ PyObject* ManagedHeap__gcnew(ManagedHeap* self, py_Type type, int slots, int uds
|
||||
|
||||
// external implementation
|
||||
void ManagedHeap__mark(ManagedHeap* self);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -1,10 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#include "pocketpy/interpreter/frame.h"
|
||||
|
||||
typedef struct Generator{
|
||||
Frame* frame;
|
||||
int state;
|
||||
} Generator;
|
||||
|
||||
void pk_newgenerator(py_Ref out, Frame* frame, int slots);
|
@ -1,6 +1,14 @@
|
||||
#pragma once
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
void pk__add_module_pkpy();
|
||||
void pk__add_module_os();
|
||||
void pk__add_module_math();
|
||||
void pk__add_module_dis();
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -6,6 +6,10 @@
|
||||
#include "pocketpy/interpreter/frame.h"
|
||||
#include "pocketpy/interpreter/modules.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct py_TypeInfo {
|
||||
py_Name name;
|
||||
py_Type base;
|
||||
@ -65,9 +69,6 @@ bool pk__normalize_index(int* index, int length);
|
||||
void pk_list__mark(void* ud, void (*marker)(py_TValue*));
|
||||
void pk_dict__mark(void* ud, void (*marker)(py_TValue*));
|
||||
|
||||
bool pk_wrapper__self(int argc, py_Ref argv);
|
||||
bool pk_wrapper__NotImplementedError(int argc, py_Ref argv);
|
||||
|
||||
typedef enum FrameResult {
|
||||
RES_RETURN,
|
||||
RES_CALL,
|
||||
@ -125,6 +126,9 @@ py_Type pk_super__register();
|
||||
py_Type pk_property__register();
|
||||
py_Type pk_staticmethod__register();
|
||||
py_Type pk_classmethod__register();
|
||||
py_Type pk_generator__register();
|
||||
|
||||
py_TValue pk_builtins__register();
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -9,6 +9,10 @@
|
||||
#include "pocketpy/common/utils.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct PyObject PyObject;
|
||||
typedef struct VM VM;
|
||||
extern VM* pk_current_vm;
|
||||
@ -31,3 +35,7 @@ typedef struct py_TValue {
|
||||
// 16 bytes to make py_arg() macro work
|
||||
static_assert(sizeof(py_CFunction) <= 8, "sizeof(py_CFunction) > 8");
|
||||
static_assert(sizeof(py_TValue) == 16, "sizeof(py_TValue) != 16");
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -10,6 +10,10 @@
|
||||
#include "pocketpy/objects/namedict.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#define BC_NOARG 0
|
||||
#define BC_KEEPLINE -1
|
||||
|
||||
@ -140,4 +144,8 @@ typedef struct Function {
|
||||
} Function;
|
||||
|
||||
void Function__ctor(Function* self, FuncDecl_ decl, py_TValue* module);
|
||||
void Function__dtor(Function* self);
|
||||
void Function__dtor(Function* self);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -7,6 +7,10 @@
|
||||
#include "pocketpy/objects/object.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct{
|
||||
SourceData_ src;
|
||||
int lineno;
|
||||
@ -16,3 +20,7 @@ typedef struct{
|
||||
void py_BaseException__set_lineno(py_Ref, int lineno, const CodeObject* code);
|
||||
int py_BaseException__get_lineno(py_Ref, const CodeObject* code);
|
||||
void py_BaseException__stpush(py_Ref, SourceData_ src, int lineno, const char* func_name);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -5,6 +5,10 @@
|
||||
#include "pocketpy/objects/base.h"
|
||||
#include <stdint.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#define SMALLMAP_T__HEADER
|
||||
#define K uint16_t
|
||||
#define V py_TValue
|
||||
@ -12,3 +16,6 @@
|
||||
#include "pocketpy/xmacros/smallmap.h"
|
||||
#undef SMALLMAP_T__HEADER
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -3,6 +3,10 @@
|
||||
#include "pocketpy/objects/namedict.h"
|
||||
#include "pocketpy/objects/base.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct PyObject {
|
||||
py_Type type; // we have a duplicated type here for convenience
|
||||
bool gc_is_large;
|
||||
@ -25,3 +29,7 @@ void* PyObject__userdata(PyObject* self);
|
||||
|
||||
PyObject* PyObject__new(py_Type type, int slots, int size);
|
||||
void PyObject__delete(PyObject* self);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
@ -6,6 +6,10 @@
|
||||
#include "pocketpy/common/sstream.h"
|
||||
#include "pocketpy/common/vector.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
struct SourceData {
|
||||
RefCounted rc;
|
||||
enum py_CompileMode mode;
|
||||
@ -34,3 +38,7 @@ void SourceData__snapshot(const struct SourceData* self,
|
||||
int lineno,
|
||||
const char* cursor,
|
||||
const char* name);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -385,7 +385,7 @@ void py_clearexc(py_StackRef p0);
|
||||
#define ValueError(...) py_exception(tp_ValueError, __VA_ARGS__)
|
||||
#define IndexError(...) py_exception(tp_IndexError, __VA_ARGS__)
|
||||
#define ImportError(...) py_exception(tp_ImportError, __VA_ARGS__)
|
||||
#define ZeroDivisionError(...) py_exception(tp_ZeroDivisionError, __VA_ARGS__)
|
||||
#define NotImplementedError() py_exception(tp_NotImplementedError, "")
|
||||
#define AttributeError(self, n) \
|
||||
py_exception(tp_AttributeError, "'%t' object has no attribute '%n'", (self)->type, (n))
|
||||
#define UnboundLocalError(n) \
|
||||
@ -451,6 +451,7 @@ int py_list_len(py_Ref self);
|
||||
void py_list_append(py_Ref self, py_Ref val);
|
||||
void py_list_clear(py_Ref self);
|
||||
void py_list_insert(py_Ref self, int i, py_Ref val);
|
||||
void py_list_reverse(py_Ref self);
|
||||
|
||||
py_TmpRef py_dict_getitem(py_Ref self, py_Ref key) PY_RAISE;
|
||||
void py_dict_setitem(py_Ref self, py_Ref key, py_Ref val) PY_RAISE;
|
||||
@ -516,10 +517,9 @@ enum py_PredefinedTypes {
|
||||
tp_NoneType,
|
||||
tp_NotImplementedType,
|
||||
tp_ellipsis,
|
||||
tp_generator,
|
||||
/* builtin exceptions */
|
||||
tp_StopIteration,
|
||||
tp_SyntaxError,
|
||||
tp_StopIteration,
|
||||
/* builtin exceptions */
|
||||
tp_StackOverflowError,
|
||||
tp_IOError,
|
||||
tp_OSError,
|
||||
|
@ -27,12 +27,21 @@ with open("include/pocketpy/common/_generated.h", "wt", encoding='utf-8', newlin
|
||||
data = '''#pragma once
|
||||
// generated by prebuild.py
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
const char* load_kPythonLib(const char* name);
|
||||
|
||||
'''
|
||||
for key in sorted(sources.keys()):
|
||||
value = sources[key]
|
||||
data += f'extern const char kPythonLibs_{key}[];\n'
|
||||
data += '''
|
||||
#ifdef __cplusplus
|
||||
} // extern "C"
|
||||
#endif
|
||||
'''
|
||||
f.write(data)
|
||||
|
||||
with open("src/common/_generated.c", "wt", encoding='utf-8', newline='\n') as f:
|
||||
|
80
python/_set.py
Normal file
80
python/_set.py
Normal file
@ -0,0 +1,80 @@
|
||||
class set:
|
||||
def __init__(self, iterable=None):
|
||||
iterable = iterable or []
|
||||
self._a = {}
|
||||
self.update(iterable)
|
||||
|
||||
def add(self, elem):
|
||||
self._a[elem] = None
|
||||
|
||||
def discard(self, elem):
|
||||
self._a.pop(elem, None)
|
||||
|
||||
def remove(self, elem):
|
||||
del self._a[elem]
|
||||
|
||||
def clear(self):
|
||||
self._a.clear()
|
||||
|
||||
def update(self, other):
|
||||
for elem in other:
|
||||
self.add(elem)
|
||||
|
||||
def __len__(self):
|
||||
return len(self._a)
|
||||
|
||||
def copy(self):
|
||||
return set(self._a.keys())
|
||||
|
||||
def __and__(self, other):
|
||||
return {elem for elem in self if elem in other}
|
||||
|
||||
def __sub__(self, other):
|
||||
return {elem for elem in self if elem not in other}
|
||||
|
||||
def __or__(self, other):
|
||||
ret = self.copy()
|
||||
ret.update(other)
|
||||
return ret
|
||||
|
||||
def __xor__(self, other):
|
||||
_0 = self - other
|
||||
_1 = other - self
|
||||
return _0 | _1
|
||||
|
||||
def union(self, other):
|
||||
return self | other
|
||||
|
||||
def intersection(self, other):
|
||||
return self & other
|
||||
|
||||
def difference(self, other):
|
||||
return self - other
|
||||
|
||||
def symmetric_difference(self, other):
|
||||
return self ^ other
|
||||
|
||||
def __eq__(self, other):
|
||||
if not isinstance(other, set):
|
||||
return NotImplemented
|
||||
return len(self ^ other) == 0
|
||||
|
||||
def isdisjoint(self, other):
|
||||
return len(self & other) == 0
|
||||
|
||||
def issubset(self, other):
|
||||
return len(self - other) == 0
|
||||
|
||||
def issuperset(self, other):
|
||||
return len(other - self) == 0
|
||||
|
||||
def __contains__(self, elem):
|
||||
return elem in self._a
|
||||
|
||||
def __repr__(self):
|
||||
if len(self) == 0:
|
||||
return 'set()'
|
||||
return '{'+ ', '.join([repr(i) for i in self._a.keys()]) + '}'
|
||||
|
||||
def __iter__(self):
|
||||
return iter(self._a.keys())
|
@ -1,4 +1,4 @@
|
||||
from pkpy import next as __pkpy_next
|
||||
from pkpy import next as __builtins_next
|
||||
|
||||
def all(iterable):
|
||||
for i in iterable:
|
||||
@ -37,8 +37,8 @@ def zip(a, b):
|
||||
a = iter(a)
|
||||
b = iter(b)
|
||||
while True:
|
||||
ai = __pkpy_next(a)
|
||||
bi = __pkpy_next(b)
|
||||
ai = __builtins_next(a)
|
||||
bi = __builtins_next(b)
|
||||
if ai is StopIteration or bi is StopIteration:
|
||||
break
|
||||
yield ai, bi
|
||||
@ -182,83 +182,36 @@ def long(*args, **kwargs):
|
||||
import _long
|
||||
return _long.long(*args, **kwargs)
|
||||
|
||||
class set:
|
||||
def __init__(self, iterable=None):
|
||||
iterable = iterable or []
|
||||
self._a = {}
|
||||
self.update(iterable)
|
||||
|
||||
def add(self, elem):
|
||||
self._a[elem] = None
|
||||
|
||||
def discard(self, elem):
|
||||
self._a.pop(elem, None)
|
||||
# builtin exceptions
|
||||
class StackOverflowError(Exception): pass
|
||||
class IOError(Exception): pass
|
||||
class NotImplementedError(Exception): pass
|
||||
class TypeError(Exception): pass
|
||||
class IndexError(Exception): pass
|
||||
class ValueError(Exception): pass
|
||||
class RuntimeError(Exception): pass
|
||||
class ZeroDivisionError(Exception): pass
|
||||
class NameError(Exception): pass
|
||||
class UnboundLocalError(Exception): pass
|
||||
class AttributeError(Exception): pass
|
||||
class ImportError(Exception): pass
|
||||
class AssertionError(Exception): pass
|
||||
|
||||
def remove(self, elem):
|
||||
del self._a[elem]
|
||||
|
||||
def clear(self):
|
||||
self._a.clear()
|
||||
class KeyError(Exception):
|
||||
def __init__(self, key=...):
|
||||
self.key = key
|
||||
if key is ...:
|
||||
super().__init__()
|
||||
else:
|
||||
super().__init__(repr(key))
|
||||
|
||||
def update(self, other):
|
||||
for elem in other:
|
||||
self.add(elem)
|
||||
|
||||
def __len__(self):
|
||||
return len(self._a)
|
||||
|
||||
def copy(self):
|
||||
return set(self._a.keys())
|
||||
|
||||
def __and__(self, other):
|
||||
return {elem for elem in self if elem in other}
|
||||
|
||||
def __sub__(self, other):
|
||||
return {elem for elem in self if elem not in other}
|
||||
|
||||
def __or__(self, other):
|
||||
ret = self.copy()
|
||||
ret.update(other)
|
||||
return ret
|
||||
|
||||
def __xor__(self, other):
|
||||
_0 = self - other
|
||||
_1 = other - self
|
||||
return _0 | _1
|
||||
|
||||
def union(self, other):
|
||||
return self | other
|
||||
|
||||
def intersection(self, other):
|
||||
return self & other
|
||||
|
||||
def difference(self, other):
|
||||
return self - other
|
||||
|
||||
def symmetric_difference(self, other):
|
||||
return self ^ other
|
||||
|
||||
def __eq__(self, other):
|
||||
if not isinstance(other, set):
|
||||
return NotImplemented
|
||||
return len(self ^ other) == 0
|
||||
|
||||
def isdisjoint(self, other):
|
||||
return len(self & other) == 0
|
||||
|
||||
def issubset(self, other):
|
||||
return len(self - other) == 0
|
||||
|
||||
def issuperset(self, other):
|
||||
return len(other - self) == 0
|
||||
|
||||
def __contains__(self, elem):
|
||||
return elem in self._a
|
||||
def __str__(self):
|
||||
if self.key is ...:
|
||||
return ''
|
||||
return str(self.key)
|
||||
|
||||
def __repr__(self):
|
||||
if len(self) == 0:
|
||||
return 'set()'
|
||||
return '{'+ ', '.join([repr(i) for i in self._a.keys()]) + '}'
|
||||
|
||||
def __iter__(self):
|
||||
return iter(self._a.keys())
|
||||
if self.key is ...:
|
||||
return 'KeyError()'
|
||||
return f'KeyError({self.key!r})'
|
||||
|
171
python/colorsys.py
Normal file
171
python/colorsys.py
Normal file
@ -0,0 +1,171 @@
|
||||
"""Conversion functions between RGB and other color systems.
|
||||
|
||||
This modules provides two functions for each color system ABC:
|
||||
|
||||
rgb_to_abc(r, g, b) --> a, b, c
|
||||
abc_to_rgb(a, b, c) --> r, g, b
|
||||
|
||||
All inputs and outputs are triples of floats in the range [0.0...1.0]
|
||||
(with the exception of I and Q, which covers a slightly larger range).
|
||||
Inputs outside the valid range may cause exceptions or invalid outputs.
|
||||
|
||||
Supported color systems:
|
||||
RGB: Red, Green, Blue components
|
||||
YIQ: Luminance, Chrominance (used by composite video signals)
|
||||
HLS: Hue, Luminance, Saturation
|
||||
HSV: Hue, Saturation, Value
|
||||
"""
|
||||
|
||||
# References:
|
||||
# http://en.wikipedia.org/wiki/YIQ
|
||||
# http://en.wikipedia.org/wiki/HLS_color_space
|
||||
# http://en.wikipedia.org/wiki/HSV_color_space
|
||||
|
||||
__all__ = ["rgb_to_yiq","yiq_to_rgb","rgb_to_hls","hls_to_rgb",
|
||||
"rgb_to_hsv","hsv_to_rgb"]
|
||||
|
||||
# Some floating point constants
|
||||
|
||||
ONE_THIRD = 1.0/3.0
|
||||
ONE_SIXTH = 1.0/6.0
|
||||
TWO_THIRD = 2.0/3.0
|
||||
|
||||
# YIQ: used by composite video signals (linear combinations of RGB)
|
||||
# Y: perceived grey level (0.0 == black, 1.0 == white)
|
||||
# I, Q: color components
|
||||
#
|
||||
# There are a great many versions of the constants used in these formulae.
|
||||
# The ones in this library uses constants from the FCC version of NTSC.
|
||||
|
||||
def rgb_to_yiq(r, g, b):
|
||||
y = 0.30*r + 0.59*g + 0.11*b
|
||||
i = 0.74*(r-y) - 0.27*(b-y)
|
||||
q = 0.48*(r-y) + 0.41*(b-y)
|
||||
return (y, i, q)
|
||||
|
||||
def yiq_to_rgb(y, i, q):
|
||||
# r = y + (0.27*q + 0.41*i) / (0.74*0.41 + 0.27*0.48)
|
||||
# b = y + (0.74*q - 0.48*i) / (0.74*0.41 + 0.27*0.48)
|
||||
# g = y - (0.30*(r-y) + 0.11*(b-y)) / 0.59
|
||||
|
||||
r = y + 0.9468822170900693*i + 0.6235565819861433*q
|
||||
g = y - 0.27478764629897834*i - 0.6356910791873801*q
|
||||
b = y - 1.1085450346420322*i + 1.7090069284064666*q
|
||||
|
||||
if r < 0.0:
|
||||
r = 0.0
|
||||
if g < 0.0:
|
||||
g = 0.0
|
||||
if b < 0.0:
|
||||
b = 0.0
|
||||
if r > 1.0:
|
||||
r = 1.0
|
||||
if g > 1.0:
|
||||
g = 1.0
|
||||
if b > 1.0:
|
||||
b = 1.0
|
||||
return (r, g, b)
|
||||
|
||||
|
||||
# HLS: Hue, Luminance, Saturation
|
||||
# H: position in the spectrum
|
||||
# L: color lightness
|
||||
# S: color saturation
|
||||
|
||||
def rgb_to_hls(r, g, b):
|
||||
maxc = max(r, g, b)
|
||||
minc = min(r, g, b)
|
||||
sumc = (maxc+minc)
|
||||
rangec = (maxc-minc)
|
||||
l = sumc/2.0
|
||||
if minc == maxc:
|
||||
return 0.0, l, 0.0
|
||||
if l <= 0.5:
|
||||
s = rangec / sumc
|
||||
else:
|
||||
s = rangec / (2.0-maxc-minc) # Not always 2.0-sumc: gh-106498.
|
||||
rc = (maxc-r) / rangec
|
||||
gc = (maxc-g) / rangec
|
||||
bc = (maxc-b) / rangec
|
||||
if r == maxc:
|
||||
h = bc-gc
|
||||
elif g == maxc:
|
||||
h = 2.0+rc-bc
|
||||
else:
|
||||
h = 4.0+gc-rc
|
||||
# h = (h/6.0) % 1.0
|
||||
h = h / 6.0
|
||||
h = h - int(h)
|
||||
return h, l, s
|
||||
|
||||
def hls_to_rgb(h, l, s):
|
||||
if s == 0.0:
|
||||
return l, l, l
|
||||
if l <= 0.5:
|
||||
m2 = l * (1.0+s)
|
||||
else:
|
||||
m2 = l+s-(l*s)
|
||||
m1 = 2.0*l - m2
|
||||
return (_v(m1, m2, h+ONE_THIRD), _v(m1, m2, h), _v(m1, m2, h-ONE_THIRD))
|
||||
|
||||
def _v(m1, m2, hue):
|
||||
# hue = hue % 1.0
|
||||
hue = hue - int(hue)
|
||||
if hue < ONE_SIXTH:
|
||||
return m1 + (m2-m1)*hue*6.0
|
||||
if hue < 0.5:
|
||||
return m2
|
||||
if hue < TWO_THIRD:
|
||||
return m1 + (m2-m1)*(TWO_THIRD-hue)*6.0
|
||||
return m1
|
||||
|
||||
|
||||
# HSV: Hue, Saturation, Value
|
||||
# H: position in the spectrum
|
||||
# S: color saturation ("purity")
|
||||
# V: color brightness
|
||||
|
||||
def rgb_to_hsv(r, g, b):
|
||||
maxc = max(r, g, b)
|
||||
minc = min(r, g, b)
|
||||
rangec = (maxc-minc)
|
||||
v = maxc
|
||||
if minc == maxc:
|
||||
return 0.0, 0.0, v
|
||||
s = rangec / maxc
|
||||
rc = (maxc-r) / rangec
|
||||
gc = (maxc-g) / rangec
|
||||
bc = (maxc-b) / rangec
|
||||
if r == maxc:
|
||||
h = bc-gc
|
||||
elif g == maxc:
|
||||
h = 2.0+rc-bc
|
||||
else:
|
||||
h = 4.0+gc-rc
|
||||
# h = (h/6.0) % 1.0
|
||||
h = h / 6.0
|
||||
h = h - int(h)
|
||||
return h, s, v
|
||||
|
||||
def hsv_to_rgb(h, s, v):
|
||||
if s == 0.0:
|
||||
return v, v, v
|
||||
i = int(h*6.0) # XXX assume int() truncates!
|
||||
f = (h*6.0) - i
|
||||
p = v*(1.0 - s)
|
||||
q = v*(1.0 - s*f)
|
||||
t = v*(1.0 - s*(1.0-f))
|
||||
i = i%6
|
||||
if i == 0:
|
||||
return v, t, p
|
||||
if i == 1:
|
||||
return q, v, p
|
||||
if i == 2:
|
||||
return p, v, t
|
||||
if i == 3:
|
||||
return p, q, v
|
||||
if i == 4:
|
||||
return t, p, v
|
||||
if i == 5:
|
||||
return v, p, q
|
||||
# Cannot get here
|
17
run_c_binding_test.sh
Normal file
17
run_c_binding_test.sh
Normal file
@ -0,0 +1,17 @@
|
||||
cd c_bindings
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake ..
|
||||
cmake --build . --config Release
|
||||
|
||||
./test_c_bindings > binding_test_scratch
|
||||
|
||||
echo "checking results (they should be identical)"
|
||||
diff -q -s binding_test_scratch ../test_answers.txt
|
||||
|
||||
if [ $? -eq 1 ]
|
||||
then
|
||||
echo "ERROR: c binding test failed"
|
||||
exit 1
|
||||
fi
|
@ -1,10 +1,10 @@
|
||||
set -e
|
||||
|
||||
python prebuild.py
|
||||
|
||||
SRC=$(find src/ -name "*.c")
|
||||
SRC_C=$(find src/ -name "*.c")
|
||||
SRC_CPP=$(find src/ -name "*.cpp")
|
||||
SRC="$SRC_C $SRC_CPP"
|
||||
|
||||
gcc -pg -Og -std=c11 -Wfatal-errors -o main $SRC src2/main.c -Iinclude
|
||||
g++ -pg -Og -std=c++17 -frtti -Wfatal-errors -o main $SRC src2/main.cpp -Iinclude
|
||||
./main benchmarks/fib.py
|
||||
gprof main gmon.out > gprof.txt
|
||||
rm gmon.out
|
||||
|
@ -1,10 +1,10 @@
|
||||
set -e
|
||||
|
||||
python prebuild.py
|
||||
|
||||
SRC=$(find src/ -name "*.c")
|
||||
SRC_C=$(find src/ -name "*.c")
|
||||
SRC_CPP=$(find src/ -name "*.cpp")
|
||||
SRC="$SRC_C $SRC_CPP"
|
||||
|
||||
clang -std=c11 --coverage -O1 -Wfatal-errors -o main src2/main.c $SRC -Iinclude -DPK_ENABLE_OS=1 -DPK_DEBUG_PRECOMPILED_EXEC=1 -DPK_ENABLE_PROFILER=1
|
||||
clang++ -std=c++17 --coverage -O1 -stdlib=libc++ -frtti -Wfatal-errors -o main src2/main.cpp $SRC -Iinclude -DPK_ENABLE_OS=1 -DPK_DEBUG_PRECOMPILED_EXEC=1 -DPK_ENABLE_PROFILER=1
|
||||
|
||||
python scripts/run_tests.py
|
||||
|
||||
|
43
scripts/build_references.py
Normal file
43
scripts/build_references.py
Normal file
@ -0,0 +1,43 @@
|
||||
import re
|
||||
|
||||
filepath = 'include/pocketpy/vm.h'
|
||||
|
||||
with open(filepath, 'r', encoding='utf-8') as f:
|
||||
lines = f.readlines()
|
||||
|
||||
REGION_PATTERN = re.compile(r'#if PK_REGION\("(.+)"\)')
|
||||
|
||||
current_region = None
|
||||
output = []
|
||||
|
||||
def parse_line(line: str):
|
||||
output.append(line)
|
||||
|
||||
for line in lines:
|
||||
if current_region:
|
||||
if line.startswith('#endif'):
|
||||
current_region = None
|
||||
output.append('```\n\n')
|
||||
else:
|
||||
parse_line(line.strip(' '))
|
||||
else:
|
||||
m = REGION_PATTERN.match(line)
|
||||
if m:
|
||||
current_region = m.group(1)
|
||||
output.append(f'### {current_region}\n')
|
||||
output.append('```cpp\n')
|
||||
|
||||
with open('docs/references.md', 'w', encoding='utf-8') as f:
|
||||
f.write('''---
|
||||
label: References
|
||||
icon: code
|
||||
order: 2
|
||||
---
|
||||
|
||||
This page contains all useful methods of `VM` class.
|
||||
|
||||
''')
|
||||
content = ''.join(output)
|
||||
# replace {...} to ; (multi-line match)
|
||||
content = re.sub(r'\{[^}]+?\}', r';', content, flags=re.DOTALL)
|
||||
f.write(content)
|
@ -10,7 +10,7 @@ def get_all_files(root: str):
|
||||
continue
|
||||
if file.startswith('_'):
|
||||
continue
|
||||
if not file.endswith('.c') and not file.endswith('.h') and not file.endswith('.hpp'):
|
||||
if not file.endswith('.cpp') and not file.endswith('.h') and not file.endswith('.hpp'):
|
||||
continue
|
||||
yield fullpath
|
||||
|
||||
|
File diff suppressed because one or more lines are too long
@ -232,10 +232,6 @@ int c11__byte_index_to_unicode(const char* data, int n) {
|
||||
}
|
||||
|
||||
//////////////
|
||||
bool c11_bytes__eq(c11_bytes* self, c11_bytes* other) {
|
||||
if(self->size != other->size) return false;
|
||||
return memcmp(self->data, other->data, self->size) == 0;
|
||||
}
|
||||
|
||||
int c11_sv__cmp(c11_sv self, c11_sv other) {
|
||||
int res = strncmp(self.data, other.data, c11__min(self.size, other.size));
|
||||
|
@ -134,7 +134,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
case OP_LOAD_FUNCTION: {
|
||||
FuncDecl_ decl = c11__getitem(FuncDecl_, &frame->co->func_decls, byte.arg);
|
||||
Function* ud = py_newobject(SP(), tp_function, 0, sizeof(Function));
|
||||
Function__ctor(ud, decl, frame->module);
|
||||
Function__ctor(ud, decl, &frame->module);
|
||||
if(decl->nested) {
|
||||
ud->closure = FastLocals__to_namedict(frame->locals, frame->co);
|
||||
py_Name name = py_name(decl->code.name->data);
|
||||
@ -173,7 +173,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
PUSH(tmp);
|
||||
DISPATCH();
|
||||
}
|
||||
tmp = py_getdict(frame->module, name);
|
||||
tmp = py_getdict(&frame->module, name);
|
||||
if(tmp != NULL) {
|
||||
PUSH(tmp);
|
||||
DISPATCH();
|
||||
@ -193,7 +193,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
PUSH(tmp);
|
||||
DISPATCH();
|
||||
}
|
||||
tmp = py_getdict(frame->module, name);
|
||||
tmp = py_getdict(&frame->module, name);
|
||||
if(tmp != NULL) {
|
||||
PUSH(tmp);
|
||||
DISPATCH();
|
||||
@ -208,7 +208,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
}
|
||||
case OP_LOAD_GLOBAL: {
|
||||
py_Name name = byte.arg;
|
||||
py_Ref tmp = py_getdict(frame->module, name);
|
||||
py_Ref tmp = py_getdict(&frame->module, name);
|
||||
if(tmp != NULL) {
|
||||
PUSH(tmp);
|
||||
DISPATCH();
|
||||
@ -237,7 +237,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
DISPATCH();
|
||||
}
|
||||
// load global if attribute not found
|
||||
tmp = py_getdict(frame->module, name);
|
||||
tmp = py_getdict(&frame->module, name);
|
||||
if(tmp) {
|
||||
PUSH(tmp);
|
||||
DISPATCH();
|
||||
@ -300,13 +300,13 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
// }
|
||||
}
|
||||
} else {
|
||||
py_setdict(frame->module, name, TOP());
|
||||
py_setdict(&frame->module, name, TOP());
|
||||
}
|
||||
POP();
|
||||
DISPATCH();
|
||||
}
|
||||
case OP_STORE_GLOBAL: {
|
||||
py_setdict(frame->module, byte.arg, TOP());
|
||||
py_setdict(&frame->module, byte.arg, TOP());
|
||||
POP();
|
||||
DISPATCH();
|
||||
}
|
||||
@ -361,7 +361,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
// }
|
||||
}
|
||||
} else {
|
||||
bool ok = py_deldict(frame->module, name);
|
||||
bool ok = py_deldict(&frame->module, name);
|
||||
if(!ok) {
|
||||
NameError(name);
|
||||
goto __ERROR;
|
||||
@ -371,7 +371,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
}
|
||||
case OP_DELETE_GLOBAL: {
|
||||
py_Name name = byte.arg;
|
||||
bool ok = py_deldict(frame->module, name);
|
||||
bool ok = py_deldict(&frame->module, name);
|
||||
if(!ok) {
|
||||
NameError(name);
|
||||
goto __ERROR;
|
||||
@ -701,9 +701,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
DISPATCH();
|
||||
}
|
||||
case OP_YIELD_VALUE: {
|
||||
py_assign(py_retval(), TOP());
|
||||
POP();
|
||||
return RES_YIELD;
|
||||
assert(false);
|
||||
}
|
||||
/////////
|
||||
case OP_LIST_APPEND: {
|
||||
@ -802,7 +800,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
ImportError("cannot import name '%n'", name);
|
||||
goto __ERROR;
|
||||
} else {
|
||||
py_setdict(frame->module, name, value);
|
||||
py_setdict(&frame->module, name, value);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
@ -811,7 +809,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
if(!kv->key) continue;
|
||||
c11_sv name = py_name2sv(kv->key);
|
||||
if(name.size == 0 || name.data[0] == '_') continue;
|
||||
py_setdict(frame->module, kv->key, &kv->value);
|
||||
py_setdict(&frame->module, kv->key, &kv->value);
|
||||
}
|
||||
}
|
||||
POP();
|
||||
@ -857,7 +855,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
}
|
||||
POP();
|
||||
py_Type type =
|
||||
pk_newtype(py_name2str(name), base, frame->module, NULL, true, false);
|
||||
pk_newtype(py_name2str(name), base, &frame->module, NULL, true, false);
|
||||
PUSH(py_tpobject(type));
|
||||
self->__curr_class = TOP();
|
||||
DISPATCH();
|
||||
@ -866,7 +864,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
// [cls or decorated]
|
||||
py_Name name = byte.arg;
|
||||
// set into f_globals
|
||||
py_setdict(frame->module, name, TOP());
|
||||
py_setdict(&frame->module, name, TOP());
|
||||
|
||||
if(py_istype(TOP(), tp_type)) {
|
||||
// call on_end_subclass
|
||||
@ -950,7 +948,7 @@ FrameResult VM__run_top_frame(VM* self) {
|
||||
py_TValue* tmp = c11__at(py_TValue, &frame->co->consts, byte.arg);
|
||||
const char* string = py_tostr(tmp);
|
||||
// TODO: optimize this
|
||||
if(!py_exec(string, "<eval>", EVAL_MODE, frame->module)) goto __ERROR;
|
||||
if(!py_exec(string, "<eval>", EVAL_MODE, &frame->module)) goto __ERROR;
|
||||
PUSH(py_retval());
|
||||
DISPATCH();
|
||||
}
|
||||
|
@ -1,7 +1,6 @@
|
||||
#include "pocketpy/interpreter/frame.h"
|
||||
#include "pocketpy/objects/codeobject.h"
|
||||
#include "pocketpy/objects/object.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
void ValueStack__ctor(ValueStack* self) {
|
||||
self->sp = self->begin;
|
||||
@ -36,7 +35,7 @@ UnwindTarget* UnwindTarget__new(UnwindTarget* next, int iblock, int offset) {
|
||||
void UnwindTarget__delete(UnwindTarget* self) { free(self); }
|
||||
|
||||
Frame* Frame__new(const CodeObject* co,
|
||||
py_GlobalRef module,
|
||||
py_TValue* module,
|
||||
py_StackRef function,
|
||||
py_StackRef p0,
|
||||
py_StackRef locals) {
|
||||
@ -45,7 +44,7 @@ Frame* Frame__new(const CodeObject* co,
|
||||
self->f_back = NULL;
|
||||
self->ip = (Bytecode*)co->codes.data - 1;
|
||||
self->co = co;
|
||||
self->module = module;
|
||||
self->module = *module;
|
||||
self->function = function;
|
||||
self->p0 = p0;
|
||||
self->locals = locals;
|
||||
|
@ -1,22 +0,0 @@
|
||||
#include "pocketpy/interpreter/generator.h"
|
||||
#include "pocketpy/interpreter/frame.h"
|
||||
#include "pocketpy/interpreter/vm.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
|
||||
void pk_newgenerator(py_Ref out, Frame* frame, int slots) {
|
||||
Generator* ud = py_newobject(out, tp_generator, slots, sizeof(Generator));
|
||||
ud->frame = frame;
|
||||
ud->state = 0;
|
||||
}
|
||||
|
||||
static bool generator__next__(int argc, py_Ref argv){
|
||||
return true;
|
||||
}
|
||||
|
||||
py_Type pk_generator__register() {
|
||||
py_Type type = pk_newtype("generator", tp_object, NULL, NULL, false, true);
|
||||
|
||||
py_bindmagic(type, __iter__, pk_wrapper__self);
|
||||
py_bindmagic(type, __next__, generator__next__);
|
||||
return type;
|
||||
}
|
@ -2,7 +2,6 @@
|
||||
#include "pocketpy/common/memorypool.h"
|
||||
#include "pocketpy/common/sstream.h"
|
||||
#include "pocketpy/common/utils.h"
|
||||
#include "pocketpy/interpreter/generator.h"
|
||||
#include "pocketpy/objects/base.h"
|
||||
#include "pocketpy/common/_generated.h"
|
||||
#include "pocketpy/pocketpy.h"
|
||||
@ -127,7 +126,9 @@ void VM__ctor(VM* self) {
|
||||
validate(tp_NotImplementedType,
|
||||
pk_newtype("NotImplementedType", tp_object, NULL, NULL, false, true));
|
||||
validate(tp_ellipsis, pk_newtype("ellipsis", tp_object, NULL, NULL, false, true));
|
||||
validate(tp_generator, pk_generator__register());
|
||||
|
||||
validate(tp_SyntaxError, pk_newtype("SyntaxError", tp_Exception, NULL, NULL, false, true));
|
||||
validate(tp_StopIteration, pk_newtype("StopIteration", tp_Exception, NULL, NULL, false, true));
|
||||
|
||||
self->builtins = pk_builtins__register();
|
||||
|
||||
@ -139,8 +140,6 @@ void VM__ctor(VM* self) {
|
||||
validate(tp_##name, type); \
|
||||
} while(0)
|
||||
|
||||
INJECT_BUILTIN_EXC(StopIteration);
|
||||
INJECT_BUILTIN_EXC(SyntaxError);
|
||||
INJECT_BUILTIN_EXC(StackOverflowError);
|
||||
INJECT_BUILTIN_EXC(IOError);
|
||||
INJECT_BUILTIN_EXC(OSError);
|
||||
@ -161,26 +160,11 @@ void VM__ctor(VM* self) {
|
||||
#undef validate
|
||||
|
||||
/* Setup Public Builtin Types */
|
||||
py_Type public_types[] = {
|
||||
tp_object,
|
||||
tp_type,
|
||||
tp_int,
|
||||
tp_float,
|
||||
tp_bool,
|
||||
tp_str,
|
||||
tp_list,
|
||||
tp_tuple,
|
||||
tp_slice,
|
||||
tp_range,
|
||||
tp_bytes,
|
||||
tp_dict,
|
||||
tp_property,
|
||||
tp_staticmethod,
|
||||
tp_classmethod,
|
||||
tp_super,
|
||||
tp_BaseException,
|
||||
tp_Exception,
|
||||
};
|
||||
py_Type public_types[] = {tp_object, tp_type, tp_int, tp_float,
|
||||
tp_bool, tp_str, tp_list, tp_tuple,
|
||||
tp_slice, tp_range, tp_bytes, tp_dict,
|
||||
tp_property, tp_staticmethod, tp_classmethod, tp_super,
|
||||
tp_BaseException, tp_Exception, tp_StopIteration, tp_SyntaxError};
|
||||
|
||||
for(int i = 0; i < c11__count_array(public_types); i++) {
|
||||
py_TypeInfo* ti = c11__at(py_TypeInfo, &self->types, public_types[i]);
|
||||
@ -197,13 +181,11 @@ void VM__ctor(VM* self) {
|
||||
|
||||
// add python builtins
|
||||
do {
|
||||
bool ok;
|
||||
ok = py_exec(kPythonLibs_builtins, "<builtins>", EXEC_MODE, &self->builtins);
|
||||
if(!ok) goto __ABORT;
|
||||
break;
|
||||
__ABORT:
|
||||
py_printexc();
|
||||
c11__abort("failed to load python builtins!");
|
||||
bool ok = py_exec(kPythonLibs__set, "<builtins>", EXEC_MODE, &self->builtins);
|
||||
if(!ok) {
|
||||
py_printexc();
|
||||
c11__abort("failed to load python builtins!");
|
||||
}
|
||||
} while(0);
|
||||
|
||||
self->main = *py_newmodule("__main__");
|
||||
@ -453,14 +435,9 @@ FrameResult VM__vectorcall(VM* self, uint16_t argc, uint16_t kwargc, bool opcall
|
||||
// submit the call
|
||||
VM__push_frame(self, Frame__new(co, &fn->module, p0, p0, argv));
|
||||
return opcall ? RES_CALL : VM__run_top_frame(self);
|
||||
case FuncType_GENERATOR: {
|
||||
bool ok = prepare_py_call(self->__vectorcall_buffer, argv, p1, kwargc, fn->decl);
|
||||
if(!ok) return RES_ERROR;
|
||||
Frame* frame = Frame__new(co, &fn->module, p0, p0, argv);
|
||||
pk_newgenerator(py_retval(), frame, 0);
|
||||
self->stack.sp = p0;
|
||||
return RES_RETURN;
|
||||
}
|
||||
case FuncType_GENERATOR:
|
||||
assert(false);
|
||||
break;
|
||||
// prepare_py_call(__vectorcall_buffer, args, kwargs, fn.decl);
|
||||
// s_data.reset(p0);
|
||||
// callstack.emplace(nullptr, co, fn._module, callable.get(), nullptr);
|
||||
@ -581,7 +558,7 @@ void ManagedHeap__mark(ManagedHeap* self) {
|
||||
}
|
||||
// mark frame
|
||||
for(Frame* frame = vm->top_frame; frame; frame = frame->f_back) {
|
||||
mark_value(frame->module);
|
||||
mark_value(&frame->module);
|
||||
}
|
||||
// mark vm's registers
|
||||
mark_value(&vm->last_retval);
|
||||
@ -647,14 +624,4 @@ void pk_print_stack(VM* self, Frame* frame, Bytecode byte) {
|
||||
byte.arg,
|
||||
stack_str->data);
|
||||
c11_string__delete(stack_str);
|
||||
}
|
||||
|
||||
bool pk_wrapper__self(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
py_assign(py_retval(), argv);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool pk_wrapper__NotImplementedError(int argc, py_Ref argv){
|
||||
return py_exception(tp_NotImplementedError, "");
|
||||
}
|
@ -68,7 +68,7 @@ int py_import(const char* path_cstr) {
|
||||
c11_sv top_filename = c11_string__sv(vm->top_frame->co->src->filename);
|
||||
int is_init = c11_sv__endswith(top_filename, (c11_sv){"__init__.py", 11});
|
||||
|
||||
py_Ref package = py_getdict(vm->top_frame->module, __path__);
|
||||
py_Ref package = py_getdict(&vm->top_frame->module, __path__);
|
||||
c11_sv package_sv = py_tosv(package);
|
||||
if(package_sv.size == 0) {
|
||||
return ImportError("attempted relative import with no known parent package");
|
||||
@ -170,6 +170,14 @@ static bool builtins_len(int argc, py_Ref argv) {
|
||||
return py_len(argv);
|
||||
}
|
||||
|
||||
static bool builtins_reversed(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
// convert _0 to list object
|
||||
if(!py_tpcall(tp_list, 1, argv)) return false;
|
||||
py_list_reverse(py_retval());
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool builtins_hex(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
PY_CHECK_ARG_TYPE(0, tp_int);
|
||||
@ -213,6 +221,23 @@ static bool builtins_next(int argc, py_Ref argv) {
|
||||
return py_exception(tp_StopIteration, "");
|
||||
}
|
||||
|
||||
static bool builtins_sorted(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(3);
|
||||
// convert _0 to list object
|
||||
if(!py_tpcall(tp_list, 1, py_arg(0))) return false;
|
||||
py_push(py_retval()); // duptop
|
||||
py_push(py_retval()); // [| <list>]
|
||||
bool ok = py_pushmethod(py_name("sort")); // [| list.sort, <list>]
|
||||
if(!ok) return false;
|
||||
py_push(py_arg(1)); // [| list.sort, <list>, key]
|
||||
py_push(py_arg(2)); // [| list.sort, <list>, key, reverse]
|
||||
ok = py_vectorcall(2, 0); // [| ]
|
||||
if(!ok) return false;
|
||||
py_assign(py_retval(), py_peek(-1));
|
||||
py_pop();
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool builtins_hash(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
py_i64 val;
|
||||
@ -226,9 +251,41 @@ static bool builtins_abs(int argc, py_Ref argv) {
|
||||
return pk_callmagic(__abs__, 1, argv);
|
||||
}
|
||||
|
||||
static bool builtins_divmod(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
return pk_callmagic(__divmod__, 2, argv);
|
||||
static bool builtins_sum(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
|
||||
if(!py_iter(py_arg(0))) return false;
|
||||
py_push(py_retval()); // iter
|
||||
|
||||
py_i64 total_i64 = 0;
|
||||
py_f64 total_f64 = 0.0;
|
||||
bool is_float = false;
|
||||
while(true) {
|
||||
int res = py_next(py_peek(-1));
|
||||
if(res == -1) {
|
||||
py_pop();
|
||||
return false;
|
||||
}
|
||||
if(res == 0) break;
|
||||
|
||||
py_Ref item = py_retval();
|
||||
switch(item->type) {
|
||||
case tp_int: total_i64 += item->_i64; break;
|
||||
case tp_float:
|
||||
is_float = true;
|
||||
total_f64 += item->_f64;
|
||||
break;
|
||||
default: return TypeError("sum() expects an iterable of numbers");
|
||||
}
|
||||
}
|
||||
|
||||
if(is_float) {
|
||||
py_newfloat(py_retval(), total_f64 + total_i64);
|
||||
} else {
|
||||
py_newint(py_retval(), total_i64);
|
||||
}
|
||||
py_pop();
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool builtins_print(int argc, py_Ref argv) {
|
||||
@ -252,18 +309,23 @@ static bool builtins_print(int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool NoneType__repr__(int argc, py_Ref argv) {
|
||||
py_newstr(py_retval(), "None");
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool builtins_exec(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
PY_CHECK_ARG_TYPE(0, tp_str);
|
||||
Frame* frame = pk_current_vm->top_frame;
|
||||
return py_exec(py_tostr(argv), "<exec>", EXEC_MODE, frame->module);
|
||||
return py_exec(py_tostr(argv), "<exec>", EXEC_MODE, &frame->module);
|
||||
}
|
||||
|
||||
static bool builtins_eval(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
PY_CHECK_ARG_TYPE(0, tp_str);
|
||||
Frame* frame = pk_current_vm->top_frame;
|
||||
return py_exec(py_tostr(argv), "<eval>", EVAL_MODE, frame->module);
|
||||
return py_exec(py_tostr(argv), "<eval>", EVAL_MODE, &frame->module);
|
||||
}
|
||||
|
||||
static bool builtins_isinstance(int argc, py_Ref argv) {
|
||||
@ -347,57 +409,24 @@ static bool builtins_delattr(int argc, py_Ref argv) {
|
||||
return py_delattr(py_arg(0), name);
|
||||
}
|
||||
|
||||
static bool builtins_chr(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
PY_CHECK_ARG_TYPE(0, tp_int);
|
||||
py_i64 val = py_toint(py_arg(0));
|
||||
if(val < 0 || val > 128) { return ValueError("chr() arg not in range(128)"); }
|
||||
py_newstrn(py_retval(), (const char*)&val, 1);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool builtins_ord(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
PY_CHECK_ARG_TYPE(0, tp_str);
|
||||
c11_sv sv = py_tosv(py_arg(0));
|
||||
if(sv.size != 1) {
|
||||
return TypeError("ord() expected a character, but string of length %d found", sv.size);
|
||||
}
|
||||
py_newint(py_retval(), sv.data[0]);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool NoneType__repr__(int argc, py_Ref argv) {
|
||||
py_newstr(py_retval(), "None");
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool ellipsis__repr__(int argc, py_Ref argv) {
|
||||
py_newstr(py_retval(), "Ellipsis");
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool NotImplementedType__repr__(int argc, py_Ref argv) {
|
||||
py_newstr(py_retval(), "NotImplemented");
|
||||
return true;
|
||||
}
|
||||
|
||||
py_TValue pk_builtins__register() {
|
||||
py_Ref builtins = py_newmodule("builtins");
|
||||
py_bindfunc(builtins, "repr", builtins_repr);
|
||||
py_bindfunc(builtins, "exit", builtins_exit);
|
||||
py_bindfunc(builtins, "len", builtins_len);
|
||||
py_bindfunc(builtins, "reversed", builtins_reversed);
|
||||
py_bindfunc(builtins, "hex", builtins_hex);
|
||||
py_bindfunc(builtins, "iter", builtins_iter);
|
||||
py_bindfunc(builtins, "next", builtins_next);
|
||||
py_bindfunc(builtins, "hash", builtins_hash);
|
||||
py_bindfunc(builtins, "abs", builtins_abs);
|
||||
py_bindfunc(builtins, "divmod", builtins_divmod);
|
||||
py_bindfunc(builtins, "sum", builtins_sum);
|
||||
|
||||
py_bindfunc(builtins, "exec", builtins_exec);
|
||||
py_bindfunc(builtins, "eval", builtins_eval);
|
||||
|
||||
py_bind(builtins, "print(*args, sep=' ', end='\\n')", builtins_print);
|
||||
py_bind(builtins, "sorted(iterable, key=None, reverse=False)", builtins_sorted);
|
||||
|
||||
py_bindfunc(builtins, "isinstance", builtins_isinstance);
|
||||
py_bindfunc(builtins, "issubclass", builtins_issubclass);
|
||||
@ -407,13 +436,8 @@ py_TValue pk_builtins__register() {
|
||||
py_bindfunc(builtins, "hasattr", builtins_hasattr);
|
||||
py_bindfunc(builtins, "delattr", builtins_delattr);
|
||||
|
||||
py_bindfunc(builtins, "chr", builtins_chr);
|
||||
py_bindfunc(builtins, "ord", builtins_ord);
|
||||
|
||||
// some patches
|
||||
// None __repr__
|
||||
py_bindmagic(tp_NoneType, __repr__, NoneType__repr__);
|
||||
py_bindmagic(tp_ellipsis, __repr__, ellipsis__repr__);
|
||||
py_bindmagic(tp_NotImplementedType, __repr__, NotImplementedType__repr__);
|
||||
return *builtins;
|
||||
}
|
||||
|
||||
|
@ -457,6 +457,12 @@ py_Type pk_dict__register() {
|
||||
}
|
||||
|
||||
//////////////////////////
|
||||
static bool dict_items__iter__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
*py_retval() = *argv;
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool dict_items__next__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
DictIterator* iter = py_touserdata(py_arg(0));
|
||||
@ -472,7 +478,7 @@ static bool dict_items__next__(int argc, py_Ref argv) {
|
||||
|
||||
py_Type pk_dict_items__register() {
|
||||
py_Type type = pk_newtype("dict_items", tp_object, NULL, NULL, false, true);
|
||||
py_bindmagic(type, __iter__, pk_wrapper__self);
|
||||
py_bindmagic(type, __iter__, dict_items__iter__);
|
||||
py_bindmagic(type, __next__, dict_items__next__);
|
||||
return type;
|
||||
}
|
||||
|
@ -59,6 +59,11 @@ void py_list_insert(py_Ref self, int i, py_Ref val) {
|
||||
c11_vector__insert(py_TValue, userdata, i, *val);
|
||||
}
|
||||
|
||||
void py_list_reverse(py_Ref self) {
|
||||
List* userdata = py_touserdata(self);
|
||||
c11__reverse(py_TValue, userdata);
|
||||
}
|
||||
|
||||
////////////////////////////////
|
||||
static bool list__len__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
|
@ -36,6 +36,10 @@ py_Type pk_classmethod__register(){
|
||||
}
|
||||
|
||||
/* boundmethod */
|
||||
static bool boundmethod__new__(int argc, py_Ref argv) {
|
||||
return NotImplementedError();
|
||||
}
|
||||
|
||||
static bool boundmethod__self__getter(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
py_assign(py_retval(), py_getslot(argv, 0));
|
||||
@ -51,6 +55,7 @@ static bool boundmethod__func__getter(int argc, py_Ref argv) {
|
||||
py_Type pk_boundmethod__register(){
|
||||
py_Type type = pk_newtype("boundmethod", tp_object, NULL, NULL, false, true);
|
||||
|
||||
py_bindmagic(type, __new__, boundmethod__new__);
|
||||
py_bindproperty(type, "__self__", boundmethod__self__getter, NULL);
|
||||
py_bindproperty(type, "__func__", boundmethod__func__getter, NULL);
|
||||
return type;
|
||||
|
@ -83,6 +83,8 @@ static bool float__truediv__(int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
#define ZeroDivisionError(msg) false
|
||||
|
||||
static bool number__pow__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
if(py_isint(&argv[0]) && py_isint(&argv[1])) {
|
||||
@ -122,7 +124,7 @@ static bool int__floordiv__(int argc, py_Ref argv) {
|
||||
py_i64 lhs = py_toint(&argv[0]);
|
||||
if(py_isint(&argv[1])) {
|
||||
py_i64 rhs = py_toint(&argv[1]);
|
||||
if(rhs == 0) return ZeroDivisionError("integer division or modulo by zero");
|
||||
if(rhs == 0) return -1;
|
||||
py_newint(py_retval(), lhs / rhs);
|
||||
} else {
|
||||
py_newnotimplemented(py_retval());
|
||||
@ -143,19 +145,6 @@ static bool int__mod__(int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool int__divmod__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
PY_CHECK_ARG_TYPE(1, tp_int);
|
||||
py_i64 lhs = py_toint(&argv[0]);
|
||||
py_i64 rhs = py_toint(&argv[1]);
|
||||
if(rhs == 0) return ZeroDivisionError("integer division or modulo by zero");
|
||||
py_newtuple(py_retval(), 2);
|
||||
ldiv_t res = ldiv(lhs, rhs);
|
||||
py_newint(py_getslot(py_retval(), 0), res.quot);
|
||||
py_newint(py_getslot(py_retval(), 1), res.rem);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool int__invert__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
py_i64 val = py_toint(&argv[0]);
|
||||
@ -470,10 +459,9 @@ void pk_number__register() {
|
||||
py_bindmagic(tp_int, __pow__, number__pow__);
|
||||
py_bindmagic(tp_float, __pow__, number__pow__);
|
||||
|
||||
// __floordiv__ & __mod__ & __divmod__
|
||||
// __floordiv__ & __mod__
|
||||
py_bindmagic(tp_int, __floordiv__, int__floordiv__);
|
||||
py_bindmagic(tp_int, __mod__, int__mod__);
|
||||
py_bindmagic(tp_int, __divmod__, int__divmod__);
|
||||
|
||||
// int.__invert__ & int.<BITWISE OP>
|
||||
py_bindmagic(tp_int, __invert__, int__invert__);
|
||||
|
@ -68,6 +68,12 @@ static bool range_iterator__new__(int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool range_iterator__iter__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
*py_retval() = *py_arg(0);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool range_iterator__next__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
RangeIterator* ud = py_touserdata(py_arg(0));
|
||||
@ -85,7 +91,7 @@ py_Type pk_range_iterator__register() {
|
||||
py_Type type = pk_newtype("range_iterator", tp_object, NULL, NULL, false, true);
|
||||
|
||||
py_bindmagic(type, __new__, range_iterator__new__);
|
||||
py_bindmagic(type, __iter__, pk_wrapper__self);
|
||||
py_bindmagic(type, __iter__, range_iterator__iter__);
|
||||
py_bindmagic(type, __next__, range_iterator__next__);
|
||||
return type;
|
||||
}
|
@ -190,7 +190,7 @@ static bool str__getitem__(int argc, py_Ref argv) {
|
||||
}
|
||||
|
||||
#define DEF_STR_CMP_OP(op, __f, __cond) \
|
||||
static bool str##op(int argc, py_Ref argv) { \
|
||||
static bool str##op(int argc, py_Ref argv) { \
|
||||
PY_CHECK_ARGC(2); \
|
||||
c11_string* self = py_touserdata(&argv[0]); \
|
||||
if(py_arg(1)->type != tp_str) { \
|
||||
@ -355,11 +355,17 @@ static bool str__strip_impl(bool left, bool right, int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool str_strip(int argc, py_Ref argv) { return str__strip_impl(true, true, argc, argv); }
|
||||
static bool str_strip(int argc, py_Ref argv) {
|
||||
return str__strip_impl(true, true, argc, argv);
|
||||
}
|
||||
|
||||
static bool str_lstrip(int argc, py_Ref argv) { return str__strip_impl(true, false, argc, argv); }
|
||||
static bool str_lstrip(int argc, py_Ref argv) {
|
||||
return str__strip_impl(true, false, argc, argv);
|
||||
}
|
||||
|
||||
static bool str_rstrip(int argc, py_Ref argv) { return str__strip_impl(false, true, argc, argv); }
|
||||
static bool str_rstrip(int argc, py_Ref argv) {
|
||||
return str__strip_impl(false, true, argc, argv);
|
||||
}
|
||||
|
||||
static bool str_zfill(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
@ -417,9 +423,13 @@ static bool str__widthjust_impl(bool left, int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool str_ljust(int argc, py_Ref argv) { return str__widthjust_impl(true, argc, argv); }
|
||||
static bool str_ljust(int argc, py_Ref argv) {
|
||||
return str__widthjust_impl(true, argc, argv);
|
||||
}
|
||||
|
||||
static bool str_rjust(int argc, py_Ref argv) { return str__widthjust_impl(false, argc, argv); }
|
||||
static bool str_rjust(int argc, py_Ref argv) {
|
||||
return str__widthjust_impl(false, argc, argv);
|
||||
}
|
||||
|
||||
static bool str_find(int argc, py_Ref argv) {
|
||||
if(argc > 3) return TypeError("find() takes at most 3 arguments");
|
||||
@ -443,15 +453,6 @@ static bool str_index(int argc, py_Ref argv) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool str_encode(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
int size;
|
||||
const char* data = py_tostrn(argv, &size);
|
||||
unsigned char* p = py_newbytes(py_retval(), size);
|
||||
memcpy(p, data, size);
|
||||
return true;
|
||||
}
|
||||
|
||||
py_Type pk_str__register() {
|
||||
py_Type type = pk_newtype("str", tp_object, NULL, NULL, false, true);
|
||||
// no need to dtor because the memory is controlled by the object
|
||||
@ -491,10 +492,15 @@ py_Type pk_str__register() {
|
||||
py_bindmethod(tp_str, "rjust", str_rjust);
|
||||
py_bindmethod(tp_str, "find", str_find);
|
||||
py_bindmethod(tp_str, "index", str_index);
|
||||
py_bindmethod(tp_str, "encode", str_encode);
|
||||
return type;
|
||||
}
|
||||
|
||||
static bool str_iterator__iter__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
*py_retval() = argv[0];
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool str_iterator__next__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
int* ud = py_touserdata(&argv[0]);
|
||||
@ -511,127 +517,14 @@ static bool str_iterator__next__(int argc, py_Ref argv) {
|
||||
py_Type pk_str_iterator__register() {
|
||||
py_Type type = pk_newtype("str_iterator", tp_object, NULL, NULL, false, true);
|
||||
|
||||
py_bindmagic(type, __iter__, pk_wrapper__self);
|
||||
py_bindmagic(type, __iter__, str_iterator__iter__);
|
||||
py_bindmagic(type, __next__, str_iterator__next__);
|
||||
return type;
|
||||
}
|
||||
|
||||
static bool bytes__new__(int argc, py_Ref argv){
|
||||
if(argc == 1){
|
||||
py_newbytes(py_retval(), 0);
|
||||
return true;
|
||||
}
|
||||
if(argc > 2) return TypeError("bytes() takes at most 1 argument");
|
||||
int length;
|
||||
py_TValue* p = pk_arrayview(&argv[1], &length);
|
||||
if(!p) return TypeError("bytes() argument must be a list or tuple");
|
||||
unsigned char* data = py_newbytes(py_retval(), length);
|
||||
for(int i = 0; i < length; i++){
|
||||
if(!py_checktype(&p[i], tp_int)) return false;
|
||||
py_i64 v = py_toint(&p[i]);
|
||||
if(v < 0 || v > 255) return ValueError("bytes must be in range(0, 256)");
|
||||
data[i] = v;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool bytes__repr__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
c11_bytes* self = py_touserdata(&argv[0]);
|
||||
c11_sbuf buf;
|
||||
c11_sbuf__ctor(&buf);
|
||||
c11_sbuf__write_char(&buf, 'b');
|
||||
c11_sbuf__write_quoted(&buf, (c11_sv){(const char*)self->data, self->size}, '\'');
|
||||
c11_sbuf__py_submit(&buf, py_retval());
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool bytes__getitem__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
int size;
|
||||
unsigned char* data = py_tobytes(&argv[0], &size);
|
||||
py_Ref _1 = py_arg(1);
|
||||
if(_1->type == tp_int) {
|
||||
int index = py_toint(_1);
|
||||
if(!pk__normalize_index(&index, size)) return false;
|
||||
py_newint(py_retval(), data[index]);
|
||||
return true;
|
||||
} else if(_1->type == tp_slice) {
|
||||
int start, stop, step;
|
||||
bool ok = pk__parse_int_slice(_1, size, &start, &stop, &step);
|
||||
if(!ok) return false;
|
||||
c11_vector res;
|
||||
c11_vector__ctor(&res, sizeof(unsigned char));
|
||||
for(int i = start; step > 0 ? i < stop : i > stop; i += step) {
|
||||
c11_vector__push(unsigned char, &res, data[i]);
|
||||
}
|
||||
unsigned char* p = py_newbytes(py_retval(), res.count);
|
||||
memcpy(p, res.data, res.count);
|
||||
c11_vector__dtor(&res);
|
||||
return true;
|
||||
} else {
|
||||
return TypeError("bytes indices must be integers");
|
||||
}
|
||||
}
|
||||
|
||||
static bool bytes__eq__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
c11_bytes* self = py_touserdata(&argv[0]);
|
||||
if(!py_istype(&argv[1], tp_bytes)) {
|
||||
py_newnotimplemented(py_retval());
|
||||
} else {
|
||||
c11_bytes* other = py_touserdata(&argv[1]);
|
||||
py_newbool(py_retval(), c11_bytes__eq(self, other));
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool bytes__ne__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
c11_bytes* self = py_touserdata(&argv[0]);
|
||||
if(!py_istype(&argv[1], tp_bytes)) {
|
||||
py_newnotimplemented(py_retval());
|
||||
} else {
|
||||
c11_bytes* other = py_touserdata(&argv[1]);
|
||||
py_newbool(py_retval(), !c11_bytes__eq(self, other));
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool bytes__add__(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(2);
|
||||
c11_bytes* self = py_touserdata(&argv[0]);
|
||||
if(py_arg(1)->type != tp_bytes) {
|
||||
py_newnotimplemented(py_retval());
|
||||
} else {
|
||||
c11_bytes* other = py_touserdata(&argv[1]);
|
||||
unsigned char* p = py_newbytes(py_retval(), self->size + other->size);
|
||||
memcpy(p, self->data, self->size);
|
||||
memcpy(p + self->size, other->data, other->size);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool bytes_decode(int argc, py_Ref argv) {
|
||||
PY_CHECK_ARGC(1);
|
||||
int size;
|
||||
unsigned char* data = py_tobytes(&argv[0], &size);
|
||||
py_newstrn(py_retval(), (const char*)data, size);
|
||||
return true;
|
||||
}
|
||||
|
||||
py_Type pk_bytes__register() {
|
||||
py_Type type = pk_newtype("bytes", tp_object, NULL, NULL, false, true);
|
||||
// no need to dtor because the memory is controlled by the object
|
||||
|
||||
py_bindmagic(tp_bytes, __new__, bytes__new__);
|
||||
py_bindmagic(tp_bytes, __repr__, bytes__repr__);
|
||||
py_bindmagic(tp_bytes, __getitem__, bytes__getitem__);
|
||||
py_bindmagic(tp_bytes, __eq__, bytes__eq__);
|
||||
py_bindmagic(tp_bytes, __ne__, bytes__ne__);
|
||||
py_bindmagic(tp_bytes, __add__, bytes__add__);
|
||||
|
||||
py_bindmethod(tp_bytes, "decode", bytes_decode);
|
||||
return type;
|
||||
}
|
||||
|
||||
|
@ -13,8 +13,7 @@ assert b'\xff\xee' == b'\xff\xee'
|
||||
a = '测试123'
|
||||
assert a == a.encode().decode()
|
||||
|
||||
assert chr(0) == '\x00'
|
||||
assert ord('\x00') == 0
|
||||
|
||||
|
||||
# test slice
|
||||
s = b"football"
|
||||
@ -34,8 +33,4 @@ assert a[::-1] == b"!dlroW ,olleH"
|
||||
assert a[::2] == b"Hlo ol!"
|
||||
assert a[2:5:2] == b"lo"
|
||||
assert a[5:2:-1] == b",ol"
|
||||
assert a[5:2:-2] == b",l"
|
||||
|
||||
assert bytes() == b''
|
||||
assert bytes((65,)) == b'A'
|
||||
assert bytes([0, 1, 2, 3]) == b'\x00\x01\x02\x03'
|
||||
assert a[5:2:-2] == b",l"
|
21
tests/70_base64.py
Normal file
21
tests/70_base64.py
Normal file
@ -0,0 +1,21 @@
|
||||
a = '测试 123'
|
||||
a = a.encode()
|
||||
|
||||
import base64
|
||||
|
||||
b = base64.b64encode(a)
|
||||
c = base64.b64decode(b)
|
||||
|
||||
assert a == c
|
||||
|
||||
|
||||
data = [66, 110, 145, 18, 176, 13, 255, 202, 173, 109, 178, 194, 171, 198, 143, 24, 113, 46, 70, 94, 71, 140, 159, 191, 134, 230, 190, 224, 223, 94, 217, 20, 241, 138, 104, 120, 249, 91, 134, 48, 108, 49, 0, 249, 235, 225, 228, 190, 63, 204, 216, 102, 153, 51, 79, 221, 234, 252, 231, 156, 74, 23, 131, 161, 172, 157, 26, 15, 88, 28, 21, 170, 86, 177, 177, 249, 111, 230, 35, 180, 61, 140, 33, 14, 74, 238, 253, 19, 177, 76, 249, 21, 35, 105, 24, 136, 187, 121, 71, 202, 239, 235, 71, 126, 60, 37, 83, 186, 102, 114, 95, 212, 81, 48, 102, 167, 208, 66, 250, 132, 199, 137, 141, 231, 126, 219, 125, 1, 86, 87, 132, 161, 55, 166, 192, 27, 95, 27, 237, 225, 32, 240, 234, 160, 247, 143, 241, 232, 195, 117, 83, 133, 69, 178, 239, 123, 144, 172, 34, 43, 56, 136, 184, 68, 65, 70, 61, 164, 109, 134, 142, 153, 125, 154, 62, 117, 166, 86, 234, 39, 73, 207, 67, 91, 88, 220, 43, 148, 201, 185, 128, 93, 151, 210, 167, 82, 87, 246, 171, 125, 210, 46, 60, 156, 4, 173, 219, 149, 24, 226, 63, 176, 92, 103, 126, 201, 254, 6, 186, 233, 165, 169, 237, 141, 252, 0, 195, 212, 222, 186, 103, 15, 137, 41, 251, 16, 163, 22, 177, 232, 205, 58, 50, 205, 89, 249, 38, 45, 98, 42, 155, 33, 225, 232, 16, 157, 91, 246, 207, 164, 150, 214, 76, 151, 179, 203, 67, 194, 213, 83, 2, 106, 109, 254, 15, 110, 168, 19, 114, 185, 174, 20, 106, 141, 116, 222, 205, 135, 222, 110, 90, 27, 61, 6, 118, 50, 155, 6, 224, 213, 109, 98, 252, 84, 166, 77, 124, 187, 187, 113, 173, 45, 17, 232, 208, 126, 248, 239, 18, 33, 205, 117, 44, 32, 223, 1, 221, 210, 41, 67, 28, 218, 218, 161, 209, 11, 93, 250, 96, 2, 43, 157, 217, 134, 183, 24, 105, 177, 74, 214, 18, 114, 191, 64, 195, 94, 194, 19, 115, 211, 103, 49, 218, 87, 8, 199, 50, 225, 174, 222, 75, 23, 159, 76, 56, 208, 224, 172, 48, 197, 126, 159, 191, 80, 216, 148, 30, 114, 231, 142, 100, 159, 67, 77, 190, 64, 182, 21, 108, 4, 232, 73, 145, 247, 196, 220, 197, 234, 55, 241, 212, 115, 115, 142, 172, 248, 132, 117, 115, 107, 176, 230, 130, 189, 160, 150, 63, 79, 253, 240, 113, 61, 222, 46, 102, 118, 100, 208, 170, 0, 60, 154, 102, 168, 241, 159, 146, 71, 55, 244, 123, 82, 49, 64, 231, 190, 49, 51, 16, 111, 153, 209, 208, 116, 19, 68, 139, 208, 105, 248, 80, 12, 237, 29, 63, 80, 127, 1, 118, 22, 39, 83, 25, 220, 75, 31, 152, 16, 94, 254, 141, 55, 6, 89, 45, 247, 229, 209, 239, 223, 226, 124, 50, 51, 219, 110, 100, 251, 122, 53, 166, 63, 43, 116, 190, 114, 169, 72, 18, 190, 55, 4, 249, 3, 200, 99, 0, 37, 94, 50, 58, 37, 56, 154, 18, 154, 127, 123, 187, 123, 110, 131, 14, 185, 76, 193, 11, 227, 36, 184, 88, 3, 222, 126, 32, 143, 125, 180, 104, 142, 84, 22, 53, 2, 38, 188, 187, 51, 163, 189, 25, 215, 94, 190, 196, 213, 155, 23, 84, 206, 237, 125, 76, 185, 12, 111, 201, 249, 101, 50, 217, 32, 3, 37, 49, 177, 4, 10, 123, 29, 126, 106, 108, 246, 89, 42, 182, 135, 11, 152, 122, 12, 23, 159, 212, 53, 44, 244, 48, 251, 130, 109, 191, 76, 148, 226, 83, 55, 225, 100, 196, 166, 171, 108, 91, 67, 226, 207, 143, 73, 81, 95, 69, 92, 141, 150, 108, 168, 235, 1, 33, 160, 158, 62, 149, 0, 200, 228, 176, 38, 112, 18, 253, 239, 107, 214, 17, 22, 112, 255, 117, 155, 248, 59, 113, 100, 145, 101, 245, 113, 230, 167, 58, 232, 195, 51, 76, 26, 7, 94, 201, 198, 96, 93, 8, 231, 60, 139, 37, 191, 37, 101, 155, 83, 246, 181, 109, 149, 241, 96, 168, 126, 232, 54, 230, 197, 179, 214, 148, 79, 13, 27, 195, 164, 146, 183, 129, 82, 82, 177, 2, 255, 8, 85, 214, 83, 244, 237, 143, 104, 107, 28, 215, 178, 46, 71, 175, 186, 77, 191, 93, 13, 204, 154, 234, 193, 231, 49, 27, 7, 66, 53, 170, 63, 3, 172, 177, 176, 255, 249, 116, 172, 165, 78, 64, 218, 147, 214, 206, 68, 42, 186, 119, 75, 28, 141, 187, 117, 21, 89, 69, 96, 79, 211, 1, 141]
|
||||
data = bytes(data)
|
||||
encoded = base64.b64encode(data)
|
||||
|
||||
res = 'Qm6RErAN/8qtbbLCq8aPGHEuRl5HjJ+/hua+4N9e2RTximh4+VuGMGwxAPnr4eS+P8zYZpkzT93q/OecSheDoaydGg9YHBWqVrGx+W/mI7Q9jCEOSu79E7FM+RUjaRiIu3lHyu/rR348JVO6ZnJf1FEwZqfQQvqEx4mN537bfQFWV4ShN6bAG18b7eEg8Oqg94/x6MN1U4VFsu97kKwiKziIuERBRj2kbYaOmX2aPnWmVuonSc9DW1jcK5TJuYBdl9KnUlf2q33SLjycBK3blRjiP7BcZ37J/ga66aWp7Y38AMPU3rpnD4kp+xCjFrHozToyzVn5Ji1iKpsh4egQnVv2z6SW1kyXs8tDwtVTAmpt/g9uqBNyua4Uao103s2H3m5aGz0GdjKbBuDVbWL8VKZNfLu7ca0tEejQfvjvEiHNdSwg3wHd0ilDHNraodELXfpgAiud2Ya3GGmxStYScr9Aw17CE3PTZzHaVwjHMuGu3ksXn0w40OCsMMV+n79Q2JQecueOZJ9DTb5AthVsBOhJkffE3MXqN/HUc3OOrPiEdXNrsOaCvaCWP0/98HE93i5mdmTQqgA8mmao8Z+SRzf0e1IxQOe+MTMQb5nR0HQTRIvQafhQDO0dP1B/AXYWJ1MZ3EsfmBBe/o03Blkt9+XR79/ifDIz225k+3o1pj8rdL5yqUgSvjcE+QPIYwAlXjI6JTiaEpp/e7t7boMOuUzBC+MkuFgD3n4gj320aI5UFjUCJry7M6O9GddevsTVmxdUzu19TLkMb8n5ZTLZIAMlMbEECnsdfmps9lkqtocLmHoMF5/UNSz0MPuCbb9MlOJTN+FkxKarbFtD4s+PSVFfRVyNlmyo6wEhoJ4+lQDI5LAmcBL972vWERZw/3Wb+DtxZJFl9XHmpzrowzNMGgdeycZgXQjnPIslvyVlm1P2tW2V8WCofug25sWz1pRPDRvDpJK3gVJSsQL/CFXWU/Ttj2hrHNeyLkevuk2/XQ3MmurB5zEbB0I1qj8DrLGw//l0rKVOQNqT1s5EKrp3SxyNu3UVWUVgT9MBjQ=='
|
||||
assert encoded.decode() == res
|
||||
|
||||
decoded = base64.b64decode(encoded)
|
||||
assert decoded == data
|
||||
|
50
tests/80_csv.py
Normal file
50
tests/80_csv.py
Normal file
@ -0,0 +1,50 @@
|
||||
import csv
|
||||
def test(data: str, expected):
|
||||
ret = list(csv.reader(data.splitlines()))
|
||||
assert ret==expected, f"Expected {expected}, got {ret}"
|
||||
|
||||
test("""a,b,c
|
||||
1,2,3
|
||||
""", [['a', 'b', 'c'], ['1', '2', '3']])
|
||||
|
||||
test("""a,b,c
|
||||
1,2,"3"
|
||||
""", [['a', 'b', 'c'], ['1', '2', '3']])
|
||||
|
||||
test("""a,b,c
|
||||
1,2,"3,,"
|
||||
""", [['a', 'b', 'c'], ['1', '2', '3,,']])
|
||||
|
||||
test("""a,b,c
|
||||
1,2,'3'
|
||||
""", [['a', 'b', 'c'], ['1', '2', '\'3\'']])
|
||||
|
||||
test('''a,b,c
|
||||
1,2,"123"""
|
||||
''', [['a', 'b', 'c'], ['1', '2', '123"']])
|
||||
|
||||
test("""a,b,c,
|
||||
1,2,3,
|
||||
""", [['a', 'b', 'c', ''], ['1', '2', '3', '']])
|
||||
|
||||
test("""a,b ,c,
|
||||
1,"22""33",3
|
||||
""", [['a', 'b ', 'c', ''], ['1', '22"33', '3']])
|
||||
|
||||
# newline
|
||||
test('''a,b,c
|
||||
1,2,"3,
|
||||
4"
|
||||
5,"a,""
|
||||
b",7
|
||||
''', [['a', 'b', 'c'], ['1', '2', '3,\n 4'], ['5', 'a,"\nb', '7']])
|
||||
|
||||
ret = csv.DictReader("""a,b,c
|
||||
1,2,3
|
||||
"4",5,6
|
||||
""".splitlines())
|
||||
|
||||
assert list(ret)==[
|
||||
{'a': '1', 'b': '2', 'c': '3'},
|
||||
{'a': '4', 'b': '5', 'c': '6'},
|
||||
]
|
Loading…
x
Reference in New Issue
Block a user