graphdump.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /*
  2. * Filter graphs to bad ASCII-art
  3. * Copyright (c) 2012 Nicolas George
  4. *
  5. * This file is part of FFmpeg.
  6. *
  7. * FFmpeg is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * FFmpeg is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with FFmpeg; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. */
  21. #include <string.h>
  22. #include "libavutil/channel_layout.h"
  23. #include "libavutil/bprint.h"
  24. #include "libavutil/pixdesc.h"
  25. #include "avfilter.h"
  26. #include "avfiltergraph.h"
  27. static int print_link_prop(AVBPrint *buf, AVFilterLink *link)
  28. {
  29. char *format;
  30. char layout[64];
  31. AVBPrint dummy_buffer = { 0 };
  32. if (!buf)
  33. buf = &dummy_buffer;
  34. switch (link->type) {
  35. case AVMEDIA_TYPE_VIDEO:
  36. format = av_x_if_null(av_get_pix_fmt_name(link->format), "?");
  37. av_bprintf(buf, "[%dx%d %d:%d %s]", link->w, link->h,
  38. link->sample_aspect_ratio.num,
  39. link->sample_aspect_ratio.den,
  40. format);
  41. break;
  42. case AVMEDIA_TYPE_AUDIO:
  43. av_get_channel_layout_string(layout, sizeof(layout),
  44. link->channels, link->channel_layout);
  45. format = av_x_if_null(av_get_sample_fmt_name(link->format), "?");
  46. av_bprintf(buf, "[%dHz %s:%s]",
  47. (int)link->sample_rate, format, layout);
  48. break;
  49. default:
  50. av_bprintf(buf, "?");
  51. break;
  52. }
  53. return buf->len;
  54. }
  55. static void avfilter_graph_dump_to_buf(AVBPrint *buf, AVFilterGraph *graph)
  56. {
  57. unsigned i, j, x, e;
  58. for (i = 0; i < graph->nb_filters; i++) {
  59. AVFilterContext *filter = graph->filters[i];
  60. unsigned max_src_name = 0, max_dst_name = 0;
  61. unsigned max_in_name = 0, max_out_name = 0;
  62. unsigned max_in_fmt = 0, max_out_fmt = 0;
  63. unsigned width, height, in_indent;
  64. unsigned lname = strlen(filter->name);
  65. unsigned ltype = strlen(filter->filter->name);
  66. for (j = 0; j < filter->nb_inputs; j++) {
  67. AVFilterLink *l = filter->inputs[j];
  68. unsigned ln = strlen(l->src->name) + 1 + strlen(l->srcpad->name);
  69. max_src_name = FFMAX(max_src_name, ln);
  70. max_in_name = FFMAX(max_in_name, strlen(l->dstpad->name));
  71. max_in_fmt = FFMAX(max_in_fmt, print_link_prop(NULL, l));
  72. }
  73. for (j = 0; j < filter->nb_outputs; j++) {
  74. AVFilterLink *l = filter->outputs[j];
  75. unsigned ln = strlen(l->dst->name) + 1 + strlen(l->dstpad->name);
  76. max_dst_name = FFMAX(max_dst_name, ln);
  77. max_out_name = FFMAX(max_out_name, strlen(l->srcpad->name));
  78. max_out_fmt = FFMAX(max_out_fmt, print_link_prop(NULL, l));
  79. }
  80. in_indent = max_src_name + max_in_name + max_in_fmt;
  81. in_indent += in_indent ? 4 : 0;
  82. width = FFMAX(lname + 2, ltype + 4);
  83. height = FFMAX3(2, filter->nb_inputs, filter->nb_outputs);
  84. av_bprint_chars(buf, ' ', in_indent);
  85. av_bprintf(buf, "+");
  86. av_bprint_chars(buf, '-', width);
  87. av_bprintf(buf, "+\n");
  88. for (j = 0; j < height; j++) {
  89. unsigned in_no = j - (height - filter->nb_inputs ) / 2;
  90. unsigned out_no = j - (height - filter->nb_outputs) / 2;
  91. /* Input link */
  92. if (in_no < filter->nb_inputs) {
  93. AVFilterLink *l = filter->inputs[in_no];
  94. e = buf->len + max_src_name + 2;
  95. av_bprintf(buf, "%s:%s", l->src->name, l->srcpad->name);
  96. av_bprint_chars(buf, '-', e - buf->len);
  97. e = buf->len + max_in_fmt + 2 +
  98. max_in_name - strlen(l->dstpad->name);
  99. print_link_prop(buf, l);
  100. av_bprint_chars(buf, '-', e - buf->len);
  101. av_bprintf(buf, "%s", l->dstpad->name);
  102. } else {
  103. av_bprint_chars(buf, ' ', in_indent);
  104. }
  105. /* Filter */
  106. av_bprintf(buf, "|");
  107. if (j == (height - 2) / 2) {
  108. x = (width - lname) / 2;
  109. av_bprintf(buf, "%*s%-*s", x, "", width - x, filter->name);
  110. } else if (j == (height - 2) / 2 + 1) {
  111. x = (width - ltype - 2) / 2;
  112. av_bprintf(buf, "%*s(%s)%*s", x, "", filter->filter->name,
  113. width - ltype - 2 - x, "");
  114. } else {
  115. av_bprint_chars(buf, ' ', width);
  116. }
  117. av_bprintf(buf, "|");
  118. /* Output link */
  119. if (out_no < filter->nb_outputs) {
  120. AVFilterLink *l = filter->outputs[out_no];
  121. unsigned ln = strlen(l->dst->name) + 1 +
  122. strlen(l->dstpad->name);
  123. e = buf->len + max_out_name + 2;
  124. av_bprintf(buf, "%s", l->srcpad->name);
  125. av_bprint_chars(buf, '-', e - buf->len);
  126. e = buf->len + max_out_fmt + 2 +
  127. max_dst_name - ln;
  128. print_link_prop(buf, l);
  129. av_bprint_chars(buf, '-', e - buf->len);
  130. av_bprintf(buf, "%s:%s", l->dst->name, l->dstpad->name);
  131. }
  132. av_bprintf(buf, "\n");
  133. }
  134. av_bprint_chars(buf, ' ', in_indent);
  135. av_bprintf(buf, "+");
  136. av_bprint_chars(buf, '-', width);
  137. av_bprintf(buf, "+\n");
  138. av_bprintf(buf, "\n");
  139. }
  140. }
  141. char *avfilter_graph_dump(AVFilterGraph *graph, const char *options)
  142. {
  143. AVBPrint buf;
  144. char *dump;
  145. av_bprint_init(&buf, 0, 0);
  146. avfilter_graph_dump_to_buf(&buf, graph);
  147. av_bprint_init(&buf, buf.len + 1, buf.len + 1);
  148. avfilter_graph_dump_to_buf(&buf, graph);
  149. av_bprint_finalize(&buf, &dump);
  150. return dump;
  151. }