#define LOG_TAG "hwservicemanager"
#include "TokenManager.h"
#include <android-base/logging.h>
#include <functional>
#include <log/log.h>
#include <openssl/hmac.h>
#include <openssl/rand.h>
namespace android {
namespace hidl {
namespace token {
namespace V1_0 {
namespace implementation {
static void ReadRandomBytes(uint8_t *buf, size_t len) {
int fd = TEMP_FAILURE_RETRY(open("/dev/urandom", O_RDONLY | O_CLOEXEC | O_NOFOLLOW));
if (fd == -1) {
ALOGE("%s: cannot read /dev/urandom", __func__);
return;
}
size_t n;
while ((n = TEMP_FAILURE_RETRY(read(fd, buf, len))) > 0) {
len -= n;
buf += n;
}
if (len > 0) {
ALOGW("%s: there are %d bytes skipped", __func__, (int)len);
}
close(fd);
}
TokenManager::TokenManager() {
ReadRandomBytes(mKey.data(), mKey.size());
}
// Methods from ::android::hidl::token::V1_0::ITokenManager follow.
Return<void> TokenManager::createToken(const sp<IBase>& store, createToken_cb hidl_cb) {
TokenInterface interface = generateToken(store);
if (interface.interface == nullptr) {
hidl_cb({});
return Void();
}
uint64_t id = getTokenId(interface.token);
if (id == TOKEN_ID_NONE) {
hidl_cb({});
return Void();
}
mMap[id] = interface;
hidl_cb(interface.token);
return Void();
}
std::unordered_map<uint64_t, TokenManager::TokenInterface>::const_iterator
TokenManager::lookupToken(const hidl_vec<uint8_t> &token) {
uint64_t tokenId = getTokenId(token);
if (tokenId == TOKEN_ID_NONE) {
return mMap.end();
}
auto it = mMap.find(tokenId);
if (it == mMap.end()) {
return mMap.end();
}
const TokenInterface &interface = it->second;
if (!constantTimeCompare(token, interface.token)) {
ALOGE("Fetch of token with invalid hash.");
return mMap.end();
}
return it;
}
Return<bool> TokenManager::unregister(const hidl_vec<uint8_t> &token) {
auto it = lookupToken(token);
if (it == mMap.end()) {
return false;
}
mMap.erase(it);
return true;
}
Return<sp<IBase>> TokenManager::get(const hidl_vec<uint8_t> &token) {
auto it = lookupToken(token);
if (it == mMap.end()) {
return nullptr;
}
return it->second.interface;
}
TokenManager::TokenInterface TokenManager::generateToken(const sp<IBase> &interface) {
uint64_t id = ++mTokenIndex;
std::array<uint8_t, EVP_MAX_MD_SIZE> hmac;
uint32_t hmacSize;
uint8_t *hmacOut = HMAC(EVP_sha256(),
mKey.data(), mKey.size(),
(uint8_t*) &id, ID_SIZE,
hmac.data(), &hmacSize);
if (hmacOut == nullptr ||
hmacOut != hmac.data()) {
ALOGE("Generating token failed, got %p.", hmacOut);
return { nullptr, {} };
}
// only care about the first HMAC_SIZE bytes of the HMAC
const hidl_vec<uint8_t> &token = TokenManager::getToken(id, hmac.data(), hmacSize);
return { interface, token };
}
// static
__attribute__((optnone))
bool TokenManager::constantTimeCompare(const hidl_vec<uint8_t> &t1, const hidl_vec<uint8_t> &t2) {
if (t1.size() != t2.size()) {
return false;
}
uint8_t x = 0;
for (size_t i = 0; i < t1.size(); i++) {
x |= t1[i] ^ t2[i];
}
return x == 0;
}
// static
uint64_t TokenManager::getTokenId(const hidl_vec<uint8_t> &token) {
if (token.size() < ID_SIZE) {
return TOKEN_ID_NONE;
}
uint64_t id = 0;
for (size_t i = 0; i < ID_SIZE; i++) {
id |= token[i] << i;
}
return id;
}
// static
hidl_vec<uint8_t> TokenManager::getToken(const uint64_t id, const uint8_t *hmac, uint64_t hmacSize) {
hidl_vec<uint8_t> token;
token.resize(ID_SIZE + hmacSize);
for (size_t i = 0; i < ID_SIZE; i++) {
token[i] = (id >> i) & 0xFF;
}
for (size_t i = 0; i < hmacSize; i++) {
token[i + ID_SIZE] = hmac[i];
}
return token;
}
} // namespace implementation
} // namespace V1_0
} // namespace token
} // namespace hidl
} // namespace android