encoder.c 19 KB

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