aja-common.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173
  1. #include "aja-card-manager.hpp"
  2. #include "aja-common.hpp"
  3. #include "aja-ui-props.hpp"
  4. #include "aja-props.hpp"
  5. #include <ajantv2/includes/ntv2debug.h>
  6. #include <ajantv2/includes/ntv2devicescanner.h>
  7. #include <ajantv2/includes/ntv2devicefeatures.h>
  8. #include <ajantv2/includes/ntv2signalrouter.h>
  9. #include <ajantv2/includes/ntv2utils.h>
  10. void filter_io_selection_input_list(const std::string &cardID,
  11. const std::string &channelOwner,
  12. obs_property_t *list)
  13. {
  14. auto &cardManager = aja::CardManager::Instance();
  15. auto cardEntry = cardManager.GetCardEntry(cardID);
  16. if (!cardEntry) {
  17. blog(LOG_DEBUG,
  18. "filter_io_selection_input_list: Card Entry not found for %s",
  19. cardID.c_str());
  20. return;
  21. }
  22. NTV2DeviceID deviceID = DEVICE_ID_NOTFOUND;
  23. CNTV2Card *card = cardEntry->GetCard();
  24. if (card)
  25. deviceID = card->GetDeviceID();
  26. // Gray out the IOSelection list items that are in use by other plugin instances
  27. for (size_t idx = 0; idx < obs_property_list_item_count(list); idx++) {
  28. auto io_select = static_cast<IOSelection>(
  29. obs_property_list_item_int(list, idx));
  30. if (io_select == IOSelection::Invalid) {
  31. obs_property_list_item_disable(list, idx, false);
  32. continue;
  33. }
  34. bool enabled = cardEntry->InputSelectionReady(
  35. io_select, deviceID, channelOwner);
  36. obs_property_list_item_disable(list, idx, !enabled);
  37. blog(LOG_DEBUG, "IOSelection %s = %s",
  38. aja::IOSelectionToString(io_select).c_str(),
  39. enabled ? "enabled" : "disabled");
  40. }
  41. }
  42. void filter_io_selection_output_list(const std::string &cardID,
  43. const std::string &channelOwner,
  44. obs_property_t *list)
  45. {
  46. auto &cardManager = aja::CardManager::Instance();
  47. auto cardEntry = cardManager.GetCardEntry(cardID);
  48. if (!cardEntry) {
  49. blog(LOG_DEBUG,
  50. "filter_io_selection_output_list: Card Entry not found for %s",
  51. cardID.c_str());
  52. return;
  53. }
  54. NTV2DeviceID deviceID = DEVICE_ID_NOTFOUND;
  55. CNTV2Card *card = cardEntry->GetCard();
  56. if (card)
  57. deviceID = card->GetDeviceID();
  58. // Gray out the IOSelection list items that are in use by other plugin instances
  59. for (size_t idx = 0; idx < obs_property_list_item_count(list); idx++) {
  60. auto io_select = static_cast<IOSelection>(
  61. obs_property_list_item_int(list, idx));
  62. if (io_select == IOSelection::Invalid) {
  63. obs_property_list_item_disable(list, idx, false);
  64. continue;
  65. }
  66. bool enabled = cardEntry->OutputSelectionReady(
  67. io_select, deviceID, channelOwner);
  68. obs_property_list_item_disable(list, idx, !enabled);
  69. blog(LOG_DEBUG, "IOSelection %s = %s",
  70. aja::IOSelectionToString(io_select).c_str(),
  71. enabled ? "enabled" : "disabled");
  72. }
  73. }
  74. void populate_io_selection_input_list(const std::string &cardID,
  75. const std::string &channelOwner,
  76. NTV2DeviceID deviceID,
  77. obs_property_t *list)
  78. {
  79. obs_property_list_clear(list);
  80. obs_property_list_add_int(list,
  81. obs_module_text(kUIPropIOSelectNone.text),
  82. static_cast<long long>(IOSelection::Invalid));
  83. for (auto i = 0; i < static_cast<int32_t>(IOSelection::NumIOSelections);
  84. i++) {
  85. auto ioSelect = static_cast<IOSelection>(i);
  86. if (ioSelect == IOSelection::AnalogIn)
  87. continue;
  88. if (aja::DeviceCanDoIOSelectionIn(deviceID, ioSelect)) {
  89. obs_property_list_add_int(
  90. list,
  91. aja::IOSelectionToString(ioSelect).c_str(),
  92. static_cast<long long>(ioSelect));
  93. }
  94. }
  95. filter_io_selection_input_list(cardID, channelOwner, list);
  96. }
  97. void populate_io_selection_output_list(const std::string &cardID,
  98. const std::string &channelOwner,
  99. NTV2DeviceID deviceID,
  100. obs_property_t *list)
  101. {
  102. obs_property_list_clear(list);
  103. obs_property_list_add_int(list,
  104. obs_module_text(kUIPropIOSelectNone.text),
  105. static_cast<long long>(IOSelection::Invalid));
  106. if (deviceID == DEVICE_ID_TTAP_PRO) {
  107. obs_property_list_add_int(
  108. list, "SDI & HDMI",
  109. static_cast<long long>(IOSelection::HDMIMonitorOut));
  110. } else {
  111. for (auto i = 0;
  112. i < static_cast<int32_t>(IOSelection::NumIOSelections);
  113. i++) {
  114. auto ioSelect = static_cast<IOSelection>(i);
  115. if (ioSelect == IOSelection::Invalid ||
  116. ioSelect == IOSelection::AnalogOut)
  117. continue;
  118. if (aja::DeviceCanDoIOSelectionOut(deviceID,
  119. ioSelect)) {
  120. obs_property_list_add_int(
  121. list,
  122. aja::IOSelectionToString(ioSelect)
  123. .c_str(),
  124. static_cast<long long>(ioSelect));
  125. }
  126. }
  127. }
  128. filter_io_selection_output_list(cardID, channelOwner, list);
  129. }
  130. void populate_video_format_list(NTV2DeviceID deviceID, obs_property_t *list,
  131. NTV2VideoFormat genlockFormat, bool want4KHFR)
  132. {
  133. VideoFormatList videoFormats = {};
  134. VideoStandardList orderedStandards = {};
  135. orderedStandards.push_back(NTV2_STANDARD_525);
  136. orderedStandards.push_back(NTV2_STANDARD_625);
  137. if (NTV2DeviceCanDoHDVideo(deviceID)) {
  138. orderedStandards.push_back(NTV2_STANDARD_720);
  139. orderedStandards.push_back(NTV2_STANDARD_1080);
  140. orderedStandards.push_back(NTV2_STANDARD_1080p);
  141. orderedStandards.push_back(NTV2_STANDARD_2K);
  142. orderedStandards.push_back(NTV2_STANDARD_2Kx1080p);
  143. orderedStandards.push_back(NTV2_STANDARD_2Kx1080i);
  144. }
  145. if (NTV2DeviceCanDo4KVideo(deviceID)) {
  146. orderedStandards.push_back(NTV2_STANDARD_3840i);
  147. orderedStandards.push_back(NTV2_STANDARD_3840x2160p);
  148. if (want4KHFR)
  149. orderedStandards.push_back(NTV2_STANDARD_3840HFR);
  150. orderedStandards.push_back(NTV2_STANDARD_4096i);
  151. orderedStandards.push_back(NTV2_STANDARD_4096x2160p);
  152. if (want4KHFR)
  153. orderedStandards.push_back(NTV2_STANDARD_4096HFR);
  154. }
  155. aja::GetSortedVideoFormats(deviceID, orderedStandards, videoFormats);
  156. for (const auto &vf : videoFormats) {
  157. bool addFormat = true;
  158. // Filter formats by framerate family if specified
  159. if (genlockFormat != NTV2_FORMAT_UNKNOWN)
  160. addFormat = IsMultiFormatCompatible(genlockFormat, vf);
  161. if (addFormat) {
  162. std::string name = NTV2VideoFormatToString(vf, true);
  163. obs_property_list_add_int(list, name.c_str(), (int)vf);
  164. }
  165. }
  166. }
  167. void populate_pixel_format_list(NTV2DeviceID deviceID, obs_property_t *list)
  168. {
  169. const NTV2PixelFormat supported_pix_fmts[] = {kDefaultAJAPixelFormat,
  170. NTV2_FBF_24BIT_BGR};
  171. for (auto &&pf : supported_pix_fmts) {
  172. if (NTV2DeviceCanDoFrameBufferFormat(deviceID, pf)) {
  173. obs_property_list_add_int(
  174. list,
  175. NTV2FrameBufferFormatToString(pf, true).c_str(),
  176. static_cast<long long>(pf));
  177. }
  178. }
  179. }
  180. void populate_sdi_transport_list(obs_property_t *list, IOSelection io,
  181. NTV2DeviceID deviceID, bool capture)
  182. {
  183. if (capture) {
  184. obs_property_list_add_int(list, obs_module_text("Auto"),
  185. kAutoDetect);
  186. }
  187. for (int i = 0; i < (int)SDITransport::Unknown; i++) {
  188. SDITransport sdi_trx = static_cast<SDITransport>(i);
  189. if (sdi_trx == SDITransport::SDI6G ||
  190. sdi_trx == SDITransport::SDI12G) {
  191. if (!NTV2DeviceCanDo12GSDI(deviceID))
  192. continue;
  193. }
  194. // Disabling 12G in Output plugin until AJA 4K HFR bug is fixed
  195. if (!capture && sdi_trx == SDITransport::SDI12G)
  196. continue;
  197. obs_property_list_add_int(
  198. list, aja::SDITransportToString(sdi_trx).c_str(),
  199. static_cast<long long>(sdi_trx));
  200. }
  201. }
  202. void populate_sdi_4k_transport_list(obs_property_t *list)
  203. {
  204. obs_property_list_add_int(
  205. list,
  206. aja::SDITransport4KToString(SDITransport4K::Squares).c_str(),
  207. static_cast<long long>(SDITransport4K::Squares));
  208. obs_property_list_add_int(
  209. list,
  210. aja::SDITransport4KToString(SDITransport4K::TwoSampleInterleave)
  211. .c_str(),
  212. static_cast<long long>(SDITransport4K::TwoSampleInterleave));
  213. }
  214. bool aja_video_format_changed(obs_properties_t *props, obs_property_t *list,
  215. obs_data_t *settings)
  216. {
  217. UNUSED_PARAMETER(list);
  218. auto vid_fmt = static_cast<NTV2VideoFormat>(
  219. obs_data_get_int(settings, kUIPropVideoFormatSelect.id));
  220. size_t itemCount = obs_property_list_item_count(list);
  221. bool itemFound = false;
  222. for (size_t i = 0; i < itemCount; i++) {
  223. int itemFormat = obs_property_list_item_int(list, i);
  224. if (itemFormat == vid_fmt) {
  225. itemFound = true;
  226. break;
  227. }
  228. }
  229. if (!itemFound) {
  230. obs_property_list_insert_int(list, 0, "", vid_fmt);
  231. obs_property_list_item_disable(list, 0, true);
  232. return true;
  233. }
  234. obs_property_t *sdi_4k_trx =
  235. obs_properties_get(props, kUIPropSDITransport4K.id);
  236. obs_property_set_visible(sdi_4k_trx, NTV2_IS_4K_VIDEO_FORMAT(vid_fmt));
  237. return true;
  238. }
  239. namespace aja {
  240. video_format AJAPixelFormatToOBSVideoFormat(NTV2PixelFormat pf)
  241. {
  242. video_format obs_video_format = VIDEO_FORMAT_NONE;
  243. switch (pf) {
  244. case NTV2_FBF_8BIT_YCBCR:
  245. obs_video_format = VIDEO_FORMAT_UYVY;
  246. break;
  247. case NTV2_FBF_24BIT_RGB:
  248. case NTV2_FBF_24BIT_BGR:
  249. obs_video_format = VIDEO_FORMAT_BGR3;
  250. break;
  251. case NTV2_FBF_ARGB:
  252. case NTV2_FBF_ABGR:
  253. case NTV2_FBF_RGBA:
  254. obs_video_format = VIDEO_FORMAT_BGRA;
  255. break;
  256. case NTV2_FBF_10BIT_YCBCR:
  257. case NTV2_FBF_10BIT_RGB:
  258. case NTV2_FBF_8BIT_YCBCR_YUY2:
  259. case NTV2_FBF_10BIT_DPX:
  260. case NTV2_FBF_10BIT_YCBCR_DPX:
  261. case NTV2_FBF_8BIT_DVCPRO:
  262. case NTV2_FBF_8BIT_YCBCR_420PL3:
  263. case NTV2_FBF_8BIT_HDV:
  264. case NTV2_FBF_10BIT_YCBCRA:
  265. case NTV2_FBF_10BIT_DPX_LE:
  266. case NTV2_FBF_48BIT_RGB:
  267. case NTV2_FBF_12BIT_RGB_PACKED:
  268. case NTV2_FBF_PRORES_DVCPRO:
  269. case NTV2_FBF_PRORES_HDV:
  270. case NTV2_FBF_10BIT_RGB_PACKED:
  271. case NTV2_FBF_10BIT_ARGB:
  272. case NTV2_FBF_16BIT_ARGB:
  273. case NTV2_FBF_8BIT_YCBCR_422PL3:
  274. case NTV2_FBF_10BIT_RAW_RGB:
  275. case NTV2_FBF_10BIT_RAW_YCBCR:
  276. case NTV2_FBF_10BIT_YCBCR_420PL3_LE:
  277. case NTV2_FBF_10BIT_YCBCR_422PL3_LE:
  278. case NTV2_FBF_10BIT_YCBCR_420PL2:
  279. case NTV2_FBF_10BIT_YCBCR_422PL2:
  280. case NTV2_FBF_8BIT_YCBCR_420PL2:
  281. case NTV2_FBF_8BIT_YCBCR_422PL2:
  282. default:
  283. obs_video_format = VIDEO_FORMAT_NONE;
  284. break;
  285. }
  286. return obs_video_format;
  287. }
  288. void GetSortedVideoFormats(NTV2DeviceID id, const VideoStandardList &standards,
  289. VideoFormatList &videoFormats)
  290. {
  291. if (standards.empty())
  292. return;
  293. VideoFormatMap videoFormatMap;
  294. // Bin all the formats based on video standard
  295. for (size_t i = (size_t)NTV2_FORMAT_UNKNOWN;
  296. i < (size_t)NTV2_MAX_NUM_VIDEO_FORMATS; i++) {
  297. NTV2VideoFormat fmt = (NTV2VideoFormat)i;
  298. NTV2Standard standard = GetNTV2StandardFromVideoFormat(fmt);
  299. if (id != DEVICE_ID_NOTFOUND &&
  300. NTV2DeviceCanDoVideoFormat(id, fmt)) {
  301. if (videoFormatMap.count(standard)) {
  302. videoFormatMap.at(standard).push_back(fmt);
  303. } else {
  304. std::vector<NTV2VideoFormat> v;
  305. v.push_back(fmt);
  306. videoFormatMap.insert(
  307. std::pair<NTV2Standard,
  308. std::vector<NTV2VideoFormat>>(
  309. standard, v));
  310. }
  311. }
  312. }
  313. for (size_t v = (size_t)NTV2_STANDARD_1080;
  314. v < (size_t)NTV2_NUM_STANDARDS; v++) {
  315. NTV2Standard standard = (NTV2Standard)v;
  316. if (videoFormatMap.count(standard)) {
  317. std::sort(videoFormatMap.at(standard).begin(),
  318. videoFormatMap.at(standard).end(),
  319. [&](const NTV2VideoFormat &d1,
  320. const NTV2VideoFormat &d2) {
  321. std::string d1Str, d2Str;
  322. d1Str = NTV2VideoFormatToString(d1);
  323. d2Str = NTV2VideoFormatToString(d2);
  324. return d1Str < d2Str;
  325. });
  326. }
  327. }
  328. for (size_t v = 0; v < standards.size(); v++) {
  329. NTV2Standard standard = standards.at(v);
  330. if (videoFormatMap.count(standard)) {
  331. for (size_t i = 0;
  332. i < videoFormatMap.at(standard).size(); i++) {
  333. NTV2VideoFormat vf =
  334. videoFormatMap.at(standard).at(i);
  335. videoFormats.push_back(vf);
  336. }
  337. }
  338. }
  339. }
  340. NTV2VideoFormat HandleSpecialCaseFormats(IOSelection io, NTV2VideoFormat vf,
  341. NTV2DeviceID id)
  342. {
  343. // 1080p Level-B formats and ST372M
  344. if (NTV2_VIDEO_FORMAT_IS_B(vf) &&
  345. !(IsSDITwoWireIOSelection(io) && NTV2_IS_HD_VIDEO_FORMAT(vf))) {
  346. vf = aja::GetLevelAFormatForLevelBFormat(vf);
  347. }
  348. // UHD/4K Square Division auto-detect
  349. if ((io == IOSelection::SDI1__4 || io == IOSelection::SDI5__8) &&
  350. NTV2_IS_HD_VIDEO_FORMAT(vf)) {
  351. vf = GetQuadSizedVideoFormat(vf, true);
  352. }
  353. // Kona5/io4K+ auto-detection of UHD/4K 6G/12G SDI formats.
  354. if (aja::IsSDIOneWireIOSelection(io) && NTV2_IS_4K_VIDEO_FORMAT(vf) &&
  355. !NTV2_IS_SQUARE_DIVISION_FORMAT(vf) &&
  356. !NTV2DeviceCanDo12gRouting(id)) {
  357. vf = GetQuadSizedVideoFormat(GetQuarterSizedVideoFormat(vf));
  358. }
  359. return vf;
  360. }
  361. NTV2Channel WidgetIDToChannel(NTV2WidgetID id)
  362. {
  363. #if AJA_NTV2_SDK_VERSION_MAJOR <= 16 && AJA_NTV2_SDK_VERSION_MINOR <= 2
  364. // Workaround for bug in NTV2 SDK <= 16.2 where NTV2_WgtSDIMonOut1 maps incorrectly to NTV2_CHANNEL1.
  365. if (id == NTV2_WgtSDIMonOut1)
  366. return NTV2_CHANNEL5;
  367. #endif
  368. return CNTV2SignalRouter::WidgetIDToChannel(id);
  369. }
  370. uint32_t CardNumFramestores(NTV2DeviceID id)
  371. {
  372. auto numFramestores = NTV2DeviceGetNumFrameStores(id);
  373. if (id == DEVICE_ID_CORVIDHBR) {
  374. numFramestores = 1;
  375. }
  376. return numFramestores;
  377. }
  378. uint32_t CardNumAudioSystems(NTV2DeviceID id)
  379. {
  380. if (id == DEVICE_ID_KONALHI || id == DEVICE_ID_KONALHEPLUS)
  381. return 2;
  382. return NTV2DeviceGetNumAudioSystems(id);
  383. }
  384. // IO4K and IO4K+ perform SDI Monitor Output on "SDI5" and "Framestore 4".
  385. bool CardCanDoSDIMonitorOutput(NTV2DeviceID id)
  386. {
  387. return (id == DEVICE_ID_IO4K || id == DEVICE_ID_IO4KPLUS);
  388. }
  389. // Cards with a dedicated HDMI Monitor Input, tied to "Framestore 4".
  390. bool CardCanDoHDMIMonitorInput(NTV2DeviceID id)
  391. {
  392. return (id == DEVICE_ID_IO4K || id == DEVICE_ID_IO4KUFC ||
  393. id == DEVICE_ID_IO4KPLUS || id == DEVICE_ID_IOXT ||
  394. id == DEVICE_ID_IOX3 || id == DEVICE_ID_KONALHI);
  395. }
  396. // Cards with a dedicated HDMI Monitor Output, tied to "Framestore 4".
  397. bool CardCanDoHDMIMonitorOutput(NTV2DeviceID id)
  398. {
  399. return (id == DEVICE_ID_IO4K || id == DEVICE_ID_IO4KPLUS ||
  400. id == DEVICE_ID_IOXT || id == DEVICE_ID_IOX3 ||
  401. id == DEVICE_ID_KONA4 || id == DEVICE_ID_KONA5 ||
  402. id == DEVICE_ID_KONA5_8K || id == DEVICE_ID_KONA5_2X4K ||
  403. id == DEVICE_ID_KONA5_8KMK);
  404. }
  405. // Cards capable of 1x SDI at 6G/12G.
  406. bool CardCanDo1xSDI12G(NTV2DeviceID id)
  407. {
  408. return (id == DEVICE_ID_KONA5_8K || id == DEVICE_ID_KONA5_8KMK ||
  409. id == DEVICE_ID_KONA5 || id == DEVICE_ID_KONA5_2X4K ||
  410. id == DEVICE_ID_IO4KPLUS || id == DEVICE_ID_CORVID44_12G);
  411. }
  412. // Check for 3G level-B SDI on the wire.
  413. bool Is3GLevelB(CNTV2Card *card, NTV2Channel channel)
  414. {
  415. if (!card)
  416. return false;
  417. bool levelB = false;
  418. auto deviceID = card->GetDeviceID();
  419. UWord channelIndex = static_cast<UWord>(channel);
  420. if (NTV2DeviceCanDo3GIn(deviceID, channelIndex) ||
  421. NTV2DeviceCanDo12GIn(deviceID, channelIndex)) {
  422. if (!card->GetSDIInput3GbPresent(levelB, channel))
  423. return false;
  424. }
  425. return levelB;
  426. }
  427. // Get the 3G Level-A enum for a 3G Level-B format enum.
  428. NTV2VideoFormat GetLevelAFormatForLevelBFormat(NTV2VideoFormat vf)
  429. {
  430. NTV2VideoFormat result = vf;
  431. switch (vf) {
  432. default:
  433. break;
  434. case NTV2_FORMAT_1080p_5000_B:
  435. result = NTV2_FORMAT_1080p_5000_A;
  436. break;
  437. case NTV2_FORMAT_1080p_5994_B:
  438. result = NTV2_FORMAT_1080p_5994_A;
  439. break;
  440. case NTV2_FORMAT_1080p_6000_B:
  441. result = NTV2_FORMAT_1080p_6000_A;
  442. break;
  443. case NTV2_FORMAT_1080p_2K_4795_B:
  444. result = NTV2_FORMAT_1080p_2K_4795_A;
  445. break;
  446. case NTV2_FORMAT_1080p_2K_4800_B:
  447. result = NTV2_FORMAT_1080p_2K_4800_A;
  448. break;
  449. case NTV2_FORMAT_1080p_2K_5000_B:
  450. result = NTV2_FORMAT_1080p_2K_5000_A;
  451. break;
  452. case NTV2_FORMAT_1080p_2K_5994_B:
  453. result = NTV2_FORMAT_1080p_2K_5994_A;
  454. break;
  455. case NTV2_FORMAT_1080p_2K_6000_B:
  456. result = NTV2_FORMAT_1080p_2K_6000_A;
  457. break;
  458. }
  459. return result;
  460. }
  461. NTV2VideoFormat InterlacedFormatForPsfFormat(NTV2VideoFormat vf)
  462. {
  463. NTV2VideoFormat result = vf;
  464. switch (vf) {
  465. default:
  466. break;
  467. case NTV2_FORMAT_1080psf_2500_2:
  468. result = NTV2_FORMAT_1080i_5000;
  469. break;
  470. case NTV2_FORMAT_1080psf_2997_2:
  471. result = NTV2_FORMAT_1080i_5994;
  472. break;
  473. }
  474. return result;
  475. }
  476. // Certain cards only have 1 SDI spigot.
  477. bool IsSingleSDIDevice(NTV2DeviceID id)
  478. {
  479. return (id == DEVICE_ID_TTAP_PRO || id == DEVICE_ID_KONA1);
  480. }
  481. bool IsIODevice(NTV2DeviceID id)
  482. {
  483. return (id == DEVICE_ID_IOXT || id == DEVICE_ID_IOX3 ||
  484. id == DEVICE_ID_IO4K || id == DEVICE_ID_IO4KPLUS ||
  485. id == DEVICE_ID_IOIP_2022 || id == DEVICE_ID_IOIP_2110);
  486. }
  487. /* Kona5 retail firmware and io4K+ have limited support for 6G/12G SDI
  488. * where SDI 1 is capable of capture and SDI 3 is capable of output.
  489. */
  490. bool IsRetail12GSDICard(NTV2DeviceID id)
  491. {
  492. return (id == DEVICE_ID_KONA5 || id == DEVICE_ID_IO4KPLUS);
  493. }
  494. bool IsOutputOnlyDevice(NTV2DeviceID id)
  495. {
  496. return id == DEVICE_ID_TTAP_PRO;
  497. }
  498. std::string SDITransportToString(SDITransport mode)
  499. {
  500. std::string str = "";
  501. switch (mode) {
  502. case SDITransport::SingleLink:
  503. str = "SD/HD Single Link";
  504. break;
  505. case SDITransport::HDDualLink:
  506. str = "HD Dual-Link";
  507. break;
  508. case SDITransport::SDI3Ga:
  509. str = "3G Level-A (3Ga)";
  510. break;
  511. case SDITransport::SDI3Gb:
  512. str = "3G Level-B (3Gb)";
  513. break;
  514. case SDITransport::SDI6G:
  515. str = "6G";
  516. break;
  517. case SDITransport::SDI12G:
  518. str = "12G";
  519. break;
  520. case SDITransport::Unknown:
  521. str = "Unknown";
  522. break;
  523. }
  524. return str;
  525. }
  526. std::string SDITransport4KToString(SDITransport4K mode)
  527. {
  528. std::string str = "";
  529. switch (mode) {
  530. case SDITransport4K::Squares:
  531. str = "Squares";
  532. break;
  533. case SDITransport4K::TwoSampleInterleave:
  534. str = "2SI";
  535. break;
  536. default:
  537. case SDITransport4K::Unknown:
  538. str = "Unknown";
  539. break;
  540. }
  541. return str;
  542. }
  543. std::string IOSelectionToString(IOSelection io)
  544. {
  545. std::string str;
  546. switch (io) {
  547. case IOSelection::SDI1:
  548. str = "SDI 1";
  549. break;
  550. case IOSelection::SDI2:
  551. str = "SDI 2";
  552. break;
  553. case IOSelection::SDI3:
  554. str = "SDI 3";
  555. break;
  556. case IOSelection::SDI4:
  557. str = "SDI 4";
  558. break;
  559. case IOSelection::SDI5:
  560. str = "SDI 5";
  561. break;
  562. case IOSelection::SDI6:
  563. str = "SDI 6";
  564. break;
  565. case IOSelection::SDI7:
  566. str = "SDI 7";
  567. break;
  568. case IOSelection::SDI8:
  569. str = "SDI 8";
  570. break;
  571. case IOSelection::SDI1_2:
  572. str = "SDI 1 & 2";
  573. break;
  574. case IOSelection::SDI3_4:
  575. str = "SDI 3 & 4";
  576. break;
  577. case IOSelection::SDI5_6:
  578. str = "SDI 5 & 6";
  579. break;
  580. case IOSelection::SDI7_8:
  581. str = "SDI 7 & 8";
  582. break;
  583. case IOSelection::SDI1__4:
  584. str = "SDI 1-4";
  585. break;
  586. case IOSelection::SDI5__8:
  587. str = "SDI 5-8";
  588. break;
  589. case IOSelection::HDMI1:
  590. str = "HDMI 1";
  591. break;
  592. case IOSelection::HDMI2:
  593. str = "HDMI 2";
  594. break;
  595. case IOSelection::HDMI3:
  596. str = "HDMI 3";
  597. break;
  598. case IOSelection::HDMI4:
  599. str = "HDMI 4";
  600. break;
  601. case IOSelection::HDMIMonitorIn:
  602. str = "HDMI IN";
  603. break;
  604. case IOSelection::HDMIMonitorOut:
  605. str = "HDMI OUT";
  606. break;
  607. case IOSelection::AnalogIn:
  608. str = "ANALOG IN";
  609. break;
  610. case IOSelection::AnalogOut:
  611. str = "ANALOG OUT";
  612. break;
  613. case IOSelection::Invalid:
  614. str = "Invalid";
  615. break;
  616. }
  617. return str;
  618. }
  619. void IOSelectionToInputSources(IOSelection io, NTV2InputSourceSet &inputSources)
  620. {
  621. switch (io) {
  622. case IOSelection::SDI1:
  623. inputSources.insert(NTV2_INPUTSOURCE_SDI1);
  624. break;
  625. case IOSelection::SDI2:
  626. inputSources.insert(NTV2_INPUTSOURCE_SDI2);
  627. break;
  628. case IOSelection::SDI3:
  629. inputSources.insert(NTV2_INPUTSOURCE_SDI3);
  630. break;
  631. case IOSelection::SDI4:
  632. inputSources.insert(NTV2_INPUTSOURCE_SDI4);
  633. break;
  634. case IOSelection::SDI5:
  635. inputSources.insert(NTV2_INPUTSOURCE_SDI5);
  636. break;
  637. case IOSelection::SDI6:
  638. inputSources.insert(NTV2_INPUTSOURCE_SDI6);
  639. break;
  640. case IOSelection::SDI7:
  641. inputSources.insert(NTV2_INPUTSOURCE_SDI7);
  642. break;
  643. case IOSelection::SDI8:
  644. inputSources.insert(NTV2_INPUTSOURCE_SDI8);
  645. break;
  646. case IOSelection::SDI1_2:
  647. inputSources.insert(NTV2_INPUTSOURCE_SDI1);
  648. inputSources.insert(NTV2_INPUTSOURCE_SDI2);
  649. break;
  650. case IOSelection::SDI3_4:
  651. inputSources.insert(NTV2_INPUTSOURCE_SDI3);
  652. inputSources.insert(NTV2_INPUTSOURCE_SDI4);
  653. break;
  654. case IOSelection::SDI5_6:
  655. inputSources.insert(NTV2_INPUTSOURCE_SDI5);
  656. inputSources.insert(NTV2_INPUTSOURCE_SDI6);
  657. break;
  658. case IOSelection::SDI7_8:
  659. inputSources.insert(NTV2_INPUTSOURCE_SDI7);
  660. inputSources.insert(NTV2_INPUTSOURCE_SDI8);
  661. break;
  662. case IOSelection::SDI1__4:
  663. inputSources.insert(NTV2_INPUTSOURCE_SDI1);
  664. inputSources.insert(NTV2_INPUTSOURCE_SDI2);
  665. inputSources.insert(NTV2_INPUTSOURCE_SDI3);
  666. inputSources.insert(NTV2_INPUTSOURCE_SDI4);
  667. break;
  668. case IOSelection::SDI5__8:
  669. inputSources.insert(NTV2_INPUTSOURCE_SDI5);
  670. inputSources.insert(NTV2_INPUTSOURCE_SDI6);
  671. inputSources.insert(NTV2_INPUTSOURCE_SDI7);
  672. inputSources.insert(NTV2_INPUTSOURCE_SDI8);
  673. break;
  674. case IOSelection::HDMI1:
  675. inputSources.insert(NTV2_INPUTSOURCE_HDMI1);
  676. break;
  677. case IOSelection::HDMI2:
  678. inputSources.insert(NTV2_INPUTSOURCE_HDMI2);
  679. break;
  680. case IOSelection::HDMI3:
  681. inputSources.insert(NTV2_INPUTSOURCE_HDMI3);
  682. break;
  683. case IOSelection::HDMI4:
  684. inputSources.insert(NTV2_INPUTSOURCE_HDMI4);
  685. break;
  686. case IOSelection::HDMIMonitorIn:
  687. inputSources.insert(NTV2_INPUTSOURCE_HDMI1);
  688. break;
  689. case IOSelection::AnalogIn:
  690. inputSources.insert(NTV2_INPUTSOURCE_ANALOG1);
  691. break;
  692. default:
  693. case IOSelection::HDMIMonitorOut:
  694. case IOSelection::AnalogOut:
  695. case IOSelection::Invalid:
  696. break;
  697. }
  698. }
  699. void IOSelectionToOutputDests(IOSelection io,
  700. NTV2OutputDestinations &outputDests)
  701. {
  702. switch (io) {
  703. case IOSelection::SDI1:
  704. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI1);
  705. break;
  706. case IOSelection::SDI2:
  707. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI2);
  708. break;
  709. case IOSelection::SDI3:
  710. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI3);
  711. break;
  712. case IOSelection::SDI4:
  713. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI4);
  714. break;
  715. case IOSelection::SDI5:
  716. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI5);
  717. break;
  718. case IOSelection::SDI6:
  719. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI6);
  720. break;
  721. case IOSelection::SDI7:
  722. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI7);
  723. break;
  724. case IOSelection::SDI8:
  725. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI8);
  726. break;
  727. case IOSelection::SDI1_2:
  728. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI1);
  729. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI2);
  730. break;
  731. case IOSelection::SDI3_4:
  732. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI3);
  733. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI4);
  734. break;
  735. case IOSelection::SDI5_6:
  736. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI5);
  737. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI6);
  738. break;
  739. case IOSelection::SDI7_8:
  740. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI7);
  741. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI8);
  742. break;
  743. case IOSelection::SDI1__4:
  744. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI1);
  745. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI2);
  746. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI3);
  747. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI4);
  748. break;
  749. case IOSelection::SDI5__8:
  750. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI5);
  751. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI6);
  752. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI7);
  753. outputDests.insert(NTV2_OUTPUTDESTINATION_SDI8);
  754. break;
  755. case IOSelection::HDMIMonitorOut:
  756. outputDests.insert(NTV2_OUTPUTDESTINATION_HDMI);
  757. break;
  758. case IOSelection::AnalogOut:
  759. outputDests.insert(NTV2_OUTPUTDESTINATION_ANALOG);
  760. break;
  761. default:
  762. case IOSelection::HDMI1:
  763. case IOSelection::HDMI2:
  764. case IOSelection::HDMI3:
  765. case IOSelection::HDMI4:
  766. case IOSelection::HDMIMonitorIn:
  767. case IOSelection::AnalogIn:
  768. case IOSelection::Invalid:
  769. break;
  770. }
  771. }
  772. bool DeviceCanDoIOSelectionIn(NTV2DeviceID id, IOSelection io)
  773. {
  774. // Hide "HDMI1" list selection on devices which have a discrete "HDMI IN" port.
  775. if (io == IOSelection::HDMI1 && CardCanDoHDMIMonitorInput(id)) {
  776. return false;
  777. }
  778. if (io == IOSelection::HDMIMonitorIn && id == DEVICE_ID_KONAHDMI) {
  779. return false;
  780. }
  781. NTV2InputSourceSet inputSources;
  782. if (io != IOSelection::Invalid) {
  783. IOSelectionToInputSources(io, inputSources);
  784. size_t numSrcs = inputSources.size();
  785. size_t canDo = 0;
  786. if (numSrcs > 0) {
  787. for (auto &&inp : inputSources) {
  788. if (NTV2DeviceCanDoInputSource(id, inp))
  789. canDo++;
  790. }
  791. if (canDo == numSrcs)
  792. return true;
  793. }
  794. }
  795. return false;
  796. }
  797. bool DeviceCanDoIOSelectionOut(NTV2DeviceID id, IOSelection io)
  798. {
  799. NTV2OutputDestinations outputDests;
  800. if (io != IOSelection::Invalid) {
  801. IOSelectionToOutputDests(io, outputDests);
  802. size_t numOuts = outputDests.size();
  803. size_t canDo = 0;
  804. if (numOuts > 0) {
  805. for (auto &&out : outputDests) {
  806. if (NTV2DeviceCanDoOutputDestination(id, out))
  807. canDo++;
  808. }
  809. if (canDo == numOuts)
  810. return true;
  811. }
  812. }
  813. return false;
  814. }
  815. bool IsSDIOneWireIOSelection(IOSelection io)
  816. {
  817. bool result = false;
  818. switch (io) {
  819. case IOSelection::SDI1:
  820. case IOSelection::SDI2:
  821. case IOSelection::SDI3:
  822. case IOSelection::SDI4:
  823. case IOSelection::SDI5:
  824. case IOSelection::SDI6:
  825. case IOSelection::SDI7:
  826. case IOSelection::SDI8:
  827. result = true;
  828. break;
  829. default:
  830. result = false;
  831. }
  832. return result;
  833. }
  834. bool IsSDITwoWireIOSelection(IOSelection io)
  835. {
  836. bool result = false;
  837. switch (io) {
  838. case IOSelection::SDI1_2:
  839. case IOSelection::SDI3_4:
  840. case IOSelection::SDI5_6:
  841. case IOSelection::SDI7_8:
  842. result = true;
  843. break;
  844. default:
  845. result = false;
  846. }
  847. return result;
  848. }
  849. bool IsSDIFourWireIOSelection(IOSelection io)
  850. {
  851. bool result = false;
  852. switch (io) {
  853. case IOSelection::SDI1__4:
  854. case IOSelection::SDI5__8:
  855. result = true;
  856. break;
  857. default:
  858. result = false;
  859. }
  860. return result;
  861. }
  862. bool IsMonitorOutputSelection(NTV2DeviceID id, IOSelection io)
  863. {
  864. if (CardCanDoSDIMonitorOutput(id) && io == IOSelection::SDI5)
  865. return true;
  866. if (CardCanDoHDMIMonitorOutput(id) && io == IOSelection::HDMIMonitorOut)
  867. return true;
  868. return false;
  869. }
  870. bool IsIOSelectionSDI(IOSelection io)
  871. {
  872. if (io == IOSelection::SDI1 || io == IOSelection::SDI2 ||
  873. io == IOSelection::SDI3 || io == IOSelection::SDI4 ||
  874. io == IOSelection::SDI5 || io == IOSelection::SDI6 ||
  875. io == IOSelection::SDI7 || io == IOSelection::SDI8 ||
  876. io == IOSelection::SDI1_2 || io == IOSelection::SDI3_4 ||
  877. io == IOSelection::SDI5_6 || io == IOSelection::SDI7_8 ||
  878. io == IOSelection::SDI1__4 || io == IOSelection::SDI5__8) {
  879. return true;
  880. }
  881. return false;
  882. }
  883. bool IsIOSelectionHDMI(IOSelection io)
  884. {
  885. if (io == IOSelection::HDMI1 || io == IOSelection::HDMI2 ||
  886. io == IOSelection::HDMI3 || io == IOSelection::HDMI4 ||
  887. io == IOSelection::HDMIMonitorIn ||
  888. io == IOSelection::HDMIMonitorOut) {
  889. return true;
  890. }
  891. return false;
  892. }
  893. std::string MakeCardID(CNTV2Card &card)
  894. {
  895. std::string cardID;
  896. if (card.GetSerialNumberString(cardID)) {
  897. // Try to construct CardID from device ID and serial number...
  898. cardID = NTV2DeviceIDToString(card.GetDeviceID(), false) + "_" +
  899. cardID;
  900. } else {
  901. // ...otherwise fall back to the CNTV2DeviceScanner method.
  902. cardID = CNTV2DeviceScanner::GetDeviceRefName(card);
  903. }
  904. return cardID;
  905. }
  906. RasterDefinition DetermineRasterDefinition(NTV2VideoFormat vf)
  907. {
  908. RasterDefinition def = RasterDefinition::Unknown;
  909. if (NTV2_IS_SD_VIDEO_FORMAT(vf)) {
  910. def = RasterDefinition::SD;
  911. } else if (NTV2_IS_HD_VIDEO_FORMAT(vf)) {
  912. def = RasterDefinition::HD;
  913. } else if (NTV2_IS_QUAD_FRAME_FORMAT(vf)) {
  914. def = RasterDefinition::UHD_4K;
  915. } else if (NTV2_IS_QUAD_QUAD_FORMAT(vf)) {
  916. def = RasterDefinition::UHD2_8K;
  917. } else {
  918. def = RasterDefinition::Unknown;
  919. }
  920. return def;
  921. }
  922. inline bool IsStandard1080i(NTV2Standard standard)
  923. {
  924. if (standard == NTV2_STANDARD_1080 ||
  925. standard == NTV2_STANDARD_2Kx1080i) {
  926. return true;
  927. }
  928. return false;
  929. }
  930. inline bool IsStandard1080p(NTV2Standard standard)
  931. {
  932. if (standard == NTV2_STANDARD_1080p || standard == NTV2_STANDARD_2K ||
  933. standard == NTV2_STANDARD_2Kx1080p) {
  934. return true;
  935. }
  936. return false;
  937. }
  938. VPIDStandard DetermineVPIDStandard(NTV2DeviceID id, IOSelection io,
  939. NTV2VideoFormat vf, NTV2PixelFormat pf,
  940. SDITransport trx, SDITransport4K t4k)
  941. {
  942. VPIDStandard vpid = VPIDStandard_Unknown;
  943. auto rd = aja::DetermineRasterDefinition(vf);
  944. auto standard = GetNTV2StandardFromVideoFormat(vf);
  945. bool is_rgb = NTV2_IS_FBF_RGB(pf);
  946. bool is_hfr =
  947. NTV2_IS_HIGH_NTV2FrameRate(GetNTV2FrameRateFromVideoFormat(vf));
  948. if (rd == RasterDefinition::SD) {
  949. vpid = VPIDStandard_483_576;
  950. } else if (rd == RasterDefinition::HD) {
  951. vpid = VPIDStandard_1080;
  952. if (aja::IsSDIOneWireIOSelection(io)) {
  953. if (is_rgb) {
  954. if (standard == NTV2_STANDARD_720) {
  955. if (trx == SDITransport::SingleLink) {
  956. vpid = VPIDStandard_720;
  957. } else if (trx ==
  958. SDITransport::SDI3Ga) {
  959. vpid = VPIDStandard_720_3Ga;
  960. } else if (trx ==
  961. SDITransport::SDI3Gb) {
  962. vpid = VPIDStandard_720_3Gb;
  963. }
  964. } else if (IsStandard1080p(standard)) {
  965. if (trx == SDITransport::SingleLink) {
  966. vpid = VPIDStandard_1080;
  967. } else if (trx ==
  968. SDITransport::SDI3Ga) {
  969. vpid = VPIDStandard_1080_3Ga;
  970. } else if (trx ==
  971. SDITransport::SDI3Gb) {
  972. vpid = VPIDStandard_1080_3Gb;
  973. }
  974. }
  975. } else {
  976. if (standard == NTV2_STANDARD_720) {
  977. vpid = VPIDStandard_720;
  978. } else if (IsStandard1080i(standard)) {
  979. vpid = VPIDStandard_1080;
  980. } else if (IsStandard1080p(standard) &&
  981. trx == SDITransport::SDI3Ga) {
  982. vpid = VPIDStandard_1080_3Ga;
  983. } else if (IsStandard1080p(standard) &&
  984. trx == SDITransport::SDI3Gb) {
  985. vpid = VPIDStandard_1080_3Gb;
  986. }
  987. }
  988. } else if (aja::IsSDITwoWireIOSelection(io)) {
  989. if (is_rgb) {
  990. if (standard == NTV2_STANDARD_720) {
  991. if (trx == SDITransport::SDI3Ga)
  992. vpid = VPIDStandard_720_3Ga;
  993. else if (trx == SDITransport::SDI3Gb)
  994. vpid = VPIDStandard_720_3Gb;
  995. } else if (IsStandard1080p(standard)) {
  996. if (trx == SDITransport::HDDualLink) {
  997. vpid = VPIDStandard_1080_DualLink;
  998. } else if (trx == SDITransport::SDI3Ga)
  999. vpid = VPIDStandard_1080_Dual_3Ga;
  1000. else if (trx == SDITransport::SDI3Gb)
  1001. vpid = VPIDStandard_1080_Dual_3Gb;
  1002. }
  1003. } else {
  1004. if (IsStandard1080p(standard) &&
  1005. trx == SDITransport::HDDualLink) {
  1006. vpid = VPIDStandard_1080_DualLink;
  1007. }
  1008. }
  1009. }
  1010. } else if (rd == RasterDefinition::UHD_4K) {
  1011. if (aja::IsSDIOneWireIOSelection(io)) {
  1012. if (is_rgb) {
  1013. if (trx == SDITransport::SDI6G) {
  1014. vpid = VPIDStandard_2160_DualLink;
  1015. } else if (trx == SDITransport::SDI12G) {
  1016. vpid = VPIDStandard_2160_DualLink_12Gb;
  1017. }
  1018. } else {
  1019. // YCbCr
  1020. if (trx == SDITransport::SDI6G) {
  1021. vpid = VPIDStandard_2160_Single_6Gb;
  1022. } else if (trx == SDITransport::SDI12G) {
  1023. vpid = VPIDStandard_2160_Single_12Gb;
  1024. } else {
  1025. vpid = VPIDStandard_2160_Single_6Gb; // fallback
  1026. }
  1027. }
  1028. } else if (aja::IsSDITwoWireIOSelection(io)) {
  1029. if (is_rgb) {
  1030. } else {
  1031. // YCbCr
  1032. if (t4k == SDITransport4K::Squares) {
  1033. vpid = VPIDStandard_1080;
  1034. } else if (t4k ==
  1035. SDITransport4K::TwoSampleInterleave) {
  1036. if (is_hfr &&
  1037. trx == SDITransport::SDI3Ga) {
  1038. vpid = VPIDStandard_2160_QuadLink_3Ga;
  1039. } else if (is_hfr &&
  1040. trx == SDITransport::SDI3Gb) {
  1041. vpid = VPIDStandard_2160_QuadDualLink_3Gb;
  1042. } else {
  1043. vpid = VPIDStandard_2160_DualLink;
  1044. }
  1045. }
  1046. }
  1047. } else if (aja::IsSDIFourWireIOSelection(io)) {
  1048. if (is_rgb) {
  1049. if (t4k == SDITransport4K::Squares) {
  1050. if (trx == SDITransport::SDI3Ga) {
  1051. vpid = VPIDStandard_1080_3Ga;
  1052. } else if (trx ==
  1053. SDITransport::SDI3Gb) {
  1054. vpid = VPIDStandard_1080_DualLink_3Gb;
  1055. }
  1056. }
  1057. } else {
  1058. // YCbCr
  1059. if (t4k == SDITransport4K::Squares) {
  1060. if (trx == SDITransport::SDI3Ga) {
  1061. vpid = VPIDStandard_1080_3Ga;
  1062. } else if (trx ==
  1063. SDITransport::SDI3Gb) {
  1064. vpid = VPIDStandard_1080_DualLink_3Gb;
  1065. } else {
  1066. vpid = VPIDStandard_1080;
  1067. }
  1068. } else if (t4k ==
  1069. SDITransport4K::TwoSampleInterleave) {
  1070. if (trx == SDITransport::SDI3Ga) {
  1071. vpid = VPIDStandard_2160_QuadLink_3Ga;
  1072. } else if (trx ==
  1073. SDITransport::SDI3Gb) {
  1074. vpid = VPIDStandard_2160_QuadDualLink_3Gb;
  1075. }
  1076. }
  1077. }
  1078. }
  1079. } else if (rd == RasterDefinition::UHD2_8K) {
  1080. }
  1081. return vpid;
  1082. }
  1083. std::vector<NTV2DeviceID> MultiViewCards()
  1084. {
  1085. return {DEVICE_ID_IOX3};
  1086. }
  1087. } // aja