Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

414 lines
10 KiB

/*+---------------------------------------------------------------------------
//
// mapio.c - mapped file i/o routines
//
// History:
// 9/4/97 DougP Create this header
allow and deal with input map files of zero length
11/20/97 DougP Move these routines from misc.c to here
Add option to spec codepage
//
// ©1997 Microsoft Corporation
//----------------------------------------------------------------------------*/
#include <windows.h>
#include <assert.h>
#include "NLGlib.h"
//#ifdef WINCE
void Assert(x)
{
if (x)
MessageBox(0,"assert","assert",MB_OK);
}
//#endif
BOOL WINAPI CloseMapFile(PMFILE pmf)
{
if (pmf==NULL) {
return FALSE;
}
// only unmap what existed - DougP
if (pmf->pvMap && !UnmapViewOfFile(pmf->pvMap)) {
return FALSE;
}
// ditto
if (pmf->hFileMap && !CloseHandle(pmf->hFileMap)) {
return FALSE;
}
if (!CloseHandle(pmf->hFile)) {
return FALSE;
}
NLGFreeMemory(pmf);
return TRUE;
}
PMFILE WINAPI OpenMapFileWorker(const WCHAR * pwszFileName,BOOL fDstUnicode)
{
PMFILE pmf;
const WCHAR * pwszExt;
if (!fNLGNewMemory(&pmf, sizeof(MFILE)))
{
goto Error;
}
pmf->fDstUnicode = fDstUnicode;
#ifdef WINCE
pmf->hFile = CreateFileForMapping(
#else
pmf->hFile = CMN_CreateFileW(
#endif
pwszFileName, GENERIC_READ, FILE_SHARE_READ,
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (pmf->hFile == INVALID_HANDLE_VALUE)
{
goto Error;
}
pmf->cbSize1 = GetFileSize(pmf->hFile, &pmf->cbSize2);
if (pmf->cbSize1 == 0xFFFFFFFF)
{
CMN_OutputSystemErrW(L"Can't get size for", pwszFileName);
CloseHandle(pmf->hFile);
goto Error;
}
else if (pmf->cbSize1 == 0)
{
// can't map a zero length file so mark this appropriately
pmf->hFileMap = 0;
pmf->pvMap = 0;
pmf->fSrcUnicode = TRUE;
}
else
{
#ifdef WINCE
pmf->hFileMap = CreateFileMapping(pmf->hFile, NULL, PAGE_READONLY, 0, 0, NULL);
#else
pmf->hFileMap = CreateFileMappingA(pmf->hFile, NULL, PAGE_READONLY, 0, 0, NULL);
#endif
if (pmf->hFileMap == NULL)
{
CMN_OutputSystemErrW(L"Can't Map", pwszFileName);
CloseHandle(pmf->hFile);
goto Error;
}
// Map the entire file starting at the first byte
//
pmf->pvMap = MapViewOfFile(pmf->hFileMap, FILE_MAP_READ, 0, 0, 0);
if (pmf->pvMap == NULL)
{
CloseHandle(pmf->hFileMap);
CloseHandle(pmf->hFile);
goto Error;
}
// HACK: Since IsTextUnicode returns false for sorted stem files, preset
// unicode status here based on utf file extension
pwszExt = pwszFileName;
while (*pwszExt && *pwszExt != L'.' ) pwszExt++;
if (*pwszExt && !wcscmp(pwszExt, L".utf"))
{
pmf->fSrcUnicode = TRUE;
}
else if (pmf->cbSize1 >= 2 && *(WORD *)pmf->pvMap == 0xFEFF)
{
// Safe to assume that anything starting with a BOM is Unicode
pmf->fSrcUnicode = TRUE;
}
else
{
#ifdef WINCE
pmf->fSrcUnicode = TRUE;
#else
pmf->fSrcUnicode = IsTextUnicode(pmf->pvMap, pmf->cbSize1, NULL);
#endif
}
if (pmf->fSrcUnicode)
{
pmf->pwsz = (WCHAR *)pmf->pvMap;
}
else
{
pmf->psz = (PSTR)pmf->pvMap;
}
}
pmf->uCodePage = CP_ACP; // DWP - use default unless client specifies otherwise
return pmf;
Error:
if (pmf)
{
NLGFreeMemory(pmf);
}
return NULL;
}
#ifndef WINCE
PMFILE WINAPI OpenMapFileA(const char * pszFileName)
{
WCHAR * pwszFileName;
DWORD cchFileNameLen;
int iRet;
cchFileNameLen = lstrlenA(pszFileName) + 1;
if (!fNLGNewMemory(&pwszFileName, cchFileNameLen * sizeof(WCHAR)))
{
return NULL;
}
iRet = MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, pszFileName, -1,
pwszFileName, cchFileNameLen);
if (iRet ==0)
{
NLGFreeMemory(pwszFileName);
return NULL;
}
return (OpenMapFileWorker(pwszFileName, FALSE));
}
#endif
BOOL WINAPI ResetMap(PMFILE pmf)
{
if (pmf == NULL) {
return FALSE;
}
if (pmf->fSrcUnicode) {
pmf->pwsz = (WCHAR*)pmf->pvMap;
if (*pmf->pwsz == 0xFEFF) {
pmf->pwsz++;
}
} else {
pmf->psz = (CHAR*)pmf->pvMap;
}
return TRUE;
}
// Same side effect as GetMapLine (incrememnt map pointer) but without returning contents
// in buffer. This is useful in situations where the line may be longer than the max cch and
// when the buffer isn't actually needed (counting lines, etc.)
//
BOOL WINAPI NextMapLine(PMFILE pmf)
{
DWORD cbOffset;
if (!pmf || !pmf->hFileMap) // check for zero length file
return FALSE;
if (pmf->fSrcUnicode)
{
WCHAR wch;
cbOffset = (DWORD) ((PBYTE)pmf->pwsz - (PBYTE)pmf->pvMap);
// test for EOF
Assert (cbOffset <= pmf->cbSize1);
if (cbOffset == pmf->cbSize1)
return FALSE;
while (cbOffset < pmf->cbSize1)
{
cbOffset += sizeof(WCHAR);
wch = *pmf->pwsz++;
// Break out if this is the newline or the control-Z
if (wch == 0x001A || wch == L'\n')
break;
}
}
else
{
CHAR ch;
cbOffset = (DWORD) ((PBYTE)pmf->psz - (PBYTE)pmf->pvMap);
// test for EOF
Assert (cbOffset <= pmf->cbSize1);
if (cbOffset == pmf->cbSize1)
return FALSE;
while (cbOffset < pmf->cbSize1)
{
cbOffset += sizeof(CHAR);
ch = *pmf->psz++;
// Break out if this is the newline or the control-Z
if (ch == 0x1A || ch == '\n')
break;
}
}
return TRUE;
}
PVOID WINAPI GetMapLine(PVOID pv0, DWORD cbMac, PMFILE pmf)
{
PVOID pv1;
DWORD cbOffset, cbBuff;
Assert(pv0);
// Make sure that the buffer is at least as big as the caller says it is.
// (If the buffer was allocated with the debug memory allocator, this access
// should cause an exception if pv0 isn't at least cbMac bytes long.
Assert(((char *)pv0)[cbMac-1] == ((char *)pv0)[cbMac-1]);
if (!pmf || !pmf->hFileMap) // check for zero length file
return NULL;
if (pmf->fSrcUnicode != pmf->fDstUnicode)
{
if (!fNLGNewMemory(&pv1, cbMac))
return NULL;
cbBuff = cbMac;
}
else
{
pv1 = pv0;
}
if (pmf->fSrcUnicode)
{
WCHAR wch, *pwsz = pv1;
cbOffset = (DWORD) ((PBYTE)pmf->pwsz - (PBYTE)pmf->pvMap);
// test for EOF
Assert (cbOffset <= pmf->cbSize1);
if (cbOffset == pmf->cbSize1)
goto Error;
// don't want deal with odd-sized buffers
if (cbMac % sizeof(WCHAR) != 0)
cbMac -= (cbMac % sizeof(WCHAR));
// reserve space for terminating 0
//
Assert (cbMac > 0);
cbMac -= sizeof(WCHAR);
while (cbOffset < pmf->cbSize1)
{
cbOffset += sizeof(WCHAR);
wch = *pmf->pwsz++;
switch (wch)
{
case L'\r':
case L'\n': // end of line
case 0xFEFF: // Unicode BOM
break;
case 0x001A: // ctrl-Z
wch = L'\n'; // Replace it so that the last line can be read
break;
default:
*pwsz++ = wch;
cbMac -= sizeof(WCHAR);
}
// Break out if this is the newline or buffer is full
if (wch == L'\n' || cbMac <= 0)
break;
}
*pwsz = L'\0';
}
else
{
CHAR ch, *psz = pv1;
cbOffset = (DWORD) ((PBYTE)pmf->psz - (PBYTE)pmf->pvMap);
// test for EOF
Assert (cbOffset <= pmf->cbSize1);
if (cbOffset == pmf->cbSize1)
goto Error;
// reserve space for terminating 0
//
Assert (cbMac > 0);
cbMac -= sizeof(CHAR);
while (cbOffset < pmf->cbSize1)
{
cbOffset += sizeof(CHAR);
ch = *pmf->psz++;
switch (ch)
{
case '\r':
case '\n': // end of line
break;
case 0x1A: // ctrl-Z
ch = '\n'; // Replace it so that the last line can be read
break;
default:
cbMac -= sizeof(CHAR);
*psz++ = ch;
}
// Break out if this is the newline or buffer is full
if (ch == '\n' || cbMac <= 0)
break;
}
*psz = '\0';
}
if (pmf->fSrcUnicode != pmf->fDstUnicode)
{
DWORD cch = cbBuff; // our argument is a count of bytes
if (pmf->fDstUnicode)
{
// MultiByteToWideChar wants the size of the destination in wide characters
cch /= sizeof(WCHAR);
cch = MultiByteToWideChar(pmf->uCodePage, MB_PRECOMPOSED,(PSTR)pv1,-1, (WCHAR *)pv0,cch);
}
else
{
cch = WideCharToMultiByte(pmf->uCodePage, 0, (WCHAR *)pv1, -1, (PSTR)pv0, cch, NULL, NULL);
}
if (cch == 0)
{
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
{
// Ignore truncation (for consistency with no-conversion cases)
//
if (pmf->fDstUnicode)
{
((WCHAR *)pv0)[(cbBuff / sizeof(WCHAR)) - 1] = 0;
}
else
{
((CHAR *)pv0)[cbBuff - 1] = 0;
}
}
else
{
// not a truncation error
NLGFreeMemory(pv1);
return NULL;
}
}
NLGFreeMemory(pv1);
}
return(pv0);
Error:
if (pmf->fSrcUnicode != pmf->fDstUnicode)
{
NLGFreeMemory(pv1);
}
return NULL;
}