shithub: riscv

Download patch

ref: 16e08adb32778864035cc582c3a9163d3032ca4b
parent: e81e1a4aeddad2bc612c9c5243573250b6ff33a4
author: cinap_lenrek <[email protected]>
date: Thu Oct 23 19:11:49 EDT 2014

efi: add initial pxe support (v4 only)

--- a/sys/src/boot/efi/efi.c
+++ b/sys/src/boot/efi/efi.c
@@ -2,15 +2,14 @@
 #include "fns.h"
 #include "efi.h"
 
-enum {
-	MAXPATH = 128,
-};
-
-UINTN MK;
+UINTN MK;
 EFI_HANDLE IH;
 EFI_SYSTEM_TABLE *ST;
-EFI_FILE_PROTOCOL *root;
 
+void* (*open)(char *name);
+int (*read)(void *f, void *data, int len);
+void (*close)(void *f);
+
 void
 putc(int c)
 {
@@ -18,7 +17,7 @@
 
 	w[0] = c;
 	w[1] = 0;
-	eficall(2, ST->ConOut->OutputString, ST->ConOut, w);
+	eficall(ST->ConOut->OutputString, ST->ConOut, w);
 }
 
 int
@@ -26,7 +25,7 @@
 {
 	EFI_INPUT_KEY k;
 
-	if(eficall(2, ST->ConIn->ReadKeyStroke, ST->ConIn, &k))
+	if(eficall(ST->ConIn->ReadKeyStroke, ST->ConIn, &k))
 		return 0;
 	return k.UnicodeChar;
 }
@@ -34,79 +33,16 @@
 void
 usleep(int us)
 {
-	eficall(1, ST->BootServices->Stall, (UINTN)us);
+	eficall(ST->BootServices->Stall, (UINTN)us);
 }
 
 void
 unload(void)
 {
-	eficall(2, ST->BootServices->ExitBootServices, IH, MK);
+	eficall(ST->BootServices->ExitBootServices, IH, MK);
 }
 
-void
-fsinit(void)
-{
-	EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *fs;
-
-	fs = nil;
-	root = nil;
-	if(eficall(3, ST->BootServices->LocateProtocol,
-		&EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID, nil, &fs))
-		return;
-	if(eficall(2, fs->OpenVolume, fs, &root)){
-		root = nil;
-		return;
-	}
-}
-
 static void
-towpath(CHAR16 *w, int nw, char *s)
-{
-	int i;
-
-	for(i=0; *s && i<nw-1; i++){
-		*w = *s++;
-		if(*w == '/')
-			*w = '\\';
-		w++;
-	}
-	*w = 0;
-}
-
-EFI_FILE_PROTOCOL*
-fswalk(EFI_FILE_PROTOCOL *dir, char *name)
-{
-	CHAR16 wname[MAXPATH];
-	EFI_FILE_PROTOCOL *fp;
-
-	towpath(wname, MAXPATH, name);
-
-	fp = nil;
-	if(eficall(5, dir->Open, dir, &fp, wname, (UINT64)1, (UINT64)1))
-		return nil;
-	return fp;
-}
-
-
-int
-read(void *f, void *data, int len)
-{
-	UINTN size;
-
-	size = len;
-	if(eficall(3, ((EFI_FILE_PROTOCOL*)f)->Read, f, &size, data))
-		return 0;
-	return (int)size;
-}
-
-void
-close(void *f)
-{
-	eficall(1, ((EFI_FILE_PROTOCOL*)f)->Close, f);
-}
-
-
-static void
 memconf(char **cfg)
 {
 	static uchar memtype[EfiMaxMemoryType] = {
@@ -134,7 +70,7 @@
 	mapsize = sizeof(mapbuf);
 	entsize = sizeof(EFI_MEMORY_DESCRIPTOR);
 	entvers = 1;
-	if(eficall(5, ST->BootServices->GetMemoryMap, &mapsize, mapbuf, &MK, &entsize, &entvers))
+	if(eficall(ST->BootServices->GetMemoryMap, &mapsize, mapbuf, &MK, &entsize, &entvers))
 		return;
 
 	s = *cfg;
@@ -165,6 +101,16 @@
 static void
 acpiconf(char **cfg)
 {
+	static EFI_GUID ACPI_20_TABLE_GUID = {
+		0x8868e871, 0xe4f1, 0x11d3,
+		0xbc, 0x22, 0x00, 0x80,
+		0xc7, 0x3c, 0x88, 0x81,
+	};
+	static EFI_GUID ACPI_10_TABLE_GUID = {
+		0xeb9d2d30, 0x2d88, 0x11d3,
+		0x9a, 0x16, 0x00, 0x90,
+		0x27, 0x3f, 0xc1, 0x4d,
+	};
 	EFI_CONFIGURATION_TABLE *t;
 	uintptr pa;
 	char *s;
@@ -220,6 +166,11 @@
 static void
 screenconf(char **cfg)
 {
+	static EFI_GUID EFI_GRAPHICS_OUTPUT_PROTOCOL_GUID = {
+		0x9042a9de, 0x23dc, 0x4a38,
+		0x96, 0xfb, 0x7a, 0xde,
+		0xd0, 0x80, 0x51, 0x6a,
+	};
 	EFI_GRAPHICS_OUTPUT_PROTOCOL *gop;
 	EFI_HANDLE *Handles;
 	UINTN Count;
@@ -231,13 +182,13 @@
 
 	Count = 0;
 	Handles = nil;
-	if(eficall(5, ST->BootServices->LocateHandleBuffer,
+	if(eficall(ST->BootServices->LocateHandleBuffer,
 		ByProtocol, &EFI_GRAPHICS_OUTPUT_PROTOCOL_GUID, nil, &Count, &Handles))
 		return;
 
 	for(i=0; i<Count; i++){
 		gop = nil;
-		if(eficall(3, ST->BootServices->HandleProtocol,
+		if(eficall(ST->BootServices->HandleProtocol,
 			Handles[i], &EFI_GRAPHICS_OUTPUT_PROTOCOL_GUID, &gop))
 			continue;
 
@@ -331,7 +282,7 @@
 }
 
 EFI_STATUS
-main(EFI_HANDLE ih, EFI_SYSTEM_TABLE *st)
+efimain(EFI_HANDLE ih, EFI_SYSTEM_TABLE *st)
 {
 	char path[MAXPATH], *kern;
 	void *f;
@@ -339,12 +290,13 @@
 	IH = ih;
 	ST = st;
 
-	fsinit();
+	f = pxeinit();
+	if(f == nil)
+		f = fsinit();
 
-	f = fswalk(root, "/plan9.ini");
 	for(;;){
 		kern = configure(f, path);
-		f = fswalk(root, kern);
+		f = open(kern);
 		if(f == nil){
 			print("not found\n");
 			continue;
--- a/sys/src/boot/efi/efi.h
+++ b/sys/src/boot/efi/efi.h
@@ -4,6 +4,7 @@
 typedef ushort	UINT16;
 typedef ulong	UINT32;
 typedef uvlong	UINT64;
+typedef UINT8	BOOLEAN;
 
 typedef uintptr	UINTN;
 
@@ -87,41 +88,6 @@
 	EFI_GRAPHICS_OUTPUT_PROTOCOL_MODE *Mode;
 } EFI_GRAPHICS_OUTPUT_PROTOCOL;
 
-EFI_GUID EFI_GRAPHICS_OUTPUT_PROTOCOL_GUID = {
-	0x9042a9de, 0x23dc, 0x4a38,
-	0x96, 0xfb, 0x7a, 0xde,
-	0xd0, 0x80, 0x51, 0x6a,
-};
-
-typedef struct {
-	UINT64		Revision;
-	void		*Open;
-	void		*Close;
-	void		*Delete;
-	void		*Read;
-	void		*Write;
-	void		*GetPosition;
-	void		*SetPosition;
-	void		*GetInfo;
-	void		*SetInfo;
-	void		*Flush;
-	void		*OpenEx;
-	void		*ReadEx;
-	void		*WriteEx;
-	void		*FlushEx;
-} EFI_FILE_PROTOCOL;
-
-typedef struct {
-	UINT64		Revision;
-	void		*OpenVolume;
-} EFI_SIMPLE_FILE_SYSTEM_PROTOCOL;
-
-EFI_GUID EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID = {
-	0x0964e5b22, 0x6459, 0x11d2,
-	0x8e, 0x39, 0x00, 0xa0,
-	0xc9, 0x69, 0x72, 0x3b,
-};
-
 enum {
 	EfiReservedMemoryType,
 	EfiLoaderCode,
@@ -243,19 +209,6 @@
 	void		*QueryVariableInfo;
 } EFI_RUNTIME_SERVICES;
 
-
-EFI_GUID ACPI_20_TABLE_GUID = {
-	0x8868e871, 0xe4f1, 0x11d3,
-	0xbc, 0x22, 0x00, 0x80,
-	0xc7, 0x3c, 0x88, 0x81,
-};
-
-EFI_GUID ACPI_10_TABLE_GUID = {
-	0xeb9d2d30, 0x2d88, 0x11d3,
-	0x9a, 0x16, 0x00, 0x90,
-	0x27, 0x3f, 0xc1, 0x4d,
-};
-
 typedef struct {
 	EFI_GUID	VendorGuid;
 	void		*VendorTable;
@@ -282,3 +235,6 @@
 	UINTN			NumberOfTableEntries;
 	EFI_CONFIGURATION_TABLE	*ConfigurationTable;
 } EFI_SYSTEM_TABLE;
+
+extern EFI_SYSTEM_TABLE *ST;
+extern EFI_HANDLE IH;
--- a/sys/src/boot/efi/fns.h
+++ b/sys/src/boot/efi/fns.h
@@ -1,11 +1,20 @@
+enum {
+	MAXPATH = 128,
+};
+
 extern char hex[];
 
 void usleep(int t);
 void jump(void *pc);
 
-int read(void *f, void *data, int len);
+void* pxeinit(void);
+void* fsinit(void);
+
+void* (*open)(char *name);
+int (*read)(void *f, void *data, int len);
+void (*close)(void *f);
+
 int readn(void *f, void *data, int len);
-void close(void *f);
 void unload(void);
 
 int getc(void);
@@ -25,5 +34,5 @@
 char *hexfmt(char *s, int i, uvlong a);
 char *decfmt(char *s, int i, ulong a);
 
-long eficall(long narg, void *proc, ...);
+uintptr eficall(void *proc, ...);
 void eficonfig(char **cfg);
--- /dev/null
+++ b/sys/src/boot/efi/fs.c
@@ -1,0 +1,106 @@
+#include <u.h>
+#include "fns.h"
+#include "efi.h"
+
+typedef struct {
+	UINT64		Revision;
+	void		*Open;
+	void		*Close;
+	void		*Delete;
+	void		*Read;
+	void		*Write;
+	void		*GetPosition;
+	void		*SetPosition;
+	void		*GetInfo;
+	void		*SetInfo;
+	void		*Flush;
+	void		*OpenEx;
+	void		*ReadEx;
+	void		*WriteEx;
+	void		*FlushEx;
+} EFI_FILE_PROTOCOL;
+
+typedef struct {
+	UINT64		Revision;
+	void		*OpenVolume;
+} EFI_SIMPLE_FILE_SYSTEM_PROTOCOL;
+
+static
+EFI_GUID EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID = {
+	0x0964e5b22, 0x6459, 0x11d2,
+	0x8e, 0x39, 0x00, 0xa0,
+	0xc9, 0x69, 0x72, 0x3b,
+};
+
+static
+EFI_FILE_PROTOCOL *fsroot;
+
+static void
+towpath(CHAR16 *w, int nw, char *s)
+{
+	int i;
+
+	for(i=0; *s && i<nw-1; i++){
+		*w = *s++;
+		if(*w == '/')
+			*w = '\\';
+		w++;
+	}
+	*w = 0;
+}
+
+static void*
+fsopen(char *name)
+{
+	CHAR16 wname[MAXPATH];
+	EFI_FILE_PROTOCOL *fp;
+
+	if(fsroot == nil)
+		return nil;
+
+	towpath(wname, MAXPATH, name);
+
+	fp = nil;
+	if(eficall(fsroot->Open, fsroot, &fp, wname, (UINT64)1, (UINT64)1))
+		return nil;
+	return fp;
+}
+
+static int
+fsread(void *f, void *data, int len)
+{
+	UINTN size;
+
+	size = len;
+	if(eficall(((EFI_FILE_PROTOCOL*)f)->Read, f, &size, data))
+		return 0;
+	return (int)size;
+}
+
+static void
+fsclose(void *f)
+{
+	eficall(((EFI_FILE_PROTOCOL*)f)->Close, f);
+}
+
+void*
+fsinit(void)
+{
+	EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *fs;
+
+	fs = nil;
+	fsroot = nil;
+	if(eficall(ST->BootServices->LocateProtocol,
+		&EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID, nil, &fs))
+		return nil;
+	if(eficall(fs->OpenVolume, fs, &fsroot)){
+		fsroot = nil;
+		return nil;
+	}
+
+	read = fsread;
+	close = fsclose;
+	open = fsopen;
+
+	return fsopen("/plan9.ini");
+}
--- a/sys/src/boot/efi/mkfile
+++ b/sys/src/boot/efi/mkfile
@@ -13,7 +13,7 @@
 clean:V:
 	rm -f *.[68] *.out $TARG
 
-bootia32.efi:	pe32.8 efi.8 sub.8
+bootia32.efi:	pe32.8 efi.8 fs.8 pxe.8 sub.8
 	8l -l -H3 -T$IMAGEBASE -o $target $prereq
 
 pe32.8:	pe32.s
@@ -22,6 +22,12 @@
 efi.8:	efi.c efi.h
 	8c $CFLAGS efi.c
 
+fs.8:	fs.c efi.h
+	8c $CFLAGS fs.c
+
+pxe.8:	pxe.c efi.h
+	8c $CFLAGS pxe.c
+
 sub.8:	sub.c
 	8c $CFLAGS sub.c
 
@@ -28,7 +34,7 @@
 %.8:	$HFILES
 
 
-bootx64.efi:	pe64.6 efi.6 sub.6
+bootx64.efi:	pe64.6 efi.6 fs.6 pxe.6 sub.6
 	6l -l -s -R1 -T$IMAGEBASE -o bootx64.out $prereq
 	dd -if bootx64.out -bs 1 -iseek 40 >$target
 
@@ -37,6 +43,12 @@
 
 efi.6:	efi.c efi.h
 	6c $CFLAGS efi.c
+
+fs.6:	fs.c efi.h
+	6c $CFLAGS fs.c
+
+pxe.6:	pxe.c efi.h
+	6c $CFLAGS pxe.c
 
 sub.6:	sub.c
 	6c $CFLAGS sub.c
--- a/sys/src/boot/efi/pe32.s
+++ b/sys/src/boot/efi/pe32.s
@@ -129,7 +129,7 @@
 	MOVL $edata-IMAGEBASE(SB), CX
 	CLD
 	REP; MOVSB
-	MOVL $main(SB), DI
+	MOVL $efimain(SB), DI
 	MOVL DI, (SP)
 	RET
 
@@ -138,12 +138,22 @@
 	MOVL 4(SP), AX
 	JMP *AX
 
-
 TEXT eficall(SB), 1, $0
-	MOVL 0(SP), DI	/* saved by callee */
-	MOVL 8(SP), AX
-	ADDL $12, SP
+	MOVL SP, SI
+	MOVL SP, DI
+	MOVL $(4*16), CX
+	SUBL CX, DI
+	ANDL $~15ULL, DI
+	SUBL $8, DI
+
+	MOVL 4(SI), AX
+	LEAL 8(DI), SP
+
+	CLD
+	REP; MOVSB
+	SUBL $(4*16), SI
+
 	CALL AX
-	SUBL $12, SP
-	MOVL DI, 0(SP)
+
+	MOVL SI, SP
 	RET
--- a/sys/src/boot/efi/pe64.s
+++ b/sys/src/boot/efi/pe64.s
@@ -136,25 +136,28 @@
 	REP; MOVSB
 
 	MOVQ 16(SP), BP
-	MOVQ $main(SB), DI
+	MOVQ $efimain(SB), DI
 	MOVQ DI, (SP)
 	RET
 
 TEXT eficall(SB), 1, $-4
-	MOVQ 0(SP), DI	/* saved by callee */
-	MOVQ 16(SP), AX
-	ADDQ $24, SP
+	MOVQ SP, SI
+	MOVQ SP, DI
+	MOVL $(8*16), CX
+	SUBQ CX, DI
+	ANDQ $~15ULL, DI
+	LEAQ 16(DI), SP
+	CLD
+	REP; MOVSB
+	SUBQ $(8*16), SI
 
-	/* arguments in regisyers */
 	MOVQ 0(SP), CX
 	MOVQ 8(SP), DX
 	MOVQ 16(SP), R8
 	MOVQ 24(SP), R9
+	CALL BP
 
-	CALL AX
-
-	SUBQ $24, SP
-	MOVQ DI, 0(SP)
+	MOVQ SI, SP
 	RET
 
 #include "mem.h"
--- /dev/null
+++ b/sys/src/boot/efi/pxe.c
@@ -1,0 +1,359 @@
+#include <u.h>
+#include "fns.h"
+#include "efi.h"
+
+typedef UINT16	EFI_PXE_BASE_CODE_UDP_PORT;
+
+typedef struct {
+	UINT8		Addr[4];
+} EFI_IPv4_ADDRESS;
+
+typedef struct {
+	UINT8		Addr[16];
+} EFI_IPv6_ADDRESS;
+
+typedef union {
+	UINT32			Addr[4];
+	EFI_IPv4_ADDRESS	v4;
+	EFI_IPv6_ADDRESS	v6;
+} EFI_IP_ADDRESS;
+
+typedef struct {
+	UINT8		Addr[32];
+} EFI_MAC_ADDRESS;
+
+typedef struct {
+	UINT8		BootpOpcode;
+	UINT8		BootpHwType;
+	UINT8		BootpHwAddrLen;
+	UINT8		BootpGateHops;
+	UINT32		BootpIdent;
+	UINT16		BootpSeconds;
+	UINT16		BootpFlags;
+	UINT8		BootpCiAddr[4];
+	UINT8		BootpYiAddr[4];
+	UINT8		BootpSiAddr[4];
+	UINT8		BootpGiAddr[4];
+	UINT8		BootpHwAddr[16];
+	UINT8		BootpSrvName[64];
+	UINT8		BootpBootFile[128];
+	UINT32		DhcpMagik;
+	UINT8		DhcpOptions[56];		
+} EFI_PXE_BASE_CODE_DHCPV4_PACKET;
+
+typedef struct {
+	BOOLEAN		Started;
+	BOOLEAN		Ipv6Available;
+	BOOLEAN		Ipv6Supported;
+	BOOLEAN		UsingIpv6;
+	BOOLEAN		BisSupported;
+	BOOLEAN		BisDetected;
+	BOOLEAN		AutoArp;
+	BOOLEAN		SendGUID;
+	BOOLEAN		DhcpDiscoverValid;
+	BOOLEAN		DhcpAckReceived;
+	BOOLEAN		ProxyOfferReceived;
+	BOOLEAN		PxeDiscoverValid;
+	BOOLEAN		PxeReplyReceived;
+	BOOLEAN		PxeBisReplyReceived;
+	BOOLEAN		IcmpErrorReceived;
+	BOOLEAN		TftpErrorReceived;
+	BOOLEAN		MakeCallbacks;
+
+	UINT8		TTL;
+	UINT8		ToS;
+
+	UINT8		Reserved;
+
+	UINT8		StationIp[16];
+	UINT8		SubnetMask[16];
+
+	UINT8		DhcpDiscover[1472];
+	UINT8		DhcpAck[1472];
+} EFI_PXE_BASE_CODE_MODE;
+
+typedef struct {
+	UINT64		Revision;
+	void		*Start;
+	void		*Stop;
+	void		*Dhcp;
+	void		*Discover;
+	void		*Mtftp;
+	void		*UdpWrite;
+	void		*UdpRead;
+	void		*SetIpFilter;
+	void		*Arp;
+	void		*SetParameters;
+	void		*SetStationIp;
+	void		*SetPackets;
+	EFI_PXE_BASE_CODE_MODE	*Mode;
+} EFI_PXE_BASE_CODE_PROTOCOL;
+
+
+enum {
+	Tftp_READ	= 1,
+	Tftp_WRITE	= 2,
+	Tftp_DATA	= 3,
+	Tftp_ACK	= 4,
+	Tftp_ERROR	= 5,
+	Tftp_OACK	= 6,
+
+	TftpPort	= 69,
+
+	Segsize		= 512,
+};
+
+static
+EFI_GUID EFI_PXE_BASE_CODE_PROTOCOL_GUID = {
+	0x03C4E603, 0xAC28, 0x11D3,
+	0x9A, 0x2D, 0x00, 0x90,
+	0x27, 0x3F, 0xC1, 0x4D,
+};
+
+static
+EFI_PXE_BASE_CODE_PROTOCOL *pxe;
+
+static
+EFI_PXE_BASE_CODE_DHCPV4_PACKET *dhcp;
+
+typedef struct Tftp Tftp;
+struct Tftp
+{
+	EFI_IP_ADDRESS sip;
+	EFI_IP_ADDRESS dip;
+
+	EFI_PXE_BASE_CODE_UDP_PORT sport;
+	EFI_PXE_BASE_CODE_UDP_PORT dport;
+
+	char *rp;
+	char *ep;
+
+	int seq;
+	int eof;
+	
+	char pkt[2+2+Segsize];
+	char nul;
+};
+
+static void
+puts(void *x, ushort v)
+{
+	uchar *p = x;
+
+	p[1] = (v>>8) & 0xFF;
+	p[0] = v & 0xFF;
+}
+
+static ushort
+gets(void *x)
+{
+	uchar *p = x;
+
+	return p[1]<<8 | p[0];
+}
+
+static void
+hnputs(void *x, ushort v)
+{
+	uchar *p = x;
+
+	p[0] = (v>>8) & 0xFF;
+	p[1] = v & 0xFF;
+}
+
+static ushort
+nhgets(void *x)
+{
+	uchar *p = x;
+
+	return p[0]<<8 | p[1];
+}
+
+enum {
+	ANY_SRC_IP	= 0x0001,
+	ANY_SRC_PORT	= 0x0002,
+	ANY_DEST_IP	= 0x0004,
+	ANY_DEST_PORT	= 0x0008,
+	USE_FILTER	= 0x0010,
+	MAY_FRAGMENT	= 0x0020,
+};
+
+static int
+udpread(EFI_IP_ADDRESS *sip, EFI_IP_ADDRESS *dip, 
+	EFI_PXE_BASE_CODE_UDP_PORT *sport, 
+	EFI_PXE_BASE_CODE_UDP_PORT dport, 
+	int *len, void *data)
+{
+	UINTN size;
+
+	size = *len;
+	if(eficall(pxe->UdpRead, pxe, (UINTN)ANY_SRC_PORT, dip, &dport, sip, sport, nil, nil, &size, data))
+		return -1;
+
+	*len = size;
+	return 0;
+}
+
+static int
+udpwrite(EFI_IP_ADDRESS *dip,
+	EFI_PXE_BASE_CODE_UDP_PORT sport, 
+	EFI_PXE_BASE_CODE_UDP_PORT dport,
+	int len, void *data)
+{
+	UINTN size;
+
+	size = len;
+	if(eficall(pxe->UdpWrite, pxe, (UINTN)MAY_FRAGMENT, dip, &dport, nil, nil, &sport, nil, nil, &size, data))
+		return -1;
+
+	return 0;
+}
+
+static int
+pxeread(void *f, void *data, int len)
+{
+	Tftp *t = f;
+	int seq, n;
+
+	while(!t->eof && t->rp >= t->ep){
+		for(;;){
+			n = sizeof(t->pkt);
+			if(udpread(&t->dip, &t->sip, &t->dport, t->sport, &n, t->pkt))
+				continue;
+			if(n >= 4)
+				break;
+		}
+		switch(nhgets(t->pkt)){
+		case Tftp_DATA:
+			seq = nhgets(t->pkt+2);
+			if(seq > t->seq){
+				putc('?');
+				continue;
+			}
+			hnputs(t->pkt, Tftp_ACK);
+			while(udpwrite(&t->dip, t->sport, t->dport, 4, t->pkt))
+				putc('!');
+			if(seq < t->seq){
+				putc('@');
+				continue;
+			}
+			t->seq = seq+1;
+			n -= 4;
+			t->rp = t->pkt + 4;
+			t->ep = t->rp + n;
+			t->eof = n < Segsize;
+			break;
+		case Tftp_ERROR:
+			print(t->pkt+4);
+			print("\n");
+		default:
+			t->eof = 1;
+			return -1;
+		}
+		break;
+	}
+	n = t->ep - t->rp;
+	if(len > n)
+		len = n;
+	memmove(data, t->rp, len);
+	t->rp += len;
+	return len;
+}
+
+static void
+pxeclose(void *f)
+{
+	Tftp *t = f;
+	t->eof = 1;
+}
+
+
+static int
+tftpopen(Tftp *t, char *path)
+{
+	static EFI_PXE_BASE_CODE_UDP_PORT xport = 6666;
+	int r, n;
+	char *p;
+
+	t->sport = xport++;
+	t->dport = 0;
+	t->rp = t->ep = 0;
+	t->seq = 1;
+	t->eof = 0;
+	t->nul = 0;
+	p = t->pkt;
+	hnputs(p, Tftp_READ); p += 2;
+	n = strlen(path)+1;
+	memmove(p, path, n); p += n;
+	memmove(p, "octet", 6); p += 6;
+	n = p - t->pkt;
+	for(;;){
+		if(r = udpwrite(&t->dip, t->sport, TftpPort, n, t->pkt))
+			break;
+		if(r = pxeread(t, 0, 0))
+			break;
+		return 0;
+	}
+	pxeclose(t);
+	return r;
+}
+
+static void*
+pxeopen(char *name)
+{
+	static Tftp t[1];
+
+	memset(t, 0, sizeof(Tftp));
+
+	memmove(&t->dip, dhcp->BootpSiAddr, 4);
+	memmove(&t->sip, pxe->Mode->StationIp, 4);
+
+	if(tftpopen(t, name) == 0)
+		return t;
+	return nil;
+}
+
+void*
+pxeinit(void)
+{
+	EFI_HANDLE *Handles;
+	UINTN Count;
+	char ini[24];
+	uchar *mac;
+	void *f;
+	int i;
+
+	pxe = nil;
+	dhcp = nil;
+	Count = 0;
+	Handles = nil;
+	if(eficall(ST->BootServices->LocateHandleBuffer,
+		ByProtocol, &EFI_PXE_BASE_CODE_PROTOCOL_GUID, nil, &Count, &Handles))
+		return nil;
+
+	for(i=0; i<Count; i++){
+		pxe = nil;
+		if(eficall(ST->BootServices->HandleProtocol,
+			Handles[i], &EFI_PXE_BASE_CODE_PROTOCOL_GUID, &pxe))
+			continue;
+		if(pxe->Mode != nil && pxe->Mode->Started)
+			break;
+	}
+	dhcp = (EFI_PXE_BASE_CODE_DHCPV4_PACKET*)pxe->Mode->DhcpAck;
+
+	mac = dhcp->BootpHwAddr;
+	memmove(ini, "/cfg/pxe/", 9);
+	for(i=0; i<6; i++){
+		ini[9+i*2+0] = hex[mac[i] >> 4];
+		ini[9+i*2+1] = hex[mac[i] & 0xF];
+	}
+	ini[9+12] = '\0';
+
+	open = pxeopen;
+	read = pxeread;
+	close = pxeclose;
+
+	if((f = pxeopen(ini)) != nil)
+		return f;
+	return pxeopen("/cfg/pxe/default");
+}