mirror of
				https://github.com/pocketpy/pocketpy
				synced 2025-10-25 14:00:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			266 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			266 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #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 definition */
 | |
|     // vm->curr_class = NULL;
 | |
|     vm->curr_decl_based_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 = "<exception str() failed>";
 | |
|     } 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;
 | |
|     
 | |
|     if(vm->trace_info.func && !py_istype(exc, tp_StopIteration)) {
 | |
|         py_Frame* frame = vm->top_frame;
 | |
|         vm->trace_info.func(frame, TRACE_EVENT_EXCEPTION);
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool KeyError(py_Ref key) {
 | |
|     bool ok = py_tpcall(tp_KeyError, 1, key);
 | |
|     if(!ok) return false;
 | |
|     return py_raise(py_retval());
 | |
| } |