mirror of
https://github.com/reactos/wine.git
synced 2025-02-09 21:55:12 +00:00
![Alexandre Julliard](/assets/img/avatar_default.png)
Wed Jan 31 10:58:00 1996 Alexandre Julliard <julliard@sunsite.unc.edu> * [configure.in] Added --with-dll option to build libwine.so. * [controls/listbox.c] Fixed ListBoxDirectory(), DlgDirSelect() and DlgDirList(). Hopefully their behavior is correct now. * [controls/menu.c] Use SEGPTRs in ChangeMenu(), InsertMenu(), AppendMenu() and ModifyMenu() for the item data, to avoid corrupting the pointer for owner-drawn items. * [controls/static.c] Attempt to load OEM icons for SS_ICON controls. Probably not entirely correct. Don't clip the text output. * [files/directory.c] Add temp dir and Windows dir to environment. * [files/dos_fs.c] Fixed a few path handling bugs in DOSFS_GetUnixFileName(). Cache last used directory in DOSFS_FindNext() to avoid quadratic search time. * [files/drive.c] New format for drives configuration in wine.conf; allows specifying the type, label and serial number of a drive. * [files/file.c] New function FILE_OpenUnixFile to make sure we don't open a directory instead of a file. Fixed DOSFS_GetUnixFileName() check_last flag in FILE_MakeDir(). * [files/profile.c] Rewrote profile handling. Should be closer to Windows behavior now. New function PROFILE_GetWineIniString() to get a string from wine.conf. Support environment variables in wine.conf. * [loader/task.c] Fixed the order of deletion in TASK_DeleteTask() to avoid memory corruption. * [memory/global.c] Create a discarded block on GlobalAlloc() if the size is 0; thanks to John Harvey for noticing this. * [memory/local.c] LOCAL_GetHeap: make sure the pointer is valid before checking magic number. * [misc/main.c] Moved profile and registry saving to ExitWindows(), so we don't try to save them in case of a crash. * [miscemu/int21.c] INT21_GetFreeDiskSpace: try to compute the cluster size from the filesystem size instead of hard-coding it to 64. Fixed functions 0x3f and 0x40 to use _hread and _hwrite to allow reading or writing 65535 bytes (thanks to Bruce Milner for this one). * [windows/message.c] Fixed bug in linked-list handling in MSG_DeleteQueue(). Simplified SetMessageQueue(). * [wine.ini] [wine.man] Updated for new drives configuration format. Tue Jan 30 11:24:46 1996 William Magro <wmagro@tc.cornell.edu> * [controls/edit.c] Implemented ES_PASSWORD style, EM_SETPASSWORDCHAR and EM_GETPASSWORDCHAR messages. * [controls/widgets.c] Adjusted class creation flags to better match values Windows uses. * [include/windows.h] Fixed ES_NOHIDESEL typo. * [loader/ne_image.c] Added detection for zero offset in RADDR fixups. Quicken was in an infinite loop here. Mon Jan 29 20:12:22 1996 Albrecht Kleine <kleine@ak.sax.de> * [files/dos_fs.c] Bugfix: range error in month value (0..11 set to 1..12). * [windows/caret.c] Changed ROP2-mode to R2_NOTXORPEN in CARET_Callback for pulsed appearance of the caret. * [windows/mdi.c] [include/mdi.h] Changed MDITile(): added a new parameter WORD wParam for WM_MDITILE second tiling method (MDITILE_HORIZONTAL in wParam) as used in Win3.1 Sun Jan 28 14:20:00 1996 Cameron Heide <heide@ee.ualberta.ca> * [miscemu/int2f.c] Added a small bit of MSCDEX emulation. * [windows/alias.c] ALIAS_RegisterAlias was returning the hash value when it should have been returning the record number. Sat Jan 27 10:53:51 1996 Jim Peterson <jspeter@birch.ee.vt.edu> * [include/shell.h] [include/wintypes.h] Moved definition of HKEY and LPHKEY types to include/wintypes.h. Declared FONTENUMPROC in wintypes.h. * [include/windows.h] Added definition of KERNINGPAIR and LPKERNINGPAIR types. Added declarations for CopyCursor(), CopyIcon(), EnumFontFamilies(), ExtractIcon(), FatalAppExit(), FindExecutable(), GetClipCursor(), GetKerningPairs(), GetQueueStatus(), GetRasterizerCaps(), IsGDIObject(), IsMenu(), IsTask(), RegCloseKey(), RegCreateKey(), RegDeleteKey(), RegEnumKey(), RegOpenKey(), RegQueryValue(), RegSetValue(), ResetDC(), ShellExecute(), SystemParametersInfo(), and wsprintf(). * [tools/makehtml.pl] [documentation/apiw.index] New files that scan windows.h, commdlg.h, and toolhelp.h and output an HTML sorted list with optional links to www.willows.com and a tally of unimplemented APIW functions. * [objects/cursoricon.c] Added Win32 versions of CopyIcon() and CopyCursor() for use in libwine. * [win32/resource.c] [win32/winprocs.c] Added '#include "libres.h"' and explicit declarations of windows procs in order to avoid warnings. * [windows/utility.c] Added Win32 version of MulDiv() for libwine. * [*/*] [include/windows.h] Changed several function declarations to comply more strictly to the windows API (without, hopefully, altering their functionality). * [controls/menu.c] Made the return value of CheckMenuItem be the previous state of the menu item if it was found, otherwise -1 as specified in the SDK. This conflicts with the APIW specification, which says it should return TRUE if successful, otherwise FALSE. * [include/windows.h] Added obsolete WM_SIZE message wParam names for compatibility. Added WinHelp() command constants, even though they are not yet supported. * [rc/winerc.c] Tidied up transform_binary_file(). In argument checking, flattened any invalid characters specified with the prefix argument. * [library/libres.c] Made FindResource() case-insensitive when parameter 'name' is a string. Sat Jan 27 02:30 1996 Uwe Bonnes <bon@elektron.ikp.physik.th-darmstadt.de * [files/drive.c] If root "/" is given in wine.conf, use it as last resort. * [files/file.c] Report ER_AccessDenied it disk ist not writable More Debug Output * [miscemu/int21.c] Squeezed some bugs in ExtendedOpenCreateFile * [windows/winpos.c] Some windows may not be moved or resized. We are missing some structures to be exact, but the approach should help in some cases and make things worse in much fewer. Fri Jan 26 10:24:00 1996 Thomas Sandford <t.d.g.sandford@prds-grn.demon.co.uk> * [loader/pe_image.c] fixup_imports: Find builtins for Borland style entries, too Fri Jan 26 10:24:00 1996 Martin von Loewis <loewis@informatik.hu-berlin.de> * [controls/menu.c] LoadMenu: branch to Win32 for PE modules * [if1632/gdi.spec][if1632/kernel32.spec][if1632/user32.spec] DeleteObject, GetPixel, SetPixel,WritePrivateProfileStringA, WriteProfileStringA, EmptyClipboard, EnableMenuItem, EnableScrollBar, EnableWindow, InvalidateRect, SetWindowTextA, WinHelpA: new relays DrawTextA, MoveToEx, GetClientRect, InvalidateRect, LoadBitmapA/W, LoadAcceleratorsA/W, LoadMenu[Indirect]A/W, LoadStringA/W: changed to convert parameters or naming convention * [include/kernel32.h][include/wintypes.h] moved WCHAR, defined LPWSTR * [include/string32.h][win32/string32.c][include/struct32.h] New files * [loader/module.h] LoadModule: exit after returning from PE_LoadModule * [loader/pe_image.c] my_wcstombs: isascii does not work on Linux for Unicode PE_LoadImage: Handle directories * [misc/user32.c] USER32_RECT32to16, USER32_RECT16to32: new functions implemented new user32 relays * [misc/newfns.c] WIN32_WinHelpA: new function * [win32/param32.c] New file * [win32/resource.c] GetResDirEntry: added support for named entries WIN32_LoadAcceleratorsW: invoke *32 resource functions WIN32_LoadBitmapA: convert name to unicode if appropriate WIN32_ParseMenu: new function implemented new resource functions from user32.spec Wed Jan 24 18:09:00 1996 Alex Korobka <alex@phm30.pharm.sunysb.edu> * [objects/cursoricon.c] GetIconId() and LoadIconHandler() functions. * [windows/mdi.c] Better maximization support, TranslateMDISysAccel() function, misc improvements. * [windows/defwnd.c] Fix for WM_WINDOWPOSCHANGED message handler. * [windows/winpos.c] Rewrote WindowFromPoint() function. Sun Jan 21 1996 17:05:09 Marcus Meissner <msmeissn@faui01.informatik.uni-erlangen.de> * [include/toolhelp.h] [misc/toolhelp.c] Added Notify(Un)Register, but no callbacks yet. Fri Jan 19 01:43:37 1996 Victor Schneider <root@tailor.roman.org> * [Makefile.in] Added target for libwine.so.1.0. * [library/winmain.c] For WINELIBDLL, _WinMain just returns hInstance instead of calling WinMain(). * [misc/main.c] For WINELIBDLL, renamed main() to _wine_main() for calling from the stub main function. * [library/winestub.c] (new file) Provides a stub main() function for using libwine.so. Tue Jan 16 11:04:34 1996 Anand Kumria <akumria@ozemail.com.au> * [winsocket.c] Fix EPERM problem. * [global.c] Attempt to do some sanity checking in MemManInfo(). * [Changelog] Fix changelog oversight for previous entry.
705 lines
20 KiB
C
705 lines
20 KiB
C
/*
|
|
* DOS file system functions
|
|
*
|
|
* Copyright 1993 Erik Bos
|
|
* Copyright 1996 Alexandre Julliard
|
|
*/
|
|
|
|
#include <ctype.h>
|
|
#include <dirent.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <sys/stat.h>
|
|
#include <time.h>
|
|
#ifdef __svr4__
|
|
#include <sys/statfs.h>
|
|
#endif
|
|
|
|
#include "windows.h"
|
|
#include "dos_fs.h"
|
|
#include "drive.h"
|
|
#include "file.h"
|
|
#include "msdos.h"
|
|
#include "stddebug.h"
|
|
#include "debug.h"
|
|
|
|
/* Chars we don't want to see in DOS file names */
|
|
#define INVALID_DOS_CHARS "*?<>|\"+=,;[] \345"
|
|
|
|
static const char *DOSFS_Devices[][2] =
|
|
{
|
|
{ "CON", "" },
|
|
{ "PRN", "" },
|
|
{ "NUL", "/dev/null" },
|
|
{ "AUX", "" },
|
|
{ "LPT1", "" },
|
|
{ "LPT2", "" },
|
|
{ "LPT3", "" },
|
|
{ "LPT4", "" },
|
|
{ "COM1", "" },
|
|
{ "COM2", "" },
|
|
{ "COM3", "" },
|
|
{ "COM4", "" }
|
|
};
|
|
|
|
#define GET_DRIVE(path) \
|
|
(((path)[1] == ':') ? toupper((path)[0]) - 'A' : DOSFS_CurDrive)
|
|
|
|
/* DOS extended error status */
|
|
WORD DOS_ExtendedError;
|
|
BYTE DOS_ErrorClass;
|
|
BYTE DOS_ErrorAction;
|
|
BYTE DOS_ErrorLocus;
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_ValidDOSName
|
|
*
|
|
* Return 1 if Unix file 'name' is also a valid MS-DOS name
|
|
* (i.e. contains only valid DOS chars, lower-case only, fits in 8.3 format).
|
|
* File name can be terminated by '\0', '\\' or '/'.
|
|
*/
|
|
static int DOSFS_ValidDOSName( const char *name )
|
|
{
|
|
static const char invalid_chars[] = INVALID_DOS_CHARS "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
|
const char *p = name;
|
|
int len = 0;
|
|
|
|
if (*p == '.')
|
|
{
|
|
/* Check for "." and ".." */
|
|
p++;
|
|
if (*p == '.') p++;
|
|
/* All other names beginning with '.' are invalid */
|
|
return (IS_END_OF_NAME(*p));
|
|
}
|
|
while (!IS_END_OF_NAME(*p))
|
|
{
|
|
if (strchr( invalid_chars, *p )) return 0; /* Invalid char */
|
|
if (*p == '.') break; /* Start of the extension */
|
|
if (++len > 8) return 0; /* Name too long */
|
|
p++;
|
|
}
|
|
if (*p != '.') return 1; /* End of name */
|
|
p++;
|
|
if (IS_END_OF_NAME(*p)) return 0; /* Empty extension not allowed */
|
|
len = 0;
|
|
while (!IS_END_OF_NAME(*p))
|
|
{
|
|
if (strchr( invalid_chars, *p )) return 0; /* Invalid char */
|
|
if (*p == '.') return 0; /* Second extension not allowed */
|
|
if (++len > 3) return 0; /* Extension too long */
|
|
p++;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_CheckDotDot
|
|
*
|
|
* Remove all '.' and '..' at the beginning of 'name'.
|
|
*/
|
|
static const char * DOSFS_CheckDotDot( const char *name, char *buffer,
|
|
char sep , int *len )
|
|
{
|
|
char *p = buffer + strlen(buffer);
|
|
|
|
while (*name == '.')
|
|
{
|
|
if (IS_END_OF_NAME(name[1]))
|
|
{
|
|
name++;
|
|
while ((*name == '\\') || (*name == '/')) name++;
|
|
}
|
|
else if ((name[1] == '.') && IS_END_OF_NAME(name[2]))
|
|
{
|
|
name += 2;
|
|
while ((*name == '\\') || (*name == '/')) name++;
|
|
while ((p > buffer) && (*p != sep)) { p--; (*len)++; }
|
|
*p = '\0'; /* Remove trailing separator */
|
|
}
|
|
else break;
|
|
}
|
|
return name;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_ToDosFCBFormat
|
|
*
|
|
* Convert a file name to DOS FCB format (8+3 chars, padded with blanks),
|
|
* expanding wild cards and converting to upper-case in the process.
|
|
* File name can be terminated by '\0', '\\' or '/'.
|
|
* Return NULL if the name is not a valid DOS name.
|
|
*/
|
|
const char *DOSFS_ToDosFCBFormat( const char *name )
|
|
{
|
|
static const char invalid_chars[] = INVALID_DOS_CHARS;
|
|
static char buffer[12];
|
|
const char *p = name;
|
|
int i;
|
|
|
|
/* Check for "." and ".." */
|
|
if (*p == '.')
|
|
{
|
|
p++;
|
|
strcpy( buffer, ". " );
|
|
if (*p == '.') p++;
|
|
return (!*p || (*p == '/') || (*p == '\\')) ? buffer : NULL;
|
|
}
|
|
|
|
for (i = 0; i < 8; i++)
|
|
{
|
|
switch(*p)
|
|
{
|
|
case '\0':
|
|
case '\\':
|
|
case '/':
|
|
case '.':
|
|
buffer[i] = ' ';
|
|
break;
|
|
case '?':
|
|
p++;
|
|
/* fall through */
|
|
case '*':
|
|
buffer[i] = '?';
|
|
break;
|
|
default:
|
|
if (strchr( invalid_chars, *p )) return NULL;
|
|
buffer[i] = toupper(*p);
|
|
p++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (*p == '*')
|
|
{
|
|
/* Skip all chars after wildcard up to first dot */
|
|
while (*p && (*p != '/') && (*p != '\\') && (*p != '.')) p++;
|
|
}
|
|
else
|
|
{
|
|
/* Check if name too long */
|
|
if (*p && (*p != '/') && (*p != '\\') && (*p != '.')) return NULL;
|
|
}
|
|
if (*p == '.') p++; /* Skip dot */
|
|
|
|
for (i = 8; i < 11; i++)
|
|
{
|
|
switch(*p)
|
|
{
|
|
case '\0':
|
|
case '\\':
|
|
case '/':
|
|
buffer[i] = ' ';
|
|
break;
|
|
case '.':
|
|
return NULL; /* Second extension not allowed */
|
|
case '?':
|
|
p++;
|
|
/* fall through */
|
|
case '*':
|
|
buffer[i] = '?';
|
|
break;
|
|
default:
|
|
if (strchr( invalid_chars, *p )) return NULL;
|
|
buffer[i] = toupper(*p);
|
|
p++;
|
|
break;
|
|
}
|
|
}
|
|
buffer[11] = '\0';
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_ToDosDTAFormat
|
|
*
|
|
* Convert a file name from FCB to DTA format (name.ext, null-terminated)
|
|
* converting to upper-case in the process.
|
|
* File name can be terminated by '\0', '\\' or '/'.
|
|
* Return NULL if the name is not a valid DOS name.
|
|
*/
|
|
const char *DOSFS_ToDosDTAFormat( const char *name )
|
|
{
|
|
static char buffer[13];
|
|
char *p;
|
|
|
|
memcpy( buffer, name, 8 );
|
|
for (p = buffer + 8; (p > buffer) && (p[-1] == ' '); p--);
|
|
*p++ = '.';
|
|
memcpy( p, name + 8, 3 );
|
|
for (p += 3; p[-1] == ' '; p--);
|
|
if (p[-1] == '.') p--;
|
|
*p = '\0';
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_Match
|
|
*
|
|
* Check a DOS file name against a mask (both in FCB format).
|
|
*/
|
|
static int DOSFS_Match( const char *mask, const char *name )
|
|
{
|
|
int i;
|
|
for (i = 11; i > 0; i--, mask++, name++)
|
|
if ((*mask != '?') && (*mask != *name)) return 0;
|
|
return 1;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_ToDosDateTime
|
|
*
|
|
* Convert a Unix time in the DOS date/time format.
|
|
*/
|
|
void DOSFS_ToDosDateTime( time_t *unixtime, WORD *pDate, WORD *pTime )
|
|
{
|
|
struct tm *tm = localtime( unixtime );
|
|
if (pTime)
|
|
*pTime = (tm->tm_hour << 11) + (tm->tm_min << 5) + (tm->tm_sec / 2);
|
|
if (pDate)
|
|
*pDate = ((tm->tm_year - 80) << 9) + ((tm->tm_mon + 1) << 5)
|
|
+ tm->tm_mday;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_Hash
|
|
*
|
|
* Transform a Unix file name into a hashed DOS name. If the name is a valid
|
|
* DOS name, it is converted to upper-case; otherwise it is replaced by a
|
|
* hashed version that fits in 8.3 format.
|
|
* File name can be terminated by '\0', '\\' or '/'.
|
|
*/
|
|
static const char *DOSFS_Hash( const char *name, int dir_format )
|
|
{
|
|
static const char invalid_chars[] = INVALID_DOS_CHARS "~.";
|
|
static const char hash_chars[32] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ012345";
|
|
|
|
static char buffer[13];
|
|
const char *p, *ext;
|
|
char *dst;
|
|
unsigned short hash;
|
|
int i;
|
|
|
|
if (dir_format) strcpy( buffer, " " );
|
|
|
|
if (DOSFS_ValidDOSName( name ))
|
|
{
|
|
/* Check for '.' and '..' */
|
|
if (*name == '.')
|
|
{
|
|
buffer[0] = '.';
|
|
if (!dir_format) buffer[1] = buffer[2] = '\0';
|
|
if (name[1] == '.') buffer[1] = '.';
|
|
return buffer;
|
|
}
|
|
|
|
/* Simply copy the name, converting to uppercase */
|
|
|
|
for (dst = buffer; !IS_END_OF_NAME(*name) && (*name != '.'); name++)
|
|
*dst++ = toupper(*name);
|
|
if (*name == '.')
|
|
{
|
|
if (dir_format) dst = buffer + 8;
|
|
else *dst++ = '.';
|
|
for (name++; !IS_END_OF_NAME(*name); name++)
|
|
*dst++ = toupper(*name);
|
|
}
|
|
if (!dir_format) *dst = '\0';
|
|
}
|
|
else
|
|
{
|
|
/* Compute the hash code of the file name */
|
|
/* If you know something about hash functions, feel free to */
|
|
/* insert a better algorithm here... */
|
|
for (p = name, hash = 0xbeef; !IS_END_OF_NAME(p[1]); p++)
|
|
hash = (hash << 3) ^ (hash >> 5) ^ *p ^ (p[1] << 8);
|
|
hash = (hash << 3) ^ (hash >> 5) ^ *p; /* Last character */
|
|
|
|
/* Find last dot for start of the extension */
|
|
for (p = name+1, ext = NULL; !IS_END_OF_NAME(*p); p++)
|
|
if (*p == '.') ext = p;
|
|
if (ext && IS_END_OF_NAME(ext[1]))
|
|
ext = NULL; /* Empty extension ignored */
|
|
|
|
/* Copy first 4 chars, replacing invalid chars with '_' */
|
|
for (i = 4, p = name, dst = buffer; i > 0; i--, p++)
|
|
{
|
|
if (IS_END_OF_NAME(*p) || (p == ext)) break;
|
|
*dst++ = strchr( invalid_chars, *p ) ? '_' : toupper(*p);
|
|
}
|
|
/* Pad to 5 chars with '~' */
|
|
while (i-- >= 0) *dst++ = '~';
|
|
|
|
/* Insert hash code converted to 3 ASCII chars */
|
|
*dst++ = hash_chars[(hash >> 10) & 0x1f];
|
|
*dst++ = hash_chars[(hash >> 5) & 0x1f];
|
|
*dst++ = hash_chars[hash & 0x1f];
|
|
|
|
/* Copy the first 3 chars of the extension (if any) */
|
|
if (ext)
|
|
{
|
|
if (!dir_format) *dst++ = '.';
|
|
for (i = 3, ext++; (i > 0) && !IS_END_OF_NAME(*ext); i--, ext++)
|
|
*dst++ = toupper(*ext);
|
|
}
|
|
if (!dir_format) *dst = '\0';
|
|
}
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_FindUnixName
|
|
*
|
|
* Find the Unix file name in a given directory that corresponds to
|
|
* a file name (either in Unix or DOS format).
|
|
* File name can be terminated by '\0', '\\' or '/'.
|
|
* Return 1 if OK, 0 if no file name matches.
|
|
*/
|
|
static int DOSFS_FindUnixName( const char *path, const char *name,
|
|
char *buffer, int maxlen )
|
|
{
|
|
DIR *dir;
|
|
struct dirent *dirent;
|
|
|
|
const char *dos_name = DOSFS_ToDosFCBFormat( name );
|
|
const char *p = strchr( name, '/' );
|
|
int len = p ? (int)(p - name) : strlen(name);
|
|
|
|
dprintf_dosfs( stddeb, "DOSFS_FindUnixName: %s %s\n", path, name );
|
|
|
|
if ((p = strchr( name, '\\' ))) len = MIN( (int)(p - name), len );
|
|
|
|
if (!(dir = opendir( path )))
|
|
{
|
|
dprintf_dosfs( stddeb, "DOSFS_FindUnixName(%s,%s): can't open dir\n",
|
|
path, name );
|
|
return 0;
|
|
}
|
|
while ((dirent = readdir( dir )) != NULL)
|
|
{
|
|
/* Check against Unix name */
|
|
if ((len == strlen(dirent->d_name) &&
|
|
!memcmp( dirent->d_name, name, len ))) break;
|
|
if (dos_name)
|
|
{
|
|
/* Check against hashed DOS name */
|
|
const char *hash_name = DOSFS_Hash( dirent->d_name, TRUE );
|
|
if (!strcmp( dos_name, hash_name )) break;
|
|
}
|
|
}
|
|
if (dirent) lstrcpyn( buffer, dirent->d_name, maxlen );
|
|
closedir( dir );
|
|
dprintf_dosfs( stddeb, "DOSFS_FindUnixName(%s,%s) -> %s\n",
|
|
path, name, dirent ? buffer : "** Not found **" );
|
|
return (dirent != NULL);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_IsDevice
|
|
*
|
|
* Check if a DOS file name represents a DOS device. Returns the name
|
|
* of the associated Unix device, or NULL if not found.
|
|
*/
|
|
const char *DOSFS_IsDevice( const char *name )
|
|
{
|
|
int i;
|
|
const char *p;
|
|
|
|
if (name[1] == ':') name += 2;
|
|
if ((p = strrchr( name, '/' ))) name = p + 1;
|
|
if ((p = strrchr( name, '\\' ))) name = p + 1;
|
|
for (i = 0; i < sizeof(DOSFS_Devices)/sizeof(DOSFS_Devices[0]); i++)
|
|
{
|
|
const char *dev = DOSFS_Devices[i][0];
|
|
if (!lstrncmpi( dev, name, strlen(dev) ))
|
|
{
|
|
p = name + strlen( dev );
|
|
if (*p == ':') p++;
|
|
if (!*p || (*p == '.')) return DOSFS_Devices[i][1];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_GetUnixFileName
|
|
*
|
|
* Convert a file name (DOS or mixed DOS/Unix format) to a valid Unix name.
|
|
* Return NULL if one of the path components does not exist. The last path
|
|
* component is only checked if 'check_last' is non-zero.
|
|
*/
|
|
const char * DOSFS_GetUnixFileName( const char * name, int check_last )
|
|
{
|
|
static char buffer[MAX_PATHNAME_LEN];
|
|
int drive, len, found;
|
|
char *p, *root;
|
|
|
|
dprintf_dosfs( stddeb, "DOSFS_GetUnixFileName: %s\n", name );
|
|
if (name[1] == ':')
|
|
{
|
|
drive = toupper(name[0]) - 'A';
|
|
name += 2;
|
|
}
|
|
else if (name[0] == '/') /* Absolute Unix path? */
|
|
{
|
|
if ((drive = DRIVE_FindDriveRoot( &name )) == -1)
|
|
{
|
|
fprintf( stderr, "Warning: %s not accessible from a DOS drive\n",
|
|
name );
|
|
/* Assume it really was a DOS name */
|
|
drive = DRIVE_GetCurrentDrive();
|
|
}
|
|
}
|
|
else drive = DRIVE_GetCurrentDrive();
|
|
|
|
if (!DRIVE_IsValid(drive))
|
|
{
|
|
DOS_ERROR( ER_InvalidDrive, EC_MediaError, SA_Abort, EL_Disk );
|
|
return NULL;
|
|
}
|
|
lstrcpyn( buffer, DRIVE_GetRoot(drive), MAX_PATHNAME_LEN );
|
|
if (buffer[1]) root = buffer + strlen(buffer);
|
|
else root = buffer; /* root directory */
|
|
|
|
if ((*name == '\\') || (*name == '/'))
|
|
{
|
|
while ((*name == '\\') || (*name == '/')) name++;
|
|
}
|
|
else
|
|
{
|
|
lstrcpyn( root + 1, DRIVE_GetUnixCwd(drive),
|
|
MAX_PATHNAME_LEN - (int)(root - buffer) - 1 );
|
|
if (root[1]) *root = '/';
|
|
}
|
|
|
|
p = buffer[1] ? buffer + strlen(buffer) : buffer;
|
|
len = MAX_PATHNAME_LEN - strlen(buffer);
|
|
found = 1;
|
|
while (*name && found)
|
|
{
|
|
const char *newname = DOSFS_CheckDotDot( name, root, '/', &len );
|
|
if (newname != name)
|
|
{
|
|
p = root + strlen(root);
|
|
name = newname;
|
|
continue;
|
|
}
|
|
if (len <= 1)
|
|
{
|
|
DOS_ERROR( ER_PathNotFound, EC_NotFound, SA_Abort, EL_Disk );
|
|
return NULL;
|
|
}
|
|
if ((found = DOSFS_FindUnixName( buffer, name, p+1, len-1 )))
|
|
{
|
|
*p = '/';
|
|
len -= strlen(p);
|
|
p += strlen(p);
|
|
while (!IS_END_OF_NAME(*name)) name++;
|
|
}
|
|
else if (!check_last)
|
|
{
|
|
*p++ = '/';
|
|
for (len--; !IS_END_OF_NAME(*name) && (len > 1); name++, len--)
|
|
*p++ = tolower(*name);
|
|
*p = '\0';
|
|
}
|
|
while ((*name == '\\') || (*name == '/')) name++;
|
|
}
|
|
if (!found)
|
|
{
|
|
if (check_last)
|
|
{
|
|
DOS_ERROR( ER_FileNotFound, EC_NotFound, SA_Abort, EL_Disk );
|
|
return NULL;
|
|
}
|
|
if (*name) /* Not last */
|
|
{
|
|
DOS_ERROR( ER_PathNotFound, EC_NotFound, SA_Abort, EL_Disk );
|
|
return NULL;
|
|
}
|
|
}
|
|
if (!buffer[0]) strcpy( buffer, "/" );
|
|
dprintf_dosfs( stddeb, "DOSFS_GetUnixFileName: returning %s\n", buffer );
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_GetDosTrueName
|
|
*
|
|
* Convert a file name (DOS or Unix format) to a complete DOS name.
|
|
* Return NULL if the path name is invalid or too long.
|
|
* The unix_format flag is a hint that the file name is in Unix format.
|
|
*/
|
|
const char * DOSFS_GetDosTrueName( const char *name, int unix_format )
|
|
{
|
|
static char buffer[MAX_PATHNAME_LEN];
|
|
int drive, len;
|
|
char *p;
|
|
|
|
dprintf_dosfs( stddeb, "DOSFS_GetDosTrueName(%s,%d)\n", name, unix_format);
|
|
if (name[1] == ':')
|
|
{
|
|
drive = toupper(name[0]) - 'A';
|
|
name += 2;
|
|
}
|
|
else if (name[0] == '/') /* Absolute Unix path? */
|
|
{
|
|
if ((drive = DRIVE_FindDriveRoot( &name )) == -1)
|
|
{
|
|
fprintf( stderr, "Warning: %s not accessible from a DOS drive\n",
|
|
name );
|
|
/* Assume it really was a DOS name */
|
|
drive = DRIVE_GetCurrentDrive();
|
|
}
|
|
}
|
|
else drive = DRIVE_GetCurrentDrive();
|
|
|
|
if (!DRIVE_IsValid(drive))
|
|
{
|
|
DOS_ERROR( ER_InvalidDrive, EC_MediaError, SA_Abort, EL_Disk );
|
|
return NULL;
|
|
}
|
|
|
|
p = buffer;
|
|
*p++ = 'A' + drive;
|
|
*p++ = ':';
|
|
if (IS_END_OF_NAME(*name))
|
|
{
|
|
while ((*name == '\\') || (*name == '/')) name++;
|
|
}
|
|
else
|
|
{
|
|
*p++ = '\\';
|
|
lstrcpyn( p, DRIVE_GetDosCwd(drive), sizeof(buffer) - 3 );
|
|
if (*p) p += strlen(p); else p--;
|
|
}
|
|
*p = '\0';
|
|
len = MAX_PATHNAME_LEN - (int)(p - buffer);
|
|
|
|
while (*name)
|
|
{
|
|
const char *newname = DOSFS_CheckDotDot( name, buffer+2, '\\', &len );
|
|
if (newname != name)
|
|
{
|
|
p = buffer + strlen(buffer);
|
|
name = newname;
|
|
continue;
|
|
}
|
|
if (len <= 1)
|
|
{
|
|
DOS_ERROR( ER_PathNotFound, EC_NotFound, SA_Abort, EL_Disk );
|
|
return NULL;
|
|
}
|
|
*p++ = '\\';
|
|
if (unix_format) /* Hash it into a DOS name */
|
|
{
|
|
lstrcpyn( p, DOSFS_Hash( name, FALSE ), len );
|
|
len -= strlen(p);
|
|
p += strlen(p);
|
|
while (!IS_END_OF_NAME(*name)) name++;
|
|
}
|
|
else /* Already DOS format, simply upper-case it */
|
|
{
|
|
while (!IS_END_OF_NAME(*name) && (len > 1))
|
|
{
|
|
*p++ = toupper(*name);
|
|
name++;
|
|
len--;
|
|
}
|
|
*p = '\0';
|
|
}
|
|
while ((*name == '\\') || (*name == '/')) name++;
|
|
}
|
|
if (!buffer[2])
|
|
{
|
|
buffer[2] = '\\';
|
|
buffer[3] = '\0';
|
|
}
|
|
dprintf_dosfs( stddeb, "DOSFS_GetDosTrueName: returning %s\n", buffer );
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DOSFS_FindNext
|
|
*
|
|
* Find the next matching file. Return the number of entries read to find
|
|
* the matching one, or 0 if no more entries.
|
|
*/
|
|
int DOSFS_FindNext( const char *path, const char *mask, int drive,
|
|
BYTE attr, int skip, DOS_DIRENT *entry )
|
|
{
|
|
static DIR *dir = NULL;
|
|
struct dirent *dirent;
|
|
int count = 0;
|
|
static char buffer[MAX_PATHNAME_LEN];
|
|
static int cur_pos = 0;
|
|
char *p;
|
|
const char *hash_name;
|
|
|
|
if ((attr & ~(FA_UNUSED | FA_ARCHIVE | FA_RDONLY)) == FA_LABEL)
|
|
{
|
|
time_t now = time(NULL);
|
|
if (skip) return 0;
|
|
strcpy( entry->name, DRIVE_GetLabel( drive ) );
|
|
entry->attr = FA_LABEL;
|
|
entry->size = 0;
|
|
DOSFS_ToDosDateTime( &now, &entry->date, &entry->time );
|
|
return 1;
|
|
}
|
|
|
|
/* Check the cached directory */
|
|
if (dir && !strcmp( buffer, path ) && (cur_pos <= skip)) skip -= cur_pos;
|
|
else /* Not in the cache, open it anew */
|
|
{
|
|
dprintf_dosfs( stddeb, "DOSFS_FindNext: cache miss, path=%s skip=%d buf=%s cur=%d\n",
|
|
path, skip, buffer, cur_pos );
|
|
cur_pos = skip;
|
|
if (dir) closedir(dir);
|
|
if (!(dir = opendir( path ))) return 0;
|
|
lstrcpyn( buffer, path, sizeof(buffer) - 1 );
|
|
}
|
|
strcat( buffer, "/" );
|
|
p = buffer + strlen(buffer);
|
|
attr |= FA_UNUSED | FA_ARCHIVE | FA_RDONLY;
|
|
|
|
while ((dirent = readdir( dir )) != NULL)
|
|
{
|
|
if (skip-- > 0) continue;
|
|
count++;
|
|
hash_name = DOSFS_Hash( dirent->d_name, TRUE );
|
|
if (!DOSFS_Match( mask, hash_name )) continue;
|
|
lstrcpyn( p, dirent->d_name, sizeof(buffer) - (int)(p - buffer) );
|
|
|
|
if (!FILE_Stat( buffer, &entry->attr, &entry->size,
|
|
&entry->date, &entry->time ))
|
|
{
|
|
fprintf( stderr, "DOSFS_FindNext: can't stat %s\n", buffer );
|
|
continue;
|
|
}
|
|
if (entry->attr & ~attr) continue;
|
|
strcpy( entry->name, hash_name );
|
|
lstrcpyn( entry->unixname, dirent->d_name, sizeof(entry->unixname) );
|
|
dprintf_dosfs( stddeb, "DOSFS_FindNext: returning %s %02x %ld\n",
|
|
entry->name, entry->attr, entry->size );
|
|
cur_pos += count;
|
|
p[-1] = '\0'; /* Remove trailing slash in buffer */
|
|
return count;
|
|
}
|
|
closedir( dir );
|
|
dir = NULL;
|
|
return 0; /* End of directory */
|
|
}
|