plus.c 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. /**
  2. * \file plus.c
  3. *
  4. * \brief Vector library - update topo structure (lower level functions)
  5. *
  6. * Lower level functions for reading/writing/manipulating vectors.
  7. *
  8. * This program is free software under the GNU General Public License
  9. * (>=v2). Read the file COPYING that comes with GRASS for details.
  10. *
  11. * \author CERL (probably Dave Gerdes), Radim Blazek
  12. *
  13. * \date 2001-2006
  14. */
  15. #include <grass/config.h>
  16. #include <sys/types.h>
  17. #include <unistd.h>
  18. #include <stdlib.h>
  19. #include <stdio.h>
  20. #include <grass/gis.h>
  21. #include <grass/vector.h>
  22. #include <grass/glocale.h>
  23. /*!
  24. * \brief Initialize Plus_head structure
  25. *
  26. * \param[in,out] Plus pointer to Plus_head structure
  27. *
  28. * \return 1
  29. */
  30. int dig_init_plus(struct Plus_head *Plus)
  31. {
  32. G_debug(3, "dig_init_plus()");
  33. Plus->Version_Major = 0;
  34. Plus->Version_Minor = 0;
  35. Plus->Back_Major = 0;
  36. Plus->Back_Minor = 0;
  37. Plus->with_z = 0;
  38. Plus->off_t_size = 0;
  39. Plus->box.N = 0;
  40. Plus->box.S = 0;
  41. Plus->box.E = 0;
  42. Plus->box.W = 0;
  43. Plus->box.T = 0;
  44. Plus->box.B = 0;
  45. Plus->built = GV_BUILD_NONE;
  46. Plus->Node = NULL;
  47. Plus->Line = NULL;
  48. Plus->Area = NULL;
  49. Plus->Isle = NULL;
  50. Plus->n_nodes = 0;
  51. Plus->n_edges = 0;
  52. Plus->n_lines = 0;
  53. Plus->n_areas = 0;
  54. Plus->n_isles = 0;
  55. Plus->n_volumes = 0;
  56. Plus->n_holes = 0;
  57. Plus->alloc_nodes = 0;
  58. Plus->alloc_edges = 0;
  59. Plus->alloc_lines = 0;
  60. Plus->alloc_areas = 0;
  61. Plus->alloc_isles = 0;
  62. Plus->alloc_volumes = 0;
  63. Plus->alloc_holes = 0;
  64. Plus->n_plines = 0;
  65. Plus->n_llines = 0;
  66. Plus->n_blines = 0;
  67. Plus->n_clines = 0;
  68. Plus->n_flines = 0;
  69. Plus->n_klines = 0;
  70. Plus->Node_offset = 0L;
  71. Plus->Line_offset = 0L;
  72. Plus->Area_offset = 0L;
  73. Plus->Isle_offset = 0L;
  74. Plus->Volume_offset = 0L;
  75. Plus->Hole_offset = 0L;
  76. Plus->Node_spidx_offset = 0L;
  77. Plus->Line_spidx_offset = 0L;
  78. Plus->Area_spidx_offset = 0L;
  79. Plus->Isle_spidx_offset = 0L;
  80. Plus->Face_spidx_offset = 0L;
  81. Plus->Volume_spidx_offset = 0L;
  82. Plus->Hole_spidx_offset = 0L;
  83. /* dig_spidx_init(Plus); */
  84. dig_cidx_init(Plus);
  85. return 1;
  86. }
  87. /*!
  88. * \brief Free Plus->Node structure
  89. *
  90. * \param[in] Plus pointer to Plus_head structure
  91. */
  92. void dig_free_plus_nodes(struct Plus_head *Plus)
  93. {
  94. int i;
  95. struct P_node *Node;
  96. G_debug(2, "dig_free_plus_nodes()");
  97. /* Nodes */
  98. if (Plus->Node) { /* it may be that header only is loaded */
  99. for (i = 1; i <= Plus->n_nodes; i++) {
  100. Node = Plus->Node[i];
  101. if (Node == NULL)
  102. continue;
  103. dig_free_node(Node);
  104. }
  105. G_free(Plus->Node);
  106. }
  107. Plus->Node = NULL;
  108. Plus->n_nodes = 0;
  109. Plus->alloc_nodes = 0;
  110. }
  111. /*!
  112. * \brief Free Plus->Line structure
  113. *
  114. * \param[in] Plus pointer to Plus_head structure
  115. */
  116. void dig_free_plus_lines(struct Plus_head *Plus)
  117. {
  118. int i;
  119. struct P_line *Line;
  120. G_debug(2, "dig_free_plus_lines()");
  121. /* Lines */
  122. if (Plus->Line) { /* it may be that header only is loaded */
  123. for (i = 1; i <= Plus->n_lines; i++) {
  124. Line = Plus->Line[i];
  125. if (Line == NULL)
  126. continue;
  127. dig_free_line(Line);
  128. }
  129. G_free(Plus->Line);
  130. }
  131. Plus->Line = NULL;
  132. Plus->n_lines = 0;
  133. Plus->alloc_lines = 0;
  134. Plus->n_plines = 0;
  135. Plus->n_llines = 0;
  136. Plus->n_blines = 0;
  137. Plus->n_clines = 0;
  138. Plus->n_flines = 0;
  139. Plus->n_klines = 0;
  140. }
  141. /*!
  142. * \brief Free Plus->Area structure
  143. *
  144. * \param[in] Plus pointer to Plus_head structure
  145. */
  146. void dig_free_plus_areas(struct Plus_head *Plus)
  147. {
  148. int i;
  149. struct P_area *Area;
  150. G_debug(2, "dig_free_plus_areas()");
  151. /* Areas */
  152. if (Plus->Area) { /* it may be that header only is loaded */
  153. for (i = 1; i <= Plus->n_areas; i++) {
  154. Area = Plus->Area[i];
  155. if (Area == NULL)
  156. continue;
  157. dig_free_area(Area);
  158. }
  159. G_free(Plus->Area);
  160. }
  161. Plus->Area = NULL;
  162. Plus->n_areas = 0;
  163. Plus->alloc_areas = 0;
  164. }
  165. /*!
  166. * \brief Free Plus->Isle structure
  167. *
  168. * \param[in] Plus pointer to Plus_head structure
  169. */
  170. void dig_free_plus_isles(struct Plus_head *Plus)
  171. {
  172. int i;
  173. struct P_isle *Isle;
  174. G_debug(2, "dig_free_plus_isles()");
  175. /* Isles */
  176. if (Plus->Isle) { /* it may be that header only is loaded */
  177. for (i = 1; i <= Plus->n_isles; i++) {
  178. Isle = Plus->Isle[i];
  179. if (Isle == NULL)
  180. continue;
  181. dig_free_isle(Isle);
  182. }
  183. G_free(Plus->Isle);
  184. }
  185. Plus->Isle = NULL;
  186. Plus->n_isles = 0;
  187. Plus->alloc_isles = 0;
  188. }
  189. /*!
  190. * \brief Free Plus structure.
  191. *
  192. * Structure is not inited and dig_init_plus() should follow.
  193. *
  194. * \param[in] Plus pointer to Plus_head structure
  195. */
  196. void dig_free_plus(struct Plus_head *Plus)
  197. {
  198. G_debug(2, "dig_free_plus()");
  199. dig_free_plus_nodes(Plus);
  200. dig_free_plus_lines(Plus);
  201. dig_free_plus_areas(Plus);
  202. dig_free_plus_isles(Plus);
  203. dig_cidx_free(Plus);
  204. }
  205. /*!
  206. * \brief Reads topo file to topo structure.
  207. *
  208. * \param[in,out] Plus pointer to Plus_head structure
  209. * \param[in] plus topo file
  210. * \param[in] head_only read only head
  211. *
  212. * \return 1 on success
  213. * \return 0 on error
  214. */
  215. int dig_load_plus(struct Plus_head *Plus, struct gvfile * plus, int head_only)
  216. {
  217. int i;
  218. G_debug(1, "dig_load_plus()");
  219. /* TODO
  220. if (do_checks)
  221. dig_do_file_checks (map, map->plus_file, map->digit_file);
  222. */
  223. /* free and init old */
  224. dig_init_plus(Plus);
  225. /* Now let's begin reading the Plus file nodes, lines, areas and isles */
  226. if (dig_Rd_Plus_head(plus, Plus) == -1)
  227. return 0;
  228. if (head_only)
  229. return 1;
  230. dig_set_cur_port(&(Plus->port));
  231. /* Nodes */
  232. if (dig_fseek(plus, Plus->Node_offset, 0) == -1)
  233. G_fatal_error(_("Unable read topology for nodes"));
  234. dig_alloc_nodes(Plus, Plus->n_nodes);
  235. for (i = 1; i <= Plus->n_nodes; i++) {
  236. if (dig_Rd_P_node(Plus, i, plus) == -1)
  237. G_fatal_error(_("Unable to read topology for node %d"), i);
  238. }
  239. /* Lines */
  240. if (dig_fseek(plus, Plus->Line_offset, 0) == -1)
  241. G_fatal_error(_("Unable read topology for lines"));
  242. dig_alloc_lines(Plus, Plus->n_lines);
  243. for (i = 1; i <= Plus->n_lines; i++) {
  244. if (dig_Rd_P_line(Plus, i, plus) == -1)
  245. G_fatal_error(_("Unable to read topology for line %d"), i);
  246. }
  247. /* Areas */
  248. if (dig_fseek(plus, Plus->Area_offset, 0) == -1)
  249. G_fatal_error(_("Unable to read topo for areas"));
  250. dig_alloc_areas(Plus, Plus->n_areas);
  251. for (i = 1; i <= Plus->n_areas; i++) {
  252. if (dig_Rd_P_area(Plus, i, plus) == -1)
  253. G_fatal_error(_("Unable read topology for area %d"), i);
  254. }
  255. /* Isles */
  256. if (dig_fseek(plus, Plus->Isle_offset, 0) == -1)
  257. G_fatal_error(_("Unable to read topology for isles"));
  258. dig_alloc_isles(Plus, Plus->n_isles);
  259. for (i = 1; i <= Plus->n_isles; i++) {
  260. if (dig_Rd_P_isle(Plus, i, plus) == -1)
  261. G_fatal_error(_("Unable to read topology for isle %d"), i);
  262. }
  263. return (1);
  264. }
  265. /*!
  266. * \brief Writes topo structure to topo file
  267. *
  268. * \param[in,out] fp_plus topo file
  269. * \param[in] Plus pointer to Plus_head structure
  270. *
  271. * \return 0 on success
  272. * \return -1 on error
  273. */
  274. int dig_write_plus_file(struct gvfile * fp_plus, struct Plus_head *Plus)
  275. {
  276. dig_set_cur_port(&(Plus->port));
  277. dig_rewind(fp_plus);
  278. if (dig_Wr_Plus_head(fp_plus, Plus) < 0) {
  279. G_warning(_("Unable to write head to plus file"));
  280. return (-1);
  281. }
  282. if (dig_write_nodes(fp_plus, Plus) < 0) {
  283. G_warning(_("Unable to write nodes to plus file"));
  284. return (-1);
  285. }
  286. if (dig_write_lines(fp_plus, Plus) < 0) {
  287. G_warning(_("Unable to write lines to plus file"));
  288. return (-1);
  289. }
  290. if (dig_write_areas(fp_plus, Plus) < 0) {
  291. G_warning(_("Unable to write areas to plus file"));
  292. return (-1);
  293. }
  294. if (dig_write_isles(fp_plus, Plus) < 0) {
  295. G_warning(_("Unable to write isles to plus file"));
  296. return (-1);
  297. }
  298. dig_rewind(fp_plus);
  299. if (dig_Wr_Plus_head(fp_plus, Plus) < 0) {
  300. G_warning(_("Unable to write head to plus file"));
  301. return (-1);
  302. }
  303. dig_fflush(fp_plus);
  304. return (0);
  305. } /* write_plus_file() */
  306. /*!
  307. * \brief Writes topo structure (nodes) to topo file
  308. *
  309. * \param[in,out] fp_plus topo file
  310. * \param[in] Plus pointer to Plus_head structure
  311. *
  312. * \return 0 on success
  313. * \return -1 on error
  314. */
  315. int dig_write_nodes(struct gvfile * plus, struct Plus_head *Plus)
  316. {
  317. int i;
  318. Plus->Node_offset = dig_ftell(plus);
  319. for (i = 1; i <= Plus->n_nodes; i++) {
  320. if (dig_Wr_P_node(Plus, i, plus) < 0)
  321. return (-1);
  322. }
  323. return (0);
  324. } /* write_nodes() */
  325. /*!
  326. * \brief Writes topo structure (lines) to topo file
  327. *
  328. * \param[in,out] fp_plus topo file
  329. * \param[in] Plus pointer to Plus_head structure
  330. *
  331. * \return 0 on success
  332. * \return -1 on error
  333. */
  334. int dig_write_lines(struct gvfile * plus, struct Plus_head *Plus)
  335. {
  336. int i;
  337. Plus->Line_offset = dig_ftell(plus);
  338. for (i = 1; i <= Plus->n_lines; i++) {
  339. if (dig_Wr_P_line(Plus, i, plus) < 0)
  340. return (-1);
  341. }
  342. return (0);
  343. } /* write_line() */
  344. /*!
  345. * \brief Writes topo structure (areas) to topo file
  346. *
  347. * \param[in,out] fp_plus topo file
  348. * \param[in] Plus pointer to Plus_head structure
  349. *
  350. * \return 0 on success
  351. * \return -1 on error
  352. */
  353. int dig_write_areas(struct gvfile * plus, struct Plus_head *Plus)
  354. {
  355. int i;
  356. Plus->Area_offset = dig_ftell(plus);
  357. for (i = 1; i <= Plus->n_areas; i++) {
  358. if (dig_Wr_P_area(Plus, i, plus) < 0)
  359. return (-1);
  360. }
  361. return (0);
  362. } /* write_areas() */
  363. /*!
  364. * \brief Writes topo structure (isles) to topo file
  365. *
  366. * \param[in,out] fp_plus topo file
  367. * \param[in] Plus pointer to Plus_head structure
  368. *
  369. * \return 0 on success
  370. * \return -1 on error
  371. */
  372. int dig_write_isles(struct gvfile * plus, struct Plus_head *Plus)
  373. {
  374. int i;
  375. Plus->Isle_offset = dig_ftell(plus);
  376. for (i = 1; i <= Plus->n_isles; i++) {
  377. if (dig_Wr_P_isle(Plus, i, plus) < 0)
  378. return (-1);
  379. }
  380. return (0);
  381. } /* write_isles() */