encoder.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941
  1. #include <util/darray.h>
  2. #include <util/dstr.h>
  3. #include <obs-module.h>
  4. #ifndef _WIN32
  5. #include <AudioToolbox/AudioToolbox.h>
  6. #endif
  7. #define CA_LOG(level, format, ...) \
  8. blog(level, "[CoreAudio encoder]: " format, ##__VA_ARGS__)
  9. #define CA_LOG_ENCODER(format_name, encoder, level, format, ...) \
  10. blog(level, "[CoreAudio %s: '%s']: " format, \
  11. format_name, obs_encoder_get_name(encoder), \
  12. ##__VA_ARGS__)
  13. #define CA_BLOG(level, format, ...) \
  14. CA_LOG_ENCODER(ca->format_name, ca->encoder, level, format, \
  15. ##__VA_ARGS__)
  16. #ifdef _WIN32
  17. #include "windows-imports.h"
  18. #endif
  19. struct ca_encoder {
  20. obs_encoder_t *encoder;
  21. const char *format_name;
  22. UInt32 format_id;
  23. const UInt32 *allowed_formats;
  24. size_t num_allowed_formats;
  25. AudioConverterRef converter;
  26. size_t output_buffer_size;
  27. uint8_t *output_buffer;
  28. size_t out_frames_per_packet;
  29. size_t in_packets;
  30. size_t in_frame_size;
  31. size_t in_bytes_required;
  32. DARRAY(uint8_t) input_buffer;
  33. size_t bytes_read;
  34. uint64_t total_samples;
  35. uint64_t samples_per_second;
  36. uint8_t *extra_data;
  37. uint32_t extra_data_size;
  38. size_t channels;
  39. };
  40. typedef struct ca_encoder ca_encoder;
  41. static const char *aac_get_name(void)
  42. {
  43. return obs_module_text("CoreAudioAAC");
  44. }
  45. static const char *code_to_str(OSStatus code)
  46. {
  47. switch (code) {
  48. #define HANDLE_CODE(c) case c: return #c
  49. HANDLE_CODE(kAudio_UnimplementedError);
  50. HANDLE_CODE(kAudio_FileNotFoundError);
  51. HANDLE_CODE(kAudio_FilePermissionError);
  52. HANDLE_CODE(kAudio_TooManyFilesOpenError);
  53. HANDLE_CODE(kAudio_BadFilePathError);
  54. HANDLE_CODE(kAudio_ParamError);
  55. HANDLE_CODE(kAudio_MemFullError);
  56. HANDLE_CODE(kAudioConverterErr_FormatNotSupported);
  57. HANDLE_CODE(kAudioConverterErr_OperationNotSupported);
  58. HANDLE_CODE(kAudioConverterErr_PropertyNotSupported);
  59. HANDLE_CODE(kAudioConverterErr_InvalidInputSize);
  60. HANDLE_CODE(kAudioConverterErr_InvalidOutputSize);
  61. HANDLE_CODE(kAudioConverterErr_UnspecifiedError);
  62. HANDLE_CODE(kAudioConverterErr_BadPropertySizeError);
  63. HANDLE_CODE(kAudioConverterErr_RequiresPacketDescriptionsError);
  64. HANDLE_CODE(kAudioConverterErr_InputSampleRateOutOfRange);
  65. HANDLE_CODE(kAudioConverterErr_OutputSampleRateOutOfRange);
  66. #undef HANDLE_CODE
  67. default: break;
  68. }
  69. return NULL;
  70. }
  71. static void log_osstatus(int log_level, ca_encoder *ca, const char *context,
  72. OSStatus code)
  73. {
  74. #ifndef _WIN32
  75. CFErrorRef err = CFErrorCreate(kCFAllocatorDefault,
  76. kCFErrorDomainOSStatus, code, NULL);
  77. CFStringRef str = CFErrorCopyDescription(err);
  78. CFIndex length = CFStringGetLength(str);
  79. CFIndex max_size = CFStringGetMaximumSizeForEncoding(length,
  80. kCFStringEncodingUTF8);
  81. char *c_str = malloc(max_size);
  82. if (CFStringGetCString(str, c_str, max_size, kCFStringEncodingUTF8)) {
  83. if (ca)
  84. CA_BLOG(log_level, "Error in %s: %s", context, c_str);
  85. else
  86. CA_LOG(log_level, "Error in %s: %s", context, c_str);
  87. } else {
  88. #endif
  89. const char *code_str = code_to_str(code);
  90. if (ca)
  91. CA_BLOG(log_level, "Error in %s: %s%s%d%s", context,
  92. code_str ? code_str : "",
  93. code_str ? " (" : "",
  94. (int)code,
  95. code_str ? ")" : "");
  96. else
  97. CA_LOG(log_level, "Error in %s: %s%s%d%s", context,
  98. code_str ? code_str : "",
  99. code_str ? " (" : "",
  100. (int)code,
  101. code_str ? ")" : "");
  102. #ifndef _WIN32
  103. }
  104. free(c_str);
  105. CFRelease(str);
  106. CFRelease(err);
  107. #endif
  108. }
  109. static const char *format_id_to_str(UInt32 format_id)
  110. {
  111. #define FORMAT_TO_STR(x) case x: return #x
  112. switch (format_id) {
  113. FORMAT_TO_STR(kAudioFormatLinearPCM);
  114. FORMAT_TO_STR(kAudioFormatAC3);
  115. FORMAT_TO_STR(kAudioFormat60958AC3);
  116. FORMAT_TO_STR(kAudioFormatAppleIMA4);
  117. FORMAT_TO_STR(kAudioFormatMPEG4AAC);
  118. FORMAT_TO_STR(kAudioFormatMPEG4CELP);
  119. FORMAT_TO_STR(kAudioFormatMPEG4HVXC);
  120. FORMAT_TO_STR(kAudioFormatMPEG4TwinVQ);
  121. FORMAT_TO_STR(kAudioFormatMACE3);
  122. FORMAT_TO_STR(kAudioFormatMACE6);
  123. FORMAT_TO_STR(kAudioFormatULaw);
  124. FORMAT_TO_STR(kAudioFormatALaw);
  125. FORMAT_TO_STR(kAudioFormatQDesign);
  126. FORMAT_TO_STR(kAudioFormatQDesign2);
  127. FORMAT_TO_STR(kAudioFormatQUALCOMM);
  128. FORMAT_TO_STR(kAudioFormatMPEGLayer1);
  129. FORMAT_TO_STR(kAudioFormatMPEGLayer2);
  130. FORMAT_TO_STR(kAudioFormatMPEGLayer3);
  131. FORMAT_TO_STR(kAudioFormatTimeCode);
  132. FORMAT_TO_STR(kAudioFormatMIDIStream);
  133. FORMAT_TO_STR(kAudioFormatParameterValueStream);
  134. FORMAT_TO_STR(kAudioFormatAppleLossless);
  135. FORMAT_TO_STR(kAudioFormatMPEG4AAC_HE);
  136. FORMAT_TO_STR(kAudioFormatMPEG4AAC_LD);
  137. FORMAT_TO_STR(kAudioFormatMPEG4AAC_ELD);
  138. FORMAT_TO_STR(kAudioFormatMPEG4AAC_ELD_SBR);
  139. FORMAT_TO_STR(kAudioFormatMPEG4AAC_HE_V2);
  140. FORMAT_TO_STR(kAudioFormatMPEG4AAC_Spatial);
  141. FORMAT_TO_STR(kAudioFormatAMR);
  142. FORMAT_TO_STR(kAudioFormatAudible);
  143. FORMAT_TO_STR(kAudioFormatiLBC);
  144. FORMAT_TO_STR(kAudioFormatDVIIntelIMA);
  145. FORMAT_TO_STR(kAudioFormatMicrosoftGSM);
  146. FORMAT_TO_STR(kAudioFormatAES3);
  147. }
  148. #undef FORMAT_TO_STR
  149. return "Unknown format";
  150. }
  151. static void aac_destroy(void *data)
  152. {
  153. ca_encoder *ca = data;
  154. if (ca->converter)
  155. AudioConverterDispose(ca->converter);
  156. da_free(ca->input_buffer);
  157. bfree(ca->extra_data);
  158. bfree(ca->output_buffer);
  159. bfree(ca);
  160. }
  161. typedef void (*bitrate_enumeration_func)(void *data, UInt32 min, UInt32 max);
  162. static bool enumerate_bitrates(ca_encoder *ca, AudioConverterRef converter,
  163. bitrate_enumeration_func enum_func, void *data)
  164. {
  165. if (!converter && ca)
  166. converter = ca->converter;
  167. UInt32 size;
  168. OSStatus code = AudioConverterGetPropertyInfo(converter,
  169. kAudioConverterApplicableEncodeBitRates,
  170. &size, NULL);
  171. if (code) {
  172. log_osstatus(LOG_WARNING, ca,
  173. "AudioConverterGetPropertyInfo(bitrates)",
  174. code);
  175. return false;
  176. }
  177. if (!size) {
  178. if (ca)
  179. CA_BLOG(LOG_WARNING, "Query for applicable bitrates "
  180. "returned 0 size");
  181. else
  182. CA_LOG(LOG_WARNING, "Query for applicable bitrates "
  183. "returned 0 size");
  184. return false;
  185. }
  186. AudioValueRange *bitrates = malloc(size);
  187. code = AudioConverterGetProperty(converter,
  188. kAudioConverterApplicableEncodeBitRates,
  189. &size, bitrates);
  190. if (code) {
  191. log_osstatus(LOG_WARNING, ca,
  192. "AudioConverterGetProperty(bitrates)", code);
  193. return false;
  194. }
  195. size_t num_bitrates = size / sizeof(AudioValueRange);
  196. for (size_t i = 0; i < num_bitrates; i++)
  197. enum_func(data, (UInt32)bitrates[i].mMinimum,
  198. (UInt32)bitrates[i].mMaximum);
  199. free(bitrates);
  200. return num_bitrates > 0;
  201. }
  202. struct validate_bitrate_helper {
  203. UInt32 bitrate;
  204. bool valid;
  205. };
  206. typedef struct validate_bitrate_helper validate_bitrate_helper;
  207. static void validate_bitrate_func(void *data, UInt32 min, UInt32 max)
  208. {
  209. validate_bitrate_helper *valid = data;
  210. if (valid->bitrate >= min && valid->bitrate <= max)
  211. valid->valid = true;
  212. }
  213. static bool bitrate_valid(ca_encoder *ca, AudioConverterRef converter,
  214. UInt32 bitrate)
  215. {
  216. validate_bitrate_helper helper = {
  217. .bitrate = bitrate,
  218. .valid = false,
  219. };
  220. enumerate_bitrates(ca, converter, validate_bitrate_func, &helper);
  221. return helper.valid;
  222. }
  223. static bool create_encoder(ca_encoder *ca, AudioStreamBasicDescription *in,
  224. AudioStreamBasicDescription *out,
  225. UInt32 format_id, UInt32 bitrate, UInt32 rate_control)
  226. {
  227. #define STATUS_CHECK(c) \
  228. code = c; \
  229. if (code) { \
  230. log_osstatus(LOG_WARNING, ca, #c, code); \
  231. return false; \
  232. }
  233. AudioStreamBasicDescription out_ = {
  234. .mSampleRate = (Float64)ca->samples_per_second,
  235. .mChannelsPerFrame = (UInt32)ca->channels,
  236. .mBytesPerFrame = 0,
  237. .mFramesPerPacket = 0,
  238. .mBitsPerChannel = 0,
  239. .mFormatID = format_id,
  240. .mFormatFlags = 0
  241. };
  242. UInt32 size = sizeof(*out);
  243. OSStatus code;
  244. STATUS_CHECK(AudioFormatGetProperty(kAudioFormatProperty_FormatInfo,
  245. 0, NULL, &size, &out_));
  246. *out = out_;
  247. STATUS_CHECK(AudioConverterNew(in, out, &ca->converter))
  248. STATUS_CHECK(AudioConverterSetProperty(ca->converter,
  249. kAudioCodecPropertyBitRateControlMode,
  250. sizeof(rate_control), &rate_control));
  251. if (!bitrate_valid(ca, NULL, bitrate)) {
  252. CA_BLOG(LOG_WARNING, "Encoder does not support bitrate %u for "
  253. "format %s (0x%x)",
  254. (uint32_t)bitrate, format_id_to_str(format_id),
  255. (uint32_t)format_id);
  256. return false;
  257. }
  258. ca->format_id = format_id;
  259. return true;
  260. #undef STATUS_CHECK
  261. }
  262. static const UInt32 aac_formats[] = {
  263. kAudioFormatMPEG4AAC_HE_V2,
  264. kAudioFormatMPEG4AAC_HE,
  265. kAudioFormatMPEG4AAC,
  266. };
  267. static const UInt32 aac_lc_formats[] = {
  268. kAudioFormatMPEG4AAC,
  269. };
  270. static void *aac_create(obs_data_t *settings, obs_encoder_t *encoder)
  271. {
  272. #define STATUS_CHECK(c) \
  273. code = c; \
  274. if (code) { \
  275. log_osstatus(LOG_ERROR, ca, #c, code); \
  276. goto free; \
  277. }
  278. UInt32 bitrate = (UInt32)obs_data_get_int(settings, "bitrate") * 1000;
  279. if (!bitrate) {
  280. CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
  281. "Invalid bitrate specified");
  282. return NULL;
  283. }
  284. const enum audio_format format = AUDIO_FORMAT_FLOAT;
  285. if (is_audio_planar(format)) {
  286. CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
  287. "Got non-interleaved audio format %d", format);
  288. return NULL;
  289. }
  290. ca_encoder *ca = bzalloc(sizeof(ca_encoder));
  291. ca->encoder = encoder;
  292. ca->format_name = "AAC";
  293. audio_t *audio = obs_encoder_audio(encoder);
  294. const struct audio_output_info *aoi = audio_output_get_info(audio);
  295. ca->channels = audio_output_get_channels(audio);
  296. ca->samples_per_second = audio_output_get_sample_rate(audio);
  297. size_t bytes_per_frame = get_audio_size(format, aoi->speakers, 1);
  298. size_t bits_per_channel = get_audio_bytes_per_channel(format) * 8;
  299. AudioStreamBasicDescription in = {
  300. .mSampleRate = (Float64)ca->samples_per_second,
  301. .mChannelsPerFrame = (UInt32)ca->channels,
  302. .mBytesPerFrame = (UInt32)bytes_per_frame,
  303. .mFramesPerPacket = 1,
  304. .mBytesPerPacket = (UInt32)(1 * bytes_per_frame),
  305. .mBitsPerChannel = (UInt32)bits_per_channel,
  306. .mFormatID = kAudioFormatLinearPCM,
  307. .mFormatFlags = kAudioFormatFlagsNativeEndian |
  308. kAudioFormatFlagIsPacked |
  309. kAudioFormatFlagIsFloat |
  310. 0
  311. };
  312. AudioStreamBasicDescription out;
  313. UInt32 rate_control = kAudioCodecBitRateControlMode_Constant;
  314. #define USE_FORMATS(x) { \
  315. ca->allowed_formats = x; \
  316. ca->num_allowed_formats = sizeof(x)/sizeof(x[0]); \
  317. }
  318. if (obs_data_get_bool(settings, "allow he-aac")) {
  319. USE_FORMATS(aac_formats);
  320. } else {
  321. USE_FORMATS(aac_lc_formats);
  322. }
  323. #undef USE_FORMATS
  324. bool encoder_created = false;
  325. for (size_t i = 0; i < ca->num_allowed_formats; i++) {
  326. UInt32 format_id = ca->allowed_formats[i];
  327. CA_BLOG(LOG_INFO, "Trying format %s (0x%x)",
  328. format_id_to_str(format_id),
  329. (uint32_t)format_id);
  330. if (!create_encoder(ca, &in, &out, format_id, bitrate,
  331. rate_control))
  332. continue;
  333. encoder_created = true;
  334. break;
  335. }
  336. if (!encoder_created) {
  337. CA_BLOG(LOG_ERROR, "Could not create encoder for "
  338. "selected format%s",
  339. ca->num_allowed_formats == 1 ? "" : "s");
  340. goto free;
  341. }
  342. OSStatus code;
  343. UInt32 converter_quality = kAudioConverterQuality_Max;
  344. STATUS_CHECK(AudioConverterSetProperty(ca->converter,
  345. kAudioConverterCodecQuality,
  346. sizeof(converter_quality), &converter_quality));
  347. STATUS_CHECK(AudioConverterSetProperty(ca->converter,
  348. kAudioConverterEncodeBitRate,
  349. sizeof(bitrate), &bitrate));
  350. UInt32 size = sizeof(in);
  351. STATUS_CHECK(AudioConverterGetProperty(ca->converter,
  352. kAudioConverterCurrentInputStreamDescription,
  353. &size, &in));
  354. size = sizeof(out);
  355. STATUS_CHECK(AudioConverterGetProperty(ca->converter,
  356. kAudioConverterCurrentOutputStreamDescription,
  357. &size, &out));
  358. ca->in_frame_size = in.mBytesPerFrame;
  359. ca->in_packets = out.mFramesPerPacket / in.mFramesPerPacket;
  360. ca->in_bytes_required = ca->in_packets * ca->in_frame_size;
  361. ca->out_frames_per_packet = out.mFramesPerPacket;
  362. da_init(ca->input_buffer);
  363. ca->output_buffer_size = out.mBytesPerPacket;
  364. if (out.mBytesPerPacket == 0) {
  365. UInt32 max_packet_size = 0;
  366. size = sizeof(max_packet_size);
  367. code = AudioConverterGetProperty(ca->converter,
  368. kAudioConverterPropertyMaximumOutputPacketSize,
  369. &size, &max_packet_size);
  370. if (code) {
  371. log_osstatus(LOG_WARNING, ca,
  372. "AudioConverterGetProperty(PacketSz)",
  373. code);
  374. ca->output_buffer_size = 32768;
  375. } else {
  376. ca->output_buffer_size = max_packet_size;
  377. }
  378. }
  379. ca->output_buffer = bmalloc(ca->output_buffer_size);
  380. const char *format_name =
  381. out.mFormatID == kAudioFormatMPEG4AAC_HE_V2 ? "HE-AAC v2" :
  382. out.mFormatID == kAudioFormatMPEG4AAC_HE ? "HE-AAC" : "AAC";
  383. CA_BLOG(LOG_INFO, "settings:\n"
  384. "\tmode: %s\n"
  385. "\tbitrate: %u\n"
  386. "\tsample rate: %llu\n"
  387. "\tcbr: %s\n"
  388. "\toutput buffer: %lu",
  389. format_name, (unsigned int)bitrate / 1000,
  390. ca->samples_per_second,
  391. rate_control == kAudioCodecBitRateControlMode_Constant ?
  392. "on" : "off",
  393. (unsigned long)ca->output_buffer_size);
  394. return ca;
  395. free:
  396. aac_destroy(ca);
  397. return NULL;
  398. }
  399. static OSStatus complex_input_data_proc(AudioConverterRef inAudioConverter,
  400. UInt32 *ioNumberDataPackets, AudioBufferList *ioData,
  401. AudioStreamPacketDescription **outDataPacketDescription,
  402. void *inUserData)
  403. {
  404. UNUSED_PARAMETER(inAudioConverter);
  405. UNUSED_PARAMETER(outDataPacketDescription);
  406. ca_encoder *ca = inUserData;
  407. if (ca->bytes_read) {
  408. da_erase_range(ca->input_buffer, 0, ca->bytes_read);
  409. ca->bytes_read = 0;
  410. }
  411. if (ca->input_buffer.num < ca->in_bytes_required) {
  412. *ioNumberDataPackets = 0;
  413. ioData->mBuffers[0].mData = NULL;
  414. return 1;
  415. }
  416. *ioNumberDataPackets =
  417. (UInt32)(ca->in_bytes_required / ca->in_frame_size);
  418. ioData->mNumberBuffers = 1;
  419. ioData->mBuffers[0].mData = ca->input_buffer.array;
  420. ioData->mBuffers[0].mNumberChannels = (UInt32)ca->channels;
  421. ioData->mBuffers[0].mDataByteSize = (UInt32)ca->in_bytes_required;
  422. ca->bytes_read += ca->in_packets * ca->in_frame_size;
  423. return 0;
  424. }
  425. static bool aac_encode(void *data, struct encoder_frame *frame,
  426. struct encoder_packet *packet, bool *received_packet)
  427. {
  428. ca_encoder *ca = data;
  429. da_push_back_array(ca->input_buffer, frame->data[0],
  430. frame->linesize[0]);
  431. if ((ca->input_buffer.num - ca->bytes_read) < ca->in_bytes_required)
  432. return true;
  433. UInt32 packets = 1;
  434. AudioBufferList buffer_list = {
  435. .mNumberBuffers = 1,
  436. .mBuffers = { {
  437. .mNumberChannels = (UInt32)ca->channels,
  438. .mDataByteSize = (UInt32)ca->output_buffer_size,
  439. .mData = ca->output_buffer,
  440. } },
  441. };
  442. AudioStreamPacketDescription out_desc = { 0 };
  443. OSStatus code = AudioConverterFillComplexBuffer(ca->converter,
  444. complex_input_data_proc, ca, &packets,
  445. &buffer_list, &out_desc);
  446. if (code && code != 1) {
  447. log_osstatus(LOG_ERROR, ca, "AudioConverterFillComplexBuffer",
  448. code);
  449. return false;
  450. }
  451. if (!(*received_packet = packets > 0))
  452. return true;
  453. packet->pts = ca->total_samples;
  454. packet->dts = ca->total_samples;
  455. packet->timebase_num = 1;
  456. packet->timebase_den = (uint32_t)ca->samples_per_second;
  457. packet->type = OBS_ENCODER_AUDIO;
  458. packet->size = out_desc.mDataByteSize;
  459. packet->data =
  460. (uint8_t*)buffer_list.mBuffers[0].mData + out_desc.mStartOffset;
  461. ca->total_samples += ca->bytes_read / ca->in_frame_size;
  462. return true;
  463. }
  464. static void aac_audio_info(void *data, struct audio_convert_info *info)
  465. {
  466. UNUSED_PARAMETER(data);
  467. info->format = AUDIO_FORMAT_FLOAT;
  468. }
  469. static size_t aac_frame_size(void *data)
  470. {
  471. ca_encoder *ca = data;
  472. return ca->out_frames_per_packet;
  473. }
  474. /* The following code was extracted from encca_aac.c in HandBrake's libhb */
  475. #define MP4ESDescrTag 0x03
  476. #define MP4DecConfigDescrTag 0x04
  477. #define MP4DecSpecificDescrTag 0x05
  478. // based off of mov_mp4_read_descr_len from mov.c in ffmpeg's libavformat
  479. static int read_descr_len(uint8_t **buffer)
  480. {
  481. int len = 0;
  482. int count = 4;
  483. while (count--)
  484. {
  485. int c = *(*buffer)++;
  486. len = (len << 7) | (c & 0x7f);
  487. if (!(c & 0x80))
  488. break;
  489. }
  490. return len;
  491. }
  492. // based off of mov_mp4_read_descr from mov.c in ffmpeg's libavformat
  493. static int read_descr(uint8_t **buffer, int *tag)
  494. {
  495. *tag = *(*buffer)++;
  496. return read_descr_len(buffer);
  497. }
  498. // based off of mov_read_esds from mov.c in ffmpeg's libavformat
  499. static void read_esds_desc_ext(uint8_t* desc_ext, uint8_t **buffer,
  500. uint32_t *size, bool version_flags)
  501. {
  502. uint8_t *esds = desc_ext;
  503. int tag, len;
  504. *size = 0;
  505. if (version_flags)
  506. esds += 4; // version + flags
  507. read_descr(&esds, &tag);
  508. esds += 2; // ID
  509. if (tag == MP4ESDescrTag)
  510. esds++; // priority
  511. read_descr(&esds, &tag);
  512. if (tag == MP4DecConfigDescrTag) {
  513. esds++; // object type id
  514. esds++; // stream type
  515. esds += 3; // buffer size db
  516. esds += 4; // max bitrate
  517. esds += 4; // average bitrate
  518. len = read_descr(&esds, &tag);
  519. if (tag == MP4DecSpecificDescrTag) {
  520. *buffer = bzalloc(len + 8);
  521. if (*buffer) {
  522. memcpy(*buffer, esds, len);
  523. *size = len;
  524. }
  525. }
  526. }
  527. }
  528. /* extracted code ends here */
  529. static void query_extra_data(ca_encoder *ca)
  530. {
  531. UInt32 size = 0;
  532. OSStatus code;
  533. code = AudioConverterGetPropertyInfo(ca->converter,
  534. kAudioConverterCompressionMagicCookie,
  535. &size, NULL);
  536. if (code) {
  537. log_osstatus(LOG_ERROR, ca,
  538. "AudioConverterGetPropertyInfo(magic_cookie)",
  539. code);
  540. return;
  541. }
  542. if (!size) {
  543. CA_BLOG(LOG_WARNING, "Got 0 data size info for magic_cookie");
  544. return;
  545. }
  546. uint8_t *extra_data = malloc(size);
  547. code = AudioConverterGetProperty(ca->converter,
  548. kAudioConverterCompressionMagicCookie,
  549. &size, extra_data);
  550. if (code) {
  551. log_osstatus(LOG_ERROR, ca,
  552. "AudioConverterGetProperty(magic_cookie)",
  553. code);
  554. goto free;
  555. }
  556. if (!size) {
  557. CA_BLOG(LOG_WARNING, "Got 0 data size for magic_cookie");
  558. goto free;
  559. }
  560. read_esds_desc_ext(extra_data, &ca->extra_data, &ca->extra_data_size,
  561. false);
  562. free:
  563. free(extra_data);
  564. }
  565. static bool aac_extra_data(void *data, uint8_t **extra_data, size_t *size)
  566. {
  567. ca_encoder *ca = data;
  568. if (!ca->extra_data)
  569. query_extra_data(ca);
  570. if (!ca->extra_data_size)
  571. return false;
  572. *extra_data = ca->extra_data;
  573. *size = ca->extra_data_size;
  574. return true;
  575. }
  576. static AudioConverterRef get_default_converter(UInt32 format_id)
  577. {
  578. UInt32 bytes_per_frame = 8;
  579. UInt32 channels = 2;
  580. UInt32 bits_per_channel = bytes_per_frame / channels * 8;
  581. AudioStreamBasicDescription in = {
  582. .mSampleRate = 44100,
  583. .mChannelsPerFrame = channels,
  584. .mBytesPerFrame = bytes_per_frame,
  585. .mFramesPerPacket = 1,
  586. .mBytesPerPacket = 1 * bytes_per_frame,
  587. .mBitsPerChannel = bits_per_channel,
  588. .mFormatID = kAudioFormatLinearPCM,
  589. .mFormatFlags = kAudioFormatFlagsNativeEndian |
  590. kAudioFormatFlagIsPacked |
  591. kAudioFormatFlagIsFloat |
  592. 0
  593. };
  594. AudioStreamBasicDescription out = {
  595. .mSampleRate = 44100,
  596. .mChannelsPerFrame = channels,
  597. .mBytesPerFrame = 0,
  598. .mFramesPerPacket = 0,
  599. .mBitsPerChannel = 0,
  600. .mFormatID = format_id,
  601. .mFormatFlags = 0
  602. };
  603. UInt32 size = sizeof(out);
  604. OSStatus code = AudioFormatGetProperty(kAudioFormatProperty_FormatInfo,
  605. 0, NULL, &size, &out);
  606. if (code) {
  607. log_osstatus(LOG_WARNING, NULL,
  608. "AudioFormatGetProperty(format_info)", code);
  609. return NULL;
  610. }
  611. AudioConverterRef converter;
  612. code = AudioConverterNew(&in, &out, &converter);
  613. if (code) {
  614. log_osstatus(LOG_WARNING, NULL, "AudioConverterNew", code);
  615. return NULL;
  616. }
  617. return converter;
  618. }
  619. static AudioConverterRef aac_default_converter(void)
  620. {
  621. return get_default_converter(kAudioFormatMPEG4AAC);
  622. }
  623. static AudioConverterRef he_aac_default_converter(void)
  624. {
  625. return get_default_converter(kAudioFormatMPEG4AAC_HE);
  626. }
  627. struct find_matching_bitrate_helper {
  628. UInt32 bitrate;
  629. UInt32 best_match;
  630. int diff;
  631. };
  632. typedef struct find_matching_bitrate_helper find_matching_bitrate_helper;
  633. static void find_matching_bitrate_func(void *data, UInt32 min, UInt32 max)
  634. {
  635. find_matching_bitrate_helper *helper = data;
  636. int min_diff = abs((int)helper->bitrate - (int)min);
  637. int max_diff = abs((int)helper->bitrate - (int)max);
  638. if (min_diff < helper->diff) {
  639. helper->best_match = min;
  640. helper->diff = min_diff;
  641. }
  642. if (max_diff < helper->diff) {
  643. helper->best_match = max;
  644. helper->diff = max_diff;
  645. }
  646. }
  647. static UInt32 find_matching_bitrate(UInt32 bitrate)
  648. {
  649. find_matching_bitrate_helper helper = {
  650. .bitrate = bitrate * 1000,
  651. .best_match = 0,
  652. .diff = INT_MAX,
  653. };
  654. AudioConverterRef converter = aac_default_converter();
  655. if (!converter) {
  656. CA_LOG(LOG_ERROR, "Could not get converter to match "
  657. "default bitrate");
  658. return bitrate;
  659. }
  660. bool has_bitrates = enumerate_bitrates(NULL, converter,
  661. find_matching_bitrate_func, &helper);
  662. AudioConverterDispose(converter);
  663. if (!has_bitrates) {
  664. CA_LOG(LOG_ERROR, "No bitrates found while matching "
  665. "default bitrate");
  666. AudioConverterDispose(converter);
  667. return bitrate;
  668. }
  669. if (helper.best_match != helper.bitrate)
  670. CA_LOG(LOG_INFO, "Returning closest matching bitrate %u "
  671. "instead of requested bitrate %u",
  672. (uint32_t)helper.best_match / 1000,
  673. (uint32_t)bitrate);
  674. return helper.best_match / 1000;
  675. }
  676. static void aac_defaults(obs_data_t *settings)
  677. {
  678. obs_data_set_default_int(settings, "bitrate",
  679. find_matching_bitrate(128));
  680. obs_data_set_default_bool(settings, "allow he-aac", true);
  681. }
  682. struct add_bitrates_helper {
  683. DARRAY(UInt32) bitrates;
  684. };
  685. typedef struct add_bitrates_helper add_bitrates_helper;
  686. static void add_bitrates_func(void *data, UInt32 min, UInt32 max)
  687. {
  688. add_bitrates_helper *helper = data;
  689. if (da_find(helper->bitrates, &min, 0) == DARRAY_INVALID)
  690. da_push_back(helper->bitrates, &min);
  691. if (da_find(helper->bitrates, &max, 0) == DARRAY_INVALID)
  692. da_push_back(helper->bitrates, &max);
  693. }
  694. static int bitrate_compare(const void *data1, const void *data2)
  695. {
  696. const UInt32 *bitrate1 = data1;
  697. const UInt32 *bitrate2 = data2;
  698. return (int)*bitrate1 - (int)*bitrate2;
  699. }
  700. static void add_bitrates(obs_property_t *prop, ca_encoder *ca)
  701. {
  702. add_bitrates_helper helper = { 0 };
  703. const size_t num_formats = ca ?
  704. ca->num_allowed_formats :
  705. sizeof(aac_formats)/sizeof(aac_formats[0]);
  706. const UInt32 *allowed_formats = ca ? ca->allowed_formats : aac_formats;
  707. for (size_t i = 0; i < num_formats; i++)
  708. enumerate_bitrates(ca,
  709. get_default_converter(allowed_formats[i]),
  710. add_bitrates_func, &helper);
  711. if (!helper.bitrates.num) {
  712. CA_BLOG(LOG_ERROR, "Enumeration found no available bitrates");
  713. return;
  714. }
  715. qsort(helper.bitrates.array, helper.bitrates.num, sizeof(UInt32),
  716. bitrate_compare);
  717. struct dstr str = { 0 };
  718. for (size_t i = 0; i < helper.bitrates.num; i++) {
  719. dstr_printf(&str, "%u",
  720. (uint32_t)helper.bitrates.array[i]/1000);
  721. obs_property_list_add_int(prop, str.array,
  722. helper.bitrates.array[i]/1000);
  723. }
  724. dstr_free(&str);
  725. da_free(helper.bitrates);
  726. }
  727. static obs_properties_t *aac_properties(void *data)
  728. {
  729. ca_encoder *ca = data;
  730. obs_properties_t *props = obs_properties_create();
  731. obs_property_t *p = obs_properties_add_list(props, "bitrate",
  732. obs_module_text("Bitrate"),
  733. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  734. add_bitrates(p, ca);
  735. obs_properties_add_bool(props, "allow he-aac",
  736. obs_module_text("AllowHEAAC"));
  737. return props;
  738. }
  739. static struct obs_encoder_info aac_info = {
  740. .id = "CoreAudio_AAC",
  741. .type = OBS_ENCODER_AUDIO,
  742. .codec = "AAC",
  743. .get_name = aac_get_name,
  744. .destroy = aac_destroy,
  745. .create = aac_create,
  746. .encode = aac_encode,
  747. .get_frame_size = aac_frame_size,
  748. .get_audio_info = aac_audio_info,
  749. .get_extra_data = aac_extra_data,
  750. .get_defaults = aac_defaults,
  751. .get_properties = aac_properties,
  752. };
  753. OBS_DECLARE_MODULE()
  754. OBS_MODULE_USE_DEFAULT_LOCALE("coreaudio-encoder", "en-US")
  755. bool obs_module_load(void)
  756. {
  757. #ifdef _WIN32
  758. if (!load_core_audio()) {
  759. CA_LOG(LOG_WARNING, "Couldn't load CoreAudio AAC encoder");
  760. return true;
  761. }
  762. CA_LOG(LOG_INFO, "Adding CoreAudio AAC encoder");
  763. #endif
  764. obs_register_encoder(&aac_info);
  765. return true;
  766. }
  767. #ifdef _WIN32
  768. void obs_module_unload(void)
  769. {
  770. unload_core_audio();
  771. }
  772. #endif