mirror of https://github.com/acidanthera/audk.git
1760 lines
44 KiB
C
1760 lines
44 KiB
C
/** @file
|
|
Floppy Peim to support Recovery function from Floppy device.
|
|
|
|
Copyright (c) 2006 - 2018, 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);
|
|
}
|