1 /*
2 * Copyright (C) 2019 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 "GnssHalTest"
18
19 #include <gnss_hal_test.h>
20 #include <chrono>
21 #include "Utils.h"
22
23 #include <gtest/gtest.h>
24
25 using ::android::hardware::gnss::common::Utils;
26
27 // Implementations for the main test class for GNSS HAL
SetUp()28 void GnssHalTest::SetUp() {
29 gnss_hal_ = IGnss::getService(GetParam());
30 ASSERT_NE(gnss_hal_, nullptr);
31
32 SetUpGnssCallback();
33 }
34
TearDown()35 void GnssHalTest::TearDown() {
36 if (gnss_hal_ != nullptr) {
37 gnss_hal_->cleanup();
38 gnss_hal_ = nullptr;
39 }
40
41 // Set to nullptr to destruct the callback event queues and warn of any unprocessed events.
42 gnss_cb_ = nullptr;
43 }
44
SetUpGnssCallback()45 void GnssHalTest::SetUpGnssCallback() {
46 gnss_cb_ = new GnssCallback();
47 ASSERT_NE(gnss_cb_, nullptr);
48
49 auto result = gnss_hal_->setCallback_2_1(gnss_cb_);
50 if (!result.isOk()) {
51 ALOGE("result of failed setCallback %s", result.description().c_str());
52 }
53
54 ASSERT_TRUE(result.isOk());
55 ASSERT_TRUE(result);
56
57 /*
58 * All capabilities, name and systemInfo callbacks should trigger
59 */
60 EXPECT_TRUE(gnss_cb_->capabilities_cbq_.retrieve(gnss_cb_->last_capabilities_, TIMEOUT_SEC));
61 EXPECT_TRUE(gnss_cb_->info_cbq_.retrieve(gnss_cb_->last_info_, TIMEOUT_SEC));
62 EXPECT_TRUE(gnss_cb_->name_cbq_.retrieve(gnss_cb_->last_name_, TIMEOUT_SEC));
63
64 EXPECT_EQ(gnss_cb_->capabilities_cbq_.calledCount(), 1);
65 EXPECT_EQ(gnss_cb_->info_cbq_.calledCount(), 1);
66 EXPECT_EQ(gnss_cb_->name_cbq_.calledCount(), 1);
67 }
68
StopAndClearLocations()69 void GnssHalTest::StopAndClearLocations() {
70 const auto result = gnss_hal_->stop();
71
72 EXPECT_TRUE(result.isOk());
73 EXPECT_TRUE(result);
74
75 /*
76 * Clear notify/waiting counter, allowing up till the timeout after
77 * the last reply for final startup messages to arrive (esp. system
78 * info.)
79 */
80 while (gnss_cb_->location_cbq_.retrieve(gnss_cb_->last_location_, TIMEOUT_SEC)) {
81 }
82 gnss_cb_->location_cbq_.reset();
83 }
84
SetPositionMode(const int min_interval_msec,const bool low_power_mode)85 void GnssHalTest::SetPositionMode(const int min_interval_msec, const bool low_power_mode) {
86 const int kPreferredAccuracy = 0; // Ideally perfect (matches GnssLocationProvider)
87 const int kPreferredTimeMsec = 0; // Ideally immediate
88
89 const auto result = gnss_hal_->setPositionMode_1_1(
90 IGnss::GnssPositionMode::MS_BASED, IGnss::GnssPositionRecurrence::RECURRENCE_PERIODIC,
91 min_interval_msec, kPreferredAccuracy, kPreferredTimeMsec, low_power_mode);
92
93 ASSERT_TRUE(result.isOk());
94 EXPECT_TRUE(result);
95 }
96
StartAndCheckFirstLocation()97 bool GnssHalTest::StartAndCheckFirstLocation() {
98 const auto result = gnss_hal_->start();
99
100 EXPECT_TRUE(result.isOk());
101 EXPECT_TRUE(result);
102
103 /*
104 * GnssLocationProvider support of AGPS SUPL & XtraDownloader is not available in VTS,
105 * so allow time to demodulate ephemeris over the air.
106 */
107 const int kFirstGnssLocationTimeoutSeconds = 75;
108
109 EXPECT_TRUE(gnss_cb_->location_cbq_.retrieve(gnss_cb_->last_location_,
110 kFirstGnssLocationTimeoutSeconds));
111 int locationCalledCount = gnss_cb_->location_cbq_.calledCount();
112 EXPECT_EQ(locationCalledCount, 1);
113
114 if (locationCalledCount > 0) {
115 // don't require speed on first fix
116 CheckLocation(gnss_cb_->last_location_, false);
117 return true;
118 }
119 return false;
120 }
121
CheckLocation(const GnssLocation_2_0 & location,bool check_speed)122 void GnssHalTest::CheckLocation(const GnssLocation_2_0& location, bool check_speed) {
123 const bool check_more_accuracies =
124 (gnss_cb_->info_cbq_.calledCount() > 0 && gnss_cb_->last_info_.yearOfHw >= 2017);
125
126 Utils::checkLocation(location.v1_0, check_speed, check_more_accuracies);
127 }
128
StartAndCheckLocations(int count)129 void GnssHalTest::StartAndCheckLocations(int count) {
130 const int kMinIntervalMsec = 500;
131 const int kLocationTimeoutSubsequentSec = 2;
132 const bool kLowPowerMode = false;
133
134 SetPositionMode(kMinIntervalMsec, kLowPowerMode);
135
136 EXPECT_TRUE(StartAndCheckFirstLocation());
137
138 for (int i = 1; i < count; i++) {
139 EXPECT_TRUE(gnss_cb_->location_cbq_.retrieve(gnss_cb_->last_location_,
140 kLocationTimeoutSubsequentSec));
141 int locationCalledCount = gnss_cb_->location_cbq_.calledCount();
142 EXPECT_EQ(locationCalledCount, i + 1);
143 // Don't cause confusion by checking details if no location yet
144 if (locationCalledCount > 0) {
145 // Should be more than 1 location by now, but if not, still don't check first fix speed
146 CheckLocation(gnss_cb_->last_location_, locationCalledCount > 1);
147 }
148 }
149 }
150
startLocationAndGetNonGpsConstellation(const int locations_to_await,const int gnss_sv_info_list_timeout)151 GnssConstellationType GnssHalTest::startLocationAndGetNonGpsConstellation(
152 const int locations_to_await, const int gnss_sv_info_list_timeout) {
153 gnss_cb_->location_cbq_.reset();
154 StartAndCheckLocations(locations_to_await);
155 const int location_called_count = gnss_cb_->location_cbq_.calledCount();
156
157 // Tolerate 1 less sv status to handle edge cases in reporting.
158 int sv_info_list_cbq_size = gnss_cb_->sv_info_list_cbq_.size();
159 EXPECT_GE(sv_info_list_cbq_size + 1, locations_to_await);
160 ALOGD("Observed %d GnssSvInfo, while awaiting %d Locations (%d received)",
161 sv_info_list_cbq_size, locations_to_await, location_called_count);
162
163 // Find first non-GPS constellation to blacklist
164 GnssConstellationType constellation_to_blacklist = GnssConstellationType::UNKNOWN;
165 for (int i = 0; i < sv_info_list_cbq_size; ++i) {
166 hidl_vec<IGnssCallback_2_1::GnssSvInfo> sv_info_vec;
167 gnss_cb_->sv_info_list_cbq_.retrieve(sv_info_vec, gnss_sv_info_list_timeout);
168 for (uint32_t iSv = 0; iSv < sv_info_vec.size(); iSv++) {
169 const auto& gnss_sv = sv_info_vec[iSv];
170 if ((gnss_sv.v2_0.v1_0.svFlag & IGnssCallback_1_0::GnssSvFlags::USED_IN_FIX) &&
171 (gnss_sv.v2_0.constellation != GnssConstellationType::UNKNOWN) &&
172 (gnss_sv.v2_0.constellation != GnssConstellationType::GPS)) {
173 // found a non-GPS constellation
174 constellation_to_blacklist = gnss_sv.v2_0.constellation;
175 break;
176 }
177 }
178 if (constellation_to_blacklist != GnssConstellationType::UNKNOWN) {
179 break;
180 }
181 }
182
183 if (constellation_to_blacklist == GnssConstellationType::UNKNOWN) {
184 ALOGI("No non-GPS constellations found, constellation blacklist test less effective.");
185 // Proceed functionally to blacklist something.
186 constellation_to_blacklist = GnssConstellationType::GLONASS;
187 }
188
189 return constellation_to_blacklist;
190 }
191
GnssCallback()192 GnssHalTest::GnssCallback::GnssCallback()
193 : info_cbq_("system_info"),
194 name_cbq_("name"),
195 capabilities_cbq_("capabilities"),
196 location_cbq_("location"),
197 sv_info_list_cbq_("sv_info") {}
198
gnssSetSystemInfoCb(const IGnssCallback_1_0::GnssSystemInfo & info)199 Return<void> GnssHalTest::GnssCallback::gnssSetSystemInfoCb(
200 const IGnssCallback_1_0::GnssSystemInfo& info) {
201 ALOGI("Info received, year %d", info.yearOfHw);
202 info_cbq_.store(info);
203 return Void();
204 }
205
gnssSetCapabilitesCb(uint32_t capabilities)206 Return<void> GnssHalTest::GnssCallback::gnssSetCapabilitesCb(uint32_t capabilities) {
207 ALOGI("Capabilities received %d", capabilities);
208 capabilities_cbq_.store(capabilities);
209 return Void();
210 }
211
gnssSetCapabilitiesCb_2_0(uint32_t capabilities)212 Return<void> GnssHalTest::GnssCallback::gnssSetCapabilitiesCb_2_0(uint32_t capabilities) {
213 ALOGI("Capabilities (v2.0) received %d", capabilities);
214 capabilities_cbq_.store(capabilities);
215 return Void();
216 }
217
gnssSetCapabilitiesCb_2_1(uint32_t capabilities)218 Return<void> GnssHalTest::GnssCallback::gnssSetCapabilitiesCb_2_1(uint32_t capabilities) {
219 ALOGI("Capabilities (v2.1) received %d", capabilities);
220 capabilities_cbq_.store(capabilities);
221 return Void();
222 }
223
gnssNameCb(const android::hardware::hidl_string & name)224 Return<void> GnssHalTest::GnssCallback::gnssNameCb(const android::hardware::hidl_string& name) {
225 ALOGI("Name received: %s", name.c_str());
226 name_cbq_.store(name);
227 return Void();
228 }
229
gnssLocationCb(const GnssLocation_1_0 & location)230 Return<void> GnssHalTest::GnssCallback::gnssLocationCb(const GnssLocation_1_0& location) {
231 ALOGI("Location received");
232 GnssLocation_2_0 location_v2_0;
233 location_v2_0.v1_0 = location;
234 return gnssLocationCbImpl(location_v2_0);
235 }
236
gnssLocationCb_2_0(const GnssLocation_2_0 & location)237 Return<void> GnssHalTest::GnssCallback::gnssLocationCb_2_0(const GnssLocation_2_0& location) {
238 ALOGI("Location (v2.0) received");
239 return gnssLocationCbImpl(location);
240 }
241
gnssLocationCbImpl(const GnssLocation_2_0 & location)242 Return<void> GnssHalTest::GnssCallback::gnssLocationCbImpl(const GnssLocation_2_0& location) {
243 location_cbq_.store(location);
244 return Void();
245 }
246
gnssSvStatusCb(const IGnssCallback_1_0::GnssSvStatus &)247 Return<void> GnssHalTest::GnssCallback::gnssSvStatusCb(const IGnssCallback_1_0::GnssSvStatus&) {
248 ALOGI("gnssSvStatusCb");
249 return Void();
250 }
251
gnssSvStatusCb_2_1(const hidl_vec<IGnssCallback_2_1::GnssSvInfo> & svInfoList)252 Return<void> GnssHalTest::GnssCallback::gnssSvStatusCb_2_1(
253 const hidl_vec<IGnssCallback_2_1::GnssSvInfo>& svInfoList) {
254 ALOGI("gnssSvStatusCb_2_1. Size = %d", (int)svInfoList.size());
255 sv_info_list_cbq_.store(svInfoList);
256 return Void();
257 }
258
gnssMeasurementCb_2_1(const IGnssMeasurementCallback_2_1::GnssData & data)259 Return<void> GnssHalTest::GnssMeasurementCallback::gnssMeasurementCb_2_1(
260 const IGnssMeasurementCallback_2_1::GnssData& data) {
261 ALOGD("GnssMeasurement v2.1 received. Size = %d", (int)data.measurements.size());
262 measurement_cbq_.store(data);
263 return Void();
264 }
265
setCapabilitiesCb(uint32_t capabilities)266 Return<void> GnssHalTest::GnssMeasurementCorrectionsCallback::setCapabilitiesCb(
267 uint32_t capabilities) {
268 ALOGI("GnssMeasurementCorrectionsCallback capabilities received %d", capabilities);
269 capabilities_cbq_.store(capabilities);
270 return Void();
271 }
272
gnssAntennaInfoCb(const hidl_vec<IGnssAntennaInfoCallback::GnssAntennaInfo> & gnssAntennaInfos)273 Return<void> GnssHalTest::GnssAntennaInfoCallback::gnssAntennaInfoCb(
274 const hidl_vec<IGnssAntennaInfoCallback::GnssAntennaInfo>& gnssAntennaInfos) {
275 ALOGD("GnssAntennaInfo v2.1 received. Size = %d", (int)gnssAntennaInfos.size());
276 antenna_info_cbq_.store(gnssAntennaInfos);
277 return Void();
278 }