blob: f5bbbfc532cbc8136699c3ea3ad18cf609276b1e [file] [log] [blame]
/*
* dhcpcd - DHCP client daemon
* Copyright (c) 2006-2013 Roy Marples <roy@marples.name>
* All rights reserved
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``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 OR CONTRIBUTORS 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 <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
#include "common.h"
#include "dhcpcd.h"
#include "if-options.h"
#include "platform.h"
static const char *mproc =
#if defined(__alpha__)
"system type"
#elif defined(__arm__)
"Hardware"
#elif defined(__avr32__)
"cpu family"
#elif defined(__bfin__)
"BOARD Name"
#elif defined(__cris__)
"cpu model"
#elif defined(__frv__)
"System"
#elif defined(__i386__) || defined(__x86_64__)
"vendor_id"
#elif defined(__ia64__)
"vendor"
#elif defined(__hppa__)
"model"
#elif defined(__m68k__)
"MMU"
#elif defined(__mips__)
"system type"
#elif defined(__powerpc__) || defined(__powerpc64__)
"machine"
#elif defined(__s390__) || defined(__s390x__)
"Manufacturer"
#elif defined(__sh__)
"machine"
#elif defined(sparc) || defined(__sparc__)
"cpu"
#elif defined(__vax__)
"cpu"
#else
NULL
#endif
;
#ifdef INET6
static char **restore;
static ssize_t nrestore;
#endif
char *
hardware_platform(void)
{
FILE *fp;
char *buf, *p;
if (mproc == NULL) {
errno = EINVAL;
return NULL;
}
fp = fopen("/proc/cpuinfo", "r");
if (fp == NULL)
return NULL;
p = NULL;
while ((buf = get_line(fp))) {
if (strncmp(buf, mproc, strlen(mproc)) == 0) {
p = strchr(buf, ':');
if (p != NULL && ++p != NULL) {
while (*p == ' ')
p++;
break;
}
}
}
fclose(fp);
if (p == NULL)
errno = ESRCH;
return p;
}
#ifdef INET6
static int
check_proc_int(const char *path)
{
FILE *fp;
char *buf;
fp = fopen(path, "r");
if (fp == NULL)
return -1;
buf = get_line(fp);
fclose(fp);
if (buf == NULL)
return -1;
return atoi(buf);
}
static ssize_t
write_path(const char *path, const char *val)
{
FILE *fp;
ssize_t r;
fp = fopen(path, "w");
if (fp == NULL)
return -1;
r = fprintf(fp, "%s\n", val);
fclose(fp);
return r;
}
static const char *prefix = "/proc/sys/net/ipv6/conf";
static void
restore_kernel_ra(void)
{
char path[256];
#ifndef DEBUG_MEMORY
if (options & DHCPCD_FORKED)
return;
#endif
for (nrestore--; nrestore >= 0; nrestore--) {
#ifdef DEBUG_MEMORY
if (!(options & DHCPCD_FORKED)) {
#endif
syslog(LOG_INFO, "%s: restoring Kernel IPv6 RA support",
restore[nrestore]);
snprintf(path, sizeof(path), "%s/%s/accept_ra",
prefix, restore[nrestore]);
if (write_path(path, "1") == -1 && errno != ENOENT)
syslog(LOG_ERR, "write_path: %s: %m", path);
#ifdef DEBUG_MEMORY
}
free(restore[nrestore]);
#endif
}
#ifdef DEBUG_MEMORY
free(restore);
#endif
}
int
check_ipv6(const char *ifname)
{
static int ipv6_checked = 0;
int r, ex, i;
char path[256], *p, **nrest;
if (ifname == NULL) {
if (ipv6_checked)
return 1;
ipv6_checked = 1;
ifname = "all";
ex = 1;
} else
ex = 0;
snprintf(path, sizeof(path), "%s/%s/accept_ra", prefix, ifname);
r = check_proc_int(path);
if (r == -1)
/* The sysctl probably doesn't exist, but this isn't an
* error as such so just log it and continue */
syslog(errno == ENOENT ? LOG_DEBUG : LOG_WARNING,
"%s: %m", path);
else if (r == 0)
options |= DHCPCD_IPV6RA_OWN;
else if (options & DHCPCD_IPV6RA_OWN) {
syslog(LOG_INFO, "%s: disabling Kernel IPv6 RA support",
ifname);
if (write_path(path, "0") == -1) {
syslog(LOG_ERR, "write_path: %s: %m", path);
return 0;
}
for (i = 0; i < nrestore; i++)
if (strcmp(restore[i], ifname) == 0)
break;
if (i == nrestore) {
p = strdup(ifname);
if (p == NULL) {
syslog(LOG_ERR, "%s: %m", __func__);
goto forward;
}
nrest = realloc(restore,
(nrestore + 1) * sizeof(char *));
if (nrest == NULL) {
syslog(LOG_ERR, "%s: %m", __func__);
goto forward;
}
restore = nrest;
restore[nrestore++] = p;
}
if (ex)
atexit(restore_kernel_ra);
}
forward:
if (r != 2) {
snprintf(path, sizeof(path), "%s/%s/forwarding",
prefix, ifname);
r = check_proc_int(path);
if (r == -1) {
/* The sysctl probably doesn't exist, but this isn't an
* error as such so just log it and continue */
syslog(errno == ENOENT ? LOG_DEBUG : LOG_WARNING,
"%s: %m", path);
} else if (r != 0) {
syslog(LOG_WARNING,
"%s: configured as a router, not a host", ifname);
return 0;
}
}
return 1;
}
int
ipv6_dadtransmits(const char *ifname)
{
char path[256];
int r;
if (ifname == NULL)
ifname = "default";
snprintf(path, sizeof(path), "%s/%s/dad_transmits", prefix, ifname);
r = check_proc_int(path);
return r < 0 ? 0 : r;
}
#endif