| /* |
| * Copyright (C) 2022 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include <aidl/android/hardware/biometrics/fingerprint/BnSessionCallback.h> |
| #include <android/binder_process.h> |
| #include <fingerprint.sysprop.h> |
| #include <gtest/gtest.h> |
| |
| #include <android-base/logging.h> |
| |
| #include "FakeFingerprintEngine.h" |
| #include "FakeFingerprintEngineUdfps.h" |
| #include "Fingerprint.h" |
| |
| using namespace ::android::fingerprint::virt; |
| using namespace ::aidl::android::hardware::biometrics::fingerprint; |
| using namespace ::aidl::android::hardware::keymaster; |
| |
| namespace aidl::android::hardware::biometrics::fingerprint { |
| |
| class TestSessionCallback : public BnSessionCallback { |
| public: |
| ndk::ScopedAStatus onChallengeGenerated(int64_t /*challenge*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onChallengeRevoked(int64_t /*challenge*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onError(fingerprint::Error /*error*/, int32_t /*vendorCode*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onEnrollmentProgress(int32_t /*enrollmentId*/, |
| int32_t /*remaining*/) override { |
| mEnrollmentProgress++; |
| return ndk::ScopedAStatus::ok(); |
| }; |
| |
| ::ndk::ScopedAStatus onAuthenticationSucceeded(int32_t /*enrollmentId*/, |
| const keymaster::HardwareAuthToken&) override { |
| mAuthenticationSuccess++; |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onAuthenticationFailed() override { |
| mAuthenticationFailure++; |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onInteractionDetected() override { |
| mDetectInteraction++; |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ndk::ScopedAStatus onAcquired(AcquiredInfo /*info*/, int32_t /*vendorCode*/) override { |
| return ndk::ScopedAStatus::ok(); |
| } |
| ::ndk::ScopedAStatus onEnrollmentsEnumerated( |
| const std::vector<int32_t>& /*enrollmentIds*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onEnrollmentsRemoved( |
| const std::vector<int32_t>& /*enrollmentIds*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onAuthenticatorIdRetrieved(int64_t /*authenticatorId*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onAuthenticatorIdInvalidated(int64_t /*authenticatorId*/) override { |
| return ndk::ScopedAStatus::ok(); |
| }; |
| ::ndk::ScopedAStatus onLockoutPermanent() override { return ndk::ScopedAStatus::ok(); }; |
| ndk::ScopedAStatus onLockoutTimed(int64_t /* timeout */) override { |
| return ndk::ScopedAStatus::ok(); |
| } |
| ndk::ScopedAStatus onLockoutCleared() override { return ndk::ScopedAStatus::ok(); } |
| ndk::ScopedAStatus onSessionClosed() override { return ndk::ScopedAStatus::ok(); } |
| |
| int32_t getAuthenticationCount() { return mAuthenticationSuccess + mAuthenticationFailure; } |
| int32_t getDetectInteractionCount() { return mDetectInteraction; } |
| |
| int32_t mAuthenticationSuccess = 0; |
| int32_t mAuthenticationFailure = 0; |
| int32_t mEnrollmentProgress = 0; |
| int32_t mDetectInteraction = 0; |
| }; |
| |
| class FakeFingerprintEngineUdfpsTest : public ::testing::Test { |
| protected: |
| void SetUp() override {} |
| |
| void TearDown() override { |
| // reset to default |
| Fingerprint::cfg().set<std::string>("sensor_location", ""); |
| } |
| |
| FakeFingerprintEngineUdfps mEngine; |
| }; |
| |
| bool isDefaultLocation(SensorLocation& sc) { |
| return (sc.sensorLocationX == FakeFingerprintEngineUdfps::defaultSensorLocationX && |
| sc.sensorLocationY == FakeFingerprintEngineUdfps::defaultSensorLocationY && |
| sc.sensorRadius == FakeFingerprintEngineUdfps::defaultSensorRadius && sc.display == ""); |
| } |
| |
| TEST_F(FakeFingerprintEngineUdfpsTest, getSensorLocationOk) { |
| auto loc = "100:200:30"; |
| Fingerprint::cfg().set<std::string>("sensor_location", loc); |
| SensorLocation sc = mEngine.getSensorLocation(); |
| ASSERT_TRUE(sc.sensorLocationX == 100); |
| ASSERT_TRUE(sc.sensorLocationY == 200); |
| ASSERT_TRUE(sc.sensorRadius == 30); |
| |
| loc = "100:200:30:screen1"; |
| Fingerprint::cfg().set<std::string>("sensor_location", loc); |
| sc = mEngine.getSensorLocation(); |
| ASSERT_TRUE(sc.sensorLocationX == 100); |
| ASSERT_TRUE(sc.sensorLocationY == 200); |
| ASSERT_TRUE(sc.sensorRadius == 30); |
| ASSERT_TRUE(sc.display == "screen1"); |
| } |
| |
| TEST_F(FakeFingerprintEngineUdfpsTest, getSensorLocationBad) { |
| const std::vector<std::string> badStr{"", "100", "10:20", "10,20,5", "a:b:c"}; |
| SensorLocation sc; |
| for (const auto& s : badStr) { |
| Fingerprint::cfg().set<std::string>("sensor_location", s); |
| sc = mEngine.getSensorLocation(); |
| ASSERT_TRUE(isDefaultLocation(sc)); |
| } |
| } |
| |
| TEST_F(FakeFingerprintEngineUdfpsTest, initialization) { |
| ASSERT_TRUE(mEngine.getWorkMode() == FakeFingerprintEngineUdfps::WorkMode::kIdle); |
| } |
| |
| TEST_F(FakeFingerprintEngineUdfpsTest, authenticate) { |
| std::shared_ptr<TestSessionCallback> cb = ndk::SharedRefBase::make<TestSessionCallback>(); |
| std::promise<void> cancel; |
| mEngine.notifyFingerdown(); |
| mEngine.authenticateImpl(cb.get(), 1, cancel.get_future()); |
| ASSERT_TRUE(mEngine.getWorkMode() == FakeFingerprintEngineUdfps::WorkMode::kAuthenticate); |
| mEngine.onPointerDownImpl(1, 2, 3, 4.0, 5.0); |
| ASSERT_EQ(cb->getAuthenticationCount(), 0); |
| mEngine.onUiReadyImpl(); |
| ASSERT_EQ(cb->getAuthenticationCount(), 1); |
| } |
| |
| TEST_F(FakeFingerprintEngineUdfpsTest, enroll) { |
| std::shared_ptr<TestSessionCallback> cb = ndk::SharedRefBase::make<TestSessionCallback>(); |
| std::promise<void> cancel; |
| keymaster::HardwareAuthToken hat{.mac = {5, 6}}; |
| Fingerprint::cfg().set<std::string>("next_enrollment", "5:0,0:true"); |
| mEngine.notifyFingerdown(); |
| mEngine.enrollImpl(cb.get(), hat, cancel.get_future()); |
| ASSERT_TRUE(mEngine.getWorkMode() == FakeFingerprintEngineUdfps::WorkMode::kEnroll); |
| mEngine.onPointerDownImpl(1, 2, 3, 4.0, 5.0); |
| ASSERT_EQ(cb->mEnrollmentProgress, 0); |
| mEngine.onUiReadyImpl(); |
| ASSERT_TRUE(cb->mEnrollmentProgress > 0); |
| } |
| |
| TEST_F(FakeFingerprintEngineUdfpsTest, detectInteraction) { |
| Fingerprint::cfg().set<bool>("detect_interaction", true); |
| Fingerprint::cfg().setopt<OptIntVec>("enrollments", {1, 2}); |
| Fingerprint::cfg().set<std::int32_t>("enrollment_hit", 2); |
| Fingerprint::cfg().set<std::string>("operation_detect_interaction_acquired", ""); |
| std::shared_ptr<TestSessionCallback> cb = ndk::SharedRefBase::make<TestSessionCallback>(); |
| std::promise<void> cancel; |
| mEngine.notifyFingerdown(); |
| mEngine.detectInteractionImpl(cb.get(), cancel.get_future()); |
| ASSERT_TRUE(mEngine.getWorkMode() == FakeFingerprintEngineUdfps::WorkMode::kDetectInteract); |
| mEngine.onPointerDownImpl(1, 2, 3, 4.0, 5.0); |
| ASSERT_EQ(cb->getDetectInteractionCount(), 0); |
| mEngine.onUiReadyImpl(); |
| ASSERT_EQ(cb->getDetectInteractionCount(), 1); |
| } |
| // More |
| } // namespace aidl::android::hardware::biometrics::fingerprint |