265 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			265 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|              LUFA Library
 | |
|      Copyright (C) Dean Camera, 2013.
 | |
| 
 | |
|   dean [at] fourwalledcubicle [dot] com
 | |
|            www.lufa-lib.org
 | |
| */
 | |
| 
 | |
| /*
 | |
|   Copyright 2013  Dean Camera (dean [at] fourwalledcubicle [dot] com)
 | |
| 
 | |
|   Permission to use, copy, modify, distribute, and sell this
 | |
|   software and its documentation for any purpose is hereby granted
 | |
|   without fee, provided that the above copyright notice appear in
 | |
|   all copies and that both that the copyright notice and this
 | |
|   permission notice and warranty disclaimer appear in supporting
 | |
|   documentation, and that the name of the author not be used in
 | |
|   advertising or publicity pertaining to distribution of the
 | |
|   software without specific, written prior permission.
 | |
| 
 | |
|   The author disclaims all warranties with regard to this
 | |
|   software, including all implied warranties of merchantability
 | |
|   and fitness.  In no event shall the author be liable for any
 | |
|   special, indirect or consequential damages or any damages
 | |
|   whatsoever resulting from loss of use, data or profits, whether
 | |
|   in an action of contract, negligence or other tortious action,
 | |
|   arising out of or in connection with the use or performance of
 | |
|   this software.
 | |
| */
 | |
| 
 | |
| /** \file
 | |
|  *
 | |
|  *  DHCP Server Application. When connected to the uIP stack, this will send IP configuration settings to a
 | |
|  *  DHCP client on the network.
 | |
|  */
 | |
| 
 | |
| #define  INCLUDE_FROM_DHCPSERVERAPP_C
 | |
| #include "DHCPServerApp.h"
 | |
| 
 | |
| #if defined(ENABLE_DHCP_SERVER) || defined(__DOXYGEN__)
 | |
| 
 | |
| struct uip_conn* BroadcastConnection;
 | |
| 
 | |
| uint8_t LeasedIPs[255 / 8];
 | |
| 
 | |
| /** Initialization function for the DHCP server. */
 | |
| void DHCPServerApp_Init(void)
 | |
| {
 | |
| 	/* Listen on port 67 for DHCP server connections from hosts */
 | |
| 	uip_listen(HTONS(DHCP_SERVER_PORT));
 | |
| 
 | |
| 	/* Create a new UDP connection to the DHCP server port for the DHCP solicitation */
 | |
| 	struct uip_udp_conn* BroadcastConnection = uip_udp_new(&uip_broadcast_addr, HTONS(DHCP_CLIENT_PORT));
 | |
| 
 | |
| 	/* If the connection was successfully created, bind it to the local DHCP client port */
 | |
| 	if (BroadcastConnection != NULL)
 | |
| 	  uip_udp_bind(BroadcastConnection, HTONS(DHCP_SERVER_PORT));
 | |
| 
 | |
| 	/* Set all IP addresses as unleased */
 | |
| 	memset(LeasedIPs, 0x00, sizeof(LeasedIPs));
 | |
| }
 | |
| 
 | |
| /** uIP stack application callback for the DHCP server. This function must be called each time the TCP/IP stack
 | |
|  *  needs a UDP packet to be processed.
 | |
|  */
 | |
| void DHCPServerApp_Callback(void)
 | |
| {
 | |
| 	DHCP_Header_t* const AppData     = (DHCP_Header_t*)uip_appdata;
 | |
| 	uint16_t             AppDataSize = 0;
 | |
| 
 | |
| 	/* Only process when new data arrives - don't retransmit lost packets */
 | |
| 	if (uip_newdata())
 | |
| 	{
 | |
| 		/* Get the DHCP message type (if present), otherwise early-abort */
 | |
| 		uint8_t DHCPMessageType;
 | |
| 		if (!(DHCPCommon_GetOption(AppData->Options, DHCP_OPTION_MSG_TYPE, &DHCPMessageType)))
 | |
| 			return;
 | |
| 
 | |
| 		uip_ipaddr_t        Netmask, GatewayIPAddress, PreferredClientIP;
 | |
| 		struct uip_eth_addr RemoteMACAddress;
 | |
| 		uint32_t            TransactionID;
 | |
| 
 | |
| 		/* Get configured network mask, gateway IP and extract out DHCP transaction ID and remote IP */
 | |
| 		uip_getnetmask(&Netmask);
 | |
| 		uip_getdraddr(&GatewayIPAddress);
 | |
| 		memcpy(&RemoteMACAddress, &AppData->ClientHardwareAddress, sizeof(struct uip_eth_addr));
 | |
| 		TransactionID = AppData->TransactionID;
 | |
| 
 | |
| 		/* Try to extract out the client's preferred IP address if it is indicated in the packet */
 | |
| 		if (!(DHCPCommon_GetOption(AppData->Options, DHCP_OPTION_REQ_IPADDR, &PreferredClientIP)))
 | |
| 		  memcpy(&PreferredClientIP, &uip_all_zeroes_addr, sizeof(uip_ipaddr_t));
 | |
| 
 | |
| 		switch (DHCPMessageType)
 | |
| 		{
 | |
| 			case DHCP_DISCOVER:
 | |
| 				/* If no preference was made or the preferred IP is already taken, find a new address */
 | |
| 				if (DHCPServerApp_CheckIfIPLeased(&PreferredClientIP))
 | |
| 				  DHCPServerApp_GetUnleasedIP(&PreferredClientIP);
 | |
| 
 | |
| 				/* Create a new DHCP OFFER packet with the offered IP address */
 | |
| 				AppDataSize += DHCPServerApp_FillDHCPHeader(AppData, DHCP_OFFER, &RemoteMACAddress, &PreferredClientIP, TransactionID);
 | |
| 
 | |
| 				/* Add network mask and router information to the list of DHCP OFFER packet options */
 | |
| 				AppDataSize += DHCPCommon_SetOption(AppData->Options, DHCP_OPTION_SUBNET_MASK,
 | |
| 													sizeof(uip_ipaddr_t), &Netmask);
 | |
| 				AppDataSize += DHCPCommon_SetOption(AppData->Options, DHCP_OPTION_ROUTER,
 | |
| 					                                sizeof(uip_ipaddr_t), &GatewayIPAddress);
 | |
| 
 | |
| 				/* Send the DHCP OFFER packet */
 | |
| 				uip_poll_conn(BroadcastConnection);
 | |
| 				memcpy(&uip_udp_conn->ripaddr, &uip_broadcast_addr, sizeof(uip_ipaddr_t));
 | |
| 				uip_udp_send(AppDataSize);
 | |
| 
 | |
| 				break;
 | |
| 			case DHCP_REQUEST:
 | |
| 				/* Check to see if the requested IP address has already been leased to a client */
 | |
| 				if (!(DHCPServerApp_CheckIfIPLeased(&PreferredClientIP)))
 | |
| 				{
 | |
| 					/* Create a new DHCP ACK packet to accept the IP address lease */
 | |
| 					AppDataSize += DHCPServerApp_FillDHCPHeader(AppData, DHCP_ACK, &RemoteMACAddress, &PreferredClientIP, TransactionID);
 | |
| 
 | |
| 					/* Add network mask and router information to the list of DHCP ACK packet options */
 | |
| 					AppDataSize += DHCPCommon_SetOption(AppData->Options, DHCP_OPTION_SUBNET_MASK,
 | |
| 														sizeof(uip_ipaddr_t), &Netmask);
 | |
| 					AppDataSize += DHCPCommon_SetOption(AppData->Options, DHCP_OPTION_ROUTER,
 | |
| 					                                    sizeof(uip_ipaddr_t), &GatewayIPAddress);
 | |
| 
 | |
| 					/* Mark the requested IP as leased to a client */
 | |
| 					DHCPServerApp_LeaseIP(&PreferredClientIP);
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					/* Create a new DHCP NAK packet to reject the requested allocation */
 | |
| 					AppDataSize += DHCPServerApp_FillDHCPHeader(AppData, DHCP_NAK, &RemoteMACAddress, &uip_all_zeroes_addr, TransactionID);
 | |
| 				}
 | |
| 
 | |
| 				/* Send the DHCP ACK or NAK packet */
 | |
| 				uip_poll_conn(BroadcastConnection);
 | |
| 				memcpy(&uip_udp_conn->ripaddr, &uip_broadcast_addr, sizeof(uip_ipaddr_t));
 | |
| 				uip_udp_send(AppDataSize);
 | |
| 
 | |
| 				break;
 | |
| 			case DHCP_RELEASE:
 | |
| 				/* Mark the IP address as released in the allocation table */
 | |
| 				DHCPServerApp_UnleaseIP(&uip_udp_conn->ripaddr);
 | |
| 				break;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /** Fills the DHCP packet response with the appropriate BOOTP header for DHCP. This fills out all the required
 | |
|  *  fields, leaving only the additional DHCP options to be added to the packet before it is sent to the DHCP client.
 | |
|  *
 | |
|  *  \param[out] DHCPHeader             Location in the packet buffer where the BOOTP header should be written to
 | |
|  *  \param[in]  DHCPMessageType        DHCP Message type, such as DHCP_DISCOVER
 | |
|  *  \param[in]  ClientHardwareAddress  Client MAC address the created transaction should be directed to
 | |
|  *  \param[in]  PreferredClientIP      Preferred IP that should be given to the client if it is unallocated
 | |
|  *  \param[in]  TransactionID          Transaction ID the created transaction should be associated with
 | |
|  *
 | |
|  *  \return Size in bytes of the created DHCP packet
 | |
|  */
 | |
| static uint16_t DHCPServerApp_FillDHCPHeader(DHCP_Header_t* const DHCPHeader,
 | |
|                                              const uint8_t DHCPMessageType,
 | |
|                                              const struct uip_eth_addr* const ClientHardwareAddress,
 | |
| 											 const uip_ipaddr_t* const PreferredClientIP,
 | |
|                                              const uint32_t TransactionID)
 | |
| {
 | |
| 	/* Erase existing packet data so that we start will all 0x00 DHCP header data */
 | |
|  	memset(DHCPHeader, 0, sizeof(DHCP_Header_t));
 | |
| 
 | |
| 	DHCPHeader->Operation             = DHCPMessageType;
 | |
| 	DHCPHeader->HardwareType          = DHCP_HTYPE_ETHERNET;
 | |
| 	DHCPHeader->HardwareAddressLength = sizeof(MACAddress);
 | |
| 	DHCPHeader->Hops                  = 0;
 | |
| 	DHCPHeader->TransactionID         = TransactionID;
 | |
| 	DHCPHeader->ElapsedSeconds        = 0;
 | |
| 	DHCPHeader->Flags                 = 0;
 | |
| 	memcpy(&DHCPHeader->NextServerIP, &uip_hostaddr, sizeof(uip_ipaddr_t));
 | |
| 	memcpy(&DHCPHeader->YourIP, PreferredClientIP, sizeof(uip_ipaddr_t));
 | |
| 	memcpy(&DHCPHeader->ClientHardwareAddress, ClientHardwareAddress, sizeof(struct uip_eth_addr));
 | |
| 	DHCPHeader->Cookie                = DHCP_MAGIC_COOKIE;
 | |
| 
 | |
| 	/* Add a DHCP message type and terminator options to the start of the DHCP options field */
 | |
| 	DHCPHeader->Options[0]            = DHCP_OPTION_MSG_TYPE;
 | |
| 	DHCPHeader->Options[1]            = 1;
 | |
| 	DHCPHeader->Options[2]            = DHCPMessageType;
 | |
| 	DHCPHeader->Options[3]            = DHCP_OPTION_END;
 | |
| 
 | |
| 	/* Calculate the total number of bytes added to the outgoing packet */
 | |
| 	return (sizeof(DHCP_Header_t) + 4);
 | |
| }
 | |
| 
 | |
| /** Checks to see if the nominated IP address has already been allocated to a client.
 | |
|  *
 | |
|  *  \param[in] IPAddress  IP Address whose lease status should be checked
 | |
|  *
 | |
|  *  \pre The IP address must be within the same /24 subnet as the virtual webserver.
 | |
|  *
 | |
|  *  \return Boolean true if the IP has already been leased to a client, false otherwise.
 | |
|  */
 | |
| static bool DHCPServerApp_CheckIfIPLeased(const uip_ipaddr_t* const IPAddress)
 | |
| {
 | |
| 	uint8_t Byte = (IPAddress->u8[3] / 8);
 | |
| 	uint8_t Mask = (1 << (IPAddress->u8[3] % 8));
 | |
| 
 | |
| 	/* Make sure that the requested IP address isn't already leased to the virtual server or another client */
 | |
| 	if (IPAddress->u8[3] && !(IPAddress->u8[3] == uip_hostaddr.u8[3]) && !(LeasedIPs[Byte] & Mask))
 | |
| 	  return false;
 | |
| 	else
 | |
| 	  return true;
 | |
| }
 | |
| 
 | |
| /** Retrieves the next unleased IP in the IP address pool.
 | |
|  *
 | |
|  *  \param[out] NewIPAddress  Location where the generated IP Address should be stored
 | |
|  */
 | |
| static void DHCPServerApp_GetUnleasedIP(uip_ipaddr_t* const NewIPAddress)
 | |
| {
 | |
| 	uip_ipaddr_copy(NewIPAddress, &uip_hostaddr);
 | |
| 
 | |
| 	/** Look through the current subnet, skipping the broadcast and zero IP addresses */
 | |
| 	for (uint8_t IP = 1; IP < 254; IP++)
 | |
| 	{
 | |
| 		/* Update new IP address to lease with the current IP address to test */
 | |
| 		NewIPAddress->u8[3] = IP;
 | |
| 
 | |
| 		/* If we've found an unleased IP, abort with the updated IP stored for the called */
 | |
| 		if (!(DHCPServerApp_CheckIfIPLeased(NewIPAddress)))
 | |
| 		  return;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /** Marks the given IP Address as leased in the address pool, so that it will not be
 | |
|  *  allocated to another client unless it is first released.
 | |
|  *
 | |
|  *  \param[in] IPAddress  IP Address to mark as leased
 | |
|  *
 | |
|  *  \pre The IP address must be within the same /24 subnet as the virtual webserver.
 | |
|  */
 | |
| static void DHCPServerApp_LeaseIP(const uip_ipaddr_t* const IPAddress)
 | |
| {
 | |
| 	uint8_t Byte = (IPAddress->u8[3] / 8);
 | |
| 	uint8_t Mask = (1 << (IPAddress->u8[3] % 8));
 | |
| 
 | |
| 	/* Mark the IP address as leased in the allocation table */
 | |
| 	LeasedIPs[Byte] |= Mask;
 | |
| }
 | |
| 
 | |
| /** Marks the given IP Address as not leased in the address pool, so that it can be
 | |
|  *  allocated to another client upon request.
 | |
|  *
 | |
|  *  \param[in] IPAddress  IP Address to mark as not leased
 | |
|  *
 | |
|  *  \pre The IP address must be within the same /24 subnet as the virtual webserver.
 | |
|  */
 | |
| static void DHCPServerApp_UnleaseIP(const uip_ipaddr_t* const IPAddress)
 | |
| {
 | |
| 	uint8_t Byte = (IPAddress->u8[3] / 8);
 | |
| 	uint8_t Mask = (1 << (IPAddress->u8[3] % 8));
 | |
| 
 | |
| 	/* Mark the IP address as unleased in the allocation table */
 | |
| 	LeasedIPs[Byte] &= ~Mask;
 | |
| }
 | |
| #endif
 | |
| 
 | 
