/* * main.c * main() routine, printer functions * * Copyright (c) 2011 William Pitcock . * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include #include #include "pkg.h" /* we are compatible with 0.26 of pkg-config */ #define PKGCONFIG_VERSION_EQUIV "0.26" static int want_version = 0; static int want_cflags = 0; static int want_libs = 0; static int want_modversion = 0; static int want_static = 0; static int want_requires = 0; static int want_variables = 0; static int want_digraph = 0; static int maximum_traverse_depth = 2; static char *required_pkgconfig_version = NULL; static char *required_module_version = NULL; static char *want_variable = NULL; static void print_cflags(pkg_t *pkg, void *unused) { if (pkg->cflags != NULL) printf("%s ", pkg->cflags); } static void print_libs(pkg_t *pkg, void *unused) { if (pkg->libs != NULL) printf("%s ", pkg->libs); } static void print_modversion(pkg_t *pkg, void *unused) { if (pkg->version != NULL) printf("%s\n", pkg->version); } static void print_variable(pkg_t *pkg, void *unused) { char *variable; variable = tuple_find(pkg->vars, want_variable); if (variable != NULL) printf("%s ", variable); } static void print_variables(pkg_t *pkg, void *unused) { pkg_tuple_t *node; foreach_list_entry(pkg->vars, node) printf("%s\n", node->key); } static void print_requires(pkg_t *pkg, void *unused) { pkg_dependency_t *node; foreach_list_entry(pkg->requires, node) { printf("%s", node->package); if (node->version != NULL) printf(" %s %s", pkg_get_comparator(node), node->version); printf("\n"); } } static void print_digraph_node(pkg_t *pkg, void *unused) { pkg_dependency_t *node; printf("\"%s\" [fontname=Sans fontsize=8]\n", pkg->id); foreach_list_entry(pkg->requires, node) { printf("\"%s\" -- \"%s\" [fontname=Sans fontsize=8]\n", node->package, pkg->id); } } typedef struct pkg_queue_ { struct pkg_queue_ *prev, *next; const char *package; } pkg_queue_t; static pkg_queue_t * pkg_queue_push(pkg_queue_t *parent, const char *package) { pkg_queue_t *pkgq = calloc(sizeof(pkg_queue_t), 1); pkgq->package = package; pkgq->prev = parent; if (pkgq->prev != NULL) pkgq->prev->next = pkgq; return pkgq; } int pkg_queue_walk(pkg_queue_t *head) { int wanted_something = 0; pkg_queue_t *pkgq; pkg_t world = (pkg_t){ .id = "world", .realname = "virtual" }; /* if maximum_traverse_depth is one, then we will not traverse deeper * than our virtual package. */ if (!maximum_traverse_depth) maximum_traverse_depth = -1; else if (maximum_traverse_depth > 0) maximum_traverse_depth++; foreach_list_entry(head, pkgq) { pkg_dependency_t *pkgdep; pkgdep = parse_deplist(&world, pkgq->package); world.requires = pkg_dependency_append(world.requires, pkgdep); } /* we should verify that the graph is complete before attempting to compute cflags etc. */ pkg_verify_graph(&world, maximum_traverse_depth); if (want_digraph) { printf("graph deptree {\n"); printf("edge [color=blue len=7.5 fontname=Sans fontsize=8]\n"); printf("node [fontname=Sans fontsize=8]\n"); pkg_traverse(&world, print_digraph_node, NULL, maximum_traverse_depth); printf("}\n"); return EXIT_SUCCESS; } if (want_modversion) { wanted_something = 0; want_cflags = 0; want_libs = 0; pkg_traverse(&world, print_modversion, NULL, 2); } if (want_variables) { wanted_something = 0; want_cflags = 0; want_libs = 0; pkg_traverse(&world, print_variables, NULL, 2); } if (want_requires) { pkg_dependency_t *iter; wanted_something = 0; want_cflags = 0; want_libs = 0; foreach_list_entry(world.requires, iter) { pkg_t *pkg; pkg = pkg_verify_dependency(iter); print_requires(pkg, NULL); } } if (want_cflags) { wanted_something++; pkg_traverse(&world, print_cflags, NULL, maximum_traverse_depth); } if (want_libs) { wanted_something++; pkg_traverse(&world, print_libs, NULL, maximum_traverse_depth); } if (want_variable) { wanted_something++; pkg_traverse(&world, print_variable, NULL, 2); } if (wanted_something) printf("\n"); return EXIT_SUCCESS; } static void version(void) { printf("%s %s\n", PACKAGE_NAME, PACKAGE_VERSION); } int main(int argc, const char *argv[]) { int ret; poptContext opt_context; pkg_queue_t *pkgq = NULL; pkg_queue_t *pkgq_head = NULL; struct poptOption options[] = { { "version", 0, POPT_ARG_NONE, &want_version, 0, "output pkgconf version" }, { "atleast-version", 0, POPT_ARG_STRING, &required_module_version, 0, "require specified version of a package" }, { "atleast-pkgconfig-version", 0, POPT_ARG_STRING, &required_pkgconfig_version, 0, "require compatibility level with specified version of pkg-config" }, { "libs", 0, POPT_ARG_NONE, &want_libs, 0, "output all linker flags" }, { "cflags", 0, POPT_ARG_NONE, &want_cflags, 0, "output all compiler flags" }, { "modversion", 0, POPT_ARG_NONE, &want_modversion, 0, "output package version" }, { "variable", 0, POPT_ARG_STRING, &want_variable, 0, "get the value of the specified variable" }, { "exists", 0, POPT_ARG_NONE, NULL, 0, "return 0 if all packages present" }, { "print-errors", 0, POPT_ARG_NONE, NULL, 0, "dummy option for pkg-config compatibility" }, { "short-errors", 0, POPT_ARG_NONE, NULL, 0, "dummy option for pkg-config compatibility" }, { "maximum-traverse-depth", 0, POPT_ARG_INT, &maximum_traverse_depth, 0, "limits maximum traversal depth of the computed dependency graph" }, { "static", 0, POPT_ARG_NONE, &want_static, 0, "be more aggressive when walking the dependency graph (for intermediary static linking deps)" }, { "print-requires", 0, POPT_ARG_NONE, &want_requires, 0, "print required dependencies of the computed dependency graph" }, { "print-variables", 0, POPT_ARG_NONE, &want_variables, 0, "print variables provided by a package" }, { "digraph", 0, POPT_ARG_NONE, &want_digraph, 0, "build a graphviz digraph (dot) graph of the computed dependency graph" }, POPT_AUTOHELP { NULL, 0, 0, NULL, 0 } }; opt_context = poptGetContext(NULL, argc, argv, options, 0); ret = poptGetNextOpt(opt_context); if (ret != -1) { fprintf(stderr, "%s: %s\n", poptBadOption(opt_context, POPT_BADOPTION_NOALIAS), poptStrerror(ret)); return EXIT_FAILURE; } if (want_version) { version(); return EXIT_SUCCESS; } if (want_static) maximum_traverse_depth++; if (required_pkgconfig_version != NULL) { if (pkg_compare_version(PKGCONFIG_VERSION_EQUIV, required_pkgconfig_version) >= 0) return EXIT_SUCCESS; return EXIT_FAILURE; } if (required_module_version != NULL) { pkg_t *pkg; const char *package; package = poptGetArg(opt_context); if (package == NULL) return EXIT_SUCCESS; pkg = pkg_find(package); if (pkg == NULL) return EXIT_FAILURE; if (pkg_compare_version(pkg->version, required_module_version) >= 0) return EXIT_SUCCESS; return EXIT_FAILURE; } while (1) { const char *package = poptGetArg(opt_context); if (package == NULL) break; pkgq = pkg_queue_push(pkgq, package); if (pkgq_head == NULL) pkgq_head = pkgq; } poptFreeContext(opt_context); if (pkgq_head == NULL) { fprintf(stderr, "Please specify at least one package name on the command line.\n"); return EXIT_FAILURE; } return pkg_queue_walk(pkgq_head); }