From 5a3bf82f41005c5f62b85611650220e397389aa8 Mon Sep 17 00:00:00 2001 From: Quincey Koziol Date: Fri, 19 Apr 2024 20:39:42 -0500 Subject: [PATCH] Initial refactor of H5E code to reduce interaction with H5I package Signed-off-by: Quincey Koziol --- bin/make_err | 35 +++++++++++++++++--- src/H5E.c | 83 ++++++++++++++--------------------------------- src/H5Eint.c | 84 +++++++++++++++++++++++++++++++++++------------- src/H5Epkg.h | 10 +++++- src/H5Eprivate.h | 2 +- src/H5private.h | 6 ++-- 6 files changed, 131 insertions(+), 89 deletions(-) diff --git a/bin/make_err b/bin/make_err index 064d66c9456..618a2df46ac 100755 --- a/bin/make_err +++ b/bin/make_err @@ -223,6 +223,8 @@ sub create_init ($) { my $desc; # Description of error message my $sect_name; # Section of minor error messages my $sect_desc; # Description of section + my $first_major = 0; # Whether the first major error code was saved + my $first_minor = 0; # Whether the first minor error code was saved # Rename previous file # rename "${prefix}${file}", "${prefix}${file}~" or die "unable to make backup"; @@ -241,12 +243,22 @@ sub create_init ($) { print HEADER "/* Major error codes */\n"; print HEADER "/*********************/\n\n"; foreach $name (keys %major) { - print HEADER " "x(0*$indent),"assert(${name}_g==(-1));\n"; + print HEADER " "x(0*$indent),"assert(${name}_g==H5I_INVALID_HID);\n"; print HEADER " "x(0*$indent),"if((msg = H5E__create_msg(cls, H5E_MAJOR, \"${major{$name}}\"))==NULL)\n"; print HEADER " "x(1*$indent),"HGOTO_ERROR(H5E_ERROR, H5E_CANTINIT, FAIL, \"error message initialization failed\");\n"; print HEADER " "x(0*$indent),"if((${name}_g = H5I_register(H5I_ERROR_MSG, msg, false))<0)\n"; print HEADER " "x(1*$indent),"HGOTO_ERROR(H5E_ERROR, H5E_CANTREGISTER, FAIL, \"can't register error message\");\n"; + if ($first_major == 0) { + print HEADER " "x(0*$indent),"\n/* Remember first major error code ID */\n"; + print HEADER " "x(0*$indent),"assert(H5E_first_maj_id_g==H5I_INVALID_HID);\n"; + print HEADER " "x(0*$indent),"H5E_first_maj_id_g = ${name}_g;\n\n"; + $first_major = 1; } + $last_name = $name; + } + print HEADER " "x(0*$indent),"\n/* Remember last major error code ID */\n"; + print HEADER " "x(0*$indent),"assert(H5E_last_maj_id_g==H5I_INVALID_HID);\n"; + print HEADER " "x(0*$indent),"H5E_last_maj_id_g = ${last_name}_g;\n\n"; # Iterate over all the minor error sections print HEADER "\n/*********************/\n"; @@ -257,13 +269,24 @@ sub create_init ($) { # Iterate over all the minor errors in each section for $name ( @{$section_list{$sect_name}}) { - print HEADER " "x(0*$indent),"assert(${name}_g==(-1));\n"; + print HEADER " "x(0*$indent),"assert(${name}_g==H5I_INVALID_HID);\n"; print HEADER " "x(0*$indent),"if((msg = H5E__create_msg(cls, H5E_MINOR, \"${minor{$name}}\"))==NULL)\n"; print HEADER " "x(1*$indent),"HGOTO_ERROR(H5E_ERROR, H5E_CANTINIT, FAIL, \"error message initialization failed\");\n"; print HEADER " "x(0*$indent),"if((${name}_g = H5I_register(H5I_ERROR_MSG, msg, true))<0)\n"; print HEADER " "x(1*$indent),"HGOTO_ERROR(H5E_ERROR, H5E_CANTREGISTER, FAIL, \"can't register error message\");\n"; + + if ($first_minor == 0) { + print HEADER " "x(0*$indent),"\n/* Remember first minor error code ID */\n"; + print HEADER " "x(0*$indent),"assert(H5E_first_min_id_g==H5I_INVALID_HID);\n"; + print HEADER " "x(0*$indent),"H5E_first_min_id_g = ${name}_g;\n\n"; + $first_minor = 1; + } + $last_name = $name; } } + print HEADER " "x(0*$indent),"\n/* Remember last minor error code ID */\n"; + print HEADER " "x(0*$indent),"assert(H5E_last_min_id_g==H5I_INVALID_HID);\n"; + print HEADER " "x(0*$indent),"H5E_last_min_id_g = ${last_name}_g;\n"; print_endprotect(*HEADER, $file); @@ -299,7 +322,9 @@ sub create_term ($) { foreach $name (keys %major) { print HEADER " "x($indent),"\n${name}_g="; } - print HEADER " (-1);\n"; + print HEADER " H5I_INVALID_HID;\n"; + print HEADER " "x(0*$indent),"H5E_first_maj_id_g = H5I_INVALID_HID;\n\n"; + print HEADER " "x(0*$indent),"H5E_last_maj_id_g = H5I_INVALID_HID;\n\n"; # Iterate over all the minor error sections print HEADER "\n/* Reset minor error IDs */\n"; @@ -311,7 +336,9 @@ sub create_term ($) { print HEADER " "x($indent),"\n${name}_g="; } } - print HEADER " (-1);\n"; + print HEADER " H5I_INVALID_HID;\n"; + print HEADER " "x(0*$indent),"H5E_first_min_id_g = H5I_INVALID_HID;\n\n"; + print HEADER " "x(0*$indent),"H5E_last_min_id_g = H5I_INVALID_HID;\n\n"; print_endprotect(*HEADER, $file); diff --git a/src/H5E.c b/src/H5E.c index 755172c755d..4f2edd78ebf 100644 --- a/src/H5E.c +++ b/src/H5E.c @@ -91,6 +91,12 @@ static herr_t H5E__append_stack(H5E_t *dst_estack, const H5E_t *src_stack); /* Package Variables */ /*********************/ +/* First & last major and minor error codes registered by the library */ +hid_t H5E_first_maj_id_g = H5I_INVALID_HID; +hid_t H5E_last_maj_id_g = H5I_INVALID_HID; +hid_t H5E_first_min_id_g = H5I_INVALID_HID; +hid_t H5E_last_min_id_g = H5I_INVALID_HID; + /*****************************/ /* Library Private Variables */ /*****************************/ @@ -435,11 +441,11 @@ H5E__register_class(const char *cls_name, const char *lib_name, const char *vers HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, NULL, "memory allocation failed"); /* Duplicate string information */ - if (NULL == (cls->cls_name = H5MM_xstrdup(cls_name))) + if (NULL == (cls->cls_name = strdup(cls_name))) HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, NULL, "memory allocation failed"); - if (NULL == (cls->lib_name = H5MM_xstrdup(lib_name))) + if (NULL == (cls->lib_name = strdup(lib_name))) HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, NULL, "memory allocation failed"); - if (NULL == (cls->lib_vers = H5MM_xstrdup(version))) + if (NULL == (cls->lib_vers = strdup(version))) HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, NULL, "memory allocation failed"); /* Set the return value */ @@ -738,7 +744,7 @@ H5E__create_msg(H5E_cls_t *cls, H5E_type_t msg_type, const char *msg_str) /* Fill new message object */ msg->cls = cls; msg->type = msg_type; - if (NULL == (msg->msg = H5MM_xstrdup(msg_str))) + if (NULL == (msg->msg = strdup(msg_str))) HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, NULL, "memory allocation failed"); /* Set return value */ @@ -884,24 +890,11 @@ H5E__get_current_stack(void) current_error = &(current_stack->slot[u]); new_error = &(estack_copy->slot[u]); - /* Increment the IDs to indicate that they are used in this stack */ - if (H5I_inc_ref(current_error->cls_id, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, NULL, "unable to increment ref count on error class"); - new_error->cls_id = current_error->cls_id; - if (H5I_inc_ref(current_error->maj_num, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, NULL, "unable to increment ref count on error message"); - new_error->maj_num = current_error->maj_num; - if (H5I_inc_ref(current_error->min_num, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, NULL, "unable to increment ref count on error message"); - new_error->min_num = current_error->min_num; - /* The 'func' & 'file' strings are statically allocated (by the compiler) - * there's no need to duplicate them. - */ - new_error->func_name = current_error->func_name; - new_error->file_name = current_error->file_name; - new_error->line = current_error->line; - if (NULL == (new_error->desc = H5MM_xstrdup(current_error->desc))) - HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, NULL, "memory allocation failed"); + /* Set error stack entry */ + if (H5E__set_stack_entry(new_error, current_error->file_name, current_error->func_name, + current_error->line, current_error->cls_id, current_error->maj_num, + current_error->min_num, current_error->desc) < 0) + HGOTO_ERROR(H5E_ERROR, H5E_CANTSET, NULL, "can't set error entry"); } /* end for */ /* Copy the "automatic" error reporting information */ @@ -997,24 +990,11 @@ H5E__set_current_stack(H5E_t *estack) current_error = &(current_stack->slot[u]); new_error = &(estack->slot[u]); - /* Increment the IDs to indicate that they are used in this stack */ - if (H5I_inc_ref(new_error->cls_id, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, FAIL, "unable to increment ref count on error class"); - current_error->cls_id = new_error->cls_id; - if (H5I_inc_ref(new_error->maj_num, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, FAIL, "unable to increment ref count on error class"); - current_error->maj_num = new_error->maj_num; - if (H5I_inc_ref(new_error->min_num, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, FAIL, "unable to increment ref count on error class"); - current_error->min_num = new_error->min_num; - /* The 'func' & 'file' strings are statically allocated (by the compiler) - * there's no need to duplicate them. - */ - current_error->func_name = new_error->func_name; - current_error->file_name = new_error->file_name; - current_error->line = new_error->line; - if (NULL == (current_error->desc = H5MM_xstrdup(new_error->desc))) - HGOTO_ERROR(H5E_RESOURCE, H5E_NOSPACE, FAIL, "memory allocation failed"); + /* Set error stack entry */ + if (H5E__set_stack_entry(current_error, new_error->file_name, new_error->func_name, new_error->line, + new_error->cls_id, new_error->maj_num, new_error->min_num, + new_error->desc) < 0) + HGOTO_ERROR(H5E_ERROR, H5E_CANTSET, FAIL, "can't set error entry"); } /* end for */ done: @@ -1635,24 +1615,11 @@ H5E__append_stack(H5E_t *dst_stack, const H5E_t *src_stack) src_error = &(src_stack->slot[u]); dst_error = &(dst_stack->slot[dst_stack->nused]); - /* Increment the IDs to indicate that they are used in this stack */ - if (H5I_inc_ref(src_error->cls_id, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, FAIL, "unable to increment ref count on error class"); - dst_error->cls_id = src_error->cls_id; - if (H5I_inc_ref(src_error->maj_num, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, FAIL, "unable to increment ref count on error message"); - dst_error->maj_num = src_error->maj_num; - if (H5I_inc_ref(src_error->min_num, false) < 0) - HGOTO_ERROR(H5E_ERROR, H5E_CANTINC, FAIL, "unable to increment ref count on error message"); - dst_error->min_num = src_error->min_num; - /* The 'func' & 'file' strings are statically allocated (by the compiler) - * there's no need to duplicate them. - */ - dst_error->func_name = src_error->func_name; - dst_error->file_name = src_error->file_name; - dst_error->line = src_error->line; - if (NULL == (dst_error->desc = H5MM_xstrdup(src_error->desc))) - HGOTO_ERROR(H5E_ERROR, H5E_CANTALLOC, FAIL, "memory allocation failed"); + /* Set error stack entry */ + if (H5E__set_stack_entry(dst_error, src_error->file_name, src_error->func_name, src_error->line, + src_error->cls_id, src_error->maj_num, src_error->min_num, + src_error->desc) < 0) + HGOTO_ERROR(H5E_ERROR, H5E_CANTSET, FAIL, "can't set error entry"); /* Increment # of errors in destination stack */ dst_stack->nused++; diff --git a/src/H5Eint.c b/src/H5Eint.c index 70848ecd7c2..aaa471cb1f0 100644 --- a/src/H5Eint.c +++ b/src/H5Eint.c @@ -712,6 +712,50 @@ H5E__push_stack(H5E_t *estack, const char *file, const char *func, unsigned line if (NULL == (estack = H5E__get_my_stack())) HGOTO_DONE(FAIL); + /* + * Push the error if there's room. Otherwise just forget it. + */ + if (estack->nused < H5E_NSLOTS) { + if (H5E__set_stack_entry(&estack->slot[estack->nused], file, func, line, cls_id, maj_id, min_id, + desc) < 0) + HGOTO_DONE(FAIL); + estack->nused++; + } /* end if */ + +done: + FUNC_LEAVE_NOAPI(ret_value) +} /* end H5E__push_stack() */ + +/*------------------------------------------------------------------------- + * Function: H5E__set_stack_entry + * + * Purpose: Sets the information for a given stack entry. + * + * Return: SUCCEED/FAIL + * + *------------------------------------------------------------------------- + */ +herr_t +H5E__set_stack_entry(H5E_error2_t *err_entry, const char *file, const char *func, unsigned line, hid_t cls_id, + hid_t maj_id, hid_t min_id, const char *desc) +{ + herr_t ret_value = SUCCEED; /* Return value */ + + /* + * WARNING: We cannot call HERROR() from within this function or else we + * could enter infinite recursion. Furthermore, we also cannot + * call any other HDF5 macro or function which might call + * HERROR(). HERROR() is called by HRETURN_ERROR() which could + * be called by FUNC_ENTER(). + */ + FUNC_ENTER_PACKAGE_NOERR + + /* Sanity check */ + assert(err_entry); + assert(cls_id > 0); + assert(maj_id > 0); + assert(min_id > 0); + /* * Don't fail if arguments are bad. Instead, substitute some default * value. @@ -723,36 +767,32 @@ H5E__push_stack(H5E_t *estack, const char *file, const char *func, unsigned line if (!desc) desc = "No description given"; - /* - * Push the error if there's room. Otherwise just forget it. - */ - assert(estack); - - if (estack->nused < H5E_NSLOTS) { /* Increment the IDs to indicate that they are used in this stack */ + /* Note: don't waste time incrementing library internal error IDs */ + if (cls_id != H5E_ERR_CLS_g) if (H5I_inc_ref_noherr(cls_id, false) < 0) HGOTO_DONE(FAIL); - estack->slot[estack->nused].cls_id = cls_id; + err_entry->cls_id = cls_id; + if (maj_id < H5E_first_maj_id_g || maj_id > H5E_last_maj_id_g) if (H5I_inc_ref_noherr(maj_id, false) < 0) HGOTO_DONE(FAIL); - estack->slot[estack->nused].maj_num = maj_id; + err_entry->maj_num = maj_id; + if (min_id < H5E_first_min_id_g || min_id > H5E_last_min_id_g) if (H5I_inc_ref_noherr(min_id, false) < 0) HGOTO_DONE(FAIL); - estack->slot[estack->nused].min_num = min_id; + err_entry->min_num = min_id; /* The 'func' & 'file' strings are statically allocated (by the compiler) * there's no need to duplicate them. */ - estack->slot[estack->nused].func_name = func; - estack->slot[estack->nused].file_name = file; - estack->slot[estack->nused].line = line; - if (NULL == (estack->slot[estack->nused].desc = H5MM_xstrdup(desc))) + err_entry->func_name = func; + err_entry->file_name = file; + err_entry->line = line; + if (NULL == (err_entry->desc = strdup(desc))) HGOTO_DONE(FAIL); - estack->nused++; - } /* end if */ done: FUNC_LEAVE_NOAPI(ret_value) -} /* end H5E__push_stack() */ +} /* end H5E__set_stack_entry() */ /*------------------------------------------------------------------------- * Function: H5E__clear_entries @@ -783,10 +823,14 @@ H5E__clear_entries(H5E_t *estack, size_t nentries) /* Decrement the IDs to indicate that they are no longer used by this stack */ /* (In reverse order that they were incremented, so that reference counts work well) */ + /* Note: don't decrement library internal error IDs, since they weren't incremented */ + if (error->min_num < H5E_first_min_id_g || error->min_num > H5E_last_min_id_g) if (H5I_dec_ref(error->min_num) < 0) HGOTO_ERROR(H5E_ERROR, H5E_CANTDEC, FAIL, "unable to decrement ref count on error message"); + if (error->maj_num < H5E_first_maj_id_g || error->maj_num > H5E_last_maj_id_g) if (H5I_dec_ref(error->maj_num) < 0) HGOTO_ERROR(H5E_ERROR, H5E_CANTDEC, FAIL, "unable to decrement ref count on error message"); + if (error->cls_id != H5E_ERR_CLS_g) if (H5I_dec_ref(error->cls_id) < 0) HGOTO_ERROR(H5E_ERROR, H5E_CANTDEC, FAIL, "unable to decrement ref count on error class"); @@ -880,16 +924,13 @@ H5E__pop(H5E_t *estack, size_t count) *------------------------------------------------------------------------- */ herr_t -H5E_dump_api_stack(bool is_api) +H5E_dump_api_stack(void) { + H5E_t *estack = H5E__get_my_stack(); herr_t ret_value = SUCCEED; /* Return value */ FUNC_ENTER_NOAPI_NOERR - /* Only dump the error stack during an API call */ - if (is_api) { - H5E_t *estack = H5E__get_my_stack(); - assert(estack); #ifdef H5_NO_DEPRECATED_SYMBOLS @@ -905,7 +946,6 @@ H5E_dump_api_stack(bool is_api) (void)((estack->auto_op.func2)(H5E_DEFAULT, estack->auto_data)); } /* end else */ #endif /* H5_NO_DEPRECATED_SYMBOLS */ - } /* end if */ FUNC_LEAVE_NOAPI(ret_value) } /* end H5E_dump_api_stack() */ diff --git a/src/H5Epkg.h b/src/H5Epkg.h index 546e389c971..52b1f9eda03 100644 --- a/src/H5Epkg.h +++ b/src/H5Epkg.h @@ -117,7 +117,13 @@ struct H5E_t { * The current error stack. */ H5_DLLVAR H5E_t H5E_stack_g[1]; -#endif /* H5_HAVE_THREADSAFE */ +#endif + +/* First & last major and minor error codes registered by the library */ +H5_DLLVAR hid_t H5E_first_maj_id_g; +H5_DLLVAR hid_t H5E_last_maj_id_g; +H5_DLLVAR hid_t H5E_first_min_id_g; +H5_DLLVAR hid_t H5E_last_min_id_g; /******************************/ /* Package Private Prototypes */ @@ -128,6 +134,8 @@ H5_DLL H5E_t *H5E__get_stack(void); #endif /* H5_HAVE_THREADSAFE */ H5_DLL herr_t H5E__push_stack(H5E_t *estack, const char *file, const char *func, unsigned line, hid_t cls_id, hid_t maj_id, hid_t min_id, const char *desc); +H5_DLL herr_t H5E__set_stack_entry(H5E_error2_t *err_entry, const char *file, const char *func, unsigned line, + hid_t cls_id, hid_t maj_id, hid_t min_id, const char *desc); H5_DLL ssize_t H5E__get_msg(const H5E_msg_t *msg_ptr, H5E_type_t *type, char *msg, size_t size); H5_DLL herr_t H5E__print(const H5E_t *estack, FILE *stream, bool bk_compat); H5_DLL herr_t H5E__walk(const H5E_t *estack, H5E_direction_t direction, const H5E_walk_op_t *op, diff --git a/src/H5Eprivate.h b/src/H5Eprivate.h index de812f33a71..fd86e96806b 100644 --- a/src/H5Eprivate.h +++ b/src/H5Eprivate.h @@ -188,6 +188,6 @@ H5_DLL herr_t H5E_init(void); H5_DLL herr_t H5E_printf_stack(H5E_t *estack, const char *file, const char *func, unsigned line, hid_t cls_id, hid_t maj_id, hid_t min_id, const char *fmt, ...) H5_ATTR_FORMAT(printf, 8, 9); H5_DLL herr_t H5E_clear_stack(H5E_t *estack); -H5_DLL herr_t H5E_dump_api_stack(bool is_api); +H5_DLL herr_t H5E_dump_api_stack(void); #endif /* H5Eprivate_H */ diff --git a/src/H5private.h b/src/H5private.h index aa8c20537e4..7a69b127b4a 100644 --- a/src/H5private.h +++ b/src/H5private.h @@ -1523,7 +1523,7 @@ H5_DLL herr_t H5CX_pop(bool update_dxpl_props); } \ H5_POP_FUNC \ if (err_occurred) \ - (void)H5E_dump_api_stack(true); \ + (void)H5E_dump_api_stack(); \ FUNC_LEAVE_API_THREADSAFE \ return (ret_value); \ } \ @@ -1535,7 +1535,7 @@ H5_DLL herr_t H5CX_pop(bool update_dxpl_props); } /*end scope from end of FUNC_ENTER*/ \ H5_POP_FUNC \ if (err_occurred) \ - (void)H5E_dump_api_stack(true); \ + (void)H5E_dump_api_stack(); \ FUNC_LEAVE_API_THREADSAFE \ return (ret_value); \ } \ @@ -1558,7 +1558,7 @@ H5_DLL herr_t H5CX_pop(bool update_dxpl_props); ; \ } /*end scope from end of FUNC_ENTER*/ \ if (err_occurred) \ - (void)H5E_dump_api_stack(true); \ + (void)H5E_dump_api_stack(); \ FUNC_LEAVE_API_THREADSAFE \ return (ret_value); \ } \