/****************************************************************************** * $Id$ * * Project: High Performance Image Reprojector * Purpose: Test program for high performance warper API. * Author: Frank Warmerdam * ****************************************************************************** * Copyright (c) 2002, i3 - information integration and imaging * Fort Collin, CO * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. ****************************************************************************/ #include "gdalwarper.h" #include "cpl_string.h" #include "ogr_spatialref.h" #include "ogr_api.h" CPL_CVSID("$Id$"); //static CPLString InsertCenterLong( GDALDatasetH hDS, CPLString osWKT ); static void LoadCutline( const char *pszCutlineDSName, const char *pszCLayer, const char *pszCWHERE, const char *pszCSQL, void **phCutlineRet ); static void TransformCutlineToSource( GDALDatasetH hSrcDS, void *hCutline, char ***ppapszWarpOptions ); static GDALDatasetH GDALWarpCreateOutput( char **papszSrcFiles, const char *pszFilename, const char *pszFormat, char **papszTO, char ***ppapszCreateOptions, GDALDataType eDT ); static double dfMinX=0.0, dfMinY=0.0, dfMaxX=0.0, dfMaxY=0.0; static double dfXRes=0.0, dfYRes=0.0; static int nForcePixels=0, nForceLines=0, bQuiet = FALSE; static int bEnableDstAlpha = FALSE, bEnableSrcAlpha = FALSE; static int bVRT = FALSE; /******************************************************************************/ /*! \page gdalwarp gdalwarp image reprojection and warping utility \section wsynopsis SYNOPSIS \htmlonly Usage: \endhtmlonly \verbatim gdalwarp [--help-general] [--formats] [-s_srs srs_def] [-t_srs srs_def] [-to "NAME=VALUE"] [-order n] [-tps] [-rpc] [-geoloc] [-et err_threshold] [-te xmin ymin xmax ymax] [-tr xres yres] [-ts width height] [-wo "NAME=VALUE"] [-ot Byte/Int16/...] [-wt Byte/Int16] [-srcnodata "value [value...]"] [-dstnodata "value [value...]"] -dstalpha [-r resampling_method] [-wm memory_in_mb] [-multi] [-q] [-cutline datasource] [-cl layer] [-cwhere expression] [-csql statement] [-cblend dist_in_pixels] [-of format] [-co "NAME=VALUE"]* srcfile* dstfile \endverbatim \section wdescription DESCRIPTION

The gdalwarp utility is an image mosaicing, reprojection and warping utility. The program can reproject to any supported projection, and can also apply GCPs stored with the image if the image is "raw" with control information.

-s_srs srs def:
source spatial reference set. The coordinate systems that can be passed are anything supported by the OGRSpatialReference.SetFromUserInput() call, which includes EPSG PCS and GCSes (ie. EPSG:4296), PROJ.4 declarations (as above), or the name of a .prf file containing well known text.
-t_srs srs_def:
target spatial reference set. The coordinate systems that can be passed are anything supported by the OGRSpatialReference.SetFromUserInput() call, which includes EPSG PCS and GCSes (ie. EPSG:4296), PROJ.4 declarations (as above), or the name of a .prf file containing well known text.
-to NAME=VALUE:
set a transformer option suitable to pass to GDALCreateGenImgProjTransformer2().
-order n:
order of polynomial used for warping (1 to 3). The default is to select a polynomial order based on the number of GCPs.
-tps:
Force use of thin plate spline transformer based on available GCPs.
-rpc:
Force use of RPCs.
-geoloc:
Force use of Geolocation Arrays.
-et err_threshold:
error threshold for transformation approximation (in pixel units - defaults to 0.125).
-te xmin ymin xmax ymax:
set georeferenced extents of output file to be created (in target SRS).
-tr xres yres:
set output file resolution (in target georeferenced units)
-ts width height:
set output file size in pixels and lines
-wo "NAME=VALUE":
Set a warp options. The GDALWarpOptions::papszWarpOptions docs show all options. Multiple -wo options may be listed.
-ot type:
For the output bands to be of the indicated data type.
-wt type:
Working pixel data type. The data type of pixels in the source image and destination image buffers.
-r resampling_method:
Resampling method to use. Available methods are:
near
:
nearest neighbour resampling (default, fastest algorithm, worst interpolation quality).
bilinear
:
bilinear resampling.
cubic
:
cubic resampling.
cubicspline
:
cubic spline resampling.
lanczos
:
Lanczos windowed sinc resampling.
-srcnodata value [value...]:
Set nodata masking values for input bands (different values can be supplied for each band). If more than one value is supplied all values should be quoted to keep them together as a single operating system argument. Masked values will not be used in interpolation. Use a value of None to ignore intrinsic nodata settings on the source dataset.
-dstnodata value [value...]:
Set nodata values for output bands (different values can be supplied for each band). If more than one value is supplied all values should be quoted to keep them together as a single operating system argument. New files will be initialized to this value and if possible the nodata value will be recorded in the output file.
-dstalpha:
Create an output alpha band to identify nodata (unset/transparent) pixels.
-wm memory_in_mb:
Set the amount of memory (in megabytes) that the warp API is allowed to use for caching.
-multi:
Use multithreaded warping implementation. Multiple threads will be used to process chunks of image and perform input/output operation simultaneously.
-q:
Be quiet.
-of format:
Select the output format. The default is GeoTIFF (GTiff). Use the short format name.
-co "NAME=VALUE":
passes a creation option to the output format driver. Multiple -co options may be listed. See format specific documentation for legal creation options for each format.
-cutline datasource:
Enable use of a blend cutline from the name OGR support datasource.
-cl layername:
Select the named layer from the cutline datasource.
-cwhere expression:
Restrict desired cutline features based on attribute query.
-csql query:
Select cutline features using an SQL query instead of from a layer with -cl.
-cblend distance:
Set a blend distance to use to blend over cutlines (in pixels).
srcfile:
The source file name(s).
dstfile:
The destination file name.
Mosaicing into an existing output file is supported if the output file already exists. The spatial extent of the existing file will not be modified to accomodate new data, so you may have to remove it in that case. Polygon cutlines may be used to restrict the the area of the destination file that may be updated, including blending. Cutline features must be in the georeferenced units of the destination file.

\section wexample EXAMPLE For instance, an eight bit spot scene stored in GeoTIFF with control points mapping the corners to lat/long could be warped to a UTM projection with a command like this:

\verbatim gdalwarp -t_srs '+proj=utm +zone=11 +datum=WGS84' raw_spot.tif utm11.tif \endverbatim For instance, the second channel of an ASTER image stored in HDF with control points mapping the corners to lat/long could be warped to a UTM projection with a command like this:

\verbatim gdalwarp HDF4_SDS:ASTER_L1B:"pg-PR1B0000-2002031402_100_001":2 pg-PR1B0000-2002031402_100_001_2.tif \endverbatim \if man \section wauthor AUTHORS Frank Warmerdam , Silke Reimer \endif */ /************************************************************************/ /* Usage() */ /************************************************************************/ static void Usage() { printf( "Usage: gdalwarp [--help-general] [--formats]\n" " [-s_srs srs_def] [-t_srs srs_def] [-to \"NAME=VALUE\"]\n" " [-order n] [-tps] [-rpc] [-geoloc] [-et err_threshold]\n" " [-te xmin ymin xmax ymax] [-tr xres yres] [-ts width height]\n" " [-wo \"NAME=VALUE\"] [-ot Byte/Int16/...] [-wt Byte/Int16]\n" " [-srcnodata \"value [value...]\"] [-dstnodata \"value [value...]\"] -dstalpha\n" " [-r resampling_method] [-wm memory_in_mb] [-multi] [-q]\n" " [-cutline datasource] [-cl layer] [-cwhere expression]\n" " [-csql statement] [-cblend dist_in_pixels]\n" " [-of format] [-co \"NAME=VALUE\"]*\n" " srcfile* dstfile\n" "\n" "Available resampling methods:\n" " near (default), bilinear, cubic, cubicspline, lanczos.\n" ); exit( 1 ); } /************************************************************************/ /* SanitizeSRS */ /************************************************************************/ char *SanitizeSRS( const char *pszUserInput ) { OGRSpatialReferenceH hSRS; char *pszResult = NULL; CPLErrorReset(); hSRS = OSRNewSpatialReference( NULL ); if( OSRSetFromUserInput( hSRS, pszUserInput ) == OGRERR_NONE ) OSRExportToWkt( hSRS, &pszResult ); else { CPLError( CE_Failure, CPLE_AppDefined, "Translating source or target SRS failed:\n%s", pszUserInput ); exit( 1 ); } OSRDestroySpatialReference( hSRS ); return pszResult; } /************************************************************************/ /* main() */ /************************************************************************/ int main( int argc, char ** argv ) { GDALDatasetH hDstDS; const char *pszFormat = "GTiff"; char **papszSrcFiles = NULL; char *pszDstFilename = NULL; int bCreateOutput = FALSE, i; void *hTransformArg, *hGenImgProjArg=NULL, *hApproxArg=NULL; char **papszWarpOptions = NULL; double dfErrorThreshold = 0.125; double dfWarpMemoryLimit = 0.0; GDALTransformerFunc pfnTransformer = NULL; char **papszCreateOptions = NULL; GDALDataType eOutputType = GDT_Unknown, eWorkingType = GDT_Unknown; GDALResampleAlg eResampleAlg = GRA_NearestNeighbour; const char *pszSrcNodata = NULL; const char *pszDstNodata = NULL; int bMulti = FALSE; char **papszTO = NULL; char *pszCutlineDSName = NULL; char *pszCLayer = NULL, *pszCWHERE = NULL, *pszCSQL = NULL; void *hCutline = NULL; /* Check that we are running against at least GDAL 1.6 */ /* Note to developers : if we use newer API, please change the requirement */ if (atoi(GDALVersionInfo("VERSION_NUM")) < 1600) { fprintf(stderr, "At least, GDAL >= 1.6.0 is required for this version of %s, " "which was compiled against GDAL %s\n", argv[0], GDAL_RELEASE_NAME); exit(1); } GDALAllRegister(); argc = GDALGeneralCmdLineProcessor( argc, &argv, 0 ); if( argc < 1 ) exit( -argc ); /* -------------------------------------------------------------------- */ /* Parse arguments. */ /* -------------------------------------------------------------------- */ for( i = 1; i < argc; i++ ) { if( EQUAL(argv[i], "--utility_version") ) { printf("%s was compiled against GDAL %s and is running against GDAL %s\n", argv[0], GDAL_RELEASE_NAME, GDALVersionInfo("RELEASE_NAME")); return 0; } else if( EQUAL(argv[i],"-co") && i < argc-1 ) { papszCreateOptions = CSLAddString( papszCreateOptions, argv[++i] ); bCreateOutput = TRUE; } else if( EQUAL(argv[i],"-wo") && i < argc-1 ) { papszWarpOptions = CSLAddString( papszWarpOptions, argv[++i] ); } else if( EQUAL(argv[i],"-multi") ) { bMulti = TRUE; } else if( EQUAL(argv[i],"-q") ) { bQuiet = TRUE; } else if( EQUAL(argv[i],"-dstalpha") ) { bEnableDstAlpha = TRUE; } else if( EQUAL(argv[i],"-srcalpha") ) { bEnableSrcAlpha = TRUE; } else if( EQUAL(argv[i],"-of") && i < argc-1 ) { pszFormat = argv[++i]; bCreateOutput = TRUE; if( EQUAL(pszFormat,"VRT") ) bVRT = TRUE; } else if( EQUAL(argv[i],"-t_srs") && i < argc-1 ) { char *pszSRS = SanitizeSRS(argv[++i]); papszTO = CSLSetNameValue( papszTO, "DST_SRS", pszSRS ); CPLFree( pszSRS ); } else if( EQUAL(argv[i],"-s_srs") && i < argc-1 ) { char *pszSRS = SanitizeSRS(argv[++i]); papszTO = CSLSetNameValue( papszTO, "SRC_SRS", pszSRS ); CPLFree( pszSRS ); } else if( EQUAL(argv[i],"-order") && i < argc-1 ) { papszTO = CSLSetNameValue( papszTO, "MAX_GCP_ORDER", argv[++i] ); } else if( EQUAL(argv[i],"-tps") ) { papszTO = CSLSetNameValue( papszTO, "METHOD", "GCP_TPS" ); } else if( EQUAL(argv[i],"-rpc") ) { papszTO = CSLSetNameValue( papszTO, "METHOD", "RPC" ); } else if( EQUAL(argv[i],"-geoloc") ) { papszTO = CSLSetNameValue( papszTO, "METHOD", "GEOLOC_ARRAY" ); } else if( EQUAL(argv[i],"-to") && i < argc-1 ) { papszTO = CSLAddString( papszTO, argv[++i] ); } else if( EQUAL(argv[i],"-et") && i < argc-1 ) { dfErrorThreshold = atof(argv[++i]); } else if( EQUAL(argv[i],"-wm") && i < argc-1 ) { if( atof(argv[i+1]) < 10000 ) dfWarpMemoryLimit = atof(argv[i+1]) * 1024 * 1024; else dfWarpMemoryLimit = atof(argv[i+1]); i++; } else if( EQUAL(argv[i],"-srcnodata") && i < argc-1 ) { pszSrcNodata = argv[++i]; } else if( EQUAL(argv[i],"-dstnodata") && i < argc-1 ) { pszDstNodata = argv[++i]; } else if( EQUAL(argv[i],"-tr") && i < argc-2 ) { dfXRes = atof(argv[++i]); dfYRes = fabs(atof(argv[++i])); if( dfXRes == 0 || dfYRes == 0 ) { printf( "Wrong value for -tr parameters\n"); Usage(); exit( 2 ); } bCreateOutput = TRUE; } else if( EQUAL(argv[i],"-ot") && i < argc-1 ) { int iType; for( iType = 1; iType < GDT_TypeCount; iType++ ) { if( GDALGetDataTypeName((GDALDataType)iType) != NULL && EQUAL(GDALGetDataTypeName((GDALDataType)iType), argv[i+1]) ) { eOutputType = (GDALDataType) iType; } } if( eOutputType == GDT_Unknown ) { printf( "Unknown output pixel type: %s\n", argv[i+1] ); Usage(); exit( 2 ); } i++; bCreateOutput = TRUE; } else if( EQUAL(argv[i],"-wt") && i < argc-1 ) { int iType; for( iType = 1; iType < GDT_TypeCount; iType++ ) { if( GDALGetDataTypeName((GDALDataType)iType) != NULL && EQUAL(GDALGetDataTypeName((GDALDataType)iType), argv[i+1]) ) { eWorkingType = (GDALDataType) iType; } } if( eWorkingType == GDT_Unknown ) { printf( "Unknown output pixel type: %s\n", argv[i+1] ); Usage(); exit( 2 ); } i++; } else if( EQUAL(argv[i],"-ts") && i < argc-2 ) { nForcePixels = atoi(argv[++i]); nForceLines = atoi(argv[++i]); bCreateOutput = TRUE; } else if( EQUAL(argv[i],"-te") && i < argc-4 ) { dfMinX = atof(argv[++i]); dfMinY = atof(argv[++i]); dfMaxX = atof(argv[++i]); dfMaxY = atof(argv[++i]); bCreateOutput = TRUE; } else if( EQUAL(argv[i],"-rn") ) eResampleAlg = GRA_NearestNeighbour; else if( EQUAL(argv[i],"-rb") ) eResampleAlg = GRA_Bilinear; else if( EQUAL(argv[i],"-rc") ) eResampleAlg = GRA_Cubic; else if( EQUAL(argv[i],"-rcs") ) eResampleAlg = GRA_CubicSpline; else if( EQUAL(argv[i],"-r") && i < argc - 1 ) { if ( EQUAL(argv[++i], "near") ) eResampleAlg = GRA_NearestNeighbour; else if ( EQUAL(argv[i], "bilinear") ) eResampleAlg = GRA_Bilinear; else if ( EQUAL(argv[i], "cubic") ) eResampleAlg = GRA_Cubic; else if ( EQUAL(argv[i], "cubicspline") ) eResampleAlg = GRA_CubicSpline; else if ( EQUAL(argv[i], "lanczos") ) eResampleAlg = GRA_Lanczos; else { printf( "Unknown resampling method: \"%s\".\n", argv[i] ); Usage(); } } else if( EQUAL(argv[i],"-cutline") && i < argc-1 ) { pszCutlineDSName = argv[++i]; } else if( EQUAL(argv[i],"-cwhere") && i < argc-1 ) { pszCWHERE = argv[++i]; } else if( EQUAL(argv[i],"-cl") && i < argc-1 ) { pszCLayer = argv[++i]; } else if( EQUAL(argv[i],"-csql") && i < argc-1 ) { pszCSQL = argv[++i]; } else if( EQUAL(argv[i],"-cblend") && i < argc-1 ) { papszWarpOptions = CSLSetNameValue( papszWarpOptions, "CUTLINE_BLEND_DIST", argv[++i] ); } else if( argv[i][0] == '-' ) Usage(); else papszSrcFiles = CSLAddString( papszSrcFiles, argv[i] ); } /* -------------------------------------------------------------------- */ /* The last filename in the file list is really our destination */ /* file. */ /* -------------------------------------------------------------------- */ if( CSLCount(papszSrcFiles) > 1 ) { pszDstFilename = papszSrcFiles[CSLCount(papszSrcFiles)-1]; papszSrcFiles[CSLCount(papszSrcFiles)-1] = NULL; } if( pszDstFilename == NULL ) Usage(); /* -------------------------------------------------------------------- */ /* Does the output dataset already exist? */ /* -------------------------------------------------------------------- */ CPLPushErrorHandler( CPLQuietErrorHandler ); hDstDS = GDALOpen( pszDstFilename, GA_Update ); CPLPopErrorHandler(); if( hDstDS != NULL && bCreateOutput ) { fprintf( stderr, "Output dataset %s exists,\n" "but some commandline options were provided indicating a new dataset\n" "should be created. Please delete existing dataset and run again.\n", pszDstFilename ); exit( 1 ); } /* -------------------------------------------------------------------- */ /* If not, we need to create it. */ /* -------------------------------------------------------------------- */ int bInitDestSetForFirst = FALSE; if( hDstDS == NULL ) { hDstDS = GDALWarpCreateOutput( papszSrcFiles, pszDstFilename,pszFormat, papszTO, &papszCreateOptions, eOutputType ); bCreateOutput = TRUE; if( CSLFetchNameValue( papszWarpOptions, "INIT_DEST" ) == NULL && pszDstNodata == NULL ) { papszWarpOptions = CSLSetNameValue(papszWarpOptions, "INIT_DEST", "0"); bInitDestSetForFirst = TRUE; } else if( CSLFetchNameValue( papszWarpOptions, "INIT_DEST" ) == NULL ) { papszWarpOptions = CSLSetNameValue(papszWarpOptions, "INIT_DEST", "NO_DATA" ); bInitDestSetForFirst = TRUE; } CSLDestroy( papszCreateOptions ); papszCreateOptions = NULL; } if( hDstDS == NULL ) exit( 1 ); /* -------------------------------------------------------------------- */ /* If we have a cutline datasource read it and attach it in the */ /* warp options. */ /* -------------------------------------------------------------------- */ if( pszCutlineDSName != NULL ) { LoadCutline( pszCutlineDSName, pszCLayer, pszCWHERE, pszCSQL, &hCutline ); } /* -------------------------------------------------------------------- */ /* Loop over all source files, processing each in turn. */ /* -------------------------------------------------------------------- */ int iSrc; for( iSrc = 0; papszSrcFiles[iSrc] != NULL; iSrc++ ) { GDALDatasetH hSrcDS; /* -------------------------------------------------------------------- */ /* Open this file. */ /* -------------------------------------------------------------------- */ hSrcDS = GDALOpen( papszSrcFiles[iSrc], GA_ReadOnly ); if( hSrcDS == NULL ) exit( 2 ); if( !bQuiet ) printf( "Processing input file %s.\n", papszSrcFiles[iSrc] ); /* -------------------------------------------------------------------- */ /* Warns if the file has a color table and something more */ /* complicated than nearest neighbour resampling is asked */ /* -------------------------------------------------------------------- */ if ( eResampleAlg != GRA_NearestNeighbour && GDALGetRasterColorTable(GDALGetRasterBand(hSrcDS, 1)) != NULL) { if( !bQuiet ) fprintf( stderr, "Warning: Input file %s has a color table, which will likely lead to " "bad results when using a resampling method other than " "nearest neighbour. Converting the dataset prior to 24/32 bit " "is advised.\n", papszSrcFiles[iSrc] ); } /* -------------------------------------------------------------------- */ /* Do we have a source alpha band? */ /* -------------------------------------------------------------------- */ if( GDALGetRasterColorInterpretation( GDALGetRasterBand(hSrcDS,GDALGetRasterCount(hSrcDS)) ) == GCI_AlphaBand && !bEnableSrcAlpha ) { bEnableSrcAlpha = TRUE; if( !bQuiet ) printf( "Using band %d of source image as alpha.\n", GDALGetRasterCount(hSrcDS) ); } /* -------------------------------------------------------------------- */ /* Create a transformation object from the source to */ /* destination coordinate system. */ /* -------------------------------------------------------------------- */ hTransformArg = hGenImgProjArg = GDALCreateGenImgProjTransformer2( hSrcDS, hDstDS, papszTO ); if( hTransformArg == NULL ) exit( 1 ); pfnTransformer = GDALGenImgProjTransform; /* -------------------------------------------------------------------- */ /* Warp the transformer with a linear approximator unless the */ /* acceptable error is zero. */ /* -------------------------------------------------------------------- */ if( dfErrorThreshold != 0.0 ) { hTransformArg = hApproxArg = GDALCreateApproxTransformer( GDALGenImgProjTransform, hGenImgProjArg, dfErrorThreshold); pfnTransformer = GDALApproxTransform; } /* -------------------------------------------------------------------- */ /* Clear temporary INIT_DEST settings after the first image. */ /* -------------------------------------------------------------------- */ if( bInitDestSetForFirst && iSrc == 1 ) papszWarpOptions = CSLSetNameValue( papszWarpOptions, "INIT_DEST", NULL ); /* -------------------------------------------------------------------- */ /* Setup warp options. */ /* -------------------------------------------------------------------- */ GDALWarpOptions *psWO = GDALCreateWarpOptions(); psWO->papszWarpOptions = CSLDuplicate(papszWarpOptions); psWO->eWorkingDataType = eWorkingType; psWO->eResampleAlg = eResampleAlg; psWO->hSrcDS = hSrcDS; psWO->hDstDS = hDstDS; psWO->pfnTransformer = pfnTransformer; psWO->pTransformerArg = hTransformArg; if( !bQuiet ) psWO->pfnProgress = GDALTermProgress; if( dfWarpMemoryLimit != 0.0 ) psWO->dfWarpMemoryLimit = dfWarpMemoryLimit; /* -------------------------------------------------------------------- */ /* Setup band mapping. */ /* -------------------------------------------------------------------- */ if( bEnableSrcAlpha ) psWO->nBandCount = GDALGetRasterCount(hSrcDS) - 1; else psWO->nBandCount = GDALGetRasterCount(hSrcDS); psWO->panSrcBands = (int *) CPLMalloc(psWO->nBandCount*sizeof(int)); psWO->panDstBands = (int *) CPLMalloc(psWO->nBandCount*sizeof(int)); for( i = 0; i < psWO->nBandCount; i++ ) { psWO->panSrcBands[i] = i+1; psWO->panDstBands[i] = i+1; } /* -------------------------------------------------------------------- */ /* Setup alpha bands used if any. */ /* -------------------------------------------------------------------- */ if( bEnableSrcAlpha ) psWO->nSrcAlphaBand = GDALGetRasterCount(hSrcDS); if( !bEnableDstAlpha && GDALGetRasterCount(hDstDS) == psWO->nBandCount+1 && GDALGetRasterColorInterpretation( GDALGetRasterBand(hDstDS,GDALGetRasterCount(hDstDS))) == GCI_AlphaBand ) { if( !bQuiet ) printf( "Using band %d of destination image as alpha.\n", GDALGetRasterCount(hDstDS) ); bEnableDstAlpha = TRUE; } if( bEnableDstAlpha ) psWO->nDstAlphaBand = GDALGetRasterCount(hDstDS); /* -------------------------------------------------------------------- */ /* Setup NODATA options. */ /* -------------------------------------------------------------------- */ if( pszSrcNodata != NULL && !EQUALN(pszSrcNodata,"n",1) ) { char **papszTokens = CSLTokenizeString( pszSrcNodata ); int nTokenCount = CSLCount(papszTokens); psWO->padfSrcNoDataReal = (double *) CPLMalloc(psWO->nBandCount*sizeof(double)); psWO->padfSrcNoDataImag = (double *) CPLMalloc(psWO->nBandCount*sizeof(double)); for( i = 0; i < psWO->nBandCount; i++ ) { if( i < nTokenCount ) { CPLStringToComplex( papszTokens[i], psWO->padfSrcNoDataReal + i, psWO->padfSrcNoDataImag + i ); } else { psWO->padfSrcNoDataReal[i] = psWO->padfSrcNoDataReal[i-1]; psWO->padfSrcNoDataImag[i] = psWO->padfSrcNoDataImag[i-1]; } } CSLDestroy( papszTokens ); psWO->papszWarpOptions = CSLSetNameValue(psWO->papszWarpOptions, "UNIFIED_SRC_NODATA", "YES" ); } /* -------------------------------------------------------------------- */ /* If -srcnodata was not specified, but the data has nodata */ /* values, use them. */ /* -------------------------------------------------------------------- */ if( pszSrcNodata == NULL ) { int bHaveNodata = FALSE; double dfReal = 0.0; for( i = 0; !bHaveNodata && i < psWO->nBandCount; i++ ) { GDALRasterBandH hBand = GDALGetRasterBand( hSrcDS, i+1 ); dfReal = GDALGetRasterNoDataValue( hBand, &bHaveNodata ); } if( bHaveNodata ) { if( !bQuiet ) printf( "Using internal nodata values (eg. %g) for image %s.\n", dfReal, papszSrcFiles[iSrc] ); psWO->padfSrcNoDataReal = (double *) CPLMalloc(psWO->nBandCount*sizeof(double)); psWO->padfSrcNoDataImag = (double *) CPLMalloc(psWO->nBandCount*sizeof(double)); for( i = 0; i < psWO->nBandCount; i++ ) { GDALRasterBandH hBand = GDALGetRasterBand( hSrcDS, i+1 ); dfReal = GDALGetRasterNoDataValue( hBand, &bHaveNodata ); if( bHaveNodata ) { psWO->padfSrcNoDataReal[i] = dfReal; psWO->padfSrcNoDataImag[i] = 0.0; } else { psWO->padfSrcNoDataReal[i] = -123456.789; psWO->padfSrcNoDataImag[i] = 0.0; } } } } /* -------------------------------------------------------------------- */ /* If the output dataset was created, and we have a destination */ /* nodata value, go through marking the bands with the information.*/ /* -------------------------------------------------------------------- */ if( pszDstNodata != NULL ) { char **papszTokens = CSLTokenizeString( pszDstNodata ); int nTokenCount = CSLCount(papszTokens); psWO->padfDstNoDataReal = (double *) CPLMalloc(psWO->nBandCount*sizeof(double)); psWO->padfDstNoDataImag = (double *) CPLMalloc(psWO->nBandCount*sizeof(double)); for( i = 0; i < psWO->nBandCount; i++ ) { if( i < nTokenCount ) { CPLStringToComplex( papszTokens[i], psWO->padfDstNoDataReal + i, psWO->padfDstNoDataImag + i ); } else { psWO->padfDstNoDataReal[i] = psWO->padfDstNoDataReal[i-1]; psWO->padfDstNoDataImag[i] = psWO->padfDstNoDataImag[i-1]; } if( bCreateOutput ) { GDALSetRasterNoDataValue( GDALGetRasterBand( hDstDS, psWO->panDstBands[i] ), psWO->padfDstNoDataReal[i] ); } } CSLDestroy( papszTokens ); } /* -------------------------------------------------------------------- */ /* If we have a cutline, transform it into the source */ /* pixel/line coordinate system and insert into warp options. */ /* -------------------------------------------------------------------- */ if( hCutline != NULL ) { TransformCutlineToSource( hSrcDS, hCutline, &(psWO->papszWarpOptions) ); } /* -------------------------------------------------------------------- */ /* If we are producing VRT output, then just initialize it with */ /* the warp options and write out now rather than proceeding */ /* with the operations. */ /* -------------------------------------------------------------------- */ if( bVRT ) { if( GDALInitializeWarpedVRT( hDstDS, psWO ) != CE_None ) exit( 1 ); GDALClose( hDstDS ); GDALClose( hSrcDS ); /* The warped VRT will clean itself the transformer used */ /* So we have only to destroy the hGenImgProjArg if we */ /* have wrapped it inside the hApproxArg */ if (pfnTransformer == GDALApproxTransform) { if( hGenImgProjArg != NULL ) GDALDestroyGenImgProjTransformer( hGenImgProjArg ); } GDALDestroyWarpOptions( psWO ); CPLFree( pszDstFilename ); CSLDestroy( argv ); CSLDestroy( papszSrcFiles ); CSLDestroy( papszWarpOptions ); CSLDestroy( papszTO ); GDALDumpOpenDatasets( stderr ); GDALDestroyDriverManager(); return 0; } /* -------------------------------------------------------------------- */ /* Initialize and execute the warp. */ /* -------------------------------------------------------------------- */ GDALWarpOperation oWO; if( oWO.Initialize( psWO ) == CE_None ) { if( bMulti ) oWO.ChunkAndWarpMulti( 0, 0, GDALGetRasterXSize( hDstDS ), GDALGetRasterYSize( hDstDS ) ); else oWO.ChunkAndWarpImage( 0, 0, GDALGetRasterXSize( hDstDS ), GDALGetRasterYSize( hDstDS ) ); } /* -------------------------------------------------------------------- */ /* Cleanup */ /* -------------------------------------------------------------------- */ if( hApproxArg != NULL ) GDALDestroyApproxTransformer( hApproxArg ); if( hGenImgProjArg != NULL ) GDALDestroyGenImgProjTransformer( hGenImgProjArg ); GDALDestroyWarpOptions( psWO ); GDALClose( hSrcDS ); } /* -------------------------------------------------------------------- */ /* Final Cleanup. */ /* -------------------------------------------------------------------- */ GDALClose( hDstDS ); CPLFree( pszDstFilename ); CSLDestroy( argv ); CSLDestroy( papszSrcFiles ); CSLDestroy( papszWarpOptions ); CSLDestroy( papszTO ); GDALDumpOpenDatasets( stderr ); GDALDestroyDriverManager(); #ifdef OGR_ENABLED if( hCutline != NULL ) OGR_G_DestroyGeometry( (OGRGeometryH) hCutline ); OGRCleanupAll(); #endif return 0; } /************************************************************************/ /* GDALWarpCreateOutput() */ /* */ /* Create the output file based on various commandline options, */ /* and the input file. */ /************************************************************************/ static GDALDatasetH GDALWarpCreateOutput( char **papszSrcFiles, const char *pszFilename, const char *pszFormat, char **papszTO, char ***ppapszCreateOptions, GDALDataType eDT ) { GDALDriverH hDriver; GDALDatasetH hDstDS; void *hTransformArg; GDALColorTableH hCT = NULL; double dfWrkMinX=0, dfWrkMaxX=0, dfWrkMinY=0, dfWrkMaxY=0; double dfWrkResX=0, dfWrkResY=0; int nDstBandCount = 0; /* -------------------------------------------------------------------- */ /* Find the output driver. */ /* -------------------------------------------------------------------- */ hDriver = GDALGetDriverByName( pszFormat ); if( hDriver == NULL || GDALGetMetadataItem( hDriver, GDAL_DCAP_CREATE, NULL ) == NULL ) { int iDr; printf( "Output driver `%s' not recognised or does not support\n", pszFormat ); printf( "direct output file creation. The following format drivers are configured\n" "and support direct output:\n" ); for( iDr = 0; iDr < GDALGetDriverCount(); iDr++ ) { GDALDriverH hDriver = GDALGetDriver(iDr); if( GDALGetMetadataItem( hDriver, GDAL_DCAP_CREATE, NULL) != NULL ) { printf( " %s: %s\n", GDALGetDriverShortName( hDriver ), GDALGetDriverLongName( hDriver ) ); } } printf( "\n" ); exit( 1 ); } /* -------------------------------------------------------------------- */ /* For virtual output files, we have to set a special subclass */ /* of dataset to create. */ /* -------------------------------------------------------------------- */ if( bVRT ) *ppapszCreateOptions = CSLSetNameValue( *ppapszCreateOptions, "SUBCLASS", "VRTWarpedDataset" ); /* -------------------------------------------------------------------- */ /* Loop over all input files to collect extents. */ /* -------------------------------------------------------------------- */ int iSrc; char *pszThisTargetSRS = (char*)CSLFetchNameValue( papszTO, "DST_SRS" ); if( pszThisTargetSRS != NULL ) pszThisTargetSRS = CPLStrdup( pszThisTargetSRS ); for( iSrc = 0; papszSrcFiles[iSrc] != NULL; iSrc++ ) { GDALDatasetH hSrcDS; const char *pszThisSourceSRS = CSLFetchNameValue(papszTO,"SRC_SRS"); hSrcDS = GDALOpen( papszSrcFiles[iSrc], GA_ReadOnly ); if( hSrcDS == NULL ) exit( 1 ); /* -------------------------------------------------------------------- */ /* Check that there's at least one raster band */ /* -------------------------------------------------------------------- */ if ( GDALGetRasterCount(hSrcDS) == 0 ) { fprintf(stderr, "Input file %s has no raster bands.\n", papszSrcFiles[iSrc] ); exit( 1 ); } if( eDT == GDT_Unknown ) eDT = GDALGetRasterDataType(GDALGetRasterBand(hSrcDS,1)); /* -------------------------------------------------------------------- */ /* If we are processing the first file, and it has a color */ /* table, then we will copy it to the destination file. */ /* -------------------------------------------------------------------- */ if( iSrc == 0 ) { nDstBandCount = GDALGetRasterCount(hSrcDS); hCT = GDALGetRasterColorTable( GDALGetRasterBand(hSrcDS,1) ); if( hCT != NULL ) { hCT = GDALCloneColorTable( hCT ); if( !bQuiet ) printf( "Copying color table from %s to new file.\n", papszSrcFiles[iSrc] ); } } /* -------------------------------------------------------------------- */ /* Get the sourcesrs from the dataset, if not set already. */ /* -------------------------------------------------------------------- */ if( pszThisSourceSRS == NULL ) { const char *pszMethod = CSLFetchNameValue( papszTO, "METHOD" ); if( GDALGetProjectionRef( hSrcDS ) != NULL && strlen(GDALGetProjectionRef( hSrcDS )) > 0 && (pszMethod == NULL || EQUAL(pszMethod,"GEOTRANSFORM")) ) pszThisSourceSRS = GDALGetProjectionRef( hSrcDS ); else if( GDALGetGCPProjection( hSrcDS ) != NULL && strlen(GDALGetGCPProjection(hSrcDS)) > 0 && GDALGetGCPCount( hSrcDS ) > 1 && (pszMethod == NULL || EQUALN(pszMethod,"GCP_",4)) ) pszThisSourceSRS = GDALGetGCPProjection( hSrcDS ); else if( pszMethod != NULL && EQUAL(pszMethod,"RPC") ) pszThisSourceSRS = SRS_WKT_WGS84; else pszThisSourceSRS = ""; } if( pszThisTargetSRS == NULL ) pszThisTargetSRS = CPLStrdup( pszThisSourceSRS ); /* -------------------------------------------------------------------- */ /* Create a transformation object from the source to */ /* destination coordinate system. */ /* -------------------------------------------------------------------- */ hTransformArg = GDALCreateGenImgProjTransformer2( hSrcDS, NULL, papszTO ); if( hTransformArg == NULL ) { CPLFree( pszThisTargetSRS ); GDALClose( hSrcDS ); return NULL; } /* -------------------------------------------------------------------- */ /* Get approximate output definition. */ /* -------------------------------------------------------------------- */ double adfThisGeoTransform[6]; double adfExtent[4]; int nThisPixels, nThisLines; if( GDALSuggestedWarpOutput2( hSrcDS, GDALGenImgProjTransform, hTransformArg, adfThisGeoTransform, &nThisPixels, &nThisLines, adfExtent, 0 ) != CE_None ) { CPLFree( pszThisTargetSRS ); GDALClose( hSrcDS ); return NULL; } /* -------------------------------------------------------------------- */ /* Expand the working bounds to include this region, ensure the */ /* working resolution is no more than this resolution. */ /* -------------------------------------------------------------------- */ if( dfWrkMaxX == 0.0 && dfWrkMinX == 0.0 ) { dfWrkMinX = adfExtent[0]; dfWrkMaxX = adfExtent[2]; dfWrkMaxY = adfExtent[3]; dfWrkMinY = adfExtent[1]; dfWrkResX = adfThisGeoTransform[1]; dfWrkResY = ABS(adfThisGeoTransform[5]); } else { dfWrkMinX = MIN(dfWrkMinX,adfExtent[0]); dfWrkMaxX = MAX(dfWrkMaxX,adfExtent[2]); dfWrkMaxY = MAX(dfWrkMaxY,adfExtent[3]); dfWrkMinY = MIN(dfWrkMinY,adfExtent[1]); dfWrkResX = MIN(dfWrkResX,adfThisGeoTransform[1]); dfWrkResY = MIN(dfWrkResY,ABS(adfThisGeoTransform[5])); } GDALDestroyGenImgProjTransformer( hTransformArg ); GDALClose( hSrcDS ); } /* -------------------------------------------------------------------- */ /* Did we have any usable sources? */ /* -------------------------------------------------------------------- */ if( nDstBandCount == 0 ) { CPLError( CE_Failure, CPLE_AppDefined, "No usable source images." ); CPLFree( pszThisTargetSRS ); return NULL; } /* -------------------------------------------------------------------- */ /* Turn the suggested region into a geotransform and suggested */ /* number of pixels and lines. */ /* -------------------------------------------------------------------- */ double adfDstGeoTransform[6]; int nPixels, nLines; adfDstGeoTransform[0] = dfWrkMinX; adfDstGeoTransform[1] = dfWrkResX; adfDstGeoTransform[2] = 0.0; adfDstGeoTransform[3] = dfWrkMaxY; adfDstGeoTransform[4] = 0.0; adfDstGeoTransform[5] = -1 * dfWrkResY; nPixels = (int) ((dfWrkMaxX - dfWrkMinX) / dfWrkResX + 0.5); nLines = (int) ((dfWrkMaxY - dfWrkMinY) / dfWrkResY + 0.5); /* -------------------------------------------------------------------- */ /* Did the user override some parameters? */ /* -------------------------------------------------------------------- */ if( dfXRes != 0.0 && dfYRes != 0.0 ) { CPLAssert( nForcePixels == 0 && nForceLines == 0 ); if( dfMinX == 0.0 && dfMinY == 0.0 && dfMaxX == 0.0 && dfMaxY == 0.0 ) { dfMinX = adfDstGeoTransform[0]; dfMaxX = adfDstGeoTransform[0] + adfDstGeoTransform[1] * nPixels; dfMaxY = adfDstGeoTransform[3]; dfMinY = adfDstGeoTransform[3] + adfDstGeoTransform[5] * nLines; } nPixels = (int) ((dfMaxX - dfMinX + (dfXRes/2.0)) / dfXRes); nLines = (int) ((dfMaxY - dfMinY + (dfYRes/2.0)) / dfYRes); adfDstGeoTransform[0] = dfMinX; adfDstGeoTransform[3] = dfMaxY; adfDstGeoTransform[1] = dfXRes; adfDstGeoTransform[5] = -dfYRes; } else if( nForcePixels != 0 && nForceLines != 0 ) { if( dfMinX == 0.0 && dfMinY == 0.0 && dfMaxX == 0.0 && dfMaxY == 0.0 ) { dfMinX = dfWrkMinX; dfMaxX = dfWrkMaxX; dfMaxY = dfWrkMaxY; dfMinY = dfWrkMinY; } dfXRes = (dfMaxX - dfMinX) / nForcePixels; dfYRes = (dfMaxY - dfMinY) / nForceLines; adfDstGeoTransform[0] = dfMinX; adfDstGeoTransform[3] = dfMaxY; adfDstGeoTransform[1] = dfXRes; adfDstGeoTransform[5] = -dfYRes; nPixels = nForcePixels; nLines = nForceLines; } else if( nForcePixels != 0 ) { if( dfMinX == 0.0 && dfMinY == 0.0 && dfMaxX == 0.0 && dfMaxY == 0.0 ) { dfMinX = dfWrkMinX; dfMaxX = dfWrkMaxX; dfMaxY = dfWrkMaxY; dfMinY = dfWrkMinY; } dfXRes = (dfMaxX - dfMinX) / nForcePixels; dfYRes = dfXRes; adfDstGeoTransform[0] = dfMinX; adfDstGeoTransform[3] = dfMaxY; adfDstGeoTransform[1] = dfXRes; adfDstGeoTransform[5] = -dfYRes; nPixels = nForcePixels; nLines = (int) ((dfMaxY - dfMinY + (dfYRes/2.0)) / dfYRes); } else if( nForceLines != 0 ) { if( dfMinX == 0.0 && dfMinY == 0.0 && dfMaxX == 0.0 && dfMaxY == 0.0 ) { dfMinX = dfWrkMinX; dfMaxX = dfWrkMaxX; dfMaxY = dfWrkMaxY; dfMinY = dfWrkMinY; } dfYRes = (dfMaxY - dfMinY) / nForceLines; dfXRes = dfYRes; adfDstGeoTransform[0] = dfMinX; adfDstGeoTransform[3] = dfMaxY; adfDstGeoTransform[1] = dfXRes; adfDstGeoTransform[5] = -dfYRes; nPixels = (int) ((dfMaxX - dfMinX + (dfXRes/2.0)) / dfXRes); nLines = nForceLines; } else if( dfMinX != 0.0 || dfMinY != 0.0 || dfMaxX != 0.0 || dfMaxY != 0.0 ) { dfXRes = adfDstGeoTransform[1]; dfYRes = fabs(adfDstGeoTransform[5]); nPixels = (int) ((dfMaxX - dfMinX + (dfXRes/2.0)) / dfXRes); nLines = (int) ((dfMaxY - dfMinY + (dfYRes/2.0)) / dfYRes); adfDstGeoTransform[0] = dfMinX; adfDstGeoTransform[3] = dfMaxY; } /* -------------------------------------------------------------------- */ /* Do we want to generate an alpha band in the output file? */ /* -------------------------------------------------------------------- */ if( bEnableSrcAlpha ) nDstBandCount--; if( bEnableDstAlpha ) nDstBandCount++; /* -------------------------------------------------------------------- */ /* Create the output file. */ /* -------------------------------------------------------------------- */ if( !bQuiet ) printf( "Creating output file that is %dP x %dL.\n", nPixels, nLines ); hDstDS = GDALCreate( hDriver, pszFilename, nPixels, nLines, nDstBandCount, eDT, *ppapszCreateOptions ); if( hDstDS == NULL ) { CPLFree( pszThisTargetSRS ); return NULL; } /* -------------------------------------------------------------------- */ /* Write out the projection definition. */ /* -------------------------------------------------------------------- */ GDALSetProjection( hDstDS, pszThisTargetSRS ); GDALSetGeoTransform( hDstDS, adfDstGeoTransform ); /* -------------------------------------------------------------------- */ /* Try to set color interpretation of output file alpha band. */ /* TODO: We should likely try to copy the other bands too. */ /* -------------------------------------------------------------------- */ if( bEnableDstAlpha ) { GDALSetRasterColorInterpretation( GDALGetRasterBand( hDstDS, nDstBandCount ), GCI_AlphaBand ); } /* -------------------------------------------------------------------- */ /* Copy the color table, if required. */ /* -------------------------------------------------------------------- */ if( hCT != NULL ) { GDALSetRasterColorTable( GDALGetRasterBand(hDstDS,1), hCT ); GDALDestroyColorTable( hCT ); } CPLFree( pszThisTargetSRS ); return hDstDS; } /************************************************************************/ /* GeoTransform_Transformer() */ /* */ /* Convert points from georef coordinates to pixel/line based */ /* on a geotransform. */ /************************************************************************/ class GeoTransform_Transformer : public OGRCoordinateTransformation { public: double adfGeoTransform[6]; virtual OGRSpatialReference *GetSourceCS() { return NULL; } virtual OGRSpatialReference *GetTargetCS() { return NULL; } virtual int Transform( int nCount, double *x, double *y, double *z = NULL ) { return TransformEx( nCount, x, y, z, NULL ); } virtual int TransformEx( int nCount, double *x, double *y, double *z = NULL, int *pabSuccess = NULL ) { int i; for( i = 0; i < nCount; i++ ) { double x_out, y_out; x_out = adfGeoTransform[0] + x[i] * adfGeoTransform[1] + y[i] * adfGeoTransform[2]; y_out = adfGeoTransform[3] + x[i] * adfGeoTransform[4] + y[i] * adfGeoTransform[5]; x[i] = x_out; y[i] = y_out; if( pabSuccess ) pabSuccess[i] = TRUE; } return TRUE; } }; /************************************************************************/ /* LoadCutline() */ /* */ /* Load blend cutline from OGR datasource and attach in warp */ /* options, after potentially transforming to destination */ /* pixel/line coordinates. */ /************************************************************************/ static void LoadCutline( const char *pszCutlineDSName, const char *pszCLayer, const char *pszCWHERE, const char *pszCSQL, void **phCutlineRet ) { #ifndef OGR_ENABLED CPLError( CE_Failure, CPLE_AppDefined, "Request to load a cutline failed, this build does not support OGR features.\n" ); exit( 1 ); #else // def OGR_ENABLED OGRRegisterAll(); /* -------------------------------------------------------------------- */ /* Open source vector dataset. */ /* -------------------------------------------------------------------- */ OGRDataSourceH hSrcDS; hSrcDS = OGROpen( pszCutlineDSName, FALSE, NULL ); if( hSrcDS == NULL ) exit( 1 ); /* -------------------------------------------------------------------- */ /* Get the source layer */ /* -------------------------------------------------------------------- */ OGRLayerH hLayer = NULL; if( pszCSQL != NULL ) hLayer = OGR_DS_ExecuteSQL( hSrcDS, pszCSQL, NULL, NULL ); else if( pszCLayer != NULL ) hLayer = OGR_DS_GetLayerByName( hSrcDS, pszCLayer ); else hLayer = OGR_DS_GetLayer( hSrcDS, 0 ); if( hLayer == NULL ) { fprintf( stderr, "Failed to identify source layer from datasource.\n" ); exit( 1 ); } /* -------------------------------------------------------------------- */ /* Apply WHERE clause if there is one. */ /* -------------------------------------------------------------------- */ if( pszCWHERE != NULL ) OGR_L_SetAttributeFilter( hLayer, pszCWHERE ); /* -------------------------------------------------------------------- */ /* Collect the geometries from this layer, and build list of */ /* burn values. */ /* -------------------------------------------------------------------- */ OGRFeatureH hFeat; OGRGeometryH hMultiPolygon = OGR_G_CreateGeometry( wkbMultiPolygon ); OGR_L_ResetReading( hLayer ); while( (hFeat = OGR_L_GetNextFeature( hLayer )) != NULL ) { OGRGeometryH hGeom = OGR_F_GetGeometryRef(hFeat); if( hGeom == NULL ) { fprintf( stderr, "ERROR: Cutline feature without a geometry.\n" ); exit( 1 ); } OGRwkbGeometryType eType = wkbFlatten(OGR_G_GetGeometryType( hGeom )); if( eType == wkbPolygon ) OGR_G_AddGeometry( hMultiPolygon, hGeom ); else if( eType == wkbMultiPolygon ) { int iGeom; for( iGeom = 0; iGeom < OGR_G_GetGeometryCount( hGeom ); iGeom++ ) { OGR_G_AddGeometry( hMultiPolygon, OGR_G_GetGeometryRef(hGeom,iGeom) ); } } else { fprintf( stderr, "ERROR: Cutline not of polygon type.\n" ); exit( 1 ); } OGR_F_Destroy( hFeat ); } if( OGR_G_GetGeometryCount( hMultiPolygon ) == 0 ) { fprintf( stderr, "ERROR: Did not get any cutline features.\n" ); exit( 1 ); } /* -------------------------------------------------------------------- */ /* Ensure the coordinate system gets set on the geometry. */ /* -------------------------------------------------------------------- */ OGR_G_AssignSpatialReference( hMultiPolygon, OGR_L_GetSpatialRef(hLayer) ); *phCutlineRet = (void *) hMultiPolygon; /* -------------------------------------------------------------------- */ /* Cleanup */ /* -------------------------------------------------------------------- */ if( pszCSQL != NULL ) OGR_DS_ReleaseResultSet( hSrcDS, hLayer ); OGR_DS_Destroy( hSrcDS ); #endif } /************************************************************************/ /* TransformCutlineToSource() */ /************************************************************************/ static void TransformCutlineToSource( GDALDatasetH hSrcDS, void *hCutline, char ***ppapszWarpOptions ) { #ifdef OGR_ENABLED OGRGeometryH hMultiPolygon = OGR_G_Clone( (OGRGeometryH) hCutline ); /* -------------------------------------------------------------------- */ /* Checkout that SRS are the same. */ /* -------------------------------------------------------------------- */ OGRSpatialReferenceH hRasterSRS = NULL; if( GDALGetProjectionRef( hSrcDS ) != NULL ) { char *pszProjection; pszProjection = (char *) GDALGetProjectionRef( hSrcDS ); hRasterSRS = OSRNewSpatialReference(NULL); if( OSRImportFromWkt( hRasterSRS, &pszProjection ) != CE_None ) { OSRDestroySpatialReference(hRasterSRS); hRasterSRS = NULL; } } OGRSpatialReferenceH hSrcSRS = OGR_G_GetSpatialReference( hMultiPolygon ); if( hRasterSRS != NULL && hSrcSRS != NULL ) { if( OSRIsSame(hSrcSRS, hRasterSRS) == FALSE ) { fprintf(stderr, "Warning : the source raster dataset and the input vector layer do not have\n" "the same SRS. Results will be probably incorrect.\n"); } } else if( hRasterSRS != NULL && hSrcSRS == NULL ) { fprintf(stderr, "Warning : the output raster dataset has a SRS, but the input vector layer\n" "not. Results may be incorrect.\n"); } else if( hRasterSRS == NULL && hSrcSRS != NULL ) { fprintf(stderr, "Warning : the input vector layer has a SRS, but the output raster dataset not.\n" "Results may be incorrect.\n"); } if( hRasterSRS != NULL ) OSRDestroySpatialReference(hRasterSRS); /* -------------------------------------------------------------------- */ /* Transform the geometry to pixel/line coordinates. */ /* -------------------------------------------------------------------- */ GeoTransform_Transformer oTransformer; double adfGT[6]; GDALGetGeoTransform( hSrcDS, adfGT ); GDALInvGeoTransform( adfGT, oTransformer.adfGeoTransform ); OGR_G_Transform( hMultiPolygon, (OGRCoordinateTransformationH) &oTransformer ); /* -------------------------------------------------------------------- */ /* Convert aggregate geometry into WKT. */ /* -------------------------------------------------------------------- */ char *pszWKT = NULL; OGR_G_ExportToWkt( hMultiPolygon, &pszWKT ); OGR_G_DestroyGeometry( hMultiPolygon ); *ppapszWarpOptions = CSLSetNameValue( *ppapszWarpOptions, "CUTLINE", pszWKT ); CPLFree( pszWKT ); #endif } /************************************************************************/ /* InsertCenterLong() */ /* */ /* Insert a CENTER_LONG Extension entry on a GEOGCS to indicate */ /* the center longitude of the dataset for wrapping purposes. */ /************************************************************************/ #ifdef notdef static CPLString InsertCenterLong( GDALDatasetH hDS, CPLString osWKT ) { if( !EQUALN(osWKT.c_str(), "GEOGCS[", 7) ) return osWKT; if( strstr(osWKT,"EXTENSION[\"CENTER_LONG") != NULL ) return osWKT; /* -------------------------------------------------------------------- */ /* For now we only do this if we have a geotransform since */ /* other forms require a bunch of extra work. */ /* -------------------------------------------------------------------- */ double adfGeoTransform[6]; if( GDALGetGeoTransform( hDS, adfGeoTransform ) != CE_None ) return osWKT; /* -------------------------------------------------------------------- */ /* Compute min/max longitude based on testing the four corners. */ /* -------------------------------------------------------------------- */ double dfMinLong, dfMaxLong; int nXSize = GDALGetRasterXSize( hDS ); int nYSize = GDALGetRasterYSize( hDS ); dfMinLong = MIN(MIN(adfGeoTransform[0] + 0 * adfGeoTransform[1] + 0 * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + 0 * adfGeoTransform[2]), MIN(adfGeoTransform[0] + 0 * adfGeoTransform[1] + nYSize * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + nYSize * adfGeoTransform[2])); dfMaxLong = MAX(MAX(adfGeoTransform[0] + 0 * adfGeoTransform[1] + 0 * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + 0 * adfGeoTransform[2]), MAX(adfGeoTransform[0] + 0 * adfGeoTransform[1] + nYSize * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + nYSize * adfGeoTransform[2])); if( dfMaxLong - dfMinLong > 360.0 ) return osWKT; /* -------------------------------------------------------------------- */ /* Insert center long. */ /* -------------------------------------------------------------------- */ OGRSpatialReference oSRS( osWKT ); double dfCenterLong = (dfMaxLong + dfMinLong) / 2.0; OGR_SRSNode *poExt; poExt = new OGR_SRSNode( "EXTENSION" ); poExt->AddChild( new OGR_SRSNode( "CENTER_LONG" ) ); poExt->AddChild( new OGR_SRSNode( CPLString().Printf("%g",dfCenterLong) )); oSRS.GetRoot()->AddChild( poExt->Clone() ); delete poExt; /* -------------------------------------------------------------------- */ /* Convert back to wkt. */ /* -------------------------------------------------------------------- */ char *pszWKT = NULL; oSRS.exportToWkt( &pszWKT ); osWKT = pszWKT; CPLFree( pszWKT ); return osWKT; } #endif