2020-09-30 17:12:32 +02:00

438 lines
8.8 KiB
C++

/*++
Copyright (c) 1998 Microsoft Corporation
Module Name:
bgtask.cxx
Abstract:
Contains background task and support functions
Contents:
LoadBackgroundTaskMgr
UnloadBackgroundTaskMgr
NotifyBackgroundTaskMgr
CreateAndQueueBackgroundWorkItem
BackgroundTaskMgr::QueueBackgroundWorkItem
BackgroundTaskMgr::DeQueueAndRunBackgroundWorkItem
BackgroundTaskMgr::CreateBackgroundFsm
BackgroundTaskMgr::Release
BackgroundTaskMgr::HasBandwidth
CFsm_BackgroundTask::RunSM
CFsm_BackgroundTask::DoSendReq
CFsm_BackgroundTask::~CFsm_BackgroundTask
Author:
Danpo Zhang (danpoz) 06-26-98
Environment:
Win32 user-mode
Revision History:
06-26-1998 danpoz
Created
--*/
#include <wininetp.h>
#include <perfdiag.hxx>
BackgroundTaskMgr* g_BGTaskMgr = NULL;
// API: Init global BackgroundTaskManager
BOOL
LoadBackgroundTaskMgr()
{
if( g_BGTaskMgr )
return TRUE;
BackgroundTaskMgr* bgMgr = NULL;
bgMgr = new BackgroundTaskMgr();
if( !bgMgr)
return FALSE;
g_BGTaskMgr = bgMgr;
return TRUE;
}
// API: Unload global BackgroundTaskManager
void
UnloadBackgroundTaskMgr()
{
if( g_BGTaskMgr )
{
//BUGBUG
//what to do with unfinished task?
delete g_BGTaskMgr;
}
g_BGTaskMgr = NULL;
}
// API: Select thread notifis now is a good time to do background task
DWORD
NotifyBackgroundTaskMgr()
{
DWORD error;
// can we run another background item?
if( !g_BGTaskMgr->HasBandwidth() )
{
error = ERROR_SUCCESS;
goto quit;
}
// get a background FSM if there is any
g_BGTaskMgr->DeQueueAndRunBackgroundWorkItem();
error = ERROR_SUCCESS;
quit:
return error;
}
// create a background task (fsm) and queue it on the background task
// list, the task item will be picked up later by a free async worker
// thread
DWORD
CreateAndQueueBackgroundWorkItem(
IN LPCSTR szUrl
)
{
DEBUG_ENTER((DBG_ASYNC,
Dword,
"CreateAndQueueBackgroundWorkItem",
"%q",
szUrl
));
DWORD error;
CFsm* pFsm = NULL;
INET_ASSERT( szUrl );
// get new fsm
pFsm = g_BGTaskMgr->CreateBackgroundFsm(szUrl);
if( !pFsm )
{
error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
// queue fsm
error = g_BGTaskMgr->QueueBackgroundWorkItem(pFsm);
if( error != ERROR_SUCCESS )
{
// delete the fsm to avoid leak
delete pFsm;
}
quit:
DEBUG_LEAVE(error);
return error;
}
DWORD
BackgroundTaskMgr::QueueBackgroundWorkItem(
IN CFsm* pFsm
)
{
DEBUG_ENTER((DBG_ASYNC,
Dword,
"ICAsyncThread::QueueBackgroundWorkItem",
"%#x",
pFsm
));
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
DWORD error = ERROR_INTERNET_INTERNAL_ERROR;
INET_ASSERT(lpThreadInfo != NULL);
if( lpThreadInfo != NULL )
{
_bgTaskQueue.Acquire();
_bgTaskQueue.Insert((CPriorityListEntry *)pFsm->List());
lpThreadInfo->Fsm = NULL;
_bgTaskQueue.Release();
error = ERROR_SUCCESS;
}
DEBUG_LEAVE(error);
return error;
}
DWORD
BackgroundTaskMgr::DeQueueAndRunBackgroundWorkItem()
{
// LOCK list
_bgTaskQueue.Acquire();
// check the list to see if it is empty
if( _bgTaskQueue.Head() != _bgTaskQueue.Self() )
{
PLIST_ENTRY pEntry = NULL;
PLIST_ENTRY pPrev = NULL;
pPrev = _bgTaskQueue.Self();
pEntry = ((CPriorityListEntry*)pPrev)->Next();
CFsm* pFsm = ContainingFsm(pEntry);
INET_ASSERT(pFsm);
// deQueued, we can remove the task item from the queue
// remove from the blocked list
_bgTaskQueue.Remove((CPriorityListEntry *)pFsm);
// increment the active running fsm count
InterlockedIncrement(&_lActiveFsm);
// this fsm will be picked up by a waken worker thread
pFsm->QueueWorkItem();
}
// UNLOCK list
_bgTaskQueue.Release();
return ERROR_SUCCESS;
}
CFsm*
BackgroundTaskMgr::CreateBackgroundFsm(LPCSTR szUrl)
{
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
// HACK - HACK for TESTING...
// correct solution is when lpThreadInfo->Fsm != NULL
// we do not create a fsm so that we always make sure
// there is one fsm on the select thread
if( lpThreadInfo != NULL )
{
lpThreadInfo->Fsm = NULL;
}
return new CFsm_BackgroundTask(this, szUrl);
}
BOOL
BackgroundTaskMgr::HasBandwidth()
{
// only one fsm can be picked at anytime
return !_lActiveFsm;
}
BackgroundTaskMgr::BackgroundTaskMgr()
: _lActiveFsm(0)
{
}
void
BackgroundTaskMgr::NotifyFsmDone()
{
InterlockedDecrement(&_lActiveFsm);
}
DWORD
CFsm_BackgroundTask::RunSM(
IN CFsm * Fsm
)
{
DEBUG_ENTER((DBG_ASYNC,
Dword,
"CFsm_BackgroundTask::RunSM",
"%#x",
Fsm
));
CFsm_BackgroundTask* fsm = (CFsm_BackgroundTask*) Fsm;
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
BOOL fIsAsyncWorkerThread = TRUE;
if( lpThreadInfo )
{
fIsAsyncWorkerThread = lpThreadInfo->IsAsyncWorkerThread;
lpThreadInfo->IsAsyncWorkerThread = FALSE;
}
switch( fsm->GetState() ) {
case FSM_STATE_INIT:
case FSM_STATE_CONTINUE:
fsm->DoSendReq();
default:
break;
}
fsm->SetDone(0);
if( lpThreadInfo )
{
lpThreadInfo->IsAsyncWorkerThread = fIsAsyncWorkerThread;
}
DEBUG_LEAVE(0);
return 0;
}
typedef HRESULT (WINAPI * pfnObtainUA)(DWORD, LPSTR, DWORD*);
DWORD
CFsm_BackgroundTask::DoSendReq()
{
HINTERNET hInternet = NULL;
HINTERNET hRequest = NULL;
CHAR szBuffer[4000];
DWORD dwBytesRead;
BOOL fSuccess;
DWORD error;
BOOL fUAFromUrlmon = FALSE;
CHAR* pszUA = NULL;
HINSTANCE hinst = GetModuleHandle("urlmon.dll");
if( hinst )
{
pfnObtainUA pfnUA = (pfnObtainUA)
GetProcAddress(hinst, "ObtainUserAgentString");
if( pfnUA )
{
DWORD dwSize = MAX_PATH;
pszUA = new CHAR[dwSize];
if( pszUA )
{
HRESULT hr = (*pfnUA)(0, pszUA, &dwSize);
if( S_OK == hr )
{
fUAFromUrlmon = TRUE;
}
else if( E_OUTOFMEMORY == hr )
{
// the original pszUA is allocated too small
// we need bigger buffer size (returned by dwSize)
delete [] pszUA;
pszUA = new CHAR[dwSize];
if( pszUA )
{
hr = (*pfnUA)(0, pszUA, &dwSize);
if( S_OK == hr )
{
fUAFromUrlmon = TRUE;
}
}
} // original buffer too small, create bigger one
}
} // get the pFN to UA agent
} // urlmon.dll is loaded
if( fUAFromUrlmon && pszUA)
{
hInternet = InternetOpen(
pszUA,
INTERNET_OPEN_TYPE_PRECONFIG,
NULL,
NULL,
0 );
}
else
{
hInternet = InternetOpen(
"Mozilla/4.0 (compatible; MSIE 5.01; Win32)",
INTERNET_OPEN_TYPE_PRECONFIG,
NULL,
NULL,
0 );
}
if( !hInternet )
{
goto quit;
}
hRequest = InternetOpenUrl(
hInternet,
m_lpszUrl,
"Accept: */*\r\n",
(DWORD) -1,
INTERNET_FLAG_RESYNCHRONIZE | INTERNET_FLAG_BGUPDATE | INTERNET_FLAG_KEEP_CONNECTION,
INTERNET_NO_CALLBACK
);
if( !hRequest )
{
DWORD dwLastErr = 0;
dwLastErr = GetLastError();
goto quit;
}
do {
dwBytesRead = 0;
fSuccess = InternetReadFile(
hRequest,
szBuffer,
sizeof(szBuffer)-1,
&dwBytesRead
);
if( !fSuccess )
{
goto quit;
}
} while ( dwBytesRead != 0 );
error = ERROR_SUCCESS;
quit:
if( hRequest )
InternetCloseHandle(hRequest);
if( hInternet )
InternetCloseHandle(hInternet);
if( fUAFromUrlmon && pszUA )
delete [] pszUA;
return error;
}
CFsm_BackgroundTask::~CFsm_BackgroundTask()
{
DELETE_MANDATORY_PARAM(m_lpszUrl);
m_pMgr->NotifyFsmDone();
}