summaryrefslogtreecommitdiff
path: root/usr/src/lib/libdhcpagent/common/dhcpagent_util.c
diff options
context:
space:
mode:
authorstevel@tonic-gate <none@none>2005-06-14 00:00:00 -0700
committerstevel@tonic-gate <none@none>2005-06-14 00:00:00 -0700
commit7c478bd95313f5f23a4c958a745db2134aa03244 (patch)
treec871e58545497667cbb4b0a4f2daf204743e1fe7 /usr/src/lib/libdhcpagent/common/dhcpagent_util.c
downloadillumos-gate-7c478bd95313f5f23a4c958a745db2134aa03244.tar.gz
OpenSolaris Launch
Diffstat (limited to 'usr/src/lib/libdhcpagent/common/dhcpagent_util.c')
-rw-r--r--usr/src/lib/libdhcpagent/common/dhcpagent_util.c264
1 files changed, 264 insertions, 0 deletions
diff --git a/usr/src/lib/libdhcpagent/common/dhcpagent_util.c b/usr/src/lib/libdhcpagent/common/dhcpagent_util.c
new file mode 100644
index 0000000000..18189a14cf
--- /dev/null
+++ b/usr/src/lib/libdhcpagent/common/dhcpagent_util.c
@@ -0,0 +1,264 @@
+/*
+ * CDDL HEADER START
+ *
+ * The contents of this file are subject to the terms of the
+ * Common Development and Distribution License, Version 1.0 only
+ * (the "License"). You may not use this file except in compliance
+ * with the License.
+ *
+ * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
+ * or http://www.opensolaris.org/os/licensing.
+ * See the License for the specific language governing permissions
+ * and limitations under the License.
+ *
+ * When distributing Covered Code, include this CDDL HEADER in each
+ * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
+ * If applicable, add the following below this CDDL HEADER, with the
+ * fields enclosed by brackets "[]" replaced with your own identifying
+ * information: Portions Copyright [yyyy] [name of copyright owner]
+ *
+ * CDDL HEADER END
+ */
+/*
+ * Copyright 2005 Sun Microsystems, Inc. All rights reserved.
+ * Use is subject to license terms.
+ */
+
+#pragma ident "%Z%%M% %I% %E% SMI"
+
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <sys/time.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include "dhcpagent_ipc.h"
+#include "dhcpagent_util.h"
+
+/*
+ * Strings returned by dhcp_status_hdr_string() and
+ * dhcp_status_reply_to_string(). The first define is the header line, and
+ * the second defines line printed underneath.
+ * The spacing of fields must match.
+ */
+#define DHCP_STATUS_HDR "Interface State Sent Recv Declined Flags\n"
+#define DHCP_STATUS_STR "%-10s %-12s %5d %5d %9d "
+
+static const char *time_to_string(time_t abs_time);
+
+/*
+ * dhcp_state_to_string(): given a state, provides the state's name
+ *
+ * input: DHCPSTATE: the state to get the name of
+ * output: const char *: the state's name
+ */
+
+const char *
+dhcp_state_to_string(DHCPSTATE state)
+{
+ const char *states[] = {
+ "INIT",
+ "SELECTING",
+ "REQUESTING",
+ "BOUND",
+ "RENEWING",
+ "REBINDING",
+ "INFORMATION",
+ "INIT_REBOOT",
+ "ADOPTING",
+ "INFORM_SENT"
+ };
+
+ if (state < 0 || state >= DHCP_NSTATES)
+ return ("<unknown>");
+
+ return (states[state]);
+}
+
+/*
+ * dhcp_string_to_request(): maps a string into a request code
+ *
+ * input: const char *: the string to map
+ * output: dhcp_ipc_type_t: the request code, or -1 if unknown
+ */
+
+dhcp_ipc_type_t
+dhcp_string_to_request(const char *request)
+{
+ static struct {
+ const char *string;
+ dhcp_ipc_type_t type;
+ } types[] = {
+ { "drop", DHCP_DROP },
+ { "extend", DHCP_EXTEND },
+ { "inform", DHCP_INFORM },
+ { "ping", DHCP_PING },
+ { "release", DHCP_RELEASE },
+ { "start", DHCP_START },
+ { "status", DHCP_STATUS }
+ };
+
+ unsigned int i;
+
+ for (i = 0; i < (sizeof (types) / sizeof (*types)); i++)
+ if (strcmp(types[i].string, request) == 0)
+ return (types[i].type);
+
+ return (-1);
+}
+
+/*
+ * dhcp_start_agent(): starts the agent if not already running
+ *
+ * input: int: number of seconds to wait for agent to start (-1 is forever)
+ * output: int: 0 on success, -1 on failure
+ */
+
+int
+dhcp_start_agent(int timeout)
+{
+ int error;
+ time_t start_time = time(NULL);
+ dhcp_ipc_request_t *request;
+ dhcp_ipc_reply_t *reply;
+
+ /*
+ * just send a dummy request to the agent to find out if it's
+ * up. we do this instead of directly connecting to it since
+ * we want to make sure we follow its IPC conventions
+ * (otherwise, it will log warnings to syslog).
+ */
+
+ request = dhcp_ipc_alloc_request(DHCP_PING, "", NULL, 0,
+ DHCP_TYPE_NONE);
+ if (request == NULL)
+ return (-1);
+
+ error = dhcp_ipc_make_request(request, &reply, 0);
+ if (error == 0) {
+ free(reply);
+ free(request);
+ return (0);
+ }
+ if (error != DHCP_IPC_E_CONNECT) {
+ free(request);
+ return (-1);
+ }
+
+ switch (fork()) {
+
+ case -1:
+ free(request);
+ return (-1);
+
+ case 0:
+ (void) execl(DHCP_AGENT_PATH, DHCP_AGENT_PATH, (char *)0);
+ _exit(EXIT_FAILURE);
+
+ default:
+ break;
+ }
+
+ while ((timeout != -1) && (time(NULL) - start_time < timeout)) {
+ error = dhcp_ipc_make_request(request, &reply, 0);
+ if (error == 0) {
+ free(reply);
+ free(request);
+ return (0);
+ } else if (error != DHCP_IPC_E_CONNECT)
+ break;
+ (void) sleep(1);
+ }
+
+ free(request);
+ return (-1);
+}
+
+/*
+ * dhcp_status_hdr_string(): Return a string suitable to use as the header
+ * when printing DHCP_STATUS reply.
+ * output: const char *: newline terminated printable string
+ */
+const char *
+dhcp_status_hdr_string(void)
+{
+ return (DHCP_STATUS_HDR);
+}
+
+/*
+ * time_to_string(): Utility routine for printing time
+ *
+ * input: time_t *: time_t to stringify
+ * output: const char *: printable time
+ */
+static const char *
+time_to_string(time_t abs_time)
+{
+ static char time_buf[24];
+ time_t tm = abs_time;
+
+ if (tm == DHCP_PERM)
+ return ("Never");
+
+ if (strftime(time_buf, sizeof (time_buf), "%m/%d/%Y %R",
+ localtime(&tm)) == 0)
+ return ("<unknown>");
+
+ return (time_buf);
+}
+
+/*
+ * dhcp_status_reply_to_string(): Return DHCP IPC reply of type DHCP_STATUS
+ * as a printable string
+ *
+ * input: dhcp_reply_t *: contains the status structure to print
+ * output: const char *: newline terminated printable string
+ */
+const char *
+dhcp_status_reply_to_string(dhcp_ipc_reply_t *reply)
+{
+ static char str[1024];
+ size_t reply_size;
+ dhcp_status_t *status;
+
+ status = dhcp_ipc_get_data(reply, &reply_size, NULL);
+ if (reply_size < DHCP_STATUS_VER1_SIZE)
+ return ("<Internal error: status msg size>\n");
+
+ (void) snprintf(str, sizeof (str), DHCP_STATUS_STR,
+ status->if_name, dhcp_state_to_string(status->if_state),
+ status->if_sent, status->if_recv, status->if_bad_offers);
+
+ if (status->if_dflags & DHCP_IF_PRIMARY)
+ (void) strlcat(str, "[PRIMARY] ", sizeof (str));
+
+ if (status->if_dflags & DHCP_IF_BOOTP)
+ (void) strlcat(str, "[BOOTP] ", sizeof (str));
+
+ if (status->if_dflags & DHCP_IF_FAILED)
+ (void) strlcat(str, "[FAILED] ", sizeof (str));
+
+ if (status->if_dflags & DHCP_IF_BUSY)
+ (void) strlcat(str, "[BUSY] ", sizeof (str));
+
+ (void) strlcat(str, "\n", sizeof (str));
+
+ switch (status->if_state) {
+ case BOUND:
+ case RENEWING:
+ case REBINDING:
+ break;
+ default:
+ return (str);
+ }
+
+ (void) strlcat(str, "(Began, Expires, Renew) = (", sizeof (str));
+ (void) strlcat(str, time_to_string(status->if_began), sizeof (str));
+ (void) strlcat(str, ", ", sizeof (str));
+ (void) strlcat(str, time_to_string(status->if_lease), sizeof (str));
+ (void) strlcat(str, ", ", sizeof (str));
+ (void) strlcat(str, time_to_string(status->if_t1), sizeof (str));
+ (void) strlcat(str, ")\n", sizeof (str));
+ return (str);
+}