head.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. /*
  2. ****************************************************************************
  3. *
  4. * MODULE: Vector library
  5. *
  6. * AUTHOR(S): Original author CERL, probably Dave Gerdes.
  7. * Update to GRASS 5.7 Radim Blazek.
  8. *
  9. * PURPOSE: Lower level functions for reading/writing/manipulating vectors.
  10. *
  11. * COPYRIGHT: (C) 2001 by the GRASS Development Team
  12. *
  13. * This program is free software under the GNU General Public
  14. * License (>=v2). Read the file COPYING that comes with GRASS
  15. * for details.
  16. *
  17. *****************************************************************************/
  18. #include <grass/config.h>
  19. #include <sys/types.h>
  20. #include <string.h>
  21. #include <grass/gis.h>
  22. #include <grass/vector.h>
  23. int dig__write_head(struct Map_info *Map)
  24. {
  25. unsigned char buf[10];
  26. long length = GV_COOR_HEAD_SIZE;
  27. G_debug(1, "dig__write_head()");
  28. dig_set_cur_port(&(Map->head.port));
  29. dig_fseek(&(Map->dig_fp), 0L, 0);
  30. /* bytes 1 - 5 */
  31. buf[0] = Map->head.Version_Major;
  32. buf[1] = Map->head.Version_Minor;
  33. buf[2] = Map->head.Back_Major;
  34. buf[3] = Map->head.Back_Minor;
  35. buf[4] = Map->head.port.byte_order;
  36. if (0 >= dig__fwrite_port_C(buf, 5, &(Map->dig_fp)))
  37. return (0);
  38. /* increase header size for new vectors, already set in V1_open_new_nat() */
  39. length = Map->head.head_size;
  40. /* bytes 6 - 9 : header size */
  41. if (0 >= dig__fwrite_port_L(&length, 1, &(Map->dig_fp)))
  42. return (0);
  43. /* byte 10 : dimension 2D or 3D */
  44. buf[0] = Map->head.with_z;
  45. if (0 >= dig__fwrite_port_C(buf, 1, &(Map->dig_fp)))
  46. return (0);
  47. /* bytes 11 - 18 : size of coordinate file */
  48. G_debug(1, "write coor size (%"PRI_OFF_T") to head", Map->head.size);
  49. if (Map->head.head_size >= GV_COOR_HEAD_SIZE + 4) {
  50. if (Map->head.size > PORT_LONG_MAX) {
  51. /* can only happen when sizeof(off_t) == 8 */
  52. if (0 >= dig__fwrite_port_O(&(Map->head.size), 1, &(Map->dig_fp), sizeof(off_t)))
  53. return (0);
  54. }
  55. else {
  56. /* write twice to fill the space and set offset (account for sizeof(off_t) == 4) */
  57. if (0 >= dig__fwrite_port_O(&(Map->head.size), 1, &(Map->dig_fp), 4))
  58. return (0);
  59. if (0 >= dig__fwrite_port_O(&(Map->head.size), 1, &(Map->dig_fp), 4))
  60. return (0);
  61. }
  62. }
  63. else {
  64. /* old vector with shorter coor head size got modified */
  65. /* bytes 11 - 14 : size of coordinate file */
  66. if (0 >= dig__fwrite_port_O(&(Map->head.size), 1, &(Map->dig_fp), 4))
  67. return (0);
  68. }
  69. G_debug(2, "coor body offset %"PRI_OFF_T, dig_ftell(&(Map->dig_fp)));
  70. return (1);
  71. }
  72. int dig__read_head(struct Map_info *Map)
  73. {
  74. unsigned char buf[10];
  75. struct Port_info port;
  76. dig_fseek(&(Map->dig_fp), 0L, 0);
  77. /* bytes 1 - 5 */
  78. if (0 >= dig__fread_port_C(buf, 5, &(Map->dig_fp)))
  79. return (0);
  80. Map->head.Version_Major = buf[0];
  81. Map->head.Version_Minor = buf[1];
  82. Map->head.Back_Major = buf[2];
  83. Map->head.Back_Minor = buf[3];
  84. Map->head.port.byte_order = buf[4];
  85. G_debug(2,
  86. "Coor header: file version %d.%d , supported from GRASS version %d.%d",
  87. Map->head.Version_Major, Map->head.Version_Minor,
  88. Map->head.Back_Major, Map->head.Back_Minor);
  89. G_debug(2, " byte order %d", Map->head.port.byte_order);
  90. /* check version numbers */
  91. if (Map->head.Version_Major > GV_COOR_VER_MAJOR ||
  92. Map->head.Version_Minor > GV_COOR_VER_MINOR) {
  93. /* The file was created by GRASS library with higher version than this one */
  94. if (Map->head.Back_Major > GV_COOR_VER_MAJOR ||
  95. Map->head.Back_Minor > GV_COOR_VER_MINOR) {
  96. /* This version of GRASS lib is lower than the oldest which can read this format */
  97. G_fatal_error
  98. ("Vector 'coor' format version %d.%d is not supported by this version of GRASS. "
  99. "Update your GRASS.", Map->head.Version_Major,
  100. Map->head.Version_Minor);
  101. return (-1);
  102. }
  103. G_warning
  104. ("Your GRASS version does not fully support vector format %d.%d."
  105. " Consider to upgrade GRASS.", Map->head.Version_Major,
  106. Map->head.Version_Minor);
  107. }
  108. dig_init_portable(&port, Map->head.port.byte_order);
  109. dig_set_cur_port(&port);
  110. /* bytes 6 - 9 : header size */
  111. if (0 >= dig__fread_port_L(&(Map->head.head_size), 1, &(Map->dig_fp)))
  112. return (0);
  113. G_debug(2, " header size %ld", Map->head.head_size);
  114. /* byte 10 : dimension 2D or 3D */
  115. if (0 >= dig__fread_port_C(buf, 1, &(Map->dig_fp)))
  116. return (0);
  117. Map->head.with_z = buf[0];
  118. G_debug(2, " with_z %d", Map->head.with_z);
  119. /* Map->head.size holds stats value */
  120. if (Map->head.size > PORT_LONG_MAX && Map->head.head_size >= GV_COOR_HEAD_SIZE + 4) {
  121. /* bytes 11 - 18 : size of coordinate file */
  122. if (0 >= dig__fread_port_O(&(Map->head.size), 1, &(Map->dig_fp), sizeof(off_t)))
  123. return (0);
  124. }
  125. else {
  126. /* bytes 11 - 14 : size of coordinate file */
  127. if (0 >= dig__fread_port_O(&(Map->head.size), 1, &(Map->dig_fp), 4))
  128. return (0);
  129. }
  130. G_debug(2, " coor size %"PRI_OFF_T, Map->head.size);
  131. /* Go to end of header, file may be written by new version of GRASS with longer header */
  132. dig_fseek(&(Map->dig_fp), Map->head.head_size, SEEK_SET);
  133. return (1);
  134. }