2007-04-05 16:48:42 +05:30
|
|
|
/*
|
|
|
|
rc-depend
|
|
|
|
rc service dependency and ordering
|
|
|
|
Copyright 2006-2007 Gentoo Foundation
|
|
|
|
Released under the GPLv2
|
|
|
|
*/
|
|
|
|
|
2007-10-04 19:08:47 +05:30
|
|
|
#define APPLET "rc-depend"
|
|
|
|
|
2007-04-05 16:48:42 +05:30
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
2007-10-04 19:08:47 +05:30
|
|
|
#include <getopt.h>
|
|
|
|
#include <errno.h>
|
2007-04-05 16:48:42 +05:30
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2007-07-31 21:35:56 +05:30
|
|
|
#include "builtins.h"
|
2007-04-05 16:48:42 +05:30
|
|
|
#include "einfo.h"
|
|
|
|
#include "rc.h"
|
|
|
|
#include "rc-misc.h"
|
|
|
|
#include "strlist.h"
|
|
|
|
|
2007-09-29 22:12:08 +05:30
|
|
|
rc_depinfo_t *_rc_deptree_load (void) {
|
|
|
|
if (rc_deptree_update_needed ()) {
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
ebegin ("Caching service dependencies");
|
|
|
|
retval = rc_deptree_update ();
|
2007-10-05 15:46:14 +05:30
|
|
|
eend (retval ? 0 : -1, "Failed to update the dependency tree");
|
2007-09-29 22:12:08 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
return (rc_deptree_load ());
|
|
|
|
}
|
|
|
|
|
2007-10-04 19:08:47 +05:30
|
|
|
static char *applet = NULL;
|
|
|
|
|
|
|
|
#include "_usage.h"
|
|
|
|
#define getoptstring "t:suT" getoptstring_COMMON
|
|
|
|
static struct option longopts[] = {
|
2007-10-12 05:31:33 +05:30
|
|
|
{ "type", 1, NULL, 't'},
|
2007-10-04 19:08:47 +05:30
|
|
|
{ "notrace", 0, NULL, 'T'},
|
|
|
|
{ "strict", 0, NULL, 's'},
|
|
|
|
{ "update", 0, NULL, 'u'},
|
|
|
|
longopts_COMMON
|
|
|
|
};
|
|
|
|
static const char * const longopts_help[] = {
|
|
|
|
"Type(s) of dependency to list",
|
|
|
|
"Don't trace service dependencies",
|
|
|
|
"Only use what is in the runlevels",
|
|
|
|
"Force an update of the dependency tree",
|
|
|
|
longopts_help_COMMON
|
|
|
|
};
|
|
|
|
#include "_usage.c"
|
|
|
|
|
2007-07-31 21:35:56 +05:30
|
|
|
int rc_depend (int argc, char **argv)
|
2007-04-05 16:48:42 +05:30
|
|
|
{
|
2007-04-11 18:14:47 +05:30
|
|
|
char **types = NULL;
|
|
|
|
char **services = NULL;
|
|
|
|
char **depends = NULL;
|
2007-10-04 19:08:47 +05:30
|
|
|
char **list;
|
2007-04-11 18:14:47 +05:30
|
|
|
rc_depinfo_t *deptree = NULL;
|
|
|
|
char *service;
|
|
|
|
int options = RC_DEP_TRACE;
|
|
|
|
bool first = true;
|
|
|
|
int i;
|
|
|
|
bool update = false;
|
2007-10-12 05:31:33 +05:30
|
|
|
char *runlevel = xstrdup( getenv ("RC_SOFTLEVEL"));
|
2007-10-04 19:08:47 +05:30
|
|
|
int opt;
|
|
|
|
char *token;
|
|
|
|
|
|
|
|
applet = argv[0];
|
|
|
|
|
|
|
|
while ((opt = getopt_long (argc, argv, getoptstring,
|
|
|
|
longopts, (int *) 0)) != -1)
|
|
|
|
{
|
|
|
|
switch (opt) {
|
|
|
|
case 's':
|
|
|
|
options |= RC_DEP_STRICT;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
while ((token = strsep (&optarg, ",")))
|
|
|
|
rc_strlist_addu (&types, token);
|
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
update = true;
|
|
|
|
break;
|
|
|
|
case 'T':
|
|
|
|
options &= RC_DEP_TRACE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case_RC_COMMON_GETOPT
|
2007-04-11 18:14:47 +05:30
|
|
|
}
|
2007-10-04 19:08:47 +05:30
|
|
|
}
|
2007-04-11 18:14:47 +05:30
|
|
|
|
2007-10-04 19:08:47 +05:30
|
|
|
if (update) {
|
|
|
|
bool u = false;
|
|
|
|
ebegin ("Caching service dependencies");
|
|
|
|
u = rc_deptree_update ();
|
|
|
|
eend (u ? 0 : -1, "%s: %s", applet, strerror (errno));
|
|
|
|
if (! u)
|
|
|
|
eerrorx ("Failed to update the dependency tree");
|
|
|
|
}
|
2007-04-11 18:14:47 +05:30
|
|
|
|
2007-10-04 19:08:47 +05:30
|
|
|
if (! (deptree = _rc_deptree_load ()))
|
|
|
|
eerrorx ("failed to load deptree");
|
|
|
|
|
2007-10-12 05:31:33 +05:30
|
|
|
if (! runlevel)
|
|
|
|
runlevel = rc_runlevel_get ();
|
|
|
|
|
2007-10-04 19:08:47 +05:30
|
|
|
while (optind < argc) {
|
|
|
|
list = NULL;
|
|
|
|
rc_strlist_add (&list, argv[optind]);
|
|
|
|
errno = 0;
|
2007-10-15 20:10:53 +05:30
|
|
|
depends = rc_deptree_depends (deptree, NULL, (const char **) list,
|
|
|
|
runlevel, 0);
|
2007-10-04 19:08:47 +05:30
|
|
|
if (! depends && errno == ENOENT)
|
|
|
|
eerror ("no dependency info for service `%s'", argv[optind]);
|
|
|
|
else
|
|
|
|
rc_strlist_add (&services, argv[optind]);
|
|
|
|
|
|
|
|
rc_strlist_free (depends);
|
|
|
|
rc_strlist_free (list);
|
|
|
|
optind++;
|
2007-04-11 18:14:47 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (! services) {
|
|
|
|
rc_strlist_free (types);
|
2007-09-29 22:12:08 +05:30
|
|
|
rc_deptree_free (deptree);
|
2007-10-12 05:31:33 +05:30
|
|
|
free (runlevel);
|
2007-04-11 18:14:47 +05:30
|
|
|
if (update)
|
|
|
|
return (EXIT_SUCCESS);
|
|
|
|
eerrorx ("no services specified");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we don't have any types, then supply some defaults */
|
|
|
|
if (! types) {
|
2007-09-18 17:06:55 +05:30
|
|
|
rc_strlist_add (&types, "ineed");
|
|
|
|
rc_strlist_add (&types, "iuse");
|
2007-04-11 18:14:47 +05:30
|
|
|
}
|
|
|
|
|
2007-10-15 20:10:53 +05:30
|
|
|
depends = rc_deptree_depends (deptree, (const char **) types,
|
|
|
|
(const char **) services, runlevel, options);
|
2007-04-11 18:14:47 +05:30
|
|
|
|
|
|
|
if (depends) {
|
|
|
|
STRLIST_FOREACH (depends, service, i) {
|
|
|
|
if (first)
|
|
|
|
first = false;
|
|
|
|
else
|
|
|
|
printf (" ");
|
|
|
|
|
|
|
|
if (service)
|
|
|
|
printf ("%s", service);
|
|
|
|
|
|
|
|
}
|
|
|
|
printf ("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
rc_strlist_free (types);
|
|
|
|
rc_strlist_free (services);
|
|
|
|
rc_strlist_free (depends);
|
2007-09-29 22:12:08 +05:30
|
|
|
rc_deptree_free (deptree);
|
2007-10-12 05:31:33 +05:30
|
|
|
free (runlevel);
|
2007-04-11 18:14:47 +05:30
|
|
|
return (EXIT_SUCCESS);
|
2007-04-05 16:48:42 +05:30
|
|
|
}
|