mirror of https://github.com/tongzx/nt5src
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.
215 lines
5.4 KiB
215 lines
5.4 KiB
|
|
//
|
|
// unbuffered rapid disk i/o class. Provides streaming write to disk using
|
|
// unbuffered, overlapped i/o via large buffers. Inter-thread sync
|
|
// must be provided elsewhere.
|
|
//
|
|
|
|
#ifndef _DIRECTIO_H_
|
|
#define _DIRECTIO_H_
|
|
|
|
#ifdef _WIN32
|
|
#ifdef CHICAGO
|
|
#include "disk32.h"
|
|
#endif
|
|
|
|
// all 'tunable' constants now found in CFileStream::EnsureBuffersValid()
|
|
|
|
// maximum number of buffers that can be requested
|
|
#define NR_OF_BUFFERS 4
|
|
|
|
// min read size
|
|
#define MIN_READ_SIZE (12 * 1024)
|
|
|
|
// --- we use these internally ----
|
|
// unbuffered i/o handler class. requires copy of data.
|
|
// will round reads and writes to correct sector size, and
|
|
// will pre-read if start location of read or write is not in buffer.
|
|
// read or write will terminate early if insufficient space or data.
|
|
// writes must be explicitly initiated from the buffer to disk
|
|
// by calling commit. reads will be initiated by the Read function, or may
|
|
// be initiated offline using ReadAhead.
|
|
|
|
class CFileBuffer {
|
|
|
|
public:
|
|
// initiate to an invalid (no buffer ready) state
|
|
CFileBuffer();
|
|
|
|
// allocate memory and become idle.
|
|
#ifdef CHICAGO
|
|
BOOL Init(DWORD nBytesPerSector, DWORD buffersize, LPQIO pqio);
|
|
#else
|
|
BOOL Init(DWORD nBytesPerSector, DWORD buffersize, HANDLE hfile);
|
|
#endif
|
|
|
|
// revert to invalid state (eg when streaming stops)
|
|
void FreeMemory();
|
|
|
|
|
|
// write some data to buffer (must be committed separately)
|
|
// filesize parameter is the current file size before this write
|
|
// (used to control reading of partial sectors).
|
|
BOOL Write(DWORD pos, LPBYTE pData, DWORD count, DWORD filesize,
|
|
DWORD * pBytesWritten);
|
|
|
|
// read data from buffer (will seek and read if necessary first)
|
|
BOOL Read(DWORD pos, LPBYTE pData, DWORD count,
|
|
DWORD filelength, DWORD * pBytesRead);
|
|
|
|
// does this position occur anywhere within the current buffer ?
|
|
// needs to know current eof for some cases (writing beyond eof
|
|
// if eof is within this buffer is ok to this buffer).
|
|
BOOL QueryPosition(DWORD pos, DWORD filesize);
|
|
|
|
// what is the first file position after this buffer's valid data
|
|
DWORD GetNextPosition();
|
|
|
|
// initiate a read-ahead
|
|
void ReadAhead(DWORD start, DWORD filelength);
|
|
|
|
|
|
// initiate the i/o from the buffer
|
|
BOOL Commit();
|
|
|
|
// wait for any pending commit to complete
|
|
BOOL WaitComplete();
|
|
|
|
// is the buffer idle - FALSE if currently busy or invalid
|
|
BOOL IsIdle() {
|
|
return (m_State == Idle);
|
|
};
|
|
|
|
// calls commit if dirty before freeing everything.
|
|
~CFileBuffer();
|
|
|
|
private:
|
|
|
|
// non-blocking check to see if pending i/o is complete and ok
|
|
BOOL CheckComplete();
|
|
|
|
BOOL ReadIntoBuffer(int offset, DWORD pos, DWORD count);
|
|
|
|
DWORD_PTR RoundPosToSector(DWORD_PTR pos)
|
|
{
|
|
// positions round down to the previous sector start
|
|
return (pos / m_BytesPerSector) * m_BytesPerSector;
|
|
};
|
|
|
|
DWORD_PTR RoundSizeToSector(DWORD_PTR size)
|
|
{
|
|
// sizes round up to total sector count
|
|
return ((size + m_BytesPerSector - 1) / m_BytesPerSector)
|
|
* m_BytesPerSector;
|
|
}
|
|
|
|
|
|
// buffer states
|
|
enum BufferState { Idle, Busy, Invalid, ErrorOccurred };
|
|
|
|
BufferState m_State;
|
|
BOOL m_bDirty;
|
|
LPBYTE m_pBuffer; // buffer with start addr rounded
|
|
LPBYTE m_pAllocedMem; // buffer before rounding
|
|
DWORD m_TotalSize; // allocated buffer size
|
|
DWORD m_DataLength; // bytes of valid data in buffer
|
|
DWORD m_Position; // file position of start of buffer
|
|
DWORD m_BytesPerSector; // sector boundaries are important
|
|
DWORD m_FileLength; // actual file size (not rounded)
|
|
|
|
|
|
#ifdef CHICAGO
|
|
QIOBUF m_qiobuf;
|
|
LPQIO m_pqio;
|
|
#else
|
|
OVERLAPPED m_Overlapped;
|
|
HANDLE m_hFile;
|
|
#endif
|
|
};
|
|
|
|
|
|
|
|
class CFileStream {
|
|
|
|
public:
|
|
CFileStream(); // does not do much (cannot return error)
|
|
|
|
BOOL Open(LPTSTR file, BOOL bWrite, BOOL bTruncate);
|
|
|
|
BOOL Seek(DWORD pos);
|
|
|
|
BOOL Write(LPBYTE pData, DWORD count, DWORD * pbyteswritten);
|
|
|
|
BOOL Read(LPBYTE pData, DWORD count, DWORD * pbytesread);
|
|
|
|
DWORD GetCurrentPosition();
|
|
|
|
BOOL StartStreaming(); // default (write if opened for write)
|
|
BOOL StartWriteStreaming();
|
|
BOOL StartReadStreaming();
|
|
BOOL StopStreaming();
|
|
|
|
// wait for all transfers to complete.
|
|
BOOL CommitAndWait();
|
|
|
|
// destructor will call Commit()
|
|
~CFileStream();
|
|
|
|
private:
|
|
|
|
// enable extra buffers for streaming
|
|
BOOL EnsureBuffersValid();
|
|
|
|
// advance to next buffer
|
|
int NextBuffer(int i) {
|
|
return (i + 1) % m_NrValid;
|
|
};
|
|
|
|
|
|
// unbuffered i/o is only allowed in multiples of this
|
|
DWORD m_SectorSize;
|
|
|
|
CFileBuffer m_Buffers[NR_OF_BUFFERS];
|
|
|
|
|
|
// how many buffers are valid ?
|
|
int m_NrValid;
|
|
|
|
// which is the current buffer
|
|
int m_Current;
|
|
|
|
// which buffer has the highest position - we will issue the
|
|
// readahead when we start using this buffer
|
|
int m_HighestBuffer;
|
|
|
|
// next read/write position within file
|
|
DWORD m_Position;
|
|
|
|
enum StreamingState { Invalid, Stopped, Reading, Writing };
|
|
StreamingState m_State;
|
|
|
|
DWORD m_Size; // current file size
|
|
|
|
// file handle
|
|
#ifdef CHICAGO
|
|
QIO m_qio;
|
|
#define m_hFile m_qio.hFile
|
|
#else
|
|
HANDLE m_hFile;
|
|
#endif
|
|
|
|
// if opened for writing, then default streaming mode is write
|
|
BOOL m_bWrite;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif //_WIN32
|
|
|
|
|
|
#endif // _DIRECTIO_H_
|
|
|
|
|
|
|