Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

473 lines
12 KiB

/*++
Copyright (c) 1997 Microsoft Corporation
Module Name:
rom.c
Abstract:
Boot loader ROM routines.
Author:
Chuck Lenzmeier (chuckl) December 27, 1996
Revision History:
Notes:
--*/
#include "precomp.h"
#pragma hdrstop
#include <bldrx86.h>
#define PhysToSeg(x) (USHORT)((ULONG)(x) >> 4) & 0xffff
#define PhysToOff(x) (USHORT)((ULONG)(x) & 0x0f)
#include <pxe_cmn.h>
#include <pxe_api.h>
#include <tftp_api.h>
#include <udp_api.h>
#include <undi_api.h>
#include <dhcp.h>
#include <pxe.h>
USHORT NetUnicastUdpDestinationPort = 0;
#if 0
USHORT NetMulticastUdpDestinationPort;
ULONG NetMulticastUdpDestinationAddress;
USHORT NetMulticastUdpSourcePort;
ULONG NetMulticastUdpSourceAddress;
#endif
#if 0 && DBG
#include <stdio.h>
VOID
RomDumpRawData (
IN PUCHAR DataStart,
IN ULONG DataLength,
IN ULONG Offset
);
ULONG RomMaxDumpLength = 64;
#endif
#if 0
//
// chuckl: Don't do this. We added it as part of a solution a problem
// with DEC cards and the boot floppy. We disabled broadcasts in
// startrom\i386\main.c so that the card wouldn't overflow and lock up,
// but we need to have broadcasts enabled in case the server needs to
// ARP us. The purpose of this routine is to enable/disable broadcasts
// during the receive loop, but that seems to put Compaq cards to sleep.
// So we need to leave broadcasts enabled all the time. The DEC card
// problem will have to be fixed another way.
//
VOID
RomSetBroadcastStatus(
BOOLEAN Enable
)
{
t_PXENV_UNDI_SET_PACKET_FILTER UndiSetPF;
USHORT status;
UndiSetPF.Status = 0;
if (Enable) {
UndiSetPF.filter = FLTR_DIRECTED | FLTR_BRDCST;
} else {
UndiSetPF.filter = FLTR_DIRECTED;
}
status = NETPC_ROM_SERVICES( PXENV_UNDI_SET_PACKET_FILTER, &UndiSetPF );
if ((status != 0) || (UndiSetPF.Status != 0)) {
DPRINT( ERROR, ("RomSetBroadcastStatus: set packet filter failed %lx, %lx\n", status, UndiSetPF.Status ));
}
}
#endif
VOID
RomSetReceiveStatus (
IN USHORT UnicastUdpDestinationPort
#if 0
,
IN USHORT MulticastUdpDestinationPort,
IN ULONG MulticastUdpDestinationAddress,
IN USHORT MulticastUdpSourcePort,
IN ULONG MulticastUdpSourceAddress
#endif
)
{
USHORT status;
PUCHAR multicastAddress;
union {
t_PXENV_UDP_OPEN UdpOpen;
t_PXENV_UNDI_SHUTDOWN UndiShutdown;
} command;
if ( UnicastUdpDestinationPort != 0 ) {
//
// If we haven't opened UDP in the ROM yet, do so now.
//
if ( NetUnicastUdpDestinationPort == 0 ) {
command.UdpOpen.Status = 0;
*(UINT32 *)command.UdpOpen.SrcIp = 0;
status = NETPC_ROM_SERVICES( PXENV_UDP_OPEN, &command );
if ( status != 0 ) {
DPRINT( ERROR, ("RomSetReceiveStatus: error %d from UDP_OPEN\n", status) );
}
}
NetUnicastUdpDestinationPort = UnicastUdpDestinationPort;
#if 0
NetMulticastUdpDestinationPort = MulticastUdpDestinationPort;
NetMulticastUdpDestinationAddress = MulticastUdpDestinationAddress;
NetMulticastUdpSourceAddress = MulticastUdpSourceAddress;
NetMulticastUdpSourceAddress = MulticastUdpSourceAddress;
#endif
} else {
//
// This is a loader shutdown notification. Shut the NIC down.
//
// NB: This is irreversible!
//
command.UndiShutdown.Status = 0;
status = NETPC_ROM_SERVICES( PXENV_UNDI_SHUTDOWN, &command );
}
return;
} // RomSetReceiveStatus
ULONG
RomSendUdpPacket (
IN PVOID Buffer,
IN ULONG Length,
IN ULONG RemoteHost,
IN USHORT RemotePort
)
{
USHORT status;
t_PXENV_UDP_WRITE command;
UCHAR tmpBuffer[MAXIMUM_TFTP_PACKET_LENGTH];
#if 0 && DBG
DPRINT( SEND_RECEIVE, ("RomSendUdpPacket: sending this packet:\n") );
IF_DEBUG(SEND_RECEIVE) {
RomDumpRawData( Buffer, Length, 0 );
}
#endif
Length = ( MAXIMUM_TFTP_PACKET_LENGTH < Length ) ? MAXIMUM_TFTP_PACKET_LENGTH : Length;
RtlCopyMemory( tmpBuffer, Buffer, Length );
command.Status = 0;
//
// Determine whether we need to send via the gateway.
//
if ( (RemoteHost & NetLocalSubnetMask) == (NetLocalIpAddress & NetLocalSubnetMask) ) {
*(UINT32 *)command.GatewayIp = 0;
} else {
*(UINT32 *)command.GatewayIp = NetGatewayIpAddress;
}
*(UINT32 *)command.DestIp = RemoteHost;
command.DestPort = RemotePort;
command.SrcPort = NetUnicastUdpDestinationPort;
command.BufferSize = (USHORT)Length;
command.BufferOffset = PhysToOff(tmpBuffer);
command.BufferSegment = PhysToSeg(tmpBuffer);
//DbgPrint( "UDP write pktaddr %lx = %x:%x\n", tmpBuffer, command.BufferSegment, command.BufferOffset );
status = NETPC_ROM_SERVICES( PXENV_UDP_WRITE, &command );
//DbgPrint( "UDP write status = %x\n", command.Status );
if ( status == 0 ) {
return Length;
} else {
return 0;
}
} // RomSendUdpPacket
ULONG
RomReceiveUdpPacket (
IN PVOID Buffer,
IN ULONG Length,
IN ULONG Timeout,
OUT PULONG RemoteHost,
OUT PUSHORT RemotePort
)
{
USHORT status;
t_PXENV_UDP_READ command;
ULONG startTime;
UCHAR tmpBuffer[MAXIMUM_TFTP_PACKET_LENGTH];
//
// Turn on broadcasts while in the receive loop, in case
// the other end needs to ARP to find us.
//
#if 0
RomSetBroadcastStatus(TRUE);
#endif
startTime = SysGetRelativeTime();
if ( Timeout < 2 ) Timeout = 2;
while ( (SysGetRelativeTime() - startTime) < Timeout ) {
command.Status = 0;
*(UINT32 *)command.SrcIp = 0;
*(UINT32 *)command.DestIp = 0;
command.SrcPort = 0;
command.DestPort = 0;
command.BufferSize = (USHORT)Length;
command.BufferOffset = PhysToOff(tmpBuffer);
command.BufferSegment = PhysToSeg(tmpBuffer);
//DbgPrint( "UDP read pktaddr %lx = %x:%x\n", tmpBuffer, command.BufferSegment, command.BufferOffset );
status = NETPC_ROM_SERVICES( PXENV_UDP_READ, &command );
if ( *(UINT32 *)command.SrcIp == 0 ) {
continue;
}
//DbgPrint( "UDP read status = %x, src ip/port = %d.%d.%d.%d/%d, length = %x\n", command.Status,
// command.SrcIp[0], command.SrcIp[1], command.SrcIp[2], command.SrcIp[3],
// SWAP_WORD(command.SrcPort), command.BufferSize );
//DbgPrint( " dest ip/port = %d.%d.%d.%d/%d\n",
// command.DestIp[0], command.DestIp[1], command.DestIp[2], command.DestIp[3],
// SWAP_WORD(command.DestPort) );
#if 0
if ( (command.DestIp[0] < 224) || (command.DestIp[0] > 239) ) {
#endif
//
// This is a directed IP packet.
//
if ( !COMPARE_IP_ADDRESSES(command.DestIp, &NetLocalIpAddress) ||
(command.DestPort != NetUnicastUdpDestinationPort)
) {
// DPRINT( ERROR, (" Directed UDP packet to wrong port\n") );
continue;
}
#if 0
} else {
//
// This is a multicast IP packet.
//
if ( !COMPARE_IP_ADDRESSES(command.SrcIp, &NetMulticastUdpSourceAddress) ||
!COMPARE_IP_ADDRESSES(command.DestIp, &NetMulticastUdpDestinationAddress) ||
(command.SrcPort != NetMulticastUdpSourcePort) ||
(command.DestPort != NetMulticastUdpDestinationPort) ) {
DPRINT( ERROR, (" Multicast UDP packet with wrong source/destination\n") );
continue;
}
}
#endif
//
// We want this packet.
//
goto packet_received;
}
//
// Timeout.
//
DPRINT( SEND_RECEIVE, ("RomReceiveUdpPacket: timeout\n") );
#if 0
RomSetBroadcastStatus(FALSE); // turn off broadcast reception
#endif
return 0;
packet_received:
//
// Packet received.
//
RtlCopyMemory( Buffer, tmpBuffer, command.BufferSize );
*RemoteHost = *(UINT32 *)command.SrcIp;
*RemotePort = command.SrcPort;
#if 0 && DBG
if ( command.BufferSize != 0 ) {
DPRINT( SEND_RECEIVE, ("RomReceiveUdpPacket: received this packet:\n") );
IF_DEBUG(SEND_RECEIVE) {
RomDumpRawData( Buffer, command.BufferSize, 0 );
}
}
#endif
#if 0
RomSetBroadcastStatus(FALSE); // turn off broadcast reception
#endif
return command.BufferSize;
} // RomReceiveUdpPacket
ULONG
RomGetNicType (
OUT t_PXENV_UNDI_GET_NIC_TYPE *NicType
)
{
return NETPC_ROM_SERVICES( PXENV_UNDI_GET_NIC_TYPE, NicType );
}
#if 0 && DBG
VOID
RomDumpRawData (
IN PUCHAR DataStart,
IN ULONG DataLength,
IN ULONG Offset
)
{
ULONG lastByte;
UCHAR lineBuffer[88];
PUCHAR bufferPtr;
if ( DataLength > RomMaxDumpLength ) {
DataLength = RomMaxDumpLength;
}
for ( lastByte = Offset + DataLength; Offset < lastByte; Offset += 16 ) {
ULONG i;
bufferPtr = lineBuffer;
sprintf( bufferPtr, " %08x %04x: ", &DataStart[Offset], Offset );
bufferPtr += 18;
for ( i = 0; i < 16 && Offset + i < lastByte; i++ ) {
sprintf( bufferPtr, "%02x", (UCHAR)DataStart[Offset + i] & (UCHAR)0xFF );
bufferPtr += 2;
if ( i == 7 ) {
*bufferPtr++ = '-';
} else {
*bufferPtr++ = ' ';
}
}
//
// Print enough spaces so that the ASCII display lines up.
//
for ( ; i < 16; i++ ) {
*bufferPtr++ = ' ';
*bufferPtr++ = ' ';
*bufferPtr++ = ' ';
}
*bufferPtr++ = ' ';
*bufferPtr++ = ' ';
*bufferPtr++ = '*';
for ( i = 0; i < 16 && Offset + i < lastByte; i++ ) {
if ( isprint( DataStart[Offset + i] ) ) {
*bufferPtr++ = (CCHAR)DataStart[Offset + i];
} else {
*bufferPtr++ = '.';
}
}
*bufferPtr = 0;
DbgPrint( "%s*\n", lineBuffer );
}
return;
} // RomDumpRawData
#endif // DBG
ARC_STATUS
RomMtftpReadFile (
IN PUCHAR FileName,
IN PVOID Buffer,
IN ULONG BufferLength,
IN ULONG ServerIPAddress, // network byte order
IN ULONG MCastIPAddress, // network byte order
IN USHORT MCastCPort, // network byte order
IN USHORT MCastSPort, // network byte order
IN USHORT Timeout,
IN USHORT Delay,
OUT PULONG DownloadSize
)
{
USHORT status;
t_PXENV_TFTP_READ_FILE tftp;
t_PXENV_UDP_CLOSE udpclose;
if (DownloadSize != NULL) {
*DownloadSize = 0;
}
memset( &tftp, 0 , sizeof( tftp ) );
strcpy(tftp.FileName, FileName);
tftp.BufferSize = BufferLength;
tftp.BufferOffset = (UINT32)Buffer;
if ( (ServerIPAddress & NetLocalSubnetMask) == (NetLocalIpAddress & NetLocalSubnetMask) ) {
*((UINT32 *)tftp.GatewayIPAddress) = 0;
} else {
*((UINT32 *)tftp.GatewayIPAddress) = NetGatewayIpAddress;
}
*((UINT32 *)tftp.ServerIPAddress) = ServerIPAddress;
*((UINT32 *)tftp.McastIPAddress) = MCastIPAddress;
tftp.TFTPClntPort = MCastCPort;
tftp.TFTPSrvPort = MCastSPort;
tftp.TFTPOpenTimeOut = Timeout;
tftp.TFTPReopenDelay = Delay;
// make sure that any UDP sessions are already closed.
status = NETPC_ROM_SERVICES( PXENV_UDP_CLOSE, &udpclose );
status = NETPC_ROM_SERVICES( PXENV_TFTP_READ_FILE, &tftp );
if (status != PXENV_EXIT_SUCCESS) {
return EINVAL;
}
if (DownloadSize != NULL) {
*DownloadSize = tftp.BufferSize;
}
return ESUCCESS;
}