diff options
author | Theo de Raadt <deraadt@cvs.openbsd.org> | 2006-08-29 03:55:10 +0000 |
---|---|---|
committer | Theo de Raadt <deraadt@cvs.openbsd.org> | 2006-08-29 03:55:10 +0000 |
commit | a652f988dfa9a2e2c1c9cfca94dff4dbb275a3cf (patch) | |
tree | bd1c12745606a8c47c34dbfcd3431253eea3e539 /sbin/dhclient/dispatch.c | |
parent | 80010d966b61c52ba3916dc4df1db9977f0e3b27 (diff) |
some net if devices do not media negotiate (for link) until brought up.
(for some drivers, this is a bug. for others, it is part of how they work)
therefore before doing the 10-second link test, we must bring the if up.
ok krw, tested marco ckuethe
Diffstat (limited to 'sbin/dhclient/dispatch.c')
-rw-r--r-- | sbin/dhclient/dispatch.c | 59 |
1 files changed, 57 insertions, 2 deletions
diff --git a/sbin/dhclient/dispatch.c b/sbin/dhclient/dispatch.c index 5b4f350eb45..5ce380f61cb 100644 --- a/sbin/dhclient/dispatch.c +++ b/sbin/dhclient/dispatch.c @@ -1,4 +1,4 @@ -/* $OpenBSD: dispatch.c,v 1.32 2005/05/24 03:11:12 todd Exp $ */ +/* $OpenBSD: dispatch.c,v 1.33 2006/08/29 03:55:09 deraadt Exp $ */ /* * Copyright 2004 Henning Brauer <henning@openbsd.org> @@ -220,7 +220,6 @@ another: } while (1); } - void got_one(struct protocol *l) { @@ -269,6 +268,62 @@ got_one(struct protocol *l) } int +interface_link_forceup(char *ifname) +{ + struct ifreq ifr; + int sock, ret = 0; + + if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) == -1) + error("Can't create socket"); + + memset(&ifr, 0, sizeof(ifr)); + strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name)); + if (ioctl(sock, SIOCGIFFLAGS, (caddr_t)&ifr) == -1) { + close(sock); + return (-1); + } + + if ((ifr.ifr_flags & (IFF_UP|IFF_RUNNING)) != (IFF_UP|IFF_RUNNING)) { + ifr.ifr_flags |= IFF_UP; + if (ioctl(sock, SIOCSIFFLAGS, (caddr_t)&ifr) == -1) { + close(sock); + return (-1); + } + close(sock); + return (0); + } + close(sock); + return (1); +} + +void +interface_link_forcedown(char *ifname) +{ + struct ifreq ifr; + int sock; + + if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) == -1) + error("Can't create socket"); + + memset(&ifr, 0, sizeof(ifr)); + strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name)); + if (ioctl(sock, SIOCGIFFLAGS, (caddr_t)&ifr) == -1) { + close(sock); + return; + } + + if ((ifr.ifr_flags & IFF_UP) == IFF_UP) { + ifr.ifr_flags &= ~IFF_UP; + if (ioctl(sock, SIOCSIFFLAGS, (caddr_t)&ifr) == -1) { + close(sock); + return; + } + } + + close(sock); +} + +int interface_status(struct interface_info *ifinfo) { char *ifname = ifinfo->name; |