py/runtime: Add mp_load_method_protected helper which catches exceptions
This new helper function acts like mp_load_method_maybe but is wrapped in an NLR handler so it can catch exceptions. It prevents AttributeError from propagating out, and optionally all other exceptions. This helper can be used to fully implement hasattr (see follow-up commit), and also for cases where mp_load_method_maybe is used but it must now raise an exception.
This commit is contained in:
parent
eb88803ac8
commit
bc87b862fd
16
py/runtime.c
16
py/runtime.c
|
@ -1086,6 +1086,22 @@ void mp_load_method(mp_obj_t base, qstr attr, mp_obj_t *dest) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Acts like mp_load_method_maybe but catches AttributeError, and all other exceptions if requested
|
||||||
|
void mp_load_method_protected(mp_obj_t obj, qstr attr, mp_obj_t *dest, bool catch_all_exc) {
|
||||||
|
nlr_buf_t nlr;
|
||||||
|
if (nlr_push(&nlr) == 0) {
|
||||||
|
mp_load_method_maybe(obj, attr, dest);
|
||||||
|
nlr_pop();
|
||||||
|
} else {
|
||||||
|
if (!catch_all_exc
|
||||||
|
&& !mp_obj_is_subclass_fast(MP_OBJ_FROM_PTR(((mp_obj_base_t*)nlr.ret_val)->type),
|
||||||
|
MP_OBJ_FROM_PTR(&mp_type_AttributeError))) {
|
||||||
|
// Re-raise the exception
|
||||||
|
nlr_raise(MP_OBJ_FROM_PTR(nlr.ret_val));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void mp_store_attr(mp_obj_t base, qstr attr, mp_obj_t value) {
|
void mp_store_attr(mp_obj_t base, qstr attr, mp_obj_t value) {
|
||||||
DEBUG_OP_printf("store attr %p.%s <- %p\n", base, qstr_str(attr), value);
|
DEBUG_OP_printf("store attr %p.%s <- %p\n", base, qstr_str(attr), value);
|
||||||
mp_obj_type_t *type = mp_obj_get_type(base);
|
mp_obj_type_t *type = mp_obj_get_type(base);
|
||||||
|
|
|
@ -132,6 +132,7 @@ mp_obj_t mp_load_attr(mp_obj_t base, qstr attr);
|
||||||
void mp_convert_member_lookup(mp_obj_t obj, const mp_obj_type_t *type, mp_obj_t member, mp_obj_t *dest);
|
void mp_convert_member_lookup(mp_obj_t obj, const mp_obj_type_t *type, mp_obj_t member, mp_obj_t *dest);
|
||||||
void mp_load_method(mp_obj_t base, qstr attr, mp_obj_t *dest);
|
void mp_load_method(mp_obj_t base, qstr attr, mp_obj_t *dest);
|
||||||
void mp_load_method_maybe(mp_obj_t base, qstr attr, mp_obj_t *dest);
|
void mp_load_method_maybe(mp_obj_t base, qstr attr, mp_obj_t *dest);
|
||||||
|
void mp_load_method_protected(mp_obj_t obj, qstr attr, mp_obj_t *dest, bool catch_all_exc);
|
||||||
void mp_load_super_method(qstr attr, mp_obj_t *dest);
|
void mp_load_super_method(qstr attr, mp_obj_t *dest);
|
||||||
void mp_store_attr(mp_obj_t base, qstr attr, mp_obj_t val);
|
void mp_store_attr(mp_obj_t base, qstr attr, mp_obj_t val);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue