blob: 4dc44f16c7a8b447a7049d197a4e01df9e99e6d9 [file] [log] [blame]
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -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#include <aidl/Gtest.h>
17#include <aidl/Vintf.h>
18#include <aidl/android/hardware/biometrics/face/BnFace.h>
19#include <aidl/android/hardware/biometrics/face/BnSessionCallback.h>
20
21#include <android/binder_manager.h>
22#include <android/binder_process.h>
23
Ilya Matyukhincc2b6942021-03-23 23:08:41 -070024#include <chrono>
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070025#include <future>
26
27namespace aidl::android::hardware::biometrics::face {
28namespace {
29
Ilya Matyukhincc2b6942021-03-23 23:08:41 -070030using namespace std::literals::chrono_literals;
31
Ilya Matyukhin046020b2021-07-03 00:05:47 +000032using aidl::android::hardware::common::NativeHandle;
33
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070034constexpr int kSensorId = 0;
35constexpr int kUserId = 0;
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070036
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070037class SessionCallback : public BnSessionCallback {
38 public:
Ilya Matyukhin046020b2021-07-03 00:05:47 +000039 ndk::ScopedAStatus onChallengeGenerated(int64_t challenge) override {
40 auto lock = std::lock_guard{mMutex};
41 mOnChallengeGeneratedInvoked = true;
42 mGeneratedChallenge = challenge;
43 mCv.notify_one();
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070044 return ndk::ScopedAStatus::ok();
45 }
46
Ilya Matyukhin046020b2021-07-03 00:05:47 +000047 ndk::ScopedAStatus onChallengeRevoked(int64_t challenge) override {
48 auto lock = std::lock_guard{mMutex};
49 mOnChallengeRevokedInvoked = true;
50 mRevokedChallenge = challenge;
51 mCv.notify_one();
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070052 return ndk::ScopedAStatus::ok();
53 }
54
Ilya Matyukhine13bc812021-01-19 16:04:04 -080055 ndk::ScopedAStatus onAuthenticationFrame(const AuthenticationFrame& /*frame*/) override {
56 return ndk::ScopedAStatus::ok();
57 }
58
59 ndk::ScopedAStatus onEnrollmentFrame(const EnrollmentFrame& /*frame*/) override {
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070060 return ndk::ScopedAStatus::ok();
61 }
62
Ilya Matyukhin046020b2021-07-03 00:05:47 +000063 ndk::ScopedAStatus onError(Error error, int32_t /*vendorCode*/) override {
Ilya Matyukhin13d84062021-06-07 16:56:11 -070064 auto lock = std::lock_guard<std::mutex>{mMutex};
65 mError = error;
Ilya Matyukhin13d84062021-06-07 16:56:11 -070066 mOnErrorInvoked = true;
67 mCv.notify_one();
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070068 return ndk::ScopedAStatus::ok();
69 }
70
71 ndk::ScopedAStatus onEnrollmentProgress(int32_t /*enrollmentId*/,
72 int32_t /*remaining*/) override {
73 return ndk::ScopedAStatus::ok();
74 }
75
76 ndk::ScopedAStatus onAuthenticationSucceeded(
77 int32_t /*enrollmentId*/, const keymaster::HardwareAuthToken& /*hat*/) override {
78 return ndk::ScopedAStatus::ok();
79 }
80
81 ndk::ScopedAStatus onAuthenticationFailed() override { return ndk::ScopedAStatus::ok(); }
82
83 ndk::ScopedAStatus onLockoutTimed(int64_t /*durationMillis*/) override {
84 return ndk::ScopedAStatus::ok();
85 }
86
87 ndk::ScopedAStatus onLockoutPermanent() override { return ndk::ScopedAStatus::ok(); }
88
89 ndk::ScopedAStatus onLockoutCleared() override { return ndk::ScopedAStatus::ok(); }
90
91 ndk::ScopedAStatus onInteractionDetected() override { return ndk::ScopedAStatus::ok(); }
92
93 ndk::ScopedAStatus onEnrollmentsEnumerated(
94 const std::vector<int32_t>& /*enrollmentIds*/) override {
Ilya Matyukhin046020b2021-07-03 00:05:47 +000095 auto lock = std::lock_guard{mMutex};
96 mOnEnrollmentsEnumeratedInvoked = true;
97 mCv.notify_one();
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -070098 return ndk::ScopedAStatus::ok();
99 }
100
101 ndk::ScopedAStatus onEnrollmentsRemoved(
102 const std::vector<int32_t>& /*enrollmentIds*/) override {
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000103 auto lock = std::lock_guard{mMutex};
104 mOnEnrollmentsRemovedInvoked = true;
105 mCv.notify_one();
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700106 return ndk::ScopedAStatus::ok();
107 }
108
Ilya Matyukhin9fcf6b12021-04-14 13:43:06 -0700109 ndk::ScopedAStatus onFeaturesRetrieved(const std::vector<Feature>& /*features*/) override {
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000110 auto lock = std::lock_guard{mMutex};
111 mOnFeaturesRetrievedInvoked = true;
112 mCv.notify_one();
Ilya Matyukhinf2d38862021-01-22 11:48:59 -0800113 return ndk::ScopedAStatus::ok();
114 }
115
Ilya Matyukhin9fcf6b12021-04-14 13:43:06 -0700116 ndk::ScopedAStatus onFeatureSet(Feature /*feature*/) override {
Ilya Matyukhinf2d38862021-01-22 11:48:59 -0800117 return ndk::ScopedAStatus::ok();
118 }
119
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700120 ndk::ScopedAStatus onAuthenticatorIdRetrieved(int64_t /*authenticatorId*/) override {
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000121 auto lock = std::lock_guard{mMutex};
122 mOnAuthenticatorIdRetrievedInvoked = true;
123 mCv.notify_one();
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700124 return ndk::ScopedAStatus::ok();
125 }
126
Kevin Chynf7890cc2021-01-11 17:08:36 -0800127 ndk::ScopedAStatus onAuthenticatorIdInvalidated(int64_t /*newAuthenticatorId*/) override {
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000128 auto lock = std::lock_guard{mMutex};
129 mOnAuthenticatorIdInvalidatedInvoked = true;
130 mCv.notify_one();
Kevin Chynf7890cc2021-01-11 17:08:36 -0800131 return ndk::ScopedAStatus::ok();
132 }
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700133
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700134 ndk::ScopedAStatus onSessionClosed() override {
Ilya Matyukhin13d84062021-06-07 16:56:11 -0700135 auto lock = std::lock_guard<std::mutex>{mMutex};
136 mOnSessionClosedInvoked = true;
137 mCv.notify_one();
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700138 return ndk::ScopedAStatus::ok();
139 }
Ilya Matyukhincbbfa932021-03-22 13:25:15 -0700140
Ilya Matyukhin13d84062021-06-07 16:56:11 -0700141 std::mutex mMutex;
142 std::condition_variable mCv;
143 Error mError = Error::UNKNOWN;
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000144 int64_t mGeneratedChallenge = 0;
145 int64_t mRevokedChallenge = 0;
146 bool mOnChallengeGeneratedInvoked = false;
147 bool mOnChallengeRevokedInvoked = false;
Ilya Matyukhin13d84062021-06-07 16:56:11 -0700148 bool mOnErrorInvoked = false;
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000149 bool mOnEnrollmentsEnumeratedInvoked = false;
150 bool mOnEnrollmentsRemovedInvoked = false;
151 bool mOnFeaturesRetrievedInvoked = false;
152 bool mOnAuthenticatorIdRetrievedInvoked = false;
153 bool mOnAuthenticatorIdInvalidatedInvoked = false;
Ilya Matyukhin13d84062021-06-07 16:56:11 -0700154 bool mOnSessionClosedInvoked = false;
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700155};
156
157class Face : public testing::TestWithParam<std::string> {
158 protected:
159 void SetUp() override {
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000160 // Prepare the callback.
161 mCb = ndk::SharedRefBase::make<SessionCallback>();
162
163 int retries = 0;
164 bool isOk = false;
165 // If the first attempt to create a session fails, we try to create a session again. The
166 // first attempt might fail if the framework already has an active session. The AIDL
167 // contract doesn't allow to create a new session without closing the old one. However, we
168 // can't close the framework's session from VTS. The expectation here is that the HAL will
169 // crash after the first illegal attempt to create a session, then it will restart, and then
170 // we'll be able to create a session.
171 do {
172 // Get an instance of the HAL.
173 AIBinder* binder = AServiceManager_waitForService(GetParam().c_str());
174 ASSERT_NE(binder, nullptr);
175 mHal = IFace::fromBinder(ndk::SpAIBinder(binder));
176
177 // Create a session.
178 isOk = mHal->createSession(kSensorId, kUserId, mCb, &mSession).isOk();
179 ++retries;
180 } while (!isOk && retries < 2);
181
182 ASSERT_TRUE(isOk);
183 }
184
185 void TearDown() override {
186 // Close the mSession.
187 ASSERT_TRUE(mSession->close().isOk());
188
189 // Make sure the mSession is closed.
190 auto lock = std::unique_lock<std::mutex>(mCb->mMutex);
191 mCb->mCv.wait(lock, [this] { return mCb->mOnSessionClosedInvoked; });
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700192 }
193
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700194 std::shared_ptr<IFace> mHal;
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000195 std::shared_ptr<SessionCallback> mCb;
196 std::shared_ptr<ISession> mSession;
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700197};
198
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000199TEST_P(Face, GetSensorPropsWorksTest) {
200 std::vector<SensorProps> sensorProps;
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700201
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000202 // Call the method.
203 ASSERT_TRUE(mHal->getSensorProps(&sensorProps).isOk());
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700204
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000205 // Make sure the sensorProps aren't empty.
206 ASSERT_FALSE(sensorProps.empty());
207 ASSERT_FALSE(sensorProps[0].commonProps.componentInfo.empty());
208}
209
210TEST_P(Face, EnrollWithBadHatResultsInErrorTest) {
211 // Call the method.
212 auto hat = keymaster::HardwareAuthToken{};
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700213 std::shared_ptr<common::ICancellationSignal> cancellationSignal;
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000214 ASSERT_TRUE(
215 mSession->enroll(hat, EnrollmentType::DEFAULT, {}, NativeHandle{}, &cancellationSignal)
216 .isOk());
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700217
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000218 // Make sure an error is returned.
219 auto lock = std::unique_lock{mCb->mMutex};
220 mCb->mCv.wait(lock, [this] { return mCb->mOnErrorInvoked; });
221}
222
223TEST_P(Face, GenerateChallengeProducesUniqueChallengesTest) {
224 static constexpr int kIterations = 100;
225
226 auto challenges = std::set<int>{};
227 for (unsigned int i = 0; i < kIterations; ++i) {
228 // Call the method.
229 ASSERT_TRUE(mSession->generateChallenge().isOk());
230
231 // Check that the generated challenge is unique and not 0.
232 auto lock = std::unique_lock{mCb->mMutex};
233 mCb->mCv.wait(lock, [this] { return mCb->mOnChallengeGeneratedInvoked; });
234 ASSERT_NE(mCb->mGeneratedChallenge, 0);
235 ASSERT_EQ(challenges.find(mCb->mGeneratedChallenge), challenges.end());
236
237 challenges.insert(mCb->mGeneratedChallenge);
238 mCb->mOnChallengeGeneratedInvoked = false;
239 }
240}
241
242TEST_P(Face, RevokeChallengeWorksForNonexistentChallengeTest) {
243 const int64_t nonexistentChallenge = 123;
244
245 // Call the method.
246 ASSERT_TRUE(mSession->revokeChallenge(nonexistentChallenge).isOk());
247
248 // Check that the challenge is revoked and matches the requested challenge.
249 auto lock = std::unique_lock{mCb->mMutex};
250 mCb->mCv.wait(lock, [this] { return mCb->mOnChallengeRevokedInvoked; });
251 ASSERT_EQ(mCb->mRevokedChallenge, nonexistentChallenge);
252}
253
254TEST_P(Face, RevokeChallengeWorksForExistentChallengeTest) {
255 // Generate a challenge.
256 ASSERT_TRUE(mSession->generateChallenge().isOk());
257
258 // Wait for the result.
259 auto lock = std::unique_lock{mCb->mMutex};
260 mCb->mCv.wait(lock, [this] { return mCb->mOnChallengeGeneratedInvoked; });
Ilya Matyukhin13d84062021-06-07 16:56:11 -0700261 lock.unlock();
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700262
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000263 // Revoke the challenge.
264 ASSERT_TRUE(mSession->revokeChallenge(mCb->mGeneratedChallenge).isOk());
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700265
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000266 // Check that the challenge is revoked and matches the requested challenge.
Ilya Matyukhin13d84062021-06-07 16:56:11 -0700267 lock.lock();
Ilya Matyukhin046020b2021-07-03 00:05:47 +0000268 mCb->mCv.wait(lock, [this] { return mCb->mOnChallengeRevokedInvoked; });
269 ASSERT_EQ(mCb->mRevokedChallenge, mCb->mGeneratedChallenge);
270}
271
272TEST_P(Face, EnumerateEnrollmentsWorksTest) {
273 // Call the method.
274 ASSERT_TRUE(mSession->enumerateEnrollments().isOk());
275
276 // Wait for the result.
277 auto lock = std::unique_lock{mCb->mMutex};
278 mCb->mCv.wait(lock, [this] { return mCb->mOnEnrollmentsEnumeratedInvoked; });
279}
280
281TEST_P(Face, RemoveEnrollmentsWorksTest) {
282 // Call the method.
283 ASSERT_TRUE(mSession->removeEnrollments({}).isOk());
284
285 // Wait for the result.
286 auto lock = std::unique_lock{mCb->mMutex};
287 mCb->mCv.wait(lock, [this] { return mCb->mOnEnrollmentsRemovedInvoked; });
288}
289
290TEST_P(Face, GetFeaturesWorksTest) {
291 // Call the method.
292 ASSERT_TRUE(mSession->getFeatures().isOk());
293
294 // Wait for the result.
295 auto lock = std::unique_lock{mCb->mMutex};
296 mCb->mCv.wait(lock, [this] { return mCb->mOnFeaturesRetrievedInvoked; });
297}
298
299TEST_P(Face, GetAuthenticatorIdWorksTest) {
300 // Call the method.
301 ASSERT_TRUE(mSession->getAuthenticatorId().isOk());
302
303 // Wait for the result.
304 auto lock = std::unique_lock{mCb->mMutex};
305 mCb->mCv.wait(lock, [this] { return mCb->mOnAuthenticatorIdRetrievedInvoked; });
306}
307
308TEST_P(Face, InvalidateAuthenticatorIdWorksTest) {
309 // Call the method.
310 ASSERT_TRUE(mSession->invalidateAuthenticatorId().isOk());
311
312 // Wait for the result.
313 auto lock = std::unique_lock{mCb->mMutex};
314 mCb->mCv.wait(lock, [this] { return mCb->mOnAuthenticatorIdInvalidatedInvoked; });
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700315}
316
317GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(Face);
318INSTANTIATE_TEST_SUITE_P(IFace, Face,
319 testing::ValuesIn(::android::getAidlHalInstanceNames(IFace::descriptor)),
320 ::android::PrintInstanceNameToString);
321
322} // namespace
Ilya Matyukhincc2b6942021-03-23 23:08:41 -0700323} // namespace aidl::android::hardware::biometrics::face
Ilya Matyukhin3b542cd2020-10-12 18:23:46 -0700324
325int main(int argc, char** argv) {
326 ::testing::InitGoogleTest(&argc, argv);
327 ABinderProcess_setThreadPoolMaxThreadCount(1);
328 ABinderProcess_startThreadPool();
329 return RUN_ALL_TESTS();
330}