1573 lines
39 KiB
C++
1573 lines
39 KiB
C++
|
|
//+============================================================================
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1994 - 1998.
|
|
//
|
|
// File: hntfsstg.cxx
|
|
//
|
|
// This file provides the NFF (NTFS Flat File) IStream implementation.
|
|
//
|
|
// History:
|
|
// 5/6/98 MikeHill
|
|
// - Use CoTaskMem rather than new/delete.
|
|
//
|
|
//+============================================================================
|
|
|
|
#include <pch.cxx>
|
|
#include <tstr.h>
|
|
#include "cli.hxx"
|
|
#include "expparam.hxx"
|
|
|
|
#ifdef _MAC_NODOC
|
|
ASSERTDATA // File-specific data for FnAssert
|
|
#endif
|
|
|
|
DECLARE_INFOLEVEL(nff)
|
|
|
|
#ifndef DEB_INFO
|
|
#define DEB_INFO DEB_USER1
|
|
#endif
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::AddRef (IUnknown)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
ULONG
|
|
CNtfsStream::AddRef()
|
|
{
|
|
LONG cRefs;
|
|
|
|
cRefs = InterlockedIncrement( &_cRefs );
|
|
|
|
nffDebug((DEB_REFCOUNT, "CNtfsStream::AddRef(this=%x) == %d\n",
|
|
this, cRefs));
|
|
return cRefs;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Release (IUnknown)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
ULONG
|
|
CNtfsStream::Release()
|
|
{
|
|
ULONG ulRet = InterlockedDecrement( &_cRefs );
|
|
|
|
if( 0 == ulRet )
|
|
{
|
|
RemoveSelfFromList();
|
|
delete this;
|
|
}
|
|
nffDebug((DEB_REFCOUNT, "CNtfsStream::Release(this=%x) == %d\n",
|
|
this, ulRet));
|
|
|
|
return( ulRet );
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::AddRef (IUnknown)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::QueryInterface(
|
|
REFIID riid,
|
|
void** ppv )
|
|
{
|
|
HRESULT sc=S_OK;
|
|
|
|
#if DBG == 1
|
|
WCHAR strIID[64];
|
|
StringFromGUID2(riid, strIID, 64);
|
|
nffDebug(( DEB_TRACE | DEB_REFCOUNT,
|
|
"CNtfsStream::QueryInterface( %ws )\n", strIID ));
|
|
#endif
|
|
|
|
NFF_VALIDATE( QueryInterface( riid, ppv ) );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
if( IsEqualIID( riid, IID_IUnknown )
|
|
||
|
|
IsEqualIID( riid, IID_IStream )
|
|
||
|
|
IsEqualIID( riid, IID_ISequentialStream ) )
|
|
{
|
|
*ppv = static_cast<IStream*>(this);
|
|
AddRef();
|
|
return( S_OK );
|
|
}
|
|
else if( IsEqualIID( riid, IID_IMappedStream ))
|
|
{
|
|
*ppv = static_cast<IMappedStream*>(&_nffMappedStream);
|
|
AddRef();
|
|
return( S_OK );
|
|
}
|
|
else if( IsEqualIID( riid, IID_ILockBytes ))
|
|
{
|
|
*ppv = static_cast<ILockBytes*>(this);
|
|
AddRef();
|
|
return( S_OK );
|
|
}
|
|
#if DBG == 1
|
|
else if( IsEqualIID( riid, IID_IStorageTest ))
|
|
{
|
|
*ppv = static_cast<IStorageTest*>(this);
|
|
AddRef();
|
|
return( S_OK );
|
|
}
|
|
#endif // #if DBG
|
|
else
|
|
{
|
|
nffDebug(( DEB_TRACE | DEB_REFCOUNT,
|
|
"CNtfsStream::QueryInterface() Failed E_NOINTERFACE\n" ));
|
|
return( E_NOINTERFACE );
|
|
}
|
|
|
|
EH_Err:
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Seek (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Seek(
|
|
LARGE_INTEGER dlibMove,
|
|
DWORD dwOrigin,
|
|
ULARGE_INTEGER *puliNewPos)
|
|
{
|
|
HRESULT sc = S_OK;
|
|
LARGE_INTEGER liFileSize;
|
|
LARGE_INTEGER liNewPos;
|
|
|
|
nffDebug(( DEB_TRACE, "CNtfsStream::Seek( %x:%08x, %d, %p );\n",
|
|
dlibMove.HighPart, dlibMove.LowPart,
|
|
dwOrigin, puliNewPos ));
|
|
|
|
NFF_VALIDATE( Seek( dlibMove, dwOrigin, puliNewPos ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
switch( dwOrigin )
|
|
{
|
|
case STREAM_SEEK_SET:
|
|
liNewPos.QuadPart = dlibMove.QuadPart;
|
|
break;
|
|
|
|
case STREAM_SEEK_CUR:
|
|
liNewPos.QuadPart = _liCurrentSeekPosition.QuadPart + dlibMove.QuadPart;
|
|
break;
|
|
|
|
case STREAM_SEEK_END:
|
|
liFileSize.LowPart = GetFileSize( _hFile,
|
|
(ULONG*)(&liFileSize.HighPart) );
|
|
|
|
if( 0xFFFFFFFF == liFileSize.LowPart && NO_ERROR != GetLastError() )
|
|
{
|
|
nffChk( HRESULT_FROM_WIN32( GetLastError() ) );
|
|
}
|
|
|
|
liNewPos.QuadPart = liFileSize.QuadPart + dlibMove.QuadPart;
|
|
break;
|
|
|
|
default:
|
|
nffChk(STG_E_INVALIDPARAMETER);
|
|
break;
|
|
}
|
|
|
|
// Compatibility with Docfile. Seeking < 0 fails.
|
|
if( liNewPos.QuadPart < 0 )
|
|
nffErr( EH_Err, STG_E_INVALIDFUNCTION );
|
|
|
|
_liCurrentSeekPosition = liNewPos;
|
|
|
|
|
|
// If desired, give the caller the now-current seek position.
|
|
if( NULL != puliNewPos )
|
|
*puliNewPos = _liCurrentSeekPosition;
|
|
|
|
EH_Err:
|
|
Unlock();
|
|
return( sc );
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::SetSize (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::SetSize(
|
|
ULARGE_INTEGER uliNewSize)
|
|
{
|
|
HRESULT sc = S_OK;
|
|
CLargeInteger liEOF;
|
|
|
|
if ( uliNewSize.HighPart != 0 )
|
|
nffErr(EH_Err, STG_E_INVALIDFUNCTION);
|
|
|
|
nffDebug(( DEB_ITRACE | DEB_INFO | DEB_WRITE,
|
|
"CNtfsStream::SetSize(%x:%x) hdl=%x, stream='%ws'\n",
|
|
uliNewSize.QuadPart,
|
|
_hFile,
|
|
_pwcsName ));
|
|
|
|
NFF_VALIDATE( SetSize( uliNewSize ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
// If this stream is mapped, set the size accordingly
|
|
|
|
if( _nffMappedStream.IsMapped() )
|
|
{
|
|
_nffMappedStream.SetSize( uliNewSize.LowPart, TRUE, NULL, &sc );
|
|
}
|
|
else
|
|
{
|
|
sc = SetFileSize( CULargeInteger(uliNewSize) );
|
|
}
|
|
|
|
if( !FAILED(sc) )
|
|
sc = S_OK;
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc);
|
|
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::CopyTo (IStream)
|
|
//
|
|
// There's no way of knowing what the IStream is to which we're copying, so
|
|
// we have to assume that we might be copying to ourself. And given that
|
|
// assumption, we have to deal with the case that this is an overlapping
|
|
// copy (e.g., "copy 10 bytes from offset 0 to offset 5").
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::CopyTo(
|
|
IStream *pstm,
|
|
ULARGE_INTEGER cb,
|
|
ULARGE_INTEGER *pcbRead,
|
|
ULARGE_INTEGER *pcbWritten)
|
|
{
|
|
nffXTrace( "CNtfsStream::CopyTo" );
|
|
|
|
HRESULT sc = S_OK;
|
|
PVOID pv = NULL;
|
|
ULONG cbRead = 0, cbWritten = 0;
|
|
CULargeInteger cbReadTotal = 0, cbWrittenTotal = 0;
|
|
CLargeInteger liZero = 0;
|
|
CULargeInteger uliOriginalSourcePosition, uliOriginalDestPosition;
|
|
CULargeInteger cbSourceSize, cbDestSize;
|
|
ULONG cbPerCopy = 0;
|
|
STATSTG statstg;
|
|
CULargeInteger cbRequested = cb;
|
|
BOOL fCopyForward;
|
|
|
|
NFF_VALIDATE( CopyTo( pstm, cb, pcbRead, pcbWritten ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
if( NULL == pstm)
|
|
nffErr( EH_Err, STG_E_INVALIDPARAMETER );
|
|
|
|
// Determine how much we'll copy at a time.
|
|
// As of this writing, STREAMBUFFERSIZE is 8192 bytes
|
|
|
|
if( cbRequested > STREAMBUFFERSIZE )
|
|
cbPerCopy = STREAMBUFFERSIZE;
|
|
else
|
|
cbPerCopy = cbRequested.LowPart;
|
|
|
|
// ------------------------------------------------------------------
|
|
// Get the current stream sizes/positions, and adjust the destination
|
|
// size if necessary
|
|
// ------------------------------------------------------------------
|
|
|
|
nffChk( this->Seek( liZero, STREAM_SEEK_CUR, &uliOriginalSourcePosition ) );
|
|
|
|
nffChk( pstm->Seek( liZero, STREAM_SEEK_CUR, &uliOriginalDestPosition ) );
|
|
|
|
nffChk( this->Stat( &statstg, STATFLAG_NONAME ) );
|
|
|
|
cbSourceSize = statstg.cbSize;
|
|
|
|
nffChk( pstm->Stat( &statstg, STATFLAG_NONAME ) );
|
|
|
|
cbDestSize = statstg.cbSize;
|
|
|
|
// Ensure the sizes are valid (we can't handle anything with the high bit
|
|
// set, because Seek takes a signed offset).
|
|
|
|
if( static_cast<CLargeInteger>(cbSourceSize) < 0
|
|
||
|
|
static_cast<CLargeInteger>(cbDestSize) < 0 )
|
|
{
|
|
nffErr( EH_Err, STG_E_INVALIDHEADER );
|
|
}
|
|
|
|
// Don't copy more than the source stream has available
|
|
if( cbRequested > cbSourceSize - uliOriginalSourcePosition )
|
|
cbRequested = cbSourceSize - uliOriginalSourcePosition;
|
|
|
|
// If necessary, grow the destination stream.
|
|
|
|
if( cbSourceSize - uliOriginalSourcePosition > cbDestSize - uliOriginalDestPosition )
|
|
{
|
|
cbDestSize = cbSourceSize - uliOriginalSourcePosition + uliOriginalDestPosition;
|
|
nffChk( pstm->SetSize( cbDestSize ) );
|
|
}
|
|
|
|
// ----------------------
|
|
// Allocate a copy buffer
|
|
// ----------------------
|
|
|
|
nffMem( pv = CoTaskMemAlloc( cbPerCopy ) );
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Determine if we're copying forwards (high seek position to low) or backwards.
|
|
// -----------------------------------------------------------------------------
|
|
|
|
fCopyForward = TRUE;
|
|
if( uliOriginalSourcePosition < uliOriginalDestPosition )
|
|
{
|
|
// E.g., say we're copying 15 bytes from offset 0 to offset 5,
|
|
// and we're only able to copy 10 bytes at a time.
|
|
// If we copy bytes 0-9 to offset 5, we'll end up overwriting
|
|
// bytes 10-14, and be unable to complete the copy.
|
|
// So instead, we'll copy bytes 5-14 to offset 10, and finish
|
|
// up by copying bytes 0-4 to offset 5.
|
|
|
|
fCopyForward = FALSE;
|
|
|
|
// To do this kind of backwards copy, we need to start by seeking
|
|
// towards the end of the stream.
|
|
|
|
CULargeInteger uliNewSourcePosition, uliNewDestPosition;
|
|
|
|
uliNewSourcePosition = cbSourceSize - cbPerCopy;
|
|
nffChk( this->Seek( uliNewSourcePosition, STREAM_SEEK_SET, NULL ) );
|
|
|
|
uliNewDestPosition = cbDestSize - cbPerCopy;
|
|
nffChk( pstm->Seek( uliNewDestPosition, STREAM_SEEK_SET, NULL ) );
|
|
|
|
}
|
|
|
|
// --------------
|
|
// Copy in chunks
|
|
// --------------
|
|
|
|
cbPerCopy = cbRequested > cbPerCopy ? cbPerCopy : cbRequested.LowPart;
|
|
while( cbRequested > 0 )
|
|
{
|
|
// Read from the source
|
|
nffChk( this->Read( pv, cbPerCopy, &cbRead ) );
|
|
|
|
if( cbRead != cbPerCopy )
|
|
nffErr(EH_Err, STG_E_READFAULT);
|
|
|
|
cbReadTotal += cbRead;
|
|
|
|
// Write to the dest
|
|
nffChk( pstm->Write( pv, cbPerCopy, &cbWritten ) );
|
|
|
|
if( cbWritten != cbPerCopy )
|
|
nffErr( EH_Err, STG_E_WRITEFAULT );
|
|
|
|
cbWrittenTotal += cbWritten;
|
|
|
|
// Adjust the amount remaining to be copied
|
|
cbRequested -= cbPerCopy;
|
|
|
|
|
|
// Determine how much to copy in the next iteration (this will
|
|
// always be cbPerCopy until the last iteration). If copying
|
|
// backwards, we need to manually adjust the seek pointer.
|
|
|
|
cbPerCopy = (cbRequested > cbPerCopy) ? cbPerCopy : cbRequested.LowPart;
|
|
if( !fCopyForward )
|
|
{
|
|
nffChk( this->Seek( -static_cast<CLargeInteger>(cbPerCopy),
|
|
STREAM_SEEK_CUR, NULL ) );
|
|
|
|
nffChk( pstm->Seek( -static_cast<CLargeInteger>(cbPerCopy),
|
|
STREAM_SEEK_CUR, NULL ) );
|
|
}
|
|
|
|
}
|
|
|
|
// If we were backward-copying, adjust the seek pointers
|
|
// as if we had forward-copied
|
|
|
|
if( !fCopyForward )
|
|
{
|
|
uliOriginalSourcePosition += cbReadTotal;
|
|
nffChk( this->Seek( uliOriginalSourcePosition, STREAM_SEEK_SET, NULL ) );
|
|
|
|
uliOriginalDestPosition += cbWrittenTotal;
|
|
nffChk( pstm->Seek( uliOriginalDestPosition, STREAM_SEEK_SET, NULL ) );
|
|
}
|
|
|
|
// ----
|
|
// Exit
|
|
// ----
|
|
|
|
if( NULL != pcbRead )
|
|
*pcbRead = cbReadTotal;
|
|
if( NULL != pcbWritten )
|
|
*pcbWritten = cbWrittenTotal;
|
|
|
|
EH_Err:
|
|
|
|
if( NULL != pv )
|
|
CoTaskMemFree(pv);
|
|
|
|
Unlock();
|
|
return(sc);
|
|
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Commit (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Commit( DWORD grfCommitFlags )
|
|
{
|
|
nffXTrace( "CNtfsStream::Commit" );
|
|
HRESULT sc = S_OK;
|
|
|
|
NFF_VALIDATE( Commit( grfCommitFlags ) );
|
|
if(~(STGC_DANGEROUSLYCOMMITMERELYTODISKCACHE|STGC_DEFAULT) & grfCommitFlags)
|
|
nffErr( EH_Err, STG_E_INVALIDFLAG );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChkTo ( EH_Unlock, CheckReverted() );
|
|
|
|
if( !(STGC_DANGEROUSLYCOMMITMERELYTODISKCACHE & grfCommitFlags) )
|
|
{
|
|
if( !FlushFileBuffers( _hFile ))
|
|
sc = LAST_SCODE;
|
|
}
|
|
|
|
EH_Unlock:
|
|
Unlock();
|
|
EH_Err:
|
|
return sc;
|
|
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Revert (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Revert(void)
|
|
{
|
|
nffXTrace( "CNtfsStream::Revert" );
|
|
// We only support direct-mode.
|
|
|
|
return CheckReverted();
|
|
}
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::LockRegion (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::LockRegion(
|
|
ULARGE_INTEGER libOffset,
|
|
ULARGE_INTEGER cb,
|
|
DWORD dwLockType)
|
|
{
|
|
nffXTrace( "CNtfsStream::LockRegion" );
|
|
HRESULT sc = S_OK;
|
|
|
|
NFF_VALIDATE( LockRegion( libOffset, cb, dwLockType ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
if (dwLockType != LOCK_EXCLUSIVE && dwLockType != LOCK_ONLYONCE)
|
|
nffErr( EH_Err, STG_E_INVALIDFUNCTION );
|
|
|
|
|
|
if( !LockFile( _hFile, libOffset.LowPart, libOffset.HighPart,
|
|
cb.LowPart, cb.HighPart))
|
|
{
|
|
nffErr( EH_Err, LAST_SCODE );
|
|
}
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::UnlockRegion (ILockBytes)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::UnlockRegion(
|
|
ULARGE_INTEGER libOffset,
|
|
ULARGE_INTEGER cb,
|
|
DWORD dwLockType)
|
|
{
|
|
nffXTrace( "CNtfsStream::UnlockRegion" );
|
|
HRESULT sc = S_OK;
|
|
|
|
NFF_VALIDATE( UnlockRegion( libOffset, cb, dwLockType ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
if (dwLockType != LOCK_EXCLUSIVE && dwLockType != LOCK_ONLYONCE)
|
|
{
|
|
nffErr( EH_Err, STG_E_INVALIDFUNCTION );
|
|
}
|
|
|
|
if( !UnlockFile(_hFile, libOffset.LowPart, libOffset.HighPart,
|
|
cb.LowPart, cb.HighPart))
|
|
{
|
|
nffErr( EH_Err, LAST_SCODE );
|
|
}
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Stat (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Stat(
|
|
STATSTG *pstatstg,
|
|
DWORD grfStatFlag)
|
|
{
|
|
nffXTrace( "CNtfsStream::Stat" );
|
|
STATSTG statstg;
|
|
HRESULT sc = S_OK;
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
FILE_ACCESS_INFORMATION file_access_information;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
BY_HANDLE_FILE_INFORMATION ByHandleFileInformation;
|
|
|
|
NFF_VALIDATE( Stat( pstatstg, grfStatFlag ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChkTo ( EH_Lock, CheckReverted() );
|
|
|
|
ZeroMemory((void*)&statstg, sizeof(STATSTG));
|
|
|
|
// Get the name, if desired
|
|
|
|
if( (STATFLAG_NONAME & grfStatFlag) )
|
|
statstg.pwcsName = NULL;
|
|
else
|
|
{
|
|
nffAssert( NULL != _pwcsName );
|
|
|
|
nffMem( statstg.pwcsName = reinterpret_cast<WCHAR*>
|
|
( CoTaskMemAlloc( sizeof(WCHAR)*(wcslen(_pwcsName) + 1) )));
|
|
wcscpy( statstg.pwcsName, _pwcsName );
|
|
}
|
|
|
|
// Get the type
|
|
statstg.type = STGTY_STREAM;
|
|
|
|
statstg.grfLocksSupported = LOCK_EXCLUSIVE | LOCK_ONLYONCE;
|
|
|
|
// Get the size & times.
|
|
|
|
if( !GetFileInformationByHandle( _hFile, &ByHandleFileInformation ))
|
|
nffErr( EH_Err, LAST_SCODE );
|
|
|
|
statstg.cbSize.LowPart = ByHandleFileInformation.nFileSizeLow;
|
|
statstg.cbSize.HighPart = ByHandleFileInformation.nFileSizeHigh;
|
|
|
|
// We get a time back in ByHandleFileInformation, but it's the file's times,
|
|
// not the streams times. So really the stream times are not supported, and
|
|
// we'll just set them to zero.
|
|
|
|
statstg.mtime = statstg.atime = statstg.ctime = CFILETIME(0);
|
|
|
|
// Get the STGM modes
|
|
statstg.grfMode = _grfMode & ~STGM_CREATE;
|
|
|
|
*pstatstg = statstg;
|
|
|
|
EH_Err:
|
|
if( FAILED(sc) && NULL != statstg.pwcsName )
|
|
CoTaskMemFree( statstg.pwcsName );
|
|
|
|
EH_Lock:
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Clone (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Clone(
|
|
IStream** ppstm)
|
|
{
|
|
nffXTrace( "CNtfsStream::Clone" );
|
|
return( E_NOTIMPL );
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Read (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Read(
|
|
void* pv,
|
|
ULONG cb,
|
|
ULONG* pcbRead)
|
|
{
|
|
nffXTrace( "CNtfsStream::Read" );
|
|
HRESULT sc = S_OK;
|
|
ULONG cbRead = 0;
|
|
|
|
nffDebug(( DEB_ITRACE, "Read( pv=0x%x, cb=0x%x );\n", pv, cb ));
|
|
|
|
NFF_VALIDATE( Read( pv, cb, pcbRead ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
nffChk( ReadAt( _liCurrentSeekPosition, pv, cb, &cbRead ) );
|
|
|
|
_liCurrentSeekPosition += cbRead;
|
|
nffDebug(( DEB_ITRACE, "Read() read %x bytes.\n", cbRead ));
|
|
if( NULL != pcbRead )
|
|
*pcbRead = cbRead;
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::ReadAt (ILockBytes)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::ReadAt(
|
|
ULARGE_INTEGER ulOffset,
|
|
void* pv,
|
|
ULONG cb,
|
|
ULONG* pcbRead)
|
|
{
|
|
nffXTrace( "CNtfsStream::ReadAt" );
|
|
HRESULT sc = S_OK;
|
|
|
|
nffDebug(( DEB_ITRACE, "ReadAt( off=%x:%x, pv=0x%x, cb=0x%x );\n",
|
|
ulOffset, pv, cb ));
|
|
|
|
NFF_VALIDATE( ReadAt( ulOffset, pv, cb, pcbRead ) );
|
|
|
|
if( static_cast<LONG>(ulOffset.HighPart) < 0 )
|
|
return TYPE_E_SIZETOOBIG;
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
// Is this stream mapped?
|
|
if( _nffMappedStream.IsMapped() )
|
|
{
|
|
// This stream is mapped. We'll read directly from the mapping buffer.
|
|
_nffMappedStream.Read( pv, _liCurrentSeekPosition.LowPart, &cb );
|
|
|
|
if( NULL != pcbRead )
|
|
*pcbRead = cb;
|
|
}
|
|
else
|
|
{
|
|
// No, just read from the file.
|
|
|
|
nffChk( SyncReadAtFile( ulOffset, pv, cb, pcbRead ) );
|
|
}
|
|
|
|
nffDebug(( DEB_ITRACE, "ReadAt() read %x bytes.\n", *pcbRead ));
|
|
|
|
|
|
EH_Err:
|
|
Unlock();
|
|
return( sc );
|
|
}
|
|
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Write (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Write(
|
|
const void* pv,
|
|
ULONG cb,
|
|
ULONG* pcbWritten)
|
|
{
|
|
nffXTrace( "CNtfsStream::Write" );
|
|
HRESULT sc = S_OK;
|
|
ULONG cbWritten = 0;
|
|
|
|
nffDebug(( DEB_ITRACE, "Write( pv=0x%x, cb=0x%x );\n", pv, cb ));
|
|
|
|
NFF_VALIDATE( Write( pv, cb, pcbWritten ) );
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
nffChk(WriteAt( _liCurrentSeekPosition, pv, cb, &cbWritten ));
|
|
|
|
_liCurrentSeekPosition += cbWritten;
|
|
|
|
nffDebug(( DEB_ITRACE, "Write() wrote %x bytes.\n", cbWritten ));
|
|
|
|
if( NULL != pcbWritten )
|
|
*pcbWritten = cbWritten;
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::WriteAt (ILockBytes)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::WriteAt(
|
|
ULARGE_INTEGER ulOffset,
|
|
const void* pv,
|
|
ULONG cb,
|
|
ULONG* pcbWritten)
|
|
{
|
|
nffXTrace( "CNtfsStream::WriteAt" );
|
|
HRESULT sc = S_OK;
|
|
|
|
nffDebug(( DEB_ITRACE, "WriteAt( off=%x:%x, pv=0x%x, cb=0x%x );\n",
|
|
ulOffset, pv, cb ));
|
|
|
|
NFF_VALIDATE( WriteAt( ulOffset, pv, cb, pcbWritten ) );
|
|
|
|
if( ((LONG)(ulOffset.HighPart)) < 0 )
|
|
return( TYPE_E_SIZETOOBIG );
|
|
|
|
|
|
Lock( INFINITE );
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
// Is this stream mapped?
|
|
if( _nffMappedStream.IsMapped() )
|
|
{
|
|
// This stream is mapped, we'll take the Write to the mapping.
|
|
|
|
ULONG iPosition = _nffMappedStream.SizeOfMapping() - _liCurrentSeekPosition.LowPart;
|
|
|
|
if( cb > iPosition )
|
|
{
|
|
_nffMappedStream.SetSize( iPosition + cb, TRUE, NULL, &sc );
|
|
nffChk(sc);
|
|
}
|
|
|
|
_nffMappedStream.Write( pv, _liCurrentSeekPosition.LowPart, &cb );
|
|
|
|
if( NULL != pcbWritten )
|
|
*pcbWritten = cb;
|
|
}
|
|
else
|
|
{
|
|
// No, just write to the file.
|
|
|
|
nffChk( SyncWriteAtFile( ulOffset, pv, cb, pcbWritten ) );
|
|
}
|
|
|
|
nffDebug(( DEB_ITRACE, "WriteAt() wrote %x bytes.\n", *pcbWritten ));
|
|
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Method: CNtfsStream::Flush (IStream)
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Flush()
|
|
{
|
|
HRESULT sc = S_OK;
|
|
|
|
Lock( INFINITE );
|
|
nffChk( CheckReverted() );
|
|
|
|
_nffMappedStream.Flush(&sc);
|
|
|
|
EH_Err:
|
|
Unlock();
|
|
return(sc);
|
|
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------
|
|
//
|
|
// Member: CNtfsStream Constructor
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
CNtfsStream::CNtfsStream( CNtfsStorage *pnffstg,
|
|
IBlockingLock *pBlockingLock )
|
|
#pragma warning(disable: 4355)
|
|
: _nffMappedStream( this ),
|
|
#pragma warning(default: 4355)
|
|
_pnffstg( pnffstg )
|
|
{
|
|
nffXTrace( "CNtfsStream::CNtfsStream" );
|
|
nffDebug(( DEB_REFCOUNT, "new CNtfsStream Constructed with cRefs=1\n" ));
|
|
|
|
_sig = NTFSSTREAM_SIG;
|
|
_cRefs = 1;
|
|
_grfMode = 0;
|
|
_hFile = INVALID_HANDLE_VALUE;
|
|
_liCurrentSeekPosition = 0;
|
|
_pnffstmNext = NULL;
|
|
_pnffstmPrev = NULL;
|
|
_pwcsName = NULL;
|
|
|
|
nffAssert( NULL != pBlockingLock );
|
|
_pBlockingLock = pBlockingLock;
|
|
_pBlockingLock->AddRef();
|
|
|
|
_ovlp.Internal = _ovlp.InternalHigh = 0;
|
|
_ovlp.Offset = _ovlp.OffsetHigh = 0;
|
|
_ovlp.hEvent = NULL;
|
|
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------
|
|
//
|
|
// Member: CNtfsStream Destructor
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
CNtfsStream::~CNtfsStream()
|
|
{
|
|
|
|
nffXTrace( "CNtfsStream::~CNtfsStream" );
|
|
nffDebug(( DEB_INFO, "CNtfsStream: Close 0x%x.\n", _hFile ));
|
|
|
|
// Shut down the mapped stream
|
|
_nffMappedStream.ShutDown();
|
|
|
|
// Close the file
|
|
if( INVALID_HANDLE_VALUE != _hFile )
|
|
NtClose( _hFile );
|
|
|
|
if( NULL != _ovlp.hEvent )
|
|
CloseHandle( _ovlp.hEvent );
|
|
|
|
if( NULL != _pwcsName )
|
|
CoTaskMemFree( _pwcsName );
|
|
|
|
// Release the object that provides access to the tree lock.
|
|
nffAssert( NULL != _pBlockingLock );
|
|
_pBlockingLock->Release();
|
|
|
|
_sig = NTFSSTREAM_SIGDEL;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------
|
|
//
|
|
// Member: CNtfsStream::Init
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Init(
|
|
HANDLE hFile, // File handle of this NTFS Stream.
|
|
DWORD grfMode, // Open Modes
|
|
const OLECHAR * pwcsName, // Name of the Stream
|
|
CNtfsStream *pnffstm) // Previously Open NTFS Stream (list)
|
|
{
|
|
HRESULT sc=S_OK;
|
|
HANDLE ev;
|
|
|
|
nffITrace( "CNtfsStream::Init" );
|
|
|
|
// We now own this file handle, and are responsible for closing it.
|
|
_hFile = hFile;
|
|
|
|
// Save the STGM_ flags so we can return them in a Stat call.
|
|
_grfMode = grfMode;
|
|
|
|
// Save the stream name
|
|
|
|
if( NULL != _pwcsName )
|
|
{
|
|
CoTaskMemFree( _pwcsName );
|
|
_pwcsName = NULL;
|
|
}
|
|
|
|
if( NULL != pwcsName )
|
|
{
|
|
nffMem( _pwcsName = reinterpret_cast<WCHAR*>
|
|
( CoTaskMemAlloc( sizeof(WCHAR)*(wcslen(pwcsName) + 1) )));
|
|
wcscpy( _pwcsName, pwcsName );
|
|
}
|
|
|
|
// All the streams live on a list held by the root Storage.
|
|
if(NULL != pnffstm)
|
|
InsertSelfIntoList(pnffstm);
|
|
|
|
if( NULL == _ovlp.hEvent )
|
|
{
|
|
ev = CreateEvent(NULL, // Security Attributes.
|
|
TRUE, // Manual Reset, Flag.
|
|
FALSE, // Initial State = Signaled, Flag.
|
|
NULL); // Object Name.
|
|
|
|
if( NULL == ev)
|
|
nffChk( LAST_SCODE );
|
|
|
|
_ovlp.hEvent = ev;
|
|
}
|
|
|
|
nffChk( _nffMappedStream.Init( _hFile ));
|
|
|
|
EH_Err:
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::ShutDown
|
|
//
|
|
// Flush data, Close File handle and mark the object as reverted.
|
|
// This is called when the Storage is released and when the Oplock Breaks.
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::ShutDown()
|
|
{
|
|
nffITrace( "CNtfsStream::ShutDown" );
|
|
HRESULT sc=S_OK;
|
|
|
|
if( INVALID_HANDLE_VALUE == _hFile )
|
|
return S_OK;
|
|
|
|
nffDebug(( DEB_INFO, "CNtfsStream::ShutDown(%x)\n", _hFile ));
|
|
|
|
//
|
|
// Shut down the mapped stream
|
|
//
|
|
_nffMappedStream.ShutDown();
|
|
|
|
//
|
|
// Close the file/stream handle and mark the IStream object as
|
|
// Reverted by giving the file handle an invalid value.
|
|
//
|
|
CloseHandle(_hFile);
|
|
_hFile = INVALID_HANDLE_VALUE;
|
|
|
|
// We don't need the parent CNtfsStorage any longer, and more importantly it could
|
|
// be going away.
|
|
_pnffstg = NULL;
|
|
|
|
//
|
|
// Remove IStream object from the Storage's list
|
|
//
|
|
RemoveSelfFromList();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::Rename
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Rename(
|
|
const WCHAR *pwcsName,
|
|
BOOL fOverWrite )
|
|
{
|
|
IO_STATUS_BLOCK io_status_block;
|
|
PFILE_RENAME_INFORMATION pFileRenInfo=NULL; // _alloca()'ed
|
|
NTSTATUS status;
|
|
|
|
CNtfsStreamName nsnName = pwcsName; // Convert to ":name:$DATA"
|
|
|
|
LONG cbBufSize=0;
|
|
HRESULT sc=S_OK;
|
|
|
|
|
|
nffDebug(( DEB_INFO | DEB_WRITE, "CNtfsStream::Rename(%ws -> %ws)\n",
|
|
_pwcsName, pwcsName ));
|
|
|
|
Lock( INFINITE );
|
|
nffChk( CheckReverted() );
|
|
|
|
// Size and allocate a FILE_RENAME_INFOMATION buffer. The size argument
|
|
// to NtSetInformationFile must be correct, so subtract 1 WCHAR for the
|
|
// FileName[1] in the struct we are not using.
|
|
//
|
|
cbBufSize = sizeof(FILE_RENAME_INFORMATION) - sizeof(WCHAR);
|
|
cbBufSize += nsnName.Count() * sizeof(WCHAR);
|
|
pFileRenInfo = (PFILE_RENAME_INFORMATION) _alloca( cbBufSize );
|
|
|
|
// Load the FILE_RENAME_INFORMATION structure
|
|
|
|
pFileRenInfo->ReplaceIfExists = (BOOLEAN) fOverWrite;
|
|
pFileRenInfo->RootDirectory = NULL;
|
|
pFileRenInfo->FileNameLength = nsnName.Count() * sizeof(WCHAR);
|
|
wcscpy( pFileRenInfo->FileName, (const WCHAR*)nsnName );
|
|
|
|
// Rename the stream
|
|
|
|
status = NtSetInformationFile( _hFile,
|
|
&io_status_block,
|
|
pFileRenInfo,
|
|
cbBufSize,
|
|
FileRenameInformation );
|
|
if( !NT_SUCCESS(status) )
|
|
{
|
|
nffChk( sc = NtStatusToScode(status) );
|
|
}
|
|
|
|
// Discard the old name and Save the new name
|
|
|
|
CoTaskMemFree( _pwcsName );
|
|
|
|
// reuse cbBufSize
|
|
cbBufSize = sizeof(WCHAR) * (wcslen(pwcsName)+1);
|
|
|
|
nffMem( _pwcsName = (WCHAR*) CoTaskMemAlloc( cbBufSize ));
|
|
wcscpy( _pwcsName, pwcsName );
|
|
|
|
sc = S_OK;
|
|
|
|
EH_Err:
|
|
|
|
Unlock();
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::Delete
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::Delete()
|
|
{
|
|
HRESULT sc=S_OK;
|
|
|
|
nffDebug(( DEB_INFO | DEB_WRITE, "Delete(stm=%ws)\n", _pwcsName ));
|
|
|
|
nffChk( CheckReverted() );
|
|
|
|
if( IsWriteable() )
|
|
nffChk( DeleteStream( &_hFile ));
|
|
|
|
EH_Err:
|
|
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::DeleteStream
|
|
//
|
|
// This method is static so that it can be called globally.
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT // static
|
|
CNtfsStream::DeleteStream( HANDLE *phStream )
|
|
{
|
|
HRESULT sc=S_OK;
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
FILE_DISPOSITION_INFORMATION Disposition;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
// Execute the following statement:
|
|
// Disposition.DeleteFile = TRUE;
|
|
// We can't actually write the code that way, because "DeleteFile" is #defined to
|
|
// "DeleteFileW".
|
|
|
|
nffDebug(( DEB_INFO | DEB_WRITE, "DeleteStream(hdl=%x)\n", *phStream ));
|
|
|
|
*reinterpret_cast<BOOLEAN*>(&Disposition) = TRUE;
|
|
DfpAssert( sizeof(Disposition) == sizeof(BOOLEAN) );
|
|
|
|
// Mark the file for delete on close
|
|
// Note that if this is the Contents stream we can delete it successfully,
|
|
// but NTFS actually just truncates it to zero length.
|
|
|
|
status = NtSetInformationFile(
|
|
*phStream,
|
|
&IoStatusBlock,
|
|
&Disposition,
|
|
sizeof(Disposition),
|
|
FileDispositionInformation
|
|
);
|
|
if( !NT_SUCCESS(status) )
|
|
{
|
|
nffErr( EH_Err, NtStatusToScode(status) );
|
|
}
|
|
|
|
NtClose( *phStream );
|
|
*phStream = INVALID_HANDLE_VALUE;
|
|
|
|
EH_Err:
|
|
|
|
return( sc );
|
|
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream::SetFileSize (private, non-interface method)
|
|
//
|
|
// Set the size of the _hFile. This is used by the IStream & IMappedStream
|
|
// SetSize methods
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT // private
|
|
CNtfsStream::SetFileSize( const CULargeInteger &uliNewSize )
|
|
{
|
|
nffITrace( "CNtfsStream::SetFileSize" );
|
|
HRESULT sc = S_OK;
|
|
CLargeInteger liEOF;
|
|
|
|
// We have to convert uliNewSize into a LARGE_INTEGER, so ensure that it can
|
|
// be cast without loss of data.
|
|
|
|
liEOF = static_cast<CLargeInteger>(uliNewSize);
|
|
if( liEOF < 0 )
|
|
nffErr( EH_Err, STG_E_INVALIDPARAMETER );
|
|
|
|
// Move to what will be the new end-of-file position.
|
|
|
|
liEOF.LowPart = SetFilePointer( _hFile, liEOF.LowPart,
|
|
&liEOF.HighPart, FILE_BEGIN );
|
|
if( 0xFFFFFFFF == liEOF.LowPart && NO_ERROR != GetLastError() )
|
|
nffErr( EH_Err, LAST_SCODE );
|
|
|
|
// Set this as the new eof
|
|
|
|
if( !SetEndOfFile( _hFile ))
|
|
nffErr( EH_Err, LAST_SCODE );
|
|
|
|
EH_Err:
|
|
|
|
return( sc );
|
|
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------
|
|
//
|
|
// Member: CNtfsStream::InsertSelfIntoList/RemoveSelfFromList
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
// We are passed a "current" element of a doubly linked list.
|
|
// Insert "this" right after the given List element.
|
|
//
|
|
VOID
|
|
CNtfsStream::InsertSelfIntoList(CNtfsStream *pnffstmCurrent)
|
|
{
|
|
nffDebug(( DEB_ITRACE, "CNtfsStream::InsertSelfIntoList this=%x\n", this ));
|
|
|
|
// If we're already in the list, or there is no list, then we're done.
|
|
if( NULL != _pnffstmNext || NULL == pnffstmCurrent )
|
|
return;
|
|
|
|
// "this" point back to the current element
|
|
// and points forward to current's next element
|
|
_pnffstmPrev = pnffstmCurrent;
|
|
_pnffstmNext = pnffstmCurrent->_pnffstmNext;
|
|
|
|
// the current element now points forward to "this"
|
|
// and if the next is not NULL it points back at this.
|
|
pnffstmCurrent->_pnffstmNext = this;
|
|
if(NULL != _pnffstmNext)
|
|
_pnffstmNext->_pnffstmPrev = this;
|
|
}
|
|
|
|
VOID
|
|
CNtfsStream::RemoveSelfFromList()
|
|
{
|
|
nffDebug(( DEB_ITRACE, "CNtfsStream::RemoveSelfFromList this=%x\n", this ));
|
|
// My next element's previous pointer is given my previous pointer.
|
|
if(NULL != _pnffstmNext)
|
|
_pnffstmNext->_pnffstmPrev = _pnffstmPrev;
|
|
|
|
// My previous element's next pointer is given my next pointer.
|
|
if(NULL != _pnffstmPrev)
|
|
_pnffstmPrev->_pnffstmNext = _pnffstmNext;
|
|
|
|
_pnffstmNext = NULL;
|
|
_pnffstmPrev = NULL;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::SyncReadAtFile
|
|
//
|
|
// Synopsis: Provide synchronous IO for a file handle open in
|
|
// asynchronous mode.
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::SyncReadAtFile(
|
|
ULARGE_INTEGER ulOffset,
|
|
PVOID pv,
|
|
ULONG cb,
|
|
PULONG pcbRead)
|
|
{
|
|
HRESULT sc=S_OK;
|
|
LONG err=ERROR_SUCCESS;
|
|
//
|
|
// We use the single OVERLAPPED structure in the object.
|
|
// This saves us from creating an Event everytime. We
|
|
// require the TreeMutex will keep us single threaded.
|
|
//
|
|
_ovlp.Offset = ulOffset.LowPart;
|
|
_ovlp.OffsetHigh = ulOffset.HighPart;
|
|
|
|
nffDebug(( DEB_ITRACE | DEB_INFO | DEB_READ,
|
|
"SyncReadAtFile(_hFile=0x%x off=%x:%x, pv=0x%x, cb=0x%x )"
|
|
" stream='%ws'\n",
|
|
_hFile, ulOffset.HighPart, ulOffset.LowPart,
|
|
pv, cb, _pwcsName ));
|
|
|
|
if( !ReadFile( _hFile, pv, cb, pcbRead, &_ovlp ) )
|
|
{
|
|
err = GetLastError();
|
|
|
|
if( ERROR_IO_PENDING == err )
|
|
{
|
|
// Only wait if ReadFile errored and returned ERROR_IO_PENDING.
|
|
// In that case clear "err" and continue.
|
|
//
|
|
err = ERROR_SUCCESS;
|
|
if( !GetOverlappedResult( _hFile, &_ovlp, pcbRead, TRUE) )
|
|
{
|
|
err = GetLastError();
|
|
}
|
|
}
|
|
}
|
|
|
|
if(ERROR_SUCCESS != err && ERROR_HANDLE_EOF != err)
|
|
nffChk( HRESULT_FROM_WIN32( err ) );
|
|
|
|
nffDebug(( DEB_INFO, "SyncReadAtFile() read 0x%x bytes.\n", *pcbRead ));
|
|
|
|
EH_Err:
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::SyncWriteAtFile
|
|
//
|
|
// Synopsis: Provide synchronous IO for a file handle open in
|
|
// asynchronous mode.
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::SyncWriteAtFile(
|
|
ULARGE_INTEGER ulOffset,
|
|
const void *pv,
|
|
ULONG cb,
|
|
PULONG pcbWritten)
|
|
{
|
|
HRESULT sc=S_OK;
|
|
//
|
|
// We use the single OVERLAPPED structure in the object.
|
|
// We require the TreeMutex will keep us single threaded.
|
|
//
|
|
_ovlp.Offset = ulOffset.LowPart;
|
|
_ovlp.OffsetHigh = ulOffset.HighPart;
|
|
|
|
nffDebug(( DEB_ITRACE | DEB_INFO | DEB_WRITE,
|
|
"SyncWriteAtFile(_hFile=0x%x, off=%x:%x, pv=0x%x, cb=0x%x );"
|
|
" stream='%ws'\n",
|
|
_hFile, ulOffset.HighPart, ulOffset.LowPart,
|
|
pv, cb, _pwcsName ));
|
|
|
|
//
|
|
// We expect either OK or FALSE/ERROR_IO_PENDING.
|
|
//
|
|
if( !WriteFile( _hFile, pv, cb, pcbWritten, &_ovlp ) )
|
|
{
|
|
if( ERROR_IO_PENDING != GetLastError() )
|
|
nffChk( HRESULT_FROM_WIN32( GetLastError() ) );
|
|
}
|
|
|
|
if( !GetOverlappedResult( _hFile, &_ovlp, pcbWritten, TRUE) )
|
|
nffChk( HRESULT_FROM_WIN32( GetLastError() ) );
|
|
|
|
nffDebug(( DEB_ITRACE | DEB_INFO,
|
|
"SyncWriteAtFile() wrote 0x%x bytes.\n", *pcbWritten ));
|
|
|
|
EH_Err:
|
|
return sc;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::MarkStreamAux
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::MarkStreamAux( const MARK_HANDLE_INFO& mhi )
|
|
{
|
|
nffDebug(( DEB_INFO | DEB_STATCTRL | DEB_ITRACE,
|
|
"MarkStreamAux() hdl=%x, stream='%ws'\n", _hFile, _pwcsName ));
|
|
|
|
return MarkFileHandleAux( _hFile, mhi );
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Static Non-Interface::MarkFileHandleAux
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::MarkFileHandleAux(
|
|
HANDLE hFile,
|
|
const MARK_HANDLE_INFO& mhi )
|
|
{
|
|
nffITrace( "CNtfsStream::MarkStreamAux" );
|
|
|
|
HRESULT sc=S_OK;
|
|
ULONG cbReturned;
|
|
|
|
if( INVALID_HANDLE_VALUE == hFile )
|
|
{
|
|
nffDebug(( DEB_IWARN, "CNtfsStream::MarkStreamAux on hFile == -1\n" ));
|
|
return S_OK;
|
|
}
|
|
|
|
nffAssert( USN_SOURCE_AUXILIARY_DATA == mhi.UsnSourceInfo);
|
|
|
|
nffDebug(( DEB_STATCTRL | DEB_ITRACE,
|
|
"MarkFileHandleAux hdl=%x\n", hFile ));
|
|
|
|
nffBool( DeviceIoControl( hFile,
|
|
FSCTL_MARK_HANDLE,
|
|
(void*)&mhi, sizeof(mhi),
|
|
NULL, 0,
|
|
&cbReturned,
|
|
NULL ) );
|
|
EH_Err:
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Non-Interface::SetStreamTime
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::SetStreamTime( const FILETIME* pctime,
|
|
const FILETIME* patime,
|
|
const FILETIME* pmtime)
|
|
{
|
|
nffDebug((DEB_INFO | DEB_STATCTRL | DEB_ITRACE,
|
|
"SetStreamTime() hdl=%x, stream='%ws'\n",
|
|
_hFile, _pwcsName ));
|
|
|
|
return SetFileHandleTime( _hFile, pctime, patime, pmtime );
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CNtfsStream Static Non-Interface::SetFileHandleTime
|
|
//
|
|
//+----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
CNtfsStream::SetFileHandleTime( HANDLE hFile,
|
|
const FILETIME* pctime,
|
|
const FILETIME* patime,
|
|
const FILETIME* pmtime)
|
|
{
|
|
HRESULT sc=S_OK;
|
|
|
|
if( INVALID_HANDLE_VALUE == hFile )
|
|
{
|
|
nffDebug(( DEB_IWARN, "CNtfsStream::SetFileHandleTime on hFile == -1\n" ));
|
|
return S_OK;
|
|
}
|
|
|
|
nffDebug(( DEB_STATCTRL | DEB_ITRACE,
|
|
"SetFileHandleTime(%p %p %p) hdl=%x\n",
|
|
pctime, patime, pmtime, hFile ));
|
|
|
|
nffBool( ::SetFileTime( hFile, pctime, patime, pmtime ) );
|
|
EH_Err:
|
|
return sc;
|
|
}
|
|
|
|
|
|
#if DBG
|
|
HRESULT STDMETHODCALLTYPE
|
|
CNtfsStream::UseNTFS4Streams( BOOL fUseNTFS4Streams )
|
|
{
|
|
return( _nffMappedStream.UseNTFS4Streams( fUseNTFS4Streams ));
|
|
}
|
|
#endif // #if DBG
|
|
|
|
#if DBG
|
|
HRESULT STDMETHODCALLTYPE
|
|
CNtfsStream::GetFormatVersion(WORD *pw)
|
|
{
|
|
return( E_NOTIMPL );
|
|
}
|
|
#endif // #if DBG
|
|
|
|
#if DBG
|
|
HRESULT STDMETHODCALLTYPE
|
|
CNtfsStream::SimulateLowMemory( BOOL fSimulate )
|
|
{
|
|
return( _nffMappedStream.SimulateLowMemory( fSimulate ));
|
|
}
|
|
#endif // #if DBG
|
|
|
|
#if DBG
|
|
HRESULT STDMETHODCALLTYPE
|
|
CNtfsStream::GetLockCount()
|
|
{
|
|
return( NULL == _pnffstg ? 0 : _pnffstg->GetLockCount() );
|
|
}
|
|
#endif // #if DBG
|
|
|
|
#if DBG
|
|
HRESULT STDMETHODCALLTYPE
|
|
CNtfsStream::IsDirty()
|
|
{
|
|
return( E_NOTIMPL );
|
|
}
|
|
#endif // #if DBG
|