plugin-main.cpp 11 KB

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