/* * (C) Copyright 1992, ..., 2014 the "DOSEMU-Development-Team". * * for details see file COPYING in the DOSEMU distribution */ #include #include #include #ifdef HAVE_LIBBSD #include #endif #include #include #include #include #include #include #include #include #include #include #include "emu.h" #include "redirect.h" #include "mfs.h" #include "mangle.h" #include "dos2linux.h" #include "bios.h" #include "int.h" #include "lfn.h" #define EOS '\0' #define BACKSLASH '\\' #define SLASH '/' struct lfndir { int drive; struct mfs_dir *dir; unsigned dirattr; unsigned psp; char pattern[PATH_MAX]; char dirbase[PATH_MAX]; }; #define MAX_OPEN_DIRS 20 struct lfndir *lfndirs[MAX_OPEN_DIRS]; char lfn_create_fpath[PATH_MAX]; static unsigned long long unix_to_win_time(time_t ut) { return ((unsigned long long)ut + (369 * 365 + 89)*24*60*60ULL) * 10000000; } static time_t win_to_unix_time(unsigned long long wt) { return (wt / 10000000) - (369 * 365 + 89)*24*60*60ULL; } /* XXX this just adds trailing / for compatibility. * Adjust the code to not require that. */ static int get_redirection_root1(int drive, char *presourceStr, int resourceLength) { int ret = get_redirection_root(drive, presourceStr, resourceLength - 1); if (ret > 1) { if (presourceStr) { presourceStr[ret] = '/'; presourceStr[ret + 1] = '\0'; } ret++; } return ret; } static int close_dirhandle(int handle) { struct lfndir *dir; if (handle < MAX_OPEN_DIRS && lfndirs[handle]) { dir = lfndirs[handle]; if (dir->dir) dos_closedir(dir->dir); free(dir); lfndirs[handle] = NULL; return 1; } return 0; } /* close all findfirst handles belonging to the current psp */ void close_dirhandles(unsigned psp) { int dirhandle; for (dirhandle = 1; dirhandle < MAX_OPEN_DIRS; dirhandle++) { struct lfndir *dir = lfndirs[dirhandle]; if (dir && dir->psp == psp) { close_dirhandle(dirhandle); } } } static int vfat_search(char *dest, const char *src, const char *_path, int alias) { struct mfs_dir *dir; struct mfs_dirent *de; char *path, *slash; int ret = 0; d_printf("LFN: vfat_search src=%s path=%s alias=%d\n", src, _path, alias); path = strdup(_path); if (!path) return 0; slash = strrchr(path, '/'); if (!slash) { free(path); return 0; } *slash = '\0'; dir = dos_opendir(path); free(path); if (dir == NULL) return 0; if (dir->dir == NULL) while ((de = dos_readdir(dir)) != NULL) { d_printf("LFN: vfat_search %s %s\n", de->d_name, de->d_long_name); if ((strcasecmp(de->d_long_name, src) == 0) || (strcasecmp(de->d_name, src) == 0)) { strlcpy(dest, alias ? de->d_name : de->d_long_name, 260); ret = 1; break; } } else d_printf("LFN: vfat_search (not VFAT)\n"); dos_closedir(dir); return ret; } /* input: fpath = unix path current_drive = drive for DOS path alias=1: mangle, alias=0: don't mangle output: dest = DOS path */ static int make_unmake_dos_mangled_path(char *dest, const char *fpath, int current_drive, int alias) { char *src; char *fpath2; char root[MAX_RESOURCE_LENGTH_EXT]; int root_len = get_redirection_root1(current_drive, root, sizeof(root)); /* root ends with / and fpath may not, so -1 */ if (root_len <= 0 || strncmp(fpath, root, root_len - 1) != 0) return -1; *dest++ = current_drive + 'A'; *dest++ = ':'; *dest = '\\'; if (strlen(fpath) <= root_len) { dest[1] = 0; return 0; } fpath2 = strdup(fpath); src = fpath2 + root_len; if (*src == '/') src++; while (src != NULL && *src != '\0') { char *src2 = strchr(src, '/'); if (src2 == src) break; if (src - 1 > fpath) src[-1] = '\0'; if (src2 != NULL) *src2++ = '\0'; dest++; d_printf("LFN: src=%s len=%zd\n", src, strlen(src)); if (!strcmp(src, "..") || !strcmp(src, ".")) { strcpy(dest, src); } else if (!vfat_search(dest, src, fpath2, alias)) { if (!name_ufs_to_dos(dest, src) || alias) { name_convert(dest, MANGLE); strupperDOS(dest); } } dest += strlen(dest); *dest = '\\'; if (src - 1 > fpath) src[-1] = '/'; src = src2; } if (dest[-1] == ':') dest++; *dest = '\0'; free(fpath2); return 0; } /* truename function, adapted from the FreeDOS kernel truename; (Svante Frey, James Tabor, Steffen Kaiser, Bart Oldeman, Ron Cemer) but now for LFNs */ #define PATH_ERROR goto errRet #define PATHLEN 256 #define PNE_WILDCARD 1 #define PNE_WILDCARD_STAR 2 #define addChar(c) \ { \ if (p >= dest + 256) PATH_ERROR; /* path too long */ \ *p++ = c; \ } static const char *get_root(const char * fname) { /* find the end */ register unsigned length = strlen(fname); char c; /* now back up to first path seperator or start */ fname += length; while (length) { length--; c = *--fname; if (c == '/' || c == '\\' || c == ':') { fname++; break; } } return fname; } /* helper for truename: parses either name or extension */ static int parse_name(const char **src, char **cp, char *dest) { int retval = 0; char *p = *cp; char c; while(1) switch(c=*(*src)++) { case '/': case '\\': case '\0': /* delete trailing periods and spaces */ p[0] = '\0'; while ((p[-1] == '.' || p[-1] == ' ') && p != *cp) p--; if (p == *cp) return -1; /* keep one trailing period if a * was encountered */ if (p[0] == '.' && (retval & PNE_WILDCARD_STAR)) p++; *cp = p; return retval; case '*': retval |= PNE_WILDCARD_STAR; /* fall through */ case '?': retval |= PNE_WILDCARD; /* fall through */ default: addChar(c); } errRet: return -1; } static int truename(char *dest, const char *src, int allowwildcards, int *r_drv) { int i; int result; int gotAnyWildcards = 0; char *p = dest; /* dynamic pointer into dest */ char *rootPos; char src0; enum { DONT_ADD, ADD, ADD_UNLESS_LAST } addSep; const char *froot = get_root(src); d_printf("truename(%s)\n", src); /* In opposite of the TRUENAME shell command, an empty string is rejected by MS DOS 6 */ src0 = src[0]; if (src0 == '\0') return -FILE_NOT_FOUND; if (src0 == '\\' && src[1] == '\\') { const char *unc_src = src; /* Flag UNC paths and short circuit processing. Set current LDT */ /* to sentinel (offset 0xFFFF) for redirector processing. */ d_printf("Truename: UNC detected\n"); do { src0 = unc_src[0]; addChar(src0); unc_src++; } while (src0); WRITE_DWORDP(&sda[sda_cds_off], 0xFFFFFFFF); d_printf("Returning path: \"%s\"\n", dest); /* Flag as network - drive bits are empty but shouldn't get */ /* referenced for network with empty current_ldt. */ return 0; } /* Do we have a drive? */ if (src[1] == ':') result = toupperDOS(src0) - 'A'; else result = sda_cur_drive(sda); *r_drv = result; if (result < 0 || result >= MAX_DRIVE || result >= get_lastdrive()) return -DISK_DRIVE_INVALID; /* LFN support is only for MFS drives, not Physical, Join or Subst */ if (get_redirection_root1(result, NULL, 0) <= 0) return -DISK_DRIVE_INVALID; // I wonder why it was necessary to update the current cds ptr in the sda // WRITE_WORDP(&sda[sda_cds_off], lol_cdsfarptr(lol).offset + result * cds_record_size); dest[0] = result + 'A'; dest[1] = ':'; /* Do we have a drive? */ if (src[1] == ':') src += 2; /* check for a device */ dest[2] = '\\'; froot = get_root(src); if (is_dos_device(froot)) { if (froot == src || froot == src + 5) { if (froot == src + 5) { int j; memcpy(dest + 3, src, 5); for (j = 0; j < 5; j++) dest[3 + j] = toupperDOS(dest[3 + j]); if (dest[3] == '/') dest[3] = '\\'; if (dest[7] == '/') dest[7] = '\\'; } if (froot == src || memcmp(dest + 3, "\\DEV\\", 5) == 0) { /* \dev\nul -> c:/nul */ dest[2] = '/'; src = froot; } } } /* Make fully-qualified logical path */ /* register these two used characters and the \0 terminator byte */ /* we always append the current dir to stat the drive; the only exceptions are devices without paths */ rootPos = p = dest + 2; if (*p != '/') { /* i.e., it's a backslash! */ char curpath[MAX_PATH_LENGTH]; if (getCWD_r(result, curpath, MAX_PATH_LENGTH) == 0) { d_printf("SUBSTing from: %s\n", curpath); /* What to do now: the logical drive letter will be replaced by the hidden portion of the associated path. This is necessary for NETWORK and SUBST drives. For local drives it should not harm. This is actually the reverse mechanism of JOINED drives. */ memcpy(dest, curpath, CDS_DEFAULT_ROOT_LEN); rootPos = p = dest + CDS_DEFAULT_ROOT_LEN; *p = '\\'; /* force backslash! */ p++; if (curpath[CDS_DEFAULT_ROOT_LEN] == '\0') p[0] = '\0'; else strcpy(p, &curpath[CDS_DEFAULT_ROOT_LEN + 1]); if (*src != '\\' && *src != '/') p += strlen(p); else /* skip the absolute path marker */ src++; /* remove trailing separator */ if (p[-1] == '\\') p--; } } /* append the path specified in src */ addSep = ADD; /* add separator */ while (*src) { /* New segment. If any wildcards in previous segment(s), this is an invalid path. */ if (gotAnyWildcards) return -PATH_NOT_FOUND; switch (*src++) { case '/': case '\\': /* skip multiple separators (duplicated slashes) */ addSep = ADD; break; case '.': /* special directory component */ switch (*src) { case '/': case '\\': case '\0': /* current path -> ignore */ addSep = ADD_UNLESS_LAST; /* If (/ or \) && no ++src --> addSep = ADD next turn */ continue; /* next char */ case '.': /* maybe ".." entry */ another_dot: switch (src[1]) { case '/': case '\\': case '\0': case '.': /* remove last path component */ while (*--p != '\\') if (p <= rootPos) /* already on root */ return -PATH_NOT_FOUND; /* the separator was removed -> add it again */ ++src; /* skip the second dot */ if (*src == '.') goto another_dot; /* If / or \, next turn will find them and assign addSep = ADD */ addSep = ADD_UNLESS_LAST; continue; /* next char */ } } default: /* normal component */ if (addSep != DONT_ADD) { /* append backslash */ addChar(*rootPos); addSep = DONT_ADD; } --src; /* first character skipped in switch() */ i = parse_name(&src, &p, dest); if (i == -1) PATH_ERROR; if (i & PNE_WILDCARD) gotAnyWildcards = TRUE; --src; /* terminator or separator was skipped */ break; } } if (gotAnyWildcards && !allowwildcards) return -PATH_NOT_FOUND; if (addSep == ADD || p == dest + 2) { /* MS DOS preserves a trailing '\\', so an access to "C:\\DOS\\" or "CDS.C\\" fails. */ /* But don't add the separator, if the last component was ".." */ /* we must also add a seperator if dest = "c:" */ addChar('\\'); } *p = '\0'; /* add the string terminator */ d_printf("Absolute logical path: \"%s\"\n", dest); return 0; errRet: /* The error is either PATHNOTFND or FILENOTFND depending on if it is not the last component */ return strchr(src, '/') == 0 && strchr(src, '\\') == 0 ? -FILE_NOT_FOUND : -PATH_NOT_FOUND; } static inline void build_ufs_path(char *ufs, const char *path, int drive) { build_ufs_path_(ufs, path, drive, 0); } static int lfn_error(int errorcode) { _AX = errorcode; WRITE_WORDP((unsigned char *)&sda_error_code(sda), errorcode); CARRY; return 1; } static int build_truename(char *dest, const char *src, int mode) { int dd = -1; int err; d_printf("LFN: build_posix_path: %s\n", src); err = truename(dest, src, mode, &dd); switch (err) { case 0: break; case -DISK_DRIVE_INVALID: assert(dd >= 0); if (get_redirection_root1(dd, NULL, 0) <= 0) return -2; /* no break */ default: lfn_error(-err); return -1; } if (src[0] == '\\' && src[1] == '\\') return -2; if (dd < 0 || dd >= MAX_DRIVE || get_redirection_root1(dd, NULL, 0) <= 0) return -2; #if 0 // Not sure what this is for? if (!((cds_flags(drive_cds(dd))) & CDS_FLAG_REMOTE) || !drives[dd & 0x1f].root) return -2; #endif return dd; } static int build_posix_path(char *dest, const char *src, int allowwildcards) { char filename[PATH_MAX]; int dd; dd = build_truename(filename, src, allowwildcards); if (dd < 0) return dd; build_ufs_path(dest, filename, dd); return dd; } /* wildcard match routine, losely based upon the GLIBC fnmatch routine */ static int recur_match(const char *pattern, const char *string) { unsigned char c; while ((c = *pattern++) != '\0') { if (c == '*') { while ((c = *pattern++) == '?' || c == '*') { if (c == '?' && *string++ == '\0') return 1; } if (c == '\0') /* * at end of pattern, matches anything */ return 0; for (pattern--; *string != '\0'; string++) { /* recursive search if rest of pattern matches a string part */ if (recur_match(pattern, string) == 0) return 0; } /* no match, failure... */ return 1; } else if (c == '?') { if (*string++ == '\0') return 1; } else { if ((unsigned char)pattern[-1] != toupperDOS(*string++)) return 1; } } /* at end of string, then no differences, success == 0 */ /* also check for a trailing dot so that *. works */ return !((*string == '\0') || (*string == '.' && string[1] == '\0')); } static int wild_match(const char *pattern, char *string) { char *dotpos; int rc; size_t slen = strlen(string); /* add a trailing period if there is no period, so that *.* matches */ dotpos = NULL; if (!strchr(string, '.')) { dotpos = string + slen; strcpy(dotpos, "."); slen++; } rc = recur_match(pattern, string); if (dotpos) *dotpos = '\0'; return rc; } static int lfn_sfn_match(const char *pattern, struct mfs_dirent *de, char *lfn, char *sfn) { if (!name_ufs_to_dos(lfn, de->d_long_name)) { name_convert(lfn, MANGLE); strupperDOS(lfn); } name_ufs_to_dos(sfn, de->d_name); name_convert(sfn, MANGLE); return wild_match(pattern, lfn) != 0 && wild_match(pattern, sfn) != 0; } static int make_finddata(const char *fpath, uint8_t attrs, struct stat *st, char *name_lfn, char *name_8_3, dosaddr_t dest); static int getfindnext(struct mfs_dirent *de, const struct lfndir *dir) { char name_8_3[PATH_MAX]; char name_lfn[PATH_MAX]; struct stat st; char *fpath; unsigned int dest; uint8_t attrs; int ret; if (lfn_sfn_match(dir->pattern, de, name_lfn, name_8_3) != 0) return 0; if (!strcmp(de->d_long_name,".") || !strcmp(de->d_long_name,"..")) { if (strlen(dir->dirbase) <= get_redirection_root1(dir->drive, NULL, 0)) return 0; } fpath = malloc(strlen(dir->dirbase) + 1 + strlen(de->d_long_name) + 1); strcpy(fpath, dir->dirbase); strcat(fpath, "/"); strcat(fpath, de->d_long_name); d_printf("LFN: findnext %s\n", fpath); if (stat(fpath, &st) != 0) { free(fpath); return 0; } if (st.st_mode & S_IFDIR) { if ((dir->dirattr & DIRECTORY) == 0) { free(fpath); return 0; } } else { if ((dir->dirattr >> 8) & DIRECTORY) { free(fpath); return 0; } } dest = SEGOFF2LINEAR(_ES, _DI); attrs = get_dos_attr(fpath, st.st_mode); ret = make_finddata(fpath, attrs, &st, name_lfn, name_8_3, dest); free(fpath); return ret; } static int make_finddata(const char *fpath, uint8_t attrs, struct stat *st, char *name_lfn, char *name_8_3, dosaddr_t dest) { MEMSET_DOS(dest, 0, 0x20); WRITE_BYTE(dest, attrs); WRITE_DWORD(dest + 0x20, st->st_size); WRITE_DWORD(dest + 0x1c, st->st_size >> 32); if (_SI == 1) { u_short date, time; d_printf("LFN: using DOS date/time\n"); time_to_dos(st->st_mtime, &date, &time); WRITE_WORD(dest+0x16, date); WRITE_WORD(dest+0x14, time); time_to_dos(st->st_ctime, &date, &time); WRITE_WORD(dest+0x6, date); WRITE_WORD(dest+0x4, time); time_to_dos(st->st_atime, &date, &time); WRITE_WORD(dest+0xe, date); WRITE_WORD(dest+0xc, time); } else { unsigned long long wtime; d_printf("LFN: using WIN date/time\n"); wtime = unix_to_win_time(st->st_mtime); WRITE_DWORD(dest+0x14, wtime); WRITE_DWORD(dest+0x18, wtime >> 32); wtime = unix_to_win_time(st->st_ctime); WRITE_DWORD(dest+0x4, wtime); WRITE_DWORD(dest+0x8, wtime >> 32); wtime = unix_to_win_time(st->st_atime); WRITE_DWORD(dest+0xc, wtime); WRITE_DWORD(dest+0x10, wtime >> 32); } MEMCPY_2DOS(dest + 0x2c, name_lfn, strlen(name_lfn)+1); WRITE_BYTE(dest + 0x130, 0); strupperDOS(name_8_3); if (strcmp(name_8_3, name_lfn) != 0) { MEMCPY_2DOS(dest + 0x130, name_8_3, strlen(name_8_3)+1); } return 1; } static void call_dos_helper(int ah) { fake_call_to(LFN_HELPER_SEG, LFN_HELPER_OFF); _AH = ah; } static int wildcard_delete(char *fpath, int drive) { struct mfs_dir *dir; struct mfs_dirent *de; struct stat st; unsigned dirattr = _CX; char *pattern, *slash; int doserr; slash = strrchr(fpath, '/'); d_printf("LFN: posix:'%s'\n", fpath); if (slash - 2 > fpath && slash[-2] == '/' && slash[-1] == '.') slash -= 2; *slash = '\0'; if (slash == fpath) strcpy(fpath, "/"); /* XXX check for device (special dir entry) */ if (!find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), NULL) || is_dos_device(fpath)) { d_printf("LFN: Get failed: '%s'\n", fpath); return lfn_error(PATH_NOT_FOUND); } slash = fpath + strlen(fpath); dir = dos_opendir(fpath); if (dir == NULL) { d_printf("LFN: Get failed: '%s'\n", fpath); free(dir); return lfn_error(PATH_NOT_FOUND); } pattern = malloc(strlen(slash + 1) + 1); name_ufs_to_dos(pattern, slash + 1); strupperDOS(pattern); d_printf("LFN: wildcard delete '%s' '%s' %x\n", pattern, fpath, dirattr); doserr = FILE_NOT_FOUND; while ((de = dos_readdir(dir))) { char name_8_3[PATH_MAX]; char name_lfn[PATH_MAX]; if (lfn_sfn_match(pattern, de, name_lfn, name_8_3) == 0) { *slash = '/'; strcpy(slash + 1, de->d_long_name); d_printf("LFN: wildcard delete '%s'\n", fpath); /* don't remove directories */ if ((dirattr >> 8) & DIRECTORY) continue; if (unlink(fpath) == -1) { int unixerr = errno; if (unixerr == EISDIR) continue; d_printf("LFN: Delete failed(%s) '%s'\n", strerror(unixerr), fpath); doserr = (unixerr == EACCES) ? ACCESS_DENIED : FILE_NOT_FOUND; break; } d_printf("LFN: Deleted '%s'\n", fpath); doserr = 0; } } free(pattern); dos_closedir(dir); if (doserr) return lfn_error(doserr); return 1; } /* the general idea here is: - convert given pathname to a Unix path by prepending the relevant "lredir" directory and flipping slashes - then we must do a case-*insensitive* and SFN matching search to find out if the path exists. This is done by find_file(); this can be a little slow but is necessary for compatibility. - following that the operation can be done directly or must be passed on to DOS using SFNs (chdir, open) using a small BIOS stub. - the SFNs are the Samba style SFNs that DOSEMU has used for as long as I can remember. */ static int mfs_lfn_(void) { char *cwd; char fpath[PATH_MAX]; char fpath2[PATH_MAX]; int drive, dirhandle = 0, rc, doserrno = FILE_NOT_FOUND; unsigned int dest = SEGOFF2LINEAR(_ES, _DI); char *src = MK_FP32(_DS, _DX); struct stat st; struct utimbuf utimbuf; size_t size; struct mfs_dirent *de; char *slash; struct lfndir *dir = NULL; d_printf("LFN: doing LFN!, AX=%x DL=%x\n", _AX, _DL); NOCARRY; #if 0 if (_AH == 0x57) { char *filename; int fd; if (_AL < 4 || _AL > 7) return 0; filename = handle_to_filename(_BX, &fd); if (filename == NULL) return 0; if (fstat(fd, &st)) return lfn_error(HANDLE_INVALID); d_printf("LFN: handle function for BX=%x, path=%s, fd=%d\n", _BX, filename, fd); switch (_AL) { case 0x04: /* get last access date and time */ time_to_dos(st.st_atime, &_DX, &_CX); _CX = 0; break; case 0x05: /* set last access date */ utimbuf.modtime = st.st_mtime; utimbuf.actime = time_to_unix(_DX, _CX); if (dos_utime(filename, &utimbuf) != 0) return lfn_error(ACCESS_DENIED); break; case 0x06: /* get creation date/time */ time_to_dos(st.st_ctime, &_DX, &_CX); _SI = (st.st_ctime & 1) ? 100 : 0; /* fall through */ case 0x07: /* set creation date/time, impossible in Linux */ return 1; } return 1; } /* else _AH == 0x71 */ #endif switch (_AL) { case 0x0D: /* reset drive, nothing to do */ break; case 0x39: /* mkdir */ case 0x3a: /* rmdir */ d_printf("LFN: %sdir %s\n", _AL == 0x39 ? "mk" : "rm", src); drive = build_truename(fpath, src, 0); if (drive < 0) return drive + 2; rc = (_AL == 0x39 ? dos_mkdir : dos_rmdir)(fpath, drive, 1); if (rc) return lfn_error(rc); break; case 0x3b: /* chdir */ { char *d = MK_FP32(BIOSSEG, LFN_short_name); Debug0((dbg_fd, "set directory to: %s\n", src)); d_printf("LFN: chdir '%s'\n", src); drive = build_posix_path(fpath, src, 0); if (drive < 0) return drive + 2; if (!find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), NULL)|| !S_ISDIR(st.st_mode)) return lfn_error(PATH_NOT_FOUND); make_unmake_dos_mangled_path(d, fpath, drive, 1); d_printf("LFN: New CWD will be %s\n", d); call_dos_helper(0x3b); break; } case 0x41: /* remove file */ drive = build_posix_path(fpath, src, _SI); if (drive < 0) return drive + 2; if (is_redirection_ro(drive)) return lfn_error(ACCESS_DENIED); if (is_dos_device(fpath)) return lfn_error(FILE_NOT_FOUND); if (_SI == 1) return wildcard_delete(fpath, drive); if (!find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), &doserrno)) return lfn_error(doserrno); d_printf("LFN: deleting %s\n", fpath); if (unlink(fpath) != 0) return lfn_error(FILE_NOT_FOUND); break; case 0x42: /* long seek (EDR-DOS compatible) */ fake_call_to(LFN_HELPER_SEG, LFN_42_HELPER_OFF); break; case 0x43: /* get/set file attributes */ d_printf("LFN: attribute %s %d\n", src, _BL); drive = build_posix_path(fpath, src, 0); if (drive < 0) return drive + 2; if (is_redirection_ro(drive) && (_BL < 8) && (_BL & 1)) return lfn_error(ACCESS_DENIED); if (!find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), &doserrno) || is_dos_device(fpath)) { d_printf("LFN: Get failed: '%s'\n", fpath); return lfn_error(doserrno); } utimbuf.actime = st.st_atime; utimbuf.modtime = st.st_mtime; switch (_BL) { case 0: /* retrieve attributes */ _CX = get_dos_attr(fpath, st.st_mode); break; case 1: /* set attributes */ if (!(st.st_mode & S_IWGRP)) return lfn_error(ACCESS_DENIED); /* allow changing attrs only on files, not dirs */ if (!S_ISREG(st.st_mode)) break; if (set_dos_attr(fpath, _CX) != 0) return lfn_error(ACCESS_DENIED); break; case 2: /* get physical size of uncompressed file */ _DX = st.st_size >> 16; _AX = st.st_size & 0xffff; break; case 3: /* set last write date/time */ utimbuf.modtime = time_to_unix(_DI, _CX); if (dos_utime(fpath, &utimbuf) != 0) return lfn_error(ACCESS_DENIED); break; case 4: /* get last write date/time */ time_to_dos(st.st_mtime, &_DI, &_CX); break; case 5: /* set last access date */ utimbuf.actime = time_to_unix(_DI, _CX); if (dos_utime(fpath, &utimbuf) != 0) return lfn_error(ACCESS_DENIED); break; case 6: /* get last access date */ { unsigned short scratch; time_to_dos(st.st_atime, &_DI, &scratch); break; } case 7: /* set creation date/time, impossible in Linux */ break; case 8: /* get creation date/time */ time_to_dos(st.st_ctime, &_DI, &_CX); _SI = (st.st_ctime & 1) ? 100 : 0; break; } break; case 0x47: /* get current directory */ if (_DL == 0) drive = sda_cur_drive(sda); else drive = _DL - 1; if (drive < 0 || drive >= MAX_DRIVE) return lfn_error(DISK_DRIVE_INVALID); if (get_redirection_root1(drive, NULL, 0) <= 0) return 0; cwd = getCWD(drive); dest = SEGOFF2LINEAR(_DS, _SI); build_ufs_path(fpath, cwd, drive); d_printf("LFN: getcwd %s %s\n", cwd, fpath); find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), NULL); d_printf("LFN: getcwd %s %s\n", cwd, fpath); d_printf("LFN: %d %#x %s\n", drive, dest, fpath+get_redirection_root1(drive, NULL, 0)); make_unmake_dos_mangled_path(fpath2, fpath, drive, 0); MEMCPY_2DOS(dest, fpath2 + 3, strlen(fpath2 + 3) + 1); break; case 0x4e: /* find first */ { drive = build_posix_path(fpath, src, 1); if (drive < 0) return drive + 2; slash = strrchr(fpath, '/'); d_printf("LFN: posix:%s\n", fpath); *slash++ = '\0'; /* note: DJGPP doesn't like "0" as a directory handle */ for (dirhandle = 1; dirhandle < MAX_OPEN_DIRS; dirhandle++) if (lfndirs[dirhandle] == NULL) break; if (dirhandle == MAX_OPEN_DIRS) { d_printf("LFN: too many dirs open\n"); return lfn_error(NO_MORE_FILES); } dir = malloc(sizeof *dir); if (slash == fpath + 1) strcpy(dir->dirbase, "/"); else strcpy(dir->dirbase, fpath); dir->dirattr = _CX; dir->drive = drive; dir->psp = sda_cur_psp(sda); name_ufs_to_dos(dir->pattern, slash); strupperDOS(dir->pattern); if (((_CX & (VOLUME_LABEL|DIRECTORY)) == VOLUME_LABEL) && (strcmp(slash, "*.*") == 0 || strcmp(slash, "*") == 0)) { char lfn[260]; dest = SEGOFF2LINEAR(_ES, _DI); MEMSET_DOS(dest, 0, 0x20); WRITE_BYTE(dest, VOLUME_LABEL); get_volume_label(NULL, NULL, lfn, drive); MEMCPY_2DOS(dest + 0x2c, lfn, strlen(lfn) + 1); WRITE_BYTE(dest + 0x130, 0); d_printf("LFN: get volume label: %s\n", lfn); lfndirs[dirhandle] = dir; dir->dir = NULL; _AX = dirhandle; _CX = 0; return 1; } /* XXX check for device (special dir entry) */ if (!find_file(dir->dirbase, &st, get_redirection_root1(drive, NULL, 0), NULL) || is_dos_device(fpath)) { d_printf("LFN: Get failed: '%s'\n", fpath); free(dir); return lfn_error(NO_MORE_FILES); } dir->dir = dos_opendir(dir->dirbase); if (dir->dir == NULL) { d_printf("LFN: Get failed: '%s'\n", fpath); free(dir); return lfn_error(NO_MORE_FILES); } d_printf("LFN: findfirst %s %s %s %x\n", slash, src, dir->dirbase, _CX); lfndirs[dirhandle] = dir; /* fall through! */ } case 0x4f: /* find next */ case 0xa2: { if (_AL != 0x4e && _BX < MAX_OPEN_DIRS) { dirhandle = _BX; dir = lfndirs[dirhandle]; } if (dir == NULL) return 0; if (dir->dir == NULL) return lfn_error(NO_MORE_FILES); do { de = dos_readdir(dir->dir); if (de == NULL) { dos_closedir(dir->dir); dir->dir = NULL; if (_AL == 0x4e) close_dirhandle(dirhandle); return lfn_error(NO_MORE_FILES); } d_printf("LFN: findnext %s %x\n", de->d_long_name, dir->dirattr); } while (!getfindnext(de, dir)); if (_AL != 0x4e) _AX = 0x4f00 + dirhandle; else _AX = dirhandle; _CX = 0; break; } case 0x56: /* rename file */ { int drive2, rc; const char *d = MK_FP32(_ES, _DI); d_printf("LFN: rename to %s\n", d); drive = build_truename(fpath2, d, 0); if (drive < 0) return drive + 2; d_printf("LFN: rename from %s\n", src); drive2 = build_truename(fpath, src, 0); if (drive2 < 0) return drive2 + 2; if (drive != drive2) return lfn_error(NOT_SAME_DEV); rc = dos_rename_lfn(fpath, fpath2, drive); if (rc) return lfn_error(rc); break; } case 0x60: /* truename */ { int i; char filename[PATH_MAX]; src = MK_FP32(_DS, _SI); d_printf("LFN: truename %s, cl=%d\n", src, _CL); i = 0; if (src[0] && src[1] == ':') i = 2; for (; src[i]; i++) { if (!VALID_DOS_PCHAR(src + i) && strchr("\\/.",src[i]) == 0 && (_CL == 2 || strchr(" +,;=[]",src[i])==0)) { return lfn_error(FILE_NOT_FOUND); } } drive = build_truename(filename, src, !_CL); if (drive < 0) return drive + 2; d_printf("LFN: %s\n", fpath); if (_CL == 1 || _CL == 2) { build_ufs_path(fpath, filename, drive); if (!find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), &doserrno)) return lfn_error(doserrno); make_unmake_dos_mangled_path(filename, fpath, drive, 2 - _CL); } else { strupperDOS(filename); } d_printf("LFN: %s\n", filename); MEMCPY_2DOS(dest, filename, strlen(filename) + 1); break; } case 0x6c: /* create/open */ { char *d = MK_FP32(BIOSSEG, LFN_short_name); src = MK_FP32(_DS, _SI); d_printf("LFN: open %s\n", src); drive = build_posix_path(fpath, src, 0); if (drive < 0) return drive + 2; if (is_dos_device(fpath)) { strcpy(d, strrchr(fpath, '/') + 1); } else { slash = strrchr(fpath, '/'); strlcpy(fpath2, slash, sizeof(fpath2)); *slash = '\0'; if (slash != fpath && !find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), NULL)) return lfn_error(PATH_NOT_FOUND); strcat(fpath, fpath2); if (!find_file(fpath, &st, get_redirection_root1(drive, NULL, 0), NULL) && (_DX & 0x10)) { if (is_redirection_ro(drive)) return lfn_error(ACCESS_DENIED); strcpy(lfn_create_fpath, fpath); } make_unmake_dos_mangled_path(d, fpath, drive, 1); } _AL = 0; call_dos_helper(0x6c); break; } case 0xa0: /* get volume info */ drive = build_posix_path(fpath, src, 0); if (drive < 0) return drive + 2; size = _CX; _AX = 0; _BX = 0x4002; _CX = 255; _DX = 260; if (size >= 4) MEMCPY_2DOS(dest, "MFS", 4); break; case 0xa1: /* findclose */ d_printf("LFN: findclose %x\n", _BX); return close_dirhandle(_BX); case 0xa6: /* get file info by handle */ fake_call_to(LFN_HELPER_SEG, LFN_A6_HELPER_OFF); break; case 0xa7: /* file time to DOS time and v.v. */ if (_BL == 0) { src = MK_FP32(_DS, _SI); time_to_dos( win_to_unix_time(*(unsigned long long *)src), &_DX, &_CX); _BH = 0; } else { unsigned long long wtime; wtime = unix_to_win_time(time_to_unix(_DX, _CX)); WRITE_DWORD(dest, wtime); WRITE_DWORD(dest + 4, wtime >> 32); } break; case 0xa8: /* generate short filename */ { src = MK_FP32(_DS, _SI); StrnCpy(fpath, src, sizeof(fpath) - 1); name_convert(fpath, MANGLE); strupperDOS(fpath); if (_DH == 0) { char d[8+3]; extract_filename(fpath, d, d + 8); MEMCPY_2DOS(dest, fpath, 11); } else MEMCPY_2DOS(dest, fpath, strlen(fpath) + 1); d_printf("LFN: name convert %s %s %x\n", src, fpath, _DH); break; } case 0xa9: /* server create or open file */ case 0xaa: /* create/terminate/query subst */ default: return 0; } return 1; /* finished: back to caller */ } int mfs_lfn(void) { int carry, ret; if (!mfs_enabled) { CARRY; return 0; } carry = isset_CF(); ret = mfs_lfn_(); /* preserve carry if we forward the LFN request */ if (ret == 0 && carry) CARRY; return ret; }