Commit 5c227a9d authored by Alexandre Julliard's avatar Alexandre Julliard

wininet: Add a critical section to protect the read-ahead buffer.

parent 47d927f7
...@@ -1376,6 +1376,7 @@ static void HTTPREQ_Destroy(WININETHANDLEHEADER *hdr) ...@@ -1376,6 +1376,7 @@ static void HTTPREQ_Destroy(WININETHANDLEHEADER *hdr)
HeapFree(GetProcessHeap(), 0, lpwhr->lpszCacheFile); HeapFree(GetProcessHeap(), 0, lpwhr->lpszCacheFile);
} }
DeleteCriticalSection( &lpwhr->read_section );
WININET_Release(&lpwhr->lpHttpSession->hdr); WININET_Release(&lpwhr->lpHttpSession->hdr);
if (lpwhr->pAuthInfo) if (lpwhr->pAuthInfo)
...@@ -1615,7 +1616,7 @@ static DWORD HTTPREQ_SetOption(WININETHANDLEHEADER *hdr, DWORD option, void *buf ...@@ -1615,7 +1616,7 @@ static DWORD HTTPREQ_SetOption(WININETHANDLEHEADER *hdr, DWORD option, void *buf
return ERROR_INTERNET_INVALID_OPTION; return ERROR_INTERNET_INVALID_OPTION;
} }
/* read some more data into the read buffer */ /* read some more data into the read buffer (the read section must be held) */
static BOOL read_more_data( WININETHTTPREQW *req, int maxlen ) static BOOL read_more_data( WININETHTTPREQW *req, int maxlen )
{ {
int len; int len;
...@@ -1634,7 +1635,7 @@ static BOOL read_more_data( WININETHTTPREQW *req, int maxlen ) ...@@ -1634,7 +1635,7 @@ static BOOL read_more_data( WININETHTTPREQW *req, int maxlen )
return TRUE; return TRUE;
} }
/* remove some amount of data from the read buffer */ /* remove some amount of data from the read buffer (the read section must be held) */
static void remove_data( WININETHTTPREQW *req, int count ) static void remove_data( WININETHTTPREQW *req, int count )
{ {
if (!(req->read_size -= count)) req->read_pos = 0; if (!(req->read_size -= count)) req->read_pos = 0;
...@@ -1645,6 +1646,7 @@ static BOOL read_line( WININETHTTPREQW *req, LPSTR buffer, DWORD *len ) ...@@ -1645,6 +1646,7 @@ static BOOL read_line( WININETHTTPREQW *req, LPSTR buffer, DWORD *len )
{ {
int count, bytes_read, pos = 0; int count, bytes_read, pos = 0;
EnterCriticalSection( &req->read_section );
for (;;) for (;;)
{ {
char *eol = memchr( req->read_buf + req->read_pos, '\n', req->read_size ); char *eol = memchr( req->read_buf + req->read_pos, '\n', req->read_size );
...@@ -1662,14 +1664,15 @@ static BOOL read_line( WININETHTTPREQW *req, LPSTR buffer, DWORD *len ) ...@@ -1662,14 +1664,15 @@ static BOOL read_line( WININETHTTPREQW *req, LPSTR buffer, DWORD *len )
remove_data( req, bytes_read ); remove_data( req, bytes_read );
if (eol) break; if (eol) break;
if (!read_more_data( req, -1 )) return FALSE; if (!read_more_data( req, -1 ) || !req->read_size)
if (!req->read_size)
{ {
*len = 0; *len = 0;
TRACE( "returning empty string\n" ); TRACE( "returning empty string\n" );
LeaveCriticalSection( &req->read_section );
return FALSE; return FALSE;
} }
} }
LeaveCriticalSection( &req->read_section );
if (pos < *len) if (pos < *len)
{ {
...@@ -1681,7 +1684,7 @@ static BOOL read_line( WININETHTTPREQW *req, LPSTR buffer, DWORD *len ) ...@@ -1681,7 +1684,7 @@ static BOOL read_line( WININETHTTPREQW *req, LPSTR buffer, DWORD *len )
return TRUE; return TRUE;
} }
/* discard data contents until we reach end of line */ /* discard data contents until we reach end of line (the read section must be held) */
static BOOL discard_eol( WININETHTTPREQW *req ) static BOOL discard_eol( WININETHTTPREQW *req )
{ {
do do
...@@ -1698,7 +1701,7 @@ static BOOL discard_eol( WININETHTTPREQW *req ) ...@@ -1698,7 +1701,7 @@ static BOOL discard_eol( WININETHTTPREQW *req )
return TRUE; return TRUE;
} }
/* read the size of the next chunk */ /* read the size of the next chunk (the read section must be held) */
static BOOL start_next_chunk( WININETHTTPREQW *req ) static BOOL start_next_chunk( WININETHTTPREQW *req )
{ {
DWORD chunk_size = 0; DWORD chunk_size = 0;
...@@ -1738,7 +1741,7 @@ static BOOL start_next_chunk( WININETHTTPREQW *req ) ...@@ -1738,7 +1741,7 @@ static BOOL start_next_chunk( WININETHTTPREQW *req )
} }
} }
/* return the size of data available to be read immediately */ /* return the size of data available to be read immediately (the read section must be held) */
static DWORD get_avail_data( WININETHTTPREQW *req ) static DWORD get_avail_data( WININETHTTPREQW *req )
{ {
if (req->read_chunked && (req->dwContentLength == ~0u || req->dwContentLength == req->dwContentRead)) if (req->read_chunked && (req->dwContentLength == ~0u || req->dwContentLength == req->dwContentRead))
...@@ -1746,7 +1749,7 @@ static DWORD get_avail_data( WININETHTTPREQW *req ) ...@@ -1746,7 +1749,7 @@ static DWORD get_avail_data( WININETHTTPREQW *req )
return min( req->read_size, req->dwContentLength - req->dwContentRead ); return min( req->read_size, req->dwContentLength - req->dwContentRead );
} }
/* check if we have reached the end of the data to read */ /* check if we have reached the end of the data to read (the read section must be held) */
static BOOL end_of_read_data( WININETHTTPREQW *req ) static BOOL end_of_read_data( WININETHTTPREQW *req )
{ {
if (req->read_chunked) return (req->dwContentLength == 0); if (req->read_chunked) return (req->dwContentLength == 0);
...@@ -1754,6 +1757,7 @@ static BOOL end_of_read_data( WININETHTTPREQW *req ) ...@@ -1754,6 +1757,7 @@ static BOOL end_of_read_data( WININETHTTPREQW *req )
return (req->dwContentLength == req->dwContentRead); return (req->dwContentLength == req->dwContentRead);
} }
/* fetch some more data into the read buffer (the read section must be held) */
static BOOL refill_buffer( WININETHTTPREQW *req ) static BOOL refill_buffer( WININETHTTPREQW *req )
{ {
int len = sizeof(req->read_buf); int len = sizeof(req->read_buf);
...@@ -1777,6 +1781,7 @@ static void HTTP_ReceiveRequestData(WININETHTTPREQW *req, BOOL first_notif) ...@@ -1777,6 +1781,7 @@ static void HTTP_ReceiveRequestData(WININETHTTPREQW *req, BOOL first_notif)
TRACE("%p\n", req); TRACE("%p\n", req);
EnterCriticalSection( &req->read_section );
if (refill_buffer( req )) { if (refill_buffer( req )) {
iar.dwResult = (DWORD_PTR)req->hdr.hInternet; iar.dwResult = (DWORD_PTR)req->hdr.hInternet;
iar.dwError = first_notif ? 0 : get_avail_data(req); iar.dwError = first_notif ? 0 : get_avail_data(req);
...@@ -1784,15 +1789,18 @@ static void HTTP_ReceiveRequestData(WININETHTTPREQW *req, BOOL first_notif) ...@@ -1784,15 +1789,18 @@ static void HTTP_ReceiveRequestData(WININETHTTPREQW *req, BOOL first_notif)
iar.dwResult = 0; iar.dwResult = 0;
iar.dwError = INTERNET_GetLastError(); iar.dwError = INTERNET_GetLastError();
} }
LeaveCriticalSection( &req->read_section );
INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_REQUEST_COMPLETE, &iar, INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_REQUEST_COMPLETE, &iar,
sizeof(INTERNET_ASYNC_RESULT)); sizeof(INTERNET_ASYNC_RESULT));
} }
/* read data from the http connection (the read section must be held) */
static DWORD HTTPREQ_Read(WININETHTTPREQW *req, void *buffer, DWORD size, DWORD *read, BOOL sync) static DWORD HTTPREQ_Read(WININETHTTPREQW *req, void *buffer, DWORD size, DWORD *read, BOOL sync)
{ {
int len, bytes_read = 0; int len, bytes_read = 0;
EnterCriticalSection( &req->read_section );
if (req->read_chunked && (req->dwContentLength == ~0u || req->dwContentLength == req->dwContentRead)) if (req->read_chunked && (req->dwContentLength == ~0u || req->dwContentLength == req->dwContentRead))
{ {
if (!start_next_chunk( req )) goto done; if (!start_next_chunk( req )) goto done;
...@@ -1818,6 +1826,8 @@ done: ...@@ -1818,6 +1826,8 @@ done:
*read = bytes_read; *read = bytes_read;
TRACE( "retrieved %u bytes (%u/%u)\n", bytes_read, req->dwContentRead, req->dwContentLength ); TRACE( "retrieved %u bytes (%u/%u)\n", bytes_read, req->dwContentRead, req->dwContentLength );
LeaveCriticalSection( &req->read_section );
if(req->lpszCacheFile) { if(req->lpszCacheFile) {
BOOL res; BOOL res;
DWORD dwBytesWritten; DWORD dwBytesWritten;
...@@ -1879,6 +1889,18 @@ static DWORD HTTPREQ_ReadFileExA(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSA *bu ...@@ -1879,6 +1889,18 @@ static DWORD HTTPREQ_ReadFileExA(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSA *bu
{ {
WORKREQUEST workRequest; WORKREQUEST workRequest;
if (TryEnterCriticalSection( &req->read_section ))
{
if (get_avail_data(req))
{
res = HTTPREQ_Read(req, buffers->lpvBuffer, buffers->dwBufferLength,
&buffers->dwBufferLength, FALSE);
LeaveCriticalSection( &req->read_section );
goto done;
}
LeaveCriticalSection( &req->read_section );
}
workRequest.asyncproc = HTTPREQ_AsyncReadFileExAProc; workRequest.asyncproc = HTTPREQ_AsyncReadFileExAProc;
workRequest.hdr = WININET_AddRef(&req->hdr); workRequest.hdr = WININET_AddRef(&req->hdr);
workRequest.u.InternetReadFileExA.lpBuffersOut = buffers; workRequest.u.InternetReadFileExA.lpBuffersOut = buffers;
...@@ -1891,6 +1913,7 @@ static DWORD HTTPREQ_ReadFileExA(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSA *bu ...@@ -1891,6 +1913,7 @@ static DWORD HTTPREQ_ReadFileExA(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSA *bu
res = HTTPREQ_Read(req, buffers->lpvBuffer, buffers->dwBufferLength, &buffers->dwBufferLength, res = HTTPREQ_Read(req, buffers->lpvBuffer, buffers->dwBufferLength, &buffers->dwBufferLength,
!(flags & IRF_NO_WAIT)); !(flags & IRF_NO_WAIT));
done:
if (res == ERROR_SUCCESS) { if (res == ERROR_SUCCESS) {
DWORD size = buffers->dwBufferLength; DWORD size = buffers->dwBufferLength;
INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_RESPONSE_RECEIVED, INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_RESPONSE_RECEIVED,
...@@ -1935,10 +1958,22 @@ static DWORD HTTPREQ_ReadFileExW(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSW *bu ...@@ -1935,10 +1958,22 @@ static DWORD HTTPREQ_ReadFileExW(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSW *bu
INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_RECEIVING_RESPONSE, NULL, 0); INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_RECEIVING_RESPONSE, NULL, 0);
if ((hdr->dwFlags & INTERNET_FLAG_ASYNC) && !get_avail_data(req)) if (hdr->dwFlags & INTERNET_FLAG_ASYNC)
{ {
WORKREQUEST workRequest; WORKREQUEST workRequest;
if (TryEnterCriticalSection( &req->read_section ))
{
if (get_avail_data(req))
{
res = HTTPREQ_Read(req, buffers->lpvBuffer, buffers->dwBufferLength,
&buffers->dwBufferLength, FALSE);
LeaveCriticalSection( &req->read_section );
goto done;
}
LeaveCriticalSection( &req->read_section );
}
workRequest.asyncproc = HTTPREQ_AsyncReadFileExWProc; workRequest.asyncproc = HTTPREQ_AsyncReadFileExWProc;
workRequest.hdr = WININET_AddRef(&req->hdr); workRequest.hdr = WININET_AddRef(&req->hdr);
workRequest.u.InternetReadFileExW.lpBuffersOut = buffers; workRequest.u.InternetReadFileExW.lpBuffersOut = buffers;
...@@ -1951,6 +1986,7 @@ static DWORD HTTPREQ_ReadFileExW(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSW *bu ...@@ -1951,6 +1986,7 @@ static DWORD HTTPREQ_ReadFileExW(WININETHANDLEHEADER *hdr, INTERNET_BUFFERSW *bu
res = HTTPREQ_Read(req, buffers->lpvBuffer, buffers->dwBufferLength, &buffers->dwBufferLength, res = HTTPREQ_Read(req, buffers->lpvBuffer, buffers->dwBufferLength, &buffers->dwBufferLength,
!(flags & IRF_NO_WAIT)); !(flags & IRF_NO_WAIT));
done:
if (res == ERROR_SUCCESS) { if (res == ERROR_SUCCESS) {
DWORD size = buffers->dwBufferLength; DWORD size = buffers->dwBufferLength;
INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_RESPONSE_RECEIVED, INTERNET_SendCallback(&req->hdr, req->hdr.dwContext, INTERNET_STATUS_RESPONSE_RECEIVED,
...@@ -1988,34 +2024,42 @@ static DWORD HTTPREQ_QueryDataAvailable(WININETHANDLEHEADER *hdr, DWORD *availab ...@@ -1988,34 +2024,42 @@ static DWORD HTTPREQ_QueryDataAvailable(WININETHANDLEHEADER *hdr, DWORD *availab
TRACE("(%p %p %x %lx)\n", req, available, flags, ctx); TRACE("(%p %p %x %lx)\n", req, available, flags, ctx);
if (!(*available = get_avail_data( req ))) if (req->lpHttpSession->lpAppInfo->hdr.dwFlags & INTERNET_FLAG_ASYNC)
{ {
if (end_of_read_data( req )) return ERROR_SUCCESS; WORKREQUEST workRequest;
if (req->lpHttpSession->lpAppInfo->hdr.dwFlags & INTERNET_FLAG_ASYNC) /* never wait, if we can't enter the section we queue an async request right away */
if (TryEnterCriticalSection( &req->read_section ))
{ {
WORKREQUEST workRequest; if ((*available = get_avail_data( req ))) goto done;
if (end_of_read_data( req )) goto done;
LeaveCriticalSection( &req->read_section );
}
workRequest.asyncproc = HTTPREQ_AsyncQueryDataAvailableProc; workRequest.asyncproc = HTTPREQ_AsyncQueryDataAvailableProc;
workRequest.hdr = WININET_AddRef( &req->hdr ); workRequest.hdr = WININET_AddRef( &req->hdr );
INTERNET_AsyncCall(&workRequest); INTERNET_AsyncCall(&workRequest);
return ERROR_IO_PENDING; return ERROR_IO_PENDING;
}
else
{
refill_buffer( req );
*available = get_avail_data( req );
}
} }
EnterCriticalSection( &req->read_section );
if (!(*available = get_avail_data( req )) && !end_of_read_data( req ))
{
refill_buffer( req );
*available = get_avail_data( req );
}
done:
if (*available == sizeof(req->read_buf)) /* check if we have even more pending in the socket */ if (*available == sizeof(req->read_buf)) /* check if we have even more pending in the socket */
{ {
DWORD extra; DWORD extra;
if (NETCON_query_data_available(&req->netConnection, &extra)) if (NETCON_query_data_available(&req->netConnection, &extra))
*available = min( *available + extra, req->dwContentLength - req->dwContentRead ); *available = min( *available + extra, req->dwContentLength - req->dwContentRead );
} }
LeaveCriticalSection( &req->read_section );
TRACE( "returning %u\n", *available ); TRACE( "returning %u\n", *available );
return ERROR_SUCCESS; return ERROR_SUCCESS;
...@@ -2075,6 +2119,7 @@ HINTERNET WINAPI HTTP_HttpOpenRequestW(LPWININETHTTPSESSIONW lpwhs, ...@@ -2075,6 +2119,7 @@ HINTERNET WINAPI HTTP_HttpOpenRequestW(LPWININETHTTPSESSIONW lpwhs,
lpwhr->hdr.lpfnStatusCB = lpwhs->hdr.lpfnStatusCB; lpwhr->hdr.lpfnStatusCB = lpwhs->hdr.lpfnStatusCB;
lpwhr->hdr.dwInternalFlags = lpwhs->hdr.dwInternalFlags & INET_CALLBACKW; lpwhr->hdr.dwInternalFlags = lpwhs->hdr.dwInternalFlags & INET_CALLBACKW;
lpwhr->dwContentLength = ~0u; lpwhr->dwContentLength = ~0u;
InitializeCriticalSection( &lpwhr->read_section );
WININET_AddRef( &lpwhs->hdr ); WININET_AddRef( &lpwhs->hdr );
lpwhr->lpHttpSession = lpwhs; lpwhr->lpHttpSession = lpwhs;
......
...@@ -193,8 +193,6 @@ typedef struct ...@@ -193,8 +193,6 @@ typedef struct
WININET_NETCONNECTION netConnection; WININET_NETCONNECTION netConnection;
LPWSTR lpszVersion; LPWSTR lpszVersion;
LPWSTR lpszStatusText; LPWSTR lpszStatusText;
DWORD dwContentLength; /* total number of bytes to be read */
DWORD dwContentRead; /* bytes of the content read so far */
DWORD dwBytesToWrite; DWORD dwBytesToWrite;
DWORD dwBytesWritten; DWORD dwBytesWritten;
HTTPHEADERW *pCustHeaders; HTTPHEADERW *pCustHeaders;
...@@ -203,6 +201,10 @@ typedef struct ...@@ -203,6 +201,10 @@ typedef struct
LPWSTR lpszCacheFile; LPWSTR lpszCacheFile;
struct HttpAuthInfo *pAuthInfo; struct HttpAuthInfo *pAuthInfo;
struct HttpAuthInfo *pProxyAuthInfo; struct HttpAuthInfo *pProxyAuthInfo;
CRITICAL_SECTION read_section; /* section to protect the following fields */
DWORD dwContentLength; /* total number of bytes to be read */
DWORD dwContentRead; /* bytes of the content read so far */
BOOL read_chunked; /* are we reading in chunked mode? */ BOOL read_chunked; /* are we reading in chunked mode? */
DWORD read_pos; /* current read position in read_buf */ DWORD read_pos; /* current read position in read_buf */
DWORD read_size; /* valid data size in read_buf */ DWORD read_size; /* valid data size in read_buf */
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment