<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML//EN"> <html> <head> <title> Description of nc_test </title> </head> <body bgcolor="#00009C" text=#FFFF00 link=#FFFFFF alink=#00FF00 vlink=#FFFFFF> <center> <h1> Description of <tt>nc_test</tt> </h1> </a> <h2>Harvey Davies</h2> <h3>7 August 1996</h3> </center> <h3> TERMINOLOGY </h3> <dl> <dt>Test <dd>Process of testing one netCDF library function. <dt>Comparison <dd>Successful match of value of individual variable/attribute element read (possibly after put) with that expected. <dt>Error <dd>Non-zero status returned by netCDF library function. <dt>Failure <dd>Unexpected result (including error status). Possible causes: <ul> <li>absent data file <li>wrong permissions <li>bug in netCDF library function </ul> <dt>Blunder <dd>Bug in <tt>nc_test</tt>. </dl> <h3> PROGRAM OPERATION AND COMMAND-LINE OPTIONS </h3> All printed output is written to <tt>stderr</tt>, not <tt>stdout</tt>. This is intended to facilitate debugging. <p> There is a separate test for each netCDF library function (apart from <tt>test_nc_enddef</tt> which simply calls <tt>test_nc_redef</tt>, which tests both <tt>nc_redef</tt> and <tt>nc_enddef</tt>). Each test generates one line of results if there are no failures and additional lines if there are failures. The simplest such line for no failures is one such as: <pre> *** Testing nc_open ... ok </pre> Many tests count the number of comparisons and if there are no failures print a line such as: <pre> *** Testing nc_get_var1_short ... 682 good comparisons. ok </pre> Each failure produces a line until the limit (per test) specified by the <tt>-n</tt> option (default 8) is reached. E.g. running the full test in a read-only directory produces: <pre> *** Testing nc_open ... FAILURE at line 119 of test_read.c: nc_create: Permission denied FAILURE at line 124 of test_read.c: nc_open: No such file or directory FAILURE at line 129 of test_read.c: remove of scratch.nc failed ### 3 FAILURES TESTING nc_open! ### </pre> The total number of failures is printed at the end. E.g. <pre> Total number of failures: 60 </pre> If there are no failures then <tt>nc_test</tt> exits with exit-status 0, otherwise 1. <p> The following usage instructions are produced by the command <tt>nc_test -h</tt>: <pre> nc_test [-hrv] [-n <MAX_NMPT>] [-h] Print help [-c] Create file test.nc (Do not do tests) [-r] Just do read-only tests [-v] Verbose mode [-n <MAX_NMPT>] max. number of messages per test (Default: 8) </pre> The <tt>-c</tt> option creates the read-only test file test.nc and does nothing else. (Originally I had a separate program, but it was simpler to combine them.) The utility <tt>make</tt> will run <tt>nc_test -c</tt> to recreate <tt>test.nc</tt> whenever <tt>nc_test</tt> is changed, but I suggest including <tt>test.nc</tt> in the distribution so users do not normally have to do this step. <p> As mentioned above, it is possible to run the full test in a read-only directory. (This currently generates 60 failures.) But in a read-only directory it is better to use the <tt>-r</tt> option which suppresses all actions which write, so there should be no failures. <p> The option <tt>-v</tt> sets verbose mode, in which additional details are produced after certain failures. (This is not consistent across tests. An earlier version also produced details of each comparison, whether successful or not; but this proved too voluminous.) <p> The option <tt>-n <i>MAX_NMPT</i></tt> sets the maximum number of messages printed per test. This defaults to 8. <h3> SOURCE FILES </h3> <dl> <dt><tt>nc_test.c</tt> <dd>Driver <dt><tt>tests.h</tt> <dd>Main include file <dt><tt>error.c</tt> <dd>Functions concerned with low-level printing and failure handling. <dt><tt>error.h</tt> <dd>Include file related to error.c. <dt><tt>util.c</tt> <dd>Various utility functions for generating data, comparisons, etc. <dt><tt>test_get.m4</tt> <dd>Read-only type-specific tests. <dt><tt>test_put.m4</tt> <dd>Write type-specific tests. <dt><tt>test_read.c</tt> <dd>Other read-only tests (in fact there is some writing). <dt><tt>test_write.c</tt> <dd>Other write tests. </dl> <p> The <tt>wc</tt> utility gives the following counts of lines, words and characters: <pre> 22 70 622 error.h 324 1000 9226 tests.h 366 1303 11029 test_get.m4 610 1941 16469 test_put.m4 52 155 1159 error.c 277 692 7645 nc_test.c 1599 5908 45979 test_read.c 1882 6887 55625 test_write.c 728 2623 19434 util.c 5860 20579 167188 total </pre> <h3> DATA FILES </h3> Two netCDF files are used. One (<tt>test.nc</tt>) is read-only, except that the <tt>-c</tt> option creates it. The other (<tt>scratch.nc</tt>) has a lifetime within the period of any test which does writing. <p> The file <tt>tests.h</tt> is used merely as an example of a non-netCDF file. <h3> INTERNAL METADATA DESCRIBING TEST DATA </h3> The test metadata defining the file <tt>tests.nc</tt> (and for many tests also <tt>scratch.nc</tt>) is described by parameters (e.g. <tt>NVARS</tt>, the number of variables) defined in <tt>tests.h</tt> and global variables with the prefix <tt>dim</tt>, <tt>var</tt>, <tt>att</tt> or <tt>gatt</tt> (global attribute). E.g. a netCDF variable is described by global variables such as <tt>var_name</tt>, <tt>var_type</tt> and <tt>var_dimid</tt>. Note that the data values of netCDF variables and attributes are not stored in this way. Instead, these are generated by function <tt>hash</tt> each time they are needed. <p> The need for separate global attribute tests is obviated by accessing <tt>att/gatt</tt> variables via macros which give global values for a varid of -1. E.g. <tt>ATT_NAME(varid,j)</tt> gives <tt>gatt_name[j]</tt> if varid is -1, otherwise it gives <tt>att_name[varid][j]</tt>. Thus attribute tests will typically vary <tt>varid</tt> from -1 to <tt>NVARS-1</tt>. <p> The function <tt>init_gvars</tt> (in <tt>util.c</tt>) defines the values of these global variables defining the metadata. It defines variable names such as <tt>sr23</tt> for a 3D short dimensioned <tt>UNLIMITED</tt> (record) x 2 x 3. The maximum dimension size is 4. There is a vector of length 1, 2, 3 and 4 for each data type. However for ranks 2 and 3 (<tt>MAX_RANK</tt>) there is only one type for each shape, but there is an example of each possible shape (including one with the <tt>UNLIMITED</tt> dimension). This produces a total of 136 (<tt>NVARS</tt>) variables. <p> The command <pre> ncdump -h test.nc | grep char </pre> produces the following list of all the character variables: <pre> char c ; char cr(Dr) ; char c1(D1) ; char c2(D2) ; char c3(D3) ; char c4(D4) ; char cr1(Dr, D1) ; char c13(D1, D3) ; char c31(D3, D1) ; char c43(D4, D3) ; char cr21(Dr, D2, D1) ; char cr33(Dr, D3, D3) ; char c111(D1, D1, D1) ; char c123(D1, D2, D3) ; char c141(D1, D4, D1) ; char c213(D2, D1, D3) ; char c231(D2, D3, D1) ; char c243(D2, D4, D3) ; char c321(D3, D2, D1) ; char c333(D3, D3, D3) ; char c411(D4, D1, D1) ; char c423(D4, D2, D3) ; char c441(D4, D4, D1) ; </pre> Given that there are two records (<tt>Dr</tt> = 2 currently), one can use this to calculate the number of character elements as: 1+2+1+2+3+4+2+3+3+12+4+18+1+6+4+6+6+24+6+27+4+24+16 = 179. This corresponds to the number of comparisons for characters given in output such as: <pre> *** Testing nc_get_vara_text ... 179 good comparisons. ok </pre> The number of comparisons for numeric types is more difficult to calculate because it depends on whether the data values are within the ranges for the external and internal types. However one can do quite a number of consistency checks. For example the number of comparisons done by the <tt>(void *)</tt> functions (e.g. <tt>nc_get_var1</tt>) is 1386 which equals the sum of that for <tt>nc_get_var1_double</tt> (1207, the total number of numeric elements) and the above value of 179. <p> There is one global attribute of each data-type and lengths vary from 1 to 6. Only the six scalar variables have attributes and their lengths vary from 0 to 5. <h3> GENERATION OF TEST DATA VALUES </h3> Variable and attribute data values are generated by function <tt>hash</tt> in file <tt>util.c</tt>. The type <tt>double</tt> result is a function of the data-type, rank (-1 for attribute) and index vector. <p> Vectors are treated specially. Index 0 gives the minimum for the type and index 1 gives the maximum. For types other than <tt>NC_CHAR</tt> and <tt>NC_DOUBLE</tt>, index 2 gives a value just less than the minimum and index 3 gives a value just more than the maximum. <p> Otherwise <tt>hash</tt> uses an algorithm which always generates a valid value for the type. <p> There is also a 4-argument wrapper function <tt>hash4</tt> which has an additional argument specifying whether the internal type is <tt>uchar</tt>. This handles the special <tt>NC_BYTE/uchar</tt> adjustment. <h3> ncdump of <tt>test.nc</tt> </h3> The following is the start of output from <tt>ncdump test.nc</tt>: <pre> netcdf test { dimensions: Dr = UNLIMITED ; // (2 currently) D1 = 1 ; D2 = 2 ; D3 = 3 ; D4 = 4 ; variables: char c ; byte b ; b:c = "" ; short s ; s:b = '\200' ; s:s = -32768s, 32767s ; long l ; l:l = -2147483648, 2147483647, 2147483647 ; l:f = -3.4028235e+38f, 3.4028235e+38f, -3.4028235e+38f, 3.4028235e+38f ; l:d = -1.797693134862316e+308, 1.797693134862316e+308, -1., 1., 660. ; float f ; double d ; d:c = "�AZ$&" ; char cr(Dr) ; byte br(Dr) ; short sr(Dr) ; long lr(Dr) ; float fr(Dr) ; double dr(Dr) ; char c1(D1) ; byte b1(D1) ; short s1(D1) ; long l1(D1) ; float f1(D1) ; double d1(D1) ; char c2(D2) ; byte b2(D2) ; short s2(D2) ; long l2(D2) ; float f2(D2) ; double d2(D2) ; char c3(D3) ; byte b3(D3) ; short s3(D3) ; long l3(D3) ; float f3(D3) ; double d3(D3) ; char c4(D4) ; byte b4(D4) ; short s4(D4) ; long l4(D4) ; float f4(D4) ; double d4(D4) ; char cr1(Dr, D1) ; byte br2(Dr, D2) ; short sr3(Dr, D3) ; long lr4(Dr, D4) ; float f11(D1, D1) ; double d12(D1, D2) ; char c13(D1, D3) ; byte b14(D1, D4) ; short s21(D2, D1) ; long l22(D2, D2) ; float f23(D2, D3) ; double d24(D2, D4) ; char c31(D3, D1) ; byte b32(D3, D2) ; short s33(D3, D3) ; long l34(D3, D4) ; float f41(D4, D1) ; double d42(D4, D2) ; char c43(D4, D3) ; byte b44(D4, D4) ; short sr11(Dr, D1, D1) ; long lr12(Dr, D1, D2) ; float fr13(Dr, D1, D3) ; double dr14(Dr, D1, D4) ; char cr21(Dr, D2, D1) ; byte br22(Dr, D2, D2) ; short sr23(Dr, D2, D3) ; long lr24(Dr, D2, D4) ; float fr31(Dr, D3, D1) ; double dr32(Dr, D3, D2) ; char cr33(Dr, D3, D3) ; byte br34(Dr, D3, D4) ; short sr41(Dr, D4, D1) ; long lr42(Dr, D4, D2) ; float fr43(Dr, D4, D3) ; double dr44(Dr, D4, D4) ; char c111(D1, D1, D1) ; byte b112(D1, D1, D2) ; short s113(D1, D1, D3) ; long l114(D1, D1, D4) ; float f121(D1, D2, D1) ; double d122(D1, D2, D2) ; char c123(D1, D2, D3) ; byte b124(D1, D2, D4) ; short s131(D1, D3, D1) ; long l132(D1, D3, D2) ; float f133(D1, D3, D3) ; double d134(D1, D3, D4) ; char c141(D1, D4, D1) ; byte b142(D1, D4, D2) ; short s143(D1, D4, D3) ; long l144(D1, D4, D4) ; float f211(D2, D1, D1) ; double d212(D2, D1, D2) ; char c213(D2, D1, D3) ; byte b214(D2, D1, D4) ; short s221(D2, D2, D1) ; long l222(D2, D2, D2) ; float f223(D2, D2, D3) ; double d224(D2, D2, D4) ; char c231(D2, D3, D1) ; byte b232(D2, D3, D2) ; short s233(D2, D3, D3) ; long l234(D2, D3, D4) ; float f241(D2, D4, D1) ; double d242(D2, D4, D2) ; char c243(D2, D4, D3) ; byte b244(D2, D4, D4) ; short s311(D3, D1, D1) ; long l312(D3, D1, D2) ; float f313(D3, D1, D3) ; double d314(D3, D1, D4) ; char c321(D3, D2, D1) ; byte b322(D3, D2, D2) ; short s323(D3, D2, D3) ; long l324(D3, D2, D4) ; float f331(D3, D3, D1) ; double d332(D3, D3, D2) ; char c333(D3, D3, D3) ; byte b334(D3, D3, D4) ; short s341(D3, D4, D1) ; long l342(D3, D4, D2) ; float f343(D3, D4, D3) ; double d344(D3, D4, D4) ; char c411(D4, D1, D1) ; byte b412(D4, D1, D2) ; short s413(D4, D1, D3) ; long l414(D4, D1, D4) ; float f421(D4, D2, D1) ; double d422(D4, D2, D2) ; char c423(D4, D2, D3) ; byte b424(D4, D2, D4) ; short s431(D4, D3, D1) ; long l432(D4, D3, D2) ; float f433(D4, D3, D3) ; double d434(D4, D3, D4) ; char c441(D4, D4, D1) ; byte b442(D4, D4, D2) ; short s443(D4, D4, D3) ; long l444(D4, D4, D4) ; // global attributes: :Gc = "" ; :Gb = '\200', '\177' ; :Gs = -32768s, 32767s, 32767s ; :Gl = -2147483648, 2147483647, 2147483647, 2147483647 ; :Gf = -3.4028235e+38f, 3.4028235e+38f, -3.4028235e+38f, 3.4028235e+38f, 531.f ; :Gd = -1.797693134862316e+308, 1.797693134862316e+308, -1., 1., 660., 650. ; data: c = "\002" ; b = 254 ; s = -5 ; l = -20 ; f = -9 ; d = -10 ; cr = "\000\377" ; br = 128, 127 ; sr = -32768, 32767 ; lr = -2147483648, 2147483647 ; fr = -3.402823e+38 , 3.402823e+38 ; dr = -1.79769313486232e+308, 1.79769313486232e+308 ; c1 = "" ; b1 = 128 ; s1 = -32768 ; </pre> <h3> HANDLING OF FAILURES </h3> The macro <tt>IF</tt> is used in place of an ordinary <tt>if</tt> as in: <pre> IF (err) error("nc_open: %s", nc_strerror(err)); </pre> <tt>IF</tt> acts like an ordinary <tt>if</tt> except for two side effects. Firstly, it prints a failure message. Secondly, it increments the global variable <tt>nfails</tt>, which contains a count of the failures in the current test. The true condition should always correspond to a failure. There can be an <tt>else</tt>. <p> The macro <tt>IF</tt> is defined in file <tt>error.h</tt> as follows: <pre> #define IF(EXPR) if (ifFail(EXPR, __LINE__, __FILE__)) </pre> The function <tt>ifFail</tt> is defined in file <tt>error.c</tt> as follows: <pre> int ifFail(const int expr, const int line, const char *file) { if (expr) { ++nfails; error("\n\tFAILURE at line %d of %s: ", line, file); } return expr; } </pre> <p> The driver calls each test function and prints summary messages using the following macro <tt>NC_TEST</tt> defined in file <tt>nc_test.c</tt>: <pre> #define NC_TEST(func) \ print( "*** Testing " #func " ... ");\ nfails = 0;\ test_ ## func();\ nfailsTotal += nfails;\ if (verbose) \ print("\n"); \ if ( nfails == 0) \ print( "ok\n");\ else\ print( "\n\t### %d FAILURES TESTING %s! ###\n", nfails, #func) </pre> <h3> INDIVIDUAL TESTS </h3> The file <tt>scratch.nc</tt> is created at the start, and deleted at the end, of each test that uses it. <p> An attempt has been made to generate all possible errors (including all possible illegal argument values) and check their error status values. Where some arguments could be <tt>NULL</tt>, this <tt>NULL</tt> alternative is typically chosen for odd variable ID values. <p> In most cases the get/put tests read/write each variable/attribute element exactly once. <p> The <tt>get_vara/put_vara</tt> tests split each dimension at a randomly selected point. So a matrix is divided into four sub-matrices. <p> The <tt>get_vars/put_vars</tt> tests do the same splitting, but also randomly select a stride vector for each sub-array. The whole sub-array is covered by repeating with different start-index-vectors. (The number of different starts is the product of the elements of the stride vector.) <p> The buffer for get/put att, <tt>var1</tt>, <tt>vara</tt> and <tt>vars</tt> is used for only a single get/put. However the buffer for <tt>varm</tt> tests is a bit-image of the external variable and each get/put transfers some elements, filling in the gaps. Otherwise, these <tt>varm</tt> tests are similar to the <tt>vars</tt> tests. <h3> TESTING DONE </h3> Tests were run on slim, binnie and laraine. These tests consisted of: <pre> nc_test -c # in writable directory nc_test # in writable directory nc_test -r # in read-only directory nc_test # in read-only directory </pre> <p> A full test on slim gave the following timings: <pre> real 0m17.67s user 0m7.81s sys 0m1.15s </pre> <h3> CURRENT STATUS AND WORK REMAINING </h3> Currently there are 95 tests implemented. There are 32 tests not yet implemented, corresponding to library functions have not yet been implemented. These correspond to the eight types for each of <tt>nc_get_vars_</tt><i>TYPE</I>, <tt>nc_get_varm_</tt><i>TYPE</I>, <tt>nc_put_vars_</tt><i>TYPE</I>, <tt>nc_put_varm_</tt><i>TYPE</I>. However these will all be generated by four <tt>m4</tt> macros, which will be similar to the existing <tt>nc_get_vars</tt>, <tt>nc_get_varm</tt>, <tt>nc_put_vars</tt>, <tt>nc_put_varm</tt>. <p> The code to test negative strides is currently commented out, because this feature has not yet been implemented in the library. </body> </html>