📄 interface.c
字号:
/* interface.c - DHCP server and relay agent network interface library *//* Copyright 1984 - 2001 Wind River Systems, Inc. */#include "copyright_wrs.h"/*modification history--------------------01f,10oct01,rae fixed modhist line01e,06oct97,spm split interface name into device name and unit number01d,26aug97,spm moved routines from common_subr.c not used by DHCP client01c,06may97,spm changed memory access to align IP header on four byte boundary; removed unused timeout variable01b,18apr97,spm added conditional include DHCPS_DEBUG for displayed output01a,07apr97,spm created by modifying WIDE project DHCP implementation*//*DESCRIPTIONThis library contains the code used by the DHCP server and relay agents tomonitor one or more network interfaces for incoming DHCP messages. The monitorroutine is triggered on a semaphore given by an Ethernet input hook. Thecorresponding message, read from a pre-allocated ring buffer, identifies theoriginating network interface and the amount of received data. The data itselfis then copied from another ring buffer into a temporary interface-specificbuffer.INCLUDE_FILES: None*//* * WIDE Project DHCP Implementation * Copyright (c) 1995 Akihiro Tominaga * Copyright (c) 1995 WIDE Project * All rights reserved. * * Permission to use, copy, modify and distribute this software and its * documentation is hereby granted, provided only with the following * conditions are satisfied: * * 1. Both the copyright notice and this permission notice appear in * all copies of the software, derivative works or modified versions, * and any portions thereof, and that both notices appear in * supporting documentation. * 2. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by WIDE Project and * its contributors. * 3. Neither the name of WIDE Project nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE DEVELOPER ``AS IS'' AND WIDE * PROJECT DISCLAIMS ANY LIABILITY OF ANY KIND FOR ANY DAMAGES * WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE. ALSO, THERE * IS NO WARRANTY IMPLIED OR OTHERWISE, NOR IS SUPPORT PROVIDED. * * Feedback of the results generated from any improvements or * extensions made to this software would be much appreciated. * Any such feedback should be sent to: * * Akihiro Tominaga * WIDE Project * Keio University, Endo 5322, Kanagawa, Japan * (E-mail: dhcp-dist@wide.ad.jp) * * WIDE project has the rights to redistribute these changes. */#include "vxWorks.h"#include "ioLib.h" /* ioctl() declaration */#include "vxLib.h" /* checksum() declaration */#include "sockLib.h"#include "logLib.h"#include "rngLib.h"#include <stdio.h>#include <stdlib.h>#include <fcntl.h>#include <string.h>#include <unistd.h>#include <sys/ioctl.h>#include <sys/socket.h>#include <net/if.h>#include <netinet/in_systm.h>#include <netinet/in.h>#include "dhcp/dhcp.h"#include "dhcp/common.h"#include "dhcp/common_subr.h"/********************************************************************************* open_if - initialize per-interface data structure** This routine sets the IP address and subnet mask entries for each* interface monitored by the DHCP server or relay agent.** RETURNS: 0 if successful, or -1 on error.** ERRNO: N/A** NOMANUAL*/int open_if ( struct if_info *ifinfo /* pointer to interface descriptor */ ) { int n; struct ifreq ifreq; bzero ( (char *)&ifreq, sizeof(ifreq)); sprintf (ifreq.ifr_name, "%s%d", ifinfo->name, ifinfo->unit); /* * Initialize the interface information (subnet and IP address). */ n = socket (AF_INET, SOCK_DGRAM, 0); if (n < 0) return (-1); if (ioctl (n, SIOCGIFNETMASK, (int)&ifreq) < 0) {#ifdef DHCPS_DEBUG logMsg("Error: Can't retrieve netmask in open_if().\n", 0, 0, 0, 0, 0, 0);#endif close (n); return (-1); } ifinfo->subnetmask.s_addr = ( (struct sockaddr_in *)&ifreq.ifr_addr)->sin_addr.s_addr; if (ioctl(n, SIOCGIFADDR, (int)&ifreq) < 0) {#ifdef DHCPS_DEBUG logMsg("Error: can't retrieve address in open_if().\n", 0, 0, 0, 0, 0, 0);#endif close (n); return (-1); } ifinfo->ipaddr.s_addr = ( (struct sockaddr_in *)&ifreq.ifr_addr)->sin_addr.s_addr; close(n); /* * Store size of preallocated buffer. Offset used to align IP header on * 4-byte boundary for Sun BSP's. */ ifinfo->buf_size = DHCP_MSG_SIZE + DHCPS_OFF; return (0); }/********************************************************************************* read_interfaces - extract arriving messages from shared buffers** This function waits for the arrival of a DHCP message (extracted by* a BPF device) and copies the message into the private buffer* associated with the receiving interface.** RETURNS: Pointer to data structure of receiving interface, or NULL on error.** ERRNO: N/A** NOMANUAL*/struct if_info * read_interfaces ( struct if_info *iflist, /* pointer to interface descriptors */ int *n, /* buffer for actual length of message */ int maxSize /* length of (internal) buffer for message */ ) { struct if_info *ifp = NULL; /* descriptor of receiver's interface */ int status; /* message extraction results */ fd_set readFds; int fileNum = 0; int maxFileNum = 0; /* Wait for an arriving DHCP message. */ FD_ZERO (&readFds); ifp = iflist; while (ifp != NULL) { fileNum = ifp->bpfDev; FD_SET (fileNum, &readFds); if (fileNum > maxFileNum) maxFileNum = fileNum; ifp = ifp->next; } maxFileNum++; /* Adjust maximum for select() routine. */#ifdef DHCPS_DEBUG logMsg ("DHCP server: waiting for data.\n", 0, 0, 0, 0, 0, 0);#endif status = select (maxFileNum, &readFds, NULL, NULL, NULL); if (status <= 0) return (NULL);#ifdef DHCPS_DEBUG logMsg ("DHCP server: read_interfaces() received new message.\n", 0, 0, 0, 0, 0, 0);#endif /* Find the interface descriptor for the first available DHCP message. */ ifp = iflist; while (ifp != NULL) { fileNum = ifp->bpfDev; if (FD_ISSET (fileNum, &readFds)) break; ifp = ifp->next; } if (ifp == NULL) {#ifdef DHCPS_DEBUG logMsg ("Warning: invalid network device in read_interfaces()\n", 0, 0, 0, 0, 0, 0);#endif return (NULL); } status = read (fileNum, ifp->buf, maxSize); if (status <= 0) {#ifdef DHCPS_DEBUG logMsg ("Warning: error reading DHCP message.\n", 0, 0, 0, 0, 0, 0);#endif return (NULL); } *n = status; return (ifp); }/********************************************************************************* get_ipsum - retrieve the IP header checksum** This routine fetches the checksum for the given IP header.** RETURNS: Value of checksum in network byte order.** ERRNO: N/A** NOMANUAL*/u_short get_ipsum ( struct ip * pIph /* IP header */ ) { pIph->ip_sum = 0;#if BSD<44 return (checksum ( (u_short *)pIph, (pIph->ip_v_hl & 0xf) << 2));#else return (checksum ( (u_short *)pIph, pIph->ip_hl << 2));#endif }/********************************************************************************* check_ipsum - verify the IP header checksum** This routine retrieves the checksum for the given IP header and compares* it to the received checksum.** RETURNS: TRUE if checksums match, or FALSE otherwise.** ERRNO: N/A** NOMANUAL*/BOOL check_ipsum ( struct ip * pIph /* Received IP header */ ) { u_short ripcksum; /* received IP checksum */ ripcksum = pIph->ip_sum; return (ripcksum == get_ipsum (pIph)); }/********************************************************************************* get_udpsum - retrieve the UDP header checksum** This routine fetches the checksum for a UDP header.** RETURNS: Value of checksum in network byte order.** ERRNO: N/A** NOMANUAL*/u_short get_udpsum ( struct ip * pIph, /* IP header */ struct udphdr * pUdph /* UDP header */ ) { struct ps_udph UdpPh; /* UDP pseudo-header */ bzero ( (char *)&UdpPh, sizeof (UdpPh)); UdpPh.srcip.s_addr = pIph->ip_src.s_addr; UdpPh.dstip.s_addr = pIph->ip_dst.s_addr; UdpPh.zero = 0; UdpPh.prto = IPPROTO_UDP; UdpPh.ulen = pUdph->uh_ulen; pUdph->uh_sum = 0; return (udp_cksum (&UdpPh, (char *)pUdph, ntohs (UdpPh.ulen))); }/********************************************************************************* check_udpsum - verify the IP header checksum** This routine retrieves the checksum for a UDP header and compares it* to the received checksum.** RETURNS: TRUE if checksums match, or FALSE otherwise.** ERRNO: N/A** NOMANUAL*/int check_udpsum ( struct ip * pIph, /* Received IP header */ struct udphdr * pUdph /* Received UDP header */ ) { u_short rudpcksum; /* received UDP checksum */ if (pUdph->uh_sum == 0) return(TRUE); rudpcksum = pUdph->uh_sum; return (rudpcksum == get_udpsum (pIph, pUdph)); }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -