plugin-main.cpp 10 KB

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