obs-libfdk.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. #include <obs-module.h>
  2. #ifdef DEBUG
  3. # ifndef _DEBUG
  4. # define _DEBUG
  5. # endif
  6. # undef DEBUG
  7. #endif
  8. #include <fdk-aac/aacenc_lib.h>
  9. static const char *libfdk_get_error(AACENC_ERROR err)
  10. {
  11. switch(err) {
  12. case AACENC_OK:
  13. return "No error";
  14. case AACENC_INVALID_HANDLE:
  15. return "Invalid handle";
  16. case AACENC_MEMORY_ERROR:
  17. return "Memory allocation error";
  18. case AACENC_UNSUPPORTED_PARAMETER:
  19. return "Unsupported parameter";
  20. case AACENC_INVALID_CONFIG:
  21. return "Invalid config";
  22. case AACENC_INIT_ERROR:
  23. return "Initialization error";
  24. case AACENC_INIT_AAC_ERROR:
  25. return "AAC library initialization error";
  26. case AACENC_INIT_SBR_ERROR:
  27. return "SBR library initialization error";
  28. case AACENC_INIT_TP_ERROR:
  29. return "Transport library initialization error";
  30. case AACENC_INIT_META_ERROR:
  31. return "Metadata library initialization error";
  32. case AACENC_ENCODE_ERROR:
  33. return "Encoding error";
  34. case AACENC_ENCODE_EOF:
  35. return "End of file";
  36. default:
  37. return "Unknown error";
  38. }
  39. }
  40. typedef struct libfdk_encoder {
  41. obs_encoder_t encoder;
  42. int channels, sample_rate;
  43. HANDLE_AACENCODER fdkhandle;
  44. AACENC_InfoStruct info;
  45. uint64_t total_samples;
  46. int frame_size_bytes;
  47. uint8_t *packet_buffer;
  48. int packet_buffer_size;
  49. } libfdk_encoder_t;
  50. static const char *libfdk_getname(void)
  51. {
  52. return obs_module_text("LibFDK");
  53. }
  54. static obs_properties_t libfdk_properties(void)
  55. {
  56. obs_properties_t props = obs_properties_create();
  57. obs_properties_add_int(props, "bitrate",
  58. obs_module_text("Bitrate"), 32, 256, 32);
  59. obs_properties_add_bool(props, "afterburner",
  60. obs_module_text("Afterburner"));
  61. return props;
  62. }
  63. static void libfdk_defaults(obs_data_t settings)
  64. {
  65. obs_data_set_default_int(settings, "bitrate", 128);
  66. obs_data_set_default_bool(settings, "afterburner", true);
  67. }
  68. #define CHECK_LIBFDK(r) \
  69. if((err = (r)) != AACENC_OK) { \
  70. blog(LOG_ERROR, #r " failed: %s", libfdk_get_error(err)); \
  71. goto fail; \
  72. }
  73. static void *libfdk_create(obs_data_t settings, obs_encoder_t encoder)
  74. {
  75. bool hasFdkHandle = false;
  76. libfdk_encoder_t *enc = 0;
  77. int bitrate = (int)obs_data_get_int(settings, "bitrate") * 1000;
  78. int afterburner = obs_data_get_bool(settings, "afterburner") ? 1 : 0;
  79. audio_t audio = obs_encoder_audio(encoder);
  80. int mode = 0;
  81. AACENC_ERROR err;
  82. if (!bitrate) {
  83. blog(LOG_ERROR, "Invalid bitrate");
  84. return NULL;
  85. }
  86. enc = bzalloc(sizeof(libfdk_encoder_t));
  87. enc->encoder = encoder;
  88. enc->channels = (int)audio_output_channels(audio);
  89. enc->sample_rate = audio_output_samplerate(audio);
  90. switch(enc->channels) {
  91. case 1:
  92. mode = MODE_1;
  93. break;
  94. case 2:
  95. mode = MODE_2;
  96. break;
  97. case 3:
  98. mode = MODE_1_2;
  99. break;
  100. case 4:
  101. mode = MODE_1_2_1;
  102. break;
  103. case 5:
  104. mode = MODE_1_2_2;
  105. break;
  106. case 6:
  107. mode = MODE_1_2_2_1;
  108. break;
  109. default:
  110. blog(LOG_ERROR, "Invalid channel count");
  111. goto fail;
  112. }
  113. CHECK_LIBFDK(aacEncOpen(&enc->fdkhandle, 0, enc->channels));
  114. hasFdkHandle = true;
  115. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_AOT,
  116. 2)); // MPEG-4 AAC-LC
  117. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_SAMPLERATE,
  118. enc->sample_rate));
  119. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_CHANNELMODE, mode));
  120. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_CHANNELORDER, 1));
  121. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_BITRATEMODE, 0));
  122. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_BITRATE, bitrate));
  123. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_TRANSMUX, 0));
  124. CHECK_LIBFDK(aacEncoder_SetParam(enc->fdkhandle, AACENC_AFTERBURNER,
  125. afterburner));
  126. CHECK_LIBFDK(aacEncEncode(enc->fdkhandle, NULL, NULL, NULL, NULL));
  127. CHECK_LIBFDK(aacEncInfo(enc->fdkhandle, &enc->info));
  128. enc->frame_size_bytes = enc->info.frameLength * 2 * enc->channels;
  129. enc->packet_buffer_size = enc->channels * 768;
  130. if(enc->packet_buffer_size < 8192)
  131. enc->packet_buffer_size = 8192;
  132. enc->packet_buffer = bmalloc(enc->packet_buffer_size);
  133. blog(LOG_INFO, "libfdk_aac encoder created");
  134. blog(LOG_INFO, "libfdk_aac bitrate: %d, channels: %d",
  135. bitrate / 1000, enc->channels);
  136. return enc;
  137. fail:
  138. if(hasFdkHandle)
  139. aacEncClose(&enc->fdkhandle);
  140. if(enc->packet_buffer)
  141. bfree(enc->packet_buffer);
  142. if(enc)
  143. bfree(enc);
  144. blog(LOG_WARNING, "libfdk_aac encoder creation failed");
  145. return 0;
  146. }
  147. static void libfdk_destroy(void *data)
  148. {
  149. libfdk_encoder_t *enc = data;
  150. aacEncClose(&enc->fdkhandle);
  151. bfree(enc->packet_buffer);
  152. bfree(enc);
  153. blog(LOG_INFO, "libfdk_aac encoder destroyed");
  154. }
  155. static bool libfdk_encode(void *data, struct encoder_frame *frame,
  156. struct encoder_packet *packet, bool *received_packet)
  157. {
  158. libfdk_encoder_t *enc = data;
  159. AACENC_BufDesc in_buf = { 0 };
  160. AACENC_BufDesc out_buf = { 0 };
  161. AACENC_InArgs in_args = { 0 };
  162. AACENC_OutArgs out_args = { 0 };
  163. int in_identifier = IN_AUDIO_DATA;
  164. int in_size, in_elem_size;
  165. int out_identifier = OUT_BITSTREAM_DATA;
  166. int out_size, out_elem_size;
  167. void *in_ptr;
  168. void *out_ptr;
  169. AACENC_ERROR err;
  170. in_ptr = frame->data[0];
  171. in_size = enc->frame_size_bytes;
  172. in_elem_size = 2;
  173. in_args.numInSamples = enc->info.frameLength * enc->channels;
  174. in_buf.numBufs = 1;
  175. in_buf.bufs = &in_ptr;
  176. in_buf.bufferIdentifiers = &in_identifier;
  177. in_buf.bufSizes = &in_size;
  178. in_buf.bufElSizes = &in_elem_size;
  179. out_ptr = enc->packet_buffer;
  180. out_size = enc->packet_buffer_size;
  181. out_elem_size = 1;
  182. out_buf.numBufs = 1;
  183. out_buf.bufs = &out_ptr;
  184. out_buf.bufferIdentifiers = &out_identifier;
  185. out_buf.bufSizes = &out_size;
  186. out_buf.bufElSizes = &out_elem_size;
  187. if((err = aacEncEncode(enc->fdkhandle, &in_buf, &out_buf, &in_args,
  188. &out_args)) != AACENC_OK) {
  189. blog(LOG_ERROR, "Failed to encode frame: %s", libfdk_get_error(err));
  190. return false;
  191. }
  192. enc->total_samples += enc->info.frameLength;
  193. if(out_args.numOutBytes == 0) {
  194. *received_packet = false;
  195. return true;
  196. }
  197. *received_packet = true;
  198. packet->pts = enc->total_samples -
  199. enc->info.encoderDelay; // TODO: Just a guess, find out if that's actualy right
  200. packet->dts = enc->total_samples - enc->info.encoderDelay;
  201. packet->data = enc->packet_buffer;
  202. packet->size = out_args.numOutBytes;
  203. packet->type = OBS_ENCODER_AUDIO;
  204. packet->timebase_num = 1;
  205. packet->timebase_den = enc->sample_rate;
  206. return true;
  207. }
  208. static bool libfdk_extra_data(void *data, uint8_t **extra_data, size_t *size)
  209. {
  210. libfdk_encoder_t *enc = data;
  211. *size = enc->info.confSize;
  212. *extra_data = enc->info.confBuf;
  213. return true;
  214. }
  215. static bool libfdk_audio_info(void *data, struct audio_convert_info *info)
  216. {
  217. UNUSED_PARAMETER(data);
  218. memset(info, 0, sizeof(struct audio_convert_info));
  219. info->format = AUDIO_FORMAT_16BIT;
  220. return true;
  221. }
  222. static size_t libfdk_frame_size(void *data)
  223. {
  224. libfdk_encoder_t *enc = data;
  225. return enc->info.frameLength;
  226. }
  227. struct obs_encoder_info obs_libfdk_encoder = {
  228. .id = "libfdk_aac",
  229. .type = OBS_ENCODER_AUDIO,
  230. .codec = "AAC",
  231. .get_name = libfdk_getname,
  232. .create = libfdk_create,
  233. .destroy = libfdk_destroy,
  234. .encode = libfdk_encode,
  235. .get_frame_size = libfdk_frame_size,
  236. .get_defaults = libfdk_defaults,
  237. .get_properties = libfdk_properties,
  238. .get_extra_data = libfdk_extra_data,
  239. .get_audio_info = libfdk_audio_info
  240. };
  241. bool obs_module_load(void)
  242. {
  243. obs_register_encoder(&obs_libfdk_encoder);
  244. return true;
  245. }
  246. OBS_DECLARE_MODULE()
  247. OBS_MODULE_USE_DEFAULT_LOCALE("obs-libfdk", "en-US")