1 /*
2 * Copyright (c) 2022-2023 Shenzhen Kaihong DID Co., Ltd..
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "codec_share_buffer.h"
17 #include <hdf_base.h>
18 #include <securec.h>
19 #include <unistd.h>
20 #include "codec_log_wrapper.h"
21
22 namespace OHOS {
23 namespace Codec {
24 namespace Omx {
CodecShareBuffer(struct OmxCodecBuffer & codecBuffer)25 CodecShareBuffer::CodecShareBuffer(struct OmxCodecBuffer &codecBuffer) : ICodecBuffer(codecBuffer)
26 {}
27
~CodecShareBuffer()28 CodecShareBuffer::~CodecShareBuffer()
29 {
30 if (shMem_ != nullptr) {
31 shMem_->UnmapAshmem();
32 shMem_->CloseAshmem();
33 shMem_ = nullptr;
34 }
35 }
36
SetAshMem(std::shared_ptr<OHOS::Ashmem> shMem)37 void CodecShareBuffer::SetAshMem(std::shared_ptr<OHOS::Ashmem> shMem)
38 {
39 shMem_ = shMem;
40 }
41
Create(struct OmxCodecBuffer & codecBuffer)42 OHOS::sptr<ICodecBuffer> CodecShareBuffer::Create(struct OmxCodecBuffer &codecBuffer)
43 {
44 int shardFd = (int)reinterpret_cast<uintptr_t>(codecBuffer.buffer);
45 if (shardFd < 0) {
46 CODEC_LOGE("shardFd < 0");
47 return nullptr;
48 }
49 int size = OHOS::AshmemGetSize(shardFd);
50 std::shared_ptr<OHOS::Ashmem> sharedMem = std::make_shared<OHOS::Ashmem>(shardFd, size);
51 bool mapd = false;
52 if (codecBuffer.type == READ_WRITE_TYPE) {
53 mapd = sharedMem->MapReadAndWriteAshmem();
54 } else {
55 mapd = sharedMem->MapReadOnlyAshmem();
56 }
57 if (!mapd) {
58 CODEC_LOGE("MapReadAndWriteAshmem or MapReadOnlyAshmem return false");
59 return nullptr;
60 }
61
62 codecBuffer.buffer = nullptr;
63 codecBuffer.bufferLen = 0;
64 CodecShareBuffer *buffer = new CodecShareBuffer(codecBuffer);
65 buffer->SetAshMem(sharedMem);
66
67 return OHOS::sptr<ICodecBuffer>(buffer);
68 }
69
Allocate(struct OmxCodecBuffer & codecBuffer)70 OHOS::sptr<ICodecBuffer> CodecShareBuffer::Allocate(struct OmxCodecBuffer &codecBuffer)
71 {
72 codecBuffer.bufferType = CODEC_BUFFER_TYPE_AVSHARE_MEM_FD;
73 // create shared memory
74 int sharedFD = AshmemCreate(nullptr, codecBuffer.allocLen);
75
76 std::shared_ptr<Ashmem> sharedMemory = std::make_shared<Ashmem>(sharedFD, codecBuffer.allocLen);
77 codecBuffer.type = READ_WRITE_TYPE;
78 bool mapd = false;
79 if (codecBuffer.type == READ_WRITE_TYPE) {
80 mapd = sharedMemory->MapReadAndWriteAshmem();
81 } else {
82 mapd = sharedMemory->MapReadOnlyAshmem();
83 }
84 if (!mapd) {
85 CODEC_LOGE("MapReadAndWriteAshmem or MapReadOnlyAshmem return false");
86 return nullptr;
87 }
88 codecBuffer.offset = 0;
89 codecBuffer.filledLen = 0;
90
91 CodecShareBuffer *buffer = new CodecShareBuffer(codecBuffer);
92 codecBuffer.buffer = reinterpret_cast<uint8_t *>(sharedFD);
93 codecBuffer.bufferLen = sizeof(int);
94 buffer->SetAshMem(sharedMemory);
95 return OHOS::sptr<ICodecBuffer>(buffer);
96 }
97
FillOmxBuffer(struct OmxCodecBuffer & codecBuffer,OMX_BUFFERHEADERTYPE & omxBuffer)98 int32_t CodecShareBuffer::FillOmxBuffer(struct OmxCodecBuffer &codecBuffer, OMX_BUFFERHEADERTYPE &omxBuffer)
99 {
100 if (!CheckInvalid(codecBuffer) || codecBuffer_.type != READ_WRITE_TYPE) {
101 CODEC_LOGE("CheckInvalid return false or mem has no right to write ");
102 return HDF_ERR_INVALID_PARAM;
103 }
104
105 ReleaseFd(codecBuffer);
106
107 return ICodecBuffer::FillOmxBuffer(codecBuffer, omxBuffer);
108 }
109
EmptyOmxBuffer(struct OmxCodecBuffer & codecBuffer,OMX_BUFFERHEADERTYPE & omxBuffer)110 int32_t CodecShareBuffer::EmptyOmxBuffer(struct OmxCodecBuffer &codecBuffer, OMX_BUFFERHEADERTYPE &omxBuffer)
111 {
112 if (!CheckInvalid(codecBuffer)) {
113 CODEC_LOGE("shMem_ is null or CheckInvalid return false");
114 return HDF_ERR_INVALID_PARAM;
115 }
116
117 ReleaseFd(codecBuffer);
118
119 void *sharedPtr = const_cast<void *>(shMem_->ReadFromAshmem(codecBuffer.filledLen, codecBuffer.offset));
120 if (!sharedPtr) {
121 CODEC_LOGE("omxBuffer.length [%{public}d omxBuffer.offset[%{public}d]", codecBuffer.filledLen,
122 codecBuffer.offset);
123 return HDF_ERR_INVALID_PARAM;
124 }
125 auto ret = memcpy_s(omxBuffer.pBuffer + codecBuffer.offset, codecBuffer.allocLen - codecBuffer.offset, sharedPtr,
126 codecBuffer.filledLen);
127 if (ret != EOK) {
128 CODEC_LOGE("memcpy_s ret [%{public}d", ret);
129 return HDF_ERR_INVALID_PARAM;
130 }
131 return ICodecBuffer::EmptyOmxBuffer(codecBuffer, omxBuffer);
132 }
133
FreeBuffer(struct OmxCodecBuffer & codecBuffer)134 int32_t CodecShareBuffer::FreeBuffer(struct OmxCodecBuffer &codecBuffer)
135 {
136 if (!CheckInvalid(codecBuffer)) {
137 CODEC_LOGE("shMem_ is null or CheckInvalid return false");
138 return HDF_ERR_INVALID_PARAM;
139 }
140
141 ReleaseFd(codecBuffer);
142
143 shMem_->UnmapAshmem();
144 shMem_->CloseAshmem();
145 shMem_ = nullptr;
146 return HDF_SUCCESS;
147 }
148
GetBuffer()149 uint8_t *CodecShareBuffer::GetBuffer()
150 {
151 return nullptr;
152 }
153
EmptyOmxBufferDone(OMX_BUFFERHEADERTYPE & omxBuffer)154 int32_t CodecShareBuffer::EmptyOmxBufferDone(OMX_BUFFERHEADERTYPE &omxBuffer)
155 {
156 return ICodecBuffer::EmptyOmxBufferDone(omxBuffer);
157 }
158
FillOmxBufferDone(OMX_BUFFERHEADERTYPE & omxBuffer)159 int32_t CodecShareBuffer::FillOmxBufferDone(OMX_BUFFERHEADERTYPE &omxBuffer)
160 {
161 if (shMem_ == nullptr || !shMem_->WriteToAshmem(omxBuffer.pBuffer, omxBuffer.nFilledLen, omxBuffer.nOffset)) {
162 CODEC_LOGE("write to ashmem fail");
163 return HDF_ERR_INVALID_PARAM;
164 }
165
166 return ICodecBuffer::FillOmxBufferDone(omxBuffer);
167 }
168
CheckInvalid(struct OmxCodecBuffer & codecBuffer)169 bool CodecShareBuffer::CheckInvalid(struct OmxCodecBuffer &codecBuffer)
170 {
171 if (!ICodecBuffer::CheckInvalid(codecBuffer) || shMem_ == nullptr) {
172 CODEC_LOGE("shMem_ is null or CheckInvalid return false");
173 return false;
174 }
175 return true;
176 }
177
ReleaseFd(struct OmxCodecBuffer & codecBuffer)178 void CodecShareBuffer::ReleaseFd(struct OmxCodecBuffer &codecBuffer)
179 {
180 // close the fd, if fd is sent by codecBuffer
181 if (codecBuffer.buffer != nullptr) {
182 int fd = (int)reinterpret_cast<uintptr_t>(codecBuffer.buffer);
183 close(fd);
184 codecBuffer.buffer = 0;
185 codecBuffer.bufferLen = 0;
186 }
187 }
188 } // namespace Omx
189 } // namespace Codec
190 } // namespace OHOS