decklink-source.cpp 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. #include <obs-module.h>
  2. #include "const.h"
  3. #include "DecklinkInput.hpp"
  4. #include "decklink-device.hpp"
  5. #include "decklink-device-discovery.hpp"
  6. #include "decklink-devices.hpp"
  7. static void decklink_enable_buffering(DeckLinkInput *decklink, bool enabled)
  8. {
  9. obs_source_t *source = decklink->GetSource();
  10. obs_source_set_async_unbuffered(source, !enabled);
  11. decklink->buffering = enabled;
  12. }
  13. static void decklink_deactivate_when_not_showing(DeckLinkInput *decklink, bool dwns)
  14. {
  15. decklink->dwns = dwns;
  16. }
  17. static void *decklink_create(obs_data_t *settings, obs_source_t *source)
  18. {
  19. DeckLinkInput *decklink = new DeckLinkInput(source, deviceEnum);
  20. obs_source_set_async_decoupled(source, true);
  21. decklink_enable_buffering(decklink,
  22. obs_data_get_bool(settings, BUFFERING));
  23. obs_source_update(source, settings);
  24. return decklink;
  25. }
  26. static void decklink_destroy(void *data)
  27. {
  28. DeckLinkInput *decklink = (DeckLinkInput *)data;
  29. delete decklink;
  30. }
  31. static void decklink_update(void *data, obs_data_t *settings)
  32. {
  33. DeckLinkInput *decklink = (DeckLinkInput *)data;
  34. const char *hash = obs_data_get_string(settings, DEVICE_HASH);
  35. long long id = obs_data_get_int(settings, MODE_ID);
  36. BMDPixelFormat pixelFormat = (BMDPixelFormat)obs_data_get_int(settings,
  37. PIXEL_FORMAT);
  38. video_colorspace colorSpace = (video_colorspace)obs_data_get_int(settings,
  39. COLOR_SPACE);
  40. video_range_type colorRange = (video_range_type)obs_data_get_int(settings,
  41. COLOR_RANGE);
  42. int chFmtInt = (int)obs_data_get_int(settings, CHANNEL_FORMAT);
  43. if (chFmtInt == 7)
  44. chFmtInt = SPEAKERS_5POINT1;
  45. else if (chFmtInt < SPEAKERS_UNKNOWN || chFmtInt > SPEAKERS_7POINT1)
  46. chFmtInt = 2;
  47. speaker_layout channelFormat = (speaker_layout)chFmtInt;
  48. decklink_enable_buffering(decklink,
  49. obs_data_get_bool(settings, BUFFERING));
  50. decklink_deactivate_when_not_showing(decklink,
  51. obs_data_get_bool(settings, DEACTIVATE_WNS));
  52. ComPtr<DeckLinkDevice> device;
  53. device.Set(deviceEnum->FindByHash(hash));
  54. decklink->SetPixelFormat(pixelFormat);
  55. decklink->SetColorSpace(colorSpace);
  56. decklink->SetColorRange(colorRange);
  57. decklink->SetChannelFormat(channelFormat);
  58. decklink->hash = std::string(hash);
  59. decklink->swap = obs_data_get_bool(settings, SWAP);
  60. decklink->Activate(device, id);
  61. }
  62. static void decklink_show(void *data)
  63. {
  64. DeckLinkInput *decklink = (DeckLinkInput *)data;
  65. obs_source_t *source = decklink->GetSource();
  66. bool showing = obs_source_showing(source);
  67. if (decklink->dwns && showing && !decklink->Capturing()) {
  68. ComPtr<DeckLinkDevice> device;
  69. device.Set(deviceEnum->FindByHash(decklink->hash.c_str()));
  70. decklink->Activate(device, decklink->id);
  71. }
  72. }
  73. static void decklink_hide(void *data)
  74. {
  75. DeckLinkInput *decklink = (DeckLinkInput *)data;
  76. obs_source_t *source = decklink->GetSource();
  77. bool showing = obs_source_showing(source);
  78. if (decklink->dwns && showing)
  79. decklink->Deactivate();
  80. }
  81. static void decklink_get_defaults(obs_data_t *settings)
  82. {
  83. obs_data_set_default_bool(settings, BUFFERING, false);
  84. obs_data_set_default_int(settings, PIXEL_FORMAT, bmdFormat8BitYUV);
  85. obs_data_set_default_int(settings, COLOR_SPACE, VIDEO_CS_DEFAULT);
  86. obs_data_set_default_int(settings, COLOR_RANGE, VIDEO_RANGE_DEFAULT);
  87. obs_data_set_default_int(settings, CHANNEL_FORMAT, SPEAKERS_STEREO);
  88. obs_data_set_default_bool(settings, SWAP, false);
  89. }
  90. static const char *decklink_get_name(void*)
  91. {
  92. return obs_module_text("BlackmagicDevice");
  93. }
  94. static bool decklink_device_changed(obs_properties_t *props,
  95. obs_property_t *list, obs_data_t *settings)
  96. {
  97. const char *name = obs_data_get_string(settings, DEVICE_NAME);
  98. const char *hash = obs_data_get_string(settings, DEVICE_HASH);
  99. const char *mode = obs_data_get_string(settings, MODE_NAME);
  100. long long modeId = obs_data_get_int(settings, MODE_ID);
  101. size_t itemCount = obs_property_list_item_count(list);
  102. bool itemFound = false;
  103. for (size_t i = 0; i < itemCount; i++) {
  104. const char *curHash = obs_property_list_item_string(list, i);
  105. if (strcmp(hash, curHash) == 0) {
  106. itemFound = true;
  107. break;
  108. }
  109. }
  110. if (!itemFound) {
  111. obs_property_list_insert_string(list, 0, name, hash);
  112. obs_property_list_item_disable(list, 0, true);
  113. }
  114. obs_property_t *modeList = obs_properties_get(props, MODE_ID);
  115. obs_property_t *channelList = obs_properties_get(props, CHANNEL_FORMAT);
  116. obs_property_list_clear(modeList);
  117. obs_property_list_clear(channelList);
  118. obs_property_list_add_int(channelList, TEXT_CHANNEL_FORMAT_NONE,
  119. SPEAKERS_UNKNOWN);
  120. obs_property_list_add_int(channelList, TEXT_CHANNEL_FORMAT_2_0CH,
  121. SPEAKERS_STEREO);
  122. ComPtr<DeckLinkDevice> device;
  123. device.Set(deviceEnum->FindByHash(hash));
  124. if (!device) {
  125. obs_property_list_add_int(modeList, mode, modeId);
  126. obs_property_list_item_disable(modeList, 0, true);
  127. } else {
  128. const std::vector<DeckLinkDeviceMode*> &modes =
  129. device->GetInputModes();
  130. for (DeckLinkDeviceMode *mode : modes) {
  131. obs_property_list_add_int(modeList,
  132. mode->GetName().c_str(),
  133. mode->GetId());
  134. }
  135. if (device->GetMaxChannel() >= 8) {
  136. obs_property_list_add_int(channelList,
  137. TEXT_CHANNEL_FORMAT_2_1CH, SPEAKERS_2POINT1);
  138. obs_property_list_add_int(channelList,
  139. TEXT_CHANNEL_FORMAT_4_0CH, SPEAKERS_4POINT0);
  140. obs_property_list_add_int(channelList,
  141. TEXT_CHANNEL_FORMAT_4_1CH, SPEAKERS_4POINT1);
  142. obs_property_list_add_int(channelList,
  143. TEXT_CHANNEL_FORMAT_5_1CH, SPEAKERS_5POINT1);
  144. obs_property_list_add_int(channelList,
  145. TEXT_CHANNEL_FORMAT_7_1CH, SPEAKERS_7POINT1);
  146. }
  147. }
  148. return true;
  149. }
  150. static bool color_format_changed(obs_properties_t *props,
  151. obs_property_t *list, obs_data_t *settings);
  152. static bool mode_id_changed(obs_properties_t *props,
  153. obs_property_t *list, obs_data_t *settings)
  154. {
  155. long long id = obs_data_get_int(settings, MODE_ID);
  156. list = obs_properties_get(props, PIXEL_FORMAT);
  157. obs_property_set_visible(list, id != MODE_ID_AUTO);
  158. return color_format_changed(props, nullptr, settings);
  159. }
  160. static bool color_format_changed(obs_properties_t *props,
  161. obs_property_t *list, obs_data_t *settings)
  162. {
  163. long long id = obs_data_get_int(settings, MODE_ID);
  164. BMDPixelFormat pixelFormat = (BMDPixelFormat)obs_data_get_int(settings,
  165. PIXEL_FORMAT);
  166. list = obs_properties_get(props, COLOR_SPACE);
  167. obs_property_set_visible(list,
  168. id != MODE_ID_AUTO && pixelFormat == bmdFormat8BitYUV);
  169. list = obs_properties_get(props, COLOR_RANGE);
  170. obs_property_set_visible(list,
  171. id == MODE_ID_AUTO || pixelFormat == bmdFormat8BitYUV);
  172. return true;
  173. }
  174. static obs_properties_t *decklink_get_properties(void *data)
  175. {
  176. obs_properties_t *props = obs_properties_create();
  177. obs_property_t *list = obs_properties_add_list(props, DEVICE_HASH,
  178. TEXT_DEVICE, OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  179. obs_property_set_modified_callback(list, decklink_device_changed);
  180. fill_out_devices(list);
  181. list = obs_properties_add_list(props, MODE_ID, TEXT_MODE,
  182. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  183. obs_property_set_modified_callback(list, mode_id_changed);
  184. list = obs_properties_add_list(props, PIXEL_FORMAT,
  185. TEXT_PIXEL_FORMAT, OBS_COMBO_TYPE_LIST,
  186. OBS_COMBO_FORMAT_INT);
  187. obs_property_set_modified_callback(list, color_format_changed);
  188. obs_property_list_add_int(list, "8-bit YUV", bmdFormat8BitYUV);
  189. obs_property_list_add_int(list, "8-bit BGRA", bmdFormat8BitBGRA);
  190. list = obs_properties_add_list(props, COLOR_SPACE, TEXT_COLOR_SPACE,
  191. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  192. obs_property_list_add_int(list, TEXT_COLOR_SPACE_DEFAULT, VIDEO_CS_DEFAULT);
  193. obs_property_list_add_int(list, "BT.601", VIDEO_CS_601);
  194. obs_property_list_add_int(list, "BT.709", VIDEO_CS_709);
  195. list = obs_properties_add_list(props, COLOR_RANGE, TEXT_COLOR_RANGE,
  196. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  197. obs_property_list_add_int(list, TEXT_COLOR_RANGE_DEFAULT, VIDEO_RANGE_DEFAULT);
  198. obs_property_list_add_int(list, TEXT_COLOR_RANGE_PARTIAL, VIDEO_RANGE_PARTIAL);
  199. obs_property_list_add_int(list, TEXT_COLOR_RANGE_FULL, VIDEO_RANGE_FULL);
  200. list = obs_properties_add_list(props, CHANNEL_FORMAT,
  201. TEXT_CHANNEL_FORMAT, OBS_COMBO_TYPE_LIST,
  202. OBS_COMBO_FORMAT_INT);
  203. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_NONE,
  204. SPEAKERS_UNKNOWN);
  205. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_2_0CH,
  206. SPEAKERS_STEREO);
  207. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_2_1CH,
  208. SPEAKERS_2POINT1);
  209. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_4_0CH,
  210. SPEAKERS_4POINT0);
  211. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_4_1CH,
  212. SPEAKERS_4POINT1);
  213. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_5_1CH,
  214. SPEAKERS_5POINT1);
  215. obs_property_list_add_int(list, TEXT_CHANNEL_FORMAT_7_1CH,
  216. SPEAKERS_7POINT1);
  217. obs_property_t *swap = obs_properties_add_bool(props, SWAP, TEXT_SWAP);
  218. obs_property_set_long_description(swap, TEXT_SWAP_TOOLTIP);
  219. obs_properties_add_bool(props, BUFFERING, TEXT_BUFFERING);
  220. obs_properties_add_bool(props, DEACTIVATE_WNS, TEXT_DWNS);
  221. UNUSED_PARAMETER(data);
  222. return props;
  223. }
  224. struct obs_source_info create_decklink_source_info()
  225. {
  226. struct obs_source_info decklink_source_info = {};
  227. decklink_source_info.id = "decklink-input";
  228. decklink_source_info.type = OBS_SOURCE_TYPE_INPUT;
  229. decklink_source_info.output_flags = OBS_SOURCE_ASYNC_VIDEO | OBS_SOURCE_AUDIO | OBS_SOURCE_DO_NOT_DUPLICATE;
  230. decklink_source_info.create = decklink_create;
  231. decklink_source_info.destroy = decklink_destroy;
  232. decklink_source_info.get_defaults = decklink_get_defaults;
  233. decklink_source_info.get_name = decklink_get_name;
  234. decklink_source_info.get_properties = decklink_get_properties;
  235. decklink_source_info.update = decklink_update;
  236. decklink_source_info.show = decklink_show;
  237. decklink_source_info.hide = decklink_hide;
  238. return decklink_source_info;
  239. }