1affae2bfSwdenk /* 2affae2bfSwdenk * (C) Copyright 2001 3affae2bfSwdenk * Wolfgang Denk, DENX Software Engineering, wd@denx.de. 4affae2bfSwdenk * 5affae2bfSwdenk * See file CREDITS for list of people who contributed to this 6affae2bfSwdenk * project. 7affae2bfSwdenk * 8affae2bfSwdenk * This program is free software; you can redistribute it and/or 9affae2bfSwdenk * modify it under the terms of the GNU General Public License as 10affae2bfSwdenk * published by the Free Software Foundation; either version 2 of 11affae2bfSwdenk * the License, or (at your option) any later version. 12affae2bfSwdenk * 13affae2bfSwdenk * This program is distributed in the hope that it will be useful, 14affae2bfSwdenk * but WITHOUT ANY WARRANTY; without even the implied warranty of 15affae2bfSwdenk * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 16affae2bfSwdenk * GNU General Public License for more details. 17affae2bfSwdenk * 18affae2bfSwdenk * You should have received a copy of the GNU General Public License 19affae2bfSwdenk * along with this program; if not, write to the Free Software 20affae2bfSwdenk * Foundation, Inc., 59 Temple Place, Suite 330, Boston, 21affae2bfSwdenk * MA 02111-1307 USA 22affae2bfSwdenk */ 23affae2bfSwdenk 24affae2bfSwdenk #include <common.h> 25affae2bfSwdenk #include <command.h> 26affae2bfSwdenk #include <ide.h> 27735dd97bSGrant Likely #include <part.h> 28affae2bfSwdenk 29affae2bfSwdenk #undef PART_DEBUG 30affae2bfSwdenk 31affae2bfSwdenk #ifdef PART_DEBUG 32affae2bfSwdenk #define PRINTF(fmt,args...) printf (fmt ,##args) 33affae2bfSwdenk #else 34affae2bfSwdenk #define PRINTF(fmt,args...) 35affae2bfSwdenk #endif 36affae2bfSwdenk 37cde5c64dSJon Loeliger #if (defined(CONFIG_CMD_IDE) || \ 38c7057b52SDave Liu defined(CONFIG_CMD_SATA) || \ 39cde5c64dSJon Loeliger defined(CONFIG_CMD_SCSI) || \ 40cde5c64dSJon Loeliger defined(CONFIG_CMD_USB) || \ 411968e615Swdenk defined(CONFIG_MMC) || \ 421968e615Swdenk defined(CONFIG_SYSTEMACE) ) 43affae2bfSwdenk 44735dd97bSGrant Likely struct block_drvr { 45735dd97bSGrant Likely char *name; 46735dd97bSGrant Likely block_dev_desc_t* (*get_dev)(int dev); 47735dd97bSGrant Likely }; 48735dd97bSGrant Likely 49735dd97bSGrant Likely static const struct block_drvr block_drvr[] = { 50cde5c64dSJon Loeliger #if defined(CONFIG_CMD_IDE) 51735dd97bSGrant Likely { .name = "ide", .get_dev = ide_get_dev, }, 52735dd97bSGrant Likely #endif 53c7057b52SDave Liu #if defined(CONFIG_CMD_SATA) 54c7057b52SDave Liu {.name = "sata", .get_dev = sata_get_dev, }, 55c7057b52SDave Liu #endif 56cde5c64dSJon Loeliger #if defined(CONFIG_CMD_SCSI) 57735dd97bSGrant Likely { .name = "scsi", .get_dev = scsi_get_dev, }, 58735dd97bSGrant Likely #endif 59cde5c64dSJon Loeliger #if defined(CONFIG_CMD_USB) && defined(CONFIG_USB_STORAGE) 60735dd97bSGrant Likely { .name = "usb", .get_dev = usb_stor_get_dev, }, 61735dd97bSGrant Likely #endif 62735dd97bSGrant Likely #if defined(CONFIG_MMC) 63735dd97bSGrant Likely { .name = "mmc", .get_dev = mmc_get_dev, }, 64735dd97bSGrant Likely #endif 65735dd97bSGrant Likely #if defined(CONFIG_SYSTEMACE) 66735dd97bSGrant Likely { .name = "ace", .get_dev = systemace_get_dev, }, 67735dd97bSGrant Likely #endif 68735dd97bSGrant Likely { }, 69735dd97bSGrant Likely }; 70735dd97bSGrant Likely 71751bb571SStefan Roese DECLARE_GLOBAL_DATA_PTR; 72751bb571SStefan Roese 73735dd97bSGrant Likely block_dev_desc_t *get_dev(char* ifname, int dev) 74735dd97bSGrant Likely { 75735dd97bSGrant Likely const struct block_drvr *drvr = block_drvr; 76751bb571SStefan Roese block_dev_desc_t* (*reloc_get_dev)(int dev); 77751bb571SStefan Roese 786c7cac8cSStefan Roese while (drvr->name) { 79751bb571SStefan Roese reloc_get_dev = drvr->get_dev + gd->reloc_off; 80751bb571SStefan Roese if (strncmp(ifname, drvr->name, strlen(drvr->name)) == 0) 81751bb571SStefan Roese return reloc_get_dev(dev); 82735dd97bSGrant Likely drvr++; 83735dd97bSGrant Likely } 84735dd97bSGrant Likely return NULL; 85735dd97bSGrant Likely } 86735dd97bSGrant Likely #else 87735dd97bSGrant Likely block_dev_desc_t *get_dev(char* ifname, int dev) 88735dd97bSGrant Likely { 89735dd97bSGrant Likely return NULL; 90735dd97bSGrant Likely } 91735dd97bSGrant Likely #endif 92735dd97bSGrant Likely 93cde5c64dSJon Loeliger #if (defined(CONFIG_CMD_IDE) || \ 94c7057b52SDave Liu defined(CONFIG_CMD_SATA) || \ 95cde5c64dSJon Loeliger defined(CONFIG_CMD_SCSI) || \ 96cde5c64dSJon Loeliger defined(CONFIG_CMD_USB) || \ 97735dd97bSGrant Likely defined(CONFIG_MMC) || \ 98735dd97bSGrant Likely defined(CONFIG_SYSTEMACE) ) 99735dd97bSGrant Likely 100affae2bfSwdenk /* ------------------------------------------------------------------------- */ 101affae2bfSwdenk /* 102affae2bfSwdenk * reports device info to the user 103affae2bfSwdenk */ 104affae2bfSwdenk void dev_print (block_dev_desc_t *dev_desc) 105affae2bfSwdenk { 10642dfe7a1Swdenk #ifdef CONFIG_LBA48 107c40b2956Swdenk uint64_t lba512; /* number of blocks if 512bytes block size */ 108c40b2956Swdenk #else 109c40b2956Swdenk lbaint_t lba512; 110c40b2956Swdenk #endif 111affae2bfSwdenk 1128ec6e332STor Krill switch (dev_desc->if_type) { 113574b3195SDetlev Zundel case IF_TYPE_SCSI: 114574b3195SDetlev Zundel printf ("(%d:%d) Vendor: %s Prod.: %s Rev: %s\n", 115574b3195SDetlev Zundel dev_desc->target,dev_desc->lun, 116affae2bfSwdenk dev_desc->vendor, 117affae2bfSwdenk dev_desc->product, 118affae2bfSwdenk dev_desc->revision); 119574b3195SDetlev Zundel break; 120*6e24a1ebSRemy Bohmer case IF_TYPE_ATAPI: 121574b3195SDetlev Zundel case IF_TYPE_IDE: 122574b3195SDetlev Zundel case IF_TYPE_SATA: 123574b3195SDetlev Zundel printf ("Model: %s Firm: %s Ser#: %s\n", 124574b3195SDetlev Zundel dev_desc->vendor, 125574b3195SDetlev Zundel dev_desc->revision, 126574b3195SDetlev Zundel dev_desc->product); 127574b3195SDetlev Zundel break; 128*6e24a1ebSRemy Bohmer case IF_TYPE_SD: 129*6e24a1ebSRemy Bohmer case IF_TYPE_MMC: 13047bebe34SNícolas Carneiro Lebedenco case IF_TYPE_USB: 13147bebe34SNícolas Carneiro Lebedenco printf ("Vendor: %s Rev: %s Prod: %s\n", 13247bebe34SNícolas Carneiro Lebedenco dev_desc->vendor, 13347bebe34SNícolas Carneiro Lebedenco dev_desc->revision, 13447bebe34SNícolas Carneiro Lebedenco dev_desc->product); 13547bebe34SNícolas Carneiro Lebedenco break; 136*6e24a1ebSRemy Bohmer case IF_TYPE_DOC: 137*6e24a1ebSRemy Bohmer puts("device type DOC\n"); 138*6e24a1ebSRemy Bohmer return; 1398ec6e332STor Krill case IF_TYPE_UNKNOWN: 140*6e24a1ebSRemy Bohmer puts("device type unknown\n"); 141*6e24a1ebSRemy Bohmer return; 142574b3195SDetlev Zundel default: 143*6e24a1ebSRemy Bohmer printf("Unhandled device type: %i\n", dev_desc->if_type); 144574b3195SDetlev Zundel return; 145affae2bfSwdenk } 146affae2bfSwdenk puts (" Type: "); 147affae2bfSwdenk if (dev_desc->removable) 148affae2bfSwdenk puts ("Removable "); 149affae2bfSwdenk switch (dev_desc->type & 0x1F) { 150726c0f1eSDetlev Zundel case DEV_TYPE_HARDDISK: 151726c0f1eSDetlev Zundel puts ("Hard Disk"); 152affae2bfSwdenk break; 153726c0f1eSDetlev Zundel case DEV_TYPE_CDROM: 154726c0f1eSDetlev Zundel puts ("CD ROM"); 155affae2bfSwdenk break; 156726c0f1eSDetlev Zundel case DEV_TYPE_OPDISK: 157726c0f1eSDetlev Zundel puts ("Optical Device"); 158affae2bfSwdenk break; 159726c0f1eSDetlev Zundel case DEV_TYPE_TAPE: 160726c0f1eSDetlev Zundel puts ("Tape"); 161affae2bfSwdenk break; 162726c0f1eSDetlev Zundel default: 163726c0f1eSDetlev Zundel printf ("# %02X #", dev_desc->type & 0x1F); 164affae2bfSwdenk break; 165affae2bfSwdenk } 166affae2bfSwdenk puts ("\n"); 167affae2bfSwdenk if ((dev_desc->lba * dev_desc->blksz)>0L) { 168affae2bfSwdenk ulong mb, mb_quot, mb_rem, gb, gb_quot, gb_rem; 169c40b2956Swdenk lbaint_t lba; 1706e592385Swdenk 171c40b2956Swdenk lba = dev_desc->lba; 172affae2bfSwdenk 173c40b2956Swdenk lba512 = (lba * (dev_desc->blksz/512)); 174affae2bfSwdenk mb = (10 * lba512) / 2048; /* 2048 = (1024 * 1024) / 512 MB */ 175affae2bfSwdenk /* round to 1 digit */ 176affae2bfSwdenk mb_quot = mb / 10; 177affae2bfSwdenk mb_rem = mb - (10 * mb_quot); 178affae2bfSwdenk 179affae2bfSwdenk gb = mb / 1024; 180affae2bfSwdenk gb_quot = gb / 10; 181affae2bfSwdenk gb_rem = gb - (10 * gb_quot); 18242dfe7a1Swdenk #ifdef CONFIG_LBA48 1836e592385Swdenk if (dev_desc->lba48) 184c40b2956Swdenk printf (" Supports 48-bit addressing\n"); 185c40b2956Swdenk #endif 18642dfe7a1Swdenk #if defined(CFG_64BIT_LBA) && defined(CFG_64BIT_VSPRINTF) 187c40b2956Swdenk printf (" Capacity: %ld.%ld MB = %ld.%ld GB (%qd x %ld)\n", 188c40b2956Swdenk mb_quot, mb_rem, 189c40b2956Swdenk gb_quot, gb_rem, 190c40b2956Swdenk lba, 191c40b2956Swdenk dev_desc->blksz); 192c40b2956Swdenk #else 193affae2bfSwdenk printf (" Capacity: %ld.%ld MB = %ld.%ld GB (%ld x %ld)\n", 194affae2bfSwdenk mb_quot, mb_rem, 195affae2bfSwdenk gb_quot, gb_rem, 196c40b2956Swdenk (ulong)lba, 197affae2bfSwdenk dev_desc->blksz); 198c40b2956Swdenk #endif 199affae2bfSwdenk } else { 200affae2bfSwdenk puts (" Capacity: not available\n"); 201affae2bfSwdenk } 202affae2bfSwdenk } 203b3aff0cbSJon Loeliger #endif 204affae2bfSwdenk 205cde5c64dSJon Loeliger #if (defined(CONFIG_CMD_IDE) || \ 206c7057b52SDave Liu defined(CONFIG_CMD_SATA) || \ 207cde5c64dSJon Loeliger defined(CONFIG_CMD_SCSI) || \ 208cde5c64dSJon Loeliger defined(CONFIG_CMD_USB) || \ 2097fac3f69SHaavard Skinnemoen defined(CONFIG_MMC) || \ 2103f85ce27Swdenk defined(CONFIG_SYSTEMACE) ) 211affae2bfSwdenk 212affae2bfSwdenk #if defined(CONFIG_MAC_PARTITION) || \ 213affae2bfSwdenk defined(CONFIG_DOS_PARTITION) || \ 214c7de829cSwdenk defined(CONFIG_ISO_PARTITION) || \ 215c7de829cSwdenk defined(CONFIG_AMIGA_PARTITION) 216affae2bfSwdenk 217affae2bfSwdenk void init_part (block_dev_desc_t * dev_desc) 218affae2bfSwdenk { 219affae2bfSwdenk #ifdef CONFIG_ISO_PARTITION 220affae2bfSwdenk if (test_part_iso(dev_desc) == 0) { 221affae2bfSwdenk dev_desc->part_type = PART_TYPE_ISO; 222affae2bfSwdenk return; 223affae2bfSwdenk } 224affae2bfSwdenk #endif 225affae2bfSwdenk 226affae2bfSwdenk #ifdef CONFIG_MAC_PARTITION 227affae2bfSwdenk if (test_part_mac(dev_desc) == 0) { 228affae2bfSwdenk dev_desc->part_type = PART_TYPE_MAC; 229affae2bfSwdenk return; 230affae2bfSwdenk } 231affae2bfSwdenk #endif 232affae2bfSwdenk 233affae2bfSwdenk #ifdef CONFIG_DOS_PARTITION 234affae2bfSwdenk if (test_part_dos(dev_desc) == 0) { 235affae2bfSwdenk dev_desc->part_type = PART_TYPE_DOS; 236affae2bfSwdenk return; 237affae2bfSwdenk } 238affae2bfSwdenk #endif 239c7de829cSwdenk 240c7de829cSwdenk #ifdef CONFIG_AMIGA_PARTITION 241c7de829cSwdenk if (test_part_amiga(dev_desc) == 0) { 242c7de829cSwdenk dev_desc->part_type = PART_TYPE_AMIGA; 243c7de829cSwdenk return; 244c7de829cSwdenk } 245c7de829cSwdenk #endif 246affae2bfSwdenk } 247affae2bfSwdenk 248affae2bfSwdenk 2493e3b9569SPeter Pearse int get_partition_info (block_dev_desc_t *dev_desc, int part 2503e3b9569SPeter Pearse , disk_partition_t *info) 251affae2bfSwdenk { 252affae2bfSwdenk switch (dev_desc->part_type) { 253affae2bfSwdenk #ifdef CONFIG_MAC_PARTITION 254affae2bfSwdenk case PART_TYPE_MAC: 255affae2bfSwdenk if (get_partition_info_mac(dev_desc,part,info) == 0) { 256affae2bfSwdenk PRINTF ("## Valid MAC partition found ##\n"); 257affae2bfSwdenk return (0); 258affae2bfSwdenk } 259affae2bfSwdenk break; 260affae2bfSwdenk #endif 261affae2bfSwdenk 262affae2bfSwdenk #ifdef CONFIG_DOS_PARTITION 263affae2bfSwdenk case PART_TYPE_DOS: 264affae2bfSwdenk if (get_partition_info_dos(dev_desc,part,info) == 0) { 265affae2bfSwdenk PRINTF ("## Valid DOS partition found ##\n"); 266affae2bfSwdenk return (0); 267affae2bfSwdenk } 268affae2bfSwdenk break; 269affae2bfSwdenk #endif 270affae2bfSwdenk 271affae2bfSwdenk #ifdef CONFIG_ISO_PARTITION 272affae2bfSwdenk case PART_TYPE_ISO: 273affae2bfSwdenk if (get_partition_info_iso(dev_desc,part,info) == 0) { 274affae2bfSwdenk PRINTF ("## Valid ISO boot partition found ##\n"); 275affae2bfSwdenk return (0); 276affae2bfSwdenk } 277affae2bfSwdenk break; 278affae2bfSwdenk #endif 279c7de829cSwdenk 280c7de829cSwdenk #ifdef CONFIG_AMIGA_PARTITION 281c7de829cSwdenk case PART_TYPE_AMIGA: 282c7de829cSwdenk if (get_partition_info_amiga(dev_desc, part, info) == 0) 283c7de829cSwdenk { 284c7de829cSwdenk PRINTF ("## Valid Amiga partition found ##\n"); 285c7de829cSwdenk return (0); 286c7de829cSwdenk } 287c7de829cSwdenk break; 288c7de829cSwdenk #endif 289affae2bfSwdenk default: 290affae2bfSwdenk break; 291affae2bfSwdenk } 292affae2bfSwdenk return (-1); 293affae2bfSwdenk } 294affae2bfSwdenk 295affae2bfSwdenk static void print_part_header (const char *type, block_dev_desc_t * dev_desc) 296affae2bfSwdenk { 297affae2bfSwdenk puts ("\nPartition Map for "); 298affae2bfSwdenk switch (dev_desc->if_type) { 299726c0f1eSDetlev Zundel case IF_TYPE_IDE: 300726c0f1eSDetlev Zundel puts ("IDE"); 301affae2bfSwdenk break; 302726c0f1eSDetlev Zundel case IF_TYPE_SATA: 303726c0f1eSDetlev Zundel puts ("SATA"); 304c7057b52SDave Liu break; 305726c0f1eSDetlev Zundel case IF_TYPE_SCSI: 306726c0f1eSDetlev Zundel puts ("SCSI"); 307affae2bfSwdenk break; 308726c0f1eSDetlev Zundel case IF_TYPE_ATAPI: 309726c0f1eSDetlev Zundel puts ("ATAPI"); 310affae2bfSwdenk break; 311726c0f1eSDetlev Zundel case IF_TYPE_USB: 312726c0f1eSDetlev Zundel puts ("USB"); 313affae2bfSwdenk break; 314726c0f1eSDetlev Zundel case IF_TYPE_DOC: 315726c0f1eSDetlev Zundel puts ("DOC"); 316affae2bfSwdenk break; 317726c0f1eSDetlev Zundel default: 318726c0f1eSDetlev Zundel puts ("UNKNOWN"); 319affae2bfSwdenk break; 320affae2bfSwdenk } 321affae2bfSwdenk printf (" device %d -- Partition Type: %s\n\n", 322affae2bfSwdenk dev_desc->dev, type); 323affae2bfSwdenk } 324affae2bfSwdenk 325affae2bfSwdenk void print_part (block_dev_desc_t * dev_desc) 326affae2bfSwdenk { 327affae2bfSwdenk 328affae2bfSwdenk switch (dev_desc->part_type) { 329affae2bfSwdenk #ifdef CONFIG_MAC_PARTITION 330affae2bfSwdenk case PART_TYPE_MAC: 331affae2bfSwdenk PRINTF ("## Testing for valid MAC partition ##\n"); 332affae2bfSwdenk print_part_header ("MAC", dev_desc); 333affae2bfSwdenk print_part_mac (dev_desc); 334affae2bfSwdenk return; 335affae2bfSwdenk #endif 336affae2bfSwdenk #ifdef CONFIG_DOS_PARTITION 337affae2bfSwdenk case PART_TYPE_DOS: 338affae2bfSwdenk PRINTF ("## Testing for valid DOS partition ##\n"); 339affae2bfSwdenk print_part_header ("DOS", dev_desc); 340affae2bfSwdenk print_part_dos (dev_desc); 341affae2bfSwdenk return; 342affae2bfSwdenk #endif 343affae2bfSwdenk 344affae2bfSwdenk #ifdef CONFIG_ISO_PARTITION 345affae2bfSwdenk case PART_TYPE_ISO: 346affae2bfSwdenk PRINTF ("## Testing for valid ISO Boot partition ##\n"); 347affae2bfSwdenk print_part_header ("ISO", dev_desc); 348affae2bfSwdenk print_part_iso (dev_desc); 349affae2bfSwdenk return; 350affae2bfSwdenk #endif 351c7de829cSwdenk 352c7de829cSwdenk #ifdef CONFIG_AMIGA_PARTITION 353c7de829cSwdenk case PART_TYPE_AMIGA: 354c7de829cSwdenk PRINTF ("## Testing for a valid Amiga partition ##\n"); 355c7de829cSwdenk print_part_header ("AMIGA", dev_desc); 356c7de829cSwdenk print_part_amiga (dev_desc); 357c7de829cSwdenk return; 358c7de829cSwdenk #endif 359affae2bfSwdenk } 360affae2bfSwdenk puts ("## Unknown partition table\n"); 361affae2bfSwdenk } 362affae2bfSwdenk 363affae2bfSwdenk 364affae2bfSwdenk #else /* neither MAC nor DOS nor ISO partition configured */ 3653e3b9569SPeter Pearse # error neither CONFIG_MAC_PARTITION nor CONFIG_DOS_PARTITION 3663e3b9569SPeter Pearse # error nor CONFIG_ISO_PARTITION configured! 367affae2bfSwdenk #endif 368affae2bfSwdenk 369cde5c64dSJon Loeliger #endif 370