blob: 22aef90204a6da63c6cfa409a9beff4e11472df5 [file] [log] [blame]
Yipeng Cao48618f62020-03-19 18:11:16 -07001/*
2 * Copyright (C) 2020 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "NmeaFixInfo"
18
19#include <Constants.h>
20#include <NmeaFixInfo.h>
21#include <Utils.h>
22#include <log/log.h>
23#include <sys/stat.h>
24#include <sys/types.h>
25#include <unistd.h>
26#include <utils/SystemClock.h>
27#include <limits>
28#include <sstream>
29#include <string>
30#include <vector>
31
32namespace android {
33namespace hardware {
34namespace gnss {
35namespace common {
36
Yuchen He42b2d0a2022-01-12 04:39:37 +000037using aidl::android::hardware::gnss::ElapsedRealtime;
38using aidl::android::hardware::gnss::GnssLocation;
39
Yipeng Cao48618f62020-03-19 18:11:16 -070040NmeaFixInfo::NmeaFixInfo() : hasGMCRecord(false), hasGGARecord(false) {}
41
42float NmeaFixInfo::getAltitudeMeters() const {
43 return altitudeMeters;
44}
45
46float NmeaFixInfo::checkAndConvertToFloat(const std::string& sentence) {
47 if (sentence.empty()) {
48 return std::numeric_limits<float>::quiet_NaN();
49 }
50 return std::stof(sentence);
51}
52
53float NmeaFixInfo::getBearingAccuracyDegrees() const {
54 // Current NMEA doesn't contains beaing accuracy inforamtion
55 return kMockBearingAccuracyDegrees;
56}
57float NmeaFixInfo::getBearingDegrees() const {
58 return bearingDegrees;
59}
60
61float NmeaFixInfo::getHorizontalAccuracyMeters() const {
62 // Current NMEA doesn't contains horizontal accuracy inforamtion
63 return kMockHorizontalAccuracyMeters;
64}
65
66float NmeaFixInfo::getLatDeg() const {
67 return latDeg;
68}
69
70float NmeaFixInfo::getLngDeg() const {
71 return lngDeg;
72}
73
74float NmeaFixInfo::getSpeedAccuracyMetersPerSecond() const {
75 // Current NMEA doesn't contains speed accuracy inforamtion
76 return kMockSpeedAccuracyMetersPerSecond;
77}
78
79float NmeaFixInfo::getSpeedMetersPerSec() const {
80 return speedMetersPerSec;
81}
82
83int64_t NmeaFixInfo::getTimestamp() const {
84 return timestamp;
85}
86
87float NmeaFixInfo::getVerticalAccuracyMeters() const {
88 // Current NMEA doesn't contains vertical accuracy inforamtion
89 return kMockVerticalAccuracyMeters;
90}
91
92int64_t NmeaFixInfo::nmeaPartsToTimestamp(const std::string& timeStr, const std::string& dateStr) {
93 /**
94 * In NMEA format, the full time can only get from the $GPRMC record, see
95 * the following example:
96 * $GPRMC,213204.00,A,3725.371240,N,12205.589239,W,000.0,000.0,290819,,,A*49
97 * the datetime is stored in two parts, 213204 and 290819, which means
98 * 2019/08/29 21:32:04, however for in unix the year starts from 1900, we
99 * need to add the offset.
100 */
101 struct tm tm;
102 const int32_t unixYearOffset = 100;
103 tm.tm_mday = std::stoi(dateStr.substr(0, 2).c_str());
104 tm.tm_mon = std::stoi(dateStr.substr(2, 2).c_str()) - 1;
105 tm.tm_year = std::stoi(dateStr.substr(4, 2).c_str()) + unixYearOffset;
106 tm.tm_hour = std::stoi(timeStr.substr(0, 2).c_str());
107 tm.tm_min = std::stoi(timeStr.substr(2, 2).c_str());
108 tm.tm_sec = std::stoi(timeStr.substr(4, 2).c_str());
109 return static_cast<int64_t>(mktime(&tm) - timezone);
110}
111
112bool NmeaFixInfo::isValidFix() const {
113 return hasGMCRecord && hasGGARecord;
114}
115
116void NmeaFixInfo::parseGGALine(const std::vector<std::string>& sentenceValues) {
117 if (sentenceValues.size() == 0 || sentenceValues[0].compare(GPGA_RECORD_TAG) != 0) {
118 return;
119 }
120 // LatDeg, need covert to degree, if it is 'N', should be negative value
121 this->latDeg = std::stof(sentenceValues[2].substr(0, 2)) +
122 (std::stof(sentenceValues[2].substr(2)) / 60.0);
123 if (sentenceValues[3].compare("N") != 0) {
124 this->latDeg *= -1;
125 }
126
127 // LngDeg, need covert to degree, if it is 'E', should be negative value
128 this->lngDeg = std::stof(sentenceValues[4].substr(0, 3)) +
129 std::stof(sentenceValues[4].substr(3)) / 60.0;
130 if (sentenceValues[5].compare("E") != 0) {
131 this->lngDeg *= -1;
132 }
133
134 this->altitudeMeters = std::stof(sentenceValues[9]);
135
136 this->hDop = sentenceValues[8].empty() ? std::numeric_limits<float>::quiet_NaN()
137 : std::stof(sentenceValues[8]);
138 this->hasGGARecord = true;
139}
140
141void NmeaFixInfo::parseRMCLine(const std::vector<std::string>& sentenceValues) {
142 if (sentenceValues.size() == 0 || sentenceValues[0].compare(GPRMC_RECORD_TAG) != 0) {
143 return;
144 }
145 this->speedMetersPerSec = checkAndConvertToFloat(sentenceValues[7]);
146 this->bearingDegrees = checkAndConvertToFloat(sentenceValues[8]);
147 this->timestamp = nmeaPartsToTimestamp(sentenceValues[1], sentenceValues[9]);
148 this->hasGMCRecord = true;
149}
150
151/** invalid the current NmeaFixInfo */
152void NmeaFixInfo::reset() {
153 this->altitudeMeters = 0;
154 this->bearingDegrees = 0;
155 this->fixId = 0;
156 this->hasGMCRecord = false;
157 this->hasGGARecord = false;
158 this->latDeg = 0;
159 this->lngDeg = 0;
160 this->hDop = 0;
161 this->vDop = 0;
162 this->satelliteCount = 0;
163 this->speedMetersPerSec = 0;
164 this->timestamp = 0;
165}
166
167void NmeaFixInfo::splitStr(const std::string& line, const char& delimiter,
168 std::vector<std::string>& out) {
169 std::istringstream iss(line);
170 std::string item;
171 while (std::getline(iss, item, delimiter)) {
172 out.push_back(item);
173 }
174}
175
176NmeaFixInfo& NmeaFixInfo::operator=(const NmeaFixInfo& rhs) {
177 if (this == &rhs) return *this;
178 this->altitudeMeters = rhs.altitudeMeters;
179 this->bearingDegrees = rhs.bearingDegrees;
180 this->fixId = rhs.fixId;
181 this->hasGMCRecord = rhs.hasGMCRecord;
182 this->hasGGARecord = rhs.hasGGARecord;
183 this->hDop = rhs.hDop;
184 this->vDop = rhs.vDop;
185 this->latDeg = rhs.latDeg;
186 this->lngDeg = rhs.lngDeg;
187 this->satelliteCount = rhs.satelliteCount;
188 this->speedMetersPerSec = rhs.speedMetersPerSec;
189 this->timestamp = rhs.timestamp;
190
191 return *this;
192}
193
194/**
195 * Parses the input string in NMEA format and convert to GnssLocation.
196 * Currently version only cares about $GPGGA and $GPRMC records. but we
197 * can easily extend to other types supported by NMEA if needed.
198 */
199std::unique_ptr<V2_0::GnssLocation> NmeaFixInfo::getLocationFromInputStr(
200 const std::string& inputStr) {
201 std::vector<std::string> nmeaRecords;
202 splitStr(inputStr, LINE_SEPARATOR, nmeaRecords);
203 NmeaFixInfo nmeaFixInfo;
204 NmeaFixInfo candidateFixInfo;
205 uint32_t fixId = 0;
206 double lastTimeStamp = 0;
207 for (const auto& line : nmeaRecords) {
Yipeng Cao80451ef2020-12-23 17:04:29 -0800208 if (line.compare(0, strlen(GPGA_RECORD_TAG), GPGA_RECORD_TAG) != 0 &&
209 line.compare(0, strlen(GPRMC_RECORD_TAG), GPRMC_RECORD_TAG) != 0) {
210 continue;
211 }
Yipeng Cao48618f62020-03-19 18:11:16 -0700212 std::vector<std::string> sentenceValues;
213 splitStr(line, COMMA_SEPARATOR, sentenceValues);
Yipeng Cao80451ef2020-12-23 17:04:29 -0800214 if (sentenceValues.size() < MIN_COL_NUM) {
215 continue;
216 }
Yipeng Cao48618f62020-03-19 18:11:16 -0700217 double currentTimeStamp = std::stof(sentenceValues[1]);
218 // If see a new timestamp, report correct location.
219 if ((currentTimeStamp - lastTimeStamp) > TIMESTAMP_EPSILON &&
220 candidateFixInfo.isValidFix()) {
221 nmeaFixInfo = candidateFixInfo;
222 candidateFixInfo.reset();
223 fixId++;
224 }
225 if (line.compare(0, strlen(GPGA_RECORD_TAG), GPGA_RECORD_TAG) == 0) {
226 candidateFixInfo.fixId = fixId;
227 candidateFixInfo.parseGGALine(sentenceValues);
228 } else if (line.compare(0, strlen(GPRMC_RECORD_TAG), GPRMC_RECORD_TAG) == 0) {
229 candidateFixInfo.parseRMCLine(sentenceValues);
230 }
231 }
232 if (candidateFixInfo.isValidFix()) {
233 nmeaFixInfo = candidateFixInfo;
234 candidateFixInfo.reset();
235 }
236 if (!nmeaFixInfo.isValidFix()) {
237 return nullptr;
238 }
239 return nmeaFixInfo.toGnssLocation();
240}
241
242/**
Yuchen He42b2d0a2022-01-12 04:39:37 +0000243 * Convert V2_0::GnssLocation to aidl::GnssLocation.
244 */
245std::unique_ptr<GnssLocation> NmeaFixInfo::getAidlLocationFromInputStr(
246 const std::string& inputStr) {
247 std::unique_ptr<V2_0::GnssLocation> locationV2 = getLocationFromInputStr(inputStr);
248 if (locationV2 == nullptr) {
249 return nullptr;
250 }
251
252 ElapsedRealtime elapsedRealtime = {
253 .flags = ElapsedRealtime::HAS_TIMESTAMP_NS | ElapsedRealtime::HAS_TIME_UNCERTAINTY_NS,
254 .timestampNs = ::android::elapsedRealtimeNano(),
255 // This is an hardcoded value indicating a 1ms of uncertainty between the two clocks.
256 // In an actual implementation provide an estimate of the synchronization uncertainty
257 // or don't set the field.
258 .timeUncertaintyNs = 1020400};
259
260 GnssLocation location = {
261 .gnssLocationFlags = locationV2->v1_0.gnssLocationFlags,
262 .latitudeDegrees = locationV2->v1_0.latitudeDegrees,
263 .longitudeDegrees = locationV2->v1_0.longitudeDegrees,
264 .altitudeMeters = locationV2->v1_0.altitudeMeters,
265 .speedMetersPerSec = locationV2->v1_0.speedMetersPerSec,
266 .bearingDegrees = locationV2->v1_0.bearingDegrees,
267 .horizontalAccuracyMeters = locationV2->v1_0.horizontalAccuracyMeters,
268 .verticalAccuracyMeters = locationV2->v1_0.verticalAccuracyMeters,
269 .speedAccuracyMetersPerSecond = locationV2->v1_0.speedAccuracyMetersPerSecond,
270 .bearingAccuracyDegrees = locationV2->v1_0.bearingAccuracyDegrees,
271 .timestampMillis = locationV2->v1_0.timestamp,
272 .elapsedRealtime = elapsedRealtime};
273 return std::make_unique<GnssLocation>(location);
274}
275
276/**
Yipeng Cao48618f62020-03-19 18:11:16 -0700277 * Parses the input string in NMEA format and convert to GnssLocation.
278 */
279std::unique_ptr<V2_0::GnssLocation> NmeaFixInfo::toGnssLocation() const {
280 const V2_0::ElapsedRealtime currentOsTimestamp = {
281 .flags = V2_0::ElapsedRealtimeFlags::HAS_TIMESTAMP_NS |
282 V2_0::ElapsedRealtimeFlags::HAS_TIME_UNCERTAINTY_NS,
283 .timestampNs = static_cast<uint64_t>(::android::elapsedRealtimeNano()),
284 // This is an hardcoded value indicating a 1ms of uncertainty between the two clocks.
285 // In an actual implementation provide an estimate of the synchronization uncertainty
286 // or don't set the field.
287 .timeUncertaintyNs = 1000000};
288
289 V1_0::GnssLocation locationV1 = {
290 .gnssLocationFlags = 0xFF,
291 .latitudeDegrees = this->getLatDeg(),
292 .longitudeDegrees = this->getLngDeg(),
293 .altitudeMeters = this->getAltitudeMeters(),
294 .speedMetersPerSec = this->getSpeedMetersPerSec(),
295 .bearingDegrees = this->getBearingDegrees(),
296 .horizontalAccuracyMeters = this->getHorizontalAccuracyMeters(),
297 .verticalAccuracyMeters = this->getVerticalAccuracyMeters(),
298 .speedAccuracyMetersPerSecond = this->getSpeedAccuracyMetersPerSecond(),
299 .bearingAccuracyDegrees = this->getBearingAccuracyDegrees(),
300 .timestamp = this->getTimestamp()};
301
302 V2_0::GnssLocation locationV2 = {.v1_0 = locationV1, .elapsedRealtime = currentOsTimestamp};
303
304 return std::make_unique<V2_0::GnssLocation>(locationV2);
305}
306
307} // namespace common
308} // namespace gnss
309} // namespace hardware
310} // namespace android