2004-06-11 01:35:48 +08:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
2007-02-07 22:56:24 +08:00
|
|
|
* Copyright by The HDF Group. *
|
2004-06-11 01:35:48 +08:00
|
|
|
* Copyright by the Board of Trustees of the University of Illinois. *
|
|
|
|
* All rights reserved. *
|
|
|
|
* *
|
|
|
|
* This file is part of HDF5. The full HDF5 copyright notice, including *
|
|
|
|
* terms governing use, modification, and redistribution, is contained in *
|
|
|
|
* the files COPYING and Copyright.html. COPYING can be found at the root *
|
|
|
|
* of the source code distribution tree; Copyright.html can be found at the *
|
|
|
|
* root level of an installed copy of the electronic HDF5 document set and *
|
|
|
|
* is linked from the top-level documents page. It can also be found at *
|
2007-02-07 22:56:24 +08:00
|
|
|
* http://hdfgroup.org/HDF5/doc/Copyright.html. If you do not have *
|
|
|
|
* access to either file, you may request a copy from help@hdfgroup.org. *
|
2004-06-11 01:35:48 +08:00
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "h5tools_ref.h"
|
|
|
|
#include "H5private.h"
|
2005-08-26 04:16:40 +08:00
|
|
|
#include "H5SLprivate.h"
|
2004-06-11 01:35:48 +08:00
|
|
|
#include "h5tools.h"
|
|
|
|
#include "h5tools_utils.h"
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Table to look up a path name for an object
|
|
|
|
* reference.
|
|
|
|
*
|
|
|
|
* This table stores mappings of reference -> path
|
|
|
|
* for all objects in the file that may be the target of
|
|
|
|
* an object reference.
|
|
|
|
*
|
|
|
|
* The 'path' is an absolute path by which the object
|
|
|
|
* can be accessed. When an object has > 1 such path,
|
|
|
|
* only one will be used in the table, with no particular
|
|
|
|
* method of selecting which one.
|
|
|
|
*/
|
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
typedef struct {
|
|
|
|
haddr_t objno; /* Object ID (i.e. address) */
|
|
|
|
const char *path; /* Object path */
|
|
|
|
} ref_path_node_t;
|
|
|
|
|
|
|
|
static H5SL_t *ref_path_table = NULL; /* the "table" (implemented with a skip list) */
|
|
|
|
static hid_t thefile;
|
2004-06-11 01:35:48 +08:00
|
|
|
|
|
|
|
extern char *progname;
|
|
|
|
extern int d_status;
|
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
static int ref_path_table_put(const char *, haddr_t objno);
|
|
|
|
static hbool_t ref_path_table_find(haddr_t objno);
|
2007-09-13 23:44:56 +08:00
|
|
|
static herr_t fill_ref_path_table_cb(hid_t group, const char *obj_name,
|
|
|
|
const H5L_info_t *linfo, void *op_data);
|
2005-08-26 04:16:40 +08:00
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: free_ref_path_info
|
|
|
|
*
|
|
|
|
* Purpose: Free the key for a reference path table node
|
|
|
|
*
|
|
|
|
* Return: Non-negative on success, negative on failure
|
|
|
|
*
|
|
|
|
* Programmer: Quincey Koziol
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
static herr_t
|
|
|
|
free_ref_path_info(void *item, void UNUSED *key, void UNUSED *operator_data/*in,out*/)
|
|
|
|
{
|
|
|
|
ref_path_node_t *node = (ref_path_node_t *)item;
|
|
|
|
|
2005-09-13 11:21:00 +08:00
|
|
|
HDfree((void *)node->path);
|
2005-08-26 04:16:40 +08:00
|
|
|
HDfree(node);
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: term_ref_path_table
|
|
|
|
*
|
|
|
|
* Purpose: Terminate the reference path table
|
|
|
|
*
|
|
|
|
* Return: Non-negative on success, negative on failure
|
|
|
|
*
|
|
|
|
* Programmer: Quincey Koziol
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
term_ref_path_table(void)
|
|
|
|
{
|
|
|
|
/* Destroy reference path table, freeing all memory */
|
|
|
|
if(ref_path_table)
|
|
|
|
H5SL_destroy(ref_path_table, free_ref_path_info, NULL);
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
2004-06-11 01:35:48 +08:00
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: ref_path_table_lookup
|
|
|
|
*
|
|
|
|
* Purpose: Looks up a table entry given a path name.
|
|
|
|
* Used during construction of the table.
|
|
|
|
*
|
|
|
|
* Return: The table entre (pte) or NULL if not in the
|
|
|
|
* table.
|
|
|
|
*
|
|
|
|
* Programmer: REMcG
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
2005-08-26 04:16:40 +08:00
|
|
|
haddr_t
|
2004-06-11 01:35:48 +08:00
|
|
|
ref_path_table_lookup(const char *thepath)
|
|
|
|
{
|
2007-09-13 23:44:56 +08:00
|
|
|
H5O_info_t oi;
|
|
|
|
|
|
|
|
/* Allow lookups on the root group, even though it doesn't have any link info */
|
|
|
|
if(HDstrcmp(thepath, "/")) {
|
|
|
|
H5L_info_t li;
|
|
|
|
|
|
|
|
/* Check for external link first, so we don't return the OID of an object in another file */
|
|
|
|
if(H5Lget_info(thefile, thepath, &li, H5P_DEFAULT) < 0)
|
|
|
|
return HADDR_UNDEF;
|
|
|
|
|
|
|
|
/* UD links can't be followed, so they always "dangle" like soft links. */
|
|
|
|
if(li.type >= H5L_TYPE_UD_MIN)
|
2006-01-24 04:46:34 +08:00
|
|
|
return HADDR_UNDEF;
|
|
|
|
} /* end if */
|
|
|
|
|
2007-09-13 23:44:56 +08:00
|
|
|
/* Get the object info now */
|
|
|
|
/* (returns failure for dangling soft links) */
|
|
|
|
if(H5Oget_info(thefile, thepath, &oi, H5P_DEFAULT) < 0)
|
|
|
|
return HADDR_UNDEF;
|
2006-01-24 04:46:34 +08:00
|
|
|
|
|
|
|
/* All existing objects in the file had better be in the table */
|
2007-09-13 23:44:56 +08:00
|
|
|
HDassert(ref_path_table_find(oi.addr));
|
2006-01-24 04:46:34 +08:00
|
|
|
|
|
|
|
/* Return OID */
|
2007-09-13 23:44:56 +08:00
|
|
|
return(oi.addr);
|
2005-08-26 04:16:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: ref_path_table_find
|
|
|
|
*
|
|
|
|
* Purpose: Looks up a table entry given a object number.
|
|
|
|
* Used during construction of the table.
|
|
|
|
*
|
|
|
|
* Return: TRUE/FALSE on success, can't fail
|
|
|
|
*
|
|
|
|
* Programmer: Quincey Koziol
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
static hbool_t
|
|
|
|
ref_path_table_find(haddr_t objno)
|
|
|
|
{
|
|
|
|
HDassert(ref_path_table);
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
if(H5SL_search(ref_path_table, &objno) == NULL)
|
|
|
|
return FALSE;
|
|
|
|
else
|
|
|
|
return TRUE;
|
2004-06-11 01:35:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: ref_path_table_put
|
|
|
|
*
|
2005-08-26 04:16:40 +08:00
|
|
|
* Purpose: Enter the 'obj' with 'path' in the table (assumes its not
|
|
|
|
* already there)
|
|
|
|
*
|
2004-06-11 01:35:48 +08:00
|
|
|
* Create an object reference, pte, and store them
|
|
|
|
* in the table.
|
|
|
|
*
|
2005-08-26 04:16:40 +08:00
|
|
|
* NOTE: Takes ownership of the path name string passed in!
|
|
|
|
*
|
|
|
|
* Return: Non-negative on success, negative on failure
|
2004-06-11 01:35:48 +08:00
|
|
|
*
|
|
|
|
* Programmer: REMcG
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
2005-08-26 04:16:40 +08:00
|
|
|
static int
|
|
|
|
ref_path_table_put(const char *path, haddr_t objno)
|
2004-06-11 01:35:48 +08:00
|
|
|
{
|
2005-08-26 04:16:40 +08:00
|
|
|
ref_path_node_t *new_node;
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
HDassert(ref_path_table);
|
|
|
|
HDassert(path);
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
if((new_node = HDmalloc(sizeof(ref_path_node_t))) == NULL)
|
|
|
|
return(-1);
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
new_node->objno = objno;
|
|
|
|
new_node->path = path;
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
return(H5SL_insert(ref_path_table, new_node, &(new_node->objno)));
|
2004-06-11 01:35:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* counter used to disambiguate multiple instances of same object.
|
|
|
|
*/
|
|
|
|
int xid = 1;
|
|
|
|
|
2006-12-07 03:32:20 +08:00
|
|
|
int get_next_xid(void) {
|
2005-08-26 04:16:40 +08:00
|
|
|
return xid++;
|
2004-06-11 01:35:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This counter is used to create fake object ID's
|
|
|
|
* The idea is to set it to the largest possible offest, which
|
|
|
|
* minimizes the chance of collision with a real object id.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
haddr_t fake_xid = HADDR_MAX;
|
|
|
|
haddr_t
|
2006-12-07 03:32:20 +08:00
|
|
|
get_fake_xid (void) {
|
2005-08-26 04:16:40 +08:00
|
|
|
return (fake_xid--);
|
2004-06-11 01:35:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* for an object that does not have an object id (e.g., soft link),
|
|
|
|
* create a table entry with a fake object id as the key.
|
2005-08-26 04:16:40 +08:00
|
|
|
*
|
|
|
|
* Assumes 'path' is for an object that is not in the table.
|
|
|
|
*
|
2004-06-11 01:35:48 +08:00
|
|
|
*/
|
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
haddr_t
|
2004-06-11 01:35:48 +08:00
|
|
|
ref_path_table_gen_fake(const char *path)
|
|
|
|
{
|
2005-08-26 04:16:40 +08:00
|
|
|
const char *dup_path;
|
|
|
|
haddr_t fake_objno;
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
if((dup_path = HDstrdup(path)) == NULL)
|
|
|
|
return HADDR_UNDEF;
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
/* Generate fake ID for string */
|
|
|
|
fake_objno = get_fake_xid();
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
/* Insert "fake" object into table (takes ownership of path) */
|
|
|
|
ref_path_table_put(dup_path, fake_objno);
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
return(fake_objno);
|
2004-06-11 01:35:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: lookup_ref_path
|
|
|
|
*
|
|
|
|
* Purpose: Lookup the path to the object with refernce 'ref'.
|
|
|
|
*
|
|
|
|
* Return: Return a path to the object, or NULL if not found.
|
|
|
|
*
|
|
|
|
* Programmer: REMcG
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
2005-08-26 04:16:40 +08:00
|
|
|
const char *
|
|
|
|
lookup_ref_path(haddr_t ref)
|
2004-06-11 01:35:48 +08:00
|
|
|
{
|
2005-08-26 04:16:40 +08:00
|
|
|
ref_path_node_t *node;
|
2004-06-11 01:35:48 +08:00
|
|
|
|
2005-08-26 04:16:40 +08:00
|
|
|
/* Be safer for h5ls */
|
|
|
|
if(!ref_path_table)
|
|
|
|
return(NULL);
|
|
|
|
|
|
|
|
node = H5SL_search(ref_path_table, &ref);
|
|
|
|
|
|
|
|
return(node ? node->path : NULL);
|
2004-06-11 01:35:48 +08:00
|
|
|
}
|
|
|
|
|
2007-09-13 23:44:56 +08:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: fill_ref_path_table_cb
|
|
|
|
*
|
|
|
|
* Purpose: Called by interator to create references for
|
|
|
|
* all objects and enter them in the table.
|
|
|
|
*
|
|
|
|
* Return: Error status.
|
|
|
|
*
|
|
|
|
* Programmer: REMcG
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
static herr_t
|
|
|
|
fill_ref_path_table_cb(hid_t group, const char *obj_name, const H5L_info_t *linfo,
|
|
|
|
void *op_data)
|
|
|
|
{
|
|
|
|
if(linfo->type == H5L_TYPE_HARD) {
|
|
|
|
H5O_info_t oinfo;
|
|
|
|
|
|
|
|
H5Oget_info(group, obj_name, &oinfo, H5P_DEFAULT);
|
|
|
|
|
|
|
|
/* Check if the object is already in the path table */
|
|
|
|
if(!ref_path_table_find(oinfo.addr)) {
|
|
|
|
const char *obj_prefix = (const char *)op_data;
|
|
|
|
size_t tmp_len;
|
|
|
|
char *thepath;
|
|
|
|
|
|
|
|
/* Compute length for this object's path */
|
|
|
|
tmp_len = HDstrlen(obj_prefix) + HDstrlen(obj_name) + 2;
|
|
|
|
|
|
|
|
/* Allocate room for the path for this object */
|
|
|
|
if((thepath = (char *)HDmalloc(tmp_len)) == NULL)
|
|
|
|
return FAIL;
|
|
|
|
|
|
|
|
/* Build the name for this object */
|
|
|
|
HDstrcpy(thepath, obj_prefix);
|
|
|
|
HDstrcat(thepath, "/");
|
|
|
|
HDstrcat(thepath, obj_name);
|
|
|
|
|
|
|
|
/* Insert the object into the path table (takes ownership of the path) */
|
|
|
|
ref_path_table_put(thepath, oinfo.addr);
|
|
|
|
|
|
|
|
if(oinfo.type == H5O_TYPE_GROUP) {
|
|
|
|
/* Iterate over objects in this group, using this group's
|
|
|
|
* name as their prefix
|
|
|
|
*/
|
|
|
|
if(H5Literate(group, obj_name, H5_INDEX_NAME, H5_ITER_INC, NULL, fill_ref_path_table_cb, thepath, H5P_DEFAULT) < 0) {
|
|
|
|
error_msg(progname, "unable to dump group \"%s\"\n", thepath);
|
|
|
|
d_status = EXIT_FAILURE;
|
|
|
|
} /* end if */
|
|
|
|
} /* end if */
|
|
|
|
} /* end if */
|
|
|
|
} /* end if */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-06-11 01:35:48 +08:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: fill_ref_path_table
|
|
|
|
*
|
|
|
|
* Purpose: Called by interator to create references for
|
|
|
|
* all objects and enter them in the table.
|
|
|
|
*
|
|
|
|
* Return: Error status.
|
|
|
|
*
|
|
|
|
* Programmer: REMcG
|
|
|
|
*
|
|
|
|
* Modifications:
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
herr_t
|
2007-09-13 23:44:56 +08:00
|
|
|
fill_ref_path_table(hid_t fid)
|
2004-06-11 01:35:48 +08:00
|
|
|
{
|
2007-09-13 23:44:56 +08:00
|
|
|
H5O_info_t oinfo;
|
|
|
|
char *root_path;
|
|
|
|
|
|
|
|
/* Set file ID for later queries (XXX: this should be fixed) */
|
|
|
|
thefile = fid;
|
|
|
|
|
|
|
|
/* Create skip list to store reference path information */
|
|
|
|
if((ref_path_table = H5SL_create(H5SL_TYPE_HADDR, 0.5, (size_t)16))==NULL)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
/* Build the name for root group */
|
|
|
|
root_path = HDstrdup("/");
|
|
|
|
|
|
|
|
/* Get info for root group */
|
|
|
|
H5Oget_info(fid, root_path, &oinfo, H5P_DEFAULT);
|
|
|
|
|
|
|
|
/* Insert the root group into the path table (takes ownership of path) */
|
|
|
|
ref_path_table_put(root_path, oinfo.addr);
|
|
|
|
|
|
|
|
/* Iterate over objects in this file */
|
|
|
|
if(H5Literate(fid, root_path, H5_INDEX_NAME, H5_ITER_INC, NULL, fill_ref_path_table_cb, (void *)"", H5P_DEFAULT) < 0) {
|
|
|
|
error_msg(progname, "unable to dump root group\n");
|
|
|
|
d_status = EXIT_FAILURE;
|
|
|
|
} /* end if */
|
2004-06-11 01:35:48 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|