1
0
mirror of git://projects.qi-hardware.com/wernermisc.git synced 2025-01-05 18:40:15 +02:00
wernermisc/qpkg/qpkg.c
Werner Almesberger 9d208dd855 qpkg: added parsing and recording of Provides data
- pkg.h (enum flags): new flags QPKG_PROVIDED to indicate that the
  package is only a provision, not an installable package
- fixup.h, fixup.c (complete_provisions): search for provisions that do
  not have a real package associated with them, create a package for
  them, and mark it as QPKG_PROVIDED
- qpkg.c (do_fixups): invoke complete_provisions
- pkg.h (struct pkg): add list of (virtual or real) packages a given
  package provides
- pkg.c (new_pkg): initialize pkg->provides
- pkg.c (free_pkg): deallocate pkg->provides
- gobble.c (gobble_buf): parse the list of packages a package provides
- TODO: updated status of Provides support
2010-11-22 19:30:09 -03:00

186 lines
3.5 KiB
C

/*
* qpkg.c - Quick package database query
*
* Written 2010 by Werner Almesberger
* Copyright 2010 Werner Almesberger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*/
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include "jrb.h"
#include "id.h"
#include "prereq.h"
#include "gobble.h"
#include "fixup.h"
#include "pkg.h"
#include "qpkg.h"
struct tree *packages = NULL;
struct tree *versions = NULL;
int debug = 0;
static void list_all_packages(void)
{
const struct jrb *n;
const struct pkg *pkg;
for (n = jrb_first(packages->root); n != jrb_nil(packages->root);
n = jrb_next(n)) {
const struct id *id = n->key;
for (pkg = n->val; pkg; pkg = pkg->more) {
printf("%.*s", ID2PF(id));
if (!pkg)
printf(" (virtual)");
else {
if (pkg->version)
printf(" (%.*s)", ID2PF(pkg->version));
if (pkg->more)
printf(" +++");
}
printf("\n");
}
}
}
static void list_one_package(const char *name)
{
const struct jrb *n = find_id(packages, name, strlen(name));
const struct pkg *pkg;
if (!n) {
fprintf(stderr, "no such package \"%s\"\n", name);
exit(1);
}
for (pkg = n->val; pkg; pkg = pkg->more)
printf("%.*s\n", ID2PF(pkg->version));
}
static void find_prereq(const char *name, const char *version)
{
const struct jrb *n = find_id(packages, name, strlen(name));
struct pkg *pkg;
struct pkg **pkgs;
if (!n) {
fprintf(stderr, "no such package \"%s\"\n", name);
exit(1);
}
pkg = n->val;
if (!pkg) {
fprintf(stderr, "package %s is a ghost\n", name);
exit(1);
}
if (version) {
n = find_id(versions, version, strlen(version));
if (!n) {
fprintf(stderr, "no such version\"%s\"\n", version);
exit(1);
}
while (pkg && pkg->version != n->key)
pkg = pkg->more;
if (pkg) {
fprintf(stderr, "no %s version\"%s\" found\n",
name, version);
exit(1);
}
}
pkgs = prereq(pkg);
if (!pkgs) {
fprintf(stderr, "can't resolve %s%s%s\n",
name, version ? " version " : "", version ? version : "");
exit(1);
}
while (*pkgs) {
const char *file = (*pkgs)->filename;
const char *end;
assert(file);
end = strchr(file, '\n');
printf("%.*s\n", (int) (end-file), file);
pkgs++;
}
}
static void do_fixups(void)
{
sort_versions();
complete_provisions();
}
static void usage(const char *name)
{
fprintf(stderr,
"usage: %s options [pkg-list ...] list [pkg]\n"
" %s options [pkg-list ...] prereq pkg [version]\n\n"
" -d enable debugging output\n"
, name, name);
exit(1);
}
int main(int argc, char **argv)
{
int arg;
packages = make_tree(comp_id);
versions = make_tree(comp_id);
if (argc == 1)
usage(*argv);
for (arg = 1; arg != argc; arg++) {
if (!strcmp(argv[arg], "-d")) {
debug = 1;
continue;
}
if (*argv[arg] == '-')
usage(*argv);
if (!strcmp(argv[arg], "list")) {
do_fixups();
switch (argc-arg) {
case 1:
list_all_packages();
return 0;
case 2:
list_one_package(argv[arg+1]);
return 0;
default:
usage(*argv);
}
} else if (!strcmp(argv[arg], "prereq")) {
do_fixups();
switch (argc-arg) {
case 2:
find_prereq(argv[arg+1], NULL);
return 0;
case 3:
find_prereq(argv[arg+1], argv[arg+2]);
return 0;
default:
usage(*argv);
}
} else {
gobble(argv[arg]);
}
}
return 0;
}