commit - edd6dd9f2f1c93acebcaaa1421088f724f9e7132
commit + 0698bd5fd36695ee913ac16345bcaf7dc83d2dbf
blob - 4011e0f3cb3a14e6359f82f82497db29b47dcd89
blob + 90c757c613ac4b53c77d0c90ea82967be741cfe8
--- ctl.c
+++ ctl.c
#include <sys/stat.h>
#include <sys/socket.h>
#include <sys/syslimits.h>
+#include <net/if.h>
#include "externs.h"
#include "editing.h"
#include "ctl.h"
void edit_crontab(char *, char **, char *);
void install_crontab(char *, char **, char *);
void call_editor(char *, char **, char *);
+void start_dhcpd(char *, char **, char *);
int edit_file(char *, mode_t, char *, char **);
void ctl_symlink(char *, char *, char *);
int rule_writeline(char *, mode_t, char *);
char *ctl_dhcp_test[] = { DHCPD, "-nc", REQTEMP, NULL };
struct ctl ctl_dhcp[] = {
{ "enable", "enable DHCPd daemon",
- { DHCPD, "-c", REQTEMP, "-l", DHCPLEASES, NULL }, NULL, DB_X_ENABLE,
- T_EXEC },
+ { DHCPD, "-c", REQTEMP, "-l", DHCPLEASES, NULL }, start_dhcpd,
+ DB_X_ENABLE, T_HANDLER_FILL1 },
{ "disable", "disable DHCPd daemon",
{ PKILL, table, "dhcpd", NULL }, NULL, DB_X_DISABLE, T_EXEC },
{ "edit", "edit,test and stage DHCPd config",
printf("%% Unable to remove temporary file %s: %s\n",
file, strerror(errno));
}
+
+void
+start_dhcpd(char *name, char **args, char *z)
+{
+ struct if_nameindex *ifn_list, *ifnp;
+ char **p, **dhcpd_args = NULL;
+ size_t nargs = 0, niface = 0;
+ int ifs, i;
+
+ if (cli_rtable == 0) {
+ cmdargs(name, args);
+ return;
+ }
+
+ /*
+ * For rdomains other than zero dhcpd(8) expects a list of
+ * interfaces on its command line. If no interface arguments
+ * are given then dhcpd will move itself into rdomain zero
+ * so we really must specify a list here.
+ *
+ * All named interfaces must be part of the same rdomain. We
+ * provide the list of all interfaces in our current rdomain.
+ * dhcpd will listen on any with matching subnets in dhcpd.conf.
+ */
+ if ((ifn_list = if_nameindex()) == NULL) {
+ printf("%% %s: if_nameindex failed\n", __func__);
+ return;
+ }
+
+ p = args;
+ while (*p) {
+ nargs++;
+ p++;
+ }
+
+ if ((ifs = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
+ printf("%% %s socket: %s\n", __func__, strerror(errno));
+ goto done;
+ }
+
+ for (ifnp = ifn_list; ifnp->if_name != NULL; ifnp++) {
+ int flags, rdomain;
+
+ flags = get_ifflags(ifnp->if_name, ifs);
+ if ((flags & IFF_LOOPBACK) ||
+ (flags & IFF_POINTOPOINT) ||
+ (!(flags & IFF_BROADCAST)))
+ continue;
+
+ rdomain = get_rdomain(ifs, ifnp->if_name);
+ if (rdomain == cli_rtable)
+ niface++;
+ }
+
+ dhcpd_args = calloc(1 + nargs + niface + 1, sizeof(char *));
+ if (dhcpd_args == NULL) {
+ printf("%% calloc: %s\n", strerror(errno));
+ goto done;
+ }
+
+ dhcpd_args[0] = name;
+ for (i = 1; i < nargs + 1; i++)
+ dhcpd_args[i] = args[i - 1];
+ for (ifnp = ifn_list; ifnp->if_name != NULL; ifnp++) {
+ int flags, rdomain;
+
+ flags = get_ifflags(ifnp->if_name, ifs);
+ if ((flags & IFF_LOOPBACK) ||
+ (flags & IFF_POINTOPOINT) ||
+ (!(flags & IFF_BROADCAST)))
+ continue;
+
+ rdomain = get_rdomain(ifs, ifnp->if_name);
+ if (rdomain == cli_rtable)
+ dhcpd_args[i++] = ifnp->if_name;
+ }
+ dhcpd_args[i] = NULL;
+
+ cmdargs(name, dhcpd_args);
+done:
+ if_freenameindex(ifn_list);
+ free(dhcpd_args);
+}