Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: Viewshed Generator
4 : * Purpose: Viewshed Generator mainline.
5 : * Author: Tamas Szekeres <szekerest@gmail.com>
6 : *
7 : ******************************************************************************
8 : *
9 : * SPDX-License-Identifier: MIT
10 : ****************************************************************************/
11 :
12 : #include <limits>
13 :
14 : #include "commonutils.h"
15 : #include "gdal.h"
16 : #include "gdalargumentparser.h"
17 :
18 : #include "viewshed/cumulative.h"
19 : #include "viewshed/viewshed.h"
20 :
21 : namespace gdal
22 : {
23 :
24 : namespace
25 : {
26 :
27 : struct Options
28 : {
29 : viewshed::Options opts;
30 : std::string osSrcFilename;
31 : int nBandIn{1};
32 : bool bQuiet;
33 : };
34 :
35 : /// Parse arguments into options structure.
36 : ///
37 : /// \param argParser Argument parser
38 : /// \param aosArgv Command line options as a string list
39 : /// \return Command line parsed as options
40 21 : Options parseArgs(GDALArgumentParser &argParser, const CPLStringList &aosArgv)
41 : {
42 21 : Options localOpts;
43 :
44 21 : viewshed::Options &opts = localOpts.opts;
45 :
46 21 : argParser.add_output_format_argument(opts.outputFormat);
47 21 : argParser.add_argument("-ox")
48 21 : .store_into(opts.observer.x)
49 42 : .metavar("<value>")
50 21 : .help(_("The X position of the observer (in SRS units)."));
51 :
52 21 : argParser.add_argument("-oy")
53 21 : .store_into(opts.observer.y)
54 42 : .metavar("<value>")
55 21 : .help(_("The Y position of the observer (in SRS units)."));
56 :
57 21 : argParser.add_argument("-oz")
58 21 : .default_value(2)
59 21 : .store_into(opts.observer.z)
60 42 : .metavar("<value>")
61 21 : .nargs(1)
62 : .help(_("The height of the observer above the DEM surface in the "
63 21 : "height unit of the DEM."));
64 :
65 21 : argParser.add_argument("-vv")
66 21 : .default_value(255)
67 21 : .store_into(opts.visibleVal)
68 42 : .metavar("<value>")
69 21 : .nargs(1)
70 21 : .help(_("Pixel value to set for visible areas."));
71 :
72 21 : argParser.add_argument("-iv")
73 21 : .default_value(0)
74 21 : .store_into(opts.invisibleVal)
75 42 : .metavar("<value>")
76 21 : .nargs(1)
77 21 : .help(_("Pixel value to set for invisible areas."));
78 :
79 21 : argParser.add_argument("-ov")
80 21 : .default_value(0)
81 21 : .store_into(opts.outOfRangeVal)
82 42 : .metavar("<value>")
83 21 : .nargs(1)
84 : .help(
85 : _("Pixel value to set for the cells that fall outside of the range "
86 21 : "specified by the observer location and the maximum distance."));
87 :
88 21 : argParser.add_creation_options_argument(opts.creationOpts);
89 :
90 21 : argParser.add_argument("-a_nodata")
91 21 : .default_value(-1.0)
92 21 : .store_into(opts.nodataVal)
93 42 : .metavar("<value>")
94 21 : .nargs(1)
95 : .help(_("The value to be set for the cells in the output raster that "
96 21 : "have no data."));
97 :
98 21 : argParser.add_argument("-tz")
99 21 : .default_value(0.0)
100 21 : .store_into(opts.targetHeight)
101 42 : .metavar("<value>")
102 21 : .nargs(1)
103 : .help(_("The height of the target above the DEM surface in the height "
104 21 : "unit of the DEM."));
105 :
106 21 : argParser.add_argument("-md")
107 21 : .default_value(0)
108 21 : .store_into(opts.maxDistance)
109 42 : .metavar("<value>")
110 21 : .nargs(1)
111 21 : .help(_("Maximum distance from observer to compute visibility."));
112 :
113 21 : argParser.add_argument("-j")
114 21 : .default_value(3)
115 21 : .store_into(opts.numJobs)
116 42 : .metavar("<value>")
117 21 : .nargs(1)
118 : .help(_(
119 21 : "Number of relative simultaneous jobs to run in cumulative mode"));
120 :
121 : // Value for standard atmospheric refraction. See
122 : // doc/source/programs/gdal_viewshed.rst
123 21 : argParser.add_argument("-cc")
124 21 : .default_value(0.85714)
125 21 : .store_into(opts.curveCoeff)
126 42 : .metavar("<value>")
127 21 : .nargs(1)
128 : .help(_("Coefficient to consider the effect of the curvature and "
129 21 : "refraction."));
130 :
131 21 : argParser.add_argument("-b")
132 21 : .default_value(localOpts.nBandIn)
133 21 : .store_into(localOpts.nBandIn)
134 42 : .metavar("<value>")
135 21 : .nargs(1)
136 21 : .help(_("Select an input band band containing the DEM data."));
137 :
138 21 : argParser.add_argument("-om")
139 21 : .choices("NORMAL", "DEM", "GROUND", "ACCUM")
140 42 : .metavar("NORMAL|DEM|GROUND|ACCUM")
141 : .action(
142 37 : [&into = opts.outputMode](const std::string &value)
143 : {
144 8 : if (EQUAL(value.c_str(), "DEM"))
145 1 : into = viewshed::OutputMode::DEM;
146 7 : else if (EQUAL(value.c_str(), "GROUND"))
147 1 : into = viewshed::OutputMode::Ground;
148 6 : else if (EQUAL(value.c_str(), "ACCUM"))
149 1 : into = viewshed::OutputMode::Cumulative;
150 : else
151 5 : into = viewshed::OutputMode::Normal;
152 21 : })
153 21 : .nargs(1)
154 21 : .help(_("Sets what information the output contains."));
155 :
156 21 : argParser.add_argument("-os")
157 21 : .default_value(10)
158 21 : .store_into(opts.observerSpacing)
159 42 : .metavar("<value>")
160 21 : .nargs(1)
161 21 : .help(_("Spacing between observer cells when using cumulative mode."));
162 :
163 21 : argParser.add_quiet_argument(&localOpts.bQuiet);
164 :
165 21 : argParser.add_argument("src_filename")
166 21 : .store_into(localOpts.osSrcFilename)
167 21 : .metavar("<src_filename>");
168 :
169 21 : argParser.add_argument("dst_filename")
170 21 : .store_into(opts.outputFilename)
171 21 : .metavar("<dst_filename>");
172 :
173 : try
174 : {
175 21 : argParser.parse_args(aosArgv);
176 : }
177 2 : catch (const std::exception &err)
178 : {
179 2 : argParser.display_error_and_usage(err);
180 2 : std::exit(1);
181 : }
182 19 : return localOpts;
183 : }
184 :
185 : /// Validate specified options.
186 : ///
187 : /// \param localOpts Options to validate
188 : /// \param argParser Argument parser
189 19 : void validateArgs(Options &localOpts, const GDALArgumentParser &argParser)
190 : {
191 19 : viewshed::Options &opts = localOpts.opts;
192 :
193 19 : if (opts.maxDistance < 0)
194 : {
195 0 : CPLError(CE_Failure, CPLE_AppDefined,
196 : "Max distance must be non-negative.");
197 0 : exit(2);
198 : }
199 :
200 19 : if (opts.outputFormat.empty())
201 : {
202 : opts.outputFormat =
203 12 : GetOutputDriverForRaster(opts.outputFilename.c_str());
204 12 : if (opts.outputFormat.empty())
205 : {
206 0 : exit(2);
207 : }
208 : }
209 :
210 19 : if (opts.outputMode != viewshed::OutputMode::Cumulative)
211 : {
212 54 : for (const char *opt : {"-os", "-j"})
213 36 : if (argParser.is_used(opt))
214 : {
215 0 : std::string err = "Option " + std::string(opt) +
216 0 : " can only be used in cumulative mode.";
217 0 : CPLError(CE_Failure, CPLE_AppDefined, "%s", err.c_str());
218 0 : exit(2);
219 : }
220 : }
221 :
222 19 : if (opts.outputMode == viewshed::OutputMode::Cumulative)
223 : {
224 6 : for (const char *opt : {"-ox", "-oy", "-vv", "-iv", "-md"})
225 5 : if (argParser.is_used(opt))
226 : {
227 0 : std::string err = "Option " + std::string(opt) +
228 0 : " can't be used in cumulative mode.";
229 0 : CPLError(CE_Failure, CPLE_AppDefined, "%s", err.c_str());
230 0 : exit(2);
231 : }
232 : }
233 : else
234 : {
235 51 : for (const char *opt : {"-ox", "-oy"})
236 35 : if (!argParser.is_used(opt))
237 : {
238 : std::string err =
239 4 : "Option " + std::string(opt) + " is required.";
240 2 : CPLError(CE_Failure, CPLE_AppDefined, "%s", err.c_str());
241 2 : exit(2);
242 : }
243 : }
244 :
245 : // For double values that are out of range for byte raster output,
246 : // set to zero. Values less than zero are sentinel as NULL nodata.
247 31 : if (opts.outputMode == viewshed::OutputMode::Normal &&
248 14 : opts.nodataVal > std::numeric_limits<uint8_t>::max())
249 0 : opts.nodataVal = 0;
250 17 : }
251 :
252 : // Adjust the coefficient of curvature for non-earth SRS.
253 : /// \param curveCoeff Current curve coefficient
254 : /// \param hSrcDS Source dataset
255 : /// \return Adjusted curve coefficient.
256 12 : double adjustCurveCoeff(double curveCoeff, GDALDatasetH hSrcDS)
257 : {
258 : const OGRSpatialReference *poSRS =
259 12 : GDALDataset::FromHandle(hSrcDS)->GetSpatialRef();
260 12 : if (poSRS)
261 : {
262 : OGRErr eSRSerr;
263 10 : const double dfSemiMajor = poSRS->GetSemiMajor(&eSRSerr);
264 10 : if (eSRSerr != OGRERR_FAILURE &&
265 10 : fabs(dfSemiMajor - SRS_WGS84_SEMIMAJOR) >
266 : 0.05 * SRS_WGS84_SEMIMAJOR)
267 : {
268 1 : curveCoeff = 1.0;
269 1 : CPLDebug("gdal_viewshed",
270 : "Using -cc=1.0 as a non-Earth CRS has been detected");
271 : }
272 : }
273 12 : return curveCoeff;
274 : }
275 :
276 : } // unnamed namespace
277 : } // namespace gdal
278 :
279 : /************************************************************************/
280 : /* main() */
281 : /************************************************************************/
282 :
283 22 : MAIN_START(argc, argv)
284 : {
285 : using namespace gdal;
286 :
287 22 : EarlySetConfigOptions(argc, argv);
288 :
289 22 : GDALAllRegister();
290 :
291 22 : argc = GDALGeneralCmdLineProcessor(argc, &argv, 0);
292 37 : CPLStringList aosArgv;
293 22 : aosArgv.Assign(argv, /* bTakeOwnership= */ true);
294 22 : if (argc < 1)
295 1 : std::exit(-argc);
296 :
297 57 : GDALArgumentParser argParser(aosArgv[0], /* bForBinary=*/true);
298 :
299 : argParser.add_description(
300 21 : _("Calculates a viewshed raster from an input raster DEM."));
301 :
302 : argParser.add_epilog(_("For more details, consult "
303 21 : "https://gdal.org/programs/gdal_viewshed.html"));
304 :
305 21 : Options localOpts = parseArgs(argParser, aosArgv);
306 19 : viewshed::Options &opts = localOpts.opts;
307 :
308 19 : validateArgs(localOpts, argParser);
309 :
310 : /* -------------------------------------------------------------------- */
311 : /* Open source raster file. */
312 : /* -------------------------------------------------------------------- */
313 : GDALDatasetH hSrcDS =
314 17 : GDALOpen(localOpts.osSrcFilename.c_str(), GA_ReadOnly);
315 17 : if (hSrcDS == nullptr)
316 1 : exit(2);
317 :
318 16 : GDALRasterBandH hBand = GDALGetRasterBand(hSrcDS, localOpts.nBandIn);
319 16 : if (hBand == nullptr)
320 : {
321 1 : CPLError(CE_Failure, CPLE_AppDefined,
322 : "Band %d does not exist on dataset.", localOpts.nBandIn);
323 1 : exit(2);
324 : }
325 :
326 15 : if (!argParser.is_used("-cc"))
327 12 : opts.curveCoeff = adjustCurveCoeff(opts.curveCoeff, hSrcDS);
328 :
329 : /* -------------------------------------------------------------------- */
330 : /* Invoke. */
331 : /* -------------------------------------------------------------------- */
332 :
333 : GDALDatasetH hDstDS;
334 :
335 : bool bSuccess;
336 15 : if (opts.outputMode == viewshed::OutputMode::Cumulative)
337 : {
338 1 : viewshed::Cumulative oViewshed(opts);
339 1 : bSuccess = oViewshed.run(localOpts.osSrcFilename,
340 1 : localOpts.bQuiet ? GDALDummyProgress
341 : : GDALTermProgress);
342 : }
343 : else
344 : {
345 28 : viewshed::Viewshed oViewshed(opts);
346 14 : bSuccess = oViewshed.run(hBand, localOpts.bQuiet ? GDALDummyProgress
347 : : GDALTermProgress);
348 14 : hDstDS = GDALDataset::FromHandle(oViewshed.output().release());
349 14 : GDALClose(hSrcDS);
350 14 : if (GDALClose(hDstDS) != CE_None)
351 0 : bSuccess = false;
352 : }
353 :
354 15 : GDALDestroyDriverManager();
355 15 : OGRCleanupAll();
356 :
357 15 : return bSuccess ? 0 : 1;
358 : }
359 :
360 0 : MAIN_END
|