gKernelArgs.boot_disk.cd is now only set to true, if the selected boot volume
is still the CD-ROM. BIOSDrive::ReadAt() now tries to read a specific block up to 3 times before failing - after the second retry, it will also reset the disk system. get_ext_drive_parameters() will now fail if the BIOS fills in the device_parameters structure incorrectly (just tested some values against zero, that at least helps in the case of one of my systems). git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@14401 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
e84506daca
commit
597416d1dd
@ -6,6 +6,7 @@
|
||||
|
||||
#include "bios.h"
|
||||
|
||||
#include <KernelExport.h>
|
||||
#include <boot/platform.h>
|
||||
#include <boot/partitions.h>
|
||||
#include <boot/stdio.h>
|
||||
@ -27,6 +28,7 @@ extern uint8 gBootDriveID;
|
||||
extern uint32 gBootPartitionOffset;
|
||||
|
||||
// int 0x13 definitions
|
||||
#define BIOS_RESET_DISK_SYSTEM 0x0000
|
||||
#define BIOS_READ 0x0200
|
||||
#define BIOS_GET_DRIVE_PARAMETERS 0x0800
|
||||
#define BIOS_IS_EXT_PRESENT 0x4100
|
||||
@ -141,6 +143,7 @@ class BIOSDrive : public Node {
|
||||
|
||||
bool HasParameters() const { return fHasParameters; }
|
||||
const drive_parameters &Parameters() const { return fParameters; }
|
||||
uint8 DriveID() const { return fDriveID; }
|
||||
|
||||
protected:
|
||||
uint8 fDriveID;
|
||||
@ -153,11 +156,11 @@ class BIOSDrive : public Node {
|
||||
|
||||
|
||||
static void
|
||||
check_cd_boot(void)
|
||||
check_cd_boot(BIOSDrive *drive)
|
||||
{
|
||||
gKernelArgs.boot_disk.cd = false;
|
||||
|
||||
if (gBootDriveID != 0)
|
||||
if (drive->DriveID() != 0)
|
||||
return;
|
||||
|
||||
struct bios_regs regs;
|
||||
@ -190,6 +193,8 @@ static status_t
|
||||
get_ext_drive_parameters(uint8 drive, drive_parameters *targetParameters)
|
||||
{
|
||||
drive_parameters *parameter = (drive_parameters *)kDataSegmentScratch;
|
||||
|
||||
memset(parameter, 0, sizeof(drive_parameters);
|
||||
parameter->parameters_size = sizeof(drive_parameters);
|
||||
|
||||
struct bios_regs regs;
|
||||
@ -198,7 +203,11 @@ get_ext_drive_parameters(uint8 drive, drive_parameters *targetParameters)
|
||||
regs.esi = (addr_t)parameter - kDataSegmentBase;
|
||||
call_bios(0x13, ®s);
|
||||
|
||||
if (regs.flags & CARRY_FLAG)
|
||||
// filter out faulty BIOS return codes
|
||||
if ((regs.flags & CARRY_FLAG) != 0
|
||||
|| parameter->heads == 0
|
||||
|| parameter->sectors_per_track == 0
|
||||
|| parameter->sectors == 0)
|
||||
return B_ERROR;
|
||||
|
||||
memcpy(targetParameters, parameter, sizeof(drive_parameters));
|
||||
@ -380,8 +389,10 @@ BIOSDrive::BIOSDrive(uint8 driveID)
|
||||
if (get_ext_drive_parameters(driveID, &fParameters) != B_OK) {
|
||||
// old style CHS support
|
||||
|
||||
if (get_drive_parameters(driveID, &fParameters) != B_OK)
|
||||
if (get_drive_parameters(driveID, &fParameters) != B_OK) {
|
||||
dprintf("getting drive parameters for: %u failed!\n", fDriveID);
|
||||
return;
|
||||
}
|
||||
|
||||
TRACE((" cylinders: %lu, heads: %lu, sectors: %lu, bytes_per_sector: %u\n",
|
||||
fParameters.cylinders, fParameters.heads, fParameters.sectors_per_track,
|
||||
@ -431,7 +442,7 @@ BIOSDrive::ReadAt(void *cookie, off_t pos, void *buffer, size_t bufferSize)
|
||||
uint32 blocksLeft = (bufferSize + offset + fBlockSize - 1) / fBlockSize;
|
||||
int32 totalBytesRead = 0;
|
||||
|
||||
TRACE(("BIOS reads %lu bytes from %Ld (offset = %lu), drive %ld\n",
|
||||
TRACE(("BIOS reads %lu bytes from %Ld (offset = %lu), drive %u\n",
|
||||
blocksLeft * fBlockSize, pos * fBlockSize, offset, fDriveID));
|
||||
|
||||
uint32 scratchSize = 24 * 1024 / fBlockSize;
|
||||
@ -476,7 +487,11 @@ BIOSDrive::ReadAt(void *cookie, off_t pos, void *buffer, size_t bufferSize)
|
||||
if (cylinder >= fParameters.cylinders)
|
||||
return B_BAD_VALUE;
|
||||
|
||||
// try to read from the device more than once, just to make sure it'll work
|
||||
struct bios_regs regs;
|
||||
int32 tries = 3;
|
||||
|
||||
while (tries-- > 0) {
|
||||
regs.eax = BIOS_READ | blocksRead;
|
||||
regs.edx = fDriveID | (head << 8);
|
||||
regs.ecx = sector | ((cylinder >> 2) & 0xc0) | ((cylinder & 0xff) << 8);
|
||||
@ -484,9 +499,26 @@ BIOSDrive::ReadAt(void *cookie, off_t pos, void *buffer, size_t bufferSize)
|
||||
regs.ebx = kExtraSegmentScratch;
|
||||
call_bios(0x13, ®s);
|
||||
|
||||
if (regs.flags & CARRY_FLAG)
|
||||
if ((regs.flags & CARRY_FLAG) == 0)
|
||||
break;
|
||||
|
||||
if (tries < 2) {
|
||||
// reset disk system
|
||||
regs.eax = BIOS_RESET_DISK_SYSTEM;
|
||||
regs.edx = fDriveID;
|
||||
call_bios(0x13, ®s);
|
||||
}
|
||||
|
||||
// wait a bit between the retries (1/20 sec)
|
||||
spin(50000);
|
||||
}
|
||||
|
||||
if (regs.flags & CARRY_FLAG) {
|
||||
dprintf("reading %ld bytes from drive %u failed at %Ld\n",
|
||||
blocksRead, fDriveID, pos);
|
||||
return B_ERROR;
|
||||
}
|
||||
}
|
||||
|
||||
uint32 bytesRead = fBlockSize * blocksRead - offset;
|
||||
// copy no more than bufferSize bytes
|
||||
@ -588,7 +620,7 @@ platform_register_boot_device(Node *device)
|
||||
BIOSDrive *drive = (BIOSDrive *)device;
|
||||
|
||||
gKernelArgs.platform_args.boot_drive_number = gBootDriveID;
|
||||
check_cd_boot();
|
||||
check_cd_boot(drive);
|
||||
|
||||
if (drive->HasParameters()) {
|
||||
const drive_parameters ¶meters = drive->Parameters();
|
||||
|
Loading…
Reference in New Issue
Block a user