Mon Jun 19 20:29:50 1995 Alexandre Julliard (julliard@sunsite.unc.edu) * [debugger/*.c] Modified debugger to use segmented pointers everywhere. * [if1632/shell.spec] [if1632/sound.spec] [if1632/user.spec] Declared all functions that return only 16-bit as 'pascal16'. * [include/ldt.h] [memory/ldt.c] Export LDT_EntryToBytes (new) and LDT_BytesToEntry for DPMI. Maintain a copy of the selector flags, removing the need to make a system call to retrieve an LDT entry. * [loader/module.c] Fixed bug with module file handle cache. * [loader/ne_resource.c] Fixed file name bug in NE_AccessResource(). * [loader/resource.c] Fixed bug in LoadIcon() that caused wrong colors to be used for the icon mask. * [loader/signal.c] Moved instruction emulation to miscemu/instr.c. * [misc/dos_fs.c] [miscemu/int21.c] Lots of small fixes, thanks to Morten Welinder. * [miscemu/dpmi.c] More complete DPMI emulation. * [miscemu/instr.c] Added support for prefixes in instructions to emulate. * [miscemu/int2f.c] Use register macros instead of destroying the high part of 32-bit registers. * [objects/dc.c] Fixed bug in GetDCState() that failed to clear the new DC. * [rc/sysres.rc] Removed dialogs 11 and 12 that were never used. * [tools/build.c] 'pascal16' generated functions did not save %dx. Removed use of %fs to access the stack. %ds is no longer initialized before calling a 16-bit routine. * [windows/defwnd.c] Accept a NULL pointer as window title. * [windows/mdi.c] MDICascade: skip iconic windows. Implemented CalcChildScroll(). * [windows/utility.c] Fixed MulDiv() for illegal values. * [windows/win.c] Fixed X error in CreateWindowEx() when WM_NCCALCSIZE returned a zero width or height. Sun Jun 18 22:22:30 MET DST 1995 Fons Botman (botman@inter.nl.net) * [controls/edit.c] Fixed "uninitalized" message which -Wall couldnt see to be ok in EDIT_WriteText. * [include/debug.h] Added define for extra checks in API definitions during debugging. * [loader/ne_image.c] Added newline in NE_FixupPrologs to avoid long lines. * [misc/dos_fs.c] Added extra safety check in DOS_ValidDrive. * [misc/exec.c] Fixed definition of ExitWindows. Sun Jun 18 21:16:08 1995 Bernd Schmidt <crux@pool.informatik.rwth-aachen.de> * [controls/edit.c] Some fixes, mostly for memory management, but also for text selection and tab postitions. General cleanup. Notepad.exe now works. * [controls/combo.c] Fix: the hwnd field of the DRAWITEMSTRUCT should always be that of the combo box, not the ComboLBox that belongs to it. * [controls/listbox.c] Handle itemID field correctly throughout. * [memory/local.c] Implemented flag LMEM_ZEROINIT. LocalReAlloc() could trash the heap. Fixed. * [objects/font.c] FONT_MatchFont(): don't get confused by negative widths. Fixed a segfault in EnumFonts(). * [objects/text.c] DrawText(): DT_CALCRECT implies DT_NOCLIP. * [objects/dcvalues.c] MAKELONG was used with bad parameters in DC_GET_X_Y. * [windows/dialog.c] Don't show the dialog if WS_VISIBLE isn't set in the template. * [windows/utility.c] UTILITY_convertArgs(): Never pass an expression containing ++ into a macro... * [windows/win.c] SetParent() should unlink the window before changing the parent. * [windows/message.c] Don't call timer functions via CallWindowProc(), since it checks whether hwnd==0 and does not call the function in that case. * [miscemu/instr.c] Ignore interrupt 0x3D, for VBRUN300.DLL. * [misc/commdlg.c] Don't rely on the itemData field of the DRAWITEMSTRUCT to contain a pointer to the item text. * [if1632/relay.c] Disable OLE and DDEML DLLs by default, since they contain nothing but stubs anyway. SHELL, COMMDLG and WIN87EM are left enabled, although some programs may work better without them. * [multimedia/*.c] [include/multimedia.h] [include/driver.h] Begun cleaning things up a little. Replaced printfs with dprintf_ macros, made functions static where possible, and some other minor changes. Sun Jun 11 23:19:10 1995 Martin von Loewis <martin@informatik.hu-berlin.de> * [debugger/dbg.y][debugger/dbg.l] Removed special handling for FILE_IDENTIFER, because it caused problems with x/<format> statements. * [debugger/info.c] Use SC_ESP instead of SC_EIP for stack dump. * [misc/compobj.c][if1632/compobj.spec] CoBuildVersion, CoInitialize, CoUninitialize: new functions * [misc/ole2.c][if1632/ole2.spec][misc/Imakefile][include/ole2.h] New files ole2.c, ole2.h OleBuildVersion, OleInitialize, OleUninitialize: new functions * [if1632/ole2disp.spec] Added missing ordinals above 109 * [misc/ole2nls.c][if1632/ole2nls.spec][include/winnls.h] New file winnls.h GetLocaleInfoA: new function * [if1632/shell.spec] Added FindEnvironmentString as stub * [misc/olecli.c][if1632/olecli.spec] OleIsDcMeta: New function * [objects/font][misc/gdi.spec] GetKerningPairs: new function * [misc/shell.c] ShellExecute: Implemented support for starting programs * [if1632/user.spec] Inserted missing relay to GetClipCursor Sun Jun 11 20:34:47 1995 Bernd Schmidt <crux@pool.informatik.rwth-aachen.de> * [controls/edit.c] Fix a problem with the local heap. * [include/wintypes.h] Fixed wrong declarations of CATCHBUF and LPCATCHBUF. * [include/mdi.h] [windows/mdi.c] This code still assumed segmented address==linear address. Fixed. * [include/msdos.h] [misc/dos_fs.c] The filemask field of the dosdirent structure could be overrun. Fixed. If you had a file called foobar and a file called foo, trying to FindFile(foo) could accidentally find file foobar instead. Fixed. * [misc/file.c] OpenFile(): Always return the full pathname in ofs->szPathName. This also fixes GetModuleFilename(). Prevent _lclose() from closing stderr or stdout. * [misc/profile.c] Search for .ini files in the path of the current module as well. (Needed by Lotus Organizer.) * [loader/task.c] [loader/ne_image.c] [loader/module.c] [memory/local.c] Local heaps are now initialized by InitTask() for executables. DLLs have to call LocalInit() themselves, LocalInit() has to put the heap at the end of the segment when called with start==0. We no longer allocate the DGROUP with 64k on startup, but grow the local heap in LOCAL_GetBlock() when necessary. * [loader/module.c] LoadLibrary() should call LoadModule() in all cases, even if the DLL is already loaded, to ensure that the reference count is correct. * [loader/ne_image.c] Some changes to function prolog fixup. Does anyone know exactly how this is supposed to work? I am only guessing here. In NE_InitializeDLLs(), initialize the DLLs a module refers to before the module itself. * [loader/task.c] Initialize instance data at the beginning of the DGROUP in InitTask(). * [memory/local.c] Some fixes for moveable blocks. * [memory/selector.c] All the IsBad*Pointer() functions returned exactly the wrong boolean value in all cases! * [objects/bitblt.c] Fixed another null pointer dereference in debugging output. * [objects/font.c] Some more recovery possibilities for FONT_MatchFont() if a specified font does not exist. * [windows/win.c] The dialog code may call CreateWindowEx with an integer in windowName. This happens for static icon controls that expect a resource ID as the window name. CreateWindowEx() used to crash. Fixed. * [windows/class.c] [windows/win.c] Window classes are owned by modules, not instances. Changed RegisterClass(), UnregisterClass(), GetClassInfo() and CreateWindowEx() accordingly. Sat Jun 10 16:10:53 1995 Olaf Flebbe <o.flebbe@science-computing.uni-tuebingen.de> * [miscemu/int21.c] clock.exe was displaying incorrect year. Fri Jun 9 20:36:56 1995 Victor Schneider <tailor@crl.com> * [include/cursor.h] [windows/cursor.c] Implemented CreateCursorIconIndirect().
449 lines
12 KiB
C
449 lines
12 KiB
C
/************************************************************************
|
|
* FILE.C Copyright (C) 1993 John Burton
|
|
*
|
|
* File I/O routines for the Linux Wine Project.
|
|
*
|
|
* WARNING : Many options of OpenFile are not yet implemeted.
|
|
*
|
|
* NOV 93 Erik Bos (erik@xs4all.nl)
|
|
* - removed ParseDosFileName, and DosDrive structures.
|
|
* - structures dynamically configured at runtime.
|
|
* - _lopen modified to use DOS_GetUnixFileName.
|
|
* - Existing functions modified to use dosfs functions.
|
|
* - Added _llseek, _lcreat, GetDriveType, GetTempDrive,
|
|
* GetWindowsDirectory, GetSystemDirectory, GetTempFileName.
|
|
*
|
|
************************************************************************/
|
|
|
|
#include <stdio.h>
|
|
#include <fcntl.h>
|
|
#include <limits.h>
|
|
#include <unistd.h>
|
|
#include <time.h>
|
|
#include <sys/stat.h>
|
|
#include <string.h>
|
|
#include "dos_fs.h"
|
|
#include "windows.h"
|
|
#include "msdos.h"
|
|
#include "options.h"
|
|
#include "stddebug.h"
|
|
#include "debug.h"
|
|
|
|
#define MAX_PATH 255
|
|
|
|
char WindowsDirectory[256], SystemDirectory[256], TempDirectory[256];
|
|
|
|
/***************************************************************************
|
|
_lopen
|
|
|
|
Emulate the _lopen windows call
|
|
***************************************************************************/
|
|
INT _lopen (LPSTR lpPathName, INT iReadWrite)
|
|
{
|
|
int handle;
|
|
char *UnixFileName;
|
|
int mode = 0;
|
|
|
|
dprintf_file(stddeb, "_lopen: open('%s', %X);\n", lpPathName, iReadWrite);
|
|
if ((UnixFileName = DOS_GetUnixFileName(lpPathName)) == NULL)
|
|
return HFILE_ERROR;
|
|
switch(iReadWrite & 3)
|
|
{
|
|
case OF_READ: mode = O_RDONLY; break;
|
|
case OF_WRITE: mode = O_WRONLY; break;
|
|
case OF_READWRITE: mode = O_RDWR; break;
|
|
}
|
|
handle = open( UnixFileName, mode );
|
|
if (( handle == -1 ) && Options.allowReadOnly)
|
|
handle = open( UnixFileName, O_RDONLY );
|
|
|
|
dprintf_file(stddeb, "_lopen: open: %s (handle %d)\n", UnixFileName, handle);
|
|
|
|
if (handle == -1)
|
|
return HFILE_ERROR;
|
|
else
|
|
return handle;
|
|
}
|
|
|
|
/***************************************************************************
|
|
_lread
|
|
***************************************************************************/
|
|
INT _lread (INT hFile, LPSTR lpBuffer, WORD wBytes)
|
|
{
|
|
int result;
|
|
|
|
dprintf_file(stddeb, "_lread: handle %d, buffer = %ld, length = %d\n",
|
|
hFile, (long) lpBuffer, wBytes);
|
|
|
|
result = read (hFile, lpBuffer, wBytes);
|
|
|
|
if (result == -1)
|
|
return HFILE_ERROR;
|
|
else
|
|
return result;
|
|
}
|
|
|
|
/****************************************************************************
|
|
_lwrite
|
|
****************************************************************************/
|
|
INT _lwrite (INT hFile, LPCSTR lpBuffer, WORD wBytes)
|
|
{
|
|
int result;
|
|
|
|
dprintf_file(stddeb, "_lwrite: handle %d, buffer = %ld, length = %d\n",
|
|
hFile, (long) lpBuffer, wBytes);
|
|
|
|
result = write (hFile, lpBuffer, wBytes);
|
|
|
|
if (result == -1)
|
|
return HFILE_ERROR;
|
|
else
|
|
return result;
|
|
}
|
|
|
|
/***************************************************************************
|
|
_lclose
|
|
***************************************************************************/
|
|
INT _lclose (INT hFile)
|
|
{
|
|
dprintf_file(stddeb, "_lclose: handle %d\n", hFile);
|
|
|
|
if (hFile == 1 || hFile == 2) {
|
|
fprintf( stderr, "Program attempted to close stdout or stderr!\n" );
|
|
return 0;
|
|
}
|
|
if (close (hFile))
|
|
return HFILE_ERROR;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
/**************************************************************************
|
|
OpenFile
|
|
**************************************************************************/
|
|
INT OpenFile (LPSTR lpFileName, LPOFSTRUCT ofs, WORD wStyle)
|
|
{
|
|
char filename[MAX_PATH+1];
|
|
int action;
|
|
struct stat s;
|
|
struct tm *now;
|
|
int res, handle;
|
|
int verify_time = 0;
|
|
|
|
dprintf_file(stddeb,"Openfile(%s,<struct>,%d)\n",lpFileName,wStyle);
|
|
|
|
action = wStyle & 0xff00;
|
|
|
|
|
|
/* OF_CREATE is completly different from all other options, so
|
|
handle it first */
|
|
|
|
if (action & OF_CREATE)
|
|
{
|
|
char *unixfilename;
|
|
|
|
if (!(action & OF_REOPEN)) strcpy(ofs->szPathName, lpFileName);
|
|
ofs->cBytes = sizeof(OFSTRUCT);
|
|
ofs->fFixedDisk = FALSE;
|
|
ofs->nErrCode = 0;
|
|
*((int*)ofs->reserved) = 0;
|
|
|
|
if ((unixfilename = DOS_GetUnixFileName (ofs->szPathName)) == NULL)
|
|
{
|
|
errno_to_doserr();
|
|
ofs->nErrCode = ExtendedError;
|
|
return -1;
|
|
}
|
|
handle = open (unixfilename, (wStyle & 0x0003) | O_CREAT, 0x666);
|
|
if (handle == -1)
|
|
{
|
|
errno_to_doserr();
|
|
ofs->nErrCode = ExtendedError;
|
|
}
|
|
return handle;
|
|
}
|
|
|
|
|
|
/* If path isn't given, try to find the file. */
|
|
|
|
if (!(action & OF_REOPEN))
|
|
{
|
|
char temp[MAX_PATH + 1];
|
|
|
|
if(index(lpFileName,'\\') || index(lpFileName,'/') ||
|
|
index(lpFileName,':'))
|
|
{
|
|
strcpy (filename,lpFileName);
|
|
goto found;
|
|
}
|
|
/* Try current directory */
|
|
if (DOS_FindFile(temp, MAX_PATH, lpFileName, NULL, ".")) {
|
|
strcpy(filename, DOS_GetDosFileName(temp));
|
|
goto found;
|
|
}
|
|
|
|
/* Try Windows directory */
|
|
GetWindowsDirectory(filename, MAX_PATH);
|
|
if (DOS_FindFile(temp, MAX_PATH, lpFileName, NULL, filename)) {
|
|
strcpy(filename, DOS_GetDosFileName(temp));
|
|
goto found;
|
|
}
|
|
|
|
/* Try Windows system directory */
|
|
GetSystemDirectory(filename, MAX_PATH);
|
|
if (DOS_FindFile(temp, MAX_PATH, lpFileName, NULL, filename)) {
|
|
strcpy(filename, DOS_GetDosFileName(temp));
|
|
goto found;
|
|
}
|
|
|
|
/* Try the path of the current executable */
|
|
if (GetCurrentTask())
|
|
{
|
|
char *p;
|
|
GetModuleFileName( GetCurrentTask(), filename, MAX_PATH );
|
|
if ((p = strrchr( filename, '\\' )))
|
|
{
|
|
p[1] = '\0';
|
|
if (DOS_FindFile(temp, MAX_PATH, lpFileName, NULL, filename)) {
|
|
strcpy(filename, DOS_GetDosFileName(temp));
|
|
goto found;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Try all directories in path */
|
|
|
|
if (DOS_FindFile(temp,MAX_PATH,lpFileName,NULL,WindowsPath))
|
|
{
|
|
strcpy(filename, DOS_GetDosFileName(temp));
|
|
goto found;
|
|
}
|
|
/* ??? shouldn't we give an error here? */
|
|
strcpy (filename, lpFileName);
|
|
|
|
found:
|
|
|
|
ofs->cBytes = sizeof(OFSTRUCT);
|
|
ofs->fFixedDisk = FALSE;
|
|
strcpy(ofs->szPathName, filename);
|
|
ofs->nErrCode = 0;
|
|
if (!(action & OF_VERIFY))
|
|
*((int*)ofs->reserved) = 0;
|
|
}
|
|
|
|
|
|
if (action & OF_PARSE)
|
|
return 0;
|
|
|
|
if (action & OF_DELETE)
|
|
return unlink(ofs->szPathName);
|
|
|
|
|
|
/* Now on to getting some information about that file */
|
|
|
|
if ((res = stat(DOS_GetUnixFileName(ofs->szPathName), &s)))
|
|
{
|
|
errno_to_doserr();
|
|
ofs->nErrCode = ExtendedError;
|
|
return -1;
|
|
}
|
|
|
|
now = localtime (&s.st_mtime);
|
|
|
|
if (action & OF_VERIFY)
|
|
verify_time = *((int*)ofs->reserved);
|
|
|
|
*((WORD*)(&ofs->reserved[2]))=
|
|
((now->tm_hour * 0x2000) + (now->tm_min * 0x20) + (now->tm_sec / 2));
|
|
*((WORD*)(&ofs->reserved[0]))=
|
|
((now->tm_year * 0x200) + (now->tm_mon * 0x20) + now->tm_mday);
|
|
|
|
|
|
if (action & OF_VERIFY)
|
|
return (verify_time != *((int*)ofs->reserved));
|
|
|
|
if (action & OF_EXIST)
|
|
return 0;
|
|
|
|
if ((handle = _lopen( ofs->szPathName, wStyle )) == -1)
|
|
{
|
|
ofs->nErrCode = 2; /* not found */
|
|
return -1;
|
|
}
|
|
return handle;
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
SetHandleCount
|
|
|
|
Changes the number of file handles available to the application. Since
|
|
Linux isn't limited to 20 files, this one's easy. - SL
|
|
**************************************************************************/
|
|
|
|
#if !defined (OPEN_MAX)
|
|
/* This one is for the Sun */
|
|
#define OPEN_MAX _POSIX_OPEN_MAX
|
|
#endif
|
|
WORD SetHandleCount (WORD wNumber)
|
|
{
|
|
dprintf_file(stddeb,"SetHandleCount(%d)\n",wNumber);
|
|
return((wNumber<OPEN_MAX) ? wNumber : OPEN_MAX);
|
|
}
|
|
|
|
/***************************************************************************
|
|
_llseek
|
|
***************************************************************************/
|
|
LONG _llseek (INT hFile, LONG lOffset, INT nOrigin)
|
|
{
|
|
int origin;
|
|
|
|
dprintf_file(stddeb, "_llseek: handle %d, offset %ld, origin %d\n",
|
|
hFile, lOffset, nOrigin);
|
|
|
|
switch (nOrigin) {
|
|
case 1: origin = SEEK_CUR;
|
|
break;
|
|
case 2: origin = SEEK_END;
|
|
break;
|
|
default: origin = SEEK_SET;
|
|
break;
|
|
}
|
|
|
|
return lseek(hFile, lOffset, origin);
|
|
}
|
|
|
|
/***************************************************************************
|
|
_lcreat
|
|
***************************************************************************/
|
|
INT _lcreat (LPSTR lpszFilename, INT fnAttribute)
|
|
{
|
|
int handle;
|
|
char *UnixFileName;
|
|
|
|
dprintf_file(stddeb, "_lcreat: filename %s, attributes %d\n",
|
|
lpszFilename, fnAttribute);
|
|
if ((UnixFileName = DOS_GetUnixFileName(lpszFilename)) == NULL)
|
|
return HFILE_ERROR;
|
|
handle = open (UnixFileName, O_CREAT | O_TRUNC | O_WRONLY, 426);
|
|
|
|
if (handle == -1)
|
|
return HFILE_ERROR;
|
|
else
|
|
return handle;
|
|
}
|
|
|
|
/***************************************************************************
|
|
GetDriveType
|
|
***************************************************************************/
|
|
UINT GetDriveType(INT drive)
|
|
{
|
|
|
|
dprintf_file(stddeb,"GetDriveType %c:\n",'A'+drive);
|
|
|
|
if (!DOS_ValidDrive(drive))
|
|
return DRIVE_DOESNOTEXIST;
|
|
|
|
if (drive == 0 || drive == 1)
|
|
return DRIVE_REMOVABLE;
|
|
|
|
return DRIVE_FIXED;
|
|
}
|
|
|
|
/***************************************************************************
|
|
GetTempDrive
|
|
***************************************************************************/
|
|
BYTE GetTempDrive(BYTE chDriveLetter)
|
|
{
|
|
dprintf_file(stddeb,"GetTempDrive (%d)\n",chDriveLetter);
|
|
if (TempDirectory[1] == ':') return TempDirectory[0];
|
|
else return 'C';
|
|
}
|
|
|
|
/***************************************************************************
|
|
GetWindowsDirectory
|
|
***************************************************************************/
|
|
UINT GetWindowsDirectory(LPSTR lpszSysPath, UINT cbSysPath)
|
|
{
|
|
if (cbSysPath < strlen(WindowsDirectory))
|
|
*lpszSysPath = 0;
|
|
else
|
|
strcpy(lpszSysPath, WindowsDirectory);
|
|
|
|
dprintf_file(stddeb,"GetWindowsDirectory (%s)\n",lpszSysPath);
|
|
|
|
ChopOffSlash(lpszSysPath);
|
|
return(strlen(lpszSysPath));
|
|
}
|
|
/***************************************************************************
|
|
GetSystemDirectory
|
|
***************************************************************************/
|
|
UINT GetSystemDirectory(LPSTR lpszSysPath, UINT cbSysPath)
|
|
{
|
|
if (cbSysPath < strlen(SystemDirectory))
|
|
*lpszSysPath = 0;
|
|
else
|
|
strcpy(lpszSysPath, SystemDirectory);
|
|
|
|
dprintf_file(stddeb,"GetSystemDirectory (%s)\n",lpszSysPath);
|
|
|
|
ChopOffSlash(lpszSysPath);
|
|
return(strlen(lpszSysPath));
|
|
}
|
|
/***************************************************************************
|
|
GetTempFileName
|
|
***************************************************************************/
|
|
INT GetTempFileName(BYTE bDriveLetter, LPCSTR lpszPrefixString, UINT uUnique, LPSTR lpszTempFileName)
|
|
{
|
|
int unique;
|
|
int handle;
|
|
char tempname[256];
|
|
|
|
if (uUnique == 0)
|
|
unique = time(NULL)%99999L;
|
|
else
|
|
unique = uUnique%99999L;
|
|
|
|
strcpy(tempname,lpszPrefixString);
|
|
tempname[3]='\0';
|
|
|
|
sprintf(lpszTempFileName,"%s\\%s%d.tmp", TempDirectory, tempname,
|
|
unique);
|
|
|
|
ToDos(lpszTempFileName);
|
|
|
|
dprintf_file(stddeb,"GetTempFilename: %c %s %d => %s\n",bDriveLetter,
|
|
lpszPrefixString,uUnique,lpszTempFileName);
|
|
if ((handle = _lcreat (lpszTempFileName, 0x0000)) == -1) {
|
|
fprintf(stderr,"GetTempFilename: can't create temp file '%s' !\n", lpszTempFileName);
|
|
}
|
|
else
|
|
close(handle);
|
|
|
|
return unique;
|
|
}
|
|
|
|
/***************************************************************************
|
|
SetErrorMode
|
|
***************************************************************************/
|
|
WORD SetErrorMode(WORD x)
|
|
{
|
|
dprintf_file(stdnimp,"wine: SetErrorMode %4x (ignored)\n",x);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/***************************************************************************
|
|
_hread
|
|
***************************************************************************/
|
|
LONG _hread(INT hf, LPSTR hpvBuffer, LONG cbBuffer)
|
|
{
|
|
return read(hf, hpvBuffer, cbBuffer);
|
|
}
|
|
/***************************************************************************
|
|
_hwrite
|
|
***************************************************************************/
|
|
LONG _hwrite(INT hf, LPCSTR hpvBuffer, LONG cbBuffer)
|
|
{
|
|
return write(hf, hpvBuffer, cbBuffer);
|
|
}
|