From 83a1d8b8d3d54c58c11e2437d03a484c730942e5 Mon Sep 17 00:00:00 2001 From: "ol.sc" Date: Sat, 11 Feb 2012 01:50:48 +0000 Subject: [PATCH] Added support for VLIR programs. git-svn-id: svn://svn.cc65.org/cc65/trunk@5523 b7a2c559-68d2-44c3-8de9-860c34a00d81 --- targetutil/geos-apple/convert.c | 157 +++++++++++++++++++++++++------- 1 file changed, 123 insertions(+), 34 deletions(-) diff --git a/targetutil/geos-apple/convert.c b/targetutil/geos-apple/convert.c index cafccca99..5630ea6a7 100644 --- a/targetutil/geos-apple/convert.c +++ b/targetutil/geos-apple/convert.c @@ -5,6 +5,8 @@ #include #include +unsigned char info_signature[3] = {3, 21, 63 | 0x80}; + dhandle_t dhandle; struct dir_entry_t { @@ -43,22 +45,28 @@ union { unsigned char addr_hi[254]; unsigned char size_hi[2]; } content; -} index_block; +} index_block, master_block, vlir_block; union { unsigned char bytes[512]; struct { - unsigned char info_block[256]; + unsigned reserved; + unsigned char info_block[254]; unsigned char vlir_records[128]; struct dir_entry_t dir_entry; } content; } header_block; -static void err_exit(char *operation) +static void err_exit(char *operation, unsigned char oserr) { - fprintf(stderr, "%s - err:%02x - %s", - operation, (int)_oserror, _stroserror(_oserror)); + if (oserr) { + fprintf(stderr, "%s - err:%02x - %s", + operation, (int)_oserror, _stroserror(_oserror)); + } else { + fprintf(stderr, "%s", + operation); + } getchar(); exit(EXIT_FAILURE); } @@ -91,11 +99,11 @@ static unsigned get_dir_entry(char* p_name) of relative path and current drive for free */ dir = opendir(d_name); if (!dir) { - err_exit("opendir"); + err_exit("opendir", 1); } dirent = readdir(dir); if (!dirent) { - err_exit("readdir"); + err_exit("readdir", 1); } /* Field header_pointer directly follows field last_mod */ @@ -104,11 +112,11 @@ static unsigned get_dir_entry(char* p_name) /* DEV_NUM is set to the drive accessed above */ dhandle = dio_open(*(driveid_t*)0xBF30); if (!dhandle) { - err_exit("dio_open"); + err_exit("dio_open", 1); } if (dio_read(dhandle, cur_addr, &dir_block)) { - err_exit("dio_read.1"); + err_exit("dio_read.1", 1); } /* Get directory entry infos from directory header */ @@ -130,12 +138,12 @@ static unsigned get_dir_entry(char* p_name) cur_addr = dir_block.content.next_block; if (!cur_addr) { _mappederrno(0x46); - err_exit("dio_read.2"); + err_exit("dio_read.2", 1); } /* Read next directory block */ if (dio_read(dhandle, cur_addr, &dir_block)) { - err_exit("dio_read.3"); + err_exit("dio_read.3", 1); } /* Start with first entry in next block */ @@ -159,10 +167,12 @@ static unsigned get_dir_entry(char* p_name) int main(int argc, char* argv[]) { - char input[80]; - char* p_name; - unsigned dir_addr; - unsigned header_addr; + char input[80]; + char* p_name; + unsigned dir_addr; + unsigned header_addr; + unsigned char index; + unsigned long size; if (argc > 1) { p_name = argv[1]; @@ -174,47 +184,126 @@ int main(int argc, char* argv[]) dir_addr = get_dir_entry(p_name); if (dio_read(dhandle, dir_entry->key_pointer, &index_block)) { - err_exit("dio_read.4"); + err_exit("dio_read.4", 1); } header_addr = index_block.content.addr_lo[0] | index_block.content.addr_hi[0] << 8; if (dio_read(dhandle, header_addr, &header_block)) { - err_exit("dio_read.5"); + err_exit("dio_read.5", 1); + } + + for (index = 0; index < sizeof(info_signature); ++index) { + if (header_block.content.info_block[index] != info_signature[index]) { + err_exit("file signature mismatch", 0); + } + } + + if (header_block.content.dir_entry.storage_length.storage_type == 2) + { + printf("Sequential file\n"); + + memmove(&index_block.content.addr_lo[0], + &index_block.content.addr_lo[1], sizeof(index_block.content.addr_lo) - 1); + memmove(&index_block.content.addr_hi[0], + &index_block.content.addr_hi[1], sizeof(index_block.content.addr_hi) - 1); + + size = (unsigned long)(header_block.content.dir_entry.size[0]) | + (unsigned long)(header_block.content.dir_entry.size[1]) << 8 | + (unsigned long)(header_block.content.dir_entry.size[2]) << 16; + + } else { + unsigned vlir_addr; + unsigned long vlir_size; + unsigned char vlir_blocks; + unsigned char record = 0; + + printf("VLIR file\n"); + + index = 1; + size = 0; + + while (1) { + vlir_addr = index_block.content.addr_lo[index] | + index_block.content.addr_hi[index] << 8; + ++index; + + if (vlir_addr == 0) { + break; + } + + while (header_block.content.vlir_records[record] == 0xFF) { + master_block.content.addr_lo[record] = 0xFF; + master_block.content.addr_hi[record] = 0xFF; + ++record; + } + + master_block.content.addr_lo[record] = (unsigned char)(vlir_addr ); + master_block.content.addr_hi[record] = (unsigned char)(vlir_addr >> 8); + ++record; + + if (dio_read(dhandle, vlir_addr, &vlir_block)) { + err_exit("dio_read.6", 1); + } + + vlir_size = (unsigned long)(vlir_block.content.size_lo[1]) | + (unsigned long)(vlir_block.content.size_hi[1]) << 8 | + (unsigned long)(vlir_block.content.size_lo[0]) << 16 | + (unsigned long)(vlir_block.content.size_hi[0]) << 24; + + printf("VLIR %u size %lu bytes\n", record, vlir_size); + + vlir_blocks = (unsigned char)((vlir_size + 511) / 512); + + memcpy(&vlir_block.content.addr_lo[0], + &index_block.content.addr_lo[index], vlir_blocks); + memcpy(&vlir_block.content.addr_hi[0], + &index_block.content.addr_hi[index], vlir_blocks); + + if (dio_write(dhandle, vlir_addr, &vlir_block)) { + err_exit("dio_write.1", 1); + } + + index += vlir_blocks; + size += vlir_size; + } + + index_block = master_block; + } + + printf("File size %lu bytes\n", size); + + index_block.content.size_lo[1] = (unsigned char)(size ); + index_block.content.size_hi[1] = (unsigned char)(size >> 8); + index_block.content.size_lo[0] = (unsigned char)(size >> 16); + index_block.content.size_hi[0] = (unsigned char)(size >> 24); + + if (dio_write(dhandle, dir_entry->key_pointer, &index_block)) { + err_exit("dio_write.2", 1); } dir_entry->storage_length = header_block.content.dir_entry.storage_length; memcpy(dir_entry->file_name, header_block.content.dir_entry.file_name, 15); dir_entry->file_type = header_block.content.dir_entry.file_type; - memcpy(dir_entry->size, header_block.content.dir_entry.size, 3); + dir_entry->size[0] = (unsigned char)(size ); + dir_entry->size[1] = (unsigned char)(size >> 8); + dir_entry->size[2] = (unsigned char)(size >> 16); dir_entry->creation = header_block.content.dir_entry.creation; dir_entry->version = header_block.content.dir_entry.version; dir_entry->min_version = header_block.content.dir_entry.min_version; dir_entry->aux_type = header_addr; dir_entry->last_mod = header_block.content.dir_entry.last_mod; - memmove(&index_block.content.addr_lo[0], - &index_block.content.addr_lo[1], sizeof(index_block.content.addr_lo) - 1); - memmove(&index_block.content.addr_hi[0], - &index_block.content.addr_hi[1], sizeof(index_block.content.addr_hi) - 1); - - index_block.content.size_lo[1] = dir_entry->size[0]; - index_block.content.size_hi[1] = dir_entry->size[1]; - index_block.content.size_lo[0] = dir_entry->size[2]; - index_block.content.size_hi[0] = 0; - if (dio_write(dhandle, dir_addr, &dir_block)) { - err_exit("dio_write.1"); - } - - if (dio_write(dhandle, dir_entry->key_pointer, &index_block)) { - err_exit("dio_write.2"); + err_exit("dio_write.3", 1); } if (dio_close(dhandle)) { - err_exit("dio_close"); + err_exit("dio_close", 1); } + printf("Conversion successful"); + getchar(); return EXIT_SUCCESS; } \ No newline at end of file