d3ed3a2079
This reverts commit 76f21b2676
.
Reason for revert: b/36637075
Bug: 36637075
Change-Id: Ica737cf96d14086aae7918f8bf2f86a36555d03b
158 lines
5.3 KiB
C++
158 lines
5.3 KiB
C++
/*
|
|
* Copyright (C) 2015 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.
|
|
*/
|
|
#define LOG_TAG "KeystoreOperation"
|
|
|
|
#include "operation.h"
|
|
|
|
#include <algorithm>
|
|
|
|
namespace keystore {
|
|
using namespace android;
|
|
|
|
OperationMap::OperationMap(IBinder::DeathRecipient* deathRecipient)
|
|
: mDeathRecipient(deathRecipient) {}
|
|
|
|
sp<IBinder> OperationMap::addOperation(uint64_t handle, uint64_t keyid, KeyPurpose purpose,
|
|
const OperationMap::km_device_t& dev,
|
|
const sp<IBinder>& appToken,
|
|
KeyCharacteristics&& characteristics, bool pruneable) {
|
|
sp<IBinder> token = new BBinder();
|
|
mMap[token] = Operation(handle, keyid, purpose, dev, std::move(characteristics), appToken);
|
|
if (pruneable) {
|
|
mLru.push_back(token);
|
|
}
|
|
if (mAppTokenMap.find(appToken) == mAppTokenMap.end()) {
|
|
appToken->linkToDeath(mDeathRecipient);
|
|
}
|
|
mAppTokenMap[appToken].push_back(token);
|
|
return token;
|
|
}
|
|
|
|
bool OperationMap::getOperation(const sp<IBinder>& token, uint64_t* outHandle, uint64_t* outKeyid,
|
|
KeyPurpose* outPurpose, km_device_t* outDevice,
|
|
const KeyCharacteristics** outCharacteristics) {
|
|
if (!outHandle || !outDevice) {
|
|
return false;
|
|
}
|
|
auto entry = mMap.find(token);
|
|
if (entry == mMap.end()) {
|
|
return false;
|
|
}
|
|
updateLru(token);
|
|
|
|
*outHandle = entry->second.handle;
|
|
*outKeyid = entry->second.keyid;
|
|
*outPurpose = entry->second.purpose;
|
|
*outDevice = entry->second.device;
|
|
if (outCharacteristics) {
|
|
*outCharacteristics = &entry->second.characteristics;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void OperationMap::updateLru(const sp<IBinder>& token) {
|
|
auto lruEntry = std::find(mLru.begin(), mLru.end(), token);
|
|
if (lruEntry != mLru.end()) {
|
|
mLru.erase(lruEntry);
|
|
mLru.push_back(token);
|
|
}
|
|
}
|
|
|
|
bool OperationMap::removeOperation(const sp<IBinder>& token) {
|
|
auto entry = mMap.find(token);
|
|
if (entry == mMap.end()) {
|
|
return false;
|
|
}
|
|
sp<IBinder> appToken = entry->second.appToken;
|
|
mMap.erase(entry);
|
|
auto lruEntry = std::find(mLru.begin(), mLru.end(), token);
|
|
if (lruEntry != mLru.end()) {
|
|
mLru.erase(lruEntry);
|
|
}
|
|
removeOperationTracking(token, appToken);
|
|
return true;
|
|
}
|
|
|
|
void OperationMap::removeOperationTracking(const sp<IBinder>& token, const sp<IBinder>& appToken) {
|
|
auto appEntry = mAppTokenMap.find(appToken);
|
|
if (appEntry == mAppTokenMap.end()) {
|
|
ALOGE("Entry for %p contains unmapped application token %p", token.get(), appToken.get());
|
|
return;
|
|
}
|
|
auto tokenEntry = std::find(appEntry->second.begin(), appEntry->second.end(), token);
|
|
appEntry->second.erase(tokenEntry);
|
|
// Stop listening for death if all operations tied to the token have finished.
|
|
if (appEntry->second.size() == 0) {
|
|
appToken->unlinkToDeath(mDeathRecipient);
|
|
mAppTokenMap.erase(appEntry);
|
|
}
|
|
}
|
|
|
|
bool OperationMap::hasPruneableOperation() const {
|
|
return mLru.size() != 0;
|
|
}
|
|
|
|
size_t OperationMap::getPruneableOperationCount() const {
|
|
return mLru.size();
|
|
}
|
|
|
|
sp<IBinder> OperationMap::getOldestPruneableOperation() {
|
|
if (!hasPruneableOperation()) {
|
|
return sp<IBinder>(NULL);
|
|
}
|
|
return mLru[0];
|
|
}
|
|
|
|
bool OperationMap::getOperationAuthToken(const sp<IBinder>& token,
|
|
const HardwareAuthToken** outToken) {
|
|
auto entry = mMap.find(token);
|
|
if (entry == mMap.end()) {
|
|
return false;
|
|
}
|
|
*outToken = entry->second.authToken.get();
|
|
return true;
|
|
}
|
|
|
|
bool OperationMap::setOperationAuthToken(const sp<IBinder>& token,
|
|
const HardwareAuthToken* authToken) {
|
|
auto entry = mMap.find(token);
|
|
if (entry == mMap.end()) {
|
|
return false;
|
|
}
|
|
entry->second.authToken.reset(new HardwareAuthToken);
|
|
*entry->second.authToken = *authToken;
|
|
return true;
|
|
}
|
|
|
|
std::vector<sp<IBinder>> OperationMap::getOperationsForToken(const sp<IBinder>& appToken) {
|
|
auto appEntry = mAppTokenMap.find(appToken);
|
|
if (appEntry != mAppTokenMap.end()) {
|
|
return appEntry->second;
|
|
} else {
|
|
return std::vector<sp<IBinder>>();
|
|
}
|
|
}
|
|
|
|
OperationMap::Operation::Operation(uint64_t handle_, uint64_t keyid_, KeyPurpose purpose_,
|
|
const OperationMap::km_device_t& device_,
|
|
KeyCharacteristics&& characteristics_, sp<IBinder> appToken_)
|
|
: handle(handle_), keyid(keyid_), purpose(purpose_), device(device_),
|
|
characteristics(characteristics_), appToken(appToken_) {}
|
|
|
|
OperationMap::Operation::Operation()
|
|
: handle(0), keyid(0), device(nullptr), characteristics(), appToken(nullptr) {}
|
|
|
|
} // namespace android
|