From d1017e771e1bfad59061844c0f32a7306289e335 Mon Sep 17 00:00:00 2001 From: germeier Date: Sat, 28 Sep 2002 00:32:41 +0000 Subject: reading support on external memory should now be correct. --- libmpio/defs.h | 19 +++- libmpio/fat.c | 11 +- libmpio/io.c | 343 ++++++++++++++++++++++++++++++++++----------------------- libmpio/io.h | 15 ++- libmpio/mpio.c | 26 +++-- 5 files changed, 255 insertions(+), 159 deletions(-) (limited to 'libmpio') diff --git a/libmpio/defs.h b/libmpio/defs.h index 68a87ca..2cdb225 100644 --- a/libmpio/defs.h +++ b/libmpio/defs.h @@ -1,7 +1,7 @@ /* -*- linux-c -*- */ /* - * $Id: defs.h,v 1.10 2002/09/24 15:38:03 germeier Exp $ + * $Id: defs.h,v 1.11 2002/09/28 00:32:41 germeier Exp $ * * Library for USB MPIO-* * @@ -71,6 +71,17 @@ typedef enum { FTYPE_MUSIC = 0x01, typedef BYTE (*mpio_callback_t)(int, int) ; typedef BYTE (*mpio_callback_init_t)(mpio_mem_t, int, int) ; +/* zone lookup table */ +#define MPIO_ZONE_MAX 8 /* 8* 16MB = 128MB */ +#define MPIO_ZONE_PBLOCKS 1024 /* physical blocks per zone */ +#define MPIO_ZONE_LBLOCKS 1000 /* logical blocks per zone */ +typedef WORD mpio_zonetable_t[MPIO_ZONE_MAX][MPIO_ZONE_PBLOCKS]; + +#define MPIO_BLOCK_FREE 0xffff +#define MPIO_BLOCK_DEFECT 0xffee +#define MPIO_BLOCK_CIS 0xaaaa +#define MPIO_BLOCK_NOT_FOUND 0xcccccccc + /* filenames */ #define MPIO_FILENAME_LEN 129 typedef BYTE mpio_filename_t[MPIO_FILENAME_LEN]; @@ -97,9 +108,6 @@ typedef BYTE mpio_filename_t[MPIO_FILENAME_LEN]; #define CMD_SIZE 0x40 -#define OFFSET_CIS 0x20 /* always fixed? :salmoon */ -#define OFFSET_MBR 0x40 /* always fixed? :salmoon */ - #define INFO_LINE 81 /* error codes */ @@ -183,6 +191,9 @@ typedef struct { int max_blocks; BYTE * spare; + /* lookup table for phys.<->log. block mapping */ + mpio_zonetable_t zonetable; + /* seems to be a fixed size according to the Samsung documentation */ BYTE dir[DIR_SIZE]; /* file index */ diff --git a/libmpio/fat.c b/libmpio/fat.c index 1a09f7b..5690206 100644 --- a/libmpio/fat.c +++ b/libmpio/fat.c @@ -1,6 +1,6 @@ /* * - * $Id: fat.c,v 1.14 2002/09/24 15:38:03 germeier Exp $ + * $Id: fat.c,v 1.15 2002/09/28 00:32:41 germeier Exp $ * * Library for USB MPIO-* * @@ -112,16 +112,15 @@ mpio_bootblocks_read (mpio_t *m, mpio_mem_t mem) /* TODO: check a few things more, just to be sure */ /* read CIS (just in case it might me usefull) */ - /* fixed @ offset 0x20 */ - if (mpio_io_sector_read(m, mem, 0x20, sm->cis)) + if (mpio_io_sector_read(m, mem, MPIO_BLOCK_CIS, sm->cis)) { debug("error reading CIS\n"); return 1; } /* read MBR */ - /* fixed @ offset 0x40 */ - if (mpio_io_sector_read(m, mem, 0x40, sm->mbr)) + /* the MBR is always located @ logical block 0, sector 0! */ + if (mpio_io_sector_read(m, mem, 0, sm->mbr)) { debug("error reading MBR\n"); return 1; @@ -142,7 +141,7 @@ mpio_bootblocks_read (mpio_t *m, mpio_mem_t mem) cylinder = (int)((*(pe+0x02) >> 6) * 0x100 + *(pe + 0x03)); sm->pbr_offset=(cylinder * sm->geo.NumHead + head ) * - sm->geo.NumSector + sector - 1 + OFFSET_MBR; + sm->geo.NumSector + sector - 1 /*+ OFFSET_MBR */; /* read PBR */ if (mpio_io_sector_read(m, mem, sm->pbr_offset, sm->pbr)) diff --git a/libmpio/io.c b/libmpio/io.c index 7a5b58e..6a11866 100644 --- a/libmpio/io.c +++ b/libmpio/io.c @@ -2,7 +2,7 @@ /* * - * $Id: io.c,v 1.16 2002/09/23 22:38:03 germeier Exp $ + * $Id: io.c,v 1.17 2002/09/28 00:32:41 germeier Exp $ * * Library for USB MPIO-* * @@ -45,11 +45,11 @@ #include "ecc.h" BYTE model2externalmem(mpio_model_t); -WORD index2blockaddress(WORD); -WORD cluster2blockaddress(DWORD, BYTE); -int cluster2block(int mem, int sector); +WORD blockaddress_encode(WORD); +WORD blockaddress_decode(BYTE *); void fatentry2hw(mpio_fatentry_t *, BYTE *, DWORD *); +/* small hack to handle external addressing on different MPIO models */ BYTE model2externalmem(mpio_model_t model) { @@ -68,91 +68,10 @@ model2externalmem(mpio_model_t model) return m; } -void -fatentry2hw(mpio_fatentry_t *f, BYTE *chip, DWORD *address) -{ - mpio_smartmedia_t *sm; - - if (f->mem == MPIO_INTERNAL_MEM) - { - sm = &f->m->internal; - /* hexdump((char *)&f->entry, 4); */ - /* hexdump((char *)&f->hw_address, 4); */ - *chip = f->hw_address / 0x1000000; - *address = f->hw_address & 0x0ffffff; - } - if (f->mem == MPIO_EXTERNAL_MEM) - { - sm = &f->m->external; - *chip = MPIO_EXTERNAL_MEM; - *address = cluster2block(sm->size, f->entry); - *address *= BLOCK_SECTORS; - - /* add offset to start of "data" area! */ - *address += (sm->dir_offset + DIR_NUM - (2 * BLOCK_SECTORS)); - } - return; -} - - - -/* - * HELP! - * - * somebody explain me these values!!! - * - */ - -int -cluster2block(int mem, int sector) -{ - int a = sector; - - /* No Zone-based block management for SmartMedia below 32MB !!*/ - - /* The jumps are defect sectors on a specific SM medium and have to - be handled with a proper logic block -> physical block which - include a bad block management */ - -/* if (mem == 32) */ -/* { */ -/* if (sector >= 830) */ -/* a++; */ -/* if (a >= 1001) */ -/* a += 21; */ -/* } */ - -/* if (mem == 64) */ -/* { */ -/* /\* I'm so large in *not* knowing! *\/ */ -/* if (sector >= 89) */ -/* a++; */ -/* if (a >= 1000) */ -/* a += 21; */ -/* if (a >= 2021) */ -/* a += 24; */ -/* if (a >= 3045) */ -/* a += 24; */ -/* /\* WHAT? *\/ */ -/* if (a >= 3755) */ -/* a++; */ -/* } */ - - if (mem > 16) - { - /* two blocks are already spent elsewhere */ - /* question is: where (CIS and ??) */ - if (sector >= 998) - a += 22; - /* ... and then add 24 empty blocks every 1000 sectors */ - a += ((sector - 998) / 1000 * 24); - } - - return a; -} +/* encoding and decoding of blockaddresses */ WORD -index2blockaddress(WORD ba) +blockaddress_encode(WORD ba) { WORD addr; BYTE p = 0, c = 0; @@ -182,48 +101,195 @@ index2blockaddress(WORD ba) return addr; } + WORD -cluster2blockaddress(DWORD index, BYTE size) +blockaddress_decode(BYTE *entry) { - DWORD ba; - WORD block_address; + WORD ba; + WORD value; + BYTE p=0, high, low; + int i, t; + + /* check for "easy" defect block */ + t=1; + for(i=0; i<0x10; i++) + if (entry[i] != 0) + t=0; + if (t) + return MPIO_BLOCK_DEFECT; + + /* check for "easy" defect block */ + t=1; + for(i=0; i<0x10; i++) + if (entry[i] != 0xff) + t=0; + if (t) + return MPIO_BLOCK_FREE; + + /* check for "strange" errors */ + if ((entry[6] != entry[11]) || + (entry[7] != entry[12])) + { + debug("error: different block addresses found:\n"); + hexdumpn(1, entry, 0x10); + return MPIO_BLOCK_DEFECT; + } + + ba = entry[6] * 0x100 + entry[7]; + + /* special values */ + if (ba == 0xffff) + return MPIO_BLOCK_DEFECT; + if (ba == 0x0000) + return MPIO_BLOCK_CIS; + + /* check parity */ + value = ba; + while (value) + { + if (value & 0x01) + p ^= 0x01; + + value /= 2; + } + if (p) + { + debug("parity error found in block address: %2x\n", ba); + return MPIO_BLOCK_DEFECT; + } + + high = ((ba / 0x100) & 0x07); + low = ((ba & 0x0ff) / 2); + + return (high * 0x80 + low); +} - /* The jumps are defect sectors on a specific SM medium and have to - be handled with a proper logic block -> physical block which - include a bad block management */ +/* foobar */ + +void +fatentry2hw(mpio_fatentry_t *f, BYTE *chip, DWORD *address) +{ + mpio_smartmedia_t *sm; - if (index<0x40) + if (f->mem == MPIO_INTERNAL_MEM) + { + sm = &f->m->internal; + /* hexdump((char *)&f->entry, 4); */ + /* hexdump((char *)&f->hw_address, 4); */ + *chip = f->hw_address / 0x1000000; + *address = f->hw_address & 0x0ffffff; + } + if (f->mem == MPIO_EXTERNAL_MEM) { - block_address=0; + sm = &f->m->external; + *chip = MPIO_EXTERNAL_MEM; + *address = mpio_zone_block_find(f->m, f->mem, f->entry); + debugn(3, "mager: %06x (logical: %04x)\n", *address, f->entry); + } + return; +} + +/* + * zone management + */ + +int +mpio_zone_init(mpio_t *m, mpio_cmd_t mem) +{ + mpio_smartmedia_t *sm; + int i; + int zone, block, e; + + if (mem != MPIO_EXTERNAL_MEM) + { + debug("called function with wrong memory selection!\n"); + return -1; + } + sm = &m->external; + + for(i=0; imax_blocks; i++) + { + zone = i / MPIO_ZONE_PBLOCKS; + block= i % MPIO_ZONE_PBLOCKS; + + e = i * 0x10; + + sm->zonetable[zone][block]=blockaddress_decode(sm->spare+e); + + hexdumpn(4, sm->spare+e, 0x10); + debugn(2, "decoded: %03x\n", sm->zonetable[zone][block]); + } + +} + +DWORD +mpio_zone_block_find(mpio_t *m, mpio_cmd_t mem, DWORD lblock) +{ + mpio_smartmedia_t *sm; + int i, f, v; + int zone, block; + + if (mem != MPIO_EXTERNAL_MEM) + { + debug("called function with wrong memory selection!\n"); + return -1; + } + sm = &m->external; + + /* OK, I can't explain this right now, but it does work, + * see Software Driver v3.0, page 31 + */ + v = lblock + (sm->size/64); + + if (lblock == MPIO_BLOCK_CIS) + { + zone = 0; + block = MPIO_BLOCK_CIS; } else { - if (index >= 0x8000) - { - ba = ((index % 0x8000) / 0x20); -/* if (size == 64) */ -/* { */ -/* if (index >= 0x1d5e0) */ -/* ba--; */ -/* } */ - } else { - ba = (index / 0x20) - 2; -/* if (size == 32) */ -/* { */ -/* if (ba >= 0x33f) */ -/* ba--; */ -/* } */ -/* if (size == 64) */ -/* { */ -/* if (ba >= 0x05b) */ -/* ba--; */ -/* } */ + zone = v / MPIO_ZONE_LBLOCKS; + block = v % MPIO_ZONE_LBLOCKS; + } + + f=0; + for (i=(MPIO_ZONE_PBLOCKS-1); i >=0; i--) + { + + if (sm->zonetable[zone][i]==block) + { + f++; + v=i; } - block_address= index2blockaddress(ba); - debugn(2, "block-foo: %06x %04x %04x\n", index, ba, block_address); + } + + if (f>1) + debug("found more than one block, using first one\n"); + + if (!f) + { + debug("block not found\n"); + return MPIO_BLOCK_NOT_FOUND; } - return block_address; + return ((zone * BLOCK_SECTORS * MPIO_ZONE_PBLOCKS ) + v * BLOCK_SECTORS); } +void +mpio_zone_block_set(mpio_t *m, mpio_cmd_t mem, DWORD pblock) +{ + int zone, block, pb; + + pb = pblock / BLOCK_SECTORS; + zone = pb / MPIO_ZONE_PBLOCKS; + block = pb % MPIO_ZONE_PBLOCKS; + + m->external.zonetable[zone][block] = MPIO_BLOCK_FREE ; + +} + +DWORD +mpio_zone_block_free(mpio_t *m, mpio_cmd_t mem, DWORD lblock) +{ +} /* * low-low level functions @@ -419,7 +485,7 @@ int mpio_io_sector_read(mpio_t *m, BYTE mem, DWORD index, BYTE *output) { mpio_smartmedia_t *sm=0; - + DWORD sector; int nwrite, nread; BYTE cmdpacket[CMD_SIZE], recvbuff[SECTOR_TRANS]; @@ -431,7 +497,24 @@ mpio_io_sector_read(mpio_t *m, BYTE mem, DWORD index, BYTE *output) exit (-1); } - mpio_io_set_cmdpacket (m, GET_SECTOR, mem, index, sm->size, 0, cmdpacket); + if (mem == MPIO_INTERNAL_MEM) + { + sector = index; + } else { + /* find the correct physical block first! */ + if (index == MPIO_BLOCK_CIS) + { + sector = mpio_zone_block_find(m, mem, index); + } else { + sector = mpio_zone_block_find(m, mem, /* yuck */ + ((index / 0x20) - (sm->size/64))); + sector+= (index % 0x20); + } + } + + debugn (2, "sector: %8x (%06x)\n", index, sector); + + mpio_io_set_cmdpacket (m, GET_SECTOR, mem, sector, sm->size, 0, cmdpacket); debugn (5, "\n>>> MPIO\n"); hexdump (cmdpacket, sizeof(cmdpacket)); @@ -533,8 +616,8 @@ mpio_io_sector_write(mpio_t *m, BYTE mem, DWORD index, BYTE *input) memset(sendbuff + SECTOR_SIZE, 0xff, 0x10); memcpy(sendbuff, input, SECTOR_SIZE); - if (mem==MPIO_EXTERNAL_MEM) - block_address = cluster2blockaddress(index, sm->size); +/* if (mem==MPIO_EXTERNAL_MEM) */ +/* block_address = cluster2blockaddress(index, sm->size); */ { /* generate ECC information for spare area ! */ @@ -584,7 +667,6 @@ mpio_io_block_read(mpio_t *m, BYTE mem, mpio_fatentry_t *f, BYTE *output) BYTE chip; DWORD address; BYTE cmdpacket[CMD_SIZE], recvbuff[BLOCK_TRANS]; - DWORD block_address, ba; if (mem == MPIO_INTERNAL_MEM) sm = &m->internal; if (mem == MPIO_EXTERNAL_MEM) sm = &m->external; @@ -630,18 +712,6 @@ mpio_io_block_read(mpio_t *m, BYTE mem, mpio_fatentry_t *f, BYTE *output) ((recvbuff +(i * SECTOR_TRANS) + SECTOR_SIZE + 8)))) debug ("ECC error @ (%02x : %06x)\n", chip, address); - - if (i==0) - { - block_address = cluster2blockaddress(address, sm->size); - - ba = recvbuff[(i * SECTOR_TRANS) + SECTOR_SIZE + 0x06] * 0x100 + - recvbuff[(i * SECTOR_TRANS) + SECTOR_SIZE + 0x07] ; - if (block_address != ba) - debugn(2,"different block_addresses during read: %04x vs. %04x\n", - block_address, ba); - } - } memcpy(output + (i * SECTOR_SIZE), @@ -739,6 +809,7 @@ mpio_io_block_delete(mpio_t *m, BYTE mem, mpio_fatentry_t *f) if (mem == MPIO_EXTERNAL_MEM) sm = &m->external; fatentry2hw(f, &chip, &address); + mpio_zone_block_free(m, mem, address); /* Send command packet to MPIO */ @@ -818,7 +889,7 @@ mpio_io_block_write(mpio_t *m, BYTE mem, mpio_fatentry_t *f, BYTE *data) /* fill in block information */ if (mem == MPIO_EXTERNAL_MEM) { - block_address = cluster2blockaddress(address, sm->size); +/* block_address = cluster2blockaddress(address, sm->size); */ ba = (block_address / 0x100) & 0xff; sendbuff[(i * SECTOR_TRANS) + SECTOR_SIZE + 0x06] = ba; diff --git a/libmpio/io.h b/libmpio/io.h index cd8b70c..44b3cbe 100644 --- a/libmpio/io.h +++ b/libmpio/io.h @@ -2,7 +2,7 @@ /* * - * $Id: io.h,v 1.8 2002/09/23 22:38:03 germeier Exp $ + * $Id: io.h,v 1.9 2002/09/28 00:32:41 germeier Exp $ * * Library for USB MPIO-* * @@ -33,6 +33,19 @@ #include "defs.h" +/* phys.<->log. block mapping */ +int mpio_zone_init(mpio_t *, mpio_cmd_t); +/* context, memory bank, logical block */ +/* returns address of physical block! */ +DWORD mpio_zone_block_find(mpio_t *, mpio_cmd_t, DWORD); +/* context, memory bank, logical block, physical address */ +/* mark physical block free (internal maintenance) */ +void mpio_zone_block_set(mpio_t *, mpio_cmd_t, DWORD); +/* context, memory bank, logical block */ +/* find used physical block and mark it as unused! */ +/* returns address of physical block! (to delete the physical block!) */ +DWORD mpio_zone_block_free(mpio_t *, mpio_cmd_t, DWORD); + /* real I/O */ int mpio_io_set_cmdpacket(mpio_t *, mpio_cmd_t, mpio_mem_t, DWORD, BYTE, BYTE, BYTE *); diff --git a/libmpio/mpio.c b/libmpio/mpio.c index 6e2fd45..330a744 100644 --- a/libmpio/mpio.c +++ b/libmpio/mpio.c @@ -1,6 +1,6 @@ /* * - * $Id: mpio.c,v 1.27 2002/09/24 15:38:03 germeier Exp $ + * $Id: mpio.c,v 1.28 2002/09/28 00:32:41 germeier Exp $ * * Library for USB MPIO-* * @@ -193,11 +193,6 @@ mpio_init_external(mpio_t *m) "mpio-devel@lists.sourceforge.net\n"); } - mpio_bootblocks_read(m, MPIO_EXTERNAL_MEM); - sm->fat = malloc(SECTOR_SIZE*sm->fat_size); - mpio_fat_read(m, MPIO_EXTERNAL_MEM, NULL); - mpio_rootdir_read(m, MPIO_EXTERNAL_MEM); - /* for reading the spare area later! */ sm->max_blocks = sm->size/16*1024; /* 1 cluster == 16 KB */ sm->spare = malloc(sm->max_blocks * 0x10); @@ -272,11 +267,20 @@ mpio_init(mpio_callback_init_t progress_callback) mpio_fat_read(new_mpio, MPIO_INTERNAL_MEM, progress_callback); /* read the spare area (for block mapping) */ - sm = &new_mpio->external; - mpio_io_spare_read(new_mpio, MPIO_EXTERNAL_MEM, 0, - sm->size, 0, sm->spare, - (sm->max_blocks * 0x10), progress_callback); + if (new_mpio->external.id) + { + sm = &new_mpio->external; + mpio_io_spare_read(new_mpio, MPIO_EXTERNAL_MEM, 0, + sm->size, 0, sm->spare, + (sm->max_blocks * 0x10), progress_callback); + mpio_zone_init(new_mpio, MPIO_EXTERNAL_MEM); + mpio_bootblocks_read(new_mpio, MPIO_EXTERNAL_MEM); + sm->fat = malloc(SECTOR_SIZE*sm->fat_size); + mpio_fat_read(new_mpio, MPIO_EXTERNAL_MEM, NULL); + mpio_rootdir_read(new_mpio, MPIO_EXTERNAL_MEM); + } + return new_mpio; } @@ -400,8 +404,6 @@ mpio_file_get(mpio_t *m, mpio_mem_t mem, mpio_filename_t filename, do { - debugn(2, "entry: %4x\n", f->entry); - mpio_io_block_read(m, mem, f, block); if (filesize > BLOCK_SIZE) { -- cgit v1.2.3