1
0

decklink-output.cpp 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  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 "../../libobs/media-io/video-scaler.h"
  9. static void decklink_output_destroy(void *data)
  10. {
  11. auto *decklink = (DeckLinkOutput *)data;
  12. delete decklink;
  13. }
  14. static void *decklink_output_create(obs_data_t *settings, obs_output_t *output)
  15. {
  16. auto *decklinkOutput = new DeckLinkOutput(output, deviceEnum);
  17. decklinkOutput->deviceHash = obs_data_get_string(settings, DEVICE_HASH);
  18. decklinkOutput->modeID = obs_data_get_int(settings, MODE_ID);
  19. return decklinkOutput;
  20. }
  21. static void decklink_output_update(void *data, obs_data_t *settings)
  22. {
  23. auto *decklink = (DeckLinkOutput *)data;
  24. decklink->deviceHash = obs_data_get_string(settings, DEVICE_HASH);
  25. decklink->modeID = obs_data_get_int(settings, MODE_ID);
  26. }
  27. static bool decklink_output_start(void *data)
  28. {
  29. auto *decklink = (DeckLinkOutput *)data;
  30. struct obs_audio_info aoi;
  31. if (!obs_get_audio_info(&aoi)) {
  32. blog(LOG_WARNING, "No active audio");
  33. return false;
  34. }
  35. decklink->audio_samplerate = aoi.samples_per_sec;
  36. decklink->audio_planes = 2;
  37. decklink->audio_size = get_audio_size(AUDIO_FORMAT_16BIT, aoi.speakers, 1);
  38. decklink->start_timestamp = 0;
  39. ComPtr<DeckLinkDevice> device;
  40. device.Set(deviceEnum->FindByHash(decklink->deviceHash));
  41. DeckLinkDeviceMode *mode = device->FindOutputMode(decklink->modeID);
  42. decklink->SetSize(mode->GetWidth(), mode->GetHeight());
  43. struct video_scale_info to = {};
  44. to.format = VIDEO_FORMAT_UYVY;
  45. to.width = mode->GetWidth();
  46. to.height = mode->GetHeight();
  47. obs_output_set_video_conversion(decklink->GetOutput(), &to);
  48. decklink->Activate(device, decklink->modeID);
  49. struct audio_convert_info conversion = {};
  50. conversion.format = AUDIO_FORMAT_16BIT;
  51. conversion.speakers = SPEAKERS_STEREO;
  52. conversion.samples_per_sec = 48000; // Only format the decklink supports
  53. obs_output_set_audio_conversion(decklink->GetOutput(), &conversion);
  54. if (!obs_output_begin_data_capture(decklink->GetOutput(), 0))
  55. return false;
  56. return true;
  57. }
  58. static void decklink_output_stop(void *data, uint64_t)
  59. {
  60. auto *decklink = (DeckLinkOutput *)data;
  61. obs_output_end_data_capture(decklink->GetOutput());
  62. ComPtr<DeckLinkDevice> device;
  63. device.Set(deviceEnum->FindByHash(decklink->deviceHash));
  64. decklink->Deactivate();
  65. }
  66. static void decklink_output_raw_video(void *data, struct video_data *frame)
  67. {
  68. auto *decklink = (DeckLinkOutput *)data;
  69. if (!decklink->start_timestamp)
  70. decklink->start_timestamp = frame->timestamp;
  71. decklink->DisplayVideoFrame(frame);
  72. }
  73. static bool prepare_audio(DeckLinkOutput *decklink,
  74. const struct audio_data *frame,
  75. struct audio_data *output)
  76. {
  77. *output = *frame;
  78. if (frame->timestamp < decklink->start_timestamp) {
  79. uint64_t duration = (uint64_t)frame->frames * 1000000000 /
  80. (uint64_t)decklink->audio_samplerate;
  81. uint64_t end_ts = frame->timestamp + duration;
  82. uint64_t cutoff;
  83. if (end_ts <= decklink->start_timestamp)
  84. return false;
  85. cutoff = decklink->start_timestamp - frame->timestamp;
  86. output->timestamp += cutoff;
  87. cutoff *= (uint64_t)decklink->audio_samplerate / 1000000000;
  88. for (size_t i = 0; i < decklink->audio_planes; i++)
  89. output->data[i] += decklink->audio_size *
  90. (uint32_t)cutoff;
  91. output->frames -= (uint32_t)cutoff;
  92. }
  93. return true;
  94. }
  95. static void decklink_output_raw_audio(void *data, struct audio_data *frames)
  96. {
  97. auto *decklink = (DeckLinkOutput *)data;
  98. struct audio_data in;
  99. if (!decklink->start_timestamp)
  100. return;
  101. if (!prepare_audio(decklink, frames, &in))
  102. return;
  103. decklink->WriteAudio(&in);
  104. }
  105. static bool decklink_output_device_changed(obs_properties_t *props,
  106. obs_property_t *list, obs_data_t *settings)
  107. {
  108. const char *name = obs_data_get_string(settings, DEVICE_NAME);
  109. const char *hash = obs_data_get_string(settings, DEVICE_HASH);
  110. const char *mode = obs_data_get_string(settings, MODE_NAME);
  111. long long modeId = obs_data_get_int(settings, MODE_ID);
  112. size_t itemCount = obs_property_list_item_count(list);
  113. bool itemFound = false;
  114. for (size_t i = 0; i < itemCount; i++) {
  115. const char *curHash = obs_property_list_item_string(list, i);
  116. if (strcmp(hash, curHash) == 0) {
  117. itemFound = true;
  118. break;
  119. }
  120. }
  121. if (!itemFound) {
  122. obs_property_list_insert_string(list, 0, name, hash);
  123. obs_property_list_item_disable(list, 0, true);
  124. }
  125. obs_property_t *modeList = obs_properties_get(props, MODE_ID);
  126. obs_property_list_clear(modeList);
  127. ComPtr<DeckLinkDevice> device;
  128. device.Set(deviceEnum->FindByHash(hash));
  129. if (!device) {
  130. obs_property_list_add_int(modeList, mode, modeId);
  131. obs_property_list_item_disable(modeList, 0, true);
  132. } else {
  133. const std::vector<DeckLinkDeviceMode*> &modes =
  134. device->GetOutputModes();
  135. for (DeckLinkDeviceMode *mode : modes) {
  136. obs_property_list_add_int(modeList,
  137. mode->GetName().c_str(),
  138. mode->GetId());
  139. }
  140. }
  141. return true;
  142. }
  143. static obs_properties_t *decklink_output_properties(void *unused)
  144. {
  145. UNUSED_PARAMETER(unused);
  146. obs_properties_t *props = obs_properties_create();
  147. obs_property_t *list = obs_properties_add_list(props, DEVICE_HASH,
  148. TEXT_DEVICE, OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  149. obs_property_set_modified_callback(list, decklink_output_device_changed);
  150. fill_out_devices(list);
  151. obs_properties_add_list(props,
  152. MODE_ID, TEXT_MODE, OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  153. obs_properties_add_bool(props, AUTO_START, TEXT_AUTO_START);
  154. return props;
  155. }
  156. static const char *decklink_output_get_name(void*)
  157. {
  158. return obs_module_text("BlackmagicDevice");
  159. }
  160. struct obs_output_info create_decklink_output_info()
  161. {
  162. struct obs_output_info decklink_output_info = {};
  163. decklink_output_info.id = "decklink_output";
  164. decklink_output_info.flags = OBS_OUTPUT_AV;
  165. decklink_output_info.get_name = decklink_output_get_name;
  166. decklink_output_info.create = decklink_output_create;
  167. decklink_output_info.destroy = decklink_output_destroy;
  168. decklink_output_info.start = decklink_output_start;
  169. decklink_output_info.stop = decklink_output_stop;
  170. decklink_output_info.get_properties = decklink_output_properties;
  171. decklink_output_info.raw_video = decklink_output_raw_video;
  172. decklink_output_info.raw_audio = decklink_output_raw_audio;
  173. decklink_output_info.update = decklink_output_update;
  174. return decklink_output_info;
  175. }