123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620 |
- /****************************************************************************
- *
- * MODULE: r.spread
- * AUTHOR(S): Jianping Xu, 1995 (original contributor)
- * Center for Remote Sensing and Spatial Analysis (CRSSA)
- * Rutgers University.
- * Andreas.Lange <andreas.lange rhein-main.de>, Eric G. Miller <egm2 jps.net>,
- * Markus Neteler <neteler itc.it>, Roberto Flor <flor itc.it>,
- * Brad Douglas <rez touchofmadness.com>,
- * Glynn Clements <glynn gclements.plus.com>, Jachym Cepicky <jachym les-ejk.cz>
- * PURPOSE:
- * This is the main program for simulating elliptical spread.
- *
- * It
- * 1) determines the earliest time a phenomenon REACHES to a
- * map cell, NOT the time that cell is EXHAUSTED.
- * 3) If a cell is spread barrier, a no-data value is assigned
- * to it.
- * COPYRIGHT: (C) 2000-2006 by the GRASS Development Team
- *
- * This program is free software under the GNU General Public
- * License (>=v2). Read the file COPYING that comes with GRASS
- * for details.
- *
- *****************************************************************************/
- #include <stdio.h>
- #include <stdlib.h>
- #include <math.h>
- #include <sys/types.h>
- #include <unistd.h>
- #include <grass/gis.h>
- #include <grass/raster.h>
- #include <grass/glocale.h>
- #include "cmd_line.h"
- #include "costHa.h"
- #include "cell_ptrHa.h"
- #include "local_proto.h"
- #define DATA(map, r, c) (map)[(r) * ncols + (c)]
- CELL *cell;
- CELL *x_cell;
- CELL *y_cell;
- CELL *map_max;
- CELL *map_dir;
- CELL *map_base;
- CELL *map_spotdist;
- CELL *map_velocity;
- CELL *map_mois;
- float *map_out;
- CELL *map_x_out;
- CELL *map_y_out;
- CELL *map_visit;
- char buf[400];
- float zero = 0.0;
- float neg = -2.0;
- int BARRIER = 0;
- int max_fd, dir_fd, base_fd, start_fd;
- int spotdist_fd, velocity_fd, mois_fd;
- int cum_fd, x_fd, y_fd;
- int nrows, ncols;
- long heap_len;
- struct Cell_head window;
- struct costHa *heap;
- int main(int argc, char *argv[])
- {
- int col, row;
- /* to menage start (source) raster map */
- struct Range start_range;
- CELL start_range_min, start_range_max;
- int start_is_time; /* 0 or 1 */
- struct
- {
- struct Option *max, *dir, *base, *start,
- *spotdist, *velocity, *mois,
- *least, *comp_dens, *init_time,
- *time_lag, *backdrop, *out, *x_out, *y_out;
- } parm;
- struct
- {
- /* please, remove display before GRASS 7 released */
- struct Flag *display, *spotting, *start_is_time;
- } flag;
- struct GModule *module;
- /* initialize access to database and create temporary files */
- G_gisinit(argv[0]);
- /* Set description */
- module = G_define_module();
- G_add_keyword(_("raster"));
- G_add_keyword(_("fire"));
- G_add_keyword(_("spread"));
- G_add_keyword(_("hazard"));
- G_add_keyword(_("model"));
- module->label =
- _("Simulates elliptically anisotropic spread.");
- module->description =
- _("Generates a raster map of the cumulative time of spread, "
- "given raster maps containing the rates of spread (ROS), "
- "the ROS directions and the spread origins. "
- "It optionally produces raster maps to contain backlink UTM "
- "coordinates for tracing spread paths. "
- "Usable for fire spread simulations.");
- parm.base = G_define_option();
- parm.base->key = "base_ros";
- parm.base->type = TYPE_STRING;
- parm.base->required = YES;
- parm.base->gisprompt = "old,cell,raster";
- parm.base->guisection = _("Input");
- parm.base->label =
- _("Raster map containing base ROS (cm/min)");
- parm.base->description =
- _("Name of an existing raster map layer in the user's "
- "current mapset search path containing the ROS values in the directions "
- "perpendicular to maximum ROSes' (cm/minute). These ROSes are also the ones "
- "without the effect of directional factors.");
-
- parm.max = G_define_option();
- parm.max->key = "max_ros";
- parm.max->type = TYPE_STRING;
- parm.max->required = YES;
- parm.max->gisprompt = "old,cell,raster";
- parm.max->guisection = _("Input");
- parm.max->label =
- _("Raster map containing maximal ROS (cm/min)");
- parm.max->description =
- _("Name of an existing raster map layer in the user's current "
- "mapset search path containing the maximum ROS values (cm/minute).");
- parm.dir = G_define_option();
- parm.dir->key = "direction_ros";
- parm.dir->type = TYPE_STRING;
- parm.dir->required = YES;
- parm.dir->gisprompt = "old,cell,raster";
- parm.dir->guisection = _("Input");
- parm.dir->label =
- _("Raster map containing directions of maximal ROS (degree)");
- parm.dir->description =
- _("Name of an existing raster map layer in the user's "
- "current mapset search path containing directions of the maximum ROSes, "
- "clockwise from north (degree)."); /* TODO: clockwise from north? see r.ros */
- parm.start = G_define_option();
- parm.start->key = "start";
- parm.start->type = TYPE_STRING;
- parm.start->required = YES;
- parm.start->gisprompt = "old,cell,raster";
- parm.start->guisection = _("Input");
- parm.start->label =
- _("Raster map containing starting sources");
- parm.start->description =
- _("Name of an existing raster map layer in the "
- "user's current mapset search path containing starting locations of the "
- "spread phenomenon. Any positive integers in this map are recognized as "
- "starting sources (seeds).");
- parm.spotdist = G_define_option();
- parm.spotdist->key = "spotting_distance";
- parm.spotdist->type = TYPE_STRING;
- parm.spotdist->gisprompt = "old,cell,raster";
- parm.spotdist->guisection = _("Input");
- parm.spotdist->label =
- _("Raster map containing maximal spotting distance (m, required with -s)");
- parm.spotdist->description =
- _("Name of an existing raster map layer in "
- "the user's current mapset search path containing the maximum potential "
- "spotting distances (meters).");
- parm.velocity = G_define_option();
- parm.velocity->key = "wind_speed";
- parm.velocity->type = TYPE_STRING;
- parm.velocity->gisprompt = "old,cell,raster";
- parm.velocity->guisection = _("Input");
- parm.velocity->label =
- _("Raster map containing midflame wind speed (ft/min, required with -s)");
- parm.velocity->description =
- _("Name of an existing raster map layer in the "
- "user's current mapset search path containing wind velocities at half of "
- "the average flame height (feet/minute).");
- parm.mois = G_define_option();
- parm.mois->key = "fuel_moisture";
- parm.mois->type = TYPE_STRING;
- parm.mois->gisprompt = "old,cell,raster";
- parm.mois->guisection = _("Input");
- parm.mois->label =
- _("Raster map containing fine fuel moisture of the cell receiving a spotting firebrand (%, required with -s)");
- parm.mois->description =
- _("Name of an existing raster map layer in the "
- "user's current mapset search path containing the 1-hour (<.25\") fuel "
- "moisture (percentage content multiplied by 100).");
- parm.least = G_define_option();
- parm.least->key = "least_size";
- parm.least->type = TYPE_STRING;
- parm.least->key_desc = "odd int";
- parm.least->options = "3,5,7,9,11,13,15";
- parm.least->label =
- _("Basic sampling window size needed to meet certain accuracy (3)"); /* TODO: what is 3 here? default? */
- parm.least->description =
- _("An odd integer ranging 3 - 15 indicating "
- "the basic sampling window size within which all cells will be considered "
- "to see whether they will be reached by the current spread cell. The default "
- "number is 3 which means a 3x3 window.");
- parm.comp_dens = G_define_option();
- parm.comp_dens->key = "comp_dens";
- parm.comp_dens->type = TYPE_STRING;
- parm.comp_dens->key_desc = "decimal";
- parm.comp_dens->label =
- _("Sampling density for additional computing (range: 0.0 - 1.0 (0.5))"); /* TODO: again, what is 0.5?, TODO: range not set */
- parm.comp_dens->description =
- _("A decimal number ranging 0.0 - 1.0 indicating "
- "additional sampling cells will be considered to see whether they will be "
- "reached by the current spread cell. The closer to 1.0 the decimal number "
- "is, the longer the program will run and the higher the simulation accuracy "
- "will be. The default number is 0.5.");
- parm.init_time = G_define_option();
- parm.init_time->key = "init_time";
- parm.init_time->type = TYPE_STRING;
- parm.init_time->key_desc = "int (>= 0)"; /* TODO: move to ->options */
- parm.init_time->answer = "0";
- parm.init_time->label =
- _("Initial time for current simulation (0) (min)");
- parm.init_time->description =
- _("A non-negative number specifying the initial "
- "time for the current spread simulation (minutes). This is useful when multiple "
- "phase simulation is conducted. The default time is 0.");
- parm.time_lag = G_define_option();
- parm.time_lag->key = "lag";
- parm.time_lag->type = TYPE_STRING;
- parm.time_lag->key_desc = "int (>= 0)"; /* TODO: move to ->options */
- parm.time_lag->label =
- _("Simulating time duration LAG (fill the region) (min)"); /* TODO: what does this mean? */
- parm.time_lag->description =
- _("A non-negative integer specifying the simulating "
- "duration time lag (minutes). The default is infinite, but the program will "
- "terminate when the current geographic region/mask has been filled. It also "
- "controls the computational time, the shorter the time lag, the faster the "
- "program will run.");
- /* TODO: what's this? probably display, so remove */
- parm.backdrop = G_define_option();
- parm.backdrop->key = "backdrop";
- parm.backdrop->type = TYPE_STRING;
- parm.backdrop->gisprompt = "old,cell,raster";
- parm.backdrop->label =
- _("Name of raster map as a display backdrop");
- parm.backdrop->description =
- _("Name of an existing raster map layer in the "
- "user's current mapset search path to be used as the background on which "
- "the \"live\" movement will be shown.");
- parm.out = G_define_option();
- parm.out->key = "output";
- parm.out->type = TYPE_STRING;
- parm.out->required = YES;
- parm.out->gisprompt = "new,cell,raster";
- parm.out->guisection = _("Output");
- parm.out->label =
- _("Raster map to contain output spread time (min)");
- parm.out->description =
- _("Name of the new raster map layer to contain "
- "the results of the cumulative spread time needed for a phenomenon to reach "
- "each cell from the starting sources (minutes).");
- parm.x_out = G_define_option();
- parm.x_out->key = "x_output";
- parm.x_out->type = TYPE_STRING;
- parm.x_out->gisprompt = "new,cell,raster";
- parm.x_out->guisection = _("Output");
- parm.x_out->label =
- _("Name of raster map to contain X back coordinates");
- parm.x_out->description =
- _("Name of the new raster map layer to contain "
- "the results of backlink information in UTM easting coordinates for each "
- "cell.");
- parm.y_out = G_define_option();
- parm.y_out->key = "y_output";
- parm.y_out->type = TYPE_STRING;
- parm.y_out->gisprompt = "new,cell,raster";
- parm.y_out->guisection = _("Output");
- parm.y_out->label =
- _("Name of raster map to contain Y back coordinates");
- parm.y_out->description =
- _("Name of the new raster map layer to contain "
- "the results of backlink information in UTM northing coordinates for each "
- "cell.");
- #if 0
- flag.display = G_define_flag();
- flag.display->key = 'd';
- flag.display->label = _("DISPLAY 'live' spread process on screen");
- flag.display->description =
- _("Display the 'live' simulation on screen. A graphics window "
- "must be opened and selected before using this option.");
- #endif
- flag.spotting = G_define_flag();
- flag.spotting->key = 's';
- flag.spotting->description = _("Consider spotting effect (for wildfires)");
- flag.start_is_time = G_define_flag();
- flag.start_is_time->key = 'i';
- flag.start_is_time->label = _("Use start raster map values in"
- " output spread time raster map");
- flag.start_is_time->description = _("Designed to be used with output"
- " of previous run of r.spread when computing spread iteratively."
- " The values in start raster map are considered as time."
- " Allowed values in raster map are from zero"
- " to the value of init_time option."
- " If not enabled, init_time is used in the area of start raster map");
- /* Parse command line */
- if (G_parser(argc, argv))
- exit(EXIT_FAILURE);
- /* FIXME - allow seed to be specified for repeatability */
- G_srand48_auto();
- display = NULL;
- spotting = flag.spotting->answer;
- max_layer = parm.max->answer;
- dir_layer = parm.dir->answer;
- base_layer = parm.base->answer;
- start_layer = parm.start->answer;
- backdrop_layer = parm.backdrop->answer;
- out_layer = parm.out->answer;
- if (parm.x_out->answer) {
- x_out = 1;
- x_out_layer = parm.x_out->answer;
- }
- if (parm.y_out->answer) {
- y_out = 1;
- y_out_layer = parm.y_out->answer;
- }
- if (spotting) {
- if (!
- (parm.spotdist->answer && parm.velocity->answer &&
- parm.mois->answer)) {
- G_warning
- ("SPOTTING DISTANCE, fuel MOISTURE, or wind VELOCITY map not given w/ -s");
- G_usage();
- exit(EXIT_FAILURE);
- }
- else {
- spotdist_layer = parm.spotdist->answer;
- velocity_layer = parm.velocity->answer;
- mois_layer = parm.mois->answer;
- }
- }
- /*Check the given the least sampling size, assign the default if needed */
- if (parm.least->answer)
- least = atoi(parm.least->answer);
- else
- least = 3;
- /*Check the given computing density, assign the default if needed */
- if (parm.comp_dens->answer) {
- comp_dens = atof(parm.comp_dens->answer);
- if (comp_dens < 0.0 || comp_dens > 1.0) {
- G_warning("Illegal computing density <%s>",
- parm.comp_dens->answer);
- G_usage();
- exit(EXIT_FAILURE);
- }
- }
- else {
- comp_dens = 0.5;
- }
- /*Check the given initial time and simulation time lag, assign the default if needed */
- init_time = atoi(parm.init_time->answer);
- if (init_time < 0) {
- G_warning("Illegal initial time <%s>", parm.init_time->answer);
- G_usage();
- exit(EXIT_FAILURE);
- }
- if (parm.time_lag->answer) {
- time_lag = atoi(parm.time_lag->answer);
- if (time_lag < 0) {
- G_warning("Illegal simulating time lag <%s>",
- parm.time_lag->answer);
- G_usage();
- exit(EXIT_FAILURE);
- }
- }
- else {
- time_lag = 99999;
- }
- /* Get database window parameters */
- G_get_window(&window);
- /* find number of rows and columns in window */
- nrows = Rast_window_rows();
- ncols = Rast_window_cols();
- /*transfor measurement unit from meters to centimeters due to ROS unit
- *if the input ROSs are in m/min units, cancell the following*/
- window.ns_res = 100 * window.ns_res;
- window.ew_res = 100 * window.ew_res;
- /* Initialize display screens */
- #if 0
- if (display)
- display_init();
- #endif
- /* Check if input layers exists in data base */
- if (G_find_raster2(max_layer, "") == NULL)
- G_fatal_error("Raster map <%s> not found", max_layer);
- if (G_find_raster2(dir_layer, "") == NULL)
- G_fatal_error(_("Raster map <%s> not found"), dir_layer);
- if (G_find_raster2(base_layer, "") == NULL)
- G_fatal_error(_("Raster map <%s> not found"), base_layer);
- if (G_find_raster2(start_layer, "") == NULL)
- G_fatal_error(_("Raster map <%s> not found"), start_layer);
- if (spotting) {
- if (G_find_raster2(spotdist_layer, "") == NULL)
- G_fatal_error(_("Raster map <%s> not found"), spotdist_layer);
- if (G_find_raster2(velocity_layer, "") == NULL)
- G_fatal_error(_("Raster map <%s> not found"), velocity_layer);
- if (G_find_raster2(mois_layer, "") == NULL)
- G_fatal_error(_("Raster map <%s> not found"), mois_layer);
- }
- /* Open input cell layers for reading */
- max_fd = Rast_open_old(max_layer, G_find_raster2(max_layer, ""));
- dir_fd = Rast_open_old(dir_layer, G_find_raster2(dir_layer, ""));
- base_fd = Rast_open_old(base_layer, G_find_raster2(base_layer, ""));
- if (spotting) {
- spotdist_fd =
- Rast_open_old(spotdist_layer, G_find_raster2(spotdist_layer, ""));
- velocity_fd =
- Rast_open_old(velocity_layer, G_find_raster2(velocity_layer, ""));
- mois_fd = Rast_open_old(mois_layer, G_find_raster2(mois_layer, ""));
- }
- /* Allocate memories for a row */
- cell = Rast_allocate_c_buf();
- if (x_out)
- x_cell = Rast_allocate_c_buf();
- if (y_out)
- y_cell = Rast_allocate_c_buf();
- /* Allocate memories for a map */
- map_max = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- map_dir = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- map_base = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- map_visit = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- map_out = (float *)G_calloc(nrows * ncols + 1, sizeof(float));
- if (spotting) {
- map_spotdist = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- map_velocity = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- map_mois = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- }
- if (x_out)
- map_x_out = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- if (y_out)
- map_y_out = (CELL *) G_calloc(nrows * ncols + 1, sizeof(CELL));
- /* Write the input layers in the map "arrays" */
- G_message(_("Reading inputs..."));
- for (row = 0; row < nrows; row++) {
- G_percent(row, nrows, 2);
- Rast_get_c_row(max_fd, cell, row);
- for (col = 0; col < ncols; col++)
- DATA(map_max, row, col) = cell[col];
- Rast_get_c_row(dir_fd, cell, row);
- for (col = 0; col < ncols; col++)
- DATA(map_dir, row, col) = cell[col];
- Rast_get_c_row(base_fd, cell, row);
- for (col = 0; col < ncols; col++)
- DATA(map_base, row, col) = cell[col];
- if (spotting) {
- Rast_get_c_row(spotdist_fd, cell, row);
- for (col = 0; col < ncols; col++)
- DATA(map_spotdist, row, col) = cell[col];
- Rast_get_c_row(velocity_fd, cell, row);
- for (col = 0; col < ncols; col++)
- DATA(map_velocity, row, col) = cell[col];
- Rast_get_c_row(mois_fd, cell, row);
- for (col = 0; col < ncols; col++)
- DATA(map_mois, row, col) = cell[col];
- }
- }
- G_percent(row, nrows, 2);
- /* Scan the START layer searching for starting points.
- * Create an array of starting points (min_heap) ordered by costs.
- */
- start_fd = Rast_open_old(start_layer, G_find_raster2(start_layer, ""));
- Rast_read_range(start_layer, G_find_file("cell", start_layer, ""),
- &start_range);
- Rast_get_range_min_max(&start_range, &start_range_min, &start_range_max);
- start_is_time = flag.start_is_time->answer;
- /* values higher than init_time are unexpected and may cause segfaults */
- if (start_is_time && start_range_max > init_time)
- G_fatal_error(_("Maximum of start raster map is grater than init_time"
- " (%d > %d)"), start_range_max, init_time);
- /* values lower then zero does not make sense for time */
- if (start_is_time && start_range_min < 0)
- G_fatal_error(_("Minimum of start raster map is less than zero"
- " (%d < 0)"), start_range_min);
- /* Initialize the heap */
- heap =
- (struct costHa *)G_calloc(nrows * ncols + 1, sizeof(struct costHa));
- heap_len = 0;
- G_message(_("Reading %s..."), start_layer);
- G_debug(1, "Collecting origins...");
- collect_ori(start_fd, start_is_time);
- G_debug(1, "Done");
- /* Major computation of spread time */
- G_debug(1, "Spreading...");
- spread();
- G_debug(1, "Done");
- /* Open cumulative cost layer (and x, y direction layers) for writing */
- cum_fd = Rast_open_c_new(out_layer);
- if (x_out)
- x_fd = Rast_open_c_new(x_out_layer);
- if (y_out)
- y_fd = Rast_open_c_new(y_out_layer);
- /* prepare output -- adjust from cm to m */
- window.ew_res = window.ew_res / 100;
- window.ns_res = window.ns_res / 100;
- /* copy maps in ram to output maps */
- ram2out();
- G_free(map_max);
- G_free(map_dir);
- G_free(map_base);
- G_free(map_out);
- G_free(map_visit);
- if (x_out)
- G_free(map_x_out);
- if (y_out)
- G_free(map_y_out);
- if (spotting) {
- G_free(map_spotdist);
- G_free(map_mois);
- G_free(map_velocity);
- }
- Rast_close(max_fd);
- Rast_close(dir_fd);
- Rast_close(base_fd);
- Rast_close(start_fd);
- Rast_close(cum_fd);
- if (x_out)
- Rast_close(x_fd);
- if (y_out)
- Rast_close(y_fd);
- if (spotting) {
- Rast_close(spotdist_fd);
- Rast_close(velocity_fd);
- Rast_close(mois_fd);
- }
- /* close graphics */
- #if 0
- if (display)
- display_close();
- #endif
- exit(EXIT_SUCCESS);
- }
|