propagator/dhcp.c
Sergey Bolshakov 48d26b2d15 - cleaned up
2007-03-03 00:48:18 +03:00

616 lines
14 KiB
C

/*
* Guillaume Cottenceau (gc@mandrakesoft.com)
*
* Copyright 2000 MandrakeSoft
*
* This software may be freely redistributed under the terms of the GNU
* public license.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
/*
* Portions from Erik Troan (ewt@redhat.com)
*
* Copyright 1996 Red Hat Software
*
*/
/*
* Portions from GRUB -- GRand Unified Bootloader
* Copyright (C) 2000 Free Software Foundation, Inc.
*/
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
// #include <sys/socket.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <arpa/inet.h>
#include <net/route.h>
#include <errno.h>
#include <net/ethernet.h>
#include <netinet/ip.h>
#include <netinet/udp.h>
#include <sys/time.h>
#include <time.h>
#include <fcntl.h>
#include <sys/poll.h>
#include "stage1.h"
#include "log.h"
#include "network.h"
#include "frontend.h"
#include "dhcp.h"
typedef int bp_int32;
typedef short bp_int16;
#define BOOTP_OPTION_NETMASK 1
#define BOOTP_OPTION_GATEWAY 3
#define BOOTP_OPTION_DNS 6
#define BOOTP_OPTION_HOSTNAME 12
#define BOOTP_OPTION_DOMAIN 15
#define BOOTP_OPTION_BROADCAST 28
#define DHCP_OPTION_REQADDR 50
#define DHCP_OPTION_LEASE 51
#define DHCP_OPTION_TYPE 53
#define DHCP_OPTION_SERVER 54
#define DHCP_OPTION_OPTIONREQ 55
#define DHCP_OPTION_MAXSIZE 57
#define BOOTP_CLIENT_PORT 68
#define BOOTP_SERVER_PORT 67
#define BOOTP_OPCODE_REQUEST 1
#define BOOTP_OPCODE_REPLY 2
#define DHCP_TYPE_DISCOVER 1
#define DHCP_TYPE_OFFER 2
#define DHCP_TYPE_REQUEST 3
#define DHCP_TYPE_ACK 5
#define DHCP_TYPE_RELEASE 7
#define BOOTP_VENDOR_LENGTH 64
#define DHCP_VENDOR_LENGTH 340
struct bootp_request {
char opcode;
char hw;
char hwlength;
char hopcount;
bp_int32 id;
bp_int16 secs;
bp_int16 flags;
bp_int32 ciaddr, yiaddr, server_ip, bootp_gw_ip;
char hwaddr[16];
char servername[64];
char bootfile[128];
unsigned char vendor[DHCP_VENDOR_LENGTH];
} ;
static const char vendor_cookie[] = { 99, 130, 83, 99, 255 };
static unsigned int verify_checksum(void * buf2, int length2)
{
unsigned int csum = 0;
unsigned short * sp;
for (sp = (unsigned short *) buf2; length2 > 0; (length2 -= 2), sp++)
csum += *sp;
while (csum >> 16)
csum = (csum & 0xffff) + (csum >> 16);
return (csum == 0xffff);
}
static int initial_setup_interface(char * device, int s) {
struct sockaddr_in * addrp;
struct ifreq req;
struct rtentry route;
int true = 1;
addrp = (struct sockaddr_in *) &req.ifr_addr;
strcpy(req.ifr_name, device);
addrp->sin_family = AF_INET;
addrp->sin_port = 0;
memset(&addrp->sin_addr, 0, sizeof(addrp->sin_addr));
req.ifr_flags = 0; /* take it down */
if (ioctl(s, SIOCSIFFLAGS, &req)) {
log_perror("SIOCSIFFLAGS (downing)");
return -1;
}
addrp->sin_family = AF_INET;
addrp->sin_addr.s_addr = htonl(0);
if (ioctl(s, SIOCSIFADDR, &req)) {
log_perror("SIOCSIFADDR");
return -1;
}
req.ifr_flags = IFF_UP | IFF_BROADCAST | IFF_RUNNING;
if (ioctl(s, SIOCSIFFLAGS, &req)) {
log_perror("SIOCSIFFLAGS (upping)");
return -1;
}
memset(&route, 0, sizeof(route));
memcpy(&route.rt_gateway, addrp, sizeof(*addrp));
addrp->sin_family = AF_INET;
addrp->sin_port = 0;
addrp->sin_addr.s_addr = INADDR_ANY;
memcpy(&route.rt_dst, addrp, sizeof(*addrp));
memcpy(&route.rt_genmask, addrp, sizeof(*addrp));
route.rt_dev = device;
route.rt_flags = RTF_UP;
route.rt_metric = 0;
if (ioctl(s, SIOCADDRT, &route)) {
if (errno != EEXIST) {
close(s);
log_perror("SIOCADDRT");
return -1;
}
}
if (setsockopt(s, SOL_SOCKET, SO_BROADCAST, &true, sizeof(true))) {
close(s);
log_perror("setsockopt");
return -1;
}
/* I need to sleep a bit in order for kernel to finish init of the
network device; this would allow to not send further multiple
dhcp requests when only one is needed. */
wait_message("Bringing up networking...");
sleep(2);
remove_wait_message();
return 0;
}
void set_missing_ip_info(struct interface_info * intf)
{
bp_int32 ipNum = *((bp_int32 *) &intf->ip);
bp_int32 nmNum;
if (intf->netmask.s_addr == 0)
inet_aton(guess_netmask(inet_ntoa(intf->ip)), &intf->netmask);
nmNum = *((bp_int32 *) &intf->netmask);
if (intf->broadcast.s_addr == 0)
*((bp_int32 *) &intf->broadcast) = (ipNum & nmNum) | ~(nmNum);
if (intf->network.s_addr == 0)
*((bp_int32 *) &intf->network) = ipNum & nmNum;
}
static void parse_reply(struct bootp_request * breq, struct interface_info * intf)
{
unsigned char * chptr;
unsigned char option, length;
memcpy(&intf->ip, &breq->yiaddr, 4);
chptr = breq->vendor;
chptr += 4;
while (*chptr != 0xFF && (void *) chptr < (void *) breq->vendor + DHCP_VENDOR_LENGTH) {
char tmp_str[500];
option = *chptr++;
if (!option)
continue;
length = *chptr++;
switch (option) {
case BOOTP_OPTION_DNS:
memcpy(&dns_server, chptr, sizeof(dns_server));
log_message("got dns %s", inet_ntoa(dns_server));
if (length >= sizeof(dns_server)*2) {
memcpy(&dns_server2, chptr+sizeof(dns_server), sizeof(dns_server2));
log_message("got dns2 %s", inet_ntoa(dns_server2));
}
break;
case BOOTP_OPTION_NETMASK:
memcpy(&intf->netmask, chptr, sizeof(intf->netmask));
log_message("got netmask %s", inet_ntoa(intf->netmask));
break;
case BOOTP_OPTION_DOMAIN:
memcpy(tmp_str, chptr, length);
tmp_str[length] = '\0';
domain = strdup(tmp_str);
log_message("got domain %s", domain);
break;
case BOOTP_OPTION_BROADCAST:
memcpy(&intf->broadcast, chptr, sizeof(intf->broadcast));
log_message("got broadcast %s", inet_ntoa(intf->broadcast));
break;
case BOOTP_OPTION_GATEWAY:
memcpy(&gateway, chptr, sizeof(gateway));
log_message("got gateway %s", inet_ntoa(gateway));
break;
case BOOTP_OPTION_HOSTNAME:
memcpy(tmp_str, chptr, length);
tmp_str[length] = '\0';
hostname = strdup(tmp_str);
log_message("got hostname %s", hostname);
break;
}
chptr += length;
}
set_missing_ip_info(intf);
}
static void init_vendor_codes(struct bootp_request * breq) {
memcpy(breq->vendor, vendor_cookie, sizeof(vendor_cookie));
}
static char gen_hwaddr[16];
static int prepare_request(struct bootp_request * breq, int sock, char * device)
{
struct ifreq req;
memset(breq, 0, sizeof(*breq));
breq->opcode = BOOTP_OPCODE_REQUEST;
strcpy(req.ifr_name, device);
if (ioctl(sock, SIOCGIFHWADDR, &req)) {
log_perror("SIOCSIFHWADDR");
return -1;
}
breq->hw = 1; /* ethernet */
breq->hwlength = IFHWADDRLEN;
memcpy(breq->hwaddr, req.ifr_hwaddr.sa_data, IFHWADDRLEN);
memcpy(gen_hwaddr, req.ifr_hwaddr.sa_data, IFHWADDRLEN);
breq->hopcount = 0;
init_vendor_codes(breq);
return 0;
}
static int get_vendor_code(struct bootp_request * bresp, unsigned char option, void * data)
{
unsigned char * chptr;
unsigned int length, theOption;
chptr = bresp->vendor + 4;
while (*chptr != 0xFF && *chptr != option) {
theOption = *chptr++;
if (!theOption)
continue;
length = *chptr++;
chptr += length;
}
if (*chptr++ == 0xff)
return 1;
length = *chptr++;
memcpy(data, chptr, length);
return 0;
}
static int currticks(void)
{
struct timeval tv;
long csecs;
int ticks_per_csec, ticks_per_usec;
/* Note: 18.2 ticks/sec. */
gettimeofday (&tv, 0);
csecs = tv.tv_sec / 10;
ticks_per_csec = csecs * 182;
ticks_per_usec = (((tv.tv_sec - csecs * 10) * 1000000 + tv.tv_usec) * 182 / 10000000);
return ticks_per_csec + ticks_per_usec;
}
#define BACKOFF_LIMIT 7
#define TICKS_PER_SEC 18
#define MAX_ARP_RETRIES 4
static void rfc951_sleep(int exp)
{
static long seed = 0;
long q;
unsigned long tmo;
if (exp > BACKOFF_LIMIT)
exp = BACKOFF_LIMIT;
if (!seed)
/* Initialize linear congruential generator. */
seed = (currticks () + *(long *) &gen_hwaddr + ((short *) gen_hwaddr)[2]);
/* Simplified version of the LCG given in Bruce Scheier's
"Applied Cryptography". */
q = seed / 53668;
if ((seed = 40014 * (seed - 53668 * q) - 12211 * q) < 0)
seed += 2147483563l;
/* Compute mask. */
for (tmo = 63; tmo <= 60 * TICKS_PER_SEC && --exp > 0; tmo = 2 * tmo + 1)
;
/* Sleep. */
log_message("<sleep>");
for (tmo = (tmo & seed) + currticks (); currticks () < tmo;);
}
static int handle_transaction(int s, struct bootp_request * breq, struct bootp_request * bresp,
struct sockaddr_in * server_addr, int dhcp_type)
{
struct pollfd polls;
int i, j;
int retry = 1;
int sin;
char eth_packet[ETH_FRAME_LEN];
struct iphdr * ip_hdr;
struct udphdr * udp_hdr;
unsigned char type;
unsigned long starttime;
int timeout = 1;
breq->id = starttime = currticks();
breq->secs = 0;
sin = socket(AF_PACKET, SOCK_DGRAM, ntohs(ETH_P_IP));
if (sin < 0) {
log_perror("af_packet socket");
return -1;
}
while (retry <= MAX_ARP_RETRIES) {
i = sizeof(*breq);
if (sendto(s, breq, i, 0, (struct sockaddr *) server_addr, sizeof(*server_addr)) != i) {
close(s);
log_perror("sendto");
return -1;
}
polls.fd = sin;
polls.events = POLLIN;
while (poll(&polls, 1, timeout*1000) == 1) {
if ((j = recv(sin, eth_packet, sizeof(eth_packet), 0)) == -1) {
log_perror("recv");
continue;
}
/* We need to do some basic sanity checking of the header */
if (j < (sizeof(*ip_hdr) + sizeof(*udp_hdr)))
continue;
ip_hdr = (void *) eth_packet;
if (!verify_checksum(ip_hdr, sizeof(*ip_hdr)))
continue;
if (ntohs(ip_hdr->tot_len) > j)
continue;
j = ntohs(ip_hdr->tot_len);
if (ip_hdr->protocol != IPPROTO_UDP)
continue;
udp_hdr = (void *) (eth_packet + sizeof(*ip_hdr));
if (ntohs(udp_hdr->source) != BOOTP_SERVER_PORT)
continue;
if (ntohs(udp_hdr->dest) != BOOTP_CLIENT_PORT)
continue;
/* Go on with this packet; it looks sane */
/* Originally copied sizeof (*bresp) - this is a security
problem due to a potential underflow of the source
buffer. Also, it trusted that the packet was properly
0xFF terminated, which is not true in the case of the
DHCP server on Cisco 800 series ISDN router. */
memset (bresp, 0xFF, sizeof (*bresp));
memcpy (bresp, (char *) udp_hdr + sizeof (*udp_hdr), j - sizeof (*ip_hdr) - sizeof (*udp_hdr));
/* sanity checks */
if (bresp->id != breq->id)
continue;
if (bresp->opcode != BOOTP_OPCODE_REPLY)
continue;
if (bresp->hwlength != breq->hwlength)
continue;
if (memcmp(bresp->hwaddr, breq->hwaddr, bresp->hwlength))
continue;
if (get_vendor_code(bresp, DHCP_OPTION_TYPE, &type) || type != dhcp_type)
continue;
if (memcmp(bresp->vendor, vendor_cookie, 4))
continue;
return 0;
}
rfc951_sleep(retry);
breq->secs = htons ((currticks () - starttime) / 20);
retry++;
timeout *= 2;
if (timeout > 5)
timeout = 5;
}
return -1;
}
static void add_vendor_code(struct bootp_request * breq, unsigned char option, unsigned char length, void * data)
{
unsigned char * chptr;
int theOption, theLength;
chptr = breq->vendor;
chptr += 4;
while (*chptr != 0xFF && *chptr != option) {
theOption = *chptr++;
if (!theOption) continue;
theLength = *chptr++;
chptr += theLength;
}
*chptr++ = option;
*chptr++ = length;
memcpy(chptr, data, length);
chptr[length] = 0xff;
}
enum return_type perform_dhcp(struct interface_info * intf)
{
int s, i;
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
struct sockaddr_in broadcast_addr;
struct bootp_request breq, bresp;
unsigned char messageType;
unsigned int lease;
short aShort;
int num_options;
char requested_options[50];
if (strncmp(intf->device, "eth", 3)) {
stg1_error_message("DHCP available only for Ethernet networking.");
return RETURN_ERROR;
}
s = socket(AF_INET, SOCK_DGRAM, 0);
if (s < 0) {
log_perror("socket");
return RETURN_ERROR;
}
if (initial_setup_interface(intf->device, s) != 0) {
close(s);
return RETURN_ERROR;
}
if (prepare_request(&breq, s, intf->device) != 0) {
close(s);
return RETURN_ERROR;
}
messageType = DHCP_TYPE_DISCOVER;
add_vendor_code(&breq, DHCP_OPTION_TYPE, 1, &messageType);
memset(&client_addr.sin_addr, 0, sizeof(&client_addr.sin_addr));
client_addr.sin_family = AF_INET;
client_addr.sin_port = htons(BOOTP_CLIENT_PORT); /* bootp client */
if (bind(s, (struct sockaddr *) &client_addr, sizeof(client_addr))) {
log_perror("bind");
return RETURN_ERROR;
}
broadcast_addr.sin_family = AF_INET;
broadcast_addr.sin_port = htons(BOOTP_SERVER_PORT); /* bootp server */
memset(&broadcast_addr.sin_addr, 0xff, sizeof(broadcast_addr.sin_addr)); /* broadcast */
log_message("DHCP: sending DISCOVER");
wait_message("Sending DHCP request...");
i = handle_transaction(s, &breq, &bresp, &broadcast_addr, DHCP_TYPE_OFFER);
remove_wait_message();
if (i != 0) {
stg1_error_message("No DHCP reply received.");
close(s);
return RETURN_ERROR;
}
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(BOOTP_SERVER_PORT); /* bootp server */
if (get_vendor_code(&bresp, DHCP_OPTION_SERVER, &server_addr.sin_addr)) {
close(s);
log_message("DHCPOFFER didn't include server address");
return RETURN_ERROR;
}
init_vendor_codes(&breq);
messageType = DHCP_TYPE_REQUEST;
add_vendor_code(&breq, DHCP_OPTION_TYPE, 1, &messageType);
add_vendor_code(&breq, DHCP_OPTION_SERVER, 4, &server_addr.sin_addr);
add_vendor_code(&breq, DHCP_OPTION_REQADDR, 4, &bresp.yiaddr);
aShort = ntohs(sizeof(struct bootp_request));
add_vendor_code(&breq, DHCP_OPTION_MAXSIZE, 2, &aShort);
num_options = 0;
requested_options[num_options++] = BOOTP_OPTION_NETMASK;
requested_options[num_options++] = BOOTP_OPTION_GATEWAY;
requested_options[num_options++] = BOOTP_OPTION_DNS;
requested_options[num_options++] = BOOTP_OPTION_DOMAIN;
requested_options[num_options++] = BOOTP_OPTION_BROADCAST;
add_vendor_code(&breq, DHCP_OPTION_OPTIONREQ, num_options, requested_options);
/* request a lease of 1 hour */
i = htonl(60 * 60);
add_vendor_code(&breq, DHCP_OPTION_LEASE, 4, &i);
log_message("DHCP: sending REQUEST");
i = handle_transaction(s, &breq, &bresp, &broadcast_addr, DHCP_TYPE_ACK);
if (i != 0) {
close(s);
return RETURN_ERROR;
}
if (get_vendor_code(&bresp, DHCP_OPTION_LEASE, &lease)) {
log_message("failed to get lease time\n");
return RETURN_ERROR;
}
lease = ntohl(lease);
close(s);
intf->netmask.s_addr = 0;
intf->broadcast.s_addr = 0;
intf->network.s_addr = 0;
parse_reply(&bresp, intf);
return RETURN_OK;
}