Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

354 lines
9.5 KiB

#pragma once
//+============================================================================
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1994 - 1998.
//
// File: hntfsstg.hxx
//
// This file provides the NFF (NTFS Flat File) IStream implementation.
//
// History:
//
//+============================================================================
#include "reserved.hxx"
#include "nffmstm.hxx"
//
// NTFS Stream and NTFS Storage are debugged together under the same
// infolevel and debug sub-system. The Stream header is included first
//
DECLARE_DEBUG(nff);
#ifdef DEB_INFO
#undef DEB_INFO
#endif
#define DEB_INFO DEB_USER1
#define DEB_REFCOUNT DEB_USER2
#define DEB_READ DEB_USER3
#define DEB_WRITE DEB_USER4
#define DEB_OPENS DEB_USER5
#define DEB_STATCTRL DEB_USER6
#define DEB_OPLOCK DEB_USER7
#if DBG == 1
#define nffAssert(e) Win4Assert(e)
#define nffVerify(e) Win4Assert(e)
#define nffDebug(x) nffInlineDebugOut x
#define nffXTrace(x) nffCDbgTrace dbg_( DEB_TRACE, x )
#define nffITrace(x) nffCDbgTrace dbg_( DEB_ITRACE, x )
// nffDebugOut is called from the Chk/Err macros
#define nffDebugOut(x) nffInlineDebugOut x
#else
#define nffAssert(e)
#define nffVerify(e) (e)
#define nffDebug(x)
#define nffXTrace(x)
#define nffITrace(x)
#endif
#define nffErr(l, e) ErrJmp(nff, l, e, sc)
#define nffChkTo(l, e) if (FAILED(sc = (e))) nffErr(l, sc) else 1
#define nffChk(e) nffChkTo(EH_Err, e)
#define nffHChkTo(l, e) if (FAILED(sc = DfGetScode(e))) nffErr(l, sc) else 1
#define nffHChk(e) nffHChkTo(EH_Err, e)
#define nffMemTo(l, e) \
if ((e) == NULL) nffErr(l, STG_E_INSUFFICIENTMEMORY) else 1
#define nffMem(e) nffMemTo(EH_Err, e)
#define nffBoolTo(l, e) if (!(e)) nffErr(l, LAST_STG_SCODE) else 1
#define nffBool(e) nffBoolTo(EH_Err, e)
#define NFF_VALIDATE(x) EXP_VALIDATE(nff, x)
#define NTFSSTREAM_SIG LONGSIG('N','T','S','T')
#define NTFSSTREAM_SIGDEL LONGSIG('N','T','S','t')
////////////////////////////////////////////////////////////////
// IStream for an NTFS file stream. Hungarian Prefix "nffstm"
//
class CNtfsStream : public IStream,
public ILockBytes // For use in e.g. StgCreateStorageOnILockBytes
#if DBG
, public IStorageTest
#endif
{
friend class CNtfsStorage;
friend class CNFFMappedStream;
// ------------
// Construction
// ------------
public:
CNtfsStream( CNtfsStorage *pnffstg, IBlockingLock *pBlockingLock );
virtual ~CNtfsStream();
virtual HRESULT Init( HANDLE hFile,
DWORD grfMode,
const OLECHAR *pwcsName,
CNtfsStream *pnffstm );
// --------
// IUnknown
// --------
public:
STDMETHODIMP QueryInterface(REFIID riid, void **ppvObject);
STDMETHODIMP_(ULONG) AddRef(void);
STDMETHODIMP_(ULONG) Release(void);
// -------
// IStream
// -------
public:
HRESULT STDMETHODCALLTYPE Read(
/* [length_is][size_is][out] */ void __RPC_FAR *pv,
/* [in] */ ULONG cb,
/* [out] */ ULONG __RPC_FAR *pcbRead);
HRESULT STDMETHODCALLTYPE Write(
/* [size_is][in] */ const void __RPC_FAR *pv,
/* [in] */ ULONG cb,
/* [out] */ ULONG __RPC_FAR *pcbWritten);
HRESULT STDMETHODCALLTYPE Seek(
/* [in] */ LARGE_INTEGER dlibMove,
/* [in] */ DWORD dwOrigin,
/* [out] */ ULARGE_INTEGER __RPC_FAR *plibNewPosition);
HRESULT STDMETHODCALLTYPE SetSize(
/* [in] */ ULARGE_INTEGER libNewSize);
HRESULT STDMETHODCALLTYPE CopyTo(
/* [unique][in] */ IStream __RPC_FAR *pstm,
/* [in] */ ULARGE_INTEGER cb,
/* [out] */ ULARGE_INTEGER __RPC_FAR *pcbRead,
/* [out] */ ULARGE_INTEGER __RPC_FAR *pcbWritten);
HRESULT STDMETHODCALLTYPE Commit(
/* [in] */ DWORD grfCommitFlags);
HRESULT STDMETHODCALLTYPE Revert(void);
HRESULT STDMETHODCALLTYPE LockRegion(
/* [in] */ ULARGE_INTEGER libOffset,
/* [in] */ ULARGE_INTEGER cb,
/* [in] */ DWORD dwLockType);
HRESULT STDMETHODCALLTYPE UnlockRegion(
/* [in] */ ULARGE_INTEGER libOffset,
/* [in] */ ULARGE_INTEGER cb,
/* [in] */ DWORD dwLockType);
HRESULT STDMETHODCALLTYPE Stat(
/* [out] */ STATSTG __RPC_FAR *pstatstg,
/* [in] */ DWORD grfStatFlag);
HRESULT STDMETHODCALLTYPE Clone(
/* [out] */ IStream __RPC_FAR *__RPC_FAR *ppstm);
// ----------
// ILockBytes
// ----------
public:
HRESULT STDMETHODCALLTYPE ReadAt(
/* [in] */ ULARGE_INTEGER ulOffset,
/* [length_is][size_is][out] */ void __RPC_FAR *pv,
/* [in] */ ULONG cb,
/* [out] */ ULONG __RPC_FAR *pcbRead);
HRESULT STDMETHODCALLTYPE WriteAt(
/* [in] */ ULARGE_INTEGER ulOffset,
/* [size_is][in] */ const void __RPC_FAR *pv,
/* [in] */ ULONG cb,
/* [out] */ ULONG __RPC_FAR *pcbWritten);
HRESULT STDMETHODCALLTYPE Flush(void);
public:
inline BOOL IsWriteable();
HRESULT CheckReverted();
// ------------
// IStorageTest
// ------------
public:
#if DBG
STDMETHOD(UseNTFS4Streams)( BOOL fUseNTFS4Streams );
STDMETHOD(GetFormatVersion)(WORD *pw);
STDMETHOD(SimulateLowMemory)( BOOL fSimulate );
STDMETHOD(GetLockCount)();
STDMETHOD(IsDirty)();
#endif
// ----------------
// Internal Methods
// ----------------
protected:
virtual HRESULT ShutDown();
void InsertSelfIntoList(CNtfsStream * pnffstmList);
void RemoveSelfFromList();
HRESULT Delete();
private:
HRESULT SetFileSize( const CULargeInteger &uliNewSize );
HRESULT Rename( const WCHAR *pwcsName, BOOL fOverWrite );
inline HRESULT Lock( DWORD dwTimeout );
inline HRESULT Unlock();
static HRESULT DeleteStream( HANDLE *phStream );
HRESULT SyncReadAtFile( ULARGE_INTEGER ulOffset,
PVOID pv, ULONG cb, PULONG pcbRead );
HRESULT SyncWriteAtFile( ULARGE_INTEGER ulOffset,
const void *pv, ULONG cb, PULONG pcbWritten );
HANDLE GetFileHandle(); // Used by friend CNtfsStorage.
HRESULT MarkStreamAux( const MARK_HANDLE_INFO& mhi );
static HRESULT MarkFileHandleAux( HANDLE hFile, const MARK_HANDLE_INFO& mhi );
HRESULT SetStreamTime( const FILETIME*, const FILETIME*, const FILETIME* );
static HRESULT SetFileHandleTime( HANDLE hFile, const FILETIME*, const FILETIME*, const FILETIME* );
const WCHAR* GetName() const;
// --------------
// Internal State
// --------------
private:
WCHAR * _pwcsName;
CNFFMappedStream _nffMappedStream;
DWORD _grfMode; // The mode used to open the IStream
HANDLE _hFile; // File represented by this stream
IBlockingLock * _pBlockingLock; // The lock to use for mutual exclusion
ULONG _sig; // Class signature
LONG _cRefs; // Reference count
CNtfsStorage * _pnffstg; // Not ref-counted, NULL-ed in ShutDown
// This class maintains its own copy of the seek pointer, different from
// the underlying file's. This is necessary so that the IStream methods mantain
// a consistent seek location, even when methods on e.g. IMappedStream are called.
CLargeInteger _liCurrentSeekPosition;
CNtfsStream * _pnffstmPrev; // links for the list of open streams.
CNtfsStream * _pnffstmNext;
OVERLAPPED _ovlp; // structure used for Async IO.
}; // class CNtfsStream
inline HANDLE
CNtfsStream::GetFileHandle()
{
return _hFile;
}
inline HRESULT
CNtfsStream::CheckReverted()
{
if(INVALID_HANDLE_VALUE == _hFile)
return STG_E_REVERTED;
return S_OK;
}
inline const WCHAR*
CNtfsStream::GetName() const
{
return _pwcsName;
}
inline HRESULT
CNtfsStream::Lock( DWORD dwTimeout )
{
return( _pBlockingLock->Lock( dwTimeout ));
}
inline HRESULT
CNtfsStream::Unlock()
{
return( _pBlockingLock->Unlock() );
}
inline BOOL
CNtfsStream::IsWriteable()
{
return( GrfModeIsWriteable( _grfMode ));
}
//+----------------------------------------------------------------------------
//
// Class: CNtfsUpdateStreamForPropStg
//
// This class wraps the update stream handle, used by
// CNFFMappedStream. See that class declaration for a description.
//
//+----------------------------------------------------------------------------
class CNtfsUpdateStreamForPropStg : public CNtfsStream
{
public:
CNtfsUpdateStreamForPropStg( CNtfsStorage *pnffstg, IBlockingLock *_pBlockingLock );
~CNtfsUpdateStreamForPropStg();
protected:
virtual HRESULT ShutDown();
}; // class CNtfsStreamForPropStg
inline
CNtfsUpdateStreamForPropStg::CNtfsUpdateStreamForPropStg( CNtfsStorage *pnffstg, IBlockingLock *pBlockingLock )
: CNtfsStream( pnffstg, pBlockingLock )
{
}
inline
CNtfsUpdateStreamForPropStg::~CNtfsUpdateStreamForPropStg()
{
// If the CNFFMappedStream was shutdown without flushing, and it couldn't
// save and fix up the update stream, then there's nothing we can
// do to recover and we should delete the stream.
// In the normal path, CNFFMappedStream::~CNFFMappedStream calls
// ReplaceOriginalWithUpdate(DONT_CREATE_NEW_UPDATE_STREAM),
// and subsequently our handle is closed, and CheckReverted
// returns STG_E_REVERTED.
if( SUCCEEDED(CheckReverted()) )
Delete();
}