mirror of
				https://git.proxmox.com/git/mirror_edk2
				synced 2025-10-25 18:00:42 +00:00 
			
		
		
		
	 45295cf5a5
			
		
	
	
		45295cf5a5
		
	
	
	
	
		
			
			Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Eric Dong <eric.dong@intel.com> Reviewed-by: Jeff Fan <jeff.fan@intel.com> git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@15592 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			1760 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1760 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
| Floppy Peim to support Recovery function from Floppy device.
 | |
| 
 | |
| Copyright (c) 2006 - 2014, Intel Corporation. All rights reserved.<BR>
 | |
|   
 | |
| This program and the accompanying materials
 | |
| are licensed and made available under the terms and conditions
 | |
| of the BSD License which accompanies this distribution.  The
 | |
| full text of the license may be found at
 | |
| http://opensource.org/licenses/bsd-license.php
 | |
| 
 | |
| THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
 | |
| 
 | |
| **/
 | |
| 
 | |
| 
 | |
| #include "FloppyPeim.h"
 | |
| 
 | |
| 
 | |
| PEI_DMA_TABLE      mRegisterTable[] = {
 | |
|   //
 | |
|   // DMA2: Clear Byte Ptr, Enable
 | |
|   //
 | |
|   {
 | |
|     R_8237_DMA_CBPR_CH4_7,
 | |
|     0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_COMMAND_CH4_7,
 | |
|     0
 | |
|   },
 | |
|   //
 | |
|   // DMA1: Clear Byte Ptr, Enable
 | |
|   //
 | |
|   {
 | |
|     R_8237_DMA_CBPR_CH0_3,
 | |
|     0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_COMMAND_CH0_3,
 | |
|     0
 | |
|   },
 | |
|   //
 | |
|   // Configure Channel 4 for Cascade Mode
 | |
|   // Clear DMA Request and enable DREQ
 | |
|   //
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH4_7,
 | |
|     V_8237_DMA_CHMODE_CASCADE | 0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH4_7,
 | |
|     0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH4_7,
 | |
|     0
 | |
|   },
 | |
|   //
 | |
|   // Configure DMA1 (Channels 0-3) for Single Mode
 | |
|   // Clear DMA Request and enable DREQ
 | |
|   //
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH0_3,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH0_3,
 | |
|     0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH0_3,
 | |
|     0
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH0_3,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 1
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH0_3,
 | |
|     1
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH0_3,
 | |
|     1
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH0_3,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 2
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH0_3,
 | |
|     2
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH0_3,
 | |
|     2
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH0_3,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 3
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH0_3,
 | |
|     3
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH0_3,
 | |
|     3
 | |
|   },
 | |
|   //
 | |
|   // Configure DMA2 (Channels 5-7) for Single Mode
 | |
|   // Clear DMA Request and enable DREQ
 | |
|   //
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH4_7,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 1
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH4_7,
 | |
|     1
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH4_7,
 | |
|     1
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH4_7,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 2
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH4_7,
 | |
|     2
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH4_7,
 | |
|     2
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_CHMODE_CH4_7,
 | |
|     V_8237_DMA_CHMODE_SINGLE | 3
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_STA_CH4_7,
 | |
|     3
 | |
|   },
 | |
|   {
 | |
|     R_8237_DMA_WRSMSK_CH4_7,
 | |
|     3
 | |
|   }
 | |
| };
 | |
| 
 | |
| //
 | |
| // Table of diskette parameters of various diskette types 
 | |
| //
 | |
| DISKET_PARA_TABLE  DiskPara[9] = {
 | |
|   {
 | |
|     0x09,
 | |
|     0x50,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x27,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x14,
 | |
|     0x80
 | |
|   },
 | |
|   {
 | |
|     0x09,
 | |
|     0x2a,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x27,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x40
 | |
|   },
 | |
|   {
 | |
|     0x0f,
 | |
|     0x54,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x0
 | |
|   },
 | |
|   {
 | |
|     0x09,
 | |
|     0x50,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x80
 | |
|   },
 | |
|   {
 | |
|     0x09,
 | |
|     0x2a,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x80
 | |
|   },
 | |
|   {
 | |
|     0x12,
 | |
|     0x1b,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x0
 | |
|   },
 | |
|   {
 | |
|     0x09,
 | |
|     0x2a,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x80
 | |
|   },
 | |
|   {
 | |
|     0x12,
 | |
|     0x1b,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0x0
 | |
|   },
 | |
|   {
 | |
|     0x24,
 | |
|     0x1b,
 | |
|     0xff,
 | |
|     0x2,
 | |
|     0x4f,
 | |
|     0x4,
 | |
|     0x25,
 | |
|     0x0f,
 | |
|     0xc0
 | |
|   }
 | |
| };
 | |
| 
 | |
| //
 | |
| // Byte per sector corresponding to various device types.
 | |
| //
 | |
| UINTN    BytePerSector[6] = { 0, 256, 512, 1024, 2048, 4096 };
 | |
| 
 | |
| FDC_BLK_IO_DEV mBlockIoDevTemplate = {
 | |
|   FDC_BLK_IO_DEV_SIGNATURE,
 | |
|   {
 | |
|     FdcGetNumberOfBlockDevices,
 | |
|     FdcGetBlockDeviceMediaInfo,
 | |
|     FdcReadBlocks,
 | |
|   },
 | |
|   {
 | |
|     (EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST),
 | |
|     &gEfiPeiVirtualBlockIoPpiGuid,
 | |
|     NULL
 | |
|   },
 | |
|   0,
 | |
|   {{0}}
 | |
| };
 | |
| 
 | |
| /**
 | |
|   Wait and check if bits for DIO and RQM of FDC Main Status Register
 | |
|   indicates FDC is ready for read or write.
 | |
| 
 | |
|   Before writing to FDC or reading from FDC, the Host must examine
 | |
|   the bit7(RQM) and bit6(DIO) of the Main Status Register.
 | |
|   That is to say:
 | |
|    Command bytes can not be written to Data Register unless RQM is 1 and DIO is 0.
 | |
|    Result bytes can not be read from Data Register unless RQM is 1 and DIO is 1.
 | |
| 
 | |
|   @param  FdcBlkIoDev       Instance of FDC_BLK_IO_DEV.
 | |
|   @param  DataIn            Indicates data input or output.
 | |
|                             TRUE means input.
 | |
|                             FALSE means output.
 | |
|   @param  TimeoutInMseconds  Timeout value to wait.
 | |
|   
 | |
|   @retval EFI_SUCCESS       FDC is ready.
 | |
|   @retval EFI_NOT_READY     FDC is not ready within the specified time period.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FdcDRQReady (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   IN BOOLEAN          DataIn,
 | |
|   IN UINTN            TimeoutInMseconds
 | |
|   )
 | |
| {
 | |
|   UINTN   Delay;
 | |
|   UINT8   StatusRegister;
 | |
|   UINT8   BitInOut;
 | |
| 
 | |
|   //
 | |
|   // Check bit6 of Main Status Register.
 | |
|   //
 | |
|   BitInOut = 0;
 | |
|   if (DataIn) {
 | |
|     BitInOut = BIT6;
 | |
|   }
 | |
| 
 | |
|   Delay = ((TimeoutInMseconds * STALL_1_MSECOND) / FDC_CHECK_INTERVAL) + 1;
 | |
|   do {
 | |
|     StatusRegister = IoRead8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_MSR));
 | |
|     if ((StatusRegister & MSR_RQM) == MSR_RQM && (StatusRegister & MSR_DIO) == BitInOut) {
 | |
|       //
 | |
|       // FDC is ready
 | |
|       //
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     MicroSecondDelay (FDC_SHORT_DELAY);
 | |
|   } while (--Delay > 0);
 | |
| 
 | |
|   if (Delay == 0) {
 | |
|     //
 | |
|     // FDC is not ready within the specified time period
 | |
|     //
 | |
|     return EFI_NOT_READY;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read a byte from FDC data register.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Pointer          Pointer to buffer to hold data read from FDC.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Byte successfully read.
 | |
|   @retval EFI_DEVICE_ERROR FDC is not ready.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| DataInByte (
 | |
|   IN  FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   OUT UINT8            *Pointer
 | |
|   )
 | |
| {
 | |
|   UINT8 Data;
 | |
| 
 | |
|   //
 | |
|   // Wait for 1ms and detect the FDC is ready to be read
 | |
|   //
 | |
|   if (FdcDRQReady (FdcBlkIoDev, TRUE, 1) != EFI_SUCCESS) {
 | |
|     //
 | |
|     // FDC is not ready.
 | |
|     //
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   Data = IoRead8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DTR));
 | |
|   MicroSecondDelay (FDC_SHORT_DELAY);
 | |
|   *Pointer = Data;
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Write a byte to FDC data register.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Pointer          Pointer to data to write.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Byte successfully written.
 | |
|   @retval EFI_DEVICE_ERROR FDC is not ready.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| DataOutByte (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   IN UINT8            *Pointer
 | |
|   )
 | |
| {
 | |
|   UINT8 Data;
 | |
| 
 | |
|   //
 | |
|   // Wait for 1ms and detect the FDC is ready to be written
 | |
|   //
 | |
|   if (FdcDRQReady (FdcBlkIoDev, FALSE, 1) != EFI_SUCCESS) {
 | |
|     //
 | |
|     // FDC is not ready.
 | |
|     //
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   Data = *Pointer;
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DTR), Data);
 | |
|   MicroSecondDelay (FDC_SHORT_DELAY);
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get Sts0 and Pcn status from FDC
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV
 | |
|   @param  Sts0             Value of Sts0
 | |
|   @param  Pcn              Value of Pcn
 | |
| 
 | |
|   @retval EFI_SUCCESS      Successfully retrieved status value of Sts0 and Pcn.
 | |
|   @retval EFI_DEVICE_ERROR Fail to send SENSE_INT_STATUS_CMD.
 | |
|   @retval EFI_DEVICE_ERROR Fail to read Sts0.
 | |
|   @retval EFI_DEVICE_ERROR Fail to read Pcn.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| SenseIntStatus (
 | |
|   IN  FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   OUT UINT8            *Sts0,
 | |
|   OUT UINT8            *Pcn
 | |
|   )
 | |
| {
 | |
|   UINT8 Command;
 | |
| 
 | |
|   Command = SENSE_INT_STATUS_CMD;
 | |
| 
 | |
|   if (DataOutByte (FdcBlkIoDev, &Command) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   if (DataInByte (FdcBlkIoDev, Sts0) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   if (DataInByte (FdcBlkIoDev, Pcn) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Issue Specify command.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Specify command successfully issued.
 | |
|   @retval EFI_DEVICE_ERROR FDC device has errors.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| Specify (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev
 | |
|   )
 | |
| {
 | |
|   FDC_SPECIFY_CMD Command;
 | |
|   UINTN           Index;
 | |
|   UINT8           *Pointer;
 | |
| 
 | |
|   ZeroMem (&Command, sizeof (FDC_SPECIFY_CMD));
 | |
|   Command.CommandCode = SPECIFY_CMD;
 | |
|   //
 | |
|   // set SRT, HUT
 | |
|   //
 | |
|   Command.SrtHut = 0xdf;
 | |
|   //
 | |
|   // 0xdf;
 | |
|   // set HLT and DMA
 | |
|   //
 | |
|   Command.HltNd = 0x02;
 | |
| 
 | |
|   Pointer            = (UINT8 *) (&Command);
 | |
|   for (Index = 0; Index < sizeof (FDC_SPECIFY_CMD); Index++) {
 | |
|     if (DataOutByte (FdcBlkIoDev, Pointer++) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Wait until busy bit is cleared.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  DevPos           Position of FDC (Driver A or B)
 | |
|   @param  TimeoutInMseconds Timeout value to wait.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Busy bit has been cleared before timeout.
 | |
|   @retval EFI_TIMEOUT      Time goes out before busy bit is cleared.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FdcWaitForBSYClear (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   IN UINT8            DevPos,
 | |
|   IN UINTN            TimeoutInMseconds
 | |
|   )
 | |
| {
 | |
|   UINTN   Delay;
 | |
|   UINT8   StatusRegister;
 | |
|   UINT8   Mask;
 | |
| 
 | |
|   //
 | |
|   // How to determine drive and command are busy or not: by the bits of Main Status Register
 | |
|   // bit0: Drive 0 busy (drive A)
 | |
|   // bit1: Drive 1 busy (drive B)
 | |
|   // bit4: Command busy
 | |
|   //
 | |
|   // set mask: for drive A set bit0 & bit4; for drive B set bit1 & bit4
 | |
|   //
 | |
|   Mask  = (UINT8) ((DevPos == 0 ? MSR_DAB : MSR_DBB) | MSR_CB);
 | |
| 
 | |
|   Delay = ((TimeoutInMseconds * STALL_1_MSECOND) / FDC_CHECK_INTERVAL) + 1;
 | |
| 
 | |
|   do {
 | |
|     StatusRegister = IoRead8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_MSR));
 | |
| 
 | |
|     if ((StatusRegister & Mask) == 0x00) {
 | |
|       //
 | |
|       // not busy
 | |
|       //
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     MicroSecondDelay (FDC_SHORT_DELAY);
 | |
|   } while (--Delay > 0);
 | |
| 
 | |
|   if (Delay == 0) {
 | |
|     return EFI_TIMEOUT;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Reset FDC device.
 | |
| 
 | |
|   @param  FdcBlkIoDev  Instance of FDC_BLK_IO_DEV
 | |
|   @param  DevPos       Index of FDC device.
 | |
| 
 | |
|   @retval EFI_SUCCESS      FDC device successfully reset.
 | |
|   @retval EFI_DEVICE_ERROR Fail to reset FDC device.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FdcReset (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   IN UINT8            DevPos
 | |
|   )
 | |
| {
 | |
|   UINT8 Data;
 | |
|   UINT8 Sts0;
 | |
|   UINT8 Pcn;
 | |
|   UINTN Index;
 | |
| 
 | |
|   //
 | |
|   // Reset specified Floppy Logic Drive according to Fdd -> Disk
 | |
|   // Set Digital Output Register(DOR) to do reset work
 | |
|   //    bit0 & bit1 of DOR : Drive Select
 | |
|   //    bit2 : Reset bit
 | |
|   //    bit3 : DMA and Int bit
 | |
|   // Reset : A "0" written to bit2 resets the FDC, this reset will remain active until
 | |
|   //       a "1" is written to this bit.
 | |
|   // Reset step 1:
 | |
|   //    use bit0 & bit1 to  select the logic drive
 | |
|   //    write "0" to bit2
 | |
|   //
 | |
|   Data = 0x0;
 | |
|   Data = (UINT8) (Data | (SELECT_DRV & DevPos));
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DOR), Data);
 | |
| 
 | |
|   //
 | |
|   // Wait some time, at least 120us.
 | |
|   //
 | |
|   MicroSecondDelay (FDC_RESET_DELAY);
 | |
|   //
 | |
|   // Reset step 2:
 | |
|   //    write "1" to bit2
 | |
|   //    write "1" to bit3 : enable DMA
 | |
|   //
 | |
|   Data |= 0x0C;
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DOR), Data);
 | |
| 
 | |
|   MicroSecondDelay (FDC_RESET_DELAY);
 | |
| 
 | |
|   //
 | |
|   // Wait until specified floppy logic drive is not busy
 | |
|   //
 | |
|   if (FdcWaitForBSYClear (FdcBlkIoDev, DevPos, 1) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
|   //
 | |
|   // Set the Transfer Data Rate
 | |
|   //
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_CCR), 0x0);
 | |
| 
 | |
|   MicroSecondDelay (FDC_MEDIUM_DELAY);
 | |
| 
 | |
|   //
 | |
|   // Issue Sense interrupt command for each drive (totally 4 drives)
 | |
|   //
 | |
|   for (Index = 0; Index < 4; Index++) {
 | |
|     if (SenseIntStatus (FdcBlkIoDev, &Sts0, &Pcn) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|   }
 | |
|   //
 | |
|   // Issue Specify command
 | |
|   //
 | |
|   if (Specify (FdcBlkIoDev) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Turn on the motor of floppy drive.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Info             Information of floppy device.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Motor is successfully turned on.
 | |
|   @retval EFI_SUCCESS      Motor is already on.
 | |
|   @retval EFI_DEVICE_ERROR Busy bit of FDC cannot be cleared.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| MotorOn (
 | |
|   IN FDC_BLK_IO_DEV             *FdcBlkIoDev,
 | |
|   IN OUT PEI_FLOPPY_DEVICE_INFO *Info
 | |
|   )
 | |
| {
 | |
|   UINT8 Data;
 | |
|   UINT8 DevPos;
 | |
| 
 | |
|   //
 | |
|   // Control of the floppy drive motors is a big pain. If motor is off, you have to turn it
 | |
|   // on first. But you can not leave the motor on all the time, since that would wear out the
 | |
|   // disk. On the other hand, if you turn the motor off after each operation, the system performance
 | |
|   // will be awful. The compromise used in this driver is to leave the motor on for 2 seconds after
 | |
|   // each operation. If a new operation is started in that interval(2s), the motor need not be
 | |
|   // turned on again. If no new operation is started, a timer goes off and the motor is turned off.
 | |
|   //
 | |
|   DevPos = Info->DevPos;
 | |
| 
 | |
|   //
 | |
|   // If the Motor is already on, just return EFI_SUCCESS.
 | |
|   //
 | |
|   if (Info->MotorOn) {
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
|   //
 | |
|   // The drive's motor is off, so need turn it on.
 | |
|   // First check if command and drive are busy or not.
 | |
|   //
 | |
|   if (FdcWaitForBSYClear (FdcBlkIoDev, DevPos, 1) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
|   //
 | |
|   // for drive A: 1CH, drive B: 2DH
 | |
|   //
 | |
|   Data = 0x0C;
 | |
|   Data = (UINT8) (Data | (SELECT_DRV & DevPos));
 | |
|   if (DevPos == 0) {
 | |
|     Data |= DRVA_MOTOR_ON;
 | |
|   } else {
 | |
|     Data |= DRVB_MOTOR_ON;
 | |
|   }
 | |
| 
 | |
|   Info->MotorOn = FALSE;
 | |
| 
 | |
|   //
 | |
|   // Turn on the motor and wait for some time to ensure it takes effect.
 | |
|   //
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DOR), Data);
 | |
|   MicroSecondDelay (FDC_LONG_DELAY);
 | |
| 
 | |
|   Info->MotorOn = TRUE;
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Turn off the motor of floppy drive.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Info             Information of floppy device.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| MotorOff (
 | |
|   IN FDC_BLK_IO_DEV             *FdcBlkIoDev,
 | |
|   IN OUT PEI_FLOPPY_DEVICE_INFO *Info
 | |
|   )
 | |
| {
 | |
|   UINT8 Data;
 | |
|   UINT8 DevPos;
 | |
| 
 | |
|   DevPos = Info->DevPos;
 | |
| 
 | |
|   if (!Info->MotorOn) {
 | |
|     return;
 | |
|   }
 | |
|   //
 | |
|   // The motor is on, so need motor off
 | |
|   //
 | |
|   Data = 0x0C;
 | |
|   Data = (UINT8) (Data | (SELECT_DRV & DevPos));
 | |
| 
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DOR), Data);
 | |
|   MicroSecondDelay (FDC_SHORT_DELAY);
 | |
| 
 | |
|   Info->MotorOn = FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Recalibrate the FDC device.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Info             Information of floppy device.
 | |
| 
 | |
|   @retval EFI_SUCCESS      FDC successfully recalibrated.
 | |
|   @retval EFI_DEVICE_ERROR Fail to send RECALIBRATE_CMD.
 | |
|   @retval EFI_DEVICE_ERROR Fail to get status value of Sts0 and Pcn.
 | |
|   @retval EFI_DEVICE_ERROR Fail to recalibrate FDC device.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| Recalibrate (
 | |
|   IN FDC_BLK_IO_DEV             *FdcBlkIoDev,
 | |
|   IN OUT PEI_FLOPPY_DEVICE_INFO *Info
 | |
|   )
 | |
| {
 | |
|   FDC_COMMAND_PACKET2 Command;
 | |
|   UINTN               Index;
 | |
|   UINT8               Sts0;
 | |
|   UINT8               Pcn;
 | |
|   UINT8               *Pointer;
 | |
|   UINT8               Count;
 | |
|   UINT8               DevPos;
 | |
| 
 | |
|   DevPos  = Info->DevPos;
 | |
| 
 | |
|   //
 | |
|   // We would try twice.
 | |
|   //
 | |
|   Count   = 2;
 | |
|   while (Count > 0) {
 | |
|     ZeroMem (&Command, sizeof (FDC_COMMAND_PACKET2));
 | |
|     Command.CommandCode = RECALIBRATE_CMD;
 | |
|     //
 | |
|     // drive select
 | |
|     //
 | |
|     if (DevPos == 0) {
 | |
|       Command.DiskHeadSel = 0;
 | |
|     } else {
 | |
|       Command.DiskHeadSel = 1;
 | |
|     }
 | |
| 
 | |
|     Pointer = (UINT8 *) (&Command);
 | |
|     for (Index = 0; Index < sizeof (FDC_COMMAND_PACKET2); Index++) {
 | |
|       if (DataOutByte (FdcBlkIoDev, Pointer++) != EFI_SUCCESS) {
 | |
|         return EFI_DEVICE_ERROR;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     MicroSecondDelay (FDC_RECALIBRATE_DELAY);
 | |
| 
 | |
|     if (SenseIntStatus (FdcBlkIoDev, &Sts0, &Pcn) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
| 
 | |
|     if ((Sts0 & 0xf0) == BIT5 && Pcn == 0) {
 | |
|       //
 | |
|       // Recalibration is successful. 
 | |
|       //
 | |
|       Info->Pcn = 0;
 | |
|       Info->NeedRecalibrate = FALSE;
 | |
| 
 | |
|       return EFI_SUCCESS;
 | |
|     } else {
 | |
|       //
 | |
|       // Recalibration is not successful. Try again.
 | |
|       // If trial is used out, return EFI_DEVICE_ERROR.
 | |
|       //
 | |
|       Count--;
 | |
|       if (Count == 0) {
 | |
|         return EFI_DEVICE_ERROR;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Seek for the cylinder according to given LBA.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Info             Information of floppy device.
 | |
|   @param  Lba              LBA for which to seek for cylinder.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Successfully moved to the destination cylinder.
 | |
|   @retval EFI_SUCCESS      Destination cylinder is just the present cylinder.
 | |
|   @retval EFI_DEVICE_ERROR Fail to move to the destination cylinder.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| Seek (
 | |
|   IN     FDC_BLK_IO_DEV         *FdcBlkIoDev,
 | |
|   IN OUT PEI_FLOPPY_DEVICE_INFO *Info,
 | |
|   IN     EFI_PEI_LBA            Lba
 | |
|   )
 | |
| {
 | |
|   FDC_SEEK_CMD      Command;
 | |
|   DISKET_PARA_TABLE *Para;
 | |
|   UINT8             EndOfTrack;
 | |
|   UINT8             Head;
 | |
|   UINT8             Cylinder;
 | |
|   UINT8             Sts0;
 | |
|   UINT8             *Pointer;
 | |
|   UINT8             Pcn;
 | |
|   UINTN             Index;
 | |
|   UINT8             Gap;
 | |
|   UINT8             DevPos;
 | |
| 
 | |
|   DevPos = Info->DevPos;
 | |
|   if (Info->NeedRecalibrate) {
 | |
|     if (Recalibrate (FdcBlkIoDev, Info) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|     //
 | |
|     // Recalibrate Success
 | |
|     //
 | |
|     Info->NeedRecalibrate = FALSE;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Get the base of disk parameter information corresponding to its type.
 | |
|   //
 | |
|   Para        = (DISKET_PARA_TABLE *) ((UINT8 *) DiskPara + sizeof (DISKET_PARA_TABLE) * Info->Type);
 | |
|   EndOfTrack  = Para->EndOfTrack;
 | |
|   //
 | |
|   // Calculate cylinder based on Lba and EOT
 | |
|   //
 | |
|   Cylinder = (UINT8) ((UINTN) Lba / EndOfTrack / 2);
 | |
| 
 | |
|   //
 | |
|   // If the dest cylinder is the present cylinder, unnecessary to do the seek operation
 | |
|   //
 | |
|   if (Info->Pcn == Cylinder) {
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Calculate the head : 0 or 1
 | |
|   //
 | |
|   Head = (UINT8) ((UINTN) Lba / EndOfTrack % 2);
 | |
| 
 | |
|   ZeroMem (&Command, sizeof (FDC_SEEK_CMD));
 | |
|   Command.CommandCode = SEEK_CMD;
 | |
|   if (DevPos == 0) {
 | |
|     Command.DiskHeadSel = 0;
 | |
|   } else {
 | |
|     Command.DiskHeadSel = 1;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Send command to move to destination cylinder.
 | |
|   //
 | |
|   Command.DiskHeadSel = (UINT8) (Command.DiskHeadSel | (Head << 2));
 | |
|   Command.NewCylinder = Cylinder;
 | |
| 
 | |
|   Pointer = (UINT8 *) (&Command);
 | |
|   for (Index = 0; Index < sizeof (FDC_SEEK_CMD); Index++) {
 | |
|     if (DataOutByte (FdcBlkIoDev, Pointer++) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   MicroSecondDelay (FDC_SHORT_DELAY);
 | |
| 
 | |
|   //
 | |
|   // Calculate waiting time, which is proportional to the gap between destination
 | |
|   // cylinder and present cylinder.
 | |
|   //
 | |
|   if (Info->Pcn > Cylinder) {
 | |
|     Gap = (UINT8) (Info->Pcn - Cylinder);
 | |
|   } else {
 | |
|     Gap = (UINT8) (Cylinder - Info->Pcn);
 | |
|   }
 | |
| 
 | |
|   MicroSecondDelay ((Gap + 1) * FDC_LONG_DELAY);
 | |
| 
 | |
|   //
 | |
|   // Confirm if the new cylinder is the destination and status is correct.
 | |
|   //
 | |
|   if (SenseIntStatus (FdcBlkIoDev, &Sts0, &Pcn) != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   if ((Sts0 & 0xf0) == BIT5) {
 | |
|     Info->Pcn             = Command.NewCylinder;
 | |
|     Info->NeedRecalibrate = FALSE;
 | |
|     return EFI_SUCCESS;
 | |
|   } else {
 | |
|     Info->NeedRecalibrate = TRUE;
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Check if diskette is changed.
 | |
| 
 | |
|   @param  FdcBlkIoDev       Instance of FDC_BLK_IO_DEV
 | |
|   @param  Info              Information of floppy device.
 | |
| 
 | |
|   @retval EFI_SUCCESS       Diskette is not changed.
 | |
|   @retval EFI_MEDIA_CHANGED Diskette is changed.
 | |
|   @retval EFI_NO_MEDIA      No diskette.
 | |
|   @retval EFI_DEVICE_ERROR  Fail to do the seek or recalibrate operation.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| DisketChanged (
 | |
|   IN FDC_BLK_IO_DEV             *FdcBlkIoDev,
 | |
|   IN OUT PEI_FLOPPY_DEVICE_INFO *Info
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   UINT8       Data;
 | |
| 
 | |
|   //
 | |
|   // Check change line
 | |
|   //
 | |
|   Data = IoRead8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DIR));
 | |
| 
 | |
|   MicroSecondDelay (FDC_SHORT_DELAY);
 | |
| 
 | |
|   if ((Data & DIR_DCL) == DIR_DCL) {
 | |
|     if (Info->Pcn != 0) {
 | |
|       Status = Recalibrate (FdcBlkIoDev, Info);
 | |
|     } else {
 | |
|       Status = Seek (FdcBlkIoDev, Info, 0x30);
 | |
|     }
 | |
| 
 | |
|     if (Status != EFI_SUCCESS) {
 | |
|       //
 | |
|       // Fail to do the seek or recalibrate operation
 | |
|       //
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
| 
 | |
|     Data = IoRead8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_DIR));
 | |
| 
 | |
|     MicroSecondDelay (FDC_SHORT_DELAY);
 | |
| 
 | |
|     if ((Data & DIR_DCL) == DIR_DCL) {
 | |
|       return EFI_NO_MEDIA;
 | |
|     }
 | |
| 
 | |
|     return EFI_MEDIA_CHANGED;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Detects if FDC device exists.
 | |
| 
 | |
|   @param  FdcBlkIoDev  Instance of FDC_BLK_IO_DEV
 | |
|   @param  Info         Information of floppy device.
 | |
|   @param  MediaInfo    Information of floppy media.
 | |
| 
 | |
|   @retval TRUE         FDC device exists and is working properly.
 | |
|   @retval FALSE        FDC device does not exist or cannot work properly.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| DiscoverFdcDevice (
 | |
|   IN  FDC_BLK_IO_DEV             *FdcBlkIoDev,
 | |
|   IN  OUT PEI_FLOPPY_DEVICE_INFO *Info,
 | |
|   OUT EFI_PEI_BLOCK_IO_MEDIA     *MediaInfo
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS        Status;
 | |
|   DISKET_PARA_TABLE *Para;
 | |
| 
 | |
|   Status = MotorOn (FdcBlkIoDev, Info);
 | |
|   if (Status != EFI_SUCCESS) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   Status = Recalibrate (FdcBlkIoDev, Info);
 | |
| 
 | |
|   if (Status != EFI_SUCCESS) {
 | |
|     MotorOff (FdcBlkIoDev, Info);
 | |
|     return FALSE;
 | |
|   }
 | |
|   //
 | |
|   // Set Media Parameter
 | |
|   //
 | |
|   MediaInfo->DeviceType   = LegacyFloppy;
 | |
|   MediaInfo->MediaPresent = TRUE;
 | |
| 
 | |
|   //
 | |
|   // Check Media
 | |
|   //
 | |
|   Status = DisketChanged (FdcBlkIoDev, Info);
 | |
|   if (Status == EFI_NO_MEDIA) {
 | |
|     //
 | |
|     // No diskette in floppy.
 | |
|     //
 | |
|     MediaInfo->MediaPresent = FALSE;    
 | |
|   } else if (Status != EFI_MEDIA_CHANGED && Status != EFI_SUCCESS) {
 | |
|     //
 | |
|     // EFI_DEVICE_ERROR
 | |
|     //
 | |
|     MotorOff (FdcBlkIoDev, Info);
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   MotorOff (FdcBlkIoDev, Info);
 | |
| 
 | |
|   //
 | |
|   // Get the base of disk parameter information corresponding to its type.
 | |
|   //
 | |
|   Para                  = (DISKET_PARA_TABLE *) ((UINT8 *) DiskPara + sizeof (DISKET_PARA_TABLE) * Info->Type);
 | |
| 
 | |
|   MediaInfo->BlockSize  = BytePerSector[Para->Number];
 | |
|   MediaInfo->LastBlock  = Para->EndOfTrack * 2 * (Para->MaxTrackNum + 1) - 1;
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Enumerate floppy device
 | |
| 
 | |
|   @param  FdcBlkIoDev  Instance of floppy device controller
 | |
| 
 | |
|   @return Number of FDC devices.
 | |
| 
 | |
| **/
 | |
| UINT8
 | |
| FdcEnumeration (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev
 | |
|   )
 | |
| {
 | |
|   UINT8                   DevPos;
 | |
|   UINT8                   DevNo;
 | |
|   EFI_PEI_BLOCK_IO_MEDIA  MediaInfo;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   DevNo = 0;
 | |
| 
 | |
|   //
 | |
|   // DevPos=0 means Drive A, 1 means Drive B.
 | |
|   //
 | |
|   for (DevPos = 0; DevPos < 2; DevPos++) {
 | |
|     //
 | |
|     // Detecting device presence
 | |
|     //
 | |
|     REPORT_STATUS_CODE (EFI_PROGRESS_CODE, EFI_PERIPHERAL_REMOVABLE_MEDIA + EFI_P_PC_PRESENCE_DETECT);
 | |
| 
 | |
|     //
 | |
|     // Reset FDC
 | |
|     //
 | |
|     Status = FdcReset (FdcBlkIoDev, DevPos);
 | |
| 
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     FdcBlkIoDev->DeviceInfo[DevPos].DevPos          = DevPos;
 | |
|     FdcBlkIoDev->DeviceInfo[DevPos].Pcn             = 0;
 | |
|     FdcBlkIoDev->DeviceInfo[DevPos].MotorOn         = FALSE;
 | |
|     FdcBlkIoDev->DeviceInfo[DevPos].NeedRecalibrate = TRUE;
 | |
|     FdcBlkIoDev->DeviceInfo[DevPos].Type            = FdcType1440K1440K;
 | |
| 
 | |
|     //
 | |
|     // Discover FDC device
 | |
|     //
 | |
|     if (DiscoverFdcDevice (FdcBlkIoDev, &(FdcBlkIoDev->DeviceInfo[DevPos]), &MediaInfo)) {
 | |
|       FdcBlkIoDev->DeviceInfo[DevNo].DevPos           = DevPos;
 | |
| 
 | |
|       FdcBlkIoDev->DeviceInfo[DevNo].Pcn              = FdcBlkIoDev->DeviceInfo[DevPos].Pcn;
 | |
|       FdcBlkIoDev->DeviceInfo[DevNo].MotorOn          = FdcBlkIoDev->DeviceInfo[DevPos].MotorOn;
 | |
|       FdcBlkIoDev->DeviceInfo[DevNo].NeedRecalibrate  = FdcBlkIoDev->DeviceInfo[DevPos].NeedRecalibrate;
 | |
|       FdcBlkIoDev->DeviceInfo[DevNo].Type             = FdcBlkIoDev->DeviceInfo[DevPos].Type;
 | |
| 
 | |
|       CopyMem (
 | |
|         &(FdcBlkIoDev->DeviceInfo[DevNo].MediaInfo),
 | |
|         &MediaInfo,
 | |
|         sizeof (EFI_PEI_BLOCK_IO_MEDIA)
 | |
|         );
 | |
| 
 | |
|       DevNo++;
 | |
|     } else {
 | |
|       //
 | |
|       // Assume controller error
 | |
|       //
 | |
|       REPORT_STATUS_CODE (
 | |
|         EFI_ERROR_CODE | EFI_ERROR_MINOR,
 | |
|         EFI_PERIPHERAL_REMOVABLE_MEDIA + EFI_P_EC_CONTROLLER_ERROR
 | |
|         );
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   FdcBlkIoDev->DeviceCount = DevNo;
 | |
|   return DevNo;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Checks result reflected by FDC_RESULT_PACKET.
 | |
| 
 | |
|   @param  Result           FDC_RESULT_PACKET read from FDC after certain operation.
 | |
|   @param  Info             Information of floppy device.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Result is healthy.
 | |
|   @retval EFI_DEVICE_ERROR Result is not healthy.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| CheckResult (
 | |
|   IN  FDC_RESULT_PACKET         *Result,
 | |
|   OUT PEI_FLOPPY_DEVICE_INFO    *Info
 | |
|   )
 | |
| {
 | |
|   if ((Result->Status0 & STS0_IC) != IC_NT) {
 | |
|     if ((Result->Status0 & STS0_SE) == BIT5) {
 | |
|       //
 | |
|       // Seek error
 | |
|       //
 | |
|       Info->NeedRecalibrate = TRUE;
 | |
|     }
 | |
| 
 | |
|     Info->NeedRecalibrate = TRUE;
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
|   //
 | |
|   // Check Status Register1
 | |
|   //
 | |
|   if ((Result->Status1 & (STS1_EN | STS1_DE | STS1_OR | STS1_ND | STS1_NW | STS1_MA)) != 0) {
 | |
|     Info->NeedRecalibrate = TRUE;
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
|   //
 | |
|   // Check Status Register2
 | |
|   //
 | |
|   if ((Result->Status2 & (STS2_CM | STS2_DD | STS2_WC | STS2_BC | STS2_MD)) != 0) {
 | |
|     Info->NeedRecalibrate = TRUE;
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Fill parameters for command packet.
 | |
| 
 | |
|   @param  Info    Information of floppy device.
 | |
|   @param  Lba     Logical block address.
 | |
|   @param  Command Command for which for fill parameters.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| FillPara (
 | |
|   IN  PEI_FLOPPY_DEVICE_INFO *Info,
 | |
|   IN  EFI_PEI_LBA            Lba,
 | |
|   OUT FDC_COMMAND_PACKET1    *Command
 | |
|   )
 | |
| {
 | |
|   DISKET_PARA_TABLE *Para;
 | |
|   UINT8             EndOfTrack;
 | |
|   UINT8             DevPos;
 | |
| 
 | |
|   DevPos      = Info->DevPos;
 | |
| 
 | |
|   //
 | |
|   // Get the base of disk parameter information corresponding to its type.
 | |
|   //
 | |
|   Para        = (DISKET_PARA_TABLE *) ((UINT8 *) DiskPara + sizeof (DISKET_PARA_TABLE) * Info->Type);
 | |
| 
 | |
|   EndOfTrack  = Para->EndOfTrack;
 | |
| 
 | |
|   if (DevPos == 0) {
 | |
|     Command->DiskHeadSel = 0;
 | |
|   } else {
 | |
|     Command->DiskHeadSel = 1;
 | |
|   }
 | |
| 
 | |
|   Command->Cylinder    = (UINT8) ((UINTN) Lba / EndOfTrack / 2);
 | |
|   Command->Head        = (UINT8) ((UINTN) Lba / EndOfTrack % 2);
 | |
|   Command->Sector      = (UINT8) ((UINT8) ((UINTN) Lba % EndOfTrack) + 1);
 | |
|   Command->DiskHeadSel = (UINT8) (Command->DiskHeadSel | (Command->Head << 2));
 | |
|   Command->Number      = Para->Number;
 | |
|   Command->EndOfTrack  = Para->EndOfTrack;
 | |
|   Command->GapLength   = Para->GapLength;
 | |
|   Command->DataLength  = Para->DataLength;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Setup specifed FDC device.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  DevPos           Index of FDC device.
 | |
| 
 | |
|   @retval EFI_SUCCESS      FDC device successfully set up.
 | |
|   @retval EFI_DEVICE_ERROR FDC device has errors.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| Setup (
 | |
|   IN  FDC_BLK_IO_DEV  *FdcBlkIoDev,
 | |
|   IN  UINT8           DevPos
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
| 
 | |
|   IoWrite8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_CCR), 0x0);
 | |
| 
 | |
|   MicroSecondDelay (FDC_MEDIUM_DELAY);
 | |
| 
 | |
|   Status = Specify (FdcBlkIoDev);
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Setup DMA channels to read data.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Buffer           Memory buffer for DMA transfer.
 | |
|   @param  BlockSize        the number of the bytes in one block.
 | |
|   @param  NumberOfBlocks   Number of blocks to read.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SetDMA (
 | |
|   IN FDC_BLK_IO_DEV   *FdcBlkIoDev,
 | |
|   IN VOID             *Buffer,
 | |
|   IN UINTN            BlockSize,
 | |
|   IN UINTN            NumberOfBlocks
 | |
|   )
 | |
| {
 | |
|   UINT8 Data;
 | |
|   UINTN Count;
 | |
| 
 | |
|   //
 | |
|   // Mask DMA channel 2;
 | |
|   //
 | |
|   IoWrite8 (R_8237_DMA_WRSMSK_CH0_3, B_8237_DMA_WRSMSK_CMS | 2);
 | |
| 
 | |
|   //
 | |
|   // Clear first/last flip flop
 | |
|   //
 | |
|   IoWrite8 (R_8237_DMA_CBPR_CH0_3, B_8237_DMA_WRSMSK_CMS | 2);
 | |
| 
 | |
|   //
 | |
|   // Set mode
 | |
|   //
 | |
|   IoWrite8 (R_8237_DMA_CHMODE_CH0_3, V_8237_DMA_CHMODE_SINGLE | V_8237_DMA_CHMODE_IO2MEM | 2);
 | |
| 
 | |
|   //
 | |
|   // Set base address and page register
 | |
|   //
 | |
|   Data = (UINT8) (UINTN) Buffer;
 | |
|   IoWrite8 (R_8237_DMA_BASE_CA_CH2, Data);
 | |
|   Data = (UINT8) ((UINTN) Buffer >> 8);
 | |
|   IoWrite8 (R_8237_DMA_BASE_CA_CH2, Data);
 | |
| 
 | |
|   Data = (UINT8) ((UINTN) Buffer >> 16);
 | |
|   IoWrite8 (R_8237_DMA_MEM_LP_CH2, Data);
 | |
| 
 | |
|   //
 | |
|   // Set count register
 | |
|   //
 | |
|   Count = BlockSize * NumberOfBlocks - 1;
 | |
|   Data  = (UINT8) (Count & 0xff);
 | |
|   IoWrite8 (R_8237_DMA_BASE_CC_CH2, Data);
 | |
|   Data = (UINT8) (Count >> 8);
 | |
|   IoWrite8 (R_8237_DMA_BASE_CC_CH2, Data);
 | |
| 
 | |
|   //
 | |
|   // Clear channel 2 mask
 | |
|   //
 | |
|   IoWrite8 (R_8237_DMA_WRSMSK_CH0_3, 0x02);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   According to the block range specified by Lba and NumberOfBlocks, calculate
 | |
|   the number of blocks in the same sector, which can be transferred in a batch.
 | |
| 
 | |
|   @param  Info           Information of floppy device.
 | |
|   @param  Lba            Start address of block range.
 | |
|   @param  NumberOfBlocks Number of blocks of the range.
 | |
| 
 | |
|   @return Number of blocks in the same sector.
 | |
| 
 | |
| **/
 | |
| UINTN
 | |
| GetTransferBlockCount (
 | |
|   IN  PEI_FLOPPY_DEVICE_INFO *Info,
 | |
|   IN  EFI_PEI_LBA            Lba,
 | |
|   IN  UINTN                  NumberOfBlocks
 | |
|   )
 | |
| {
 | |
|   DISKET_PARA_TABLE *Para;
 | |
|   UINT8             EndOfTrack;
 | |
|   UINT8             Head;
 | |
|   UINT8             SectorsInTrack;
 | |
| 
 | |
|   //
 | |
|   // Get the base of disk parameter information corresponding to its type.
 | |
|   //
 | |
|   Para            = (DISKET_PARA_TABLE *) ((UINT8 *) DiskPara + sizeof (DISKET_PARA_TABLE) * Info->Type);
 | |
| 
 | |
|   EndOfTrack      = Para->EndOfTrack;
 | |
|   Head            = (UINT8) ((UINTN) Lba / EndOfTrack % 2);
 | |
| 
 | |
|   SectorsInTrack  = (UINT8) (EndOfTrack * (2 - Head) - (UINT8) ((UINTN) Lba % EndOfTrack));
 | |
|   if (SectorsInTrack < NumberOfBlocks) {
 | |
|     //
 | |
|     // Not all the block range locates in the same sector
 | |
|     //
 | |
|     return SectorsInTrack;
 | |
|   } else {
 | |
|     //
 | |
|     // All the block range is in the same sector.
 | |
|     //
 | |
|     return NumberOfBlocks;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read data sector from FDC device.
 | |
| 
 | |
|   @param  FdcBlkIoDev      Instance of FDC_BLK_IO_DEV.
 | |
|   @param  Info             Information of floppy device.
 | |
|   @param  Buffer           Buffer to setup for DMA.
 | |
|   @param  Lba              The start address to read.
 | |
|   @param  NumberOfBlocks   Number of blocks to read.
 | |
| 
 | |
|   @retval EFI_SUCCESS      Data successfully read out.
 | |
|   @retval EFI_DEVICE_ERROR FDC device has errors.
 | |
|   @retval EFI_TIMEOUT      Command does not take effect in time.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| ReadDataSector (
 | |
|   IN     FDC_BLK_IO_DEV         *FdcBlkIoDev,
 | |
|   IN OUT PEI_FLOPPY_DEVICE_INFO *Info,
 | |
|   IN     VOID                   *Buffer,
 | |
|   IN     EFI_PEI_LBA            Lba,
 | |
|   IN     UINTN                  NumberOfBlocks
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS          Status;
 | |
|   FDC_COMMAND_PACKET1 Command;
 | |
|   FDC_RESULT_PACKET   Result;
 | |
|   UINTN               Index;
 | |
|   UINTN               Times;
 | |
|   UINT8               *Pointer;
 | |
| 
 | |
|   Status = Seek (FdcBlkIoDev, Info, Lba);
 | |
|   if (Status != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Set up DMA
 | |
|   //
 | |
|   SetDMA (FdcBlkIoDev, Buffer, Info->MediaInfo.BlockSize, NumberOfBlocks);
 | |
| 
 | |
|   //
 | |
|   // Allocate Read command packet
 | |
|   //
 | |
|   ZeroMem (&Command, sizeof (FDC_COMMAND_PACKET1));
 | |
|   Command.CommandCode = READ_DATA_CMD | CMD_MT | CMD_MFM | CMD_SK;
 | |
| 
 | |
|   //
 | |
|   // Fill parameters for command.
 | |
|   //
 | |
|   FillPara (Info, Lba, &Command);
 | |
| 
 | |
|   //
 | |
|   // Write command bytes to FDC
 | |
|   //
 | |
|   Pointer = (UINT8 *) (&Command);
 | |
|   for (Index = 0; Index < sizeof (FDC_COMMAND_PACKET1); Index++) {
 | |
|     if (DataOutByte (FdcBlkIoDev, Pointer++) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Wait for some time until command takes effect.
 | |
|   //
 | |
|   Times = (STALL_1_SECOND / FDC_CHECK_INTERVAL) + 1;
 | |
|   do {
 | |
|     if ((IoRead8 ((UINT16) (PcdGet16 (PcdFdcBaseAddress) + FDC_REGISTER_MSR)) & 0xc0) == 0xc0) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     MicroSecondDelay (FDC_SHORT_DELAY);
 | |
|   } while (--Times > 0);
 | |
| 
 | |
|   if (Times == 0) {
 | |
|     //
 | |
|     // Command fails to take effect in time, return EFI_TIMEOUT.
 | |
|     //
 | |
|     return EFI_TIMEOUT;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Read result bytes from FDC
 | |
|   //
 | |
|   Pointer = (UINT8 *) (&Result);
 | |
|   for (Index = 0; Index < sizeof (FDC_RESULT_PACKET); Index++) {
 | |
|     if (DataInByte (FdcBlkIoDev, Pointer++) != EFI_SUCCESS) {
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return CheckResult (&Result, Info);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Gets the count of block I/O devices that one specific block driver detects.
 | |
| 
 | |
|   This function is used for getting the count of block I/O devices that one 
 | |
|   specific block driver detects.  To the PEI ATAPI driver, it returns the number
 | |
|   of all the detected ATAPI devices it detects during the enumeration process. 
 | |
|   To the PEI legacy floppy driver, it returns the number of all the legacy 
 | |
|   devices it finds during its enumeration process. If no device is detected, 
 | |
|   then the function will return zero.  
 | |
|   
 | |
|   @param[in]  PeiServices          General-purpose services that are available 
 | |
|                                    to every PEIM.
 | |
|   @param[in]  This                 Indicates the EFI_PEI_RECOVERY_BLOCK_IO_PPI 
 | |
|                                    instance.
 | |
|   @param[out] NumberBlockDevices   The number of block I/O devices discovered.
 | |
| 
 | |
|   @retval     EFI_SUCCESS          Operation performed successfully.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| FdcGetNumberOfBlockDevices (
 | |
|   IN   EFI_PEI_SERVICES                  **PeiServices,
 | |
|   IN   EFI_PEI_RECOVERY_BLOCK_IO_PPI     *This,
 | |
|   OUT  UINTN                             *NumberBlockDevices
 | |
|   )
 | |
| {
 | |
|   FDC_BLK_IO_DEV  *FdcBlkIoDev;
 | |
| 
 | |
|   FdcBlkIoDev = NULL;
 | |
| 
 | |
|   FdcBlkIoDev         = PEI_RECOVERY_FDC_FROM_BLKIO_THIS (This);
 | |
| 
 | |
|   *NumberBlockDevices = FdcBlkIoDev->DeviceCount;
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Gets a block device's media information.
 | |
| 
 | |
|   This function will provide the caller with the specified block device's media 
 | |
|   information. If the media changes, calling this function will update the media 
 | |
|   information accordingly.
 | |
| 
 | |
|   @param[in]  PeiServices   General-purpose services that are available to every
 | |
|                             PEIM
 | |
|   @param[in]  This          Indicates the EFI_PEI_RECOVERY_BLOCK_IO_PPI instance.
 | |
|   @param[in]  DeviceIndex   Specifies the block device to which the function wants 
 | |
|                             to talk. Because the driver that implements Block I/O 
 | |
|                             PPIs will manage multiple block devices, the PPIs that 
 | |
|                             want to talk to a single device must specify the 
 | |
|                             device index that was assigned during the enumeration
 | |
|                             process. This index is a number from one to 
 | |
|                             NumberBlockDevices.
 | |
|   @param[out] MediaInfo     The media information of the specified block media.  
 | |
|                             The caller is responsible for the ownership of this 
 | |
|                             data structure.
 | |
|   
 | |
|   @retval EFI_SUCCESS        Media information about the specified block device 
 | |
|                              was obtained successfully.
 | |
|   @retval EFI_DEVICE_ERROR   Cannot get the media information due to a hardware 
 | |
|                              error.
 | |
|   @retval Others             Other failure occurs.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| FdcGetBlockDeviceMediaInfo (
 | |
|   IN   EFI_PEI_SERVICES                     **PeiServices,
 | |
|   IN   EFI_PEI_RECOVERY_BLOCK_IO_PPI        *This,
 | |
|   IN   UINTN                                DeviceIndex,
 | |
|   OUT  EFI_PEI_BLOCK_IO_MEDIA               *MediaInfo
 | |
|   )
 | |
| {
 | |
|   UINTN           DeviceCount;
 | |
|   FDC_BLK_IO_DEV  *FdcBlkIoDev;
 | |
|   BOOLEAN         Healthy;
 | |
|   UINTN           Index;
 | |
| 
 | |
|   FdcBlkIoDev = NULL;
 | |
| 
 | |
|   if (This == NULL || MediaInfo == NULL) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   FdcBlkIoDev = PEI_RECOVERY_FDC_FROM_BLKIO_THIS (This);
 | |
| 
 | |
|   DeviceCount = FdcBlkIoDev->DeviceCount;
 | |
| 
 | |
|   //
 | |
|   // DeviceIndex is a value from 1 to NumberBlockDevices.
 | |
|   //
 | |
|   if ((DeviceIndex < 1) || (DeviceIndex > DeviceCount) || (DeviceIndex > 2)) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   Index = DeviceIndex - 1;
 | |
|   //
 | |
|   // Probe media and retrieve latest media information
 | |
|   //
 | |
|   Healthy = DiscoverFdcDevice (
 | |
|               FdcBlkIoDev,
 | |
|               &FdcBlkIoDev->DeviceInfo[Index],
 | |
|               MediaInfo
 | |
|               );
 | |
| 
 | |
|   if (!Healthy) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   CopyMem (
 | |
|     &(FdcBlkIoDev->DeviceInfo[Index].MediaInfo),
 | |
|     MediaInfo,
 | |
|     sizeof (EFI_PEI_BLOCK_IO_MEDIA)
 | |
|     );
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Reads the requested number of blocks from the specified block device.
 | |
| 
 | |
|   The function reads the requested number of blocks from the device. All the 
 | |
|   blocks are read, or an error is returned. If there is no media in the device,
 | |
|   the function returns EFI_NO_MEDIA.
 | |
| 
 | |
|   @param[in]  PeiServices   General-purpose services that are available to 
 | |
|                             every PEIM.
 | |
|   @param[in]  This          Indicates the EFI_PEI_RECOVERY_BLOCK_IO_PPI instance.
 | |
|   @param[in]  DeviceIndex   Specifies the block device to which the function wants 
 | |
|                             to talk. Because the driver that implements Block I/O 
 | |
|                             PPIs will manage multiple block devices, the PPIs that 
 | |
|                             want to talk to a single device must specify the device 
 | |
|                             index that was assigned during the enumeration process. 
 | |
|                             This index is a number from one to NumberBlockDevices.
 | |
|   @param[in]  StartLBA      The starting logical block address (LBA) to read from
 | |
|                             on the device
 | |
|   @param[in]  BufferSize    The size of the Buffer in bytes. This number must be
 | |
|                             a multiple of the intrinsic block size of the device.
 | |
|   @param[out] Buffer        A pointer to the destination buffer for the data.
 | |
|                             The caller is responsible for the ownership of the 
 | |
|                             buffer.
 | |
|                          
 | |
|   @retval EFI_SUCCESS             The data was read correctly from the device.
 | |
|   @retval EFI_DEVICE_ERROR        The device reported an error while attempting 
 | |
|                                   to perform the read operation.
 | |
|   @retval EFI_INVALID_PARAMETER   The read request contains LBAs that are not 
 | |
|                                   valid, or the buffer is not properly aligned.
 | |
|   @retval EFI_NO_MEDIA            There is no media in the device.
 | |
|   @retval EFI_BAD_BUFFER_SIZE     The BufferSize parameter is not a multiple of
 | |
|                                   the intrinsic block size of the device.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| FdcReadBlocks (
 | |
|   IN   EFI_PEI_SERVICES                  **PeiServices,
 | |
|   IN   EFI_PEI_RECOVERY_BLOCK_IO_PPI     *This,
 | |
|   IN   UINTN                             DeviceIndex,
 | |
|   IN   EFI_PEI_LBA                       StartLBA,
 | |
|   IN   UINTN                             BufferSize,
 | |
|   OUT  VOID                              *Buffer
 | |
|   )
 | |
| {
 | |
|   EFI_PEI_BLOCK_IO_MEDIA MediaInfo;
 | |
|   EFI_STATUS            Status;
 | |
|   UINTN                 Count;
 | |
|   UINTN                 NumberOfBlocks;
 | |
|   UINTN                 BlockSize;
 | |
|   FDC_BLK_IO_DEV        *FdcBlkIoDev;
 | |
|   VOID                  *MemPage;
 | |
| 
 | |
|   FdcBlkIoDev = NULL;
 | |
|   ZeroMem (&MediaInfo, sizeof (EFI_PEI_BLOCK_IO_MEDIA));
 | |
| 
 | |
|   if (This == NULL) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   FdcBlkIoDev = PEI_RECOVERY_FDC_FROM_BLKIO_THIS (This);
 | |
| 
 | |
|   if (Buffer == NULL) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   Status = FdcGetBlockDeviceMediaInfo (PeiServices, This, DeviceIndex, &MediaInfo);
 | |
|   if (Status != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   if (!MediaInfo.MediaPresent) {
 | |
|     return EFI_NO_MEDIA;
 | |
|   }
 | |
| 
 | |
|   BlockSize = MediaInfo.BlockSize;
 | |
| 
 | |
|   //
 | |
|   // If BufferSize cannot be divided by block size of FDC device,
 | |
|   // return EFI_BAD_BUFFER_SIZE.
 | |
|   //
 | |
|   if (BufferSize % BlockSize != 0) {
 | |
|     return EFI_BAD_BUFFER_SIZE;
 | |
|   }
 | |
| 
 | |
|   NumberOfBlocks = BufferSize / BlockSize;
 | |
| 
 | |
|   if ((StartLBA + NumberOfBlocks - 1) > FdcBlkIoDev->DeviceInfo[DeviceIndex - 1].MediaInfo.LastBlock) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   MemPage = AllocatePages (EFI_SIZE_TO_PAGES (BufferSize));
 | |
|   if ((MemPage == NULL) || ((UINTN) MemPage >= ISA_MAX_MEMORY_ADDRESS)) {
 | |
|     //
 | |
|     // If fail to allocate memory under ISA_MAX_MEMORY_ADDRESS, designate the address space for DMA
 | |
|     //
 | |
|     MemPage = (VOID *) ((UINTN) (UINT32) 0x0f00000);
 | |
|   }
 | |
|   Status = MotorOn (FdcBlkIoDev, &(FdcBlkIoDev->DeviceInfo[DeviceIndex - 1]));
 | |
|   if (Status != EFI_SUCCESS) {
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| 
 | |
|   Status = Setup (FdcBlkIoDev, FdcBlkIoDev->DeviceInfo[DeviceIndex - 1].DevPos);
 | |
|   if (Status != EFI_SUCCESS) {
 | |
|     MotorOff (FdcBlkIoDev, &(FdcBlkIoDev->DeviceInfo[DeviceIndex - 1]));
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
|   //
 | |
|   // Read data in batches.
 | |
|   // Blocks in the same cylinder are read out in a batch.
 | |
|   //
 | |
|   while ((Count = GetTransferBlockCount (
 | |
|                     &(FdcBlkIoDev->DeviceInfo[DeviceIndex - 1]),
 | |
|                     StartLBA,
 | |
|                     NumberOfBlocks
 | |
|                     )) != 0 && Status == EFI_SUCCESS) {
 | |
|     Status = ReadDataSector (
 | |
|                FdcBlkIoDev,
 | |
|                &(FdcBlkIoDev->DeviceInfo[DeviceIndex - 1]),
 | |
|                MemPage,
 | |
|                StartLBA,
 | |
|                Count
 | |
|                );
 | |
|     CopyMem (Buffer, MemPage, BlockSize * Count);
 | |
|     StartLBA += Count;
 | |
|     NumberOfBlocks -= Count;
 | |
|     Buffer = (VOID *) ((UINTN) Buffer + Count * BlockSize);
 | |
|   }
 | |
| 
 | |
|   MotorOff (FdcBlkIoDev, &(FdcBlkIoDev->DeviceInfo[DeviceIndex - 1]));
 | |
| 
 | |
|   switch (Status) {
 | |
|   case EFI_SUCCESS:
 | |
|     return EFI_SUCCESS;
 | |
| 
 | |
|   default:
 | |
|     FdcReset (FdcBlkIoDev, FdcBlkIoDev->DeviceInfo[DeviceIndex - 1].DevPos);
 | |
|     return EFI_DEVICE_ERROR;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Initializes the floppy disk controller and installs FDC Block I/O PPI.
 | |
| 
 | |
|   @param  FileHandle            Handle of the file being invoked.
 | |
|   @param  PeiServices           Describes the list of possible PEI Services.
 | |
| 
 | |
|   @retval EFI_SUCCESS           Successfully initialized FDC and installed PPI.
 | |
|   @retval EFI_NOT_FOUND         Cannot find FDC device.
 | |
|   @retval EFI_OUT_OF_RESOURCES  Have no enough memory to create instance or descriptors.
 | |
|   @retval Other                 Fail to install FDC Block I/O PPI.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| FdcPeimEntry (
 | |
|   IN  EFI_PEI_FILE_HANDLE         FileHandle,
 | |
|   IN  CONST EFI_PEI_SERVICES      **PeiServices
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS            Status;
 | |
|   FDC_BLK_IO_DEV        *FdcBlkIoDev;
 | |
|   UINTN                 DeviceCount;
 | |
|   UINT32                Index;
 | |
| 
 | |
|   Status = PeiServicesRegisterForShadow (FileHandle);
 | |
|   if (!EFI_ERROR (Status)) {
 | |
|     return Status;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Allocate memory for instance of FDC_BLK_IO_DEV and copy initial value
 | |
|   // from template to it. 
 | |
|   //
 | |
|   FdcBlkIoDev = AllocatePages (EFI_SIZE_TO_PAGES(sizeof (FDC_BLK_IO_DEV)));
 | |
|   if (FdcBlkIoDev == NULL) {
 | |
|     return EFI_OUT_OF_RESOURCES;
 | |
|   }
 | |
|   CopyMem (FdcBlkIoDev, &mBlockIoDevTemplate, sizeof (mBlockIoDevTemplate));
 | |
| 
 | |
|   //
 | |
|   // Initialize DMA controller to enable all channels.
 | |
|   //
 | |
|   for (Index = 0; Index < sizeof (mRegisterTable) / sizeof (PEI_DMA_TABLE); Index++) {
 | |
|     IoWrite8 (mRegisterTable[Index].Register, mRegisterTable[Index].Value);
 | |
|   }
 | |
|   REPORT_STATUS_CODE (EFI_PROGRESS_CODE, EFI_PERIPHERAL_REMOVABLE_MEDIA + EFI_P_PC_INIT);
 | |
| 
 | |
|   //
 | |
|   // Enumerate FDC devices.
 | |
|   //
 | |
|   DeviceCount = FdcEnumeration (FdcBlkIoDev);
 | |
|   if (DeviceCount == 0) {
 | |
|     return EFI_NOT_FOUND;
 | |
|   }
 | |
| 
 | |
|   FdcBlkIoDev->PpiDescriptor.Ppi = &FdcBlkIoDev->FdcBlkIo;
 | |
| 
 | |
|   return PeiServicesInstallPpi (&FdcBlkIoDev->PpiDescriptor);
 | |
| }
 |