#include #include #include "igcparser.h" static int str2int(const char *str, size_t len) { int res = 0; for (const char *sp = str; sp < str + len; sp++) { if (::isdigit(*sp)) res = res * 10 + *sp - '0'; else return -1; } return res; } static bool readLat(const char *data, qreal &lat) { int d = str2int(data, 2); int mi = str2int(data + 2, 2); int mf = str2int(data + 4, 3); if (d < 0 || mi < 0 || mf < 0) return false; if (!(data[7] == 'N' || data[7] == 'S')) return false; lat = d + (((qreal)mi + (qreal)mf/1000) / 60); if (lat > 90) return false; if (data[7] == 'S') lat = -lat; return true; } static bool readLon(const char *data, qreal &lon) { int d = str2int(data, 3); int mi = str2int(data + 3, 2); int mf = str2int(data + 5, 3); if (d < 0 || mi < 0 || mf < 0) return false; if (!(data[8] == 'E' || data[8] == 'W')) return false; lon = d + (((qreal)mi + (qreal)mf/1000) / 60); if (lon > 180) return false; if (data[8] == 'W') lon = -lon; return true; } static bool readAltitude(const char *data, qreal &ele) { int p; if (!(data[0] == 'A' || data[0] == 'V')) return false; if (data[1] == '-') p = str2int(data + 2, 4); else p = str2int(data + 1, 5); int g = str2int(data + 6, 5); if (p < 0 || g < 0) return false; if (data[0] == 'A') ele = (qreal)g; else ele = NAN; return true; } static bool readTimestamp(const char *data, QTime &time) { int h = str2int(data, 2); int m = str2int(data + 2, 2); int s = str2int(data + 4, 2); if (h < 0 || m < 0 || s < 0) return false; time = QTime(h, m, s); return true; } static bool readARecord(const char *line, qint64 len) { if (len < 9 || line[0] != 'A') return false; for (int i = 1; i < 7; i++) if (!::isprint(line[i])) return false; return true; } bool IGCParser::readHRecord(const char *line, qint64 len) { if (len < 10 || ::strncmp(line, "HFDTE", 5)) return true; int d = str2int(line + 5, 2); int m = str2int(line + 7, 2); int y = str2int(line + 9, 2); if (y < 0 || m < 0 || d < 0) { _errorString = "Invalid date header format"; return false; } _date = QDate(2000 + y, m, d); if (!_date.isValid()) { _errorString = "Invalid date"; return false; } return true; } bool IGCParser::readBRecord(const char *line, qint64 len) { qreal lat, lon, ele; QTime time; if (len < 35) return false; if (!readTimestamp(line + 1, time)) { _errorString = "Invalid timestamp"; return false; } if (!readLat(line + 7, lat)) { _errorString = "Invalid latitude"; return false; } if (!readLon(line + 15, lon)) { _errorString = "Invalid longitude"; return false; } if (!readAltitude(line + 24, ele)) { _errorString = "Invalid altitude"; return false; } if (time < _time) _date.addDays(1); _time = time; Trackpoint t(Coordinates(lon, lat)); t.setTimestamp(QDateTime(_date, _time, Qt::UTC)); t.setElevation(ele); _tracks.last().append(t); return true; } bool IGCParser::readCRecord(const char *line, qint64 len) { qreal lat, lon; if (len < 18) return false; if (!readLat(line + 1, lat)) { _errorString = "Invalid latitude"; return false; } if (!readLon(line + 9, lon)) { _errorString = "Invalid longitude"; return false; } if (!(lat == 0 && lon == 0)) { QByteArray ba(line + 18, len - 19); Waypoint w(Coordinates(lon, lat)); w.setName(QString(ba.trimmed())); _routes.last().append(w); } return true; } bool IGCParser::loadFile(QFile *file) { qint64 len; char line[76 + 2 + 1 + 1]; bool route = false, track = false; _errorLine = 1; _errorString.clear(); while ((len = file->readLine(line, sizeof(line))) > 0) { if (len < 0) { _errorString = "I/O error"; return false; } else if (len > (qint64)sizeof(line) - 1) { _errorString = "Line limit exceeded"; return false; } if (_errorLine == 1) { if (!readARecord(line, len)) { _errorString = "Invalid/missing A record"; return false; } } else { if (line[0] == 'H') { if (!readHRecord(line, len)) return false; } else if (line[0] == 'C') { if (route) { if (!readCRecord(line, len)) return false; } else { route = true; _routes.append(RouteData()); } } else if (line[0] == 'B') { if (_date.isNull()) { _errorString = "Missing date header"; return false; } if (!track) { _tracks.append(TrackData()); _time = QTime(0, 0); track = true; } if (!readBRecord(line, len)) return false; } } _errorLine++; } return true; }