#include "su.h" #include "pxe_cmn.h" #include "pxe_api.h" #include "tftp_api.h" #include "udp_api.h" #include "dhcp.h" #include "pxe.h" #define htons( a ) ((((a) & 0xFF00) >> 8) |\ (((a) & 0x00FF) << 8)) // // packet - Work buffer used to hold DHCP ACK and BINL REPLY packets. // These packets will be read into this buffer using the // PXENV API service PXENV_GET_BINL_INFO. // BOOTPLAYER packet; // // PxenvApiCall() - see su.asm for details // extern UINT16 PxenvApiCall( UINT16 service, void far *param ); // // GetPacket() // // Description: // Get cached packet from PXENV API. // // Passed: // packet := Far pointer to packet buffer // packet_type := see pxe_api.h for PXENV_PACKET_TYPE_xxx #defines // // Returns: // -1 := Packet could not be transfered to buffer // size := Number of bytes transfered into packet buffer // // Warning: // No check is made to see if buffer is actually large enough to // hold the entire packet. The buffer should be of type BOOTPLAYER. // SHORT GetPacket( void far *packet, UINT16 packet_type ) { t_PXENV_GET_BINL_INFO gbi; // // Check for invalid parameters // if (packet == NULL) { BlPrint("\nGetPacket() NULL pointers\n"); return -1; } // // Request size of packet by sending a size of zero. // gbi.packet_type = packet_type; gbi.buffer_size = 0; if (PxenvApiCall(PXENV_GET_BINL_INFO, &gbi) != PXENV_EXIT_SUCCESS) { BlPrint("\nGetPacket() PXENV API FAILURE #1\n"); return -1; } // // Transfer cached packet into buffer. // gbi.buffer_offset = FP_OFF(packet); gbi.buffer_segment = FP_SEG(packet); if (PxenvApiCall(PXENV_GET_BINL_INFO, &gbi) != PXENV_EXIT_SUCCESS) { BlPrint("\nGetPacket() PXENV API FAILURE #2\n"); return -1; } return (SHORT)gbi.buffer_size; } // // pFindOption() // // Description: // Find the desired Option in an options string // internal routine to allow FindOption/FindVendorOption // to share code. // // Passed: // Option := Option to be found // options := options string // pLength := IN Length of options string (16bit max) // OUT Length of option. (8bit max) // // Returns: // pointer to option data; NULL if not present // UINT32 * pFindOption( UINT8 Option, UINT8 *options, UINT16 *pLength ) { UINT8 *end; if (options == NULL || pLength == NULL) { return NULL; } end = options + *pLength; *pLength = 0; // // walk down the packet looking for the desired option // type. be sure to check that the option will not // walk off the end of a malformed packet. // use length to indicate whether a valid // option was found // while ((options < end) && (*options != 0xFF) ) { // // step over option pads // if ( *options == DHCP_PAD ) { options++; } else { if ( end <= options + 2 || end <= options + 2 + options[1] ) { // // invalid option. it walked past the end of the packet // break; } if ( *options == Option ) { // // found the option. break out of loop // *pLength = options[1]; break; } else { options += 2 + options[1]; } } } return (*pLength != 0) ? (UINT32 *)(options + 2) : NULL; } // // FindOption() // // Description: // Find the subnet mask option in a DHCP packet. // // Passed: // Packet := IN Pointer to DHCP packet. // PacketLength := IN Length of DHCP packet. // pLength := OUT Length of option. // // Returns: // pointer to option data; NULL if not present // UINT32 * FindOption( UINT8 Option, BOOTPLAYER *Packet, UINT16 PacketLength, UINT8 *pLength ) { UINT32 *retOption; UINT16 length; // // Verify parameters // if ( *((ULONG *)Packet->vendor.v.magic) != VM_RFC1048 ) { return NULL; } length = PacketLength; retOption = pFindOption(Option, (UINT8 *)&Packet->vendor.v.flags, &length ); if (pLength != NULL) { *pLength = (UINT8)length; } return retOption; } UCHAR * FindVendorOption( UINT8 Option, UINT8 VendorOption, BOOTPLAYER *Packet, UINT16 PacketLength, UINT8 * pLength ) { UINT8 *start; UINT16 cb; UCHAR *retOption; if (pLength != NULL) { *pLength = 0; } start = (UINT8*)FindOption( Option, Packet, PacketLength, (UINT8*)&cb ); if (start == NULL) { return NULL; } retOption = (UCHAR *)pFindOption( VendorOption, start, &cb ); if (pLength) { *pLength = (UINT8)cb; } return retOption; } // // strlen() // // Description: // Works like std C. // int strlen(UCHAR *s1) { int n = 0; if (s1 != NULL) while (*s1++) ++n; return n; } // // strcpy() // // Description: // Works like std C. // UCHAR * strcpy(UCHAR *s1, UCHAR *s2) { UCHAR *s = s1; if (s1 != NULL && s2 != NULL) while ((*s1++ = *s2++) != 0) ; return s; } // // strncpy() // // Description: // Works like std C. // UCHAR * strncpy(UCHAR *s1, UCHAR *s2, int n) { UCHAR *s = s1; if (s1 != NULL && s2 != NULL && n > 0) while (n--) if ((*s1++ = *s2++) == 0) break; return s; } // // memset() // // Description: // Works like std C. // PUCHAR memset( PUCHAR Destination, UCHAR Value, int Length ) { while (Length--) { *Destination++ = Value; } return Destination; } // // PxenvTftp() // // Description: // Try to transfer the protect-mode loader using information from // DHCP ACK and BINL REPLY packets. // // Passed: // DownloadAddr := Physical address, in client machine, to transfer to. // FileName := File name sent down in BINL REPLY packet. // BOOLEAN PxenvTftp( ) { UINT16 status; UINT16 packetLength; t_PXENV_TFTP_READ_FILE tftp; int pathLength; UINT32 clientIp; UINT32 serverIp; UINT32 gatewayIp; UINT32 *optionPtr; UINT32 subnetMask; UCHAR *FileName; UCHAR cb; UCHAR *optionVendor; // // Get the DHCP ACK packet. // if ((packetLength = GetPacket(&packet, PXENV_PACKET_TYPE_DHCP_ACK)) == -1) { return TRUE; } // // Get client IP address, server IP address, default gateway IP address, // and subnet mask from the DHCP ACK packet. // clientIp = *(UINT32 *)packet.yip; serverIp = *(UINT32 *)packet.sip; //BlPrint("PxenvTftp: DHCP ACK yip = %lx, sip = %lx\n", *(UINT32 *)packet.yip, *(UINT32 *)packet.sip); optionPtr = FindOption( DHCP_ROUTER, &packet, packetLength, NULL ); if ( optionPtr != NULL ) { //BlPrint("PxenvTftp: DHCP ACK router = %lx\n", *optionPtr); gatewayIp = *optionPtr; } else { //BlPrint("PxenvTftp: DHCP ACK gip = %lx\n", *(UINT32 *)packet.gip); gatewayIp = *(UINT32 *)packet.gip; } optionPtr = FindOption( DHCP_SUBNET, &packet, packetLength, NULL ); if ( optionPtr != NULL ) { //BlPrint("PxenvTftp: DHCP ACK subnet = %lx\n", *optionPtr); subnetMask = *optionPtr; } else { //BlPrint("PxenvTftp: DHCP ACK subnet not specified\n"); subnetMask = 0; } // // Get the BINL REPLY packet. // if ((packetLength = GetPacket(&packet, PXENV_PACKET_TYPE_BINL_REPLY)) == -1) { return TRUE; } // // Values for client IP address, server IP address, default gateway IP address, // and subnet mask that are present in the BINL REPLY packet override those // in the DHCP ACK packet. // if ( *(UINT32 *)packet.yip != 0 ) { clientIp = *(UINT32 *)packet.yip; } if ( *(UINT32 *)packet.sip != 0 ) { serverIp = *(UINT32 *)packet.sip; } //BlPrint("PxenvTftp: BINL REPLY yip = %lx, sip = %lx\n", *(UINT32 *)packet.yip, *(UINT32 *)packet.sip); optionPtr = FindOption( DHCP_ROUTER, &packet, packetLength, NULL ); if ( optionPtr != NULL ) { //BlPrint("PxenvTftp: BINL REPLY router = %lx\n", *optionPtr); gatewayIp = *optionPtr; } else if ( *(UINT32 *)packet.gip != 0 ) { //BlPrint("PxenvTftp: BINL REPLY router = %lx\n", *(UINT32 *)packet.gip); gatewayIp = *(UINT32 *)packet.gip; } optionPtr = FindOption( DHCP_SUBNET, &packet, packetLength, NULL ); if ( optionPtr != NULL ) { //BlPrint("PxenvTftp: BINL REPLY subnet = %lx\n", *optionPtr); subnetMask = *optionPtr; } // // Determine whether we need to send packets via the gateway. // //BlPrint("PxenvTftp: clientIp = %lx, serverIp = %lx, subnet = %lx\n", clientIp, serverIp, subnetMask); //BlPrint(" router = %lx\n", gatewayIp); if ( (clientIp & subnetMask) == (serverIp & subnetMask) ) { //BlPrint("PxenvTftp: subnets match. clearing router address\n"); gatewayIp = 0; } //PxenvApiCall(-1, NULL); // // Now fill in the TFTP TRANSFER parameter structure // memset( (PUCHAR)&tftp, 0, sizeof( tftp ) ); // // Find the name and path of the NTLDR that we are going to download. // This is specified by a DHCP Vendor option tag. If this tag // is missing we will default to NTLDR and the same path as // startrom.com. // FileName = (UCHAR*)FindOption( DHCP_LOADER_PATH, &packet, packetLength, &cb ); if ( FileName == NULL ) { // // We could not find the DHCP_LOADER_PATH. We will use the default name of // \NTLDR where the is the same as that used // to download startrom.com // strncpy(tftp.FileName, packet.bootfile, sizeof(tftp.FileName) - sizeof("NTLDR")); tftp.FileName[sizeof(tftp.FileName) - 1] = '\0'; pathLength = strlen(tftp.FileName); while (pathLength > 0) { --pathLength; if (tftp.FileName[pathLength] == '\\') { ++pathLength; // advance it past the '\' break; } } strcpy(tftp.FileName + pathLength, "NTLDR"); } else { // We found the DHCP_LOADER_PATH option. We will use that // as is to download the loader, unless it is too large. // Note that since the DHCP_LOADER_PATH size might include a // null terminator, we need to check to make sure that it // just might fit if ((cb > sizeof(tftp.FileName)) || ((cb == sizeof(tftp.FileName)) && (FileName[cb] != '\0'))) { //BlPrint("PxenvTftp: DHCP_LOADER_PATH is too large = %s\n", FileName); return TRUE; } strncpy(tftp.FileName, FileName, cb); tftp.FileName[sizeof(tftp.FileName) - 1] = '\0'; } // // Loader will be transfered to 1MB region and must not be more // than to 2MB in length. // tftp.BufferSize = 0x200000L; tftp.BufferOffset = 0x100000L; // // Set the Server and gateway address // *((UINT32 *)tftp.ServerIPAddress) = serverIp; *((UINT32 *)tftp.GatewayIPAddress) = gatewayIp; // // Check whether we are going to use multicast download or not. The // multicast options are set in a DHCP option tag (DHCP_LOADER_MCAST_OPTIONS). // These are encapsulated options and work the same way as Vendor options. // If these are missing then unicast transfer will be used. // optionVendor = FindVendorOption( DHCP_LOADER_MCAST_OPTIONS, PXE_MTFTP_IP, &packet, packetLength, &cb ); if ( optionVendor != NULL && cb == 4 ) { *(UINT32*)tftp.McastIPAddress = *(UINT32*)optionVendor; optionVendor = FindVendorOption( DHCP_LOADER_MCAST_OPTIONS, PXE_MTFTP_CPORT, &packet, packetLength, &cb ); if (optionVendor == NULL || cb != 2) { return TRUE; } tftp.TFTPClntPort = htons( *(UINT16*)optionVendor ); optionVendor = FindVendorOption( DHCP_LOADER_MCAST_OPTIONS, PXE_MTFTP_SPORT, &packet, packetLength, &cb ); if (optionVendor == NULL || cb != 2) { return TRUE; } tftp.TFTPSrvPort = htons( *(UINT16*)optionVendor ); optionVendor = FindVendorOption( DHCP_LOADER_MCAST_OPTIONS, PXE_MTFTP_TMOUT, &packet, packetLength, &cb ); if (optionVendor == NULL || cb != 1) { return TRUE; } tftp.TFTPOpenTimeOut = *(UINT8*)optionVendor; optionVendor = FindVendorOption( DHCP_LOADER_MCAST_OPTIONS, PXE_MTFTP_DELAY, &packet, packetLength, &cb ); if (optionVendor == NULL || cb != 1) { return TRUE; } tftp.TFTPReopenDelay = *(UINT8*)optionVendor; } #if DBG BlPrint("Downloading Loader:\n"); BlPrint("FileName = %s\n", tftp.FileName ); BlPrint("BufferSize = %lx\n", tftp.BufferSize ); BlPrint("BufferOffset = %lx\n", tftp.BufferOffset ); BlPrint("ServerIPAddress = %d.%d.%d.%d\n", tftp.ServerIPAddress[0], tftp.ServerIPAddress[1], tftp.ServerIPAddress[2], tftp.ServerIPAddress[3] ); BlPrint("GatewayIPAddress = %d.%d.%d.%d\n", tftp.GatewayIPAddress[0], tftp.GatewayIPAddress[1], tftp.GatewayIPAddress[2], tftp.GatewayIPAddress[3] ); BlPrint("McastIPAddress = %d.%d.%d.%d\n", tftp.McastIPAddress[0], tftp.McastIPAddress[1], tftp.McastIPAddress[2], tftp.McastIPAddress[3] ); BlPrint("TFTPClntPort = %d\n", htons( tftp.TFTPClntPort ) ); BlPrint("TFTPSrvPort = %d\n", htons( tftp.TFTPSrvPort ) ); BlPrint("TFTPOpenTimeOut = %d\n", tftp.TFTPOpenTimeOut ); BlPrint("TFTPReopenDelay = %d\n", tftp.TFTPReopenDelay ); BlPrint("\n\nPress any key to start download...\n" ); _asm { push ax mov ax, 0 int 16h pop ax } #endif // // Transfer image from TFTP server // status = PxenvApiCall(PXENV_TFTP_READ_FILE, &tftp); if (status != PXENV_EXIT_SUCCESS) { return TRUE; } return FALSE; } /* EOF - mtftp.c */