• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /** @file
2   DiskIo driver that lays on every BlockIo protocol in the system.
3   DiskIo converts a block oriented device to a byte oriented device.
4 
5   Disk access may have to handle unaligned request about sector boundaries.
6   There are three cases:
7     UnderRun - The first byte is not on a sector boundary or the read request is
8                less than a sector in length.
9     Aligned  - A read of N contiguous sectors.
10     OverRun  - The last byte is not on a sector boundary.
11 
12 Copyright (c) 2006 - 2016, Intel Corporation. All rights reserved.<BR>
13 This program and the accompanying materials
14 are licensed and made available under the terms and conditions of the BSD License
15 which accompanies this distribution.  The full text of the license may be found at
16 http://opensource.org/licenses/bsd-license.php
17 
18 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
19 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
20 
21 **/
22 
23 #include "DiskIo.h"
24 
25 //
26 // Driver binding protocol implementation for DiskIo driver.
27 //
28 EFI_DRIVER_BINDING_PROTOCOL gDiskIoDriverBinding = {
29   DiskIoDriverBindingSupported,
30   DiskIoDriverBindingStart,
31   DiskIoDriverBindingStop,
32   0xa,
33   NULL,
34   NULL
35 };
36 
37 //
38 // Template for DiskIo private data structure.
39 // The pointer to BlockIo protocol interface is assigned dynamically.
40 //
41 DISK_IO_PRIVATE_DATA        gDiskIoPrivateDataTemplate = {
42   DISK_IO_PRIVATE_DATA_SIGNATURE,
43   {
44     EFI_DISK_IO_PROTOCOL_REVISION,
45     DiskIoReadDisk,
46     DiskIoWriteDisk
47   },
48   {
49     EFI_DISK_IO2_PROTOCOL_REVISION,
50     DiskIo2Cancel,
51     DiskIo2ReadDiskEx,
52     DiskIo2WriteDiskEx,
53     DiskIo2FlushDiskEx
54   }
55 };
56 
57 /**
58   Test to see if this driver supports ControllerHandle.
59 
60   @param  This                Protocol instance pointer.
61   @param  ControllerHandle    Handle of device to test
62   @param  RemainingDevicePath Optional parameter use to pick a specific child
63                               device to start.
64 
65   @retval EFI_SUCCESS         This driver supports this device
66   @retval EFI_ALREADY_STARTED This driver is already running on this device
67   @retval other               This driver does not support this device
68 
69 **/
70 EFI_STATUS
71 EFIAPI
DiskIoDriverBindingSupported(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE ControllerHandle,IN EFI_DEVICE_PATH_PROTOCOL * RemainingDevicePath OPTIONAL)72 DiskIoDriverBindingSupported (
73   IN EFI_DRIVER_BINDING_PROTOCOL  *This,
74   IN EFI_HANDLE                   ControllerHandle,
75   IN EFI_DEVICE_PATH_PROTOCOL     *RemainingDevicePath OPTIONAL
76   )
77 {
78   EFI_STATUS            Status;
79   EFI_BLOCK_IO_PROTOCOL *BlockIo;
80 
81   //
82   // Open the IO Abstraction(s) needed to perform the supported test.
83   //
84   Status = gBS->OpenProtocol (
85                   ControllerHandle,
86                   &gEfiBlockIoProtocolGuid,
87                   (VOID **) &BlockIo,
88                   This->DriverBindingHandle,
89                   ControllerHandle,
90                   EFI_OPEN_PROTOCOL_BY_DRIVER
91                   );
92   if (EFI_ERROR (Status)) {
93     return Status;
94   }
95 
96   //
97   // Close the I/O Abstraction(s) used to perform the supported test.
98   //
99   gBS->CloseProtocol (
100          ControllerHandle,
101          &gEfiBlockIoProtocolGuid,
102          This->DriverBindingHandle,
103          ControllerHandle
104          );
105   return EFI_SUCCESS;
106 }
107 
108 
109 /**
110   Start this driver on ControllerHandle by opening a Block IO protocol and
111   installing a Disk IO protocol on ControllerHandle.
112 
113   @param  This                 Protocol instance pointer.
114   @param  ControllerHandle     Handle of device to bind driver to
115   @param  RemainingDevicePath  Optional parameter use to pick a specific child
116                                device to start.
117 
118   @retval EFI_SUCCESS          This driver is added to ControllerHandle
119   @retval EFI_ALREADY_STARTED  This driver is already running on ControllerHandle
120   @retval other                This driver does not support this device
121 
122 **/
123 EFI_STATUS
124 EFIAPI
DiskIoDriverBindingStart(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE ControllerHandle,IN EFI_DEVICE_PATH_PROTOCOL * RemainingDevicePath OPTIONAL)125 DiskIoDriverBindingStart (
126   IN EFI_DRIVER_BINDING_PROTOCOL  *This,
127   IN EFI_HANDLE                   ControllerHandle,
128   IN EFI_DEVICE_PATH_PROTOCOL     *RemainingDevicePath OPTIONAL
129   )
130 {
131   EFI_STATUS            Status;
132   DISK_IO_PRIVATE_DATA  *Instance;
133   EFI_TPL               OldTpl;
134 
135   Instance = NULL;
136 
137   OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
138 
139   //
140   // Connect to the Block IO and Block IO2 interface on ControllerHandle.
141   //
142   Status = gBS->OpenProtocol (
143                   ControllerHandle,
144                   &gEfiBlockIoProtocolGuid,
145                   (VOID **) &gDiskIoPrivateDataTemplate.BlockIo,
146                   This->DriverBindingHandle,
147                   ControllerHandle,
148                   EFI_OPEN_PROTOCOL_BY_DRIVER
149                   );
150   if (EFI_ERROR (Status)) {
151     goto ErrorExit1;
152   }
153 
154   Status = gBS->OpenProtocol (
155                   ControllerHandle,
156                   &gEfiBlockIo2ProtocolGuid,
157                   (VOID **) &gDiskIoPrivateDataTemplate.BlockIo2,
158                   This->DriverBindingHandle,
159                   ControllerHandle,
160                   EFI_OPEN_PROTOCOL_BY_DRIVER
161                   );
162   if (EFI_ERROR (Status)) {
163     gDiskIoPrivateDataTemplate.BlockIo2 = NULL;
164   }
165 
166   //
167   // Initialize the Disk IO device instance.
168   //
169   Instance = AllocateCopyPool (sizeof (DISK_IO_PRIVATE_DATA), &gDiskIoPrivateDataTemplate);
170   if (Instance == NULL) {
171     Status = EFI_OUT_OF_RESOURCES;
172     goto ErrorExit;
173   }
174 
175   //
176   // The BlockSize and IoAlign of BlockIo and BlockIo2 should equal.
177   //
178   ASSERT ((Instance->BlockIo2 == NULL) ||
179           ((Instance->BlockIo->Media->IoAlign == Instance->BlockIo2->Media->IoAlign) &&
180            (Instance->BlockIo->Media->BlockSize == Instance->BlockIo2->Media->BlockSize)
181           ));
182 
183   InitializeListHead (&Instance->TaskQueue);
184   EfiInitializeLock (&Instance->TaskQueueLock, TPL_NOTIFY);
185   Instance->SharedWorkingBuffer = AllocateAlignedPages (
186                                     EFI_SIZE_TO_PAGES (PcdGet32 (PcdDiskIoDataBufferBlockNum) * Instance->BlockIo->Media->BlockSize),
187                                     Instance->BlockIo->Media->IoAlign
188                                     );
189   if (Instance->SharedWorkingBuffer == NULL) {
190     Status = EFI_OUT_OF_RESOURCES;
191     goto ErrorExit;
192   }
193 
194   //
195   // Install protocol interfaces for the Disk IO device.
196   //
197   if (Instance->BlockIo2 != NULL) {
198     Status = gBS->InstallMultipleProtocolInterfaces (
199                     &ControllerHandle,
200                     &gEfiDiskIoProtocolGuid,  &Instance->DiskIo,
201                     &gEfiDiskIo2ProtocolGuid, &Instance->DiskIo2,
202                     NULL
203                     );
204   } else {
205     Status = gBS->InstallMultipleProtocolInterfaces (
206                     &ControllerHandle,
207                     &gEfiDiskIoProtocolGuid,  &Instance->DiskIo,
208                     NULL
209                     );
210   }
211 
212 ErrorExit:
213   if (EFI_ERROR (Status)) {
214     if (Instance != NULL && Instance->SharedWorkingBuffer != NULL) {
215       FreeAlignedPages (
216         Instance->SharedWorkingBuffer,
217         EFI_SIZE_TO_PAGES (PcdGet32 (PcdDiskIoDataBufferBlockNum) * Instance->BlockIo->Media->BlockSize)
218         );
219     }
220 
221     if (Instance != NULL) {
222       FreePool (Instance);
223     }
224 
225     gBS->CloseProtocol (
226            ControllerHandle,
227            &gEfiBlockIoProtocolGuid,
228            This->DriverBindingHandle,
229            ControllerHandle
230            );
231   }
232 
233 ErrorExit1:
234   gBS->RestoreTPL (OldTpl);
235   return Status;
236 }
237 
238 /**
239   Stop this driver on ControllerHandle by removing Disk IO protocol and closing
240   the Block IO protocol on ControllerHandle.
241 
242   @param  This              Protocol instance pointer.
243   @param  ControllerHandle  Handle of device to stop driver on
244   @param  NumberOfChildren  Number of Handles in ChildHandleBuffer. If number of
245                             children is zero stop the entire bus driver.
246   @param  ChildHandleBuffer List of Child Handles to Stop.
247 
248   @retval EFI_SUCCESS       This driver is removed ControllerHandle
249   @retval other             This driver was not removed from this device
250 
251 **/
252 EFI_STATUS
253 EFIAPI
DiskIoDriverBindingStop(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE ControllerHandle,IN UINTN NumberOfChildren,IN EFI_HANDLE * ChildHandleBuffer)254 DiskIoDriverBindingStop (
255   IN  EFI_DRIVER_BINDING_PROTOCOL    *This,
256   IN  EFI_HANDLE                     ControllerHandle,
257   IN  UINTN                          NumberOfChildren,
258   IN  EFI_HANDLE                     *ChildHandleBuffer
259   )
260 {
261   EFI_STATUS            Status;
262   EFI_DISK_IO_PROTOCOL  *DiskIo;
263   EFI_DISK_IO2_PROTOCOL *DiskIo2;
264   DISK_IO_PRIVATE_DATA  *Instance;
265   BOOLEAN               AllTaskDone;
266 
267   //
268   // Get our context back.
269   //
270   Status = gBS->OpenProtocol (
271                   ControllerHandle,
272                   &gEfiDiskIoProtocolGuid,
273                   (VOID **) &DiskIo,
274                   This->DriverBindingHandle,
275                   ControllerHandle,
276                   EFI_OPEN_PROTOCOL_GET_PROTOCOL
277                   );
278   if (EFI_ERROR (Status)) {
279     return Status;
280   }
281   Status = gBS->OpenProtocol (
282                   ControllerHandle,
283                   &gEfiDiskIo2ProtocolGuid,
284                   (VOID **) &DiskIo2,
285                   This->DriverBindingHandle,
286                   ControllerHandle,
287                   EFI_OPEN_PROTOCOL_GET_PROTOCOL
288                   );
289   if (EFI_ERROR (Status)) {
290     DiskIo2 = NULL;
291   }
292 
293   Instance = DISK_IO_PRIVATE_DATA_FROM_DISK_IO (DiskIo);
294 
295   if (DiskIo2 != NULL) {
296     //
297     // Call BlockIo2::Reset() to terminate any in-flight non-blocking I/O requests
298     //
299     ASSERT (Instance->BlockIo2 != NULL);
300     Status = Instance->BlockIo2->Reset (Instance->BlockIo2, FALSE);
301     if (EFI_ERROR (Status)) {
302       return Status;
303     }
304     Status = gBS->UninstallMultipleProtocolInterfaces (
305                     ControllerHandle,
306                     &gEfiDiskIoProtocolGuid,  &Instance->DiskIo,
307                     &gEfiDiskIo2ProtocolGuid, &Instance->DiskIo2,
308                     NULL
309                     );
310   } else {
311     Status = gBS->UninstallMultipleProtocolInterfaces (
312                     ControllerHandle,
313                     &gEfiDiskIoProtocolGuid,  &Instance->DiskIo,
314                     NULL
315                     );
316   }
317   if (!EFI_ERROR (Status)) {
318 
319     do {
320       EfiAcquireLock (&Instance->TaskQueueLock);
321       AllTaskDone = IsListEmpty (&Instance->TaskQueue);
322       EfiReleaseLock (&Instance->TaskQueueLock);
323     } while (!AllTaskDone);
324 
325     FreeAlignedPages (
326       Instance->SharedWorkingBuffer,
327       EFI_SIZE_TO_PAGES (PcdGet32 (PcdDiskIoDataBufferBlockNum) * Instance->BlockIo->Media->BlockSize)
328       );
329 
330     Status = gBS->CloseProtocol (
331                     ControllerHandle,
332                     &gEfiBlockIoProtocolGuid,
333                     This->DriverBindingHandle,
334                     ControllerHandle
335                     );
336     ASSERT_EFI_ERROR (Status);
337     if (DiskIo2 != NULL) {
338       Status = gBS->CloseProtocol (
339                       ControllerHandle,
340                       &gEfiBlockIo2ProtocolGuid,
341                       This->DriverBindingHandle,
342                       ControllerHandle
343                       );
344       ASSERT_EFI_ERROR (Status);
345     }
346 
347     FreePool (Instance);
348   }
349 
350   return Status;
351 }
352 
353 
354 /**
355   Destroy the sub task.
356 
357   @param Instance     Pointer to the DISK_IO_PRIVATE_DATA.
358   @param Subtask      Subtask.
359 
360   @return LIST_ENTRY *  Pointer to the next link of subtask.
361 **/
362 LIST_ENTRY *
DiskIoDestroySubtask(IN DISK_IO_PRIVATE_DATA * Instance,IN DISK_IO_SUBTASK * Subtask)363 DiskIoDestroySubtask (
364   IN DISK_IO_PRIVATE_DATA     *Instance,
365   IN DISK_IO_SUBTASK          *Subtask
366   )
367 {
368   LIST_ENTRY               *Link;
369 
370   if (Subtask->Task != NULL) {
371     EfiAcquireLock (&Subtask->Task->SubtasksLock);
372   }
373   Link = RemoveEntryList (&Subtask->Link);
374   if (Subtask->Task != NULL) {
375     EfiReleaseLock (&Subtask->Task->SubtasksLock);
376   }
377 
378   if (!Subtask->Blocking) {
379     if (Subtask->WorkingBuffer != NULL) {
380       FreeAlignedPages (
381         Subtask->WorkingBuffer,
382         Subtask->Length < Instance->BlockIo->Media->BlockSize
383         ? EFI_SIZE_TO_PAGES (Instance->BlockIo->Media->BlockSize)
384         : EFI_SIZE_TO_PAGES (Subtask->Length)
385         );
386     }
387     if (Subtask->BlockIo2Token.Event != NULL) {
388       gBS->CloseEvent (Subtask->BlockIo2Token.Event);
389     }
390   }
391   FreePool (Subtask);
392 
393   return Link;
394 }
395 
396 /**
397   The callback for the BlockIo2 ReadBlocksEx/WriteBlocksEx.
398   @param  Event                 Event whose notification function is being invoked.
399   @param  Context               The pointer to the notification function's context,
400                                 which points to the DISK_IO_SUBTASK instance.
401 **/
402 VOID
403 EFIAPI
DiskIo2OnReadWriteComplete(IN EFI_EVENT Event,IN VOID * Context)404 DiskIo2OnReadWriteComplete (
405   IN EFI_EVENT            Event,
406   IN VOID                 *Context
407   )
408 {
409   DISK_IO_SUBTASK       *Subtask;
410   DISK_IO2_TASK         *Task;
411   EFI_STATUS            TransactionStatus;
412   DISK_IO_PRIVATE_DATA  *Instance;
413 
414   Subtask           = (DISK_IO_SUBTASK *) Context;
415   TransactionStatus = Subtask->BlockIo2Token.TransactionStatus;
416   Task              = Subtask->Task;
417   Instance          = Task->Instance;
418 
419   ASSERT (Subtask->Signature  == DISK_IO_SUBTASK_SIGNATURE);
420   ASSERT (Instance->Signature == DISK_IO_PRIVATE_DATA_SIGNATURE);
421   ASSERT (Task->Signature     == DISK_IO2_TASK_SIGNATURE);
422 
423   if ((Subtask->WorkingBuffer != NULL) && !EFI_ERROR (TransactionStatus) &&
424       (Task->Token != NULL) && !Subtask->Write
425      ) {
426     CopyMem (Subtask->Buffer, Subtask->WorkingBuffer + Subtask->Offset, Subtask->Length);
427   }
428 
429   DiskIoDestroySubtask (Instance, Subtask);
430 
431   if (EFI_ERROR (TransactionStatus) || IsListEmpty (&Task->Subtasks)) {
432     if (Task->Token != NULL) {
433       //
434       // Signal error status once the subtask is failed.
435       // Or signal the last status once the last subtask is finished.
436       //
437       Task->Token->TransactionStatus = TransactionStatus;
438       gBS->SignalEvent (Task->Token->Event);
439 
440       //
441       // Mark token to NULL indicating the Task is a dead task.
442       //
443       Task->Token = NULL;
444     }
445   }
446 }
447 
448 /**
449   Create the subtask.
450 
451   @param Write         TRUE: Write request; FALSE: Read request.
452   @param Lba           The starting logical block address to read from on the device.
453   @param Offset        The starting byte offset to read from the LBA.
454   @param Length        The number of bytes to read from the device.
455   @param WorkingBuffer The aligned buffer to hold the data for reading or writing.
456   @param Buffer        The buffer to hold the data for reading or writing.
457   @param Blocking      TRUE: Blocking request; FALSE: Non-blocking request.
458 
459   @return A pointer to the created subtask.
460 **/
461 DISK_IO_SUBTASK *
DiskIoCreateSubtask(IN BOOLEAN Write,IN UINT64 Lba,IN UINT32 Offset,IN UINTN Length,IN VOID * WorkingBuffer,OPTIONAL IN VOID * Buffer,IN BOOLEAN Blocking)462 DiskIoCreateSubtask (
463   IN BOOLEAN          Write,
464   IN UINT64           Lba,
465   IN UINT32           Offset,
466   IN UINTN            Length,
467   IN VOID             *WorkingBuffer,  OPTIONAL
468   IN VOID             *Buffer,
469   IN BOOLEAN          Blocking
470   )
471 {
472   DISK_IO_SUBTASK       *Subtask;
473   EFI_STATUS            Status;
474 
475   Subtask = AllocateZeroPool (sizeof (DISK_IO_SUBTASK));
476   if (Subtask == NULL) {
477     return NULL;
478   }
479   Subtask->Signature     = DISK_IO_SUBTASK_SIGNATURE;
480   Subtask->Write         = Write;
481   Subtask->Lba           = Lba;
482   Subtask->Offset        = Offset;
483   Subtask->Length        = Length;
484   Subtask->WorkingBuffer = WorkingBuffer;
485   Subtask->Buffer        = Buffer;
486   Subtask->Blocking      = Blocking;
487   if (!Blocking) {
488     Status = gBS->CreateEvent (
489                     EVT_NOTIFY_SIGNAL,
490                     TPL_NOTIFY,
491                     DiskIo2OnReadWriteComplete,
492                     Subtask,
493                     &Subtask->BlockIo2Token.Event
494                     );
495     if (EFI_ERROR (Status)) {
496       FreePool (Subtask);
497       return NULL;
498     }
499   }
500   DEBUG ((
501     EFI_D_BLKIO,
502     "  %c:Lba/Offset/Length/WorkingBuffer/Buffer = %016lx/%08x/%08x/%08x/%08x\n",
503     Write ? 'W': 'R', Lba, Offset, Length, WorkingBuffer, Buffer
504     ));
505 
506   return Subtask;
507 }
508 
509 /**
510   Create the subtask list.
511 
512   @param Instance            Pointer to the DISK_IO_PRIVATE_DATA.
513   @param Write               TRUE: Write request; FALSE: Read request.
514   @param Offset              The starting byte offset to read from the device.
515   @param BufferSize          The size in bytes of Buffer. The number of bytes to read from the device.
516   @param Buffer              A pointer to the buffer for the data.
517   @param Blocking            TRUE: Blocking request; FALSE: Non-blocking request.
518   @param SharedWorkingBuffer The aligned buffer to hold the data for reading or writing.
519   @param Subtasks            The subtask list header.
520 
521   @retval TRUE  The subtask list is created successfully.
522   @retval FALSE The subtask list is not created.
523 **/
524 BOOLEAN
DiskIoCreateSubtaskList(IN DISK_IO_PRIVATE_DATA * Instance,IN BOOLEAN Write,IN UINT64 Offset,IN UINTN BufferSize,IN VOID * Buffer,IN BOOLEAN Blocking,IN VOID * SharedWorkingBuffer,IN OUT LIST_ENTRY * Subtasks)525 DiskIoCreateSubtaskList (
526   IN DISK_IO_PRIVATE_DATA  *Instance,
527   IN BOOLEAN               Write,
528   IN UINT64                Offset,
529   IN UINTN                 BufferSize,
530   IN VOID                  *Buffer,
531   IN BOOLEAN               Blocking,
532   IN VOID                  *SharedWorkingBuffer,
533   IN OUT LIST_ENTRY        *Subtasks
534   )
535 {
536   UINT32                BlockSize;
537   UINT32                IoAlign;
538   UINT64                Lba;
539   UINT64                OverRunLba;
540   UINT32                UnderRun;
541   UINT32                OverRun;
542   UINT8                 *BufferPtr;
543   UINTN                 Length;
544   UINTN                 DataBufferSize;
545   DISK_IO_SUBTASK       *Subtask;
546   VOID                  *WorkingBuffer;
547   LIST_ENTRY            *Link;
548 
549   DEBUG ((EFI_D_BLKIO, "DiskIo: Create subtasks for task: Offset/BufferSize/Buffer = %016lx/%08x/%08x\n", Offset, BufferSize, Buffer));
550 
551   BlockSize = Instance->BlockIo->Media->BlockSize;
552   IoAlign   = Instance->BlockIo->Media->IoAlign;
553   if (IoAlign == 0) {
554     IoAlign = 1;
555   }
556 
557   Lba       = DivU64x32Remainder (Offset, BlockSize, &UnderRun);
558   BufferPtr = (UINT8 *) Buffer;
559 
560   //
561   // Special handling for zero BufferSize
562   //
563   if (BufferSize == 0) {
564     Subtask = DiskIoCreateSubtask (Write, Lba, UnderRun, 0, NULL, BufferPtr, Blocking);
565     if (Subtask == NULL) {
566       goto Done;
567     }
568     InsertTailList (Subtasks, &Subtask->Link);
569     return TRUE;
570   }
571 
572   if (UnderRun != 0) {
573     Length = MIN (BlockSize - UnderRun, BufferSize);
574     if (Blocking) {
575       WorkingBuffer = SharedWorkingBuffer;
576     } else {
577       WorkingBuffer = AllocateAlignedPages (EFI_SIZE_TO_PAGES (BlockSize), IoAlign);
578       if (WorkingBuffer == NULL) {
579         goto Done;
580       }
581     }
582     if (Write) {
583       //
584       // A half write operation can be splitted to a blocking block-read and half write operation
585       // This can simplify the sub task processing logic
586       //
587       Subtask = DiskIoCreateSubtask (FALSE, Lba, 0, BlockSize, NULL, WorkingBuffer, TRUE);
588       if (Subtask == NULL) {
589         goto Done;
590       }
591       InsertTailList (Subtasks, &Subtask->Link);
592     }
593 
594     Subtask = DiskIoCreateSubtask (Write, Lba, UnderRun, Length, WorkingBuffer, BufferPtr, Blocking);
595     if (Subtask == NULL) {
596       goto Done;
597     }
598     InsertTailList (Subtasks, &Subtask->Link);
599 
600     BufferPtr  += Length;
601     Offset     += Length;
602     BufferSize -= Length;
603     Lba ++;
604   }
605 
606   OverRunLba  = Lba + DivU64x32Remainder (BufferSize, BlockSize, &OverRun);
607   BufferSize -= OverRun;
608 
609   if (OverRun != 0) {
610     if (Blocking) {
611       WorkingBuffer = SharedWorkingBuffer;
612     } else {
613       WorkingBuffer = AllocateAlignedPages (EFI_SIZE_TO_PAGES (BlockSize), IoAlign);
614       if (WorkingBuffer == NULL) {
615         goto Done;
616       }
617     }
618     if (Write) {
619       //
620       // A half write operation can be splitted to a blocking block-read and half write operation
621       // This can simplify the sub task processing logic
622       //
623       Subtask = DiskIoCreateSubtask (FALSE, OverRunLba, 0, BlockSize, NULL, WorkingBuffer, TRUE);
624       if (Subtask == NULL) {
625         goto Done;
626       }
627       InsertTailList (Subtasks, &Subtask->Link);
628     }
629 
630     Subtask = DiskIoCreateSubtask (Write, OverRunLba, 0, OverRun, WorkingBuffer, BufferPtr + BufferSize, Blocking);
631     if (Subtask == NULL) {
632       goto Done;
633     }
634     InsertTailList (Subtasks, &Subtask->Link);
635   }
636 
637   if (OverRunLba > Lba) {
638     //
639     // If the DiskIo maps directly to a BlockIo device do the read.
640     //
641     if (ALIGN_POINTER (BufferPtr, IoAlign) == BufferPtr) {
642       Subtask = DiskIoCreateSubtask (Write, Lba, 0, BufferSize, NULL, BufferPtr, Blocking);
643       if (Subtask == NULL) {
644         goto Done;
645       }
646       InsertTailList (Subtasks, &Subtask->Link);
647 
648       BufferPtr  += BufferSize;
649       Offset     += BufferSize;
650       BufferSize -= BufferSize;
651 
652     } else {
653       if (Blocking) {
654         //
655         // Use the allocated buffer instead of the original buffer
656         // to avoid alignment issue.
657         //
658         for (; Lba < OverRunLba; Lba += PcdGet32 (PcdDiskIoDataBufferBlockNum)) {
659           DataBufferSize = MIN (BufferSize, PcdGet32 (PcdDiskIoDataBufferBlockNum) * BlockSize);
660 
661           Subtask = DiskIoCreateSubtask (Write, Lba, 0, DataBufferSize, SharedWorkingBuffer, BufferPtr, Blocking);
662           if (Subtask == NULL) {
663             goto Done;
664           }
665           InsertTailList (Subtasks, &Subtask->Link);
666 
667           BufferPtr  += DataBufferSize;
668           Offset     += DataBufferSize;
669           BufferSize -= DataBufferSize;
670         }
671       } else {
672         WorkingBuffer = AllocateAlignedPages (EFI_SIZE_TO_PAGES (BufferSize), IoAlign);
673         if (WorkingBuffer == NULL) {
674           //
675           // If there is not enough memory, downgrade to blocking access
676           //
677           DEBUG ((EFI_D_VERBOSE, "DiskIo: No enough memory so downgrade to blocking access\n"));
678           if (!DiskIoCreateSubtaskList (Instance, Write, Offset, BufferSize, BufferPtr, TRUE, SharedWorkingBuffer, Subtasks)) {
679             goto Done;
680           }
681         } else {
682           Subtask = DiskIoCreateSubtask (Write, Lba, 0, BufferSize, WorkingBuffer, BufferPtr, Blocking);
683           if (Subtask == NULL) {
684             goto Done;
685           }
686           InsertTailList (Subtasks, &Subtask->Link);
687         }
688 
689         BufferPtr  += BufferSize;
690         Offset     += BufferSize;
691         BufferSize -= BufferSize;
692       }
693     }
694   }
695 
696   ASSERT (BufferSize == 0);
697 
698   return TRUE;
699 
700 Done:
701   //
702   // Remove all the subtasks.
703   //
704   for (Link = GetFirstNode (Subtasks); !IsNull (Subtasks, Link); ) {
705     Subtask = CR (Link, DISK_IO_SUBTASK, Link, DISK_IO_SUBTASK_SIGNATURE);
706     Link = DiskIoDestroySubtask (Instance, Subtask);
707   }
708   return FALSE;
709 }
710 
711 /**
712   Terminate outstanding asynchronous requests to a device.
713 
714   @param This                   Indicates a pointer to the calling context.
715 
716   @retval EFI_SUCCESS           All outstanding requests were successfully terminated.
717   @retval EFI_DEVICE_ERROR      The device reported an error while performing the cancel
718                                 operation.
719 **/
720 EFI_STATUS
721 EFIAPI
DiskIo2Cancel(IN EFI_DISK_IO2_PROTOCOL * This)722 DiskIo2Cancel (
723   IN EFI_DISK_IO2_PROTOCOL *This
724   )
725 {
726   DISK_IO_PRIVATE_DATA  *Instance;
727   DISK_IO2_TASK         *Task;
728   LIST_ENTRY            *Link;
729 
730   Instance = DISK_IO_PRIVATE_DATA_FROM_DISK_IO2 (This);
731 
732   EfiAcquireLock (&Instance->TaskQueueLock);
733 
734   for (Link = GetFirstNode (&Instance->TaskQueue)
735     ; !IsNull (&Instance->TaskQueue, Link)
736     ; Link = GetNextNode (&Instance->TaskQueue, Link)
737     ) {
738     Task = CR (Link, DISK_IO2_TASK, Link, DISK_IO2_TASK_SIGNATURE);
739 
740     if (Task->Token != NULL) {
741       Task->Token->TransactionStatus = EFI_ABORTED;
742       gBS->SignalEvent (Task->Token->Event);
743       //
744       // Set Token to NULL so that the further BlockIo2 responses will be ignored
745       //
746       Task->Token = NULL;
747     }
748   }
749 
750   EfiReleaseLock (&Instance->TaskQueueLock);
751 
752   return EFI_SUCCESS;
753 }
754 
755 /**
756   Remove the completed tasks from Instance->TaskQueue. Completed tasks are those who don't have any subtasks.
757 
758   @param Instance    Pointer to the DISK_IO_PRIVATE_DATA.
759 
760   @retval TRUE       The Instance->TaskQueue is empty after the completed tasks are removed.
761   @retval FALSE      The Instance->TaskQueue is not empty after the completed tasks are removed.
762 **/
763 BOOLEAN
DiskIo2RemoveCompletedTask(IN DISK_IO_PRIVATE_DATA * Instance)764 DiskIo2RemoveCompletedTask (
765   IN DISK_IO_PRIVATE_DATA     *Instance
766   )
767 {
768   BOOLEAN                     QueueEmpty;
769   LIST_ENTRY                  *Link;
770   DISK_IO2_TASK               *Task;
771 
772   QueueEmpty = TRUE;
773 
774   EfiAcquireLock (&Instance->TaskQueueLock);
775   for (Link = GetFirstNode (&Instance->TaskQueue); !IsNull (&Instance->TaskQueue, Link); ) {
776     Task = CR (Link, DISK_IO2_TASK, Link, DISK_IO2_TASK_SIGNATURE);
777     if (IsListEmpty (&Task->Subtasks)) {
778       Link = RemoveEntryList (&Task->Link);
779       ASSERT (Task->Token == NULL);
780       FreePool (Task);
781     } else {
782       Link = GetNextNode (&Instance->TaskQueue, Link);
783       QueueEmpty = FALSE;
784     }
785   }
786   EfiReleaseLock (&Instance->TaskQueueLock);
787 
788   return QueueEmpty;
789 }
790 
791 /**
792   Common routine to access the disk.
793 
794   @param Instance    Pointer to the DISK_IO_PRIVATE_DATA.
795   @param Write       TRUE: Write operation; FALSE: Read operation.
796   @param MediaId     ID of the medium to access.
797   @param Offset      The starting byte offset on the logical block I/O device to access.
798   @param Token       A pointer to the token associated with the transaction.
799                      If this field is NULL, synchronous/blocking IO is performed.
800   @param  BufferSize            The size in bytes of Buffer. The number of bytes to read from the device.
801   @param  Buffer                A pointer to the destination buffer for the data.
802                                 The caller is responsible either having implicit or explicit ownership of the buffer.
803 **/
804 EFI_STATUS
DiskIo2ReadWriteDisk(IN DISK_IO_PRIVATE_DATA * Instance,IN BOOLEAN Write,IN UINT32 MediaId,IN UINT64 Offset,IN EFI_DISK_IO2_TOKEN * Token,IN UINTN BufferSize,IN UINT8 * Buffer)805 DiskIo2ReadWriteDisk (
806   IN DISK_IO_PRIVATE_DATA     *Instance,
807   IN BOOLEAN                  Write,
808   IN UINT32                   MediaId,
809   IN UINT64                   Offset,
810   IN EFI_DISK_IO2_TOKEN       *Token,
811   IN UINTN                    BufferSize,
812   IN UINT8                    *Buffer
813   )
814 {
815   EFI_STATUS             Status;
816   EFI_BLOCK_IO_PROTOCOL  *BlockIo;
817   EFI_BLOCK_IO2_PROTOCOL *BlockIo2;
818   EFI_BLOCK_IO_MEDIA     *Media;
819   LIST_ENTRY             *Link;
820   LIST_ENTRY             *NextLink;
821   LIST_ENTRY             Subtasks;
822   DISK_IO_SUBTASK        *Subtask;
823   DISK_IO2_TASK          *Task;
824   EFI_TPL                OldTpl;
825   BOOLEAN                Blocking;
826   BOOLEAN                SubtaskBlocking;
827   LIST_ENTRY             *SubtasksPtr;
828 
829   Task      = NULL;
830   BlockIo   = Instance->BlockIo;
831   BlockIo2  = Instance->BlockIo2;
832   Media     = BlockIo->Media;
833   Status    = EFI_SUCCESS;
834   Blocking  = (BOOLEAN) ((Token == NULL) || (Token->Event == NULL));
835 
836   if (Blocking) {
837     //
838     // Wait till pending async task is completed.
839     //
840     while (!DiskIo2RemoveCompletedTask (Instance));
841 
842     SubtasksPtr = &Subtasks;
843   } else {
844     DiskIo2RemoveCompletedTask (Instance);
845     Task = AllocatePool (sizeof (DISK_IO2_TASK));
846     if (Task == NULL) {
847       return EFI_OUT_OF_RESOURCES;
848     }
849 
850     EfiAcquireLock (&Instance->TaskQueueLock);
851     InsertTailList (&Instance->TaskQueue, &Task->Link);
852     EfiReleaseLock (&Instance->TaskQueueLock);
853 
854     Task->Signature = DISK_IO2_TASK_SIGNATURE;
855     Task->Instance  = Instance;
856     Task->Token     = Token;
857     EfiInitializeLock (&Task->SubtasksLock, TPL_NOTIFY);
858 
859     SubtasksPtr = &Task->Subtasks;
860   }
861 
862   InitializeListHead (SubtasksPtr);
863   if (!DiskIoCreateSubtaskList (Instance, Write, Offset, BufferSize, Buffer, Blocking, Instance->SharedWorkingBuffer, SubtasksPtr)) {
864     if (Task != NULL) {
865       FreePool (Task);
866     }
867     return EFI_OUT_OF_RESOURCES;
868   }
869   ASSERT (!IsListEmpty (SubtasksPtr));
870 
871   OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
872   for ( Link = GetFirstNode (SubtasksPtr), NextLink = GetNextNode (SubtasksPtr, Link)
873       ; !IsNull (SubtasksPtr, Link)
874       ; Link = NextLink, NextLink = GetNextNode (SubtasksPtr, NextLink)
875       ) {
876     Subtask         = CR (Link, DISK_IO_SUBTASK, Link, DISK_IO_SUBTASK_SIGNATURE);
877     Subtask->Task   = Task;
878     SubtaskBlocking = Subtask->Blocking;
879 
880     ASSERT ((Subtask->Length % Media->BlockSize == 0) || (Subtask->Length < Media->BlockSize));
881 
882     if (Subtask->Write) {
883       //
884       // Write
885       //
886       if (Subtask->WorkingBuffer != NULL) {
887         //
888         // A sub task before this one should be a block read operation, causing the WorkingBuffer filled with the entire one block data.
889         //
890         CopyMem (Subtask->WorkingBuffer + Subtask->Offset, Subtask->Buffer, Subtask->Length);
891       }
892 
893       if (SubtaskBlocking) {
894         Status = BlockIo->WriteBlocks (
895                             BlockIo,
896                             MediaId,
897                             Subtask->Lba,
898                             (Subtask->Length % Media->BlockSize == 0) ? Subtask->Length : Media->BlockSize,
899                             (Subtask->WorkingBuffer != NULL) ? Subtask->WorkingBuffer : Subtask->Buffer
900                             );
901       } else {
902         Status = BlockIo2->WriteBlocksEx (
903                              BlockIo2,
904                              MediaId,
905                              Subtask->Lba,
906                              &Subtask->BlockIo2Token,
907                              (Subtask->Length % Media->BlockSize == 0) ? Subtask->Length : Media->BlockSize,
908                              (Subtask->WorkingBuffer != NULL) ? Subtask->WorkingBuffer : Subtask->Buffer
909                              );
910       }
911 
912     } else {
913       //
914       // Read
915       //
916       if (SubtaskBlocking) {
917         Status = BlockIo->ReadBlocks (
918                             BlockIo,
919                             MediaId,
920                             Subtask->Lba,
921                             (Subtask->Length % Media->BlockSize == 0) ? Subtask->Length : Media->BlockSize,
922                             (Subtask->WorkingBuffer != NULL) ? Subtask->WorkingBuffer : Subtask->Buffer
923                             );
924         if (!EFI_ERROR (Status) && (Subtask->WorkingBuffer != NULL)) {
925           CopyMem (Subtask->Buffer, Subtask->WorkingBuffer + Subtask->Offset, Subtask->Length);
926         }
927       } else {
928         Status = BlockIo2->ReadBlocksEx (
929                              BlockIo2,
930                              MediaId,
931                              Subtask->Lba,
932                              &Subtask->BlockIo2Token,
933                              (Subtask->Length % Media->BlockSize == 0) ? Subtask->Length : Media->BlockSize,
934                              (Subtask->WorkingBuffer != NULL) ? Subtask->WorkingBuffer : Subtask->Buffer
935                              );
936       }
937     }
938 
939     if (SubtaskBlocking || EFI_ERROR (Status)) {
940       //
941       // Make sure the subtask list only contains non-blocking subtasks.
942       // Remove failed non-blocking subtasks as well because the callback won't be called.
943       //
944       DiskIoDestroySubtask (Instance, Subtask);
945     }
946 
947     if (EFI_ERROR (Status)) {
948       break;
949     }
950   }
951 
952   gBS->RaiseTPL (TPL_NOTIFY);
953 
954   //
955   // Remove all the remaining subtasks when failure.
956   // We shouldn't remove all the tasks because the non-blocking requests have been submitted and cannot be canceled.
957   //
958   if (EFI_ERROR (Status)) {
959     while (!IsNull (SubtasksPtr, NextLink)) {
960       Subtask = CR (NextLink, DISK_IO_SUBTASK, Link, DISK_IO_SUBTASK_SIGNATURE);
961       NextLink = DiskIoDestroySubtask (Instance, Subtask);
962     }
963   }
964 
965   //
966   // It's possible that the non-blocking subtasks finish before raising TPL to NOTIFY,
967   // so the subtasks list might be empty at this point.
968   //
969   if (!Blocking && IsListEmpty (SubtasksPtr)) {
970     EfiAcquireLock (&Instance->TaskQueueLock);
971     RemoveEntryList (&Task->Link);
972     EfiReleaseLock (&Instance->TaskQueueLock);
973 
974     if (!EFI_ERROR (Status) && (Task->Token != NULL)) {
975       //
976       // Task->Token should be set to NULL by the DiskIo2OnReadWriteComplete
977       // It it's not, that means the non-blocking request was downgraded to blocking request.
978       //
979       DEBUG ((EFI_D_VERBOSE, "DiskIo: Non-blocking request was downgraded to blocking request, signal event directly.\n"));
980       Task->Token->TransactionStatus = Status;
981       gBS->SignalEvent (Task->Token->Event);
982     }
983 
984     FreePool (Task);
985   }
986 
987   gBS->RestoreTPL (OldTpl);
988 
989   return Status;
990 }
991 
992 /**
993   Reads a specified number of bytes from a device.
994 
995   @param This                   Indicates a pointer to the calling context.
996   @param MediaId                ID of the medium to be read.
997   @param Offset                 The starting byte offset on the logical block I/O device to read from.
998   @param Token                  A pointer to the token associated with the transaction.
999                                 If this field is NULL, synchronous/blocking IO is performed.
1000   @param  BufferSize            The size in bytes of Buffer. The number of bytes to read from the device.
1001   @param  Buffer                A pointer to the destination buffer for the data.
1002                                 The caller is responsible either having implicit or explicit ownership of the buffer.
1003 
1004   @retval EFI_SUCCESS           If Event is NULL (blocking I/O): The data was read correctly from the device.
1005                                 If Event is not NULL (asynchronous I/O): The request was successfully queued for processing.
1006                                                                          Event will be signaled upon completion.
1007   @retval EFI_DEVICE_ERROR      The device reported an error while performing the write.
1008   @retval EFI_NO_MEDIA          There is no medium in the device.
1009   @retval EFI_MEDIA_CHNAGED     The MediaId is not for the current medium.
1010   @retval EFI_INVALID_PARAMETER The read request contains device addresses that are not valid for the device.
1011   @retval EFI_OUT_OF_RESOURCES  The request could not be completed due to a lack of resources.
1012 
1013 **/
1014 EFI_STATUS
1015 EFIAPI
DiskIo2ReadDiskEx(IN EFI_DISK_IO2_PROTOCOL * This,IN UINT32 MediaId,IN UINT64 Offset,IN OUT EFI_DISK_IO2_TOKEN * Token,IN UINTN BufferSize,OUT VOID * Buffer)1016 DiskIo2ReadDiskEx (
1017   IN EFI_DISK_IO2_PROTOCOL        *This,
1018   IN UINT32                       MediaId,
1019   IN UINT64                       Offset,
1020   IN OUT EFI_DISK_IO2_TOKEN       *Token,
1021   IN UINTN                        BufferSize,
1022   OUT VOID                        *Buffer
1023   )
1024 {
1025   return DiskIo2ReadWriteDisk (
1026            DISK_IO_PRIVATE_DATA_FROM_DISK_IO2 (This),
1027            FALSE, MediaId, Offset, Token, BufferSize, (UINT8 *) Buffer
1028            );
1029 }
1030 
1031 /**
1032   Writes a specified number of bytes to a device.
1033 
1034   @param This        Indicates a pointer to the calling context.
1035   @param MediaId     ID of the medium to be written.
1036   @param Offset      The starting byte offset on the logical block I/O device to write to.
1037   @param Token       A pointer to the token associated with the transaction.
1038                      If this field is NULL, synchronous/blocking IO is performed.
1039   @param BufferSize  The size in bytes of Buffer. The number of bytes to write to the device.
1040   @param Buffer      A pointer to the buffer containing the data to be written.
1041 
1042   @retval EFI_SUCCESS           If Event is NULL (blocking I/O): The data was written correctly to the device.
1043                                 If Event is not NULL (asynchronous I/O): The request was successfully queued for processing.
1044                                                                          Event will be signaled upon completion.
1045   @retval EFI_WRITE_PROTECTED   The device cannot be written to.
1046   @retval EFI_DEVICE_ERROR      The device reported an error while performing the write operation.
1047   @retval EFI_NO_MEDIA          There is no medium in the device.
1048   @retval EFI_MEDIA_CHNAGED     The MediaId is not for the current medium.
1049   @retval EFI_INVALID_PARAMETER The write request contains device addresses that are not valid for the device.
1050   @retval EFI_OUT_OF_RESOURCES  The request could not be completed due to a lack of resources.
1051 
1052 **/
1053 EFI_STATUS
1054 EFIAPI
DiskIo2WriteDiskEx(IN EFI_DISK_IO2_PROTOCOL * This,IN UINT32 MediaId,IN UINT64 Offset,IN OUT EFI_DISK_IO2_TOKEN * Token,IN UINTN BufferSize,IN VOID * Buffer)1055 DiskIo2WriteDiskEx (
1056   IN EFI_DISK_IO2_PROTOCOL        *This,
1057   IN UINT32                       MediaId,
1058   IN UINT64                       Offset,
1059   IN OUT EFI_DISK_IO2_TOKEN       *Token,
1060   IN UINTN                        BufferSize,
1061   IN VOID                         *Buffer
1062   )
1063 {
1064   return DiskIo2ReadWriteDisk (
1065            DISK_IO_PRIVATE_DATA_FROM_DISK_IO2 (This),
1066            TRUE, MediaId, Offset, Token, BufferSize, (UINT8 *) Buffer
1067            );
1068 }
1069 
1070 /**
1071   The callback for the BlockIo2 FlushBlocksEx.
1072   @param  Event                 Event whose notification function is being invoked.
1073   @param  Context               The pointer to the notification function's context,
1074                                 which points to the DISK_IO2_FLUSH_TASK instance.
1075 **/
1076 VOID
1077 EFIAPI
DiskIo2OnFlushComplete(IN EFI_EVENT Event,IN VOID * Context)1078 DiskIo2OnFlushComplete (
1079   IN EFI_EVENT                 Event,
1080   IN VOID                      *Context
1081   )
1082 {
1083   DISK_IO2_FLUSH_TASK             *Task;
1084 
1085   gBS->CloseEvent (Event);
1086 
1087   Task = (DISK_IO2_FLUSH_TASK *) Context;
1088   ASSERT (Task->Signature == DISK_IO2_FLUSH_TASK_SIGNATURE);
1089   Task->Token->TransactionStatus = Task->BlockIo2Token.TransactionStatus;
1090   gBS->SignalEvent (Task->Token->Event);
1091 
1092   FreePool (Task);
1093 }
1094 
1095 /**
1096   Flushes all modified data to the physical device.
1097 
1098   @param This        Indicates a pointer to the calling context.
1099   @param Token       A pointer to the token associated with the transaction.
1100                      If this field is NULL, synchronous/blocking IO is performed.
1101 
1102   @retval EFI_SUCCESS           If Event is NULL (blocking I/O): The data was flushed successfully to the device.
1103                                 If Event is not NULL (asynchronous I/O): The request was successfully queued for processing.
1104                                                                          Event will be signaled upon completion.
1105   @retval EFI_WRITE_PROTECTED   The device cannot be written to.
1106   @retval EFI_DEVICE_ERROR      The device reported an error while performing the write operation.
1107   @retval EFI_NO_MEDIA          There is no medium in the device.
1108   @retval EFI_OUT_OF_RESOURCES  The request could not be completed due to a lack of resources.
1109 **/
1110 EFI_STATUS
1111 EFIAPI
DiskIo2FlushDiskEx(IN EFI_DISK_IO2_PROTOCOL * This,IN OUT EFI_DISK_IO2_TOKEN * Token)1112 DiskIo2FlushDiskEx (
1113   IN EFI_DISK_IO2_PROTOCOL        *This,
1114   IN OUT EFI_DISK_IO2_TOKEN       *Token
1115   )
1116 {
1117   EFI_STATUS                      Status;
1118   DISK_IO2_FLUSH_TASK             *Task;
1119   DISK_IO_PRIVATE_DATA            *Private;
1120 
1121   Private = DISK_IO_PRIVATE_DATA_FROM_DISK_IO2 (This);
1122 
1123   if ((Token != NULL) && (Token->Event != NULL)) {
1124     Task = AllocatePool (sizeof (DISK_IO2_FLUSH_TASK));
1125     if (Task == NULL) {
1126       return EFI_OUT_OF_RESOURCES;
1127     }
1128 
1129     Status = gBS->CreateEvent (
1130                     EVT_NOTIFY_SIGNAL,
1131                     TPL_CALLBACK,
1132                     DiskIo2OnFlushComplete,
1133                     Task,
1134                     &Task->BlockIo2Token.Event
1135                     );
1136     if (EFI_ERROR (Status)) {
1137       FreePool (Task);
1138       return Status;
1139     }
1140     Task->Signature = DISK_IO2_FLUSH_TASK_SIGNATURE;
1141     Task->Token     = Token;
1142     Status = Private->BlockIo2->FlushBlocksEx (Private->BlockIo2, &Task->BlockIo2Token);
1143     if (EFI_ERROR (Status)) {
1144       gBS->CloseEvent (Task->BlockIo2Token.Event);
1145       FreePool (Task);
1146     }
1147   } else {
1148     Status = Private->BlockIo2->FlushBlocksEx (Private->BlockIo2, NULL);
1149   }
1150 
1151   return Status;
1152 }
1153 
1154 /**
1155   Read BufferSize bytes from Offset into Buffer.
1156   Reads may support reads that are not aligned on
1157   sector boundaries. There are three cases:
1158     UnderRun - The first byte is not on a sector boundary or the read request is
1159                less than a sector in length.
1160     Aligned  - A read of N contiguous sectors.
1161     OverRun  - The last byte is not on a sector boundary.
1162 
1163   @param  This                  Protocol instance pointer.
1164   @param  MediaId               Id of the media, changes every time the media is replaced.
1165   @param  Offset                The starting byte offset to read from
1166   @param  BufferSize            Size of Buffer
1167   @param  Buffer                Buffer containing read data
1168 
1169   @retval EFI_SUCCESS           The data was read correctly from the device.
1170   @retval EFI_DEVICE_ERROR      The device reported an error while performing the read.
1171   @retval EFI_NO_MEDIA          There is no media in the device.
1172   @retval EFI_MEDIA_CHNAGED     The MediaId does not matched the current device.
1173   @retval EFI_INVALID_PARAMETER The read request contains device addresses that are not
1174                                 valid for the device.
1175 
1176 **/
1177 EFI_STATUS
1178 EFIAPI
DiskIoReadDisk(IN EFI_DISK_IO_PROTOCOL * This,IN UINT32 MediaId,IN UINT64 Offset,IN UINTN BufferSize,OUT VOID * Buffer)1179 DiskIoReadDisk (
1180   IN EFI_DISK_IO_PROTOCOL  *This,
1181   IN UINT32                MediaId,
1182   IN UINT64                Offset,
1183   IN UINTN                 BufferSize,
1184   OUT VOID                 *Buffer
1185   )
1186 {
1187   return DiskIo2ReadWriteDisk (
1188            DISK_IO_PRIVATE_DATA_FROM_DISK_IO (This),
1189            FALSE, MediaId, Offset, NULL, BufferSize, (UINT8 *) Buffer
1190            );
1191 }
1192 
1193 
1194 /**
1195   Writes BufferSize bytes from Buffer into Offset.
1196   Writes may require a read modify write to support writes that are not
1197   aligned on sector boundaries. There are three cases:
1198     UnderRun - The first byte is not on a sector boundary or the write request
1199                is less than a sector in length. Read modify write is required.
1200     Aligned  - A write of N contiguous sectors.
1201     OverRun  - The last byte is not on a sector boundary. Read modified write
1202                required.
1203 
1204   @param  This       Protocol instance pointer.
1205   @param  MediaId    Id of the media, changes every time the media is replaced.
1206   @param  Offset     The starting byte offset to read from
1207   @param  BufferSize Size of Buffer
1208   @param  Buffer     Buffer containing read data
1209 
1210   @retval EFI_SUCCESS           The data was written correctly to the device.
1211   @retval EFI_WRITE_PROTECTED   The device can not be written to.
1212   @retval EFI_DEVICE_ERROR      The device reported an error while performing the write.
1213   @retval EFI_NO_MEDIA          There is no media in the device.
1214   @retval EFI_MEDIA_CHNAGED     The MediaId does not matched the current device.
1215   @retval EFI_INVALID_PARAMETER The write request contains device addresses that are not
1216                                  valid for the device.
1217 
1218 **/
1219 EFI_STATUS
1220 EFIAPI
DiskIoWriteDisk(IN EFI_DISK_IO_PROTOCOL * This,IN UINT32 MediaId,IN UINT64 Offset,IN UINTN BufferSize,IN VOID * Buffer)1221 DiskIoWriteDisk (
1222   IN EFI_DISK_IO_PROTOCOL  *This,
1223   IN UINT32                MediaId,
1224   IN UINT64                Offset,
1225   IN UINTN                 BufferSize,
1226   IN VOID                  *Buffer
1227   )
1228 {
1229   return DiskIo2ReadWriteDisk (
1230            DISK_IO_PRIVATE_DATA_FROM_DISK_IO (This),
1231            TRUE, MediaId, Offset, NULL, BufferSize, (UINT8 *) Buffer
1232            );
1233 }
1234 
1235 /**
1236   The user Entry Point for module DiskIo. The user code starts with this function.
1237 
1238   @param[in] ImageHandle    The firmware allocated handle for the EFI image.
1239   @param[in] SystemTable    A pointer to the EFI System Table.
1240 
1241   @retval EFI_SUCCESS       The entry point is executed successfully.
1242   @retval other             Some error occurs when executing this entry point.
1243 
1244 **/
1245 EFI_STATUS
1246 EFIAPI
InitializeDiskIo(IN EFI_HANDLE ImageHandle,IN EFI_SYSTEM_TABLE * SystemTable)1247 InitializeDiskIo (
1248   IN EFI_HANDLE           ImageHandle,
1249   IN EFI_SYSTEM_TABLE     *SystemTable
1250   )
1251 {
1252   EFI_STATUS              Status;
1253 
1254   //
1255   // Install driver model protocol(s).
1256   //
1257   Status = EfiLibInstallDriverBindingComponentName2 (
1258              ImageHandle,
1259              SystemTable,
1260              &gDiskIoDriverBinding,
1261              ImageHandle,
1262              &gDiskIoComponentName,
1263              &gDiskIoComponentName2
1264              );
1265   ASSERT_EFI_ERROR (Status);
1266 
1267   return Status;
1268 }
1269