Windows2000/private/inet/wininet/http/prefetch.cxx
2020-09-30 17:12:32 +02:00

475 lines
12 KiB
C++

/*++
Copyright (c) 1996 Microsoft Corporation
Module Name:
prefetch.cxx
Abstract:
Functions and methods to support http random read access.
Contents:
AttemptReadFromFile
SetStreamPointer
SetLengthFromCache
ReadLoop_Fsm
WriteResponseBufferToCache
WriteQueryBufferToCache
Author:
Rajeev Dujari (rajeevd) March 1996
Revision History:
Ahsan Kabir (akabir) November 1997
--*/
#include <wininetp.h>
#define READLOOP_BUFSIZE 2048
BOOL
HTTP_REQUEST_HANDLE_OBJECT::AttemptReadFromFile(
LPVOID lpBuf,
DWORD cbToRead,
DWORD* pcbRead
)
{
DEBUG_ENTER((DBG_CACHE,
Bool,
"INTERNET_CONNECT_HANDLE_OBJECT::AttemptReadFromFile",
"%#x, %d, %#x",
lpBuf,
cbToRead,
pcbRead
));
BOOL fSuccess;
DWORD dwBytesToCopy = 0;
if (!cbToRead)
{
*pcbRead = 0;
DEBUG_LEAVE(TRUE);
return TRUE;
}
if (IsCacheReadInProgress())
{
INET_ASSERT(_VirtualCacheFileSize == _RealCacheFileSize);
// Entire read should be satisfied from cache.
*pcbRead = cbToRead;
if (ReadUrlCacheEntryStream
(_hCacheStream, _dwCurrentStreamPosition, lpBuf, pcbRead, 0))
{
AdvanceReadPosition (*pcbRead);
DEBUG_LEAVE(TRUE);
return TRUE;
}
else
{
*pcbRead = 0;
DEBUG_LEAVE(FALSE);
return FALSE;
}
}
else if (IsCacheWriteInProgress())
{
// See if the read is completely within the file.
if (!IsEndOfFile() && _dwCurrentStreamPosition + cbToRead > _VirtualCacheFileSize)
{
DEBUG_PRINT(HTTP, ERROR, ("AttemptRead Failed streampos=%d cbToRead=%d, _VitrualCacheFileSize=%d\n",
_dwCurrentStreamPosition, cbToRead, _VirtualCacheFileSize));
DEBUG_LEAVE(FALSE);
return FALSE;
}
HANDLE hfRead;
hfRead = GetDownloadFileReadHandle();
if (hfRead == INVALID_HANDLE_VALUE)
{
DEBUG_LEAVE(FALSE);
return FALSE;
}
// Read the data from the file.
SetFilePointer (hfRead, _dwCurrentStreamPosition, NULL, FILE_BEGIN);
fSuccess = ReadFile (hfRead, lpBuf, cbToRead, pcbRead, NULL);
if (!fSuccess)
{
DEBUG_LEAVE(FALSE);
return FALSE;
}
AdvanceReadPosition (*pcbRead);
DEBUG_LEAVE(TRUE);
return TRUE;
}
else
{
DEBUG_LEAVE(FALSE);
return FALSE;
}
}
// Called from InternetSetFilePointer
DWORD
HTTP_REQUEST_HANDLE_OBJECT::SetStreamPointer(
LONG lDistanceToMove,
DWORD dwMoveMethod
)
{
DWORD dwErr = ERROR_SUCCESS;
// Fail if data is not from cache or going to cache.
if (!IsCacheReadInProgress() && !IsCacheWriteInProgress())
{
dwErr = ERROR_INTERNET_INVALID_OPERATION;
goto done;
}
// BUGBUG: we don't handle chunked transfer, new with http 1.1
if (IsChunkEncoding())
{
dwErr = ERROR_INTERNET_INVALID_OPERATION;
goto done;
}
switch (dwMoveMethod)
{
case FILE_BEGIN:
_dwCurrentStreamPosition = (DWORD) lDistanceToMove;
break;
case FILE_CURRENT:
if (lDistanceToMove < 0
&& ((DWORD) -lDistanceToMove) > _dwCurrentStreamPosition)
{
dwErr = ERROR_NEGATIVE_SEEK;
}
else
{
_dwCurrentStreamPosition += lDistanceToMove;
}
break;
case FILE_END:
if (!IsContentLength())
dwErr = ERROR_INTERNET_INVALID_OPERATION;
else if (lDistanceToMove < 0 && ((DWORD) -lDistanceToMove) > _ContentLength)
dwErr = ERROR_NEGATIVE_SEEK;
else
_dwCurrentStreamPosition = _ContentLength + lDistanceToMove;
break;
default:
dwErr = ERROR_INVALID_PARAMETER;
break;
}
done:
if (dwErr == ERROR_SUCCESS)
{
if (IsKeepAlive() && IsContentLength())
_BytesRemaining = _ContentLength - _dwCurrentStreamPosition;
if (_VirtualCacheFileSize > _dwCurrentStreamPosition)
SetAvailableDataLength (_VirtualCacheFileSize - _dwCurrentStreamPosition);
else
SetAvailableDataLength (0);
return _dwCurrentStreamPosition;
}
else
{
SetLastError (dwErr);
return (DWORD) -1L;
}
}
DWORD
CFsm_ReadLoop::RunSM(
IN CFsm * Fsm
)
{
DEBUG_ENTER((DBG_HTTP,
Dword,
"CFsm_ReadLoop::RunSM",
"%#x",
Fsm
));
DWORD error;
HTTP_REQUEST_HANDLE_OBJECT * pRequest;
CFsm_ReadLoop * stateMachine = (CFsm_ReadLoop *)Fsm;
pRequest = (HTTP_REQUEST_HANDLE_OBJECT *)Fsm->GetContext();
switch (Fsm->GetState()) {
case FSM_STATE_INIT:
case FSM_STATE_CONTINUE:
error = pRequest->ReadLoop_Fsm(stateMachine);
break;
default:
error = ERROR_INTERNET_INTERNAL_ERROR;
Fsm->SetDone(ERROR_INTERNET_INTERNAL_ERROR);
INET_ASSERT(FALSE);
break;
}
DEBUG_LEAVE(error);
return error;
}
DWORD
HTTP_REQUEST_HANDLE_OBJECT::ReadLoop_Fsm(
IN CFsm_ReadLoop * Fsm
)
{
DEBUG_ENTER((DBG_HTTP,
Dword,
"HTTP_REQUEST_HANDLE_OBJECT::ReadLoop_Fsm",
"%#x",
Fsm
));
CFsm_ReadLoop & fsm = *Fsm;
DWORD dwErr = fsm.GetError();
if (fsm.GetState() == FSM_STATE_CONTINUE)
{
goto receive_continue;
}
// Set the goal for reading from the socket.
fsm.m_dwReadEnd = _dwCurrentStreamPosition +
((fsm.m_dwSocketFlags & SF_NO_WAIT)? 1 : fsm.m_cbReadIn);
// Flush any data in response buffer to download file.
dwErr = WriteResponseBufferToCache();
if (dwErr != ERROR_SUCCESS)
goto done;
// Flush any data in query buffer to download file.
dwErr = WriteQueryBufferToCache();
if (dwErr != ERROR_SUCCESS)
goto done;
// BUGBUG: what if HaveReadFileExData?
// Allocate receive buffer. We could optimize this to use
// the client buffer or query buffer when available to avoid
// reading from the file data that was just written.
fsm.m_cbBuf = READLOOP_BUFSIZE;
if (!(fsm.m_pBuf = (PVOID) new BYTE [fsm.m_cbBuf]))
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto done;
}
HANDLE hfRead;
hfRead = GetDownloadFileReadHandle();
if (hfRead == INVALID_HANDLE_VALUE)
{
dwErr = GetLastError();
DEBUG_PRINT(CACHE, ERROR, ("Cache: createfile failed error=%ld\n", dwErr));
goto done;
}
while (1)
{
// Calculate amount of data available for next read.
if (_VirtualCacheFileSize > _dwCurrentStreamPosition)
SetAvailableDataLength (_VirtualCacheFileSize - _dwCurrentStreamPosition);
else
SetAvailableDataLength (0);
// Check if pending request can be satisfied from file.
if (_EndOfFile || _VirtualCacheFileSize >= fsm.m_dwReadEnd)
{
INET_ASSERT (AvailableDataLength());
if (fsm.m_pRead)
{
// The client wants us to fill the buffer.
if (fsm.m_dwSocketFlags & SF_NO_WAIT)
{
// If the app is greedy, give it all we've got.
if (fsm.m_cbReadIn > AvailableDataLength())
fsm.m_cbReadIn = AvailableDataLength();
}
else
{
INET_ASSERT (fsm.m_cbReadIn <= AvailableDataLength());
}
}
// Read the data from the file.
LONG lRet;
lRet = SetFilePointer
(hfRead, _dwCurrentStreamPosition, NULL, FILE_BEGIN);
INET_ASSERT ((DWORD) lRet == _dwCurrentStreamPosition);
if (lRet == -1)
{
dwErr = GetLastError();
goto done;
}
BOOL fRet;
fRet = ReadFile
(hfRead, fsm.m_pRead, fsm.m_cbReadIn, fsm.m_pcbReadOut, NULL);
if (!fRet)
{
dwErr = GetLastError();
goto done;
}
AdvanceReadPosition (*fsm.m_pcbReadOut);
INET_ASSERT (dwErr == ERROR_SUCCESS);
goto done;
}
INET_ASSERT (!_EndOfFile);
INET_ASSERT (_Socket);
fsm.m_cbRead = fsm.m_cbBuf;
if (IsKeepAlive() && fsm.m_cbRead > _BytesInSocket)
fsm.m_cbRead = _BytesInSocket;
fsm.m_cbRecv = 0;
// Get some more data from the socket.
dwErr = _Socket->Receive
(
&fsm.m_pBuf, // IN OUT LPVOID* lplpBuffer,
&fsm.m_cbBuf, // IN OUT LPDWORD lpdwBufferLength,
&fsm.m_cbRead, // IN OUT LPDWORD lpdwBufferRemaining,
&fsm.m_cbRecv, // IN OUT LPDWORD lpdwBytesReceived,
0, // IN DWORD dwExtraSpace,
fsm.m_dwSocketFlags, // IN DWORD dwFlags,
&_EndOfFile // OUT LPBOOL lpbEof
);
if (dwErr == ERROR_IO_PENDING)
{
if (fsm.m_dwSocketFlags & SF_NO_WAIT)
{
// InternetReadFileEx can set IRF_NO_WAIT. If we must go async
// in this case, morph the request into a QueryDataAvailable
// and app will call again to get the data after notification.
fsm.m_pRead = NULL;
fsm.m_cbReadIn = 1;
fsm.m_pcbReadOut = NULL;
}
goto done;
}
receive_continue:
if (dwErr != ERROR_SUCCESS)
{
DEBUG_PRINT(HTTP,
ERROR,
("error %d on socket %#x\n",
dwErr,
_Socket->GetSocket()));
goto done;
}
// Append data to download file.
dwErr = WriteCache ((PBYTE) fsm.m_pBuf, fsm.m_cbRecv);
if (dwErr != ERROR_SUCCESS)
goto done;
// If content length is known, check for EOF.
if (IsKeepAlive())
{
_BytesInSocket -= fsm.m_cbRecv;
if (!_BytesInSocket)
_EndOfFile = TRUE;
}
if (_EndOfFile)
{
// Set handle state.
SetState(HttpRequestStateReopen);
if (!IsKeepAlive())
CloseConnection(FALSE);
SetData(FALSE);
}
} // end while (1)
done:
if (dwErr != ERROR_IO_PENDING)
{
delete [] fsm.m_pBuf;
fsm.SetDone();
if (dwErr != ERROR_SUCCESS)
{
DEBUG_PRINT(HTTP, ERROR, ("Readloop: Error = %d\n", dwErr));
SetState(HttpRequestStateError);
LocalEndCacheWrite(FALSE);
}
}
DEBUG_LEAVE(dwErr);
return dwErr;
}
// Called from ReadLoop to write data in response buffer to download file.
DWORD HTTP_REQUEST_HANDLE_OBJECT::WriteResponseBufferToCache (VOID)
{
DWORD cbData = _BytesReceived - _DataOffset;
if (!cbData)
return ERROR_SUCCESS;
DWORD dwErr = WriteCache
(((LPBYTE) _ResponseBuffer) + _DataOffset, cbData);
_DataOffset += cbData;
DEBUG_PRINT(HTTP, ERROR, ("WriteResponseBufferToCache: Error = %d", dwErr));
return dwErr;
}
// Called from ReadLoop to write data in query buffer to download file.
DWORD HTTP_REQUEST_HANDLE_OBJECT::WriteQueryBufferToCache (VOID)
{
if (!_QueryBytesAvailable)
return ERROR_SUCCESS;
DWORD dwErr = WriteCache
(((LPBYTE) _QueryBuffer) + _QueryOffset, _QueryBytesAvailable);
_QueryOffset += _QueryBytesAvailable;
_QueryBytesAvailable = 0;
DEBUG_PRINT(HTTP, ERROR, ("WriteQueryBufferToCache: Error = %d", dwErr));
return dwErr;
}