#include "pocketpy/objects/codeobject.h" #include "pocketpy/objects/error.h" #include "pocketpy/pocketpy.h" #include "pocketpy/common/utils.h" #include "pocketpy/objects/object.h" #include "pocketpy/interpreter/vm.h" #include "pocketpy/common/sstream.h" typedef struct BaseExceptionFrame { SourceData_ src; int lineno; c11_string* name; } BaseExceptionFrame; typedef struct BaseException { c11_vector /*T=BaseExceptionFrame*/ stacktrace; } BaseException; void py_BaseException__stpush(py_Ref self, SourceData_ src, int lineno, const char* func_name) { BaseException* ud = py_touserdata(self); if(ud->stacktrace.length >= 7) return; BaseExceptionFrame* frame = c11_vector__emplace(&ud->stacktrace); PK_INCREF(src); frame->src = src; frame->lineno = lineno; frame->name = func_name ? c11_string__new(func_name) : NULL; } static void BaseException__dtor(void* ud) { BaseException* self = (BaseException*)ud; c11__foreach(BaseExceptionFrame, &self->stacktrace, it) { PK_DECREF(it->src); if(it->name) c11_string__delete(it->name); } c11_vector__dtor(&self->stacktrace); } static bool _py_BaseException__new__(int argc, py_Ref argv) { py_Type cls = py_totype(argv); BaseException* ud = py_newobject(py_retval(), cls, 2, sizeof(BaseException)); c11_vector__ctor(&ud->stacktrace, sizeof(BaseExceptionFrame)); return true; } static bool _py_BaseException__init__(int argc, py_Ref argv) { py_newnone(py_retval()); if(argc == 1 + 0) return true; if(argc == 1 + 1) { py_setslot(py_arg(0), 0, py_arg(1)); return true; } return TypeError("__init__() takes at most 1 arguments but %d were given", argc - 1); } static bool _py_BaseException__repr__(int argc, py_Ref argv) { c11_sbuf ss; c11_sbuf__ctor(&ss); pk_sprintf(&ss, "%t(", argv->type); py_Ref arg = py_getslot(argv, 0); if(!py_isnil(arg)) { if(!py_repr(arg)) return false; c11_sbuf__write_sv(&ss, py_tosv(py_retval())); } c11_sbuf__write_char(&ss, ')'); c11_sbuf__py_submit(&ss, py_retval()); return true; } static bool _py_BaseException__str__(int argc, py_Ref argv) { c11_sbuf ss; c11_sbuf__ctor(&ss); py_Ref arg = py_getslot(argv, 0); if(!py_isnil(arg)) { if(argv->type == tp_KeyError) { if(!py_repr(arg)) return false; } else { if(!py_str(arg)) return false; } c11_sbuf__write_sv(&ss, py_tosv(py_retval())); } c11_sbuf__py_submit(&ss, py_retval()); return true; } static bool BaseException_args(int argc, py_Ref argv){ PY_CHECK_ARGC(1); py_Ref arg = py_getslot(argv, 0); if(!py_isnil(arg)) { py_Ref p = py_newtuple(py_retval(), 1); p[0] = *arg; }else{ py_newtuple(py_retval(), 0); } return true; } static bool StopIteration_value(int argc, py_Ref argv) { PY_CHECK_ARGC(1); py_Ref arg = py_getslot(argv, 0); if(py_isnil(arg)) { py_newnone(py_retval()); }else{ py_assign(py_retval(), arg); } return true; } py_Type pk_BaseException__register() { py_Type type = pk_newtype("BaseException", tp_object, NULL, BaseException__dtor, false, false); py_bindmagic(type, __new__, _py_BaseException__new__); py_bindmagic(type, __init__, _py_BaseException__init__); py_bindmagic(type, __repr__, _py_BaseException__repr__); py_bindmagic(type, __str__, _py_BaseException__str__); py_bindproperty(type, "args", BaseException_args, NULL); return type; } py_Type pk_Exception__register() { py_Type type = pk_newtype("Exception", tp_BaseException, NULL, NULL, false, false); return type; } py_Type pk_StopIteration__register() { py_Type type = pk_newtype("StopIteration", tp_Exception, NULL, NULL, false, false); py_bindproperty(type, "value", StopIteration_value, NULL); return type; } ////////////////////////////////////////////////// bool py_checkexc(bool ignore_handled) { VM* vm = pk_current_vm; if(ignore_handled && vm->is_curr_exc_handled) return false; return !py_isnil(&vm->curr_exception); } bool py_matchexc(py_Type type) { VM* vm = pk_current_vm; if(vm->is_curr_exc_handled) return false; if(py_isnil(&vm->curr_exception)) return false; bool ok = py_issubclass(vm->curr_exception.type, type); if(ok) { // if match, then the exception is handled vm->is_curr_exc_handled = true; vm->last_retval = vm->curr_exception; } return ok; } void py_clearexc(py_StackRef p0) { VM* vm = pk_current_vm; vm->curr_exception = *py_NIL(); vm->is_curr_exc_handled = false; /* Don't clear this, because StopIteration() may corrupt the class defination */ // vm->__curr_class = NULL; vm->__curr_function = NULL; if(p0) vm->stack.sp = p0; } void py_printexc() { char* msg = py_formatexc(); if(!msg) return; pk_current_vm->callbacks.print(msg); pk_current_vm->callbacks.print("\n"); PK_FREE(msg); } static void c11_sbuf__write_exc(c11_sbuf* self, py_Ref exc) { if(true) { c11_sbuf__write_cstr(self, "Traceback (most recent call last):\n"); } BaseException* ud = py_touserdata(exc); for(int i = ud->stacktrace.length - 1; i >= 0; i--) { BaseExceptionFrame* frame = c11__at(BaseExceptionFrame, &ud->stacktrace, i); SourceData__snapshot(frame->src, self, frame->lineno, NULL, frame->name ? frame->name->data : NULL); c11_sbuf__write_char(self, '\n'); } const char* name = py_tpname(exc->type); const char* message; bool ok = py_str(exc); if(!ok || !py_isstr(py_retval())) { message = ""; } else { message = py_tostr(py_retval()); } c11_sbuf__write_cstr(self, name); c11_sbuf__write_cstr(self, ": "); c11_sbuf__write_cstr(self, message); } char* py_formatexc() { VM* vm = pk_current_vm; if(py_isnil(&vm->curr_exception)) return NULL; // when you call `py_formatexc()`, you are handling the exception vm->is_curr_exc_handled = true; c11_sbuf ss; c11_sbuf__ctor(&ss); py_Ref inner = py_getslot(&vm->curr_exception, 1); if(py_isnil(inner)) { c11_sbuf__write_exc(&ss, &vm->curr_exception); } else { c11_sbuf__write_exc(&ss, inner); c11_sbuf__write_cstr( &ss, "\n\nDuring handling of the above exception, another exception occurred:\n\n"); c11_sbuf__write_exc(&ss, &vm->curr_exception); } c11_string* res = c11_sbuf__submit(&ss); char* dup = PK_MALLOC(res->size + 1); memcpy(dup, res->data, res->size); dup[res->size] = '\0'; c11_string__delete(res); return dup; } bool py_exception(py_Type type, const char* fmt, ...) { c11_sbuf buf; c11_sbuf__ctor(&buf); va_list args; va_start(args, fmt); pk_vsprintf(&buf, fmt, args); va_end(args); py_Ref message = py_pushtmp(); c11_sbuf__py_submit(&buf, message); bool ok = py_tpcall(type, 1, message); if(!ok) return false; py_pop(); return py_raise(py_retval()); } bool py_raise(py_Ref exc) { assert(py_isinstance(exc, tp_BaseException)); VM* vm = pk_current_vm; if(!py_isnil(&vm->curr_exception)) { // inner exception py_setslot(exc, 1, &vm->curr_exception); } vm->curr_exception = *exc; vm->is_curr_exc_handled = false; return false; } bool KeyError(py_Ref key) { bool ok = py_tpcall(tp_KeyError, 1, key); if(!ok) return false; return py_raise(py_retval()); }