mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-06 15:34:44 +08:00
615 lines
17 KiB
HTML
615 lines
17 KiB
HTML
|
<!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 = "<22>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>
|