faad.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643
  1. /*
  2. * Squeezelite - lightweight headless squeezebox emulator
  3. *
  4. * (c) Adrian Smith 2012-2015, triode1@btinternet.com
  5. * Ralph Irving 2015-2017, ralph_irving@hotmail.com
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation, either version 3 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. *
  20. */
  21. #include "squeezelite.h"
  22. #include <neaacdec.h>
  23. #define WRAPBUF_LEN 2048
  24. struct chunk_table {
  25. u32_t sample, offset;
  26. };
  27. struct faad {
  28. NeAACDecHandle hAac;
  29. u8_t type;
  30. // following used for mp4 only
  31. u32_t consume;
  32. u32_t pos;
  33. u32_t sample;
  34. u32_t nextchunk;
  35. void *stsc;
  36. u32_t skip;
  37. u64_t samples;
  38. u64_t sttssamples;
  39. bool empty;
  40. struct chunk_table *chunkinfo;
  41. // faad symbols to be dynamically loaded
  42. #if !LINKALL
  43. NeAACDecConfigurationPtr (* NeAACDecGetCurrentConfiguration)(NeAACDecHandle);
  44. unsigned char (* NeAACDecSetConfiguration)(NeAACDecHandle, NeAACDecConfigurationPtr);
  45. NeAACDecHandle (* NeAACDecOpen)(void);
  46. void (* NeAACDecClose)(NeAACDecHandle);
  47. long (* NeAACDecInit)(NeAACDecHandle, unsigned char *, unsigned long, unsigned long *, unsigned char *);
  48. char (* NeAACDecInit2)(NeAACDecHandle, unsigned char *pBuffer, unsigned long, unsigned long *, unsigned char *);
  49. void *(* NeAACDecDecode)(NeAACDecHandle, NeAACDecFrameInfo *, unsigned char *, unsigned long);
  50. char *(* NeAACDecGetErrorMessage)(unsigned char);
  51. #endif
  52. };
  53. static struct faad *a;
  54. extern log_level loglevel;
  55. extern struct buffer *streambuf;
  56. extern struct buffer *outputbuf;
  57. extern struct streamstate stream;
  58. extern struct outputstate output;
  59. extern struct decodestate decode;
  60. extern struct processstate process;
  61. #define LOCK_S mutex_lock(streambuf->mutex)
  62. #define UNLOCK_S mutex_unlock(streambuf->mutex)
  63. #define LOCK_O mutex_lock(outputbuf->mutex)
  64. #define UNLOCK_O mutex_unlock(outputbuf->mutex)
  65. #if PROCESS
  66. #define LOCK_O_direct if (decode.direct) mutex_lock(outputbuf->mutex)
  67. #define UNLOCK_O_direct if (decode.direct) mutex_unlock(outputbuf->mutex)
  68. #define IF_DIRECT(x) if (decode.direct) { x }
  69. #define IF_PROCESS(x) if (!decode.direct) { x }
  70. #else
  71. #define LOCK_O_direct mutex_lock(outputbuf->mutex)
  72. #define UNLOCK_O_direct mutex_unlock(outputbuf->mutex)
  73. #define IF_DIRECT(x) { x }
  74. #define IF_PROCESS(x)
  75. #endif
  76. #if LINKALL
  77. #define NEAAC(h, fn, ...) (NeAACDec ## fn)(__VA_ARGS__)
  78. #else
  79. #define NEAAC(h, fn, ...) (h)->NeAACDec##fn(__VA_ARGS__)
  80. #endif
  81. // minimal code for mp4 file parsing to extract audio config and find media data
  82. // adapted from faad2/common/mp4ff
  83. u32_t mp4_desc_length(u8_t **buf) {
  84. u8_t b;
  85. u8_t num_bytes = 0;
  86. u32_t length = 0;
  87. do {
  88. b = **buf;
  89. *buf += 1;
  90. num_bytes++;
  91. length = (length << 7) | (b & 0x7f);
  92. } while ((b & 0x80) && num_bytes < 4);
  93. return length;
  94. }
  95. // read mp4 header to extract config data
  96. static int read_mp4_header(unsigned long *samplerate_p, unsigned char *channels_p) {
  97. size_t bytes = min(_buf_used(streambuf), _buf_cont_read(streambuf));
  98. char type[5];
  99. u32_t len;
  100. while (bytes >= 8) {
  101. // count trak to find the first playable one
  102. static unsigned trak, play;
  103. u32_t consume;
  104. len = unpackN((u32_t *)streambuf->readp);
  105. memcpy(type, streambuf->readp + 4, 4);
  106. type[4] = '\0';
  107. if (!strcmp(type, "moov")) {
  108. trak = 0;
  109. play = 0;
  110. }
  111. if (!strcmp(type, "trak")) {
  112. trak++;
  113. }
  114. // extract audio config from within esds and pass to DecInit2
  115. if (!strcmp(type, "esds") && bytes > len) {
  116. unsigned config_len;
  117. u8_t *ptr = streambuf->readp + 12;
  118. if (*ptr++ == 0x03) {
  119. mp4_desc_length(&ptr);
  120. ptr += 4;
  121. } else {
  122. ptr += 3;
  123. }
  124. mp4_desc_length(&ptr);
  125. ptr += 13;
  126. if (*ptr++ != 0x05) {
  127. LOG_WARN("error parsing esds");
  128. return -1;
  129. }
  130. config_len = mp4_desc_length(&ptr);
  131. if (NEAAC(a, Init2, a->hAac, ptr, config_len, samplerate_p, channels_p) == 0) {
  132. LOG_DEBUG("playable aac track: %u", trak);
  133. play = trak;
  134. }
  135. }
  136. // extract the total number of samples from stts
  137. if (!strcmp(type, "stts") && bytes > len) {
  138. u32_t i;
  139. u8_t *ptr = streambuf->readp + 12;
  140. u32_t entries = unpackN((u32_t *)ptr);
  141. ptr += 4;
  142. for (i = 0; i < entries; ++i) {
  143. u32_t count = unpackN((u32_t *)ptr);
  144. u32_t size = unpackN((u32_t *)(ptr + 4));
  145. a->sttssamples += count * size;
  146. ptr += 8;
  147. }
  148. LOG_DEBUG("total number of samples contained in stts: " FMT_u64, a->sttssamples);
  149. }
  150. // stash sample to chunk info, assume it comes before stco
  151. if (!strcmp(type, "stsc") && bytes > len && !a->chunkinfo) {
  152. a->stsc = malloc(len - 12);
  153. if (a->stsc == NULL) {
  154. LOG_WARN("malloc fail");
  155. return -1;
  156. }
  157. memcpy(a->stsc, streambuf->readp + 12, len - 12);
  158. }
  159. // build offsets table from stco and stored stsc
  160. if (!strcmp(type, "stco") && bytes > len && play == trak) {
  161. u32_t i;
  162. // extract chunk offsets
  163. u8_t *ptr = streambuf->readp + 12;
  164. u32_t entries = unpackN((u32_t *)ptr);
  165. ptr += 4;
  166. a->chunkinfo = malloc(sizeof(struct chunk_table) * (entries + 1));
  167. if (a->chunkinfo == NULL) {
  168. LOG_WARN("malloc fail");
  169. return -1;
  170. }
  171. for (i = 0; i < entries; ++i) {
  172. a->chunkinfo[i].offset = unpackN((u32_t *)ptr);
  173. a->chunkinfo[i].sample = 0;
  174. ptr += 4;
  175. }
  176. a->chunkinfo[i].sample = 0;
  177. a->chunkinfo[i].offset = 0;
  178. // fill in first sample id for each chunk from stored stsc
  179. if (a->stsc) {
  180. u32_t stsc_entries = unpackN((u32_t *)a->stsc);
  181. u32_t sample = 0;
  182. u32_t last = 0, last_samples = 0;
  183. u8_t *ptr = (u8_t *)a->stsc + 4;
  184. while (stsc_entries--) {
  185. u32_t first = unpackN((u32_t *)ptr);
  186. u32_t samples = unpackN((u32_t *)(ptr + 4));
  187. if (last) {
  188. for (i = last - 1; i < first - 1; ++i) {
  189. a->chunkinfo[i].sample = sample;
  190. sample += last_samples;
  191. }
  192. }
  193. if (stsc_entries == 0) {
  194. for (i = first - 1; i < entries; ++i) {
  195. a->chunkinfo[i].sample = sample;
  196. sample += samples;
  197. }
  198. }
  199. last = first;
  200. last_samples = samples;
  201. ptr += 12;
  202. }
  203. free(a->stsc);
  204. a->stsc = NULL;
  205. }
  206. }
  207. // found media data, advance to start of first chunk and return
  208. if (!strcmp(type, "mdat")) {
  209. _buf_inc_readp(streambuf, 8);
  210. a->pos += 8;
  211. bytes -= 8;
  212. if (play) {
  213. LOG_DEBUG("type: mdat len: %u pos: %u", len, a->pos);
  214. if (a->chunkinfo && a->chunkinfo[0].offset > a->pos) {
  215. u32_t skip = a->chunkinfo[0].offset - a->pos;
  216. LOG_DEBUG("skipping: %u", skip);
  217. if (skip <= bytes) {
  218. _buf_inc_readp(streambuf, skip);
  219. a->pos += skip;
  220. } else {
  221. a->consume = skip;
  222. }
  223. }
  224. a->sample = a->nextchunk = 1;
  225. return 1;
  226. } else {
  227. LOG_DEBUG("type: mdat len: %u, no playable track found", len);
  228. return -1;
  229. }
  230. }
  231. // parse key-value atoms within ilst ---- entries to get encoder padding within iTunSMPB entry for gapless
  232. if (!strcmp(type, "----") && bytes > len) {
  233. u8_t *ptr = streambuf->readp + 8;
  234. u32_t remain = len - 8, size;
  235. if (!memcmp(ptr + 4, "mean", 4) && (size = unpackN((u32_t *)ptr)) < remain) {
  236. ptr += size; remain -= size;
  237. }
  238. if (!memcmp(ptr + 4, "name", 4) && (size = unpackN((u32_t *)ptr)) < remain && !memcmp(ptr + 12, "iTunSMPB", 8)) {
  239. ptr += size; remain -= size;
  240. }
  241. if (!memcmp(ptr + 4, "data", 4) && remain > 16 + 48) {
  242. // data is stored as hex strings: 0 start end samples
  243. u32_t b, c; u64_t d;
  244. if (sscanf((const char *)(ptr + 16), "%x %x %x " FMT_x64, &b, &b, &c, &d) == 4) {
  245. LOG_DEBUG("iTunSMPB start: %u end: %u samples: " FMT_u64, b, c, d);
  246. if (a->sttssamples && a->sttssamples < b + c + d) {
  247. LOG_DEBUG("reducing samples as stts count is less");
  248. d = a->sttssamples - (b + c);
  249. }
  250. a->skip = b;
  251. a->samples = d;
  252. }
  253. }
  254. }
  255. // default to consuming entire box
  256. consume = len;
  257. // read into these boxes so reduce consume
  258. if (!strcmp(type, "moov") || !strcmp(type, "trak") || !strcmp(type, "mdia") || !strcmp(type, "minf") || !strcmp(type, "stbl") ||
  259. !strcmp(type, "udta") || !strcmp(type, "ilst")) {
  260. consume = 8;
  261. }
  262. // special cases which mix mix data in the enclosing box which we want to read into
  263. if (!strcmp(type, "stsd")) consume = 16;
  264. if (!strcmp(type, "mp4a")) consume = 36;
  265. if (!strcmp(type, "meta")) consume = 12;
  266. // consume rest of box if it has been parsed (all in the buffer) or is not one we want to parse
  267. if (bytes >= consume) {
  268. LOG_DEBUG("type: %s len: %u consume: %u", type, len, consume);
  269. _buf_inc_readp(streambuf, consume);
  270. a->pos += consume;
  271. bytes -= consume;
  272. } else if ( !(!strcmp(type, "esds") || !strcmp(type, "stts") || !strcmp(type, "stsc") ||
  273. !strcmp(type, "stco") || !strcmp(type, "----")) ) {
  274. LOG_DEBUG("type: %s len: %u consume: %u - partial consume: %u", type, len, consume, bytes);
  275. _buf_inc_readp(streambuf, bytes);
  276. a->pos += bytes;
  277. a->consume = consume - bytes;
  278. break;
  279. } else {
  280. break;
  281. }
  282. }
  283. return 0;
  284. }
  285. static decode_state faad_decode(void) {
  286. size_t bytes_total;
  287. size_t bytes_wrap;
  288. static NeAACDecFrameInfo info;
  289. s32_t *iptr;
  290. bool endstream;
  291. frames_t frames;
  292. LOCK_S;
  293. bytes_total = _buf_used(streambuf);
  294. bytes_wrap = min(bytes_total, _buf_cont_read(streambuf));
  295. if (stream.state <= DISCONNECT && !bytes_total) {
  296. UNLOCK_S;
  297. return DECODE_COMPLETE;
  298. }
  299. if (a->consume) {
  300. u32_t consume = min(a->consume, bytes_wrap);
  301. LOG_DEBUG("consume: %u of %u", consume, a->consume);
  302. _buf_inc_readp(streambuf, consume);
  303. a->pos += consume;
  304. a->consume -= consume;
  305. UNLOCK_S;
  306. return DECODE_RUNNING;
  307. }
  308. if (decode.new_stream) {
  309. int found = 0;
  310. static unsigned char channels;
  311. static unsigned long samplerate;
  312. if (a->type == '2') {
  313. // adts stream - seek for header
  314. while (bytes_wrap >= 2 && (*(streambuf->readp) != 0xFF || (*(streambuf->readp + 1) & 0xF6) != 0xF0)) {
  315. _buf_inc_readp(streambuf, 1);
  316. bytes_total--;
  317. bytes_wrap--;
  318. }
  319. if (bytes_wrap >= 2) {
  320. long n = NEAAC(a, Init, a->hAac, streambuf->readp, bytes_wrap, &samplerate, &channels);
  321. if (n < 0) {
  322. found = -1;
  323. } else {
  324. _buf_inc_readp(streambuf, n);
  325. found = 1;
  326. }
  327. }
  328. } else {
  329. // mp4 - read header
  330. found = read_mp4_header(&samplerate, &channels);
  331. }
  332. if (found == 1) {
  333. LOG_INFO("samplerate: %u channels: %u", samplerate, channels);
  334. bytes_total = _buf_used(streambuf);
  335. bytes_wrap = min(bytes_total, _buf_cont_read(streambuf));
  336. LOCK_O;
  337. LOG_INFO("setting track_start");
  338. output.next_sample_rate = decode_newstream(samplerate, output.supported_rates);
  339. IF_DSD( output.next_fmt = PCM; )
  340. output.track_start = outputbuf->writep;
  341. if (output.fade_mode) _checkfade(true);
  342. decode.new_stream = false;
  343. UNLOCK_O;
  344. } else if (found == -1) {
  345. LOG_WARN("error reading stream header");
  346. UNLOCK_S;
  347. return DECODE_ERROR;
  348. } else {
  349. // not finished header parsing come back next time
  350. UNLOCK_S;
  351. return DECODE_RUNNING;
  352. }
  353. }
  354. if (bytes_wrap < WRAPBUF_LEN && bytes_total > WRAPBUF_LEN) {
  355. // make a local copy of frames which may have wrapped round the end of streambuf
  356. static u8_t buf[WRAPBUF_LEN];
  357. memcpy(buf, streambuf->readp, bytes_wrap);
  358. memcpy(buf + bytes_wrap, streambuf->buf, WRAPBUF_LEN - bytes_wrap);
  359. iptr = NEAAC(a, Decode, a->hAac, &info, buf, WRAPBUF_LEN);
  360. } else {
  361. iptr = NEAAC(a, Decode, a->hAac, &info, streambuf->readp, bytes_wrap);
  362. }
  363. if (info.error) {
  364. LOG_WARN("error: %u %s", info.error, NEAAC(a, GetErrorMessage, info.error));
  365. }
  366. endstream = false;
  367. // mp4 end of chunk - skip to next offset
  368. if (a->chunkinfo && a->chunkinfo[a->nextchunk].offset && a->sample++ == a->chunkinfo[a->nextchunk].sample) {
  369. if (a->chunkinfo[a->nextchunk].offset > a->pos) {
  370. u32_t skip = a->chunkinfo[a->nextchunk].offset - a->pos;
  371. if (skip != info.bytesconsumed) {
  372. LOG_DEBUG("skipping to next chunk pos: %u consumed: %u != skip: %u", a->pos, info.bytesconsumed, skip);
  373. }
  374. if (bytes_total >= skip) {
  375. _buf_inc_readp(streambuf, skip);
  376. a->pos += skip;
  377. } else {
  378. a->consume = skip;
  379. }
  380. a->nextchunk++;
  381. } else {
  382. LOG_ERROR("error: need to skip backwards!");
  383. endstream = true;
  384. }
  385. // adts and mp4 when not at end of chunk
  386. } else if (info.bytesconsumed != 0) {
  387. _buf_inc_readp(streambuf, info.bytesconsumed);
  388. a->pos += info.bytesconsumed;
  389. // error which doesn't advance streambuf - end
  390. } else {
  391. endstream = true;
  392. }
  393. UNLOCK_S;
  394. if (endstream) {
  395. LOG_WARN("unable to decode further");
  396. return DECODE_ERROR;
  397. }
  398. if (!info.samples) {
  399. a->empty = true;
  400. return DECODE_RUNNING;
  401. }
  402. frames = info.samples / info.channels;
  403. if (a->skip) {
  404. u32_t skip;
  405. if (a->empty) {
  406. a->empty = false;
  407. a->skip -= frames;
  408. LOG_DEBUG("gapless: first frame empty, skipped %u frames at start", frames);
  409. }
  410. skip = min(frames, a->skip);
  411. LOG_DEBUG("gapless: skipping %u frames at start", skip);
  412. frames -= skip;
  413. a->skip -= skip;
  414. iptr += skip * info.channels;
  415. }
  416. if (a->samples) {
  417. if (a->samples < frames) {
  418. LOG_DEBUG("gapless: trimming %u frames from end", frames - a->samples);
  419. frames = (frames_t)a->samples;
  420. }
  421. a->samples -= frames;
  422. }
  423. LOG_SDEBUG("write %u frames", frames);
  424. LOCK_O_direct;
  425. while (frames > 0) {
  426. frames_t f;
  427. frames_t count;
  428. s32_t *optr;
  429. IF_DIRECT(
  430. f = _buf_cont_write(outputbuf) / BYTES_PER_FRAME;
  431. optr = (s32_t *)outputbuf->writep;
  432. );
  433. IF_PROCESS(
  434. f = process.max_in_frames;
  435. optr = (s32_t *)process.inbuf;
  436. );
  437. f = min(f, frames);
  438. count = f;
  439. if (info.channels == 2) {
  440. while (count--) {
  441. *optr++ = *iptr++ << 8;
  442. *optr++ = *iptr++ << 8;
  443. }
  444. } else if (info.channels == 1) {
  445. while (count--) {
  446. *optr++ = *iptr << 8;
  447. *optr++ = *iptr++ << 8;
  448. }
  449. } else {
  450. LOG_WARN("unsupported number of channels");
  451. }
  452. frames -= f;
  453. IF_DIRECT(
  454. _buf_inc_writep(outputbuf, f * BYTES_PER_FRAME);
  455. );
  456. IF_PROCESS(
  457. process.in_frames = f;
  458. if (frames) LOG_ERROR("unhandled case");
  459. );
  460. }
  461. UNLOCK_O_direct;
  462. return DECODE_RUNNING;
  463. }
  464. static void faad_open(u8_t size, u8_t rate, u8_t chan, u8_t endianness) {
  465. NeAACDecConfigurationPtr conf;
  466. LOG_INFO("opening %s stream", size == '2' ? "adts" : "mp4");
  467. a->type = size;
  468. a->pos = a->consume = a->sample = a->nextchunk = 0;
  469. if (a->chunkinfo) {
  470. free(a->chunkinfo);
  471. }
  472. if (a->stsc) {
  473. free(a->stsc);
  474. }
  475. a->chunkinfo = NULL;
  476. a->stsc = NULL;
  477. a->skip = 0;
  478. a->samples = 0;
  479. a->sttssamples = 0;
  480. a->empty = false;
  481. if (a->hAac) {
  482. NEAAC(a, Close, a->hAac);
  483. }
  484. a->hAac = NEAAC(a, Open);
  485. conf = NEAAC(a, GetCurrentConfiguration, a->hAac);
  486. conf->outputFormat = FAAD_FMT_24BIT;
  487. conf->downMatrix = 1;
  488. if (!NEAAC(a, SetConfiguration, a->hAac, conf)) {
  489. LOG_WARN("error setting config");
  490. };
  491. }
  492. static void faad_close(void) {
  493. NEAAC(a, Close, a->hAac);
  494. a->hAac = NULL;
  495. if (a->chunkinfo) {
  496. free(a->chunkinfo);
  497. a->chunkinfo = NULL;
  498. }
  499. if (a->stsc) {
  500. free(a->stsc);
  501. a->stsc = NULL;
  502. }
  503. }
  504. static bool load_faad() {
  505. #if !LINKALL
  506. void *handle = dlopen(LIBFAAD, RTLD_NOW);
  507. char *err;
  508. if (!handle) {
  509. LOG_INFO("dlerror: %s", dlerror());
  510. return false;
  511. }
  512. a->NeAACDecGetCurrentConfiguration = dlsym(handle, "NeAACDecGetCurrentConfiguration");
  513. a->NeAACDecSetConfiguration = dlsym(handle, "NeAACDecSetConfiguration");
  514. a->NeAACDecOpen = dlsym(handle, "NeAACDecOpen");
  515. a->NeAACDecClose = dlsym(handle, "NeAACDecClose");
  516. a->NeAACDecInit = dlsym(handle, "NeAACDecInit");
  517. a->NeAACDecInit2 = dlsym(handle, "NeAACDecInit2");
  518. a->NeAACDecDecode = dlsym(handle, "NeAACDecDecode");
  519. a->NeAACDecGetErrorMessage = dlsym(handle, "NeAACDecGetErrorMessage");
  520. if ((err = dlerror()) != NULL) {
  521. LOG_INFO("dlerror: %s", err);
  522. return false;
  523. }
  524. LOG_INFO("loaded "LIBFAAD"");
  525. #endif
  526. return true;
  527. }
  528. struct codec *register_faad(void) {
  529. static struct codec ret = {
  530. 'a', // id
  531. "aac", // types
  532. WRAPBUF_LEN, // min read
  533. 20480, // min space
  534. faad_open, // open
  535. faad_close, // close
  536. faad_decode, // decode
  537. };
  538. a = malloc(sizeof(struct faad));
  539. if (!a) {
  540. return NULL;
  541. }
  542. a->hAac = NULL;
  543. a->chunkinfo = NULL;
  544. a->stsc = NULL;
  545. if (!load_faad()) {
  546. return NULL;
  547. }
  548. LOG_INFO("using faad to decode aac");
  549. return &ret;
  550. }