You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
248 lines
8.1 KiB
248 lines
8.1 KiB
/*
|
|
* Copyright (C) 2020 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 "Session.h"
|
|
|
|
#include <android-base/logging.h>
|
|
|
|
#include "CancellationSignal.h"
|
|
|
|
namespace aidl::android::hardware::biometrics::fingerprint {
|
|
|
|
Session::Session(int sensorId, int userId, std::shared_ptr<ISessionCallback> cb,
|
|
FakeFingerprintEngine* engine, WorkerThread* worker)
|
|
: mSensorId(sensorId),
|
|
mUserId(userId),
|
|
mCb(std::move(cb)),
|
|
mEngine(engine),
|
|
mWorker(worker),
|
|
mScheduledState(SessionState::IDLING),
|
|
mCurrentState(SessionState::IDLING) {
|
|
CHECK_GE(mSensorId, 0);
|
|
CHECK_GE(mUserId, 0);
|
|
CHECK(mEngine);
|
|
CHECK(mWorker);
|
|
CHECK(mCb);
|
|
}
|
|
|
|
void Session::scheduleStateOrCrash(SessionState state) {
|
|
// TODO(b/166800618): call enterIdling from the terminal callbacks and restore these checks.
|
|
// CHECK(mScheduledState == SessionState::IDLING);
|
|
// CHECK(mCurrentState == SessionState::IDLING);
|
|
mScheduledState = state;
|
|
}
|
|
|
|
void Session::enterStateOrCrash(SessionState state) {
|
|
CHECK(mScheduledState == state);
|
|
mCurrentState = state;
|
|
mScheduledState = SessionState::IDLING;
|
|
}
|
|
|
|
void Session::enterIdling() {
|
|
// TODO(b/166800618): call enterIdling from the terminal callbacks and rethink this conditional.
|
|
if (mCurrentState != SessionState::CLOSED) {
|
|
mCurrentState = SessionState::IDLING;
|
|
}
|
|
}
|
|
|
|
bool Session::isClosed() {
|
|
return mCurrentState == SessionState::CLOSED;
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::generateChallenge() {
|
|
LOG(INFO) << "generateChallenge";
|
|
scheduleStateOrCrash(SessionState::GENERATING_CHALLENGE);
|
|
|
|
mWorker->schedule(Callable::from([this] {
|
|
enterStateOrCrash(SessionState::GENERATING_CHALLENGE);
|
|
mEngine->generateChallengeImpl(mCb.get());
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::revokeChallenge(int64_t challenge) {
|
|
LOG(INFO) << "revokeChallenge";
|
|
scheduleStateOrCrash(SessionState::REVOKING_CHALLENGE);
|
|
|
|
mWorker->schedule(Callable::from([this, challenge] {
|
|
enterStateOrCrash(SessionState::REVOKING_CHALLENGE);
|
|
mEngine->revokeChallengeImpl(mCb.get(), challenge);
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::enroll(const keymaster::HardwareAuthToken& hat,
|
|
std::shared_ptr<common::ICancellationSignal>* out) {
|
|
LOG(INFO) << "enroll";
|
|
scheduleStateOrCrash(SessionState::ENROLLING);
|
|
|
|
std::promise<void> cancellationPromise;
|
|
auto cancFuture = cancellationPromise.get_future();
|
|
|
|
mWorker->schedule(Callable::from([this, hat, cancFuture = std::move(cancFuture)] {
|
|
enterStateOrCrash(SessionState::ENROLLING);
|
|
if (shouldCancel(cancFuture)) {
|
|
mCb->onError(Error::CANCELED, 0 /* vendorCode */);
|
|
} else {
|
|
mEngine->enrollImpl(mCb.get(), hat);
|
|
}
|
|
enterIdling();
|
|
}));
|
|
|
|
*out = SharedRefBase::make<CancellationSignal>(std::move(cancellationPromise));
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::authenticate(int64_t operationId,
|
|
std::shared_ptr<common::ICancellationSignal>* out) {
|
|
LOG(INFO) << "authenticate";
|
|
scheduleStateOrCrash(SessionState::AUTHENTICATING);
|
|
|
|
std::promise<void> cancPromise;
|
|
auto cancFuture = cancPromise.get_future();
|
|
|
|
mWorker->schedule(Callable::from([this, operationId, cancFuture = std::move(cancFuture)] {
|
|
enterStateOrCrash(SessionState::AUTHENTICATING);
|
|
if (shouldCancel(cancFuture)) {
|
|
mCb->onError(Error::CANCELED, 0 /* vendorCode */);
|
|
} else {
|
|
mEngine->authenticateImpl(mCb.get(), operationId);
|
|
}
|
|
enterIdling();
|
|
}));
|
|
|
|
*out = SharedRefBase::make<CancellationSignal>(std::move(cancPromise));
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::detectInteraction(std::shared_ptr<common::ICancellationSignal>* out) {
|
|
LOG(INFO) << "detectInteraction";
|
|
scheduleStateOrCrash(SessionState::DETECTING_INTERACTION);
|
|
|
|
std::promise<void> cancellationPromise;
|
|
auto cancFuture = cancellationPromise.get_future();
|
|
|
|
mWorker->schedule(Callable::from([this, cancFuture = std::move(cancFuture)] {
|
|
enterStateOrCrash(SessionState::DETECTING_INTERACTION);
|
|
if (shouldCancel(cancFuture)) {
|
|
mCb->onError(Error::CANCELED, 0 /* vendorCode */);
|
|
} else {
|
|
mEngine->detectInteractionImpl(mCb.get());
|
|
}
|
|
enterIdling();
|
|
}));
|
|
|
|
*out = SharedRefBase::make<CancellationSignal>(std::move(cancellationPromise));
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::enumerateEnrollments() {
|
|
LOG(INFO) << "enumerateEnrollments";
|
|
scheduleStateOrCrash(SessionState::ENUMERATING_ENROLLMENTS);
|
|
|
|
mWorker->schedule(Callable::from([this] {
|
|
enterStateOrCrash(SessionState::ENUMERATING_ENROLLMENTS);
|
|
mEngine->enumerateEnrollmentsImpl(mCb.get());
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::removeEnrollments(const std::vector<int32_t>& enrollmentIds) {
|
|
LOG(INFO) << "removeEnrollments";
|
|
scheduleStateOrCrash(SessionState::REMOVING_ENROLLMENTS);
|
|
|
|
mWorker->schedule(Callable::from([this, enrollmentIds] {
|
|
enterStateOrCrash(SessionState::REMOVING_ENROLLMENTS);
|
|
mEngine->removeEnrollmentsImpl(mCb.get(), enrollmentIds);
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::getAuthenticatorId() {
|
|
LOG(INFO) << "getAuthenticatorId";
|
|
scheduleStateOrCrash(SessionState::GETTING_AUTHENTICATOR_ID);
|
|
|
|
mWorker->schedule(Callable::from([this] {
|
|
enterStateOrCrash(SessionState::GETTING_AUTHENTICATOR_ID);
|
|
mEngine->getAuthenticatorIdImpl(mCb.get());
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::invalidateAuthenticatorId() {
|
|
LOG(INFO) << "invalidateAuthenticatorId";
|
|
scheduleStateOrCrash(SessionState::INVALIDATING_AUTHENTICATOR_ID);
|
|
|
|
mWorker->schedule(Callable::from([this] {
|
|
enterStateOrCrash(SessionState::INVALIDATING_AUTHENTICATOR_ID);
|
|
mEngine->invalidateAuthenticatorIdImpl(mCb.get());
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::resetLockout(const keymaster::HardwareAuthToken& hat) {
|
|
LOG(INFO) << "resetLockout";
|
|
scheduleStateOrCrash(SessionState::RESETTING_LOCKOUT);
|
|
|
|
mWorker->schedule(Callable::from([this, hat] {
|
|
enterStateOrCrash(SessionState::RESETTING_LOCKOUT);
|
|
mEngine->resetLockoutImpl(mCb.get(), hat);
|
|
enterIdling();
|
|
}));
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::close() {
|
|
LOG(INFO) << "close";
|
|
// TODO(b/166800618): call enterIdling from the terminal callbacks and restore this check.
|
|
// CHECK(mCurrentState == SessionState::IDLING) << "Can't close a non-idling session.
|
|
// Crashing.";
|
|
mCurrentState = SessionState::CLOSED;
|
|
mCb->onSessionClosed();
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::onPointerDown(int32_t /*pointerId*/, int32_t /*x*/, int32_t /*y*/,
|
|
float /*minor*/, float /*major*/) {
|
|
LOG(INFO) << "onPointerDown";
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::onPointerUp(int32_t /*pointerId*/) {
|
|
LOG(INFO) << "onPointerUp";
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus Session::onUiReady() {
|
|
LOG(INFO) << "onUiReady";
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
} // namespace aidl::android::hardware::biometrics::fingerprint
|