| /*************************************************************************** |
| * __________ __ ___. |
| * Open \______ \ ____ ____ | | _\_ |__ _______ ___ |
| * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / |
| * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < |
| * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ |
| * \/ \/ \/ \/ \/ |
| * $Id$ |
| * |
| * Copyright (C) 2002 by Björn Stenberg |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License |
| * as published by the Free Software Foundation; either version 2 |
| * of the License, or (at your option) any later version. |
| * |
| * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY |
| * KIND, either express or implied. |
| * |
| ****************************************************************************/ |
| #include <stdio.h> |
| #include "ata.h" |
| #include "debug.h" |
| #include "fat.h" |
| #ifdef HAVE_HOTSWAP |
| #include "hotswap.h" |
| #include "dir.h" /* for release_dirs() */ |
| #include "file.h" /* for release_files() */ |
| #endif |
| #include "disk.h" |
| #include <string.h> |
| |
| /* Partition table entry layout: |
| ----------------------- |
| 0: 0x80 - active |
| 1: starting head |
| 2: starting sector |
| 3: starting cylinder |
| 4: partition type |
| 5: end head |
| 6: end sector |
| 7: end cylinder |
| 8-11: starting sector (LBA) |
| 12-15: nr of sectors in partition |
| */ |
| |
| #define BYTES2INT32(array,pos) \ |
| ((long)array[pos] | ((long)array[pos+1] << 8 ) | \ |
| ((long)array[pos+2] << 16 ) | ((long)array[pos+3] << 24 )) |
| |
| static const unsigned char fat_partition_types[] = { |
| 0x0b, 0x1b, /* FAT32 + hidden variant */ |
| 0x0c, 0x1c, /* FAT32 (LBA) + hidden variant */ |
| #ifdef HAVE_FAT16SUPPORT |
| 0x04, 0x14, /* FAT16 <= 32MB + hidden variant */ |
| 0x06, 0x16, /* FAT16 > 32MB + hidden variant */ |
| 0x0e, 0x1e, /* FAT16 (LBA) + hidden variant */ |
| #endif |
| }; |
| |
| static struct partinfo part[8]; /* space for 4 partitions on 2 drives */ |
| static int vol_drive[NUM_VOLUMES]; /* mounted to which drive (-1 if none) */ |
| |
| #ifdef MAX_LOG_SECTOR_SIZE |
| int disk_sector_multiplier = 1; |
| #endif |
| |
| struct partinfo* disk_init(IF_MV_NONVOID(int drive)) |
| { |
| int i; |
| unsigned char sector[512]; |
| #ifdef HAVE_MULTIVOLUME |
| /* For each drive, start at a different position, in order not to destroy |
| the first entry of drive 0. |
| That one is needed to calculate config sector position. */ |
| struct partinfo* pinfo = &part[drive*4]; |
| if ((size_t)drive >= sizeof(part)/sizeof(*part)/4) |
| return NULL; /* out of space in table */ |
| #else |
| struct partinfo* pinfo = part; |
| #endif |
| |
| ata_read_sectors(IF_MV2(drive,) 0,1, §or); |
| #ifndef CREATIVE_ZVx |
| /* check that the boot sector is initialized */ |
| if ( (sector[510] != 0x55) || |
| (sector[511] != 0xaa)) { |
| DEBUGF("Bad boot sector signature\n"); |
| return NULL; |
| } |
| |
| /* parse partitions */ |
| for ( i=0; i<4; i++ ) { |
| unsigned char* ptr = sector + 0x1be + 16*i; |
| pinfo[i].type = ptr[4]; |
| pinfo[i].start = BYTES2INT32(ptr, 8); |
| pinfo[i].size = BYTES2INT32(ptr, 12); |
| |
| DEBUGF("Part%d: Type %02x, start: %08lx size: %08lx\n", |
| i,pinfo[i].type,pinfo[i].start,pinfo[i].size); |
| |
| /* extended? */ |
| if ( pinfo[i].type == 5 ) { |
| /* not handled yet */ |
| } |
| } |
| #else |
| struct partition_struct |
| { |
| unsigned int end; |
| unsigned int start; |
| char name[8]; |
| }; |
| struct hdd_struct |
| { |
| unsigned char MBLK[4]; |
| int sector_size; |
| long long total_disk_size; |
| struct partition_struct partitions[4]; |
| }; |
| struct hdd_struct* hdd_struct = (struct hdd_struct*)sector; |
| |
| if(hdd_struct->MBLK[0] != 0x4B || |
| hdd_struct->MBLK[1] != 0x4C || |
| hdd_struct->MBLK[2] != 0x42 || |
| hdd_struct->MBLK[3] != 0x4D) /* 0x4B4C424D = KLBM */ |
| { |
| DEBUGF("Bad boot sector signature\n"); |
| return NULL; |
| } |
| else |
| { |
| /* parse partitions */ |
| for ( i=0; i<4; i++ ) { |
| if(hdd_struct->partitions[i].name[0] != 0) |
| { |
| pinfo[i].type = ( strcmp(hdd_struct->partitions[i].name, "cfs") == 0 ? PARTITION_TYPE_FAT32_LBA : 0); |
| pinfo[i].start = hdd_struct->partitions[i].start; |
| pinfo[i].size = (hdd_struct->partitions[i].end - hdd_struct->partitions[i].start); |
| |
| DEBUGF("Part%d: Type %02x, start: %08lx size: %08lx\n", |
| i,pinfo[i].type,pinfo[i].start,pinfo[i].size); |
| } |
| } |
| } |
| #endif |
| return pinfo; |
| } |
| |
| struct partinfo* disk_partinfo(int partition) |
| { |
| return &part[partition]; |
| } |
| |
| int disk_mount_all(void) |
| { |
| int mounted; |
| int i; |
| |
| #ifdef HAVE_HOTSWAP |
| card_enable_monitoring(false); |
| #endif |
| |
| fat_init(); /* reset all mounted partitions */ |
| for (i=0; i<NUM_VOLUMES; i++) |
| vol_drive[i] = -1; /* mark all as unassigned */ |
| |
| mounted = disk_mount(0); |
| #ifdef HAVE_HOTSWAP |
| if (card_detect()) |
| { |
| mounted += disk_mount(1); /* try 2nd "drive", too */ |
| } |
| |
| card_enable_monitoring(true); |
| #endif |
| |
| return mounted; |
| } |
| |
| static int get_free_volume(void) |
| { |
| int i; |
| for (i=0; i<NUM_VOLUMES; i++) |
| { |
| if (vol_drive[i] == -1) /* unassigned? */ |
| return i; |
| } |
| |
| return -1; /* none found */ |
| } |
| |
| int disk_mount(int drive) |
| { |
| int mounted = 0; /* reset partition-on-drive flag */ |
| int volume = get_free_volume(); |
| struct partinfo* pinfo = disk_init(IF_MV(drive)); |
| |
| if (pinfo == NULL) |
| { |
| return 0; |
| } |
| #if defined(TOSHIBA_GIGABEAT_S) || defined(CREATIVE_ZVx) |
| int i = 1; /* For the Gigabeat S, we mount the second partition */ |
| #else |
| int i = 0; |
| #endif |
| for (; volume != -1 && i<4; i++) |
| { |
| if (memchr(fat_partition_types, pinfo[i].type, |
| sizeof(fat_partition_types)) == NULL) |
| continue; /* not an accepted partition type */ |
| |
| #ifdef MAX_LOG_SECTOR_SIZE |
| int j; |
| |
| for (j = 1; j <= (MAX_LOG_SECTOR_SIZE/SECTOR_SIZE); j <<= 1) |
| { |
| if (!fat_mount(IF_MV2(volume,) IF_MV2(drive,) pinfo[i].start * j)) |
| { |
| pinfo[i].start *= j; |
| pinfo[i].size *= j; |
| mounted++; |
| vol_drive[volume] = drive; /* remember the drive for this volume */ |
| volume = get_free_volume(); /* prepare next entry */ |
| if (drive == 0) |
| disk_sector_multiplier = j; |
| break; |
| } |
| } |
| #else |
| if (!fat_mount(IF_MV2(volume,) IF_MV2(drive,) pinfo[i].start)) |
| { |
| mounted++; |
| vol_drive[volume] = drive; /* remember the drive for this volume */ |
| volume = get_free_volume(); /* prepare next entry */ |
| } |
| #endif |
| } |
| |
| if (mounted == 0 && volume != -1) /* none of the 4 entries worked? */ |
| { /* try "superfloppy" mode */ |
| DEBUGF("No partition found, trying to mount sector 0.\n"); |
| if (!fat_mount(IF_MV2(volume,) IF_MV2(drive,) 0)) |
| { |
| mounted = 1; |
| vol_drive[volume] = drive; /* remember the drive for this volume */ |
| } |
| } |
| return mounted; |
| } |
| |
| #ifdef HAVE_HOTSWAP |
| int disk_unmount(int drive) |
| { |
| int unmounted = 0; |
| int i; |
| for (i=0; i<NUM_VOLUMES; i++) |
| { |
| if (vol_drive[i] == drive) |
| { /* force releasing resources */ |
| vol_drive[i] = -1; /* mark unused */ |
| unmounted++; |
| release_files(i); |
| release_dirs(i); |
| fat_unmount(i, false); |
| } |
| } |
| |
| return unmounted; |
| } |
| #endif /* #ifdef HAVE_HOTSWAP */ |