1
0
mirror of https://github.com/UzixLS/KernelEx.git synced 2025-07-18 23:11:19 +03:00
Files
KernelEx/apilibs/kexbases/Kernel32/FileApis_fix.c
2018-11-03 16:21:44 +03:00

185 lines
6.6 KiB
C
Executable File

/*
* KernelEx
* Copyright (C) 2008-2009, Xeno86, Tihiy
*
* This file is part of KernelEx source code.
*
* KernelEx is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published
* by the Free Software Foundation; version 2 of the License.
*
* KernelEx is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GNU Make; see the file COPYING. If not, write to
* the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
#include <windows.h>
/* MAKE_EXPORT CreateFileA_fix=CreateFileA */
HANDLE WINAPI CreateFileA_fix(LPCSTR lpFileName, DWORD dwDesiredAccess,
DWORD dwShareMode, LPSECURITY_ATTRIBUTES lpSecurityAttributes,
DWORD dwCreationDistribution, DWORD dwFlagsAndAttributes, HANDLE hTemplateFile)
{
/* translate NT-specific access rights to generic ones */
if (dwDesiredAccess & ~0xf0010100)
{
DWORD oldaccess = dwDesiredAccess;
dwDesiredAccess &= 0xf0010100;
if (oldaccess & FILE_READ_DATA)
dwDesiredAccess |= GENERIC_READ;
if (oldaccess & FILE_WRITE_DATA)
dwDesiredAccess |= GENERIC_WRITE;
if (oldaccess & FILE_EXECUTE)
dwDesiredAccess |= GENERIC_EXECUTE;
}
//FILE_SHARE_DELETE is not supported
dwShareMode &= ~FILE_SHARE_DELETE;
// hTemplate has to be NULL on 9x
hTemplateFile = NULL;
// special case: overlapped I/O
if (dwFlagsAndAttributes & FILE_FLAG_OVERLAPPED)
{
DWORD lasterr = GetLastError();
HANDLE h;
h = CreateFileA(lpFileName, dwDesiredAccess, dwShareMode, lpSecurityAttributes,
dwCreationDistribution, dwFlagsAndAttributes, hTemplateFile);
if (h != INVALID_HANDLE_VALUE)
return h;
SetLastError(lasterr);
dwFlagsAndAttributes &= ~FILE_FLAG_OVERLAPPED;
}
return CreateFileA(lpFileName, dwDesiredAccess, dwShareMode, lpSecurityAttributes,
dwCreationDistribution, dwFlagsAndAttributes, hTemplateFile);
}
/* MAKE_EXPORT ReadFile_fix=ReadFile */
BOOL WINAPI ReadFile_fix(HANDLE hFile, LPVOID lpBuffer, DWORD nNumberOfBytesToRead,
LPDWORD lpNumberOfBytesRead, LPOVERLAPPED lpOverlapped)
{
DWORD lasterr = GetLastError();
if (ReadFile(hFile, lpBuffer, nNumberOfBytesToRead, lpNumberOfBytesRead,
lpOverlapped))
return TRUE;
if (lpOverlapped && GetLastError() == ERROR_INVALID_PARAMETER)
{
LONG high = lpOverlapped->OffsetHigh;
SetLastError(lasterr);
if ((SetFilePointer(hFile, lpOverlapped->Offset, &high, FILE_BEGIN)
== INVALID_SET_FILE_POINTER && GetLastError() != NO_ERROR))
return FALSE;
ResetEvent(lpOverlapped->hEvent);
lpOverlapped->Internal = STATUS_PENDING;
lpOverlapped->InternalHigh = 0;
BOOL result =
ReadFile(hFile, lpBuffer, nNumberOfBytesToRead, &lpOverlapped->InternalHigh, 0);
lasterr = GetLastError();
lpOverlapped->Internal = STATUS_WAIT_0;
SetEvent(lpOverlapped->hEvent);
SetLastError(lasterr);
if (lpNumberOfBytesRead)
*lpNumberOfBytesRead = lpOverlapped->InternalHigh;
return result;
}
return FALSE;
}
/* MAKE_EXPORT WriteFile_fix=WriteFile */
BOOL WINAPI WriteFile_fix(HANDLE hFile, LPCVOID lpBuffer, DWORD nNumberOfBytesToWrite,
LPDWORD lpNumberOfBytesWritten, LPOVERLAPPED lpOverlapped)
{
DWORD lasterr = GetLastError();
if (WriteFile(hFile, lpBuffer, nNumberOfBytesToWrite,
lpNumberOfBytesWritten, lpOverlapped))
return TRUE;
if (lpOverlapped && GetLastError() == ERROR_INVALID_PARAMETER)
{
LONG high = lpOverlapped->OffsetHigh;
SetLastError(lasterr);
if ((SetFilePointer(hFile, lpOverlapped->Offset, &high, FILE_BEGIN)
== INVALID_SET_FILE_POINTER && GetLastError() != NO_ERROR))
return FALSE;
ResetEvent(lpOverlapped->hEvent);
lpOverlapped->Internal = STATUS_PENDING;
lpOverlapped->InternalHigh = 0;
BOOL result =
WriteFile(hFile, lpBuffer, nNumberOfBytesToWrite, &lpOverlapped->InternalHigh, 0);
lasterr = GetLastError();
lpOverlapped->Internal = STATUS_WAIT_0;
SetEvent(lpOverlapped->hEvent);
SetLastError(lasterr);
if (lpNumberOfBytesWritten)
*lpNumberOfBytesWritten = lpOverlapped->InternalHigh;
return result;
}
return FALSE;
}
/* MAKE_EXPORT GetTempFileNameA_fix=GetTempFileNameA */
UINT WINAPI GetTempFileNameA_fix(LPCSTR lpPathName, LPCSTR lpPrefixString, UINT uUnique, LPTSTR lpTempFileName)
{
static int g_tempprefix = 0;
char temppref[2];
if (!lpPathName)
lpPathName = "\\";
if (!lpPrefixString)
{
g_tempprefix++;
g_tempprefix %= 5;
temppref[0] = '0' + g_tempprefix;
temppref[1] = '\0';
lpPrefixString = temppref;
}
return GetTempFileNameA(lpPathName,lpPrefixString,uUnique,lpTempFileName);
}
/* MAKE_EXPORT GetDiskFreeSpaceA_fix=GetDiskFreeSpaceA */
BOOL WINAPI GetDiskFreeSpaceA_fix(LPCSTR lpRootPathName, LPDWORD lpSectorsPerCluster,
LPDWORD lpBytesPerSector, LPDWORD lpNumberOfFreeClusters,
LPDWORD lpTotalNumberOfClusters)
{
char newRootPath[4] = { "X:\\" };
if (lstrlenA(lpRootPathName) == 2 && lpRootPathName[1] == ':') //GetDiskFreeSpace fails on paths like C:
{
newRootPath[0] = lpRootPathName[0];
lpRootPathName = newRootPath;
}
BOOL ret = GetDiskFreeSpaceA(lpRootPathName,lpSectorsPerCluster,lpBytesPerSector,lpNumberOfFreeClusters,lpTotalNumberOfClusters);
if (!ret) //more suprisingly, it may fail on paths with two extensions (e.g. c:\game.exe.lnk)
{
char shortPath[MAX_PATH];
if (GetShortPathName(lpRootPathName,shortPath,MAX_PATH))
{
//GetDiskFreeSpace will still fail on short path name, so we check drive root, having path validated
newRootPath[0] = shortPath[0];
ret = GetDiskFreeSpaceA(newRootPath,lpSectorsPerCluster,lpBytesPerSector,lpNumberOfFreeClusters,lpTotalNumberOfClusters);
}
}
return ret;
}
/* MAKE_EXPORT GetDiskFreeSpaceExA_fix=GetDiskFreeSpaceExA */
BOOL WINAPI GetDiskFreeSpaceExA_fix(LPCSTR lpDirectoryName, PULARGE_INTEGER lpFreeBytesAvailable,
PULARGE_INTEGER lpTotalNumberOfBytes, PULARGE_INTEGER lpTotalNumberOfFreeBytes)
{
//GetDiskFreeSpaceEx does not fail on paths like C: but still fails on on paths with two extensions
BOOL ret = GetDiskFreeSpaceExA(lpDirectoryName,lpFreeBytesAvailable,lpTotalNumberOfBytes,lpTotalNumberOfFreeBytes);
if (!ret)
{
char shortPath[MAX_PATH];
if (GetShortPathName(lpDirectoryName,shortPath,MAX_PATH))
ret = GetDiskFreeSpaceExA(shortPath,lpFreeBytesAvailable,lpTotalNumberOfBytes,lpTotalNumberOfFreeBytes);
}
return ret;
}