| version 1.3, 2004/01/09 07:27:15 | version 1.18, 2005/05/20 13:59:46 | 
| Line 1 | Line 1 | 
 |  | #include        "compiler.h" | 
 |  |  | 
 |  | #if defined(SUPPORT_HOSTDRV) | 
 |  |  | 
 | /* | /* | 
 | ゲストOS(DOS)からホストOS(Win)にアクセスするの〜 | ゲストOS(DOS)からホストOS(Win)にアクセスするの〜 | 
| Line 6 | Line 9 | 
 | 更に、手抜き版だし(マテ | 更に、手抜き版だし(マテ | 
 | */ | */ | 
 |  |  | 
 | #include        "compiler.h" |  | 
 | #include        "dosio.h" | #include        "dosio.h" | 
 | #include        "cpucore.h" | #include        "cpucore.h" | 
 | #include        "pccore.h" | #include        "pccore.h" | 
 |  | #include        "iocore.h" | 
 | #include        "hostdrv.h" | #include        "hostdrv.h" | 
 | #include        "hostdrvs.h" | #include        "hostdrvs.h" | 
 | #include        "hostdrv.tbl" | #include        "hostdrv.tbl" | 
| Line 21 | Line 24 | 
 | #define ROOTPATH_NAME           "\\\\HOSTDRV\\" | #define ROOTPATH_NAME           "\\\\HOSTDRV\\" | 
 | #define ROOTPATH_SIZE           (sizeof(ROOTPATH_NAME) - 1) | #define ROOTPATH_SIZE           (sizeof(ROOTPATH_NAME) - 1) | 
 |  |  | 
| static const char ROOTPATH[ROOTPATH_SIZE] = ROOTPATH_NAME; | static const char ROOTPATH[ROOTPATH_SIZE + 1] = ROOTPATH_NAME; | 
| static const HDRVDIR hdd_volume = {"_HOSTDRIVE_", 0, 0x08}; | static const HDRVDIR hdd_volume = {"_HOSTDRIVE_", 0, 0, 0, 0x08, {0}, {0}}; | 
| static const HDRVDIR hdd_owner  = {".          ", 0, 0x10}; | static const HDRVDIR hdd_owner  = {".          ", 0, 0, 0, 0x10, {0}, {0}}; | 
| static const HDRVDIR hdd_parent = {"..         ", 0, 0x10}; | static const HDRVDIR hdd_parent = {"..         ", 0, 0, 0, 0x10, {0}, {0}}; | 
|  |  | 
 |  |  | 
 | //      see int2159-BX0000 | //      see int2159-BX0000 | 
 | enum { | enum { | 
| Line 64  static void fetch_if4dos(void) { | Line 66  static void fetch_if4dos(void) { | 
 | REG16   seg; | REG16   seg; | 
 | IF4DOS  if4dos; | IF4DOS  if4dos; | 
 |  |  | 
| off = i286_memoryread_w(IF4DOSPTR_ADDR); | off = MEMR_READ16(IF4DOSPTR_SEG, IF4DOSPTR_OFF + 0); | 
| seg = i286_memoryread_w(IF4DOSPTR_ADDR + 2); | seg = MEMR_READ16(IF4DOSPTR_SEG, IF4DOSPTR_OFF + 2); | 
| i286_memstr_read(seg, off, &if4dos, sizeof(if4dos)); | MEMR_READS(seg, off, &if4dos, sizeof(if4dos)); | 
| hostdrv.drive_no = if4dos.drive_no; | hostdrv.stat.drive_no = if4dos.drive_no; | 
| hostdrv.dosver_major = if4dos.dosver_major; | hostdrv.stat.dosver_major = if4dos.dosver_major; | 
| hostdrv.dosver_minor = if4dos.dosver_minor; | hostdrv.stat.dosver_minor = if4dos.dosver_minor; | 
| hostdrv.sda_off = LOADINTELWORD(if4dos.sda_off); | hostdrv.stat.sda_off = LOADINTELWORD(if4dos.sda_off); | 
| hostdrv.sda_seg = LOADINTELWORD(if4dos.sda_seg); | hostdrv.stat.sda_seg = LOADINTELWORD(if4dos.sda_seg); | 
|  |  | 
| TRACEOUT(("hostdrv.drive_no = %d", if4dos.drive_no)); | TRACEOUT(("hostdrv:drive_no = %d", if4dos.drive_no)); | 
| TRACEOUT(("hostdrv.dosver = %d.%.2d", if4dos.dosver_major, if4dos.dosver_minor)); | TRACEOUT(("hostdrv:dosver = %d.%.2d", if4dos.dosver_major, if4dos.dosver_minor)); | 
| TRACEOUT(("hostdrv.sda = %.4x:%.4x", hostdrv.sda_seg, hostdrv.sda_off)); | TRACEOUT(("hostdrv.sda = %.4x:%.4x", hostdrv.stat.sda_seg, hostdrv.stat.sda_off)); | 
 | } | } | 
 |  |  | 
 |  |  | 
 | static void fetch_intr_regs(INTRST is) { | static void fetch_intr_regs(INTRST is) { | 
 |  |  | 
| i286_memstr_read(CPU_SS, CPU_BP, &is->r, sizeof(is->r)); | MEMR_READS(CPU_SS, CPU_BP, &is->r, sizeof(is->r)); | 
 | } | } | 
 |  |  | 
 | static void store_intr_regs(INTRST is) { | static void store_intr_regs(INTRST is) { | 
 |  |  | 
| i286_memstr_write(CPU_SS, CPU_BP, &is->r, sizeof(is->r)); | MEMR_WRITES(CPU_SS, CPU_BP, &is->r, sizeof(is->r)); | 
 | } | } | 
 |  |  | 
 |  |  | 
| Line 95  static void fetch_sda_currcds(SDACDS sc) | Line 97  static void fetch_sda_currcds(SDACDS sc) | 
 | REG16   off; | REG16   off; | 
 | REG16   seg; | REG16   seg; | 
 |  |  | 
| if (hostdrv.dosver_major == 3) { | if (hostdrv.stat.dosver_major == 3) { | 
| i286_memstr_read(hostdrv.sda_seg, hostdrv.sda_off, | MEMR_READS(hostdrv.stat.sda_seg, hostdrv.stat.sda_off, | 
 | &sc->ver3.sda, sizeof(sc->ver3.sda)); | &sc->ver3.sda, sizeof(sc->ver3.sda)); | 
 | off = LOADINTELWORD(sc->ver3.sda.cdsptr.off); | off = LOADINTELWORD(sc->ver3.sda.cdsptr.off); | 
 | seg = LOADINTELWORD(sc->ver3.sda.cdsptr.seg); | seg = LOADINTELWORD(sc->ver3.sda.cdsptr.seg); | 
| i286_memstr_read(seg, off, &sc->ver3.cds, sizeof(sc->ver3.cds)); | MEMR_READS(seg, off, &sc->ver3.cds, sizeof(sc->ver3.cds)); | 
 | } | } | 
 | else { | else { | 
| i286_memstr_read(hostdrv.sda_seg, hostdrv.sda_off, | MEMR_READS(hostdrv.stat.sda_seg, hostdrv.stat.sda_off, | 
 | &sc->ver4.sda, sizeof(sc->ver4.sda)); | &sc->ver4.sda, sizeof(sc->ver4.sda)); | 
 | off = LOADINTELWORD(sc->ver4.sda.cdsptr.off); | off = LOADINTELWORD(sc->ver4.sda.cdsptr.off); | 
 | seg = LOADINTELWORD(sc->ver4.sda.cdsptr.seg); | seg = LOADINTELWORD(sc->ver4.sda.cdsptr.seg); | 
| i286_memstr_read(seg, off, &sc->ver4.cds, sizeof(sc->ver4.cds)); | MEMR_READS(seg, off, &sc->ver4.cds, sizeof(sc->ver4.cds)); | 
 | } | } | 
 | } | } | 
 |  |  | 
| Line 116  static void store_sda_currcds(SDACDS sc) | Line 118  static void store_sda_currcds(SDACDS sc) | 
 | REG16   off; | REG16   off; | 
 | REG16   seg; | REG16   seg; | 
 |  |  | 
| if (hostdrv.dosver_major == 3) { | if (hostdrv.stat.dosver_major == 3) { | 
| i286_memstr_write(hostdrv.sda_seg, hostdrv.sda_off, | MEMR_WRITES(hostdrv.stat.sda_seg, hostdrv.stat.sda_off, | 
 | &sc->ver3.sda, sizeof(sc->ver3.sda)); | &sc->ver3.sda, sizeof(sc->ver3.sda)); | 
 | off = LOADINTELWORD(sc->ver3.sda.cdsptr.off); | off = LOADINTELWORD(sc->ver3.sda.cdsptr.off); | 
 | seg = LOADINTELWORD(sc->ver3.sda.cdsptr.seg); | seg = LOADINTELWORD(sc->ver3.sda.cdsptr.seg); | 
| i286_memstr_write(seg, off, &sc->ver3.cds, sizeof(sc->ver3.cds)); | MEMR_WRITES(seg, off, &sc->ver3.cds, sizeof(sc->ver3.cds)); | 
 | } | } | 
 | else { | else { | 
| i286_memstr_write(hostdrv.sda_seg, hostdrv.sda_off, | MEMR_WRITES(hostdrv.stat.sda_seg, hostdrv.stat.sda_off, | 
 | &sc->ver4.sda, sizeof(sc->ver4.sda)); | &sc->ver4.sda, sizeof(sc->ver4.sda)); | 
 | off = LOADINTELWORD(sc->ver4.sda.cdsptr.off); | off = LOADINTELWORD(sc->ver4.sda.cdsptr.off); | 
 | seg = LOADINTELWORD(sc->ver4.sda.cdsptr.seg); | seg = LOADINTELWORD(sc->ver4.sda.cdsptr.seg); | 
| i286_memstr_write(seg, off, &sc->ver4.cds, sizeof(sc->ver4.cds)); | MEMR_WRITES(seg, off, &sc->ver4.cds, sizeof(sc->ver4.cds)); | 
 | } | } | 
 | } | } | 
 |  |  | 
| Line 140  static void fetch_sft(INTRST is, SFTREC | Line 142  static void fetch_sft(INTRST is, SFTREC | 
 |  |  | 
 | off = LOADINTELWORD(is->r.w.di); | off = LOADINTELWORD(is->r.w.di); | 
 | seg = LOADINTELWORD(is->r.w.es); | seg = LOADINTELWORD(is->r.w.es); | 
| i286_memstr_read(seg, off, sft, sizeof(_SFTREC)); | MEMR_READS(seg, off, sft, sizeof(_SFTREC)); | 
 | } | } | 
 |  |  | 
 | static void store_sft(INTRST is, SFTREC sft) { | static void store_sft(INTRST is, SFTREC sft) { | 
| Line 150  static void store_sft(INTRST is, SFTREC | Line 152  static void store_sft(INTRST is, SFTREC | 
 |  |  | 
 | off = LOADINTELWORD(is->r.w.di); | off = LOADINTELWORD(is->r.w.di); | 
 | seg = LOADINTELWORD(is->r.w.es); | seg = LOADINTELWORD(is->r.w.es); | 
| i286_memstr_write(seg, off, sft, sizeof(_SFTREC)); | MEMR_WRITES(seg, off, sft, sizeof(_SFTREC)); | 
 | } | } | 
 |  |  | 
 |  |  | 
| Line 160  static void store_srch(INTRST is) { | Line 162  static void store_srch(INTRST is) { | 
 |  |  | 
 | // SDA内のSRCHRECにセット | // SDA内のSRCHRECにセット | 
 | srchrec = is->srchrec_ptr; | srchrec = is->srchrec_ptr; | 
| srchrec->drive_no = 0xc0 | hostdrv.drive_no; | srchrec->drive_no = 0xc0 | hostdrv.stat.drive_no; | 
 | CopyMemory(srchrec->srch_mask, is->fcbname_ptr, 11); | CopyMemory(srchrec->srch_mask, is->fcbname_ptr, 11); | 
 | srchrec->attr_mask = *is->srch_attr_ptr; | srchrec->attr_mask = *is->srch_attr_ptr; | 
 | STOREINTELWORD(srchrec->dir_entry_no, ((UINT16)-1)); | STOREINTELWORD(srchrec->dir_entry_no, ((UINT16)-1)); | 
| Line 171  static void store_dir(INTRST is, const H | Line 173  static void store_dir(INTRST is, const H | 
 |  |  | 
 | DIRREC  dirrec; | DIRREC  dirrec; | 
 | UINT8   attr; | UINT8   attr; | 
 |  | UINT16  reg; | 
 |  |  | 
 | // SDA内のDIRRECにセット | // SDA内のDIRRECにセット | 
 | dirrec = is->dirrec_ptr; | dirrec = is->dirrec_ptr; | 
| Line 180  static void store_dir(INTRST is, const H | Line 183  static void store_dir(INTRST is, const H | 
 | attr |= 0x01; | attr |= 0x01; | 
 | } | } | 
 | dirrec->file_attr = attr; | dirrec->file_attr = attr; | 
| STOREINTELDWORD(dirrec->file_time, 0);          // di->datetime | reg = 0; | 
| STOREINTELDWORD(dirrec->start_sector, ((UINT32)-1)); | if (di->caps & FLICAPS_TIME) { | 
|  | reg |= (di->time.hour & 0x1f) << 11; | 
|  | reg |= (di->time.minute & 0x3f) << 5; | 
|  | reg |= (di->time.second & 0x3e) >> 1; | 
|  | } | 
|  | STOREINTELWORD(dirrec->file_time, reg); | 
|  | reg = 0; | 
|  | if (di->caps & FLICAPS_DATE) { | 
|  | reg |= ((di->date.year - 1980) & 0x7f) << 9; | 
|  | reg |= (di->date.month & 0x0f) << 5; | 
|  | reg |= di->date.day & 0x1f; | 
|  | } | 
|  | STOREINTELWORD(dirrec->file_date, reg); | 
|  | STOREINTELWORD(dirrec->start_sector, ((UINT16)-1)); | 
 | STOREINTELDWORD(dirrec->file_size, di->size); | STOREINTELDWORD(dirrec->file_size, di->size); | 
 | } | } | 
 |  |  | 
 | static void fill_sft(INTRST is, SFTREC sft, UINT num, HDRVDIR *di) { | static void fill_sft(INTRST is, SFTREC sft, UINT num, HDRVDIR *di) { | 
 |  |  | 
 | UINT8   attr; | UINT8   attr; | 
 |  | UINT16  reg; | 
 |  |  | 
 | attr = di->attr; | attr = di->attr; | 
 | if (!IS_PERMITWRITE) { | if (!IS_PERMITWRITE) { | 
| Line 195  static void fill_sft(INTRST is, SFTREC s | Line 212  static void fill_sft(INTRST is, SFTREC s | 
 | } | } | 
 | sft->file_attr = attr; | sft->file_attr = attr; | 
 | STOREINTELWORD(sft->start_sector, (UINT16)num); | STOREINTELWORD(sft->start_sector, (UINT16)num); | 
| STOREINTELDWORD(sft->file_time, 0);                                     // di->datetime |  | 
|  | reg = 0; | 
|  | if (di->caps & FLICAPS_TIME) { | 
|  | reg |= (di->time.hour & 0x1f) << 11; | 
|  | reg |= (di->time.minute & 0x3f) << 5; | 
|  | reg |= (di->time.second & 0x3e) >> 1; | 
|  | } | 
|  | STOREINTELWORD(sft->file_time, reg); | 
|  | reg = 0; | 
|  | if (di->caps & FLICAPS_DATE) { | 
|  | reg |= ((di->date.year - 1980) & 0x7f) << 9; | 
|  | reg |= (di->date.month & 0x0f) << 5; | 
|  | reg |= di->date.day & 0x1f; | 
|  | } | 
|  | STOREINTELWORD(sft->file_date, reg); | 
 | STOREINTELDWORD(sft->file_size, di->size); | STOREINTELDWORD(sft->file_size, di->size); | 
 | STOREINTELWORD(sft->dir_sector, (UINT16)-1); | STOREINTELWORD(sft->dir_sector, (UINT16)-1); | 
 | sft->dir_entry_no = (UINT8)-1; | sft->dir_entry_no = (UINT8)-1; | 
 | CopyMemory(sft->file_name, is->fcbname_ptr, 11); | CopyMemory(sft->file_name, is->fcbname_ptr, 11); | 
 |  | TRACEOUT(("open -> size %d", di->size)); | 
 | } | } | 
 |  |  | 
 | static void init_sft(SFTREC sft) { | static void init_sft(SFTREC sft) { | 
| Line 210  static void init_sft(SFTREC sft) { | Line 242  static void init_sft(SFTREC sft) { | 
 | else { | else { | 
 | sft->open_mode[0] &= 0x0f; | sft->open_mode[0] &= 0x0f; | 
 | } | } | 
| sft->dev_info_word[0] = (BYTE)(0x40 | hostdrv.drive_no); | sft->dev_info_word[0] = (UINT8)(0x40 | hostdrv.stat.drive_no); | 
 | sft->dev_info_word[1] = 0x80; | sft->dev_info_word[1] = 0x80; | 
 | STOREINTELDWORD(sft->dev_drvr_ptr, 0); | STOREINTELDWORD(sft->dev_drvr_ptr, 0); | 
 | STOREINTELDWORD(sft->file_pos, 0); | STOREINTELDWORD(sft->file_pos, 0); | 
| Line 253  static void setup_ptrs(INTRST is, SDACDS | Line 285  static void setup_ptrs(INTRST is, SDACDS | 
 | char    *rootpath; | char    *rootpath; | 
 | int             off; | int             off; | 
 |  |  | 
| if (hostdrv.dosver_major == 3) { | if (hostdrv.stat.dosver_major == 3) { | 
 | is->fcbname_ptr = sc->ver3.sda.fcb_name; | is->fcbname_ptr = sc->ver3.sda.fcb_name; | 
 | is->filename_ptr = sc->ver3.sda.file_name + ROOTPATH_SIZE - 1; | is->filename_ptr = sc->ver3.sda.file_name + ROOTPATH_SIZE - 1; | 
 | is->fcbname_ptr_2 = sc->ver3.sda.fcb_name_2; | is->fcbname_ptr_2 = sc->ver3.sda.fcb_name_2; | 
| Line 307  static BOOL pathishostdrv(INTRST is, SDA | Line 339  static BOOL pathishostdrv(INTRST is, SDA | 
 | } | } | 
 |  |  | 
 |  |  | 
| static BOOL read_data(UINT num, UINT32 pos, UINT size, | static BOOL read_data(UINT num, UINT32 pos, UINT size, UINT seg, UINT off) { | 
| UINT16 seg, UINT16 off) { |  | 
 |  |  | 
 | HDRVFILE        hdf; | HDRVFILE        hdf; | 
 | FILEH           fh; | FILEH           fh; | 
| BYTE            work[1024]; | UINT8           work[1024]; | 
 | UINT            r; | UINT            r; | 
 |  |  | 
 | hdf = (HDRVFILE)listarray_getitem(hostdrv.fhdl, num); | hdf = (HDRVFILE)listarray_getitem(hostdrv.fhdl, num); | 
| Line 328  static BOOL read_data(UINT num, UINT32 p | Line 359  static BOOL read_data(UINT num, UINT32 p | 
 | if (file_read(fh, work, r) != r) { | if (file_read(fh, work, r) != r) { | 
 | return(FAILURE); | return(FAILURE); | 
 | } | } | 
| i286_memstr_write(seg, off, work, r); | MEMR_WRITES(seg, off, work, r); | 
 | off += r; | off += r; | 
 | size -= r; | size -= r; | 
 | } | } | 
 | return(SUCCESS); | return(SUCCESS); | 
 | } | } | 
 |  |  | 
| static BOOL write_data(UINT num, UINT32 pos, UINT size, | static BOOL write_data(UINT num, UINT32 pos, UINT size, UINT seg, UINT off) { | 
| UINT16 seg, UINT16 off) { |  | 
 |  |  | 
 | HDRVFILE        hdf; | HDRVFILE        hdf; | 
 | FILEH           fh; | FILEH           fh; | 
| BYTE            work[1024]; | UINT8           work[1024]; | 
 | UINT            r; | UINT            r; | 
 |  |  | 
 | hdf = (HDRVFILE)listarray_getitem(hostdrv.fhdl, num); | hdf = (HDRVFILE)listarray_getitem(hostdrv.fhdl, num); | 
| Line 357  static BOOL write_data(UINT num, UINT32 | Line 387  static BOOL write_data(UINT num, UINT32 | 
 | else { | else { | 
 | do { | do { | 
 | r = min(size, sizeof(work)); | r = min(size, sizeof(work)); | 
| i286_memstr_read(seg, off, work, r); | MEMR_READS(seg, off, work, r); | 
 | if (file_write(fh, work, r) != r) { | if (file_write(fh, work, r) != r) { | 
 | return(FAILURE); | return(FAILURE); | 
 | } | } | 
| Line 396  const HDRVDIR *di; | Line 426  const HDRVDIR *di; | 
 | store_srch(is); | store_srch(is); | 
 |  |  | 
 | ret = FAILURE; | ret = FAILURE; | 
| pos = hostdrv.flistpos; | pos = hostdrv.stat.flistpos; | 
 | do { | do { | 
 | if (pos == 0) { | if (pos == 0) { | 
 | di = &hdd_owner; | di = &hdd_owner; | 
| Line 407  const HDRVDIR *di; | Line 437  const HDRVDIR *di; | 
 | else { | else { | 
 | hdl = listarray_getitem(hostdrv.flist, pos - 2); | hdl = listarray_getitem(hostdrv.flist, pos - 2); | 
 | if (hdl == NULL) { | if (hdl == NULL) { | 
 |  | listarray_destroy(hostdrv.flist); | 
 |  | hostdrv.flist = NULL; | 
 | break; | break; | 
 | } | } | 
 | di = &hdl->di; | di = &hdl->di; | 
| Line 414  const HDRVDIR *di; | Line 446  const HDRVDIR *di; | 
 | pos++; | pos++; | 
 | ret = find_file1(is, di); | ret = find_file1(is, di); | 
 | } while(ret != SUCCESS); | } while(ret != SUCCESS); | 
| hostdrv.flistpos = pos; | hostdrv.stat.flistpos = pos; | 
 | return(ret); | return(ret); | 
 | } | } | 
 |  |  | 
| Line 441  static void change_currdir(INTRST intrst | Line 473  static void change_currdir(INTRST intrst | 
 | } | } | 
 |  |  | 
 | ptr = intrst->filename_ptr; | ptr = intrst->filename_ptr; | 
 |  | TRACEOUT(("change_currdir %s", intrst->filename_ptr)); | 
 | if (ptr[0] == '\0') {                                                   // るーと | if (ptr[0] == '\0') {                                                   // るーと | 
 | strcpy(intrst->filename_ptr, "\\"); | strcpy(intrst->filename_ptr, "\\"); | 
 | strcpy(intrst->current_path, intrst->filename_ptr); | strcpy(intrst->current_path, intrst->filename_ptr); | 
| Line 448  static void change_currdir(INTRST intrst | Line 481  static void change_currdir(INTRST intrst | 
 | succeed(intrst); | succeed(intrst); | 
 | return; | return; | 
 | } | } | 
| if ((is_wildcards(intrst->fcbname_ptr) != FALSE) || | if ((strlen(intrst->filename_ptr) >= (67 - ROOTPATH_SIZE)) || | 
|  | (is_wildcards(intrst->fcbname_ptr) != FALSE) || | 
 | (hostdrvs_getrealpath(&hdp, ptr) != SUCCESS) || | (hostdrvs_getrealpath(&hdp, ptr) != SUCCESS) || | 
 | (hdp.di.fcbname[0] == ' ') || (!(hdp.di.attr & 0x10))) { | (hdp.di.fcbname[0] == ' ') || (!(hdp.di.attr & 0x10))) { | 
 | fail(intrst, ERR_PATHNOTFOUND); | fail(intrst, ERR_PATHNOTFOUND); | 
| Line 472  static void close_file(INTRST intrst) { | Line 506  static void close_file(INTRST intrst) { | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 | setup_ptrs(intrst, &sc); | setup_ptrs(intrst, &sc); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 504  static void commit_file(INTRST intrst) { | Line 538  static void commit_file(INTRST intrst) { | 
 | fetch_sda_currcds(&sc); | fetch_sda_currcds(&sc); | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 526  static void read_file(INTRST intrst) { | Line 560  static void read_file(INTRST intrst) { | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 | setup_ptrs(intrst, &sc); | setup_ptrs(intrst, &sc); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 552  static void read_file(INTRST intrst) { | Line 586  static void read_file(INTRST intrst) { | 
 | fail(intrst, ERR_READFAULT); | fail(intrst, ERR_READFAULT); | 
 | return; | return; | 
 | } | } | 
 |  |  | 
 | file_pos += cx; | file_pos += cx; | 
 | STOREINTELDWORD(sft.file_pos, file_pos); | STOREINTELDWORD(sft.file_pos, file_pos); | 
 |  |  | 
 | store_sft(intrst, &sft); | store_sft(intrst, &sft); | 
| store_sda_currcds(&sc); | //      store_sda_currcds(&sc);                                         // ver0.74 Yui / sdaは変更無し | 
 | succeed(intrst); | succeed(intrst); | 
 | } | } | 
 |  |  | 
| Line 573  static void write_file(INTRST intrst) { | Line 608  static void write_file(INTRST intrst) { | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 | setup_ptrs(intrst, &sc); | setup_ptrs(intrst, &sc); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 618  static void lock_file(INTRST intrst) { | Line 653  static void lock_file(INTRST intrst) { | 
 | fetch_sda_currcds(&sc); | fetch_sda_currcds(&sc); | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 635  static void unlock_file(INTRST intrst) { | Line 670  static void unlock_file(INTRST intrst) { | 
 | fetch_sda_currcds(&sc); | fetch_sda_currcds(&sc); | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 662  static void set_fileattr(INTRST intrst) | Line 697  static void set_fileattr(INTRST intrst) | 
 | fail(intrst, ERR_ACCESSDENIED); | fail(intrst, ERR_ACCESSDENIED); | 
 | return; | return; | 
 | } | } | 
| attr = i286_memword_read(CPU_SS, CPU_BP + sizeof(IF4INTR)) & 0x37; | attr = MEMR_READ16(CPU_SS, CPU_BP + sizeof(IF4INTR)) & 0x37; | 
 |  |  | 
 | // 成功したことにする... | // 成功したことにする... | 
 | succeed(intrst); | succeed(intrst); | 
| Line 679  static void get_fileattr(INTRST intrst) | Line 714  static void get_fileattr(INTRST intrst) | 
 | return; | return; | 
 | } | } | 
 |  |  | 
 |  | TRACEOUT(("get_fileattr: ->%s", intrst->fcbname_ptr)); | 
 | if ((is_wildcards(intrst->fcbname_ptr)) || | if ((is_wildcards(intrst->fcbname_ptr)) || | 
 | (hostdrvs_getrealpath(&hdp, intrst->filename_ptr) != SUCCESS)) { | (hostdrvs_getrealpath(&hdp, intrst->filename_ptr) != SUCCESS)) { | 
 | fail(intrst, ERR_FILENOTFOUND); | fail(intrst, ERR_FILENOTFOUND); | 
 | return; | return; | 
 | } | } | 
 |  | TRACEOUT(("get_fileattr: %s - %x", hdp.path, hdp.di.attr)); | 
 | ax = hdp.di.attr & 0x37; | ax = hdp.di.attr & 0x37; | 
 | if (!IS_PERMITWRITE) { | if (!IS_PERMITWRITE) { | 
 | ax |= 0x01; | ax |= 0x01; | 
| Line 703  static void rename_file(INTRST intrst) { | Line 740  static void rename_file(INTRST intrst) { | 
 | return; | return; | 
 | } | } | 
 |  |  | 
 |  | // ワイルドカードくるんで要修正… | 
 | if ((hostdrvs_getrealpath(&hdp1, intrst->filename_ptr) != SUCCESS) || | if ((hostdrvs_getrealpath(&hdp1, intrst->filename_ptr) != SUCCESS) || | 
 | (hostdrvs_getrealpath(&hdp2, intrst->filename_ptr_2) != SUCCESS)) { | (hostdrvs_getrealpath(&hdp2, intrst->filename_ptr_2) != SUCCESS)) { | 
 | fail(intrst, ERR_PATHNOTFOUND); | fail(intrst, ERR_PATHNOTFOUND); | 
| Line 712  static void rename_file(INTRST intrst) { | Line 750  static void rename_file(INTRST intrst) { | 
 | fail(intrst, ERR_ACCESSDENIED); | fail(intrst, ERR_ACCESSDENIED); | 
 | } | } | 
 |  |  | 
 |  |  | 
 | /* 13 */ | /* 13 */ | 
 | static void delete_file(INTRST intrst) { | static void delete_file(INTRST intrst) { | 
 |  |  | 
| Line 723  static void delete_file(INTRST intrst) { | Line 760  static void delete_file(INTRST intrst) { | 
 | return; | return; | 
 | } | } | 
 |  |  | 
 |  | // ワイルドカードくるんで要修正… | 
 | if ((hostdrvs_getrealpath(&hdp, intrst->filename_ptr) != SUCCESS) || | if ((hostdrvs_getrealpath(&hdp, intrst->filename_ptr) != SUCCESS) || | 
 | (hdp.di.attr & 0x10)) { | (hdp.di.attr & 0x10)) { | 
 | fail(intrst, ERR_PATHNOTFOUND); | fail(intrst, ERR_PATHNOTFOUND); | 
| Line 745  static void open_file(INTRST intrst) { | Line 783  static void open_file(INTRST intrst) { | 
 | if (pathishostdrv(intrst, &sc) != SUCCESS) { | if (pathishostdrv(intrst, &sc) != SUCCESS) { | 
 | return; | return; | 
 | } | } | 
| fetch_sft(intrst, &sft); | //      fetch_sft(intrst, &sft); | 
|  | ZeroMemory(&sft, sizeof(sft)); | 
 |  |  | 
 | if ((is_wildcards(intrst->fcbname_ptr)) || | if ((is_wildcards(intrst->fcbname_ptr)) || | 
 | (hostdrvs_getrealpath(&hdp, intrst->filename_ptr) != SUCCESS) || | (hostdrvs_getrealpath(&hdp, intrst->filename_ptr) != SUCCESS) || | 
| Line 865  static void find_first(INTRST intrst) { | Line 904  static void find_first(INTRST intrst) { | 
 | flist = hostdrv.flist; | flist = hostdrv.flist; | 
 | if (flist) { | if (flist) { | 
 | hostdrv.flist = NULL; | hostdrv.flist = NULL; | 
| hostdrv.flistpos = 0; | hostdrv.stat.flistpos = 0; | 
 | listarray_destroy(flist); | listarray_destroy(flist); | 
 | } | } | 
 |  |  | 
| Line 885  static void find_first(INTRST intrst) { | Line 924  static void find_first(INTRST intrst) { | 
 | } | } | 
 | TRACEOUT(("find_first %s -> %s", intrst->current_path, hdp.path)); | TRACEOUT(("find_first %s -> %s", intrst->current_path, hdp.path)); | 
 | hostdrv.flist = hostdrvs_getpathlist(hdp.path); | hostdrv.flist = hostdrvs_getpathlist(hdp.path); | 
| hostdrv.flistpos = 0; | hostdrv.stat.flistpos = 0; | 
 | if (find_file(intrst) != SUCCESS) { | if (find_file(intrst) != SUCCESS) { | 
 | fail(intrst, ERR_PATHNOTFOUND); | fail(intrst, ERR_PATHNOTFOUND); | 
 | return; | return; | 
| Line 900  static void find_next(INTRST intrst) { | Line 939  static void find_next(INTRST intrst) { | 
 |  |  | 
 | _SDACDS         sc; | _SDACDS         sc; | 
 | SRCHREC         srchrec; | SRCHREC         srchrec; | 
 | //      char            *curpath; |  | 
 |  |  | 
 | fetch_sda_currcds(&sc); | fetch_sda_currcds(&sc); | 
 | setup_ptrs(intrst, &sc); | setup_ptrs(intrst, &sc); | 
 |  |  | 
 | srchrec = intrst->srchrec_ptr; | srchrec = intrst->srchrec_ptr; | 
 | if ((!(srchrec->drive_no & 0x40)) || | if ((!(srchrec->drive_no & 0x40)) || | 
| ((srchrec->drive_no & 0x1f) != hostdrv.drive_no)) { | ((srchrec->drive_no & 0x1f) != hostdrv.stat.drive_no)) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
 | //      curpath = intrst->current_path; |  | 
 | if (find_file(intrst) != SUCCESS) { | if (find_file(intrst) != SUCCESS) { | 
 | fail(intrst, ERR_NOMOREFILES); | fail(intrst, ERR_NOMOREFILES); | 
 | return; | return; | 
| Line 920  static void find_next(INTRST intrst) { | Line 957  static void find_next(INTRST intrst) { | 
 | succeed(intrst); | succeed(intrst); | 
 | } | } | 
 |  |  | 
 |  | #if 1 | 
 |  | /* 1E */ | 
 |  | static void do_redir(INTRST intrst) { | 
 |  |  | 
 |  | _SDACDS         sc; | 
 |  | REG16           mode; | 
 |  | REG16           bx; | 
 |  | char            tmp[4]; | 
 |  |  | 
 |  | TRACEOUT(("do_redir")); | 
 |  | if (pathishostdrv(intrst, &sc) != SUCCESS) { | 
 |  | return; | 
 |  | } | 
 |  | mode = MEMR_READ16(CPU_SS, CPU_BP + sizeof(IF4INTR)); | 
 |  | TRACEOUT(("do_redir: %.4x", mode)); | 
 |  | switch(mode) { | 
 |  | case 0x5f02: | 
 |  | bx = LOADINTELWORD(intrst->r.w.bx); | 
 |  | if (bx) { | 
 |  | fail(intrst, 0x12); | 
 |  | return; | 
 |  | } | 
 |  | MEMR_WRITE16(CPU_DS, CPU_BX + 2, 4); | 
 |  | MEMR_WRITE16(CPU_DS, CPU_BX + 4, 1); | 
 |  | tmp[0] = (char)('A' + hostdrv.stat.drive_no); | 
 |  | tmp[1] = ':'; | 
 |  | tmp[2] = '\0'; | 
 |  | MEMR_WRITES(LOADINTELWORD(intrst->r.w.ds), | 
 |  | LOADINTELWORD(intrst->r.w.si), tmp, 3); | 
 |  | MEMR_WRITES(LOADINTELWORD(intrst->r.w.es), | 
 |  | LOADINTELWORD(intrst->r.w.di), | 
 |  | ROOTPATH, ROOTPATH_SIZE + 1); | 
 |  | break; | 
 |  |  | 
 |  | default: | 
 |  | CPU_FLAG &= ~Z_FLAG;    // chain | 
 |  | return; | 
 |  | } | 
 |  | succeed(intrst); | 
 |  | } | 
 |  | #endif | 
 |  |  | 
 | /* 21 */ | /* 21 */ | 
 | // dos4以降呼ばれることはあんまない・・・ | // dos4以降呼ばれることはあんまない・・・ | 
 | static void seek_fromend(INTRST intrst) { | static void seek_fromend(INTRST intrst) { | 
| Line 933  static void seek_fromend(INTRST intrst) | Line 1012  static void seek_fromend(INTRST intrst) | 
 | fetch_sda_currcds(&sc); | fetch_sda_currcds(&sc); | 
 | fetch_sft(intrst, &sft); | fetch_sft(intrst, &sft); | 
 |  |  | 
| if ((sft.dev_info_word[0] & 0x3f) != hostdrv.drive_no) { | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 | CPU_FLAG &= ~Z_FLAG;    // chain | CPU_FLAG &= ~Z_FLAG;    // chain | 
 | return; | return; | 
 | } | } | 
| Line 956  static void seek_fromend(INTRST intrst) | Line 1035  static void seek_fromend(INTRST intrst) | 
 | intrst->r.b.flag_l &= ~C_FLAG; | intrst->r.b.flag_l &= ~C_FLAG; | 
 | } | } | 
 |  |  | 
 |  | /* 2D */ | 
 |  | static void unknownfunc_2d(INTRST intrst) { | 
 |  |  | 
 |  | _SDACDS         sc; | 
 |  | _SFTREC         sft; | 
 |  |  | 
 |  | fetch_sda_currcds(&sc); | 
 |  | fetch_sft(intrst, &sft); | 
 |  | if ((sft.dev_info_word[0] & 0x3f) != hostdrv.stat.drive_no) { | 
 |  | CPU_FLAG &= ~Z_FLAG;    // chain | 
 |  | return; | 
 |  | } | 
 |  | #if 1 | 
 |  | TRACEOUT(("unknownfunc_2d")); | 
 |  | #else | 
 |  | intr_regs.flags &= ~C_FLAG; | 
 |  | intr_regs.ax = 2; | 
 |  | #endif | 
 |  | } | 
 |  |  | 
 |  | /* 2E */ | 
 |  | // for dos4+ | 
 |  | static void ext_openfile(INTRST intrst) { | 
 |  |  | 
 |  | _SDACDS         sc; | 
 |  | _SFTREC         sft; | 
 |  | HDRVPATH        hdp; | 
 |  | UINT            mode; | 
 |  | BOOL            create; | 
 |  | REG16           act; | 
 |  | REG16           cx; | 
 |  | FILEH           fh; | 
 |  | HDRVFILE        hdf; | 
 |  |  | 
 |  | if (pathishostdrv(intrst, &sc) != SUCCESS) { | 
 |  | return; | 
 |  | } | 
 |  | fetch_sft(intrst, &sft); | 
 |  |  | 
 |  | // ファイルを探しに〜 | 
 |  | if ((is_wildcards(intrst->fcbname_ptr)) || | 
 |  | (hostdrvs_newrealpath(&hdp, intrst->filename_ptr) != SUCCESS) || | 
 |  | (hdp.di.attr & 0x10)) { | 
 |  | fail(intrst, ERR_PATHNOTFOUND); | 
 |  | return; | 
 |  | } | 
 |  |  | 
 |  | sft.open_mode[0] = sc.ver4.sda.mode_2E[0] & 0x7f; | 
 |  | sft.open_mode[1] = sc.ver4.sda.mode_2E[1] & 0x00; | 
 |  | act     = LOADINTELWORD(sc.ver4.sda.action_2E); | 
 |  |  | 
 |  | switch(sft.open_mode[0] & 7) { | 
 |  | case 1: // write only | 
 |  | mode = HDFMODE_WRITE; | 
 |  | break; | 
 |  |  | 
 |  | case 2: // read/write | 
 |  | mode = HDFMODE_READ | HDFMODE_WRITE; | 
 |  | break; | 
 |  |  | 
 |  | default: | 
 |  | mode = HDFMODE_READ; | 
 |  | break; | 
 |  | } | 
 |  |  | 
 |  | create = FALSE; | 
 |  | if (hdp.di.exist) {                                     // ファイルが存在 | 
 |  | switch(act & 3) { | 
 |  | case 1: | 
 |  | cx = 1; | 
 |  | break; | 
 |  |  | 
 |  | case 2: | 
 |  | create = TRUE; | 
 |  | cx = 3; | 
 |  | break; | 
 |  |  | 
 |  | default: | 
 |  | fail(intrst, ERR_ACCESSDENIED); | 
 |  | return; | 
 |  | } | 
 |  | } | 
 |  | else {                                                          // 新規ファイル | 
 |  | if (act & 0x10) { | 
 |  | create = TRUE; | 
 |  | cx = 2; | 
 |  | } | 
 |  | else { | 
 |  | fail(intrst, ERR_PATHNOTFOUND); | 
 |  | return; | 
 |  | } | 
 |  | } | 
 |  |  | 
 |  | if (create) { | 
 |  | if (!IS_PERMITWRITE) { | 
 |  | fail(intrst, ERR_ACCESSDENIED); | 
 |  | return; | 
 |  | } | 
 |  | fh = file_create(hdp.path); | 
 |  | } | 
 |  | else if (mode & HDFMODE_WRITE) { | 
 |  | if (!IS_PERMITWRITE) { | 
 |  | fail(intrst, ERR_ACCESSDENIED); | 
 |  | return; | 
 |  | } | 
 |  | fh = file_open(hdp.path); | 
 |  | } | 
 |  | else { | 
 |  | fh = file_open_rb(hdp.path); | 
 |  | } | 
 |  | if (fh == FILEH_INVALID) { | 
 |  | TRACEOUT(("file open error!")); | 
 |  | fail(intrst, ERR_PATHNOTFOUND); | 
 |  | return; | 
 |  | } | 
 |  |  | 
 |  | hdf = hostdrvs_fhdlsea(hostdrv.fhdl); | 
 |  | if (hdf == NULL) { | 
 |  | file_close(fh); | 
 |  | fail(intrst, ERR_PATHNOTFOUND); | 
 |  | return; | 
 |  | } | 
 |  |  | 
 |  | hdf->hdl = (long)fh; | 
 |  | hdf->mode = mode; | 
 |  | file_cpyname(hdf->path, hdp.path, sizeof(hdf->path)); | 
 |  |  | 
 |  | STOREINTELWORD(intrst->r.w.cx, cx); | 
 |  | fill_sft(intrst, &sft, listarray_getpos(hostdrv.fhdl, hdf), &hdp.di); | 
 |  | init_sft(&sft); | 
 |  | store_sft(intrst, &sft); | 
 |  |  | 
 |  | store_sda_currcds(&sc); | 
 |  | succeed(intrst); | 
 |  | } | 
 |  |  | 
 |  |  | 
 | // ---- | // ---- | 
 |  |  | 
| Line 992  static const HDINTRFN intr_func[] = { | Line 1207  static const HDINTRFN intr_func[] = { | 
 | find_first,                     /* 1B */ | find_first,                     /* 1B */ | 
 | find_next,                      /* 1C */ | find_next,                      /* 1C */ | 
 | NULL, | NULL, | 
| NULL, | do_redir, | 
 | NULL, | NULL, | 
 | NULL, | NULL, | 
 | seek_fromend,           /* 21 */ | seek_fromend,           /* 21 */ | 
| Line 1007  static const HDINTRFN intr_func[] = { | Line 1222  static const HDINTRFN intr_func[] = { | 
 | NULL, | NULL, | 
 | NULL, | NULL, | 
 | NULL, | NULL, | 
| NULL,   //      unknownfunc_2d,         /* 2D */ | unknownfunc_2d,         /* 2D */ | 
| NULL,   //      ext_openfile            /* 2E */ | ext_openfile            /* 2E */ | 
 | }; | }; | 
 |  |  | 
 |  |  | 
| Line 1041  void hostdrv_reset(void) { | Line 1256  void hostdrv_reset(void) { | 
 |  |  | 
 | // ---- for np2sysp | // ---- for np2sysp | 
 |  |  | 
| BOOL hostdrv_mount(void) { | void hostdrv_mount(const void *arg1, long arg2) { | 
 |  |  | 
| if ((np2cfg.hdrvroot[0] == '\0') || (hostdrv.is_mount)) { | if ((np2cfg.hdrvroot[0] == '\0') || (hostdrv.stat.is_mount)) { | 
| return(FAILURE); | np2sysp_outstr("ng", 0); | 
|  | return; | 
 | } | } | 
| hostdrv.is_mount = TRUE; | hostdrv.stat.is_mount = TRUE; | 
 | fetch_if4dos(); | fetch_if4dos(); | 
| return(SUCCESS); | np2sysp_outstr("ok", 0); | 
|  | (void)arg1; | 
|  | (void)arg2; | 
 | } | } | 
 |  |  | 
| void hostdrv_unmount(void) { | void hostdrv_unmount(const void *arg1, long arg2) { | 
 |  |  | 
| if (hostdrv.is_mount) { | if (hostdrv.stat.is_mount) { | 
 | hostdrv_reset(); | hostdrv_reset(); | 
 | } | } | 
 |  | (void)arg1; | 
 |  | (void)arg2; | 
 | } | } | 
 |  |  | 
| void hostdrv_intr(void) { | void hostdrv_intr(const void *arg1, long arg2) { | 
 |  |  | 
 | _INTRST intrst; | _INTRST intrst; | 
 |  |  | 
| Line 1066  void hostdrv_intr(void) { | Line 1286  void hostdrv_intr(void) { | 
 | intrst.is_chardev = (CPU_FLAG & C_FLAG) == 0; | intrst.is_chardev = (CPU_FLAG & C_FLAG) == 0; | 
 | CPU_FLAG &= ~(C_FLAG | Z_FLAG);                         // not fcb / chain | CPU_FLAG &= ~(C_FLAG | Z_FLAG);                         // not fcb / chain | 
 |  |  | 
| if (!hostdrv.is_mount) { | if (!hostdrv.stat.is_mount) { | 
 | return; | return; | 
 | } | } | 
 |  |  | 
| Line 1074  void hostdrv_intr(void) { | Line 1294  void hostdrv_intr(void) { | 
 |  |  | 
 | TRACEOUT(("hostdrv: AL=%.2x", intrst.r.b.al)); | TRACEOUT(("hostdrv: AL=%.2x", intrst.r.b.al)); | 
 |  |  | 
| if ((intrst.r.b.al >= sizeof(intr_func) / sizeof(HDINTRFN)) || | if ((intrst.r.b.al >= NELEMENTS(intr_func)) || | 
 | (intr_func[intrst.r.b.al] == NULL)) { | (intr_func[intrst.r.b.al] == NULL)) { | 
 | return; | return; | 
 | } | } | 
| Line 1083  void hostdrv_intr(void) { | Line 1303  void hostdrv_intr(void) { | 
 | (*intr_func[intrst.r.b.al])(&intrst); | (*intr_func[intrst.r.b.al])(&intrst); | 
 |  |  | 
 | store_intr_regs(&intrst); | store_intr_regs(&intrst); | 
 |  |  | 
 |  | (void)arg1; | 
 |  | (void)arg2; | 
 |  | } | 
 |  |  | 
 |  |  | 
 |  | // ---- for statsave | 
 |  |  | 
 |  | typedef struct { | 
 |  | UINT    stat; | 
 |  | UINT    files; | 
 |  | UINT    flists; | 
 |  | } SFHDRV; | 
 |  |  | 
 |  | static BOOL fhdl_wr(void *vpItem, void *vpArg) { | 
 |  |  | 
 |  | OEMCHAR *p; | 
 |  | UINT    len; | 
 |  |  | 
 |  | p = ((HDRVFILE)vpItem)->path; | 
 |  | len = OEMSTRLEN(p); | 
 |  | statflag_write((STFLAGH)vpArg, &len, sizeof(len)); | 
 |  | if (len) { | 
 |  | if (len < MAX_PATH) { | 
 |  | ZeroMemory(p + len, (MAX_PATH - len) * sizeof(OEMCHAR)); | 
 |  | } | 
 |  | statflag_write((STFLAGH)vpArg, vpItem, sizeof(_HDRVFILE)); | 
 |  | } | 
 |  | return(FALSE); | 
 |  | } | 
 |  |  | 
 |  | static BOOL flist_wr(void *vpItem, void *vpArg) { | 
 |  |  | 
 |  | OEMCHAR *p; | 
 |  | int             len; | 
 |  |  | 
 |  | p = ((HDRVLST)vpItem)->realname; | 
 |  | len = OEMSTRLEN(p); | 
 |  | if (len < MAX_PATH) { | 
 |  | ZeroMemory(p + len, (MAX_PATH - len) * sizeof(OEMCHAR)); | 
 |  | } | 
 |  | statflag_write((STFLAGH)vpArg, vpItem, sizeof(_HDRVLST)); | 
 |  | return(FALSE); | 
 |  | } | 
 |  |  | 
 |  | int hostdrv_sfsave(STFLAGH sfh, const SFENTRY *tbl) { | 
 |  |  | 
 |  | SFHDRV  sfhdrv; | 
 |  | int             ret; | 
 |  |  | 
 |  | if (!hostdrv.stat.is_mount) { | 
 |  | return(STATFLAG_SUCCESS); | 
 |  | } | 
 |  | sfhdrv.stat = sizeof(hostdrv.stat); | 
 |  | sfhdrv.files = listarray_getitems(hostdrv.fhdl); | 
 |  | sfhdrv.flists = listarray_getitems(hostdrv.flist); | 
 |  | ret = statflag_write(sfh, &sfhdrv, sizeof(sfhdrv)); | 
 |  | ret |= statflag_write(sfh, &hostdrv.stat, sizeof(hostdrv.stat)); | 
 |  | listarray_enum(hostdrv.fhdl, fhdl_wr, sfh); | 
 |  | listarray_enum(hostdrv.flist, flist_wr, sfh); | 
 |  | (void)tbl; | 
 |  | return(ret); | 
 |  | } | 
 |  |  | 
 |  | int hostdrv_sfload(STFLAGH sfh, const SFENTRY *tbl) { | 
 |  |  | 
 |  | SFHDRV          sfhdrv; | 
 |  | int                     ret; | 
 |  | UINT            i; | 
 |  | UINT            len; | 
 |  | HDRVFILE        hdf; | 
 |  | FILEH           fh; | 
 |  | HDRVLST         hdl; | 
 |  |  | 
 |  | listarray_clr(hostdrv.fhdl); | 
 |  | listarray_clr(hostdrv.flist); | 
 |  |  | 
 |  | ret = statflag_read(sfh, &sfhdrv, sizeof(sfhdrv)); | 
 |  | if (sfhdrv.stat != sizeof(hostdrv.stat)) { | 
 |  | return(STATFLAG_FAILURE); | 
 |  | } | 
 |  | ret |= statflag_read(sfh, &hostdrv.stat, sizeof(hostdrv.stat)); | 
 |  | for (i=0; i<sfhdrv.files; i++) { | 
 |  | hdf = (HDRVFILE)listarray_append(hostdrv.fhdl, NULL); | 
 |  | if (hdf == NULL) { | 
 |  | return(STATFLAG_FAILURE); | 
 |  | } | 
 |  | ret |= statflag_read(sfh, &len, sizeof(len)); | 
 |  | if (len) { | 
 |  | ret |= statflag_read(sfh, hdf, sizeof(_HDRVFILE)); | 
 |  | if (hdf->mode & HDFMODE_WRITE) { | 
 |  | fh = file_open(hdf->path); | 
 |  | } | 
 |  | else { | 
 |  | fh = file_open_rb(hdf->path); | 
 |  | } | 
 |  | hdf->hdl = (long)fh; | 
 |  | } | 
 |  | } | 
 |  | for (i=0; i<sfhdrv.flists; i++) { | 
 |  | hdl = (HDRVLST)listarray_append(hostdrv.flist, NULL); | 
 |  | if (hdl == NULL) { | 
 |  | return(STATFLAG_FAILURE); | 
 |  | } | 
 |  | ret |= statflag_read(sfh, hdl, sizeof(_HDRVLST)); | 
 |  | } | 
 |  | (void)tbl; | 
 |  | return(ret); | 
 | } | } | 
 |  | #endif | 
 |  |  |