1 /*
2 * Copyright (C) 2018 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 */
16
17 //#define LOG_NDEBUG 0
18 #ifdef AMRNB
19 #define LOG_TAG "C2SoftAmrNbDec"
20 #else
21 #define LOG_TAG "C2SoftAmrWbDec"
22 #endif
23 #include <log/log.h>
24
25 #include <media/stagefright/foundation/MediaDefs.h>
26
27 #include <C2PlatformSupport.h>
28 #include <SimpleC2Interface.h>
29
30 #include "C2SoftAmrDec.h"
31 #include "gsmamr_dec.h"
32 #include "pvamrwbdecoder.h"
33
34 namespace android {
35
36 namespace {
37
38 #ifdef AMRNB
39 constexpr char COMPONENT_NAME[] = "c2.android.amrnb.decoder";
40 #else
41 constexpr char COMPONENT_NAME[] = "c2.android.amrwb.decoder";
42 #endif
43
44 } // namespace
45
46 class C2SoftAmrDec::IntfImpl : public SimpleInterface<void>::BaseParams {
47 public:
IntfImpl(const std::shared_ptr<C2ReflectorHelper> & helper)48 explicit IntfImpl(const std::shared_ptr<C2ReflectorHelper> &helper)
49 : SimpleInterface<void>::BaseParams(
50 helper,
51 COMPONENT_NAME,
52 C2Component::KIND_DECODER,
53 C2Component::DOMAIN_AUDIO,
54 #ifdef AMRNB
55 MEDIA_MIMETYPE_AUDIO_AMR_NB
56 #else
57 MEDIA_MIMETYPE_AUDIO_AMR_WB
58 #endif
59 ) {
60 noPrivateBuffers();
61 noInputReferences();
62 noOutputReferences();
63 noInputLatency();
64 noTimeStretch();
65 setDerivedInstance(this);
66
67 addParameter(
68 DefineParam(mAttrib, C2_PARAMKEY_COMPONENT_ATTRIBUTES)
69 .withConstValue(new C2ComponentAttributesSetting(
70 C2Component::ATTRIB_IS_TEMPORAL))
71 .build());
72
73 addParameter(
74 DefineParam(mSampleRate, C2_PARAMKEY_SAMPLE_RATE)
75 #ifdef AMRNB
76 .withDefault(new C2StreamSampleRateInfo::output(0u, 8000))
77 .withFields({C2F(mSampleRate, value).equalTo(8000)})
78 #else
79 .withDefault(new C2StreamSampleRateInfo::output(0u, 16000))
80 .withFields({C2F(mSampleRate, value).equalTo(16000)})
81 #endif
82 .withSetter((Setter<decltype(*mSampleRate)>::StrictValueWithNoDeps))
83 .build());
84
85 addParameter(
86 DefineParam(mChannelCount, C2_PARAMKEY_CHANNEL_COUNT)
87 .withDefault(new C2StreamChannelCountInfo::output(0u, 1))
88 .withFields({C2F(mChannelCount, value).equalTo(1)})
89 .withSetter((Setter<decltype(*mChannelCount)>::StrictValueWithNoDeps))
90 .build());
91
92 addParameter(
93 DefineParam(mBitrate, C2_PARAMKEY_BITRATE)
94 #ifdef AMRNB
95 .withDefault(new C2StreamBitrateInfo::input(0u, 4750))
96 .withFields({C2F(mBitrate, value).inRange(4750, 12200)})
97 #else
98 .withDefault(new C2StreamBitrateInfo::input(0u, 6600))
99 .withFields({C2F(mBitrate, value).inRange(6600, 23850)})
100 #endif
101 .withSetter(Setter<decltype(*mBitrate)>::NonStrictValueWithNoDeps)
102 .build());
103
104 addParameter(
105 DefineParam(mInputMaxBufSize, C2_PARAMKEY_INPUT_MAX_BUFFER_SIZE)
106 .withConstValue(new C2StreamMaxBufferSizeInfo::input(0u, 8192))
107 .build());
108 }
109
110 private:
111 std::shared_ptr<C2StreamSampleRateInfo::output> mSampleRate;
112 std::shared_ptr<C2StreamChannelCountInfo::output> mChannelCount;
113 std::shared_ptr<C2StreamBitrateInfo::input> mBitrate;
114 std::shared_ptr<C2StreamMaxBufferSizeInfo::input> mInputMaxBufSize;
115 };
116
C2SoftAmrDec(const char * name,c2_node_id_t id,const std::shared_ptr<IntfImpl> & intfImpl)117 C2SoftAmrDec::C2SoftAmrDec(
118 const char *name,
119 c2_node_id_t id,
120 const std::shared_ptr<IntfImpl> &intfImpl)
121 : SimpleC2Component(std::make_shared<SimpleInterface<IntfImpl>>(name, id, intfImpl)),
122 mIntf(intfImpl),
123 mAmrHandle(nullptr),
124 mDecoderBuf(nullptr),
125 mDecoderCookie(nullptr) {
126 #ifdef AMRNB
127 mIsWide = false;
128 #else
129 mIsWide = true;
130 #endif
131 }
132
~C2SoftAmrDec()133 C2SoftAmrDec::~C2SoftAmrDec() {
134 (void)onRelease();
135 }
136
onInit()137 c2_status_t C2SoftAmrDec::onInit() {
138 status_t err = initDecoder();
139 return err == OK ? C2_OK : C2_NO_MEMORY;
140 }
141
onStop()142 c2_status_t C2SoftAmrDec::onStop() {
143 if (!mIsWide) {
144 Speech_Decode_Frame_reset(mAmrHandle);
145 } else {
146 pvDecoder_AmrWb_Reset(mAmrHandle, 1 /* reset_all */);
147 }
148 mSignalledError = false;
149 mSignalledOutputEos = false;
150
151 return C2_OK;
152 }
153
onReset()154 void C2SoftAmrDec::onReset() {
155 (void)onStop();
156 }
157
onRelease()158 void C2SoftAmrDec::onRelease() {
159 if (!mIsWide) {
160 if (mAmrHandle) {
161 GSMDecodeFrameExit(&mAmrHandle);
162 }
163 mAmrHandle = nullptr;
164 } else {
165 if (mDecoderBuf) {
166 free(mDecoderBuf);
167 }
168 mDecoderBuf = nullptr;
169 mAmrHandle = nullptr;
170 mDecoderCookie = nullptr;
171 }
172 }
173
onFlush_sm()174 c2_status_t C2SoftAmrDec::onFlush_sm() {
175 return onStop();
176 }
177
initDecoder()178 status_t C2SoftAmrDec::initDecoder() {
179 if (!mIsWide) {
180 if (GSMInitDecode(&mAmrHandle, (int8_t *)"AMRNBDecoder"))
181 return UNKNOWN_ERROR;
182 } else {
183 uint32_t memReq = pvDecoder_AmrWbMemRequirements();
184 mDecoderBuf = malloc(memReq);
185 if (mDecoderBuf) {
186 pvDecoder_AmrWb_Init(&mAmrHandle, mDecoderBuf, &mDecoderCookie);
187 }
188 else {
189 return NO_MEMORY;
190 }
191 }
192 mSignalledError = false;
193 mSignalledOutputEos = false;
194
195 return OK;
196 }
197
getFrameSize(bool isWide,unsigned FM)198 static size_t getFrameSize(bool isWide, unsigned FM) {
199 static const size_t kFrameSizeNB[16] = {
200 12, 13, 15, 17, 19, 20, 26, 31,
201 5, 6, 5, 5, // SID
202 0, 0, 0, // future use
203 0 // no data
204 };
205 static const size_t kFrameSizeWB[16] = {
206 17, 23, 32, 36, 40, 46, 50, 58, 60,
207 5, // SID
208 0, 0, 0, 0, // future use
209 0, // speech lost
210 0 // no data
211 };
212
213 if (FM > 15 || (isWide && FM > 9 && FM < 14) || (!isWide && FM > 11 && FM < 15)) {
214 ALOGE("illegal AMR frame mode %d", FM);
215 return 0;
216 }
217 // add 1 for header byte
218 return (isWide ? kFrameSizeWB[FM] : kFrameSizeNB[FM]) + 1;
219 }
220
calculateNumFrames(const uint8 * input,size_t inSize,std::vector<size_t> * frameSizeList,bool isWide)221 static status_t calculateNumFrames(const uint8 *input, size_t inSize,
222 std::vector<size_t> *frameSizeList, bool isWide) {
223 for (size_t k = 0; k < inSize;) {
224 int16_t FM = ((input[0] >> 3) & 0x0f);
225 size_t frameSize = getFrameSize(isWide, FM);
226 if (frameSize == 0) {
227 return UNKNOWN_ERROR;
228 }
229 if ((inSize - k) >= frameSize) {
230 input += frameSize;
231 k += frameSize;
232 }
233 else break;
234 frameSizeList->push_back(frameSize);
235 }
236 return OK;
237 }
238
process(const std::unique_ptr<C2Work> & work,const std::shared_ptr<C2BlockPool> & pool)239 void C2SoftAmrDec::process(
240 const std::unique_ptr<C2Work> &work,
241 const std::shared_ptr<C2BlockPool> &pool) {
242 // Initialize output work
243 work->result = C2_OK;
244 work->workletsProcessed = 1u;
245 work->worklets.front()->output.flags = work->input.flags;
246
247 if (mSignalledError || mSignalledOutputEos) {
248 work->result = C2_BAD_VALUE;
249 return;
250 }
251
252 C2ReadView rView = mDummyReadView;
253 size_t inOffset = 0u;
254 size_t inSize = 0u;
255 if (!work->input.buffers.empty()) {
256 rView = work->input.buffers[0]->data().linearBlocks().front().map().get();
257 inSize = rView.capacity();
258 if (inSize && rView.error()) {
259 ALOGE("read view map failed %d", rView.error());
260 work->result = rView.error();
261 return;
262 }
263 }
264
265 bool eos = (work->input.flags & C2FrameData::FLAG_END_OF_STREAM) != 0;
266 if (inSize == 0) {
267 work->worklets.front()->output.flags = work->input.flags;
268 work->worklets.front()->output.buffers.clear();
269 work->worklets.front()->output.ordinal = work->input.ordinal;
270 if (eos) {
271 mSignalledOutputEos = true;
272 ALOGV("signalled EOS");
273 }
274 return;
275 }
276
277 ALOGV("in buffer attr. size %zu timestamp %d frameindex %d", inSize,
278 (int)work->input.ordinal.timestamp.peeku(), (int)work->input.ordinal.frameIndex.peeku());
279
280 std::vector<size_t> frameSizeList;
281 if (OK != calculateNumFrames(rView.data() + inOffset, inSize, &frameSizeList,
282 mIsWide)) {
283 work->result = C2_CORRUPTED;
284 mSignalledError = true;
285 return;
286 }
287 if (frameSizeList.empty()) {
288 ALOGE("input size smaller than expected");
289 work->result = C2_CORRUPTED;
290 mSignalledError = true;
291 return;
292 }
293
294 int16_t outSamples = mIsWide ? kNumSamplesPerFrameWB : kNumSamplesPerFrameNB;
295 size_t calOutSize = outSamples * frameSizeList.size() * sizeof(int16_t);
296 std::shared_ptr<C2LinearBlock> block;
297 C2MemoryUsage usage = { C2MemoryUsage::CPU_READ, C2MemoryUsage::CPU_WRITE };
298 c2_status_t err = pool->fetchLinearBlock(calOutSize, usage, &block);
299 if (err != C2_OK) {
300 ALOGE("fetchLinearBlock for Output failed with status %d", err);
301 work->result = C2_NO_MEMORY;
302 return;
303 }
304 C2WriteView wView = block->map().get();
305 if (wView.error()) {
306 ALOGE("write view map failed %d", wView.error());
307 work->result = wView.error();
308 return;
309 }
310
311 int16_t *output = reinterpret_cast<int16_t *>(wView.data());
312 auto it = frameSizeList.begin();
313 const uint8_t *inPtr = rView.data() + inOffset;
314 size_t inPos = 0;
315 while (inPos < inSize) {
316 if (it == frameSizeList.end()) {
317 ALOGD("unexpected trailing bytes, ignoring them");
318 break;
319 }
320 uint8_t *input = const_cast<uint8_t *>(inPtr + inPos);
321 int16_t FM = ((*input >> 3) & 0x0f);
322 if (!mIsWide) {
323 int32_t numBytesRead = AMRDecode(mAmrHandle,
324 (Frame_Type_3GPP) FM,
325 input + 1, output, MIME_IETF);
326 if (static_cast<size_t>(numBytesRead + 1) != *it) {
327 ALOGE("panic, parsed size does not match decoded size");
328 work->result = C2_CORRUPTED;
329 mSignalledError = true;
330 return;
331 }
332 } else {
333 if (FM >= 9) {
334 // Produce silence instead of comfort noise and for
335 // speech lost/no data.
336 memset(output, 0, outSamples * sizeof(int16_t));
337 } else {
338 int16_t FT;
339 int16_t numRecSamples;
340
341 mime_unsorting(const_cast<uint8_t *>(&input[1]),
342 mInputSampleBuffer, &FT, &FM, 1, &mRxState);
343 pvDecoder_AmrWb(FM, mInputSampleBuffer, output, &numRecSamples,
344 mDecoderBuf, FT, mDecoderCookie);
345 if (numRecSamples != outSamples) {
346 ALOGE("Sample output per frame incorrect");
347 work->result = C2_CORRUPTED;
348 mSignalledError = true;
349 return;
350 }
351 /* Delete the 2 LSBs (14-bit output) */
352 for (int i = 0; i < numRecSamples; ++i) {
353 output[i] &= 0xfffC;
354 }
355 }
356 }
357 inPos += *it;
358 output += outSamples;
359 ++it;
360 }
361
362 work->worklets.front()->output.flags = work->input.flags;
363 work->worklets.front()->output.buffers.clear();
364 // we filled the output buffer to (intptr_t)output - (intptr_t)wView.data()
365 // use calOutSize as that contains the expected number of samples
366 ALOGD_IF(calOutSize != ((intptr_t)output - (intptr_t)wView.data()),
367 "Expected %zu output bytes, but filled %lld",
368 calOutSize, (long long)((intptr_t)output - (intptr_t)wView.data()));
369 work->worklets.front()->output.buffers.push_back(
370 createLinearBuffer(block, 0, calOutSize));
371 work->worklets.front()->output.ordinal = work->input.ordinal;
372 if (eos) {
373 mSignalledOutputEos = true;
374 ALOGV("signalled EOS");
375 }
376 }
377
drain(uint32_t drainMode,const std::shared_ptr<C2BlockPool> & pool)378 c2_status_t C2SoftAmrDec::drain(
379 uint32_t drainMode,
380 const std::shared_ptr<C2BlockPool> &pool) {
381 (void)pool;
382 if (drainMode == NO_DRAIN) {
383 ALOGW("drain with NO_DRAIN: no-op");
384 return C2_OK;
385 }
386 if (drainMode == DRAIN_CHAIN) {
387 ALOGW("DRAIN_CHAIN not supported");
388 return C2_OMITTED;
389 }
390 return C2_OK;
391 }
392
393 class C2SoftAMRDecFactory : public C2ComponentFactory {
394 public:
C2SoftAMRDecFactory()395 C2SoftAMRDecFactory() : mHelper(std::static_pointer_cast<C2ReflectorHelper>(
396 GetCodec2PlatformComponentStore()->getParamReflector())) {
397 }
398
createComponent(c2_node_id_t id,std::shared_ptr<C2Component> * const component,std::function<void (C2Component *)> deleter)399 virtual c2_status_t createComponent(
400 c2_node_id_t id,
401 std::shared_ptr<C2Component>* const component,
402 std::function<void(C2Component*)> deleter) override {
403 *component = std::shared_ptr<C2Component>(
404 new C2SoftAmrDec(COMPONENT_NAME, id,
405 std::make_shared<C2SoftAmrDec::IntfImpl>(mHelper)),
406 deleter);
407 return C2_OK;
408 }
409
createInterface(c2_node_id_t id,std::shared_ptr<C2ComponentInterface> * const interface,std::function<void (C2ComponentInterface *)> deleter)410 virtual c2_status_t createInterface(
411 c2_node_id_t id,
412 std::shared_ptr<C2ComponentInterface>* const interface,
413 std::function<void(C2ComponentInterface*)> deleter) override {
414 *interface = std::shared_ptr<C2ComponentInterface>(
415 new SimpleInterface<C2SoftAmrDec::IntfImpl>(
416 COMPONENT_NAME, id, std::make_shared<C2SoftAmrDec::IntfImpl>(mHelper)),
417 deleter);
418 return C2_OK;
419 }
420
421 virtual ~C2SoftAMRDecFactory() override = default;
422
423 private:
424 std::shared_ptr<C2ReflectorHelper> mHelper;
425 };
426
427 } // namespace android
428
429 __attribute__((cfi_canonical_jump_table))
CreateCodec2Factory()430 extern "C" ::C2ComponentFactory* CreateCodec2Factory() {
431 ALOGV("in %s", __func__);
432 return new ::android::C2SoftAMRDecFactory();
433 }
434
435 __attribute__((cfi_canonical_jump_table))
DestroyCodec2Factory(::C2ComponentFactory * factory)436 extern "C" void DestroyCodec2Factory(::C2ComponentFactory* factory) {
437 ALOGV("in %s", __func__);
438 delete factory;
439 }
440