2004-08-18 03:54:42 +08:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
2007-02-07 22:56:24 +08:00
|
|
|
* Copyright by The HDF Group. *
|
2003-10-23 05:43:34 +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. *
|
2003-10-23 05:43:34 +08:00
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2006-03-23 02:11:24 +08:00
|
|
|
#include <string.h>
|
2003-12-05 03:35:33 +08:00
|
|
|
#include "h5repack.h"
|
2007-04-05 04:25:42 +08:00
|
|
|
#include "h5tools_utils.h"
|
2003-10-23 05:43:34 +08:00
|
|
|
|
2006-10-05 22:35:40 +08:00
|
|
|
extern char *progname;
|
2003-10-23 05:43:34 +08:00
|
|
|
|
2004-01-08 23:53:32 +08:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: init_packobject
|
|
|
|
*
|
|
|
|
* Purpose: initialize a pack_info_t structure
|
|
|
|
*
|
|
|
|
* Return: void
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
void init_packobject(pack_info_t *obj)
|
|
|
|
{
|
|
|
|
int j, k;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2004-01-08 23:53:32 +08:00
|
|
|
strcpy(obj->path,"\0");
|
|
|
|
for ( j=0; j<H5_REPACK_MAX_NFILTERS; j++)
|
|
|
|
{
|
2004-07-21 03:21:03 +08:00
|
|
|
obj->filter[j].filtn = -1;
|
2005-08-14 04:53:35 +08:00
|
|
|
for ( k=0; k<CDVALUES; k++)
|
2004-01-08 23:53:32 +08:00
|
|
|
obj->filter[j].cd_values[k] = -1;
|
|
|
|
}
|
|
|
|
obj->chunk.rank = -1;
|
|
|
|
obj->refobj_id = -1;
|
2004-04-18 12:10:09 +08:00
|
|
|
obj->layout = H5D_LAYOUT_ERROR;
|
2004-01-08 23:53:32 +08:00
|
|
|
obj->nfilters = 0;
|
|
|
|
}
|
|
|
|
|
2004-01-07 01:49:00 +08:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: aux_tblinsert_filter
|
|
|
|
*
|
|
|
|
* Purpose: auxiliary function, inserts the filter in object OBJS[ I ]
|
|
|
|
*
|
|
|
|
* Return: void
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2005-08-14 04:53:35 +08:00
|
|
|
static void aux_tblinsert_filter(pack_opttbl_t *table,
|
2006-12-07 23:48:19 +08:00
|
|
|
unsigned int I,
|
2004-01-08 23:53:32 +08:00
|
|
|
filter_info_t filt)
|
2004-01-07 01:49:00 +08:00
|
|
|
{
|
|
|
|
if (table->objs[ I ].nfilters<H5_REPACK_MAX_NFILTERS)
|
|
|
|
{
|
|
|
|
table->objs[ I ].filter[ table->objs[ I ].nfilters++ ] = filt;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-10-05 22:35:40 +08:00
|
|
|
error_msg(progname, "cannot insert the filter in this object.\
|
2004-01-07 01:49:00 +08:00
|
|
|
Maximum capacity exceeded\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-16 03:25:27 +08:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: aux_tblinsert_layout
|
|
|
|
*
|
|
|
|
* Purpose: auxiliary function, inserts the layout in object OBJS[ I ]
|
|
|
|
*
|
|
|
|
* Return: void
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2005-08-14 04:53:35 +08:00
|
|
|
static void aux_tblinsert_layout(pack_opttbl_t *table,
|
2006-12-07 23:48:19 +08:00
|
|
|
unsigned int I,
|
2004-07-16 03:25:27 +08:00
|
|
|
pack_info_t *pack)
|
|
|
|
{
|
|
|
|
int k;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2004-07-16 03:25:27 +08:00
|
|
|
table->objs[I].layout = pack->layout;
|
2005-08-14 04:53:35 +08:00
|
|
|
if (H5D_CHUNKED==pack->layout)
|
2004-08-18 03:54:42 +08:00
|
|
|
{
|
|
|
|
/* -2 means the NONE option, remove chunking
|
|
|
|
and set the layout to contiguous */
|
|
|
|
if (pack->chunk.rank==-2)
|
|
|
|
{
|
|
|
|
table->objs[I].layout = H5D_CONTIGUOUS;
|
|
|
|
table->objs[I].chunk.rank = -2;
|
|
|
|
}
|
|
|
|
/* otherwise set the chunking type */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
table->objs[I].chunk.rank = pack->chunk.rank;
|
2005-08-14 04:53:35 +08:00
|
|
|
for (k = 0; k < pack->chunk.rank; k++)
|
2004-08-18 03:54:42 +08:00
|
|
|
table->objs[I].chunk.chunk_lengths[k] = pack->chunk.chunk_lengths[k];
|
|
|
|
}
|
2004-07-16 03:25:27 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-07 01:49:00 +08:00
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: aux_inctable
|
|
|
|
*
|
|
|
|
* Purpose: auxiliary function, increases the size of the collection by N_OBJS
|
|
|
|
*
|
|
|
|
* Return: 0, ok, -1, fail
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int aux_inctable(pack_opttbl_t *table, int n_objs )
|
|
|
|
{
|
2006-11-17 04:38:05 +08:00
|
|
|
unsigned int i;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2004-01-07 01:49:00 +08:00
|
|
|
table->size += n_objs;
|
|
|
|
table->objs = (pack_info_t*)realloc(table->objs, table->size * sizeof(pack_info_t));
|
|
|
|
if (table->objs==NULL) {
|
2006-10-05 22:35:40 +08:00
|
|
|
error_msg(progname, "not enough memory for options table\n");
|
2004-01-07 01:49:00 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
for (i = table->nelems; i < table->size; i++)
|
2004-01-07 01:49:00 +08:00
|
|
|
{
|
2004-07-21 03:21:03 +08:00
|
|
|
init_packobject(&table->objs[i]);
|
2004-01-07 01:49:00 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2003-10-23 05:43:34 +08:00
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: options_table_init
|
|
|
|
*
|
|
|
|
* Purpose: init options table
|
|
|
|
*
|
|
|
|
* Return: 0, ok, -1, fail
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2003-10-29 01:40:05 +08:00
|
|
|
int options_table_init( pack_opttbl_t **tbl )
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
2006-11-17 04:38:05 +08:00
|
|
|
unsigned int i;
|
2003-10-29 01:40:05 +08:00
|
|
|
pack_opttbl_t* table = (pack_opttbl_t*) malloc(sizeof(pack_opttbl_t));
|
2003-10-23 05:43:34 +08:00
|
|
|
if (table==NULL) {
|
2006-10-05 22:35:40 +08:00
|
|
|
error_msg(progname, "not enough memory for options table\n");
|
2003-10-23 05:43:34 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-24 03:51:51 +08:00
|
|
|
table->size = 30;
|
2003-10-23 05:43:34 +08:00
|
|
|
table->nelems = 0;
|
|
|
|
table->objs = (pack_info_t*) malloc(table->size * sizeof(pack_info_t));
|
|
|
|
if (table->objs==NULL) {
|
2006-10-05 22:35:40 +08:00
|
|
|
error_msg(progname, "not enough memory for options table\n");
|
2003-10-23 05:43:34 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
|
|
|
for ( i=0; i<table->size; i++)
|
2004-01-07 01:49:00 +08:00
|
|
|
{
|
2004-07-21 03:21:03 +08:00
|
|
|
init_packobject(&table->objs[i]);
|
2004-01-07 01:49:00 +08:00
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
*tbl = table;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: options_table_free
|
|
|
|
*
|
|
|
|
* Purpose: free table memory
|
|
|
|
*
|
|
|
|
* Return: 0
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2003-10-29 01:40:05 +08:00
|
|
|
int options_table_free( pack_opttbl_t *table )
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
|
|
|
free(table->objs);
|
|
|
|
free(table);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
2003-12-31 06:59:33 +08:00
|
|
|
* Function: options_add_layout
|
2003-10-23 05:43:34 +08:00
|
|
|
*
|
2003-12-31 06:59:33 +08:00
|
|
|
* Purpose: add a layout option to the option list
|
2003-10-23 05:43:34 +08:00
|
|
|
*
|
|
|
|
* Return: 0, ok, -1, fail
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2003-12-31 06:59:33 +08:00
|
|
|
int options_add_layout( obj_list_t *obj_list,
|
|
|
|
int n_objs,
|
|
|
|
pack_info_t *pack,
|
|
|
|
pack_opttbl_t *table )
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
2006-12-07 23:48:19 +08:00
|
|
|
unsigned int i, I;
|
|
|
|
int j, added=0, found=0;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2004-01-07 01:49:00 +08:00
|
|
|
/* increase the size of the collection by N_OBJS if necessary */
|
2005-08-14 04:53:35 +08:00
|
|
|
if (table->nelems+n_objs >= table->size)
|
2004-01-07 01:49:00 +08:00
|
|
|
{
|
|
|
|
if (aux_inctable(table,n_objs)<0)
|
2003-10-23 05:43:34 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
/* search if this object is already in the table; "path" is the key */
|
|
|
|
if (table->nelems>0)
|
|
|
|
{
|
|
|
|
/* go tru the supplied list of names */
|
2005-08-14 04:53:35 +08:00
|
|
|
for (j = 0; j < n_objs; j++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
|
|
|
/* linear table search */
|
2005-08-14 04:53:35 +08:00
|
|
|
for (i = 0; i < table->nelems; i++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
|
|
|
/*already on the table */
|
|
|
|
if (strcmp(obj_list[j].obj,table->objs[i].path)==0)
|
|
|
|
{
|
|
|
|
/* already chunk info inserted for this one; exit */
|
|
|
|
if (table->objs[i].chunk.rank>0)
|
|
|
|
{
|
2006-10-05 22:35:40 +08:00
|
|
|
error_msg(progname, "chunk information already inserted for <%s>\n",obj_list[j].obj);
|
2003-10-23 05:43:34 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
2003-12-31 06:59:33 +08:00
|
|
|
/* insert the layout info */
|
2003-10-23 05:43:34 +08:00
|
|
|
else
|
|
|
|
{
|
2004-07-16 03:25:27 +08:00
|
|
|
aux_tblinsert_layout(table,i,pack);
|
2003-10-23 05:43:34 +08:00
|
|
|
found=1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} /* if */
|
|
|
|
} /* i */
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
if (found==0)
|
|
|
|
{
|
|
|
|
/* keep the grow in a temp var */
|
2005-08-14 04:53:35 +08:00
|
|
|
I = table->nelems + added;
|
2003-10-23 05:43:34 +08:00
|
|
|
added++;
|
|
|
|
strcpy(table->objs[I].path,obj_list[j].obj);
|
2004-07-16 03:25:27 +08:00
|
|
|
aux_tblinsert_layout(table,I,pack);
|
2003-12-31 06:59:33 +08:00
|
|
|
}
|
2004-07-16 03:25:27 +08:00
|
|
|
/* cases where we have an already inserted name but there is a new name also
|
|
|
|
example:
|
2005-08-14 04:53:35 +08:00
|
|
|
-f dset1:GZIP=1 -l dset1,dset2:CHUNK=20x20
|
|
|
|
dset1 is already inserted, but dset2 must also be
|
2004-07-16 03:25:27 +08:00
|
|
|
*/
|
|
|
|
else if (found==1 && strcmp(obj_list[j].obj,table->objs[i].path)!=0)
|
|
|
|
{
|
|
|
|
/* keep the grow in a temp var */
|
2005-08-14 04:53:35 +08:00
|
|
|
I = table->nelems + added;
|
2004-07-16 03:25:27 +08:00
|
|
|
added++;
|
|
|
|
strcpy(table->objs[I].path,obj_list[j].obj);
|
|
|
|
aux_tblinsert_layout(table,I,pack);
|
2003-10-23 05:43:34 +08:00
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
} /* j */
|
2003-10-23 05:43:34 +08:00
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
/* first time insertion */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* go tru the supplied list of names */
|
2005-08-14 04:53:35 +08:00
|
|
|
for (j = 0; j < n_objs; j++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
2005-08-14 04:53:35 +08:00
|
|
|
I = table->nelems + added;
|
2003-10-23 05:43:34 +08:00
|
|
|
added++;
|
|
|
|
strcpy(table->objs[I].path,obj_list[j].obj);
|
2004-07-16 03:25:27 +08:00
|
|
|
aux_tblinsert_layout(table,I,pack);
|
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
}
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
table->nelems+= added;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
2003-12-30 04:26:21 +08:00
|
|
|
* Function: options_add_filter
|
2003-10-23 05:43:34 +08:00
|
|
|
*
|
2003-12-30 04:26:21 +08:00
|
|
|
* Purpose: add a compression -f option to the option list
|
2003-10-23 05:43:34 +08:00
|
|
|
*
|
|
|
|
* Return: 0, ok, -1, fail
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2003-12-30 04:26:21 +08:00
|
|
|
int options_add_filter(obj_list_t *obj_list,
|
|
|
|
int n_objs,
|
|
|
|
filter_info_t filt,
|
|
|
|
pack_opttbl_t *table )
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2006-12-07 23:48:19 +08:00
|
|
|
unsigned int i, I;
|
|
|
|
int j, added=0, found=0;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2004-01-07 01:49:00 +08:00
|
|
|
/* increase the size of the collection by N_OBJS if necessary */
|
2005-08-14 04:53:35 +08:00
|
|
|
if (table->nelems+n_objs >= table->size)
|
2004-01-07 01:49:00 +08:00
|
|
|
{
|
|
|
|
if (aux_inctable(table,n_objs)<0)
|
2003-10-23 05:43:34 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
/* search if this object is already in the table; "path" is the key */
|
|
|
|
if (table->nelems>0)
|
|
|
|
{
|
|
|
|
/* go tru the supplied list of names */
|
2005-08-14 04:53:35 +08:00
|
|
|
for (j = 0; j < n_objs; j++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
|
|
|
/* linear table search */
|
2005-08-14 04:53:35 +08:00
|
|
|
for (i = 0; i < table->nelems; i++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
|
|
|
/*already on the table */
|
|
|
|
if (strcmp(obj_list[j].obj,table->objs[i].path)==0)
|
|
|
|
{
|
2003-12-30 04:26:21 +08:00
|
|
|
/* insert */
|
2004-01-07 01:49:00 +08:00
|
|
|
aux_tblinsert_filter(table,i,filt);
|
2003-12-30 04:26:21 +08:00
|
|
|
found=1;
|
|
|
|
break;
|
2003-10-23 05:43:34 +08:00
|
|
|
} /* if */
|
|
|
|
} /* i */
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
if (found==0)
|
|
|
|
{
|
|
|
|
/* keep the grow in a temp var */
|
2005-08-14 04:53:35 +08:00
|
|
|
I = table->nelems + added;
|
2004-07-16 03:25:27 +08:00
|
|
|
added++;
|
|
|
|
strcpy(table->objs[I].path,obj_list[j].obj);
|
|
|
|
aux_tblinsert_filter(table,I,filt);
|
|
|
|
}
|
|
|
|
/* cases where we have an already inserted name but there is a new name also
|
|
|
|
example:
|
|
|
|
-l dset1:CHUNK=20x20 -f dset1,dset2:GZIP=1
|
2005-08-14 04:53:35 +08:00
|
|
|
dset1 is already inserted, but dset2 must also be
|
2004-07-16 03:25:27 +08:00
|
|
|
*/
|
|
|
|
else if (found==1 && strcmp(obj_list[j].obj,table->objs[i].path)!=0)
|
|
|
|
{
|
|
|
|
/* keep the grow in a temp var */
|
2005-08-14 04:53:35 +08:00
|
|
|
I = table->nelems + added;
|
2003-10-23 05:43:34 +08:00
|
|
|
added++;
|
|
|
|
strcpy(table->objs[I].path,obj_list[j].obj);
|
2004-01-07 01:49:00 +08:00
|
|
|
aux_tblinsert_filter(table,I,filt);
|
2003-10-23 05:43:34 +08:00
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
} /* j */
|
2003-10-23 05:43:34 +08:00
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
/* first time insertion */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* go tru the supplied list of names */
|
2005-08-14 04:53:35 +08:00
|
|
|
for (j = 0; j < n_objs; j++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
2005-08-14 04:53:35 +08:00
|
|
|
I = table->nelems + added;
|
2003-10-23 05:43:34 +08:00
|
|
|
added++;
|
|
|
|
strcpy(table->objs[I].path,obj_list[j].obj);
|
2004-01-07 01:49:00 +08:00
|
|
|
aux_tblinsert_filter(table,I,filt);
|
2003-10-23 05:43:34 +08:00
|
|
|
}
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
table->nelems+= added;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
* Function: options_get_object
|
|
|
|
*
|
|
|
|
* Purpose: get object from table; "path" is the key
|
|
|
|
*
|
|
|
|
* Return: pack_info_t* OBJECT or NULL if not found; PATH is the key
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2003-11-11 04:59:32 +08:00
|
|
|
pack_info_t* options_get_object( const char *path,
|
2003-10-29 01:40:05 +08:00
|
|
|
pack_opttbl_t *table )
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
2006-11-17 04:38:05 +08:00
|
|
|
unsigned int i;
|
2005-08-14 04:53:35 +08:00
|
|
|
|
|
|
|
for ( i = 0; i < table->nelems; i++)
|
2003-10-23 05:43:34 +08:00
|
|
|
{
|
|
|
|
/* found it */
|
|
|
|
if (strcmp(table->objs[i].path,path)==0)
|
|
|
|
{
|
|
|
|
return (&table->objs[i]);
|
|
|
|
}
|
|
|
|
}
|
2005-08-14 04:53:35 +08:00
|
|
|
|
2003-10-23 05:43:34 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|