ref: e48a5c343d3ba76c6dfb949b5178e46a41792f72
dir: /sys/src/boot/efi/fs.c/
#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_GUID EFI_LOADED_IMAGE_PROTOCOL_GUID = { 0x5b1b31a1, 0x9562, 0x11d2, 0x8e, 0x3f, 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 > 4096 ? 4096 : 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); } int fsinit(void **pf) { EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *fs; EFI_LOADED_IMAGE_PROTOCOL *image; EFI_FILE_PROTOCOL *root; EFI_HANDLE *Handles; void *f; UINTN Count; int i; image = nil; /* locate kernel and plan9.ini by deriving a fs protocol * from the device the loader was read from. * if that fails, fall back to old method. */ if(eficall(ST->BootServices->HandleProtocol, IH, &EFI_LOADED_IMAGE_PROTOCOL_GUID, &image) == 0 && eficall(ST->BootServices->HandleProtocol, image->DeviceHandle, &EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID, &fs) == 0 && eficall(fs->OpenVolume, fs, &root) == 0){ fsroot = root; f = fsopen("/plan9.ini"); if(f != nil){ if(pf != nil) *pf = f; else fsclose(f); goto gotit; } } Count = 0; Handles = nil; if(eficall(ST->BootServices->LocateHandleBuffer, ByProtocol, &EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID, nil, &Count, &Handles)) return -1; /* * assuming the ESP is the first entry in the handle buffer, so go backwards * to scan for plan9.ini in other (9fat) filesystems first. if nothing is found * we'll be defaulting to the ESP. */ fsroot = nil; for(i=Count-1; i>=0; i--){ root = nil; fs = nil; if(eficall(ST->BootServices->HandleProtocol, Handles[i], &EFI_SIMPLE_FILE_SYSTEM_PROTOCOL_GUID, &fs)) continue; if(eficall(fs->OpenVolume, fs, &root)) continue; fsroot = root; f = fsopen("/plan9.ini"); if(f != nil){ if(pf != nil) *pf = f; else fsclose(f); break; } } if(fsroot == nil) return -1; gotit: read = fsread; close = fsclose; open = fsopen; return 0; }