Windows2003-3790/base/boot/startrom/i386/mtftp.c
2020-09-30 16:53:55 +02:00

610 lines
15 KiB
C

#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
// <path>\NTLDR where the <path> 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 */