Commit 84fdbcf2 authored by Bartlomiej Zolnierkiewicz's avatar Bartlomiej Zolnierkiewicz Committed by Linus Torvalds

[PATCH] ide: disk geometry/capacity cleanups

From Andries.Brouwer@cwi.nl.

- kill redundant, never executed code in lba_capacity_is_ok()
- add idedisk_supports_{hpa,lba48}() helpers
- don't recalculate drive->cyl for drives using LBA addressing,
  we never fall-back to CHS, so its useless and confusing
- remove wrong drive->head and drive->sect assignments for LBA-48
- don't overwrite id->lba_capacity and id->lba_capacity_2
parent e71c9413
...@@ -86,11 +86,6 @@ static int lba_capacity_is_ok (struct hd_driveid *id) ...@@ -86,11 +86,6 @@ static int lba_capacity_is_ok (struct hd_driveid *id)
{ {
unsigned long lba_sects, chs_sects, head, tail; unsigned long lba_sects, chs_sects, head, tail;
if ((id->command_set_2 & 0x0400) && (id->cfs_enable_2 & 0x0400)) {
printk("48-bit Drive: %llu \n", id->lba_capacity_2);
return 1;
}
/* /*
* The ATA spec tells large drives to return * The ATA spec tells large drives to return
* C/H/S = 16383/16/63 independent of their size. * C/H/S = 16383/16/63 independent of their size.
...@@ -1074,11 +1069,29 @@ static unsigned long long sectors_to_MB(unsigned long long n) ...@@ -1074,11 +1069,29 @@ static unsigned long long sectors_to_MB(unsigned long long n)
return n; return n;
} }
/*
* Bits 10 of command_set_1 and cfs_enable_1 must be equal,
* so on non-buggy drives we need test only one.
* However, we should also check whether these fields are valid.
*/
static inline int idedisk_supports_hpa(const struct hd_driveid *id)
{
return (id->command_set_1 & 0x0400) && (id->cfs_enable_1 & 0x0400);
}
/*
* The same here.
*/
static inline int idedisk_supports_lba48(const struct hd_driveid *id)
{
return (id->command_set_2 & 0x0400) && (id->cfs_enable_2 & 0x0400);
}
static inline void idedisk_check_hpa_lba28(ide_drive_t *drive) static inline void idedisk_check_hpa_lba28(ide_drive_t *drive)
{ {
unsigned long capacity, set_max; unsigned long capacity, set_max;
capacity = drive->id->lba_capacity; capacity = drive->capacity;
set_max = idedisk_read_native_max_address(drive); set_max = idedisk_read_native_max_address(drive);
if (set_max <= capacity) if (set_max <= capacity)
...@@ -1093,7 +1106,7 @@ static inline void idedisk_check_hpa_lba28(ide_drive_t *drive) ...@@ -1093,7 +1106,7 @@ static inline void idedisk_check_hpa_lba28(ide_drive_t *drive)
#ifdef CONFIG_IDEDISK_STROKE #ifdef CONFIG_IDEDISK_STROKE
set_max = idedisk_set_max_address(drive, set_max); set_max = idedisk_set_max_address(drive, set_max);
if (set_max) { if (set_max) {
drive->id->lba_capacity = set_max; drive->capacity = set_max;
printk(KERN_INFO "%s: Host Protected Area disabled.\n", printk(KERN_INFO "%s: Host Protected Area disabled.\n",
drive->name); drive->name);
} }
...@@ -1104,7 +1117,7 @@ static inline void idedisk_check_hpa_lba48(ide_drive_t *drive) ...@@ -1104,7 +1117,7 @@ static inline void idedisk_check_hpa_lba48(ide_drive_t *drive)
{ {
unsigned long long capacity_2, set_max_ext; unsigned long long capacity_2, set_max_ext;
capacity_2 = drive->id->lba_capacity_2; capacity_2 = drive->capacity48;
set_max_ext = idedisk_read_native_max_address_ext(drive); set_max_ext = idedisk_read_native_max_address_ext(drive);
if (set_max_ext <= capacity_2) if (set_max_ext <= capacity_2)
...@@ -1119,7 +1132,8 @@ static inline void idedisk_check_hpa_lba48(ide_drive_t *drive) ...@@ -1119,7 +1132,8 @@ static inline void idedisk_check_hpa_lba48(ide_drive_t *drive)
#ifdef CONFIG_IDEDISK_STROKE #ifdef CONFIG_IDEDISK_STROKE
set_max_ext = idedisk_set_max_address_ext(drive, set_max_ext); set_max_ext = idedisk_set_max_address_ext(drive, set_max_ext);
if (set_max_ext) { if (set_max_ext) {
drive->id->lba_capacity_2 = set_max_ext; drive->capacity48 = set_max_ext;
drive->capacity = (unsigned long) set_max_ext;
printk(KERN_INFO "%s: Host Protected Area disabled.\n", printk(KERN_INFO "%s: Host Protected Area disabled.\n",
drive->name); drive->name);
} }
...@@ -1147,32 +1161,21 @@ static void init_idedisk_capacity (ide_drive_t *drive) ...@@ -1147,32 +1161,21 @@ static void init_idedisk_capacity (ide_drive_t *drive)
* If this drive supports the Host Protected Area feature set, * If this drive supports the Host Protected Area feature set,
* then we may need to change our opinion about the drive's capacity. * then we may need to change our opinion about the drive's capacity.
*/ */
int hpa = (id->command_set_1 & 0x0400) && (id->cfs_enable_1 & 0x0400); int hpa = idedisk_supports_hpa(id);
if ((id->command_set_2 & 0x0400) && (id->cfs_enable_2 & 0x0400)) { if (idedisk_supports_lba48(id)) {
/* drive speaks 48-bit LBA */ /* drive speaks 48-bit LBA */
unsigned long long capacity_2;
drive->select.b.lba = 1; drive->select.b.lba = 1;
drive->capacity48 = id->lba_capacity_2;
drive->capacity = (unsigned long) drive->capacity48;
if (hpa) if (hpa)
idedisk_check_hpa_lba48(drive); idedisk_check_hpa_lba48(drive);
capacity_2 = id->lba_capacity_2;
drive->head = drive->bios_head = 255;
drive->sect = drive->bios_sect = 63;
drive->cyl = (unsigned int) capacity_2 / (drive->head * drive->sect);
drive->bios_cyl = drive->cyl;
drive->capacity48 = capacity_2;
drive->capacity = (unsigned long) capacity_2;
} else if ((id->capability & 2) && lba_capacity_is_ok(id)) { } else if ((id->capability & 2) && lba_capacity_is_ok(id)) {
/* drive speaks 28-bit LBA */ /* drive speaks 28-bit LBA */
unsigned long capacity;
drive->select.b.lba = 1; drive->select.b.lba = 1;
drive->capacity = id->lba_capacity;
if (hpa) if (hpa)
idedisk_check_hpa_lba28(drive); idedisk_check_hpa_lba28(drive);
capacity = id->lba_capacity;
drive->cyl = capacity / (drive->head * drive->sect);
drive->capacity = capacity;
} else { } else {
/* drive speaks boring old 28-bit CHS */ /* drive speaks boring old 28-bit CHS */
drive->capacity = drive->cyl * drive->head * drive->sect; drive->capacity = drive->cyl * drive->head * drive->sect;
...@@ -1181,7 +1184,7 @@ static void init_idedisk_capacity (ide_drive_t *drive) ...@@ -1181,7 +1184,7 @@ static void init_idedisk_capacity (ide_drive_t *drive)
static sector_t idedisk_capacity (ide_drive_t *drive) static sector_t idedisk_capacity (ide_drive_t *drive)
{ {
if (drive->id->cfs_enable_2 & 0x0400) if (idedisk_supports_lba48(drive->id))
return (drive->capacity48 - drive->sect0); return (drive->capacity48 - drive->sect0);
return (drive->capacity - drive->sect0); return (drive->capacity - drive->sect0);
} }
...@@ -1472,7 +1475,7 @@ static int probe_lba_addressing (ide_drive_t *drive, int arg) ...@@ -1472,7 +1475,7 @@ static int probe_lba_addressing (ide_drive_t *drive, int arg)
if (HWIF(drive)->addressing) if (HWIF(drive)->addressing)
return 0; return 0;
if (!(drive->id->cfs_enable_2 & 0x0400)) if (!idedisk_supports_lba48(drive->id))
return -EIO; return -EIO;
drive->addressing = arg; drive->addressing = arg;
return 0; return 0;
...@@ -1642,19 +1645,28 @@ static void idedisk_setup (ide_drive_t *drive) ...@@ -1642,19 +1645,28 @@ static void idedisk_setup (ide_drive_t *drive)
* by correcting bios_cyls: * by correcting bios_cyls:
*/ */
capacity = idedisk_capacity (drive); capacity = idedisk_capacity (drive);
if (!drive->forced_geom && drive->bios_sect && drive->bios_head) { if (!drive->forced_geom) {
unsigned int cap0 = capacity; /* truncate to 32 bits */
unsigned int cylsz, cyl; if (idedisk_supports_lba48(drive->id)) {
/* compatibility */
if (cap0 != capacity) drive->bios_sect = 63;
drive->bios_cyl = 65535; drive->bios_head = 255;
else { }
cylsz = drive->bios_sect * drive->bios_head;
cyl = cap0 / cylsz; if (drive->bios_sect && drive->bios_head) {
if (cyl > 65535) unsigned int cap0 = capacity; /* truncate to 32 bits */
cyl = 65535; unsigned int cylsz, cyl;
if (cyl > drive->bios_cyl)
drive->bios_cyl = cyl; if (cap0 != capacity)
drive->bios_cyl = 65535;
else {
cylsz = drive->bios_sect * drive->bios_head;
cyl = cap0 / cylsz;
if (cyl > 65535)
cyl = 65535;
if (cyl > drive->bios_cyl)
drive->bios_cyl = cyl;
}
} }
} }
printk(KERN_INFO "%s: %llu sectors (%llu MB)", printk(KERN_INFO "%s: %llu sectors (%llu MB)",
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment