/* * $Id: Gs3.c,v 2.2 2006/02/09 03:08:57 glynn Exp $ */ /* Gs.c Bill Brown, USACERL January 1993 */ #include #include #include #include #include #include #include /* for geoview & geodisplay in 3dview stuff */ #include "gsget.h" /* for update_attrange - might be able to move this func now */ /* The following macros are only used in the function Gs_update_attrange() */ #define INIT_MINMAX(p, nm, size, min, max, found) \ found = 0; \ p+=(size-1); \ while (size--) \ { \ if (!BM_GET_BYOFFSET(nm, size)) \ { \ min = max = *p; \ found = 1; \ break; \ } \ p--; \ } #define SET_MINMAX(p, nm, size, min, max) \ p+=(size-1); \ while(size--) \ { \ if (!BM_GET_BYOFFSET(nm, size)) \ { \ if (*p < min) \ { \ min = *p; \ } \ else if (*p > max) \ { \ max = *p; \ } \ } \ p--; \ } typedef int FILEDESC; /* #define TRACE_FUNCS */ #define NO_DATA_COL 0xffffff /************************************************************************/ /* This should be a function variable that * may be replaced by a user's function. * Or else use G_set_error_routine. */ void Gs_warning(char *str) { fprintf(stderr, "%s\n", str); return; } /************************************************************************/ /* This should be a function variable that * may be replaced by a user's function. * Or else use G_set_error_routine. */ void Gs_status(char *str) { fprintf(stderr, "%s\n", str); return; } /************************************************************************/ /* calculates distance in METERS between two points in current projection */ double Gs_distance(double *from, double *to) { static int first = 1; if (first) { first = 0; G_begin_distance_calculations(); } return G_distance(from[0], from[1], to[0], to[1]); } /************************************************************************/ /* Calling function must have already allocated space in buff for wind->rows * wind->cols floats. This routine simply loads the map into a 2d array by repetitve calls to get_f_raster_row. */ int Gs_loadmap_as_float(struct Cell_head *wind, char *map_name, float *buff, struct BM *nullmap, int *has_null) { FILEDESC cellfile; char *map_set, *nullflags; char err_buff[100]; int offset, row, col; #ifdef TRACE_FUNCS { Gs_status("Gs_loadmap_as_float"); } #endif map_set = G_find_file2("cell", map_name, ""); *has_null = 0; if (NULL == (nullflags = G_allocate_null_buf())) { sprintf(err_buff, "Not able to allocate null buffer for [%s]", map_name); Gs_warning(err_buff); exit(0); } if ((cellfile = G_open_cell_old(map_name, map_set)) == -1) { sprintf(err_buff, "Not able to open cellfile for [%s]", map_name); Gs_warning(err_buff); exit(0); } Gs_status("Loading Data"); for (row = 0; row < wind->rows; row++) { offset = row * wind->cols; G_get_f_raster_row(cellfile, &(buff[offset]), row); G_get_null_value_row(cellfile, nullflags, row); G_percent(row, wind->rows, 2); for (col = 0; col < wind->cols; col++) { if (nullflags[col] || G_is_f_null_value(buff + offset + col)) { *has_null = 1; BM_set(nullmap, col, row, 1); } /* set nm */ } } #ifdef DEBUG_MSG { fprintf(stderr, "_HAS-NULL_ = %d\n", *has_null); } #endif G_close_cell(cellfile); free(nullflags); return (1); } /************************************************************************/ /* Calling function must have already allocated space in buff for wind->rows * wind->cols integers. This routine simply loads the map into a 2d array by repetitve calls to get_map_row. */ int Gs_loadmap_as_int(struct Cell_head *wind, char *map_name, int *buff, struct BM *nullmap, int *has_null) { FILEDESC cellfile; char *map_set, *nullflags; char err_buff[100]; int offset, row, col; #ifdef TRACE_FUNCS { Gs_status("Gs_loadmap_as_int"); } #endif map_set = G_find_file2("cell", map_name, ""); *has_null = 0; if (NULL == (nullflags = G_allocate_null_buf())) { sprintf(err_buff, "Not able to allocate null buffer for [%s]", map_name); Gs_warning(err_buff); exit(0); } if ((cellfile = G_open_cell_old(map_name, map_set)) == -1) { sprintf(err_buff, "Not able to open cellfile for [%s]", map_name); Gs_warning(err_buff); exit(0); } Gs_status("Loading Data"); for (row = 0; row < wind->rows; row++) { offset = row * wind->cols; G_get_c_raster_row(cellfile, &(buff[offset]), row); G_get_null_value_row(cellfile, nullflags, row); G_percent(row, wind->rows, 2); for (col = 0; col < wind->cols; col++) { if (nullflags[col]) { *has_null = 1; BM_set(nullmap, col, row, 1); } /* set nm */ } } G_close_cell(cellfile); free(nullflags); return (1); } /*********************************************************************/ /* returns -1 if map is integer and G_read_range fails !! */ int Gs_numtype(char *filename, int *negflag) { CELL max = 0, min = 0; struct Range range; char *mapset; int shortbits, charbits, bitplace; static int max_short, max_char; static int first = 1; if (first) { max_short = max_char = 1; shortbits = 8 * sizeof(short); for (bitplace = 1; bitplace < shortbits; ++bitplace) { /*1 bit for sign */ max_short *= 2; } max_short -= 1; /* NO bits for sign, using unsigned char */ charbits = 8 * sizeof(unsigned char); for (bitplace = 0; bitplace < charbits; ++bitplace) { max_char *= 2; } max_char -= 1; first = 0; } mapset = G_find_file2("cell", filename, ""); if (G_raster_map_is_fp(filename, mapset)) { #ifdef DEBUG_MSG { fprintf(stderr, "fp map detected \n"); } #endif return (ATTY_FLOAT); } if (-1 == G_read_range(filename, mapset, &range)) { return (-1); } G_get_range_min_max(&range, &min, &max); *negflag = (min < 0); if (max < max_char && min > 0) { return (ATTY_CHAR); } if (max < max_short && min > -max_short) { return (ATTY_SHORT); } return (ATTY_INT); } /************************************************************************/ /* Calling function must have already allocated space in buff for wind->rows * wind->cols shorts. This routine simply loads the map into a 2d array by repetitve calls to get_map_row. Returns 1 on success, -1 on failure, -2 if read ok, but 1 or more values were too large (small) to fit into a short. (in which case the max (min) short is used) */ int Gs_loadmap_as_short(struct Cell_head *wind, char *map_name, short *buff, struct BM *nullmap, int *has_null) { FILEDESC cellfile; char *map_set, *nullflags; char err_buff[100]; int *ti, *tmp_buf; int offset, row, col, val, max_short, overflow, shortsize, bitplace; short *ts; #ifdef TRACE_FUNCS { Gs_status("Gs_loadmap_as_short"); } #endif overflow = 0; shortsize = 8 * sizeof(short); /* 1 bit for sign */ /* same as 2 << (shortsize-1) */ for (max_short = bitplace = 1; bitplace < shortsize; ++bitplace) { max_short *= 2; } max_short -= 1; map_set = G_find_file2("cell", map_name, ""); *has_null = 0; if (NULL == (nullflags = G_allocate_null_buf())) { sprintf(err_buff, "Not able to allocate null buffer for [%s]", map_name); Gs_warning(err_buff); exit(0); } if ((cellfile = G_open_cell_old(map_name, map_set)) == -1) { sprintf(err_buff, "Not able to open cellfile for [%s]", map_name); Gs_warning(err_buff); exit(0); } if (NULL == (tmp_buf = (int *) G_malloc(wind->cols * sizeof(int)))) { sprintf(err_buff, "out of memory"); Gs_warning(err_buff); exit(0); } Gs_status("Loading Data"); for (row = 0; row < wind->rows; row++) { offset = row * wind->cols; G_get_c_raster_row(cellfile, tmp_buf, row); G_get_null_value_row(cellfile, nullflags, row); G_percent(row, wind->rows, 2); ts = &(buff[offset]); ti = tmp_buf; for (col = 0; col < wind->cols; col++) { if (nullflags[col]) { *has_null = 1; BM_set(nullmap, col, row, 1); } else { val = *ti; if (abs(val) > max_short) { overflow = 1; /* assign floor/ceiling value? */ *ts = (short) (max_short * val / abs(val)); } else { *ts = (short) val; } } ti++; ts++; } } G_close_cell(cellfile); free(tmp_buf); free(nullflags); return (overflow ? -2 : 1); } /************************************************************************/ /* Calling function must have already allocated space in buff for wind->rows * wind->cols unsigned chars. This routine simply loads the map into a 2d array by repetitve calls to get_map_row. Returns 1 on success, -1 on failure, -2 if read ok, but 1 or more values were too large (small) to fit into an unsigned char. (in which case the max (min) char is used) Since signs of chars can be tricky, we only load positive chars between 0-255. */ int Gs_loadmap_as_char(struct Cell_head *wind, char *map_name, unsigned char *buff, struct BM *nullmap, int *has_null) { FILEDESC cellfile; char *map_set, *nullflags; char err_buff[100]; int *ti, *tmp_buf; int offset, row, col, val, max_char, overflow, charsize, bitplace; unsigned char *tc; #ifdef TRACE_FUNCS { Gs_status("Gs_loadmap_as_char"); } #endif overflow = 0; charsize = 8 * sizeof(unsigned char); /* 0 bits for sign! */ max_char = 1; for (bitplace = 0; bitplace < charsize; ++bitplace) { max_char *= 2; } max_char -= 1; map_set = G_find_file2("cell", map_name, ""); *has_null = 0; if (NULL == (nullflags = G_allocate_null_buf())) { sprintf(err_buff, "Not able to allocate null buffer for [%s]", map_name); Gs_warning(err_buff); exit(0); } if ((cellfile = G_open_cell_old(map_name, map_set)) == -1) { sprintf(err_buff, "Not able to open cellfile for [%s]", map_name); Gs_warning(err_buff); exit(0); } if (NULL == (tmp_buf = (int *) G_malloc(wind->cols * sizeof(int)))) { sprintf(err_buff, "out of memory"); Gs_warning(err_buff); exit(0); } Gs_status("Loading Data"); for (row = 0; row < wind->rows; row++) { offset = row * wind->cols; G_get_c_raster_row(cellfile, tmp_buf, row); G_get_null_value_row(cellfile, nullflags, row); tc = (unsigned char *) &(buff[offset]); ti = tmp_buf; G_percent(row, wind->rows, 2); for (col = 0; col < wind->cols; col++) { if (nullflags[col]) { *has_null = 1; BM_set(nullmap, col, row, 1); } else { val = *ti; if (val > max_char) { overflow = 1; *tc = (unsigned char) max_char; } else if (val < 0) { overflow = 1; *tc = 0; } else { *tc = (unsigned char) val; } } ti++; tc++; } } G_close_cell(cellfile); free(tmp_buf); free(nullflags); return (overflow ? -2 : 1); } /************************************************************************/ /* Calling function must have already allocated space in buff for struct BM of wind->rows & wind->cols. This routine simply loads the map into the bitmap by repetitve calls to get_map_row. Any value other than 0 in the map will set the bitmap. (may want to change later to allow specific value to set) Returns 1 on success, -1 on failure. CHANGED TO USE NULLS */ int Gs_loadmap_as_bitmap(struct Cell_head *wind, char *map_name, struct BM *buff) { FILEDESC cellfile; char *map_set, *nullflags; char err_buff[100]; int *tmp_buf; int row, col; #ifdef TRACE_FUNCS { Gs_status("Gs_loadmap_as_bitmap"); } #endif map_set = G_find_file2("cell", map_name, ""); if ((cellfile = G_open_cell_old(map_name, map_set)) == -1) { sprintf(err_buff, "Not able to open cellfile for [%s]", map_name); Gs_warning(err_buff); exit(0); } if (NULL == (tmp_buf = (int *) G_malloc(wind->cols * sizeof(int)))) { sprintf(err_buff, "out of memory"); Gs_warning(err_buff); exit(0); } if (NULL == (nullflags = G_allocate_null_buf())) { sprintf(err_buff, "Not able to allocate null buffer for [%s]", map_name); Gs_warning(err_buff); exit(0); } Gs_status("Loading Data"); for (row = 0; row < wind->rows; row++) { G_get_null_value_row(cellfile, nullflags, row); for (col = 0; col < wind->cols; col++) { if (nullflags[col]) { /* no data */ BM_set(buff, col, row, 1); } else { BM_set(buff, col, row, 0); } } } G_close_cell(cellfile); free(tmp_buf); free(nullflags); return (1); } /************************************************************************/ /* Calling function must have already allocated space in buff for range of data (256 for now) - simply calls get_color for each cat in color range */ int Gs_build_256lookup(char *filename, int *buff) { char *map; struct Colors colrules; CELL min, max, cats[256]; int i; unsigned char r[256], g[256], b[256], set[256]; Gs_status("building color table"); map = G_find_file2("cell", filename, ""); G_read_colors(filename, map, &colrules); G_get_color_range(&min, &max, &colrules); if (min < 0 || max > 255) { fprintf(stderr, "mincol: %d, maxcol: %d\n", min, max); fprintf(stderr, "WARNING: color table range doesn't match data\n"); min = min < 0 ? 0 : min; max = max > 255 ? 255 : max; } G_zero(cats, 256 * sizeof(CELL)); for (i = min; i <= max; i++) { cats[i] = i; } G_lookup_colors(cats, r, g, b, set, 256, &colrules); for (i = 0; i < 256; i++) { if (set[i]) { buff[i] = r[i] & 0xff | ((g[i] & 0xff) << 8) | ((b[i] & 0xff) << 16); } else { buff[i] = NO_DATA_COL; } } return (1); } /************************************************************************/ /* passed an array of 32 bit ints that is converted from cell values to packed colors (0xbbggrr) */ void Gs_pack_colors(char *filename, int *buff, int rows, int cols) { char *map; struct Colors colrules; unsigned char *r, *g, *b, *set; int *cur, i, j; Gs_status("translating colors"); r = (unsigned char *) G_malloc(cols); g = (unsigned char *) G_malloc(cols); b = (unsigned char *) G_malloc(cols); set = (unsigned char *) G_malloc(cols); map = G_find_file2("cell", filename, ""); G_read_colors(filename, map, &colrules); cur = buff; for (i = 0; i < rows; i++) { G_lookup_colors(cur, r, g, b, set, cols, &colrules); G_percent(i, rows, 2); for (j = 0; j < cols; j++) { if (set[j]) { cur[j] = r[j] & 0xff | ((g[j] & 0xff) << 8) | ((b[j] & 0xff) << 16); } else { cur[j] = NO_DATA_COL; } } cur = &(cur[cols]); } G_free_colors(&colrules); free(r); free(g); free(b); free(set); return; } /************************************************************************/ /* passed a array of floats that will be converted from cell values to packed colors (0xbbggrr) and float to int ** floating point data not freed here, use: gsds_free_data_buff(id, ATTY_FLOAT) */ void Gs_pack_colors_float(char *filename, float *fbuf, int *ibuf, int rows, int cols) { char *map; struct Colors colrules; unsigned char *r, *g, *b, *set; int i, j, *icur; FCELL *fcur; Gs_status("translating colors from fp"); r = (unsigned char *) G_malloc(cols); g = (unsigned char *) G_malloc(cols); b = (unsigned char *) G_malloc(cols); set = (unsigned char *) G_malloc(cols); map = G_find_file2("cell", filename, ""); G_read_colors(filename, map, &colrules); fcur = fbuf; icur = ibuf; for (i = 0; i < rows; i++) { G_lookup_f_raster_colors(fcur, r, g, b, set, cols, &colrules); G_percent(i, rows, 2); for (j = 0; j < cols; j++) { if (set[j]) { icur[j] = r[j] & 0xff | ((g[j] & 0xff) << 8) | ((b[j] & 0xff) << 16); } else { icur[j] = NO_DATA_COL; } } icur = &(icur[cols]); fcur = &(fcur[cols]); } G_free_colors(&colrules); free(r); free(g); free(b); free(set); return; } /************************************************************************/ /* Formats label as in d.what.rast -> (catval) catlabel * * */ int Gs_get_cat_label(char *filename, int drow, int dcol, char *catstr) { struct Categories cats; char *mapset; CELL *buf; DCELL *dbuf; RASTER_MAP_TYPE map_type; int fd; if ((mapset = G_find_cell(filename, "")) == NULL) { sprintf(catstr, "error"); exit(0); } if (-1 != G_read_cats(filename, mapset, &cats)) { map_type = G_raster_map_type(filename, mapset); fd = G_open_cell_old(filename, mapset); if (map_type == CELL_TYPE) { buf = G_allocate_c_raster_buf(); if (G_get_c_raster_row(fd, buf, drow) < 0) { sprintf(catstr, "error"); } else if (G_is_c_null_value(&buf[dcol])) { sprintf(catstr, "(NULL) %s", G_get_c_raster_cat(&buf[dcol], &cats)); } else { sprintf(catstr, "(%d) %s", buf[dcol], G_get_c_raster_cat(&buf[dcol], &cats)); } free(buf); } else { /* fp map */ dbuf = G_allocate_d_raster_buf(); if (G_get_d_raster_row(fd, dbuf, drow) < 0) { sprintf(catstr, "error"); } else if (G_is_d_null_value(&dbuf[dcol])) { sprintf(catstr, "(NULL) %s", G_get_d_raster_cat(&dbuf[dcol], &cats)); } else { sprintf(catstr, "(%g) %s", dbuf[dcol], G_get_d_raster_cat(&dbuf[dcol], &cats)); } free(dbuf); } } else { strcpy(catstr, "no category label"); } /* TODO: may want to keep these around for multiple queries */ G_free_cats(&cats); G_close_cell(fd); return (1); } /************************************************************************/ int Gs_save_3dview(char *vname, geoview * gv, geodisplay * gd, struct Cell_head *w, geosurf * defsurf) { char *mapset; struct G_3dview v; float zmax, zmin; GS_get_zrange(&zmin, &zmax, 0); G_get_3dview_defaults(&v, w); mapset = G_mapset(); if (mapset != NULL) { if (defsurf) { if (defsurf->draw_mode & DM_WIRE_POLY) { v.display_type = 3; } else if (defsurf->draw_mode & DM_WIRE || defsurf->draw_mode & DM_COL_WIRE) { v.display_type = 1; } else if (defsurf->draw_mode & DM_POLY) { v.display_type = 2; } v.mesh_freq = defsurf->x_modw; /* mesh resolution */ v.poly_freq = defsurf->x_mod; /* poly resolution */ v.dozero = !(defsurf->nz_topo); v.colorgrid = (defsurf->draw_mode & DM_COL_WIRE) ? 1 : 0; v.shading = (defsurf->draw_mode & DM_GOURAUD) ? 1 : 0; } if (gv->infocus) { GS_v3eq(v.from_to[TO], gv->real_to); v.from_to[TO][Z] -= zmin; GS_v3mult(v.from_to[TO], gv->scale); v.from_to[TO][Z] *= gv->vert_exag; } else { GS_v3eq(v.from_to[TO], gv->from_to[TO]); } gsd_model2real(v.from_to[TO]); GS_v3eq(v.from_to[FROM], gv->from_to[FROM]); gsd_model2real(v.from_to[FROM]); v.exag = gv->vert_exag; v.fov = gv->fov / 10.; v.twist = gv->twist; v.fringe = 0; /* not implemented here */ v.lightson = 1; /* always true, curently */ if (gv->lights[0].position[W] == 1) { /* local */ v.lightpos[X] = gv->lights[0].position[X]; v.lightpos[Y] = gv->lights[0].position[Y]; v.lightpos[Z] = gv->lights[0].position[Z]; gsd_model2real(v.lightpos); v.lightpos[W] = 1.0; /* local */ } else { v.lightpos[X] = gv->lights[0].position[X]; v.lightpos[Y] = gv->lights[0].position[Y]; v.lightpos[Z] = gv->lights[0].position[Z]; v.lightpos[W] = 0.0; /* inf */ } v.lightcol[0] = gv->lights[0].color[0]; v.lightcol[1] = gv->lights[0].color[1]; v.lightcol[2] = gv->lights[0].color[2]; v.ambient = (gv->lights[0].ambient[0] + gv->lights[0].ambient[1] + gv->lights[0].ambient[2]) / 3.; v.shine = gv->lights[0].shine; v.surfonly = 0; /* N/A - now uses constant color */ strcpy((v.pgm_id), "Nvision-ALPHA!"); return (G_put_3dview(vname, mapset, &v, w)); } else { return (-1); } } /************************************************************************/ int Gs_load_3dview(char *vname, geoview * gv, geodisplay * gd, struct Cell_head *w, geosurf * defsurf) { char *mapset; struct G_3dview v; int ret = -1; float pt[3]; mapset = G_find_file2("3d.view", vname, ""); if (mapset != NULL) { ret = G_get_3dview(vname, mapset, &v); } if (ret >= 0) { if (strcmp((v.pgm_id), "Nvision-ALPHA!")) { fprintf(stderr, "WARNING: view not saved by this program,\n"); fprintf(stderr, " there may be some inconsistancies.\n"); } /* set poly and mesh resolutions */ v.mesh_freq = (int) (v.mesh_freq * v.vwin.ns_res / w->ns_res); v.poly_freq = (int) (v.poly_freq * v.vwin.ns_res / w->ns_res); /* Set To and FROM positions */ /* TO */ pt[0] = (v.from_to[TO][X] - w->west) - w->ew_res / 2.; pt[1] = (v.from_to[TO][Y] - w->south) - w->ns_res / 2.; pt[2] = v.from_to[TO][Z]; GS_set_focus(pt); /* FROM */ pt[0] = (float) v.from_to[FROM][X]; pt[1] = (float) v.from_to[FROM][Y]; pt[2] = (float) v.from_to[FROM][Z]; GS_moveto_real(pt); if (defsurf) { int dmode = 0; GS_setall_drawres(v.poly_freq, v.poly_freq, v.mesh_freq, v.mesh_freq); while (v.display_type >= 10) { /* globe stuff not used */ v.display_type -= 10; } /* set drawing modes */ if (v.colorgrid) { dmode |= DM_COL_WIRE; } if (v.shading) { dmode |= DM_GOURAUD; } switch (v.display_type) { case 1: dmode |= DM_WIRE; break; case 2: dmode |= DM_POLY; break; case 3: dmode |= DM_WIRE_POLY; break; } GS_setall_drawmode(dmode); /* should also set nozeros here */ } /* set exaggeration */ if (v.exag) GS_set_global_exag(v.exag); /* Set FOV */ if (v.fov) { GS_set_fov((int) (v.fov > 0 ? v.fov * 10. + 0.5 : v.fov * 10. - 0.5)); } else { /* TODO: do ortho */ } /* Set twist */ if (v.twist) GS_set_twist((int) (v.twist > 0 ? v.twist + 0.5 : v.twist - 0.5)); /* TODO: OK to here - need to unravel/reverse lights stuff*** */ if (v.lightson) { /* Lights are on */ /* Light Position */ gv->lights[0].position[X] = v.lightpos[X]; gv->lights[0].position[Y] = v.lightpos[Y]; gv->lights[0].position[Z] = v.lightpos[Z]; /* Light Color */ gv->lights[0].color[0] = v.lightcol[0]; gv->lights[0].color[1] = v.lightcol[1]; gv->lights[0].color[2] = v.lightcol[2]; /* Light Shininess */ gv->lights[0].shine = v.shine; /* Light Ambient */ gv->lights[0].ambient[0] = gv->lights[0].ambient[1] = gv->lights[0].ambient[2] = v.ambient * 3.; } /* Done with lights */ GS_alldraw_wire(); } /* Done with file */ return (1); } /***********************************************************************/ /* updates no_zero ranges for att (actually no_null now) */ /* static CELL tmpc; */ int Gs_update_attrange(geosurf * gs, int desc) { long size; float min, max; typbuff *tb; struct BM *nm; int found; gs->att[desc].max_nz = gs->att[desc].min_nz = gs->att[desc].range_nz = 0.0; if (CONST_ATT == gs_get_att_src(gs, desc)) { gs->att[desc].max_nz = gs->att[desc].min_nz = gs->att[desc].constant; gs->att[desc].range_nz = 0.0; } else if (CF_COLOR_PACKED & gsds_get_changed(gs->att[desc].hdata)) { gs->att[desc].max_nz = 0xFFFFFF; gs->att[desc].min_nz = 0x010101; gs->att[desc].range_nz = 0xFFFFFF; } else { if (NULL == (tb = gsds_get_typbuff(gs->att[desc].hdata, 0))) { return (-1); } nm = tb->nm; if (tb->ib) { int *p; size = gs->rows * gs->cols; p = tb->ib; INIT_MINMAX(p, nm, size, min, max, found); if (!found) { /* all nulls! */ return (-1); } size = gs->rows * gs->cols; p = tb->ib; SET_MINMAX(p, nm, size, min, max); } else if (tb->sb) { short *p; size = gs->rows * gs->cols; p = tb->sb; INIT_MINMAX(p, nm, size, min, max, found); if (!found) { /* all nulls! */ return (-1); } size = gs->rows * gs->cols; p = tb->sb; SET_MINMAX(p, nm, size, min, max); } else if (tb->cb) { char *p; size = gs->rows * gs->cols; p = (char *) tb->cb; INIT_MINMAX(p, nm, size, min, max, found); if (!found) { /* all nulls! */ return (-1); } size = gs->rows * gs->cols; p = (char *) tb->cb; SET_MINMAX(p, nm, size, min, max); } else if (tb->fb) { float *p; size = gs->rows * gs->cols; p = tb->fb; INIT_MINMAX(p, nm, size, min, max, found); if (!found) { /* all nulls! */ return (-1); } size = gs->rows * gs->cols; p = tb->fb; SET_MINMAX(p, nm, size, min, max); } gs->att[desc].max_nz = max; gs->att[desc].min_nz = min; gs->att[desc].range_nz = gs->att[desc].max_nz - gs->att[desc].min_nz; } if (ATT_TOPO == desc) { gs->zmin = min; gs->zmax = max; gs->zrange = gs->zmax - gs->zmin; gs->zminmasked = gs->zmin; gs->zmax_nz = gs->zmax; gs->zmin_nz = gs->zmin; gs->zrange_nz = gs->zmax_nz - gs->zmin_nz; } return (1); }