summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorQuincey Koziol <koziol@hdfgroup.org>2002-10-14 19:56:13 (GMT)
committerQuincey Koziol <koziol@hdfgroup.org>2002-10-14 19:56:13 (GMT)
commit754cf917b499d4774e8b0495755741f21e29037e (patch)
treef852dfe74382c9d0c04e6a513605648113f23778
parent4629204d864aa936baca552d742e5d1c2bd1f15c (diff)
downloadhdf5-754cf917b499d4774e8b0495755741f21e29037e.zip
hdf5-754cf917b499d4774e8b0495755741f21e29037e.tar.gz
hdf5-754cf917b499d4774e8b0495755741f21e29037e.tar.bz2
[svn-r5992] Purpose:
Bug fix Description: Generic property classes where incorrectly reference counting their users, in certain cases, causing a memory leak. Solution: Cleanup reference counting code & unify in H5P_access_class() Platforms tested: FreeBSD 4.6 (sleipnir) w and w/o parallel Linux 2.2.x (eirene) w/FORTRAN & C++ Solaris 2.7 (arabica) w/FORTRAN IRIX64 6.5 (modi4) w/FORTRAN & parallel
-rw-r--r--src/H5P.c36
-rw-r--r--src/H5Ppkg.h1
2 files changed, 13 insertions, 24 deletions
diff --git a/src/H5P.c b/src/H5P.c
index 09f1ff5..e05ef3a 100644
--- a/src/H5P.c
+++ b/src/H5P.c
@@ -402,11 +402,6 @@ H5P_copy_pclass(H5P_genclass_t *pclass)
} /* end for */
} /* end if */
- /* Increment parent class's derived class value */
- if(new_pclass->parent!=NULL)
- if(H5P_access_class(new_pclass->parent,H5P_MOD_INC_CLS)<0)
- HGOTO_ERROR (H5E_PLIST, H5E_CANTINIT, NULL,"Can't increment parent class ref count");
-
/* Set the return value */
ret_value=new_pclass;
@@ -1007,16 +1002,20 @@ H5P_access_class(H5P_genclass_t *pclass, H5P_class_mod_t mod)
break;
case H5P_MOD_INC_REF: /* Increment the ID reference count*/
+ /* Reset the deleted flag if incrementing the reference count */
+ if(pclass->deleted)
+ pclass->deleted=0;
pclass->ref_count++;
break;
case H5P_MOD_DEC_REF: /* Decrement the ID reference count*/
pclass->ref_count--;
- break;
- case H5P_MOD_CHECK: /* NOOP, just check if we can delete the class */
+ /* Mark the class object as deleted if reference count drops to zero */
+ if(pclass->ref_count==0)
+ pclass->deleted=1;
break;
-
+
case H5P_MOD_ERR:
case H5P_MOD_MAX:
assert(0 && "Invalid H5P class modification");
@@ -1024,6 +1023,8 @@ H5P_access_class(H5P_genclass_t *pclass, H5P_class_mod_t mod)
/* Check if we can release the class information now */
if(pclass->deleted && pclass->plists==0 && pclass->classes==0 ) {
+ H5P_genclass_t *par_class=pclass->parent; /* Pointer to class's parent */
+
assert(pclass->name);
H5MM_xfree(pclass->name);
@@ -1031,6 +1032,10 @@ H5P_access_class(H5P_genclass_t *pclass, H5P_class_mod_t mod)
H5P_free_all_prop(pclass->props,pclass->hashsize,0);
H5MM_xfree(pclass);
+
+ /* Reduce the number of dependent classes on parent class also */
+ if(par_class!=NULL)
+ H5P_access_class(par_class, H5P_MOD_DEC_CLS);
} /* end if */
FUNC_LEAVE (SUCCEED);
@@ -5029,21 +5034,6 @@ H5P_close_class(void *_pclass)
if(H5P_access_class(pclass,H5P_MOD_DEC_REF)<0)
HGOTO_ERROR (H5E_PLIST, H5E_NOTFOUND, FAIL, "Can't decrement ID ref count");
- /* Check if the object should go away */
- if(pclass->ref_count==0) {
- /* Decrement parent class's dependant property class value! */
- if(pclass->parent)
- if (H5P_access_class(pclass->parent, H5P_MOD_DEC_CLS) < 0)
- HGOTO_ERROR (H5E_PLIST, H5E_NOTFOUND, FAIL,"Can't decrement class ref count");
-
- /* Mark class as deleted */
- pclass->deleted = 1;
-
- /* Check dependancies on this class, deleting it if allowed */
- if (H5P_access_class(pclass, H5P_MOD_CHECK) < 0)
- HGOTO_ERROR (H5E_PLIST, H5E_NOTFOUND, FAIL,"Can't check class ref count");
- } /* end if */
-
done:
FUNC_LEAVE (ret_value);
} /* H5P_close_class() */
diff --git a/src/H5Ppkg.h b/src/H5Ppkg.h
index a8609f3..475bfef 100644
--- a/src/H5Ppkg.h
+++ b/src/H5Ppkg.h
@@ -34,7 +34,6 @@ typedef enum {
H5P_MOD_DEC_LST, /* Decrement the dependant list count*/
H5P_MOD_INC_REF, /* Increment the ID reference count*/
H5P_MOD_DEC_REF, /* Decrement the ID reference count*/
- H5P_MOD_CHECK, /* Just check about deleting the class */
H5P_MOD_MAX /* Upper limit on class modifications */
} H5P_class_mod_t;