2 * Hic Est Leo -- Reading ESRI Shape Files
4 * (c) 2014--2015 Martin Mares <mj@ucw.cz>
6 * FIXME: Currently, this parser handles only the subset
7 * of shape file syntax which is used by gdal_contours.
18 #include <ucw/fastbuf.h>
20 #include <ucw/unaligned.h>
25 static osm_id_t shp_id_counter;
27 static double shp_get_double(byte *p)
29 // FIXME: This is non-portable!
35 static void shp_record_polyline(byte *buf, u32 len)
38 die("Polyline record too short: %u bytes", len);
40 u32 num_parts = get_u32_le(buf+36);
41 u32 num_points = get_u32_le(buf+40);
42 DBG("%u parts, %u points", num_parts, num_points);
44 die("Polylines with multiple parts are not supported yet");
46 if (len < 44 + 4*num_parts + 16*num_points)
47 die("Polyline record too short for %u parts and %u points: %u bytes", num_parts, num_points, len);
49 struct osm_way *w = osm_way_new(shp_id_counter++);
50 osm_obj_add_tag(&w->o, "shape", "contour");
52 byte *p = buf + 44 + 4*num_parts;
53 for (uint i=0; i < num_points; i++)
55 double x = shp_get_double(p);
56 double y = shp_get_double(p+8);
57 DBG(">>> x=%.10g y=%.10g", x, y);
58 struct osm_node *n = osm_node_new(shp_id_counter++);
61 osm_way_add_node(w, n);
66 void shp_parse(const char *name)
68 msg(L_INFO, "Loading shape file %s", name);
69 struct fastbuf *fb = bopen_file(name, O_RDONLY, NULL);
75 uns len = bread(fb, buf, 100);
76 if (len != 100 || get_u32_be(buf) != 9994)
77 die("Invalid shape file header");
79 u32 version = get_u32_le(buf+28);
81 die("Unknown shape file version %u", version);
83 u32 type = get_u32_le(buf+32);
85 die("Unsupported shape file type %u", type);
91 uns len = bread(fb, buf, 8);
95 die("Truncated record header at %ju", (uintmax_t) pos);
96 u32 recno = get_u32_be(buf);
97 u32 reclen = get_u32_be(buf+4) * 2;
98 if (recno != ++last_recno)
99 die("Unexpected record #%u (should be #%u) at %ju", recno, last_recno, (uintmax_t) pos);
100 if (reclen > GARY_SIZE(buf))
101 GARY_RESIZE(buf, reclen);
102 len = bread(fb, buf, reclen);
104 die("Truncated record: %u < %u at %ju", len, reclen, (uintmax_t) pos);
106 die("Record too short: %u bytes at %ju", reclen, (uintmax_t) pos);
108 u32 rectype = get_u32_le(buf);
109 DBG("@%ju: recno=%u len=%u type=%u", (uintmax_t) pos, recno, reclen, rectype);
111 shp_record_polyline(buf, len);