12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256 |
- #include <util/darray.h>
- #include <util/dstr.hpp>
- #include <obs-module.h>
- #include <algorithm>
- #include <cstdlib>
- #include <initializer_list>
- #include <memory>
- #include <mutex>
- #include <vector>
- #ifndef _WIN32
- #include <AudioToolbox/AudioToolbox.h>
- #endif
- #define CA_LOG(level, format, ...) \
- blog(level, "[CoreAudio encoder]: " format, ##__VA_ARGS__)
- #define CA_LOG_ENCODER(format_name, encoder, level, format, ...) \
- blog(level, "[CoreAudio %s: '%s']: " format, \
- format_name, obs_encoder_get_name(encoder), \
- ##__VA_ARGS__)
- #define CA_BLOG(level, format, ...) \
- CA_LOG_ENCODER(ca->format_name, ca->encoder, level, format, \
- ##__VA_ARGS__)
- #define CA_CO_LOG(level, format, ...) \
- do { \
- if (ca) \
- CA_BLOG(level, format, ##__VA_ARGS__); \
- else \
- CA_LOG(level, format, ##__VA_ARGS__); \
- } while (false)
- #ifdef _WIN32
- #include "windows-imports.h"
- #endif
- using namespace std;
- namespace {
- struct asbd_builder {
- AudioStreamBasicDescription asbd;
- asbd_builder &sample_rate(Float64 rate)
- {
- asbd.mSampleRate = rate;
- return *this;
- }
- asbd_builder &format_id(UInt32 format)
- {
- asbd.mFormatID = format;
- return *this;
- }
- asbd_builder &format_flags(UInt32 flags)
- {
- asbd.mFormatFlags = flags;
- return *this;
- }
- asbd_builder &bytes_per_packet(UInt32 bytes)
- {
- asbd.mBytesPerPacket = bytes;
- return *this;
- }
- asbd_builder &frames_per_packet(UInt32 frames)
- {
- asbd.mFramesPerPacket = frames;
- return *this;
- }
- asbd_builder &bytes_per_frame(UInt32 bytes)
- {
- asbd.mBytesPerFrame = bytes;
- return *this;
- }
- asbd_builder &channels_per_frame(UInt32 channels)
- {
- asbd.mChannelsPerFrame = channels;
- return *this;
- }
- asbd_builder &bits_per_channel(UInt32 bits)
- {
- asbd.mBitsPerChannel = bits;
- return *this;
- }
- };
- struct ca_encoder {
- obs_encoder_t *encoder;
- const char *format_name;
- UInt32 format_id;
- const initializer_list<UInt32> *allowed_formats;
- AudioConverterRef converter;
- size_t output_buffer_size;
- vector<uint8_t> output_buffer;
- size_t out_frames_per_packet;
- size_t in_packets;
- size_t in_frame_size;
- size_t in_bytes_required;
- DARRAY(uint8_t) input_buffer;
- size_t bytes_read;
- uint64_t total_samples;
- uint64_t samples_per_second;
- vector<uint8_t> extra_data;
- size_t channels;
- ~ca_encoder()
- {
- if (converter)
- AudioConverterDispose(converter);
- da_free(input_buffer);
- }
- };
- typedef struct ca_encoder ca_encoder;
- }
- namespace std {
- #ifndef _WIN32
- template <>
- struct default_delete<remove_pointer<CFErrorRef>::type> {
- void operator()(remove_pointer<CFErrorRef>::type *err)
- {
- CFRelease(err);
- }
- };
- template <>
- struct default_delete<remove_pointer<CFStringRef>::type> {
- void operator()(remove_pointer<CFStringRef>::type *str)
- {
- CFRelease(str);
- }
- };
- #endif
- template <>
- struct default_delete<remove_pointer<AudioConverterRef>::type> {
- void operator()(AudioConverterRef converter)
- {
- AudioConverterDispose(converter);
- }
- };
- }
- template <typename T>
- using cf_ptr = unique_ptr<typename remove_pointer<T>::type>;
- #ifndef _MSC_VER
- __attribute__((__format__(__printf__, 3, 4)))
- #endif
- static void log_to_dstr(DStr &str, ca_encoder *ca, const char *fmt, ...)
- {
- dstr prev_str = *static_cast<dstr*>(str);
- va_list args;
- va_start(args, fmt);
- dstr_vcatf(str, fmt, args);
- va_end(args);
- if (str->array)
- return;
- char array[4096];
- va_start(args, fmt);
- vsnprintf(array, 4096, fmt, args);
- va_end(args);
- array[4095] = 0;
- if (!prev_str.array && !prev_str.len)
- CA_CO_LOG(LOG_ERROR, "Could not allocate buffer for logging:"
- "\n'%s'", array);
- else
- CA_CO_LOG(LOG_ERROR, "Could not allocate buffer for logging:"
- "\n'%s'\nPrevious log entries:\n%s",
- array, prev_str.array);
- bfree(prev_str.array);
- }
- static const char *flush_log(DStr &log)
- {
- if (!log->array || !log->len)
- return "";
- if (log->array[log->len - 1] == '\n') {
- log->array[log->len - 1] = 0; //Get rid of last newline
- log->len -= 1;
- }
- return log->array;
- }
- #define CA_CO_DLOG_(level, format) \
- CA_CO_LOG(level, format "%s%s", \
- log->array ? ":\n" : "", flush_log(log))
- #define CA_CO_DLOG(level, format, ...) \
- CA_CO_LOG(level, format "%s%s", ##__VA_ARGS__, \
- log->array ? ":\n" : "", flush_log(log))
- static const char *aac_get_name(void)
- {
- return obs_module_text("CoreAudioAAC");
- }
- static const char *code_to_str(OSStatus code)
- {
- switch (code) {
- #define HANDLE_CODE(c) case c: return #c
- HANDLE_CODE(kAudio_UnimplementedError);
- HANDLE_CODE(kAudio_FileNotFoundError);
- HANDLE_CODE(kAudio_FilePermissionError);
- HANDLE_CODE(kAudio_TooManyFilesOpenError);
- HANDLE_CODE(kAudio_BadFilePathError);
- HANDLE_CODE(kAudio_ParamError);
- HANDLE_CODE(kAudio_MemFullError);
- HANDLE_CODE(kAudioConverterErr_FormatNotSupported);
- HANDLE_CODE(kAudioConverterErr_OperationNotSupported);
- HANDLE_CODE(kAudioConverterErr_PropertyNotSupported);
- HANDLE_CODE(kAudioConverterErr_InvalidInputSize);
- HANDLE_CODE(kAudioConverterErr_InvalidOutputSize);
- HANDLE_CODE(kAudioConverterErr_UnspecifiedError);
- HANDLE_CODE(kAudioConverterErr_BadPropertySizeError);
- HANDLE_CODE(kAudioConverterErr_RequiresPacketDescriptionsError);
- HANDLE_CODE(kAudioConverterErr_InputSampleRateOutOfRange);
- HANDLE_CODE(kAudioConverterErr_OutputSampleRateOutOfRange);
- #undef HANDLE_CODE
- default: break;
- }
- return NULL;
- }
- static DStr osstatus_to_dstr(OSStatus code)
- {
- DStr result;
- #ifndef _WIN32
- cf_ptr<CFErrorRef> err{CFErrorCreate(kCFAllocatorDefault,
- kCFErrorDomainOSStatus, code, NULL)};
- cf_ptr<CFStringRef> str{CFErrorCopyDescription(err.get())};
- CFIndex length = CFStringGetLength(str.get());
- CFIndex max_size = CFStringGetMaximumSizeForEncoding(length,
- kCFStringEncodingUTF8);
- dstr_ensure_capacity(result, max_size);
- if (result->array && CFStringGetCString(str.get(), result->array,
- max_size, kCFStringEncodingUTF8)) {
- dstr_resize(result, strlen(result->array));
- return result;
- }
- #endif
- const char *code_str = code_to_str(code);
- dstr_printf(result, "%s%s%d%s",
- code_str ? code_str : "",
- code_str ? " (" : "",
- static_cast<int>(code),
- code_str ? ")" : "");
- return result;
- }
- static void log_osstatus(int log_level, ca_encoder *ca, const char *context,
- OSStatus code)
- {
- DStr str = osstatus_to_dstr(code);
- if (ca)
- CA_BLOG(log_level, "Error in %s: %s", context, str->array);
- else
- CA_LOG(log_level, "Error in %s: %s", context, str->array);
- }
- static const char *format_id_to_str(UInt32 format_id)
- {
- #define FORMAT_TO_STR(x) case x: return #x
- switch (format_id) {
- FORMAT_TO_STR(kAudioFormatLinearPCM);
- FORMAT_TO_STR(kAudioFormatAC3);
- FORMAT_TO_STR(kAudioFormat60958AC3);
- FORMAT_TO_STR(kAudioFormatAppleIMA4);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC);
- FORMAT_TO_STR(kAudioFormatMPEG4CELP);
- FORMAT_TO_STR(kAudioFormatMPEG4HVXC);
- FORMAT_TO_STR(kAudioFormatMPEG4TwinVQ);
- FORMAT_TO_STR(kAudioFormatMACE3);
- FORMAT_TO_STR(kAudioFormatMACE6);
- FORMAT_TO_STR(kAudioFormatULaw);
- FORMAT_TO_STR(kAudioFormatALaw);
- FORMAT_TO_STR(kAudioFormatQDesign);
- FORMAT_TO_STR(kAudioFormatQDesign2);
- FORMAT_TO_STR(kAudioFormatQUALCOMM);
- FORMAT_TO_STR(kAudioFormatMPEGLayer1);
- FORMAT_TO_STR(kAudioFormatMPEGLayer2);
- FORMAT_TO_STR(kAudioFormatMPEGLayer3);
- FORMAT_TO_STR(kAudioFormatTimeCode);
- FORMAT_TO_STR(kAudioFormatMIDIStream);
- FORMAT_TO_STR(kAudioFormatParameterValueStream);
- FORMAT_TO_STR(kAudioFormatAppleLossless);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC_HE);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC_LD);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC_ELD);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC_ELD_SBR);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC_HE_V2);
- FORMAT_TO_STR(kAudioFormatMPEG4AAC_Spatial);
- FORMAT_TO_STR(kAudioFormatAMR);
- FORMAT_TO_STR(kAudioFormatAudible);
- FORMAT_TO_STR(kAudioFormatiLBC);
- FORMAT_TO_STR(kAudioFormatDVIIntelIMA);
- FORMAT_TO_STR(kAudioFormatMicrosoftGSM);
- FORMAT_TO_STR(kAudioFormatAES3);
- }
- #undef FORMAT_TO_STR
- return "Unknown format";
- }
- static void aac_destroy(void *data)
- {
- ca_encoder *ca = static_cast<ca_encoder*>(data);
- delete ca;
- }
- template <typename Func>
- static bool query_converter_property_raw(DStr &log, ca_encoder *ca,
- AudioFormatPropertyID property,
- const char *get_property_info, const char *get_property,
- AudioConverterRef converter, Func &&func)
- {
- UInt32 size = 0;
- OSStatus code = AudioConverterGetPropertyInfo(converter, property,
- &size, nullptr);
- if (code) {
- log_to_dstr(log, ca, "%s: %s\n", get_property_info,
- osstatus_to_dstr(code)->array);
- return false;
- }
- if (!size) {
- log_to_dstr(log, ca, "%s returned 0 size\n", get_property_info);
- return false;
- }
- vector<uint8_t> buffer;
-
- try {
- buffer.resize(size);
- } catch (...) {
- log_to_dstr(log, ca, "Failed to allocate %u bytes for %s\n",
- static_cast<uint32_t>(size), get_property);
- return false;
- }
- code = AudioConverterGetProperty(converter, property, &size,
- buffer.data());
- if (code) {
- log_to_dstr(log, ca, "%s: %s\n", get_property,
- osstatus_to_dstr(code)->array);
- return false;
- }
- func(size, static_cast<void*>(buffer.data()));
- return true;
- }
- #define EXPAND_CONVERTER_NAMES(x) x, \
- "AudioConverterGetPropertyInfo(" #x ")", \
- "AudioConverterGetProperty(" #x ")"
- template <typename Func>
- static bool enumerate_bitrates(DStr &log, ca_encoder *ca,
- AudioConverterRef converter, Func &&func)
- {
- auto helper = [&](UInt32 size, void *data)
- {
- auto range = static_cast<AudioValueRange*>(data);
- size_t num_ranges = size / sizeof(AudioValueRange);
- for (size_t i = 0; i < num_ranges; i++)
- func(static_cast<UInt32>(range[i].mMinimum),
- static_cast<UInt32>(range[i].mMaximum));
- };
- return query_converter_property_raw(log, ca, EXPAND_CONVERTER_NAMES(
- kAudioConverterApplicableEncodeBitRates),
- converter, helper);
- }
- typedef void (*bitrate_enumeration_func)(void *data, UInt32 min, UInt32 max);
- static bool enumerate_bitrates(ca_encoder *ca, AudioConverterRef converter,
- bitrate_enumeration_func enum_func, void *data)
- {
- if (!converter && ca)
- converter = ca->converter;
- UInt32 size;
- OSStatus code = AudioConverterGetPropertyInfo(converter,
- kAudioConverterApplicableEncodeBitRates,
- &size, NULL);
- if (code) {
- log_osstatus(LOG_WARNING, ca,
- "AudioConverterGetPropertyInfo(bitrates)",
- code);
- return false;
- }
- if (!size) {
- if (ca)
- CA_BLOG(LOG_WARNING, "Query for applicable bitrates "
- "returned 0 size");
- else
- CA_LOG(LOG_WARNING, "Query for applicable bitrates "
- "returned 0 size");
- return false;
- }
- size_t num_bitrates = (size + sizeof(AudioValueRange) - 1) /
- sizeof(AudioValueRange);
- vector<AudioValueRange> bitrates;
-
- try {
- bitrates.resize(num_bitrates);
- } catch (...) {
- if (ca)
- CA_BLOG(LOG_WARNING, "Could not allocate buffer while "
- "enumerating bitrates");
- else
- CA_LOG(LOG_WARNING, "Could not allocate buffer while "
- "enumerating bitrates");
- return false;
- }
- code = AudioConverterGetProperty(converter,
- kAudioConverterApplicableEncodeBitRates,
- &size, bitrates.data());
- if (code) {
- log_osstatus(LOG_WARNING, ca,
- "AudioConverterGetProperty(bitrates)", code);
- return false;
- }
- for (size_t i = 0; i < num_bitrates; i++)
- enum_func(data, (UInt32)bitrates[i].mMinimum,
- (UInt32)bitrates[i].mMaximum);
- return num_bitrates > 0;
- }
- static bool bitrate_valid(DStr &log, ca_encoder *ca,
- AudioConverterRef converter, UInt32 bitrate)
- {
- bool valid = false;
- auto helper = [&](UInt32 min_, UInt32 max_)
- {
- if (min_ == bitrate || max_ == bitrate)
- valid = true;
- };
- enumerate_bitrates(log, ca, converter, helper);
- return valid;
- }
- static bool create_encoder(DStr &log, ca_encoder *ca,
- AudioStreamBasicDescription *in,
- AudioStreamBasicDescription *out,
- UInt32 format_id, UInt32 bitrate, UInt32 rate_control)
- {
- #define STATUS_CHECK(c) \
- code = c; \
- if (code) { \
- log_to_dstr(log, ca, #c " returned %s", \
- osstatus_to_dstr(code)->array); \
- return false; \
- }
- auto out_ = asbd_builder()
- .sample_rate((Float64)ca->samples_per_second)
- .channels_per_frame((UInt32)ca->channels)
- .format_id(format_id)
- .asbd;
- UInt32 size = sizeof(*out);
- OSStatus code;
- STATUS_CHECK(AudioFormatGetProperty(kAudioFormatProperty_FormatInfo,
- 0, NULL, &size, &out_));
- *out = out_;
- STATUS_CHECK(AudioConverterNew(in, out, &ca->converter))
- STATUS_CHECK(AudioConverterSetProperty(ca->converter,
- kAudioCodecPropertyBitRateControlMode,
- sizeof(rate_control), &rate_control));
- if (!bitrate_valid(log, ca, ca->converter, bitrate)) {
- log_to_dstr(log, ca, "Encoder does not support bitrate %u "
- "for format %s (0x%x)\n",
- (uint32_t)bitrate, format_id_to_str(format_id),
- (uint32_t)format_id);
- return false;
- }
- ca->format_id = format_id;
- return true;
- #undef STATUS_CHECK
- }
- static const initializer_list<UInt32> aac_formats = {
- kAudioFormatMPEG4AAC_HE_V2,
- kAudioFormatMPEG4AAC_HE,
- kAudioFormatMPEG4AAC,
- };
- static const initializer_list<UInt32> aac_lc_formats = {
- kAudioFormatMPEG4AAC,
- };
- static void *aac_create(obs_data_t *settings, obs_encoder_t *encoder)
- {
- #define STATUS_CHECK(c) \
- code = c; \
- if (code) { \
- log_osstatus(LOG_ERROR, ca.get(), #c, code); \
- return nullptr; \
- }
- UInt32 bitrate = (UInt32)obs_data_get_int(settings, "bitrate") * 1000;
- if (!bitrate) {
- CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
- "Invalid bitrate specified");
- return NULL;
- }
- const enum audio_format format = AUDIO_FORMAT_FLOAT;
- if (is_audio_planar(format)) {
- CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
- "Got non-interleaved audio format %d", format);
- return NULL;
- }
- unique_ptr<ca_encoder> ca;
- try {
- ca.reset(new ca_encoder());
- } catch (...) {
- CA_LOG_ENCODER("AAC", encoder, LOG_ERROR,
- "Could not allocate encoder");
- return nullptr;
- }
- ca->encoder = encoder;
- ca->format_name = "AAC";
- audio_t *audio = obs_encoder_audio(encoder);
- const struct audio_output_info *aoi = audio_output_get_info(audio);
- ca->channels = audio_output_get_channels(audio);
- ca->samples_per_second = audio_output_get_sample_rate(audio);
- size_t bytes_per_frame = get_audio_size(format, aoi->speakers, 1);
- size_t bits_per_channel = get_audio_bytes_per_channel(format) * 8;
- auto in = asbd_builder()
- .sample_rate((Float64)ca->samples_per_second)
- .channels_per_frame((UInt32)ca->channels)
- .bytes_per_frame((UInt32)bytes_per_frame)
- .frames_per_packet(1)
- .bytes_per_packet((UInt32)(1 * bytes_per_frame))
- .bits_per_channel((UInt32)bits_per_channel)
- .format_id(kAudioFormatLinearPCM)
- .format_flags(kAudioFormatFlagsNativeEndian |
- kAudioFormatFlagIsPacked |
- kAudioFormatFlagIsFloat |
- 0)
- .asbd;
- AudioStreamBasicDescription out;
- UInt32 rate_control = kAudioCodecBitRateControlMode_Constant;
- if (obs_data_get_bool(settings, "allow he-aac")) {
- ca->allowed_formats = &aac_formats;
- } else {
- ca->allowed_formats = &aac_lc_formats;
- }
- DStr log;
- bool encoder_created = false;
- for (UInt32 format_id : *ca->allowed_formats) {
- log_to_dstr(log, ca.get(), "Trying format %s (0x%x)\n",
- format_id_to_str(format_id),
- (uint32_t)format_id);
- if (!create_encoder(log, ca.get(), &in, &out, format_id,
- bitrate, rate_control))
- continue;
- encoder_created = true;
- break;
- }
- if (!encoder_created) {
- CA_CO_DLOG(LOG_ERROR, "Could not create encoder for "
- "selected format%s",
- ca->allowed_formats->size() == 1 ? "" : "s");
- return nullptr;
- }
- if (log->len)
- CA_CO_DLOG_(LOG_DEBUG, "Encoder created");
- OSStatus code;
- UInt32 converter_quality = kAudioConverterQuality_Max;
- STATUS_CHECK(AudioConverterSetProperty(ca->converter,
- kAudioConverterCodecQuality,
- sizeof(converter_quality), &converter_quality));
- STATUS_CHECK(AudioConverterSetProperty(ca->converter,
- kAudioConverterEncodeBitRate,
- sizeof(bitrate), &bitrate));
- UInt32 size = sizeof(in);
- STATUS_CHECK(AudioConverterGetProperty(ca->converter,
- kAudioConverterCurrentInputStreamDescription,
- &size, &in));
- size = sizeof(out);
- STATUS_CHECK(AudioConverterGetProperty(ca->converter,
- kAudioConverterCurrentOutputStreamDescription,
- &size, &out));
- ca->in_frame_size = in.mBytesPerFrame;
- ca->in_packets = out.mFramesPerPacket / in.mFramesPerPacket;
- ca->in_bytes_required = ca->in_packets * ca->in_frame_size;
- ca->out_frames_per_packet = out.mFramesPerPacket;
- da_init(ca->input_buffer);
- ca->output_buffer_size = out.mBytesPerPacket;
- if (out.mBytesPerPacket == 0) {
- UInt32 max_packet_size = 0;
- size = sizeof(max_packet_size);
-
- code = AudioConverterGetProperty(ca->converter,
- kAudioConverterPropertyMaximumOutputPacketSize,
- &size, &max_packet_size);
- if (code) {
- log_osstatus(LOG_WARNING, ca.get(),
- "AudioConverterGetProperty(PacketSz)",
- code);
- ca->output_buffer_size = 32768;
- } else {
- ca->output_buffer_size = max_packet_size;
- }
- }
- try {
- ca->output_buffer.resize(ca->output_buffer_size);
- } catch (...) {
- CA_BLOG(LOG_ERROR, "Failed to allocate output buffer");
- return nullptr;
- }
- const char *format_name =
- out.mFormatID == kAudioFormatMPEG4AAC_HE_V2 ? "HE-AAC v2" :
- out.mFormatID == kAudioFormatMPEG4AAC_HE ? "HE-AAC" : "AAC";
- CA_BLOG(LOG_INFO, "settings:\n"
- "\tmode: %s\n"
- "\tbitrate: %u\n"
- "\tsample rate: %llu\n"
- "\tcbr: %s\n"
- "\toutput buffer: %lu",
- format_name, (unsigned int)bitrate / 1000,
- ca->samples_per_second,
- rate_control == kAudioCodecBitRateControlMode_Constant ?
- "on" : "off",
- (unsigned long)ca->output_buffer_size);
- return ca.release();
- }
- static OSStatus complex_input_data_proc(AudioConverterRef inAudioConverter,
- UInt32 *ioNumberDataPackets, AudioBufferList *ioData,
- AudioStreamPacketDescription **outDataPacketDescription,
- void *inUserData)
- {
- UNUSED_PARAMETER(inAudioConverter);
- UNUSED_PARAMETER(outDataPacketDescription);
- ca_encoder *ca = static_cast<ca_encoder*>(inUserData);
- if (ca->bytes_read) {
- da_erase_range(ca->input_buffer, 0, ca->bytes_read);
- ca->bytes_read = 0;
- }
- if (ca->input_buffer.num < ca->in_bytes_required) {
- *ioNumberDataPackets = 0;
- ioData->mBuffers[0].mData = NULL;
- return 1;
- }
- *ioNumberDataPackets =
- (UInt32)(ca->in_bytes_required / ca->in_frame_size);
- ioData->mNumberBuffers = 1;
- ioData->mBuffers[0].mData = ca->input_buffer.array;
- ioData->mBuffers[0].mNumberChannels = (UInt32)ca->channels;
- ioData->mBuffers[0].mDataByteSize = (UInt32)ca->in_bytes_required;
- ca->bytes_read += ca->in_packets * ca->in_frame_size;
- return 0;
- }
- #ifdef _MSC_VER
- // disable warning that recommends if ((foo = bar > 0) == false) over
- // if (!(foo = bar > 0))
- #pragma warning(push)
- #pragma warning(disable: 4706)
- #endif
- static bool aac_encode(void *data, struct encoder_frame *frame,
- struct encoder_packet *packet, bool *received_packet)
- {
- ca_encoder *ca = static_cast<ca_encoder*>(data);
- da_push_back_array(ca->input_buffer, frame->data[0],
- frame->linesize[0]);
- if ((ca->input_buffer.num - ca->bytes_read) < ca->in_bytes_required)
- return true;
- UInt32 packets = 1;
- AudioBufferList buffer_list = { 0 };
- buffer_list.mNumberBuffers = 1;
- buffer_list.mBuffers[0].mNumberChannels = (UInt32)ca->channels;
- buffer_list.mBuffers[0].mDataByteSize = (UInt32)ca->output_buffer_size;
- buffer_list.mBuffers[0].mData = ca->output_buffer.data();
- AudioStreamPacketDescription out_desc = { 0 };
- OSStatus code = AudioConverterFillComplexBuffer(ca->converter,
- complex_input_data_proc, ca, &packets,
- &buffer_list, &out_desc);
- if (code && code != 1) {
- log_osstatus(LOG_ERROR, ca, "AudioConverterFillComplexBuffer",
- code);
- return false;
- }
- if (!(*received_packet = packets > 0))
- return true;
- packet->pts = ca->total_samples;
- packet->dts = ca->total_samples;
- packet->timebase_num = 1;
- packet->timebase_den = (uint32_t)ca->samples_per_second;
- packet->type = OBS_ENCODER_AUDIO;
- packet->size = out_desc.mDataByteSize;
- packet->data =
- (uint8_t*)buffer_list.mBuffers[0].mData + out_desc.mStartOffset;
- ca->total_samples += ca->bytes_read / ca->in_frame_size;
- return true;
- }
- #ifdef _MSC_VER
- #pragma warning(pop)
- #endif
- static void aac_audio_info(void *data, struct audio_convert_info *info)
- {
- UNUSED_PARAMETER(data);
- info->format = AUDIO_FORMAT_FLOAT;
- }
- static size_t aac_frame_size(void *data)
- {
- ca_encoder *ca = static_cast<ca_encoder*>(data);
- return ca->out_frames_per_packet;
- }
- /* The following code was extracted from encca_aac.c in HandBrake's libhb */
- #define MP4ESDescrTag 0x03
- #define MP4DecConfigDescrTag 0x04
- #define MP4DecSpecificDescrTag 0x05
- // based off of mov_mp4_read_descr_len from mov.c in ffmpeg's libavformat
- static int read_descr_len(uint8_t **buffer)
- {
- int len = 0;
- int count = 4;
- while (count--)
- {
- int c = *(*buffer)++;
- len = (len << 7) | (c & 0x7f);
- if (!(c & 0x80))
- break;
- }
- return len;
- }
- // based off of mov_mp4_read_descr from mov.c in ffmpeg's libavformat
- static int read_descr(uint8_t **buffer, int *tag)
- {
- *tag = *(*buffer)++;
- return read_descr_len(buffer);
- }
- // based off of mov_read_esds from mov.c in ffmpeg's libavformat
- static void read_esds_desc_ext(uint8_t* desc_ext, vector<uint8_t> &buffer,
- bool version_flags)
- {
- uint8_t *esds = desc_ext;
- int tag, len;
- if (version_flags)
- esds += 4; // version + flags
- read_descr(&esds, &tag);
- esds += 2; // ID
- if (tag == MP4ESDescrTag)
- esds++; // priority
- read_descr(&esds, &tag);
- if (tag == MP4DecConfigDescrTag) {
- esds++; // object type id
- esds++; // stream type
- esds += 3; // buffer size db
- esds += 4; // max bitrate
- esds += 4; // average bitrate
- len = read_descr(&esds, &tag);
- if (tag == MP4DecSpecificDescrTag)
- try {
- buffer.assign(esds, esds + len);
- } catch (...) {
- //leave buffer empty
- }
- }
- }
- /* extracted code ends here */
- static void query_extra_data(ca_encoder *ca)
- {
- UInt32 size = 0;
- OSStatus code;
- code = AudioConverterGetPropertyInfo(ca->converter,
- kAudioConverterCompressionMagicCookie,
- &size, NULL);
- if (code) {
- log_osstatus(LOG_ERROR, ca,
- "AudioConverterGetPropertyInfo(magic_cookie)",
- code);
- return;
- }
- if (!size) {
- CA_BLOG(LOG_WARNING, "Got 0 data size info for magic_cookie");
- return;
- }
- vector<uint8_t> extra_data;
-
- try {
- extra_data.resize(size);
- } catch (...) {
- CA_BLOG(LOG_WARNING, "Could not allocate extra data buffer");
- return;
- }
- code = AudioConverterGetProperty(ca->converter,
- kAudioConverterCompressionMagicCookie,
- &size, extra_data.data());
- if (code) {
- log_osstatus(LOG_ERROR, ca,
- "AudioConverterGetProperty(magic_cookie)",
- code);
- return;
- }
- if (!size) {
- CA_BLOG(LOG_WARNING, "Got 0 data size for magic_cookie");
- return;
- }
- read_esds_desc_ext(extra_data.data(), ca->extra_data, false);
- }
- static bool aac_extra_data(void *data, uint8_t **extra_data, size_t *size)
- {
- ca_encoder *ca = static_cast<ca_encoder*>(data);
- if (!ca->extra_data.size())
- query_extra_data(ca);
- if (!ca->extra_data.size())
- return false;
- *extra_data = ca->extra_data.data();
- *size = ca->extra_data.size();
- return true;
- }
- static asbd_builder fill_common_asbd_fields(asbd_builder builder,
- bool in=false)
- {
- UInt32 bytes_per_frame = 8;
- UInt32 channels = 2;
- UInt32 bits_per_channel = bytes_per_frame / channels * 8;
- builder.channels_per_frame(channels);
- if (in) {
- builder
- .bytes_per_frame(bytes_per_frame)
- .frames_per_packet(1)
- .bytes_per_packet(1 * bytes_per_frame)
- .bits_per_channel(bits_per_channel);
- }
- return builder;
- }
- static AudioStreamBasicDescription get_default_in_asbd()
- {
- return fill_common_asbd_fields(asbd_builder(), true)
- .sample_rate(44100)
- .format_id(kAudioFormatLinearPCM)
- .format_flags(kAudioFormatFlagsNativeEndian |
- kAudioFormatFlagIsPacked |
- kAudioFormatFlagIsFloat |
- 0)
- .asbd;
- }
- static asbd_builder get_default_out_asbd_builder()
- {
- return fill_common_asbd_fields(asbd_builder())
- .sample_rate(44100);
- }
- static cf_ptr<AudioConverterRef> get_converter(DStr &log, ca_encoder *ca,
- AudioStreamBasicDescription out,
- AudioStreamBasicDescription in = get_default_in_asbd())
- {
- UInt32 size = sizeof(out);
- OSStatus code;
- #define STATUS_CHECK(x) \
- code = x; \
- if (code) { \
- log_to_dstr(log, ca, "%s: %s\n", #x, \
- osstatus_to_dstr(code)->array); \
- return nullptr; \
- }
- STATUS_CHECK(AudioFormatGetProperty(kAudioFormatProperty_FormatInfo,
- 0, NULL, &size, &out));
- AudioConverterRef converter;
- STATUS_CHECK(AudioConverterNew(&in, &out, &converter));
- return cf_ptr<AudioConverterRef>{converter};
- #undef STATUS_CHECK
- }
- static AudioConverterRef get_default_converter(UInt32 format_id)
- {
- auto out = get_default_out_asbd_builder()
- .format_id(format_id)
- .asbd;
- DStr log;
- auto converter = get_converter(log, nullptr, out);
- if (!converter) {
- CA_LOG(LOG_ERROR, "Couldn't get default converter for format "
- "%s (0x%x):\n%s",
- format_id_to_str(format_id),
- static_cast<uint32_t>(format_id),
- flush_log(log));
- return nullptr;
- }
- return converter.release();
- }
- static AudioConverterRef aac_default_converter(void)
- {
- return get_default_converter(kAudioFormatMPEG4AAC);
- }
- static AudioConverterRef he_aac_default_converter(void)
- {
- return get_default_converter(kAudioFormatMPEG4AAC_HE);
- }
- static bool find_best_match(DStr &log, ca_encoder *ca, UInt32 bitrate,
- UInt32 &best_match)
- {
- UInt32 actual_bitrate = bitrate * 1000;
- bool found_match = false;
- auto handle_bitrate = [&](UInt32 candidate)
- {
- if (abs(static_cast<intmax_t>(actual_bitrate - candidate)) <
- abs(static_cast<intmax_t>(actual_bitrate - best_match))) {
- log_to_dstr(log, ca, "Found new best match %u\n",
- static_cast<uint32_t>(candidate));
- found_match = true;
- best_match = candidate;
- }
- };
- auto helper = [&](UInt32 min_, UInt32 max_)
- {
- handle_bitrate(min_);
- if (min_ == max_)
- return;
- log_to_dstr(log, ca, "Got actual bit rate range: %u<->%u\n",
- static_cast<uint32_t>(min_),
- static_cast<uint32_t>(max_));
- handle_bitrate(max_);
- };
- for (UInt32 format_id : aac_formats) {
- log_to_dstr(log, ca, "Trying %s (0x%x)\n",
- format_id_to_str(format_id), format_id);
- auto out = get_default_out_asbd_builder()
- .format_id(format_id)
- .asbd;
- auto converter = get_converter(log, ca, out);
- if (converter)
- enumerate_bitrates(log, ca, converter.get(),
- helper);
- else
- log_to_dstr(log, ca, "Could not get converter\n");
- }
- best_match /= 1000;
- return found_match;
- }
- static UInt32 find_matching_bitrate(UInt32 bitrate)
- {
- static UInt32 match = bitrate;
- static once_flag once;
- call_once(once, [&]()
- {
- DStr log;
- ca_encoder *ca = nullptr;
- if (!find_best_match(log, ca, bitrate, match)) {
- CA_CO_DLOG(LOG_ERROR, "No matching bitrates found for "
- "target bitrate %u",
- static_cast<uint32_t>(bitrate));
- match = bitrate;
- return;
- }
- if (match != bitrate) {
- CA_CO_DLOG(LOG_INFO, "Default bitrate (%u) isn't "
- "supported, returning %u as closest match",
- static_cast<uint32_t>(bitrate),
- static_cast<uint32_t>(match));
- return;
- }
- if (log->len)
- CA_CO_DLOG(LOG_DEBUG, "Default bitrate matching log "
- "for bitrate %u",
- static_cast<uint32_t>(bitrate));
- });
- return match;
- }
- static void aac_defaults(obs_data_t *settings)
- {
- obs_data_set_default_int(settings, "bitrate",
- find_matching_bitrate(128));
- obs_data_set_default_bool(settings, "allow he-aac", true);
- }
- static vector<UInt32> get_bitrates(DStr &log, ca_encoder *ca)
- {
- vector<UInt32> bitrates;
- auto handle_bitrate = [&](UInt32 bitrate)
- {
- if (find(begin(bitrates), end(bitrates), bitrate) ==
- end(bitrates)) {
- log_to_dstr(log, ca, "Adding bitrate %u\n",
- static_cast<uint32_t>(bitrate));
- bitrates.push_back(bitrate);
- } else {
- log_to_dstr(log, ca, "Bitrate %u already added\n",
- static_cast<uint32_t>(bitrate));
- }
- };
- auto helper = [&](UInt32 min_, UInt32 max_)
- {
- handle_bitrate(min_);
- if (min_ == max_)
- return;
- log_to_dstr(log, ca, "Got actual bitrate range: %u<->%u\n",
- static_cast<uint32_t>(min_),
- static_cast<uint32_t>(max_));
- handle_bitrate(max_);
- };
- for (UInt32 format_id : (ca ? *ca->allowed_formats : aac_formats)) {
- log_to_dstr(log, ca, "Trying %s (0x%x)\n",
- format_id_to_str(format_id),
- static_cast<uint32_t>(format_id));
- auto out = get_default_out_asbd_builder()
- .format_id(format_id)
- .asbd;
- auto converter = get_converter(log, ca, out);
- if (converter)
- enumerate_bitrates(log, ca, converter.get(), helper);
- }
- return bitrates;
- }
- static void add_bitrates(obs_property_t *prop, ca_encoder *ca)
- {
- DStr log;
- auto bitrates = get_bitrates(log, ca);
- if (!bitrates.size()) {
- CA_CO_DLOG_(LOG_ERROR, "Couldn't find available bitrates");
- return;
- }
- if (log->len)
- CA_CO_DLOG_(LOG_DEBUG, "Bitrate enumeration log");
- sort(begin(bitrates), end(bitrates));
- DStr buffer;
- for (UInt32 bitrate : bitrates) {
- dstr_printf(buffer, "%u", (uint32_t)bitrate / 1000);
- obs_property_list_add_int(prop, buffer->array, bitrate / 1000);
- }
- }
- static obs_properties_t *aac_properties(void *data)
- {
- ca_encoder *ca = static_cast<ca_encoder*>(data);
- obs_properties_t *props = obs_properties_create();
- obs_property_t *p = obs_properties_add_list(props, "bitrate",
- obs_module_text("Bitrate"),
- OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
- add_bitrates(p, ca);
- obs_properties_add_bool(props, "allow he-aac",
- obs_module_text("AllowHEAAC"));
- return props;
- }
- OBS_DECLARE_MODULE()
- OBS_MODULE_USE_DEFAULT_LOCALE("coreaudio-encoder", "en-US")
- bool obs_module_load(void)
- {
- #ifdef _WIN32
- if (!load_core_audio()) {
- CA_LOG(LOG_WARNING, "Couldn't load CoreAudio AAC encoder");
- return true;
- }
- CA_LOG(LOG_INFO, "Adding CoreAudio AAC encoder");
- #endif
- struct obs_encoder_info aac_info;
- aac_info.id = "CoreAudio_AAC";
- aac_info.type = OBS_ENCODER_AUDIO;
- aac_info.codec = "AAC";
- aac_info.get_name = aac_get_name;
- aac_info.destroy = aac_destroy;
- aac_info.create = aac_create;
- aac_info.encode = aac_encode;
- aac_info.get_frame_size = aac_frame_size;
- aac_info.get_audio_info = aac_audio_info;
- aac_info.get_extra_data = aac_extra_data;
- aac_info.get_defaults = aac_defaults;
- aac_info.get_properties = aac_properties;
- obs_register_encoder(&aac_info);
- return true;
- }
- #ifdef _WIN32
- void obs_module_unload(void)
- {
- unload_core_audio();
- }
- #endif
|