PostGIS  2.4.9dev-r@@SVN_REVISION@@
rtpg_gdal.c
Go to the documentation of this file.
1 /*
2  *
3  * WKTRaster - Raster Types for PostGIS
4  * http://trac.osgeo.org/postgis/wiki/WKTRaster
5  *
6  * Copyright (C) 2011-2013 Regents of the University of California
7  * <bkpark@ucdavis.edu>
8  * Copyright (C) 2010-2011 Jorge Arevalo <jorge.arevalo@deimos-space.com>
9  * Copyright (C) 2010-2011 David Zwarg <dzwarg@azavea.com>
10  * Copyright (C) 2009-2011 Pierre Racine <pierre.racine@sbf.ulaval.ca>
11  * Copyright (C) 2009-2011 Mateusz Loskot <mateusz@loskot.net>
12  * Copyright (C) 2008-2009 Sandro Santilli <strk@kbt.io>
13  *
14  * This program is free software; you can redistribute it and/or
15  * modify it under the terms of the GNU General Public License
16  * as published by the Free Software Foundation; either version 2
17  * of the License, or (at your option) any later version.
18  *
19  * This program is distributed in the hope that it will be useful,
20  * but WITHOUT ANY WARRANTY; without even the implied warranty of
21  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22  * GNU General Public License for more details.
23  *
24  * You should have received a copy of the GNU General Public License
25  * along with this program; if not, write to the Free Software Foundation,
26  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
27  *
28  */
29 
30 #include <postgres.h>
31 #include <fmgr.h>
32 #include <funcapi.h> /* for SRF */
33 #include <utils/builtins.h> /* for text_to_cstring() */
34 #include "utils/lsyscache.h" /* for get_typlenbyvalalign */
35 #include "utils/array.h" /* for ArrayType */
36 #include "catalog/pg_type.h" /* for INT2OID, INT4OID, FLOAT4OID, FLOAT8OID and TEXTOID */
37 
38 #include "../../postgis_config.h"
39 
40 
41 #include "access/htup_details.h" /* for heap_form_tuple() */
42 
43 
44 #include "rtpostgis.h"
45 #include "rtpg_internal.h"
46 
47 /* convert GDAL raster to raster */
48 Datum RASTER_fromGDALRaster(PG_FUNCTION_ARGS);
49 
50 /* convert raster to GDAL raster */
51 Datum RASTER_asGDALRaster(PG_FUNCTION_ARGS);
52 Datum RASTER_getGDALDrivers(PG_FUNCTION_ARGS);
53 
54 /* warp a raster using GDAL Warp API */
55 Datum RASTER_GDALWarp(PG_FUNCTION_ARGS);
56 
57 /* ---------------------------------------------------------------- */
58 /* Returns raster from GDAL raster */
59 /* ---------------------------------------------------------------- */
61 Datum RASTER_fromGDALRaster(PG_FUNCTION_ARGS)
62 {
63  bytea *bytea_data;
64  uint8_t *data;
65  int data_len = 0;
66  VSILFILE *vsifp = NULL;
67  GDALDatasetH hdsSrc;
68  int srid = -1; /* -1 for NULL */
69 
70  rt_pgraster *pgraster = NULL;
72 
73  /* NULL if NULL */
74  if (PG_ARGISNULL(0))
75  PG_RETURN_NULL();
76 
77  /* get data */
78  bytea_data = (bytea *) PG_GETARG_BYTEA_P(0);
79  data = (uint8_t *) VARDATA(bytea_data);
80  data_len = VARSIZE(bytea_data) - VARHDRSZ;
81 
82  /* process srid */
83  /* NULL srid means try to determine SRID from bytea */
84  if (!PG_ARGISNULL(1))
85  srid = clamp_srid(PG_GETARG_INT32(1));
86 
87  /* create memory "file" */
88  vsifp = VSIFileFromMemBuffer("/vsimem/in.dat", data, data_len, FALSE);
89  if (vsifp == NULL) {
90  PG_FREE_IF_COPY(bytea_data, 0);
91  elog(ERROR, "RASTER_fromGDALRaster: Could not load bytea into memory file for use by GDAL");
92  PG_RETURN_NULL();
93  }
94 
95  /* register all GDAL drivers */
97 
98  /* open GDAL raster */
99  hdsSrc = rt_util_gdal_open("/vsimem/in.dat", GA_ReadOnly, 1);
100  if (hdsSrc == NULL) {
101  VSIFCloseL(vsifp);
102  PG_FREE_IF_COPY(bytea_data, 0);
103  elog(ERROR, "RASTER_fromGDALRaster: Could not open bytea with GDAL. Check that the bytea is of a GDAL supported format");
104  PG_RETURN_NULL();
105  }
106 
107 #if POSTGIS_DEBUG_LEVEL > 3
108  {
109  GDALDriverH hdrv = GDALGetDatasetDriver(hdsSrc);
110 
111  POSTGIS_RT_DEBUGF(4, "Input GDAL Raster info: %s, (%d x %d)",
112  GDALGetDriverShortName(hdrv),
113  GDALGetRasterXSize(hdsSrc),
114  GDALGetRasterYSize(hdsSrc)
115  );
116  }
117 #endif
118 
119  /* convert GDAL raster to raster */
120  raster = rt_raster_from_gdal_dataset(hdsSrc);
121 
122  GDALClose(hdsSrc);
123  VSIFCloseL(vsifp);
124  PG_FREE_IF_COPY(bytea_data, 0);
125 
126  if (raster == NULL) {
127  elog(ERROR, "RASTER_fromGDALRaster: Could not convert GDAL raster to raster");
128  PG_RETURN_NULL();
129  }
130 
131  /* apply SRID if set */
132  if (srid != -1)
133  rt_raster_set_srid(raster, srid);
134 
135  pgraster = rt_raster_serialize(raster);
136  rt_raster_destroy(raster);
137  if (!pgraster)
138  PG_RETURN_NULL();
139 
140  SET_VARSIZE(pgraster, pgraster->size);
141  PG_RETURN_POINTER(pgraster);
142 }
143 
148 Datum RASTER_asGDALRaster(PG_FUNCTION_ARGS)
149 {
150  rt_pgraster *pgraster = NULL;
152 
153  text *formattext = NULL;
154  char *format = NULL;
155  char **options = NULL;
156  text *optiontext = NULL;
157  char *option = NULL;
158  int srid = SRID_UNKNOWN;
159  char *srs = NULL;
160 
161  ArrayType *array;
162  Oid etype;
163  Datum *e;
164  bool *nulls;
165  int16 typlen;
166  bool typbyval;
167  char typalign;
168  int n = 0;
169  int i = 0;
170  int j = 0;
171 
172  uint8_t *gdal = NULL;
173  uint64_t gdal_size = 0;
174  bytea *result = NULL;
175  uint64_t result_size = 0;
176 
177  POSTGIS_RT_DEBUG(3, "RASTER_asGDALRaster: Starting");
178 
179  /* pgraster is null, return null */
180  if (PG_ARGISNULL(0)) PG_RETURN_NULL();
181  pgraster = (rt_pgraster *) PG_DETOAST_DATUM(PG_GETARG_DATUM(0));
182 
183  raster = rt_raster_deserialize(pgraster, FALSE);
184  if (!raster) {
185  PG_FREE_IF_COPY(pgraster, 0);
186  elog(ERROR, "RASTER_asGDALRaster: Could not deserialize raster");
187  PG_RETURN_NULL();
188  }
189 
190  /* format is required */
191  if (PG_ARGISNULL(1)) {
192  elog(NOTICE, "Format must be provided");
193  rt_raster_destroy(raster);
194  PG_FREE_IF_COPY(pgraster, 0);
195  PG_RETURN_NULL();
196  }
197  else {
198  formattext = PG_GETARG_TEXT_P(1);
199  format = text_to_cstring(formattext);
200  }
201 
202  POSTGIS_RT_DEBUGF(3, "RASTER_asGDALRaster: Arg 1 (format) is %s", format);
203 
204  /* process options */
205  if (!PG_ARGISNULL(2)) {
206  POSTGIS_RT_DEBUG(3, "RASTER_asGDALRaster: Processing Arg 2 (options)");
207  array = PG_GETARG_ARRAYTYPE_P(2);
208  etype = ARR_ELEMTYPE(array);
209  get_typlenbyvalalign(etype, &typlen, &typbyval, &typalign);
210 
211  switch (etype) {
212  case TEXTOID:
213  break;
214  default:
215  rt_raster_destroy(raster);
216  PG_FREE_IF_COPY(pgraster, 0);
217  elog(ERROR, "RASTER_asGDALRaster: Invalid data type for options");
218  PG_RETURN_NULL();
219  break;
220  }
221 
222  deconstruct_array(array, etype, typlen, typbyval, typalign, &e,
223  &nulls, &n);
224 
225  if (n) {
226  options = (char **) palloc(sizeof(char *) * (n + 1));
227  if (options == NULL) {
228  rt_raster_destroy(raster);
229  PG_FREE_IF_COPY(pgraster, 0);
230  elog(ERROR, "RASTER_asGDALRaster: Could not allocate memory for options");
231  PG_RETURN_NULL();
232  }
233 
234  /* clean each option */
235  for (i = 0, j = 0; i < n; i++) {
236  if (nulls[i]) continue;
237 
238  option = NULL;
239  switch (etype) {
240  case TEXTOID:
241  optiontext = (text *) DatumGetPointer(e[i]);
242  if (NULL == optiontext) break;
243  option = text_to_cstring(optiontext);
244 
245  /* trim string */
246  option = rtpg_trim(option);
247  POSTGIS_RT_DEBUGF(3, "RASTER_asGDALRaster: option is '%s'", option);
248  break;
249  }
250 
251  if (strlen(option)) {
252  options[j] = (char *) palloc(sizeof(char) * (strlen(option) + 1));
253  options[j] = option;
254  j++;
255  }
256  }
257 
258  if (j > 0) {
259  /* trim allocation */
260  options = repalloc(options, (j + 1) * sizeof(char *));
261 
262  /* add NULL to end */
263  options[j] = NULL;
264 
265  }
266  else {
267  pfree(options);
268  options = NULL;
269  }
270  }
271  }
272 
273  /* process srid */
274  /* NULL srid means use raster's srid */
275  if (PG_ARGISNULL(3))
276  srid = rt_raster_get_srid(raster);
277  else
278  srid = PG_GETARG_INT32(3);
279 
280  /* get srs from srid */
281  if (clamp_srid(srid) != SRID_UNKNOWN) {
282  srs = rtpg_getSR(srid);
283  if (NULL == srs) {
284  if (NULL != options) {
285  for (i = j - 1; i >= 0; i--) pfree(options[i]);
286  pfree(options);
287  }
288  rt_raster_destroy(raster);
289  PG_FREE_IF_COPY(pgraster, 0);
290  elog(ERROR, "RASTER_asGDALRaster: Could not find srtext for SRID (%d)", srid);
291  PG_RETURN_NULL();
292  }
293  POSTGIS_RT_DEBUGF(3, "RASTER_asGDALRaster: Arg 3 (srs) is %s", srs);
294  }
295  else
296  srs = NULL;
297 
298  POSTGIS_RT_DEBUG(3, "RASTER_asGDALRaster: Generating GDAL raster");
299  gdal = rt_raster_to_gdal(raster, srs, format, options, &gdal_size);
300 
301  /* free memory */
302  if (NULL != options) {
303  for (i = j - 1; i >= 0; i--) pfree(options[i]);
304  pfree(options);
305  }
306  if (NULL != srs) pfree(srs);
307  rt_raster_destroy(raster);
308  PG_FREE_IF_COPY(pgraster, 0);
309 
310  if (!gdal) {
311  elog(ERROR, "RASTER_asGDALRaster: Could not allocate and generate GDAL raster");
312  PG_RETURN_NULL();
313  }
314  POSTGIS_RT_DEBUGF(3, "RASTER_asGDALRaster: GDAL raster generated with %d bytes", (int) gdal_size);
315 
316  /* result is a varlena */
317  result_size = gdal_size + VARHDRSZ;
318  result = (bytea *) palloc(result_size);
319  if (NULL == result) {
320  elog(ERROR, "RASTER_asGDALRaster: Insufficient virtual memory for GDAL raster");
321  PG_RETURN_NULL();
322  }
323  SET_VARSIZE(result, result_size);
324  memcpy(VARDATA(result), gdal, VARSIZE(result) - VARHDRSZ);
325 
326  /* for test output
327  FILE *fh = NULL;
328  fh = fopen("/tmp/out.dat", "w");
329  fwrite(gdal, sizeof(uint8_t), gdal_size, fh);
330  fclose(fh);
331  */
332 
333  /* free gdal mem buffer */
334  if (gdal) CPLFree(gdal);
335 
336  POSTGIS_RT_DEBUG(3, "RASTER_asGDALRaster: Returning pointer to GDAL raster");
337  PG_RETURN_POINTER(result);
338 }
339 
340 #define VALUES_LENGTH 6
341 
346 Datum RASTER_getGDALDrivers(PG_FUNCTION_ARGS)
347 {
348  FuncCallContext *funcctx;
349  TupleDesc tupdesc;
350 
351  uint32_t drv_count;
352  rt_gdaldriver drv_set;
353  rt_gdaldriver drv_set2;
354  int call_cntr;
355  int max_calls;
356 
357  /* first call of function */
358  if (SRF_IS_FIRSTCALL()) {
359  MemoryContext oldcontext;
360 
361  /* create a function context for cross-call persistence */
362  funcctx = SRF_FIRSTCALL_INIT();
363 
364  /* switch to memory context appropriate for multiple function calls */
365  oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);
366 
367  drv_set = rt_raster_gdal_drivers(&drv_count, 1);
368  if (NULL == drv_set || !drv_count) {
369  elog(NOTICE, "No GDAL drivers found");
370  MemoryContextSwitchTo(oldcontext);
371  SRF_RETURN_DONE(funcctx);
372  }
373 
374  POSTGIS_RT_DEBUGF(3, "%d drivers returned", (int) drv_count);
375 
376  /* Store needed information */
377  funcctx->user_fctx = drv_set;
378 
379  /* total number of tuples to be returned */
380  funcctx->max_calls = drv_count;
381 
382  /* Build a tuple descriptor for our result type */
383  if (get_call_result_type(fcinfo, NULL, &tupdesc) != TYPEFUNC_COMPOSITE) {
384  ereport(ERROR, (
385  errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
386  errmsg(
387  "function returning record called in context "
388  "that cannot accept type record"
389  )
390  ));
391  }
392 
393  BlessTupleDesc(tupdesc);
394  funcctx->tuple_desc = tupdesc;
395  MemoryContextSwitchTo(oldcontext);
396  }
397 
398  /* stuff done on every call of the function */
399  funcctx = SRF_PERCALL_SETUP();
400 
401  call_cntr = funcctx->call_cntr;
402  max_calls = funcctx->max_calls;
403  tupdesc = funcctx->tuple_desc;
404  drv_set2 = funcctx->user_fctx;
405 
406  /* do when there is more left to send */
407  if (call_cntr < max_calls) {
408  Datum values[VALUES_LENGTH];
409  bool nulls[VALUES_LENGTH];
410  HeapTuple tuple;
411  Datum result;
412 
413  POSTGIS_RT_DEBUGF(3, "Result %d", call_cntr);
414 
415  memset(nulls, FALSE, sizeof(bool) * VALUES_LENGTH);
416 
417  values[0] = Int32GetDatum(drv_set2[call_cntr].idx);
418  values[1] = CStringGetTextDatum(drv_set2[call_cntr].short_name);
419  values[2] = CStringGetTextDatum(drv_set2[call_cntr].long_name);
420  values[3] = CStringGetTextDatum(drv_set2[call_cntr].create_options);
421 
422  POSTGIS_RT_DEBUGF(4, "Result %d, Index %d", call_cntr, drv_set2[call_cntr].idx);
423  POSTGIS_RT_DEBUGF(4, "Result %d, Short Name %s", call_cntr, drv_set2[call_cntr].short_name);
424  POSTGIS_RT_DEBUGF(4, "Result %d, Full Name %s", call_cntr, drv_set2[call_cntr].long_name);
425  POSTGIS_RT_DEBUGF(5, "Result %d, Create Options %s", call_cntr, drv_set2[call_cntr].create_options);
426 
427  /* build a tuple */
428  tuple = heap_form_tuple(tupdesc, values, nulls);
429 
430  /* make the tuple into a datum */
431  result = HeapTupleGetDatum(tuple);
432 
433  /* clean up */
434  pfree(drv_set2[call_cntr].short_name);
435  pfree(drv_set2[call_cntr].long_name);
436  pfree(drv_set2[call_cntr].create_options);
437 
438  SRF_RETURN_NEXT(funcctx, result);
439  }
440  /* do when there is no more left */
441  else {
442  pfree(drv_set2);
443  SRF_RETURN_DONE(funcctx);
444  }
445 }
446 
451 Datum RASTER_GDALWarp(PG_FUNCTION_ARGS)
452 {
453  rt_pgraster *pgraster = NULL;
454  rt_pgraster *pgrast = NULL;
455  rt_raster raster = NULL;
456  rt_raster rast = NULL;
457 
458  text *algtext = NULL;
459  char *algchar = NULL;
460  GDALResampleAlg alg = GRA_NearestNeighbour;
461  double max_err = 0.125;
462 
463  int src_srid = SRID_UNKNOWN;
464  char *src_srs = NULL;
465  int dst_srid = SRID_UNKNOWN;
466  char *dst_srs = NULL;
467  int no_srid = 0;
468 
469  double scale[2] = {0};
470  double *scale_x = NULL;
471  double *scale_y = NULL;
472 
473  double gridw[2] = {0};
474  double *grid_xw = NULL;
475  double *grid_yw = NULL;
476 
477  double skew[2] = {0};
478  double *skew_x = NULL;
479  double *skew_y = NULL;
480 
481  int dim[2] = {0};
482  int *dim_x = NULL;
483  int *dim_y = NULL;
484 
485  POSTGIS_RT_DEBUG(3, "RASTER_GDALWarp: Starting");
486 
487  /* pgraster is null, return null */
488  if (PG_ARGISNULL(0))
489  PG_RETURN_NULL();
490  pgraster = (rt_pgraster *) PG_DETOAST_DATUM(PG_GETARG_DATUM(0));
491 
492  /* raster */
493  raster = rt_raster_deserialize(pgraster, FALSE);
494  if (!raster) {
495  PG_FREE_IF_COPY(pgraster, 0);
496  elog(ERROR, "RASTER_GDALWarp: Could not deserialize raster");
497  PG_RETURN_NULL();
498  }
499 
500  /* resampling algorithm */
501  if (!PG_ARGISNULL(1)) {
502  algtext = PG_GETARG_TEXT_P(1);
503  algchar = rtpg_trim(rtpg_strtoupper(text_to_cstring(algtext)));
504  alg = rt_util_gdal_resample_alg(algchar);
505  }
506  POSTGIS_RT_DEBUGF(4, "Resampling algorithm: %d", alg);
507 
508  /* max error */
509  if (!PG_ARGISNULL(2)) {
510  max_err = PG_GETARG_FLOAT8(2);
511  if (max_err < 0.) max_err = 0.;
512  }
513  POSTGIS_RT_DEBUGF(4, "max_err: %f", max_err);
514 
515  /* source SRID */
516  src_srid = clamp_srid(rt_raster_get_srid(raster));
517  POSTGIS_RT_DEBUGF(4, "source SRID: %d", src_srid);
518 
519  /* target SRID */
520  if (!PG_ARGISNULL(3)) {
521  dst_srid = clamp_srid(PG_GETARG_INT32(3));
522  if (dst_srid == SRID_UNKNOWN) {
523  rt_raster_destroy(raster);
524  PG_FREE_IF_COPY(pgraster, 0);
525  elog(ERROR, "RASTER_GDALWarp: %d is an invalid target SRID", dst_srid);
526  PG_RETURN_NULL();
527  }
528  }
529  else
530  dst_srid = src_srid;
531  POSTGIS_RT_DEBUGF(4, "destination SRID: %d", dst_srid);
532 
533  /* target SRID != src SRID, error */
534  if (src_srid == SRID_UNKNOWN && dst_srid != src_srid) {
535  rt_raster_destroy(raster);
536  PG_FREE_IF_COPY(pgraster, 0);
537  elog(ERROR, "RASTER_GDALWarp: Input raster has unknown (%d) SRID", src_srid);
538  PG_RETURN_NULL();
539  }
540  /* target SRID == src SRID, no reprojection */
541  else if (dst_srid == src_srid) {
542  no_srid = 1;
543  }
544 
545  /* scale x */
546  if (!PG_ARGISNULL(4)) {
547  scale[0] = PG_GETARG_FLOAT8(4);
548  if (FLT_NEQ(scale[0], 0)) scale_x = &scale[0];
549  }
550 
551  /* scale y */
552  if (!PG_ARGISNULL(5)) {
553  scale[1] = PG_GETARG_FLOAT8(5);
554  if (FLT_NEQ(scale[1], 0)) scale_y = &scale[1];
555  }
556 
557  /* grid alignment x */
558  if (!PG_ARGISNULL(6)) {
559  gridw[0] = PG_GETARG_FLOAT8(6);
560  grid_xw = &gridw[0];
561  }
562 
563  /* grid alignment y */
564  if (!PG_ARGISNULL(7)) {
565  gridw[1] = PG_GETARG_FLOAT8(7);
566  grid_yw = &gridw[1];
567  }
568 
569  /* skew x */
570  if (!PG_ARGISNULL(8)) {
571  skew[0] = PG_GETARG_FLOAT8(8);
572  if (FLT_NEQ(skew[0], 0)) skew_x = &skew[0];
573  }
574 
575  /* skew y */
576  if (!PG_ARGISNULL(9)) {
577  skew[1] = PG_GETARG_FLOAT8(9);
578  if (FLT_NEQ(skew[1], 0)) skew_y = &skew[1];
579  }
580 
581  /* width */
582  if (!PG_ARGISNULL(10)) {
583  dim[0] = PG_GETARG_INT32(10);
584  if (dim[0] < 0) dim[0] = 0;
585  if (dim[0] > 0) dim_x = &dim[0];
586  }
587 
588  /* height */
589  if (!PG_ARGISNULL(11)) {
590  dim[1] = PG_GETARG_INT32(11);
591  if (dim[1] < 0) dim[1] = 0;
592  if (dim[1] > 0) dim_y = &dim[1];
593  }
594 
595  /* check that at least something is to be done */
596  if (
597  (dst_srid == SRID_UNKNOWN) &&
598  (scale_x == NULL) && (scale_y == NULL) &&
599  (grid_xw == NULL) && (grid_yw == NULL) &&
600  (skew_x == NULL) && (skew_y == NULL) &&
601  (dim_x == NULL) && (dim_y == NULL)
602  ) {
603  elog(NOTICE, "No resampling parameters provided. Returning original raster");
604  rt_raster_destroy(raster);
605  PG_RETURN_POINTER(pgraster);
606  }
607  /* both values of alignment must be provided if any one is provided */
608  else if (
609  (grid_xw != NULL && grid_yw == NULL) ||
610  (grid_xw == NULL && grid_yw != NULL)
611  ) {
612  elog(NOTICE, "Values must be provided for both X and Y when specifying the alignment. Returning original raster");
613  rt_raster_destroy(raster);
614  PG_RETURN_POINTER(pgraster);
615  }
616  /* both values of scale must be provided if any one is provided */
617  else if (
618  (scale_x != NULL && scale_y == NULL) ||
619  (scale_x == NULL && scale_y != NULL)
620  ) {
621  elog(NOTICE, "Values must be provided for both X and Y when specifying the scale. Returning original raster");
622  rt_raster_destroy(raster);
623  PG_RETURN_POINTER(pgraster);
624  }
625  /* scale and width/height provided */
626  else if (
627  (scale_x != NULL || scale_y != NULL) &&
628  (dim_x != NULL || dim_y != NULL)
629  ) {
630  elog(NOTICE, "Scale X/Y and width/height are mutually exclusive. Only provide one. Returning original raster");
631  rt_raster_destroy(raster);
632  PG_RETURN_POINTER(pgraster);
633  }
634 
635  /* get srses from srids */
636  if (!no_srid) {
637  /* source srs */
638  src_srs = rtpg_getSR(src_srid);
639  if (NULL == src_srs) {
640  rt_raster_destroy(raster);
641  PG_FREE_IF_COPY(pgraster, 0);
642  elog(ERROR, "RASTER_GDALWarp: Input raster has unknown SRID (%d)", src_srid);
643  PG_RETURN_NULL();
644  }
645  POSTGIS_RT_DEBUGF(4, "src srs: %s", src_srs);
646 
647  dst_srs = rtpg_getSR(dst_srid);
648  if (NULL == dst_srs) {
649  if (!no_srid) pfree(src_srs);
650  rt_raster_destroy(raster);
651  PG_FREE_IF_COPY(pgraster, 0);
652  elog(ERROR, "RASTER_GDALWarp: Target SRID (%d) is unknown", dst_srid);
653  PG_RETURN_NULL();
654  }
655  POSTGIS_RT_DEBUGF(4, "dst srs: %s", dst_srs);
656  }
657 
658  rast = rt_raster_gdal_warp(
659  raster,
660  src_srs, dst_srs,
661  scale_x, scale_y,
662  dim_x, dim_y,
663  NULL, NULL,
664  grid_xw, grid_yw,
665  skew_x, skew_y,
666  alg, max_err);
667  rt_raster_destroy(raster);
668  PG_FREE_IF_COPY(pgraster, 0);
669  if (!no_srid) {
670  pfree(src_srs);
671  pfree(dst_srs);
672  }
673  if (!rast) {
674  elog(ERROR, "RASTER_band: Could not create transformed raster");
675  PG_RETURN_NULL();
676  }
677 
678  /* add target SRID */
679  rt_raster_set_srid(rast, dst_srid);
680 
681  pgrast = rt_raster_serialize(rast);
682  rt_raster_destroy(rast);
683 
684  if (NULL == pgrast) PG_RETURN_NULL();
685 
686  POSTGIS_RT_DEBUG(3, "RASTER_GDALWarp: done");
687 
688  SET_VARSIZE(pgrast, pgrast->size);
689  PG_RETURN_POINTER(pgrast);
690 }
691 
int clamp_srid(int srid)
Return a valid SRID from an arbitrary integer Raises a notice if what comes out is different from wha...
Definition: lwutil.c:380
void * rt_raster_serialize(rt_raster raster)
Return this raster in serialized form.
Definition: rt_serialize.c:521
int rt_util_gdal_register_all(int force_register_all)
Definition: rt_util.c:334
raster
Be careful!! Zeros function&#39;s input parameter can be a (height x width) array, not (width x height): ...
Definition: rtrowdump.py:121
data
Definition: ovdump.py:103
rt_raster rt_raster_gdal_warp(rt_raster raster, const char *src_srs, const char *dst_srs, double *scale_x, double *scale_y, int *width, int *height, double *ul_xw, double *ul_yw, double *grid_xw, double *grid_yw, double *skew_x, double *skew_y, GDALResampleAlg resample_alg, double max_err)
Return a warped raster using GDAL Warp API.
Definition: rt_warp.c:178
Datum RASTER_GDALWarp(PG_FUNCTION_ARGS)
Definition: rtpg_gdal.c:451
#define POSTGIS_RT_DEBUGF(level, msg,...)
Definition: rtpostgis.h:65
#define VALUES_LENGTH
Definition: rtpg_gdal.c:340
unsigned int uint32_t
Definition: uthash.h:78
Datum RASTER_asGDALRaster(PG_FUNCTION_ARGS)
Definition: rtpg_gdal.c:148
#define SRID_UNKNOWN
Unknown SRID value.
Definition: liblwgeom.h:188
#define FLT_NEQ(x, y)
Definition: librtcore.h:2184
char * rtpg_trim(const char *input)
Datum RASTER_getGDALDrivers(PG_FUNCTION_ARGS)
Definition: rtpg_gdal.c:346
void rt_raster_set_srid(rt_raster raster, int32_t srid)
Set raster&#39;s SRID.
Definition: rt_raster.c:363
int32_t rt_raster_get_srid(rt_raster raster)
Get raster&#39;s SRID.
Definition: rt_raster.c:356
void rt_raster_destroy(rt_raster raster)
Release memory associated to a raster.
Definition: rt_raster.c:82
PG_FUNCTION_INFO_V1(RASTER_fromGDALRaster)
uint8_t * rt_raster_to_gdal(rt_raster raster, const char *srs, char *format, char **options, uint64_t *gdalsize)
Return formatted GDAL raster from raster.
Definition: rt_raster.c:1602
#define FALSE
Definition: dbfopen.c:168
Struct definitions.
Definition: librtcore.h:2201
rt_gdaldriver rt_raster_gdal_drivers(uint32_t *drv_count, uint8_t cancc)
Returns a set of available GDAL drivers.
Definition: rt_raster.c:1705
Datum RASTER_fromGDALRaster(PG_FUNCTION_ARGS)
Definition: rtpg_gdal.c:61
char * rtpg_getSR(int srid)
rt_raster rt_raster_from_gdal_dataset(GDALDatasetH ds)
Return a raster from a GDAL dataset.
Definition: rt_raster.c:2165
#define POSTGIS_RT_DEBUG(level, msg)
Definition: rtpostgis.h:61
GDALResampleAlg rt_util_gdal_resample_alg(const char *algname)
Convert cstring name to GDAL Resample Algorithm.
Definition: rt_util.c:91
unsigned char uint8_t
Definition: uthash.h:79
rt_raster rt_raster_deserialize(void *serialized, int header_only)
Return a raster from a serialized form.
Definition: rt_serialize.c:717
GDALDatasetH rt_util_gdal_open(const char *fn, GDALAccess fn_access, int shared)
Definition: rt_util.c:379
char * rtpg_strtoupper(char *str)