NetCDF
4.3.2
|
In order to make this process concrete, let us assume we plan to add an in-memory implementation of netcdf-3.
Define a –enable flag and an AM_CONFIGURE flag in configure.ac. We will use the flags –enable-netcdfm and USE_NETCDFM respectively.
Choose some prefix of characters to identify the new dispatch system. In effect we are defining a name-space. For our in-memory system, we will choose "NCM" and "ncm". NCM is used for non-static procedures to be entered into the dispatch table and ncm for all other non-static procedures.
Modify file libdispatch/ncdispatch.h as follows.
Add a index for this implementation:
#define NC_DISPATCH_NCM 5
Define an external reference to the in-memory dispatch table.
#ifdef USE_NETCDFM extern NC_Dispatch* NCM_dispatch_table; #endif
Modify file libdispatch/netcdf.c as follows.
Add a ptr to the in-memory dispatch table.
#ifdef USE_NETCDFM NC_Dispatch* NCM_dispatch_table = NULL; #endif
Add includes for any necessary header files as needed.
Define the functions necessary to fill in the dispatch table. As a rule, we assume that a new directory is defined, libsrcm, say. Within this directory, we need to define Makefile.am, the source files containing the dispatch table and the functions to be placed in the dispatch table – call them ncmdispatch.c and ncmdispatch.h. Look at libsrc/nc3dispatch.[ch] for an example.
As part of the ncmdispatch.c file, you must define the following.
NC_Dispatch NCM_dispatcher = { NC_DISPATCH_NCM, NCM_create, NCM_open, ... }; int NCM_initialize(void) { NCM_dispatch_table = &NCM_dispatcher; return NC_NOERR; }
Assuming that the in-memory library does not require any external libraries, then the Makefile.am will look something like this.
NCM_SOURCES = ncmdispatch.c ncmdispatch.h ... AM_CPPFLAGS += -I$(top_srcdir)/libsrc -I$(top_srcdir)/libdispatch libnetcdfm_la_SOURCES = $(NCM_SOURCES) noinst_LTLIBRARIES = libnetcdfm.la
Provide for the inclusion of this library in the final libnetcdf library. This is accomplished by modifying liblib/Makefile.am by adding something like the following.
if USE_NETCDFM
libnetcdf_la_LIBADD += $(top_builddir)/libsrcm/libnetcdfm.la
endif
Modify the NC_intialize function in liblib/stub.c by adding appropriate references to the NCM dispatch function.
#ifdef USE_NETCDFM extern int NCM_initialize(void); #endif ... int NC_initialize(void) { ... #ifdef USE_DAP if((stat = NCM_initialize())) return stat; #endif ... }
Add a directory of tests; ncm_test, say. The file ncm_test/Makefile.am will look something like this.
# These files are created by the tests. CLEANFILES = ... # These are the tests which are always run. TESTPROGRAMS = test1 test2 ... test1_SOURCES = test1.c ... ... # Set up the tests. check_PROGRAMS = $(TESTPROGRAMS) TESTS = $(TESTPROGRAMS) # Any extra files required by the tests EXTRA_DIST = ...
Provide for libnetcdfm to be constructed by adding the following to the top-level Makefile.am.
if USE_NETCDFM
NCM=libsrcm
NCMTESTDIR=ncm_test
endif
...
SUBDIRS = ... $(DISPATCHDIR) $(NCM) ... $(NCMTESTDIR)
The dispatch table is chosen in the NC_create and the NC_open procedures in libdispatch/netcdf.c. The decision is currently based on the following pieces of information.
The file path – this can be used to detect, for example, a DAP url versus a normal file system file.
The mode argument – this can be used to detect, for example, what kind of file to create: netcdf-3, netcdf-4, 64-bit netcdf-3, etc. For nc_open and when the file path references a real file, the contents of the file can also be used to determine the dispatch table. Although currently not used, this code could be modified to also use other pieces of information such as environment variables.
In addition to the above, there is one additional mechanism to force the use of a specific dispatch table. The procedure "NC_set_dispatch_override()" can be invoked to specify a dispatch table.
When adding a new dispatcher, it is necessary to modify NC_create and NC_open in libdispatch/netcdf.c to detect when it is appropriate to use the NCM dispatcher. Some possibilities are as follows.
Several of the entries in the dispatch table are significantly different than those of the external API.
The create table entry and the open table entry have the following signatures respectively.
int (*create)(const char *path, int cmode, size_t initialsz, int basepe, size_t *chunksizehintp, int useparallel, MPI_Comm comm, MPI_Info info, struct NC_Dispatch*, struct NC** ncp);
int (*open)(const char *path, int mode, int basepe, size_t *chunksizehintp, int use_parallel, MPI_Comm comm, MPI_Info info, NC_Dispatch*, NC** ncp);
The key difference is that these are the union of all the possible create/open signatures from the netcdf.h API. Note especially the last two parameters. The dispatch table is included in case the create function (e.g. NCM_create) needs to invoke other dispatch functions. The very last parameter is a pointer to a pointer to an NC instance. It is expected that the create function will allocate and fill in an instance of an "NC" object and return a pointer to it in the ncp parameter.