videohubctrl can be used to control Blackmagic Design Videohub SDI router device over the network.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

display.c 6.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. /*
  2. * === Display functions ===
  3. *
  4. * Blackmagic Design Videohub control application
  5. * Copyright (C) 2014 Unix Solutions Ltd.
  6. * Written by Georgi Chorbadzhiyski
  7. *
  8. * Released under MIT license.
  9. * See LICENSE-MIT.txt for license terms.
  10. *
  11. */
  12. #include <stdio.h>
  13. #include <string.h>
  14. #include "data.h"
  15. #include "cmd.h"
  16. #include "util.h"
  17. #include "display.h"
  18. static void printf_line(int len) {
  19. int i;
  20. printf(" ");
  21. for (i = 0; i < len; i++)
  22. printf("-");
  23. printf("\n");
  24. }
  25. static char format_status(char *status) {
  26. if (!strlen(status)) return ' ';
  27. if (streq(status, "BNC")) return 'B';
  28. if (streq(status, "Optical")) return 'o';
  29. if (streq(status, "None")) return 'x';
  30. if (streq(status, "RS422")) return '4'; // For serial ports
  31. return '?';
  32. }
  33. void print_device_info(struct videohub_data *d) {
  34. int len = 59;
  35. printf("Device info\n");
  36. printf_line(len);
  37. printf(" | %-26s | %-26s |\n", "Device address", d->dev_host);
  38. printf(" | %-26s | %-26s |\n", "Device port", d->dev_port);
  39. printf(" | %-26s | %-26s |\n", "Model name", d->device.model_name);
  40. printf(" | %-26s | %-26s |\n", "Unique ID", d->device.unique_id);
  41. printf(" | %-26s | %-26s |\n", "Protocol", d->device.protocol_ver);
  42. printf(" | %-26s | %-26u |\n", "Video inputs", d->inputs.num);
  43. printf(" | %-26s | %-26u |\n", "Video outputs", d->outputs.num);
  44. if (d->serial.num)
  45. printf(" | %-26s | %-26u |\n", "Serial ports", d->serial.num);
  46. if (d->device.num_video_processing_units)
  47. printf(" | %-26s | %-26u |\n", "Video processing units", d->device.num_video_processing_units);
  48. if (d->mon_outputs.num)
  49. printf(" | %-26s | %-26u |\n", "Video monitoring outputs", d->mon_outputs.num);
  50. printf_line(len);
  51. printf("\n");
  52. }
  53. void print_device_video_inputs(struct videohub_data *d) {
  54. unsigned int i, r, len = 68;
  55. if (!d->inputs.num)
  56. return;
  57. printf("Video inputs\n");
  58. printf_line(len);
  59. printf(" | ## | %-24s | n | %-24s | s |\n", "Video input name", "Routed to output");
  60. printf_line(len);
  61. for(i = 0; i < d->inputs.num; i++) {
  62. unsigned int num_outputs = 0, routed_to = 0;
  63. for(r = 0; r < d->outputs.num; r++) {
  64. if (d->outputs.port[r].routed_to == i) {
  65. num_outputs++;
  66. if (num_outputs == 1)
  67. routed_to = r; // The first output
  68. }
  69. }
  70. printf(" | %2d | %-24s | %d | ", i + 1, d->inputs.port[i].name,
  71. num_outputs);
  72. if (num_outputs == 0) {
  73. printf("%-24s | %c |\n", "-", format_status(d->inputs.port[i].status));
  74. } else {
  75. printf("%-24s | %c |\n", d->outputs.port[routed_to].name, format_status(d->inputs.port[i].status));
  76. bool first_skipped = false;
  77. for(r = 0; r < d->outputs.num; r++) {
  78. if (d->outputs.port[r].routed_to == i) {
  79. if (!first_skipped) {
  80. first_skipped = true;
  81. continue;
  82. }
  83. printf(" | %2s | %-24s | %s | %-24s | %c |\n",
  84. " ", " ", " ", d->outputs.port[r].name, ' ');
  85. }
  86. }
  87. }
  88. }
  89. printf_line(len);
  90. printf("\n");
  91. }
  92. static char port_lock_symbol(enum port_lock p) {
  93. switch(p) {
  94. case PORT_UNLOCKED : return ' ';
  95. case PORT_LOCKED : return 'O';
  96. case PORT_LOCKED_OTHER: return 'L';
  97. }
  98. return '?';
  99. }
  100. void print_device_video_outputs(struct videohub_data *d) {
  101. unsigned int i, len = 68;
  102. if (!d->outputs.num)
  103. return;
  104. printf("Video outputs\n");
  105. printf_line(len);
  106. printf(" | ## | x | %-24s | %-24s | s |\n", "Video output name", "Connected video input");
  107. printf_line(len);
  108. for(i = 0; i < d->outputs.num; i++) {
  109. printf(" | %2d | %c | %-24s | %-24s | %c |\n",
  110. i + 1,
  111. port_lock_symbol(d->outputs.port[i].lock),
  112. d->outputs.port[i].name,
  113. d->inputs.port[d->outputs.port[i].routed_to].name,
  114. format_status(d->outputs.port[i].status)
  115. );
  116. }
  117. printf_line(len);
  118. printf("\n");
  119. }
  120. void print_device_monitoring_outputs(struct videohub_data *d) {
  121. unsigned int i, len = 64;
  122. if (!d->mon_outputs.num)
  123. return;
  124. printf("Monitoring outputs\n");
  125. printf_line(len);
  126. printf(" | ## | x | %-24s | %-24s |\n", "Monitoring output name", "Connected video input");
  127. printf_line(len);
  128. for(i = 0; i < d->mon_outputs.num; i++) {
  129. printf(" | %2d | %c | %-24s | %-24s |\n",
  130. i + 1,
  131. port_lock_symbol(d->mon_outputs.port[i].lock),
  132. d->mon_outputs.port[i].name,
  133. d->inputs.port[d->mon_outputs.port[i].routed_to].name
  134. );
  135. }
  136. printf_line(len);
  137. printf("\n");
  138. }
  139. static char *dir2opt(enum serial_dir dir) {
  140. switch (dir) {
  141. case DIR_CONTROL: return "in";
  142. case DIR_SLAVE : return "out";
  143. case DIR_AUTO : return "auto";
  144. }
  145. return "auto";
  146. }
  147. void print_device_serial_ports(struct videohub_data *d) {
  148. unsigned char port_seen[MAX_PORTS];
  149. unsigned int i, len = 63;
  150. if (!d->serial.num)
  151. return;
  152. memset(port_seen, 0, sizeof(port_seen));
  153. printf("Serial ports\n");
  154. printf_line(len);
  155. printf(" | ## | x | Dir | %-18s | %-18s | s |\n", "Serial port", "Connected serial");
  156. printf_line(len);
  157. for(i = 0; i < d->serial.num; i++) {
  158. printf(" | %2d | %c | %4s | %-18s | %-18s | %c |\n",
  159. i + 1,
  160. port_lock_symbol(d->serial.port[i].lock),
  161. dir2opt(d->serial.port[i].direction),
  162. d->serial.port[i].name,
  163. (d->serial.port[i].routed_to_set && !port_seen[d->serial.port[i].routed_to]
  164. ? d->serial.port[d->serial.port[i].routed_to].name
  165. : ""),
  166. format_status(d->serial.port[i].status)
  167. );
  168. port_seen[d->serial.port[i].routed_to]++;
  169. }
  170. printf_line(len);
  171. printf("\n");
  172. }
  173. static void __print_opt(struct videohub_data *d, enum vcmd vcmd) {
  174. unsigned int i, last = 0;
  175. struct videohub_commands *v = &videohub_commands[vcmd];
  176. struct port_set *s_port = !v->ports1 ? NULL : (void *)d + v->ports1;
  177. const char *p = v->opt_prefix;
  178. for(i = 0; i < s_port->num; i++) {
  179. switch (v->type) {
  180. case PARSE_LABEL:
  181. printf(" --%s-name %2d \"%s\" \\\n", p, i + 1, s_port->port[i].name);
  182. break;
  183. case PARSE_ROUTE:
  184. if (v->cmd == CMD_SERIAL_PORT_ROUTING && !s_port->port[i].routed_to_set)
  185. continue;
  186. printf(" --%s-input %2d %2d \\\n", p, i + 1, s_port->port[i].routed_to + 1);
  187. break;
  188. case PARSE_LOCK:
  189. last = i + 1 < s_port->num;
  190. if (s_port->port[i].lock != PORT_UNLOCKED) {
  191. printf(" --%s-unlock %2d --%s-lock %2d%s\n",
  192. p, i + 1, p, i + 1, last ? " \\" : "");
  193. } else {
  194. printf(" --%s-unlock %2d%s\n", p, i + 1, last ? " \\" : "");
  195. }
  196. break;
  197. case PARSE_DIR:
  198. printf(" --%s-dir %2d %s \\\n", p, i + 1, dir2opt(s_port->port[i].direction));
  199. break;
  200. default: break;
  201. }
  202. }
  203. }
  204. void print_device_backup(struct videohub_data *d) {
  205. unsigned int i;
  206. printf("videohubctrl \\\n");
  207. for (i = 0; i < NUM_COMMANDS; i++) {
  208. if (videohub_commands[i].type == PARSE_LABEL)
  209. __print_opt(d, videohub_commands[i].cmd);
  210. }
  211. for (i = 0; i < NUM_COMMANDS; i++) {
  212. if (videohub_commands[i].type == PARSE_ROUTE)
  213. __print_opt(d, videohub_commands[i].cmd);
  214. }
  215. for (i = 0; i < NUM_COMMANDS; i++) {
  216. if (videohub_commands[i].type == PARSE_DIR)
  217. __print_opt(d, videohub_commands[i].cmd);
  218. }
  219. for (i = 0; i < NUM_COMMANDS; i++) {
  220. if (videohub_commands[i].type == PARSE_LOCK)
  221. __print_opt(d, videohub_commands[i].cmd);
  222. }
  223. printf("\n");
  224. }