aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGleb Natapov <gleb@redhat.com>2010-12-07 13:50:54 +0200
committerKevin O'Connor <kevin@koconnor.net>2010-12-12 14:12:59 -0500
commit4c90a20f86a09ffdcecd9685f11f3fe2674b8b77 (patch)
tree7982abe752449000416b4c9b647f91dbab9111d5
parent0e6f6366bc1645c151811c575d667a7676c5d53f (diff)
downloadseabios-4c90a20f86a09ffdcecd9685f11f3fe2674b8b77.tar.gz
Create separate IPL entry for each CD/DVD
Current code creates only one IPL entry of type IPL_TYPE_CDROM even if there are more then one CDROM drive present. If CDROM that the entry refers to is not bootable there is no way to retry boot from another CDROM. Fix this by creating IPL entry for each CDROM drive found. First CDROM will always be placed in IPL entry 3 for backwards compatibility. Signed-off-by: Gleb Natapov <gleb@redhat.com>
-rw-r--r--src/ata.c4
-rw-r--r--src/block.c10
-rw-r--r--src/boot.c39
-rw-r--r--src/boot.h2
-rw-r--r--src/cdrom.c7
-rw-r--r--src/disk.h3
6 files changed, 52 insertions, 13 deletions
diff --git a/src/ata.c b/src/ata.c
index e533108..d971480 100644
--- a/src/ata.c
+++ b/src/ata.c
@@ -782,8 +782,10 @@ init_drive_atapi(struct atadrive_s *dummy, u16 *buffer)
dprintf(1, "%s\n", adrive_g->drive.desc);
// fill cdidmap
- if (iscd)
+ if (iscd) {
map_cd_drive(&adrive_g->drive);
+ add_baid_cdrom(&adrive_g->drive);
+ }
return adrive_g;
}
diff --git a/src/block.c b/src/block.c
index 818c9f9..cab503c 100644
--- a/src/block.c
+++ b/src/block.c
@@ -27,6 +27,16 @@ getDrive(u8 exttype, u8 extdriveoffset)
return GLOBALFLAT2GLOBAL(drive_gf);
}
+int getDriveId(u8 exttype, struct drive_s *drive_g)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(Drives.idmap[0]); i++)
+ if (getDrive(exttype, i) == drive_g)
+ return i;
+
+ return -1;
+}
/****************************************************************
* Disk geometry translation
diff --git a/src/boot.c b/src/boot.c
index 021b8ac..9c94439 100644
--- a/src/boot.c
+++ b/src/boot.c
@@ -87,6 +87,25 @@ add_bev(u16 seg, u16 bev, u16 desc)
ie->description = d;
}
+// Add a IPL entry for BAID cdrom.
+void
+add_baid_cdrom(struct drive_s *drive_g)
+{
+ if (! CONFIG_CDROM_BOOT)
+ return;
+
+ /* put first cdrom into ipl 3 for compatability with qemu */
+ struct ipl_entry_s *ie = &IPL.bev[2];
+ if (IPL.bevcount >= ARRAY_SIZE(IPL.bev) && ie->vector)
+ return;
+
+ if (ie->vector)
+ ie = &IPL.bev[IPL.bevcount++];
+ ie->type = IPL_TYPE_CDROM;
+ ie->vector = (u32)drive_g;
+ ie->description = "DVD/CD";
+}
+
// Add a bcv entry for an expansion card harddrive or legacy option rom
void
add_bcv(u16 seg, u16 ip, u16 desc)
@@ -188,12 +207,11 @@ menu_show_harddisk(struct ipl_entry_s *ie, int menupos)
static int
menu_show_cdrom(struct ipl_entry_s *ie, int menupos)
{
- int i;
- for (i = 0; i < Drives.cdcount; i++) {
- struct drive_s *drive_g = getDrive(EXTTYPE_CD, i);
- printf("%d. DVD/CD [%s]\n", menupos + i, drive_g->desc);
- }
- return Drives.cdcount;
+ struct drive_s *drive_g = (void*)ie->vector;
+ if (!ie->vector)
+ return 0;
+ printf("%d. DVD/CD [%s]\n", menupos, drive_g->desc);
+ return 1;
}
// Show coreboot-fs menu item.
@@ -399,9 +417,14 @@ boot_cdrom(struct ipl_entry_s *ie)
{
if (! CONFIG_CDROM_BOOT)
return;
- int status = cdrom_boot(ie->subchoice);
+
+ if (!ie->vector)
+ return;
+
+ struct drive_s *drive_g = (void*)ie->vector;
+ int status = cdrom_boot(drive_g);
if (status) {
- printf("Boot failed: Could not read from CDROM (code %04x)\n", status);
+ printf("Boot failed: Could not read from CDROM %s (code %04x)\n", drive_g->desc, status);
return;
}
diff --git a/src/boot.h b/src/boot.h
index db046e3..f751d71 100644
--- a/src/boot.h
+++ b/src/boot.h
@@ -43,6 +43,8 @@ void add_bev(u16 seg, u16 bev, u16 desc);
void add_bcv(u16 seg, u16 ip, u16 desc);
struct drive_s;
void add_bcv_internal(struct drive_s *drive_g);
+void add_baid_cdrom(struct drive_s *drive_g);
+
void boot_prep(void);
#endif // __BOOT_H
diff --git a/src/cdrom.c b/src/cdrom.c
index 349e401..31ceaaa 100644
--- a/src/cdrom.c
+++ b/src/cdrom.c
@@ -241,12 +241,13 @@ atapi_is_ready(struct disk_op_s *op)
}
int
-cdrom_boot(int cdid)
+cdrom_boot(struct drive_s *drive_g)
{
struct disk_op_s dop;
+ int cdid = getDriveId(EXTTYPE_CD, drive_g);
memset(&dop, 0, sizeof(dop));
- dop.drive_g = getDrive(EXTTYPE_CD, cdid);
- if (!dop.drive_g)
+ dop.drive_g = drive_g;
+ if (!dop.drive_g || cdid < 0)
return 1;
int ret = atapi_is_ready(&dop);
diff --git a/src/disk.h b/src/disk.h
index c1ba6ae..a598bb4 100644
--- a/src/disk.h
+++ b/src/disk.h
@@ -229,6 +229,7 @@ struct drives_s {
// block.c
extern struct drives_s Drives;
struct drive_s *getDrive(u8 exttype, u8 extdriveoffset);
+int getDriveId(u8 exttype, struct drive_s *drive_g);
void setup_translation(struct drive_s *drive_g);
void map_floppy_drive(struct drive_s *drive_g);
void map_hd_drive(struct drive_s *drive_g);
@@ -249,7 +250,7 @@ extern struct drive_s *cdemu_drive_gf;
int process_cdemu_op(struct disk_op_s *op);
void cdemu_setup(void);
void cdemu_134b(struct bregs *regs);
-int cdrom_boot(int cdid);
+int cdrom_boot(struct drive_s *drive_g);
// ramdisk.c
void ramdisk_setup(void);