/*=================================================================== Microsoft IIS 5.0 (ASP) Microsoft Confidential. Copyright 1998 Microsoft Corporation. All Rights Reserved. Component: 449 negotiations w/IE File: ie449.cpp Owner: DmitryR This file contains the implementation of the 449 negotiations w/IE ===================================================================*/ #include "denpre.h" #pragma hdrstop #include "ie449.h" #include "memchk.h" /*=================================================================== Globals ===================================================================*/ C449FileMgr *m_p449FileMgr = NULL; /*=================================================================== Internal funcitons ===================================================================*/ inline BOOL FindCookie ( char *szCookiesBuf, char *szCookie, DWORD cbCookie ) { char *pch = szCookiesBuf; if (pch == NULL || *pch == '\0') return FALSE; while (1) { if (strnicmp(pch, szCookie, cbCookie) == 0) { if (pch[cbCookie] == '=') // next char must be '=' return TRUE; } // next cookie pch = strchr(pch, ';'); if (pch == NULL) break; while (*(++pch) == ' ') // skip ; and any spaces ; } return FALSE; } /*=================================================================== The API ===================================================================*/ /*=================================================================== Init449 ===================================================================*/ HRESULT Init449() { // init hash table m_p449FileMgr = new C449FileMgr; if (m_p449FileMgr == NULL) return E_OUTOFMEMORY; HRESULT hr = m_p449FileMgr->Init(); if (FAILED(hr)) { delete m_p449FileMgr; m_p449FileMgr = NULL; return hr; } return S_OK; } /*=================================================================== UnInit449 ===================================================================*/ HRESULT UnInit449() { if (m_p449FileMgr != NULL) { delete m_p449FileMgr; m_p449FileMgr = NULL; } return S_OK; } /*=================================================================== Create449Cookie Get an existing 449 cookie from cache or create a new one Parameters szName cookie name szFile script file pp449 [out] the cookie Returns HRESULT ===================================================================*/ HRESULT Create449Cookie ( char *szName, TCHAR *szFile, C449Cookie **pp449 ) { HRESULT hr = S_OK; // Get the file first C449File *pFile = NULL; hr = m_p449FileMgr->GetFile(szFile, &pFile); if (FAILED(hr)) return hr; // Create the cookie hr = C449Cookie::Create449Cookie(szName, pFile, pp449); if (FAILED(hr)) pFile->Release(); // GetFile gave it addref'd return hr; } /*=================================================================== Do449Processing Check if the browser is IE5+ there's no echo-reply: header all the cookies are present Construct and send 449 response if needed When the response is sent, HitObj is marked as 'done with session' Parameters pHitObj the request rgpCookies array of cookie requirements cCookies number of cookie requirements Returns HRESULT ===================================================================*/ HRESULT Do449Processing ( CHitObj *pHitObj, C449Cookie **rgpCookies, DWORD cCookies ) { HRESULT hr = S_OK; if (cCookies == 0) return hr; ////////////////////////////////////////// // check the browser BOOL fBrowser = FALSE; char *szBrowser = pHitObj->PIReq()->QueryPszUserAgent(); if (szBrowser == NULL || szBrowser[0] == '\0') return S_OK; // bad browser char *szMSIE = strstr(szBrowser, "MSIE "); if (szMSIE) { char chVersion = szMSIE[5]; if (chVersion >= '5' && chVersion <= '9') fBrowser = TRUE; } #ifdef TINYGET449 if (strcmp(szBrowser, "WBCLI") == 0) fBrowser = TRUE; #endif if (!fBrowser) // bad browser return S_OK; ////////////////////////////////////////// // check the cookies char *szCookie = pHitObj->PIReq()->QueryPszCookie(); // collect the arrays of pointers and sizes for not found cookies. // arrays size to at most number of cookies in the template. DWORD cNotFound = 0; DWORD cbNotFound = 0; STACK_BUFFER( tempCookies, 128 ); if (!tempCookies.Resize(cCookies * sizeof(WSABUF))) { return E_OUTOFMEMORY; } LPWSABUF rgpNotFound = (LPWSABUF)tempCookies.QueryPtr(); for (DWORD i = 0; SUCCEEDED(hr) && (i < cCookies); i++) { if (!FindCookie(szCookie, rgpCookies[i]->SzCookie(), rgpCookies[i]->CbCookie())) { // cookie not found -- append to the list hr = rgpCookies[i]->LoadFile(); if (SUCCEEDED(hr)) // ignore bad files { rgpNotFound[cNotFound].buf = rgpCookies[i]->SzBuffer(); rgpNotFound[cNotFound].len = rgpCookies[i]->CbBuffer(); cbNotFound += rgpCookies[i]->CbBuffer(); cNotFound++; } } } if (!SUCCEEDED(hr)) return hr; if (cNotFound == 0) return S_OK; // everything's found ////////////////////////////////////////// // check echo-reply header char szEcho[80]; DWORD dwEchoLen = sizeof(szEcho); if (pHitObj->PIReq()->GetServerVariableA("HTTP_MS_ECHO_REPLY", szEcho, &dwEchoLen) || GetLastError() == ERROR_INSUFFICIENT_BUFFER) { return S_OK; // already in response cycle } ////////////////////////////////////////// // send the 449 response CResponse::WriteBlocksResponse ( pHitObj->PIReq(), // WAM_EXEC_INFO cNotFound, // number of blocks rgpNotFound, // array of blocks cbNotFound, // total number of bytes in blocks NULL, // text/html "449 Retry with", // status "ms-echo-request: execute opaque=\"0\" location=\"BODY\"\r\n" // extra header ); ////////////////////////////////////////// // tell HitObj no to write anything else pHitObj->SetDoneWithSession(); return S_OK; } /*=================================================================== Do449ChangeNotification Change notification processing Parameters szFile file changed or NULL for all Returns HRESULT ===================================================================*/ HRESULT Do449ChangeNotification ( TCHAR *szFile ) { // if m_p449FileMgr is NULL, we're likely getting called after // the 449 manager has been UnInited. Return S_OK in this case. if (m_p449FileMgr == NULL) return S_OK; if (szFile) return m_p449FileMgr->Flush(szFile); else return m_p449FileMgr->FlushAll(); } /*=================================================================== Class C449File ===================================================================*/ /*=================================================================== C449File::C449File Constructor ===================================================================*/ C449File::C449File() { m_cRefs = 0; m_fNeedLoad = 1; m_szFile = NULL; m_szBuffer = NULL; m_cbBuffer = 0; m_pDME = NULL; m_hFileReadyForUse=NULL; m_hrLoadResult= E_FAIL; } /*=================================================================== C449File::~C449File Destructor ===================================================================*/ C449File::~C449File() { Assert(m_cRefs == 0); if (m_szFile) free(m_szFile); if (m_szBuffer) free(m_szBuffer); if (m_pDME) m_pDME->Release(); if(m_hFileReadyForUse != NULL) CloseHandle(m_hFileReadyForUse); } /*=================================================================== C449File::Init Init strings, Load file for the first time, Start change notifications ===================================================================*/ HRESULT C449File::Init ( TCHAR *szFile ) { // remember the name m_szFile = StringDup(szFile); if (m_szFile == NULL) return E_OUTOFMEMORY; // init link element CLinkElem::Init(m_szFile, _tcslen(m_szFile)*sizeof(TCHAR)); // Create event: manual-reset, ready-for-use event; non-signaled // Better create event before using it. m_hFileReadyForUse = IIS_CREATE_EVENT( "C449File::m_hFileReadyForUse", this, TRUE, // flag for manual-reset event FALSE // flag for initial state ); if (!m_hFileReadyForUse) return E_OUTOFMEMORY; // load m_fNeedLoad = 1; HRESULT hr = Load(); if (FAILED(hr)) return hr; // start directory notifications TCHAR *pch = _tcsrchr(m_szFile, _T('\\')); // last backslash if (pch == NULL) return E_FAIL; // bogus filename? CASPDirMonitorEntry *pDME = NULL; *pch = _T('\0'); RegisterASPDirMonitorEntry(m_szFile, &pDME); *pch = _T('\\'); m_pDME = pDME; // done return S_OK; } /*=================================================================== C449File::Load Load the file if needed ===================================================================*/ HRESULT C449File::Load() { HRESULT hr = S_OK; HANDLE hFile = INVALID_HANDLE_VALUE; HANDLE hMap = NULL; BYTE *pbBytes = NULL; DWORD dwSize = 0; // check if this thread needs to load the file if (InterlockedExchange(&m_fNeedLoad, 0) == 0) { WaitForSingleObject(m_hFileReadyForUse, INFINITE); return m_hrLoadResult; } // cleanup the existing data if any if (m_szBuffer) free(m_szBuffer); m_szBuffer = NULL; m_cbBuffer = 0; // open the file if (SUCCEEDED(hr)) { hFile = AspCreateFile( m_szFile, GENERIC_READ, // access (read-write) mode FILE_SHARE_READ, // share mode NULL, // pointer to security descriptor OPEN_EXISTING, // how to create FILE_ATTRIBUTE_NORMAL, // file attributes NULL // handle to file with attributes to copy ); if (hFile == INVALID_HANDLE_VALUE) hr = E_FAIL; } // get file size if (SUCCEEDED(hr)) { dwSize = GetFileSize(hFile, NULL); if (dwSize == 0 || dwSize == 0xFFFFFFFF) hr = E_FAIL; } // create mapping if (SUCCEEDED(hr)) { hMap = CreateFileMapping( hFile, // handle to file to map NULL, // optional security attributes PAGE_READONLY, // protection for mapping object 0, // high-order 32 bits of object size 0, // low-order 32 bits of object size NULL // name of file-mapping object ); if (hMap == NULL) hr = E_FAIL; } // map the file if (SUCCEEDED(hr)) { pbBytes = (BYTE *)MapViewOfFile( hMap, // file-mapping object to map into address space FILE_MAP_READ, // access mode 0, // high-order 32 bits of file offset 0, // low-order 32 bits of file offset 0 // number of bytes to map ); if (pbBytes == NULL) hr = E_FAIL; } // remember the bytes if (SUCCEEDED(hr)) { m_szBuffer = (char *)malloc(dwSize); if (m_szBuffer != NULL) { memcpy(m_szBuffer, pbBytes, dwSize); m_cbBuffer = dwSize; } else { hr = E_OUTOFMEMORY; } } // cleanup if (pbBytes != NULL) UnmapViewOfFile(pbBytes); if (hMap != NULL) CloseHandle(hMap); if (hFile != INVALID_HANDLE_VALUE) CloseHandle(hFile); // Set the Need Load flag or flag the read event as successful. m_hrLoadResult = hr; SetEvent(m_hFileReadyForUse); return m_hrLoadResult; } /*=================================================================== C449File::Create449File static constructor ===================================================================*/ HRESULT C449File::Create449File ( TCHAR *szFile, C449File **ppFile ) { HRESULT hr = S_OK; C449File *pFile = new C449File; if (pFile == NULL) hr = E_OUTOFMEMORY; if (SUCCEEDED(hr)) { hr = pFile->Init(szFile); } if (SUCCEEDED(hr)) { pFile->AddRef(); *ppFile = pFile; } else if (pFile) { delete pFile; } return hr; } /*=================================================================== C449File::QueryInterface C449File::AddRef C449File::Release IUnknown members for C449File object. ===================================================================*/ STDMETHODIMP C449File::QueryInterface(REFIID riid, VOID **ppv) { // should never be called Assert(FALSE); *ppv = NULL; return E_NOINTERFACE; } STDMETHODIMP_(ULONG) C449File::AddRef() { return InterlockedIncrement(&m_cRefs); } STDMETHODIMP_(ULONG) C449File::Release() { LONG cRefs = InterlockedDecrement(&m_cRefs); if (cRefs) return cRefs; delete this; return 0; } /*=================================================================== Class C449FileMgr ===================================================================*/ /*=================================================================== C449FileMgr::C449FileMgr Constructor ===================================================================*/ C449FileMgr::C449FileMgr() { INITIALIZE_CRITICAL_SECTION(&m_csLock); } /*=================================================================== C449FileMgr::~C449FileMgr Destructor ===================================================================*/ C449FileMgr::~C449FileMgr() { FlushAll(); m_ht449Files.UnInit(); DeleteCriticalSection(&m_csLock); } /*=================================================================== C449FileMgr::Init Initialization ===================================================================*/ HRESULT C449FileMgr::Init() { return m_ht449Files.Init(199); } /*=================================================================== C449FileMgr::GetFile Find file in the hash table, or create a new one ===================================================================*/ HRESULT C449FileMgr::GetFile ( TCHAR *szFile, C449File **ppFile ) { C449File *pFile = NULL; CLinkElem *pElem; Lock(); pElem = m_ht449Files.FindElem(szFile, _tcslen(szFile)*sizeof(TCHAR)); if (pElem) { // found pFile = static_cast(pElem); if (!SUCCEEDED(pFile->Load())) pFile = NULL; else pFile->AddRef(); // 1 ref to hand out } else if (SUCCEEDED(C449File::Create449File(szFile, &pFile))) { if (m_ht449Files.AddElem(pFile)) pFile->AddRef(); // 1 for hash table + 1 to hand out } UnLock(); *ppFile = pFile; return (pFile != NULL) ? S_OK : E_FAIL; } /*=================================================================== C449FileMgr::Flush Change notification for a single file ===================================================================*/ HRESULT C449FileMgr::Flush ( TCHAR *szFile ) { Lock(); CLinkElem *pElem = m_ht449Files.FindElem(szFile, _tcslen(szFile)*sizeof(TCHAR)); if (pElem) { C449File *pFile = static_cast(pElem); pFile->SetNeedLoad(); // next time reload } UnLock(); return S_OK; } /*=================================================================== C449FileMgr::FlushAll Remove all files FlushAll is always together with template flush ===================================================================*/ HRESULT C449FileMgr::FlushAll() { // Unlink from hash table first Lock(); CLinkElem *pElem = m_ht449Files.Head(); m_ht449Files.ReInit(); UnLock(); // Walk the list to remove all while (pElem) { C449File *pFile = static_cast(pElem); pElem = pElem->m_pNext; pFile->Release(); } return S_OK; } /*=================================================================== Class C449Cookie ===================================================================*/ /*=================================================================== C449Cookie::C449Cookie Constructor ===================================================================*/ C449Cookie::C449Cookie() { m_cRefs = 0; m_szName = NULL; m_cbName = 0; m_pFile = NULL; } /*=================================================================== C449Cookie::~C449Cookie Destructor ===================================================================*/ C449Cookie::~C449Cookie() { Assert(m_cRefs == 0); if (m_szName) free(m_szName); if (m_pFile) m_pFile->Release(); } /*=================================================================== C449Cookie::Init Initialize ===================================================================*/ HRESULT C449Cookie::Init ( char *szName, C449File *pFile ) { m_szName = StringDupA(szName); if (m_szName == NULL) return E_OUTOFMEMORY; m_cbName = strlen(m_szName); m_pFile = pFile; return S_OK; } /*=================================================================== C449Cookie::Create449Cookie static constructor ===================================================================*/ HRESULT C449Cookie::Create449Cookie ( char *szName, C449File *pFile, C449Cookie **pp449Cookie ) { HRESULT hr = S_OK; C449Cookie *pCookie = new C449Cookie; if (pCookie == NULL) hr = E_OUTOFMEMORY; if (SUCCEEDED(hr)) { hr = pCookie->Init(szName, pFile); } if (SUCCEEDED(hr)) { pCookie->AddRef(); *pp449Cookie = pCookie; } else if (pCookie) { delete pCookie; } return hr; } /*=================================================================== C449Cookie::QueryInterface C449Cookie::AddRef C449Cookie::Release IUnknown members for C449Cookie object. ===================================================================*/ STDMETHODIMP C449Cookie::QueryInterface(REFIID riid, VOID **ppv) { // should never be called Assert(FALSE); *ppv = NULL; return E_NOINTERFACE; } STDMETHODIMP_(ULONG) C449Cookie::AddRef() { return InterlockedIncrement(&m_cRefs); } STDMETHODIMP_(ULONG) C449Cookie::Release() { LONG cRefs = InterlockedDecrement(&m_cRefs); if (cRefs) return cRefs; delete this; return 0; }