389 lines
13 KiB
C++
389 lines
13 KiB
C++
/*
|
|
* Copyright (C) 2019 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.
|
|
*/
|
|
|
|
#ifndef ANDROID_DRMUTILS_H
|
|
#define ANDROID_DRMUTILS_H
|
|
|
|
#include <android/hardware/drm/1.0/ICryptoFactory.h>
|
|
#include <android/hardware/drm/1.0/IDrmFactory.h>
|
|
#include <android/hardware/drm/1.4/IDrmPlugin.h>
|
|
#include <android/hardware/drm/1.4/types.h>
|
|
#include <media/stagefright/MediaErrors.h>
|
|
#include <utils/Errors.h> // for status_t
|
|
#include <utils/Log.h>
|
|
#include <utils/String8.h>
|
|
#include <utils/StrongPointer.h>
|
|
#include <utils/Vector.h>
|
|
#include <algorithm>
|
|
#include <chrono>
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <cstring>
|
|
#include <ctime>
|
|
#include <deque>
|
|
#include <endian.h>
|
|
#include <iterator>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <vector>
|
|
#include <aidl/android/hardware/drm/LogMessage.h>
|
|
#include <aidl/android/hardware/drm/Status.h>
|
|
#include <aidl/android/hardware/drm/IDrmFactory.h>
|
|
|
|
using namespace ::android::hardware::drm;
|
|
using ::android::hardware::hidl_vec;
|
|
using ::android::hardware::Return;
|
|
|
|
using ::aidl::android::hardware::drm::LogPriority;
|
|
using ::aidl::android::hardware::drm::LogMessage;
|
|
using ::aidl::android::hardware::drm::Uuid;
|
|
using StatusAidl = ::aidl::android::hardware::drm::Status;
|
|
using IDrmFactoryAidl = ::aidl::android::hardware::drm::IDrmFactory;
|
|
|
|
namespace android {
|
|
|
|
struct ICrypto;
|
|
struct IDrm;
|
|
|
|
namespace DrmUtils {
|
|
|
|
// Log APIs
|
|
class LogBuffer {
|
|
public:
|
|
static const int MAX_CAPACITY = 100;
|
|
void addLog(const ::V1_4::LogMessage &log);
|
|
Vector<::V1_4::LogMessage> getLogs();
|
|
|
|
private:
|
|
std::deque<::V1_4::LogMessage> mBuffer;
|
|
std::mutex mMutex;
|
|
};
|
|
|
|
extern LogBuffer gLogBuf;
|
|
|
|
static inline int formatBuffer(char *buf, size_t size, const char *msg) {
|
|
return snprintf(buf, size, "%s", msg);
|
|
}
|
|
|
|
template <typename First, typename... Args>
|
|
static inline int formatBuffer(char *buf, size_t size, const char *fmt, First first, Args... args) {
|
|
return snprintf(buf, size, fmt, first, args...);
|
|
}
|
|
|
|
template <typename... Args>
|
|
void LogToBuffer(android_LogPriority level, const char *fmt, Args... args) {
|
|
const int LOG_BUF_SIZE = 256;
|
|
char buf[LOG_BUF_SIZE];
|
|
int len = formatBuffer(buf, LOG_BUF_SIZE, fmt, args...);
|
|
if (len <= 0) {
|
|
return;
|
|
}
|
|
__android_log_write(level, LOG_TAG, buf);
|
|
if (level >= ANDROID_LOG_INFO) {
|
|
int64_t epochTimeMs =
|
|
std::chrono::system_clock::now().time_since_epoch() / std::chrono::milliseconds(1);
|
|
gLogBuf.addLog({epochTimeMs, static_cast<::V1_4::LogPriority>(level), buf});
|
|
}
|
|
}
|
|
|
|
template <typename... Args>
|
|
void LogToBuffer(android_LogPriority level, const uint8_t uuid[16], const char *fmt, Args... args) {
|
|
uint64_t uuid2[2] = {};
|
|
std::memcpy(uuid2, uuid, sizeof(uuid2));
|
|
std::string uuidFmt("uuid=[%lx %lx] ");
|
|
uuidFmt += fmt;
|
|
LogToBuffer(level, uuidFmt.c_str(), htobe64(uuid2[0]), htobe64(uuid2[1]), args...);
|
|
}
|
|
|
|
#ifndef LOG2BE
|
|
#define LOG2BE(...) LogToBuffer(ANDROID_LOG_ERROR, __VA_ARGS__)
|
|
#define LOG2BW(...) LogToBuffer(ANDROID_LOG_WARN, __VA_ARGS__)
|
|
#define LOG2BI(...) LogToBuffer(ANDROID_LOG_INFO, __VA_ARGS__)
|
|
#define LOG2BD(...) LogToBuffer(ANDROID_LOG_DEBUG, __VA_ARGS__)
|
|
#define LOG2BV(...) LogToBuffer(ANDROID_LOG_VERBOSE, __VA_ARGS__)
|
|
#endif
|
|
|
|
bool UseDrmService();
|
|
|
|
sp<IDrm> MakeDrm(status_t *pstatus = nullptr);
|
|
|
|
sp<ICrypto> MakeCrypto(status_t *pstatus = nullptr);
|
|
|
|
template<typename BA, typename PARCEL>
|
|
void WriteByteArray(PARCEL &obj, const BA &vec) {
|
|
obj.writeInt32(vec.size());
|
|
if (vec.size()) {
|
|
obj.write(vec.data(), vec.size());
|
|
}
|
|
}
|
|
|
|
template<typename ET, typename BA, typename PARCEL>
|
|
void WriteEventToParcel(
|
|
PARCEL &obj,
|
|
ET eventType,
|
|
const BA &sessionId,
|
|
const BA &data) {
|
|
WriteByteArray(obj, sessionId);
|
|
WriteByteArray(obj, data);
|
|
obj.writeInt32(eventType);
|
|
}
|
|
|
|
template<typename BA, typename PARCEL>
|
|
void WriteExpirationUpdateToParcel(
|
|
PARCEL &obj,
|
|
const BA &sessionId,
|
|
int64_t expiryTimeInMS) {
|
|
WriteByteArray(obj, sessionId);
|
|
obj.writeInt64(expiryTimeInMS);
|
|
}
|
|
|
|
template<typename BA, typename KSL, typename PARCEL>
|
|
void WriteKeysChange(
|
|
PARCEL &obj,
|
|
const BA &sessionId,
|
|
const KSL &keyStatusList,
|
|
bool hasNewUsableKey) {
|
|
WriteByteArray(obj, sessionId);
|
|
obj.writeInt32(keyStatusList.size());
|
|
for (const auto &keyStatus : keyStatusList) {
|
|
WriteByteArray(obj, keyStatus.keyId);
|
|
obj.writeInt32(keyStatus.type);
|
|
}
|
|
obj.writeInt32(hasNewUsableKey);
|
|
}
|
|
|
|
inline Uuid toAidlUuid(const uint8_t uuid[16]) {
|
|
Uuid uuidAidl;
|
|
for (int i = 0; i < 16; ++i) uuidAidl.uuid[i] = uuid[i];
|
|
return uuidAidl;
|
|
}
|
|
|
|
std::vector<std::shared_ptr<IDrmFactoryAidl>> makeDrmFactoriesAidl();
|
|
|
|
std::vector<sp<::V1_0::IDrmFactory>> MakeDrmFactories(const uint8_t uuid[16] = nullptr);
|
|
|
|
std::vector<sp<::V1_0::IDrmPlugin>> MakeDrmPlugins(const uint8_t uuid[16],
|
|
const char *appPackageName);
|
|
|
|
std::vector<sp<::V1_0::ICryptoFactory>> MakeCryptoFactories(const uint8_t uuid[16]);
|
|
|
|
std::vector<sp<::V1_0::ICryptoPlugin>> MakeCryptoPlugins(const uint8_t uuid[16],
|
|
const void *initData, size_t initDataSize);
|
|
|
|
status_t toStatusT_1_4(::V1_4::Status status);
|
|
|
|
template<typename S>
|
|
inline status_t toStatusT(S status) {
|
|
auto err = static_cast<::V1_4::Status>(status);
|
|
return toStatusT_1_4(err);
|
|
}
|
|
|
|
template<typename T>
|
|
inline status_t toStatusT(const android::hardware::Return<T> &status) {
|
|
auto t = static_cast<T>(status);
|
|
auto err = static_cast<::V1_4::Status>(t);
|
|
return toStatusT_1_4(err);
|
|
}
|
|
|
|
inline status_t statusAidlToStatusT(::ndk::ScopedAStatus &statusAidl) {
|
|
if (statusAidl.isOk()) return OK;
|
|
if (statusAidl.getExceptionCode() != EX_SERVICE_SPECIFIC) return DEAD_OBJECT;
|
|
auto status = static_cast<StatusAidl>(statusAidl.getServiceSpecificError());
|
|
switch (status) {
|
|
case StatusAidl::OK:
|
|
return OK;
|
|
case StatusAidl::BAD_VALUE:
|
|
return BAD_VALUE;
|
|
case StatusAidl::ERROR_DRM_CANNOT_HANDLE:
|
|
return ERROR_DRM_CANNOT_HANDLE;
|
|
case StatusAidl::ERROR_DRM_DECRYPT:
|
|
return ERROR_DRM_DECRYPT;
|
|
case StatusAidl::ERROR_DRM_DEVICE_REVOKED:
|
|
return ERROR_DRM_DEVICE_REVOKED;
|
|
case StatusAidl::ERROR_DRM_FRAME_TOO_LARGE:
|
|
return ERROR_DRM_FRAME_TOO_LARGE;
|
|
case StatusAidl::ERROR_DRM_INSUFFICIENT_OUTPUT_PROTECTION:
|
|
return ERROR_DRM_INSUFFICIENT_OUTPUT_PROTECTION;
|
|
case StatusAidl::ERROR_DRM_INSUFFICIENT_SECURITY:
|
|
return ERROR_DRM_INSUFFICIENT_SECURITY;
|
|
case StatusAidl::ERROR_DRM_INVALID_STATE:
|
|
return ERROR_DRM_INVALID_STATE;
|
|
case StatusAidl::ERROR_DRM_LICENSE_EXPIRED:
|
|
return ERROR_DRM_LICENSE_EXPIRED;
|
|
case StatusAidl::ERROR_DRM_NO_LICENSE:
|
|
return ERROR_DRM_NO_LICENSE;
|
|
case StatusAidl::ERROR_DRM_NOT_PROVISIONED:
|
|
return ERROR_DRM_NOT_PROVISIONED;
|
|
case StatusAidl::ERROR_DRM_RESOURCE_BUSY:
|
|
return ERROR_DRM_RESOURCE_BUSY;
|
|
case StatusAidl::ERROR_DRM_RESOURCE_CONTENTION:
|
|
return ERROR_DRM_RESOURCE_CONTENTION;
|
|
case StatusAidl::ERROR_DRM_SESSION_LOST_STATE:
|
|
return ERROR_DRM_SESSION_LOST_STATE;
|
|
case StatusAidl::ERROR_DRM_SESSION_NOT_OPENED:
|
|
return ERROR_DRM_SESSION_NOT_OPENED;
|
|
|
|
// New in S / drm@1.4:
|
|
case StatusAidl::CANNOT_DECRYPT_ZERO_SUBSAMPLES:
|
|
return ERROR_DRM_ZERO_SUBSAMPLES;
|
|
case StatusAidl::CRYPTO_LIBRARY_ERROR:
|
|
return ERROR_DRM_CRYPTO_LIBRARY;
|
|
case StatusAidl::GENERAL_OEM_ERROR:
|
|
return ERROR_DRM_GENERIC_OEM;
|
|
case StatusAidl::GENERAL_PLUGIN_ERROR:
|
|
return ERROR_DRM_GENERIC_PLUGIN;
|
|
case StatusAidl::INIT_DATA_INVALID:
|
|
return ERROR_DRM_INIT_DATA;
|
|
case StatusAidl::KEY_NOT_LOADED:
|
|
return ERROR_DRM_KEY_NOT_LOADED;
|
|
case StatusAidl::LICENSE_PARSE_ERROR:
|
|
return ERROR_DRM_LICENSE_PARSE;
|
|
case StatusAidl::LICENSE_POLICY_ERROR:
|
|
return ERROR_DRM_LICENSE_POLICY;
|
|
case StatusAidl::LICENSE_RELEASE_ERROR:
|
|
return ERROR_DRM_LICENSE_RELEASE;
|
|
case StatusAidl::LICENSE_REQUEST_REJECTED:
|
|
return ERROR_DRM_LICENSE_REQUEST_REJECTED;
|
|
case StatusAidl::LICENSE_RESTORE_ERROR:
|
|
return ERROR_DRM_LICENSE_RESTORE;
|
|
case StatusAidl::LICENSE_STATE_ERROR:
|
|
return ERROR_DRM_LICENSE_STATE;
|
|
case StatusAidl::MALFORMED_CERTIFICATE:
|
|
return ERROR_DRM_CERTIFICATE_MALFORMED;
|
|
case StatusAidl::MEDIA_FRAMEWORK_ERROR:
|
|
return ERROR_DRM_MEDIA_FRAMEWORK;
|
|
case StatusAidl::MISSING_CERTIFICATE:
|
|
return ERROR_DRM_CERTIFICATE_MISSING;
|
|
case StatusAidl::PROVISIONING_CERTIFICATE_ERROR:
|
|
return ERROR_DRM_PROVISIONING_CERTIFICATE;
|
|
case StatusAidl::PROVISIONING_CONFIGURATION_ERROR:
|
|
return ERROR_DRM_PROVISIONING_CONFIG;
|
|
case StatusAidl::PROVISIONING_PARSE_ERROR:
|
|
return ERROR_DRM_PROVISIONING_PARSE;
|
|
case StatusAidl::PROVISIONING_REQUEST_REJECTED:
|
|
return ERROR_DRM_PROVISIONING_REQUEST_REJECTED;
|
|
case StatusAidl::RETRYABLE_PROVISIONING_ERROR:
|
|
return ERROR_DRM_PROVISIONING_RETRY;
|
|
case StatusAidl::SECURE_STOP_RELEASE_ERROR:
|
|
return ERROR_DRM_SECURE_STOP_RELEASE;
|
|
case StatusAidl::STORAGE_READ_FAILURE:
|
|
return ERROR_DRM_STORAGE_READ;
|
|
case StatusAidl::STORAGE_WRITE_FAILURE:
|
|
return ERROR_DRM_STORAGE_WRITE;
|
|
|
|
case StatusAidl::ERROR_DRM_UNKNOWN:
|
|
default:
|
|
return ERROR_DRM_UNKNOWN;
|
|
}
|
|
return ERROR_DRM_UNKNOWN;
|
|
}
|
|
|
|
template<typename T, typename U>
|
|
status_t GetLogMessagesAidl(const std::shared_ptr<U> &obj, Vector<::V1_4::LogMessage> &logs) {
|
|
std::shared_ptr<T> plugin = obj;
|
|
if (obj == NULL) {
|
|
LOG2BW("%s obj is null", U::descriptor);
|
|
} else if (plugin == NULL) {
|
|
LOG2BW("Cannot cast %s obj to %s plugin", U::descriptor, T::descriptor);
|
|
}
|
|
|
|
std::vector<LogMessage> pluginLogsAidl;
|
|
if (plugin != NULL) {
|
|
if(!plugin->getLogMessages(&pluginLogsAidl).isOk()) {
|
|
LOG2BW("%s::getLogMessages remote call failed", T::descriptor);
|
|
}
|
|
}
|
|
|
|
std::vector<::V1_4::LogMessage> pluginLogs;
|
|
for (LogMessage log : pluginLogsAidl) {
|
|
::V1_4::LogMessage logHidl;
|
|
logHidl.timeMs = log.timeMs;
|
|
// skip negative convert check as count of enum elements is 7
|
|
logHidl.priority = static_cast<::V1_4::LogPriority>((int32_t)log.priority);
|
|
logHidl.message = log.message;
|
|
pluginLogs.push_back(logHidl);
|
|
}
|
|
|
|
auto allLogs(gLogBuf.getLogs());
|
|
LOG2BD("framework logs size %zu; plugin logs size %zu",
|
|
allLogs.size(), pluginLogs.size());
|
|
std::copy(pluginLogs.begin(), pluginLogs.end(), std::back_inserter(allLogs));
|
|
std::sort(allLogs.begin(), allLogs.end(),
|
|
[](const ::V1_4::LogMessage &a, const ::V1_4::LogMessage &b) {
|
|
return a.timeMs < b.timeMs;
|
|
});
|
|
|
|
logs.appendVector(allLogs);
|
|
return OK;
|
|
}
|
|
|
|
template<typename T, typename U>
|
|
status_t GetLogMessages(const sp<U> &obj, Vector<::V1_4::LogMessage> &logs) {
|
|
sp<T> plugin = T::castFrom(obj);
|
|
if (obj == NULL) {
|
|
LOG2BW("%s obj is null", U::descriptor);
|
|
} else if (plugin == NULL) {
|
|
LOG2BW("Cannot cast %s obj to %s plugin", U::descriptor, T::descriptor);
|
|
}
|
|
|
|
::V1_4::Status err{};
|
|
std::vector<::V1_4::LogMessage> pluginLogs;
|
|
::V1_4::IDrmPlugin::getLogMessages_cb cb = [&](
|
|
::V1_4::Status status,
|
|
hidl_vec<::V1_4::LogMessage> hLogs) {
|
|
if (::V1_4::Status::OK != status) {
|
|
err = status;
|
|
return;
|
|
}
|
|
pluginLogs.assign(hLogs.data(), hLogs.data() + hLogs.size());
|
|
};
|
|
|
|
Return<void> hResult;
|
|
if (plugin != NULL) {
|
|
hResult = plugin->getLogMessages(cb);
|
|
}
|
|
if (!hResult.isOk()) {
|
|
LOG2BW("%s::getLogMessages remote call failed %s",
|
|
T::descriptor, hResult.description().c_str());
|
|
}
|
|
|
|
auto allLogs(gLogBuf.getLogs());
|
|
LOG2BD("framework logs size %zu; plugin logs size %zu",
|
|
allLogs.size(), pluginLogs.size());
|
|
std::copy(pluginLogs.begin(), pluginLogs.end(), std::back_inserter(allLogs));
|
|
std::sort(allLogs.begin(), allLogs.end(),
|
|
[](const ::V1_4::LogMessage &a, const ::V1_4::LogMessage &b) {
|
|
return a.timeMs < b.timeMs;
|
|
});
|
|
|
|
logs.appendVector(allLogs);
|
|
return OK;
|
|
}
|
|
|
|
std::string GetExceptionMessage(status_t err, const char *msg,
|
|
const Vector<::V1_4::LogMessage> &logs);
|
|
|
|
template<typename T>
|
|
std::string GetExceptionMessage(status_t err, const char *msg, const sp<T> &iface) {
|
|
Vector<::V1_4::LogMessage> logs;
|
|
iface->getLogMessages(logs);
|
|
return GetExceptionMessage(err, msg, logs);
|
|
}
|
|
|
|
} // namespace DrmUtils
|
|
} // namespace android
|
|
#endif // ANDROID_DRMUTILS_H
|