Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

10573 lines
346 KiB

//depot/private/jasbr/inetsrv/iis/svcs/cmp/asp/template.cpp#19 - edit change 3548 (text)
/*==============================================================================
Microsoft Denali
Microsoft Confidential.
Copyright 1996 Microsoft Corporation. All Rights Reserved.
File: template.cpp
Maintained by: DaveK
Component: source file for Denali Compiled Template object
==============================================================================*/
#include "denpre.h"
#pragma hdrstop
const int SNIPPET_SIZE = 20; // # of characters in the code snippets
#pragma warning( disable : 4509 ) // suppress SEH/destructor warnings
#pragma warning( disable : 4355 ) // ignore: "'this' used in base member init
#include "debugger.h"
#include "dbgutil.h"
#include "tlbcache.h"
#include "ie449.h"
#include "memchk.h"
#include "vecimpl.h" // Include after memchk to insure that vector uses our mem manager.
#include "Accctrl.h"
#include "aclapi.h"
// Init class statics
CTemplate::CTokenList *CTemplate::gm_pTokenList = NULL;
PTRACE_LOG CTemplate::gm_pTraceLog = NULL;
HANDLE CTemplate::sm_hSmallHeap = NULL;
HANDLE CTemplate::sm_hLargeHeap = NULL;
// Max # of opener tokens to look for
#define TOKEN_OPENERS_MAX 8
// Expose AspDoRevertHack and AspUndoRevertHack so that it can be used in template.cpp
extern VOID AspDoRevertHack( HANDLE * phToken );
extern VOID AspUndoRevertHack( HANDLE * phToken );
/*===================================================================
Private non-class support functions
===================================================================*/
static void ByteRangeFromPb(BYTE* pbSource, CByteRange& brTarget);
static BOOLB FByteRangesAreEqual(CByteRange& br1, CByteRange& br2);
static unsigned CharAdvDBCS(WORD wCodePage, char *pchStart, char *pchEnd, unsigned cCharAdv, char **ppchEnd, BOOL fForceDBCS = FALSE);
static void LineFromByteRangeAdv(CByteRange& br, CByteRange& brLine);
static void LTrimWhiteSpace(CByteRange& br);
static void RTrimWhiteSpace(CByteRange& br);
static CByteRange BrNewLine(CByteRange br);
static BOOLB FWhiteSpace(char ch, BOOLB fSpaceIsWhiteSpace = TRUE);
static BOOLB FByteRangeIsWhiteSpace(CByteRange br);
static BOOLB FTagName(BYTE* pb, UINT cb);
static void ByteAlignOffset(UINT* pcbOffset, UINT cbAlignment);
static void GetSzFromPatternInserts(char* pszPattern, UINT cInserts, char** ppszInserts, char* szReturned);
static UINT CchPathFromFilespec(LPCTSTR szFile);
static void GetPathFromParentAndFilespec(LPCTSTR szParentPath, LPCTSTR szFileSpec, LPTSTR* pszPath);
static void HandleAccessFailure(CHitObj* pHitObj, TCHAR* szFile);
static void SendToLog(DWORD dwMask, CHAR *szFileName, CHAR *szLineNum, CHAR *szShortDes, CHAR *szLongDes, CHAR *szEngine, CHitObj *pHitObj);
static HRESULT GetProgLangId(CByteRange& brEngine, PROGLANG_ID* pProgLangId);
inline
void __cdecl DebugPrintf(LPCSTR fmt, ...)
{
#if DBG
char msg[512];
va_list marker;
va_start(marker, fmt);
vsprintf(msg, fmt, marker);
va_end(marker);
OutputDebugStringA(msg);
#endif
}
/* ============================================================================
ByteRangeFromPb
Gets a byte range from a contiguous block of memory
Returns:
Nothing.
Side effects:
None.
*/
void
ByteRangeFromPb
(
BYTE* pbSource,
CByteRange& brTarget
)
{
Assert(pbSource != NULL);
brTarget.m_cb = *(ULONG*)pbSource;
brTarget.m_pb = pbSource + sizeof(ULONG);
}
/* ============================================================================
FByteRangesAreEqual
Compares two byte ranges
Returns:
BOOLB. True if byte ranges are equal, else false.
Side effects:
None.
*/
BOOLB
FByteRangesAreEqual
(
CByteRange& br1,
CByteRange& br2
)
{
if(br1.m_cb != br2.m_cb)
return FALSE;
return (!_strnicmp((LPCSTR)br1.m_pb, (LPCSTR)br2.m_pb, br1.m_cb));
}
/* ============================================================================
CharAdvDBCS
Advance "cchCharAdv" characters in a buffer
SBCS: Degenerates to simple pointer arithmatic
Arguments:
wCodePage - code page
pchStart - pointer to beginning of segment
pchEnd - pointer to just past end of segment
cCharAdv - # of characters to advance
ppchEnd - [output], contains pointer "cCharAdv" chars past pchStart
fForceDBCS - if TRUE, always use double byte algorithm.
(for verifying correct behavior of func in debug mode)
Returns:
(int) # of characters that we actually advanced
Notes:
By passing INFINITE for "cCharAdv", you can use this function to count characters
in a block
Side effects:
None.
*/
unsigned
CharAdvDBCS
(
WORD wCodePage,
char *pchStart,
char *pchEnd,
unsigned cCharAdv,
char **ppchEnd,
BOOL fForceDBCS
)
{
CPINFO CpInfo;
GetCPInfo(wCodePage, &CpInfo);
if (!fForceDBCS && CpInfo.MaxCharSize == 1)
{
char *pchT = pchStart + min(cCharAdv, unsigned(pchEnd - pchStart));
if (ppchEnd)
*ppchEnd = pchT;
#if DBG
// Verify DBCS algorithm (not often tested otherwise)
char *pchTest;
unsigned cchTest = CharAdvDBCS(wCodePage, pchStart, pchEnd, cCharAdv, &pchTest, TRUE);
Assert (cchTest == unsigned(pchT - pchStart) && pchTest == pchT);
#endif
return DIFF(pchT - pchStart);
}
else
{
int cch = 0;
char *pchNext = pchStart;
// Count DBCS characters. We have to stop before pchEnd because
// pchEnd may point past file map and CharNextExA AVs when advancing
// past allocated memory
while (cCharAdv > 0 && pchNext < pchEnd-2)
{
pchNext = *pchNext? AspCharNextA(wCodePage, pchNext) : pchNext + 1;
--cCharAdv;
++cch;
}
// We could stop on the last or the before last character
// depending on the DBCS char sequence
if (cCharAdv > 0 && pchNext == pchEnd-1)
{
// Only one byte - has to be one single byte character
++pchNext;
++cch;
}
else if (cCharAdv > 0 && pchNext == pchEnd-2)
{
// 2 bytes left - either 1 2-byte char or 2 1-byte chars
if (IsDBCSLeadByteEx(wCodePage, *pchNext))
{
++cch;
pchNext += 2;
}
else
{
// Two characters left. If cCharAdv > 1, this means that user wants to
// advance at least two more chars. Otherwise, cCharAdv == 1, and
// we advance one char
//
if (cCharAdv > 1)
{
cch += 2;
pchNext += 2;
}
else
{
Assert (cCharAdv == 1);
++cch;
++pchNext;
}
}
}
if (ppchEnd)
*ppchEnd = pchNext;
return cch;
}
}
/* ============================================================================
LineFromByteRangeAdv
Gets the first line in a byte range.
Returns:
Nothing
Side effects:
Advances source byte range to just beyond its first non-white-space line,
if one is found.
*/
void
LineFromByteRangeAdv
(
CByteRange& brSource,
CByteRange& brLine
)
{
CByteRange brTemp;
if(brSource.IsNull())
{
brLine.Nullify();
return;
}
brLine.m_pb = brSource.m_pb;
brTemp = BrNewLine(brSource);
if(brTemp.IsNull())
{
// We found no newline in a non-empty byte range:
// set line range to entire source byte range and empty source byte range
brLine.m_cb = brSource.m_cb;
brSource.Nullify();
}
else
{
// We found a newline in a non-empty byte range:
// set line range to portion of source byte range before new line;
// set source range to portion of source range after new line
brLine.m_cb = DIFF(brTemp.m_pb - brSource.m_pb);
brSource.m_pb = brTemp.m_pb + brTemp.m_cb;
brSource.m_cb -= (brLine.m_cb + brTemp.m_cb);
}
}
/* ============================================================================
LTrimWhiteSpace
Left-trim white space from byte-range
Returns:
Nothing
Side effects:
Advances byte range to just beyond its first non-white-space character.
*/
void
LTrimWhiteSpace
(
CByteRange& br
)
{
if(br.IsNull())
return;
while(FWhiteSpace(*br.m_pb))
{
br.m_pb++;
if(--br.m_cb == 0)
return;
}
}
/* ============================================================================
RTrimWhiteSpace
Right-trim white space from byte-range
*/
void
RTrimWhiteSpace(CByteRange& br)
{
if(br.IsNull())
return;
while(FWhiteSpace(*(br.m_pb + br.m_cb - 1)))
{
if(--br.m_cb == 0)
return;
}
}
/* ============================================================================
BrNewLine
Returns ptr to the first newline in a byte range
NOTE does not change byte range (since it is passed by value)
*/
CByteRange
BrNewLine(CByteRange br)
{
while(!br.IsNull())
{
if(*br.m_pb == '\r')
return CByteRange(br.m_pb, (br.m_cb > 1 && br.m_pb[1] == '\n')? 2 : 1);
else if (*br.m_pb == '\n')
return CByteRange(br.m_pb, 1);
++br.m_pb;
--br.m_cb;
}
return CByteRange();
}
/* ============================================================================
FWhiteSpace
Returns:
TRUE if ch is a white-space character, else returns FALSE
Certain character(s) (e.g. space) may be treated as
non-white-space; to do this, caller passes FALSE for
fSpaceIsWhiteSpace flag.
*/
BOOLB
FWhiteSpace(char ch, BOOLB fSpaceIsWhiteSpace)
{
switch (ch)
{
case ' ':
return fSpaceIsWhiteSpace;
case '\0':
return TRUE;
case '\a':
return TRUE;
case '\b':
return TRUE;
case '\f':
return TRUE;
case '\n':
return TRUE;
case '\r':
return TRUE;
case '\t':
return TRUE;
case '\v':
return TRUE;
default:
return FALSE;
}
}
/* ============================================================================
FByteRangeIsWhiteSpace
Is the entire input byte range white space?
NOTE input byte range is byval; caller's copy is not changed
*/
BOOLB
FByteRangeIsWhiteSpace(CByteRange br)
{
while(!br.IsNull())
{
if(!FWhiteSpace(*(br.m_pb)))
return FALSE;
br.Advance(1);
}
return TRUE;
}
/* ============================================================================
FTagName
Does pb point to a valid HTML tag name?
(i.e., is *pb a valid HTML tag name and not a substring?)
Returns
TRUE or FALSE
Side effects
None
*/
BOOLB
FTagName(BYTE* pb, UINT cb)
{
if((pb == NULL) || (cb == 0))
return FALSE;
// a valid HTML tag name must be preceded by white space ...
if( FWhiteSpace(*(pb - 1)) || *(pb - 1) == '@' )
{
// ... and followed either by white space or the tag separator
if(FWhiteSpace(*(pb + cb)))
return TRUE;
else if(*(pb + cb) == CH_ATTRIBUTE_SEPARATOR)
return TRUE;
}
return FALSE;
}
/*===================================================================
ByteAlignOffset
Byte-aligns an offset value, based on size of source data
*/
void
ByteAlignOffset
(
UINT* pcbOffset, // ptr to offset value
UINT cbAlignment // Alignment boundary
)
{
// comment the below code out so that it works for 64 bit...
// only byte-align for 2-, or 4-byte data
// since our base pointer in only aligned to a 4 byte boundary
//if(cbAlignment == 2 || cbAlignment == 4)
//{
// if current offset does not fall on a byte-aligned location for current data type,
// advance offset to next byte-aligned location
Assert(cbAlignment > 0);
--cbAlignment;
if (*pcbOffset & cbAlignment)
*pcbOffset = (*pcbOffset + cbAlignment + 1) & ~cbAlignment;
}
/* ============================================================================
GetSzFromPatternInserts
Returns a 'resolved' version of a pattern string, i.e. a new string in which
| characters have been replaced by caller-specified insert strings.
NOTE this function allocates, but caller must free
Returns:
Nothing
Side effects:
allocates memory
*/
void
GetSzFromPatternInserts
(
char* pszPattern, // 'pattern' string
UINT cInserts, // count of insert strings
char** ppszInserts, // array of ptrs to insert strings
char* szReturned // returned string MUST be allocated by caller
)
{
UINT cchRet = strlen(pszPattern); // length of return string
char* pchStartCopy = pszPattern; // ptr to start of copy range in pattern
char* pchEndCopy = pszPattern; // ptr to end of copy range in pattern
UINT cActualInserts = 0; // count of actual insert strings
// init return string to empty so we can concatenate onto it
szReturned[0] = NULL;
// zero out length of return string - we now use it to count actual length as we build return string
cchRet = 0;
while(TRUE)
{
// advance end-of-copy ptr through pattern looking for insertion points or end of string
while ((*pchEndCopy != NULL) && (IsDBCSLeadByte(*pchEndCopy) || (*pchEndCopy != '|')))
pchEndCopy = CharNextA(pchEndCopy);
// cat from start-of-copy to end-of-copy onto return string
strncat(szReturned, pchStartCopy, DIFF(pchEndCopy - pchStartCopy));
// update return string length
cchRet += DIFF(pchEndCopy - pchStartCopy);
// if we are at end of pattern, exit
if(*pchEndCopy == NULL)
goto Exit;
if(cActualInserts < cInserts)
{
// if inserts remain, cat the next one onto return string
strcat(szReturned, ppszInserts[cActualInserts]);
// update return string length
cchRet += strlen(ppszInserts[cActualInserts]);
cActualInserts++;
}
// advance end-of-copy and start-of-copy beyond insertion point
pchEndCopy++;
pchStartCopy = pchEndCopy;
}
Exit:
// null-terminate return string
szReturned[cchRet] = NULL;
}
/* ============================================================================
CchPathFromFilespec
Returns a filespec's path length (exclusive of filespec)
NOTE path string includes trailing '\' or '/'
Returns:
Length of path string
Side effects:
None
*/
UINT
CchPathFromFilespec
(
LPCTSTR szFileSpec // filespec
)
{
// BUG FIX 102010 DBCS fixes
//int ich = lstrlen(szFileSpec) - 1; // index of char to compare
//
//while(*(szFileSpec + ich) != '\\' && *(szFileSpec + ich) != '/')
// {
// if(--ich < 0)
// THROW(E_FAIL);
// }
//return (UINT) (ich + 1); // path length, including trailing '\' or '/', is char index + 1
TCHAR* p1 = _tcsrchr(szFileSpec, _T('\\'));
TCHAR* p2 = _tcsrchr(szFileSpec, _T('/')); // this wont be a DBCS trail byte.
if (p1 == NULL && p2 == NULL)
THROW(E_FAIL);
return (UINT) ((((LPTSTR)max(p1,p2) - szFileSpec)) + 1);
}
/* ============================================================================
GetPathFromParentAndFilespec
Returns an absolute path which is a 'parent' file's path concatenated with a filespec.
Returns:
absolute path (out-parameter)
Side effects:
None
*/
void
GetPathFromParentAndFilespec
(
LPCTSTR szParentPath, // parent path
LPCTSTR szFileSpec, // filespec
LPTSTR* pszPath // resolved path (out-parameter)
)
{
UINT cchParentPath = CchPathFromFilespec(szParentPath);
if ((cchParentPath + _tcslen(szFileSpec)) > MAX_PATH)
THROW(E_FAIL);
_tcsncpy(*pszPath, szParentPath, cchParentPath);
_tcscpy(*pszPath + cchParentPath, szFileSpec);
}
/* ============================================================================
HandleAccessFailure
Handles an access-denied failure
Returns:
nothing
Side effects:
none
*/
void
HandleAccessFailure
(
CHitObj* pHitObj, // browser's hitobj
TCHAR * szFile // file path of main template
)
{
Assert(pHitObj);
// debugging diagnostic print
#if DBG
STACK_BUFFER( authUserBuff, 32 );
char *szAuthUser;
DWORD cbAuthUser;
if (SERVER_GET(pHitObj->PIReq(), "AUTH_USER", &authUserBuff, &cbAuthUser)) {
szAuthUser = (char*)authUserBuff.QueryPtr();
}
else {
szAuthUser = "anonymous";
}
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "No permission to read file %S\n", szFile != NULL? szFile : pHitObj->PIReq()->QueryPszPathTranslated()));
#else
DBGPRINTF((DBG_CONTEXT, "No permission to read file %s\n", szFile != NULL? szFile : pHitObj->PIReq()->QueryPszPathTranslated()));
#endif
DBGPRINTF((DBG_CONTEXT, " The user account is \"%s\"\n", szAuthUser));
#endif
CResponse *pResponse = pHitObj->PResponse();
if (!pResponse)
return;
pHitObj->PIReq()->SetDwHttpStatusCode(401);
HandleSysError(401,3,IDE_401_3_ACCESS_DENIED,IDH_401_3_ACCESS_DENIED,pHitObj->PIReq(),pHitObj);
return;
}
/* ============================================================================
SendToLog
Sends Error Info to Log
Returns:
Nothing
Side effects:
None.
*/
void
SendToLog
(
DWORD dwMask,
CHAR *szFileName,
CHAR *szLineNum,
CHAR *szEngine,
CHAR *szErrCode,
CHAR *szShortDes,
CHAR *szLongDes,
CHitObj *pHitObj // browser's hitobj
)
{
CHAR *szFileNameT;
CHAR *szLineNumT;
CHAR *szEngineT;
CHAR *szErrCodeT;
CHAR *szShortDesT;
CHAR *szLongDesT;
if(pHitObj) {
// NOTE - szFileName is assumed to be UTF8 when UNICODE is defined
szFileNameT = StringDupA(szFileName);
szLineNumT = StringDupA(szLineNum);
szEngineT = StringDupA(szEngine);
szErrCodeT = StringDupA(szErrCode);
szShortDesT = StringDupA(szShortDes);
szLongDesT = StringDupA(szLongDes);
HandleError(szShortDesT, szLongDesT, dwMask, szFileNameT, szLineNumT, szEngineT, szErrCodeT, NULL, pHitObj);
}
}
/* ============================================================================
FreeNullify
Frees and nullifies a ptr to memory allocated with malloc.
Returns:
Nothing
Side effects:
None
*/
static void
FreeNullify
(
void** pp
)
{
if(*pp != NULL)
{
free(*pp);
*pp = NULL;
}
}
/* ============================================================================
SmallTemplateFreeNullify
Frees and nullifies a ptr to memory allocated with CTemplate::SmallMalloc.
Returns:
Nothing
Side effects:
None
*/
static void
SmallTemplateFreeNullify
(
void** pp
)
{
if(*pp != NULL)
{
CTemplate::SmallFree(*pp);
*pp = NULL;
}
}
/* ============================================================================
LargeTemplateFreeNullify
Frees and nullifies a ptr to memory allocated with CTemplate::LargeMalloc.
Returns:
Nothing
Side effects:
None
*/
static void
LargeTemplateFreeNullify
(
void** pp
)
{
if(*pp != NULL)
{
CTemplate::LargeFree(*pp);
*pp = NULL;
}
}
/* ============================================================================
GetProgLangId
Gets the prog lang id for a script engine
Returns:
Nothing
Side effects:
throws on error
*/
HRESULT
GetProgLangId
(
CByteRange& brEngine, // engine name
PROGLANG_ID* pProgLangId // prog lang id (out-parameter)
)
{
STACK_BUFFER( tempEngine, 128 );
if (!tempEngine.Resize(brEngine.m_cb + 1)) {
return E_OUTOFMEMORY;
}
LPSTR szProgLang = static_cast<LPSTR> (tempEngine.QueryPtr());
strncpy(szProgLang, (LPCSTR)brEngine.m_pb, brEngine.m_cb);
szProgLang[brEngine.m_cb] = '\0';
return g_ScriptManager.ProgLangIdOfLangName((LPCSTR) szProgLang, pProgLangId);
}
/* ****************************************************************************
CByteRange member functions
*/
/* ========================================================
CByteRange::Advance
Advances a byte range.
*/
void
CByteRange::Advance(UINT i)
{
if(i >= m_cb)
{
Nullify();
}
else
{
m_pb += i;
m_cb -= i;
}
}
/* ========================================================
CByteRange::FMatchesSz
Compares a byte range with a string, case-insensitively
*/
BOOLB
CByteRange::FMatchesSz
(
LPCSTR psz
)
{
if(IsNull() || (psz == NULL))
return FALSE;
if((ULONG)strlen(psz) != m_cb)
return FALSE;
return !_strnicmp((const char*)m_pb, psz, m_cb);
}
/* ============================================================================
CByteRange::PbString
Finds a case-insensitive string within a byte range
Returns:
Ptr to first case-insensitive occurrence of the string in this byte range;
NULL if none found.
Side effects:
None
*/
BYTE*
CByteRange::PbString
(
LPSTR psz,
LONG lCodePage
)
{
UINT cch = strlen(psz);
if(cch == 0)
return NULL;
BYTE *pbLocal = m_pb;
UINT cbLocal = m_cb;
char ch0 = psz[0];
BYTE *pbTemp = NULL;
UINT cbAdvanced = 0;
if (IsCharAlpha(ch0))
{
// cannot use strchr
while (cbLocal >= cch)
{
if (_strnicmp((const char *)pbLocal, psz, cch) == 0)
return pbLocal;
// The following code simply performs a DBCS-enabled ByteRange.Advance() action.
pbTemp = pbLocal;
pbLocal = *pbLocal? (BYTE *)AspCharNextA((WORD)lCodePage, (const char *)pbLocal) : pbLocal + 1;
cbAdvanced = DIFF(pbLocal - pbTemp);
if (cbAdvanced >= cbLocal)
{
cbLocal = 0;
pbLocal = NULL;
}
else
cbLocal -= cbAdvanced;
}
}
else
{
// can use strchr
while (cbLocal >= cch)
{
pbTemp = (BYTE *)memchr(pbLocal, ch0, cbLocal);
if (pbTemp == NULL)
break;
UINT cbOffset = DIFF(pbTemp - pbLocal);
if (cbOffset >= cbLocal)
break;
pbLocal = pbTemp;
cbLocal -= cbOffset;
if (cch <= cbLocal && _strnicmp((const char *)pbLocal, psz, cch) == 0)
return pbLocal;
// The following code simply performs a DBCS-enabled ByteRange.Advance() action.
pbTemp = pbLocal;
pbLocal = *pbLocal? (BYTE *)AspCharNextA((WORD)lCodePage, (const char *)pbLocal) : pbLocal + 1;
cbAdvanced = DIFF(pbLocal - pbTemp);
if (cbAdvanced >= cbLocal)
{
cbLocal = 0;
pbLocal = NULL;
}
else
cbLocal -= cbAdvanced;
}
}
return NULL;
}
/* ============================================================================
CByteRange::PbOneOfAspOpenerStringTokens
Finds a case-insensitive string within a byte range
that matches one of the strings passed
!!! WILL ONLY WORK IF THE FOLLOWING IS TRUE:
1) All the tokens start with the same charater (for example '<')
2) This character is not alpha (so that strchr() would work)
!!! THE ABOVE ASSUMPTIONS MAKE THE CODE WORK FASTER
Returns:
Ptr to first case-insensitive occurrence of the string in this byte range;
NULL if none found.
*pcindex is set to the index of string found
Side effects:
None
*/
BYTE*
CByteRange::PbOneOfAspOpenerStringTokens
(
LPSTR rgszTokens[],
UINT rgcchTokens[],
UINT nTokens,
UINT *pidToken
)
{
if (nTokens == 0)
return NULL;
BYTE *pb = m_pb; // pointer to unsearched remainder of the range
UINT cbRemainder = m_cb; // remaining byte range length
char ch0 = rgszTokens[0][0]; // first char of every token
while (cbRemainder > 0) {
// BUG 82331: avoid strchr() because byte range is not null-terminated
while (cbRemainder > 0 && *pb != ch0)
{
++pb;
--cbRemainder;
}
if (cbRemainder == 0)
break;
for (UINT i = 0; i < nTokens; i++) {
if ((rgcchTokens[i] <= cbRemainder)
&& (rgszTokens[i] != NULL)
&& (_strnicmp((const char *)pb, rgszTokens[i], rgcchTokens[i]) == 0)) {
*pidToken = i;
return pb;
}
}
++pb;
--cbRemainder;
}
return NULL;
}
/* ============================================================================
CByteRange::FEarlierInSourceThan
Does this byte range occur earlier in source than parameter byte range?
Returns
TRUE or FALSE
Side effects
None
*/
BOOLB
CByteRange::FEarlierInSourceThan(CByteRange& br)
{
if(br.IsNull())
return TRUE;
return(m_idSequence < br.m_idSequence);
}
/* ****************************************************************************
CTemplate member functions
*/
/* ============================================================================
CTemplate::InitClass
Initilaizes CTemplate static members
Returns:
hresult
Side effects:
allocates memory for static members
*/
HRESULT
CTemplate::InitClass
(
)
{
HRESULT hr = S_OK;
TRY
// init heaps
sm_hSmallHeap = ::HeapCreate(0, 0, 0);
sm_hLargeHeap = ::HeapCreate(0, 0, 0);
// Init token list
gm_pTokenList = new CTokenList;
if (gm_pTokenList == NULL)
return E_OUTOFMEMORY;
gm_pTokenList->Init();
CATCH(hrException)
hr = hrException;
END_TRY
return hr;
}
/* ============================================================================
CTemplate::UnInitClass
Un-initilaizes CTemplate static members
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::UnInitClass()
{
delete gm_pTokenList;
gm_pTokenList = NULL;
::HeapDestroy(sm_hLargeHeap);
if (sm_hLargeHeap != sm_hSmallHeap)
::HeapDestroy(sm_hSmallHeap);
sm_hLargeHeap = sm_hSmallHeap = NULL;
}
/* ============================================================================
CTemplate::Init
Inits template in preparation for calling Compile
Does the minimum needed
Returns:
Success or failure code
Side effects:
Allocates memory
*/
HRESULT
CTemplate::Init
(
CHitObj *pHitObj, // ptr to template's hit object
BOOL fGlobalAsa, // is this the global.asa file?
const CTemplateKey &rTemplateKey // hash table key
)
{
HRESULT hr;
// Create debug critical section
ErrInitCriticalSection(&m_csDebuggerDetach, hr);
if (FAILED(hr))
return hr;
// note critical section creation success
m_fDebuggerDetachCSInited = TRUE;
// Create event: manual-reset, ready-for-use event; non-signaled
m_hEventReadyForUse = IIS_CREATE_EVENT(
"CTemplate::m_hEventReadyForUse",
this,
TRUE, // flag for manual-reset event
FALSE // flag for initial state
);
if (!m_hEventReadyForUse)
return E_OUTOFMEMORY;
// cache GlobalAsp flag
m_fGlobalAsa = BOOLB(fGlobalAsa);
// CIsapiReqInfo better be present
if (pHitObj->PIReq() == NULL)
return E_POINTER;
// Initialize the template's code page
m_wCodePage = pHitObj->PAppln()->QueryAppConfig()->uCodePage();
m_lLCID = pHitObj->PAppln()->QueryAppConfig()->uLCID();
STACK_BUFFER( serverNameBuff, 32 );
STACK_BUFFER( serverPortBuff, 10 );
STACK_BUFFER( portSecureBuff, 8 );
DWORD cbServerName;
DWORD cbServerPort;
DWORD cbServerPortSecure;
// Construct a URL for the application
// Get the server name and port
if (!SERVER_GET(pHitObj->PIReq(), "SERVER_NAME", &serverNameBuff, &cbServerName)
|| !SERVER_GET(pHitObj->PIReq(), "SERVER_PORT", &serverPortBuff, &cbServerPort)) {
if (GetLastError() == E_OUTOFMEMORY) {
hr = E_OUTOFMEMORY;
}
else {
hr = E_FAIL;
}
return hr;
}
char *szServerPort = (char *)serverPortBuff.QueryPtr();
char *szServerName = (char *)serverNameBuff.QueryPtr();
BOOL fServerPortSecure = FALSE;
// determine if server port is secure
if (SERVER_GET(pHitObj->PIReq(), "SERVER_PORT_SECURE", &portSecureBuff, &cbServerPortSecure)) {
char *szServerPortSecure = (char *)portSecureBuff.QueryPtr();
fServerPortSecure = (szServerPortSecure[0] == '1');
}
// Get the application virtual path
TCHAR szApplnVirtPath[256];
if (FAILED(hr = FindApplicationPath(pHitObj->PIReq(), szApplnVirtPath, sizeof szApplnVirtPath)))
return hr;
TCHAR *szServerNameT;
TCHAR *szServerPortT;
#if UNICODE
CMBCSToWChar convServer;
if (FAILED(hr = convServer.Init(szServerName))) {
return hr;
}
szServerNameT = convServer.GetString();
#else
szServerNameT = szServerName;
#endif
#if UNICODE
CMBCSToWChar convPort;
if (FAILED(hr = convPort.Init(szServerPort))) {
return hr;
}
szServerPortT = convPort.GetString();
#else
szServerPortT = szServerPort;
#endif
// Allocate space for and construct the application URL
m_szApplnURL = new TCHAR [(9 /* sizeof "https://:" */ + _tcslen(szServerNameT) + _tcslen(szServerPortT) + _tcslen(szApplnVirtPath) + 1)];
if (m_szApplnURL == NULL)
return E_OUTOFMEMORY;
TCHAR *pT;
// start with the protocol prefix...
pT = strcpyEx(m_szApplnURL, fServerPortSecure? _T("https://") : _T("http://"));
// next add the servername
pT = strcpyEx(pT, szServerNameT);
// next the colon between the servername and the serverport
pT = strcpyEx(pT, _T(":"));
// next the server port
pT = strcpyEx(pT, szServerPortT);
// now the applURL is built up to the appln path. The next step will be to
// add the virtpath.
m_szApplnVirtPath = pT;
_tcscpy(m_szApplnVirtPath, szApplnVirtPath);
m_LKHashKey.dwInstanceID = rTemplateKey.dwInstanceID;
if ((m_LKHashKey.szPathTranslated = StringDup((TCHAR *)rTemplateKey.szPathTranslated)) == NULL)
return E_OUTOFMEMORY;
return S_OK;
}
/* ============================================================================
CTemplate::Compile
Compiles the template from its source file and include files, if any,
by calling GetSegmentsFromFile (to populate WorkStore),
followed by WriteTemplate (to create the template from WorkStore).
Returns:
HRESULT indicating success or type of failure
Side effects:
Indirectly allocates memory (via WriteTemplate)
Indirectly frees memory on error (via FreeGoodTemplateMemory)
*/
HRESULT
CTemplate::Compile
(
CHitObj* pHitObj
)
{
HRESULT hr = S_OK;
// The following code moved from Init() (to make Init() lighter)
Assert(pHitObj);
// Create and Init WorkStore
if (SUCCEEDED(hr))
{
// construct the workstore - bail on fail
if(NULL == (m_pWorkStore = new CWorkStore))
hr = E_OUTOFMEMORY;
}
if (SUCCEEDED(hr))
{
hr = (m_pWorkStore->m_ScriptStore).Init(pHitObj->QueryAppConfig()->szScriptLanguage(),
pHitObj->QueryAppConfig()->pCLSIDDefaultEngine());
if (hr == TYPE_E_ELEMENTNOTFOUND)
{
// default script language in registry is bogus - send error msg to browser
HandleCTemplateError(
NULL, // source file map
NULL, // ptr to source location where error occurred
IDE_TEMPLATE_BAD_PROGLANG_IN_REGISTRY, // error message id
0, // count of insert strings for error msg
NULL, // array of ptrs to error msg insert strings
pHitObj // Browser Request
);
}
if (FAILED(hr))
{
delete m_pWorkStore;
m_pWorkStore = NULL;
}
}
// Try to init the workstore and map main file - this can fail with oom, etc or user lacks permissions
if (SUCCEEDED(hr))
{
TRY
m_pWorkStore->Init();
AppendMapFile(
NULL, // file spec for this file - NULL means get filespec from pHitObj
NULL, // ptr to filemap of parent file
FALSE, // don't care
pHitObj, // ptr to template's hit object
m_fGlobalAsa // is this the global.asa file?
);
CATCH(hrException)
delete m_pWorkStore;
m_pWorkStore = NULL;
hr = hrException;
if(hr == E_USER_LACKS_PERMISSIONS)
HandleAccessFailure(pHitObj,
(m_rgpFilemaps && m_rgpFilemaps[0])? m_rgpFilemaps[0]->m_szPathTranslated : NULL);
if (m_rgpFilemaps && m_rgpFilemaps[0])
{
// empty file will fail to map but will have a handle, so we check for it here
if (0 == GetFileSize(m_rgpFilemaps[0]->m_hFile, NULL))
hr = E_SOURCE_FILE_IS_EMPTY;
m_rgpFilemaps[0]->UnmapFile();
}
if (SUCCEEDED(hr))
hr = E_FAIL; // make sure the error is set
END_TRY
}
if (SUCCEEDED(hr))
{
Assert(m_rgpFilemaps[0]);
Assert(m_rgpFilemaps[0]->m_szPathTranslated);
Assert(FImplies(!m_fGlobalAsa, (0 == _tcscmp(m_rgpFilemaps[0]->m_szPathTranslated, pHitObj->PSzCurrTemplatePhysPath()))));
Assert(FImplies(m_fGlobalAsa, (0 == _tcscmp(m_rgpFilemaps[0]->m_szPathTranslated, pHitObj->GlobalAspPath()))));
Assert(0 < m_rgpFilemaps[0]->GetSize());
}
if (FAILED(hr))
{
m_fDontCache = TRUE;
// OK, cache HR if m_fDontCache is true
// later, another thread might find this template from the cache even if the template
// has some error and marked as DontCache.
m_hrOnNoCache = hr;
m_fReadyForUse = TRUE;
SetEvent(m_hEventReadyForUse);
return hr;
}
// End of Code moved from Init()
// By default we are not in a transaction
m_ttTransacted = ttUndefined;
// By default session is required
m_fSession = TRUE;
// By default assume script exists
m_fScriptless = FALSE;
// we assert, in effect, that template is already init'ed
Assert(FImplies(!m_fGlobalAsa, (0 == _tcscmp(m_rgpFilemaps[0]->m_szPathTranslated, pHitObj->PSzCurrTemplatePhysPath()))));
Assert(FImplies(m_fGlobalAsa, (0 == _tcscmp(m_rgpFilemaps[0]->m_szPathTranslated, pHitObj->GlobalAspPath()))));
TRY
// Get source segments from source file
GetSegmentsFromFile(*(m_rgpFilemaps[0]), *m_pWorkStore, pHitObj);
/* get "language equivalents" for primary languagefrom registry
NOTE we do this here because the user can reset the primary language in the script file,
so we must wait until after GetSegmentsFromFile()
*/
GetLanguageEquivalents();
// Call WriteTemplate, which writes out template components to contiguous memory,
// resulting in a compiled template
WriteTemplate(*m_pWorkStore, pHitObj);
// Calculate the # of characters in a filemap before we unmap the file for all time.
for (unsigned i = 0; i < m_cFilemaps; ++i)
m_rgpFilemaps[i]->CountChars((WORD)m_wCodePage);
// Wrap typelibs into single IDispatch*
WrapTypeLibs(pHitObj);
m_fIsValid = TRUE;
CATCH(hrException)
// NOTE: we used to free template memory here. Now we do not because if the
// error was E_USER_LACKS_PERMISSIONS, and template is in cache, we don't want
// to sabotage future requests. There's no need to decache the template.
//
// The template destructor will free this memory anyway.
//
hr = hrException;
END_TRY
// check if scriptless
if (!m_fGlobalAsa)
{
// count various stuff to make the determination
DWORD cScriptEngines = m_pWorkStore->m_ScriptStore.CountPreliminaryEngines();
DWORD cPrimaryScriptSegments = (cScriptEngines > 0) ? m_pWorkStore->m_ScriptStore.m_ppbufSegments[0]->Count() : 0;
DWORD cObjectTags = m_pWorkStore->m_ObjectInfoStore.Count();
DWORD cHtmlSegments = m_pWorkStore->m_bufHTMLSegments.Count();
DWORD c449Cookies = m_rgp449.length();
BOOL fPageCommandsPresent = m_pWorkStore->m_fPageCommandsExecuted;
if (cScriptEngines <= 1 &&
cPrimaryScriptSegments == 0 &&
cObjectTags == 0 &&
cHtmlSegments == 1 &&
c449Cookies == 0 &&
!fPageCommandsPresent)
{
m_fScriptless = TRUE;
}
}
// free working storage - no longer needed
delete m_pWorkStore;
m_pWorkStore = NULL;
// un-map filemaps - NOTE filemaps stay around for possible post-compile errors (e.g., script failure)
UnmapFiles();
// Debugging: print data structure to debugger
IF_DEBUG(SCRIPT_DEBUGGER)
{
if (SUCCEEDED(hr))
{
DBGPRINTF((DBG_CONTEXT, "Script Compiled\n"));
for (UINT i = 0; i < m_cScriptEngines; ++i)
{
char *szEngineName;
PROGLANG_ID *pProgLangID;
const wchar_t *wszScriptText;
GetScriptBlock(i, &szEngineName, &pProgLangID, &wszScriptText);
DBGPRINTF((DBG_CONTEXT, "Engine %d, Language=\"%s\":\n", i, szEngineName));
#ifndef _NO_TRACING_
DBGINFO((DBG_CONTEXT, (char *) wszScriptText));
DBGINFO((DBG_CONTEXT, "\n"));
#else
OutputDebugStringW(wszScriptText);
OutputDebugStringA("\n");
#endif
}
#if 0
OutputDebugTables();
#endif
}
}
if (hr == E_TEMPLATE_COMPILE_FAILED_DONT_CACHE)
{
m_fDontCache = TRUE;
m_hrOnNoCache = hr;
}
// Set ready-for-use flag true and event to signaled
// NOTE we do this whether success or failure, since even a failed-compile template
// will remain in the cache to allow template cache mgr to satisfy requests on it
m_fReadyForUse = TRUE;
SetEvent(m_hEventReadyForUse);
// Note whether the template currently is debuggable
// BUG BUG: Template is debuggable or not based on first app. If shared between a debug
// & non-debug app, the first application wins.
m_fDebuggable = (BOOLB)!!pHitObj->PAppln()->FDebuggable();
return hr;
}
/* ============================================================================
CTemplate::Deliver
Delivers template to caller once template is ready for use
NOTE 'compile failure' == template is 'ready for use' but did not compile successfully;
this allows cache mgr to keep a failed template in cache in case it gets requested again
Returns
success or failure
Side effects
none
*/
HRESULT
CTemplate::Deliver
(
CHitObj* pHitObj
)
{
// NOTE: There was a compiler bug where 'ps' would not be correctly aligned,
// EVEN if it was declared to be a DWORD array, if 'ps' was nested in
// a block. Thus declare it here.
//
BYTE ps[SIZE_PRIVILEGE_SET]; // privilege set
HRESULT hr = S_OK;
// if ready flag is not yet set block until template is ready for use
if(!m_fReadyForUse)
{
WaitForSingleObject(m_hEventReadyForUse, INFINITE);
Assert(m_fReadyForUse); // when event unblocks, flag will be set
}
if (m_pbStart == NULL)
{
if (m_fDontCache && m_dwLastErrorMask == 0)
{
DBGPRINTF((DBG_CONTEXT, "template compile failed with %08x\n", m_hrOnNoCache));
DBG_ASSERT(FAILED(m_hrOnNoCache));
// Safety net: always fail, even if "m_hrOnNoCache" did not get set somehow.
hr = m_hrOnNoCache;
if (SUCCEEDED(m_hrOnNoCache))
hr = E_FAIL;
if(hr == E_USER_LACKS_PERMISSIONS)
HandleAccessFailure(pHitObj,
(m_rgpFilemaps && m_rgpFilemaps[0])? m_rgpFilemaps[0]->m_szPathTranslated : NULL);
return hr;
}
// template compile failed - NOTE null start-of-template ptr == template compile failed
// use cached error info
SendToLog( m_dwLastErrorMask,
m_pszLastErrorInfo[ILE_szFileName],
m_pszLastErrorInfo[ILE_szLineNum],
m_pszLastErrorInfo[ILE_szEngine],
m_pszLastErrorInfo[ILE_szErrorCode],
m_pszLastErrorInfo[ILE_szShortDes],
m_pszLastErrorInfo[ILE_szLongDes],
pHitObj);
hr = E_TEMPLATE_COMPILE_FAILED;
}
else if (!pHitObj->FIsBrowserRequest())
{
return hr;
}
else if (Glob(fWinNT))
// template compile succeeded - check user's file permissions
// ACLs: the following code should in future be shared with IIS (see creatfil.cxx in IIS project)
{
HANDLE hUserAccessToken = pHitObj->HImpersonate(); // current user's access token
DWORD dwPS = sizeof(ps); // privilege set size
DWORD dwGrantedAccess; // granted access mask
BOOL fAccessGranted; // access granted flag
GENERIC_MAPPING gm = { // generic mapping struct
FILE_GENERIC_READ,
FILE_GENERIC_WRITE,
FILE_GENERIC_EXECUTE,
FILE_ALL_ACCESS
};
((PRIVILEGE_SET*)&ps)->PrivilegeCount = 0; // set privilege count to 0
Assert(NULL != hUserAccessToken);
for(UINT i = 0; i < m_cFilemaps; i++)
{
if(NULL == m_rgpFilemaps[i]->m_pSecurityDescriptor)
continue;
if(!AccessCheck(
m_rgpFilemaps[i]->m_pSecurityDescriptor, // pointer to security descriptor
hUserAccessToken, // handle to client access token
FILE_GENERIC_READ, // access mask to request
&gm, // address of generic-mapping structure
(PRIVILEGE_SET*)ps, // address of privilege-set structure
&dwPS, // address of size of privilege-set structure
&dwGrantedAccess, // address of granted access mask
&fAccessGranted // address of flag indicating whether access granted
))
return E_FAIL;
if(!fAccessGranted)
{
// if access is denied on any file, handle the failure and return
HandleAccessFailure(pHitObj, m_rgpFilemaps[0]->m_szPathTranslated);
return E_USER_LACKS_PERMISSIONS;
}
}
}
// Reset the Session.CodePage to the script compilation-time codepage
// only if a code page directive was found during compilation
if (m_fCodePageSet && (!pHitObj->FHasSession() || !pHitObj->PSession()->FCodePageSet()))
{
pHitObj->SetCodePage(m_wCodePage);
}
// Reset the Session.LCID to the script compilation-time LCID
// only if an LCID directive was found during compilation
if (m_fLCIDSet && (!pHitObj->FHasSession() || !pHitObj->PSession()->FLCIDSet()))
{
pHitObj->SetLCID(m_lLCID);
}
return hr;
}
/* ============================================================================
CTemplate::CTemplate
Ctor
*/
CTemplate::CTemplate()
: m_pWorkStore(NULL),
m_fGlobalAsa(FALSE),
m_fReadyForUse(FALSE),
m_fDontAttach(FALSE),
m_hEventReadyForUse(NULL),
m_fDebuggerDetachCSInited(FALSE),
m_pbStart(NULL),
m_cbTemplate(0),
m_cRefs(1), // NOTE ctor AddRefs implicitly
m_pbErrorLocation(NULL),
m_idErrMsg(0),
m_cMsgInserts(0),
m_ppszMsgInserts(NULL),
m_cScriptEngines(0),
m_rgrgSourceInfos(NULL),
m_rgpDebugScripts(NULL),
m_rgpFilemaps(NULL),
m_cFilemaps(0),
m_rgpSegmentFilemaps(NULL),
m_cSegmentFilemapSlots(0),
m_wCodePage(CP_ACP),
m_lLCID(LOCALE_SYSTEM_DEFAULT),
m_ttTransacted(ttUndefined),
m_fSession(TRUE),
m_fScriptless(FALSE),
m_fDebuggable(FALSE),
m_fIsValid(FALSE),
m_fDontCache(FALSE),
m_fZombie(FALSE),
m_fCodePageSet(FALSE),
m_fLCIDSet(FALSE),
m_fIsPersisted(FALSE),
m_szPersistTempName(NULL),
m_szApplnVirtPath(NULL),
m_szApplnURL(NULL),
m_CPTextEvents(this, IID_IDebugDocumentTextEvents),
m_pdispTypeLibWrapper(NULL),
m_dwLastErrorMask(S_OK),
m_hrOnNoCache(S_OK),
m_cbTargetOffsetPrevT(0),
m_pHashTable(NULL)
{
for (UINT i = 0; i < ILE_MAX; i++)
{
m_pszLastErrorInfo[i] = NULL;
}
IF_DEBUG(TEMPLATE)
{
WriteRefTraceLog(gm_pTraceLog, m_cRefs, this);
}
#if PER_TEMPLATE_REFLOG
m_pTraceLog = CreateRefTraceLog (100,0);
WriteRefTraceLog (m_pTraceLog,m_cRefs, this);
#endif
}
/* ============================================================================
CTemplate::~CTemplate
Destructor
Returns:
Nothing
Side effects:
None
*/
CTemplate::~CTemplate()
{
DBGPRINTF(( DBG_CONTEXT, "Deleting template, m_cFilemaps = %d, m_rgpFilemaps %p\n", m_cFilemaps, m_rgpFilemaps));
// first, remove this template from its inc-files' template lists
// NOTE must do this before freeing template memory
RemoveFromIncFiles();
// Remove the template from the debugger's list of documents
Detach();
PersistCleanup();
if(m_rgpFilemaps)
{
for(UINT i = 0; i < m_cFilemaps; i++)
delete m_rgpFilemaps[i];
SmallTemplateFreeNullify((void**) &m_rgpFilemaps);
}
FreeGoodTemplateMemory();
if (m_pWorkStore)
delete m_pWorkStore;
//FileName, LineNum, Engine, ErrorCode, ShortDes, LongDes
for(UINT iErrInfo = 0; iErrInfo < ILE_MAX; iErrInfo++)
{
FreeNullify((void**) &m_pszLastErrorInfo[iErrInfo]);
}
if(m_hEventReadyForUse != NULL)
CloseHandle(m_hEventReadyForUse);
if (m_LKHashKey.szPathTranslated)
free((void *)m_LKHashKey.szPathTranslated);
if (m_szApplnURL)
delete [] m_szApplnURL;
if (m_fDebuggerDetachCSInited)
DeleteCriticalSection(&m_csDebuggerDetach);
if (m_pdispTypeLibWrapper)
m_pdispTypeLibWrapper->Release();
if (m_szPersistTempName)
CTemplate::LargeFree(m_szPersistTempName);
#if PER_TEMPLATE_REFLOG
DestroyRefTraceLog (m_pTraceLog);
#endif
}
/* ============================================================================
CTemplate::QueryInterface
Provides QueryInterface implementation for CTemplate
NOTE: It is arbitrary which vtable we return for IDebugDocument & IDebugDocumentInfo.
*/
HRESULT
CTemplate::QueryInterface(const GUID &uidInterface, void **ppvObj)
{
if (uidInterface == IID_IUnknown || uidInterface == IID_IDebugDocumentProvider)
*ppvObj = static_cast<IDebugDocumentProvider *>(this);
else if (uidInterface == IID_IDebugDocument || uidInterface == IID_IDebugDocumentInfo || uidInterface == IID_IDebugDocumentText)
*ppvObj = static_cast<IDebugDocumentText *>(this);
else if (uidInterface == IID_IConnectionPointContainer)
*ppvObj = static_cast<IConnectionPointContainer *>(this);
else
*ppvObj = NULL;
if (*ppvObj)
{
AddRef();
return S_OK;
}
else
return E_NOINTERFACE;
}
/* ============================================================================
CTemplate::AddRef
Adds a ref to this template, thread-safely
*/
ULONG
CTemplate::AddRef()
{
LONG cRefs = InterlockedIncrement(&m_cRefs);
Assert(FImplies(m_fIsValid,FImplies(cRefs > 1, m_pbStart != NULL)));
IF_DEBUG(TEMPLATE)
{
WriteRefTraceLog(gm_pTraceLog, cRefs, this);
}
#if PER_TEMPLATE_REFLOG
WriteRefTraceLog(m_pTraceLog, cRefs, this);
#endif
return cRefs;
}
/* ============================================================================
CTemplate::Release
Releases a ref to this template, thread-safely
*/
ULONG
CTemplate::Release()
{
LONG cRefs = InterlockedDecrement(&m_cRefs);
IF_DEBUG(TEMPLATE)
{
WriteRefTraceLog(gm_pTraceLog, cRefs, this);
}
#if PER_TEMPLATE_REFLOG
WriteRefTraceLog(m_pTraceLog, cRefs, this);
#endif
if (cRefs == 0)
delete this;
return cRefs;
}
/* ============================================================================
CTemplate::RemoveIncFile
Removes (by setting to null) an inc-file ptr from this template's inc-file list.
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::RemoveIncFile
(
CIncFile* pIncFile
)
{
// If the filemap count is non-zero the pointer to
// the array of filemaps has better not be null
DBGPRINTF(( DBG_CONTEXT, "m_cFilemaps = %d, m_rgpFilemaps %p\n", m_cFilemaps, m_rgpFilemaps));
Assert((m_cFilemaps <= 0) || (m_rgpFilemaps != NULL));
// find the inc-file in list
for(UINT i = 1; (i < m_cFilemaps) && (m_rgpFilemaps[i]->m_pIncFile != pIncFile); i++)
;
// assert that we found the inc-file in list
Assert((i < m_cFilemaps) && (m_rgpFilemaps[i]->m_pIncFile == pIncFile));
// set inc-file ptr null
m_rgpFilemaps[i]->m_pIncFile = NULL;
}
/*===================================================================
CTemplate::FTemplateObsolete
Test to see if the files this template depends on have changed since it
was compiled.
We use this in cases where we may have missed a change notification,
for example, when there were too many changes to record in our change
notification buffer. We check the last time the file was written too,
and the security descriptor, since changes to the security descriptor
aren't noted in the file last write time.
Parameters:
None
Returns:
TRUE if the template is obsolete, else FALSE
*/
BOOL CTemplate::FTemplateObsolete(VOID)
{
BOOL fStatus = FALSE;
// On Windows 95 files should not be cached
// so assume the template has changed
if (!FIsWinNT())
{
return TRUE;
}
for (UINT i = 0; i < m_cFilemaps; i++)
{
if (FFileChangedSinceCached(m_rgpFilemaps[i]->m_szPathTranslated, m_rgpFilemaps[i]->m_ftLastWriteTime))
{
// If the file write time has changed we know enough
// and can quit here
fStatus = TRUE;
break;
}
else
{
// The file hasn't been writen to, but the security descriptor may
// have chagned
// Assert on non-valid security descriptor
if (NULL != m_rgpFilemaps[i]->m_pSecurityDescriptor)
{
PSECURITY_DESCRIPTOR pSecurityDescriptor = NULL;
DWORD dwSize = m_rgpFilemaps[i]->m_dwSecDescSize;
if( 0 == GetSecDescriptor(m_rgpFilemaps[i]->m_szPathTranslated, &pSecurityDescriptor, &dwSize))
{
if (pSecurityDescriptor)
{
// if the size is not the same then set fStatus to TRUE no need to compare memory blocks.
if(dwSize != GetSecurityDescriptorLength(m_rgpFilemaps[i]->m_pSecurityDescriptor))
{
fStatus = TRUE;
}
else
{
// The size of the security descriptor hasn't changed
// but we have to compare the contents to make sure they haven't changed
fStatus = !(0 == memcmp(m_rgpFilemaps[i]->m_pSecurityDescriptor, pSecurityDescriptor, dwSize));
}
// We are done with the descriptor
free(pSecurityDescriptor);
}
else
{
// Since we failed to get a security descriptor
// assume the file has changed.
fStatus = TRUE;
}
}
}
}
// Quit as soon as we find a change
if (fStatus)
{
break;
}
}
return fStatus;
}
/* ============================================================================
CTemplate::GetSourceFileName
Returns name of source file on which this template is based
Returns
source file name
Side effects
none
*/
LPTSTR
CTemplate::GetSourceFileName(SOURCEPATHTYPE pathtype)
{
if (!m_rgpFilemaps)
{
return NULL;
}
switch (pathtype)
{
case SOURCEPATHTYPE_PHYSICAL:
return((m_rgpFilemaps[0] ? m_rgpFilemaps[0]->m_szPathTranslated : NULL));
case SOURCEPATHTYPE_VIRTUAL:
return((m_rgpFilemaps[0] ? m_rgpFilemaps[0]->m_szPathInfo : NULL));
default:
return(NULL);
}
}
/* ============================================================================
CTemplate::Count
Returns count of components of type tcomp contained in this template
Returns:
Count of components of type tcomp
Side effects:
None
*/
USHORT
CTemplate::Count
(
TEMPLATE_COMPONENT tcomp
)
{
Assert(NULL != m_pbStart);
// script engines and script blocks have the same count, stored in same slot
if(tcomp == tcompScriptEngine)
tcomp = tcompScriptBlock;
// counts are stored at start of template in sequential slots, starting with script blocks count
return * (USHORT*) ((USHORT*)m_pbStart + (tcomp - tcompScriptBlock));
}
/* ============================================================================
CTemplate::GetScriptBlock
Gets ptrs to script engine name, prog lang id and script text of i-th script block.
Returns:
Out-parameters; see below
Side effects:
None
*/
void
CTemplate::GetScriptBlock
(
UINT i, // script block id
LPSTR* pszScriptEngine, // ptr to script engine name (out-parameter)
PROGLANG_ID** ppProgLangId, // ptr to prog lang id (out-parameter)
LPCOLESTR* pwstrScriptText // ptr to wstr script text (out-parameter)
)
{
CByteRange brEngine; // engine name
CByteRange brScriptText; // script text
UINT cbAlignment; // count of bytes guid was shifted in WriteTemplate() to make it dword-aligned
BYTE* pbEngineInfo = GetAddress(tcompScriptEngine, (USHORT)i); // ptr to engine info
Assert(pbEngineInfo != NULL);
Assert(i < CountScriptEngines());
// Get engine name from start of engine info
ByteRangeFromPb(pbEngineInfo, brEngine);
ByteRangeFromPb(GetAddress(tcompScriptBlock, (USHORT)i), brScriptText);
Assert(!brEngine.IsNull());
Assert(!brScriptText.IsNull());
// Advance ptr past name to prog lang id
// length of prefix + length of name + NULL
pbEngineInfo += (sizeof(UINT) + (*pbEngineInfo) + 1);
// Get prog lang id - it will be on the next pointer sized boundary
cbAlignment = (UINT) (((DWORD_PTR) pbEngineInfo) % sizeof(DWORD));
if(cbAlignment > 0)
{pbEngineInfo += (sizeof(DWORD) - cbAlignment);}
*pszScriptEngine = (LPSTR)brEngine.m_pb;
*ppProgLangId = (PROGLANG_ID*)pbEngineInfo;
*pwstrScriptText = (LPCOLESTR)brScriptText.m_pb;
}
/* ============================================================================
CTemplate::GetObjectInfo
Returns i-th object-info in template as object name and
its clsid, scope, model
Returns:
HRESULT
Out-parameters; see below
Side effects:
*/
HRESULT
CTemplate::GetObjectInfo
(
UINT i, // object index
LPSTR* ppszObjectName, // address of object name ptr (out-parameter)
CLSID* pClsid, // address of object clsid
CompScope* pcsScope, // address of object scope
CompModel* pcmModel // address of object threading model
)
{
BYTE* pbObjectInfo = GetAddress(tcompObjectInfo, (USHORT)i); // ptr to current read location
CByteRange brName; // object name
UINT cbAlignment; // count of bytes guid was shifted in WriteTemplate() to make it dword-aligned
Assert(i < Count(tcompObjectInfo));
// Get name from start of object-info
ByteRangeFromPb(pbObjectInfo, brName);
Assert(!brName.IsNull());
// Advance ptr past name
// length of prefix + length of name + NULL
pbObjectInfo += (sizeof(UINT) + (*pbObjectInfo) + 1);
// Get clsid - it will be on the next DWORD boundary
cbAlignment = (UINT)(((DWORD_PTR) pbObjectInfo) % sizeof(DWORD));
if(cbAlignment > 0)
pbObjectInfo += (sizeof(DWORD) - cbAlignment);
*pClsid = *(CLSID*)pbObjectInfo;
pbObjectInfo += sizeof(CLSID);
// Get scope
*pcsScope = *(CompScope*)pbObjectInfo;
pbObjectInfo += sizeof(CompScope);
// Get model
*pcmModel = *(CompModel*)pbObjectInfo;
pbObjectInfo += sizeof(CompModel);
*ppszObjectName = (LPSTR)brName.m_pb;
return S_OK;
}
/* ============================================================================
CTemplate::GetHTMLBlock
Returns i-th HTML block
Parameters:
UINT i block number
LPSTR* pszHTML [out] html text
ULONG* pcbHTML [out] html text length
ULONG* pcbSrcOffs [out] offset in the source file
LPSTR* pszSrcIncFile [out] include source file name
Returns:
Nothing
Side effects:
None
*/
HRESULT
CTemplate::GetHTMLBlock
(
UINT i,
LPSTR* pszHTML,
ULONG* pcbHTML,
ULONG* pcbSrcOffs,
LPSTR* pszSrcIncFile
)
{
Assert(i < Count(tcompHTMLBlock));
// this was added due to user attempt to access the method with an invalid array offset
//
if ( i >= Count(tcompHTMLBlock) )
return E_FAIL;
// get address of the block start in template memory
BYTE *pbBlock = GetAddress(tcompHTMLBlock, (USHORT)i);
Assert(pbBlock);
// retrieve the byte range of the html code
CByteRange brHTML;
ByteRangeFromPb(pbBlock, brHTML);
*pszHTML = (LPSTR)brHTML.m_pb;
*pcbHTML = brHTML.m_cb;
// advance to the source offset
pbBlock += sizeof(ULONG); // skip prefix
pbBlock += brHTML.m_cb+1; // skip html bytes (incl. '\0')
// Add byte aligment which is done in ByteAlignOffset()
if ((reinterpret_cast<ULONG_PTR>(pbBlock)) & 3)
pbBlock = reinterpret_cast<BYTE *>((reinterpret_cast<ULONG_PTR>(pbBlock) + 4) & ~3);
*pcbSrcOffs = *((ULONG*)pbBlock);
// advance to the source name length
pbBlock += sizeof(ULONG); // skip source offset prefix
ULONG cbSrcIncFile = *((ULONG *)pbBlock); // inc file name length
pbBlock += sizeof(ULONG); // skip inc file name length
*pszSrcIncFile = (cbSrcIncFile > 0) ? (LPSTR)pbBlock : NULL;
return S_OK;
}
/* ============================================================================
CTemplate::GetScriptSourceInfo
Returns line number and source file name a given target line in a given script engine.
Returns
line number and source file name (as out-parameters)
Side effects:
None
*/
void
CTemplate::GetScriptSourceInfo
(
UINT idEngine, // script engine id
int iTargetLine, // target line number
LPTSTR* pszPathInfo, // ptr to source file virtual path (out-parameter)
LPTSTR* pszPathTranslated, // ptr to source file real path (out-parameter)
ULONG* piSourceLine, // ptr to source line number (out-parameter)
ULONG* pichSourceLine, // ptr to source file offset (out-parameter)
BOOLB* pfGuessedLine // ptr to flag: did we guess the source line?
)
{
// Initialize some out parameters
if (pszPathInfo)
*pszPathInfo = _T("?"); // In case we don't ever find the path
if (pszPathTranslated)
*pszPathTranslated = _T("?"); // In case we don't ever find the path
if (piSourceLine)
*piSourceLine = 0;
if (pichSourceLine)
*pichSourceLine = 0;
if (pfGuessedLine)
*pfGuessedLine = FALSE;
if (iTargetLine <=0)
{
return;
}
// CHANGE: The rgSourceInfo array is now ZERO based. Decrement target line
// to convert.
--iTargetLine;
// CONSIDER: Make these assertions?
if(!m_rgrgSourceInfos)
return;
if(idEngine > (m_cScriptEngines - 1)) // bug 375: check vs. array bound
return;
if(size_t(iTargetLine) >= m_rgrgSourceInfos[idEngine].length()) // bug 375: check vs. array bound
return;
vector<CSourceInfo> *prgSourceInfos = &m_rgrgSourceInfos[idEngine];
// bug 379: move backwards through target lines, starting with the caller's, until we find one whose
// fIsHTML flag is false. this handles the case where vbs flags a manufactured line as in error;
// we assume the actual error occurred at the most recent authored line
while (iTargetLine >= 0 && (*prgSourceInfos)[iTargetLine].m_fIsHTML)
{
--iTargetLine;
if (pfGuessedLine)
*pfGuessedLine = TRUE;
}
if (iTargetLine >= 0)
{
if (pszPathInfo && (*prgSourceInfos)[iTargetLine].m_pfilemap != NULL)
*pszPathInfo = (*prgSourceInfos)[iTargetLine].m_pfilemap->m_szPathInfo;
if (pszPathTranslated && (*prgSourceInfos)[iTargetLine].m_pfilemap != NULL)
*pszPathTranslated = (*prgSourceInfos)[iTargetLine].m_pfilemap->m_szPathTranslated;
if (piSourceLine)
*piSourceLine = (*prgSourceInfos)[iTargetLine].m_idLine;
if (pichSourceLine)
*pichSourceLine = (*prgSourceInfos)[iTargetLine].m_cchSourceOffset;
}
}
/* ============================================================================
CTemplate::GetPositionOfLine
Get the character offset of a line of source
(Debugger API Extended to specify a filemap)
*/
HRESULT
CTemplate::GetPositionOfLine
(
CFileMap *pFilemap,
ULONG cLineNumber,
ULONG *pcCharacterPosition
)
{
// NOTE:
// The table is not binary-searchable because include files
// will start a new line ordering
//
// Algorithm:
//
// Find the largest source line N across all engines, such that
// N <= cLineNumber and the line corresponds to an line
// in the appropriate file.
//
CSourceInfo *pSourceInfoLE = NULL;
++cLineNumber; // Convert zero-based line # to one-based
// Find the correct offset
for (unsigned idEngine = 0; idEngine < m_cScriptEngines; ++idEngine)
{
vector<CSourceInfo> *prgSourceInfos = &m_rgrgSourceInfos[idEngine];
// Loop through all lines EXCEPT the EOF line
for (unsigned j = 0; j < prgSourceInfos->length() - 1; ++j)
{
CSourceInfo *pSourceInfo = &(*prgSourceInfos)[j];
if (pFilemap == pSourceInfo->m_pfilemap &&
pSourceInfo->m_idLine <= cLineNumber &&
(pSourceInfoLE == NULL || pSourceInfo->m_idLine > pSourceInfoLE->m_idLine))
{
pSourceInfoLE = pSourceInfo;
}
}
}
// We had better be able to map all line numbers to offsets, unless they passed a bogus line
// (in which case we still find an offset)
//
Assert (pSourceInfoLE != NULL);
if (pSourceInfoLE == NULL) {
return E_FAIL;
}
*pcCharacterPosition = pSourceInfoLE->m_cchSourceOffset;
#if 0
IF_DEBUG(SCRIPT_DEBUGGER)
{
wchar_t wszSourceText[SNIPPET_SIZE + 1], wszTargetText[SNIPPET_SIZE + 1], wszDebugMessage[256];
GetScriptSnippets(
pSourceInfoLE->m_cchSourceOffset, pSourceInfoLE->m_pfilemap,
0, 0,
wszSourceText, NULL
);
DBGPRINTF((
DBG_CONTEXT,
"Source Line %d corresponds to source offset %d (Text: \"%S\")\n",
cLineNumber - 1, pSourceInfoLE->m_cchSourceOffset,
wszSourceText
));
}
#endif
return S_OK;
}
/* ============================================================================
CTemplate::GetLineOfPosition
Get the line # & offset in line of an arbitrary character offset in source
(Debugger API Extended to specify a filemap)
*/
HRESULT CTemplate::GetLineOfPosition
(
CFileMap *pFilemap,
ULONG cCharacterPosition,
ULONG *pcLineNumber,
ULONG *pcCharacterOffsetInLine
)
{
// FAIL if source offset totally off-base
if (cCharacterPosition >= pFilemap->m_cChars)
return E_FAIL;
// NOTE:
// The table is not binary-searchable because include files
// will start a new line ordering
//
// Algorithm:
//
// Find the largest source line N across all engines, such that
// N <= cLineNumber and the line corresponds to an line
// in the appropriate file.
//
CSourceInfo *pSourceInfoLE = NULL;
// Find the correct offset
for (unsigned idEngine = 0; idEngine < m_cScriptEngines; ++idEngine)
{
vector<CSourceInfo> *prgSourceInfos = &m_rgrgSourceInfos[idEngine];
// Loop through all lines EXCEPT the EOF line
for (unsigned j = 0; j < prgSourceInfos->length() - 1; ++j)
{
CSourceInfo *pSourceInfo = &(*prgSourceInfos)[j];
if (pFilemap == pSourceInfo->m_pfilemap &&
pSourceInfo->m_cchSourceOffset <= cCharacterPosition &&
(pSourceInfoLE == NULL || pSourceInfo->m_cchSourceOffset > pSourceInfoLE->m_cchSourceOffset))
{
pSourceInfoLE = pSourceInfo;
}
}
}
// We had better be able to map all offsets to line numbers, unless they passed a bogus offset
// (in which case we still find a line #, but may go out of range for the offset in line.
// That case is handled later)
//
Assert (pSourceInfoLE != NULL);
if (pSourceInfoLE == NULL) {
return E_FAIL;
}
*pcLineNumber = pSourceInfoLE->m_idLine - 1; // Convert to zero-based line #
*pcCharacterOffsetInLine = cCharacterPosition - pSourceInfoLE->m_cchSourceOffset;
#if 0
IF_DEBUG(SCRIPT_DEBUGGER)
{
wchar_t wszSourceText[SNIPPET_SIZE + 1], wszTargetText[SNIPPET_SIZE + 1], wszDebugMessage[256];
GetScriptSnippets(
pSourceInfoLE->m_cchSourceOffset, pSourceInfoLE->m_pfilemap,
0, 0,
wszSourceText, NULL
);
DBGPRINTF((
DBG_CONTEXT,
"Source offset %d corresponds to source line %d (Text: \"%S\")\n",
pSourceInfoLE->m_cchSourceOffset, *pcLineNumber,
wszSourceText
));
}
DBGPRINTF((
DBG_CONTEXT,
"Source offset %d corresponds to source line %d (Text: \"%S\")\n",
pSourceInfoLE->m_cchSourceOffset, *pcLineNumber,
wszSourceText
));
}
#endif
return S_OK;
}
/* ============================================================================
CTemplate::GetSourceOffset
Convert a character offset relative to the target script to the appropriate
offset in the source.
NOTE: offsets in the middle of a target line are converted to the
offset relative to the beginning of source line - NOT to the
precise source offset.
this is OK because debugger ultimately wants the offset of the
beginning of line. It is a lot of work to do the precise conversion
due to the translation of "=" to Response.Write & HTML to
Response.WriteBlock
Also, because of these translations, we return the length of the segment
calculated during compilation, and throw away the length the scripting
engine sent to us.
*/
void
CTemplate::GetSourceOffset
(
ULONG idEngine,
ULONG cchTargetOffset,
TCHAR **pszSourceFile,
ULONG *pcchSourceOffset,
ULONG *pcchSourceText
)
{
Assert (idEngine < m_cScriptEngines);
vector<CSourceInfo> *prgSourceInfos = &m_rgrgSourceInfos[idEngine];
// Find the closest offset in the source
// This is the largest target offset N, such that N <= cchTargetOffset
CSourceInfo *pSourceInfo;
GetBracketingPair(
cchTargetOffset, // value to search for
prgSourceInfos->begin(), prgSourceInfos->end(), // array to search
CTargetOffsetOrder(), // ordering predicate
&pSourceInfo, static_cast<CSourceInfo **>(NULL) // return values
);
// Since the first offset is zero, which is less than all other conceivable offsets,
// the offset must have been found or else there is a bug.
Assert (pSourceInfo != NULL);
Assert (cchTargetOffset >= pSourceInfo->m_cchTargetOffset);
#if 0
IF_DEBUG(SCRIPT_DEBUGGER)
{
wchar_t wszSourceText[SNIPPET_SIZE + 1], wszTargetText[SNIPPET_SIZE + 1], wszDebugMessage[256];
GetScriptSnippets(
pSourceInfo->m_cchSourceOffset, pSourceInfo->m_pfilemap,
cchTargetOffset, idEngine,
wszSourceText, wszTargetText
);
DBGPRINTF((
DBG_CONTEXT,
"Target offset %d (Text: \"%S\") corresponds to source offset %d (Text: \"%S\") (Length is %d)\n",
cchTargetOffset, wszTargetText,
pSourceInfo->m_cchSourceOffset, wszSourceText,
pSourceInfo->m_cchSourceText
));
}
#endif
*pszSourceFile = pSourceInfo->m_pfilemap->m_szPathTranslated;
*pcchSourceOffset = pSourceInfo->m_cchSourceOffset;
*pcchSourceText = pSourceInfo->m_cchSourceText;
}
/* ============================================================================
CTemplate::GetTargetOffset
Convert a character offset relative to the source script to the appropriate
offset in the target.
Returns:
TRUE - source offset corresponds to script
FALSE - source offset corresponds to HTML
NOTES:
1. This function is very slow. consider caching the value of this function
(The CTemplateDocumentContext class does this.)
2. This function returns the source offset in the master include file -
if the target offset corresponds to an offset in a header file, then
the offset to the #include line in the source is returned.
3. offsets in the middle of a target line are converted to the
offset relative to the beginning of source line - NOT to the
precise source offset.
this is OK because the debugger ultimately wants the offset of the
beginning of line. It is a lot of work to do the precise conversion
due to the translation of "=" to Response.Write & HTML to
Response.WriteBlock
CONSIDER:
Figure out a better way to do this
*/
BOOL CTemplate::GetTargetOffset
(
TCHAR *szSourceFile,
ULONG cchSourceOffset,
/* [out] */ ULONG *pidEngine,
/* [out] */ ULONG *pcchTargetOffset
)
{
// NOTE:
// The table is not binary-searchable because of two factors:
// 1. Include files will start a new line ordering
// 2. For engine 0, tagged scripts will be re-arranged in
// the target code to reside after all primary script in
// engine 0.
//
// Algorithm:
//
// Find the largest source offset N across all engines, such that
// N <= cchSourceOffset and the offset corresponds to an offset
// in the appropriate file.
//
CSourceInfo *pSourceInfoLE = NULL;
unsigned idEngineLE = 0;
// Find the correct offset
for (unsigned idEngine = 0; idEngine < m_cScriptEngines; ++idEngine)
{
vector<CSourceInfo> *prgSourceInfos = &m_rgrgSourceInfos[idEngine];
// Loop through all lines EXCEPT the EOF line
for (unsigned j = 0; j < prgSourceInfos->length() - 1; ++j)
{
CSourceInfo *pSourceInfo = &(*prgSourceInfos)[j];
if (_tcscmp(pSourceInfo->m_pfilemap->m_szPathTranslated, szSourceFile) == 0 &&
pSourceInfo->m_cchSourceOffset <= cchSourceOffset &&
(pSourceInfoLE == NULL || pSourceInfo->m_cchSourceOffset > pSourceInfoLE->m_cchSourceOffset))
{
pSourceInfoLE = pSourceInfo;
idEngineLE = idEngine;
}
}
}
// There won't be a valid offset in the case where there is no
// code corresponding to the first line in the file (this only
// occurs when the first line is whitespace, because there is no
// corresponding "Response.WriteBlock" call there)
//
// In that case, return FALSE, which will cause the caller to fail
//
if (pSourceInfoLE == NULL)
{
*pidEngine = 0;
*pcchTargetOffset = 0;
return FALSE;
}
*pidEngine = idEngineLE;
*pcchTargetOffset = pSourceInfoLE->m_cchTargetOffset;
#if 0
IF_DEBUG(SCRIPT_DEBUGGER)
{
wchar_t wszSourceText[SNIPPET_SIZE + 1], wszTargetText[SNIPPET_SIZE + 1], wszDebugMessage[256];
GetScriptSnippets(
cchSourceOffset, pSourceInfoLE->m_pfilemap,
*pcchTargetOffset, *pidEngine,
wszSourceText, wszTargetText
);
DBGPRINTF((
DBG_CONTEXT,
"Source offset %d (Text: \"%S\") corresponds to target offset %d (Text: \"%S\")\n",
cchSourceOffset, wszSourceText,
*pcchTargetOffset, wszTargetText
));
}
#endif
return !pSourceInfoLE->m_fIsHTML;
}
/* ============================================================================
CTemplate::GetActiveScript
Return a cached script from the template - only used in debug mode
*/
CActiveScriptEngine *CTemplate::GetActiveScript(ULONG idEngine)
{
if (m_rgpDebugScripts == NULL)
return NULL;
else
{
Assert (idEngine < m_cScriptEngines);
CActiveScriptEngine *pEng = m_rgpDebugScripts[idEngine];
if (pEng)
pEng->AddRef();
return pEng;
}
}
/* ============================================================================
CTemplate::AddScript
add an active script to the template object
*/
HRESULT CTemplate::AddScript(ULONG idEngine, CActiveScriptEngine *pScriptEngine)
{
if (m_rgpDebugScripts == NULL)
{
if (
(m_rgpDebugScripts = new CActiveScriptEngine *[m_cScriptEngines])
== NULL
)
{
return E_OUTOFMEMORY;
}
memset(m_rgpDebugScripts, 0, m_cScriptEngines * sizeof(CActiveScriptEngine *));
}
Assert (idEngine < m_cScriptEngines);
CActiveScriptEngine **ppScriptElem = &m_rgpDebugScripts[idEngine];
if (*ppScriptElem != NULL)
(*ppScriptElem)->Release();
*ppScriptElem = pScriptEngine;
pScriptEngine->AddRef();
// Initialize the script engine now (is currently uninitialized)
// so that the debugger user can set breakpoints.
IActiveScript *pActiveScript = pScriptEngine->GetActiveScript();
HRESULT hr;
TRY
hr = pActiveScript->SetScriptSite(static_cast<IActiveScriptSite *>(pScriptEngine));
CATCH(nExcept)
HandleErrorMissingFilename(IDE_SCRIPT_ENGINE_GPF,
NULL,
TRUE,
nExcept,
"IActiveScript::SetScriptSite()",
"CTemplate::AddScript()");
hr = nExcept;
END_TRY
if (FAILED(hr))
{
*ppScriptElem = NULL;
return E_FAIL;
}
TRY
hr = pActiveScript->SetScriptState(SCRIPTSTATE_INITIALIZED);
CATCH(nExcept)
HandleErrorMissingFilename(IDE_SCRIPT_ENGINE_GPF,
NULL,
TRUE,
nExcept,
"IActiveScript::SetScriptState()",
"CTemplate::AddScript()");
hr = nExcept;
END_TRY
if (FAILED(hr))
return E_FAIL;
return S_OK;
}
/* ============================================================================
CTemplate::AppendMapFile
Appends a filemap to the workstore and memory-maps its file
Returns:
Nothing
Side effects:
Allocates memory; throws exception on error
*/
void
CTemplate::AppendMapFile
(
LPCTSTR szFileSpec, // file spec for this file
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
BOOLB fVirtual, // is file spec virtual or relative?
CHitObj* pHitObj, // ptr to template's hit object
BOOLB fGlobalAsa // is this file the global.asa file?
)
{
// alloc or realloc as needed
if(m_cFilemaps++ == 0)
m_rgpFilemaps = (CFileMap**) CTemplate::SmallMalloc(sizeof(CFileMap*));
else
m_rgpFilemaps = (CFileMap**) CTemplate::SmallReAlloc(m_rgpFilemaps, m_cFilemaps * sizeof(CFileMap*));
if(NULL == m_rgpFilemaps)
THROW(E_OUTOFMEMORY);
if(NULL == (m_rgpFilemaps[m_cFilemaps - 1] = new CFileMap))
THROW(E_OUTOFMEMORY);
// map the file
m_rgpFilemaps[m_cFilemaps - 1]->MapFile(
szFileSpec,
m_szApplnVirtPath,
pfilemapCurrent,
fVirtual,
pHitObj,
fGlobalAsa
);
}
/* ============================================================================
CTemplate::GetSegmentsFromFile
Gets source segments from a source file by calling ExtractAndProcessSegment
until there are no more segments; populates WorkStore with info on source segments.
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::GetSegmentsFromFile
(
CFileMap& filemap, // this file's file map
CWorkStore& WorkStore, // working storage for source segments
CHitObj* pHitObj, // Browser request object
BOOL fIsHTML
)
{
CByteRange brSearch; // byte range to search for source segments
_TOKEN rgtknOpeners[TOKEN_OPENERS_MAX]; // array of permitted open tokens
UINT ctknOpeners; // count of permitted open tokens
SOURCE_SEGMENT ssegThisFile = ssegHTML; // Either HTML or <SCRIPT> segment
BOOL fPrevCodePageSet = FALSE;
UINT wPrevCodePage;
// init search range to all of file - NOTE we ignore high dword of file size
brSearch.m_pb = filemap.m_pbStartOfFile;
brSearch.m_cb = filemap.GetSize();
if (fIsHTML)
{
// populate array of permitted open tokens
ctknOpeners = 4;
rgtknOpeners[0] = CTokenList::tknOpenPrimaryScript;
rgtknOpeners[1] = CTokenList::tknOpenTaggedScript;
rgtknOpeners[2] = CTokenList::tknOpenObject;
rgtknOpeners[3] = CTokenList::tknOpenHTMLComment;
}
else
{
ctknOpeners = 1;
rgtknOpeners[0] = CTokenList::tknOpenHTMLComment;
ssegThisFile = ssegTaggedScript;
}
TRY
if ((brSearch.m_cb >= 2)
&& (((brSearch.m_pb[0] == 0xff) && (brSearch.m_pb[1] == 0xfe))
|| ((brSearch.m_pb[0] == 0xfe) && (brSearch.m_pb[1] == 0xff)))) {
ThrowError(brSearch.m_pb,IDE_TEMPLATE_UNICODE_NOTSUP);
return;
}
// check for the UTF-8 BOM mark. If present, then treat this similar to
// seeing @CODEPAGE=65001. Note that previous values are retained in the
// event that there are differing @CODEPAGE settings. This probably should
// be an error in itself, but I can imagine that this might break a lot of
// apps as more and more UTF8 files are put into use.
if ((brSearch.m_cb >= 3)
&& (brSearch.m_pb[0] == 0xEF)
&& (brSearch.m_pb[1] == 0xBB)
&& (brSearch.m_pb[2] == 0xBF)) {
pHitObj->SetCodePage(65001);
fPrevCodePageSet = m_fCodePageSet;
wPrevCodePage = m_wCodePage;
m_fCodePageSet = TRUE;
m_wCodePage = 65001;
brSearch.Advance(3);
}
// Process source segments until we run out of them, i.e. until search segment is empty
// NOTE we pass current filemap as 'parent file' to ExtractAndProcessSegment
// NOTE ExtractAndProcessSegment appends source segments to WorkStore, advancing brSearch as it goes
while(!brSearch.IsNull())
ExtractAndProcessSegment(
brSearch,
ssegThisFile,
rgtknOpeners,
ctknOpeners,
&filemap,
WorkStore,
pHitObj,
ssegThisFile == ssegTaggedScript,
fIsHTML
);
CATCH(hrException)
/*
NOTE we indicate 'generic error' by m_idErrMsg == 0; this happens as we move
up the 'include file stack' after processing a specific error (m_idErrMsg != 0).
Only the specific error is processed; generic error, we simply re-throw exception.
*/
if(m_idErrMsg != 0)
{
// process specific error
ProcessSpecificError(filemap, pHitObj);
// reset err message so next msg will be generic as we move up the stack
m_idErrMsg = 0;
}
THROW(hrException);
END_TRY
if (fPrevCodePageSet){
m_wCodePage = wPrevCodePage;
pHitObj->SetCodePage(wPrevCodePage);
}
}
#define SZ_REG_LANGUAGE_ENGINES "SYSTEM\\CurrentControlSet\\Services\\W3SVC\\ASP\\LanguageEngines\\"
/* ============================================================================
CTemplate::GetLanguageEquivalents
Gets the "Write", "WriteBlock", etc. equivalents from registry for primary scripting language
Returns
Nothing
Side effects
Throws on error
*/
void
CTemplate::GetLanguageEquivalents
(
)
{
CByteRange brPrimaryEngine;
m_pWorkStore->m_ScriptStore.m_bufEngineNames.GetItem(0, brPrimaryEngine); // 0-th engine is primary
#if DBG
/* DEBUG ONLY - to test the reg lookup code you must:
1) create the key HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W3SVC\ASP\LanguageEngines\Debug
2) put the language reg values for WriteBlock and Write under the language you want to test
*/
HANDLE hKey;
if(ERROR_SUCCESS != RegOpenKeyExA(
HKEY_LOCAL_MACHINE,
"SYSTEM\\CurrentControlSet\\Services\\W3SVC\\ASp\\LanguageEngines\\Debug",
0,
KEY_READ,
(PHKEY)&hKey
))
{
return;
}
RegCloseKey((HKEY)hKey);
#else
// if the primary language is one of the big two, return; we don't need to look up equivalents
if(brPrimaryEngine.FMatchesSz("VBScript"))
return;
if(brPrimaryEngine.FMatchesSz("JScript"))
return;
if(brPrimaryEngine.FMatchesSz("JavaScript"))
return;
if(brPrimaryEngine.FMatchesSz("LiveScript"))
return;
#endif // DBG
/* query the registry; language equivalents are stored in:
HKEY_LOCAL_MACHINE
key: SYSTEM
key: CurrentControlSet
key: Services
key: W3SVC
key: ASP
key: LanguageEngines
key: <LanguageName>
value: Write data: <replacement syntax for Response.Write(|)>
value: WriteBlock data: <replacement syntax for Response.WriteBlock(|)>
*/
STACK_BUFFER( tempRegKeyPath, 512 );
UINT cchRegKeyPath = strlen(SZ_REG_LANGUAGE_ENGINES);
if (!tempRegKeyPath.Resize(cchRegKeyPath + brPrimaryEngine.m_cb + 1)) {
SetLastError(E_OUTOFMEMORY);
return;
}
LPSTR szRegKeyPath = static_cast<LPSTR> (tempRegKeyPath.QueryPtr());
LPSTR pch = szRegKeyPath;
strcpy(pch, SZ_REG_LANGUAGE_ENGINES);
pch += cchRegKeyPath;
strncpy(pch, (const char *) brPrimaryEngine.m_pb, brPrimaryEngine.m_cb);
pch += brPrimaryEngine.m_cb;
*pch = '\0';
HANDLE hKeyScriptLanguage; // handle of script language reg key
if(ERROR_SUCCESS == RegOpenKeyExA(
HKEY_LOCAL_MACHINE, // handle constant
(const char*) szRegKeyPath, // LPCSTR lpSubKey subkey to open
0, // DWORD ulOptions reserved; must be zero
KEY_QUERY_VALUE, // REGSAM samDesired security access mask
(PHKEY) &hKeyScriptLanguage // PHKEY phkResult address of handle of open key
))
{
SetLanguageEquivalent(hKeyScriptLanguage, "Write", &(m_pWorkStore->m_szWriteOpen), &(m_pWorkStore->m_szWriteClose));
SetLanguageEquivalent(hKeyScriptLanguage, "WriteBlock", &(m_pWorkStore->m_szWriteBlockOpen), &(m_pWorkStore->m_szWriteBlockClose));
RegCloseKey((HKEY) hKeyScriptLanguage);
}
}
/* ============================================================================
CTemplate::SetLanguageEquivalent
Sets a "language equivalent" from the registry.
Returns:
language item opener and closer as out-parameters
Ex: "Response.Write(" and ")"
Side effects:
Throws on error
*/
void
CTemplate::SetLanguageEquivalent
(
HANDLE hKeyScriptLanguage, // reg key
LPSTR szLanguageItem, // reg value name - "Write", "WriteBlock", etc.
LPSTR* pszOpen, // ptr to language item opener, e.g. "Response.Write(" (out-parameter)
LPSTR* pszClose // ptr to language item closer, e.g. ")" (out-parameter)
)
{
LONG lError;
DWORD cbSyntax;
LPSTR szSyntax;
char* pchInsert;
UINT cchOpen;
UINT cchClose;
// query registry to get buffer size
lError = RegQueryValueExA(
(HKEY) hKeyScriptLanguage, // handle of key to query
szLanguageItem, // name of value to query
NULL, // reserved; must be NULL
NULL, // ptr to value type; not required
NULL, // ptr to data buffer
&cbSyntax // ptr to data buffer size
);
if(ERROR_FILE_NOT_FOUND == lError)
// if we don't find szLanguageItem in registry, return silently, leaving *pszOpen and *pszClose unchanged
return;
else if((ERROR_MORE_DATA != lError) && (ERROR_SUCCESS != lError))
THROW(lError);
Assert(cbSyntax > 0);
// allocate buffer and re-query registry to get syntax string
// NOTE RegQueryValueEx returns cbSyntax that includes room for '\0' terminator
STACK_BUFFER(tempSyntax, 64);
if (!tempSyntax.Resize(cbSyntax)) {
THROW(E_OUTOFMEMORY);
}
szSyntax = static_cast<LPSTR> (tempSyntax.QueryPtr());
lError = RegQueryValueExA(
(HKEY) hKeyScriptLanguage, // handle of key to query
szLanguageItem, // name of value to query
NULL, // reserved; must be NULL
NULL, // ptr to value type; not required
(LPBYTE) szSyntax, // ptr to data buffer
&cbSyntax // ptr to data buffer size
);
/* NOTE there is the slight possibility of ERROR_FILE_NOT_FOUND or ERROR_MORE_DATA
if the registry value was deleted or changed between the first and second calls to RegQueryValueEx.
Since this occurs with vanishingly small probability, we throw (instead of coding the re-try logic).
*/
if(ERROR_SUCCESS != lError)
THROW(lError);
pchInsert = szSyntax;
while(*pchInsert != '|' && *pchInsert != '\0')
pchInsert++;
cchOpen = DIFF(pchInsert - szSyntax);
cchClose = *pchInsert == '|'
? cbSyntax - cchOpen - 2 // found insert symbol: deduct 2 chars, 1 for insert symbol, 1 for '\0'
: cbSyntax - cchOpen - 1; // didn't find insert symbol: deduct 1 char for '\0'
Assert(FImplies(cchOpen == 0, *szSyntax == '|'));
Assert(FImplies(*pchInsert == '\0', cchClose == 0));
if(cchOpen == 0)
// opener is empty - set caller's opener ptr null
*pszOpen = NULL;
else if(cchOpen > 0)
{
// opener is non-empty - set caller's opener to opener in registry
if(NULL == (*pszOpen = (LPSTR) CTemplate::SmallMalloc(cchOpen + 1)))
THROW(E_OUTOFMEMORY);
strncpy(*pszOpen, szSyntax, cchOpen);
(*pszOpen)[cchOpen] = '\0';
}
if(cchClose == 0)
// closer is empty - set caller's closer ptr null
*pszClose = NULL;
else if(cchClose > 0)
{
// closer is non-empty - set caller's closer to closer in registry
if(NULL == (*pszClose = (LPSTR) CTemplate::SmallMalloc(cchClose + 1)))
THROW(E_OUTOFMEMORY);
strncpy(*pszClose, (pchInsert + 1), cchClose);
(*pszClose)[cchClose] = '\0';
}
}
/* ============================================================================
CTemplate::ThrowError
Sets up for processing a compile failure.
Returns:
Nothing
Side effects:
Throws error
*/
void
CTemplate::ThrowError
(
BYTE* pbErrorLocation, // ptr to error location in source file
UINT idErrMsg // error id
)
{
m_pbErrorLocation = pbErrorLocation;
m_idErrMsg = idErrMsg;
// bug 80745: always throw compile-failed-don't-cache
THROW(E_TEMPLATE_COMPILE_FAILED_DONT_CACHE);
}
/* ============================================================================
CTemplate::AppendErrorMessageInsert
Appends an error message insert to member array.
Returns:
Nothing
Side effects:
Appends to inserts array
*/
void
CTemplate::AppendErrorMessageInsert
(
BYTE* pbInsert, // ptr to insert
UINT cbInsert // length of insert
)
{
if (m_ppszMsgInserts == NULL)
{
m_ppszMsgInserts = new char*;
m_cMsgInserts = 0;
if (m_ppszMsgInserts == NULL)
return;
}
m_ppszMsgInserts[m_cMsgInserts] = new char[cbInsert + 1];
if (m_ppszMsgInserts[m_cMsgInserts] == NULL)
return;
strncpy(m_ppszMsgInserts[m_cMsgInserts], (const char*)pbInsert, cbInsert);
m_ppszMsgInserts[m_cMsgInserts++][cbInsert] = NULL;
}
/* ============================================================================
CTemplate::ThrowErrorSingleInsert
Appends a single message insert to member array and throws a compile error.
Returns:
Nothing
Side effects:
Throws error indirectly
*/
void
CTemplate::ThrowErrorSingleInsert
(
BYTE* pbErrorLocation, // ptr to error location in source file
UINT idErrMsg, // error id
BYTE* pbInsert, // ptr to insert
UINT cbInsert // length of insert
)
{
AppendErrorMessageInsert(pbInsert, cbInsert);
ThrowError(pbErrorLocation, idErrMsg);
}
/* ============================================================================
CTemplate::ProcessSpecificError
Processes a specific compile failure.
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::ProcessSpecificError
(
CFileMap& filemap, // source file map
CHitObj* pHitObj // Browser request object
)
{
// no error msg for generic failures
if(m_idErrMsg == E_FAIL || m_idErrMsg == E_OUTOFMEMORY)
return;
HandleCTemplateError(
&filemap,
m_pbErrorLocation,
m_idErrMsg,
m_cMsgInserts,
m_ppszMsgInserts,
pHitObj
);
}
/* ============================================================================
CTemplate::ShowErrorInDebugger
Display a runtime error by invoking the JIT debugger
Returns:
failure if debugger won't start
Side effects:
None.
*/
HRESULT
CTemplate::ShowErrorInDebugger
(
CFileMap* pfilemap,
UINT cchErrorLocation,
char* szDescription,
CHitObj *pHitObj,
BOOL fAttachDocument
)
{
HRESULT hr = S_OK;
char szDebugTitle[64];
if (pfilemap == NULL || szDescription == NULL || pHitObj == NULL)
return E_POINTER;
// Create a new document context for this statement
// CONSIDER: character count that we return is bogus - however our debugging
// client (Caesar's) does not use this information anyway.
//
CTemplateDocumentContext *pDebugContext = new CTemplateDocumentContext(this, cchErrorLocation, 1);
if (pDebugContext == NULL)
return E_OUTOFMEMORY;
// Make sure debug document is attached to debugger
if (fAttachDocument)
AttachTo(pHitObj->PAppln());
// Yes it does, bring up the debugger on this line
hr = InvokeDebuggerWithThreadSwitch(g_pDebugApp, DEBUGGER_UI_BRING_DOC_CONTEXT_TO_TOP, pDebugContext);
if (FAILED(hr))
goto LExit;
// Load the compiler message string
CchLoadStringOfId(IDE_TEMPLATE_ERRMSG_TITLE, szDebugTitle, sizeof szDebugTitle);
// pop up a message box with the error description
MessageBoxA(NULL, szDescription, szDebugTitle, MB_SERVICE_NOTIFICATION | MB_TOPMOST | MB_OK | MB_ICONEXCLAMATION);
LExit:
if (pDebugContext)
pDebugContext->Release();
return hr;
}
/* ============================================================================
CTemplate::HandleCTemplateError
Handles template compilation errors
Returns:
Nothing
Side effects:
None.
*/
void
CTemplate::HandleCTemplateError
(
CFileMap* pfilemap, // ptr to source file map
BYTE* pbErrorLocation, // ptr to source location where error occurred
UINT idErrMsg, // error message id
UINT cMsgInserts, // count of insert strings for error msg
char** ppszMsgInserts, // array of ptrs to error msg insert strings
CHitObj* pHitObj // Browser Request
)
{
char szErrMsgPattern[MAX_RESSTRINGSIZE]; // error message pattern
CHAR szLineNum[12];
TCHAR szFileName[512];
CHAR szShortDes[256];
CHAR szEngine[256];
CHAR szErrCode[20];
CHAR szLongDes[MAX_RESSTRINGSIZE];
CHAR szCombinedDes[sizeof szShortDes + sizeof szLongDes]; // long & short desc
DWORD dwMask;
UINT cch;
// if request ptr or ecb ptr is null, bail; we won't be able to write error msg anyway
if(pHitObj == NULL)
return;
/* if this was a security error, process it specially and bail
NOTE security error causes exception, rather than true error id
NOTE template will be destroyed anyway in this case, so no need to maintain m_pszLastErrorMessage
*/
if(idErrMsg == E_USER_LACKS_PERMISSIONS)
{
Assert(cMsgInserts == 1);
HandleAccessFailure(pHitObj,
(m_rgpFilemaps && m_rgpFilemaps[0])? m_rgpFilemaps[0]->m_szPathTranslated : NULL);
return;
}
// get error resource message
LoadErrResString(idErrMsg, &dwMask, szErrCode, szShortDes, szLongDes);
// if we have a specific error location, construct msg prefix
if(pbErrorLocation != NULL) {
Assert(pfilemap != NULL);
// get line number of error location as string
_itoa(SourceLineNumberFromPb(pfilemap, pbErrorLocation), szLineNum, 10);
}
else {
szLineNum[0] = NULL;
}
if(pfilemap != NULL) {
cch = _tcslen(pfilemap->m_szPathInfo);
_tcsncpy(szFileName, pfilemap->m_szPathInfo, cch);
}
else {
cch = 0;
}
szFileName[cch] = '\0';
//Load Default Engine from resource
cch = CchLoadStringOfId(IDS_ENGINE, szEngine, sizeof szEngine);
szEngine[cch] = '\0';
// resolve error msg pattern and inserts into actual error msg
cch = strlen(szLongDes);
memcpy(szErrMsgPattern, szLongDes, cch);
szErrMsgPattern[cch] = '\0';
// get an idea of the possibility of a buffer overrunn
UINT dwTotalLen=0;
BOOL fTooBig = FALSE;
if (cMsgInserts) {
// allow 32 characters for space, etc.
dwTotalLen = 32 + strlen(szErrMsgPattern);
for (UINT i = 0; i < cMsgInserts; i++)
dwTotalLen += strlen(ppszMsgInserts[i]);
if (dwTotalLen > sizeof szLongDes) {
cch = CchLoadStringOfId(IDE_TOOBIG, szLongDes, sizeof szLongDes);
szLongDes[cch] = '\0';
fTooBig = TRUE;
}
}
if (!fTooBig)
GetSzFromPatternInserts(szErrMsgPattern, cMsgInserts, ppszMsgInserts, szLongDes);
// attempt to bring up debugger to display the error - if we cannot then log the error
/* Find the character offset closest to cbErrorLocation. This will be
* the place where we start looping with CharNext() to get the full
* character offset.
*
* NOTE: compilation is done in two phases.
* Errors are detected and reported in phase 1.
* The DBCS mapping is created in phase 2.
*
* Therefore, we don't have the benefit of the rgByte2DBCS table
* because it doesn't exist yet. Therefore we are left with a SLOW
* loop starting at BOF. To make things not so abysmal, we don't
* do the loop on SBCS charsets. We also don't do this conversion
* unless debugging is enabled.
*/
if (FCaesars() && pHitObj->PAppln()->FDebuggable()) {
unsigned cchErrorLocation = CharAdvDBCS(
(WORD)m_wCodePage,
reinterpret_cast<char *>(pfilemap->m_pbStartOfFile),
reinterpret_cast<char *>(pbErrorLocation),
INFINITE,
NULL);
// Create the description string
char *szEnd = strcpyExA(szCombinedDes, szShortDes);
*szEnd++ = '\n';
*szEnd++ = '\n';
strcpy(szEnd, szLongDes);
ShowErrorInDebugger(pfilemap, cchErrorLocation, szCombinedDes, pHitObj, idErrMsg != IDE_TEMPLATE_CYCLIC_INCLUDE);
}
//cache the info in case we need to use later.
m_dwLastErrorMask = dwMask;
//delay NULL check to caller who use this info.
#if UNICODE
m_pszLastErrorInfo[ILE_szFileName] = StringDupUTF8(szFileName);
#else
m_pszLastErrorInfo[ILE_szFileName] = StringDupA(szFileName);
#endif
m_pszLastErrorInfo[ILE_szLineNum] = StringDupA(szLineNum);
m_pszLastErrorInfo[ILE_szEngine] = StringDupA(szEngine);
m_pszLastErrorInfo[ILE_szErrorCode] = StringDupA(szErrCode);
m_pszLastErrorInfo[ILE_szShortDes] = StringDupA(szShortDes);
m_pszLastErrorInfo[ILE_szLongDes] = StringDupA(szLongDes);
SendToLog( m_dwLastErrorMask,
m_pszLastErrorInfo[ILE_szFileName],
m_pszLastErrorInfo[ILE_szLineNum],
m_pszLastErrorInfo[ILE_szEngine],
m_pszLastErrorInfo[ILE_szErrorCode],
m_pszLastErrorInfo[ILE_szShortDes],
m_pszLastErrorInfo[ILE_szLongDes],
pHitObj);
}
/* ============================================================================
CTemplate::FreeGoodTemplateMemory
Frees memory allocated for a 'good' (successfully compiled) template.
This includes the template itself, memory to support compile-time errors
(since the entire concatenated compile-time error message is cached in
last-err-msg member), and memory to support run-time errors (since if the
template didn't compile, it can't run).
Returns
Nothing
Side effects
None
*/
void
CTemplate::FreeGoodTemplateMemory
(
)
{
UINT i;
LargeTemplateFreeNullify((void**) &m_pbStart);
SmallTemplateFreeNullify((void**) &m_rgpSegmentFilemaps);
delete[] m_rgrgSourceInfos;
m_rgrgSourceInfos = NULL;
if(m_ppszMsgInserts)
{
for(i = 0; i < m_cMsgInserts; i++)
delete m_ppszMsgInserts[i];
delete m_ppszMsgInserts;
m_ppszMsgInserts = NULL;
}
// release the collected type libs
ReleaseTypeLibs();
// release any 449-echo-cookie objects
Release449();
}
/* ============================================================================
CTemplate::UnmapFiles
Unmaps the template's filemaps.
NOTE: we keep filemap objects around so that filenames will be available for runtime errors
Returns
Nothing
Side effects
Unmaps template's filemaps
*/
void
CTemplate::UnmapFiles
(
)
{
UINT i;
for(i = 0; i < m_cFilemaps; i++)
m_rgpFilemaps[i]->UnmapFile();
}
/*===================================================================
CTemplate::ExtractAndProcessSegment
Extracts and processes leading source segment and first contained
source segment from search range.
Returns
Nothing
Side effects
None
*/
void
CTemplate::ExtractAndProcessSegment
(
CByteRange& brSearch, // byte range to search for next segment-opening token
const SOURCE_SEGMENT& ssegLeading, // type of 'leading', i.e. pre-token, source segment
_TOKEN* rgtknOpeners, // array of permitted open tokens
UINT ctknOpeners, // count of permitted open tokens
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
CWorkStore& WorkStore, // working storage for source segments
CHitObj* pHitObj, // Browser request object
BOOL fScriptTagProcessed,// has script tag been processed?
BOOL fIsHTML // are we in HTML segment?
)
{
CByteRange brLeadSegment; // byte range of leading source segment
SOURCE_SEGMENT ssegContained; // type of 'contained', i.e. post-token, source segment
CByteRange brContainedSegment; // byte range of contained source segment
_TOKEN tknOpen; // opening token
BYTE* pbTokenOpen; // ptr to opening token
_TOKEN tknClose; // closing token
BYTE* pbTokenClose; // ptr to closing token
// NOTE: If "fScriptTagProcessed" is TRUE, then "fIsHTML" must be FALSE. The reason for
// both flags is that if "fScriptTagProcessed" is FALSE, then "fIsHTML" may be either TRUE
// or FALSE (indeterminate)
//
Assert (FImplies(fScriptTagProcessed, !fIsHTML));
// If search range is empty, return
if(brSearch.IsNull())
return;
// Set ptr of leading segment to start of search segment
brLeadSegment.m_pb = brSearch.m_pb;
// get open token for contained segment
pbTokenOpen = GetOpenToken(
brSearch,
ssegLeading,
rgtknOpeners,
ctknOpeners,
&tknOpen
);
// Set count of leading segment to distance between start of search range and token
brLeadSegment.m_cb = DIFF(pbTokenOpen - brSearch.m_pb);
// Process leading segment
ProcessSegment(ssegLeading, brLeadSegment, pfilemapCurrent, WorkStore, fScriptTagProcessed, pHitObj, fIsHTML);
// If open token was 'EOF', empty out search range and return
if(tknOpen == CTokenList::tknEOF)
{
brSearch.Nullify();
return;
}
// Set contained segment type and close token based upon the opener we found
tknClose = GetComplementToken(tknOpen);
ssegContained = GetSegmentOfOpenToken(tknOpen);
if(ssegContained == ssegHTMLComment)
// for html comment segments, advance search range to open token
// NOTE keep html comment tags in segment because they must be sent to client
brSearch.Advance(DIFF(pbTokenOpen - brSearch.m_pb));
else
// for all but html comment segments, advance search range to just past open token
gm_pTokenList->MovePastToken(tknOpen, pbTokenOpen, brSearch);
// Get closing token - if none, throw error
if(NULL == (pbTokenClose = GetCloseToken(brSearch, tknClose)))
{
if(tknOpen == CTokenList::tknOpenPrimaryScript)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NO_CLOSE_PSCRIPT);
else if(tknOpen == CTokenList::tknOpenTaggedScript)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NO_CLOSE_TSCRIPT);
else if(tknOpen == CTokenList::tknOpenObject)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NO_CLOSE_OBJECT);
else if(tknOpen == CTokenList::tknOpenHTMLComment)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NO_CLOSE_HTML_COMMENT);
}
// calc contained segment
brContainedSegment.m_pb = brSearch.m_pb;
brContainedSegment.m_cb = DIFF(pbTokenClose - brSearch.m_pb);
// advance search range to just past close token
gm_pTokenList->MovePastToken(tknClose, pbTokenClose, brSearch);
// if an html comment segment, get actual segment type (e.g. might be a server-side include command)
// NOTE call may also change contained segment byte range
if(ssegContained == ssegHTMLComment)
ssegContained = SsegFromHTMLComment(brContainedSegment);
// if an html comment segment, add its close tag to contained segment
// NOTE we keep html comment tags as part of segment so we can process like any other html segment
if(ssegContained == ssegHTMLComment)
brContainedSegment.m_cb += CCH_TOKEN(tknClose);
if(ssegContained == ssegMetadata)
{
// METADATA comments are used by DESIGN time controls and we don't send
// them to the client.
// We process metadata to get to the typelib info
UINT idError = 0;
HRESULT hr = ProcessMetadataSegment(brContainedSegment, &idError, pHitObj);
if (FAILED(hr))
ThrowError(brContainedSegment.m_pb, idError);
}
else if (ssegContained == ssegFPBot)
{
}
else
{
// process contained segment
ProcessSegment(ssegContained, brContainedSegment, pfilemapCurrent, WorkStore, fScriptTagProcessed, pHitObj, fIsHTML);
}
}
/* ============================================================================
CTemplate::SsegFromHTMLComment
Determines source segment type of HTML comment.
Returns
Source segment type
Side effects
May advance segment byte range
*/
CTemplate::SOURCE_SEGMENT
CTemplate::SsegFromHTMLComment
(
CByteRange& brSegment // source segment
)
{
SOURCE_SEGMENT ssegRet = ssegHTMLComment; // return value
BYTE* pbToken; // ptr to token
if(NULL != (pbToken = gm_pTokenList->GetToken(CTokenList::tknCommandINCLUDE, brSegment, m_wCodePage)))
{
gm_pTokenList->MovePastToken(CTokenList::tknCommandINCLUDE, pbToken, brSegment);
ssegRet = ssegInclude;
}
else if(NULL != (pbToken = gm_pTokenList->GetToken(CTokenList::tknTagMETADATA, brSegment, m_wCodePage)))
{
gm_pTokenList->MovePastToken(CTokenList::tknTagMETADATA, pbToken, brSegment);
ssegRet = ssegMetadata;
}
else if(NULL != (pbToken = gm_pTokenList->GetToken(CTokenList::tknTagFPBot, brSegment, m_wCodePage)))
{
gm_pTokenList->MovePastToken(CTokenList::tknTagFPBot, pbToken, brSegment);
ssegRet = ssegFPBot;
}
return ssegRet;
}
/* ============================================================================
CTemplate::ProcessSegment
Processes a source segment based on its type.
Returns
Nothing
Side effects
None
*/
void
CTemplate::ProcessSegment
(
SOURCE_SEGMENT sseg, // segment type
CByteRange& brSegment, // segment byte range
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
CWorkStore& WorkStore, // working storage for source segments
BOOL fScriptTagProcessed, // has script tag been processed?
CHitObj* pHitObj, // Browser request object
BOOL fIsHTML // Is segment in HTML block or script?
)
{
UINT idSequence; // sequence id for this segment
// if segment is entirely white space, silently return
if(FByteRangeIsWhiteSpace(brSegment))
return;
// set local sequence id and increment member
idSequence = WorkStore.m_idCurSequence++;
// Process segment based on its type
if(sseg == ssegHTML)
ProcessHTMLSegment(brSegment, WorkStore.m_bufHTMLSegments, idSequence, pfilemapCurrent);
else if(sseg == ssegHTMLComment)
ProcessHTMLCommentSegment(brSegment, pfilemapCurrent, WorkStore, pHitObj);
else if(sseg == ssegPrimaryScript || sseg == ssegTaggedScript)
ProcessScriptSegment(sseg, brSegment, pfilemapCurrent, WorkStore, idSequence, (BOOLB)!!fScriptTagProcessed, pHitObj);
else if(sseg == ssegObject)
ProcessObjectSegment(brSegment, pfilemapCurrent, WorkStore, idSequence);
else if(sseg == ssegInclude)
{
if (! fIsHTML)
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_SSI_COMMAND);
ProcessIncludeFile(brSegment, pfilemapCurrent, WorkStore, idSequence, pHitObj, fIsHTML);
}
// malloc/realloc array if needed
if(m_cSegmentFilemapSlots == 0)
{
m_cSegmentFilemapSlots = C_SCRIPTSEGMENTSDEFAULT + C_HTMLSEGMENTSDEFAULT;
if(NULL == (m_rgpSegmentFilemaps = (CFileMap**) CTemplate::SmallMalloc(m_cSegmentFilemapSlots * sizeof(CFileMap*))))
THROW(E_OUTOFMEMORY);
}
else if(idSequence >= m_cSegmentFilemapSlots)
{
// grab twice what we had before
m_cSegmentFilemapSlots *= 2;
if(NULL == (m_rgpSegmentFilemaps = (CFileMap**) CTemplate::SmallReAlloc(m_rgpSegmentFilemaps,
m_cSegmentFilemapSlots * sizeof(CFileMap*))))
THROW(E_OUTOFMEMORY);
}
// set filemap ptr for this segment - NOTE 'parent' filemap is also current file map
m_rgpSegmentFilemaps[idSequence] = pfilemapCurrent;
}
/* ========================================================
CTemplate::ProcessHTMLSegment
Processes an HTML segment.
Returns
Nothing
Side effects
None
*/
void
CTemplate::ProcessHTMLSegment
(
CByteRange& brHTML, // html segment
CBuffer& bufHTMLBlocks, // working storage for html blocks
UINT idSequence, // segment sequence id
CFileMap* pfilemapCurrent // current filemap
)
{
if(!(brHTML.IsNull()))
// If byte range is non-empty, store it in html buffer (non-local)
bufHTMLBlocks.Append(brHTML, FALSE, idSequence, pfilemapCurrent);
}
/* ========================================================
CTemplate::ProcessHTMLCommentSegment
Processes an HTML comment segment: within an HTML comment we
honor plain text (passed through as HTML comment) and primary script.
See bug 182 for istudio scenarios that require this behavior.
Returns
Nothing
Side effects
None
*/
void
CTemplate::ProcessHTMLCommentSegment
(
CByteRange& brSegment, // segment byte range
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
CWorkStore& WorkStore, // working storage for source segments
CHitObj* pHitObj // Browser request object
)
{
_TOKEN* rgtknOpeners; // array of permitted open tokens
UINT ctknOpeners; // count of permitted open tokens
// populate array of permitted open tokens
ctknOpeners = 1;
_TOKEN tknOpeners[1];
rgtknOpeners = tknOpeners;
rgtknOpeners[0] = CTokenList::tknOpenPrimaryScript;
// Process source segments embedded within HTML comment segment
while(!brSegment.IsNull())
ExtractAndProcessSegment(
brSegment, // byte range to search for next segment-opening token
ssegHTML, // type of 'leading', i.e. pre-token, source segment
rgtknOpeners, // array of permitted open tokens
ctknOpeners, // count of permitted open tokens
pfilemapCurrent,// ptr to filemap of parent file
WorkStore, // working storage for source segments
pHitObj // Browser request object
);
}
/* ============================================================================
CTemplate::ProcessScriptSegment
Processes a script segment.
Returns
Nothing
Side effects
None
*/
void
CTemplate::ProcessScriptSegment
(
SOURCE_SEGMENT sseg, // segment type
CByteRange& brSegment, // segment byte range
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
CWorkStore& WorkStore, // working storage for scripts
UINT idSequence, // segment sequence id
BOOLB fScriptTagProcessed,// has script tag been processed?
CHitObj* pHitObj // Browser request object
)
{
CByteRange brEngine; // script engine name - NOTE constructed null
if(m_fGlobalAsa)
if(sseg == ssegPrimaryScript)
// error out on primary script if we are processing global.asa
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_GLOBAL_PSCRIPT);
if(sseg == ssegPrimaryScript)
{
CByteRange brTemp = brSegment;
LTrimWhiteSpace(brTemp);
if(*brTemp.m_pb == '@') // CONSIDER: tknTagSetPriScriptLang
{
// impossible condition: page-level @ commands can't be allowed if they have already been executed
Assert(!(WorkStore.m_fPageCommandsAllowed && WorkStore.m_fPageCommandsExecuted));
if(!WorkStore.m_fPageCommandsAllowed)
{
if(WorkStore.m_fPageCommandsExecuted)
// error out if trying to re-execute page-level @ commands
ThrowError(brSegment.m_pb, IDE_TEMPLATE_PAGE_COMMAND_REPEATED);
else
// error out if trying to execute page-level @ commands when not allowed
ThrowError(brSegment.m_pb, IDE_TEMPLATE_PAGE_COMMAND_NOT_FIRST);
}
// if we made it here, must be allowed to execute page-level @ commands AND they have not been executed
Assert((WorkStore.m_fPageCommandsAllowed && !WorkStore.m_fPageCommandsExecuted));
/* set primary script language if required
NOTE we call GetTagName to see if LANGUAGE tag occurs in tags segment; this is somewhat wasteful,
since BrValueOfTag must simply call GetTagName again. However, this scheme is easier than changing
BrValueOfTag to return a BOOL and amending all its other callers, who don't need this info.
*/
// Flags and counters used to track and validate the @ command directive
//
int nFirstPass = 1;
int nOffset = 0;
BOOLB fTagLanguage = TRUE;
BOOLB fTagCodePage = TRUE;
BOOLB fTagLCID = TRUE;
BOOLB fTagTransacted = TRUE;
BOOLB fTagSession = TRUE;
while( GetTag( brSegment, nFirstPass) )
{
nFirstPass =2;
nOffset = 0;
if ( fTagLanguage && CompTagName( brSegment, CTokenList::tknTagLanguage ) )
{
fTagLanguage = FALSE;
brEngine = BrValueOfTag(brSegment, CTokenList::tknTagLanguage);
if ( brEngine.IsNull() )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_ENGINE_NAME);
// get prog lang id
PROGLANG_ID ProgLangId;
HRESULT hr = GetProgLangId(brEngine, &ProgLangId);
if(hr == TYPE_E_ELEMENTNOTFOUND)
// if prog lang not found, throw error
ThrowErrorSingleInsert(
brEngine.m_pb,
IDE_TEMPLATE_BAD_PROGLANG,
brEngine.m_pb,
brEngine.m_cb
);
else if(FAILED(hr))
// other failure: re-throw exception code
THROW(hr);
Assert(WorkStore.m_ScriptStore.CountPreliminaryEngines() >= 1);
// Set 0-th (primary) script engine to user-specified value
WorkStore.m_ScriptStore.m_bufEngineNames.SetItem(
0, // index of item to set
brEngine, // engine name
FALSE, // item is non-local
0, // sequence id (don't care)
NULL // filemap ptr (don't care)
);
// Set 0-th (primary) prog lang id to engine's
WorkStore.m_ScriptStore.m_rgProgLangId[0] = ProgLangId;
brSegment.Advance(DIFF(brEngine.m_pb - brSegment.m_pb));
}
/* set code page if required
see NOTE above for why we call we call GetTagName.
*/
else if ( fTagCodePage && CompTagName( brSegment, CTokenList::tknTagCodePage ) )
{
fTagCodePage = FALSE;
CByteRange brCodePage = BrValueOfTag( brSegment, CTokenList::tknTagCodePage );
if ( brCodePage.IsNull() )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_CODEPAGE);
if ( brCodePage.m_cb > 10 )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_CODEPAGE);
char szCodePage[31];
strncpy( szCodePage, (char*) brCodePage.m_pb, brCodePage.m_cb );
szCodePage[ brCodePage.m_cb ] = '\0';
char *pchEnd;
UINT uCodePage = UINT( strtoul( szCodePage, &pchEnd, 10 ) );
// verify that pchEnd is the NULL
if (*pchEnd != 0)
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_CODEPAGE);
if ( FAILED( pHitObj->SetCodePage( uCodePage ) ) )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_CODEPAGE);
else
{
m_wCodePage = uCodePage;
m_fCodePageSet = TRUE;
}
brSegment.Advance(DIFF(brCodePage.m_pb - brSegment.m_pb));
}
/* set LCID if required
see NOTE above for why we call we call GetTagName.
*/
else if ( fTagLCID && CompTagName( brSegment, CTokenList::tknTagLCID ) )
{
fTagLCID = FALSE;
CByteRange brLCID = BrValueOfTag( brSegment, CTokenList::tknTagLCID );
if ( brLCID.IsNull() )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_LCID);
char szLCID[31];
strncpy( szLCID, (char*) brLCID.m_pb, brLCID.m_cb );
szLCID[ brLCID.m_cb ] = '\0';
char *pchEnd;
UINT uLCID = UINT( strtoul( szLCID, &pchEnd, 10 ) );
// verify that pchEnd is the NULL
if (*pchEnd != 0)
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_LCID);
if ( FAILED( pHitObj->SetLCID( uLCID ) ) )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_LCID);
else
{
m_lLCID = uLCID;
m_fLCIDSet = TRUE;
}
brSegment.Advance(DIFF(brLCID.m_pb - brSegment.m_pb));
}
/* Set transacted if requiured
see NOTE above for why we call GetTagName
*/
else if ( fTagTransacted && CompTagName( brSegment, CTokenList::tknTagTransacted ) )
{
STACK_BUFFER( tempTransValue, 32 );
fTagTransacted = FALSE;
CByteRange brTransacted = BrValueOfTag( brSegment, CTokenList::tknTagTransacted );
if ( brTransacted.IsNull() )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_TRANSACTED_VALUE);
if (!tempTransValue.Resize(brTransacted.m_cb + 1)) {
ThrowError(brSegment.m_pb, IDE_OOM);
}
LPSTR szTransacted = static_cast<LPSTR> (tempTransValue.QueryPtr());
strncpy(szTransacted, (LPCSTR)brTransacted.m_pb, brTransacted.m_cb);
szTransacted[brTransacted.m_cb]='\0';
if (!strcmpi(szTransacted, "REQUIRED"))
m_ttTransacted = ttRequired;
else if (!strcmpi(szTransacted, "REQUIRES_NEW"))
m_ttTransacted = ttRequiresNew;
else if (!strcmpi(szTransacted, "SUPPORTED"))
m_ttTransacted = ttSupported;
else if (!strcmpi(szTransacted, "NOT_SUPPORTED"))
m_ttTransacted = ttNotSupported;
else
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_TRANSACTED_VALUE);
brSegment.Advance(DIFF(brTransacted.m_pb - brSegment.m_pb));
}
/* Set session flag
see NOTE above for why we call GetTagName
*/
else if ( fTagSession && CompTagName( brSegment, CTokenList::tknTagSession ) )
{
STACK_BUFFER( tempSession, 16 );
fTagSession = FALSE;
CByteRange brSession = BrValueOfTag( brSegment, CTokenList::tknTagSession );
if ( brSession.IsNull() )
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_SESSION_VALUE);
if (!tempSession.Resize(brSession.m_cb + 1))
ThrowError(brSegment.m_pb, IDE_OOM);
LPSTR szSession = static_cast<LPSTR> (tempSession.QueryPtr());
strncpy(szSession, (LPCSTR)brSession.m_pb, brSession.m_cb);
szSession[brSession.m_cb]='\0';
if (strcmpi(szSession, "TRUE") == 0)
{
m_fSession = TRUE;
if (!pHitObj->PAppln()->QueryAppConfig()->fAllowSessionState())
ThrowError(brSegment.m_pb, IDE_TEMPLATE_CANT_ENABLE_SESSIONS);
}
else if (strcmpi(szSession, "FALSE") == 0)
m_fSession = FALSE;
else
ThrowError(brSegment.m_pb, IDE_TEMPLATE_BAD_SESSION_VALUE);
brSegment.Advance(DIFF(brSession.m_pb - brSegment.m_pb));
}
else
ThrowErrorSingleInsert( brSegment.m_pb,
IDE_TEMPLATE_BAD_AT_COMMAND,
brSegment.m_pb,
brSegment.m_cb
);
}
if (nFirstPass == 1)
ThrowErrorSingleInsert( brSegment.m_pb,
IDE_TEMPLATE_BAD_AT_COMMAND,
brSegment.m_pb,
brSegment.m_cb
);
// set flag true and ignore remainder of segment, since we only use this segment for page-level @ commands
WorkStore.m_fPageCommandsExecuted = TRUE;
goto LExit;
}
}
if(sseg == ssegTaggedScript)
{
if(!fScriptTagProcessed)
{
/* semantics of script-tag-processed flag:
- if false, we have a 'fresh' tagged script block, so we need to get its engine name
(which also advances past the script tag header) and then process the tagged segment
via indirect recursive call
- if true, we have aleady been called recursively, so we bypass further recursion
and simply append to store
*/
CByteRange brIncludeFile;
GetScriptEngineOfSegment(brSegment, WorkStore.m_brCurEngine, brIncludeFile);
if (! brIncludeFile.IsNull())
{
STACK_BUFFER( tempInclude, 256 );
if (!tempInclude.Resize(brIncludeFile.m_cb + 1)) {
ThrowError(brSegment.m_pb, IDE_OOM);
}
// Create Null-terminated string from brIncludeFile
char *szFileSpec = reinterpret_cast<char *>(tempInclude.QueryPtr());
memcpy(szFileSpec, brIncludeFile.m_pb, brIncludeFile.m_cb);
szFileSpec[brIncludeFile.m_cb] = 0;
if (szFileSpec[0] == '\\') // metabase stuff chokes on initial '\' char
szFileSpec[0] = '/';
// read the include file (szFileSpec & brIncludeFile in this case point to same string contents.
// however, "brIncludeFile" is used as an error location.
//
TRY
ProcessIncludeFile2(szFileSpec, brIncludeFile, pfilemapCurrent, WorkStore, idSequence, pHitObj, FALSE);
CATCH(hrException)
// The TRY/CATCH below may re-throw a IDE_TEMPLATE_BAD_PROGLANG when the
// segment being processed is tagged script with a SRC file. The reason being
// that to properly report the error, the ThrowErrorSingleInsert must be called
// from the template which contained the script tag with the bad prog lang. If
// called from the template created containing the included script, then the
// brEngine assigned below is not pointing into the included script's filemap
// which results in AVs because we can't do the pointer math to determine the
// line number.
if(hrException == IDE_TEMPLATE_BAD_PROGLANG)
// exception code is really an error message id: set err id to it
ThrowErrorSingleInsert(
WorkStore.m_brCurEngine.m_pb,
IDE_TEMPLATE_BAD_PROGLANG,
WorkStore.m_brCurEngine.m_pb,
WorkStore.m_brCurEngine.m_cb
);
else
// other exception: re-throw
THROW(hrException);
END_TRY
// done - don't process script text
return;
}
else
ProcessTaggedScriptSegment(brSegment, pfilemapCurrent, WorkStore, pHitObj);
}
brEngine = WorkStore.m_brCurEngine;
}
TRY
// append script segment to store
WorkStore.m_ScriptStore.AppendScript(brSegment, brEngine, (sseg == ssegPrimaryScript), idSequence, pfilemapCurrent);
CATCH(hrException)
// NOTE exception code from AppendScript() is overloaded: it can be an error message id or a true exception
// if the brEngine does not point to memory within the current filemap, then
// we must have come into here because of a tagged script statement with a SRC=
// attrib. In which case, we won't call ThrowError from here but will re-throw
// the error to be caught above.
if((hrException == IDE_TEMPLATE_BAD_PROGLANG)
&& (brEngine.m_pb >= pfilemapCurrent->m_pbStartOfFile)
&& (brEngine.m_pb < (pfilemapCurrent->m_pbStartOfFile + pfilemapCurrent->GetSize()))) {
// exception code is really an error message id: set err id to it
ThrowErrorSingleInsert(
brEngine.m_pb,
IDE_TEMPLATE_BAD_PROGLANG,
brEngine.m_pb,
brEngine.m_cb
);
}
else
// other exception: re-throw
THROW(hrException);
END_TRY
LExit:
// set flag to say we can no longer set primary language (must be in first script segment, if at all)
WorkStore.m_fPageCommandsAllowed = FALSE;
}
/* ========================================================
CTemplate::ProcessMetadataSegment
Parses the metadata comment for typelib information.
Returns
HRESULT
*/
HRESULT
CTemplate::ProcessMetadataSegment
(
const CByteRange& brSegment,
UINT *pidError,
CHitObj *pHitObj
)
{
// TYPELIB
if (FTagHasValue(brSegment,
CTokenList::tknTagType,
CTokenList::tknValueTypeLib))
{
return ProcessMetadataTypelibSegment(brSegment, pidError, pHitObj);
}
// METADATA INVALID in Global.asa
else if (m_fGlobalAsa)
{
ThrowError(brSegment.m_pb, IDE_TEMPLATE_METADATA_IN_GLOBAL_ASA);
return E_TEMPLATE_COMPILE_FAILED_DONT_CACHE; // to keep compiler happy; in reality doesn't return.
}
// COOKIE
else if (FTagHasValue(brSegment,
CTokenList::tknTagType,
CTokenList::tknValueCookie))
{
return ProcessMetadataCookieSegment(brSegment, pidError, pHitObj);
}
// Ignore everything else
else
{
return S_OK;
}
}
/* ========================================================
CTemplate::ProcessMetadataTypelibSegment
Parses the metadata comment for typelib information.
Returns
HRESULT
*/
HRESULT
CTemplate::ProcessMetadataTypelibSegment
(
const CByteRange& brSegment,
UINT *pidError,
CHitObj *pHitObj
)
{
// Ignore ENDSPAN segments
if (GetTagName(brSegment, CTokenList::tknTagEndspan))
{
// ENDSPAN found - ignore
return S_OK;
}
HRESULT hr;
char szFile[MAX_PATH+1];
DWORD cbFile;
// Try to get the filename
CByteRange br = BrValueOfTag(brSegment, CTokenList::tknTagFile);
if (!br.IsNull())
{
// filename present
if (br.m_cb > MAX_PATH)
{
// file too long
*pidError = IDE_TEMPLATE_BAD_TYPELIB_SPEC;
return E_FAIL;
}
memcpy(szFile, br.m_pb, br.m_cb);
cbFile = br.m_cb;
szFile[cbFile] = '\0';
}
else
{
// No filename - use GUID, version, LCID to get file
char szUUID[44]; // {} + hex chars + dashes
char szVers[16]; // "1.0", etc
char szLCID[16]; // locale id - a number
br = BrValueOfTag(brSegment, CTokenList::tknTagUUID);
if (br.IsNull() || br.m_cb > sizeof(szUUID)-3)
{
// no filename and no uuid -> invalid typelib spec
*pidError = IDE_TEMPLATE_BAD_TYPELIB_SPEC;
return E_FAIL;
}
if (br.m_pb[0] == '{')
{
// already in braces
memcpy(szUUID, br.m_pb, br.m_cb);
szUUID[br.m_cb] = '\0';
}
else
{
// enclose in {}
szUUID[0] = '{';
memcpy(szUUID+1, br.m_pb, br.m_cb);
szUUID[br.m_cb+1] = '}';
szUUID[br.m_cb+2] = '\0';
}
// Optional Version
szVers[0] = '\0';
br = BrValueOfTag(brSegment, CTokenList::tknTagVersion);
if (!br.IsNull() && br.m_cb < sizeof(szVers)-1)
{
memcpy(szVers, br.m_pb, br.m_cb);
szVers[br.m_cb] = '\0';
}
// Optional LCID
LCID lcid;
br = BrValueOfTag(brSegment, CTokenList::tknTagLCID);
if (!br.IsNull() && br.m_cb < sizeof(szLCID)-1)
{
memcpy(szLCID, br.m_pb, br.m_cb);
szLCID[br.m_cb] = '\0';
lcid = strtoul(szLCID, NULL, 16);
}
else
{
// if the LCID is not defined -> use system's
lcid = GetSystemDefaultLCID();
}
// Get TYPELIB filename from registry
hr = GetTypelibFilenameFromRegistry
(
szUUID,
szVers,
lcid,
szFile,
MAX_PATH
);
if (FAILED(hr))
{
*pidError = IDE_TEMPLATE_BAD_TYPELIB_REG_SPEC;
return hr;
}
cbFile = strlen(szFile);
}
// Convert filename to double-byte to call LoadTypeLib()
STACK_BUFFER( tempFile, MAX_PATH * sizeof(WCHAR) );
if (!tempFile.Resize((cbFile+1) * sizeof(WCHAR))) {
*pidError = IDE_OOM;
return E_FAIL;
}
LPWSTR wszFile = (LPWSTR)tempFile.QueryPtr();
if (MultiByteToWideChar(pHitObj->GetCodePage(), MB_ERR_INVALID_CHARS,
szFile, cbFile, wszFile, cbFile) == 0)
{
*pidError = IDE_OOM;
return E_FAIL;
}
wszFile[cbFile] = L'\0';
// LoadTypeLib() to get ITypeLib*
ITypeLib *ptlb = NULL;
hr = LoadTypeLib(wszFile, &ptlb);
if (FAILED(hr))
{
*pidError = IDE_TEMPLATE_LOAD_TYPELIB_FAILED;
return hr;
}
// Remember ITypeLib* in the array
Assert(ptlb);
hr = m_rgpTypeLibs.append(ptlb);
if (FAILED(hr))
{
*pidError = IDE_TEMPLATE_LOAD_TYPELIB_FAILED;
return hr;
}
return S_OK;
}
/* ========================================================
CTemplate::ProcessMetadataCookieSegment
Parses the metadata comment for cookie information.
Returns
HRESULT
*/
HRESULT
CTemplate::ProcessMetadataCookieSegment
(
const CByteRange& brSegment,
UINT *pidError,
CHitObj *pHitObj
)
{
HRESULT hr;
CByteRange br;
char *pszName;
char szFile[MAX_PATH+1];
TCHAR sztFile[MAX_PATH+1];
CMBCSToWChar convStr;
STACK_BUFFER( tempCookie, 64 );
STACK_BUFFER( tempFile, 64 );
// Try to get the cookie name
br = BrValueOfTag(brSegment, CTokenList::tknTagName);
if (br.IsNull() || (br.m_cb == 0)) {
*pidError = IDE_TEMPLATE_BAD_COOKIE_SPEC_NAME;
return E_FAIL;
}
if (!tempCookie.Resize(br.m_cb + 1)) {
*pidError = IDE_OOM;
return E_FAIL;
}
pszName = (char *)tempCookie.QueryPtr();
if (!pszName)
{
*pidError = IDE_OOM;
return E_FAIL;
}
memcpy(pszName, br.m_pb, br.m_cb);
pszName[br.m_cb] = '\0';
// Try to get the path to the script
br = BrValueOfTag(brSegment, CTokenList::tknTagSrc);
if (br.IsNull() || (br.m_cb >= MAX_PATH) || (br.m_cb == 0))
{
*pidError = IDE_TEMPLATE_BAD_COOKIE_SPEC_SRC;
return E_FAIL;
}
memcpy(szFile, br.m_pb, br.m_cb);
szFile[br.m_cb] = '\0';
// Convert file to physical path
Assert(pHitObj->PServer());
WCHAR *pCookieFile;
#if _IIS_5_1
// just use CP_ACP for 5.1 since the Core can't handle anything else anyway
if (FAILED (convStr.Init (szFile))) {
#else
// 6.0 can handle UNICODE. Convert using script code page
if (FAILED (convStr.Init (szFile,pHitObj->GetCodePage()))) {
#endif
*pidError = IDE_OOM;
return E_FAIL;
}
pCookieFile = convStr.GetString();
if (FAILED(pHitObj->PServer()->MapPathInternal(0, pCookieFile, sztFile)))
{
*pidError = IDE_TEMPLATE_BAD_COOKIE_SPEC_SRC;
return E_FAIL;
}
Normalize(sztFile);
// Construct 449-echo-cookie object
C449Cookie *p449 = NULL;
hr = Create449Cookie(pszName, sztFile, &p449);
if (FAILED(hr))
{
*pidError = IDE_TEMPLATE_LOAD_COOKIESCRIPT_FAILED;
return hr;
}
// Remember 449 cookie in the array
Assert(p449);
hr = m_rgp449.append(p449);
if (FAILED(hr)) {
*pidError = IDE_TEMPLATE_LOAD_COOKIESCRIPT_FAILED;
return hr;
}
return S_OK;
}
/* ========================================================
CTemplate::GetScriptEngineOfSegment
Returns script engine name for a script segment.
Returns
Byte range containing script engine name
Side effects
Advances segment byte range past close tag token
*/
void
CTemplate::GetScriptEngineOfSegment
(
CByteRange& brSegment, // segment byte range
CByteRange& brEngine, // script engine name
CByteRange& brInclude // value of SRC tag
)
{
BYTE* pbCloseTag; // ptr to close of start tag
// tags contained in start tag
CByteRange brTags = BrTagsFromSegment(brSegment, CTokenList::tknCloseTaggedScript, &pbCloseTag);
// if no close found, throw error
if(pbCloseTag == NULL)
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_CLOSE_TSCRIPT);
Assert(FTagHasValue(brTags, CTokenList::tknTagRunat, CTokenList::tknValueServer));
// get engine name from tags
brEngine = BrValueOfTag(brTags, CTokenList::tknTagLanguage);
if(brEngine.IsNull())
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_ENGINE_NAME);
// Get SRC attribute from tags
brInclude = BrValueOfTag(brTags, CTokenList::tknTagSrc);
// advance segment past close tag token
gm_pTokenList->MovePastToken(CTokenList::tknCloseTag, pbCloseTag, brSegment);
}
/* ========================================================
CTemplate::ProcessTaggedScriptSegment
Processes a tagged script segment: within tagged script we
honor plain text (passed through as script text) and HTML comments.
See bug 423 for istudio scenarios that require this behavior.
Returns
Nothing
Side effects
None
*/
void
CTemplate::ProcessTaggedScriptSegment
(
CByteRange& brSegment, // segment byte range
CFileMap* pfilemapCurrent,// ptr to filemap of parent file
CWorkStore& WorkStore, // working storage for source segments
CHitObj* pHitObj // Browser request object
)
{
_TOKEN* rgtknOpeners; // array of permitted open tokens
_TOKEN tknOpeners[1];
UINT ctknOpeners; // count of permitted open tokens
// populate array of permitted open tokens
ctknOpeners = 1;
rgtknOpeners = tknOpeners;
rgtknOpeners[0] = CTokenList::tknOpenHTMLComment;
// Process source segments embedded within tagged script segment
while(!brSegment.IsNull())
ExtractAndProcessSegment(
brSegment, // byte range to search for next segment-opening token
ssegTaggedScript, // type of 'leading', i.e. pre-token, source segment
rgtknOpeners, // array of permitted open tokens
ctknOpeners, // count of permitted open tokens
pfilemapCurrent, // ptr to filemap of parent file
WorkStore, // working storage for source segments
pHitObj, // Browser request object
TRUE, // script tag has been processed
FALSE // NOT in HTML segment
);
}
/* ============================================================================
CTemplate::ProcessObjectSegment
Processes an object segment.
Returns
Nothing
Side effects
throws on error
*/
void
CTemplate::ProcessObjectSegment
(
CByteRange& brSegment, // segment byte range
CFileMap* pfilemapCurrent,// ptr to filemap of parent file
CWorkStore& WorkStore, // working storage for source segments
UINT idSequence // segment sequence id
)
{
BYTE* pbCloseTag; // ptr to close of start tag
// tags contained in start tag
CByteRange brTags = BrTagsFromSegment(brSegment, CTokenList::tknCloseObject, &pbCloseTag);
// if no close found, bail on error
if(pbCloseTag == NULL)
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_CLOSE_OBJECT);
// if this is a server object (RUNAT=Server), process its tags
if(FTagHasValue(brTags, CTokenList::tknTagRunat, CTokenList::tknValueServer))
{
CLSID ClsId; // clsid
// get name value
CByteRange brName = BrValueOfTag(brTags, CTokenList::tknTagID);
// if name is null, error out
if(brName.IsNull())
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_OBJECT_NAME);
if(!FValidObjectName(brName))
ThrowErrorSingleInsert(brName.m_pb, IDE_TEMPLATE_INVALID_OBJECT_NAME, brName.m_pb, brName.m_cb);
// get values for ClassID and ProgID tags
CByteRange brClassIDText = BrValueOfTag(brTags, CTokenList::tknTagClassID);
CByteRange brProgIDText = BrValueOfTag(brTags, CTokenList::tknTagProgID);
if(!brClassIDText.IsNull())
// if we find a text class id, set clsid with it
// NOTE progid tag is ignored if classid tag exists
GetCLSIDFromBrClassIDText(brClassIDText, &ClsId);
else if(!brProgIDText.IsNull())
// else if we find a prog id, resolve it into a class id
GetCLSIDFromBrProgIDText(brProgIDText, &ClsId);
else
// else, throw error; can't create an object without at least one of classid or progid
ThrowErrorSingleInsert(brTags.m_pb, IDE_TEMPLATE_NO_CLASSID_PROGID, brName.m_pb, brName.m_cb);
// set scope; bail if bogus
CompScope csScope = csUnknown;
CByteRange brScope = BrValueOfTag(brTags, CTokenList::tknTagScope);
if(brScope.FMatchesSz(SZ_TOKEN(CTokenList::tknValuePage)) || brScope.IsNull())
// non-existent scope tag defaults to page scope
csScope = csPage;
else if(brScope.FMatchesSz(SZ_TOKEN(CTokenList::tknValueApplication)))
csScope = csAppln;
else if(brScope.FMatchesSz(SZ_TOKEN(CTokenList::tknValueSession)))
csScope = csSession;
else
ThrowError(brTags.m_pb, IDE_TEMPLATE_BAD_OBJECT_SCOPE);
if(!m_fGlobalAsa && csScope != csPage)
// error out on non-page-level object if we are processing anything but global.asa
ThrowErrorSingleInsert(brTags.m_pb, IDE_TEMPLATE_BAD_PAGE_OBJECT_SCOPE, brName.m_pb, brName.m_cb);
else if(m_fGlobalAsa && csScope == csPage)
// error out on page-level object if we are processing global.asa
ThrowErrorSingleInsert(brTags.m_pb, IDE_TEMPLATE_BAD_GLOBAL_OBJECT_SCOPE, brName.m_pb, brName.m_cb);
// set threading model
CompModel cmModel = cmSingle;
CompModelFromCLSID(ClsId, &cmModel);
// append object-info to store
WorkStore.m_ObjectInfoStore.AppendObject(brName, ClsId, csScope, cmModel, idSequence);
}
}
/* ============================================================================
CTemplate::GetCLSIDFromBrClassIDText
Sets a clsid from a byte range containing an ANSI text version of a class id
Returns
ptr to clsid (out-parameter)
Side effects
throws on error
*/
void
CTemplate::GetCLSIDFromBrClassIDText
(
CByteRange& brClassIDText,
LPCLSID pclsid
)
{
// if class id text starts with its standard object tag prefix, advance past it
if(!_strnicmp((char*)brClassIDText.m_pb, "clsid:", 6))
brClassIDText.Advance(6);
// if class id text is bracketed with {}, adjust byte range to strip them
// NOTE we always add {} below, because normal case is that they are missing from input text
if(*brClassIDText.m_pb == '{')
brClassIDText.Advance(1);
if(*(brClassIDText.m_pb + brClassIDText.m_cb - 1) == '}')
brClassIDText.m_cb--;
// Allocate a wide char string for the string version of class id
// NOTE we add 3 characters to hold {} and null terminator
OLECHAR* pszWideClassID = new WCHAR[brClassIDText.m_cb + 3];
if (NULL == pszWideClassID)
THROW(E_OUTOFMEMORY);
// start wide string class id with left brace
pszWideClassID[0] = '{';
// Convert the string class id to wide chars
if (0 == MultiByteToWideChar( CP_ACP, // ANSI code page
MB_ERR_INVALID_CHARS, // err on invalid chars
(LPCSTR)brClassIDText.m_pb, // input ANSI string version of class id
brClassIDText.m_cb, // length of input string
pszWideClassID + 1, // location for output wide string class id
brClassIDText.m_cb // size of output buffer
))
{
delete [] pszWideClassID;
THROW(E_FAIL);
}
// append right brace to wide string
pszWideClassID[brClassIDText.m_cb + 1] = '}';
// Null terminate the wide string
pszWideClassID[brClassIDText.m_cb + 2] = NULL;
// Now get the clsid from wide string class id
if(FAILED(CLSIDFromString(pszWideClassID, pclsid)))
{
delete [] pszWideClassID;
ThrowErrorSingleInsert(brClassIDText.m_pb, IDE_TEMPLATE_BAD_CLASSID, brClassIDText.m_pb, brClassIDText.m_cb);
}
if(NULL != pszWideClassID)
delete [] pszWideClassID;
}
/* ===================================================================
CTemplate::GetCLSIDFromBrProgIDText
Gets a clsid from the registry given a ProgID
Returns
ptr to clsid (out-parameter)
Side effects
throws on error
*/
void
CTemplate::GetCLSIDFromBrProgIDText
(
CByteRange& brProgIDText,
LPCLSID pclsid
)
{
// allocate a wide char string for ProgID plus null terminator
OLECHAR* pszWideProgID = new WCHAR[brProgIDText.m_cb + 1];
if (NULL == pszWideProgID)
THROW(E_OUTOFMEMORY);
// Convert the string class id to wide chars
if (0 == MultiByteToWideChar( CP_ACP, // ANSI code page
MB_ERR_INVALID_CHARS, // err on invalid chars
(LPCSTR)brProgIDText.m_pb, // input ANSI string version of prog id
brProgIDText.m_cb, // length of input string
pszWideProgID, // location for output wide string prog id
brProgIDText.m_cb // size of output buffer
))
{
delete [] pszWideProgID; pszWideProgID = NULL;
THROW(E_FAIL);
}
// Null terminate the wide string
pszWideProgID[brProgIDText.m_cb] = NULL;
// Now get clsid from ProgID
if(FAILED(CLSIDFromProgID(pszWideProgID, pclsid)))
{
delete [] pszWideProgID; pszWideProgID = NULL;
ThrowErrorSingleInsert(brProgIDText.m_pb, IDE_TEMPLATE_BAD_PROGID, brProgIDText.m_pb, brProgIDText.m_cb);
}
// Cache ProgId to CLSID mapping
g_TypelibCache.RememberProgidToCLSIDMapping(pszWideProgID, *pclsid);
if (NULL != pszWideProgID)
delete [] pszWideProgID;
}
/* ============================================================================
CTemplate::FValidObjectName
Determines whether an object name clashes with a Denali intrinsic object name.
Returns
TRUE or FALSE
Side effects
None
*/
BOOLB
CTemplate::FValidObjectName
(
CByteRange& brName // object name
)
{
if(brName.FMatchesSz(SZ_OBJ_APPLICATION))
return FALSE;
if(brName.FMatchesSz(SZ_OBJ_REQUEST))
return FALSE;
if(brName.FMatchesSz(SZ_OBJ_RESPONSE))
return FALSE;
if(brName.FMatchesSz(SZ_OBJ_SERVER))
return FALSE;
if(brName.FMatchesSz(SZ_OBJ_CERTIFICATE))
return FALSE;
if(brName.FMatchesSz(SZ_OBJ_SESSION))
return FALSE;
if(brName.FMatchesSz(SZ_OBJ_SCRIPTINGNAMESPACE))
return FALSE;
return TRUE;
}
/* ============================================================================
CTemplate::ProcessIncludeFile
Processes an include file.
Returns
Nothing
*/
void
CTemplate::ProcessIncludeFile
(
CByteRange& brSegment, // segment byte range
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
CWorkStore& WorkStore, // current working storage
UINT idSequence, // sequence #
CHitObj* pHitObj, // Browser request object pointer
BOOL fIsHTML
)
{
CByteRange brFileSpec; // filespec of include file
BOOLB fVirtual = FALSE; // is include filespec virtual?
// filespec of include file (sz)
CHAR szFileSpec[MAX_PATH + 1];
LPSTR szTemp = szFileSpec; // temp ptr to filespec
// get value of FILE tag
brFileSpec = BrValueOfTag(brSegment, CTokenList::tknTagFile);
if(brFileSpec.IsNull())
{
// if we found no FILE tag, get value of VIRTUAL tag
brFileSpec = BrValueOfTag(brSegment, CTokenList::tknTagVirtual);
fVirtual = TRUE;
}
if(brFileSpec.IsNull())
// if we found neither, error out
ThrowError(brSegment.m_pb, IDE_TEMPLATE_NO_INCLUDE_NAME);
if (brFileSpec.m_cb>MAX_PATH)
{
// return the last MAX_PATH chars of the file name. This is done
// this way to avoid a Error Too Long message when the include
// file spec is exceedingly long.
char fileNameLast[MAX_PATH+4];
strcpy(fileNameLast, "...");
strcpy(fileNameLast+3, (LPSTR)(brFileSpec.m_pb+brFileSpec.m_cb-MAX_PATH));
ThrowErrorSingleInsert(brFileSpec.m_pb,
IDE_TEMPLATE_BAD_INCLUDE,
brFileSpec.m_pb,
brFileSpec.m_cb);
}
// NOTE we manipulate temp sz to preserve szFileSpec
if(fVirtual)
{
if(*brFileSpec.m_pb == '\\')
{
// if VIRTUAL path starts with backslash, replace it with fwd slash
*szTemp++ = '/';
brFileSpec.Advance(1);
}
else if(*brFileSpec.m_pb != '/')
// if VIRTUAL path starts with anything other than fwd slash or backslash, prepend fwd slash
*szTemp++ = '/';
}
// append supplied path to temp sz
strncpy(szTemp, (LPCSTR) brFileSpec.m_pb, brFileSpec.m_cb);
szTemp[brFileSpec.m_cb] = NULL;
if(!fVirtual)
{
// if FILE filespec starts with \ or /, hurl
if(*szFileSpec == '\\' || *szFileSpec == '/')
ThrowErrorSingleInsert(
brFileSpec.m_pb,
IDE_TEMPLATE_BAD_FILE_TAG,
brFileSpec.m_pb,
brFileSpec.m_cb
);
}
// NOTE: szFileSpec is the doctored path (it possibly has "/" prepended.
// brFileSpec is used as the error location.
//
ProcessIncludeFile2(szFileSpec, brFileSpec, pfilemapCurrent, WorkStore, idSequence, pHitObj, fIsHTML);
}
/* ============================================================================
CTemplate::ProcessIncludeFile2
adds a #include file to the CTemplate and starts the template to processing
the file.
Returns
Nothing
Side effects
Calls GetSegmentsFromFile recursively
NOTE - kind of an oddball thing here. The szFileSpec in this case is
intentionally ANSI as it came from the ASP script content. It may need
to be converted to UNICODE.
*/
void
CTemplate::ProcessIncludeFile2
(
CHAR * szAnsiFileSpec, // file to include
CByteRange& brErrorLocation, // ByteRange in source where errors should be reported
CFileMap* pfilemapCurrent, // ptr to filemap of parent file
CWorkStore& WorkStore, // current working storage
UINT idSequence, // sequence #
CHitObj* pHitObj, // Browser request object pointer
BOOL fIsHTML
)
{
HRESULT hr;
TCHAR *szFileSpec;
#if UNICODE
CMBCSToWChar convFileSpec;
if (FAILED(hr = convFileSpec.Init(szAnsiFileSpec, pHitObj->GetCodePage()))) {
THROW(hr);
}
szFileSpec = convFileSpec.GetString();
#else
szFileSpec = szAnsiFileSpec;
#endif
// if parent paths are disallowed and filespec contains parent dir reference, hurl
if (!pHitObj->QueryAppConfig()->fEnableParentPaths() && _tcsstr(szFileSpec, _T("..")))
ThrowErrorSingleInsert(
brErrorLocation.m_pb,
IDE_TEMPLATE_DISALLOWED_PARENT_PATH,
brErrorLocation.m_pb,
brErrorLocation.m_cb
);
TRY
AppendMapFile(
szFileSpec,
pfilemapCurrent,
(szFileSpec[0] == _T('/')) || (szFileSpec[0] == _T('\\')), // fVirtual
pHitObj, // main file's hit object
FALSE // not the global.asa file
);
CATCH(hrException)
// MapFile() threw an exception: delete last filemap's memory and decrement filemap counter
// NOTE this is a bit hokey, but we need to do it here rather than AppendMapFile (where we allocated)
// because its other caller(s) may not want this behavior
delete m_rgpFilemaps[m_cFilemaps-- - 1];
/* NOTE exception code from MapFile() is overloaded: it can sometimes
be an error message id, sometimes a true exception
NOTE security error causes exception E_USER_LACKS_PERMISSIONS, rather than error id,
but we pass it thru as if it were an error id because the various error-catch routines
know how to handle E_USER_LACKS_PERMISSIONS specially.
*/
UINT idErrMsg;
if(hrException == IDE_TEMPLATE_CYCLIC_INCLUDE || hrException == E_USER_LACKS_PERMISSIONS)
// exception code is really an error message id: set err id to it
idErrMsg = hrException;
else if(hrException == E_COULDNT_OPEN_SOURCE_FILE)
// exception is generic couldn't-open-file : set err id to generic bad-file error
idErrMsg = IDE_TEMPLATE_BAD_INCLUDE;
else
// other exception: re-throw
THROW(hrException);
ThrowErrorSingleInsert(
brErrorLocation.m_pb,
idErrMsg,
brErrorLocation.m_pb,
brErrorLocation.m_cb
);
END_TRY
// store ptr to current file map in local before recursive call (which may increment m_cFilemaps)
CFileMap* pfilemap = m_rgpFilemaps[m_cFilemaps - 1];
// get inc-file object from cache
CIncFile* pIncFile;
if(FAILED(hr = g_IncFileMap.GetIncFile(pfilemap->m_szPathTranslated, &pIncFile)))
THROW(hr);
// add this template to inc-file's template list
if (FAILED(hr = pIncFile->AddTemplate(this)))
THROW(hr);
// set filemap's inc-file ptr
pfilemap->m_pIncFile = pIncFile;
// get source segments from include file
// bugs 1363, 1364: process include file only after we establish dependencies;
// required for cache flushing to work correctly after compile errors
GetSegmentsFromFile(*pfilemap, WorkStore, pHitObj, fIsHTML);
}
/* ===================================================================
CTemplate::GetOpenToken
Returns the token index of and a ptr to the first valid open token
in search range. For the open token to be valid, we must bypass
segments we should not process, e.g. scripts or objects not tagged as 'server'
Returns
ptr to open token; ptr to open token enum value (out-parameter)
Side effects
None
*/
BYTE*
CTemplate::GetOpenToken
(
CByteRange brSearch, // (ByVal) byte range to search for next segment-opening token
SOURCE_SEGMENT ssegLeading, // type of 'leading', i.e. pre-token, source segment
// (only used when deciding to ignore non-SSI comments)
_TOKEN* rgtknOpeners, // array of permitted open tokens
UINT ctknOpeners, // count of permitted open tokens
_TOKEN* ptknOpen // ptr to open token enum value (out-parameter)
)
{
BYTE* pbTokenOpen = NULL; // ptr to opening token
// keep getting segment-opening tokens until we find one that we need to process
while(TRUE)
{
// Get next open token in search range
*ptknOpen = gm_pTokenList->NextOpenToken(
brSearch,
rgtknOpeners,
ctknOpeners,
&pbTokenOpen,
m_wCodePage
);
/* Certain tokens must be followed immediately by white space; others need not.
NOTE it is pure coincidence that the 'white-space tokens' are also those that
get special processing below; hence we handle white space issue there.
If we ever add another 'white-space token' that doesn't require the special processing,
we will need to handle the white space issue here.
*/
/* Similar thing applies to non-include and non-metadata HTML
comments. We really don't care for them to generate their
own segments -- we can reduce number of Response.WriteBlock()
calls by considering them part of the preceding HTML segment.
*/
if (*ptknOpen == CTokenList::tknOpenHTMLComment)
{
if (ssegLeading != ssegHTML) // if not inside HTML
break; // generate a new segment
// for HTML comments check if it is an include or metadata
// and if not, this is not a separate segment - keep on looking
// for the next opener
// advance search range to just past open token
gm_pTokenList->MovePastToken(*ptknOpen, pbTokenOpen, brSearch);
// find end of comment
BYTE *pbClose = gm_pTokenList->GetToken(CTokenList::tknCloseHTMLComment,
brSearch, m_wCodePage);
if (pbClose == NULL)
{
// Error -- let other code handle this
break;
}
// construct comment byte range to limit search to it
CByteRange brComment = brSearch;
brComment.m_cb = DIFF(pbClose - brSearch.m_pb);
// look for metadata and include (only inside the comment)
if (gm_pTokenList->GetToken(CTokenList::tknCommandINCLUDE,
brComment, m_wCodePage))
{
// SSI inclide -- keep it
break;
}
else if (gm_pTokenList->GetToken(CTokenList::tknTagMETADATA,
brComment, m_wCodePage))
{
// METADATA -- keep it
break;
}
else if (gm_pTokenList->GetToken(CTokenList::tknTagFPBot,
brComment, m_wCodePage))
{
// METADATA -- keep it
break;
}
else
{
// Regular comment - ignore it
goto LKeepLooking;
}
}
else if (*ptknOpen == CTokenList::tknOpenTaggedScript || *ptknOpen == CTokenList::tknOpenObject)
{
/* if token was script or object tag, check to see if:
a) it is followed immediately by white space; if not keep looking
b) it opens a well-formed segment, i.e. one with a proper close tag; if not, throw error
c) it is designated runat-server; if not keep looking
*/
// advance search range to just past open token
gm_pTokenList->MovePastToken(*ptknOpen, pbTokenOpen, brSearch);
// bug 760: if token is not followed immediately by white space, keep looking
if(!brSearch.IsNull())
if(!FWhiteSpace(*brSearch.m_pb))
goto LKeepLooking;
// ptr to close of start tag
BYTE* pbCloseTag;
// tags contained in start tag
CByteRange brTags = BrTagsFromSegment(
brSearch,
GetComplementToken(*ptknOpen),
&pbCloseTag
);
if(pbCloseTag == NULL)
{
// if no close tag, throw error
if(*ptknOpen == CTokenList::tknOpenObject)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NO_CLOSE_OBJECT);
else if(*ptknOpen == CTokenList::tknOpenTaggedScript)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NO_CLOSE_TSCRIPT);
}
// if this is a server object (RUNAT=Server), we will process it; else keep looking
if(FTagHasValue(brTags, CTokenList::tknTagRunat, CTokenList::tknValueServer))
break;
}
else
{
// if token was other than script or object tag, or comment
// we should process segment;
// hence we have found our open token, so break
break;
}
LKeepLooking: ;
}
return pbTokenOpen;
}
/* ===================================================================
CTemplate::GetCloseToken
Returns a ptr to the next token of type tknClose.
Returns
ptr to close token
Side effects
Detects and errors out on attempt to nest tagged script or object blocks.
*/
BYTE*
CTemplate::GetCloseToken
(
CByteRange brSearch, // (ByVal) byte range to search
_TOKEN tknClose // close token
)
{
BYTE* pbTokenClose = gm_pTokenList->GetToken(tknClose, brSearch, m_wCodePage);
if(pbTokenClose != NULL)
if(tknClose == CTokenList::tknCloseTaggedScript || tknClose == CTokenList::tknCloseObject)
{
CByteRange brSegment;
BYTE* pbTokenOpen;
brSegment.m_pb = brSearch.m_pb;
brSegment.m_cb = DIFF(pbTokenClose - brSearch.m_pb);
if(NULL != (pbTokenOpen = gm_pTokenList->GetToken(GetComplementToken(tknClose), brSegment, m_wCodePage)))
{
if(tknClose == CTokenList::tknCloseTaggedScript)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NESTED_TSCRIPT);
else if(tknClose == CTokenList::tknCloseObject)
ThrowError(pbTokenOpen, IDE_TEMPLATE_NESTED_OBJECT);
}
}
return pbTokenClose;
}
/*===================================================================
CTemplate::GetComplementToken
Returns a token's compement token.
Returns
Complement token
Side effects
None
*/
_TOKEN
CTemplate::GetComplementToken
(
_TOKEN tkn
)
{
switch(tkn)
{
// open tokens
case CTokenList::tknOpenPrimaryScript:
return CTokenList::tknClosePrimaryScript;
case CTokenList::tknOpenTaggedScript:
return CTokenList::tknCloseTaggedScript;
case CTokenList::tknOpenObject:
return CTokenList::tknCloseObject;
case CTokenList::tknOpenHTMLComment:
return CTokenList::tknCloseHTMLComment;
// close tokens
case CTokenList::tknClosePrimaryScript:
return CTokenList::tknOpenPrimaryScript;
case CTokenList::tknCloseTaggedScript:
return CTokenList::tknOpenTaggedScript;
case CTokenList::tknCloseObject:
return CTokenList::tknOpenObject;
case CTokenList::tknCloseHTMLComment:
return CTokenList::tknOpenHTMLComment;
}
Assert(FALSE);
return CTokenList::tknEOF;
}
/*===================================================================
CTemplate::GetSegmentOfOpenToken
Returns the segment type of an open token.
Returns
source segment type of open token
Side effects
None
*/
CTemplate::SOURCE_SEGMENT
CTemplate::GetSegmentOfOpenToken
(
_TOKEN tknOpen
)
{
switch(tknOpen)
{
case CTokenList::tknOpenPrimaryScript:
return ssegPrimaryScript;
case CTokenList::tknOpenTaggedScript:
return ssegTaggedScript;
case CTokenList::tknOpenObject:
return ssegObject;
case CTokenList::tknOpenHTMLComment:
return ssegHTMLComment;
}
return ssegHTML;
}
/* ========================================================
CTemplate::BrTagsFromSegment
Returns the tag range from an HTML start tag
Returns
tag byte range
Side effects
none
*/
CByteRange
CTemplate::BrTagsFromSegment
(
CByteRange brSegment, // segment byte range
_TOKEN tknClose, // close token
BYTE** ppbCloseTag // ptr-to-ptr to close tag - returned to caller
)
{
CByteRange brTags; // tags return value - NOTE constructed null
// ptr to close token - NOTE null if none within segment byte range
BYTE* pbTokenClose = GetCloseToken(brSegment, tknClose);
// if no close tag found, return null tags
if(NULL == (*ppbCloseTag = gm_pTokenList->GetToken(CTokenList::tknCloseTag, brSegment, m_wCodePage)))
goto Exit;
// if next non-null close token occurs before close tag, close tag is invalid; return nulls
if((pbTokenClose != NULL) && (*ppbCloseTag > pbTokenClose ))
{
*ppbCloseTag = NULL;
goto Exit;
}
// crack tags from header tag
brTags.m_pb = brSegment.m_pb;
brTags.m_cb = DIFF(*ppbCloseTag - brSegment.m_pb);
Exit:
return brTags;
}
/* ========================================================
CTemplate::BrValueOfTag
Returns a tag's value from a byte range; null if tag is not found
NOTE value search algorithm per W3 HTML spec - see www.w3.org
Returns
byte range of tag's value
pfTagExists - does the tag exist in tags byte range? (out-parameter)
NOTE we default *pfTagExists = TRUE; most callers don't care and omit this parameter
Side effects
none
*/
CByteRange
CTemplate::BrValueOfTag
(
CByteRange brTags, // tags byte range
_TOKEN tknTagName // tag name token
)
{
CByteRange brTemp = brTags; // temp byte range
CByteRange brValue; // byte range of value for the given tag - NOTE constructed null
char chDelimiter = NULL; // value delimiter
// ptr to tag name
BYTE* pbTagName = GetTagName(brTags, tknTagName);
// If we did not find tag, return
if(pbTagName == NULL)
return brValue;
// Move past tag name token and pre-separator white space
brTemp.Advance(DIFF(pbTagName - brTags.m_pb) + CCH_TOKEN(tknTagName));
LTrimWhiteSpace(brTemp);
if(brTemp.IsNull())
goto Exit;
// If we did not find separator, return
if(*brTemp.m_pb != CH_ATTRIBUTE_SEPARATOR)
goto Exit;
// Move past separator and post-separator white space
brTemp.Advance(sizeof(CH_ATTRIBUTE_SEPARATOR));
LTrimWhiteSpace(brTemp);
if(brTemp.IsNull())
goto Exit;
// If value begins with a quote mark, cache it as delimiter
if((*brTemp.m_pb == CH_SINGLE_QUOTE) || (*brTemp.m_pb == CH_DOUBLE_QUOTE))
chDelimiter = *brTemp.m_pb;
if(chDelimiter)
{
// move past delimiter
brTemp.Advance(sizeof(chDelimiter));
if(brTemp.IsNull())
goto Exit;
}
// provisionally set value to temp byte range
brValue = brTemp;
// advance temp byte range to end of value range
while(
(chDelimiter && (*brTemp.m_pb != chDelimiter)) // if we have a delimiter, find next delimiter
|| (!chDelimiter && (!FWhiteSpace(*brTemp.m_pb))) // if we have no delimiter, find next white space
)
{
// advance temp byte range
brTemp.Advance(1);
if(brTemp.IsNull())
{
if(chDelimiter)
// we found no closing delimiter, so error out
ThrowErrorSingleInsert(brValue.m_pb, IDE_TEMPLATE_NO_ATTRIBUTE_DELIMITER,
pbTagName, CCH_TOKEN(tknTagName));
else
// value runs to end of temp byte range, so exit (since we already init'ed to temp)
goto Exit;
}
}
// set byte count so that value points to delimited range
brValue.m_cb = DIFF(brTemp.m_pb - brValue.m_pb);
Exit:
// if tag is empty, raise an error
if (brValue.IsNull())
{
ThrowErrorSingleInsert(brTags.m_pb, IDE_TEMPLATE_VALUE_REQUIRED, pbTagName, CCH_TOKEN(tknTagName));
}
// enforce mandatory tag values if required
if(tknTagName == CTokenList::tknTagRunat)
{
if(!brValue.FMatchesSz(SZ_TOKEN(CTokenList::tknValueServer)))
ThrowError(brTags.m_pb, IDE_TEMPLATE_RUNAT_NOT_SERVER);
}
return brValue;
}
/* ============================================================================
CTemplate::CompTagName
Compares characters in two buffers (case-insensitive) and returns TRUE or FALSE
Side effects
none
*/
BOOL
CTemplate::CompTagName
(
CByteRange &brTags, // tags byte range
_TOKEN tknTagName // tag name token
)
{
CByteRange brTemp = brTags; // local byte range, so we don't change tags byte range
UINT cbAttributeName = CCH_TOKEN(tknTagName); // length of tag name
LPSTR pszAttributeName = SZ_TOKEN(tknTagName); // tag name string
// search for potential matches on tag name string, case-insensitive
if(!brTemp.IsNull())
if( 0 == _memicmp( brTemp.m_pb, pszAttributeName, cbAttributeName ))
return TRUE;
return FALSE;
}
/* ============================================================================
CTemplate::GetTagName
Returns a ptr to a tag name in a byte range; null if not found
Returns
ptr to tag name
Side effects
none
*/
BYTE*
CTemplate::GetTagName
(
CByteRange brTags, // tags byte range
_TOKEN tknTagName // tag name token
)
{
CByteRange brTemp = brTags; // local byte range, so we don't change tags byte range
UINT cbAttributeName = CCH_TOKEN(tknTagName); // length of tag name
LPSTR pszAttributeName = SZ_TOKEN(tknTagName); // tag name string
// PREFIX: pszAttributeName could be NULL, though I don't think that can happen.
Assert (pszAttributeName != NULL);
while(TRUE)
{
// search for potential matches on tag name string, case-insensitive
while(!brTemp.IsNull())
{
if(0 == _strnicmp((char*)brTemp.m_pb, pszAttributeName, cbAttributeName ))
break;
brTemp.Advance(1);
}
// if we did not find tag name string at all, return 'not found'
if(brTemp.IsNull())
goto NotFound;
// if it is a valid HTML tag name, return it
if(FTagName(brTemp.m_pb, cbAttributeName))
goto Exit;
// if we found a matching but invalid substring, advance beyond it so we can search again
brTemp.Advance(cbAttributeName);
// if we have exhausted search range, return 'not found'
if(brTemp.IsNull())
goto NotFound;
}
Exit:
return brTemp.m_pb;
NotFound:
return NULL;
}
/* ============================================================================
CTemplate::GetTag
Returns a ptr to a tag name in a byte range; null if not found
Returns
ptr to tag name
Side effects
none
*/
BOOL
CTemplate::GetTag
(
CByteRange &brTags, // tags byte range
int nIndex
)
{
CByteRange brTemp = brTags; // local byte range, so we don't change tags byte range
int nTIndex = 0;
while(TRUE)
{
// locate the start of a tag by skipping past the script tag "<%" and any leading white space
//
while(!brTemp.IsNull())
{
if( *brTemp.m_pb == '<' ||
*brTemp.m_pb == '%' ||
*brTemp.m_pb == '@' ||
FWhiteSpace(*brTemp.m_pb))
{
brTemp.Advance(1);
brTags.Advance(1);
}
else
break;
}
// search for potential matches on tag name string, case-insensitive
//
while(!brTemp.IsNull())
{
if( *brTemp.m_pb == '=' || FWhiteSpace(*brTemp.m_pb))
{
nTIndex++;
break;
}
brTemp.Advance(1);
}
// if we did not find tag name string at all, return 'not found'
if(brTemp.IsNull())
goto NotFound;
// if it is a valid HTML tag name, return it
if(FTagName(brTags.m_pb, DIFF(brTemp.m_pb - brTags.m_pb)))
if(nTIndex >= nIndex)
goto Exit;
// position past named pair data and reset start and if end of byte range then
// goto NotFound
//
while(!brTemp.IsNull() && !FWhiteSpace(*brTemp.m_pb))
brTemp.Advance(1);
if(brTemp.IsNull())
goto NotFound;
else
brTags.Advance(DIFF(brTemp.m_pb - brTags.m_pb));
}
Exit:
return TRUE;
NotFound:
return FALSE;
}
/* ============================================================================
CTemplate::FTagHasValue
Do tags include tknTag=tknValue?
Returns
TRUE if tags include value, else FALSE
Side effects
none
*/
BOOLB
CTemplate::FTagHasValue
(
const CByteRange& brTags, // tags byte range to search
_TOKEN tknTag, // tag token
_TOKEN tknValue // value token
)
{
return (BrValueOfTag(brTags, tknTag) // byte range of value
.FMatchesSz(SZ_TOKEN(tknValue)));
}
/* =========================
CTemplate::CopySzAdv
Copies a string to a ptr and advances the ptr just beyond the copied string.
Returns
Nothing
Side effects
advances ptr beyond copied string
*/
void
CTemplate::CopySzAdv
(
char* pchWrite, // write location ptr
LPSTR psz // string to copy
)
{
strcpy(pchWrite, psz);
pchWrite += strlen(psz);
}
/* ============================================================================
CTemplate::WriteTemplate
Writes the template out to a contiguous block of memory.
Returns:
nothing
Side effects:
Allocates, and possibly re-allocates, memory for the template.
HERE IS HOW IT WORKS
--------------------
- an 'offset' is the count of bytes from start-of-template to a location
within template memory
- at the start of the template are 3 USHORTs, the counts of script blocks,
object-infos and HTML blocks, respectively
- next are 4 ULONGs, each an offset to a block of offsets; in order, these are:
offset-to-offset to first script engine name
offset-to-offset to first script block (the script text itself)
offset-to-offset to first object-info
offset-to-offset to first HTML block
- next are a variable number of ULONGs, each an offset to a particular
template component. In order these ULONGs are:
Offsets to Count of offsets
---------- ----------------
script engine names cScriptBlocks
script blocks cScriptBlocks
object-infos cObjectInfos
HTML blocks cHTMLBlocks
- next are the template components themselves, stored sequentially
in the following order:
script engine names
script blocks
object-infos
HTML blocks
HERE IS HOW IT LOOKS
--------------------
|--|--|--| 3 template component counts (USHORTs)
|-- --|-- --| 4 offsets to template component offsets (ULONGs)
|-- --|-- --|-- --|-- --|-- --| template component offsets (ULONGs)
|-- --| ............... |-- --|
|-- --|-- --|-- --|-- --|-- --|
| ........................... | template components
| ........................... |
| ........................... |
| ........................... |
or, mnemonically:
cS cO cH 3 template component counts (USHORTs)
offSE offSB offOb offHT 4 offsets to template component offsets (ULONGs)
|-- --|-- --|-- --|-- --|-- --| template component offsets (ULONGs)
|-- --| ............... |-- --|
|-- --|-- --|-- --|-- --|-- --|
| ........................... | template components
| ........................... |
| ........................... |
| ........................... |
*/
void
CTemplate::WriteTemplate
(
CWorkStore& WorkStore, // working storage for source segments
CHitObj* pHitObj
)
{
USHORT i; // loop index
CByteRange brWrite; // general-purpose byte range for writing stuff out
USHORT cScriptBlocks = WorkStore.CRequiredScriptEngines(m_fGlobalAsa); // count of script blocks - 1 per required engine
USHORT cObjectInfos = WorkStore.m_ObjectInfoStore.Count(); // count of object-infos
USHORT cHTMLBlocks = WorkStore.m_bufHTMLSegments.Count(); // count of HTML segments
// Calc count of offset-to-offsets == total count of all scripts, objects, etc
// NOTE we keep separate offset-to-offsets for script engine names and script text, hence 2x
USHORT cBlockPtrs = (2 * cScriptBlocks) + cObjectInfos + cHTMLBlocks;
// Calc total memory required
// NOTE header includes counts and ptr-ptrs
UINT cbRequiredHeader = (C_COUNTS_IN_HEADER * sizeof(USHORT)) + (C_OFFOFFS_IN_HEADER * sizeof(BYTE**));
UINT cbRequiredBlockPtrs = cBlockPtrs * sizeof(BYTE*);
// Init write-offset locations
// offset to location for writing the next header information; header is at start of template
UINT cbHeaderOffset = 0;
// offset to location for writing the next offset-to-offset; immediately follows header
UINT cbOffsetToOffset = cbRequiredHeader;
// offset to location for writing the next block of data; immediately follows offset-to-offsets
UINT cbDataOffset = cbOffsetToOffset + cbRequiredBlockPtrs;
// offset in source file (for html blocks)
ULONG cbSourceOffset;
// source filename (only if include file)
BYTE *pbIncFilename;
ULONG cbIncFilename;
// Allocate memory and init start-ptr; bail on fail
// NOTE here we init template member variables m_pbStart and m_cbTemplate
if(NULL == (m_pbStart = (BYTE*) CTemplate::LargeMalloc(m_cbTemplate = CB_TEMPLATE_DEFAULT)))
THROW(E_OUTOFMEMORY);
// write out template header
WriteHeader(cScriptBlocks, cObjectInfos, cHTMLBlocks, &cbHeaderOffset, &cbOffsetToOffset);
// Reset offset-to-offset ptr to beginning of its section
cbOffsetToOffset = cbRequiredHeader;
// write script engine names and prog lang ids at current of data section
for(i = 0; i < WorkStore.m_ScriptStore.CountPreliminaryEngines(); i++)
{
// bug 933: only write non-empty script engines
if(WorkStore.FScriptEngineRequired(i, m_fGlobalAsa))
{
WorkStore.m_ScriptStore.m_bufEngineNames.GetItem(i, brWrite);
WriteByteRangeAdv(brWrite, TRUE, &cbDataOffset, &cbOffsetToOffset);
MemCpyAdv(&cbDataOffset, &(WorkStore.m_ScriptStore.m_rgProgLangId[i]), sizeof(PROGLANG_ID), sizeof(DWORD));
}
}
// write the script blocks for each engine at current of data section
// NOTE we sequence this after script engine names (rather than interleave)
USHORT idEngine = 0;
for(i = 0; i < WorkStore.m_ScriptStore.CountPreliminaryEngines(); i++)
{
// bug 933: only write non-empty script engines
if(WorkStore.FScriptEngineRequired(i, m_fGlobalAsa))
{
// bug 933: we need to pass both 'preliminary' engine id (i) and id of instantiated engine (idEngine)
WriteScriptBlockOfEngine(i, idEngine, WorkStore, &cbDataOffset, &cbOffsetToOffset, pHitObj);
idEngine++;
}
}
// Write object-infos at current of data section
for(i = 0; i < cObjectInfos; i++)
{
// get i-th object info from work store
WorkStore.m_ObjectInfoStore.m_bufObjectNames.GetItem(i, brWrite);
// write object name
WriteByteRangeAdv(brWrite, TRUE, &cbDataOffset, &cbOffsetToOffset);
// write clsid, scope and model
/* CONSIDER include if we need to byte-align clsid
// NOTE byte-align clsid (16-byte), which then byte-aligns scope and model (both 4-byte)
MemCpyAdv(&cbDataOffset, &(WorkStore.m_ObjectInfoStore.m_pObjectInfos[i].m_clsid), sizeof(CLSID), TRUE); */
MemCpyAdv(&cbDataOffset, &(WorkStore.m_ObjectInfoStore.m_pObjectInfos[i].m_clsid), sizeof(CLSID), sizeof(DWORD));
MemCpyAdv(&cbDataOffset, &(WorkStore.m_ObjectInfoStore.m_pObjectInfos[i].m_scope), sizeof(CompScope));
MemCpyAdv(&cbDataOffset, &(WorkStore.m_ObjectInfoStore.m_pObjectInfos[i].m_model), sizeof(CompModel));
}
// if other than globals template, write HTML blocks at current of data section
if(!m_fGlobalAsa)
for(i = 0; i < cHTMLBlocks; i++)
{
// write byterange with html code
WorkStore.m_bufHTMLSegments.GetItem(i, brWrite);
WriteByteRangeAdv(brWrite, TRUE, &cbDataOffset, &cbOffsetToOffset);
// source offset and include file
cbSourceOffset = 0;
pbIncFilename = NULL;
cbIncFilename = 0;
if (brWrite.m_pfilemap)
{
// calculate offset from filemap
CFileMap *pFileMap = (CFileMap *)brWrite.m_pfilemap;
if (pFileMap->m_pbStartOfFile) // mapped?
{
cbSourceOffset = DIFF(brWrite.m_pb - pFileMap->m_pbStartOfFile) + 1;
if (pFileMap->GetParent() != NULL && // is include file?
pFileMap->m_szPathInfo) // path exists
{
pbIncFilename = (BYTE *)pFileMap->m_szPathInfo;
cbIncFilename = _tcslen(pFileMap->m_szPathInfo)*sizeof(TCHAR);
}
}
}
// write them
MemCpyAdv(&cbDataOffset, &cbSourceOffset, sizeof(ULONG));
MemCpyAdv(&cbDataOffset, &cbIncFilename, sizeof(ULONG));
if (cbIncFilename > 0)
MemCpyAdv(&cbDataOffset, pbIncFilename, cbIncFilename+sizeof(TCHAR));
}
// trim template memory to exactly what we used
// NOTE cbDataOffset now contains maximum reach we have written to
if(NULL == (m_pbStart = (BYTE*) CTemplate::LargeReAlloc(m_pbStart, m_cbTemplate = cbDataOffset)))
THROW(E_OUTOFMEMORY);
}
/* ============================================================================
CTemplate::WriteHeader
Writes template header, and writes vesrion stamp and source file name into
template data region.
Returns
nothing
Side effects
none
*/
void
CTemplate::WriteHeader
(
USHORT cScriptBlocks, // count of script blocks
USHORT cObjectInfos, // count of object-infos
USHORT cHTMLBlocks, // count of HTML blocks
UINT* pcbHeaderOffset, // ptr to offset value for header write location
UINT* pcbOffsetToOffset // ptr to offset value for offset-to-offset write location
)
{
// Write template component counts out at start of header
WriteShortAdv(cScriptBlocks, pcbHeaderOffset);
WriteShortAdv(cObjectInfos, pcbHeaderOffset);
WriteShortAdv(cHTMLBlocks, pcbHeaderOffset);
// Write offsets-to-offset to script engine names, script blocks, object-infos, HTML blocks
// NOTE counts of script engine names and script blocks are identical
WriteOffsetToOffset(cScriptBlocks, pcbHeaderOffset, pcbOffsetToOffset);
WriteOffsetToOffset(cScriptBlocks, pcbHeaderOffset, pcbOffsetToOffset);
WriteOffsetToOffset(cObjectInfos, pcbHeaderOffset, pcbOffsetToOffset);
WriteOffsetToOffset(cHTMLBlocks, pcbHeaderOffset, pcbOffsetToOffset);
}
/* ============================================================================
CTemplate::WriteScriptBlockOfEngine
Writes out script block for idEngine-th script engine.
NOTE segment buffer [0] contains primary script segments
segment buffer [1] contains tagged script segments of default engine
segment buffer [i] contains tagged script segments of (i-1)th engine, for i >= 2
Returns
nothing
Side effects
none
*/
void
CTemplate::WriteScriptBlockOfEngine
(
USHORT idEnginePrelim, // preliminary script engine id (assigned during template pre-processing)
USHORT idEngine, // actual script engine id (written into compiled template)
CWorkStore& WorkStore, // working storage for source segments
UINT* pcbDataOffset, // ptr to write location offset value
UINT* pcbOffsetToOffset, // ptr to offset-to-offset offset value
CHitObj* pHitObj
)
{
// NOTE works for all id's - see comment above
USHORT iTSegBuffer = idEnginePrelim + 1; // index of tagged segment buffer
CByteRange brSegment; // current script segment
UINT i; // loop index
UINT cbScriptBlockOffset; // offset to script block write-location
// count of tagged script segments
UINT cTaggedSegments = WorkStore.m_ScriptStore.m_ppbufSegments[iTSegBuffer]->Count();
// Byte-align data offset location, since next thing we will write there is script block length
// NOTE we use brSegment.m_cb generically; we really want CByteRange::m_cb
ByteAlignOffset(pcbDataOffset, sizeof(brSegment.m_cb));
// Cache current data offset location as offset to start of script block
cbScriptBlockOffset = *pcbDataOffset;
// Write offset to start of script block at current offset-to-offset offset
WriteLongAdv(cbScriptBlockOffset, pcbOffsetToOffset);
// advance data ptr (by init'ing script length value to 0)
WriteLongAdv(0, pcbDataOffset);
// reset counter that AppendSourceInfo uses
m_cbTargetOffsetPrevT = 0;
// if other than globals template and this is default script engine (prelim engine 0),
// write primary script procedure at current of data section
if(!m_fGlobalAsa)
if(idEnginePrelim == 0)
WritePrimaryScriptProcedure(0, WorkStore, pcbDataOffset, cbScriptBlockOffset + sizeof(long));
// write out tagged script segments at current of data section
for(i = 0; i < cTaggedSegments; i++)
{
WorkStore.m_ScriptStore.m_ppbufSegments[iTSegBuffer]->GetItem(i, brSegment);
WriteScriptSegment(
idEngine,
m_rgpSegmentFilemaps[brSegment.m_idSequence],
brSegment,
pcbDataOffset,
cbScriptBlockOffset + sizeof(long),
FALSE /* fAllowExprWrite - disallowed for tagged script */
);
}
// Write out null terminator
MemCpyAdv(pcbDataOffset, SZ_NULL, 1);
// convert script text to unicode, so script engine won't have to do this at runtime
// ptr to start of script is:
// ptr start of template + offset to script + size of script length
LPSTR szScript = (LPSTR) m_pbStart + cbScriptBlockOffset + sizeof(ULONG);
/* script block length is:
== EndOfScriptText - StartOfScriptText
== (current of data ptr - length of null ) - (start of primary script byte range + length of br.m_cb)
*/
ULONG cbScript = (*pcbDataOffset - sizeof(BYTE)) - (cbScriptBlockOffset + sizeof(ULONG));
/* bug 887: we append one extra "pseudo-line" to the end of source-infos array
to cover the case where the script engine reports back an error line number
that falls after end of script. We always want the "pseudo-line" to point to the
main file, so that the debugger can display something reasonable, so we pass
m_rgpFilemaps[0] as the source file, which is the main file.
*/
AppendSourceInfo(idEngine, m_rgpFilemaps[0],
NULL, // Don't calculate line #
UINT_MAX, // Don't care & calculation is expensive
UINT_MAX, // Start of script blocks
UINT_MAX, // Really don't care
0, // zero characters exist past EOF
TRUE); // Line is HTML (bogus)
// get wide string version of script text, using hitobj's code page
// NOTE we may slightly over-allocate space for wstrScript by using cbScript (e.g. if script contains DBCS).
// However, this won't matter since we call MultiByteToWideChar with -1, telling it to calc length of szScript
LPOLESTR wstrScript = NULL;
DWORD cbConvert = ( cbScript + 1 ) * 2;
STACK_BUFFER( tempScript, 2048 );
if (!tempScript.Resize(cbConvert)) {
THROW(E_OUTOFMEMORY);
}
wstrScript = (LPOLESTR)tempScript.QueryPtr();
MultiByteToWideChar( m_wCodePage, 0, szScript, -1, wstrScript, (cbScript + 1) );
// reset data offset location to start of script
*pcbDataOffset = cbScriptBlockOffset + sizeof(ULONG);
// write wide string script text over top of ansi version
MemCpyAdv(pcbDataOffset, wstrScript, sizeof(WCHAR) * cbScript);
// write wide string null terminator
MemCpyAdv(pcbDataOffset, WSTR_NULL, sizeof(WCHAR));
// write script length at start of script byte range
// NOTE we do this here because script length was initially unknown
WriteLongAdv(sizeof(WCHAR) * cbScript, &cbScriptBlockOffset);
}
/* ============================================================================
CTemplate::WritePrimaryScriptProcedure
Writes out default-engine primary script procedure.
If VBScript is default-engine, the primary script procedure contains
interleaved script commands and HTML block-writes, like this:
Sub Main
...
[script segment]
Response.WriteBlock([HTML block id])
...
[script segment]
Response.WriteBlock([HTML block id])
...
[script segment]
Response.WriteBlock([HTML block id])
...
End Sub
NOTE segment buffer [0] == primary script segments
Returns
nothing
Side effects
none
*/
void
CTemplate::WritePrimaryScriptProcedure
(
USHORT idEngine, // script engine id
CWorkStore& WorkStore, // working storage for source segments
UINT* pcbDataOffset, // ptr to write location offset value
UINT cbScriptBlockOffset // ptr to start of script engine code
)
{
USHORT cScriptSegmentsProcessed = 0; // count of script blocks processed
USHORT cHTMLBlocksProcessed = 0; // count of HTML blocks processed
CByteRange brScriptNext; // next script block to write out
CByteRange brHTMLNext; // next HTML block to write out
char szHTMLBlockID[6]; // sz representation of HTML block ID - NOTE limited to 5 digits
// count of primary script segments
USHORT cPrimaryScriptSegments = WorkStore.m_ScriptStore.m_ppbufSegments[0]->Count();
// count of HTML blocks
USHORT cHTMLBlocks = WorkStore.m_bufHTMLSegments.Count();
CFileMap* pfilemap; // file where HTML segment lives in
// get initial script segment and initial html segment
if(cPrimaryScriptSegments)
WorkStore.m_ScriptStore.m_ppbufSegments[0]->GetItem(0, brScriptNext);
if(cHTMLBlocks)
WorkStore.m_bufHTMLSegments.GetItem(0, brHTMLNext);
// While HTML block(s) or primary script segment(s) remain to be processed ...
while((cHTMLBlocksProcessed < cHTMLBlocks) || (cScriptSegmentsProcessed < cPrimaryScriptSegments))
{
// If HTML block(s) remain to be processed ...
if(cHTMLBlocksProcessed < cHTMLBlocks)
while (TRUE)
{
// Write out write-block command for each HTML segment earlier in source than next script segment
if(brHTMLNext.FEarlierInSourceThan(brScriptNext) || (cScriptSegmentsProcessed >= cPrimaryScriptSegments))
{
// append source-info for the target script line we just manufactured
pfilemap = m_rgpSegmentFilemaps[brHTMLNext.m_idSequence];
AppendSourceInfo(idEngine, pfilemap,
NULL, // Don't calculate line #
DIFF(brHTMLNext.m_pb - pfilemap->m_pbStartOfFile), // line offset
cbScriptBlockOffset,
*pcbDataOffset - cbScriptBlockOffset, // character offset in target script
CharAdvDBCS((WORD)m_wCodePage, // length of the segment
reinterpret_cast<char *>(brHTMLNext.m_pb),
reinterpret_cast<char *>(brHTMLNext.m_pb + brHTMLNext.m_cb),
INFINITE, NULL),
TRUE); // Line is HTML text
// Get block number as an sz
_itoa(cHTMLBlocksProcessed, szHTMLBlockID, 10);
// Write out write-block opener
WriteSzAsBytesAdv(WorkStore.m_szWriteBlockOpen, pcbDataOffset);
// Write out block number
WriteSzAsBytesAdv(szHTMLBlockID, pcbDataOffset);
// Write out write-block closer and newline
WriteSzAsBytesAdv(WorkStore.m_szWriteBlockClose, pcbDataOffset);
WriteSzAsBytesAdv(SZ_NEWLINE, pcbDataOffset);
if(++cHTMLBlocksProcessed >= cHTMLBlocks)
break;
// Get next HTML block
WorkStore.m_bufHTMLSegments.GetItem(cHTMLBlocksProcessed, brHTMLNext);
}
else
break;
}
// if primary script segment(s) remain to be processed ...
if(cScriptSegmentsProcessed < cPrimaryScriptSegments)
while (TRUE)
{
// Write out each primary script segment earlier in the source file than the next HTML block
if(brScriptNext.FEarlierInSourceThan(brHTMLNext) || (cHTMLBlocksProcessed >= cHTMLBlocks))
{
WriteScriptSegment(
idEngine,
m_rgpSegmentFilemaps[brScriptNext.m_idSequence],
brScriptNext,
pcbDataOffset,
cbScriptBlockOffset,
TRUE /* fAllowExprWrite - allowed for primary script */
);
if(++cScriptSegmentsProcessed >= cPrimaryScriptSegments)
break;
// Get next script segment
WorkStore.m_ScriptStore.m_ppbufSegments[0]->GetItem(cScriptSegmentsProcessed, brScriptNext);
}
else
break;
}
}
}
/* ============================================================================
CTemplate::WriteScriptSegment
Writes a script segment to template memory line-by-line.
NOTE a 'script segment' is a piece (possibly all) of a 'script block'
Returns
nothing
Side effects
none
*/
void
CTemplate::WriteScriptSegment
(
USHORT idEngine, // script engine id
CFileMap* pfilemap, // ptr to source file map
CByteRange& brScript, // byte range containing script segment
UINT* pcbDataOffset, // ptr to write location offset value
UINT cbScriptBlockOffset,// ptr to beginning of the script text
BOOL fAllowExprWrite // allow short-hand expression write?
)
{
CByteRange brLine; // byte range containing next line
UINT cbPtrOffset = 0; // ptr offset - 0 tells WriteByteRangeAdv 'ignore this'
BOOL fExpression = FALSE; // is current line an expression?
BOOL fCalcLineNumber = TRUE; // calc source line number?
BOOL fFirstLine = TRUE; // first line in script segment?
if(FByteRangeIsWhiteSpace(brScript))
return;
// trim white space from beginning of script segment
if (FIsLangVBScriptOrJScript(idEngine))
LTrimWhiteSpace(brScript);
while(!(brScript.IsNull()))
{
// fetch next line from byte range
// NOTE LineFromByteRangeAdv advances through brScript until brScript is null
LineFromByteRangeAdv(brScript, brLine);
if(FByteRangeIsWhiteSpace(brLine))
{
// if line is blank, don't process it; simply force calc of line number on next non-blank line
fCalcLineNumber = TRUE;
continue;
}
// line is non-blank; trim its white space
if (FIsLangVBScriptOrJScript(idEngine))
LTrimWhiteSpace(brLine);
RTrimWhiteSpace(brLine);
// append source-info to array; if flag is set, calc line number
// from location in source file; else, simply increment previous line number (NULL indicates this)
AppendSourceInfo(idEngine, pfilemap,
fCalcLineNumber? brLine.m_pb : NULL, // info to calc line #
DIFF(brLine.m_pb - pfilemap->m_pbStartOfFile), // line offset
cbScriptBlockOffset,
*pcbDataOffset - cbScriptBlockOffset, // character offset in target script
CharAdvDBCS((WORD)m_wCodePage, // statement length
reinterpret_cast<char *>(brLine.m_pb),
reinterpret_cast<char *>(brLine.m_pb + brLine.m_cb),
INFINITE, NULL),
FALSE); // HTML?
/* if it's true, set calc-line-number flag false
NOTE this is purely an optimization, to make the call to AppendSourceInfo faster
on subsequent calls within a contiguous block of non-blank lines
*/
if(fCalcLineNumber)
fCalcLineNumber = FALSE;
if(fAllowExprWrite && fFirstLine)
{
// bug 912: test for remainder of script segment null on temp copy of script byte range, not on actual
CByteRange brTemp = brScript;
LTrimWhiteSpace(brTemp); // NOTE will nullify brScript if it is all white space
if(brTemp.IsNull())
{
/* if
a) expr-write is allowed AND
b) this is only script line in this segment (i.e. first line in segment and remainder of segment is null)
then, test this line to see if it is an expression.
NOTE test (b) fixes bug 785
if this line is an expression, create a script command that reads
Response.Write([line contents])
*/
if(fExpression = FExpression(brLine))
{
Assert(idEngine == 0); // =expr is only enabled for primary engine
WriteSzAsBytesAdv(m_pWorkStore->m_szWriteOpen, pcbDataOffset);
}
// in this case only, set actual script to (now null) temp copy, since brScript governs while loop termination
brScript = brTemp;
}
}
Assert(FImplies(fExpression, fFirstLine)); // if an expr, must be first line in segment
Assert(FImplies(fExpression, brScript.IsNull())); // if an expr, no more script lines remain
Assert(FImplies(!fFirstLine, !fExpression)); // if not first line in segment, line cannot be expr
Assert(FImplies(!brScript.IsNull(), !fExpression)); // if script lines remain, line cannot be expr
// write out line contents
WriteScriptMinusEscapeChars(brLine, pcbDataOffset, &cbPtrOffset);
// if this line is an expression, close script command
if(fExpression)
WriteSzAsBytesAdv(m_pWorkStore->m_szWriteClose, pcbDataOffset);
// write new-line and set first-line flag false
WriteSzAsBytesAdv(SZ_NEWLINE, pcbDataOffset);
fFirstLine = FALSE;
}
}
/* ============================================================================
CTemplate::WriteScriptMinusEscapeChars
Writes a script byte range to memory, minus its escape characters, if any.
Returns:
Nothing.
Side effects:
None.
*/
void
CTemplate::WriteScriptMinusEscapeChars
(
CByteRange brScript, // (ByVal) script byte range
UINT* pcbDataOffset, // offset where data will be written
UINT* pcbPtrOffset // offset where ptr will be written
)
{
BYTE* pbToken;
while(NULL != (pbToken = gm_pTokenList->GetToken(CTokenList::tknEscapedClosePrimaryScript, brScript, m_wCodePage)))
{
CByteRange brTemp = brScript;
// set temp range to source range up to escaped-token
brTemp.m_cb = DIFF(pbToken - brTemp.m_pb);
// write out temp range and actual-token - this replaces escaped-token with actual-token
WriteByteRangeAdv(brTemp, FALSE, pcbDataOffset, pcbPtrOffset);
WriteSzAsBytesAdv(SZ_TOKEN(CTokenList::tknClosePrimaryScript), pcbDataOffset);
//advance source range past escaped-token
brScript.Advance(DIFF(pbToken - brScript.m_pb) + CCH_TOKEN(CTokenList::tknEscapedClosePrimaryScript));
}
// write remainder of source range
WriteByteRangeAdv(brScript, FALSE, pcbDataOffset, pcbPtrOffset);
}
/* ============================================================================
CTemplate::FVbsComment
Determines whether a script line is a VBS comment.
NOTE caller must ensure that brLine is non-blank and has no leading white space
Returns
TRUE if the line is a VBS comment, else FALSE
Side effects
none
*/
BOOLB
CTemplate::FVbsComment(CByteRange& brLine)
{
// CONSIDER: SCRIPTLANG generic comment token
if(!_strnicmp((LPCSTR)brLine.m_pb, SZ_TOKEN(CTokenList::tknVBSCommentSQuote), CCH_TOKEN(CTokenList::tknVBSCommentSQuote)))
return TRUE;
if(!_strnicmp((LPCSTR)brLine.m_pb, SZ_TOKEN(CTokenList::tknVBSCommentRem), CCH_TOKEN(CTokenList::tknVBSCommentRem)))
return TRUE;
return FALSE;
}
/* ============================================================================
CTemplate::FExpression
Determines whether a script line is an expression, and if so returns
just the expression in brLine.
NOTE caller must ensure that brLine has no leading white space
Returns
TRUE if the line is an expression, else FALSE
Side effects
none
*/
BOOLB
CTemplate::FExpression(CByteRange& brLine)
{
// may be whitespace (other languages besides VB & JScript will have whitespace)
char *pchLine = reinterpret_cast<char *>(brLine.m_pb);
int cchLine = brLine.m_cb;
while (cchLine > 0 && FWhiteSpace(*pchLine))
{
--cchLine;
++pchLine;
}
// if line starts with =, it is an expression: bypass =, left-trim whitespace and return true
if(cchLine > 0 && *pchLine == '=')
{
brLine.Advance(1 + DIFF(reinterpret_cast<BYTE *>(pchLine) - brLine.m_pb)); // OK to advance past whitespace now.
LTrimWhiteSpace(brLine);
return TRUE;
}
// else return false
return FALSE;
}
/**
** In the following function names:
** 'Adv' == 'advance offset after writing'
**/
/* ============================================================================
CTemplate::WriteOffsetToOffset
Writes a offset-to-offset offset (0 if no blocks) into header,
and advances header offset and offset-to-offset.
Returns:
Nothing.
Side effects:
Advances offsets.
*/
void
CTemplate::WriteOffsetToOffset
(
USHORT cBlocks, // count of blocks
UINT* pcbHeaderOffset, // ptr to header offset value
UINT* pcbOffsetToOffset // ptr to offset-to-offset value
)
{
// if blocks of this type, write offset to first of them into header;
// if no blocks of this type, write 0 into header
WriteLongAdv((cBlocks > 0) ? *pcbOffsetToOffset : 0, pcbHeaderOffset);
// advance offset-to-offset offset
*pcbOffsetToOffset += cBlocks * sizeof(ULONG);
}
/* ============================================================================
CTemplate::WriteSzAsBytesAdv
Writes a null-terminated string as bytes, i.e. without its null terminator
and advances offset
Returns:
Nothing.
Side effects:
Advances offset.
*/
void
CTemplate::WriteSzAsBytesAdv
(
LPCSTR szSource, // source string
UINT* pcbDataOffset // ptr to offset value
)
{
if((szSource == NULL) || (*szSource == '\0'))
return;
MemCpyAdv(pcbDataOffset, (void*) szSource, strlen(szSource));
}
/* ============================================================================
CTemplate::WriteByteRangeAdv
Writes a byte range to memory at template offset location *pcbDataOffset and, optionally,
writes a ptr to the written data at template offset location *pcbPtrOffset
(pass *pcbPtrOffset == 0 to avoid this)
fWriteAsBsz == FALSE --> write only byte range's data
fWriteAsBsz == TRUE --> write length, followed by data, followed by NULL
NOTE bsz == length-prefixed, null-terminated string
Returns:
Nothing.
Side effects:
Advances offset(s).
*/
void
CTemplate::WriteByteRangeAdv
(
CByteRange& brSource, // source data
BOOLB fWriteAsBsz, // write as bsz?
UINT* pcbDataOffset, // offset where data will be written
UINT* pcbPtrOffset // offset where ptr will be written
)
{
// bail if source is empty
if(brSource.IsNull())
return;
// If writing as a bsz, write length prefix
if(fWriteAsBsz)
WriteLongAdv(brSource.m_cb, pcbDataOffset);
// Write data
MemCpyAdv(pcbDataOffset, brSource.m_pb, brSource.m_cb);
// If writing as a bsz, write null terminator and advance target ptr
if(fWriteAsBsz)
MemCpyAdv(pcbDataOffset, SZ_NULL, 1);
// If caller passed a non-zero ptr offset, write offset to data there
if(*pcbPtrOffset > 0)
{
if(fWriteAsBsz)
/* if writing as a bsz ...
offset to start of data == current data offset
- null terminator
- data length
- sizeof length prefix
*/
WriteLongAdv(*pcbDataOffset - 1 - brSource.m_cb - sizeof(brSource.m_cb), pcbPtrOffset);
else
// else, offset to start of data == current data offset - data length
WriteLongAdv(*pcbDataOffset - brSource.m_cb, pcbPtrOffset);
}
}
/*===================================================================
CTemplate::MemCpyAdv
Copies from a memory location to a template offset location,
and advances offset.
Returns:
Nothing.
Side effects:
Advances offset.
Re-allocates memory if required.
*/
void
CTemplate::MemCpyAdv
(
UINT* pcbOffset, // ptr to offset value
void* pbSource, // ptr to source
ULONG cbSource, // length of source
UINT cbByteAlign // align bytes on short/long/dword boundary?
)
{
// byte-align offset location before write, if specified by caller
if(cbByteAlign > 0)
ByteAlignOffset(pcbOffset, cbByteAlign);
// calc number of bytes by which to grow allocated template memory:
// if projected reach exceeds current reach, we need to grow by the difference;
// else, no need to grow
if((*pcbOffset + cbSource) > m_cbTemplate)
{
// Reallocate space for storing local data - we grab twice what we had before
// or twice current growth requirement, whichever is more
m_cbTemplate = 2 * max(m_cbTemplate, (*pcbOffset + cbSource) - m_cbTemplate);
if(NULL == (m_pbStart = (BYTE*) CTemplate::LargeReAlloc(m_pbStart, m_cbTemplate)))
THROW(E_OUTOFMEMORY);
}
// copy source to template offset location
memcpy(m_pbStart + *pcbOffset, pbSource, cbSource);
// advance offset location
*pcbOffset += cbSource;
}
/* ============================================================================
CTemplate::GetAddress
Returns a ptr to the i-th object of type tcomp
*/
BYTE*
CTemplate::GetAddress
(
TEMPLATE_COMPONENT tcomp,
USHORT i
)
{
DWORD* pdwBase;
Assert(NULL != m_pbStart);
// refer to CTemplate::WriteTemplate comments for the structure of what this is dealing with
pdwBase = (DWORD*)(m_pbStart + (C_COUNTS_IN_HEADER * sizeof(USHORT)));
// tcomp types are ptr-to-ptrs
DWORD* pdwTcompBase = (DWORD *) (m_pbStart + pdwBase[tcomp]);
return m_pbStart + pdwTcompBase[i];
}
/* ============================================================================
CTemplate::AppendSourceInfo
Appends a source line number for the current target line
NOTE if caller passes null source ptr, we append prev source line number + 1
Returns
Nothing
Side effects
allocates memory first time thru; may realloc
*/
void
CTemplate::AppendSourceInfo
(
USHORT idEngine, // script engine id
CFileMap* pfilemap, // ptr to source file map
BYTE* pbSource, // ptr to current location in source file
ULONG cbSourceOffset, // byte offset of line in source file
ULONG cbScriptBlockOffset, // pointer to start of script text
ULONG cbTargetOffset, // character offset of line in target file
ULONG cchSourceText, // # of characters in source text
BOOL fIsHTML // TRUE if manufactured line
)
{
UINT i; // loop index
CSourceInfo si; // temporary CSourceInfo structure
vector<CSourceInfo> *prgSourceInfos; // pointer to line mapping table for the engine
ULONG cchSourceOffset = 0;// cch corresponding to cbSourceOffset
HRESULT hr = S_OK;
// if arrays are not yet allocated, allocate them
if (m_rgrgSourceInfos == NULL)
{
// transfer count of script engines from workstore to template
m_cScriptEngines = m_pWorkStore->CRequiredScriptEngines(m_fGlobalAsa);
// one source-info array per engine
if ((m_rgrgSourceInfos = new vector<CSourceInfo>[m_cScriptEngines]) == NULL)
THROW (E_OUTOFMEMORY);
}
// new script engine must be allocated in IdEngineFromBr (way upstream of this point),
// so we assert that current engine must already be covered
Assert(idEngine < m_pWorkStore->CRequiredScriptEngines(m_fGlobalAsa));
/* set current target line's source line number (SLN):
a) if caller passed a source ptr, calc SLN from the source ptr;
b) else if caller passed a filemap ptr, set SLN to prev target line's SLN plus one;
c) else set SLN to 0
semantics:
a) we have a source file location, but must calc a line # for that location
b) caller tells us (by passing NULL source file location) that this target line
immediately follows prev target line. This is an optimization because
SourceLineNumberFromPb is very slow.
change:
caller used to pass NULL filemap ptr that target line is 'manufactured'
i.e. has no corresponding authored line in source file
HOWEVER - now filemap ptr must NOT be NULL because 'manufactured' lines
are also stored in the file map array
*/
Assert (pfilemap != NULL);
prgSourceInfos = &m_rgrgSourceInfos[idEngine];
if (pbSource == NULL)
{
if (prgSourceInfos->length() == 0)
si.m_idLine = 1;
else
si.m_idLine = (*prgSourceInfos)[prgSourceInfos->length() - 1].m_idLine + 1;
}
else
si.m_idLine = SourceLineNumberFromPb(pfilemap, pbSource);
// The EOF line does not have a source offset (caller passes -1 (UINT_MAX)). For this case, no
// DBCS calculations etc. should be done. (set cchSourceOffset to UINT_MAX).
if (cbSourceOffset == UINT_MAX)
cchSourceOffset = UINT_MAX;
else
{
// BUG 80901: Source offset needs to point to the beginning of leading white space on the line
// Adjust source length by one as we decrement source offset
// Note: whitepsace is never trailing byte, so loop will work with DBCS encoded character sets
while (cbSourceOffset > 0 && strchr(" \t\v\a\f", pfilemap->m_pbStartOfFile[cbSourceOffset - 1]))
{
--cbSourceOffset;
++cchSourceText;
}
// BUG 95859
// If the cursor is on the opening token of a script block (the "<%" part of a line), the
// BP is set in the previous HTML, not in the script block, as is desired.
//
// To correct this, if we are in a script block, scan back two characters, see if it is the open
// token. If it is, set the offset back two, and add two to the length.
//
if (!fIsHTML)
{
// Skip whitespace (including newlines -- the previous step did not skip newlines)
//
ULONG cbOpen = cbSourceOffset;
while (cbOpen > 0 && strchr(" \t\v\a\f\r\n", pfilemap->m_pbStartOfFile[cbOpen - 1]))
--cbOpen;
if (cbOpen >= 2 && strncmp(reinterpret_cast<char *>(&pfilemap->m_pbStartOfFile[cbOpen - 2]), "<%", 2) == 0)
{
cbOpen -= 2;
cchSourceText += cbSourceOffset - cbOpen;
cbSourceOffset = cbOpen;
}
// Look for trailing "%>" in this snippet, and if it exists then include the end delimiter in
// the length. NOTE: No DBCS awareness needed here - if we find a lead byte we just get out
// of the loop. We are looking for <whitespace>*"%>" which is totally SBCS chars.
//
ULONG cbClose = cbSourceOffset + cchSourceText;
ULONG cbFile = pfilemap->GetSize();
while (cbClose < cbFile && strchr(" \t\v\a\f\r\n", pfilemap->m_pbStartOfFile[cbClose]))
++cbClose;
if (cbClose < cbFile && strncmp(reinterpret_cast<char *>(&pfilemap->m_pbStartOfFile[cbClose]), "%>", 2) == 0)
cchSourceText += cbClose - (cbSourceOffset + cchSourceText) + 2;
}
// BUG 82222, 85584
// Compiler marks HTML segments starting with the newline on the previous line
// if the line ends with %>.
//
// This screws up the debugger, becasue when you press <F9>, the pointer is placed
// on the line above when it should point to the start of the whitespace on the next line.
if (fIsHTML)
{
UINT cbEOF = pfilemap->GetSize(), cbRover = cbSourceOffset;
// Skip initial whitespace
while (cbRover < cbEOF && strchr(" \t\a\f", pfilemap->m_pbStartOfFile[cbRover]))
++cbRover;
// If what's left is a CR/LF pair, then advance cbSourceOffset to next line
BOOL fCR = FALSE, fLF = FALSE;
if (cbRover < cbEOF && strchr("\r\n", pfilemap->m_pbStartOfFile[cbRover]))
{
fCR = pfilemap->m_pbStartOfFile[cbRover] == '\r';
fLF = pfilemap->m_pbStartOfFile[cbRover] == '\n';
++cbRover;
Assert (fCR || fLF);
}
// we allow either <CR>, <LF>, <CR><LF>, or <LF><CR> to terminate a line,
// so look for its opposite terminator if one is found (but don't require it)
if (fCR && cbRover < cbEOF && pfilemap->m_pbStartOfFile[cbRover] == '\n')
++cbRover;
if (fLF && cbRover < cbEOF && pfilemap->m_pbStartOfFile[cbRover] == '\r')
++cbRover;
// OK, adjust cbSourceOffset now
if ((fCR || fLF) && cbRover < cbEOF)
{
cchSourceText -= cbRover - cbSourceOffset; // adjust # of chars to select
cbSourceOffset = cbRover;
}
}
// Now that we have the source offset, calculate its CCH by finding
// the last time we sampled the value, then add that to the number
// of DBCS characters from that point to the current offset.
//
// For the case of includes, it's possible offset already exists
// (if the entry was previously generated by another instance of
// #include - therefore we have to search)
COffsetInfo *pOffsetInfoLE, *pOffsetInfoGE;
GetBracketingPair(
cbSourceOffset, // value to find
pfilemap->m_rgByte2DBCS.begin(), // beginning of array
pfilemap->m_rgByte2DBCS.end(), // end of array
CByteOffsetOrder(), // search for byte offset
&pOffsetInfoLE, &pOffsetInfoGE // return values
);
// If we find an equal match, don't insert any duplicates
if (pOffsetInfoLE == NULL || pOffsetInfoLE->m_cbOffset < cbSourceOffset)
{
// if pOffsetInfoLE is NULL, it means that the array is empty -
// create the mapping of offset 0 to offset 0.
//
// In the case of the first line of a file being an include directive,
// the first executable line from the file may not start at offset zero,
// so in this case we need to create this entry AND execute the next "if"
// block.
//
if (pOffsetInfoLE == NULL)
{
COffsetInfo oiZero; // ctor will init
if (FAILED(hr = pfilemap->m_rgByte2DBCS.append(oiZero)))
THROW(hr);
pOffsetInfoLE = pfilemap->m_rgByte2DBCS.begin();
Assert (pOffsetInfoLE != NULL);
}
// If cbSourceOffset is zero, we handled it above
if (cbSourceOffset != 0)
{
cchSourceOffset = pOffsetInfoLE->m_cchOffset +
CharAdvDBCS
(
(WORD)m_wCodePage,
reinterpret_cast<char *>(pfilemap->m_pbStartOfFile + pOffsetInfoLE->m_cbOffset),
reinterpret_cast<char *>(pfilemap->m_pbStartOfFile + cbSourceOffset),
INFINITE,
NULL
);
// Now add the value to the table
COffsetInfo oi;
oi.m_cchOffset = cchSourceOffset;
oi.m_cbOffset = cbSourceOffset;
if (pOffsetInfoGE == NULL) // No offset greater
hr = pfilemap->m_rgByte2DBCS.append(oi);
else
hr = pfilemap->m_rgByte2DBCS.insertAt(DIFF(pOffsetInfoGE - pfilemap->m_rgByte2DBCS.begin()), oi);
if (FAILED(hr))
THROW(hr);
}
}
else
{
// If we're not adding anything for the table, Assert it's because there's
// a duplicate item
Assert (cbSourceOffset == pOffsetInfoLE->m_cbOffset);
cchSourceOffset = pOffsetInfoLE->m_cchOffset;
}
}
UINT cchTargetOffset = UINT_MAX;
if (cbTargetOffset != UINT_MAX)
{
// ptr to start of script is:
// ptr start of template + offset to script + size of script length
LPSTR szScript = (LPSTR) m_pbStart + cbScriptBlockOffset;
// Calculate cchTargetOffset (have the cb). The cch is the number of characters since the
// last cch calculated in the end of the array.
//
if (prgSourceInfos->length() > 0)
cchTargetOffset = (*prgSourceInfos)[prgSourceInfos->length() - 1].m_cchTargetOffset;
else
cchTargetOffset = 0;
cchTargetOffset += CharAdvDBCS
(
(WORD) m_wCodePage,
&szScript[m_cbTargetOffsetPrevT],
&szScript[cbTargetOffset],
INFINITE,
NULL
);
// Keeps track of offsets during compilation
//
m_cbTargetOffsetPrevT = cbTargetOffset;
}
// Store this record and move on.
//
si.m_pfilemap = pfilemap;
si.m_fIsHTML = fIsHTML;
si.m_cchSourceOffset = cchSourceOffset;
si.m_cchTargetOffset = cchTargetOffset;
si.m_cchSourceText = cchSourceText;
if (FAILED(prgSourceInfos->append(si)))
THROW(hr);
}
/* ============================================================================
CTemplate::SourceLineNumberFromPb
Returns the starting source line number for the given source file location
*/
UINT
CTemplate::SourceLineNumberFromPb
(
CFileMap* pfilemap, // ptr to source file map
BYTE* pbSource // ptr to current location in source file
)
{
UINT cSourceLines = 1; // count of lines into source file
CByteRange brScan; // byte range to scan for newlines
CByteRange brSOL; // start-of-line ptr
if(pbSource == NULL || pfilemap == NULL)
return 0;
// set scan range to run from start-of-template to caller's ptr
brScan.m_pb = pfilemap->m_pbStartOfFile;
brScan.m_cb = max(DIFF(pbSource - brScan.m_pb), 0);
// get newlines in scan range
brSOL = BrNewLine(brScan);
while(!brSOL.IsNull())
{
// advance start-of-line ptr and scan byte range
brScan.Advance(DIFF((brSOL.m_pb + brSOL.m_cb) - brScan.m_pb));
// increment source line counter
cSourceLines++;
// find next newline
brSOL = BrNewLine(brScan);
}
return cSourceLines;
}
/* ============================================================================
CTemplate::RemoveFromIncFiles
Removes this template from inc-files on which it depends
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::RemoveFromIncFiles
(
)
{
// NOTE we loop from 1 to count, since 0-th filemap is for main file
for(UINT i = 1; i < m_cFilemaps; i++)
{
if(NULL != m_rgpFilemaps[i]->m_pIncFile)
m_rgpFilemaps[i]->m_pIncFile->RemoveTemplate(this);
}
}
/* ****************************************************************************
IDebugDocumentProvider implementation
*/
/* ============================================================================
CTemplate::GetDocument
Return a pointer to the IDebugDocument implementation. (same object in this case)
Returns:
*ppDebugDoc is set to "this".
Notes:
always succeeds
*/
HRESULT CTemplate::GetDocument
(
IDebugDocument **ppDebugDoc
)
{
return QueryInterface(IID_IDebugDocument, reinterpret_cast<void **>(ppDebugDoc));
}
/* ============================================================================
CTemplate::GetName
Return the various names of a document.
*/
HRESULT CTemplate::GetName
(
/* [in] */ DOCUMENTNAMETYPE doctype,
/* [out] */ BSTR *pbstrName
)
{
TCHAR *szPathInfo = m_rgpFilemaps[0]->m_szPathInfo;
switch (doctype) {
case DOCUMENTNAMETYPE_APPNODE:
case DOCUMENTNAMETYPE_FILE_TAIL:
case DOCUMENTNAMETYPE_TITLE:
// Skip application path portion of the filename
{
// Make sure the template remembers the virtual path
// from the same application (it could be different
// if template is shared between two applications)
//
int cch = _tcslen(m_szApplnVirtPath);
if (_tcsncicmp(szPathInfo, m_szApplnVirtPath, cch) == 0)
szPathInfo += cch;
// Strip leading '/'
if (*szPathInfo == _T('/'))
szPathInfo++;
#if UNICODE
*pbstrName = SysAllocString(szPathInfo);
if (*pbstrName == NULL)
return E_OUTOFMEMORY;
return S_OK;
#else
return SysAllocStringFromSz(szPathInfo, 0, pbstrName, m_wCodePage);
#endif
}
case DOCUMENTNAMETYPE_URL:
// prefix with the URL, use szPathInfo for the rest of the path
{
STACK_BUFFER( tempName, MAX_PATH );
int cbURLPrefix = DIFF(m_szApplnVirtPath - m_szApplnURL) * sizeof (TCHAR);
if (!tempName.Resize(cbURLPrefix + (_tcslen(szPathInfo)*sizeof(TCHAR)) + sizeof(TCHAR))) {
return E_OUTOFMEMORY;
}
TCHAR *szURL = (TCHAR *)tempName.QueryPtr();
memcpy(szURL, m_szApplnURL, cbURLPrefix);
_tcscpy(&szURL[cbURLPrefix/sizeof(TCHAR)], szPathInfo);
#if UNICODE
*pbstrName = SysAllocString(szURL);
if (*pbstrName == NULL)
return E_OUTOFMEMORY;
return S_OK;
#else
return SysAllocStringFromSz(szURL, 0, pbstrName, m_wCodePage);
#endif
}
default:
return E_FAIL;
}
}
/* ****************************************************************************
IDebugDocumentText implementation
*/
/* ============================================================================
CTemplate::GetSize
Return the number of lines & characters in the document
*/
HRESULT CTemplate::GetSize
(
/* [out] */ ULONG *pcLines,
/* [out] */ ULONG *pcChars
)
{
/*
* NOTE: compilation is done in two phases.
* Errors are detected and reported in phase 1.
* The DBCS mapping is created in phase 2.
*
* If an error occurred during compilation, m_cChars will be equal to zero
* (Since zero length files are not compiled, m_cChars == 0 means "size
* is unknown", not "size is zero").
*/
if (m_rgpFilemaps[0]->m_cChars == 0)
{
// Likely need to remap the file, then count
BOOL fRemapTemplate = !m_rgpFilemaps[0]->FIsMapped();
if (fRemapTemplate)
TRY
m_rgpFilemaps[0]->RemapFile();
CATCH (dwException)
return E_FAIL;
END_TRY
m_rgpFilemaps[0]->CountChars((WORD)m_wCodePage);
if (fRemapTemplate)
TRY
m_rgpFilemaps[0]->UnmapFile();
CATCH (dwException)
return E_FAIL;
END_TRY
// let's hope client is not relying on # of lines - expensive to compute
*pcChars = m_rgpFilemaps[0]->m_cChars;
*pcLines = ULONG_MAX;
}
else
{
/* The last line in the line mapping array of each engine is the <<EOF>> line
* for that engine. Therefore, the # of lines is the largest <<EOF>> line
* number - 1. The EOF line always points into the main file, so there are no
* include file glitches here.
*/
ULONG cLinesMax = 0;
for (UINT i = 0; i < m_cScriptEngines; ++i)
{
ULONG cLinesCurrentEngine = m_rgrgSourceInfos[0][m_rgrgSourceInfos[0].length() - 1].m_idLine - 1;
if (cLinesCurrentEngine > cLinesMax)
cLinesMax = cLinesCurrentEngine;
}
*pcLines = cLinesMax;
*pcChars = m_rgpFilemaps[0]->m_cChars;
}
IF_DEBUG(SCRIPT_DEBUGGER) {
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "GetSize(\"%S\") returns %lu characters (%lu lines)\n", m_rgpFilemaps[0]->m_szPathTranslated, *pcChars, *pcLines));
#else
DBGPRINTF((DBG_CONTEXT, "GetSize(\"%s\") returns %lu characters (%lu lines)\n", m_rgpFilemaps[0]->m_szPathTranslated, *pcChars, *pcLines));
#endif
}
return S_OK;
}
/* ============================================================================
CTemplate::GetDocumentAttributes
Return doc attributes
*/
HRESULT CTemplate::GetDocumentAttributes
(
/* [out] */ TEXT_DOC_ATTR *ptextdocattr
)
{
// Easy way to tell debugger that we don't support editing.
*ptextdocattr = TEXT_DOC_ATTR_READONLY;
return S_OK;
}
/* ============================================================================
CTemplate::GetPositionOfLine
From a line number, return the character offset of the beginning
*/
HRESULT CTemplate::GetPositionOfLine
(
/* [in] */ ULONG cLineNumber,
/* [out] */ ULONG *pcCharacterPosition
)
{
return GetPositionOfLine(m_rgpFilemaps[0], cLineNumber, pcCharacterPosition);
}
/* ============================================================================
CTemplate::GetLineOfPosition
From a character offset, return the line number and offset within the line
*/
HRESULT CTemplate::GetLineOfPosition
(
/* [in] */ ULONG cCharacterPosition,
/* [out] */ ULONG *pcLineNumber,
/* [out] */ ULONG *pcCharacterOffsetInLine
)
{
return GetLineOfPosition(m_rgpFilemaps[0], cCharacterPosition, pcLineNumber, pcCharacterOffsetInLine);
}
/* ============================================================================
CTemplate::GetText
From a character offset and length, return the document text
*/
HRESULT CTemplate::GetText
(
ULONG cchSourceOffset,
WCHAR *pwchText,
SOURCE_TEXT_ATTR *pTextAttr,
ULONG *pcChars,
ULONG cMaxChars
)
{
return m_rgpFilemaps[0]->GetText((WORD)m_wCodePage, cchSourceOffset, pwchText, pTextAttr, pcChars, cMaxChars);
}
/* ============================================================================
CTemplate::GetPositionOfContext
Decompose a document context into the document offset & length
*/
HRESULT CTemplate::GetPositionOfContext
(
/* [in] */ IDebugDocumentContext *pUnknownDocumentContext,
/* [out] */ ULONG *pcchSourceOffset,
/* [out] */ ULONG *pcchText
)
{
// Make sure that the context is one of ours
CTemplateDocumentContext *pDocumentContext;
if (FAILED(pUnknownDocumentContext->QueryInterface(IID_IDenaliTemplateDocumentContext, reinterpret_cast<void **>(&pDocumentContext))))
return E_FAIL;
if (pcchSourceOffset)
*pcchSourceOffset = pDocumentContext->m_cchSourceOffset;
if (pcchText)
*pcchText = pDocumentContext->m_cchText;
pDocumentContext->Release();
return S_OK;
}
/* ============================================================================
CTemplate::GetContextOfPosition
Given the character position & number of characters in the document,
encapsulate this into a document context object.
*/
HRESULT CTemplate::GetContextOfPosition
(
/* [in] */ ULONG cchSourceOffset,
/* [in] */ ULONG cchText,
/* [out] */ IDebugDocumentContext **ppDocumentContext
)
{
if (
(*ppDocumentContext = new CTemplateDocumentContext(this, cchSourceOffset, cchText))
== NULL
)
return E_OUTOFMEMORY;
return S_OK;
}
/* ****************************************************************************
IConnectionPointContainer implementation
*/
/* ============================================================================
CTemplate::FindConnectionPoint
From a character offset and length, return the document text
*/
HRESULT CTemplate::FindConnectionPoint
(
const GUID &uidConnection,
IConnectionPoint **ppCP
)
{
if (uidConnection == IID_IDebugDocumentTextEvents)
return m_CPTextEvents.QueryInterface(IID_IConnectionPoint, reinterpret_cast<void **>(ppCP));
else
{
*ppCP = NULL;
return E_NOINTERFACE;
}
}
/* ============================================================================
CTemplate::AttachTo
attach this to the debugger UI tree view.
*/
HRESULT CTemplate::AttachTo
(
CAppln *pAppln
)
{
if (!m_fDontAttach && pAppln->FDebuggable())
{
// If we are already attached to this application, then ignore 2nd request
CDblLink *pNodeCurr = m_listDocNodes.PNext();
while (pNodeCurr != &m_listDocNodes)
{
if (pAppln == static_cast<CDocNodeElem *>(pNodeCurr)->m_pAppln)
return S_OK;
pNodeCurr = pNodeCurr->PNext();
}
// Create the node and store it in the linked list.
HRESULT hr;
IDebugApplicationNode *pDocRoot;
CDocNodeElem *pDocNodeElem;
// Create a document tree, showing the include file hierarchy
if (FAILED(hr = CreateDocumentTree(m_rgpFilemaps[0], &pDocRoot)))
return hr;
if (FAILED(hr = pDocRoot->Attach(pAppln->PAppRoot())))
return hr;
if ((pDocNodeElem = new CDocNodeElem(pAppln, pDocRoot)) == NULL)
return E_OUTOFMEMORY;
pDocNodeElem->AppendTo(m_listDocNodes);
pDocRoot->Release();
m_fDebuggable = TRUE;
}
return S_OK;
}
/* ============================================================================
CTemplate::DetachFrom
detach this from the debugger UI tree view.
*/
HRESULT CTemplate::DetachFrom
(
CAppln *pAppln
)
{
// Enter the CS to prevent Detach() from detaching while we are scanning
// the list (causes application ptr to be deleted twice if this occurs)
DBG_ASSERT(m_fDebuggerDetachCSInited);
EnterCriticalSection(&m_csDebuggerDetach);
// Look for the node that has this application
CDblLink *pNodeCurr = m_listDocNodes.PNext();
while (pNodeCurr != &m_listDocNodes)
{
if (pAppln == static_cast<CDocNodeElem *>(pNodeCurr)->m_pAppln)
break;
pNodeCurr = pNodeCurr->PNext();
}
// If not found (pNodeCurr points back to head), then fail
if (pNodeCurr == &m_listDocNodes)
{
LeaveCriticalSection(&m_csDebuggerDetach);
return E_FAIL;
}
// Detach the node by deleting the current element
delete pNodeCurr;
// Turn off "Debuggable" flag if last application is detached
m_fDebuggable = !m_listDocNodes.FIsEmpty();
// At this point CS not needed
LeaveCriticalSection(&m_csDebuggerDetach);
// If we have just removed ourselves from the last application,
// then we call Detach(), to remove all cached script engines now.
if (!m_fDebuggable)
Detach();
return S_OK;
}
/* ============================================================================
CTemplate::Detach
detach this from the debugger UI tree view.
*/
HRESULT CTemplate::Detach
(
)
{
// Enter the CS to prevent DetachFrom() from detaching while we are clearing
// the list (causes application ptr to be deleted twice if this occurs)
if (m_fDebuggerDetachCSInited)
EnterCriticalSection(&m_csDebuggerDetach);
// Detach all nodes
while (! m_listDocNodes.FIsEmpty())
delete m_listDocNodes.PNext();
// Done with CS
if (m_fDebuggerDetachCSInited)
LeaveCriticalSection(&m_csDebuggerDetach);
// Since we are not debuggable now, remove any script engines we may
// be holding on to. If we are detaching from change notification
// thread, queue engines to be released from debugger thread.
//
if (m_rgpDebugScripts)
{
Assert (g_dwDebugThreadId != 0);
BOOL fCalledFromDebugActivity = GetCurrentThreadId() == g_dwDebugThreadId;
for (UINT i = 0; i < m_cScriptEngines; i++)
{
CActiveScriptEngine *pEngine = m_rgpDebugScripts[i];
if (pEngine)
{
if (fCalledFromDebugActivity)
{
pEngine->FinalRelease();
}
else
{
g_ApplnMgr.AddEngine(pEngine);
pEngine->Release();
}
}
}
delete[] m_rgpDebugScripts;
m_rgpDebugScripts = NULL;
}
m_fDebuggable = FALSE;
return S_OK;
}
/* ============================================================================
CTemplate::CreateDocumentTree
Traverse the tree that we have embedded in the filemap structures,
and use it to create the include file structure
*/
HRESULT CTemplate::CreateDocumentTree
(
CFileMap *pfilemapRoot,
IDebugApplicationNode **ppDocRoot
)
{
IDebugApplicationNode *pDocNode;
HRESULT hr = S_OK;
if (pfilemapRoot == NULL || ppDocRoot == NULL)
return E_POINTER;
// Create the root node
if (FAILED(hr = g_pDebugApp->CreateApplicationNode(ppDocRoot)))
return hr;
// From the filemap information, match it up with the correct provider
// "This" is the provider for the root document, others come from Inc file cache
if (pfilemapRoot == m_rgpFilemaps[0])
{
if (FAILED(hr = (*ppDocRoot)->SetDocumentProvider(this)))
return hr;
}
else
{
CIncFile *pIncFile;
if (FAILED(hr = g_IncFileMap.GetIncFile(pfilemapRoot->m_szPathTranslated, &pIncFile)))
return hr;
if (FAILED(hr = (*ppDocRoot)->SetDocumentProvider(pIncFile)))
return hr;
// SetDocumentProvider AddRef'ed
pIncFile->Release();
}
// Create a node from all of the children and attach it to this node
CFileMap *pfilemapChild = pfilemapRoot->m_pfilemapChild;
while (pfilemapChild != NULL)
{
IDebugApplicationNode *pDocChild;
if (FAILED(hr = CreateDocumentTree(pfilemapChild, &pDocChild)))
return hr;
if (FAILED(hr = pDocChild->Attach(*ppDocRoot)))
return hr;
pfilemapChild = pfilemapChild->m_fHasSibling? pfilemapChild->m_pfilemapSibling : NULL;
}
return S_OK;
}
/* ============================================================================
CTemplate::End
Place template in non-usable state (after this is called, last ref. should
be the any currently executing scripts. The count will naturally vanish
as the scripts finish. The template should never be recycled in cache after
this call.)
REF COUNTING NOTE:
Since debugging client has a reference to the template, the template needs
to dis-associate with the debugger at a point in time before destruction.
Otherwise, the reference will never go to zero.
*/
ULONG
CTemplate::End
(
)
{
// Flag template as non-usable (for debugging)
m_fIsValid = FALSE;
Detach();
if (!m_CPTextEvents.FIsEmpty() && g_pDebugApp != NULL)
{
IEnumConnections *pConnIterator;
if (SUCCEEDED(m_CPTextEvents.EnumConnections(&pConnIterator)))
{
CONNECTDATA ConnectData;
while (pConnIterator->Next(1, &ConnectData, NULL) == S_OK)
{
IDebugDocumentTextEvents *pTextEventSink;
if (SUCCEEDED(ConnectData.pUnk->QueryInterface(IID_IDebugDocumentTextEvents, reinterpret_cast<void **>(&pTextEventSink))))
{
InvokeDebuggerWithThreadSwitch(g_pDebugApp, DEBUGGER_ON_DESTROY, pTextEventSink);
pTextEventSink->Release();
}
ConnectData.pUnk->Release();
}
pConnIterator->Release();
}
}
return Release();
}
/* ============================================================================
CTemplate::NotifyDebuggerOnPageEvent
Let debugger know about page start/end
*/
HRESULT
CTemplate::NotifyDebuggerOnPageEvent
(
BOOL fStart // TRUE = StartPage, FALSE = EndPage
)
{
CTemplateDocumentContext *pDebugContext = new CTemplateDocumentContext(this, 0, 0);
if (pDebugContext == NULL)
return E_OUTOFMEMORY;
HRESULT hr = S_OK;
if (g_pDebugApp)
hr = InvokeDebuggerWithThreadSwitch
(
g_pDebugApp,
fStart ? DEBUGGER_EVENT_ON_PAGEBEGIN : DEBUGGER_EVENT_ON_PAGEEND,
static_cast<IUnknown *>(pDebugContext)
);
pDebugContext->Release();
return hr;
}
/* ============================================================================
CTemplate::ReleaseTypeLibs
Release all typelibs collected from metadata
*/
void
CTemplate::ReleaseTypeLibs()
{
if (m_rgpTypeLibs.length() > 0)
{
for (UINT i = 0; i < m_rgpTypeLibs.length(); i++)
{
m_rgpTypeLibs[i]->Release();
}
m_rgpTypeLibs.reshape(0);
}
}
/* ============================================================================
CTemplate::WrapTypeLibs
Wrap all typelibs collected from metadata into single IDispatch *
*/
void
CTemplate::WrapTypeLibs(CHitObj *pHitObj)
{
HRESULT hr = S_OK;
Assert(m_pdispTypeLibWrapper == NULL);
if (m_rgpTypeLibs.length() > 0)
{
hr = ::WrapTypeLibs
(
m_rgpTypeLibs.begin(),
m_rgpTypeLibs.length(),
&m_pdispTypeLibWrapper
);
ReleaseTypeLibs();
}
if (FAILED(hr))
{
m_pbErrorLocation = NULL;
m_idErrMsg = IDE_TEMPLATE_WRAP_TYPELIB_FAILED;
ProcessSpecificError(*(m_rgpFilemaps[0]), pHitObj);
THROW(E_TEMPLATE_COMPILE_FAILED_DONT_CACHE);
}
}
/* ============================================================================
CTemplate::Release449
Release all 449-echo-cookie objects collected from metadata
*/
void
CTemplate::Release449()
{
if (m_rgp449.length() > 0)
{
for (UINT i = 0; i < m_rgp449.length(); i++)
{
m_rgp449[i]->Release();
}
m_rgp449.reshape(0);
}
}
/* ============================================================================
CTemplate::Do449Processing
Generate 449 response in cookie negotiations with IE when needed
*/
HRESULT
CTemplate::Do449Processing
(
CHitObj *pHitObj
)
{
if (m_rgp449.length() == 0 || pHitObj->F449Done())
return S_OK;
HRESULT hr = ::Do449Processing
(
pHitObj,
m_rgp449.begin(),
m_rgp449.length()
);
pHitObj->Set449Done();
return hr;
}
#if 0
/* ============================================================================
CTemplate::OutputDebugTables
print the debugging data structures to the debug window
*/
void
CTemplate::OutputDebugTables()
{
unsigned i, j;
wchar_t wszDebugLine[256];
CWCharToMBCS convTarget;
CWCharToMBCS convSource;
// print line mapping table
DBGPRINTF((DBG_CONTEXT, "\nEngine HTML? Line# SourceOffset Length TargetOffset TargetText__________ SourceText__________ File\n"));
for (i = 0; i < m_cScriptEngines; ++i)
for (j = 0; j < m_rgrgSourceInfos[i].length(); ++j)
{
wchar_t wszSourceText[SNIPPET_SIZE + 1], wszTargetText[SNIPPET_SIZE + 1];
CSourceInfo *pSourceInfo = &m_rgrgSourceInfos[i][j];
// DON'T display sample script text on last line of each engine
if (j == m_rgrgSourceInfos[i].length() - 1)
{
wszTargetText[0] = 0;
wszSourceText[0] = 0;
}
else
{
// Get source & target text sample
GetScriptSnippets(
pSourceInfo->m_cchSourceOffset, pSourceInfo->m_pfilemap,
pSourceInfo->m_cchTargetOffset, i,
wszSourceText, wszTargetText
);
// Actually display each line
#if 0
#ifndef _NO_TRACING_
convTarget.Init(wszTargetText);
convSource.Init(wszSourceText);
DBGINFO((DBG_CONTEXT,
"%-6d %-5s %-5d %-12d %-6d %-12d %-20s %-20s %s\n",
i,
pSourceInfo->m_fIsHTML? "Yes" : "No",
pSourceInfo->m_idLine,
pSourceInfo->m_cchSourceOffset,
pSourceInfo->m_cchSourceText,
pSourceInfo->m_cchTargetOffset,
convTarget.GetString(),
convSource.GetString(),
pSourceInfo->m_pfilemap->m_szPathTranslated));
#else
CMBCSToWChar convPath;
convPath.Init(pSourceInfo->m_pfilemap->m_szPathTranslated);
wsprintfW(
wszDebugLine,
L"%-6d %-5s %-5d %-12d %-6d %-12d %-20s %-20s %s\n",
i,
pSourceInfo->m_fIsHTML? L"Yes" : L"No",
pSourceInfo->m_idLine,
pSourceInfo->m_cchSourceOffset,
pSourceInfo->m_cchSourceText,
pSourceInfo->m_cchTargetOffset,
wszTargetText,
wszSourceText,
convPath.GetString());
OutputDebugStringW(wszDebugLine);
#endif
#endif
}
}
OutputDebugStringA("\n\n");
for (i = 0; i < m_cFilemaps; ++i)
{
CFileMap *pFilemap = m_rgpFilemaps[i];
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "DBCS mapping table for File %S:\n", pFilemap->m_szPathTranslated));
#else
DBGPRINTF((DBG_CONTEXT, "DBCS mapping table for File %s:\n", pFilemap->m_szPathTranslated));
#endif
DBGPRINTF((DBG_CONTEXT, "ByteOffset CharOffset\n"));
for (COffsetInfo *pOffsetInfo = pFilemap->m_rgByte2DBCS.begin();
pOffsetInfo < pFilemap->m_rgByte2DBCS.end();
++pOffsetInfo)
DebugPrintf("%-10d %-10d\n", pOffsetInfo->m_cbOffset, pOffsetInfo->m_cchOffset);
DBGPRINTF((DBG_CONTEXT, "\n\n"));
}
DBGPRINTF((DBG_CONTEXT, "Include File Hierarchy\n"));
OutputIncludeHierarchy(m_rgpFilemaps[0], 0);
DBGPRINTF((DBG_CONTEXT, "\n"));
}
/* ============================================================================
CTemplate::OutputIncludeHierarchy
print the lineage information that we keep around for include files.
Print all nodes on one level at the current indentation, then descend for
nested includes.
*/
void
CTemplate::OutputIncludeHierarchy
(
CFileMap* pfilemap,
int cchIndent
)
{
TCHAR szDebugString[256], *pchEnd;
for (;;)
{
pchEnd = szDebugString;
for (int i = 0; i < cchIndent; ++i)
*pchEnd++ = _T(' ');
pchEnd = strcpyEx(pchEnd, pfilemap->m_szPathTranslated);
*pchEnd++ = _T('\n');
*pchEnd = _T('\0');
DBGPRINTF((DBG_CONTEXT, szDebugString));
// Print anything that this file includes
if (pfilemap->m_pfilemapChild)
OutputIncludeHierarchy(pfilemap->m_pfilemapChild, cchIndent + 3);
// Stop when there are no more siblings on this level
if (! pfilemap->m_fHasSibling)
break;
// Advance to next sibling
pfilemap = pfilemap->m_pfilemapSibling;
}
}
/* ============================================================================
CTemplate::OutputScriptSnippets
print some script from both the source offset & its corresponding target.
Good way to visually see if the offset conversions are working.
*/
void
CTemplate::GetScriptSnippets
(
ULONG cchSourceOffset,
CFileMap *pFilemapSource,
ULONG cchTargetOffset,
ULONG idTargetEngine,
wchar_t *wszSourceText,
wchar_t *wszTargetText
)
{
// Get target text sample
if (wszTargetText)
{
char *szEngineName;
PROGLANG_ID *pProgLangID;
const wchar_t *wszScriptText;
GetScriptBlock(idTargetEngine, &szEngineName, &pProgLangID, &wszScriptText);
wszScriptText += cchTargetOffset;
int cch = wcslen(wszScriptText);
wcsncpy(wszTargetText, wszScriptText, min(cch, SNIPPET_SIZE) + 1);
wszTargetText[min(cch, SNIPPET_SIZE)] = 0;
// Convert newlines to space
wchar_t *pwch = wszTargetText;
while (*pwch != 0)
if (iswspace(*pwch++))
pwch[-1] = ' ';
}
// Get source text sample
if (wszSourceText)
{
ULONG cchMax = 0;
pFilemapSource->GetText((WORD)m_wCodePage, cchSourceOffset, wszSourceText, NULL, &cchMax, SNIPPET_SIZE);
wszSourceText[cchMax] = 0;
// Convert newlines to space
wchar_t *pwch = wszSourceText;
while (*pwch != 0)
if (iswspace(*pwch++))
pwch[-1] = ' ';
}
}
#endif
/* ============================================================================
CTemplate::BuildPersistedDACL
Builds a DACL based on the SECURITY_DESCRIPTOR already
associated with the template. The PersistedDACL is modified to include
full access for administrators and delete access for everyone.
*/
HRESULT CTemplate::BuildPersistedDACL(PACL *ppRetDACL)
{
HRESULT hr = S_OK;
BOOL bDaclPresent;
BOOL bDaclDefaulted;
PACL pSrcDACL = NULL;
EXPLICIT_ACCESS ea;
SID_IDENTIFIER_AUTHORITY WorldAuthority = SECURITY_WORLD_SID_AUTHORITY;
*ppRetDACL = NULL;
ZeroMemory(&ea, sizeof(EXPLICIT_ACCESS));
ea.grfAccessPermissions = SYNCHRONIZE | DELETE;
ea.grfAccessMode = GRANT_ACCESS;
ea.grfInheritance= SUB_CONTAINERS_AND_OBJECTS_INHERIT;
ea.Trustee.TrusteeForm = TRUSTEE_IS_SID;
if (m_rgpFilemaps[0]->m_pSecurityDescriptor == NULL) {
return S_OK;
}
if (!AllocateAndInitializeSid(&WorldAuthority,
1,
SECURITY_WORLD_RID,
0,0,0,0,0,0,0,
(PSID *)(&ea.Trustee.ptstrName)))
hr = HRESULT_FROM_WIN32(GetLastError());
else if (!GetSecurityDescriptorDacl(m_rgpFilemaps[0]->m_pSecurityDescriptor,
&bDaclPresent,
&pSrcDACL,
&bDaclDefaulted))
hr = HRESULT_FROM_WIN32(GetLastError());
else if ((hr = SetEntriesInAcl(1,
&ea,
bDaclPresent ? pSrcDACL : NULL,
ppRetDACL)) != ERROR_SUCCESS)
hr = HRESULT_FROM_WIN32(hr);
if (ea.Trustee.ptstrName)
FreeSid(ea.Trustee.ptstrName);
return hr;
}
/* ============================================================================
CTemplate::PersistData
Attempts to write the contents of the template memory to disk. Note that
the memory isn't freed here but later when the template ref count falls to
1 (indicating that the only reference to the template is the one that the
cache has on it).
*/
HRESULT CTemplate::PersistData(char *pszTempFilePath)
{
HRESULT hr = S_OK;
DWORD winErr = 0;
HANDLE hFile = NULL;
DWORD dwWritten;
HANDLE hImpersonationToken = NULL;
HANDLE hThread;
PACL pPersistDACL = NULL;
#if DBG_PERSTEMPL
DBGPRINTF((DBG_CONTEXT,
"CTemplate::PersistData() enterred.\n\tTemplate is %s\n\tPersistTempName is %s\n",
GetSourceFileName(),
m_szPersistTempName ? m_szPersistTempName : "<none>"));
#endif
// if for some reason this template has been marked as invalid, then it is
// not persistable
if (m_fIsValid == FALSE) {
hr = E_FAIL;
goto end;
}
// if it is already persisted, there is nothing to do
if (m_fIsPersisted) {
goto end;
}
// check to see if we already have a persist temp name. If a template moves
// from the persisted cache back to the memory cache, then the persisted flag
// will have been lifted but the cache name will remain as an optimization for
// future persisting.
if (m_szPersistTempName == NULL) {
hThread = GetCurrentThread();
if (OpenThreadToken( hThread,
TOKEN_READ | TOKEN_IMPERSONATE,
TRUE,
&hImpersonationToken )) {
RevertToSelf();
}
// allocate memory for this temp path
if (!(m_szPersistTempName = (LPSTR)CTemplate::LargeMalloc(MAX_PATH))) {
hr = E_OUTOFMEMORY;
}
// create the temp file. The location of the temp directory was passed
// in as an argument. The resulting tempfile name in m_szPersistTempName
// will include this path.
else if (GetTempFileNameA(pszTempFilePath,
"ASPTemplate",
0,
m_szPersistTempName) == 0) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
// build a security descriptor to use with this persisted file. It is
// comprised of the .asp's security descriptor plus a couple of DACLs
// to allow administrators full access and everyone delete access.
else if (FAILED(hr = BuildPersistedDACL(&pPersistDACL)));
else if (pPersistDACL
&& (winErr = SetNamedSecurityInfoA((LPSTR)m_szPersistTempName,
SE_FILE_OBJECT,
DACL_SECURITY_INFORMATION,
NULL,
NULL,
pPersistDACL,
NULL)))
hr = HRESULT_FROM_WIN32(winErr);
// create the file
else if ((hFile = CreateFileA(m_szPersistTempName,
GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL,
NULL)) == INVALID_HANDLE_VALUE) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
// slam out the entire contents of the template memory to the file
else if (WriteFile(hFile,
m_pbStart,
m_cbTemplate,
&dwWritten,
NULL) == 0) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
// close
else if (CloseHandle(hFile) == 0) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
else {
hFile = NULL;
}
if (FAILED(hr));
// make sure that the entire amount was written out
else if (dwWritten != m_cbTemplate) {
hr = E_FAIL;
}
if (hImpersonationToken) {
SetThreadToken(&hThread, hImpersonationToken);
CloseHandle(hImpersonationToken);
}
}
if (FAILED(hr));
else {
// if successfull, then note that the template is now persisted.
// Do an AddRef and Release as a safe way to check to see if the
// template memory can be freed.
m_fIsPersisted = TRUE;
AddRef();
Release();
}
// if errors occurred, clean up any resources.
if (hr != S_OK) {
if (hFile)
CloseHandle(hFile);
if (m_szPersistTempName)
CTemplate::LargeFree(m_szPersistTempName);
m_szPersistTempName = NULL;
}
// free the persisted SECURITY_DESCRIPTOR if allocated
if (pPersistDACL) {
LocalFree(pPersistDACL);
}
end:
#if DBG_PERSTEMPL
if (hr == S_OK) {
DBGPRINTF((DBG_CONTEXT,
"Persist Successful. TempName is %s\n",
m_szPersistTempName));
}
else {
DBGPRINTF((DBG_CONTEXT,
"Persist failed. hr = %x",
hr));
}
#endif
return hr;
}
/* ============================================================================
CTemplate::UnPersistData
Restores the template memory from disk.
*/
HRESULT CTemplate::UnPersistData()
{
HRESULT hr = S_OK;
HANDLE hFile = NULL;
DWORD dwRead;
HANDLE hImpersonationToken = NULL;
HANDLE hThread;
#if DEB_PERSTEMPL
DBGPRINTF((DBG_CONTEXT,
"CTemplate::UnPersistData() enterred.\n\tTemplate is %s\n\tTempName is %s\n",
m_rgpFilemaps[0]->m_szPathTranslated,
m_szPersistTempName));
#endif
// check to see if the template is already loaded into memory. If so, then
// all this routine needs to do is lift the IsPersisted flag.
if (m_pbStart != NULL) {
m_fIsPersisted = FALSE;
goto end;
}
hThread = GetCurrentThread();
if (OpenThreadToken( hThread,
TOKEN_READ | TOKEN_IMPERSONATE,
TRUE,
&hImpersonationToken )) {
RevertToSelf();
}
// open the temp file for read
if ((hFile = CreateFileA(m_szPersistTempName,
GENERIC_READ,
0,
NULL,
OPEN_EXISTING,
0,
NULL)) == INVALID_HANDLE_VALUE) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
// allocate the template memory
else if (!(m_pbStart = (BYTE *)CTemplate::LargeMalloc(m_cbTemplate))) {
hr = E_OUTOFMEMORY;
}
// read in the entire file
else if (ReadFile(hFile,
m_pbStart,
m_cbTemplate,
&dwRead,
NULL) == 0) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
// we're done with the file
else if (CloseHandle(hFile) == 0) {
hr = HRESULT_FROM_WIN32(GetLastError());
}
else {
hFile = NULL;
}
if (FAILED(hr));
// check to make sure we got everything
else if (m_cbTemplate != dwRead) {
hr = E_FAIL;
}
else {
// if not, pretend like this is no longer persisted. Prevents errors
// in the future.
m_fIsPersisted = FALSE;
}
if (hr != S_OK) {
// make sure that the file handle was cleaned up
if (hFile)
CloseHandle(hFile);
}
end:
if (hImpersonationToken) {
SetThreadToken(&hThread, hImpersonationToken);
CloseHandle(hImpersonationToken);
}
#if DBG_PERSTEMPL
if (hr == S_OK) {
DBGPRINTF((DBG_CONTEXT,
"UnPersist Successful\n"));
}
else {
DBGPRINTF((DBG_CONTEXT,
"UnPersist failed. hr = %x",
hr));
}
#endif
return hr;
}
/* ============================================================================
CTemplate::PersistCleanup
Cleans up the temp file and the memory holding the temp file name.
*/
HRESULT CTemplate::PersistCleanup()
{
HRESULT hr = S_OK;
HANDLE hImpersonationToken = NULL;
HANDLE hThread;
if (m_szPersistTempName == NULL) {
return (S_OK);
}
hThread = GetCurrentThread();
if (OpenThreadToken( hThread,
TOKEN_READ | TOKEN_IMPERSONATE,
TRUE,
&hImpersonationToken )) {
RevertToSelf();
}
if (DeleteFileA(m_szPersistTempName) == 0) {
hr = GetLastError();
}
else {
m_fIsPersisted = FALSE;
CTemplate::LargeFree(m_szPersistTempName);
m_szPersistTempName = NULL;
}
if (hImpersonationToken) {
SetThreadToken(&hThread, hImpersonationToken);
CloseHandle(hImpersonationToken);
}
return hr;
}
/* ****************************************************************************
CIncFile member functions
*/
/* ============================================================================
CIncFile::CIncFile
Constructor
Returns:
Nothing
Side effects:
None
*/
CIncFile::CIncFile
(
)
: m_szIncFile(NULL),
m_fCsInited(FALSE),
m_CPTextEvents(this, IID_IDebugDocumentTextEvents),
m_cRefs(0)
{ }
/* ============================================================================
CIncFile::Init
Inits the CIncFile object
Returns:
HRESULT
Side effects:
None
*/
HRESULT
CIncFile::Init
(
const TCHAR* szIncFile // file name
)
{
HRESULT hr = S_OK;
WIN32_FILE_ATTRIBUTE_DATA fad; // win32 file attributes data structure
ErrInitCriticalSection(&m_csUpdate, hr);
m_fCsInited = TRUE;
if(NULL == (m_szIncFile = (LPTSTR) CTemplate::SmallMalloc((_tcslen(szIncFile) + 1)*sizeof(TCHAR)))) {
hr = E_OUTOFMEMORY;
goto LExit;
}
_tcscpy(m_szIncFile, szIncFile);
// init hash table element base class
if(FAILED(hr = CLinkElem::Init(m_szIncFile, _tcslen(m_szIncFile)*sizeof(TCHAR))))
goto LExit;
LExit:
return hr;
}
/* ============================================================================
CIncFile::~CIncFile
Destructor
Returns:
Nothing
Side effects:
None
*/
CIncFile::~CIncFile
(
)
{
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "Include file deleted: %S\n", m_szIncFile));
#else
DBGPRINTF((DBG_CONTEXT, "Include file deleted: %s\n", m_szIncFile));
#endif
Assert(m_cRefs == 0);
SmallTemplateFreeNullify((void**) &m_szIncFile);
if(m_fCsInited)
DeleteCriticalSection(&m_csUpdate);
}
/* ============================================================================
CIncFile::GetTemplate
Get i'th template user from CIncFile
Returns:
NULL if "iTemplate" is out of range, m_rgpTemplates[iTemplate] otherwise
Side effects:
None
*/
CTemplate*
CIncFile::GetTemplate
(
int iTemplate
)
{
if (iTemplate < 0 || iTemplate >= (signed int) m_rgpTemplates.length())
return NULL;
else
return m_rgpTemplates[iTemplate];
}
/* ============================================================================
CIncFile::QueryInterface
Provides QueryInterface implementation for CIncFile
NOTE: It is arbitrary which vtable we return for IDebugDocument & IDebugDocumentInfo.
*/
HRESULT
CIncFile::QueryInterface(const GUID &uidInterface, void **ppvObj)
{
if (uidInterface == IID_IUnknown || uidInterface == IID_IDebugDocumentProvider)
*ppvObj = static_cast<IDebugDocumentProvider *>(this);
else if (uidInterface == IID_IDebugDocument || uidInterface == IID_IDebugDocumentInfo || uidInterface == IID_IDebugDocumentText)
*ppvObj = static_cast<IDebugDocumentText *>(this);
else if (uidInterface == IID_IConnectionPointContainer)
*ppvObj = static_cast<IConnectionPointContainer *>(this);
else
*ppvObj = NULL;
if (*ppvObj)
{
AddRef();
return S_OK;
}
else
return E_NOINTERFACE;
}
/* ============================================================================
CIncFile::AddRef
Adds a ref to this IncFile, thread-safely
*/
ULONG
CIncFile::AddRef()
{
InterlockedIncrement(&m_cRefs);
return m_cRefs;
}
/* ============================================================================
CIncFile::Release
Releases a ref to this IncFile, thread-safely
*/
ULONG
CIncFile::Release()
{
if (InterlockedDecrement(&m_cRefs) == 0)
{
delete this;
return 0;
}
return m_cRefs;
}
/* ****************************************************************************
IDebugDocumentProvider implementation for includes
*/
/* ============================================================================
CIncFile::GetDocument
Return a pointer to the IDebugDocument implementation. (same object in this case)
Returns:
*ppDebugDoc is set to "this".
Notes:
always succeeds
*/
HRESULT CIncFile::GetDocument
(
IDebugDocument **ppDebugDoc
)
{
return QueryInterface(IID_IDebugDocument, reinterpret_cast<void **>(ppDebugDoc));
}
/* ============================================================================
CIncFile::GetName
Return the various names of a document.
*/
HRESULT CIncFile::GetName
(
/* [in] */ DOCUMENTNAMETYPE doctype,
/* [out] */ BSTR *pbstrName
)
{
switch (doctype) {
case DOCUMENTNAMETYPE_APPNODE:
case DOCUMENTNAMETYPE_FILE_TAIL:
case DOCUMENTNAMETYPE_TITLE:
// Use the name of the include file (char after last back-slash) converted to lower case.
{
TCHAR *szFilePart = _tcsrchr(m_szIncFile, _T('\\'));
Assert (szFilePart != NULL);
#if UNICODE
*pbstrName = SysAllocString(szFilePart + 1);
if (*pbstrName == NULL) {
return E_OUTOFMEMORY;
}
#else
if (FAILED(SysAllocStringFromSz(szFilePart + 1, 0, pbstrName, CP_ACP)))
return E_FAIL;
#endif
if (*pbstrName != NULL)
_wcslwr(*pbstrName);
return S_OK;
}
case DOCUMENTNAMETYPE_URL:
// prefix with the URL, use szPathInfo for the rest of the path
{
CTemplate::CFileMap *pFilemap = GetFilemap();
if (pFilemap->FHasVirtPath()) {
STACK_BUFFER( tempName, MAX_PATH );
CTemplate *pTemplate = m_rgpTemplates[0];
int cbURLPrefix = DIFF(pTemplate->m_szApplnVirtPath - pTemplate->m_szApplnURL)*sizeof(TCHAR);
if (!tempName.Resize(cbURLPrefix + ((_tcslen(pFilemap->m_szPathInfo) + 1)*sizeof(TCHAR)))) {
return E_OUTOFMEMORY;
}
TCHAR *szURL = (TCHAR *)tempName.QueryPtr();
memcpy(szURL, pTemplate->m_szApplnURL, cbURLPrefix);
_tcscpy(&szURL[cbURLPrefix/sizeof(TCHAR)], pFilemap->m_szPathInfo);
#if UNICODE
*pbstrName = SysAllocString(szURL);
if (*pbstrName == NULL) {
return (E_OUTOFMEMORY);
}
return S_OK;
#else
return SysAllocStringFromSz(szURL, 0, pbstrName, pTemplate->m_wCodePage);
#endif
}
else {
*pbstrName = NULL;
return E_FAIL;
}
}
default:
return E_FAIL;
}
}
/* ****************************************************************************
IDebugDocumentText implementation
*/
/* ============================================================================
CIncFile::GetSize
Return the number of lines & characters in the document
*/
HRESULT CIncFile::GetSize
(
/* [out] */ ULONG *pcLines,
/* [out] */ ULONG *pcChars
)
{
CTemplate::CFileMap *pfilemap = GetFilemap();
*pcLines = ULONG_MAX;
*pcChars = pfilemap->m_cChars;
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "GetSize(\"%S\") returns %lu characters (%lu lines)\n", pfilemap->m_szPathTranslated, *pcChars, *pcLines));
#else
DBGPRINTF((DBG_CONTEXT, "GetSize(\"%s\") returns %lu characters (%lu lines)\n", pfilemap->m_szPathTranslated, *pcChars, *pcLines));
#endif
return S_OK;
}
/* ============================================================================
CTemplate::GetDocumentAttributes
Return doc attributes
*/
HRESULT CIncFile::GetDocumentAttributes
(
/* [out] */ TEXT_DOC_ATTR *ptextdocattr
)
{
// Easy way to tell debugger that we don't support editing.
*ptextdocattr = TEXT_DOC_ATTR_READONLY;
return S_OK;
}
/* ============================================================================
CIncFile::GetPositionOfLine
From a line number, return the character offset of the beginning
I don't think we need this function. It is meant to support line oriented
debuggers, of which Caesar is not one.
*/
HRESULT CIncFile::GetPositionOfLine
(
/* [in] */ ULONG cLineNumber,
/* [out] */ ULONG *pcCharacterPosition
)
{
return m_rgpTemplates[0]->GetPositionOfLine(GetFilemap(), cLineNumber, pcCharacterPosition);
}
/* ============================================================================
CIncFile::GetLineOfPosition
From a character offset, return the line number and offset within the line
I don't think we need this function. It is meant to support line oriented
debuggers, of which Caesar is not one.
*/
HRESULT CIncFile::GetLineOfPosition
(
/* [in] */ ULONG cCharacterPosition,
/* [out] */ ULONG *pcLineNumber,
/* [out] */ ULONG *pcCharacterOffsetInLine
)
{
return m_rgpTemplates[0]->GetLineOfPosition(GetFilemap(), cCharacterPosition, pcLineNumber, pcCharacterOffsetInLine);
}
/* ============================================================================
CIncFile::GetText
From a character offset and length, return the document text
*/
HRESULT CIncFile::GetText
(
ULONG cchSourceOffset,
WCHAR *pwchText,
SOURCE_TEXT_ATTR *pTextAttr,
ULONG *pcChars,
ULONG cMaxChars
)
{
return GetFilemap()->GetText((WORD)m_rgpTemplates[0]->m_wCodePage, cchSourceOffset, pwchText, pTextAttr, pcChars, cMaxChars);
}
/* ============================================================================
CIncFile::GetPositionOfContext
Decompose a document context into the document offset & length
*/
HRESULT CIncFile::GetPositionOfContext
(
/* [in] */ IDebugDocumentContext *pUnknownDocumentContext,
/* [out] */ ULONG *pcchSourceOffset,
/* [out] */ ULONG *pcchText
)
{
// Make sure that the context is one of ours
CIncFileDocumentContext *pDocumentContext;
if (FAILED(pUnknownDocumentContext->QueryInterface(IID_IDenaliIncFileDocumentContext, reinterpret_cast<void **>(&pDocumentContext))))
return E_FAIL;
if (pcchSourceOffset)
*pcchSourceOffset = pDocumentContext->m_cchSourceOffset;
if (pcchText)
*pcchText = pDocumentContext->m_cchText;
pDocumentContext->Release();
return S_OK;
}
/* ============================================================================
CIncFile::GetContextOfPosition
Given the character position & number of characters in the document,
encapsulate this into a document context object.
*/
HRESULT CIncFile::GetContextOfPosition
(
/* [in] */ ULONG cchSourceOffset,
/* [in] */ ULONG cchText,
/* [out] */ IDebugDocumentContext **ppDocumentContext
)
{
if (
(*ppDocumentContext = new CIncFileDocumentContext(this, cchSourceOffset, cchText))
== NULL
)
return E_OUTOFMEMORY;
return S_OK;
}
/* ****************************************************************************
IConnectionPointContainer implementation
*/
/* ============================================================================
CIncFile::FindConnectionPoint
From a character offset and length, return the document text
*/
HRESULT CIncFile::FindConnectionPoint
(
const GUID &uidConnection,
IConnectionPoint **ppCP
)
{
if (uidConnection == IID_IDebugDocumentTextEvents)
return m_CPTextEvents.QueryInterface(IID_IConnectionPoint, reinterpret_cast<void **>(ppCP));
else
{
*ppCP = NULL;
return E_NOINTERFACE;
}
}
/* ============================================================================
CIncFile::GetFilemap
Returns a CFileMap pointer for this include file. (Note: There are several
CFileMaps that may be used, corresponding to each template. This function
selects one of them.)
Returns:
Corresponding CFileMap
Side effects:
None
*/
CTemplate::CFileMap *
CIncFile::GetFilemap
(
)
{
// Get pointer to first template's filemaps
CTemplate::CFileMap **ppFilemapInc = &m_rgpTemplates[0]->m_rgpFilemaps[1];
BOOL fFoundInc = FALSE;
// Look for the filemap whose name corresponds to this IncFile. It had better exist
// in all template filemaps.
// NOTE: Start searching at position 1, because position 0 is the template itself.
//
for (unsigned i = 1; i < m_rgpTemplates[0]->m_cFilemaps && !fFoundInc; ++i)
if (_tcscmp(m_szIncFile, (*ppFilemapInc++)->m_szPathTranslated) == 0)
fFoundInc = TRUE;
Assert (fFoundInc);
return ppFilemapInc[-1];
}
/* ============================================================================
CIncFile::AddTemplate
Adds a template to the list of templates that include this inc-file
Returns:
HRESULT
Side effects:
None
*/
HRESULT
CIncFile::AddTemplate
(
CTemplate* pTemplate
)
{
EnterCriticalSection(&m_csUpdate);
// Add the template to the list only if it does not exist
if (m_rgpTemplates.find(pTemplate) == -1)
{
if (FAILED(m_rgpTemplates.append(pTemplate)))
{
LeaveCriticalSection(&m_csUpdate);
return E_OUTOFMEMORY;
}
// Notify the debugger that template dependency has changed
// (Ignore failure)
//
if (g_pDebugApp)
{
IF_DEBUG(SCRIPT_DEBUGGER)
DBGPRINTF((DBG_CONTEXT, "AddTemplate: Notifying debugger to refresh breakpoints\n"));
InvokeDebuggerWithThreadSwitch
(
g_pDebugApp,
DEBUGGER_EVENT_ON_REFRESH_BREAKPOINT,
static_cast<IDebugDocument *>(this)
);
}
}
LeaveCriticalSection(&m_csUpdate);
return S_OK;
}
/* ============================================================================
CIncFile::RemoveTemplate
Removes a template from the template list
Returns:
Nothing
Side effects:
Compresses the removed template's ptr out of template ptrs array (see "back-copy", below)
Decrements template count
*/
void
CIncFile::RemoveTemplate
(
CTemplate* pTemplate
)
{
EnterCriticalSection(&m_csUpdate);
// find the template in list
int i = m_rgpTemplates.find(pTemplate);
// Remove the element (If we found it - possible that this is 2nd instance of #include and was previously removed)
if (i != -1)
{
m_rgpTemplates.removeAt(i);
// Notify the debugger that template dependency has changed
// (Ignore failure)
//
if (g_pDebugApp)
{
IF_DEBUG(SCRIPT_DEBUGGER)
DBGPRINTF((DBG_CONTEXT, "RemoveTemplate: Notifying debugger to refresh breakpoints\n"));
InvokeDebuggerWithThreadSwitch
(
g_pDebugApp,
DEBUGGER_EVENT_ON_REFRESH_BREAKPOINT,
static_cast<IDebugDocument *>(this)
);
}
}
LeaveCriticalSection(&m_csUpdate);
}
/* ============================================================================
CIncFile::FlushTemplates
Flushes all of this inc-file's templates from the global template cache
Returns:
TRUE if all templates flushed, FALSE if some left
Side effects:
None
*/
BOOL
CIncFile::FlushTemplates
(
)
{
/* NOTE we have a cross-dependency with RemoveTemplate() because the following call chain
occurs when an inc-file gets flushed:
CIncFileMap::Flush
CIncFile::FlushTemplates
CTemplateCacheManager::Flush
CTemplate::RemoveFromIncFiles
CIncFile::RemoveTemplate
The problem is that RemoveTemplate() updates m_cTemplates and m_rgTemplates, so these members
will not be stable during the loop within FlushTemplates.
To get around this, we make a local copy of m_rgTemplates.
*/
EnterCriticalSection(&m_csUpdate);
STACK_BUFFER( tempTemplates, 128 );
STACK_BUFFER( tempFile, MAX_PATH );
UINT cTemplates = m_rgpTemplates.length();
if (!tempTemplates.Resize(cTemplates * sizeof(CTemplate*))) {
// failed to get memory. The best we can do is return FALSE to indicate
// that not all templates where flushed.
LeaveCriticalSection(&m_csUpdate);
return FALSE;
}
CTemplate** rgpTemplates = static_cast<CTemplate**> (tempTemplates.QueryPtr());
memcpy(rgpTemplates, m_rgpTemplates.vec(), sizeof(CTemplate *) * cTemplates);
UINT cTemplatesFlushed = 0;
for(UINT i = 0; i < cTemplates; i++)
{
// If the template is ready now, flush it
if(rgpTemplates[i]->m_fReadyForUse && !(rgpTemplates[i]->m_fDontCache))
{
// bug 917: make a local copy of template file name, since the member gets freed part way through g_TemplateCache.Flush
TCHAR* szTemp = NULL;
szTemp = rgpTemplates[i]->GetSourceFileName();
if (szTemp)
{
if (!tempFile.Resize((_tcslen(szTemp) + 1)*sizeof(TCHAR))) {
// failed on this one. Continue and try to flush as many
// as we can.
continue;
}
TCHAR *szTemplateFile = (TCHAR *)tempFile.QueryPtr();
_tcscpy(szTemplateFile, szTemp);
g_TemplateCache.Flush(szTemplateFile, MATCH_ALL_INSTANCE_IDS);
cTemplatesFlushed++;
}
}
// If the template was not ready, we don't flush. It will probably
// pick up the current include file anyway
}
LeaveCriticalSection(&m_csUpdate);
return (cTemplates == cTemplatesFlushed);
}
/* ============================================================================
CIncFile::OnIncFileDecache
Callback which we use to call onDestroy events in the debugger just before
we are removed from the IncFile cache.
REF COUNTING NOTE:
Since debugging client has a reference to the IDebugDocument, the include needs
to dis-associate with the debugger at a point in time before destruction.
Otherwise, the reference will never go to zero.
*/
void
CIncFile::OnIncFileDecache
(
)
{
if (m_CPTextEvents.FIsEmpty() || g_pDebugApp == NULL)
return;
IEnumConnections *pConnIterator;
if (SUCCEEDED(m_CPTextEvents.EnumConnections(&pConnIterator)))
{
CONNECTDATA ConnectData;
while (pConnIterator->Next(1, &ConnectData, NULL) == S_OK)
{
IDebugDocumentTextEvents *pTextEventSink;
if (SUCCEEDED(ConnectData.pUnk->QueryInterface(IID_IDebugDocumentTextEvents, reinterpret_cast<void **>(&pTextEventSink))))
{
InvokeDebuggerWithThreadSwitch(g_pDebugApp, DEBUGGER_ON_DESTROY, pTextEventSink);
pTextEventSink->Release();
}
ConnectData.pUnk->Release();
}
pConnIterator->Release();
}
}
/* ****************************************************************************
CTemplate::CBuffer member functions
*/
/* ============================================================================
CTemplate::CBuffer::CBuffer
Ctor
*/
CTemplate::CBuffer::CBuffer()
:
m_pItems(NULL),
m_cSlots(0),
m_cItems(0),
m_pbData(NULL),
m_cbData(0),
m_cbDataUsed(0)
{
}
/* ============================================================================
CTemplate::CBuffer::~CBuffer
Dtor
*/
CTemplate::CBuffer::~CBuffer()
{
if(m_pItems)
CTemplate::SmallFree(m_pItems);
if(m_pbData)
CTemplate::LargeFree(m_pbData);
}
/* ============================================================================
CTemplate::CBuffer::Init
Inits a CBuffer
*/
void
CTemplate::CBuffer::Init
(
USHORT cSlots,
ULONG cbData
)
{
m_cSlots = cSlots;
m_cbData = cbData;
// Allocate space for storing byte range items
if(!(m_pItems = (CByteRange*) CTemplate::SmallMalloc(m_cSlots * sizeof(CByteRange))))
THROW(E_OUTOFMEMORY);
// Allocate space for storing local data, if there is any
if(m_cbData > 0)
{
if(!(m_pbData = (BYTE*) CTemplate::LargeMalloc(m_cbData)))
THROW(E_OUTOFMEMORY);
}
}
/* ============================================================================
CTemplate::CBuffer::Append
Appends to a CBuffer
*/
void
CTemplate::CBuffer::Append
(
const CByteRange& br, // byte range to append
BOOL fLocal, // append local?
UINT idSequence, // segment sequence id
CFileMap* pfilemap,
BOOL fLocalString // append local as a string? (length-prefixed, null-terminated)
)
{
// calc bytes required to store byte range; allow for length prefix and null if a local string
ULONG cbRequired = (ULONG)(br.m_cb + (fLocalString ? sizeof(br.m_cb) + 1 : 0));
// If caller passed a non-local zero-length byte range, no-op and return;
// allows callers to ignore byte range size
// NOTE we store empty local byte ranges - required by token list
if(!fLocal && br.m_cb == 0)
return;
if(fLocal)
{
if((m_cbData - m_cbDataUsed) < cbRequired)
{
// Reallocate space for storing local data - we grab twice what we had before
// or twice current requirement, whichever is more
m_cbData = 2 * (m_cbData > cbRequired ? m_cbData : cbRequired);
if(!(m_pbData = (BYTE*) CTemplate::LargeReAlloc(m_pbData, m_cbData)))
THROW(E_OUTOFMEMORY);
}
// if appending as a local string, copy length-prefix to buffer
if(fLocalString)
{
memcpy(m_pbData + m_cbDataUsed, &(br.m_cb), sizeof(br.m_cb));
m_cbDataUsed += sizeof(br.m_cb);
}
// copy data to buffer
memcpy(m_pbData + m_cbDataUsed, br.m_pb, br.m_cb);
m_cbDataUsed += br.m_cb;
// if appending as a local string, copy null terminator to buffer
if(fLocalString)
*(m_pbData + m_cbDataUsed++) = NULL;
}
if(m_cItems >= m_cSlots)
{
// Reallocate space for storing byte range items - we grab twice what we had before
m_cSlots *= 2;
if(!(m_pItems = (CByteRange*) CTemplate::SmallReAlloc(m_pItems, m_cSlots * sizeof(*m_pItems))))
THROW(E_OUTOFMEMORY);
}
// Set the (new) last item to this byte range
SetItem(m_cItems++, br, fLocal, idSequence, pfilemap, fLocalString);
}
/* ============================================================================
CTemplate::CBuffer::GetItem
Gets an item from a CBuffer, as a byte range
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::CBuffer::GetItem
(
UINT i, // index of item
CByteRange& br // byte range containing returned item (out-parameter)
)
{
Assert(i < m_cItems);
// for local data, ptr is offset only; must add it to base ptr
br.m_pb = m_pItems[i].m_pb + (m_pItems[i].m_fLocal ? (DWORD_PTR) m_pbData : 0);
br.m_cb = m_pItems[i].m_cb;
br.m_fLocal = m_pItems[i].m_fLocal;
br.m_idSequence = m_pItems[i].m_idSequence;
br.m_pfilemap = m_pItems[i].m_pfilemap;
}
/* ============================================================================
CTemplate::CBuffer::SetItem
Sets a CBuffer item to a new value
Returns
Nothing
Side effects
Throws error on non-existent item index
*/
void
CTemplate::CBuffer::SetItem
(
UINT i,
const CByteRange& br, // byte range to set item to
BOOL fLocal, // is item local in buffer?
UINT idSequence, // segment sequence id
CFileMap * pfilemap, // file where segment came from
BOOL fLocalString // append local as a string? (length-prefixed, null-terminated)
)
{
// If buffer item i does not exist, fail
if(i >= m_cSlots)
THROW(E_FAIL);
// for local data, store ptr as offset only - avoids fixup after realloc
// NOTE offset == data used offset - length of data - null terminator (if local string)
m_pItems[i].m_pb = (fLocal
? (BYTE*)(m_cbDataUsed - br.m_cb -
(fLocalString
? sizeof(BYTE)
: 0
))
: (BYTE*)br.m_pb);
m_pItems[i].m_cb = br.m_cb;
m_pItems[i].m_fLocal = fLocal;
m_pItems[i].m_idSequence = idSequence;
m_pItems[i].m_pfilemap = pfilemap;
}
/* ============================================================================
CTemplate::CBuffer::PszLocal
Gets i-th locally-buffered string within the buffer.
Returns:
Ptr to locally-buffered string; NULL if not found
Side effects:
None
*/
LPSTR
CTemplate::CBuffer::PszLocal
(
UINT i // index of item to retrieve
)
{
CByteRange br;
GetItem(i, br);
if(!br.m_fLocal)
return NULL;
return (LPSTR) br.m_pb;
}
/* ****************************************************************************
CTemplate::CScriptStore member functions
*/
/* ============================================================================
CTemplate::CScriptStore::~CScriptStore
Destructor - frees memory
Returns:
nothing
Side effects:
none
*/
CTemplate::CScriptStore::~CScriptStore()
{
UINT i;
for(i = 0; i < m_cSegmentBuffers; i++)
delete m_ppbufSegments[i];
if(m_ppbufSegments != NULL)
CTemplate::SmallFree(m_ppbufSegments);
if(m_rgProgLangId != NULL)
CTemplate::SmallFree(m_rgProgLangId);
}
/* ============================================================================
CTemplate::CScriptStore::Init
Inits the script store
Returns:
nothing
Side effects:
allocates memory
*/
HRESULT
CTemplate::CScriptStore::Init
(
LPCSTR szDefaultScriptLanguage,
CLSID *pCLSIDDefaultEngine
)
{
HRESULT hr = S_OK;
UINT i;
CByteRange brDefaultScriptLanguage;
// Check for NULL pointers - can happen if Application has invalid default lang
if (szDefaultScriptLanguage == NULL || pCLSIDDefaultEngine == NULL)
return TYPE_E_ELEMENTNOTFOUND;
/* init segment buffers count based on:
- two for default engine (one primary, one tagged)
- one each for other engines (tagged only)
*/
m_cSegmentBuffers = C_SCRIPTENGINESDEFAULT + 1;
// init segments buffers
if(NULL == (m_ppbufSegments = (CBuffer**) CTemplate::SmallMalloc(m_cSegmentBuffers * sizeof(CBuffer*))))
{
hr = E_OUTOFMEMORY;
goto LExit;
}
for(i = 0; i < m_cSegmentBuffers; i++)
{
if(NULL == (m_ppbufSegments[i] = new CBuffer))
{
hr = E_OUTOFMEMORY;
goto LExit;
}
m_ppbufSegments[i]->Init((C_SCRIPTSEGMENTSDEFAULT), 0);
}
// Append default engine to script store
brDefaultScriptLanguage.m_cb = strlen(szDefaultScriptLanguage);
brDefaultScriptLanguage.m_pb = (unsigned char *)szDefaultScriptLanguage;
hr = AppendEngine(brDefaultScriptLanguage, pCLSIDDefaultEngine, /* idSequence */ 0);
LExit:
return hr;
}
/* ============================================================================
CTemplate::CScriptStore::AppendEngine
Appends a script engine to the script store
Returns:
HRESULT
Side effects:
None
*/
HRESULT
CTemplate::CScriptStore::AppendEngine
(
CByteRange& brEngine, // engine name
PROGLANG_ID* pProgLangId, // ptr to prog lang id - pass NULL to have this function get proglangid from registry
UINT idSequence // segment sequence id
)
{
HRESULT hr = S_OK;
USHORT cEngines; // count of engines
TRY
// if no engines yet, init engine names buffer
if(CountPreliminaryEngines() == 0)
m_bufEngineNames.Init(C_SCRIPTENGINESDEFAULT, 0);
// Append engine name to buffer
m_bufEngineNames.Append(brEngine, FALSE, idSequence, NULL);
CATCH(hrException)
hr = hrException;
goto LExit;
END_TRY
Assert(CountPreliminaryEngines() >= 1);
// malloc or realloc prog lang ids array
if((cEngines = CountPreliminaryEngines()) == 1)
m_rgProgLangId = (PROGLANG_ID*) CTemplate::SmallMalloc(cEngines * sizeof(PROGLANG_ID));
else
m_rgProgLangId = (PROGLANG_ID*) CTemplate::SmallReAlloc(m_rgProgLangId, cEngines * sizeof(PROGLANG_ID));
if(NULL == m_rgProgLangId)
{
hr = E_OUTOFMEMORY;
goto LExit;
}
if(NULL == pProgLangId)
// caller passed null progid ptr - get prog id from registry
hr = GetProgLangId(brEngine, &(m_rgProgLangId[cEngines - 1]));
else
// caller passed non-null progid ptr - set prog id from it
m_rgProgLangId[cEngines - 1] = *pProgLangId;
LExit:
return hr;
}
/* ============================================================================
CTemplate::CScriptStore::IdEngineFromBr
Determines the id of a script engine from its engine name
Returns:
id of script engine whose name is passed in
Side effects:
appends a new script engine name to engine names buffer
*/
USHORT
CTemplate::CScriptStore::IdEngineFromBr
(
CByteRange& brEngine, // engine name
UINT idSequence // segment sequence id
)
{
Assert(!brEngine.IsNull()); // NOTE we trap/error null engine name earlier
USHORT cKnownEngines = CountPreliminaryEngines();
// search existing names for a match; return id if found
for(USHORT i = 0; i < cKnownEngines; i++)
{
Assert(m_bufEngineNames[i]);
Assert(m_bufEngineNames[i]->m_pb);
if(FByteRangesAreEqual(*(m_bufEngineNames[i]), brEngine))
return i;
}
// if not found by name try to find by engine id
// (some engines with different names share the same id, like J[ava]Script)
if (cKnownEngines > 0)
{
PROGLANG_ID ProgLandId;
// we will get the prog lang id again inside AppendEngine() but
// because it's cached and this only happens when > 1 engine, it's alright
if (SUCCEEDED(GetProgLangId(brEngine, &ProgLandId)))
{
for(i = 0; i < cKnownEngines; i++)
{
// If matches don't append -- just return the index
if (m_rgProgLangId[i] == ProgLandId)
return i;
}
}
}
/* if we did not find engine among those already buffered
- append engine to script store
- realloc segment buffers array if necessary
- return index of last engine (the one we just appended)
*/
// append engine to script store
HRESULT hr = AppendEngine(brEngine, NULL, idSequence);
if(hr == TYPE_E_ELEMENTNOTFOUND)
// if prog lang not found, throw bad prog lang error id
THROW(IDE_TEMPLATE_BAD_PROGLANG);
else if(FAILED(hr))
// other failure: re-throw hresult
THROW(hr);
// realloc segment buffers array if necessary
if(CountPreliminaryEngines() > (m_cSegmentBuffers - 1))
{
// increment count of segment buffers
m_cSegmentBuffers++;
Assert(CountPreliminaryEngines() == m_cSegmentBuffers - 1);
// realloc array of ptrs
if(NULL == (m_ppbufSegments = (CBuffer**) CTemplate::SmallReAlloc(m_ppbufSegments, m_cSegmentBuffers * sizeof(CBuffer*))))
THROW(E_OUTOFMEMORY);
// allocate the new buffer
if(NULL == (m_ppbufSegments[m_cSegmentBuffers - 1] = new CBuffer))
THROW(E_OUTOFMEMORY);
// init the new buffer
m_ppbufSegments[m_cSegmentBuffers - 1]->Init(C_SCRIPTSEGMENTSDEFAULT, 0);
}
// return index of last engine (the one we just appended)
return (CountPreliminaryEngines() - 1);
}
/* ============================================================================
CTemplate::CScriptStore::AppendScript
Appends a script/engine pair to the store.
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::CScriptStore::AppendScript
(
CByteRange& brScript, // script text
CByteRange& brEngine, // script engine name
BOOLB fPrimary, // primary or tagged script?
UINT idSequence, // segment sequence id
CFileMap* pfilemapCurrent
)
{
USHORT iBuffer; // buffer id
Assert(fPrimary || !brEngine.IsNull()); // NOTE we trap/error null engine name earlier
Assert(m_bufEngineNames[0]); // page's primary engine must be known by this point
Assert(m_bufEngineNames[0]->m_pb);
if(fPrimary)
// if primary script (not tagged), buffer id is 0
iBuffer = 0;
else if((!fPrimary) && FByteRangesAreEqual(brEngine, /* bug 1008: primary script engine name */ *(m_bufEngineNames[0])))
// if tagged script and engine is primary, buffer id is 1
iBuffer = 1;
else
// else, buffer id is engine id plus 1
iBuffer = IdEngineFromBr(brEngine, idSequence) + 1;
// append script segment to iBuffer-th segments buffer
m_ppbufSegments[iBuffer]->Append(brScript, FALSE, idSequence, pfilemapCurrent);
}
/* ****************************************************************************
CTemplate::CObjectInfoStore member functions
*/
/* ============================================================================
CTemplate::CObjectInfoStore::~CObjectInfoStore
*/
CTemplate::CObjectInfoStore::~CObjectInfoStore
(
)
{
if(m_pObjectInfos)
CTemplate::SmallFree(m_pObjectInfos);
}
/* ============================================================================
CTemplate::CObjectInfoStore::Init
Inits the object-info store
*/
void
CTemplate::CObjectInfoStore::Init()
{
m_bufObjectNames.Init(C_OBJECTINFOS_DEFAULT, 0);
// init object-infos array
if(NULL == (m_pObjectInfos = (CObjectInfo*) CTemplate::SmallMalloc(m_bufObjectNames.CountSlots() * sizeof(CObjectInfo))))
THROW(E_OUTOFMEMORY);
}
/* ============================================================================
CTemplate::CObjectInfoStore::AppendObject
Appends an object-info to the object-info store
*/
void
CTemplate::CObjectInfoStore::AppendObject
(
CByteRange& brObjectName,
CLSID clsid,
CompScope scope,
CompModel model,
UINT idSequence
)
{
USHORT iObject = m_bufObjectNames.Count();
if(iObject >= m_bufObjectNames.CountSlots())
{
// Reallocate space for storing object-infos - we grab twice what we had before
// NOTE we keep no object count in CObjectInfoStore, but instead use count in object names buffer
(m_pObjectInfos = (CObjectInfo*)CTemplate::SmallReAlloc(m_pObjectInfos,
2 * m_bufObjectNames.CountSlots() * sizeof(CObjectInfo)));
if (m_pObjectInfos == NULL)
THROW(E_OUTOFMEMORY);
}
m_pObjectInfos[iObject].m_clsid = clsid;
m_pObjectInfos[iObject].m_scope = scope;
m_pObjectInfos[iObject].m_model = model;
m_bufObjectNames.Append(brObjectName, FALSE, idSequence, NULL);
}
/* ****************************************************************************
CTemplate::CWorkStore member functions
*/
/* ============================================================================
CTemplate::CWorkStore::CWorkStore
Constructor
Returns:
Nothing
Side effects:
None
*/
CTemplate::CWorkStore::CWorkStore
(
)
:
m_idCurSequence(0),
m_fPageCommandsExecuted(FALSE),
m_fPageCommandsAllowed(TRUE),
m_szWriteBlockOpen(g_szWriteBlockOpen),
m_szWriteBlockClose(g_szWriteBlockClose),
m_szWriteOpen(g_szWriteOpen),
m_szWriteClose(g_szWriteClose)
{ }
/* ============================================================================
CTemplate::CWorkStore::~CWorkStore
Destructor
Returns:
Nothing
Side effects:
None
*/
CTemplate::CWorkStore::~CWorkStore
(
)
{
/* if language element ptrs are anything but their constant defaults or null,
they must have been allocated during compilation - free them now
*/
if(m_szWriteBlockOpen != g_szWriteBlockOpen && m_szWriteBlockOpen != NULL)
CTemplate::SmallFree(m_szWriteBlockOpen);
if(m_szWriteBlockClose != g_szWriteBlockClose && m_szWriteBlockClose != NULL)
CTemplate::SmallFree(m_szWriteBlockClose);
if(m_szWriteOpen != g_szWriteOpen && m_szWriteOpen != NULL)
CTemplate::SmallFree(m_szWriteOpen);
if(m_szWriteClose != g_szWriteClose && m_szWriteClose != NULL)
CTemplate::SmallFree(m_szWriteClose);
}
/* ============================================================================
CTemplate::CWorkStore::Init
Inits the workstore
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::CWorkStore::Init
(
)
{
/*
NOTE init we the scriptstore separately from rest of workstore
because try-catch in CTemplate::Init() apparently doesn't work to detect
bogus script engine name; we need to get an hr back instead.
m_ScriptStore.Init(brDefaultEngine);
*/
m_ObjectInfoStore.Init();
m_bufHTMLSegments.Init(C_HTMLSEGMENTSDEFAULT, 0);
}
/* ============================================================================
CTemplate::CWorkStore::CRequiredScriptEngines
Returns the count of script engines in the script store that are required
to run the template.
NOTE this function is part of the fix for bug 933
Returns:
Count of non-empty script engines
Side effects:
None
*/
USHORT
CTemplate::CWorkStore::CRequiredScriptEngines
(
BOOL fGlobalAsa // bug 1394: is template global.asa?
)
{
USHORT cPreliminaryEngines = m_ScriptStore.CountPreliminaryEngines();
USHORT cRequiredEngines = 0;
for(USHORT i = 0; i < cPreliminaryEngines; i++)
{
if(FScriptEngineRequired(i, fGlobalAsa))
cRequiredEngines++;
}
return cRequiredEngines;
}
/* ============================================================================
CTemplate::CWorkStore::FScriptEngineRequired
Is a given preliminary script engine required to run the template?
NOTE this function is part of the fix for bug 933
Returns:
TRUE or FALSE
Side effects:
None
*/
BOOLB
CTemplate::CWorkStore::FScriptEngineRequired
(
USHORT idEnginePrelim,
BOOL fGlobalAsa // bug 1394: is template global.asa?
)
{
if(idEnginePrelim == 0)
return ( // primary engine (id 0) required if
(m_ScriptStore.m_ppbufSegments[0]->Count() > 0) // ... script buffer 0 has segments
|| (m_ScriptStore.m_ppbufSegments[1]->Count() > 0) // ... or script buffer 1 has segments
|| ((m_bufHTMLSegments.Count() > 0) && !fGlobalAsa) // ... or html buffer has segments and (bug 1394) template is not global.asa
);
// non-primary engine required if script buffer id+1 has segments
return (m_ScriptStore.m_ppbufSegments[idEnginePrelim + 1]->Count() > 0);
}
/* ****************************************************************************
CTemplate::CFileMap member functions
*/
/* ============================================================================
CTemplate::CFileMap::CFileMap
Constructor
Returns
Nothing
Side effects
None
*/
CTemplate::CFileMap::CFileMap()
:
m_szPathInfo(NULL),
m_szPathTranslated(NULL),
m_pfilemapSibling(NULL),
m_pfilemapChild(NULL),
m_hFile(NULL),
m_hMap(NULL),
m_pbStartOfFile(NULL),
m_pIncFile(NULL),
m_pSecurityDescriptor(NULL),
m_dwSecDescSize(0),
m_cChars(0),
m_pDME(NULL)
{
m_ftLastWriteTime.dwLowDateTime = 0;
m_ftLastWriteTime.dwHighDateTime = 0;
}
/* ============================================================================
CTemplate::CFileMap::~CFileMap
Destructor
Returns
Nothing
Side effects
None
*/
CTemplate::CFileMap::~CFileMap()
{
if (m_pDME)
{
m_pDME->Release();
m_pDME = NULL;
}
if(m_szPathInfo != NULL)
CTemplate::SmallFree(m_szPathInfo);
if(m_szPathTranslated != NULL)
CTemplate::SmallFree(m_szPathTranslated);
if(m_pSecurityDescriptor != NULL)
CTemplate::SmallFree(m_pSecurityDescriptor);
if (m_pIncFile != NULL)
m_pIncFile->Release();
}
/* ============================================================================
CTemplate::CFileMap::MapFile
Memory-maps a file.
Returns
Nothing
Side effects
Throws **overloaded** exception on error: exception code can sometimes be
an error message id, sometimes a true exception. Caller must handle.
*/
void
CTemplate::CFileMap::MapFile
(
LPCTSTR szFileSpec, // file spec for this file
LPCTSTR szApplnPath, // application path (in case its global.asa)
CFileMap* pfilemapParent, // ptr to filemap of parent file
BOOL fVirtual, // is file spec virtual or relative?
CHitObj* pHitObj, // ptr to template's hit object
BOOL fGlobalAsa // is this file the global.asa file?
)
{
BOOL fMustNormalize = TRUE;
BOOL fImpersonatedUser = FALSE;
HANDLE hImpersonationToken = NULL;
HANDLE hCurrentImpersonationToken = NULL;
Assert((pfilemapParent != NULL) || (pHitObj->PIReq() != NULL) || fGlobalAsa);
/* three possible cases:
1) we are processing global.asa file
2) we are processing the "main" .asp file
3) we are processing an include file
*/
if(fGlobalAsa)
{
// case 1) we are processing global.asa file
Assert(pHitObj->GlobalAspPath());
DWORD cchPathTranslated = _tcslen(pHitObj->GlobalAspPath());
m_szPathTranslated = (TCHAR *)CTemplate::SmallMalloc((cchPathTranslated+1)*sizeof(TCHAR));
if (!m_szPathTranslated)
THROW(E_OUTOFMEMORY);
_tcscpy(m_szPathTranslated, pHitObj->GlobalAspPath());
DWORD cchPathInfo = _tcslen(szApplnPath) + 11; // "/global.asa"
m_szPathInfo = (TCHAR *)CTemplate::SmallMalloc((cchPathInfo+1) * sizeof(TCHAR));
if (!m_szPathInfo)
THROW(E_OUTOFMEMORY);
_tcscpy(strcpyEx(m_szPathInfo, szApplnPath), _T("/global.asa"));
// no need to normalize in this case, since global.asa path is already normalized
Assert(IsNormalized((const TCHAR*)m_szPathTranslated));
fMustNormalize = FALSE;
m_fHasVirtPath = TRUE;
}
else if(pfilemapParent == NULL)
{
// case 2) we are processing the "main" .asp file: get path-info and path-tran from ecb
Assert(pHitObj->PIReq());
TCHAR *szVirtPath = pHitObj->PSzCurrTemplateVirtPath();
TCHAR *szPhysPath = pHitObj->PSzCurrTemplatePhysPath();
m_szPathInfo = static_cast<LPTSTR>(CTemplate::SmallMalloc((_tcslen(szVirtPath) + 1)*sizeof(TCHAR)));
m_szPathTranslated = static_cast<LPTSTR>(CTemplate::SmallMalloc((_tcslen(szPhysPath) + 1)*sizeof(TCHAR)));
if (!m_szPathInfo || !m_szPathTranslated)
THROW(E_OUTOFMEMORY);
_tcscpy(m_szPathInfo, szVirtPath);
_tcscpy(m_szPathTranslated, szPhysPath);
// no need to normalize in this case, since ecb's path-tran is already normalized
Assert(IsNormalized((const TCHAR*)m_szPathTranslated));
fMustNormalize = FALSE;
m_fHasVirtPath = TRUE;
}
else
{
/* case 3) we are processing an include file: resolve filespec into path-info and path-tran
based on whether file was included with VIRTUAL tag or FILE tag
*/
Assert(szFileSpec);
// in this case, we don't know path lengths up front so we alloc the max and realloc below
m_szPathInfo = static_cast<LPTSTR> (CTemplate::SmallMalloc((MAX_PATH + 1)*sizeof(TCHAR)));
m_szPathTranslated = static_cast<LPTSTR> (CTemplate::SmallMalloc((MAX_PATH + 1)*sizeof(TCHAR)));
if (!m_szPathInfo || !m_szPathTranslated)
THROW(E_OUTOFMEMORY);
STACK_BUFFER(tempPathT, MAX_PATH );
if (!tempPathT.Resize((_tcslen(szFileSpec) + 1)*sizeof(TCHAR))) {
THROW(E_OUTOFMEMORY);
}
LPTSTR szPathTranslatedT = (TCHAR *)tempPathT.QueryPtr(); // temp path-tran
if(fVirtual) {
DWORD dwSzLength = tempPathT.QuerySize(); // length of path string buffer
if (_tcslen(szFileSpec) > MAX_PATH)
THROW(E_FAIL);
// VIRTUAL: path-info is simply virtual filespec
_tcscpy(m_szPathInfo, szFileSpec);
// VIRTUAL: path-tran is translation of path-info
_tcscpy(szPathTranslatedT, m_szPathInfo);
if (!pHitObj->PIReq()->MapUrlToPath(szPathTranslatedT, &dwSzLength))
THROW(E_FAIL);
// Check the translated path for a UNC specified path
if ((dwSzLength >= (2*sizeof(TCHAR)))
&& (szPathTranslatedT[0] == _T('\\'))
&& (szPathTranslatedT[1] == _T('\\'))) {
// if UNC, then ask WAM for the impersonation token for
// this UNC VRoot. Silently fail.
if (pHitObj->PIReq()->ServerSupportFunction(
HSE_REQ_GET_VIRTUAL_PATH_TOKEN,
(void *)szFileSpec,
(DWORD *) &hImpersonationToken,
NULL))
{
// set the impersonation token and note that we did so
AspDoRevertHack(&hCurrentImpersonationToken);
fImpersonatedUser = ImpersonateLoggedOnUser(hImpersonationToken)
? TRUE
: FALSE;
if (!fImpersonatedUser) {
AspUndoRevertHack(&hCurrentImpersonationToken);
}
}
}
m_fHasVirtPath = TRUE;
}
else
{
TCHAR szParentDir[MAX_PATH], *szT;
_tcscpy(szParentDir, pfilemapParent->m_szPathInfo);
if ((szT = _tcsrchr(szParentDir, _T('/'))) != NULL)
*szT = _T('\0');
// If we don't allow parent paths, we can save lots of time (Always have a valid virtual path)
if (!pHitObj->QueryAppConfig()->fEnableParentPaths())
{
int strlen_szParentDir = (int)(szT - szParentDir);
if ((strlen_szParentDir + 1 + _tcslen(szFileSpec)) > MAX_PATH)
THROW(E_FAIL);
strcpyEx(strcpyEx(strcpyEx(m_szPathInfo, szParentDir), _T("/")), szFileSpec);
m_fHasVirtPath = TRUE;
}
else
{
// NOTE: If we must translate ".." paths, there is no need to verify them (by remapping)
// because: If the file does not exist, that case will show up when the file is mapped
// If we ".." ourselves out of the vroot space, (out of the app or into another app)
// DotPathToPath will detect this.
//
if (DotPathToPath(m_szPathInfo, szFileSpec, szParentDir))
m_fHasVirtPath = TRUE;
else
{
GetPathFromParentAndFilespec(pfilemapParent->m_szPathTranslated, szFileSpec, &m_szPathInfo);
m_fHasVirtPath = FALSE;
}
}
GetPathFromParentAndFilespec(pfilemapParent->m_szPathTranslated, szFileSpec, &szPathTranslatedT);
}
// bug 1214: get canonical path-tran, without . and ..
// CONSIDER check for . or .. in name before calling GetFullPathName? UNCs? what else?
GetFullPathName(
szPathTranslatedT, // LPCSTR lpFileName, // address of name of file to find path for
MAX_PATH + 1, // DWORD nBufferLength, // size, in characters, of path buffer
m_szPathTranslated, // LPSTR lpBuffer, // address of path buffer
NULL // LPSTR *lpFilePart // address of filename in path
);
// realloc path strings to only use required memory (see note above)
m_szPathInfo = static_cast<LPTSTR> (CTemplate::SmallReAlloc(m_szPathInfo, (_tcslen(m_szPathInfo) + 1)*sizeof(TCHAR)));
m_szPathTranslated = static_cast<LPTSTR> (CTemplate::SmallReAlloc(m_szPathTranslated, (_tcslen(m_szPathTranslated) + 1)*sizeof(TCHAR)));
if (!m_szPathInfo || !m_szPathTranslated)
{
if (fImpersonatedUser)
{
AspUndoRevertHack(&hCurrentImpersonationToken);
}
if (hImpersonationToken)
{
CloseHandle(hImpersonationToken);
}
THROW(E_OUTOFMEMORY);
}
}
// if required, normalize path-tran so that
// a) cyclic include check can ignore case; b) inc-file cache lookups will work
if(fMustNormalize)
Normalize(m_szPathTranslated);
Assert(IsNormalized(m_szPathTranslated));
// Bug 99071: Attempt to open the file **BEFORE** we add it to the tree of file
// dependencies. Otherwise if it fails to open, we will have
// dangling references. Since FCyclicInclude depends on us adding
// to the tree, if it is cyclic, we need to unmap then. Since that
// is a very uncommon error case, the extra overhead is probably OK
//
// RemapFile will throw if it fails. If the exception is that the source file is empty
// and we are trying to process an include file, we will handle the exception here.
// in all other cases, rethrow the exception. We do this so that an empty include file
// will be harmless, but an empty primary file will fail.
TRY
RemapFile();
CATCH(hrException)
if (hrException != E_SOURCE_FILE_IS_EMPTY || pfilemapParent == NULL)
{
if (fImpersonatedUser)
{
AspUndoRevertHack(&hCurrentImpersonationToken);
}
if (hImpersonationToken)
{
CloseHandle(hImpersonationToken);
}
THROW(hrException);
}
END_TRY
if (fImpersonatedUser)
{
AspUndoRevertHack(&hCurrentImpersonationToken);
}
if (hImpersonationToken)
{
CloseHandle(hImpersonationToken);
}
// Create the tree structure for this file
if (pfilemapParent != NULL)
{
// See if this file is already included once on this level. (Don't show duplicates in the
// debugger tree view)
//
BOOL fDuplicateExists = FALSE;
CFileMap *pFilemap = pfilemapParent->m_pfilemapChild;
while (pFilemap != NULL && !fDuplicateExists)
{
if (_tcscmp(pFilemap->m_szPathTranslated, m_szPathTranslated) == 0)
fDuplicateExists = TRUE;
pFilemap = pFilemap->m_fHasSibling? pFilemap->m_pfilemapSibling : NULL;
}
// If the include file is #include'd more than once, don't add it as a sibling.
// Rather orphan the pfilemap and just set the parent pointer.
//
if (!fDuplicateExists)
{
if (pfilemapParent->m_pfilemapChild == NULL)
pfilemapParent->m_pfilemapChild = this;
else
pfilemapParent->m_pfilemapChild->AddSibling(this);
}
}
// in both of the above code paths, we are always added as the LAST child, (or we are an orphan node)
// so it is safe to set the parent without calling SetParent()
m_fHasSibling = FALSE; // Paranoia
m_pfilemapParent = pfilemapParent;
// hurl if this file is being included by itself (perhaps indirectly)
if(FCyclicInclude(m_szPathTranslated))
{
UnmapFile();
THROW(IDE_TEMPLATE_CYCLIC_INCLUDE);
}
}
/* ============================================================================
CTemplate::CFileMap::RemapFile
map a file that was previously mapped.
Returns
Nothing
Side effects
Throws **overloaded** exception on error: exception code can sometimes be
an error message id, sometimes a true exception. Caller must handle.
Does not decrypt EASPs on remapping. Caller must decrypt if required. This
function is called by the debugger, and the debugger does not allow access
to decrypted files, so decryption is a waste of time.
*/
void
CTemplate::CFileMap::RemapFile
(
)
{
WIN32_FILE_ATTRIBUTE_DATA fad; // win32 file attributes data structure
if (FIsMapped())
return;
if(INVALID_HANDLE_VALUE == (m_hFile =
CreateFile(
m_szPathTranslated, // file name
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
)))
{
DWORD dwLastError = GetLastError();
if(dwLastError == ERROR_ACCESS_DENIED)
{
// typically, we end up here if the user has no permissions on the file
// bug 1007: however, we also end up here if the user gave us a directory name, instead of a file name
if(FAILED(AspGetFileAttributes(m_szPathTranslated, &fad)))
{
// bug 1495: file in a secured directory will end up here - we need to re-GetLastError to see if access is denied
dwLastError = GetLastError();
if(dwLastError == ERROR_ACCESS_DENIED)
{
THROW(E_USER_LACKS_PERMISSIONS);
}
// GetFileAttributes call failed; don't know why
THROW(E_FAIL);
}
else if(FILE_ATTRIBUTE_DIRECTORY & fad.dwFileAttributes)
{
// bug 1007: the user gave us a directory name
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "Failed to open file %S because it is a directory.\n", m_szPathTranslated));
#else
DBGPRINTF((DBG_CONTEXT, "Failed to open file %s because it is a directory.\n", m_szPathTranslated));
#endif
THROW(E_COULDNT_OPEN_SOURCE_FILE);
}
else
{
THROW(E_USER_LACKS_PERMISSIONS);
}
}
else
{
#if DBG
char szError[128];
if (!FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM,
NULL,
dwLastError,
0L, // lang ID - defaults to LANG_NEUTRAL
szError,
sizeof szError,
NULL) )
{
sprintf(szError, "%d", dwLastError);
}
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "Failed to open file %S\n", m_szPathTranslated));
#else
DBGPRINTF((DBG_CONTEXT, "Failed to open file %s\n", m_szPathTranslated));
#endif
DBGPRINTF((DBG_CONTEXT, " The error returned was: %s\n", szError));
#endif
THROW(E_COULDNT_OPEN_SOURCE_FILE);
}
}
// Get the file's last access time. Only do this for NT
if (Glob(fWinNT))
{
if (SUCCEEDED(AspGetFileAttributes(m_szPathTranslated, &fad)))
{
m_ftLastWriteTime.dwLowDateTime = fad.ftLastWriteTime.dwLowDateTime;
m_ftLastWriteTime.dwHighDateTime = fad.ftLastWriteTime.dwHighDateTime;
}
}
// get file's security descriptor
if(!GetSecurityDescriptor())
THROW(E_COULDNT_OPEN_SOURCE_FILE);
// map the file
if(NULL == (m_hMap =
CreateFileMapping(
m_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 (SUCCEEDED(AspGetFileAttributes(m_szPathTranslated, &fad)))
{
if(fad.nFileSizeHigh == 0 && fad.nFileSizeLow == 0)
{
#if UNICODE
DBGPRINTF((DBG_CONTEXT, "Empty source file %S\n", m_szPathTranslated));
#else
DBGPRINTF((DBG_CONTEXT, "Empty source file %s\n", m_szPathTranslated));
#endif
THROW(E_SOURCE_FILE_IS_EMPTY);
}
}
else
{
THROW(E_COULDNT_OPEN_SOURCE_FILE);
}
}
// set file's start-of-file ptr
if(NULL == (m_pbStartOfFile =
(PBYTE) MapViewOfFile(
m_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
)))
THROW(E_COULDNT_OPEN_SOURCE_FILE);
}
/* ============================================================================
CTemplate::CFileMap::SetParent
Set the parent for this filemap
*/
void
CTemplate::CFileMap::SetParent
(
CFileMap* pfilemapParent
)
{
CFileMap *pfilemap = this;
while (pfilemap->m_fHasSibling)
pfilemap = pfilemap->m_pfilemapSibling;
pfilemap->m_pfilemapParent = pfilemapParent;
}
/* ============================================================================
CTemplate::CFileMap::GetParent
Get the parent for this filemap
*/
CTemplate::CFileMap*
CTemplate::CFileMap::GetParent
(
)
{
register CFileMap *pfilemap = this;
while (pfilemap->m_fHasSibling)
pfilemap = pfilemap->m_pfilemapSibling;
return pfilemap->m_pfilemapParent;
}
/* ============================================================================
CTemplate::CFileMap::AddSibling
Add a new node as a sibling of this
*/
void
CTemplate::CFileMap::AddSibling
(
register CFileMap* pfilemapSibling
)
{
register CFileMap *pfilemap = this;
while (pfilemap->m_fHasSibling)
pfilemap = pfilemap->m_pfilemapSibling;
pfilemapSibling->m_fHasSibling = FALSE;
pfilemapSibling->m_pfilemapParent = pfilemap->m_pfilemapParent;
pfilemap->m_fHasSibling = TRUE;
pfilemap->m_pfilemapSibling = pfilemapSibling;
}
/* ============================================================================
CTemplate::CFileMap::FCyclicInclude
Is a file among this filemap's ancestors? (i.e. does it occur anywhere
in the filemap's parent chain?)
Returns
TRUE or FALSE
Side effects
None
*/
BOOL
CTemplate::CFileMap::FCyclicInclude
(
LPCTSTR szPathTranslated
)
{
CFileMap *pfilemapParent = GetParent();
if(pfilemapParent == NULL)
return FALSE;
// NOTE we ignore case because path-tran was normalized
if(_tcscmp(szPathTranslated, pfilemapParent->m_szPathTranslated) == 0)
return TRUE;
return pfilemapParent->FCyclicInclude(szPathTranslated);
}
/* ============================================================================
CTemplate::CFileMap::GetSecurityDescriptor
Gets a file's security descriptor
Returns
TRUE if we got security descriptor, else FALSE
Side effects
allocates memory
*/
BOOL
CTemplate::CFileMap::GetSecurityDescriptor
(
)
// ACLs: the following code should in future be shared with IIS (see creatfil.cxx in IIS project)
{
BOOL fRet = TRUE; // return value
BOOL fGotSecurityDescriptor; // did we get a security descriptor?
DWORD dwSecDescSizeNeeded = 0; // required size of security descriptor
DWORD dwLastError; // last error code
const SECURITY_INFORMATION si = OWNER_SECURITY_INFORMATION // security info struct
| GROUP_SECURITY_INFORMATION
| DACL_SECURITY_INFORMATION;
// we dont support security on win95
if (!Glob(fWinNT))
return(TRUE);
// get required buffer size before malloc
// NOTE this costs us an extra system call, but means the cached template will often use less memory
// we must do this up front by passing 0 buffer size because when the call succeeds it returns
// dwSecDescSizeNeeded == 0 (i.e. we can't realloc to shrink after successful call)
GetKernelObjectSecurity(
m_hFile, // handle of object to query
si, // requested information
NULL, // address of security descriptor
0, // size of buffer for security descriptor
&dwSecDescSizeNeeded // address of required size of buffer
);
if((dwLastError = GetLastError()) != ERROR_INSUFFICIENT_BUFFER)
{
// pretend everything's fine -- just NULL security descriptor
if(m_pSecurityDescriptor != NULL)
CTemplate::SmallFree(m_pSecurityDescriptor);
m_pSecurityDescriptor = NULL;
m_dwSecDescSize = 0;
if (dwLastError == ERROR_NOT_SUPPORTED)
return TRUE;
else
return FALSE;
}
// set member buffer size to just enough chunks to accommodate security descriptor size needed
m_dwSecDescSize = ((dwSecDescSizeNeeded + SECURITY_DESC_GRANULARITY - 1) / SECURITY_DESC_GRANULARITY)
* SECURITY_DESC_GRANULARITY;
// allocate memory for security descriptor
// (Note: security descriptor may already be allocated if this is a remap)
if (m_pSecurityDescriptor == NULL)
if(NULL == (m_pSecurityDescriptor = (PSECURITY_DESCRIPTOR) CTemplate::SmallMalloc(m_dwSecDescSize)))
THROW(E_OUTOFMEMORY);
// try to get security descriptor until we succeed, or until we fail for some reason other than buffer-too-small
while(TRUE)
{
// attempt to get security descriptor
fGotSecurityDescriptor = GetKernelObjectSecurity(
m_hFile, // handle of object to query
si, // requested information
m_pSecurityDescriptor, // address of security descriptor
m_dwSecDescSize, // size of buffer for security descriptor
&dwSecDescSizeNeeded // address of required size of buffer
);
// get last error immediately after call
dwLastError = fGotSecurityDescriptor
? 0 // we got a security descriptor: set last error to 0
: GetLastError(); // we didn't get a security descriptor: get last error
if(fGotSecurityDescriptor)
// we got a security descriptor, so break
// NOTE we can't realloc m_pSecurityDescriptor to free its unused memory
// because dwSecDescSizeNeeded is 0 after successful call
break;
else if(dwLastError == ERROR_INSUFFICIENT_BUFFER)
{
// we didn't get a security descriptor because buffer was too small: increase buffer size, realloc and continue.
Assert(m_dwSecDescSize < dwSecDescSizeNeeded);
// set member buffer size to just enough chunks to accommodate security descriptor size needed
m_dwSecDescSize = ((dwSecDescSizeNeeded + SECURITY_DESC_GRANULARITY - 1) / SECURITY_DESC_GRANULARITY)
* SECURITY_DESC_GRANULARITY;
if(NULL == (m_pSecurityDescriptor = (PSECURITY_DESCRIPTOR) CTemplate::SmallReAlloc(m_pSecurityDescriptor, m_dwSecDescSize)))
THROW(E_OUTOFMEMORY);
}
else
{
// we didn't get a security descriptor for some random reason: return failure
fRet = FALSE;
break;
}
}
return fRet;
}
/* ============================================================================
CTemplate::CFileMap::UnmapFile
Unmaps a memory-mapped file
NOTE this leaves the filemap's path-info and path-tran members intact
Returns
Nothing
Side effects
None
*/
void
CTemplate::CFileMap::UnmapFile
(
)
{
if(m_pbStartOfFile != NULL)
if(!UnmapViewOfFile(m_pbStartOfFile)) THROW(E_FAIL);
if(m_hMap!= NULL)
if(!CloseHandle(m_hMap)) THROW(E_FAIL);
if(m_hFile != NULL && m_hFile != INVALID_HANDLE_VALUE)
if(!CloseHandle(m_hFile)) THROW(E_FAIL);
// Null-ify ptr and handles, since MapFile checks for non-null
m_pbStartOfFile = NULL;
m_hMap = NULL;
m_hFile = NULL;
}
/* ============================================================================
CTemplate::CFileMap::CountChars
Count the number of characters in the (open) filemap
Returns:
# of characters in the file
*/
DWORD
CTemplate::CFileMap::CountChars
(
WORD wCodePage
)
{
// Bug 84284: Scripts containing object tags only do not have the DBCS table built
// (Because there is no line mapping table to base it from)
//
CTemplate::COffsetInfo *pOffsetInfoLast, oiZero;
pOffsetInfoLast = (m_rgByte2DBCS.length() == 0)
? &oiZero
: &m_rgByte2DBCS[m_rgByte2DBCS.length() - 1];
// If GetSize() fails don't count the remaining DBCS characters - otherwise an AV
DWORD cchFilemap = GetSize();
if (cchFilemap != 0xFFFFFFFF && cchFilemap != 0)
{
// Count DBCS characters
m_cChars = pOffsetInfoLast->m_cchOffset +
CharAdvDBCS(wCodePage,
reinterpret_cast<char *>(m_pbStartOfFile + pOffsetInfoLast->m_cbOffset),
reinterpret_cast<char *>(m_pbStartOfFile + cchFilemap),
INFINITE, NULL);
}
else
{
m_cChars = 0;
}
// Done counting DBCS characters
return m_cChars;
}
/* ============================================================================
CTemplate::CFileMap::GetText
From a character offset and length, return the document text
File must be mapped
*/
HRESULT CTemplate::CFileMap::GetText
(
WORD wCodePage,
ULONG cchSourceOffset,
WCHAR *pwchText,
SOURCE_TEXT_ATTR *pTextAttr,
ULONG *pcChars,
ULONG cMaxChars
)
{
ULONG cCharsCopied;
if (pcChars == NULL)
pcChars = &cCharsCopied;
// Map the file (temporarily) if not mapped
BOOL fRemapFile = !FIsMapped();
TRY
RemapFile();
CATCH (dwException)
return E_FAIL;
END_TRY
/* Find the byte offset closest to cchSourceOffset. This will be
* the place where we start looping with CharNext() to get the full
* byte offset.
*/
COffsetInfo *pOffsetInfoLE = NULL, OffsetInfoT;
/*
* NOTE: compilation is done in two phases.
* Errors are detected and reported in phase 1.
* The DBCS mapping is created in phase 2.
*
* If an error occurred during compilation, the DBCS table does not exist.
* If there is no DBCS mapping table, then pretend like we found entry with
* nearest offset == 0. (unless this is SBCS in which case nearest
* offset == cchSourceOffset)
*/
if (m_rgByte2DBCS.length() == 0)
{
CPINFO CpInfo;
GetCPInfo(wCodePage, &CpInfo);
OffsetInfoT.m_cbOffset = OffsetInfoT.m_cchOffset = (CpInfo.MaxCharSize == 1)? cchSourceOffset : 0;
pOffsetInfoLE = &OffsetInfoT;
}
else
GetBracketingPair(
cchSourceOffset, // value to search for
m_rgByte2DBCS.begin(), // beginning of array to search
m_rgByte2DBCS.end(), // end of array
CCharOffsetOrder(), // order by character offsets
&pOffsetInfoLE, // desired offset
static_cast<COffsetInfo **>(NULL) // don't care
);
/* OK - pOffsetLE->cbOffset contains the closest offset not exceeding
* cchSourceOffset. Iterate over the remainder of the characters
* to convert the cch to a cb. It had better exist!
*/
Assert (pOffsetInfoLE != NULL);
// Advance over remaining characters
char *pchStart;
CharAdvDBCS(wCodePage,
reinterpret_cast<char *>(m_pbStartOfFile + pOffsetInfoLE->m_cbOffset),
reinterpret_cast<char *>(m_pbStartOfFile + GetSize()),
cchSourceOffset - pOffsetInfoLE->m_cchOffset,
&pchStart
);
// Compute # of Characters to copy
Assert (m_cChars >= cchSourceOffset);
*pcChars = min(cMaxChars, m_cChars - cchSourceOffset);
// Compute # of Bytes to copy
char *pchEnd;
CharAdvDBCS(wCodePage,
pchStart,
reinterpret_cast<char *>(m_pbStartOfFile + GetSize()),
*pcChars,
&pchEnd
);
if (pwchText)
MultiByteToWideChar(
(WORD)wCodePage,
0,
pchStart,
DIFF(pchEnd - pchStart),
pwchText,
cMaxChars
);
// We don't support syntax coloring, so set all the character attributes to
// default color (black)
if (pTextAttr)
memset(pTextAttr, 0, *pcChars);
// Unmap the file (but only if we previously remapped it)
if (fRemapFile)
TRY
UnmapFile();
CATCH (dwException)
return E_FAIL;
END_TRY
return S_OK;
}
/* ****************************************************************************
CTemplate::CTokenList member functions
*/
/* ============================================================================
CTemplate::CTokenList::Init
Populates token list with tokens
Returns
Nothing
Side effects
None
*/
void
CTemplate::CTokenList::Init
(
)
{
// Init tokens buffer for local storage
m_bufTokens.Init(tkncAll, CB_TOKENS_DEFAULT);
// append tokens to buffer
// NOTE *** TOKENS MUST BE IN SAME ORDER AS ENUM TYPE VALUES ***
// NOTE 'superset' token must precede 'subset' token (e.g. <!--#INCLUDE before <!--)
AppendToken(tknOpenPrimaryScript, "<%");
AppendToken(tknOpenTaggedScript, "<SCRIPT");
AppendToken(tknOpenObject, "<OBJECT");
AppendToken(tknOpenHTMLComment, "<!--");
AppendToken(tknNewLine, SZ_NEWLINE);
AppendToken(tknClosePrimaryScript, "%>");
AppendToken(tknCloseTaggedScript, "</SCRIPT>");
AppendToken(tknCloseObject, "</OBJECT>");
AppendToken(tknCloseHTMLComment, "-->");
AppendToken(tknEscapedClosePrimaryScript, "%\\>");
AppendToken(tknCloseTag, ">");
AppendToken(tknCommandINCLUDE, "#INCLUDE");
AppendToken(tknTagRunat, "RUNAT");
AppendToken(tknTagLanguage, "LANGUAGE");
AppendToken(tknTagCodePage, "CODEPAGE");
AppendToken(tknTagCodePage, "LCID");
AppendToken(tknTagTransacted, "TRANSACTION");
AppendToken(tknTagSession, "ENABLESESSIONSTATE");
AppendToken(tknTagID, "ID");
AppendToken(tknTagClassID, "CLASSID");
AppendToken(tknTagProgID, "PROGID");
AppendToken(tknTagScope, "SCOPE");
AppendToken(tknTagVirtual, "VIRTUAL");
AppendToken(tknTagFile, "FILE");
AppendToken(tknTagMETADATA, "METADATA");
// AppendToken(tknTagSetPriScriptLang, "@");
AppendToken(tknTagName, "NAME");
AppendToken(tknValueTypeLib, "TYPELIB");
AppendToken(tknTagType, "TYPE");
AppendToken(tknTagUUID, "UUID");
AppendToken(tknTagVersion, "VERSION");
AppendToken(tknTagStartspan, "STARTSPAN");
AppendToken(tknTagEndspan, "ENDSPAN");
AppendToken(tknValueCookie, "COOKIE");
AppendToken(tknTagSrc, "SRC");
AppendToken(tknValueServer, "Server");
AppendToken(tknValueApplication, "Application");
AppendToken(tknValueSession, "Session");
AppendToken(tknValuePage, "Page");
AppendToken(tknVBSCommentSQuote, "'");
AppendToken(tknVBSCommentRem, "REM "); // NOTE ends with space character
AppendToken(tknTagFPBot, "webbot");
AppendToken(tknEOF, "");
AppendToken(tkncAll, "");
}
/* ============================================================================
CTemplate::CTokenList::AppendToken
Appends a string to tokens buffer
NOTE we keep the unused tkn parameter because it enforces consistency and
readability in CTemplate::CTokenList::Init(), e.g.
AppendToken(tknOpenPrimaryScript, "<%");
rather than
AppendToken("<%");
Returns:
Nothing
Side effects:
None
*/
void
CTemplate::CTokenList::AppendToken
(
_TOKEN tkn, // token value
char* sz // token string
)
{
// construct byte range from token string
CByteRange br;
br.m_pb = (BYTE*) sz;
br.m_cb = strlen(sz);
// append to tokens buffer as local string
m_bufTokens.Append(br, TRUE, 0, NULL, TRUE);
}
/* ============================================================================
CTemplate::CTokenList::NextOpenToken
Returns value of next open token in search range
Returns
token value of next open token in search range; ptr to ptr to open token (out-parameter)
Side effects
None
*/
_TOKEN
CTemplate::CTokenList::NextOpenToken
(
CByteRange& brSearch, // search byte range
TOKEN* rgtknOpeners, // array of permitted open tokens
UINT ctknOpeners, // count of permitted open tokens
BYTE** ppbToken, // ptr to ptr to open token (out-parameter)
LONG lCodePage
)
{
BYTE* pbTemp = NULL; // temp pointer
_TOKEN tkn = tknEOF; // return value
USHORT i; // loop index
// Init caller's token ptr to null
*ppbToken = NULL;
// If input is empty, return
if (brSearch.IsNull())
return tkn;
// Prepare array of LPSTR pointers to tokens.
// Do it here once, because to get LPSTR is not free.
LPSTR rgszTokens[TOKEN_OPENERS_MAX];
UINT rgcchTokens[TOKEN_OPENERS_MAX];
Assert(ctknOpeners <= TOKEN_OPENERS_MAX);
for (i = 0; i < ctknOpeners; i++)
{
LPSTR pszStr = m_bufTokens.PszLocal((UINT)(rgtknOpeners[i]));
rgszTokens[i] = pszStr;
rgcchTokens[i] = (pszStr != NULL) ? strlen(pszStr) : 0;
}
// Call a method to find one of the strings in the range
UINT idToken;
pbTemp = brSearch.PbOneOfAspOpenerStringTokens(
rgszTokens, rgcchTokens, ctknOpeners, &idToken);
if (pbTemp != NULL)
{
*ppbToken = pbTemp;
tkn = rgtknOpeners[idToken];
}
// If we found no open token, position token pointer at end of search range
if (tkn == tknEOF)
*ppbToken = brSearch.m_pb + brSearch.m_cb;
return tkn;
}
/* ============================================================================
CTemplate::CTokenList::MovePastToken
Moves a byte range past a token contained within it
*/
void
CTemplate::CTokenList::MovePastToken
(
_TOKEN tkn,
BYTE* pbToken,
CByteRange& brSearch
)
{
Assert(pbToken >= brSearch.m_pb);
Assert(brSearch.m_cb >= (DIFF(pbToken - brSearch.m_pb) + CCH_TOKEN_X(tkn)));
brSearch.Advance(DIFF(pbToken - brSearch.m_pb) + CCH_TOKEN_X(tkn));
}
/* ============================================================================
CTemplate::CTokenList::GetToken
Gets the next occurrence of a token within a byte range.
Returns:
ptr to token
Side effects
none
*/
BYTE*
CTemplate::CTokenList::GetToken
(
TOKEN tkn,
CByteRange& brSearch,
LONG lCodePage
)
{
return brSearch.PbString(m_bufTokens.PszLocal((UINT)tkn), lCodePage);
}
/* ============================================================================
The Big Three for CTemplateConnPt
NOTES:
Since this interface is embedded in CTemplate,
AddRef() and Release() delegate to the container object (because that
is the CTemplate pointer)
*/
HRESULT
CTemplateConnPt::QueryInterface(const GUID &uidInterface, void **ppvObj)
{
if (uidInterface == IID_IUnknown || uidInterface == IID_IConnectionPoint)
{
*ppvObj = this;
AddRef();
return S_OK;
}
else
{
*ppvObj = NULL;
return E_NOINTERFACE;
}
}
ULONG
CTemplateConnPt::AddRef()
{
return m_pUnkContainer->AddRef();
}
ULONG
CTemplateConnPt::Release()
{
return m_pUnkContainer->Release();
}
/* ============================================================================
Constructor for CDocNode
*/
CTemplate::CDocNodeElem::CDocNodeElem(CAppln *pAppln, IDebugApplicationNode *pDocRoot)
{
Assert (pAppln != NULL);
Assert (pDocRoot != NULL);
(m_pAppln = pAppln)->AddRef();
(m_pDocRoot = pDocRoot)->AddRef();
}
/* ============================================================================
Destructor for CDocNode
*/
CTemplate::CDocNodeElem::~CDocNodeElem()
{
m_pAppln->Release();
DestroyDocumentTree(m_pDocRoot);
}
/* ============================================================================
CTemplate::fIsLangVBScriptOrJScript(USHORT idEngine)
This function returns T/F to determine if the requested script engine
is VBScript or JScript. This function is used as an indicator to determin
if spaces need to be preserved for non MS Scripting languages
There is an assumption here that the GUIDs for VBScript and JScript will not change
Inputs
Index to a script engine
Returns
BOOL
*/
BOOLB CTemplate::FIsLangVBScriptOrJScript(USHORT idEngine)
{
// {b54f3741-5b07-11cf-a4b0-00aa004a55e8} VBScript
static const GUID uid_VBScript = {0xb54f3741, 0x5b07, 0x11cf, {0xa4, 0xb0, 0x00, 0xaa, 0x00, 0x4a, 0x55, 0xe8}};
// {f414c260-6ac0-11cf-b6d1-00aa00bbbb58} JavaScript
static const GUID uid_JScript = {0xf414c260, 0x6ac0, 0x11cf, {0xb6, 0xd1, 0x00, 0xaa, 0x00, 0xbb, 0xbb, 0x58}};
// {b54f3743-5b07-11cf-a4b0-00aa004a55e8} VBScript.Encode
static const GUID uid_VBScriptEncode = {0xb54f3743, 0x5b07, 0x11cf, {0xa4, 0xb0, 0x00, 0xaa, 0x00, 0x4a, 0x55, 0xe8}};
// {f414c262-6ac0-11cf-b6d1-00aa00bbbb58} JavaScript.Encode
static const GUID uid_JScriptEncode = {0xf414c262, 0x6ac0, 0x11cf, {0xb6, 0xd1, 0x00, 0xaa, 0x00, 0xbb, 0xbb, 0x58}};
GUID &uidLang = m_pWorkStore->m_ScriptStore.m_rgProgLangId[idEngine];
return
uidLang == uid_VBScript || uidLang == uid_VBScriptEncode ||
uidLang == uid_JScript || uidLang == uid_JScriptEncode;
}
SIZE_T
_RoundUp(
SIZE_T dwBytes)
{
#if 1
// 16KB <= dwBytes? Round up to next multiple of 4KB
if (16*1024 <= dwBytes)
dwBytes = ((dwBytes + (1<<12) - 1) >> 12) << 12;
// 4KB <= dwBytes < 16KB? Round up to next multiple of 1KB
else if (4*1024 <= dwBytes)
dwBytes = ((dwBytes + (1<<10) - 1) >> 10) << 10;
// 1KB <= dwBytes < 4KB? Round up to next multiple of 256 bytes
else if (1024 <= dwBytes)
dwBytes = ((dwBytes + (1<<8) - 1) >> 8) << 8;
// dwBytes < 1KB? Round up to next multiple of 32 bytes
else
dwBytes = ((dwBytes + (1<<5) - 1) >> 5) << 5;
#endif
return dwBytes;
}
void*
CTemplate::SmallMalloc(SIZE_T dwBytes)
{
DBG_ASSERT(sm_hSmallHeap != NULL);
dwBytes = _RoundUp(dwBytes);
return ::HeapAlloc(sm_hSmallHeap, 0, dwBytes);
}
void*
CTemplate::SmallReAlloc(void* pvMem, SIZE_T dwBytes)
{
DBG_ASSERT(sm_hSmallHeap != NULL);
dwBytes = _RoundUp(dwBytes);
return ::HeapReAlloc(sm_hSmallHeap, 0, pvMem, dwBytes);
}
void
CTemplate::SmallFree(void* pvMem)
{
DBG_ASSERT(sm_hSmallHeap != NULL);
::HeapFree(sm_hSmallHeap, 0, pvMem);
}
void*
CTemplate::LargeMalloc(SIZE_T dwBytes)
{
DBG_ASSERT(sm_hLargeHeap != NULL);
dwBytes = _RoundUp(dwBytes);
return ::HeapAlloc(sm_hLargeHeap, 0, dwBytes);
}
void*
CTemplate::LargeReAlloc(void* pvMem, SIZE_T dwBytes)
{
DBG_ASSERT(sm_hLargeHeap != NULL);
dwBytes = _RoundUp(dwBytes);
return ::HeapReAlloc(sm_hLargeHeap, 0, pvMem, dwBytes);
}
void
CTemplate::LargeFree(void* pvMem)
{
DBG_ASSERT(sm_hLargeHeap != NULL);
::HeapFree(sm_hLargeHeap, 0, pvMem);
}