blob: de1524468e65a3baba705fde1c135f08ac065ee5 [file] [log] [blame]
/*
* Copyright (C) 2017 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_NDEBUG 0
#define LOG_TAG "CasImpl"
#include <android/media/ICasListener.h>
#include <media/cas/CasAPI.h>
#include <media/CasImpl.h>
#include <media/SharedLibrary.h>
#include <utils/Log.h>
namespace android {
static Status getBinderStatus(status_t err) {
if (err == OK) {
return Status::ok();
}
if (err == BAD_VALUE) {
return Status::fromExceptionCode(Status::EX_ILLEGAL_ARGUMENT);
}
if (err == INVALID_OPERATION) {
return Status::fromExceptionCode(Status::EX_ILLEGAL_STATE);
}
return Status::fromServiceSpecificError(err);
}
static String8 sessionIdToString(const CasSessionId &sessionId) {
String8 result;
for (size_t i = 0; i < sessionId.size(); i++) {
result.appendFormat("%02x ", sessionId[i]);
}
if (result.isEmpty()) {
result.append("(null)");
}
return result;
}
CasImpl::CasImpl(const sp<ICasListener> &listener)
: mPlugin(NULL), mListener(listener) {
ALOGV("CTOR: mPlugin=%p", mPlugin);
}
CasImpl::~CasImpl() {
ALOGV("DTOR: mPlugin=%p", mPlugin);
release();
}
//static
void CasImpl::OnEvent(
void *appData,
int32_t event,
int32_t arg,
uint8_t *data,
size_t size) {
if (appData == NULL) {
ALOGE("Invalid appData!");
return;
}
CasImpl *casImpl = static_cast<CasImpl *>(appData);
casImpl->onEvent(event, arg, data, size);
}
void CasImpl::init(const sp<SharedLibrary>& library, CasPlugin *plugin) {
mLibrary = library;
mPlugin = plugin;
}
void CasImpl::onEvent(
int32_t event, int32_t arg, uint8_t *data, size_t size) {
if (mListener == NULL) {
return;
}
std::unique_ptr<CasData> eventData;
if (data != NULL && size > 0) {
eventData.reset(new CasData(data, data + size));
}
mListener->onEvent(event, arg, eventData);
}
Status CasImpl::setPrivateData(const CasData& pvtData) {
ALOGV("setPrivateData");
return getBinderStatus(mPlugin->setPrivateData(pvtData));
}
Status CasImpl::openSession(int32_t program_number, CasSessionId* sessionId) {
ALOGV("openSession: program_number=%d", program_number);
status_t err = mPlugin->openSession(program_number, sessionId);
ALOGV("openSession: session opened for program_number=%d, sessionId=%s",
program_number, sessionIdToString(*sessionId).string());
return getBinderStatus(err);
}
Status CasImpl::openSessionForStream(
int32_t program_number,
int32_t elementary_PID,
CasSessionId* sessionId) {
ALOGV("openSession: program_number=%d, elementary_PID=%d",
program_number, elementary_PID);
status_t err = mPlugin->openSession(
program_number, elementary_PID, sessionId);
ALOGV("openSession: session opened for "
"program_number=%d, elementary_PID=%d, sessionId=%s",
program_number, elementary_PID,
sessionIdToString(*sessionId).string());
return getBinderStatus(err);
}
Status CasImpl::setSessionPrivateData(
const CasSessionId &sessionId, const CasData& pvtData) {
ALOGV("setSessionPrivateData: sessionId=%s",
sessionIdToString(sessionId).string());
return getBinderStatus(mPlugin->setSessionPrivateData(sessionId, pvtData));
}
Status CasImpl::closeSession(const CasSessionId &sessionId) {
ALOGV("closeSession: sessionId=%s",
sessionIdToString(sessionId).string());
return getBinderStatus(mPlugin->closeSession(sessionId));
}
Status CasImpl::processEcm(const CasSessionId &sessionId, const ParcelableCasData& ecm) {
ALOGV("processEcm: sessionId=%s",
sessionIdToString(sessionId).string());
return getBinderStatus(mPlugin->processEcm(sessionId, ecm));
}
Status CasImpl::processEmm(const ParcelableCasData& emm) {
ALOGV("processEmm");
return getBinderStatus(mPlugin->processEmm(emm));
}
Status CasImpl::sendEvent(
int32_t event, int32_t arg, const ::std::unique_ptr<CasData> &eventData) {
ALOGV("sendEvent");
status_t err;
if (eventData == nullptr) {
err = mPlugin->sendEvent(event, arg, CasData());
} else {
err = mPlugin->sendEvent(event, arg, *eventData);
}
return getBinderStatus(err);
}
Status CasImpl::provision(const String16& provisionString) {
ALOGV("provision: provisionString=%s", String8(provisionString).string());
return getBinderStatus(mPlugin->provision(String8(provisionString)));
}
Status CasImpl::refreshEntitlements(
int32_t refreshType, const ::std::unique_ptr<CasData> &refreshData) {
ALOGV("refreshEntitlements");
status_t err;
if (refreshData == nullptr) {
err = mPlugin->refreshEntitlements(refreshType, CasData());
} else {
err = mPlugin->refreshEntitlements(refreshType, *refreshData);
}
return getBinderStatus(err);
}
Status CasImpl::release() {
ALOGV("release: mPlugin=%p", mPlugin);
if (mPlugin != NULL) {
delete mPlugin;
mPlugin = NULL;
}
return Status::ok();
}
} // namespace android