Sophie

Sophie

distrib > Fedora > 13 > i386 > by-pkgid > 7fd7c575020aa78a8e2e309ea8909f43 > files > 1281

gdal-1.6.2-6.fc13.i686.rpm

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/xhtml;charset=UTF-8"/>
<title>GDAL: GDAL Warp API Tutorial</title>
<link href="tabs.css" rel="stylesheet" type="text/css"/>
<link href="doxygen.css" rel="stylesheet" type="text/css"/>
</head>
<body>
<!-- Generated by Doxygen 1.6.2-20100208 -->
<div class="navigation" id="top">
  <div class="tabs">
    <ul>
      <li><a href="index.html"><span>Main&nbsp;Page</span></a></li>
      <li class="current"><a href="pages.html"><span>Related&nbsp;Pages</span></a></li>
      <li><a href="annotated.html"><span>Classes</span></a></li>
      <li><a href="files.html"><span>Files</span></a></li>
    </ul>
  </div>
</div>
<div class="contents">


<h1><a class="anchor" id="warptut">GDAL Warp API Tutorial </a></h1><h2><a class="anchor" id="warptut_overview">
Overview</a></h2>
<p>The GDAL Warp API (declared in <a class="el" href="gdalwarper_8h.html">gdalwarper.h</a>) provides services for high performance image warping using application provided geometric transformation functions (GDALTransformerFunc), a variety of resampling kernels, and various masking options. Files much larger than can be held in memory can be warped.</p>
<p>This tutorial demonstrates how to implement an application using the Warp API. It assumes implementation in C++ as C and Python bindings are incomplete for the Warp API. It also assumes familiarity with the <a href="gdal_datamodel.html">GDAL Data Model</a>, and the general GDAL API.</p>
<p>Applications normally perform a warp by initializing a <a class="el" href="structGDALWarpOptions.html">GDALWarpOptions</a> structure with the options to be utilized, instantiating a <a class="el" href="classGDALWarpOperation.html">GDALWarpOperation</a> based on these options, and then invoking the <a class="el" href="classGDALWarpOperation.html#a589a9b74fa370cc9eaf11bdfd9aab2ae">GDALWarpOperation::ChunkAndWarpImage()</a> method to perform the warp options internally using the <a class="el" href="classGDALWarpKernel.html">GDALWarpKernel</a> class.</p>
<h2><a class="anchor" id="warptut_simple">
A Simple Reprojection Case</a></h2>
<p>First we will construct a relatively simple example for reprojecting an image, assuming an appropriate output file already exists, and with minimal error checking.</p>
<div class="fragment"><pre class="fragment"><span class="preprocessor">#include &quot;<a class="code" href="gdalwarper_8h.html">gdalwarper.h</a>&quot;</span>

<span class="keywordtype">int</span> main()
{
    GDALDatasetH  hSrcDS, hDstDS;

    <span class="comment">// Open input and output files. </span>

    <a class="code" href="gdal_8h.html#a9d40bc998bd6ed07ccde96028e85ae26">GDALAllRegister</a>();

    hSrcDS = <a class="code" href="gdal_8h.html#ae97be045eb4701183ad332ffce29745b">GDALOpen</a>( <span class="stringliteral">&quot;in.tif&quot;</span>, <a class="code" href="gdal_8h.html#a045e3967c208993f70257bfd40c9f1d7a5a021a550b9d5640307d3c0e7e35b732">GA_ReadOnly</a> );
    hDstDS = <a class="code" href="gdal_8h.html#ae97be045eb4701183ad332ffce29745b">GDALOpen</a>( <span class="stringliteral">&quot;out.tif&quot;</span>, <a class="code" href="gdal_8h.html#a045e3967c208993f70257bfd40c9f1d7a61c6081de474ef2a756982d3c53130a2">GA_Update</a> );

    <span class="comment">// Setup warp options. </span>
    
    <a class="code" href="structGDALWarpOptions.html">GDALWarpOptions</a> *psWarpOptions = GDALCreateWarpOptions();

    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a65f68e1cf7ca641c5d0a93762d0416d5">hSrcDS</a> = hSrcDS;
    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a0550f97dd229ae14e7356588a9c74ffb">hDstDS</a> = hDstDS;

    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a9b60196e623b269e8f4905bf9e1792e2">nBandCount</a> = 1;
    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a841df78e52087580fa6c0bd811714160">panSrcBands</a> = 
        (<span class="keywordtype">int</span> *) CPLMalloc(<span class="keyword">sizeof</span>(<span class="keywordtype">int</span>) * psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a9b60196e623b269e8f4905bf9e1792e2">nBandCount</a> );
    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a841df78e52087580fa6c0bd811714160">panSrcBands</a>[0] = 1;
    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a4997d9eff29876052adb47f7928497f7">panDstBands</a> = 
        (<span class="keywordtype">int</span> *) CPLMalloc(<span class="keyword">sizeof</span>(<span class="keywordtype">int</span>) * psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a9b60196e623b269e8f4905bf9e1792e2">nBandCount</a> );
    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a4997d9eff29876052adb47f7928497f7">panDstBands</a>[0] = 1;

    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a5cf9b6e00e148b360f659c4e2ac68c01">pfnProgress</a> = <a class="code" href="gdal_8h.html#a00e9838f30867a9dbeb65e454d3bea1e">GDALTermProgress</a>;   

    <span class="comment">// Establish reprojection transformer. </span>

    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a35647845a2629da876f379f82d58a2cd">pTransformerArg</a> = 
        <a class="code" href="gdal__alg_8h.html#a7671696d085085a0bfba3c3df9ffcc0a">GDALCreateGenImgProjTransformer</a>( hSrcDS, 
                                         <a class="code" href="gdal_8h.html#a639a11014cf6c4ff30df6f21d5db9da2">GDALGetProjectionRef</a>(hSrcDS), 
                                         hDstDS,
                                         <a class="code" href="gdal_8h.html#a639a11014cf6c4ff30df6f21d5db9da2">GDALGetProjectionRef</a>(hDstDS), 
                                         FALSE, 0.0, 1 );
    psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a8899bacf99abdfd14c8bd590bd6079fa">pfnTransformer</a> = <a class="code" href="gdal__alg_8h.html#a109c26234c2f934164e29649353532b6">GDALGenImgProjTransform</a>;

    <span class="comment">// Initialize and execute the warp operation. </span>

    <a class="code" href="classGDALWarpOperation.html">GDALWarpOperation</a> oOperation;

    oOperation.<a class="code" href="classGDALWarpOperation.html#a52447611e7e196187a4667842f628d3b">Initialize</a>( psWarpOptions );
    oOperation.<a class="code" href="classGDALWarpOperation.html#a589a9b74fa370cc9eaf11bdfd9aab2ae">ChunkAndWarpImage</a>( 0, 0, 
                                  <a class="code" href="gdal_8h.html#a4ef08b38a70b6e04f25a81bd82ef0138">GDALGetRasterXSize</a>( hDstDS ), 
                                  <a class="code" href="gdal_8h.html#ae0c0af31441c6bac994f35ac26c82f99">GDALGetRasterYSize</a>( hDstDS ) );

    <a class="code" href="gdal__alg_8h.html#a5fb383c4e5197e8e37ae1265cca8124d">GDALDestroyGenImgProjTransformer</a>( psWarpOptions-&gt;<a class="code" href="structGDALWarpOptions.html#a35647845a2629da876f379f82d58a2cd">pTransformerArg</a> );
    GDALDestroyWarpOptions( psWarpOptions );

    <a class="code" href="gdal_8h.html#a0984222d45a72028fcbbf1f44831ffbc">GDALClose</a>( hDstDS );
    <a class="code" href="gdal_8h.html#a0984222d45a72028fcbbf1f44831ffbc">GDALClose</a>( hSrcDS );

    <span class="keywordflow">return</span> 0;
}
</pre></div><p>This example opens the existing input and output files (in.tif and out.tif). A <a class="el" href="structGDALWarpOptions.html">GDALWarpOptions</a> structure is allocated (GDALCreateWarpOptions() sets lots of sensible defaults for stuff, always use it for defaulting things), and the input and output file handles, and band lists are set. The panSrcBands and panDstBands lists are dynamically allocated here and will be free automatically by GDALDestroyWarpOptions(). The simple terminal output progress monitor (GDALTermProgress) is installed for reporting completion progress to the user.</p>
<p><a class="el" href="gdal__alg_8h.html#a7671696d085085a0bfba3c3df9ffcc0a">GDALCreateGenImgProjTransformer()</a> is used to initialize the reprojection transformation between the source and destination images. We assume that they already have reasonable bounds and coordinate systems set. Use of GCPs is disabled.</p>
<p>Once the options structure is ready, a <a class="el" href="classGDALWarpOperation.html">GDALWarpOperation</a> is instantiated using them, and the warp actually performed with <a class="el" href="classGDALWarpOperation.html#a589a9b74fa370cc9eaf11bdfd9aab2ae">GDALWarpOperation::ChunkAndWarpImage()</a>. Then the transformer, warp options and datasets are cleaned up.</p>
<p>Normally error check would be needed after opening files, setting up the reprojection transformer (returns NULL on failure), and initializing the warp.</p>
<h2><a class="anchor" id="warptut_options">
Other Warping Options</a></h2>
<p>The <a class="el" href="structGDALWarpOptions.html">GDALWarpOptions</a> structures contains a number of items that can be set to control warping behavior. A few of particular interest are:</p>
<ol>
<li>
<p class="startli"><a class="el" href="structGDALWarpOptions.html#aa72d8bd37f896272cd979ce9dc9d65e9">GDALWarpOptions::dfWarpMemoryLimit</a> - Set the maximum amount of memory to be used by the <a class="el" href="classGDALWarpOperation.html">GDALWarpOperation</a> when selecting a size of image chunk to operate on. The value is in bytes, and the default is likely to be conservative (small). Increasing the chunk size can help substantially in some situations but care should be taken to ensure that this size, plus the GDAL cache size plus the working set of GDAL, your application and the operating system are less than the size of RAM or else excessive swapping is likely to interfere with performance. On a system with 256MB of RAM, a value of at least 64MB (roughly 64000000 bytes) is reasonable. Note that this value does <b>not</b> include the memory used by GDAL for low level block caching.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">GDALWarpOpations::eResampleAlg - One of GRA_NearestNeighbour (the default, and fastest), GRA_Bilinear (2x2 bilinear resampling) or GRA_Cubic. The GRA_NearestNeighbour type should generally be used for thematic or colormapped images. The other resampling types may give better results for thematic images, especially when substantially changing resolution.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli"><a class="el" href="structGDALWarpOptions.html#a304bf40101dbea72b77067071919eb21">GDALWarpOptions::padfSrcNoDataReal</a> - This array (one entry per band being processed) may be setup with a "nodata" value for each band if you wish to avoid having pixels of some background value copied to the destination image.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli"><a class="anchor" id="#warpoptions"></a> <a class="el" href="structGDALWarpOptions.html#a0ed77f9917bb96c7a9aabd73d4d06e08">GDALWarpOptions::papszWarpOptions</a> - This is a string list of NAME=VALUE options passed to the warper. See the <a class="el" href="structGDALWarpOptions.html#a0ed77f9917bb96c7a9aabd73d4d06e08">GDALWarpOptions::papszWarpOptions</a> docs for all options. Supported values include:</p>
<ul>
<li>
<p class="startli">INIT_DEST=[value] or INIT_DEST=NO_DATA: This option forces the destination image to be initialized to the indicated value (for all bands) or indicates that it should be initialized to the NO_DATA value in padfDstNoDataReal/padfDstNoDataImag. If this value isn't set the destination image will be read and the source warp overlayed on it.</p>
<p class="endli"></p>
</li>
<li>
WRITE_FLUSH=YES/NO: This option forces a flush to disk of data after each chunk is processed. In some cases this helps ensure a serial writing of the output data otherwise a block of data may be written to disk each time a block of data is read for the input buffer resulting in a lot of extra seeking around the disk, and reduced IO throughput. The default at this time is NO. </li>
</ul>
<p class="endli"></p>
</li>
</ol>
<h2><a class="anchor" id="warptut_output">
Creating the Output File</a></h2>
<p>In the previous case an appropriate output file was already assumed to exist. Now we will go through a case where a new file with appropriate bounds in a new coordinate system is created. This operation doesn't relate specifically to the warp API. It is just using the transformation API.</p>
<div class="fragment"><pre class="fragment"><span class="preprocessor">#include &quot;<a class="code" href="gdalwarper_8h.html">gdalwarper.h</a>&quot;</span>
<span class="preprocessor">#include &quot;ogr_spatialref.h&quot;</span>

...

    GDALDriverH hDriver;
    <a class="code" href="gdal_8h.html#a22e22ce0a55036a96f652765793fb7a4">GDALDataType</a> eDT;
    GDALDatasetH hDstDS;
    GDALDatasetH hSrcDS;

    <span class="comment">// Open the source file. </span>

    hSrcDS = <a class="code" href="gdal_8h.html#ae97be045eb4701183ad332ffce29745b">GDALOpen</a>( <span class="stringliteral">&quot;in.tif&quot;</span>, <a class="code" href="gdal_8h.html#a045e3967c208993f70257bfd40c9f1d7a5a021a550b9d5640307d3c0e7e35b732">GA_ReadOnly</a> );
    CPLAssert( hSrcDS != NULL );
    
    <span class="comment">// Create output with same datatype as first input band. </span>

    eDT = <a class="code" href="gdal_8h.html#a2edba2a096915aa63e7ca0bf4c383bd0">GDALGetRasterDataType</a>(<a class="code" href="gdal_8h.html#a2a74e5e34528589303c1521ebfb9c162">GDALGetRasterBand</a>(hSrcDS,1));

    <span class="comment">// Get output driver (GeoTIFF format)</span>

    hDriver = <a class="code" href="gdal_8h.html#ae8ae868eef1e4773283d137b0a1adfc4">GDALGetDriverByName</a>( <span class="stringliteral">&quot;GTiff&quot;</span> );
    CPLAssert( hDriver != NULL );

    <span class="comment">// Get Source coordinate system. </span>

    <span class="keyword">const</span> <span class="keywordtype">char</span> *pszSrcWKT, *pszDstWKT = NULL;

    pszSrcWKT = <a class="code" href="gdal_8h.html#a639a11014cf6c4ff30df6f21d5db9da2">GDALGetProjectionRef</a>( hSrcDS );
    CPLAssert( pszSrcWKT != NULL &amp;&amp; strlen(pszSrcWKT) &gt; 0 );

    <span class="comment">// Setup output coordinate system that is UTM 11 WGS84. </span>

    OGRSpatialReference oSRS;

    oSRS.SetUTM( 11, TRUE );
    oSRS.SetWellKnownGeogCS( <span class="stringliteral">&quot;WGS84&quot;</span> );

    oSRS.exportToWkt( &amp;pszDstWKT );

    <span class="comment">// Create a transformer that maps from source pixel/line coordinates</span>
    <span class="comment">// to destination georeferenced coordinates (not destination </span>
    <span class="comment">// pixel line).  We do that by omitting the destination dataset</span>
    <span class="comment">// handle (setting it to NULL). </span>

    <span class="keywordtype">void</span> *hTransformArg;

    hTransformArg = 
        <a class="code" href="gdal__alg_8h.html#a7671696d085085a0bfba3c3df9ffcc0a">GDALCreateGenImgProjTransformer</a>( hSrcDS, pszSrcWKT, NULL, pszDstWKT, 
                                         FALSE, 0, 1 );
    CPLAssert( hTransformArg != NULL );

    <span class="comment">// Get approximate output georeferenced bounds and resolution for file. </span>

    <span class="keywordtype">double</span> adfDstGeoTransform[6];
    <span class="keywordtype">int</span> nPixels=0, nLines=0;
    CPLErr eErr;

    eErr = <a class="code" href="gdal__alg_8h.html#a816819e7495bfce06dbd110f7c57af65">GDALSuggestedWarpOutput</a>( hSrcDS, 
                                    GDALGenImgProjTransform, hTransformArg, 
                                    adfDstGeoTransform, &amp;nPixels, &amp;nLines );
    CPLAssert( eErr == CE_None );

    <a class="code" href="gdal__alg_8h.html#a5fb383c4e5197e8e37ae1265cca8124d">GDALDestroyGenImgProjTransformer</a>( hTransformArg );

    <span class="comment">// Create the output file.  </span>

    hDstDS = <a class="code" href="gdal_8h.html#af68516793118967e1292519cbd66442c">GDALCreate</a>( hDriver, <span class="stringliteral">&quot;out.tif&quot;</span>, nPixels, nLines, 
                         <a class="code" href="gdal_8h.html#a1b9f888aac1cb4dbc99dc1dc023174b7">GDALGetRasterCount</a>(hSrcDS), eDT, NULL );
    
    CPLAssert( hDstDS != NULL );

    <span class="comment">// Write out the projection definition. </span>

    <a class="code" href="gdal_8h.html#a145f2be5db1ac31a07a9d4389f4ace65">GDALSetProjection</a>( hDstDS, pszDstWKT );
    <a class="code" href="gdal_8h.html#ae93448112c1a7e69f2764c1aa3c6c8b5">GDALSetGeoTransform</a>( hDstDS, adfDstGeoTransform );

    <span class="comment">// Copy the color table, if required.</span>

    GDALColorTableH hCT;

    hCT = <a class="code" href="gdal_8h.html#ab4ebf9ba142ed1847cfb04143fb75c3e">GDALGetRasterColorTable</a>( <a class="code" href="gdal_8h.html#a2a74e5e34528589303c1521ebfb9c162">GDALGetRasterBand</a>(hSrcDS,1) );
    <span class="keywordflow">if</span>( hCT != NULL )
        <a class="code" href="gdal_8h.html#ade5abc76e229097e6799ab58414aeaed">GDALSetRasterColorTable</a>( <a class="code" href="gdal_8h.html#a2a74e5e34528589303c1521ebfb9c162">GDALGetRasterBand</a>(hDstDS,1), hCT );

    ... proceed with warp as before ...
</pre></div><p>Some notes on this logic:</p>
<ul>
<li>
<p class="startli">We need to create the transformer to output coordinates such that the output of the transformer is georeferenced, not pixel line coordinates since we use the transformer to map pixels around the source image into destination georeferenced coordinates.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">The <a class="el" href="gdal__alg_8h.html#a816819e7495bfce06dbd110f7c57af65">GDALSuggestedWarpOutput()</a> function will return an adfDstGeoTransform, nPixels and nLines that describes an output image size and georeferenced extents that should hold all pixels from the source image. The resolution is intended to be comparable to the source, but the output pixels are always square regardless of the shape of input pixels.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">The warper requires an output file in a format that can be "randomly" written to. This generally limits things to uncompressed formats that have an implementation of the Create() method (as opposed to CreateCopy()). To warp to compressed formats, or CreateCopy() style formats it is necessary to produce a full temporary copy of the image in a better behaved format, and then CreateCopy() it to the desired final format.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">The Warp API copies only pixels. All colormaps, georeferencing and other metadata must be copied to the destination by the application.</p>
<p class="endli"></p>
</li>
</ul>
<h2><a class="anchor" id="warptut_perfomance">
Performance Optimization</a></h2>
<p>There are a number of things that can be done to optimize the performance of the warp API.</p>
<ol>
<li>
<p class="startli">Increase the amount of memory available for the Warp API chunking so that larger chunks can be operated on at a time. This is the <a class="el" href="structGDALWarpOptions.html#aa72d8bd37f896272cd979ce9dc9d65e9">GDALWarpOptions::dfWarpMemoryLimit</a> parameter. In theory the larger the chunk size operated on the more efficient the I/O strategy, and the more efficient the approximated transformation will be. However, the sum of the warp memory and the GDAL cache should be less than RAM size, likely around 2/3 of RAM size.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">Increase the amount of memory for GDAL caching. This is especially important when working with very large input and output images that are scanline oriented. If all the input or output scanlines have to be re-read for each chunk they intersect performance may degrade greatly. Use <a class="el" href="gdal_8h.html#adfb1e95703ee577f012935869852d96c">GDALSetCacheMax()</a> to control the amount of memory available for caching within GDAL.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">Use an approximated transformation instead of exact reprojection for each pixel to be transformed. This code illustrates how an approximated transformation could be created based on a reprojection transformation, but with a given error threshold (dfErrorThreshold in output pixels).</p>
<div class="fragment"><pre class="fragment">        hTransformArg = 
            <a class="code" href="gdal__alg_8h.html#a4ec403b75384f0a71130eb009078426f">GDALCreateApproxTransformer</a>( GDALGenImgProjTransform, 
                                         hGenImgProjArg, dfErrorThreshold );
        pfnTransformer = <a class="code" href="gdal__alg_8h.html#a766ccb23b021d30d86908c08ad8d1668">GDALApproxTransform</a>;
</pre></div><p class="endli"></p>
</li>
<li>
<p class="startli">When writing to a blank output file, use the INIT_DEST option in the <a class="el" href="structGDALWarpOptions.html#a0ed77f9917bb96c7a9aabd73d4d06e08">GDALWarpOptions::papszWarpOptions</a> to cause the output chunks to be initialized to a fixed value, instead of being read from the output. This can substantially reduce unnecessary IO work.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">Use tiled input and output formats. Tiled formats allow a given chunk of source and destination imagery to be accessed without having to touch a great deal of extra image data. Large scanline oriented files can result in a great deal of wasted extra IO.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">Process all bands in one call. This ensures the transformation calculations don't have to be performed for each band.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">Use the <a class="el" href="classGDALWarpOperation.html#aa4fcccc201832f6f692fd290848cf4f1">GDALWarpOperation::ChunkAndWarpMulti()</a> method instead of <a class="el" href="classGDALWarpOperation.html#a589a9b74fa370cc9eaf11bdfd9aab2ae">GDALWarpOperation::ChunkAndWarpImage()</a>. It uses a separate thread for the IO and the actual image warp operation allowing more effective use of CPU and IO bandwidth. For this to work GDAL needs to have been built with multi-threading support (default on Win32, --with-pthreads on Unix).</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">The resampling kernels vary is work required from nearest neighbour being least, then bilinear then cubic. Don't use a more complex resampling kernel than needed.</p>
<p class="endli"></p>
</li>
<li>
<p class="startli">Avoid use of esoteric masking options so that special simplified logic case be used for common special cases. For instance, nearest neighbour resampling with no masking on 8bit data is highly optimized compared to the general case.</p>
<p class="endli"></p>
</li>
</ol>
<h2><a class="anchor" id="warptut_other_opts">
Other Masking Options</a></h2>
<p>The <a class="el" href="structGDALWarpOptions.html">GDALWarpOptions</a> include a bunch of esoteric masking capabilities, for validity masks, and density masks on input and output. Some of these are not yet implemented and others are implemented but poorly tested. Other than per-band validity masks it is advised that these features be used with caution at this time.</p>
 
<p>
$Id: warptut.dox 10252 2006-11-09 14:53:18Z fwarmerdam $
</p>
 </div>
<hr>

Generated for GDAL by 
<a href="http://www.doxygen.org/index.html"><img src="doxygen.png" alt="doxygen" align="middle" border="0"></a> 1.6.2-20100208.
</body>
</html>