plugin-main.cpp 12 KB

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