sread.c 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. /*++
  2. Copyright (c) 1998 Intel Corporation
  3. Module Name:
  4. sread.c
  5. Abstract:
  6. Simple read file access
  7. Revision History
  8. --*/
  9. #include "lib.h"
  10. #define SIMPLE_READ_SIGNATURE EFI_SIGNATURE_32('s','r','d','r')
  11. typedef struct _SIMPLE_READ_FILE {
  12. UINTN Signature;
  13. BOOLEAN FreeBuffer;
  14. VOID *Source;
  15. UINTN SourceSize;
  16. EFI_FILE_HANDLE FileHandle;
  17. } SIMPLE_READ_HANDLE;
  18. EFI_STATUS
  19. OpenSimpleReadFile (
  20. IN BOOLEAN BootPolicy,
  21. IN VOID *SourceBuffer OPTIONAL,
  22. IN UINTN SourceSize,
  23. IN OUT EFI_DEVICE_PATH **FilePath,
  24. OUT EFI_HANDLE *DeviceHandle,
  25. OUT SIMPLE_READ_FILE *SimpleReadHandle
  26. )
  27. /*++
  28. Routine Description:
  29. Opens a file for (simple) reading. The simple read abstraction
  30. will access the file either from a memory copy, from a file
  31. system interface, or from the load file interface.
  32. Arguments:
  33. Returns:
  34. A handle to access the file
  35. --*/
  36. {
  37. SIMPLE_READ_HANDLE *FHand;
  38. EFI_DEVICE_PATH *UserFilePath;
  39. EFI_DEVICE_PATH *TempFilePath;
  40. EFI_DEVICE_PATH *TempFilePathPtr;
  41. FILEPATH_DEVICE_PATH *FilePathNode;
  42. EFI_FILE_HANDLE FileHandle, LastHandle;
  43. EFI_STATUS Status;
  44. EFI_LOAD_FILE_INTERFACE *LoadFile;
  45. FHand = NULL;
  46. UserFilePath = *FilePath;
  47. //
  48. // Allocate a new simple read handle structure
  49. //
  50. FHand = AllocateZeroPool (sizeof(SIMPLE_READ_HANDLE));
  51. if (!FHand) {
  52. Status = EFI_OUT_OF_RESOURCES;
  53. goto Done;
  54. }
  55. *SimpleReadHandle = (SIMPLE_READ_FILE) FHand;
  56. FHand->Signature = SIMPLE_READ_SIGNATURE;
  57. //
  58. // If the caller passed a copy of the file, then just use it
  59. //
  60. if (SourceBuffer) {
  61. FHand->Source = SourceBuffer;
  62. FHand->SourceSize = SourceSize;
  63. *DeviceHandle = NULL;
  64. Status = EFI_SUCCESS;
  65. goto Done;
  66. }
  67. //
  68. // Attempt to access the file via a file system interface
  69. //
  70. FileHandle = NULL;
  71. Status = uefi_call_wrapper(BS->LocateDevicePath, 3, &FileSystemProtocol, FilePath, DeviceHandle);
  72. if (!EFI_ERROR(Status)) {
  73. FileHandle = LibOpenRoot (*DeviceHandle);
  74. }
  75. Status = FileHandle ? EFI_SUCCESS : EFI_UNSUPPORTED;
  76. //
  77. // To access as a filesystem, the filepath should only
  78. // contain filepath components. Follow the filepath nodes
  79. // and find the target file
  80. //
  81. FilePathNode = (FILEPATH_DEVICE_PATH *) *FilePath;
  82. while (!IsDevicePathEnd(&FilePathNode->Header)) {
  83. //
  84. // For filesystem access each node should be a filepath component
  85. //
  86. if (DevicePathType(&FilePathNode->Header) != MEDIA_DEVICE_PATH ||
  87. DevicePathSubType(&FilePathNode->Header) != MEDIA_FILEPATH_DP) {
  88. Status = EFI_UNSUPPORTED;
  89. }
  90. //
  91. // If there's been an error, stop
  92. //
  93. if (EFI_ERROR(Status)) {
  94. break;
  95. }
  96. //
  97. // Open this file path node
  98. //
  99. LastHandle = FileHandle;
  100. FileHandle = NULL;
  101. Status = uefi_call_wrapper(
  102. LastHandle->Open,
  103. 5,
  104. LastHandle,
  105. &FileHandle,
  106. FilePathNode->PathName,
  107. EFI_FILE_MODE_READ,
  108. 0
  109. );
  110. //
  111. // Close the last node
  112. //
  113. uefi_call_wrapper(LastHandle->Close, 1, LastHandle);
  114. //
  115. // Get the next node
  116. //
  117. FilePathNode = (FILEPATH_DEVICE_PATH *) NextDevicePathNode(&FilePathNode->Header);
  118. }
  119. //
  120. // If success, return the FHand
  121. //
  122. if (!EFI_ERROR(Status)) {
  123. ASSERT(FileHandle);
  124. FHand->FileHandle = FileHandle;
  125. goto Done;
  126. }
  127. //
  128. // Cleanup from filesystem access
  129. //
  130. if (FileHandle) {
  131. uefi_call_wrapper(FileHandle->Close, 1, FileHandle);
  132. FileHandle = NULL;
  133. *FilePath = UserFilePath;
  134. }
  135. //
  136. // If the error is something other then unsupported, return it
  137. //
  138. if (Status != EFI_UNSUPPORTED) {
  139. goto Done;
  140. }
  141. //
  142. // Attempt to access the file via the load file protocol
  143. //
  144. Status = LibDevicePathToInterface (&LoadFileProtocol, *FilePath, (VOID*)&LoadFile);
  145. if (!EFI_ERROR(Status)) {
  146. TempFilePath = DuplicateDevicePath (*FilePath);
  147. TempFilePathPtr = TempFilePath;
  148. Status = uefi_call_wrapper(BS->LocateDevicePath, 3, &LoadFileProtocol, &TempFilePath, DeviceHandle);
  149. FreePool (TempFilePathPtr);
  150. //
  151. // Determine the size of buffer needed to hold the file
  152. //
  153. SourceSize = 0;
  154. Status = uefi_call_wrapper(
  155. LoadFile->LoadFile,
  156. 5,
  157. LoadFile,
  158. *FilePath,
  159. BootPolicy,
  160. &SourceSize,
  161. NULL
  162. );
  163. //
  164. // We expect a buffer too small error to inform us
  165. // of the buffer size needed
  166. //
  167. if (Status == EFI_BUFFER_TOO_SMALL) {
  168. SourceBuffer = AllocatePool (SourceSize);
  169. if (SourceBuffer) {
  170. FHand->FreeBuffer = TRUE;
  171. FHand->Source = SourceBuffer;
  172. FHand->SourceSize = SourceSize;
  173. Status = uefi_call_wrapper(
  174. LoadFile->LoadFile,
  175. 5,
  176. LoadFile,
  177. *FilePath,
  178. BootPolicy,
  179. &SourceSize,
  180. SourceBuffer
  181. );
  182. }
  183. }
  184. //
  185. // If success, return FHand
  186. //
  187. if (!EFI_ERROR(Status) || Status == EFI_ALREADY_STARTED) {
  188. goto Done;
  189. }
  190. }
  191. //
  192. // Nothing else to try
  193. //
  194. DEBUG ((D_LOAD|D_WARN, "OpenSimpleReadFile: Device did not support a known load protocol\n"));
  195. Status = EFI_UNSUPPORTED;
  196. Done:
  197. //
  198. // If the file was not accessed, clean up
  199. //
  200. if (EFI_ERROR(Status) && (Status != EFI_ALREADY_STARTED)) {
  201. if (FHand) {
  202. if (FHand->FreeBuffer) {
  203. FreePool (FHand->Source);
  204. }
  205. FreePool (FHand);
  206. }
  207. }
  208. return Status;
  209. }
  210. EFI_STATUS
  211. ReadSimpleReadFile (
  212. IN SIMPLE_READ_FILE UserHandle,
  213. IN UINTN Offset,
  214. IN OUT UINTN *ReadSize,
  215. OUT VOID *Buffer
  216. )
  217. {
  218. UINTN EndPos;
  219. SIMPLE_READ_HANDLE *FHand;
  220. EFI_STATUS Status;
  221. FHand = UserHandle;
  222. ASSERT (FHand->Signature == SIMPLE_READ_SIGNATURE);
  223. if (FHand->Source) {
  224. //
  225. // Move data from our local copy of the file
  226. //
  227. EndPos = Offset + *ReadSize;
  228. if (EndPos > FHand->SourceSize) {
  229. *ReadSize = FHand->SourceSize - Offset;
  230. if (Offset >= FHand->SourceSize) {
  231. *ReadSize = 0;
  232. }
  233. }
  234. CopyMem (Buffer, (CHAR8 *) FHand->Source + Offset, *ReadSize);
  235. Status = EFI_SUCCESS;
  236. } else {
  237. //
  238. // Read data from the file
  239. //
  240. Status = uefi_call_wrapper(FHand->FileHandle->SetPosition, 2, FHand->FileHandle, Offset);
  241. if (!EFI_ERROR(Status)) {
  242. Status = uefi_call_wrapper(FHand->FileHandle->Read, 3, FHand->FileHandle, ReadSize, Buffer);
  243. }
  244. }
  245. return Status;
  246. }
  247. VOID
  248. CloseSimpleReadFile (
  249. IN SIMPLE_READ_FILE UserHandle
  250. )
  251. {
  252. SIMPLE_READ_HANDLE *FHand;
  253. FHand = UserHandle;
  254. ASSERT (FHand->Signature == SIMPLE_READ_SIGNATURE);
  255. //
  256. // Free any file handle we opened
  257. //
  258. if (FHand->FileHandle) {
  259. uefi_call_wrapper(FHand->FileHandle->Close, 1, FHand->FileHandle);
  260. }
  261. //
  262. // If we allocated the Source buffer, free it
  263. //
  264. if (FHand->FreeBuffer) {
  265. FreePool (FHand->Source);
  266. }
  267. //
  268. // Done with this simple read file handle
  269. //
  270. FreePool (FHand);
  271. }