decklink-output.cpp 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. #include <obs-module.h>
  2. #include <obs-avc.h>
  3. #include "const.h"
  4. #include "DecklinkOutput.hpp"
  5. #include "decklink-device.hpp"
  6. #include "decklink-device-discovery.hpp"
  7. #include "decklink-devices.hpp"
  8. #include <media-io/video-scaler.h>
  9. #include <util/util_uint64.h>
  10. static void decklink_output_destroy(void *data)
  11. {
  12. auto *decklink = (DeckLinkOutput *)data;
  13. delete decklink;
  14. }
  15. static void *decklink_output_create(obs_data_t *settings, obs_output_t *output)
  16. {
  17. auto *decklinkOutput = new DeckLinkOutput(output, deviceEnum);
  18. decklinkOutput->deviceHash = obs_data_get_string(settings, DEVICE_HASH);
  19. decklinkOutput->modeID = obs_data_get_int(settings, MODE_ID);
  20. decklinkOutput->keyerMode = (int)obs_data_get_int(settings, KEYER);
  21. decklinkOutput->force_sdr = obs_data_get_bool(settings, FORCE_SDR);
  22. ComPtr<DeckLinkDevice> device;
  23. device.Set(deviceEnum->FindByHash(decklinkOutput->deviceHash));
  24. if (device) {
  25. DeckLinkDeviceMode *mode =
  26. device->FindOutputMode(decklinkOutput->modeID);
  27. struct video_scale_info to = {};
  28. to.format = VIDEO_FORMAT_BGRA;
  29. to.width = mode->GetWidth();
  30. to.height = mode->GetHeight();
  31. to.range = VIDEO_RANGE_FULL;
  32. to.colorspace = (device->GetSupportsHDRMetadata() &&
  33. !decklinkOutput->force_sdr)
  34. ? VIDEO_CS_2100_PQ
  35. : VIDEO_CS_709;
  36. obs_output_set_video_conversion(output, &to);
  37. }
  38. return decklinkOutput;
  39. }
  40. static void decklink_output_update(void *data, obs_data_t *settings)
  41. {
  42. auto *decklink = (DeckLinkOutput *)data;
  43. decklink->deviceHash = obs_data_get_string(settings, DEVICE_HASH);
  44. decklink->modeID = obs_data_get_int(settings, MODE_ID);
  45. decklink->keyerMode = (int)obs_data_get_int(settings, KEYER);
  46. decklink->force_sdr = obs_data_get_bool(settings, FORCE_SDR);
  47. }
  48. static bool decklink_output_start(void *data)
  49. {
  50. auto *decklink = (DeckLinkOutput *)data;
  51. struct obs_audio_info aoi;
  52. if (!obs_get_audio_info(&aoi)) {
  53. blog(LOG_WARNING, "No active audio");
  54. return false;
  55. }
  56. if (!decklink->deviceHash || !*decklink->deviceHash)
  57. return false;
  58. decklink->audio_samplerate = aoi.samples_per_sec;
  59. decklink->audio_planes = 2;
  60. decklink->audio_size =
  61. get_audio_size(AUDIO_FORMAT_16BIT, aoi.speakers, 1);
  62. decklink->start_timestamp = 0;
  63. ComPtr<DeckLinkDevice> device;
  64. device.Set(deviceEnum->FindByHash(decklink->deviceHash));
  65. if (!device)
  66. return false;
  67. DeckLinkDeviceMode *mode = device->FindOutputMode(decklink->modeID);
  68. struct obs_video_info ovi;
  69. if (!obs_get_video_info(&ovi)) {
  70. LOG(LOG_ERROR,
  71. "Start failed: could not retrieve obs_video_info!");
  72. return false;
  73. }
  74. if (!mode->IsEqualFrameRate(ovi.fps_num, ovi.fps_den)) {
  75. LOG(LOG_ERROR, "Start failed: FPS mismatch!");
  76. return false;
  77. }
  78. decklink->SetSize(mode->GetWidth(), mode->GetHeight());
  79. device->SetKeyerMode(decklink->keyerMode);
  80. if (!decklink->Activate(device, decklink->modeID))
  81. return false;
  82. struct audio_convert_info conversion = {};
  83. conversion.format = AUDIO_FORMAT_16BIT;
  84. conversion.speakers = SPEAKERS_STEREO;
  85. conversion.samples_per_sec = 48000; // Only format the decklink supports
  86. obs_output_set_audio_conversion(decklink->GetOutput(), &conversion);
  87. if (!obs_output_begin_data_capture(decklink->GetOutput(), 0))
  88. return false;
  89. return true;
  90. }
  91. static void decklink_output_stop(void *data, uint64_t)
  92. {
  93. auto *decklink = (DeckLinkOutput *)data;
  94. obs_output_end_data_capture(decklink->GetOutput());
  95. decklink->Deactivate();
  96. }
  97. static void decklink_output_raw_video(void *data, struct video_data *frame)
  98. {
  99. auto *decklink = (DeckLinkOutput *)data;
  100. if (!decklink->start_timestamp)
  101. decklink->start_timestamp = frame->timestamp;
  102. decklink->UpdateVideoFrame(frame);
  103. }
  104. static bool prepare_audio(DeckLinkOutput *decklink,
  105. const struct audio_data *frame,
  106. struct audio_data *output)
  107. {
  108. *output = *frame;
  109. if (frame->timestamp < decklink->start_timestamp) {
  110. uint64_t duration = util_mul_div64(frame->frames, 1000000000ULL,
  111. decklink->audio_samplerate);
  112. uint64_t end_ts = frame->timestamp + duration;
  113. uint64_t cutoff;
  114. if (end_ts <= decklink->start_timestamp)
  115. return false;
  116. cutoff = decklink->start_timestamp - frame->timestamp;
  117. output->timestamp += cutoff;
  118. cutoff = util_mul_div64(cutoff, decklink->audio_samplerate,
  119. 1000000000ULL);
  120. for (size_t i = 0; i < decklink->audio_planes; i++)
  121. output->data[i] +=
  122. decklink->audio_size * (uint32_t)cutoff;
  123. output->frames -= (uint32_t)cutoff;
  124. }
  125. return true;
  126. }
  127. static void decklink_output_raw_audio(void *data, struct audio_data *frames)
  128. {
  129. auto *decklink = (DeckLinkOutput *)data;
  130. struct audio_data in;
  131. if (!decklink->start_timestamp)
  132. return;
  133. if (!prepare_audio(decklink, frames, &in))
  134. return;
  135. decklink->WriteAudio(&in);
  136. }
  137. static bool decklink_output_device_changed(obs_properties_t *props,
  138. obs_property_t *list,
  139. obs_data_t *settings)
  140. {
  141. const char *hash = obs_data_get_string(settings, DEVICE_HASH);
  142. if (*hash) {
  143. const char *name = obs_data_get_string(settings, DEVICE_NAME);
  144. const char *mode = obs_data_get_string(settings, MODE_NAME);
  145. long long modeId = obs_data_get_int(settings, MODE_ID);
  146. size_t itemCount = obs_property_list_item_count(list);
  147. bool itemFound = false;
  148. for (size_t i = 0; i < itemCount; i++) {
  149. const char *curHash =
  150. obs_property_list_item_string(list, i);
  151. if (strcmp(hash, curHash) == 0) {
  152. itemFound = true;
  153. break;
  154. }
  155. }
  156. if (!itemFound) {
  157. obs_property_list_insert_string(list, 0, name, hash);
  158. obs_property_list_item_disable(list, 0, true);
  159. }
  160. obs_property_t *modeList = obs_properties_get(props, MODE_ID);
  161. obs_property_t *keyerList = obs_properties_get(props, KEYER);
  162. obs_property_list_clear(modeList);
  163. obs_property_list_clear(keyerList);
  164. ComPtr<DeckLinkDevice> device;
  165. device.Set(deviceEnum->FindByHash(hash));
  166. if (!device) {
  167. obs_property_list_add_int(modeList, mode, modeId);
  168. obs_property_list_item_disable(modeList, 0, true);
  169. obs_property_list_item_disable(keyerList, 0, true);
  170. } else {
  171. const std::vector<DeckLinkDeviceMode *> &modes =
  172. device->GetOutputModes();
  173. struct obs_video_info ovi;
  174. if (obs_get_video_info(&ovi)) {
  175. for (DeckLinkDeviceMode *mode : modes) {
  176. if (mode->IsEqualFrameRate(
  177. ovi.fps_num, ovi.fps_den)) {
  178. obs_property_list_add_int(
  179. modeList,
  180. mode->GetName().c_str(),
  181. mode->GetId());
  182. }
  183. }
  184. }
  185. obs_property_list_add_int(keyerList, "Disabled", 0);
  186. if (device->GetSupportsExternalKeyer()) {
  187. obs_property_list_add_int(keyerList, "External",
  188. 1);
  189. }
  190. if (device->GetSupportsInternalKeyer()) {
  191. obs_property_list_add_int(keyerList, "Internal",
  192. 2);
  193. }
  194. }
  195. }
  196. return true;
  197. }
  198. static obs_properties_t *decklink_output_properties(void *unused)
  199. {
  200. UNUSED_PARAMETER(unused);
  201. obs_properties_t *props = obs_properties_create();
  202. obs_property_t *list = obs_properties_add_list(props, DEVICE_HASH,
  203. TEXT_DEVICE,
  204. OBS_COMBO_TYPE_LIST,
  205. OBS_COMBO_FORMAT_STRING);
  206. obs_property_set_modified_callback(list,
  207. decklink_output_device_changed);
  208. fill_out_devices(list);
  209. obs_properties_add_list(props, MODE_ID, TEXT_MODE, OBS_COMBO_TYPE_LIST,
  210. OBS_COMBO_FORMAT_INT);
  211. obs_properties_add_bool(props, AUTO_START, TEXT_AUTO_START);
  212. obs_properties_add_bool(props, FORCE_SDR, TEXT_FORCE_SDR);
  213. obs_properties_add_list(props, KEYER, TEXT_ENABLE_KEYER,
  214. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  215. return props;
  216. }
  217. static const char *decklink_output_get_name(void *)
  218. {
  219. return obs_module_text("BlackmagicDevice");
  220. }
  221. struct obs_output_info create_decklink_output_info()
  222. {
  223. struct obs_output_info decklink_output_info = {};
  224. decklink_output_info.id = "decklink_output";
  225. decklink_output_info.flags = OBS_OUTPUT_AV;
  226. decklink_output_info.get_name = decklink_output_get_name;
  227. decklink_output_info.create = decklink_output_create;
  228. decklink_output_info.destroy = decklink_output_destroy;
  229. decklink_output_info.start = decklink_output_start;
  230. decklink_output_info.stop = decklink_output_stop;
  231. decklink_output_info.get_properties = decklink_output_properties;
  232. decklink_output_info.raw_video = decklink_output_raw_video;
  233. decklink_output_info.raw_audio = decklink_output_raw_audio;
  234. decklink_output_info.update = decklink_output_update;
  235. return decklink_output_info;
  236. }