tools/graph2dot.c
11ab237e
 /*
  * Copyright (c) 2008-2010 Stefano Sabatini
  *
  * This file is part of FFmpeg.
  *
  * FFmpeg is free software; you can redistribute it and/or
  * modify it under the terms of the GNU Lesser General Public
  * License as published by the Free Software Foundation; either
  * version 2.1 of the License, or (at your option) any later version.
  *
  * FFmpeg is distributed in the hope that it will be useful,
  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  * Lesser General Public License for more details.
  *
  * You should have received a copy of the GNU Lesser General Public
  * License along with FFmpeg; if not, write to the Free Software
  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  */
 
09d5e02a
 #include "config.h"
 #if HAVE_UNISTD_H
11ab237e
 #include <unistd.h>             /* getopt */
09d5e02a
 #endif
1d9c2dc8
 #include <stdio.h>
 #include <string.h>
11ab237e
 
1acd2f6b
 #include "libavutil/channel_layout.h"
1d9c2dc8
 #include "libavutil/mem.h"
11ab237e
 #include "libavutil/pixdesc.h"
38f0c078
 #include "libavfilter/avfilter.h"
11ab237e
 
09d5e02a
 #if !HAVE_GETOPT
 #include "compat/getopt.c"
 #endif
 
11ab237e
 static void usage(void)
 {
cd7b6dee
     printf("Convert a libavfilter graph to a dot file.\n");
11ab237e
     printf("Usage: graph2dot [OPTIONS]\n");
     printf("\n"
            "Options:\n"
            "-i INFILE         set INFILE as input file, stdin if omitted\n"
            "-o OUTFILE        set OUTFILE as output file, stdout if omitted\n"
            "-h                print this help\n");
 }
 
 struct line {
     char data[256];
     struct line *next;
 };
 
 static void print_digraph(FILE *outfile, AVFilterGraph *graph)
 {
     int i, j;
 
     fprintf(outfile, "digraph G {\n");
     fprintf(outfile, "node [shape=box]\n");
     fprintf(outfile, "rankdir=LR\n");
 
ecade984
     for (i = 0; i < graph->nb_filters; i++) {
11ab237e
         char filter_ctx_label[128];
         const AVFilterContext *filter_ctx = graph->filters[i];
 
f054dbee
         snprintf(filter_ctx_label, sizeof(filter_ctx_label), "%s\\n(%s)",
11ab237e
                  filter_ctx->name,
                  filter_ctx->filter->name);
 
59360cd4
         for (j = 0; j < filter_ctx->nb_outputs; j++) {
11ab237e
             AVFilterLink *link = filter_ctx->outputs[j];
             if (link) {
                 char dst_filter_ctx_label[128];
                 const AVFilterContext *dst_filter_ctx = link->dst;
 
4e81b5f5
                 snprintf(dst_filter_ctx_label, sizeof(dst_filter_ctx_label),
f054dbee
                          "%s\\n(%s)",
11ab237e
                          dst_filter_ctx->name,
                          dst_filter_ctx->filter->name);
 
f054dbee
                 fprintf(outfile, "\"%s\" -> \"%s\" [ label= \"inpad:%s -> outpad:%s\\n",
                         filter_ctx_label, dst_filter_ctx_label,
                         link->srcpad->name, link->dstpad->name);
ce7266c6
 
5d329e2e
                 if (link->type == AVMEDIA_TYPE_VIDEO) {
b7f1010c
                     const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(link->format);
4e81b5f5
                     fprintf(outfile,
ce7266c6
                             "fmt:%s w:%d h:%d tb:%d/%d",
a33ed6bc
                             desc->name,
ce7266c6
                             link->w, link->h,
                             link->time_base.num, link->time_base.den);
5d329e2e
                 } else if (link->type == AVMEDIA_TYPE_AUDIO) {
                     char buf[255];
4e81b5f5
                     av_get_channel_layout_string(buf, sizeof(buf), -1,
                                                  link->channel_layout);
                     fprintf(outfile,
039e9fe0
                             "fmt:%s sr:%d cl:%s tb:%d/%d",
5d329e2e
                             av_get_sample_fmt_name(link->format),
d93a448b
                             link->sample_rate, buf,
ce7266c6
                             link->time_base.num, link->time_base.den);
5d329e2e
                 }
36330781
                 fprintf(outfile, "\" ];\n");
11ab237e
             }
         }
     }
     fprintf(outfile, "}\n");
 }
 
 int main(int argc, char **argv)
 {
     const char *outfilename = NULL;
4e81b5f5
     const char *infilename  = NULL;
     FILE *outfile           = NULL;
     FILE *infile            = NULL;
     char *graph_string      = NULL;
11ab237e
     AVFilterGraph *graph = av_mallocz(sizeof(AVFilterGraph));
     char c;
 
     av_log_set_level(AV_LOG_DEBUG);
 
     while ((c = getopt(argc, argv, "hi:o:")) != -1) {
4e81b5f5
         switch (c) {
11ab237e
         case 'h':
             usage();
             return 0;
         case 'i':
             infilename = optarg;
             break;
         case 'o':
             outfilename = optarg;
             break;
         case '?':
             return 1;
         }
     }
 
     if (!infilename || !strcmp(infilename, "-"))
         infilename = "/dev/stdin";
     infile = fopen(infilename, "r");
     if (!infile) {
cd7b6dee
         fprintf(stderr, "Failed to open input file '%s': %s\n",
4e81b5f5
                 infilename, strerror(errno));
11ab237e
         return 1;
     }
 
     if (!outfilename || !strcmp(outfilename, "-"))
         outfilename = "/dev/stdout";
     outfile = fopen(outfilename, "w");
     if (!outfile) {
cd7b6dee
         fprintf(stderr, "Failed to open output file '%s': %s\n",
4e81b5f5
                 outfilename, strerror(errno));
11ab237e
         return 1;
     }
 
     /* read from infile and put it in a buffer */
     {
         unsigned int count = 0;
         struct line *line, *last_line, *first_line;
         char *p;
         last_line = first_line = av_malloc(sizeof(struct line));
 
         while (fgets(last_line->data, sizeof(last_line->data), infile)) {
             struct line *new_line = av_malloc(sizeof(struct line));
             count += strlen(last_line->data);
             last_line->next = new_line;
4e81b5f5
             last_line       = new_line;
11ab237e
         }
         last_line->next = NULL;
 
         graph_string = av_malloc(count + 1);
         p = graph_string;
         for (line = first_line; line->next; line = line->next) {
             unsigned int l = strlen(line->data);
             memcpy(p, line->data, l);
             p += l;
         }
         *p = '\0';
     }
 
     avfilter_register_all();
 
     if (avfilter_graph_parse(graph, graph_string, NULL, NULL, NULL) < 0) {
cd7b6dee
         fprintf(stderr, "Failed to parse the graph description\n");
11ab237e
         return 1;
     }
 
2a24df93
     if (avfilter_graph_config(graph, NULL) < 0)
11ab237e
         return 1;
 
     print_digraph(outfile, graph);
     fflush(outfile);
 
     return 0;
 }