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) || \ 38*75eb82ecSunsik Kim defined(CONFIG_CMD_MG_DISK) || \ 39c7057b52SDave Liu defined(CONFIG_CMD_SATA) || \ 40cde5c64dSJon Loeliger defined(CONFIG_CMD_SCSI) || \ 41cde5c64dSJon Loeliger defined(CONFIG_CMD_USB) || \ 421968e615Swdenk defined(CONFIG_MMC) || \ 431968e615Swdenk defined(CONFIG_SYSTEMACE) ) 44affae2bfSwdenk 45735dd97bSGrant Likely struct block_drvr { 46735dd97bSGrant Likely char *name; 47735dd97bSGrant Likely block_dev_desc_t* (*get_dev)(int dev); 48735dd97bSGrant Likely }; 49735dd97bSGrant Likely 50735dd97bSGrant Likely static const struct block_drvr block_drvr[] = { 51cde5c64dSJon Loeliger #if defined(CONFIG_CMD_IDE) 52735dd97bSGrant Likely { .name = "ide", .get_dev = ide_get_dev, }, 53735dd97bSGrant Likely #endif 54c7057b52SDave Liu #if defined(CONFIG_CMD_SATA) 55c7057b52SDave Liu {.name = "sata", .get_dev = sata_get_dev, }, 56c7057b52SDave Liu #endif 57cde5c64dSJon Loeliger #if defined(CONFIG_CMD_SCSI) 58735dd97bSGrant Likely { .name = "scsi", .get_dev = scsi_get_dev, }, 59735dd97bSGrant Likely #endif 60cde5c64dSJon Loeliger #if defined(CONFIG_CMD_USB) && defined(CONFIG_USB_STORAGE) 61735dd97bSGrant Likely { .name = "usb", .get_dev = usb_stor_get_dev, }, 62735dd97bSGrant Likely #endif 63735dd97bSGrant Likely #if defined(CONFIG_MMC) 64735dd97bSGrant Likely { .name = "mmc", .get_dev = mmc_get_dev, }, 65735dd97bSGrant Likely #endif 66735dd97bSGrant Likely #if defined(CONFIG_SYSTEMACE) 67735dd97bSGrant Likely { .name = "ace", .get_dev = systemace_get_dev, }, 68735dd97bSGrant Likely #endif 69*75eb82ecSunsik Kim #if defined(CONFIG_CMD_MG_DISK) 70*75eb82ecSunsik Kim { .name = "mgd", .get_dev = mg_disk_get_dev, }, 71*75eb82ecSunsik Kim #endif 72735dd97bSGrant Likely { }, 73735dd97bSGrant Likely }; 74735dd97bSGrant Likely 75751bb571SStefan Roese DECLARE_GLOBAL_DATA_PTR; 76751bb571SStefan Roese 77735dd97bSGrant Likely block_dev_desc_t *get_dev(char* ifname, int dev) 78735dd97bSGrant Likely { 79735dd97bSGrant Likely const struct block_drvr *drvr = block_drvr; 80751bb571SStefan Roese block_dev_desc_t* (*reloc_get_dev)(int dev); 81751bb571SStefan Roese 826c7cac8cSStefan Roese while (drvr->name) { 83751bb571SStefan Roese reloc_get_dev = drvr->get_dev + gd->reloc_off; 84751bb571SStefan Roese if (strncmp(ifname, drvr->name, strlen(drvr->name)) == 0) 85751bb571SStefan Roese return reloc_get_dev(dev); 86735dd97bSGrant Likely drvr++; 87735dd97bSGrant Likely } 88735dd97bSGrant Likely return NULL; 89735dd97bSGrant Likely } 90735dd97bSGrant Likely #else 91735dd97bSGrant Likely block_dev_desc_t *get_dev(char* ifname, int dev) 92735dd97bSGrant Likely { 93735dd97bSGrant Likely return NULL; 94735dd97bSGrant Likely } 95735dd97bSGrant Likely #endif 96735dd97bSGrant Likely 97cde5c64dSJon Loeliger #if (defined(CONFIG_CMD_IDE) || \ 98*75eb82ecSunsik Kim defined(CONFIG_CMD_MG_DISK) || \ 99c7057b52SDave Liu defined(CONFIG_CMD_SATA) || \ 100cde5c64dSJon Loeliger defined(CONFIG_CMD_SCSI) || \ 101cde5c64dSJon Loeliger defined(CONFIG_CMD_USB) || \ 102735dd97bSGrant Likely defined(CONFIG_MMC) || \ 103735dd97bSGrant Likely defined(CONFIG_SYSTEMACE) ) 104735dd97bSGrant Likely 105affae2bfSwdenk /* ------------------------------------------------------------------------- */ 106affae2bfSwdenk /* 107affae2bfSwdenk * reports device info to the user 108affae2bfSwdenk */ 109affae2bfSwdenk void dev_print (block_dev_desc_t *dev_desc) 110affae2bfSwdenk { 11142dfe7a1Swdenk #ifdef CONFIG_LBA48 112c40b2956Swdenk uint64_t lba512; /* number of blocks if 512bytes block size */ 113c40b2956Swdenk #else 114c40b2956Swdenk lbaint_t lba512; 115c40b2956Swdenk #endif 116affae2bfSwdenk 1178ec6e332STor Krill switch (dev_desc->if_type) { 118574b3195SDetlev Zundel case IF_TYPE_SCSI: 119574b3195SDetlev Zundel printf ("(%d:%d) Vendor: %s Prod.: %s Rev: %s\n", 120574b3195SDetlev Zundel dev_desc->target,dev_desc->lun, 121affae2bfSwdenk dev_desc->vendor, 122affae2bfSwdenk dev_desc->product, 123affae2bfSwdenk dev_desc->revision); 124574b3195SDetlev Zundel break; 1256e24a1ebSRemy Bohmer case IF_TYPE_ATAPI: 126574b3195SDetlev Zundel case IF_TYPE_IDE: 127574b3195SDetlev Zundel case IF_TYPE_SATA: 128574b3195SDetlev Zundel printf ("Model: %s Firm: %s Ser#: %s\n", 129574b3195SDetlev Zundel dev_desc->vendor, 130574b3195SDetlev Zundel dev_desc->revision, 131574b3195SDetlev Zundel dev_desc->product); 132574b3195SDetlev Zundel break; 1336e24a1ebSRemy Bohmer case IF_TYPE_SD: 1346e24a1ebSRemy Bohmer case IF_TYPE_MMC: 13547bebe34SNícolas Carneiro Lebedenco case IF_TYPE_USB: 13647bebe34SNícolas Carneiro Lebedenco printf ("Vendor: %s Rev: %s Prod: %s\n", 13747bebe34SNícolas Carneiro Lebedenco dev_desc->vendor, 13847bebe34SNícolas Carneiro Lebedenco dev_desc->revision, 13947bebe34SNícolas Carneiro Lebedenco dev_desc->product); 14047bebe34SNícolas Carneiro Lebedenco break; 1416e24a1ebSRemy Bohmer case IF_TYPE_DOC: 1426e24a1ebSRemy Bohmer puts("device type DOC\n"); 1436e24a1ebSRemy Bohmer return; 1448ec6e332STor Krill case IF_TYPE_UNKNOWN: 1456e24a1ebSRemy Bohmer puts("device type unknown\n"); 1466e24a1ebSRemy Bohmer return; 147574b3195SDetlev Zundel default: 1486e24a1ebSRemy Bohmer printf("Unhandled device type: %i\n", dev_desc->if_type); 149574b3195SDetlev Zundel return; 150affae2bfSwdenk } 151affae2bfSwdenk puts (" Type: "); 152affae2bfSwdenk if (dev_desc->removable) 153affae2bfSwdenk puts ("Removable "); 154affae2bfSwdenk switch (dev_desc->type & 0x1F) { 155726c0f1eSDetlev Zundel case DEV_TYPE_HARDDISK: 156726c0f1eSDetlev Zundel puts ("Hard Disk"); 157affae2bfSwdenk break; 158726c0f1eSDetlev Zundel case DEV_TYPE_CDROM: 159726c0f1eSDetlev Zundel puts ("CD ROM"); 160affae2bfSwdenk break; 161726c0f1eSDetlev Zundel case DEV_TYPE_OPDISK: 162726c0f1eSDetlev Zundel puts ("Optical Device"); 163affae2bfSwdenk break; 164726c0f1eSDetlev Zundel case DEV_TYPE_TAPE: 165726c0f1eSDetlev Zundel puts ("Tape"); 166affae2bfSwdenk break; 167726c0f1eSDetlev Zundel default: 168726c0f1eSDetlev Zundel printf ("# %02X #", dev_desc->type & 0x1F); 169affae2bfSwdenk break; 170affae2bfSwdenk } 171affae2bfSwdenk puts ("\n"); 172affae2bfSwdenk if ((dev_desc->lba * dev_desc->blksz)>0L) { 173affae2bfSwdenk ulong mb, mb_quot, mb_rem, gb, gb_quot, gb_rem; 174c40b2956Swdenk lbaint_t lba; 1756e592385Swdenk 176c40b2956Swdenk lba = dev_desc->lba; 177affae2bfSwdenk 178c40b2956Swdenk lba512 = (lba * (dev_desc->blksz/512)); 179affae2bfSwdenk mb = (10 * lba512) / 2048; /* 2048 = (1024 * 1024) / 512 MB */ 180affae2bfSwdenk /* round to 1 digit */ 181affae2bfSwdenk mb_quot = mb / 10; 182affae2bfSwdenk mb_rem = mb - (10 * mb_quot); 183affae2bfSwdenk 184affae2bfSwdenk gb = mb / 1024; 185affae2bfSwdenk gb_quot = gb / 10; 186affae2bfSwdenk gb_rem = gb - (10 * gb_quot); 18742dfe7a1Swdenk #ifdef CONFIG_LBA48 1886e592385Swdenk if (dev_desc->lba48) 189c40b2956Swdenk printf (" Supports 48-bit addressing\n"); 190c40b2956Swdenk #endif 1916d0f6bcfSJean-Christophe PLAGNIOL-VILLARD #if defined(CONFIG_SYS_64BIT_LBA) && defined(CONFIG_SYS_64BIT_VSPRINTF) 1926c6166f5SMike Frysinger printf (" Capacity: %ld.%ld MB = %ld.%ld GB (%Ld x %ld)\n", 193c40b2956Swdenk mb_quot, mb_rem, 194c40b2956Swdenk gb_quot, gb_rem, 195c40b2956Swdenk lba, 196c40b2956Swdenk dev_desc->blksz); 197c40b2956Swdenk #else 198affae2bfSwdenk printf (" Capacity: %ld.%ld MB = %ld.%ld GB (%ld x %ld)\n", 199affae2bfSwdenk mb_quot, mb_rem, 200affae2bfSwdenk gb_quot, gb_rem, 201c40b2956Swdenk (ulong)lba, 202affae2bfSwdenk dev_desc->blksz); 203c40b2956Swdenk #endif 204affae2bfSwdenk } else { 205affae2bfSwdenk puts (" Capacity: not available\n"); 206affae2bfSwdenk } 207affae2bfSwdenk } 208b3aff0cbSJon Loeliger #endif 209affae2bfSwdenk 210cde5c64dSJon Loeliger #if (defined(CONFIG_CMD_IDE) || \ 211*75eb82ecSunsik Kim defined(CONFIG_CMD_MG_DISK) || \ 212c7057b52SDave Liu defined(CONFIG_CMD_SATA) || \ 213cde5c64dSJon Loeliger defined(CONFIG_CMD_SCSI) || \ 214cde5c64dSJon Loeliger defined(CONFIG_CMD_USB) || \ 2157fac3f69SHaavard Skinnemoen defined(CONFIG_MMC) || \ 2163f85ce27Swdenk defined(CONFIG_SYSTEMACE) ) 217affae2bfSwdenk 218affae2bfSwdenk #if defined(CONFIG_MAC_PARTITION) || \ 219affae2bfSwdenk defined(CONFIG_DOS_PARTITION) || \ 220c7de829cSwdenk defined(CONFIG_ISO_PARTITION) || \ 22107f3d789Srichardretanubun defined(CONFIG_AMIGA_PARTITION) || \ 22207f3d789Srichardretanubun defined(CONFIG_EFI_PARTITION) 223affae2bfSwdenk 224affae2bfSwdenk void init_part (block_dev_desc_t * dev_desc) 225affae2bfSwdenk { 226affae2bfSwdenk #ifdef CONFIG_ISO_PARTITION 227affae2bfSwdenk if (test_part_iso(dev_desc) == 0) { 228affae2bfSwdenk dev_desc->part_type = PART_TYPE_ISO; 229affae2bfSwdenk return; 230affae2bfSwdenk } 231affae2bfSwdenk #endif 232affae2bfSwdenk 233affae2bfSwdenk #ifdef CONFIG_MAC_PARTITION 234affae2bfSwdenk if (test_part_mac(dev_desc) == 0) { 235affae2bfSwdenk dev_desc->part_type = PART_TYPE_MAC; 236affae2bfSwdenk return; 237affae2bfSwdenk } 238affae2bfSwdenk #endif 239affae2bfSwdenk 24007f3d789Srichardretanubun /* must be placed before DOS partition detection */ 24107f3d789Srichardretanubun #ifdef CONFIG_EFI_PARTITION 24207f3d789Srichardretanubun if (test_part_efi(dev_desc) == 0) { 24307f3d789Srichardretanubun dev_desc->part_type = PART_TYPE_EFI; 24407f3d789Srichardretanubun return; 24507f3d789Srichardretanubun } 24607f3d789Srichardretanubun #endif 24707f3d789Srichardretanubun 248affae2bfSwdenk #ifdef CONFIG_DOS_PARTITION 249affae2bfSwdenk if (test_part_dos(dev_desc) == 0) { 250affae2bfSwdenk dev_desc->part_type = PART_TYPE_DOS; 251affae2bfSwdenk return; 252affae2bfSwdenk } 253affae2bfSwdenk #endif 254c7de829cSwdenk 255c7de829cSwdenk #ifdef CONFIG_AMIGA_PARTITION 256c7de829cSwdenk if (test_part_amiga(dev_desc) == 0) { 257c7de829cSwdenk dev_desc->part_type = PART_TYPE_AMIGA; 258c7de829cSwdenk return; 259c7de829cSwdenk } 260c7de829cSwdenk #endif 261affae2bfSwdenk } 262affae2bfSwdenk 263affae2bfSwdenk 2643e3b9569SPeter Pearse int get_partition_info (block_dev_desc_t *dev_desc, int part 2653e3b9569SPeter Pearse , disk_partition_t *info) 266affae2bfSwdenk { 267affae2bfSwdenk switch (dev_desc->part_type) { 268affae2bfSwdenk #ifdef CONFIG_MAC_PARTITION 269affae2bfSwdenk case PART_TYPE_MAC: 270affae2bfSwdenk if (get_partition_info_mac(dev_desc,part,info) == 0) { 271affae2bfSwdenk PRINTF ("## Valid MAC partition found ##\n"); 272affae2bfSwdenk return (0); 273affae2bfSwdenk } 274affae2bfSwdenk break; 275affae2bfSwdenk #endif 276affae2bfSwdenk 277affae2bfSwdenk #ifdef CONFIG_DOS_PARTITION 278affae2bfSwdenk case PART_TYPE_DOS: 279affae2bfSwdenk if (get_partition_info_dos(dev_desc,part,info) == 0) { 280affae2bfSwdenk PRINTF ("## Valid DOS partition found ##\n"); 281affae2bfSwdenk return (0); 282affae2bfSwdenk } 283affae2bfSwdenk break; 284affae2bfSwdenk #endif 285affae2bfSwdenk 286affae2bfSwdenk #ifdef CONFIG_ISO_PARTITION 287affae2bfSwdenk case PART_TYPE_ISO: 288affae2bfSwdenk if (get_partition_info_iso(dev_desc,part,info) == 0) { 289affae2bfSwdenk PRINTF ("## Valid ISO boot partition found ##\n"); 290affae2bfSwdenk return (0); 291affae2bfSwdenk } 292affae2bfSwdenk break; 293affae2bfSwdenk #endif 294c7de829cSwdenk 295c7de829cSwdenk #ifdef CONFIG_AMIGA_PARTITION 296c7de829cSwdenk case PART_TYPE_AMIGA: 297c7de829cSwdenk if (get_partition_info_amiga(dev_desc, part, info) == 0) 298c7de829cSwdenk { 299c7de829cSwdenk PRINTF ("## Valid Amiga partition found ##\n"); 300c7de829cSwdenk return (0); 301c7de829cSwdenk } 302c7de829cSwdenk break; 303c7de829cSwdenk #endif 30407f3d789Srichardretanubun 30507f3d789Srichardretanubun #ifdef CONFIG_EFI_PARTITION 30607f3d789Srichardretanubun case PART_TYPE_EFI: 30707f3d789Srichardretanubun if (get_partition_info_efi(dev_desc,part,info) == 0) { 30807f3d789Srichardretanubun PRINTF ("## Valid EFI partition found ##\n"); 30907f3d789Srichardretanubun return (0); 31007f3d789Srichardretanubun } 31107f3d789Srichardretanubun break; 31207f3d789Srichardretanubun #endif 313affae2bfSwdenk default: 314affae2bfSwdenk break; 315affae2bfSwdenk } 316affae2bfSwdenk return (-1); 317affae2bfSwdenk } 318affae2bfSwdenk 319affae2bfSwdenk static void print_part_header (const char *type, block_dev_desc_t * dev_desc) 320affae2bfSwdenk { 321affae2bfSwdenk puts ("\nPartition Map for "); 322affae2bfSwdenk switch (dev_desc->if_type) { 323726c0f1eSDetlev Zundel case IF_TYPE_IDE: 324726c0f1eSDetlev Zundel puts ("IDE"); 325affae2bfSwdenk break; 326726c0f1eSDetlev Zundel case IF_TYPE_SATA: 327726c0f1eSDetlev Zundel puts ("SATA"); 328c7057b52SDave Liu break; 329726c0f1eSDetlev Zundel case IF_TYPE_SCSI: 330726c0f1eSDetlev Zundel puts ("SCSI"); 331affae2bfSwdenk break; 332726c0f1eSDetlev Zundel case IF_TYPE_ATAPI: 333726c0f1eSDetlev Zundel puts ("ATAPI"); 334affae2bfSwdenk break; 335726c0f1eSDetlev Zundel case IF_TYPE_USB: 336726c0f1eSDetlev Zundel puts ("USB"); 337affae2bfSwdenk break; 338726c0f1eSDetlev Zundel case IF_TYPE_DOC: 339726c0f1eSDetlev Zundel puts ("DOC"); 340affae2bfSwdenk break; 341726c0f1eSDetlev Zundel default: 342726c0f1eSDetlev Zundel puts ("UNKNOWN"); 343affae2bfSwdenk break; 344affae2bfSwdenk } 345affae2bfSwdenk printf (" device %d -- Partition Type: %s\n\n", 346affae2bfSwdenk dev_desc->dev, type); 347affae2bfSwdenk } 348affae2bfSwdenk 349affae2bfSwdenk void print_part (block_dev_desc_t * dev_desc) 350affae2bfSwdenk { 351affae2bfSwdenk 352affae2bfSwdenk switch (dev_desc->part_type) { 353affae2bfSwdenk #ifdef CONFIG_MAC_PARTITION 354affae2bfSwdenk case PART_TYPE_MAC: 355affae2bfSwdenk PRINTF ("## Testing for valid MAC partition ##\n"); 356affae2bfSwdenk print_part_header ("MAC", dev_desc); 357affae2bfSwdenk print_part_mac (dev_desc); 358affae2bfSwdenk return; 359affae2bfSwdenk #endif 360affae2bfSwdenk #ifdef CONFIG_DOS_PARTITION 361affae2bfSwdenk case PART_TYPE_DOS: 362affae2bfSwdenk PRINTF ("## Testing for valid DOS partition ##\n"); 363affae2bfSwdenk print_part_header ("DOS", dev_desc); 364affae2bfSwdenk print_part_dos (dev_desc); 365affae2bfSwdenk return; 366affae2bfSwdenk #endif 367affae2bfSwdenk 368affae2bfSwdenk #ifdef CONFIG_ISO_PARTITION 369affae2bfSwdenk case PART_TYPE_ISO: 370affae2bfSwdenk PRINTF ("## Testing for valid ISO Boot partition ##\n"); 371affae2bfSwdenk print_part_header ("ISO", dev_desc); 372affae2bfSwdenk print_part_iso (dev_desc); 373affae2bfSwdenk return; 374affae2bfSwdenk #endif 375c7de829cSwdenk 376c7de829cSwdenk #ifdef CONFIG_AMIGA_PARTITION 377c7de829cSwdenk case PART_TYPE_AMIGA: 378c7de829cSwdenk PRINTF ("## Testing for a valid Amiga partition ##\n"); 379c7de829cSwdenk print_part_header ("AMIGA", dev_desc); 380c7de829cSwdenk print_part_amiga (dev_desc); 381c7de829cSwdenk return; 382c7de829cSwdenk #endif 38307f3d789Srichardretanubun 38407f3d789Srichardretanubun #ifdef CONFIG_EFI_PARTITION 38507f3d789Srichardretanubun case PART_TYPE_EFI: 38607f3d789Srichardretanubun PRINTF ("## Testing for valid EFI partition ##\n"); 38707f3d789Srichardretanubun print_part_header ("EFI", dev_desc); 38807f3d789Srichardretanubun print_part_efi (dev_desc); 38907f3d789Srichardretanubun return; 39007f3d789Srichardretanubun #endif 391affae2bfSwdenk } 392affae2bfSwdenk puts ("## Unknown partition table\n"); 393affae2bfSwdenk } 394affae2bfSwdenk 395affae2bfSwdenk 39607f3d789Srichardretanubun #else /* neither MAC nor DOS nor ISO nor AMIGA nor EFI partition configured */ 3973e3b9569SPeter Pearse # error neither CONFIG_MAC_PARTITION nor CONFIG_DOS_PARTITION 39807f3d789Srichardretanubun # error nor CONFIG_ISO_PARTITION nor CONFIG_AMIGA_PARTITION 39907f3d789Srichardretanubun # error nor CONFIG_EFI_PARTITION configured! 400affae2bfSwdenk #endif 401affae2bfSwdenk 402cde5c64dSJon Loeliger #endif 403