blob: fd75dbd3f98810c11d773df0b3aac52c292e1f7d [file] [log] [blame]
Jeff Tinkerb075caa2016-12-06 23:15:20 -08001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
Jeff Tinker972a3e32017-01-23 14:02:50 -080016#define LOG_TAG "android.hardware.drm@1.0-impl"
Jeff Tinkerb075caa2016-12-06 23:15:20 -080017
18#include "CryptoPlugin.h"
19#include "TypeConvert.h"
20
Jeff Tinkerb075caa2016-12-06 23:15:20 -080021#include <android/hidl/memory/1.0/IMemory.h>
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -080022#include <hidlmemory/mapping.h>
Steven Moreland3eb7df72017-04-06 12:15:23 -070023#include <log/log.h>
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -080024#include <media/stagefright/foundation/AString.h>
Jeff Tinkerb075caa2016-12-06 23:15:20 -080025
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -080026using android::hardware::hidl_memory;
Jeff Tinkerb075caa2016-12-06 23:15:20 -080027using android::hidl::memory::V1_0::IMemory;
28
Jeff Tinkerb075caa2016-12-06 23:15:20 -080029namespace android {
30namespace hardware {
31namespace drm {
Jeff Tinkerb075caa2016-12-06 23:15:20 -080032namespace V1_0 {
33namespace implementation {
34
Jeff Tinkerda002fe2017-01-19 14:41:11 -080035 // Methods from ::android::hardware::drm::V1_0::ICryptoPlugin follow
Jeff Tinkerb075caa2016-12-06 23:15:20 -080036 Return<bool> CryptoPlugin::requiresSecureDecoderComponent(
37 const hidl_string& mime) {
Scott Randolph89978802017-04-03 14:06:19 -070038 return mLegacyPlugin->requiresSecureDecoderComponent(mime.c_str());
Jeff Tinkerb075caa2016-12-06 23:15:20 -080039 }
40
41 Return<void> CryptoPlugin::notifyResolution(uint32_t width,
42 uint32_t height) {
43 mLegacyPlugin->notifyResolution(width, height);
44 return Void();
45 }
46
47 Return<Status> CryptoPlugin::setMediaDrmSession(
48 const hidl_vec<uint8_t>& sessionId) {
49 return toStatus(mLegacyPlugin->setMediaDrmSession(toVector(sessionId)));
50 }
51
Jeff Tinker0b3f41e2017-02-16 12:20:30 -080052 Return<void> CryptoPlugin::setSharedBufferBase(const hidl_memory& base,
53 uint32_t bufferId) {
Edwin Wong3b8a9ed2017-07-17 09:53:31 -070054 sp<IMemory> hidlMemory = mapMemory(base);
55 ALOGE_IF(hidlMemory == nullptr, "mapMemory returns nullptr");
56
57 // allow mapMemory to return nullptr
58 mSharedBufferMap[bufferId] = hidlMemory;
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -080059 return Void();
60 }
61
Jeff Tinkerb075caa2016-12-06 23:15:20 -080062 Return<void> CryptoPlugin::decrypt(bool secure,
63 const hidl_array<uint8_t, 16>& keyId,
64 const hidl_array<uint8_t, 16>& iv, Mode mode,
65 const Pattern& pattern, const hidl_vec<SubSample>& subSamples,
Jeff Tinkerda002fe2017-01-19 14:41:11 -080066 const SharedBuffer& source, uint64_t offset,
Jeff Tinker6fdbe862017-01-11 19:45:23 -080067 const DestinationBuffer& destination,
Jeff Tinkerb075caa2016-12-06 23:15:20 -080068 decrypt_cb _hidl_cb) {
69
Jeff Tinker0b3f41e2017-02-16 12:20:30 -080070 if (mSharedBufferMap.find(source.bufferId) == mSharedBufferMap.end()) {
71 _hidl_cb(Status::ERROR_DRM_CANNOT_HANDLE, 0, "source decrypt buffer base not set");
Jeff Tinkerdc8e2d02017-01-23 14:24:12 -080072 return Void();
73 }
74
Jeff Tinker0b3f41e2017-02-16 12:20:30 -080075 if (destination.type == BufferType::SHARED_MEMORY) {
76 const SharedBuffer& dest = destination.nonsecureMemory;
77 if (mSharedBufferMap.find(dest.bufferId) == mSharedBufferMap.end()) {
78 _hidl_cb(Status::ERROR_DRM_CANNOT_HANDLE, 0, "destination decrypt buffer base not set");
79 return Void();
80 }
81 }
82
Jeff Tinkerb075caa2016-12-06 23:15:20 -080083 android::CryptoPlugin::Mode legacyMode;
84 switch(mode) {
85 case Mode::UNENCRYPTED:
86 legacyMode = android::CryptoPlugin::kMode_Unencrypted;
87 break;
88 case Mode::AES_CTR:
89 legacyMode = android::CryptoPlugin::kMode_AES_CTR;
90 break;
91 case Mode::AES_CBC_CTS:
92 legacyMode = android::CryptoPlugin::kMode_AES_WV;
93 break;
94 case Mode::AES_CBC:
95 legacyMode = android::CryptoPlugin::kMode_AES_CBC;
96 break;
97 }
98 android::CryptoPlugin::Pattern legacyPattern;
99 legacyPattern.mEncryptBlocks = pattern.encryptBlocks;
100 legacyPattern.mSkipBlocks = pattern.skipBlocks;
101
102 android::CryptoPlugin::SubSample *legacySubSamples =
103 new android::CryptoPlugin::SubSample[subSamples.size()];
104
105 for (size_t i = 0; i < subSamples.size(); i++) {
106 legacySubSamples[i].mNumBytesOfClearData
107 = subSamples[i].numBytesOfClearData;
108 legacySubSamples[i].mNumBytesOfEncryptedData
109 = subSamples[i].numBytesOfEncryptedData;
110 }
111
112 AString detailMessage;
Jeff Tinker0b3f41e2017-02-16 12:20:30 -0800113 sp<IMemory> sourceBase = mSharedBufferMap[source.bufferId];
Edwin Wong3b8a9ed2017-07-17 09:53:31 -0700114 if (sourceBase == nullptr) {
115 _hidl_cb(Status::ERROR_DRM_CANNOT_HANDLE, 0, "source is a nullptr");
116 return Void();
117 }
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800118
Jeff Tinker0b3f41e2017-02-16 12:20:30 -0800119 if (source.offset + offset + source.size > sourceBase->getSize()) {
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -0800120 _hidl_cb(Status::ERROR_DRM_CANNOT_HANDLE, 0, "invalid buffer size");
121 return Void();
122 }
Jeff Tinker6fdbe862017-01-11 19:45:23 -0800123
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -0800124 uint8_t *base = static_cast<uint8_t *>
Jeff Tinker0b3f41e2017-02-16 12:20:30 -0800125 (static_cast<void *>(sourceBase->getPointer()));
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -0800126 void *srcPtr = static_cast<void *>(base + source.offset + offset);
Jeff Tinker6fdbe862017-01-11 19:45:23 -0800127
Jeff Tinker6fdbe862017-01-11 19:45:23 -0800128 void *destPtr = NULL;
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800129 if (destination.type == BufferType::SHARED_MEMORY) {
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -0800130 const SharedBuffer& destBuffer = destination.nonsecureMemory;
Jeff Tinker0b3f41e2017-02-16 12:20:30 -0800131 sp<IMemory> destBase = mSharedBufferMap[destBuffer.bufferId];
Edwin Wong3b8a9ed2017-07-17 09:53:31 -0700132 if (destBase == nullptr) {
133 _hidl_cb(Status::ERROR_DRM_CANNOT_HANDLE, 0, "destination is a nullptr");
134 return Void();
135 }
136
Jeff Tinker0b3f41e2017-02-16 12:20:30 -0800137 if (destBuffer.offset + destBuffer.size > destBase->getSize()) {
Jeff Tinkerf21cdaf2017-01-18 11:49:27 -0800138 _hidl_cb(Status::ERROR_DRM_CANNOT_HANDLE, 0, "invalid buffer size");
139 return Void();
140 }
141 destPtr = static_cast<void *>(base + destination.nonsecureMemory.offset);
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800142 } else if (destination.type == BufferType::NATIVE_HANDLE) {
143 native_handle_t *handle = const_cast<native_handle_t *>(
144 destination.secureMemory.getNativeHandle());
145 destPtr = static_cast<void *>(handle);
146 }
147 ssize_t result = mLegacyPlugin->decrypt(secure, keyId.data(), iv.data(),
Jeff Tinker6fdbe862017-01-11 19:45:23 -0800148 legacyMode, legacyPattern, srcPtr, legacySubSamples,
149 subSamples.size(), destPtr, &detailMessage);
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800150
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800151 delete[] legacySubSamples;
152
153 uint32_t status;
154 uint32_t bytesWritten;
155
156 if (result >= 0) {
157 status = android::OK;
158 bytesWritten = result;
159 } else {
Rahul Friasabd4e112017-02-27 19:17:30 -0800160 status = result;
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800161 bytesWritten = 0;
162 }
163
Jeff Tinker01f0a5a2017-01-12 09:22:18 -0800164 _hidl_cb(toStatus(status), bytesWritten, detailMessage.c_str());
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800165 return Void();
166 }
167
168} // namespace implementation
169} // namespace V1_0
Jeff Tinkerb075caa2016-12-06 23:15:20 -0800170} // namespace drm
171} // namespace hardware
172} // namespace android