#include #include #include #include #include #include /*---------------------------------------------------------------------------*/ static int verifyVolumeVertices(map, v) void *map; double v[2][2][2][3]; { if (!(G3d_isValidLocation(map, v[0][0][0][0], v[0][0][0][1], v[0][0][0][2]) && G3d_isValidLocation(map, v[0][0][1][0], v[0][0][1][1], v[0][0][1][2]) && G3d_isValidLocation(map, v[0][1][0][0], v[0][1][0][1], v[0][1][0][2]) && G3d_isValidLocation(map, v[0][1][1][0], v[0][1][1][1], v[0][1][1][2]) && G3d_isValidLocation(map, v[1][0][0][0], v[1][0][0][1], v[1][0][0][2]) && G3d_isValidLocation(map, v[1][0][1][0], v[1][0][1][1], v[1][0][1][2]) && G3d_isValidLocation(map, v[1][1][0][0], v[1][1][0][1], v[1][1][0][2]) && G3d_isValidLocation(map, v[1][1][1][0], v[1][1][1][1], v[1][1][1][2]))) G3d_fatalError("verifyCubeVertices: volume vertex out of range"); return 0; } /*---------------------------------------------------------------------------*/ static int verifyVolumeEdges(nx, ny, nz) int nx, ny, nz; { if ((nx <= 0) || (ny <= 0) || (nz <= 0)) G3d_fatalError("verifyCubeEdges: Volume edge out of range"); return 0; } /*---------------------------------------------------------------------------*/ void G3d_getVolumeA(void *map, double u[2][2][2][3], int nx, int ny, int nz, void *volumeBuf, int type) { typedef double doubleArray[3]; doubleArray *u000, *u001, *u010, *u011; doubleArray *u100, *u101, *u110, *u111; double v00[3], v01[3], v10[3], v11[3]; double v0[3], v1[3]; double v[3]; double r, rp, s, sp, t, tp; double dx, dy, dz; int x, y, z, nxp, nyp, nzp; double *doubleBuf; float *floatBuf; doubleBuf = (double *)volumeBuf; floatBuf = (float *)volumeBuf; verifyVolumeVertices(map, u); verifyVolumeEdges(nx, ny, nz); nxp = nx * 2 + 1; nyp = ny * 2 + 1; nzp = nz * 2 + 1; u000 = (doubleArray *) u[0][0][0]; u001 = (doubleArray *) u[0][0][1]; u010 = (doubleArray *) u[0][1][0]; u011 = (doubleArray *) u[0][1][1]; u100 = (doubleArray *) u[1][0][0]; u101 = (doubleArray *) u[1][0][1]; u110 = (doubleArray *) u[1][1][0]; u111 = (doubleArray *) u[1][1][1]; for (dz = 1; dz < nzp; dz += 2) { r = 1. - (rp = dz / nz / 2.); v00[0] = r * (*u000)[0] + rp * (*u100)[0]; v00[1] = r * (*u000)[1] + rp * (*u100)[1]; v00[2] = r * (*u000)[2] + rp * (*u100)[2]; v01[0] = r * (*u001)[0] + rp * (*u101)[0]; v01[1] = r * (*u001)[1] + rp * (*u101)[1]; v01[2] = r * (*u001)[2] + rp * (*u101)[2]; v10[0] = r * (*u010)[0] + rp * (*u110)[0]; v10[1] = r * (*u010)[1] + rp * (*u110)[1]; v10[2] = r * (*u010)[2] + rp * (*u110)[2]; v11[0] = r * (*u011)[0] + rp * (*u111)[0]; v11[1] = r * (*u011)[1] + rp * (*u111)[1]; v11[2] = r * (*u011)[2] + rp * (*u111)[2]; for (dy = 1; dy < nyp; dy += 2) { s = 1. - (sp = dy / ny / 2.); v0[0] = s * v00[0] + sp * v10[0]; v0[1] = s * v00[1] + sp * v10[1]; v0[2] = s * v00[2] + sp * v10[2]; v1[0] = s * v01[0] + sp * v11[0]; v1[1] = s * v01[1] + sp * v11[1]; v1[2] = s * v01[2] + sp * v11[2]; for (dx = 1; dx < nxp; dx += 2) { t = 1. - (tp = dx / nx / 2.); v[0] = t * v0[0] + tp * v1[0]; v[1] = t * v0[1] + tp * v1[1]; v[2] = t * v0[2] + tp * v1[2]; G3d_location2coord(map, v[0], v[1], v[2], &x, &y, &z); /* DEBUG printf ("(%d %d %d) (%lf %lf %lf) (%d %d %d) %lf\n", (int) dx / 2, (int) dy / 2, (int) dz / 2, v[0], v[1], v[2], x, y, z, G3d_getDoubleRegion (map, x, y, z)); */ if (type == DCELL_TYPE) *(doubleBuf + ((int)dz / 2) * nx * ny + ((int)dy / 2) * nx + (int)dx / 2) = G3d_getDoubleRegion(map, x, y, z); else *(floatBuf + ((int)dz / 2) * nx * ny + ((int)dy / 2) * nx + (int)dx / 2) = G3d_getFloatRegion(map, x, y, z); } } } } /*---------------------------------------------------------------------------*/ void G3d_getVolume(void *map, double originNorth, double originWest, double originBottom, double vxNorth, double vxWest, double vxBottom, double vyNorth, double vyWest, double vyBottom, double vzNorth, double vzWest, double vzBottom, int nx, int ny, int nz, void *volumeBuf, int type) { double u[2][2][2][3]; u[0][0][0][0] = originNorth; u[0][0][0][1] = originWest; u[0][0][0][2] = originBottom; u[0][0][1][0] = vxNorth; u[0][0][1][1] = vxWest; u[0][0][1][2] = vxBottom; u[1][0][0][0] = vzNorth; u[1][0][0][1] = vzWest; u[1][0][0][2] = vzBottom; u[1][0][1][0] = (u[0][0][1][0] - u[0][0][0][0]) + u[1][0][0][0]; u[1][0][1][1] = (u[0][0][1][1] - u[0][0][0][1]) + u[1][0][0][1]; u[1][0][1][2] = (u[0][0][1][2] - u[0][0][0][2]) + u[1][0][0][2]; u[0][1][0][0] = vyNorth; u[0][1][0][1] = vyWest; u[0][1][0][2] = vyBottom; u[0][1][1][0] = (u[0][0][1][0] - u[0][0][0][0]) + u[0][1][0][0]; u[0][1][1][1] = (u[0][0][1][1] - u[0][0][0][1]) + u[0][1][0][1]; u[0][1][1][2] = (u[0][0][1][2] - u[0][0][0][2]) + u[0][1][0][2]; u[1][1][0][0] = (u[1][0][0][0] - u[0][0][0][0]) + u[0][1][0][0]; u[1][1][0][1] = (u[1][0][0][1] - u[0][0][0][1]) + u[0][1][0][1]; u[1][1][0][2] = (u[1][0][0][2] - u[0][0][0][2]) + u[0][1][0][2]; u[1][1][1][0] = (u[1][0][0][0] - u[0][0][0][0]) + u[0][1][1][0]; u[1][1][1][1] = (u[1][0][0][1] - u[0][0][0][1]) + u[0][1][1][1]; u[1][1][1][2] = (u[1][0][0][2] - u[0][0][0][2]) + u[0][1][1][2]; G3d_getVolumeA(map, u, nx, ny, nz, volumeBuf, type); } /*---------------------------------------------------------------------------*/ void G3d_getAlignedVolume(void *map, double originNorth, double originWest, double originBottom, double lengthNorth, double lengthWest, double lengthBottom, int nx, int ny, int nz, void *volumeBuf, int type) { G3d_getVolume(map, originNorth, originWest, originBottom, originNorth + lengthNorth, originWest, originBottom, originNorth, originWest + lengthWest, originBottom, originNorth, originWest, originBottom + lengthBottom, nx, ny, nz, volumeBuf, type); } /*---------------------------------------------------------------------------*/ void G3d_makeAlignedVolumeFile(void *map, const char *fileName, double originNorth, double originWest, double originBottom, double lengthNorth, double lengthWest, double lengthBottom, int nx, int ny, int nz) { void *volumeBuf; void *mapVolume; int x, y, z, eltLength; G3D_Region region; volumeBuf = G3d_malloc(nx * ny * nz * sizeof(G3d_getFileType())); if (volumeBuf == NULL) G3d_fatalError("G3d_makeAlignedVolumeFile: error in G3d_malloc"); G3d_getAlignedVolume(map, originNorth, originWest, originBottom, lengthNorth, lengthWest, lengthBottom, nx, ny, nz, volumeBuf, G3d_getFileType()); region.north = originNorth; region.south = originNorth + lengthNorth; region.east = originWest; region.west = originWest + lengthWest; region.top = originBottom; region.bottom = originBottom + lengthBottom; region.rows = ny; region.cols = nx; region.depths = nz; mapVolume = G3d_openCellNew(fileName, G3d_getFileType(), G3D_USE_CACHE_DEFAULT, ®ion); if (mapVolume == NULL) G3d_fatalError("G3d_makeAlignedVolumeFile: error in G3d_openCellNew"); eltLength = G3d_length(G3d_getFileType()); for (z = 0; z < nz; z++) { for (y = 0; y < ny; y++) { for (x = 0; x < nx; x++) { /* G3d_putValueRegion? */ if (!G3d_putValue(mapVolume, x, y, z, G_incr_void_ptr(volumeBuf, (z * ny * nx + y * nx + x) * eltLength), G3d_fileTypeMap(mapVolume))) G3d_fatalError ("G3d_makeAlignedVolumeFile: error in G3d_putValue"); } } } if (!G3d_closeCell(mapVolume)) G3d_fatalError("G3d_makeAlignedVolumeFile: error in G3d_closeCell"); G3d_free(volumeBuf); }