encoder.cpp 28 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169
  1. #include <util/darray.h>
  2. #include <util/dstr.hpp>
  3. #include <obs-module.h>
  4. #include <algorithm>
  5. #include <cstdlib>
  6. #include <initializer_list>
  7. #include <memory>
  8. #include <mutex>
  9. #include <vector>
  10. #ifndef _WIN32
  11. #include <AudioToolbox/AudioToolbox.h>
  12. #endif
  13. #define CA_LOG(level, format, ...) \
  14. blog(level, "[CoreAudio encoder]: " format, ##__VA_ARGS__)
  15. #define CA_LOG_ENCODER(format_name, encoder, level, format, ...) \
  16. blog(level, "[CoreAudio %s: '%s']: " format, \
  17. format_name, obs_encoder_get_name(encoder), \
  18. ##__VA_ARGS__)
  19. #define CA_BLOG(level, format, ...) \
  20. CA_LOG_ENCODER(ca->format_name, ca->encoder, level, format, \
  21. ##__VA_ARGS__)
  22. #define CA_CO_LOG(level, format, ...) \
  23. do { \
  24. if (ca) \
  25. CA_BLOG(level, format, ##__VA_ARGS__); \
  26. else \
  27. CA_LOG(level, format, ##__VA_ARGS__); \
  28. } while (false)
  29. #ifdef _WIN32
  30. #include "windows-imports.h"
  31. #endif
  32. using namespace std;
  33. namespace {
  34. struct asbd_builder {
  35. AudioStreamBasicDescription asbd;
  36. asbd_builder &sample_rate(Float64 rate)
  37. {
  38. asbd.mSampleRate = rate;
  39. return *this;
  40. }
  41. asbd_builder &format_id(UInt32 format)
  42. {
  43. asbd.mFormatID = format;
  44. return *this;
  45. }
  46. asbd_builder &format_flags(UInt32 flags)
  47. {
  48. asbd.mFormatFlags = flags;
  49. return *this;
  50. }
  51. asbd_builder &bytes_per_packet(UInt32 bytes)
  52. {
  53. asbd.mBytesPerPacket = bytes;
  54. return *this;
  55. }
  56. asbd_builder &frames_per_packet(UInt32 frames)
  57. {
  58. asbd.mFramesPerPacket = frames;
  59. return *this;
  60. }
  61. asbd_builder &bytes_per_frame(UInt32 bytes)
  62. {
  63. asbd.mBytesPerFrame = bytes;
  64. return *this;
  65. }
  66. asbd_builder &channels_per_frame(UInt32 channels)
  67. {
  68. asbd.mChannelsPerFrame = channels;
  69. return *this;
  70. }
  71. asbd_builder &bits_per_channel(UInt32 bits)
  72. {
  73. asbd.mBitsPerChannel = bits;
  74. return *this;
  75. }
  76. };
  77. struct ca_encoder {
  78. obs_encoder_t *encoder;
  79. const char *format_name;
  80. UInt32 format_id;
  81. const initializer_list<UInt32> *allowed_formats;
  82. AudioConverterRef converter;
  83. size_t output_buffer_size;
  84. vector<uint8_t> output_buffer;
  85. size_t out_frames_per_packet;
  86. size_t in_packets;
  87. size_t in_frame_size;
  88. size_t in_bytes_required;
  89. DARRAY(uint8_t) input_buffer;
  90. size_t bytes_read;
  91. uint64_t total_samples;
  92. uint64_t samples_per_second;
  93. vector<uint8_t> extra_data;
  94. size_t channels;
  95. ~ca_encoder()
  96. {
  97. if (converter)
  98. AudioConverterDispose(converter);
  99. da_free(input_buffer);
  100. }
  101. };
  102. typedef struct ca_encoder ca_encoder;
  103. }
  104. namespace std {
  105. #ifndef _WIN32
  106. template <>
  107. struct default_delete<remove_pointer<CFErrorRef>::type> {
  108. void operator()(remove_pointer<CFErrorRef>::type *err)
  109. {
  110. CFRelease(err);
  111. }
  112. };
  113. template <>
  114. struct default_delete<remove_pointer<CFStringRef>::type> {
  115. void operator()(remove_pointer<CFStringRef>::type *str)
  116. {
  117. CFRelease(str);
  118. }
  119. };
  120. #endif
  121. template <>
  122. struct default_delete<remove_pointer<AudioConverterRef>::type> {
  123. void operator()(AudioConverterRef converter)
  124. {
  125. AudioConverterDispose(converter);
  126. }
  127. };
  128. }
  129. template <typename T>
  130. using cf_ptr = unique_ptr<typename remove_pointer<T>::type>;
  131. #ifndef _MSC_VER
  132. __attribute__((__format__(__printf__, 3, 4)))
  133. #endif
  134. static void log_to_dstr(DStr &str, ca_encoder *ca, const char *fmt, ...)
  135. {
  136. dstr prev_str = *static_cast<dstr*>(str);
  137. va_list args;
  138. va_start(args, fmt);
  139. dstr_vcatf(str, fmt, args);
  140. va_end(args);
  141. if (str->array)
  142. return;
  143. char array[4096];
  144. va_start(args, fmt);
  145. vsnprintf(array, 4096, fmt, args);
  146. va_end(args);
  147. array[4095] = 0;
  148. if (!prev_str.array && !prev_str.len)
  149. CA_CO_LOG(LOG_ERROR, "Could not allocate buffer for logging:"
  150. "\n'%s'", array);
  151. else
  152. CA_CO_LOG(LOG_ERROR, "Could not allocate buffer for logging:"
  153. "\n'%s'\nPrevious log entries:\n%s",
  154. array, prev_str.array);
  155. bfree(prev_str.array);
  156. }
  157. static const char *flush_log(DStr &log)
  158. {
  159. if (!log->array || !log->len)
  160. return "";
  161. if (log->array[log->len - 1] == '\n') {
  162. log->array[log->len - 1] = 0; //Get rid of last newline
  163. log->len -= 1;
  164. }
  165. return log->array;
  166. }
  167. #define CA_CO_DLOG_(level, format) \
  168. CA_CO_LOG(level, format "%s%s", \
  169. log->array ? ":\n" : "", flush_log(log))
  170. #define CA_CO_DLOG(level, format, ...) \
  171. CA_CO_LOG(level, format "%s%s", ##__VA_ARGS__, \
  172. log->array ? ":\n" : "", flush_log(log))
  173. static const char *aac_get_name(void)
  174. {
  175. return obs_module_text("CoreAudioAAC");
  176. }
  177. static const char *code_to_str(OSStatus code)
  178. {
  179. switch (code) {
  180. #define HANDLE_CODE(c) case c: return #c
  181. HANDLE_CODE(kAudio_UnimplementedError);
  182. HANDLE_CODE(kAudio_FileNotFoundError);
  183. HANDLE_CODE(kAudio_FilePermissionError);
  184. HANDLE_CODE(kAudio_TooManyFilesOpenError);
  185. HANDLE_CODE(kAudio_BadFilePathError);
  186. HANDLE_CODE(kAudio_ParamError);
  187. HANDLE_CODE(kAudio_MemFullError);
  188. HANDLE_CODE(kAudioConverterErr_FormatNotSupported);
  189. HANDLE_CODE(kAudioConverterErr_OperationNotSupported);
  190. HANDLE_CODE(kAudioConverterErr_PropertyNotSupported);
  191. HANDLE_CODE(kAudioConverterErr_InvalidInputSize);
  192. HANDLE_CODE(kAudioConverterErr_InvalidOutputSize);
  193. HANDLE_CODE(kAudioConverterErr_UnspecifiedError);
  194. HANDLE_CODE(kAudioConverterErr_BadPropertySizeError);
  195. HANDLE_CODE(kAudioConverterErr_RequiresPacketDescriptionsError);
  196. HANDLE_CODE(kAudioConverterErr_InputSampleRateOutOfRange);
  197. HANDLE_CODE(kAudioConverterErr_OutputSampleRateOutOfRange);
  198. #undef HANDLE_CODE
  199. default: break;
  200. }
  201. return NULL;
  202. }
  203. static DStr osstatus_to_dstr(OSStatus code)
  204. {
  205. DStr result;
  206. #ifndef _WIN32
  207. cf_ptr<CFErrorRef> err{CFErrorCreate(kCFAllocatorDefault,
  208. kCFErrorDomainOSStatus, code, NULL)};
  209. cf_ptr<CFStringRef> str{CFErrorCopyDescription(err.get())};
  210. CFIndex length = CFStringGetLength(str.get());
  211. CFIndex max_size = CFStringGetMaximumSizeForEncoding(length,
  212. kCFStringEncodingUTF8);
  213. dstr_ensure_capacity(result, max_size);
  214. if (result->array && CFStringGetCString(str.get(), result->array,
  215. max_size, kCFStringEncodingUTF8)) {
  216. dstr_resize(result, strlen(result->array));
  217. return result;
  218. }
  219. #endif
  220. const char *code_str = code_to_str(code);
  221. dstr_printf(result, "%s%s%d%s",
  222. code_str ? code_str : "",
  223. code_str ? " (" : "",
  224. static_cast<int>(code),
  225. code_str ? ")" : "");
  226. return result;
  227. }
  228. static void log_osstatus(int log_level, ca_encoder *ca, const char *context,
  229. OSStatus code)
  230. {
  231. DStr str = osstatus_to_dstr(code);
  232. if (ca)
  233. CA_BLOG(log_level, "Error in %s: %s", context, str->array);
  234. else
  235. CA_LOG(log_level, "Error in %s: %s", context, str->array);
  236. }
  237. static const char *format_id_to_str(UInt32 format_id)
  238. {
  239. #define FORMAT_TO_STR(x) case x: return #x
  240. switch (format_id) {
  241. FORMAT_TO_STR(kAudioFormatLinearPCM);
  242. FORMAT_TO_STR(kAudioFormatAC3);
  243. FORMAT_TO_STR(kAudioFormat60958AC3);
  244. FORMAT_TO_STR(kAudioFormatAppleIMA4);
  245. FORMAT_TO_STR(kAudioFormatMPEG4AAC);
  246. FORMAT_TO_STR(kAudioFormatMPEG4CELP);
  247. FORMAT_TO_STR(kAudioFormatMPEG4HVXC);
  248. FORMAT_TO_STR(kAudioFormatMPEG4TwinVQ);
  249. FORMAT_TO_STR(kAudioFormatMACE3);
  250. FORMAT_TO_STR(kAudioFormatMACE6);
  251. FORMAT_TO_STR(kAudioFormatULaw);
  252. FORMAT_TO_STR(kAudioFormatALaw);
  253. FORMAT_TO_STR(kAudioFormatQDesign);
  254. FORMAT_TO_STR(kAudioFormatQDesign2);
  255. FORMAT_TO_STR(kAudioFormatQUALCOMM);
  256. FORMAT_TO_STR(kAudioFormatMPEGLayer1);
  257. FORMAT_TO_STR(kAudioFormatMPEGLayer2);
  258. FORMAT_TO_STR(kAudioFormatMPEGLayer3);
  259. FORMAT_TO_STR(kAudioFormatTimeCode);
  260. FORMAT_TO_STR(kAudioFormatMIDIStream);
  261. FORMAT_TO_STR(kAudioFormatParameterValueStream);
  262. FORMAT_TO_STR(kAudioFormatAppleLossless);
  263. FORMAT_TO_STR(kAudioFormatMPEG4AAC_HE);
  264. FORMAT_TO_STR(kAudioFormatMPEG4AAC_LD);
  265. FORMAT_TO_STR(kAudioFormatMPEG4AAC_ELD);
  266. FORMAT_TO_STR(kAudioFormatMPEG4AAC_ELD_SBR);
  267. FORMAT_TO_STR(kAudioFormatMPEG4AAC_HE_V2);
  268. FORMAT_TO_STR(kAudioFormatMPEG4AAC_Spatial);
  269. FORMAT_TO_STR(kAudioFormatAMR);
  270. FORMAT_TO_STR(kAudioFormatAudible);
  271. FORMAT_TO_STR(kAudioFormatiLBC);
  272. FORMAT_TO_STR(kAudioFormatDVIIntelIMA);
  273. FORMAT_TO_STR(kAudioFormatMicrosoftGSM);
  274. FORMAT_TO_STR(kAudioFormatAES3);
  275. }
  276. #undef FORMAT_TO_STR
  277. return "Unknown format";
  278. }
  279. static void aac_destroy(void *data)
  280. {
  281. ca_encoder *ca = static_cast<ca_encoder*>(data);
  282. delete ca;
  283. }
  284. template <typename Func>
  285. static bool query_converter_property_raw(DStr &log, ca_encoder *ca,
  286. AudioFormatPropertyID property,
  287. const char *get_property_info, const char *get_property,
  288. AudioConverterRef converter, Func &&func)
  289. {
  290. UInt32 size = 0;
  291. OSStatus code = AudioConverterGetPropertyInfo(converter, property,
  292. &size, nullptr);
  293. if (code) {
  294. log_to_dstr(log, ca, "%s: %s\n", get_property_info,
  295. osstatus_to_dstr(code)->array);
  296. return false;
  297. }
  298. if (!size) {
  299. log_to_dstr(log, ca, "%s returned 0 size\n", get_property_info);
  300. return false;
  301. }
  302. vector<uint8_t> buffer;
  303. try {
  304. buffer.resize(size);
  305. } catch (...) {
  306. log_to_dstr(log, ca, "Failed to allocate %u bytes for %s\n",
  307. static_cast<uint32_t>(size), get_property);
  308. return false;
  309. }
  310. code = AudioConverterGetProperty(converter, property, &size,
  311. buffer.data());
  312. if (code) {
  313. log_to_dstr(log, ca, "%s: %s\n", get_property,
  314. osstatus_to_dstr(code)->array);
  315. return false;
  316. }
  317. func(size, static_cast<void*>(buffer.data()));
  318. return true;
  319. }
  320. #define EXPAND_CONVERTER_NAMES(x) x, \
  321. "AudioConverterGetPropertyInfo(" #x ")", \
  322. "AudioConverterGetProperty(" #x ")"
  323. template <typename Func>
  324. static bool enumerate_bitrates(DStr &log, ca_encoder *ca,
  325. AudioConverterRef converter, Func &&func)
  326. {
  327. auto helper = [&](UInt32 size, void *data)
  328. {
  329. auto range = static_cast<AudioValueRange*>(data);
  330. size_t num_ranges = size / sizeof(AudioValueRange);
  331. for (size_t i = 0; i < num_ranges; i++)
  332. func(static_cast<UInt32>(range[i].mMinimum),
  333. static_cast<UInt32>(range[i].mMaximum));
  334. };
  335. return query_converter_property_raw(log, ca, EXPAND_CONVERTER_NAMES(
  336. kAudioConverterApplicableEncodeBitRates),
  337. converter, helper);
  338. }
  339. static bool bitrate_valid(DStr &log, ca_encoder *ca,
  340. AudioConverterRef converter, UInt32 bitrate)
  341. {
  342. bool valid = false;
  343. auto helper = [&](UInt32 min_, UInt32 max_)
  344. {
  345. if (min_ == bitrate || max_ == bitrate)
  346. valid = true;
  347. };
  348. enumerate_bitrates(log, ca, converter, helper);
  349. return valid;
  350. }
  351. static bool create_encoder(DStr &log, ca_encoder *ca,
  352. AudioStreamBasicDescription *in,
  353. AudioStreamBasicDescription *out,
  354. UInt32 format_id, UInt32 bitrate, UInt32 rate_control)
  355. {
  356. #define STATUS_CHECK(c) \
  357. code = c; \
  358. if (code) { \
  359. log_to_dstr(log, ca, #c " returned %s", \
  360. osstatus_to_dstr(code)->array); \
  361. return false; \
  362. }
  363. auto out_ = asbd_builder()
  364. .sample_rate((Float64)ca->samples_per_second)
  365. .channels_per_frame((UInt32)ca->channels)
  366. .format_id(format_id)
  367. .asbd;
  368. UInt32 size = sizeof(*out);
  369. OSStatus code;
  370. STATUS_CHECK(AudioFormatGetProperty(kAudioFormatProperty_FormatInfo,
  371. 0, NULL, &size, &out_));
  372. *out = out_;
  373. STATUS_CHECK(AudioConverterNew(in, out, &ca->converter))
  374. STATUS_CHECK(AudioConverterSetProperty(ca->converter,
  375. kAudioCodecPropertyBitRateControlMode,
  376. sizeof(rate_control), &rate_control));
  377. if (!bitrate_valid(log, ca, ca->converter, bitrate)) {
  378. log_to_dstr(log, ca, "Encoder does not support bitrate %u "
  379. "for format %s (0x%x)\n",
  380. (uint32_t)bitrate, format_id_to_str(format_id),
  381. (uint32_t)format_id);
  382. return false;
  383. }
  384. ca->format_id = format_id;
  385. return true;
  386. #undef STATUS_CHECK
  387. }
  388. static const initializer_list<UInt32> aac_formats = {
  389. kAudioFormatMPEG4AAC_HE_V2,
  390. kAudioFormatMPEG4AAC_HE,
  391. kAudioFormatMPEG4AAC,
  392. };
  393. static const initializer_list<UInt32> aac_lc_formats = {
  394. kAudioFormatMPEG4AAC,
  395. };
  396. static void *aac_create(obs_data_t *settings, obs_encoder_t *encoder)
  397. {
  398. #define STATUS_CHECK(c) \
  399. code = c; \
  400. if (code) { \
  401. log_osstatus(LOG_ERROR, ca.get(), #c, code); \
  402. return nullptr; \
  403. }
  404. UInt32 bitrate = (UInt32)obs_data_get_int(settings, "bitrate") * 1000;
  405. if (!bitrate) {
  406. CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
  407. "Invalid bitrate specified");
  408. return NULL;
  409. }
  410. const enum audio_format format = AUDIO_FORMAT_FLOAT;
  411. if (is_audio_planar(format)) {
  412. CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
  413. "Got non-interleaved audio format %d", format);
  414. return NULL;
  415. }
  416. unique_ptr<ca_encoder> ca;
  417. try {
  418. ca.reset(new ca_encoder());
  419. } catch (...) {
  420. CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
  421. "Could not allocate encoder");
  422. return nullptr;
  423. }
  424. ca->encoder = encoder;
  425. ca->format_name = "AAC";
  426. audio_t *audio = obs_encoder_audio(encoder);
  427. const struct audio_output_info *aoi = audio_output_get_info(audio);
  428. ca->channels = audio_output_get_channels(audio);
  429. ca->samples_per_second = audio_output_get_sample_rate(audio);
  430. size_t bytes_per_frame = get_audio_size(format, aoi->speakers, 1);
  431. size_t bits_per_channel = get_audio_bytes_per_channel(format) * 8;
  432. auto in = asbd_builder()
  433. .sample_rate((Float64)ca->samples_per_second)
  434. .channels_per_frame((UInt32)ca->channels)
  435. .bytes_per_frame((UInt32)bytes_per_frame)
  436. .frames_per_packet(1)
  437. .bytes_per_packet((UInt32)(1 * bytes_per_frame))
  438. .bits_per_channel((UInt32)bits_per_channel)
  439. .format_id(kAudioFormatLinearPCM)
  440. .format_flags(kAudioFormatFlagsNativeEndian |
  441. kAudioFormatFlagIsPacked |
  442. kAudioFormatFlagIsFloat |
  443. 0)
  444. .asbd;
  445. AudioStreamBasicDescription out;
  446. UInt32 rate_control = kAudioCodecBitRateControlMode_Constant;
  447. if (obs_data_get_bool(settings, "allow he-aac")) {
  448. ca->allowed_formats = &aac_formats;
  449. } else {
  450. ca->allowed_formats = &aac_lc_formats;
  451. }
  452. DStr log;
  453. bool encoder_created = false;
  454. for (UInt32 format_id : *ca->allowed_formats) {
  455. log_to_dstr(log, ca.get(), "Trying format %s (0x%x)\n",
  456. format_id_to_str(format_id),
  457. (uint32_t)format_id);
  458. if (!create_encoder(log, ca.get(), &in, &out, format_id,
  459. bitrate, rate_control))
  460. continue;
  461. encoder_created = true;
  462. break;
  463. }
  464. if (!encoder_created) {
  465. CA_CO_DLOG(LOG_ERROR, "Could not create encoder for "
  466. "selected format%s",
  467. ca->allowed_formats->size() == 1 ? "" : "s");
  468. return nullptr;
  469. }
  470. if (log->len)
  471. CA_CO_DLOG_(LOG_DEBUG, "Encoder created");
  472. OSStatus code;
  473. UInt32 converter_quality = kAudioConverterQuality_Max;
  474. STATUS_CHECK(AudioConverterSetProperty(ca->converter,
  475. kAudioConverterCodecQuality,
  476. sizeof(converter_quality), &converter_quality));
  477. STATUS_CHECK(AudioConverterSetProperty(ca->converter,
  478. kAudioConverterEncodeBitRate,
  479. sizeof(bitrate), &bitrate));
  480. UInt32 size = sizeof(in);
  481. STATUS_CHECK(AudioConverterGetProperty(ca->converter,
  482. kAudioConverterCurrentInputStreamDescription,
  483. &size, &in));
  484. size = sizeof(out);
  485. STATUS_CHECK(AudioConverterGetProperty(ca->converter,
  486. kAudioConverterCurrentOutputStreamDescription,
  487. &size, &out));
  488. ca->in_frame_size = in.mBytesPerFrame;
  489. ca->in_packets = out.mFramesPerPacket / in.mFramesPerPacket;
  490. ca->in_bytes_required = ca->in_packets * ca->in_frame_size;
  491. ca->out_frames_per_packet = out.mFramesPerPacket;
  492. da_init(ca->input_buffer);
  493. ca->output_buffer_size = out.mBytesPerPacket;
  494. if (out.mBytesPerPacket == 0) {
  495. UInt32 max_packet_size = 0;
  496. size = sizeof(max_packet_size);
  497. code = AudioConverterGetProperty(ca->converter,
  498. kAudioConverterPropertyMaximumOutputPacketSize,
  499. &size, &max_packet_size);
  500. if (code) {
  501. log_osstatus(LOG_WARNING, ca.get(),
  502. "AudioConverterGetProperty(PacketSz)",
  503. code);
  504. ca->output_buffer_size = 32768;
  505. } else {
  506. ca->output_buffer_size = max_packet_size;
  507. }
  508. }
  509. try {
  510. ca->output_buffer.resize(ca->output_buffer_size);
  511. } catch (...) {
  512. CA_BLOG(LOG_ERROR, "Failed to allocate output buffer");
  513. return nullptr;
  514. }
  515. const char *format_name =
  516. out.mFormatID == kAudioFormatMPEG4AAC_HE_V2 ? "HE-AAC v2" :
  517. out.mFormatID == kAudioFormatMPEG4AAC_HE ? "HE-AAC" : "AAC";
  518. CA_BLOG(LOG_INFO, "settings:\n"
  519. "\tmode: %s\n"
  520. "\tbitrate: %u\n"
  521. "\tsample rate: %llu\n"
  522. "\tcbr: %s\n"
  523. "\toutput buffer: %lu",
  524. format_name, (unsigned int)bitrate / 1000,
  525. ca->samples_per_second,
  526. rate_control == kAudioCodecBitRateControlMode_Constant ?
  527. "on" : "off",
  528. (unsigned long)ca->output_buffer_size);
  529. return ca.release();
  530. #undef STATUS_CHECK
  531. }
  532. static OSStatus complex_input_data_proc(AudioConverterRef inAudioConverter,
  533. UInt32 *ioNumberDataPackets, AudioBufferList *ioData,
  534. AudioStreamPacketDescription **outDataPacketDescription,
  535. void *inUserData)
  536. {
  537. UNUSED_PARAMETER(inAudioConverter);
  538. UNUSED_PARAMETER(outDataPacketDescription);
  539. ca_encoder *ca = static_cast<ca_encoder*>(inUserData);
  540. if (ca->bytes_read) {
  541. da_erase_range(ca->input_buffer, 0, ca->bytes_read);
  542. ca->bytes_read = 0;
  543. }
  544. if (ca->input_buffer.num < ca->in_bytes_required) {
  545. *ioNumberDataPackets = 0;
  546. ioData->mBuffers[0].mData = NULL;
  547. return 1;
  548. }
  549. *ioNumberDataPackets =
  550. (UInt32)(ca->in_bytes_required / ca->in_frame_size);
  551. ioData->mNumberBuffers = 1;
  552. ioData->mBuffers[0].mData = ca->input_buffer.array;
  553. ioData->mBuffers[0].mNumberChannels = (UInt32)ca->channels;
  554. ioData->mBuffers[0].mDataByteSize = (UInt32)ca->in_bytes_required;
  555. ca->bytes_read += ca->in_packets * ca->in_frame_size;
  556. return 0;
  557. }
  558. #ifdef _MSC_VER
  559. // disable warning that recommends if ((foo = bar > 0) == false) over
  560. // if (!(foo = bar > 0))
  561. #pragma warning(push)
  562. #pragma warning(disable: 4706)
  563. #endif
  564. static bool aac_encode(void *data, struct encoder_frame *frame,
  565. struct encoder_packet *packet, bool *received_packet)
  566. {
  567. ca_encoder *ca = static_cast<ca_encoder*>(data);
  568. da_push_back_array(ca->input_buffer, frame->data[0],
  569. frame->linesize[0]);
  570. if ((ca->input_buffer.num - ca->bytes_read) < ca->in_bytes_required)
  571. return true;
  572. UInt32 packets = 1;
  573. AudioBufferList buffer_list = { 0 };
  574. buffer_list.mNumberBuffers = 1;
  575. buffer_list.mBuffers[0].mNumberChannels = (UInt32)ca->channels;
  576. buffer_list.mBuffers[0].mDataByteSize = (UInt32)ca->output_buffer_size;
  577. buffer_list.mBuffers[0].mData = ca->output_buffer.data();
  578. AudioStreamPacketDescription out_desc = { 0 };
  579. OSStatus code = AudioConverterFillComplexBuffer(ca->converter,
  580. complex_input_data_proc, ca, &packets,
  581. &buffer_list, &out_desc);
  582. if (code && code != 1) {
  583. log_osstatus(LOG_ERROR, ca, "AudioConverterFillComplexBuffer",
  584. code);
  585. return false;
  586. }
  587. if (!(*received_packet = packets > 0))
  588. return true;
  589. packet->pts = ca->total_samples;
  590. packet->dts = ca->total_samples;
  591. packet->timebase_num = 1;
  592. packet->timebase_den = (uint32_t)ca->samples_per_second;
  593. packet->type = OBS_ENCODER_AUDIO;
  594. packet->size = out_desc.mDataByteSize;
  595. packet->data =
  596. (uint8_t*)buffer_list.mBuffers[0].mData + out_desc.mStartOffset;
  597. ca->total_samples += ca->bytes_read / ca->in_frame_size;
  598. return true;
  599. }
  600. #ifdef _MSC_VER
  601. #pragma warning(pop)
  602. #endif
  603. static void aac_audio_info(void *data, struct audio_convert_info *info)
  604. {
  605. UNUSED_PARAMETER(data);
  606. info->format = AUDIO_FORMAT_FLOAT;
  607. }
  608. static size_t aac_frame_size(void *data)
  609. {
  610. ca_encoder *ca = static_cast<ca_encoder*>(data);
  611. return ca->out_frames_per_packet;
  612. }
  613. /* The following code was extracted from encca_aac.c in HandBrake's libhb */
  614. #define MP4ESDescrTag 0x03
  615. #define MP4DecConfigDescrTag 0x04
  616. #define MP4DecSpecificDescrTag 0x05
  617. // based off of mov_mp4_read_descr_len from mov.c in ffmpeg's libavformat
  618. static int read_descr_len(uint8_t **buffer)
  619. {
  620. int len = 0;
  621. int count = 4;
  622. while (count--)
  623. {
  624. int c = *(*buffer)++;
  625. len = (len << 7) | (c & 0x7f);
  626. if (!(c & 0x80))
  627. break;
  628. }
  629. return len;
  630. }
  631. // based off of mov_mp4_read_descr from mov.c in ffmpeg's libavformat
  632. static int read_descr(uint8_t **buffer, int *tag)
  633. {
  634. *tag = *(*buffer)++;
  635. return read_descr_len(buffer);
  636. }
  637. // based off of mov_read_esds from mov.c in ffmpeg's libavformat
  638. static void read_esds_desc_ext(uint8_t* desc_ext, vector<uint8_t> &buffer,
  639. bool version_flags)
  640. {
  641. uint8_t *esds = desc_ext;
  642. int tag, len;
  643. if (version_flags)
  644. esds += 4; // version + flags
  645. read_descr(&esds, &tag);
  646. esds += 2; // ID
  647. if (tag == MP4ESDescrTag)
  648. esds++; // priority
  649. read_descr(&esds, &tag);
  650. if (tag == MP4DecConfigDescrTag) {
  651. esds++; // object type id
  652. esds++; // stream type
  653. esds += 3; // buffer size db
  654. esds += 4; // max bitrate
  655. esds += 4; // average bitrate
  656. len = read_descr(&esds, &tag);
  657. if (tag == MP4DecSpecificDescrTag)
  658. try {
  659. buffer.assign(esds, esds + len);
  660. } catch (...) {
  661. //leave buffer empty
  662. }
  663. }
  664. }
  665. /* extracted code ends here */
  666. static void query_extra_data(ca_encoder *ca)
  667. {
  668. UInt32 size = 0;
  669. OSStatus code;
  670. code = AudioConverterGetPropertyInfo(ca->converter,
  671. kAudioConverterCompressionMagicCookie,
  672. &size, NULL);
  673. if (code) {
  674. log_osstatus(LOG_ERROR, ca,
  675. "AudioConverterGetPropertyInfo(magic_cookie)",
  676. code);
  677. return;
  678. }
  679. if (!size) {
  680. CA_BLOG(LOG_WARNING, "Got 0 data size info for magic_cookie");
  681. return;
  682. }
  683. vector<uint8_t> extra_data;
  684. try {
  685. extra_data.resize(size);
  686. } catch (...) {
  687. CA_BLOG(LOG_WARNING, "Could not allocate extra data buffer");
  688. return;
  689. }
  690. code = AudioConverterGetProperty(ca->converter,
  691. kAudioConverterCompressionMagicCookie,
  692. &size, extra_data.data());
  693. if (code) {
  694. log_osstatus(LOG_ERROR, ca,
  695. "AudioConverterGetProperty(magic_cookie)",
  696. code);
  697. return;
  698. }
  699. if (!size) {
  700. CA_BLOG(LOG_WARNING, "Got 0 data size for magic_cookie");
  701. return;
  702. }
  703. read_esds_desc_ext(extra_data.data(), ca->extra_data, false);
  704. }
  705. static bool aac_extra_data(void *data, uint8_t **extra_data, size_t *size)
  706. {
  707. ca_encoder *ca = static_cast<ca_encoder*>(data);
  708. if (!ca->extra_data.size())
  709. query_extra_data(ca);
  710. if (!ca->extra_data.size())
  711. return false;
  712. *extra_data = ca->extra_data.data();
  713. *size = ca->extra_data.size();
  714. return true;
  715. }
  716. static asbd_builder fill_common_asbd_fields(asbd_builder builder,
  717. bool in=false)
  718. {
  719. UInt32 bytes_per_frame = 8;
  720. UInt32 channels = 2;
  721. UInt32 bits_per_channel = bytes_per_frame / channels * 8;
  722. builder.channels_per_frame(channels);
  723. if (in) {
  724. builder
  725. .bytes_per_frame(bytes_per_frame)
  726. .frames_per_packet(1)
  727. .bytes_per_packet(1 * bytes_per_frame)
  728. .bits_per_channel(bits_per_channel);
  729. }
  730. return builder;
  731. }
  732. static AudioStreamBasicDescription get_default_in_asbd()
  733. {
  734. return fill_common_asbd_fields(asbd_builder(), true)
  735. .sample_rate(44100)
  736. .format_id(kAudioFormatLinearPCM)
  737. .format_flags(kAudioFormatFlagsNativeEndian |
  738. kAudioFormatFlagIsPacked |
  739. kAudioFormatFlagIsFloat |
  740. 0)
  741. .asbd;
  742. }
  743. static asbd_builder get_default_out_asbd_builder()
  744. {
  745. return fill_common_asbd_fields(asbd_builder())
  746. .sample_rate(44100);
  747. }
  748. static cf_ptr<AudioConverterRef> get_converter(DStr &log, ca_encoder *ca,
  749. AudioStreamBasicDescription out,
  750. AudioStreamBasicDescription in = get_default_in_asbd())
  751. {
  752. UInt32 size = sizeof(out);
  753. OSStatus code;
  754. #define STATUS_CHECK(x) \
  755. code = x; \
  756. if (code) { \
  757. log_to_dstr(log, ca, "%s: %s\n", #x, \
  758. osstatus_to_dstr(code)->array); \
  759. return nullptr; \
  760. }
  761. STATUS_CHECK(AudioFormatGetProperty(kAudioFormatProperty_FormatInfo,
  762. 0, NULL, &size, &out));
  763. AudioConverterRef converter;
  764. STATUS_CHECK(AudioConverterNew(&in, &out, &converter));
  765. return cf_ptr<AudioConverterRef>{converter};
  766. #undef STATUS_CHECK
  767. }
  768. static bool find_best_match(DStr &log, ca_encoder *ca, UInt32 bitrate,
  769. UInt32 &best_match)
  770. {
  771. UInt32 actual_bitrate = bitrate * 1000;
  772. bool found_match = false;
  773. auto handle_bitrate = [&](UInt32 candidate)
  774. {
  775. if (abs(static_cast<intmax_t>(actual_bitrate - candidate)) <
  776. abs(static_cast<intmax_t>(actual_bitrate - best_match))) {
  777. log_to_dstr(log, ca, "Found new best match %u\n",
  778. static_cast<uint32_t>(candidate));
  779. found_match = true;
  780. best_match = candidate;
  781. }
  782. };
  783. auto helper = [&](UInt32 min_, UInt32 max_)
  784. {
  785. handle_bitrate(min_);
  786. if (min_ == max_)
  787. return;
  788. log_to_dstr(log, ca, "Got actual bit rate range: %u<->%u\n",
  789. static_cast<uint32_t>(min_),
  790. static_cast<uint32_t>(max_));
  791. handle_bitrate(max_);
  792. };
  793. for (UInt32 format_id : aac_formats) {
  794. log_to_dstr(log, ca, "Trying %s (0x%x)\n",
  795. format_id_to_str(format_id), format_id);
  796. auto out = get_default_out_asbd_builder()
  797. .format_id(format_id)
  798. .asbd;
  799. auto converter = get_converter(log, ca, out);
  800. if (converter)
  801. enumerate_bitrates(log, ca, converter.get(),
  802. helper);
  803. else
  804. log_to_dstr(log, ca, "Could not get converter\n");
  805. }
  806. best_match /= 1000;
  807. return found_match;
  808. }
  809. static UInt32 find_matching_bitrate(UInt32 bitrate)
  810. {
  811. static UInt32 match = bitrate;
  812. static once_flag once;
  813. call_once(once, [&]()
  814. {
  815. DStr log;
  816. ca_encoder *ca = nullptr;
  817. if (!find_best_match(log, ca, bitrate, match)) {
  818. CA_CO_DLOG(LOG_ERROR, "No matching bitrates found for "
  819. "target bitrate %u",
  820. static_cast<uint32_t>(bitrate));
  821. match = bitrate;
  822. return;
  823. }
  824. if (match != bitrate) {
  825. CA_CO_DLOG(LOG_INFO, "Default bitrate (%u) isn't "
  826. "supported, returning %u as closest match",
  827. static_cast<uint32_t>(bitrate),
  828. static_cast<uint32_t>(match));
  829. return;
  830. }
  831. if (log->len)
  832. CA_CO_DLOG(LOG_DEBUG, "Default bitrate matching log "
  833. "for bitrate %u",
  834. static_cast<uint32_t>(bitrate));
  835. });
  836. return match;
  837. }
  838. static void aac_defaults(obs_data_t *settings)
  839. {
  840. obs_data_set_default_int(settings, "bitrate",
  841. find_matching_bitrate(128));
  842. obs_data_set_default_bool(settings, "allow he-aac", true);
  843. }
  844. static vector<UInt32> get_bitrates(DStr &log, ca_encoder *ca,
  845. Float64 samplerate)
  846. {
  847. vector<UInt32> bitrates;
  848. auto handle_bitrate = [&](UInt32 bitrate)
  849. {
  850. if (find(begin(bitrates), end(bitrates), bitrate) ==
  851. end(bitrates)) {
  852. log_to_dstr(log, ca, "Adding bitrate %u\n",
  853. static_cast<uint32_t>(bitrate));
  854. bitrates.push_back(bitrate);
  855. } else {
  856. log_to_dstr(log, ca, "Bitrate %u already added\n",
  857. static_cast<uint32_t>(bitrate));
  858. }
  859. };
  860. auto helper = [&](UInt32 min_, UInt32 max_)
  861. {
  862. handle_bitrate(min_);
  863. if (min_ == max_)
  864. return;
  865. log_to_dstr(log, ca, "Got actual bitrate range: %u<->%u\n",
  866. static_cast<uint32_t>(min_),
  867. static_cast<uint32_t>(max_));
  868. handle_bitrate(max_);
  869. };
  870. for (UInt32 format_id : (ca ? *ca->allowed_formats : aac_formats)) {
  871. log_to_dstr(log, ca, "Trying %s (0x%x) at %g hz\n",
  872. format_id_to_str(format_id),
  873. static_cast<uint32_t>(format_id),
  874. samplerate);
  875. auto out = get_default_out_asbd_builder()
  876. .format_id(format_id)
  877. .sample_rate(samplerate)
  878. .asbd;
  879. auto converter = get_converter(log, ca, out);
  880. if (converter)
  881. enumerate_bitrates(log, ca, converter.get(), helper);
  882. }
  883. return bitrates;
  884. }
  885. static void add_bitrates(obs_property_t *prop, ca_encoder *ca,
  886. Float64 samplerate=44100.)
  887. {
  888. DStr log;
  889. auto bitrates = get_bitrates(log, ca, samplerate);
  890. if (!bitrates.size()) {
  891. CA_CO_DLOG_(LOG_ERROR, "Couldn't find available bitrates");
  892. return;
  893. }
  894. if (log->len)
  895. CA_CO_DLOG_(LOG_DEBUG, "Bitrate enumeration log");
  896. sort(begin(bitrates), end(bitrates));
  897. DStr buffer;
  898. for (UInt32 bitrate : bitrates) {
  899. dstr_printf(buffer, "%u", (uint32_t)bitrate / 1000);
  900. obs_property_list_add_int(prop, buffer->array, bitrate / 1000);
  901. }
  902. }
  903. static obs_properties_t *aac_properties(void *data)
  904. {
  905. ca_encoder *ca = static_cast<ca_encoder*>(data);
  906. obs_properties_t *props = obs_properties_create();
  907. obs_property_t *p = obs_properties_add_list(props, "bitrate",
  908. obs_module_text("Bitrate"),
  909. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  910. add_bitrates(p, ca);
  911. obs_properties_add_bool(props, "allow he-aac",
  912. obs_module_text("AllowHEAAC"));
  913. return props;
  914. }
  915. OBS_DECLARE_MODULE()
  916. OBS_MODULE_USE_DEFAULT_LOCALE("coreaudio-encoder", "en-US")
  917. bool obs_module_load(void)
  918. {
  919. #ifdef _WIN32
  920. if (!load_core_audio()) {
  921. CA_LOG(LOG_WARNING, "Couldn't load CoreAudio AAC encoder");
  922. return true;
  923. }
  924. CA_LOG(LOG_INFO, "Adding CoreAudio AAC encoder");
  925. #endif
  926. struct obs_encoder_info aac_info;
  927. aac_info.id = "CoreAudio_AAC";
  928. aac_info.type = OBS_ENCODER_AUDIO;
  929. aac_info.codec = "AAC";
  930. aac_info.get_name = aac_get_name;
  931. aac_info.destroy = aac_destroy;
  932. aac_info.create = aac_create;
  933. aac_info.encode = aac_encode;
  934. aac_info.get_frame_size = aac_frame_size;
  935. aac_info.get_audio_info = aac_audio_info;
  936. aac_info.get_extra_data = aac_extra_data;
  937. aac_info.get_defaults = aac_defaults;
  938. aac_info.get_properties = aac_properties;
  939. obs_register_encoder(&aac_info);
  940. return true;
  941. }
  942. #ifdef _WIN32
  943. void obs_module_unload(void)
  944. {
  945. unload_core_audio();
  946. }
  947. #endif