blob: c3f3d22ed5f9092919923155619151d5c275484c [file] [log] [blame]
//-------------------------------------------------------------------------------------------------------
// Copyright (C) Microsoft. All rights reserved.
// Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
//-------------------------------------------------------------------------------------------------------
#include "RuntimeBasePch.h"
// Parser Includes
#include "RegexCommon.h"
#include "DebugWriter.h"
#include "RegexStats.h"
#include "ByteCode/ByteCodeApi.h"
#include "Library/ProfileString.h"
#include "Debug/DiagHelperMethodWrapper.h"
#include "BackendApi.h"
#if PROFILE_DICTIONARY
#include "DictionaryStats.h"
#endif
#include "Base/ScriptContextProfiler.h"
#include "Base/EtwTrace.h"
#include "Language/InterpreterStackFrame.h"
#include "Language/SourceDynamicProfileManager.h"
#include "Language/JavascriptStackWalker.h"
#include "Language/AsmJsTypes.h"
#include "Language/AsmJsModule.h"
#ifdef ASMJS_PLAT
#include "Language/AsmJsEncoder.h"
#include "Language/AsmJsCodeGenerator.h"
#endif
#ifdef ENABLE_BASIC_TELEMETRY
#include "ScriptContextTelemetry.h"
#endif
namespace Js
{
ScriptContext * ScriptContext::New(ThreadContext * threadContext)
{
AutoPtr<ScriptContext> scriptContext(HeapNew(ScriptContext, threadContext));
scriptContext->InitializeAllocations();
return scriptContext.Detach();
}
void ScriptContext::Delete(ScriptContext* scriptContext)
{
HeapDelete(scriptContext);
}
ScriptContext::ScriptContext(ThreadContext* threadContext) :
ScriptContextBase(),
interpreterArena(nullptr),
moduleSrcInfoCount(0),
// Regex globals
#if ENABLE_REGEX_CONFIG_OPTIONS
regexStatsDatabase(0),
regexDebugWriter(0),
#endif
trigramAlphabet(nullptr),
regexStacks(nullptr),
arrayMatchInit(false),
config(threadContext->GetConfig(), threadContext->IsOptimizedForManyInstances()),
#if ENABLE_BACKGROUND_PARSING
backgroundParser(nullptr),
#endif
#if ENABLE_NATIVE_CODEGEN
nativeCodeGen(nullptr),
#endif
threadContext(threadContext),
scriptStartEventHandler(nullptr),
scriptEndEventHandler(nullptr),
#ifdef FAULT_INJECTION
disposeScriptByFaultInjectionEventHandler(nullptr),
#endif
integerStringMap(this->GeneralAllocator()),
guestArena(nullptr),
raiseMessageToDebuggerFunctionType(nullptr),
transitionToDebugModeIfFirstSourceFn(nullptr),
sourceSize(0),
deferredBody(false),
isScriptContextActuallyClosed(false),
isFinalized(false),
isInvalidatedForHostObjects(false),
fastDOMenabled(false),
directHostTypeId(TypeIds_GlobalObject),
isPerformingNonreentrantWork(false),
isDiagnosticsScriptContext(false),
m_enumerateNonUserFunctionsOnly(false),
recycler(threadContext->EnsureRecycler()),
CurrentThunk(DefaultEntryThunk),
CurrentCrossSiteThunk(CrossSite::DefaultThunk),
DeferredParsingThunk(DefaultDeferredParsingThunk),
DeferredDeserializationThunk(DefaultDeferredDeserializeThunk),
DispatchDefaultInvoke(nullptr),
DispatchProfileInvoke(nullptr),
m_pBuiltinFunctionIdMap(nullptr),
diagnosticArena(nullptr),
hostScriptContext(nullptr),
scriptEngineHaltCallback(nullptr),
#if DYNAMIC_INTERPRETER_THUNK
interpreterThunkEmitter(nullptr),
#endif
#ifdef ASMJS_PLAT
asmJsInterpreterThunkEmitter(nullptr),
asmJsCodeGenerator(nullptr),
#endif
generalAllocator(_u("SC-General"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
#ifdef ENABLE_BASIC_TELEMETRY
telemetryAllocator(_u("SC-Telemetry"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
#endif
dynamicProfileInfoAllocator(_u("SC-DynProfileInfo"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
#ifdef SEPARATE_ARENA
sourceCodeAllocator(_u("SC-Code"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
regexAllocator(_u("SC-Regex"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
#endif
#ifdef NEED_MISC_ALLOCATOR
miscAllocator(_u("GC-Misc"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
#endif
inlineCacheAllocator(_u("SC-InlineCache"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
isInstInlineCacheAllocator(_u("SC-IsInstInlineCache"), threadContext->GetPageAllocator(), Throw::OutOfMemory),
hasUsedInlineCache(false),
hasProtoOrStoreFieldInlineCache(false),
hasIsInstInlineCache(false),
registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext(nullptr),
cache(nullptr),
firstInterpreterFrameReturnAddress(nullptr),
builtInLibraryFunctions(nullptr),
isWeakReferenceDictionaryListCleared(false)
#if ENABLE_PROFILE_INFO
, referencesSharedDynamicSourceContextInfo(false)
#endif
#if DBG
, isInitialized(false)
, isCloningGlobal(false)
, bindRef(MiscAllocator())
#endif
#if ENABLE_TTD
, TTDHostCallbackFunctor()
, TTDMode(TTD::TTDMode::Invalid)
, ScriptContextLogTag(TTD_INVALID_LOG_PTR_ID)
, TTDRootNestingCount(0)
, TTDWellKnownInfo(nullptr)
, TTDContextInfo(nullptr)
#endif
#ifdef REJIT_STATS
, rejitStatsMap(nullptr)
#endif
#ifdef ENABLE_BASIC_TELEMETRY
, telemetry(nullptr)
#endif
#ifdef INLINE_CACHE_STATS
, cacheDataMap(nullptr)
#endif
#ifdef FIELD_ACCESS_STATS
, fieldAccessStatsByFunctionNumber(nullptr)
#endif
, webWorkerId(Js::Constants::NonWebWorkerContextId)
, url(_u(""))
, startupComplete(false)
, isEnumeratingRecyclerObjects(false)
#ifdef EDIT_AND_CONTINUE
, activeScriptEditQuery(nullptr)
#endif
#ifdef ENABLE_SCRIPT_PROFILING
, heapEnum(nullptr)
#endif
#ifdef RECYCLER_PERF_COUNTERS
, bindReferenceCount(0)
#endif
, nextPendingClose(nullptr)
#ifdef ENABLE_SCRIPT_PROFILING
, m_fTraceDomCall(FALSE)
#endif
#ifdef ENABLE_DOM_FAST_PATH
, domFastPathIRHelperMap(nullptr)
#endif
, intConstPropsOnGlobalObject(nullptr)
, intConstPropsOnGlobalUserObject(nullptr)
#ifdef PROFILE_STRINGS
, stringProfiler(nullptr)
#endif
#ifdef PROFILE_BAILOUT_RECORD_MEMORY
, codeSize(0)
, bailOutRecordBytes(0)
, bailOutOffsetBytes(0)
#endif
, debugContext(nullptr)
{
// This may allocate memory and cause exception, but it is ok, as we all we have done so far
// are field init and those dtor will be called if exception occurs
threadContext->EnsureDebugManager();
// Don't use throwing memory allocation in ctor, as exception in ctor doesn't cause the dtor to be called
// potentially causing memory leaks
BEGIN_NO_EXCEPTION;
#ifdef RUNTIME_DATA_COLLECTION
createTime = time(nullptr);
#endif
#ifdef BGJIT_STATS
interpretedCount = maxFuncInterpret = funcJITCount = bytecodeJITCount = interpretedCallsHighPri = jitCodeUsed = funcJitCodeUsed = loopJITCount = speculativeJitCount = 0;
#endif
#ifdef PROFILE_TYPES
convertNullToSimpleCount = 0;
convertNullToSimpleDictionaryCount = 0;
convertNullToDictionaryCount = 0;
convertDeferredToDictionaryCount = 0;
convertDeferredToSimpleDictionaryCount = 0;
convertSimpleToDictionaryCount = 0;
convertSimpleToSimpleDictionaryCount = 0;
convertPathToDictionaryCount1 = 0;
convertPathToDictionaryCount2 = 0;
convertPathToDictionaryCount3 = 0;
convertPathToDictionaryCount4 = 0;
convertPathToSimpleDictionaryCount = 0;
convertSimplePathToPathCount = 0;
convertSimpleDictionaryToDictionaryCount = 0;
convertSimpleSharedDictionaryToNonSharedCount = 0;
convertSimpleSharedToNonSharedCount = 0;
simplePathTypeHandlerCount = 0;
pathTypeHandlerCount = 0;
promoteCount = 0;
cacheCount = 0;
branchCount = 0;
maxPathLength = 0;
memset(typeCount, 0, sizeof(typeCount));
memset(instanceCount, 0, sizeof(instanceCount));
#endif
#ifdef PROFILE_OBJECT_LITERALS
objectLiteralInstanceCount = 0;
objectLiteralPathCount = 0;
memset(objectLiteralCount, 0, sizeof(objectLiteralCount));
objectLiteralSimpleDictionaryCount = 0;
objectLiteralMaxLength = 0;
objectLiteralPromoteCount = 0;
objectLiteralCacheCount = 0;
objectLiteralBranchCount = 0;
#endif
#if DBG_DUMP
byteCodeDataSize = 0;
byteCodeAuxiliaryDataSize = 0;
byteCodeAuxiliaryContextDataSize = 0;
memset(byteCodeHistogram, 0, sizeof(byteCodeHistogram));
#endif
memset(propertyStrings, 0, sizeof(PropertyStringMap*)* 80);
#if DBG || defined(RUNTIME_DATA_COLLECTION)
this->allocId = threadContext->GetUnreleasedScriptContextCount();
#endif
#if DBG
this->hadProfiled = false;
#endif
#if DBG_DUMP
forinCache = 0;
forinNoCache = 0;
#endif
callCount = 0;
threadContext->GetHiResTimer()->Reset();
#ifdef PROFILE_EXEC
profiler = nullptr;
isProfilerCreated = false;
disableProfiler = false;
ensureParentInfo = false;
#endif
#ifdef PROFILE_MEM
profileMemoryDump = true;
#endif
#ifdef ENABLE_SCRIPT_PROFILING
m_pProfileCallback = nullptr;
m_pProfileCallback2 = nullptr;
m_inProfileCallback = FALSE;
CleanupDocumentContext = nullptr;
#endif
// Do this after all operations that may cause potential exceptions
threadContext->RegisterScriptContext(this);
numberAllocator.Initialize(this->GetRecycler());
#if DEBUG
m_iProfileSession = -1;
#endif
#ifdef LEAK_REPORT
this->urlRecord = nullptr;
this->isRootTrackerScriptContext = false;
#endif
PERF_COUNTER_INC(Basic, ScriptContext);
PERF_COUNTER_INC(Basic, ScriptContextActive);
END_NO_EXCEPTION;
}
void ScriptContext::InitializeAllocations()
{
this->charClassifier = Anew(GeneralAllocator(), CharClassifier, this);
this->valueOfInlineCache = AllocatorNewZ(InlineCacheAllocator, GetInlineCacheAllocator(), InlineCache);
this->toStringInlineCache = AllocatorNewZ(InlineCacheAllocator, GetInlineCacheAllocator(), InlineCache);
#ifdef REJIT_STATS
if (PHASE_STATS1(Js::ReJITPhase))
{
rejitReasonCounts = AnewArrayZ(GeneralAllocator(), uint, NumRejitReasons);
bailoutReasonCounts = Anew(GeneralAllocator(), BailoutStatsMap, GeneralAllocator());
}
#endif
#ifdef ENABLE_BASIC_TELEMETRY
this->telemetry = Anew(this->TelemetryAllocator(), ScriptContextTelemetry, *this);
#endif
#ifdef PROFILE_STRINGS
if (Js::Configuration::Global.flags.ProfileStrings)
{
stringProfiler = Anew(MiscAllocator(), StringProfiler, threadContext->GetPageAllocator());
}
#endif
intConstPropsOnGlobalObject = Anew(GeneralAllocator(), PropIdSetForConstProp, GeneralAllocator());
intConstPropsOnGlobalUserObject = Anew(GeneralAllocator(), PropIdSetForConstProp, GeneralAllocator());
this->debugContext = HeapNew(DebugContext, this);
}
void ScriptContext::EnsureClearDebugDocument()
{
if (this->sourceList)
{
this->sourceList->Map([=](uint i, RecyclerWeakReference<Js::Utf8SourceInfo>* sourceInfoWeakRef) {
Js::Utf8SourceInfo* sourceInfo = sourceInfoWeakRef->Get();
if (sourceInfo)
{
sourceInfo->ClearDebugDocument();
}
});
}
}
void ScriptContext::ShutdownClearSourceLists()
{
if (this->sourceList)
{
// In the unclean shutdown case, we might not have destroyed the script context when
// this is called- in which case, skip doing this work and simply release the source list
// so that it doesn't show up as a leak. Since we're doing unclean shutdown, it's ok to
// skip cleanup here for expediency.
if (this->isClosed)
{
this->MapFunction([this](Js::FunctionBody* functionBody) {
Assert(functionBody->GetScriptContext() == this);
functionBody->CleanupSourceInfo(true);
});
}
EnsureClearDebugDocument();
// Don't need the source list any more so ok to release
this->sourceList.Unroot(this->GetRecycler());
}
if (this->calleeUtf8SourceInfoList)
{
this->calleeUtf8SourceInfoList.Unroot(this->GetRecycler());
}
}
ScriptContext::~ScriptContext()
{
// Take etw rundown lock on this thread context. We are going to change/destroy this scriptContext.
AutoCriticalSection autocs(GetThreadContext()->GetEtwRundownCriticalSection());
// TODO: Can we move this on Close()?
ClearHostScriptContext();
if (this->hasProtoOrStoreFieldInlineCache)
{
// TODO (PersistentInlineCaches): It really isn't necessary to clear inline caches in all script contexts.
// Since this script context is being destroyed, the inline cache arena will also go away and release its
// memory back to the page allocator. Thus, we cannot leave this script context's inline caches on the
// thread context's invalidation lists. However, it should suffice to remove this script context's caches
// without touching other script contexts' caches. We could call some form of RemoveInlineCachesFromInvalidationLists()
// on the inline cache allocator, which would walk all inline caches and zap values pointed to by strongRef.
// clear out all inline caches to remove our proto inline caches from the thread context
threadContext->ClearInlineCaches();
ClearInlineCaches();
Assert(!this->hasProtoOrStoreFieldInlineCache);
}
if (this->hasIsInstInlineCache)
{
// clear out all inline caches to remove our proto inline caches from the thread context
threadContext->ClearIsInstInlineCaches();
ClearIsInstInlineCaches();
Assert(!this->hasIsInstInlineCache);
}
// Only call RemoveFromPendingClose if we are in a pending close state.
if (isClosed && !isScriptContextActuallyClosed)
{
threadContext->RemoveFromPendingClose(this);
}
this->isClosed = true;
bool closed = Close(true);
// JIT may access number allocator. Need to close the script context first,
// which will close the native code generator and abort any current job on this generator.
numberAllocator.Uninitialize();
ShutdownClearSourceLists();
if (regexStacks)
{
Adelete(RegexAllocator(), regexStacks);
regexStacks = nullptr;
}
if (javascriptLibrary != nullptr)
{
javascriptLibrary->scriptContext = nullptr;
javascriptLibrary = nullptr;
if (closed)
{
// if we just closed, we haven't unpin the object yet.
// We need to null out the script context in the global object first
// before we unpin the global object so that script context dtor doesn't get called twice
#if ENABLE_NATIVE_CODEGEN
Assert(this->IsClosedNativeCodeGenerator());
#endif
if (!GetThreadContext()->IsJSRT())
{
this->recycler->RootRelease(globalObject);
}
}
}
#if ENABLE_BACKGROUND_PARSING
if (this->backgroundParser != nullptr)
{
BackgroundParser::Delete(this->backgroundParser);
this->backgroundParser = nullptr;
}
#endif
#if ENABLE_NATIVE_CODEGEN
if (this->nativeCodeGen != nullptr)
{
DeleteNativeCodeGenerator(this->nativeCodeGen);
nativeCodeGen = NULL;
}
#endif
#if DYNAMIC_INTERPRETER_THUNK
if (this->interpreterThunkEmitter != nullptr)
{
HeapDelete(interpreterThunkEmitter);
this->interpreterThunkEmitter = NULL;
}
#endif
#ifdef ASMJS_PLAT
if (this->asmJsInterpreterThunkEmitter != nullptr)
{
HeapDelete(asmJsInterpreterThunkEmitter);
this->asmJsInterpreterThunkEmitter = nullptr;
}
if (this->asmJsCodeGenerator != nullptr)
{
HeapDelete(asmJsCodeGenerator);
this->asmJsCodeGenerator = NULL;
}
#endif
// In case there is something added to the list between close and dtor, just reset the list again
this->weakReferenceDictionaryList.Reset();
PERF_COUNTER_DEC(Basic, ScriptContext);
}
void ScriptContext::SetUrl(BSTR bstrUrl)
{
// Assumption: this method is never called multiple times
Assert(this->url != nullptr && wcslen(this->url) == 0);
charcount_t length = SysStringLen(bstrUrl) + 1; // Add 1 for the NULL.
char16* urlCopy = AnewArray(this->GeneralAllocator(), char16, length);
js_memcpy_s(urlCopy, (length - 1) * sizeof(char16), bstrUrl, (length - 1) * sizeof(char16));
urlCopy[length - 1] = _u('\0');
this->url = urlCopy;
#ifdef LEAK_REPORT
if (Js::Configuration::Global.flags.IsEnabled(Js::LeakReportFlag))
{
this->urlRecord = LeakReport::LogUrl(urlCopy, this->globalObject);
}
#endif
}
uint ScriptContext::GetNextSourceContextId()
{
Assert(this->cache);
Assert(this->cache->sourceContextInfoMap ||
this->cache->dynamicSourceContextInfoMap);
uint nextSourceContextId = 0;
if (this->cache->sourceContextInfoMap)
{
nextSourceContextId = this->cache->sourceContextInfoMap->Count();
}
if (this->cache->dynamicSourceContextInfoMap)
{
nextSourceContextId += this->cache->dynamicSourceContextInfoMap->Count();
}
return nextSourceContextId + 1;
}
// Do most of the Close() work except the final release which could delete the scriptContext.
void ScriptContext::InternalClose()
{
this->PrintStats();
isScriptContextActuallyClosed = true;
PERF_COUNTER_DEC(Basic, ScriptContextActive);
#if DBG_DUMP
if (Js::Configuration::Global.flags.TraceWin8Allocations)
{
Output::Print(_u("MemoryTrace: ScriptContext Close\n"));
Output::Flush();
}
#endif
#ifdef ENABLE_JS_ETW
EventWriteJSCRIPT_HOST_SCRIPT_CONTEXT_CLOSE(this);
#endif
#if ENABLE_TTD
if(this->TTDWellKnownInfo != nullptr)
{
TT_HEAP_DELETE(TTD::RuntimeContextInfo, this->TTDWellKnownInfo);
this->TTDWellKnownInfo = nullptr;
}
if(this->TTDContextInfo != nullptr)
{
TT_HEAP_DELETE(TTD::ScriptContextTTD, this->TTDContextInfo);
this->TTDContextInfo = nullptr;
}
#endif
#if ENABLE_PROFILE_INFO
HRESULT hr = S_OK;
BEGIN_TRANSLATE_OOM_TO_HRESULT_NESTED
{
DynamicProfileInfo::Save(this);
}
END_TRANSLATE_OOM_TO_HRESULT(hr);
#if DBG_DUMP || defined(DYNAMIC_PROFILE_STORAGE) || defined(RUNTIME_DATA_COLLECTION)
this->ClearDynamicProfileList();
#endif
#endif
#if ENABLE_NATIVE_CODEGEN
if (nativeCodeGen != nullptr)
{
Assert(!isInitialized || this->globalObject != nullptr);
CloseNativeCodeGenerator(this->nativeCodeGen);
}
#endif
if (this->fakeGlobalFuncForUndefer)
{
this->fakeGlobalFuncForUndefer->Cleanup(true);
this->fakeGlobalFuncForUndefer.Unroot(this->GetRecycler());
}
if (this->sourceList)
{
bool hasFunctions = false;
this->sourceList->MapUntil([&hasFunctions](int, RecyclerWeakReference<Utf8SourceInfo>* sourceInfoWeakRef) -> bool
{
Utf8SourceInfo* sourceInfo = sourceInfoWeakRef->Get();
if (sourceInfo)
{
hasFunctions = sourceInfo->HasFunctions();
}
return hasFunctions;
});
if (hasFunctions)
{
// We still need to walk through all the function bodies and call cleanup
// because otherwise ETW events might not get fired if a GC doesn't happen
// and the thread context isn't shut down cleanly (process detach case)
this->MapFunction([this](Js::FunctionBody* functionBody) {
Assert(functionBody->GetScriptContext() == nullptr || functionBody->GetScriptContext() == this);
functionBody->Cleanup(/* isScriptContextClosing */ true);
});
}
}
JS_ETW(EtwTrace::LogSourceUnloadEvents(this));
this->GetThreadContext()->SubSourceSize(this->GetSourceSize());
#if DYNAMIC_INTERPRETER_THUNK
if (this->interpreterThunkEmitter != nullptr)
{
this->interpreterThunkEmitter->Close();
}
#endif
#ifdef ASMJS_PLAT
if (this->asmJsInterpreterThunkEmitter != nullptr)
{
this->asmJsInterpreterThunkEmitter->Close();
}
#endif
#ifdef ENABLE_SCRIPT_PROFILING
// Stop profiling if present
DeRegisterProfileProbe(S_OK, nullptr);
#endif
if (this->diagnosticArena != nullptr)
{
HeapDelete(this->diagnosticArena);
this->diagnosticArena = nullptr;
}
if (this->debugContext != nullptr)
{
// Guard the closing and deleting of DebugContext as in meantime PDM might
// call OnBreakFlagChange
AutoCriticalSection autoDebugContextCloseCS(&debugContextCloseCS);
this->debugContext->Close();
HeapDelete(this->debugContext);
this->debugContext = nullptr;
}
// Need to print this out before the native code gen is deleted
// which will delete the codegenProfiler
#ifdef PROFILE_EXEC
if (Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag))
{
if (isProfilerCreated)
{
this->ProfilePrint();
}
if (profiler != nullptr)
{
profiler->Release();
profiler = nullptr;
}
}
#endif
#if ENABLE_PROFILE_INFO
// Release this only after native code gen is shut down, as there may be
// profile info allocated from the SourceDynamicProfileManager arena.
// The first condition might not be true if the dynamic functions have already been freed by the time
// ScriptContext closes
if (referencesSharedDynamicSourceContextInfo)
{
// For the host provided dynamic code, we may not have added any dynamic context to the dynamicSourceContextInfoMap
Assert(this->GetDynamicSourceContextInfoMap() != nullptr);
this->GetThreadContext()->ReleaseSourceDynamicProfileManagers(this->GetUrl());
}
#endif
RECYCLER_PERF_COUNTER_SUB(BindReference, bindReferenceCount);
if (this->interpreterArena)
{
ReleaseInterpreterArena();
interpreterArena = nullptr;
}
if (this->guestArena)
{
ReleaseGuestArena();
guestArena = nullptr;
}
cache = nullptr;
builtInLibraryFunctions = nullptr;
pActiveScriptDirect = nullptr;
isWeakReferenceDictionaryListCleared = true;
this->weakReferenceDictionaryList.Clear(this->GeneralAllocator());
// This can be null if the script context initialization threw
// and InternalClose gets called in the destructor code path
if (javascriptLibrary != nullptr)
{
javascriptLibrary->CleanupForClose();
javascriptLibrary->Uninitialize();
}
if (registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext != nullptr)
{
// UnregisterPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext may throw, set up the correct state first
ScriptContext ** registeredScriptContext = registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext;
ClearPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesCaches();
Assert(registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext == nullptr);
threadContext->UnregisterPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext(registeredScriptContext);
}
threadContext->ReleaseDebugManager();
}
bool ScriptContext::Close(bool inDestructor)
{
if (isScriptContextActuallyClosed)
return false;
// Limit the lock scope. We require the same lock in ~ScriptContext(), which may be called next.
{
// Take etw rundown lock on this thread context. We are going to change this scriptContext.
AutoCriticalSection autocs(GetThreadContext()->GetEtwRundownCriticalSection());
InternalClose();
}
if (!inDestructor && globalObject != nullptr)
{
//A side effect of releasing globalObject that this script context could be deleted, so the release call here
//must be the last thing in close.
#if ENABLE_NATIVE_CODEGEN
Assert(this->IsClosedNativeCodeGenerator());
#endif
if (!GetThreadContext()->IsJSRT())
{
GetRecycler()->RootRelease(globalObject);
}
}
// A script context closing is a signal to the thread context that it
// needs to do an idle GC independent of what the heuristics are
this->threadContext->SetForceOneIdleCollection();
return true;
}
PropertyString* ScriptContext::GetPropertyString2(char16 ch1, char16 ch2)
{
if (ch1 < '0' || ch1 > 'z' || ch2 < '0' || ch2 > 'z')
{
return NULL;
}
const uint i = PropertyStringMap::PStrMapIndex(ch1);
if (propertyStrings[i] == NULL)
{
return NULL;
}
const uint j = PropertyStringMap::PStrMapIndex(ch2);
return propertyStrings[i]->strLen2[j];
}
void ScriptContext::FindPropertyRecord(JavascriptString *pstName, PropertyRecord const ** propertyRecord)
{
threadContext->FindPropertyRecord(pstName, propertyRecord);
}
void ScriptContext::FindPropertyRecord(__in LPCWSTR propertyName, __in int propertyNameLength, PropertyRecord const ** propertyRecord)
{
threadContext->FindPropertyRecord(propertyName, propertyNameLength, propertyRecord);
}
JsUtil::List<const RecyclerWeakReference<Js::PropertyRecord const>*>* ScriptContext::FindPropertyIdNoCase(__in LPCWSTR propertyName, __in int propertyNameLength)
{
return threadContext->FindPropertyIdNoCase(this, propertyName, propertyNameLength);
}
PropertyId ScriptContext::GetOrAddPropertyIdTracked(JsUtil::CharacterBuffer<WCHAR> const& propName)
{
Js::PropertyRecord const * propertyRecord;
threadContext->GetOrAddPropertyId(propName, &propertyRecord);
this->TrackPid(propertyRecord);
return propertyRecord->GetPropertyId();
}
void ScriptContext::GetOrAddPropertyRecord(JsUtil::CharacterBuffer<WCHAR> const& propertyName, PropertyRecord const ** propertyRecord)
{
threadContext->GetOrAddPropertyId(propertyName, propertyRecord);
}
PropertyId ScriptContext::GetOrAddPropertyIdTracked(__in_ecount(propertyNameLength) LPCWSTR propertyName, __in int propertyNameLength)
{
Js::PropertyRecord const * propertyRecord;
threadContext->GetOrAddPropertyId(propertyName, propertyNameLength, &propertyRecord);
if (propertyNameLength == 2)
{
CachePropertyString2(propertyRecord);
}
this->TrackPid(propertyRecord);
return propertyRecord->GetPropertyId();
}
void ScriptContext::GetOrAddPropertyRecord(__in_ecount(propertyNameLength) LPCWSTR propertyName, __in int propertyNameLength, PropertyRecord const ** propertyRecord)
{
threadContext->GetOrAddPropertyId(propertyName, propertyNameLength, propertyRecord);
if (propertyNameLength == 2)
{
CachePropertyString2(*propertyRecord);
}
}
BOOL ScriptContext::IsNumericPropertyId(PropertyId propertyId, uint32* value)
{
BOOL isNumericPropertyId = threadContext->IsNumericPropertyId(propertyId, value);
#if DEBUG
PropertyRecord const * name = this->GetPropertyName(propertyId);
if (name != nullptr)
{
// Symbol properties are not numeric - description should not be used.
if (name->IsSymbol())
{
return false;
}
uint32 index;
BOOL isIndex = JavascriptArray::GetIndex(name->GetBuffer(), &index);
if (isNumericPropertyId != isIndex)
{
// WOOB 1137798: JavascriptArray::GetIndex does not handle embedded NULLs. So if we have a property
// name "1234\0", JavascriptArray::GetIndex would incorrectly accepts it as an array index property
// name.
Assert((size_t)(name->GetLength()) != wcslen(name->GetBuffer()));
}
else if (isNumericPropertyId)
{
Assert((uint32)*value == index);
}
}
#endif
return isNumericPropertyId;
}
void ScriptContext::RegisterWeakReferenceDictionary(JsUtil::IWeakReferenceDictionary* weakReferenceDictionary)
{
this->weakReferenceDictionaryList.Prepend(this->GeneralAllocator(), weakReferenceDictionary);
}
RecyclableObject *ScriptContext::GetMissingPropertyResult()
{
return GetLibrary()->GetUndefined();
}
RecyclableObject *ScriptContext::GetMissingItemResult()
{
return GetLibrary()->GetUndefined();
}
SRCINFO *ScriptContext::AddHostSrcInfo(SRCINFO const *pSrcInfo)
{
Assert(pSrcInfo != nullptr);
return RecyclerNewZ(this->GetRecycler(), SRCINFO, *pSrcInfo);
}
#ifdef PROFILE_TYPES
void ScriptContext::ProfileTypes()
{
Output::Print(_u("===============================================================================\n"));
Output::Print(_u("Types Profile\n"));
Output::Print(_u("-------------------------------------------------------------------------------\n"));
Output::Print(_u("Dynamic Type Conversions:\n"));
Output::Print(_u(" Null to Simple %8d\n"), convertNullToSimpleCount);
Output::Print(_u(" Deferred to SimpleMap %8d\n"), convertDeferredToSimpleDictionaryCount);
Output::Print(_u(" Simple to Map %8d\n"), convertSimpleToDictionaryCount);
Output::Print(_u(" Simple to SimpleMap %8d\n"), convertSimpleToSimpleDictionaryCount);
Output::Print(_u(" Path to SimpleMap (set) %8d\n"), convertPathToDictionaryCount1);
Output::Print(_u(" Path to SimpleMap (delete) %8d\n"), convertPathToDictionaryCount2);
Output::Print(_u(" Path to SimpleMap (attribute) %8d\n"), convertPathToDictionaryCount3);
Output::Print(_u(" Path to SimpleMap %8d\n"), convertPathToSimpleDictionaryCount);
Output::Print(_u(" SimplePath to Path %8d\n"), convertSimplePathToPathCount);
Output::Print(_u(" Shared SimpleMap to non-shared %8d\n"), convertSimpleSharedDictionaryToNonSharedCount);
Output::Print(_u(" Deferred to Map %8d\n"), convertDeferredToDictionaryCount);
Output::Print(_u(" Path to Map (accessor) %8d\n"), convertPathToDictionaryCount4);
Output::Print(_u(" SimpleMap to Map %8d\n"), convertSimpleDictionaryToDictionaryCount);
Output::Print(_u(" Path Cache Hits %8d\n"), cacheCount);
Output::Print(_u(" Path Branches %8d\n"), branchCount);
Output::Print(_u(" Path Promotions %8d\n"), promoteCount);
Output::Print(_u(" Path Length (max) %8d\n"), maxPathLength);
Output::Print(_u(" SimplePathTypeHandlers %8d\n"), simplePathTypeHandlerCount);
Output::Print(_u(" PathTypeHandlers %8d\n"), pathTypeHandlerCount);
Output::Print(_u("\n"));
Output::Print(_u("Type Statistics: %8s %8s\n"), _u("Types"), _u("Instances"));
Output::Print(_u(" Undefined %8d %8d\n"), typeCount[TypeIds_Undefined], instanceCount[TypeIds_Undefined]);
Output::Print(_u(" Null %8d %8d\n"), typeCount[TypeIds_Null], instanceCount[TypeIds_Null]);
Output::Print(_u(" Boolean %8d %8d\n"), typeCount[TypeIds_Boolean], instanceCount[TypeIds_Boolean]);
Output::Print(_u(" Integer %8d %8d\n"), typeCount[TypeIds_Integer], instanceCount[TypeIds_Integer]);
Output::Print(_u(" Number %8d %8d\n"), typeCount[TypeIds_Number], instanceCount[TypeIds_Number]);
Output::Print(_u(" String %8d %8d\n"), typeCount[TypeIds_String], instanceCount[TypeIds_String]);
Output::Print(_u(" Object %8d %8d\n"), typeCount[TypeIds_Object], instanceCount[TypeIds_Object]);
Output::Print(_u(" Function %8d %8d\n"), typeCount[TypeIds_Function], instanceCount[TypeIds_Function]);
Output::Print(_u(" Array %8d %8d\n"), typeCount[TypeIds_Array], instanceCount[TypeIds_Array]);
Output::Print(_u(" Date %8d %8d\n"), typeCount[TypeIds_Date], instanceCount[TypeIds_Date] + instanceCount[TypeIds_WinRTDate]);
Output::Print(_u(" Symbol %8d %8d\n"), typeCount[TypeIds_Symbol], instanceCount[TypeIds_Symbol]);
Output::Print(_u(" RegEx %8d %8d\n"), typeCount[TypeIds_RegEx], instanceCount[TypeIds_RegEx]);
Output::Print(_u(" Error %8d %8d\n"), typeCount[TypeIds_Error], instanceCount[TypeIds_Error]);
Output::Print(_u(" Proxy %8d %8d\n"), typeCount[TypeIds_Proxy], instanceCount[TypeIds_Proxy]);
Output::Print(_u(" BooleanObject %8d %8d\n"), typeCount[TypeIds_BooleanObject], instanceCount[TypeIds_BooleanObject]);
Output::Print(_u(" NumberObject %8d %8d\n"), typeCount[TypeIds_NumberObject], instanceCount[TypeIds_NumberObject]);
Output::Print(_u(" StringObject %8d %8d\n"), typeCount[TypeIds_StringObject], instanceCount[TypeIds_StringObject]);
Output::Print(_u(" SymbolObject %8d %8d\n"), typeCount[TypeIds_SymbolObject], instanceCount[TypeIds_SymbolObject]);
Output::Print(_u(" GlobalObject %8d %8d\n"), typeCount[TypeIds_GlobalObject], instanceCount[TypeIds_GlobalObject]);
Output::Print(_u(" Enumerator %8d %8d\n"), typeCount[TypeIds_Enumerator], instanceCount[TypeIds_Enumerator]);
Output::Print(_u(" Int8Array %8d %8d\n"), typeCount[TypeIds_Int8Array], instanceCount[TypeIds_Int8Array]);
Output::Print(_u(" Uint8Array %8d %8d\n"), typeCount[TypeIds_Uint8Array], instanceCount[TypeIds_Uint8Array]);
Output::Print(_u(" Uint8ClampedArray %8d %8d\n"), typeCount[TypeIds_Uint8ClampedArray], instanceCount[TypeIds_Uint8ClampedArray]);
Output::Print(_u(" Int16Array %8d %8d\n"), typeCount[TypeIds_Int16Array], instanceCount[TypeIds_Int16Array]);
Output::Print(_u(" Int16Array %8d %8d\n"), typeCount[TypeIds_Uint16Array], instanceCount[TypeIds_Uint16Array]);
Output::Print(_u(" Int32Array %8d %8d\n"), typeCount[TypeIds_Int32Array], instanceCount[TypeIds_Int32Array]);
Output::Print(_u(" Uint32Array %8d %8d\n"), typeCount[TypeIds_Uint32Array], instanceCount[TypeIds_Uint32Array]);
Output::Print(_u(" Float32Array %8d %8d\n"), typeCount[TypeIds_Float32Array], instanceCount[TypeIds_Float32Array]);
Output::Print(_u(" Float64Array %8d %8d\n"), typeCount[TypeIds_Float64Array], instanceCount[TypeIds_Float64Array]);
Output::Print(_u(" DataView %8d %8d\n"), typeCount[TypeIds_DataView], instanceCount[TypeIds_DataView]);
Output::Print(_u(" ModuleRoot %8d %8d\n"), typeCount[TypeIds_ModuleRoot], instanceCount[TypeIds_ModuleRoot]);
Output::Print(_u(" HostObject %8d %8d\n"), typeCount[TypeIds_HostObject], instanceCount[TypeIds_HostObject]);
Output::Print(_u(" VariantDate %8d %8d\n"), typeCount[TypeIds_VariantDate], instanceCount[TypeIds_VariantDate]);
Output::Print(_u(" HostDispatch %8d %8d\n"), typeCount[TypeIds_HostDispatch], instanceCount[TypeIds_HostDispatch]);
Output::Print(_u(" Arguments %8d %8d\n"), typeCount[TypeIds_Arguments], instanceCount[TypeIds_Arguments]);
Output::Print(_u(" ActivationObject %8d %8d\n"), typeCount[TypeIds_ActivationObject], instanceCount[TypeIds_ActivationObject]);
Output::Print(_u(" Map %8d %8d\n"), typeCount[TypeIds_Map], instanceCount[TypeIds_Map]);
Output::Print(_u(" Set %8d %8d\n"), typeCount[TypeIds_Set], instanceCount[TypeIds_Set]);
Output::Print(_u(" WeakMap %8d %8d\n"), typeCount[TypeIds_WeakMap], instanceCount[TypeIds_WeakMap]);
Output::Print(_u(" WeakSet %8d %8d\n"), typeCount[TypeIds_WeakSet], instanceCount[TypeIds_WeakSet]);
Output::Print(_u(" ArrayIterator %8d %8d\n"), typeCount[TypeIds_ArrayIterator], instanceCount[TypeIds_ArrayIterator]);
Output::Print(_u(" MapIterator %8d %8d\n"), typeCount[TypeIds_MapIterator], instanceCount[TypeIds_MapIterator]);
Output::Print(_u(" SetIterator %8d %8d\n"), typeCount[TypeIds_SetIterator], instanceCount[TypeIds_SetIterator]);
Output::Print(_u(" StringIterator %8d %8d\n"), typeCount[TypeIds_StringIterator], instanceCount[TypeIds_StringIterator]);
Output::Print(_u(" Generator %8d %8d\n"), typeCount[TypeIds_Generator], instanceCount[TypeIds_Generator]);
#if !DBG
Output::Print(_u(" ** Instance statistics only available on debug builds...\n"));
#endif
Output::Flush();
}
#endif
#ifdef PROFILE_OBJECT_LITERALS
void ScriptContext::ProfileObjectLiteral()
{
Output::Print(_u("===============================================================================\n"));
Output::Print(_u(" Object Lit Instances created.. %d\n"), objectLiteralInstanceCount);
Output::Print(_u(" Object Lit Path Types......... %d\n"), objectLiteralPathCount);
Output::Print(_u(" Object Lit Simple Map......... %d\n"), objectLiteralSimpleDictionaryCount);
Output::Print(_u(" Object Lit Max # of properties %d\n"), objectLiteralMaxLength);
Output::Print(_u(" Object Lit Promote count...... %d\n"), objectLiteralPromoteCount);
Output::Print(_u(" Object Lit Cache Hits......... %d\n"), objectLiteralCacheCount);
Output::Print(_u(" Object Lit Branch count....... %d\n"), objectLiteralBranchCount);
for (int i = 0; i < TypePath::MaxPathTypeHandlerLength; i++)
{
if (objectLiteralCount[i] != 0)
{
Output::Print(_u(" Object Lit properties [ %2d] .. %d\n"), i, objectLiteralCount[i]);
}
}
Output::Flush();
}
#endif
//
// Regex helpers
//
#if ENABLE_REGEX_CONFIG_OPTIONS
UnifiedRegex::RegexStatsDatabase* ScriptContext::GetRegexStatsDatabase()
{
if (regexStatsDatabase == 0)
{
ArenaAllocator* allocator = MiscAllocator();
regexStatsDatabase = Anew(allocator, UnifiedRegex::RegexStatsDatabase, allocator);
}
return regexStatsDatabase;
}
UnifiedRegex::DebugWriter* ScriptContext::GetRegexDebugWriter()
{
if (regexDebugWriter == 0)
{
ArenaAllocator* allocator = MiscAllocator();
regexDebugWriter = Anew(allocator, UnifiedRegex::DebugWriter);
}
return regexDebugWriter;
}
#endif
bool ScriptContext::DoUndeferGlobalFunctions() const
{
return CONFIG_FLAG(DeferTopLevelTillFirstCall) && !AutoSystemInfo::Data.IsLowMemoryProcess();
}
RegexPatternMruMap* ScriptContext::GetDynamicRegexMap() const
{
Assert(!isScriptContextActuallyClosed);
Assert(cache);
Assert(cache->dynamicRegexMap);
return cache->dynamicRegexMap;
}
void ScriptContext::SetTrigramAlphabet(UnifiedRegex::TrigramAlphabet * trigramAlphabet)
{
this->trigramAlphabet = trigramAlphabet;
}
UnifiedRegex::RegexStacks *ScriptContext::RegexStacks()
{
UnifiedRegex::RegexStacks * stacks = regexStacks;
if (stacks)
{
return stacks;
}
return AllocRegexStacks();
}
UnifiedRegex::RegexStacks * ScriptContext::AllocRegexStacks()
{
Assert(this->regexStacks == nullptr);
UnifiedRegex::RegexStacks * stacks = Anew(RegexAllocator(), UnifiedRegex::RegexStacks, threadContext->GetPageAllocator());
this->regexStacks = stacks;
return stacks;
}
UnifiedRegex::RegexStacks *ScriptContext::SaveRegexStacks()
{
Assert(regexStacks);
const auto saved = regexStacks;
regexStacks = nullptr;
return saved;
}
void ScriptContext::RestoreRegexStacks(UnifiedRegex::RegexStacks *const stacks)
{
Assert(stacks);
Assert(stacks != regexStacks);
if (regexStacks)
{
Adelete(RegexAllocator(), regexStacks);
}
regexStacks = stacks;
}
Js::TempArenaAllocatorObject* ScriptContext::GetTemporaryAllocator(LPCWSTR name)
{
return this->threadContext->GetTemporaryAllocator(name);
}
void ScriptContext::ReleaseTemporaryAllocator(Js::TempArenaAllocatorObject* tempAllocator)
{
AssertMsg(tempAllocator != nullptr, "tempAllocator should not be null");
this->threadContext->ReleaseTemporaryAllocator(tempAllocator);
}
Js::TempGuestArenaAllocatorObject* ScriptContext::GetTemporaryGuestAllocator(LPCWSTR name)
{
return this->threadContext->GetTemporaryGuestAllocator(name);
}
void ScriptContext::ReleaseTemporaryGuestAllocator(Js::TempGuestArenaAllocatorObject* tempGuestAllocator)
{
AssertMsg(tempGuestAllocator != nullptr, "tempAllocator should not be null");
this->threadContext->ReleaseTemporaryGuestAllocator(tempGuestAllocator);
}
void ScriptContext::InitializeCache()
{
this->cache = RecyclerNewFinalized(recycler, Cache);
this->javascriptLibrary->scriptContextCache = this->cache;
this->cache->dynamicRegexMap =
RegexPatternMruMap::New(
recycler,
REGEX_CONFIG_FLAG(DynamicRegexMruListSize) <= 0 ? 16 : REGEX_CONFIG_FLAG(DynamicRegexMruListSize));
SourceContextInfo* sourceContextInfo = RecyclerNewStructZ(this->GetRecycler(), SourceContextInfo);
sourceContextInfo->dwHostSourceContext = Js::Constants::NoHostSourceContext;
sourceContextInfo->isHostDynamicDocument = false;
sourceContextInfo->sourceContextId = Js::Constants::NoSourceContext;
this->cache->noContextSourceContextInfo = sourceContextInfo;
SRCINFO* srcInfo = RecyclerNewStructZ(this->GetRecycler(), SRCINFO);
srcInfo->sourceContextInfo = this->cache->noContextSourceContextInfo;
srcInfo->moduleID = kmodGlobal;
this->cache->noContextGlobalSourceInfo = srcInfo;
}
void ScriptContext::InitializePreGlobal()
{
this->guestArena = this->GetRecycler()->CreateGuestArena(_u("Guest"), Throw::OutOfMemory);
#if ENABLE_PROFILE_INFO
#if DBG_DUMP || defined(DYNAMIC_PROFILE_STORAGE) || defined(RUNTIME_DATA_COLLECTION)
if (DynamicProfileInfo::NeedProfileInfoList())
{
this->profileInfoList.Root(RecyclerNew(this->GetRecycler(), SListBase<DynamicProfileInfo *>), recycler);
}
#endif
#endif
#if ENABLE_BACKGROUND_PARSING
if (PHASE_ON1(Js::ParallelParsePhase))
{
this->backgroundParser = BackgroundParser::New(this);
}
#endif
#if ENABLE_NATIVE_CODEGEN
// Create the native code gen before the profiler
this->nativeCodeGen = NewNativeCodeGenerator(this);
#endif
#ifdef PROFILE_EXEC
this->CreateProfiler();
#endif
this->operationStack = Anew(GeneralAllocator(), JsUtil::Stack<Var>, GeneralAllocator());
Tick::InitType();
}
void ScriptContext::Initialize()
{
SmartFPUControl defaultControl;
InitializePreGlobal();
InitializeGlobalObject();
InitializePostGlobal();
}
void ScriptContext::InitializePostGlobal()
{
this->GetDebugContext()->Initialize();
this->GetDebugContext()->GetProbeContainer()->Initialize(this);
AssertMsg(this->CurrentThunk == DefaultEntryThunk, "Creating non default thunk while initializing");
AssertMsg(this->DeferredParsingThunk == DefaultDeferredParsingThunk, "Creating non default thunk while initializing");
AssertMsg(this->DeferredDeserializationThunk == DefaultDeferredDeserializeThunk, "Creating non default thunk while initializing");
#ifdef FIELD_ACCESS_STATS
this->fieldAccessStatsByFunctionNumber = RecyclerNew(this->recycler, FieldAccessStatsByFunctionNumberMap, recycler);
BindReference(this->fieldAccessStatsByFunctionNumber);
#endif
if (!sourceList)
{
AutoCriticalSection critSec(threadContext->GetEtwRundownCriticalSection());
sourceList.Root(RecyclerNew(this->GetRecycler(), SourceList, this->GetRecycler()), this->GetRecycler());
}
#if DYNAMIC_INTERPRETER_THUNK
interpreterThunkEmitter = HeapNew(InterpreterThunkEmitter, SourceCodeAllocator(), this->GetThreadContext()->GetThunkPageAllocators());
#endif
#ifdef ASMJS_PLAT
asmJsInterpreterThunkEmitter = HeapNew(InterpreterThunkEmitter, SourceCodeAllocator(), this->GetThreadContext()->GetThunkPageAllocators(),
true);
#endif
JS_ETW(EtwTrace::LogScriptContextLoadEvent(this));
JS_ETW(EventWriteJSCRIPT_HOST_SCRIPT_CONTEXT_START(this));
#ifdef PROFILE_EXEC
if (profiler != nullptr)
{
this->threadContext->GetRecycler()->SetProfiler(profiler->GetProfiler(), profiler->GetBackgroundRecyclerProfiler());
}
#endif
#if DBG
this->javascriptLibrary->DumpLibraryByteCode();
isInitialized = TRUE;
#endif
}
#ifdef ASMJS_PLAT
AsmJsCodeGenerator* ScriptContext::InitAsmJsCodeGenerator()
{
if( !asmJsCodeGenerator )
{
asmJsCodeGenerator = HeapNew( AsmJsCodeGenerator, this );
}
return asmJsCodeGenerator;
}
#endif
void ScriptContext::MarkForClose()
{
SaveStartupProfileAndRelease(true);
this->isClosed = true;
#ifdef LEAK_REPORT
if (this->isRootTrackerScriptContext)
{
this->GetThreadContext()->ClearRootTrackerScriptContext(this);
}
#endif
if (!threadContext->IsInScript())
{
Close(FALSE);
}
else
{
threadContext->AddToPendingScriptContextCloseList(this);
}
}
void ScriptContext::InitializeGlobalObject()
{
GlobalObject * localGlobalObject = GlobalObject::New(this);
GetRecycler()->RootAddRef(localGlobalObject);
// Assigned the global Object after we have successfully AddRef (in case of OOM)
globalObject = localGlobalObject;
globalObject->Initialize(this);
}
ArenaAllocator* ScriptContext::AllocatorForDiagnostics()
{
if (this->diagnosticArena == nullptr)
{
this->diagnosticArena = HeapNew(ArenaAllocator, _u("Diagnostic"), this->GetThreadContext()->GetDebugManager()->GetDiagnosticPageAllocator(), Throw::OutOfMemory);
}
Assert(this->diagnosticArena != nullptr);
return this->diagnosticArena;
}
void ScriptContext::PushObject(Var object)
{
operationStack->Push(object);
}
Var ScriptContext::PopObject()
{
return operationStack->Pop();
}
BOOL ScriptContext::CheckObject(Var object)
{
return operationStack->Contains(object);
}
void ScriptContext::SetHostScriptContext(HostScriptContext * hostScriptContext)
{
Assert(this->hostScriptContext == nullptr);
this->hostScriptContext = hostScriptContext;
#ifdef PROFILE_EXEC
this->ensureParentInfo = true;
#endif
}
//
// Enables chakradiag to get the HaltCallBack pointer that is implemented by
// the ScriptEngine.
//
void ScriptContext::SetScriptEngineHaltCallback(HaltCallback* scriptEngine)
{
Assert(this->scriptEngineHaltCallback == NULL);
Assert(scriptEngine != NULL);
this->scriptEngineHaltCallback = scriptEngine;
}
void ScriptContext::ClearHostScriptContext()
{
if (this->hostScriptContext != nullptr)
{
this->hostScriptContext->Delete();
#ifdef PROFILE_EXEC
this->ensureParentInfo = false;
#endif
}
}
IActiveScriptProfilerHeapEnum* ScriptContext::GetHeapEnum()
{
Assert(this->GetThreadContext());
return this->GetThreadContext()->GetHeapEnum();
}
void ScriptContext::SetHeapEnum(IActiveScriptProfilerHeapEnum* newHeapEnum)
{
Assert(this->GetThreadContext());
this->GetThreadContext()->SetHeapEnum(newHeapEnum);
}
void ScriptContext::ClearHeapEnum()
{
Assert(this->GetThreadContext());
this->GetThreadContext()->ClearHeapEnum();
}
BOOL ScriptContext::VerifyAlive(BOOL isJSFunction, ScriptContext* requestScriptContext)
{
if (isClosed)
{
if (!requestScriptContext)
{
requestScriptContext = this;
}
#if ENABLE_PROFILE_INFO
if (!GetThreadContext()->RecordImplicitException())
{
return FALSE;
}
#endif
if (isJSFunction)
{
Js::JavascriptError::MapAndThrowError(requestScriptContext, JSERR_CantExecute);
}
else
{
Js::JavascriptError::MapAndThrowError(requestScriptContext, E_ACCESSDENIED);
}
}
return TRUE;
}
void ScriptContext::VerifyAliveWithHostContext(BOOL isJSFunction, HostScriptContext* requestHostScriptContext)
{
if (requestHostScriptContext)
{
VerifyAlive(isJSFunction, requestHostScriptContext->GetScriptContext());
}
else
{
Assert(GetThreadContext()->IsJSRT() || !GetHostScriptContext()->HasCaller());
VerifyAlive(isJSFunction, NULL);
}
}
PropertyRecord const * ScriptContext::GetPropertyName(PropertyId propertyId)
{
return threadContext->GetPropertyName(propertyId);
}
PropertyRecord const * ScriptContext::GetPropertyNameLocked(PropertyId propertyId)
{
return threadContext->GetPropertyNameLocked(propertyId);
}
void ScriptContext::InitPropertyStringMap(int i)
{
propertyStrings[i] = AnewStruct(GeneralAllocator(), PropertyStringMap);
memset(propertyStrings[i]->strLen2, 0, sizeof(PropertyString*)* 80);
}
void ScriptContext::TrackPid(const PropertyRecord* propertyRecord)
{
if (IsBuiltInPropertyId(propertyRecord->GetPropertyId()) || propertyRecord->IsBound())
{
return;
}
if (-1 != this->GetLibrary()->EnsureReferencedPropertyRecordList()->AddNew(propertyRecord))
{
RECYCLER_PERF_COUNTER_INC(PropertyRecordBindReference);
}
}
void ScriptContext::TrackPid(PropertyId propertyId)
{
if (IsBuiltInPropertyId(propertyId))
{
return;
}
const PropertyRecord* propertyRecord = this->GetPropertyName(propertyId);
Assert(propertyRecord != nullptr);
this->TrackPid(propertyRecord);
}
bool ScriptContext::IsTrackedPropertyId(Js::PropertyId propertyId)
{
if (IsBuiltInPropertyId(propertyId))
{
return true;
}
const PropertyRecord* propertyRecord = this->GetPropertyName(propertyId);
Assert(propertyRecord != nullptr);
if (propertyRecord->IsBound())
{
return true;
}
JavascriptLibrary::ReferencedPropertyRecordHashSet * referencedPropertyRecords
= this->GetLibrary()->GetReferencedPropertyRecordList();
return referencedPropertyRecords && referencedPropertyRecords->Contains(propertyRecord);
}
PropertyString* ScriptContext::AddPropertyString2(const Js::PropertyRecord* propString)
{
const char16* buf = propString->GetBuffer();
const uint i = PropertyStringMap::PStrMapIndex(buf[0]);
if (propertyStrings[i] == NULL)
{
InitPropertyStringMap(i);
}
const uint j = PropertyStringMap::PStrMapIndex(buf[1]);
if (propertyStrings[i]->strLen2[j] == NULL && !isClosed)
{
propertyStrings[i]->strLen2[j] = GetLibrary()->CreatePropertyString(propString, this->GeneralAllocator());
this->TrackPid(propString);
}
return propertyStrings[i]->strLen2[j];
}
PropertyString* ScriptContext::CachePropertyString2(const PropertyRecord* propString)
{
Assert(propString->GetLength() == 2);
const char16* propertyName = propString->GetBuffer();
if ((propertyName[0] <= 'z') && (propertyName[1] <= 'z') && (propertyName[0] >= '0') && (propertyName[1] >= '0') && ((propertyName[0] > '9') || (propertyName[1] > '9')))
{
return AddPropertyString2(propString);
}
return NULL;
}
PropertyString* ScriptContext::TryGetPropertyString(PropertyId propertyId)
{
PropertyStringCacheMap* propertyStringMap = this->GetLibrary()->EnsurePropertyStringMap();
RecyclerWeakReference<PropertyString>* stringReference;
if (propertyStringMap->TryGetValue(propertyId, &stringReference))
{
PropertyString *string = stringReference->Get();
if (string != nullptr)
{
return string;
}
}
return nullptr;
}
PropertyString* ScriptContext::GetPropertyString(PropertyId propertyId)
{
PropertyString *string = TryGetPropertyString(propertyId);
if (string != nullptr)
{
return string;
}
PropertyStringCacheMap* propertyStringMap = this->GetLibrary()->EnsurePropertyStringMap();
const Js::PropertyRecord* propertyName = this->GetPropertyName(propertyId);
string = this->GetLibrary()->CreatePropertyString(propertyName);
propertyStringMap->Item(propertyId, recycler->CreateWeakReferenceHandle(string));
return string;
}
void ScriptContext::InvalidatePropertyStringCache(PropertyId propertyId, Type* type)
{
Assert(!isFinalized);
PropertyStringCacheMap* propertyStringMap = this->javascriptLibrary->GetPropertyStringMap();
if (propertyStringMap != nullptr)
{
PropertyString *string = nullptr;
RecyclerWeakReference<PropertyString>* stringReference;
if (propertyStringMap->TryGetValue(propertyId, &stringReference))
{
string = stringReference->Get();
}
if (string)
{
PropertyCache const* cache = string->GetPropertyCache();
if (cache->type == type)
{
string->ClearPropertyCache();
}
}
}
}
void ScriptContext::CleanupWeakReferenceDictionaries()
{
if (!isWeakReferenceDictionaryListCleared)
{
SListBase<JsUtil::IWeakReferenceDictionary*>::Iterator iter(&this->weakReferenceDictionaryList);
while (iter.Next())
{
JsUtil::IWeakReferenceDictionary* weakReferenceDictionary = iter.Data();
weakReferenceDictionary->Cleanup();
}
}
}
JavascriptString* ScriptContext::GetIntegerString(Var aValue)
{
return this->GetIntegerString(TaggedInt::ToInt32(aValue));
}
JavascriptString* ScriptContext::GetIntegerString(uint value)
{
if (value <= INT_MAX)
{
return this->GetIntegerString((int)value);
}
return TaggedInt::ToString(value, this);
}
JavascriptString* ScriptContext::GetIntegerString(int value)
{
// Optimize for 0-9
if (0 <= value && value <= 9)
{
return GetLibrary()->GetCharStringCache().GetStringForCharA('0' + static_cast<char>(value));
}
JavascriptString *string;
if (!this->integerStringMap.TryGetValue(value, &string))
{
// Add the string to hash table cache
// Don't add if table is getting too full. We'll be holding on to
// too many strings, and table lookup will become too slow.
if (this->integerStringMap.Count() > 1024)
{
// Use recycler memory
string = TaggedInt::ToString(value, this);
}
else
{
char16 stringBuffer[20];
TaggedInt::ToBuffer(value, stringBuffer, _countof(stringBuffer));
string = JavascriptString::NewCopySzFromArena(stringBuffer, this, this->GeneralAllocator());
this->integerStringMap.AddNew(value, string);
}
}
return string;
}
void ScriptContext::CheckEvalRestriction()
{
HRESULT hr = S_OK;
Var domError = nullptr;
HostScriptContext* hostScriptContext = this->GetHostScriptContext();
BEGIN_LEAVE_SCRIPT(this)
{
if (!FAILED(hr = hostScriptContext->CheckEvalRestriction()))
{
return;
}
hr = hostScriptContext->HostExceptionFromHRESULT(hr, &domError);
}
END_LEAVE_SCRIPT(this);
if (FAILED(hr))
{
Js::JavascriptError::MapAndThrowError(this, hr);
}
if (domError != nullptr)
{
JavascriptExceptionOperators::Throw(domError, this);
}
AssertMsg(false, "We should have thrown by now.");
Js::JavascriptError::MapAndThrowError(this, E_FAIL);
}
ParseNode* ScriptContext::ParseScript(Parser* parser,
const byte* script,
size_t cb,
SRCINFO const * pSrcInfo,
CompileScriptException * pse,
Utf8SourceInfo** ppSourceInfo,
const char16 *rootDisplayName,
LoadScriptFlag loadScriptFlag,
uint* sourceIndex)
{
if (pSrcInfo == nullptr)
{
pSrcInfo = this->cache->noContextGlobalSourceInfo;
}
LPUTF8 utf8Script = nullptr;
size_t length = cb;
size_t cbNeeded = 0;
bool isLibraryCode = ((loadScriptFlag & LoadScriptFlag_LibraryCode) == LoadScriptFlag_LibraryCode);
if ((loadScriptFlag & LoadScriptFlag_Utf8Source) != LoadScriptFlag_Utf8Source)
{
// Convert to UTF8 and then load that
length = cb / sizeof(char16);
if (!IsValidCharCount(length))
{
Js::Throw::OutOfMemory();
}
Assert(length < MAXLONG);
// Allocate memory for the UTF8 output buffer.
// We need at most 3 bytes for each Unicode code point.
// The + 1 is to include the terminating NUL.
// Nit: Technically, we know that the NUL only needs 1 byte instead of
// 3, but that's difficult to express in a SAL annotation for "EncodeInto".
size_t cbUtf8Buffer = AllocSizeMath::Mul(AllocSizeMath::Add(length, 1), 3);
utf8Script = RecyclerNewArrayLeafTrace(this->GetRecycler(), utf8char_t, cbUtf8Buffer);
cbNeeded = utf8::EncodeIntoAndNullTerminate(utf8Script, (const char16*)script, static_cast<charcount_t>(length));
#if DBG_DUMP
if(Js::Configuration::Global.flags.TraceMemory.IsEnabled(Js::ParsePhase) && Configuration::Global.flags.Verbose)
{
Output::Print(_u("Loading script.\n")
_u(" Unicode (in bytes) %u\n")
_u(" UTF-8 size (in bytes) %u\n")
_u(" Expected savings %d\n"), length * sizeof(char16), cbNeeded, length * sizeof(char16) - cbNeeded);
}
#endif
// Free unused bytes
Assert(cbNeeded + 1 <= cbUtf8Buffer);
*ppSourceInfo = Utf8SourceInfo::New(this, utf8Script, (int)length, cbNeeded, pSrcInfo, isLibraryCode);
}
else
{
// We do not own the memory passed into DefaultLoadScriptUtf8. We need to save it so we copy the memory.
if(*ppSourceInfo == nullptr)
{
// the 'length' here is not correct - we will get the length from the parser - however parser hasn't done yet.
// Once the parser is done we will update the utf8sourceinfo's lenght correctly with parser's
*ppSourceInfo = Utf8SourceInfo::New(this, script, (int)length, cb, pSrcInfo, isLibraryCode);
}
}
//
// Parse and the JavaScript code
//
HRESULT hr;
SourceContextInfo * sourceContextInfo = pSrcInfo->sourceContextInfo;
// Invoke the parser, passing in the global function name, which we will then run to execute
// the script.
// TODO: yongqu handle non-global code.
ULONG grfscr = fscrGlobalCode | ((loadScriptFlag & LoadScriptFlag_Expression) == LoadScriptFlag_Expression ? fscrReturnExpression : 0);
if(((loadScriptFlag & LoadScriptFlag_disableDeferredParse) != LoadScriptFlag_disableDeferredParse) &&
(length > Parser::GetDeferralThreshold(sourceContextInfo->IsSourceProfileLoaded())))
{
grfscr |= fscrDeferFncParse;
}
if((loadScriptFlag & LoadScriptFlag_disableAsmJs) == LoadScriptFlag_disableAsmJs)
{
grfscr |= fscrNoAsmJs;
}
if(PHASE_FORCE1(Js::EvalCompilePhase))
{
// pretend it is eval
grfscr |= (fscrEval | fscrEvalCode);
}
if((loadScriptFlag & LoadScriptFlag_isByteCodeBufferForLibrary) == LoadScriptFlag_isByteCodeBufferForLibrary)
{
grfscr |= (fscrNoAsmJs | fscrNoPreJit);
}
if(((loadScriptFlag & LoadScriptFlag_Module) == LoadScriptFlag_Module) &&
GetConfig()->IsES6ModuleEnabled())
{
grfscr |= fscrIsModuleCode;
}
if (isLibraryCode)
{
grfscr |= fscrIsLibraryCode;
}
ParseNodePtr parseTree;
if((loadScriptFlag & LoadScriptFlag_Utf8Source) == LoadScriptFlag_Utf8Source)
{
hr = parser->ParseUtf8Source(&parseTree, script, cb, grfscr, pse, &sourceContextInfo->nextLocalFunctionId, sourceContextInfo);
}
else
{
hr = parser->ParseCesu8Source(&parseTree, utf8Script, cbNeeded, grfscr, pse, &sourceContextInfo->nextLocalFunctionId, sourceContextInfo);
}
if(FAILED(hr) || parseTree == nullptr)
{
return nullptr;
}
(*ppSourceInfo)->SetParseFlags(grfscr);
//Make sure we have the body and text information available
if ((loadScriptFlag & LoadScriptFlag_Utf8Source) != LoadScriptFlag_Utf8Source)
{
*sourceIndex = this->SaveSourceNoCopy(*ppSourceInfo, static_cast<charcount_t>((*ppSourceInfo)->GetCchLength()), /*isCesu8*/ true);
}
else
{
// Update the length.
(*ppSourceInfo)->SetCchLength(parser->GetSourceIchLim());
*sourceIndex = this->SaveSourceNoCopy(*ppSourceInfo, parser->GetSourceIchLim(), /* isCesu8*/ false);
}
return parseTree;
}
JavascriptFunction* ScriptContext::LoadScript(const byte* script, size_t cb, SRCINFO const * pSrcInfo, CompileScriptException * pse, Utf8SourceInfo** ppSourceInfo, const char16 *rootDisplayName, LoadScriptFlag loadScriptFlag)
{
Assert(!this->threadContext->IsScriptActive());
Assert(pse != nullptr);
try
{
AUTO_NESTED_HANDLED_EXCEPTION_TYPE((ExceptionType)(ExceptionType_OutOfMemory | ExceptionType_StackOverflow));
Js::AutoDynamicCodeReference dynamicFunctionReference(this);
Parser parser(this);
uint sourceIndex;
JavascriptFunction * pFunction = nullptr;
ParseNodePtr parseTree = ParseScript(&parser, script, cb, pSrcInfo, pse, ppSourceInfo, rootDisplayName, loadScriptFlag, &sourceIndex);
if (parseTree != nullptr)
{
pFunction = GenerateRootFunction(parseTree, sourceIndex, &parser, (*ppSourceInfo)->GetParseFlags(), pse, rootDisplayName);
}
if (pse->ei.scode == JSERR_AsmJsCompileError)
{
Assert((loadScriptFlag & LoadScriptFlag_disableAsmJs) != LoadScriptFlag_disableAsmJs);
pse->Clear();
loadScriptFlag = (LoadScriptFlag)(loadScriptFlag | LoadScriptFlag_disableAsmJs);
return LoadScript(script, cb, pSrcInfo, pse, ppSourceInfo, rootDisplayName, loadScriptFlag);
}
#ifdef ENABLE_SCRIPT_PROFILING
if (pFunction != nullptr && this->IsProfiling())
{
RegisterScript(pFunction->GetFunctionProxy());
}
#else
Assert(!this->IsProfiling());
#endif
return pFunction;
}
catch (Js::OutOfMemoryException)
{
pse->ProcessError(nullptr, E_OUTOFMEMORY, nullptr);
return nullptr;
}
catch (Js::StackOverflowException)
{
pse->ProcessError(nullptr, VBSERR_OutOfStack, nullptr);
return nullptr;
}
}
JavascriptFunction* ScriptContext::GenerateRootFunction(ParseNodePtr parseTree, uint sourceIndex, Parser* parser, uint32 grfscr, CompileScriptException * pse, const char16 *rootDisplayName)
{
HRESULT hr;
// Get the source code to keep it alive during the bytecode generation process
LPCUTF8 source = this->GetSource(sourceIndex)->GetSource(_u("ScriptContext::GenerateRootFunction"));
Assert(source != nullptr); // Source should not have been reclaimed by now
// Generate bytecode and native code
ParseableFunctionInfo* body = NULL;
hr = GenerateByteCode(parseTree, grfscr, this, &body, sourceIndex, false, parser, pse);
this->GetSource(sourceIndex)->SetByteCodeGenerationFlags(grfscr);
if(FAILED(hr))
{
return nullptr;
}
body->SetDisplayName(rootDisplayName);
body->SetIsTopLevel(true);
JavascriptFunction* rootFunction = javascriptLibrary->CreateScriptFunction(body);
return rootFunction;
}
BOOL ScriptContext::ReserveStaticTypeIds(__in int first, __in int last)
{
return threadContext->ReserveStaticTypeIds(first, last);
}
TypeId ScriptContext::ReserveTypeIds(int count)
{
return threadContext->ReserveTypeIds(count);
}
TypeId ScriptContext::CreateTypeId()
{
return threadContext->CreateTypeId();
}
void ScriptContext::OnScriptStart(bool isRoot, bool isScript)
{
const bool isForcedEnter = this->GetDebugContext() != nullptr ? this->GetDebugContext()->GetProbeContainer()->isForcedToEnterScriptStart : false;
if (this->scriptStartEventHandler != nullptr && ((isRoot && threadContext->GetCallRootLevel() == 1) || isForcedEnter))
{
if (this->GetDebugContext() != nullptr)
{
this->GetDebugContext()->GetProbeContainer()->isForcedToEnterScriptStart = false;
}
this->scriptStartEventHandler(this);
}
#if ENABLE_NATIVE_CODEGEN
//Blue 5491: Only start codegen if isScript. Avoid it if we are not really starting script and called from risky region such as catch handler.
if (isScript)
{
NativeCodeGenEnterScriptStart(this->GetNativeCodeGenerator());
}
#endif
}
void ScriptContext::OnScriptEnd(bool isRoot, bool isForcedEnd)
{
if ((isRoot && threadContext->GetCallRootLevel() == 1) || isForcedEnd)
{
if (this->scriptEndEventHandler != nullptr)
{
this->scriptEndEventHandler(this);
}
}
}
#ifdef FAULT_INJECTION
void ScriptContext::DisposeScriptContextByFaultInjection() {
if (this->disposeScriptByFaultInjectionEventHandler != nullptr)
{
this->disposeScriptByFaultInjectionEventHandler(this);
}
}
#endif
template <bool stackProbe, bool leaveForHost>
bool ScriptContext::LeaveScriptStart(void * frameAddress)
{
ThreadContext * threadContext = this->threadContext;
if (!threadContext->IsScriptActive())
{
// we should have enter always.
AssertMsg(FALSE, "Leaving ScriptStart while script is not active.");
return false;
}
// Make sure the host function will have at least 32k of stack available.
if (stackProbe)
{
threadContext->ProbeStack(Js::Constants::MinStackCallout, this);
}
else
{
AssertMsg(ExceptionCheck::HasStackProbe(), "missing stack probe");
}
threadContext->LeaveScriptStart<leaveForHost>(frameAddress);
return true;
}
template <bool leaveForHost>
void ScriptContext::LeaveScriptEnd(void * frameAddress)
{
this->threadContext->LeaveScriptEnd<leaveForHost>(frameAddress);
}
// explicit instantiations
template bool ScriptContext::LeaveScriptStart<true, true>(void * frameAddress);
template bool ScriptContext::LeaveScriptStart<true, false>(void * frameAddress);
template bool ScriptContext::LeaveScriptStart<false, true>(void * frameAddress);
template void ScriptContext::LeaveScriptEnd<true>(void * frameAddress);
template void ScriptContext::LeaveScriptEnd<false>(void * frameAddress);
bool ScriptContext::EnsureInterpreterArena(ArenaAllocator **ppAlloc)
{
bool fNew = false;
if (this->interpreterArena == nullptr)
{
this->interpreterArena = this->GetRecycler()->CreateGuestArena(_u("Interpreter"), Throw::OutOfMemory);
fNew = true;
}
*ppAlloc = this->interpreterArena;
return fNew;
}
void ScriptContext::ReleaseInterpreterArena()
{
AssertMsg(this->interpreterArena, "No interpreter arena to release");
if (this->interpreterArena)
{
this->GetRecycler()->DeleteGuestArena(this->interpreterArena);
this->interpreterArena = nullptr;
}
}
void ScriptContext::ReleaseGuestArena()
{
AssertMsg(this->guestArena, "No guest arena to release");
if (this->guestArena)
{
this->GetRecycler()->DeleteGuestArena(this->guestArena);
this->guestArena = nullptr;
}
}
void ScriptContext::SetScriptStartEventHandler(ScriptContext::EventHandler eventHandler)
{
AssertMsg(this->scriptStartEventHandler == nullptr, "Do not support multi-cast yet");
this->scriptStartEventHandler = eventHandler;
}
void ScriptContext::SetScriptEndEventHandler(ScriptContext::EventHandler eventHandler)
{
AssertMsg(this->scriptEndEventHandler == nullptr, "Do not support multi-cast yet");
this->scriptEndEventHandler = eventHandler;
}
#ifdef FAULT_INJECTION
void ScriptContext::SetDisposeDisposeByFaultInjectionEventHandler(ScriptContext::EventHandler eventHandler)
{
AssertMsg(this->disposeScriptByFaultInjectionEventHandler == nullptr, "Do not support multi-cast yet");
this->disposeScriptByFaultInjectionEventHandler = eventHandler;
}
#endif
bool ScriptContext::SaveSourceCopy(Utf8SourceInfo* const sourceInfo, int cchLength, bool isCesu8, uint * index)
{
HRESULT hr = S_OK;
BEGIN_TRANSLATE_OOM_TO_HRESULT
{
*index = this->SaveSourceCopy(sourceInfo, cchLength, isCesu8);
}
END_TRANSLATE_OOM_TO_HRESULT(hr);
return hr == S_OK;
}
uint ScriptContext::SaveSourceCopy(Utf8SourceInfo* sourceInfo, int cchLength, bool isCesu8)
{
Utf8SourceInfo* newSource = Utf8SourceInfo::Clone(this, sourceInfo);
return SaveSourceNoCopy(newSource, cchLength, isCesu8);
}
uint ScriptContext::SaveSourceNoCopy(Utf8SourceInfo* sourceInfo, int cchLength, bool isCesu8)
{
Assert(sourceInfo->GetScriptContext() == this);
if (this->IsScriptContextInDebugMode() && !sourceInfo->GetIsLibraryCode() && !sourceInfo->IsInDebugMode())
{
sourceInfo->SetInDebugMode(true);
}
RecyclerWeakReference<Utf8SourceInfo>* sourceWeakRef = this->GetRecycler()->CreateWeakReferenceHandle<Utf8SourceInfo>(sourceInfo);
sourceInfo->SetIsCesu8(isCesu8);
{
// We can be compiling new source code while rundown thread is reading from the list, causing AV on the reader thread
// lock the list during write as well.
AutoCriticalSection autocs(GetThreadContext()->GetEtwRundownCriticalSection());
return sourceList->SetAtFirstFreeSpot(sourceWeakRef);
}
}
void ScriptContext::CloneSources(ScriptContext* sourceContext)
{
sourceContext->sourceList->Map([=](int index, RecyclerWeakReference<Utf8SourceInfo>* sourceInfo)
{
Utf8SourceInfo* info = sourceInfo->Get();
if (info)
{
CloneSource(info);
}
});
}
uint ScriptContext::CloneSource(Utf8SourceInfo* info)
{
return this->SaveSourceCopy(info, info->GetCchLength(), info->GetIsCesu8());
}
Utf8SourceInfo* ScriptContext::GetSource(uint index)
{
Assert(this->sourceList->IsItemValid(index)); // This assert should be a subset of info != null- if info was null, in the last collect, we'd have invalidated the item
Utf8SourceInfo* info = this->sourceList->Item(index)->Get();
Assert(info != nullptr); // Should still be alive if this method is being called
return info;
}
bool ScriptContext::IsItemValidInSourceList(int index)
{
return (index < this->sourceList->Count()) && this->sourceList->IsItemValid(index);
}
void ScriptContext::RecordException(JavascriptExceptionObject * exceptionObject, bool propagateToDebugger)
{
Assert(this->threadContext->GetRecordedException() == nullptr || GetThreadContext()->HasUnhandledException());
this->threadContext->SetRecordedException(exceptionObject, propagateToDebugger);
#if DBG && ENABLE_DEBUG_STACK_BACK_TRACE
exceptionObject->FillStackBackTrace();
#endif
}
void ScriptContext::RethrowRecordedException(JavascriptExceptionObject::HostWrapperCreateFuncType hostWrapperCreateFunc)
{
bool considerPassingToDebugger = false;
JavascriptExceptionObject * exceptionObject = this->GetAndClearRecordedException(&considerPassingToDebugger);
if (hostWrapperCreateFunc)
{
exceptionObject->SetHostWrapperCreateFunc(exceptionObject->GetScriptContext() != this ? hostWrapperCreateFunc : nullptr);
}
JavascriptExceptionOperators::RethrowExceptionObject(exceptionObject, this, considerPassingToDebugger);
}
Js::JavascriptExceptionObject * ScriptContext::GetAndClearRecordedException(bool *considerPassingToDebugger)
{
JavascriptExceptionObject * exceptionObject = this->threadContext->GetRecordedException();
Assert(exceptionObject != nullptr);
if (considerPassingToDebugger)
{
*considerPassingToDebugger = this->threadContext->GetPropagateException();
}
exceptionObject = exceptionObject->CloneIfStaticExceptionObject(this);
this->threadContext->SetRecordedException(nullptr);
return exceptionObject;
}
bool ScriptContext::IsInEvalMap(FastEvalMapString const& key, BOOL isIndirect, ScriptFunction **ppFuncScript)
{
EvalCacheDictionary *dict = isIndirect ? this->cache->indirectEvalCacheDictionary : this->cache->evalCacheDictionary;
if (dict == nullptr)
{
return false;
}
#ifdef PROFILE_EVALMAP
if (Configuration::Global.flags.ProfileEvalMap)
{
charcount_t len = key.str.GetLength();
if (dict->TryGetValue(key, ppFuncScript))
{
Output::Print(_u("EvalMap cache hit:\t source size = %d\n"), len);
}
else
{
Output::Print(_u("EvalMap cache miss:\t source size = %d\n"), len);
}
}
#endif
// If eval map cleanup is false, to preserve existing behavior, add it to the eval map MRU list
bool success = dict->TryGetValue(key, ppFuncScript);
if (success)
{
dict->NotifyAdd(key);
#ifdef VERBOSE_EVAL_MAP
#if DBG
dict->DumpKeepAlives();
#endif
#endif
}
return success;
}
void ScriptContext::AddToEvalMap(FastEvalMapString const& key, BOOL isIndirect, ScriptFunction *pFuncScript)
{
EvalCacheDictionary *dict = isIndirect ? this->cache->indirectEvalCacheDictionary : this->cache->evalCacheDictionary;
if (dict == nullptr)
{
EvalCacheTopLevelDictionary* evalTopDictionary = RecyclerNew(this->recycler, EvalCacheTopLevelDictionary, this->recycler);
dict = RecyclerNew(this->recycler, EvalCacheDictionary, evalTopDictionary, recycler);
if (isIndirect)
{
this->cache->indirectEvalCacheDictionary = dict;
}
else
{
this->cache->evalCacheDictionary = dict;
}
}
dict->Add(key, pFuncScript);
}
bool ScriptContext::IsInNewFunctionMap(EvalMapString const& key, ParseableFunctionInfo **ppFuncBody)
{
if (this->cache->newFunctionCache == nullptr)
{
return false;
}
// If eval map cleanup is false, to preserve existing behavior, add it to the eval map MRU list
bool success = this->cache->newFunctionCache->TryGetValue(key, ppFuncBody);
if (success)
{
this->cache->newFunctionCache->NotifyAdd(key);
#ifdef VERBOSE_EVAL_MAP
#if DBG
this->cache->newFunctionCache->DumpKeepAlives();
#endif
#endif
}
return success;
}
void ScriptContext::AddToNewFunctionMap(EvalMapString const& key, ParseableFunctionInfo *pFuncBody)
{
if (this->cache->newFunctionCache == nullptr)
{
this->cache->newFunctionCache = RecyclerNew(this->recycler, NewFunctionCache, this->recycler);
}
this->cache->newFunctionCache->Add(key, pFuncBody);
}
void ScriptContext::EnsureSourceContextInfoMap()
{
if (this->cache->sourceContextInfoMap == nullptr)
{
this->cache->sourceContextInfoMap = RecyclerNew(this->GetRecycler(), SourceContextInfoMap, this->GetRecycler());
}
}
void ScriptContext::EnsureDynamicSourceContextInfoMap()
{
if (this->cache->dynamicSourceContextInfoMap == nullptr)
{
this->cache->dynamicSourceContextInfoMap = RecyclerNew(this->GetRecycler(), DynamicSourceContextInfoMap, this->GetRecycler());
}
}
SourceContextInfo* ScriptContext::GetSourceContextInfo(uint hash)
{
SourceContextInfo * sourceContextInfo;
if (this->cache->dynamicSourceContextInfoMap && this->cache->dynamicSourceContextInfoMap->TryGetValue(hash, &sourceContextInfo))
{
return sourceContextInfo;
}
return nullptr;
}
SourceContextInfo* ScriptContext::CreateSourceContextInfo(uint hash, DWORD_PTR hostSourceContext)
{
EnsureDynamicSourceContextInfoMap();
if (this->GetSourceContextInfo(hash) != nullptr)
{
return const_cast<SourceContextInfo*>(this->cache->noContextSourceContextInfo);
}
if (this->cache->dynamicSourceContextInfoMap->Count() > INMEMORY_CACHE_MAX_PROFILE_MANAGER)
{
OUTPUT_TRACE(Js::DynamicProfilePhase, _u("Max of dynamic script profile info reached.\n"));
return const_cast<SourceContextInfo*>(this->cache->noContextSourceContextInfo);
}
// This is capped so we can continue allocating in the arena
SourceContextInfo * sourceContextInfo = RecyclerNewStructZ(this->GetRecycler(), SourceContextInfo);
sourceContextInfo->sourceContextId = this->GetNextSourceContextId();
sourceContextInfo->dwHostSourceContext = hostSourceContext;
sourceContextInfo->isHostDynamicDocument = true;
sourceContextInfo->hash = hash;
#if ENABLE_PROFILE_INFO
sourceContextInfo->sourceDynamicProfileManager = this->threadContext->GetSourceDynamicProfileManager(this->GetUrl(), hash, &referencesSharedDynamicSourceContextInfo);
#endif
// For the host provided dynamic code (if hostSourceContext is not NoHostSourceContext), do not add to dynamicSourceContextInfoMap
if (hostSourceContext == Js::Constants::NoHostSourceContext)
{
this->cache->dynamicSourceContextInfoMap->Add(hash, sourceContextInfo);
}
return sourceContextInfo;
}
//
// Makes a copy of the URL to be stored in the map.
//
SourceContextInfo * ScriptContext::CreateSourceContextInfo(DWORD_PTR sourceContext, char16 const * url, size_t len,
IActiveScriptDataCache* profileDataCache, char16 const * sourceMapUrl /*= NULL*/, size_t sourceMapUrlLen /*= 0*/)
{
// Take etw rundown lock on this thread context. We are going to init/add to sourceContextInfoMap.
AutoCriticalSection autocs(GetThreadContext()->GetEtwRundownCriticalSection());
EnsureSourceContextInfoMap();
Assert(this->GetSourceContextInfo(sourceContext, profileDataCache) == nullptr);
SourceContextInfo * sourceContextInfo = RecyclerNewStructZ(this->GetRecycler(), SourceContextInfo);
sourceContextInfo->sourceContextId = this->GetNextSourceContextId();
sourceContextInfo->dwHostSourceContext = sourceContext;
sourceContextInfo->isHostDynamicDocument = false;
#if ENABLE_PROFILE_INFO
sourceContextInfo->sourceDynamicProfileManager = nullptr;
#endif
if (url != nullptr)
{
sourceContextInfo->url = CopyString(url, len, this->SourceCodeAllocator());
JS_ETW(EtwTrace::LogSourceModuleLoadEvent(this, sourceContext, url));
}
if (sourceMapUrl != nullptr && sourceMapUrlLen != 0)
{
sourceContextInfo->sourceMapUrl = CopyString(sourceMapUrl, sourceMapUrlLen, this->SourceCodeAllocator());
}
#if ENABLE_PROFILE_INFO
if (!this->startupComplete)
{
sourceContextInfo->sourceDynamicProfileManager = SourceDynamicProfileManager::LoadFromDynamicProfileStorage(sourceContextInfo, this, profileDataCache);
Assert(sourceContextInfo->sourceDynamicProfileManager != NULL);
}
this->cache->sourceContextInfoMap->Add(sourceContext, sourceContextInfo);
#endif
return sourceContextInfo;
}
// static
const char16* ScriptContext::CopyString(const char16* str, size_t charCount, ArenaAllocator* alloc)
{
size_t length = charCount + 1; // Add 1 for the NULL.
char16* copy = AnewArray(alloc, char16, length);
js_wmemcpy_s(copy, length, str, charCount);
copy[length - 1] = _u('\0');
return copy;
}
SourceContextInfo * ScriptContext::GetSourceContextInfo(DWORD_PTR sourceContext, IActiveScriptDataCache* profileDataCache)
{
if (sourceContext == Js::Constants::NoHostSourceContext)
{
return const_cast<SourceContextInfo*>(this->cache->noContextSourceContextInfo);
}
// We only init sourceContextInfoMap, don't need to lock.
EnsureSourceContextInfoMap();
SourceContextInfo * sourceContextInfo;
if (this->cache->sourceContextInfoMap->TryGetValue(sourceContext, &sourceContextInfo))
{
#if ENABLE_PROFILE_INFO
if (profileDataCache &&
sourceContextInfo->sourceDynamicProfileManager != nullptr &&
!sourceContextInfo->sourceDynamicProfileManager->IsProfileLoadedFromWinInet() &&
!this->startupComplete)
{
bool profileLoaded = sourceContextInfo->sourceDynamicProfileManager->LoadFromProfileCache(profileDataCache, sourceContextInfo->url);
if (profileLoaded)
{
JS_ETW(EventWriteJSCRIPT_PROFILE_LOAD(sourceContextInfo->dwHostSourceContext, this));
}
}
#endif
return sourceContextInfo;
}
return nullptr;
}
SRCINFO const *
ScriptContext::GetModuleSrcInfo(Js::ModuleID moduleID)
{
if (moduleSrcInfoCount <= moduleID)
{
uint newCount = moduleID + 4; // Preallocate 4 more slots, moduleID don't usually grow much
SRCINFO const ** newModuleSrcInfo = RecyclerNewArrayZ(this->GetRecycler(), SRCINFO const*, newCount);
memcpy(newModuleSrcInfo, cache->moduleSrcInfo, sizeof(SRCINFO const *)* moduleSrcInfoCount);
cache->moduleSrcInfo = newModuleSrcInfo;
moduleSrcInfoCount = newCount;
cache->moduleSrcInfo[0] = this->cache->noContextGlobalSourceInfo;
}
SRCINFO const * si = cache->moduleSrcInfo[moduleID];
if (si == nullptr)
{
SRCINFO * newSrcInfo = RecyclerNewStructZ(this->GetRecycler(), SRCINFO);
newSrcInfo->sourceContextInfo = this->cache->noContextSourceContextInfo;
newSrcInfo->moduleID = moduleID;
cache->moduleSrcInfo[moduleID] = newSrcInfo;
si = newSrcInfo;
}
return si;
}
#if ENABLE_TTD
void ScriptContext::InitializeCoreImage_TTD()
{
AssertMsg(this->TTDWellKnownInfo == nullptr, "This should only happen once!!!");
this->TTDWellKnownInfo = TT_HEAP_NEW(TTD::RuntimeContextInfo);
bool hasCaller = this->GetHostScriptContext() ? !!this->GetHostScriptContext()->HasCaller() : false;
BEGIN_JS_RUNTIME_CALLROOT_EX(this, hasCaller)
{
this->TTDWellKnownInfo->GatherKnownObjectToPathMap(this);
}
END_JS_RUNTIME_CALL(this);
}
void ScriptContext::InitializeRecordingActionsAsNeeded_TTD()
{
this->TTDContextInfo = TT_HEAP_NEW(TTD::ScriptContextTTD, this);
this->TTDContextInfo->AddTrackedRoot(TTD_CONVERT_OBJ_TO_LOG_PTR_ID(this->GetLibrary()->GetGlobalObject()), this->GetLibrary()->GetGlobalObject());
this->ScriptContextLogTag = TTD_CONVERT_OBJ_TO_LOG_PTR_ID(this->GetLibrary()->GetGlobalObject());
this->TTDContextInfo->AddTrackedRoot(TTD_CONVERT_OBJ_TO_LOG_PTR_ID(this->GetLibrary()->GetUndefined()), this->GetLibrary()->GetUndefined());
this->TTDContextInfo->AddTrackedRoot(TTD_CONVERT_OBJ_TO_LOG_PTR_ID(this->GetLibrary()->GetNull()), this->GetLibrary()->GetNull());
this->TTDContextInfo->AddTrackedRoot(TTD_CONVERT_OBJ_TO_LOG_PTR_ID(this->GetLibrary()->GetTrue()), this->GetLibrary()->GetTrue());
this->TTDContextInfo->AddTrackedRoot(TTD_CONVERT_OBJ_TO_LOG_PTR_ID(this->GetLibrary()->GetFalse()), this->GetLibrary()->GetFalse());
#if ENABLE_TTD_STACK_STMTS
this->ForceNoNative();
#endif
}
void ScriptContext::InitializeDebuggingActionsAsNeeded_TTD()
{
this->ForceNoNative();
}
#endif
#ifdef PROFILE_EXEC
void
ScriptContext::DisableProfiler()
{
disableProfiler = true;
}
Profiler *
ScriptContext::CreateProfiler()
{
Assert(profiler == nullptr);
if (Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag))
{
this->profiler = NoCheckHeapNew(ScriptContextProfiler);
this->profiler->Initialize(GetThreadContext()->GetPageAllocator(), threadContext->GetRecycler());
#if ENABLE_NATIVE_CODEGEN
CreateProfilerNativeCodeGen(this->nativeCodeGen, this->profiler);
#endif
this->isProfilerCreated = true;
Profiler * oldProfiler = this->threadContext->GetRecycler()->GetProfiler();
this->threadContext->GetRecycler()->SetProfiler(this->profiler->GetProfiler(), this->profiler->GetBackgroundRecyclerProfiler());
return oldProfiler;
}
return nullptr;
}
void
ScriptContext::SetRecyclerProfiler()
{
Assert(Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag));
AssertMsg(this->profiler != nullptr, "Profiler tag is supplied but the profiler pointer is NULL");
if (this->ensureParentInfo)
{
this->hostScriptContext->EnsureParentInfo();
this->ensureParentInfo = false;
}
this->GetRecycler()->SetProfiler(this->profiler->GetProfiler(), this->profiler->GetBackgroundRecyclerProfiler());
}
void
ScriptContext::SetProfilerFromScriptContext(ScriptContext * scriptContext)
{
// this function needs to be called before any code gen happens so
// that access to codegenProfiler won't have concurrency issues
if (Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag))
{
Assert(this->profiler != nullptr);
Assert(this->isProfilerCreated);
Assert(scriptContext->profiler != nullptr);
Assert(scriptContext->isProfilerCreated);
scriptContext->profiler->ProfileMerge(this->profiler);
this->profiler->Release();
this->profiler = scriptContext->profiler;
this->profiler->AddRef();
this->isProfilerCreated = false;
#if ENABLE_NATIVE_CODEGEN
SetProfilerFromNativeCodeGen(this->nativeCodeGen, scriptContext->GetNativeCodeGenerator());
#endif
this->threadContext->GetRecycler()->SetProfiler(this->profiler->GetProfiler(), this->profiler->GetBackgroundRecyclerProfiler());
}
}
void
ScriptContext::ProfileBegin(Js::Phase phase)
{
AssertMsg((this->profiler != nullptr) == Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag),
"Profiler tag is supplied but the profiler pointer is NULL");
if (this->profiler)
{
if (this->ensureParentInfo)
{
this->hostScriptContext->EnsureParentInfo();
this->ensureParentInfo = false;
}
this->profiler->ProfileBegin(phase);
}
}
void
ScriptContext::ProfileEnd(Js::Phase phase)
{
AssertMsg((this->profiler != nullptr) == Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag),
"Profiler tag is supplied but the profiler pointer is NULL");
if (this->profiler)
{
this->profiler->ProfileEnd(phase);
}
}
void
ScriptContext::ProfileSuspend(Js::Phase phase, Js::Profiler::SuspendRecord * suspendRecord)
{
AssertMsg((this->profiler != nullptr) == Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag),
"Profiler tag is supplied but the profiler pointer is NULL");
if (this->profiler)
{
this->profiler->ProfileSuspend(phase, suspendRecord);
}
}
void
ScriptContext::ProfileResume(Js::Profiler::SuspendRecord * suspendRecord)
{
AssertMsg((this->profiler != nullptr) == Js::Configuration::Global.flags.IsEnabled(Js::ProfileFlag),
"Profiler tag is supplied but the profiler pointer is NULL");
if (this->profiler)
{
this->profiler->ProfileResume(suspendRecord);
}
}
void
ScriptContext::ProfilePrint()
{
if (disableProfiler)
{
return;
}
Assert(profiler != nullptr);
recycler->EnsureNotCollecting();
profiler->ProfilePrint(Js::Configuration::Global.flags.Profile.GetFirstPhase());
#if ENABLE_NATIVE_CODEGEN
ProfilePrintNativeCodeGen(this->nativeCodeGen);
#endif
}
#endif
#ifdef ENABLE_SCRIPT_PROFILING
inline void ScriptContext::CoreSetProfileEventMask(DWORD dwEventMask)
{
AssertMsg(m_pProfileCallback != NULL, "Assigning the event mask when there is no callback");
m_dwEventMask = dwEventMask;
m_fTraceFunctionCall = (dwEventMask & PROFILER_EVENT_MASK_TRACE_SCRIPT_FUNCTION_CALL);
m_fTraceNativeFunctionCall = (dwEventMask & PROFILER_EVENT_MASK_TRACE_NATIVE_FUNCTION_CALL);
m_fTraceDomCall = (dwEventMask & PROFILER_EVENT_MASK_TRACE_DOM_FUNCTION_CALL);
}
HRESULT ScriptContext::RegisterProfileProbe(IActiveScriptProfilerCallback *pProfileCallback, DWORD dwEventMask, DWORD dwContext, RegisterExternalLibraryType RegisterExternalLibrary, JavascriptMethod dispatchInvoke)
{
if (m_pProfileCallback != NULL)
{
return ACTIVPROF_E_PROFILER_PRESENT;
}
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::RegisterProfileProbe\n"));
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("Info\nThunks Address :\n"));
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("DefaultEntryThunk : 0x%08X, CrossSite::DefaultThunk : 0x%08X, DefaultDeferredParsingThunk : 0x%08X\n"), DefaultEntryThunk, CrossSite::DefaultThunk, DefaultDeferredParsingThunk);
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ProfileEntryThunk : 0x%08X, CrossSite::ProfileThunk : 0x%08X, ProfileDeferredParsingThunk : 0x%08X, ProfileDeferredDeserializeThunk : 0x%08X,\n"), ProfileEntryThunk, CrossSite::ProfileThunk, ProfileDeferredParsingThunk, ProfileDeferredDeserializeThunk);
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptType :\n"));
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("PROFILER_SCRIPT_TYPE_USER : 0, PROFILER_SCRIPT_TYPE_DYNAMIC : 1, PROFILER_SCRIPT_TYPE_NATIVE : 2, PROFILER_SCRIPT_TYPE_DOM : 3\n"));
HRESULT hr = pProfileCallback->Initialize(dwContext);
if (SUCCEEDED(hr))
{
m_pProfileCallback = pProfileCallback;
pProfileCallback->AddRef();
CoreSetProfileEventMask(dwEventMask);
if (m_fTraceDomCall)
{
if (FAILED(pProfileCallback->QueryInterface(&m_pProfileCallback2)))
{
m_fTraceDomCall = FALSE;
}
}
if (webWorkerId != Js::Constants::NonWebWorkerContextId)
{
IActiveScriptProfilerCallback3 * pProfilerCallback3;
if (SUCCEEDED(pProfileCallback->QueryInterface(&pProfilerCallback3)))
{
pProfilerCallback3->SetWebWorkerId(webWorkerId);
pProfilerCallback3->Release();
// Omitting the HRESULT since it is up to the callback to make use of the webWorker information.
}
}
#if DEBUG
StartNewProfileSession();
#endif
#if ENABLE_NATIVE_CODEGEN
NativeCodeGenerator *pNativeCodeGen = this->GetNativeCodeGenerator();
AutoOptionalCriticalSection autoAcquireCodeGenQueue(GetNativeCodeGenCriticalSection(pNativeCodeGen));
#endif
this->SetProfileMode(TRUE);
#if ENABLE_NATIVE_CODEGEN
SetProfileModeNativeCodeGen(pNativeCodeGen, TRUE);
#endif
// Register builtin functions
if (m_fTraceNativeFunctionCall)
{
hr = this->RegisterBuiltinFunctions(RegisterExternalLibrary);
if (FAILED(hr))
{
return hr;
}
}
this->RegisterAllScripts();
// Set the dispatch profiler:
this->SetDispatchProfile(TRUE, dispatchInvoke);
// Update the function objects currently present in there.
this->SetFunctionInRecyclerToProfileMode();
}
return hr;
}
HRESULT ScriptContext::SetProfileEventMask(DWORD dwEventMask)
{
if (m_pProfileCallback == NULL)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
return ACTIVPROF_E_UNABLE_TO_APPLY_ACTION;
}
HRESULT ScriptContext::DeRegisterProfileProbe(HRESULT hrReason, JavascriptMethod dispatchInvoke)
{
if (m_pProfileCallback == NULL)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::DeRegisterProfileProbe\n"));
#if ENABLE_NATIVE_CODEGEN
// Acquire the code gen working queue - we are going to change the thunks
NativeCodeGenerator *pNativeCodeGen = this->GetNativeCodeGenerator();
Assert(pNativeCodeGen);
{
AutoOptionalCriticalSection lock(GetNativeCodeGenCriticalSection(pNativeCodeGen));
this->SetProfileMode(FALSE);
SetProfileModeNativeCodeGen(pNativeCodeGen, FALSE);
// DisableJIT-TODO: Does need to happen even with JIT disabled?
// Unset the dispatch profiler:
if (dispatchInvoke != nullptr)
{
this->SetDispatchProfile(FALSE, dispatchInvoke);
}
}
#endif
m_inProfileCallback = TRUE;
HRESULT hr = m_pProfileCallback->Shutdown(hrReason);
m_inProfileCallback = FALSE;
m_pProfileCallback->Release();
m_pProfileCallback = NULL;
if (m_pProfileCallback2 != NULL)
{
m_pProfileCallback2->Release();
m_pProfileCallback2 = NULL;
}
#if DEBUG
StopProfileSession();
#endif
return hr;
}
void ScriptContext::SetProfileMode(BOOL fSet)
{
#ifdef ENABLE_SCRIPT_PROFILING
if (fSet)
{
AssertMsg(m_pProfileCallback != NULL, "In profile mode when there is no call back");
this->CurrentThunk = ProfileEntryThunk;
this->CurrentCrossSiteThunk = CrossSite::ProfileThunk;
this->DeferredParsingThunk = ProfileDeferredParsingThunk;
this->DeferredDeserializationThunk = ProfileDeferredDeserializeThunk;
this->globalObject->EvalHelper = &Js::GlobalObject::ProfileModeEvalHelper;
#if DBG
this->hadProfiled = true;
#endif
}
else
#endif
{
Assert(!fSet);
this->CurrentThunk = DefaultEntryThunk;
this->CurrentCrossSiteThunk = CrossSite::DefaultThunk;
this->DeferredParsingThunk = DefaultDeferredParsingThunk;
this->globalObject->EvalHelper = &Js::GlobalObject::DefaultEvalHelper;
// In Debug mode/Fast F12 library is still needed for built-in wrappers.
if (!(this->IsScriptContextInDebugMode() && this->IsExceptionWrapperForBuiltInsEnabled()))
{
this->javascriptLibrary->SetProfileMode(FALSE);
}
}
}
HRESULT ScriptContext::RegisterScript(Js::FunctionProxy * proxy, BOOL fRegisterScript /*default TRUE*/)
{
if (m_pProfileCallback == nullptr)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::RegisterScript, fRegisterScript : %s, IsFunctionDefer : %s\n"), IsTrueOrFalse(fRegisterScript), IsTrueOrFalse(proxy->IsDeferred()));
AssertMsg(proxy != nullptr, "Function body cannot be null when calling reporting");
AssertMsg(proxy->GetScriptContext() == this, "wrong script context while reporting the function?");
if (fRegisterScript)
{
// Register the script to the callback.
// REVIEW: do we really need to undefer everything?
HRESULT hr = proxy->EnsureDeserialized()->Parse()->ReportScriptCompiled();
if (FAILED(hr))
{
return hr;
}
}
return !proxy->IsDeferred() ? proxy->GetFunctionBody()->RegisterFunction(false) : S_OK;
}
HRESULT ScriptContext::RegisterAllScripts()
{
AssertMsg(m_pProfileCallback != nullptr, "Called register scripts when we don't have profile callback");
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::RegisterAllScripts started\n"));
// Future Work: Once Utf8SourceInfo can generate the debug document text without requiring a function body,
// this code can be considerably simplified to doing the following:
// - scriptContext->MapScript() : Fire script compiled for each script
// - scriptContext->MapFunction(): Fire function compiled for each function
this->MapScript([](Utf8SourceInfo* sourceInfo)
{
FunctionBody* functionBody = sourceInfo->GetAnyParsedFunction();
if (functionBody)
{
functionBody->ReportScriptCompiled();
}
});
// FunctionCompiled events for all functions.
this->MapFunction([](Js::FunctionBody* pFuncBody)
{
if (!pFuncBody->GetIsTopLevel() && pFuncBody->GetIsGlobalFunc())
{
// This must be the dummy function, generated due to the deferred parsing.
return;
}
pFuncBody->RegisterFunction(TRUE, TRUE); // Ignore potential failure (worst case is not profiling).
});
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::RegisterAllScripts ended\n"));
return S_OK;
}
#endif // ENABLE_SCRIPT_PROFILING
// Shuts down and recreates the native code generator. This is used when
// attaching and detaching the debugger in order to clear the list of work
// items that are pending in the JIT job queue.
// Alloc first and then free so that the native code generator is at a different address
#if ENABLE_NATIVE_CODEGEN
HRESULT ScriptContext::RecreateNativeCodeGenerator()
{
NativeCodeGenerator* oldCodeGen = this->nativeCodeGen;
HRESULT hr = S_OK;
BEGIN_TRANSLATE_OOM_TO_HRESULT_NESTED
this->nativeCodeGen = NewNativeCodeGenerator(this);
SetProfileModeNativeCodeGen(this->GetNativeCodeGenerator(), this->IsProfiling());
END_TRANSLATE_OOM_TO_HRESULT(hr);
// Delete the native code generator and recreate so that all jobs get cleared properly
// and re-jitted.
CloseNativeCodeGenerator(oldCodeGen);
DeleteNativeCodeGenerator(oldCodeGen);
return hr;
}
#endif
HRESULT ScriptContext::OnDebuggerAttached()
{
OUTPUT_TRACE(Js::DebuggerPhase, _u("ScriptContext::OnDebuggerAttached: start 0x%p\n"), this);
Js::StepController* stepController = &this->GetThreadContext()->GetDebugManager()->stepController;
if (stepController->IsActive())
{
AssertMsg(stepController->GetActivatedContext() == nullptr, "StepController should not be active when we attach.");
stepController->Deactivate(); // Defense in depth
}
bool shouldPerformSourceRundown = false;
if (this->IsScriptContextInNonDebugMode())
{
// Today we do source rundown as a part of attach to support VS attaching without
// first calling PerformSourceRundown. PerformSourceRundown will be called once
// by debugger host prior to attaching.
this->GetDebugContext()->SetDebuggerMode(Js::DebuggerMode::SourceRundown);
// Need to perform rundown only once.
shouldPerformSourceRundown = true;
}
// Rundown on all existing functions and change their thunks so that they will go to debug mode once they are called.
HRESULT hr = OnDebuggerAttachedDetached(/*attach*/ true);
// Debugger attach/detach failure is catastrophic, take down the process
DEBUGGER_ATTACHDETACH_FATAL_ERROR_IF_FAILED(hr);
// Disable QC while functions are re-parsed as this can be time consuming
AutoDisableInterrupt autoDisableInterrupt(this->threadContext->GetInterruptPoller(), true);
hr = this->GetDebugContext()->RundownSourcesAndReparse(shouldPerformSourceRundown, /*shouldReparseFunctions*/ true);
if (this->IsClosed())
{
return hr;
}
// Debugger attach/detach failure is catastrophic, take down the process
DEBUGGER_ATTACHDETACH_FATAL_ERROR_IF_FAILED(hr);
HRESULT hrEntryPointUpdate = S_OK;
BEGIN_TRANSLATE_OOM_TO_HRESULT_NESTED
#ifdef ASMJS_PLAT
TempArenaAllocatorObject* tmpAlloc = GetTemporaryAllocator(_u("DebuggerTransition"));
debugTransitionAlloc = tmpAlloc->GetAllocator();
asmJsEnvironmentMap = Anew(debugTransitionAlloc, AsmFunctionMap, debugTransitionAlloc);
#endif
// Still do the pass on the function's entrypoint to reflect its state with the functionbody's entrypoint.
this->UpdateRecyclerFunctionEntryPointsForDebugger();
#ifdef ASMJS_PLAT
auto asmEnvIter = asmJsEnvironmentMap->GetIterator();
while (asmEnvIter.IsValid())
{
// we are attaching, change frame setup for asm.js frame to javascript frame
SList<AsmJsScriptFunction *> * funcList = asmEnvIter.CurrentValue();
Assert(!funcList->Empty());
void* newEnv = AsmJsModuleInfo::ConvertFrameForJavascript(asmEnvIter.CurrentKey(), funcList->Head());
funcList->Iterate([&](AsmJsScriptFunction * func)
{
func->GetEnvironment()->SetItem(0, newEnv);
});
asmEnvIter.MoveNext();
}
// walk through and clean up the asm.js fields as a discrete step, because module might be multiply linked
auto asmCleanupIter = asmJsEnvironmentMap->GetIterator();
while (asmCleanupIter.IsValid())
{
SList<AsmJsScriptFunction *> * funcList = asmCleanupIter.CurrentValue();
Assert(!funcList->Empty());
funcList->Iterate([](AsmJsScriptFunction * func)
{
func->SetModuleMemory(nullptr);
func->GetFunctionBody()->ResetAsmJsInfo();
});
asmCleanupIter.MoveNext();
}
ReleaseTemporaryAllocator(tmpAlloc);
#endif
END_TRANSLATE_OOM_TO_HRESULT(hrEntryPointUpdate);
if (hrEntryPointUpdate != S_OK)
{
// should only be here for OOM
Assert(hrEntryPointUpdate == E_OUTOFMEMORY);
return hrEntryPointUpdate;
}
OUTPUT_TRACE(Js::DebuggerPhase, _u("ScriptContext::OnDebuggerAttached: done 0x%p, hr = 0x%X\n"), this, hr);
return hr;
}
// Reverts the script context state back to the state before debugging began.
HRESULT ScriptContext::OnDebuggerDetached()
{
OUTPUT_TRACE(Js::DebuggerPhase, _u("ScriptContext::OnDebuggerDetached: start 0x%p\n"), this);
Js::StepController* stepController = &this->GetThreadContext()->GetDebugManager()->stepController;
if (stepController->IsActive())
{
// Normally step controller is deactivated on start of dispatch (step, async break, exception, etc),
// and in the beginning of interpreter loop we check for step complete (can cause check whether current bytecode belong to stmt).
// But since it holds to functionBody/statementMaps, we have to deactivate it as func bodies are going away/reparsed.
stepController->Deactivate();
}
// Go through all existing functions and change their thunks back to using non-debug mode versions when called
// and notify the script context that the debugger has detached to allow it to revert the runtime to the proper
// state (JIT enabled).
HRESULT hr = OnDebuggerAttachedDetached(/*attach*/ false);
// Debugger attach/detach failure is catastrophic, take down the process
DEBUGGER_ATTACHDETACH_FATAL_ERROR_IF_FAILED(hr);
// Move the debugger into source rundown mode.
this->GetDebugContext()->SetDebuggerMode(Js::DebuggerMode::SourceRundown);
// Disable QC while functions are re-parsed as this can be time consuming
AutoDisableInterrupt autoDisableInterrupt(this->threadContext->GetInterruptPoller(), true);
// Force a reparse so that indirect function caches are updated.
hr = this->GetDebugContext()->RundownSourcesAndReparse(/*shouldPerformSourceRundown*/ false, /*shouldReparseFunctions*/ true);
if (this->IsClosed())
{
return hr;
}
// Debugger attach/detach failure is catastrophic, take down the process
DEBUGGER_ATTACHDETACH_FATAL_ERROR_IF_FAILED(hr);
// Still do the pass on the function's entrypoint to reflect its state with the functionbody's entrypoint.
this->UpdateRecyclerFunctionEntryPointsForDebugger();
OUTPUT_TRACE(Js::DebuggerPhase, _u("ScriptContext::OnDebuggerDetached: done 0x%p, hr = 0x%X\n"), this, hr);
return hr;
}
HRESULT ScriptContext::OnDebuggerAttachedDetached(bool attach)
{
// notify threadContext that debugger is attaching so do not do expire
struct AutoRestore
{
AutoRestore(ThreadContext* threadContext)
:threadContext(threadContext)
{
this->threadContext->GetDebugManager()->SetDebuggerAttaching(true);
}
~AutoRestore()
{
this->threadContext->GetDebugManager()->SetDebuggerAttaching(false);
}
private:
ThreadContext* threadContext;
} autoRestore(this->GetThreadContext());
if (!Js::Configuration::Global.EnableJitInDebugMode())
{
if (attach)
{
// Now force nonative, so the job will not be put in jit queue.
ForceNoNative();
}
else
{
// Take the runtime out of interpreted mode so the JIT
// queue can be exercised.
this->ForceNative();
}
}
// Invalidate all the caches.
this->threadContext->InvalidateAllProtoInlineCaches();
this->threadContext->InvalidateAllStoreFieldInlineCaches();
this->threadContext->InvalidateAllIsInstInlineCaches();
if (!attach)
{
this->UnRegisterDebugThunk();
// Remove all breakpoint probes
this->GetDebugContext()->GetProbeContainer()->RemoveAllProbes();
}
HRESULT hr = S_OK;
if (!CONFIG_FLAG(ForceDiagnosticsMode))
{
#if ENABLE_NATIVE_CODEGEN
// Recreate the native code generator so that all pending
// JIT work items will be cleared.
hr = RecreateNativeCodeGenerator();
if (FAILED(hr))
{
return hr;
}
#endif
if (attach)
{
// We need to transition to debug mode after the NativeCodeGenerator is cleared/closed. Since the NativeCodeGenerator will be working on a different thread - it may
// be checking on the DebuggerState (from ScriptContext) while emitting code.
this->GetDebugContext()->SetDebuggerMode(Js::DebuggerMode::Debugging);
#if ENABLE_NATIVE_CODEGEN
UpdateNativeCodeGeneratorForDebugMode(this->nativeCodeGen);
#endif
}
}
else if (attach)
{
this->GetDebugContext()->SetDebuggerMode(Js::DebuggerMode::Debugging);
}
BEGIN_TRANSLATE_OOM_TO_HRESULT_NESTED
{
// Remap all the function entry point thunks.
this->sourceList->Map([=](uint i, RecyclerWeakReference<Js::Utf8SourceInfo>* sourceInfoWeakRef) {
Js::Utf8SourceInfo* sourceInfo = sourceInfoWeakRef->Get();
if (sourceInfo != nullptr)
{
if (!sourceInfo->GetIsLibraryCode())
{
sourceInfo->SetInDebugMode(attach);
sourceInfo->MapFunction([](Js::FunctionBody* functionBody) {
functionBody->SetEntryToDeferParseForDebugger();
});
}
else
{
sourceInfo->MapFunction([](Js::FunctionBody* functionBody) {
functionBody->ResetEntryPoint();
});
}
}
});
}
END_TRANSLATE_OOM_TO_HRESULT(hr);
if (FAILED(hr))
{
return hr;
}
if (attach)
{
this->RegisterDebugThunk();
}
#if ENABLE_PROFILE_INFO
#if DBG_DUMP || defined(DYNAMIC_PROFILE_STORAGE) || defined(RUNTIME_DATA_COLLECTION)
// Reset the dynamic profile list
if (this->profileInfoList)
{
this->profileInfoList->Reset();
}
#endif
#endif
return hr;
}
// We use ProfileThunk under debugger.
void ScriptContext::RegisterDebugThunk(bool calledDuringAttach /*= true*/)
{
if (this->IsExceptionWrapperForBuiltInsEnabled())
{
#ifdef ENABLE_SCRIPT_PROFILING
this->CurrentThunk = ProfileEntryThunk;
this->CurrentCrossSiteThunk = CrossSite::ProfileThunk;
#if ENABLE_NATIVE_CODEGEN
SetProfileModeNativeCodeGen(this->GetNativeCodeGenerator(), TRUE);
#endif
// Set library to profile mode so that for built-ins all new instances of functions
// are created with entry point set to the ProfileThunk.
this->javascriptLibrary->SetProfileMode(true);
this->javascriptLibrary->SetDispatchProfile(true, DispatchProfileInvoke);
if (!calledDuringAttach)
{
#ifdef ENABLE_SCRIPT_PROFILING
m_fTraceDomCall = TRUE; // This flag is always needed in DebugMode to wrap external functions with DebugProfileThunk
#endif
// Update the function objects currently present in there.
this->SetFunctionInRecyclerToProfileMode(true/*enumerateNonUserFunctionsOnly*/);
}
#else
AssertMsg(false, "Profiling needs to be enabled to call RegisterDebugThunk");
#endif
}
}
void ScriptContext::UnRegisterDebugThunk()
{
if (!this->IsProfiling() && this->IsExceptionWrapperForBuiltInsEnabled())
{
this->CurrentThunk = DefaultEntryThunk;
this->CurrentCrossSiteThunk = CrossSite::DefaultThunk;
#if ENABLE_NATIVE_CODEGEN
SetProfileModeNativeCodeGen(this->GetNativeCodeGenerator(), FALSE);
#endif
if (!this->IsProfiling())
{
this->javascriptLibrary->SetProfileMode(false);
this->javascriptLibrary->SetDispatchProfile(false, DispatchDefaultInvoke);
}
}
}
#ifdef ENABLE_SCRIPT_PROFILING
HRESULT ScriptContext::RegisterBuiltinFunctions(RegisterExternalLibraryType RegisterExternalLibrary)
{
Assert(m_pProfileCallback != NULL);
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::RegisterBuiltinFunctions\n"));
HRESULT hr = S_OK;
// Consider creating ProfileArena allocator instead of General allocator
if (m_pBuiltinFunctionIdMap == NULL)
{
// Anew throws if it OOMs, so the caller into this function needs to handle that exception
m_pBuiltinFunctionIdMap = Anew(GeneralAllocator(), BuiltinFunctionIdDictionary,
GeneralAllocator(), 17);
}
this->javascriptLibrary->SetProfileMode(TRUE);
if (FAILED(hr = OnScriptCompiled(BuiltInFunctionsScriptId, PROFILER_SCRIPT_TYPE_NATIVE, NULL)))
{
return hr;
}
if (FAILED(hr = this->javascriptLibrary->ProfilerRegisterBuiltIns()))
{
return hr;
}
// External Library
if (RegisterExternalLibrary != NULL)
{
(*RegisterExternalLibrary)(this);
}
return hr;
}
void ScriptContext::SetFunctionInRecyclerToProfileMode(bool enumerateNonUserFunctionsOnly/* = false*/)
{
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::SetFunctionInRecyclerToProfileMode started (m_fTraceDomCall : %s)\n"), IsTrueOrFalse(m_fTraceDomCall));
// Mark this script context isEnumeratingRecyclerObjects
AutoEnumeratingRecyclerObjects enumeratingRecyclerObjects(this);
m_enumerateNonUserFunctionsOnly = enumerateNonUserFunctionsOnly;
this->recycler->EnumerateObjects(JavascriptLibrary::EnumFunctionClass, &ScriptContext::RecyclerEnumClassEnumeratorCallback);
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::SetFunctionInRecyclerToProfileMode ended\n"));
}
#endif // ENABLE_SCRIPT_PROFILING
void ScriptContext::UpdateRecyclerFunctionEntryPointsForDebugger()
{
// Mark this script context isEnumeratingRecyclerObjects
AutoEnumeratingRecyclerObjects enumeratingRecyclerObjects(this);
this->recycler->EnumerateObjects(JavascriptLibrary::EnumFunctionClass, &ScriptContext::RecyclerFunctionCallbackForDebugger);
}
#ifdef ASMJS_PLAT
void ScriptContext::TransitionEnvironmentForDebugger(ScriptFunction * scriptFunction)
{
if (scriptFunction->GetScriptContext()->IsScriptContextInDebugMode() &&
scriptFunction->GetFunctionBody()->IsInDebugMode() &&
scriptFunction->GetFunctionBody()->GetAsmJsFunctionInfo() != nullptr &&
scriptFunction->GetFunctionBody()->GetAsmJsFunctionInfo()->GetModuleFunctionBody() != nullptr)
{
void* env = scriptFunction->GetEnvironment()->GetItem(0);
SList<AsmJsScriptFunction*> * funcList = nullptr;
if (asmJsEnvironmentMap->TryGetValue(env, &funcList))
{
funcList->Push((AsmJsScriptFunction*)scriptFunction);
}
else
{
SList<AsmJsScriptFunction*> * newList = Anew(debugTransitionAlloc, SList<AsmJsScriptFunction*>, debugTransitionAlloc);
asmJsEnvironmentMap->AddNew(env, newList);
newList->Push((AsmJsScriptFunction*)scriptFunction);
}
}
}
#endif
/*static*/
void ScriptContext::RecyclerFunctionCallbackForDebugger(void *address, size_t size)
{
JavascriptFunction *pFunction = (JavascriptFunction *)address;
ScriptContext* scriptContext = pFunction->GetScriptContext();
if (scriptContext == nullptr || scriptContext->IsClosed())
{
// Can't enumerate from closed scriptcontext
return;
}
if (!scriptContext->IsEnumeratingRecyclerObjects())
{
return; // function not from enumerating script context
}
// Wrapped function are not allocated with the EnumClass bit
Assert(pFunction->GetFunctionInfo() != &JavascriptExternalFunction::EntryInfo::WrappedFunctionThunk);
JavascriptMethod entryPoint = pFunction->GetEntryPoint();
FunctionInfo * info = pFunction->GetFunctionInfo();
FunctionProxy * proxy = info->GetFunctionProxy();
if (proxy != info)
{
#ifdef ENABLE_SCRIPT_PROFILING
// Not a script function or, the thunk can deal with moving to the function body
Assert(proxy == nullptr || entryPoint == DefaultDeferredParsingThunk
|| entryPoint == ProfileDeferredParsingThunk
|| entryPoint == ProfileDeferredDeserializeThunk
|| entryPoint == CrossSite::ProfileThunk
|| entryPoint == DefaultDeferredDeserializeThunk
|| entryPoint == CrossSite::DefaultThunk);
#else
// Not a script function or, the thunk can deal with moving to the function body
Assert(proxy == nullptr || entryPoint == DefaultDeferredParsingThunk
|| entryPoint == DefaultDeferredDeserializeThunk
|| entryPoint == CrossSite::DefaultThunk);
#endif
// Replace entry points for built-ins/external/winrt functions so that we can wrap them with try-catch for "continue after exception".
if (!pFunction->IsScriptFunction() && IsExceptionWrapperForBuiltInsEnabled(scriptContext))
{
#ifdef ENABLE_SCRIPT_PROFILING
if (scriptContext->IsScriptContextInDebugMode())
{
// We are attaching.
// For built-ins, WinRT and DOM functions which are already in recycler, change entry points to route to debug/profile thunk.
ScriptContext::SetEntryPointToProfileThunk(pFunction);
}
else
{
// We are detaching.
// For built-ins, WinRT and DOM functions which are already in recycler, restore entry points to original.
if (!scriptContext->IsProfiling())
{
ScriptContext::RestoreEntryPointFromProfileThunk(pFunction);
}
// If we are profiling, don't change anything.
}
#else
AssertMsg(false, "Profiling needs to be enabled to change thunks for debugging");
#endif
}
return;
}
if (!proxy->IsFunctionBody())
{
// REVIEW: why we still have function that is still deferred?
return;
}
Assert(pFunction->IsScriptFunction());
// Excluding the internal library code, which is not debuggable already
if (!proxy->GetUtf8SourceInfo()->GetIsLibraryCode())
{
// Reset the constructor cache to default, so that it will not pick up the cached type, created before debugging.
// Look bug: 301517
pFunction->ResetConstructorCacheToDefault();
}
if (ScriptFunctionWithInlineCache::Is(pFunction))
{
ScriptFunctionWithInlineCache::FromVar(pFunction)->ClearInlineCacheOnFunctionObject();
}
// We should have force parsed the function, and have a function body
FunctionBody * pBody = proxy->GetFunctionBody();
#ifdef ENABLE_DEBUG_CONFIG_OPTIONS
if (scriptContext->IsScriptContextInDebugMode() && !proxy->GetUtf8SourceInfo()->GetIsLibraryCode() && !pBody->IsInDebugMode())
{
// Identifying if any function escaped for not being in debug mode. (This can be removed as a part of TFS : 935011)
Throw::FatalInternalError();
}
#endif
ScriptFunction * scriptFunction = ScriptFunction::FromVar(pFunction);
#ifdef ASMJS_PLAT
scriptContext->TransitionEnvironmentForDebugger(scriptFunction);
#endif
JavascriptMethod newEntryPoint;
if (CrossSite::IsThunk(entryPoint))
{
// Can't change from cross-site to non-cross-site, but still need to update the e.p.info on ScriptFunctionType.
newEntryPoint = entryPoint;
}
else
{
newEntryPoint = pBody->GetDirectEntryPoint(pBody->GetDefaultFunctionEntryPointInfo());
}
scriptFunction->ChangeEntryPoint(pBody->GetDefaultFunctionEntryPointInfo(), newEntryPoint);
}
#ifdef ENABLE_SCRIPT_PROFILING
void ScriptContext::RecyclerEnumClassEnumeratorCallback(void *address, size_t size)
{
// TODO: we are assuming its function because for now we are enumerating only on functions
// In future if the RecyclerNewEnumClass is used of Recyclable objects or Dynamic object, we would need a check if it is function
JavascriptFunction *pFunction = (JavascriptFunction *)address;
ScriptContext* scriptContext = pFunction->GetScriptContext();
if (scriptContext == nullptr || scriptContext->IsClosed())
{
// Can't enumerate from closed scriptcontext
return;
}
if (!scriptContext->IsEnumeratingRecyclerObjects())
{
return; // function not from enumerating script context
}
if (!scriptContext->IsTraceDomCall() && (pFunction->IsExternalFunction() || pFunction->IsWinRTFunction()))
{
return;
}
if (scriptContext->IsEnumerateNonUserFunctionsOnly() && pFunction->IsScriptFunction())
{
return;
}
// Wrapped function are not allocated with the EnumClass bit
Assert(pFunction->GetFunctionInfo() != &JavascriptExternalFunction::EntryInfo::WrappedFunctionThunk);
JavascriptMethod entryPoint = pFunction->GetEntryPoint();
FunctionProxy *proxy = pFunction->GetFunctionProxy();
if (proxy != NULL)
{
#if ENABLE_DEBUG_CONFIG_OPTIONS
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
#endif
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::RecyclerEnumClassEnumeratorCallback\n"));
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("\tFunctionProxy : 0x%08X, FunctionNumber : %s, DeferredParseAttributes : %d, EntryPoint : 0x%08X"),
(DWORD_PTR)proxy, proxy->GetDebugNumberSet(debugStringBuffer), proxy->GetAttributes(), (DWORD_PTR)entryPoint);
#if ENABLE_NATIVE_CODEGEN
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u(" (IsIntermediateCodeGenThunk : %s, isNative : %s)\n"),
IsTrueOrFalse(IsIntermediateCodeGenThunk(entryPoint)), IsTrueOrFalse(scriptContext->IsNativeAddress(entryPoint)));
#endif
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("\n"));
FunctionInfo * info = pFunction->GetFunctionInfo();
if (proxy != info)
{
// The thunk can deal with moving to the function body
#ifdef ENABLE_SCRIPT_PROFILING
Assert(entryPoint == DefaultDeferredParsingThunk || entryPoint == ProfileDeferredParsingThunk
|| entryPoint == DefaultDeferredDeserializeThunk || entryPoint == ProfileDeferredDeserializeThunk
|| entryPoint == CrossSite::DefaultThunk || entryPoint == CrossSite::ProfileThunk);
#else
Assert(entryPoint == DefaultDeferredParsingThunk
|| entryPoint == DefaultDeferredDeserializeThunk
|| entryPoint == CrossSite::DefaultThunk);
#endif
Assert(!proxy->IsDeferred());
Assert(proxy->GetFunctionBody()->GetProfileSession() == proxy->GetScriptContext()->GetProfileSession());
return;
}
#if ENABLE_NATIVE_CODEGEN
if (!IsIntermediateCodeGenThunk(entryPoint) && entryPoint != DynamicProfileInfo::EnsureDynamicProfileInfoThunk)
#endif
{
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("\t\tJs::ScriptContext::GetProfileModeThunk : 0x%08X\n"), (DWORD_PTR)Js::ScriptContext::GetProfileModeThunk(entryPoint));
ScriptFunction * scriptFunction = ScriptFunction::FromVar(pFunction);
scriptFunction->ChangeEntryPoint(proxy->GetDefaultEntryPointInfo(), Js::ScriptContext::GetProfileModeThunk(entryPoint));
#if ENABLE_NATIVE_CODEGEN
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("\tUpdated entrypoint : 0x%08X (isNative : %s)\n"), (DWORD_PTR)pFunction->GetEntryPoint(), IsTrueOrFalse(scriptContext->IsNativeAddress(entryPoint)));
#endif
}
}
else
{
ScriptContext::SetEntryPointToProfileThunk(pFunction);
}
}
// static
void ScriptContext::SetEntryPointToProfileThunk(JavascriptFunction* function)
{
JavascriptMethod entryPoint = function->GetEntryPoint();
if (entryPoint == Js::CrossSite::DefaultThunk)
{
function->SetEntryPoint(Js::CrossSite::ProfileThunk);
}
else if (entryPoint != Js::CrossSite::ProfileThunk && entryPoint != ProfileEntryThunk)
{
function->SetEntryPoint(ProfileEntryThunk);
}
}
// static
void ScriptContext::RestoreEntryPointFromProfileThunk(JavascriptFunction* function)
{
JavascriptMethod entryPoint = function->GetEntryPoint();
if (entryPoint == Js::CrossSite::ProfileThunk)
{
function->SetEntryPoint(Js::CrossSite::DefaultThunk);
}
else if (entryPoint == ProfileEntryThunk)
{
function->SetEntryPoint(function->GetFunctionInfo()->GetOriginalEntryPoint());
}
}
JavascriptMethod ScriptContext::GetProfileModeThunk(JavascriptMethod entryPoint)
{
#if ENABLE_NATIVE_CODEGEN
Assert(!IsIntermediateCodeGenThunk(entryPoint));
#endif
if (entryPoint == DefaultDeferredParsingThunk || entryPoint == ProfileDeferredParsingThunk)
{
return ProfileDeferredParsingThunk;
}
if (entryPoint == DefaultDeferredDeserializeThunk || entryPoint == ProfileDeferredDeserializeThunk)
{
return ProfileDeferredDeserializeThunk;
}
if (CrossSite::IsThunk(entryPoint))
{
return CrossSite::ProfileThunk;
}
return ProfileEntryThunk;
}
#endif // ENABLE_SCRIPT_PROFILING
#if _M_IX86
__declspec(naked)
Var ScriptContext::ProfileModeDeferredParsingThunk(RecyclableObject* function, CallInfo callInfo, ...)
{
// Register functions
__asm
{
push ebp
mov ebp, esp
lea eax, [esp + 8]
push eax
call ScriptContext::ProfileModeDeferredParse
#ifdef _CONTROL_FLOW_GUARD
// verify that the call target is valid
mov ecx, eax
call[__guard_check_icall_fptr]
mov eax, ecx
#endif
pop ebp
// Although we don't restore ESP here on WinCE, this is fine because script profiler is not shipped for WinCE.
jmp eax
}
}
#elif defined(_M_X64) || defined(_M_ARM32_OR_ARM64)
// Do nothing: the implementation of ScriptContext::ProfileModeDeferredParsingThunk is declared (appropriately decorated) in
// Language\amd64\amd64_Thunks.asm and Language\arm\arm_Thunks.asm and Language\arm64\arm64_Thunks.asm respectively.
#else
Var ScriptContext::ProfileModeDeferredParsingThunk(RecyclableObject* function, CallInfo callInfo, ...)
{
Js::Throw::NotImplemented();
return nullptr;
}
#endif
Js::JavascriptMethod ScriptContext::ProfileModeDeferredParse(ScriptFunction ** functionRef)
{
#if ENABLE_DEBUG_CONFIG_OPTIONS
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
#endif
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::ProfileModeDeferredParse FunctionNumber : %s, startEntrypoint : 0x%08X\n"), (*functionRef)->GetFunctionProxy()->GetDebugNumberSet(debugStringBuffer), (*functionRef)->GetEntryPoint());
BOOL fParsed = FALSE;
JavascriptMethod entryPoint = Js::JavascriptFunction::DeferredParseCore(functionRef, fParsed);
#ifdef ENABLE_SCRIPT_PROFILING
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("\t\tIsParsed : %s, updatedEntrypoint : 0x%08X\n"), IsTrueOrFalse(fParsed), entryPoint);
//To get the scriptContext we only need the functionProxy
FunctionProxy *pRootBody = (*functionRef)->GetFunctionProxy();
ScriptContext *pScriptContext = pRootBody->GetScriptContext();
if (pScriptContext->IsProfiling() && !pRootBody->GetFunctionBody()->HasFunctionCompiledSent())
{
pScriptContext->RegisterScript(pRootBody, FALSE /*fRegisterScript*/);
}
// We can come to this function even though we have stopped profiling.
Assert(!pScriptContext->IsProfiling() || (*functionRef)->GetFunctionBody()->GetProfileSession() == pScriptContext->GetProfileSession());
#endif
return entryPoint;
}
#if _M_IX86
__declspec(naked)
Var ScriptContext::ProfileModeDeferredDeserializeThunk(RecyclableObject* function, CallInfo callInfo, ...)
{
// Register functions
__asm
{
push ebp
mov ebp, esp
push[esp + 8]
call ScriptContext::ProfileModeDeferredDeserialize
#ifdef _CONTROL_FLOW_GUARD
// verify that the call target is valid
mov ecx, eax
call[__guard_check_icall_fptr]
mov eax, ecx
#endif
pop ebp
// Although we don't restore ESP here on WinCE, this is fine because script profiler is not shipped for WinCE.
jmp eax
}
}
#elif defined(_M_X64) || defined(_M_ARM32_OR_ARM64)
// Do nothing: the implementation of ScriptContext::ProfileModeDeferredDeserializeThunk is declared (appropriately decorated) in
// Language\amd64\amd64_Thunks.asm and Language\arm\arm_Thunks.asm respectively.
#endif
Js::JavascriptMethod ScriptContext::ProfileModeDeferredDeserialize(ScriptFunction *function)
{
#if ENABLE_DEBUG_CONFIG_OPTIONS
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
#endif
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::ProfileModeDeferredDeserialize FunctionNumber : %s\n"), function->GetFunctionProxy()->GetDebugNumberSet(debugStringBuffer));
JavascriptMethod entryPoint = Js::JavascriptFunction::DeferredDeserialize(function);
#ifdef ENABLE_SCRIPT_PROFILING
//To get the scriptContext; we only need the FunctionProxy
FunctionProxy *pRootBody = function->GetFunctionProxy();
ScriptContext *pScriptContext = pRootBody->GetScriptContext();
if (pScriptContext->IsProfiling() && !pRootBody->GetFunctionBody()->HasFunctionCompiledSent())
{
pScriptContext->RegisterScript(pRootBody, FALSE /*fRegisterScript*/);
}
// We can come to this function even though we have stopped profiling.
Assert(!pScriptContext->IsProfiling() || function->GetFunctionBody()->GetProfileSession() == pScriptContext->GetProfileSession());
#endif
return entryPoint;
}
#ifdef ENABLE_SCRIPT_PROFILING
BOOL ScriptContext::GetProfileInfo(
JavascriptFunction* function,
PROFILER_TOKEN &scriptId,
PROFILER_TOKEN &functionId)
{
BOOL fCanProfile = (m_pProfileCallback != nullptr && m_fTraceFunctionCall);
if (!fCanProfile)
{
return FALSE;
}
Js::FunctionInfo* functionInfo = function->GetFunctionInfo();
if (functionInfo->GetAttributes() & FunctionInfo::DoNotProfile)
{
return FALSE;
}
Js::FunctionBody * functionBody = functionInfo->GetFunctionBody();
if (functionBody == nullptr)
{
functionId = GetFunctionNumber(functionInfo->GetOriginalEntryPoint());
if (functionId == -1)
{
// Dom Call
return m_fTraceDomCall && (m_pProfileCallback2 != nullptr);
}
else
{
// Builtin function
scriptId = BuiltInFunctionsScriptId;
return m_fTraceNativeFunctionCall;
}
}
else if (!functionBody->GetUtf8SourceInfo()->GetIsLibraryCode() || functionBody->IsPublicLibraryCode()) // user script or public library code
{
scriptId = (PROFILER_TOKEN)functionBody->GetUtf8SourceInfo()->GetSourceInfoId();
functionId = functionBody->GetFunctionNumber();
return TRUE;
}
return FALSE;
}
#endif // ENABLE_SCRIPT_PROFILING
bool ScriptContext::IsForceNoNative()
{
bool forceNoNative = false;
if (this->IsScriptContextInSourceRundownOrDebugMode())
{
forceNoNative = this->IsInterpreted();
}
else if (!Js::Configuration::Global.EnableJitInDebugMode())
{
forceNoNative = true;
this->ForceNoNative();
}
return forceNoNative;
}
void ScriptContext::InitializeDebugging()
{
if (!this->IsScriptContextInDebugMode()) // If we already in debug mode, we would have done below changes already.
{
this->GetDebugContext()->SetDebuggerMode(Js::DebuggerMode::Debugging);
if (this->IsScriptContextInDebugMode())
{
// Note: for this we need final IsInDebugMode and NativeCodeGen initialized,
// and inside EnsureScriptContext, which seems appropriate as well,
// it's too early as debugger manager is not registered, thus IsDebuggerEnvironmentAvailable is false.
this->RegisterDebugThunk(false/*calledDuringAttach*/);
// TODO: for launch scenario for external and WinRT functions it might be too late to register debug thunk here,
// as we need the thunk registered before FunctionInfo's for built-ins, that may throw, are created.
// Need to verify. If that's the case, one way would be to enumerate and fix all external/winRT thunks here.
}
}
}
// Combined profile/debug wrapper thunk.
// - used when we profile to send profile events
// - used when we debug, only used for built-in functions
// - used when we profile and debug
Var ScriptContext::DebugProfileProbeThunk(RecyclableObject* callable, CallInfo callInfo, ...)
{
#if defined(ENABLE_SCRIPT_DEBUGGING) || defined(ENABLE_SCRIPT_PROFILING)
RUNTIME_ARGUMENTS(args, callInfo);
JavascriptFunction* function = JavascriptFunction::FromVar(callable);
ScriptContext* scriptContext = function->GetScriptContext();
PROFILER_TOKEN scriptId = -1;
PROFILER_TOKEN functionId = -1;
bool functionEnterEventSent = false;
const bool isProfilingUserCode = scriptContext->GetThreadContext()->IsProfilingUserCode();
const bool isUserCode = !function->IsLibraryCode();
char16 *pwszExtractedFunctionName = NULL;
const char16 *pwszFunctionName = NULL;
HRESULT hrOfEnterEvent = S_OK;
// We can come here when profiling is not on
// e.g. User starts profiling, we update all thinks and then stop profiling - we don't update thunk
// So we still get this call
const bool fProfile = (isUserCode || isProfilingUserCode) // Only report user code or entry library code
&& scriptContext->GetProfileInfo(function, scriptId, functionId);
if (fProfile)
{
Js::FunctionBody *pBody = function->GetFunctionBody();
if (pBody != nullptr && !pBody->HasFunctionCompiledSent())
{
pBody->RegisterFunction(false/*changeThunk*/);
}
#if DEBUG
{ // scope
Assert(scriptContext->IsProfiling());
if (pBody && pBody->GetProfileSession() != pBody->GetScriptContext()->GetProfileSession())
{
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
OUTPUT_TRACE_DEBUGONLY(Js::ScriptProfilerPhase, _u("ScriptContext::ProfileProbeThunk, ProfileSession does not match (%d != %d), functionNumber : %s, functionName : %s\n"),
pBody->GetProfileSession(), pBody->GetScriptContext()->GetProfileSession(), pBody->GetDebugNumberSet(debugStringBuffer), pBody->GetDisplayName());
}
AssertMsg(pBody == NULL || pBody->GetProfileSession() == pBody->GetScriptContext()->GetProfileSession(), "Function info wasn't reported for this profile session");
}
#endif
if (functionId == -1)
{
Var sourceString = function->GetSourceString();
// SourceString will be null for the Js::BoundFunction, don't throw Enter/Exit notification in that case.
if (sourceString != NULL)
{
if (TaggedInt::Is(sourceString))
{
PropertyId nameId = TaggedInt::ToInt32(sourceString);
pwszFunctionName = scriptContext->GetPropertyString(nameId)->GetSz();
}
else
{
// it is string because user had called in toString extract name from it
Assert(JavascriptString::Is(sourceString));
const char16 *pwszToString = ((JavascriptString *)sourceString)->GetSz();
const char16 *pwszNameStart = wcsstr(pwszToString, _u(" "));
const char16 *pwszNameEnd = wcsstr(pwszToString, _u("("));
if (pwszNameStart == nullptr || pwszNameEnd == nullptr || ((int)(pwszNameEnd - pwszNameStart) <= 0))
{
int len = ((JavascriptString *)sourceString)->GetLength() + 1;
pwszExtractedFunctionName = new char16[len];
wcsncpy_s(pwszExtractedFunctionName, len, pwszToString, _TRUNCATE);
}
else
{
int len = (int)(pwszNameEnd - pwszNameStart);
AssertMsg(len > 0, "Allocating array with zero or negative length?");
pwszExtractedFunctionName = new char16[len];
wcsncpy_s(pwszExtractedFunctionName, len, pwszNameStart + 1, _TRUNCATE);
}
pwszFunctionName = pwszExtractedFunctionName;
}
functionEnterEventSent = true;
Assert(pwszFunctionName != NULL);
hrOfEnterEvent = scriptContext->OnDispatchFunctionEnter(pwszFunctionName);
}
}
else
{
hrOfEnterEvent = scriptContext->OnFunctionEnter(scriptId, functionId);
}
scriptContext->GetThreadContext()->SetIsProfilingUserCode(isUserCode); // Update IsProfilingUserCode state
}
Var aReturn = NULL;
JavascriptMethod origEntryPoint = function->GetFunctionInfo()->GetOriginalEntryPoint();
__try
{
Assert(!function->IsScriptFunction() || function->GetFunctionProxy());
// No need to wrap script functions, also can't if the wrapper is already on the stack.
// Treat "library code" script functions, such as Intl, as built-ins:
// use the wrapper when calling them, and do not reset the wrapper when calling them.
bool isDebugWrapperEnabled = scriptContext->IsScriptContextInDebugMode() && IsExceptionWrapperForBuiltInsEnabled(scriptContext);
bool useDebugWrapper =
isDebugWrapperEnabled &&
function->IsLibraryCode() &&
!AutoRegisterIgnoreExceptionWrapper::IsRegistered(scriptContext->GetThreadContext());
OUTPUT_VERBOSE_TRACE(Js::DebuggerPhase, _u("DebugProfileProbeThunk: calling function: %s isWrapperRegistered=%d useDebugWrapper=%d\n"),
function->GetFunctionInfo()->HasBody() ? function->GetFunctionBody()->GetDisplayName() : _u("built-in/library"), AutoRegisterIgnoreExceptionWrapper::IsRegistered(scriptContext->GetThreadContext()), useDebugWrapper);
if (scriptContext->IsScriptContextInDebugMode())
{
scriptContext->GetDebugContext()->GetProbeContainer()->StartRecordingCall();
}
if (useDebugWrapper)
{
// For native use wrapper and bail out on to ignore exception.
// Extract try-catch out of hot path in normal profile mode (presence of try-catch in a function is bad for perf).
aReturn = ProfileModeThunk_DebugModeWrapper(function, scriptContext, origEntryPoint, args);
}
else
{
if (isDebugWrapperEnabled && !function->IsLibraryCode())
{
// We want to ignore exception and continue into closest user/script function down on the stack.
// Thus, if needed, reset the wrapper for the time of this call,
// so that if there is library/helper call after script function, it will use try-catch.
// Can't use smart/destructor object here because of __try__finally.
ThreadContext* threadContext = scriptContext->GetThreadContext();
bool isOrigWrapperPresent = threadContext->GetDebugManager()->GetDebuggingFlags()->IsBuiltInWrapperPresent();
if (isOrigWrapperPresent)
{
threadContext->GetDebugManager()->GetDebuggingFlags()->SetIsBuiltInWrapperPresent(false);
}
__try
{
aReturn = JavascriptFunction::CallFunction<true>(function, origEntryPoint, args);
}
__finally
{
threadContext->GetDebugManager()->GetDebuggingFlags()->SetIsBuiltInWrapperPresent(isOrigWrapperPresent);
}
}
else
{
// Can we update return address to a thunk that sends Exit event and then jmp to entry instead of Calling it.
// Saves stack space and it might be something we would be doing anyway for handling profile.Start/stop
// which can come anywhere on the stack.
aReturn = JavascriptFunction::CallFunction<true>(function, origEntryPoint, args);
}
}
}
__finally
{
if (fProfile)
{
if (hrOfEnterEvent != ACTIVPROF_E_PROFILER_ABSENT)
{
if (functionId == -1)
{
// Check whether we have sent the Enter event or not.
if (functionEnterEventSent)
{
scriptContext->OnDispatchFunctionExit(pwszFunctionName);
if (pwszExtractedFunctionName != NULL)
{
delete[]pwszExtractedFunctionName;
}
}
}
else
{
scriptContext->OnFunctionExit(scriptId, functionId);
}
}
scriptContext->GetThreadContext()->SetIsProfilingUserCode(isProfilingUserCode); // Restore IsProfilingUserCode state
}
if (scriptContext->IsScriptContextInDebugMode())
{
scriptContext->GetDebugContext()->GetProbeContainer()->EndRecordingCall(aReturn, function);
}
}
return aReturn;
#else
return nullptr;
#endif // defined(ENABLE_SCRIPT_DEBUGGING) || defined(ENABLE_SCRIPT_PROFILING)
}
#ifdef ENABLE_SCRIPT_PROFILING
// Part of ProfileModeThunk which is called in debug mode (debug or debug & profile).
Var ScriptContext::ProfileModeThunk_DebugModeWrapper(JavascriptFunction* function, ScriptContext* scriptContext, JavascriptMethod entryPoint, Arguments& args)
{
AutoRegisterIgnoreExceptionWrapper autoWrapper(scriptContext->GetThreadContext());
Var aReturn = HelperOrLibraryMethodWrapper<true>(scriptContext, [=] {
return JavascriptFunction::CallFunction<true>(function, entryPoint, args);
});
return aReturn;
}
HRESULT ScriptContext::OnScriptCompiled(PROFILER_TOKEN scriptId, PROFILER_SCRIPT_TYPE type, IUnknown *pIDebugDocumentContext)
{
// TODO : can we do a delay send of these events or can we send an event before doing all this stuff that could calculate overhead?
Assert(m_pProfileCallback != NULL);
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::OnScriptCompiled scriptId : %d, ScriptType : %d\n"), scriptId, type);
HRESULT hr = S_OK;
if ((type == PROFILER_SCRIPT_TYPE_NATIVE && m_fTraceNativeFunctionCall) ||
(type != PROFILER_SCRIPT_TYPE_NATIVE && m_fTraceFunctionCall))
{
m_inProfileCallback = TRUE;
hr = m_pProfileCallback->ScriptCompiled(scriptId, type, pIDebugDocumentContext);
m_inProfileCallback = FALSE;
}
return hr;
}
HRESULT ScriptContext::OnFunctionCompiled(
PROFILER_TOKEN functionId,
PROFILER_TOKEN scriptId,
const WCHAR *pwszFunctionName,
const WCHAR *pwszFunctionNameHint,
IUnknown *pIDebugDocumentContext)
{
Assert(m_pProfileCallback != NULL);
#ifdef ENABLE_DEBUG_CONFIG_OPTIONS
if (scriptId != BuiltInFunctionsScriptId || Js::Configuration::Global.flags.Verbose)
{
OUTPUT_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::OnFunctionCompiled scriptId : %d, functionId : %d, FunctionName : %s, FunctionNameHint : %s\n"), scriptId, functionId, pwszFunctionName, pwszFunctionNameHint);
}
#endif
HRESULT hr = S_OK;
if ((scriptId == BuiltInFunctionsScriptId && m_fTraceNativeFunctionCall) ||
(scriptId != BuiltInFunctionsScriptId && m_fTraceFunctionCall))
{
m_inProfileCallback = TRUE;
hr = m_pProfileCallback->FunctionCompiled(functionId, scriptId, pwszFunctionName, pwszFunctionNameHint, pIDebugDocumentContext);
m_inProfileCallback = FALSE;
}
return hr;
}
HRESULT ScriptContext::OnFunctionEnter(PROFILER_TOKEN scriptId, PROFILER_TOKEN functionId)
{
if (m_pProfileCallback == NULL)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
OUTPUT_VERBOSE_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::OnFunctionEnter scriptId : %d, functionId : %d\n"), scriptId, functionId);
HRESULT hr = S_OK;
if ((scriptId == BuiltInFunctionsScriptId && m_fTraceNativeFunctionCall) ||
(scriptId != BuiltInFunctionsScriptId && m_fTraceFunctionCall))
{
m_inProfileCallback = TRUE;
hr = m_pProfileCallback->OnFunctionEnter(scriptId, functionId);
m_inProfileCallback = FALSE;
}
return hr;
}
HRESULT ScriptContext::OnFunctionExit(PROFILER_TOKEN scriptId, PROFILER_TOKEN functionId)
{
if (m_pProfileCallback == NULL)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
OUTPUT_VERBOSE_TRACE(Js::ScriptProfilerPhase, _u("ScriptContext::OnFunctionExit scriptId : %d, functionId : %d\n"), scriptId, functionId);
HRESULT hr = S_OK;
if ((scriptId == BuiltInFunctionsScriptId && m_fTraceNativeFunctionCall) ||
(scriptId != BuiltInFunctionsScriptId && m_fTraceFunctionCall))
{
m_inProfileCallback = TRUE;
hr = m_pProfileCallback->OnFunctionExit(scriptId, functionId);
m_inProfileCallback = FALSE;
}
return hr;
}
HRESULT ScriptContext::FunctionExitSenderThunk(PROFILER_TOKEN functionId, PROFILER_TOKEN scriptId, ScriptContext *pScriptContext)
{
return pScriptContext->OnFunctionExit(scriptId, functionId);
}
HRESULT ScriptContext::FunctionExitByNameSenderThunk(const char16 *pwszFunctionName, ScriptContext *pScriptContext)
{
return pScriptContext->OnDispatchFunctionExit(pwszFunctionName);
}
#endif // ENABLE_SCRIPT_PROFILING
Js::PropertyId ScriptContext::GetFunctionNumber(JavascriptMethod entryPoint)
{
return (m_pBuiltinFunctionIdMap == NULL) ? -1 : m_pBuiltinFunctionIdMap->Lookup(entryPoint, -1);
}
HRESULT ScriptContext::RegisterLibraryFunction(const char16 *pwszObjectName, const char16 *pwszFunctionName, Js::PropertyId functionPropertyId, JavascriptMethod entryPoint)
{
#ifdef ENABLE_SCRIPT_PROFILING
#if DEBUG
const char16 *pwszObjectNameFromProperty = const_cast<char16 *>(GetPropertyName(functionPropertyId)->GetBuffer());
if (GetPropertyName(functionPropertyId)->IsSymbol())
{
// The spec names functions whose property is a well known symbol as the description from the symbol
// wrapped in square brackets, so verify by skipping past first bracket
Assert(!wcsncmp(pwszFunctionName + 1, pwszObjectNameFromProperty, wcslen(pwszObjectNameFromProperty)));
Assert(wcslen(pwszFunctionName) == wcslen(pwszObjectNameFromProperty) + 2);
}
else
{
Assert(!wcscmp(pwszFunctionName, pwszObjectNameFromProperty));
}
Assert(m_pBuiltinFunctionIdMap != NULL);
#endif
// Create the propertyId as object.functionName if it is not global function
// the global functions would be recognized by just functionName
// e.g. with functionName, toString, depending on objectName, it could be Object.toString, or Date.toString
char16 szTempName[70];
if (pwszObjectName != NULL)
{
// Create name as "object.function"
swprintf_s(szTempName, 70, _u("%s.%s"), pwszObjectName, pwszFunctionName);
functionPropertyId = GetOrAddPropertyIdTracked(szTempName, (uint)wcslen(szTempName));
}
Js::PropertyId cachedFunctionId;
bool keyFound = m_pBuiltinFunctionIdMap->TryGetValue(entryPoint, &cachedFunctionId);
if (keyFound)
{
// Entry point is already in the map
if (cachedFunctionId != functionPropertyId)
{
// This is the scenario where we could be using same function for multiple builtin functions
// e.g. Error.toString, WinRTError.toString etc.
// We would ignore these extra entrypoints because while profiling, identifying which object's toString is too costly for its worth
return S_OK;
}
// else is the scenario where map was created by earlier profiling session and we are yet to send function compiled for this session
}
else
{
#if DBG
m_pBuiltinFunctionIdMap->MapUntil([&](JavascriptMethod, Js::PropertyId propertyId) -> bool
{
if (functionPropertyId == propertyId)
{
Assert(false);
return true;
}
return false;
});
#endif
// throws, this must always be in a function that handles OOM
m_pBuiltinFunctionIdMap->Add(entryPoint, functionPropertyId);
}
// Use name with "Object." if its not a global function
if (pwszObjectName != NULL)
{
return OnFunctionCompiled(functionPropertyId, BuiltInFunctionsScriptId, szTempName, NULL, NULL);
}
else
{
return OnFunctionCompiled(functionPropertyId, BuiltInFunctionsScriptId, pwszFunctionName, NULL, NULL);
}
#else
return S_OK;
#endif // ENABLE_SCRIPT_PROFILING
}
void ScriptContext::BindReference(void * addr)
{
Assert(!this->isClosed);
Assert(this->guestArena);
Assert(recycler->IsValidObject(addr));
#if DBG
Assert(!bindRef.ContainsKey(addr)); // Make sure we don't bind the same pointer twice
bindRef.AddNew(addr);
#endif
javascriptLibrary->BindReference(addr);
#ifdef RECYCLER_PERF_COUNTERS
this->bindReferenceCount++;
RECYCLER_PERF_COUNTER_INC(BindReference);
#endif
}
#ifdef PROFILE_STRINGS
StringProfiler* ScriptContext::GetStringProfiler()
{
return stringProfiler;
}
#endif
void ScriptContext::FreeFunctionEntryPoint(Js::JavascriptMethod method)
{
#if ENABLE_NATIVE_CODEGEN
FreeNativeCodeGenAllocation(this, method);
#endif
}
void ScriptContext::RegisterProtoInlineCache(InlineCache *pCache, PropertyId propId)
{
hasProtoOrStoreFieldInlineCache = true;
threadContext->RegisterProtoInlineCache(pCache, propId);
}
void ScriptContext::InvalidateProtoCaches(const PropertyId propertyId)
{
threadContext->InvalidateProtoInlineCaches(propertyId);
// Because setter inline caches get registered in the store field chain, we must invalidate that
// chain whenever we invalidate the proto chain.
threadContext->InvalidateStoreFieldInlineCaches(propertyId);
#if ENABLE_NATIVE_CODEGEN
threadContext->InvalidatePropertyGuards(propertyId);
#endif
threadContext->InvalidateProtoTypePropertyCaches(propertyId);
}
void ScriptContext::InvalidateAllProtoCaches()
{
threadContext->InvalidateAllProtoInlineCaches();
// Because setter inline caches get registered in the store field chain, we must invalidate that
// chain whenever we invalidate the proto chain.
threadContext->InvalidateAllStoreFieldInlineCaches();
#if ENABLE_NATIVE_CODEGEN
threadContext->InvalidateAllPropertyGuards();
#endif
threadContext->InvalidateAllProtoTypePropertyCaches();
}
void ScriptContext::RegisterStoreFieldInlineCache(InlineCache *pCache, PropertyId propId)
{
hasProtoOrStoreFieldInlineCache = true;
threadContext->RegisterStoreFieldInlineCache(pCache, propId);
}
void ScriptContext::InvalidateStoreFieldCaches(const PropertyId propertyId)
{
threadContext->InvalidateStoreFieldInlineCaches(propertyId);
#if ENABLE_NATIVE_CODEGEN
threadContext->InvalidatePropertyGuards(propertyId);
#endif
}
void ScriptContext::InvalidateAllStoreFieldCaches()
{
threadContext->InvalidateAllStoreFieldInlineCaches();
}
void ScriptContext::RegisterIsInstInlineCache(Js::IsInstInlineCache * cache, Js::Var function)
{
Assert(JavascriptFunction::FromVar(function)->GetScriptContext() == this);
hasIsInstInlineCache = true;
threadContext->RegisterIsInstInlineCache(cache, function);
}
#if DBG
bool ScriptContext::IsIsInstInlineCacheRegistered(Js::IsInstInlineCache * cache, Js::Var function)
{
return threadContext->IsIsInstInlineCacheRegistered(cache, function);
}
#endif
void ScriptContext::CleanSourceListInternal(bool calledDuringMark)
{
bool fCleanupDocRequired = false;
for (int i = 0; i < sourceList->Count(); i++)
{
if (this->sourceList->IsItemValid(i))
{
RecyclerWeakReference<Utf8SourceInfo>* sourceInfoWeakRef = this->sourceList->Item(i);
Utf8SourceInfo* strongRef = nullptr;
if (calledDuringMark)
{
strongRef = sourceInfoWeakRef->FastGet();
}
else
{
strongRef = sourceInfoWeakRef->Get();
}
if (strongRef == nullptr)
{
this->sourceList->RemoveAt(i);
fCleanupDocRequired = true;
}
}
}
#ifdef ENABLE_SCRIPT_PROFILING
// If the sourceList got changed, in we need to refresh the nondebug document list in the profiler mode.
if (fCleanupDocRequired && m_pProfileCallback != NULL)
{
Assert(CleanupDocumentContext != NULL);
CleanupDocumentContext(this);
}
#endif // ENABLE_SCRIPT_PROFILING
}
void ScriptContext::ClearScriptContextCaches()
{
// Prevent reentrancy for the following work, which is not required to be done on every call to this function including
// reentrant calls
if (this->isPerformingNonreentrantWork || !this->hasUsedInlineCache)
{
return;
}
class AutoCleanup
{
private:
ScriptContext *const scriptContext;
public:
AutoCleanup(ScriptContext *const scriptContext) : scriptContext(scriptContext)
{
scriptContext->isPerformingNonreentrantWork = true;
}
~AutoCleanup()
{
scriptContext->isPerformingNonreentrantWork = false;
}
} autoCleanup(this);
if (this->isScriptContextActuallyClosed)
{
return;
}
Assert(this->guestArena);
Assert(this->cache);
if (EnableEvalMapCleanup())
{
// The eval map is not re-entrant, so make sure it's not in the middle of adding an entry
// Also, don't clean the eval map if the debugger is attached
if (!this->IsScriptContextInDebugMode())
{
if (this->cache->evalCacheDictionary != nullptr)
{
this->CleanDynamicFunctionCache<Js::EvalCacheTopLevelDictionary>(this->cache->evalCacheDictionary->GetDictionary());
}
if (this->cache->indirectEvalCacheDictionary != nullptr)
{
this->CleanDynamicFunctionCache<Js::EvalCacheTopLevelDictionary>(this->cache->indirectEvalCacheDictionary->GetDictionary());
}
if (this->cache->newFunctionCache != nullptr)
{
this->CleanDynamicFunctionCache<Js::NewFunctionCache>(this->cache->newFunctionCache);
}
if (this->hostScriptContext != nullptr)
{
this->hostScriptContext->CleanDynamicCodeCache();
}
}
}
if (REGEX_CONFIG_FLAG(DynamicRegexMruListSize) > 0)
{
GetDynamicRegexMap()->RemoveRecentlyUnusedItems();
}
CleanSourceListInternal(true);
}
void ScriptContext::ClearInlineCaches()
{
if (this->hasUsedInlineCache)
{
GetInlineCacheAllocator()->ZeroAll();
this->hasUsedInlineCache = false;
this->hasProtoOrStoreFieldInlineCache = false;
}
Assert(GetInlineCacheAllocator()->IsAllZero());
}
void ScriptContext::ClearIsInstInlineCaches()
{
if (this->hasIsInstInlineCache)
{
GetIsInstInlineCacheAllocator()->ZeroAll();
this->hasIsInstInlineCache = false;
}
Assert(GetIsInstInlineCacheAllocator()->IsAllZero());
}
#ifdef PERSISTENT_INLINE_CACHES
void ScriptContext::ClearInlineCachesWithDeadWeakRefs()
{
if (this->hasUsedInlineCache)
{
GetInlineCacheAllocator()->ClearCachesWithDeadWeakRefs(this->recycler);
Assert(GetInlineCacheAllocator()->HasNoDeadWeakRefs(this->recycler));
}
}
#endif
#if ENABLE_NATIVE_CODEGEN
void ScriptContext::RegisterConstructorCache(Js::PropertyId propertyId, Js::ConstructorCache* cache)
{
this->threadContext->RegisterConstructorCache(propertyId, cache);
}
#endif
void ScriptContext::RegisterPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext()
{
Assert(!IsClosed());
if (registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext == nullptr)
{
DoRegisterPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext();
}
}
void ScriptContext::DoRegisterPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext()
{
Assert(!IsClosed());
Assert(registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext == nullptr);
// this call may throw OOM
registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext = threadContext->RegisterPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext(this);
}
void ScriptContext::ClearPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesCaches()
{
Assert(registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext != nullptr);
if (!isFinalized)
{
javascriptLibrary->NoPrototypeChainsAreEnsuredToHaveOnlyWritableDataProperties();
}
// Caller will unregister the script context from the thread context
registeredPrototypeChainEnsuredToHaveOnlyWritableDataPropertiesScriptContext = nullptr;
}
JavascriptString * ScriptContext::GetLastNumberToStringRadix10(double value)
{
if (value == lastNumberToStringRadix10)
{
return cache->lastNumberToStringRadix10String;
}
return nullptr;
}
void
ScriptContext::SetLastNumberToStringRadix10(double value, JavascriptString * str)
{
lastNumberToStringRadix10 = value;
cache->lastNumberToStringRadix10String = str;
}
bool ScriptContext::GetLastUtcTimeFromStr(JavascriptString * str, double& dbl)
{
Assert(str != nullptr);
if (str != cache->lastUtcTimeFromStrString)
{
if (cache->lastUtcTimeFromStrString == nullptr
|| !JavascriptString::Equals(str, cache->lastUtcTimeFromStrString))
{
return false;
}
}
dbl = lastUtcTimeFromStr;
return true;
}
void
ScriptContext::SetLastUtcTimeFromStr(JavascriptString * str, double value)
{
lastUtcTimeFromStr = value;
cache->lastUtcTimeFromStrString = str;
}
#if ENABLE_NATIVE_CODEGEN
BOOL ScriptContext::IsNativeAddress(void * codeAddr)
{
return this->GetThreadContext()->IsNativeAddress(codeAddr);
}
#endif
bool ScriptContext::SetDispatchProfile(bool fSet, JavascriptMethod dispatchInvoke)
{
if (!fSet)
{
this->javascriptLibrary->SetDispatchProfile(false, dispatchInvoke);
return true;
}
#ifdef ENABLE_SCRIPT_PROFILING
else if (m_fTraceDomCall)
{
this->javascriptLibrary->SetDispatchProfile(true, dispatchInvoke);
return true;
}
#endif // ENABLE_SCRIPT_PROFILING
return false;
}
#ifdef ENABLE_SCRIPT_PROFILING
HRESULT ScriptContext::OnDispatchFunctionEnter(const WCHAR *pwszFunctionName)
{
if (m_pProfileCallback2 == NULL)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
HRESULT hr = S_OK;
if (m_fTraceDomCall)
{
m_inProfileCallback = TRUE;
hr = m_pProfileCallback2->OnFunctionEnterByName(pwszFunctionName, PROFILER_SCRIPT_TYPE_DOM);
m_inProfileCallback = FALSE;
}
return hr;
}
HRESULT ScriptContext::OnDispatchFunctionExit(const WCHAR *pwszFunctionName)
{
if (m_pProfileCallback2 == NULL)
{
return ACTIVPROF_E_PROFILER_ABSENT;
}
HRESULT hr = S_OK;
if (m_fTraceDomCall)
{
m_inProfileCallback = TRUE;
hr = m_pProfileCallback2->OnFunctionExitByName(pwszFunctionName, PROFILER_SCRIPT_TYPE_DOM);
m_inProfileCallback = FALSE;
}
return hr;
}
#endif // ENABLE_SCRIPT_PROFILING
void ScriptContext::SetBuiltInLibraryFunction(JavascriptMethod entryPoint, JavascriptFunction* function)
{
if (!isClosed)
{
if (builtInLibraryFunctions == NULL)
{
Assert(this->recycler);
builtInLibraryFunctions = RecyclerNew(this->recycler, BuiltInLibraryFunctionMap, this->recycler);
cache->builtInLibraryFunctions = builtInLibraryFunctions;
}
builtInLibraryFunctions->Item(entryPoint, function);
}
}
JavascriptFunction* ScriptContext::GetBuiltInLibraryFunction(JavascriptMethod entryPoint)
{
JavascriptFunction * function = NULL;
if (builtInLibraryFunctions)
{
builtInLibraryFunctions->TryGetValue(entryPoint, &function);
}
return function;
}
#ifdef ENABLE_DOM_FAST_PATH
DOMFastPathIRHelperMap* ScriptContext::EnsureDOMFastPathIRHelperMap()
{
if (domFastPathIRHelperMap == nullptr)
{
// Anew throws if it OOMs, so the caller into this function needs to handle that exception
domFastPathIRHelperMap = Anew(GeneralAllocator(), DOMFastPathIRHelperMap,
GeneralAllocator(), 17); // initial capacity set to 17; unlikely to grow much bigger.
}
return domFastPathIRHelperMap;
}
#endif
#if ENABLE_PROFILE_INFO
void ScriptContext::AddDynamicProfileInfo(FunctionBody * functionBody, WriteBarrierPtr<DynamicProfileInfo>* dynamicProfileInfo)
{
Assert(functionBody->GetScriptContext() == this);
Assert(functionBody->HasValidSourceInfo());
DynamicProfileInfo * newDynamicProfileInfo = *dynamicProfileInfo;
// If it is a dynamic script - we should create a profile info bound to the threadContext for its lifetime.
SourceContextInfo* sourceContextInfo = functionBody->GetSourceContextInfo();
SourceDynamicProfileManager* profileManager = sourceContextInfo->sourceDynamicProfileManager;
if (sourceContextInfo->IsDynamic())
{
if (profileManager != nullptr)
{
// There is an in-memory cache and dynamic profile info is coming from there
if (newDynamicProfileInfo == nullptr)
{
newDynamicProfileInfo = DynamicProfileInfo::New(this->GetRecycler(), functionBody, true /* persistsAcrossScriptContexts */);
profileManager->UpdateDynamicProfileInfo(functionBody->GetLocalFunctionId(), newDynamicProfileInfo);
*dynamicProfileInfo = newDynamicProfileInfo;
}
profileManager->MarkAsExecuted(functionBody->GetLocalFunctionId());
newDynamicProfileInfo->UpdateFunctionInfo(functionBody, this->GetRecycler());
}
else
{
if (newDynamicProfileInfo == nullptr)
{
newDynamicProfileInfo = functionBody->AllocateDynamicProfile();
}
*dynamicProfileInfo = newDynamicProfileInfo;
}
}
else
{
if (newDynamicProfileInfo == nullptr)
{
newDynamicProfileInfo = functionBody->AllocateDynamicProfile();
*dynamicProfileInfo = newDynamicProfileInfo;
}
Assert(functionBody->GetInterpretedCount() == 0);
#if DBG_DUMP || defined(DYNAMIC_PROFILE_STORAGE) || defined(RUNTIME_DATA_COLLECTION)
if (profileInfoList)
{
profileInfoList->Prepend(this->GetRecycler(), newDynamicProfileInfo);
}
#endif
if (!startupComplete)
{
Assert(profileManager);
profileManager->MarkAsExecuted(functionBody->GetLocalFunctionId());
}
}
Assert(*dynamicProfileInfo != nullptr);
}
#endif
CharClassifier const * ScriptContext::GetCharClassifier(void) const
{
return this->charClassifier;
}
void ScriptContext::OnStartupComplete()
{
JS_ETW(EventWriteJSCRIPT_ON_STARTUP_COMPLETE(this));
SaveStartupProfileAndRelease();
}
void ScriptContext::SaveStartupProfileAndRelease(bool isSaveOnClose)
{
// No need to save profiler info in JSRT scenario at this time.
if (GetThreadContext()->IsJSRT())
{
return;
}
if (!startupComplete && this->cache->sourceContextInfoMap)
{
#if ENABLE_PROFILE_INFO
this->cache->sourceContextInfoMap->Map([&](DWORD_PTR dwHostSourceContext, SourceContextInfo* info)
{
Assert(info->sourceDynamicProfileManager);
uint bytesWritten = info->sourceDynamicProfileManager->SaveToProfileCacheAndRelease(info);
if (bytesWritten > 0)
{
JS_ETW(EventWriteJSCRIPT_PROFILE_SAVE(info->dwHostSourceContext, this, bytesWritten, isSaveOnClose));
OUTPUT_TRACE(Js::DynamicProfilePhase, _u("Profile saving succeeded\n"));
}
});
#endif
}
startupComplete = true;
}
void ScriptContextBase::ClearGlobalObject()
{
#if ENABLE_NATIVE_CODEGEN
ScriptContext* scriptContext = static_cast<ScriptContext*>(this);
Assert(scriptContext->IsClosedNativeCodeGenerator());
#endif
globalObject = nullptr;
javascriptLibrary = nullptr;
}
void ScriptContext::SetFastDOMenabled()
{
fastDOMenabled = true; Assert(globalObject->GetDirectHostObject() != NULL);
}
#if DYNAMIC_INTERPRETER_THUNK
JavascriptMethod ScriptContext::GetNextDynamicAsmJsInterpreterThunk(PVOID* ppDynamicInterpreterThunk)
{
#ifdef ASMJS_PLAT
return (JavascriptMethod)this->asmJsInterpreterThunkEmitter->GetNextThunk(ppDynamicInterpreterThunk);
#else
__debugbreak();
return nullptr;
#endif
}
JavascriptMethod ScriptContext::GetNextDynamicInterpreterThunk(PVOID* ppDynamicInterpreterThunk)
{
return (JavascriptMethod)this->interpreterThunkEmitter->GetNextThunk(ppDynamicInterpreterThunk);
}
BOOL ScriptContext::IsDynamicInterpreterThunk(void* address)
{
return this->interpreterThunkEmitter->IsInHeap(address);
}
void ScriptContext::ReleaseDynamicInterpreterThunk(BYTE* address, bool addtoFreeList)
{
this->interpreterThunkEmitter->Release(address, addtoFreeList);
}
void ScriptContext::ReleaseDynamicAsmJsInterpreterThunk(BYTE* address, bool addtoFreeList)
{
#ifdef ASMJS_PLAT
this->asmJsInterpreterThunkEmitter->Release(address, addtoFreeList);
#else
Assert(UNREACHED);
#endif
}
#endif
bool ScriptContext::IsExceptionWrapperForBuiltInsEnabled()
{
return ScriptContext::IsExceptionWrapperForBuiltInsEnabled(this);
}
// static
bool ScriptContext::IsExceptionWrapperForBuiltInsEnabled(ScriptContext* scriptContext)
{
Assert(scriptContext);
return CONFIG_FLAG(EnableContinueAfterExceptionWrappersForBuiltIns);
}
bool ScriptContext::IsExceptionWrapperForHelpersEnabled(ScriptContext* scriptContext)
{
Assert(scriptContext);
return CONFIG_FLAG(EnableContinueAfterExceptionWrappersForHelpers);
}
void ScriptContextBase::SetGlobalObject(GlobalObject *globalObject)
{
#if DBG
ScriptContext* scriptContext = static_cast<ScriptContext*>(this);
Assert(scriptContext->IsCloningGlobal() && !this->globalObject);
#endif
this->globalObject = globalObject;
}
void ConvertKey(const FastEvalMapString& src, EvalMapString& dest)
{
dest.str = src.str;
dest.strict = src.strict;
dest.moduleID = src.moduleID;
dest.hash = TAGHASH((hash_t)dest.str);
}
void ScriptContext::PrintStats()
{
#if ENABLE_PROFILE_INFO
#if DBG_DUMP
DynamicProfileInfo::DumpScriptContext(this);
#endif
#ifdef RUNTIME_DATA_COLLECTION
DynamicProfileInfo::DumpScriptContextToFile(this);
#endif
#endif
#ifdef PROFILE_TYPES
if (Configuration::Global.flags.ProfileTypes)
{
ProfileTypes();
}
#endif
#ifdef PROFILE_BAILOUT_RECORD_MEMORY
if (Configuration::Global.flags.ProfileBailOutRecordMemory)
{
Output::Print(_u("CodeSize: %6d\nBailOutRecord Size: %6d\nLocalOffsets Size: %6d\n"), codeSize, bailOutRecordBytes, bailOutOffsetBytes);
}
#endif
#ifdef PROFILE_OBJECT_LITERALS
if (Configuration::Global.flags.ProfileObjectLiteral)
{
ProfileObjectLiteral();
}
#endif
#ifdef PROFILE_STRINGS
if (stringProfiler != nullptr)
{
stringProfiler->PrintAll();
Adelete(MiscAllocator(), stringProfiler);
stringProfiler = nullptr;
}
#endif
#ifdef PROFILE_MEM
if (profileMemoryDump && MemoryProfiler::IsTraceEnabled())
{
MemoryProfiler::PrintAll();
#ifdef PROFILE_RECYCLER_ALLOC
if (Js::Configuration::Global.flags.TraceMemory.IsEnabled(Js::AllPhase)
|| Js::Configuration::Global.flags.TraceMemory.IsEnabled(Js::RunPhase))
{
GetRecycler()->PrintAllocStats();
}
#endif
}
#endif
#if DBG_DUMP
if (PHASE_STATS1(Js::ByteCodePhase))
{
Output::Print(_u(" Total Bytecode size: <%d, %d, %d> = %d\n"),
byteCodeDataSize,
byteCodeAuxiliaryDataSize,
byteCodeAuxiliaryContextDataSize,
byteCodeDataSize + byteCodeAuxiliaryDataSize + byteCodeAuxiliaryContextDataSize);
}
if (Configuration::Global.flags.BytecodeHist)
{
Output::Print(_u("ByteCode Histogram\n"));
Output::Print(_u("\n"));
uint total = 0;
uint unique = 0;
for (int j = 0; j < (int)OpCode::ByteCodeLast; j++)
{
total += byteCodeHistogram[j];
if (byteCodeHistogram[j] > 0)
{
unique++;
}
}
Output::Print(_u("%9u Total executed ops\n"), total);
Output::Print(_u("\n"));
uint max = UINT_MAX;
double pctcume = 0.0;
while (true)
{
uint upper = 0;
int index = -1;
for (int j = 0; j < (int)OpCode::ByteCodeLast; j++)
{
if (OpCodeUtil::IsValidOpcode((OpCode)j) && byteCodeHistogram[j] > upper && byteCodeHistogram[j] < max)
{
index = j;
upper = byteCodeHistogram[j];
}
}
if (index == -1)
{
break;
}
max = byteCodeHistogram[index];
for (OpCode j = (OpCode)0; j < OpCode::ByteCodeLast; j++)
{
if (OpCodeUtil::IsValidOpcode(j) && max == byteCodeHistogram[(int)j])
{
double pct = ((double)max) / total;
pctcume += pct;
Output::Print(_u("%9u %5.1lf %5.1lf %04x %s\n"), max, pct * 100, pctcume * 100, j, OpCodeUtil::GetOpCodeName(j));
}
}
}
Output::Print(_u("\n"));
Output::Print(_u("Unique opcodes: %d\n"), unique);
}
#endif
#if ENABLE_NATIVE_CODEGEN
#ifdef BGJIT_STATS
// We do not care about small script contexts without much activity - unless t
if (PHASE_STATS1(Js::BGJitPhase) && (this->interpretedCount > 50 || Js::Configuration::Global.flags.IsEnabled(Js::ForceFlag)))
{
uint loopJitCodeUsed = 0;
uint bucketSize1 = 20;
uint bucketSize2 = 100;
uint size1CutOffbucketId = 4;
uint totalBuckets[15] = { 0 };
uint nativeCodeBuckets[15] = { 0 };
uint usedNativeCodeBuckets[15] = { 0 };
uint rejits[15] = { 0 };
uint zeroInterpretedFunctions = 0;
uint oneInterpretedFunctions = 0;
uint nonZeroBytecodeFunctions = 0;
Output::Print(_u("Script Context: 0x%p Url: %s\n"), this, this->url);
FunctionBody* anyFunctionBody = this->FindFunction([](FunctionBody* body) { return body != nullptr; });
if (anyFunctionBody)
{
OUTPUT_VERBOSE_STATS(Js::BGJitPhase, _u("Function list\n"));
OUTPUT_VERBOSE_STATS(Js::BGJitPhase, _u("===============================\n"));
OUTPUT_VERBOSE_STATS(Js::BGJitPhase, _u("%-24s, %-8s, %-10s, %-10s, %-10s, %-10s, %-10s\n"), _u("Function"), _u("InterpretedCount"), _u("ByteCodeInLoopSize"), _u("ByteCodeSize"), _u("IsJitted"), _u("IsUsed"), _u("NativeCodeSize"));
this->MapFunction([&](FunctionBody* body)
{
bool isNativeCode = false;
// Filtering interpreted count lowers a lot of noise
if (body->GetInterpretedCount() > 1 || Js::Configuration::Global.flags.IsEnabled(Js::ForceFlag))
{
body->MapEntryPoints([&](uint entryPointIndex, FunctionEntryPointInfo* entryPoint)
{
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
char rejit = entryPointIndex > 0 ? '*' : ' ';
isNativeCode = entryPoint->IsNativeCode() | isNativeCode;
OUTPUT_VERBOSE_STATS(Js::BGJitPhase, _u("%-20s %16s %c, %8d , %10d , %10d, %-10s, %-10s, %10d\n"),
body->GetExternalDisplayName(),
body->GetDebugNumberSet(debugStringBuffer),
rejit,
body->GetInterpretedCount(),
body->GetByteCodeInLoopCount(),
body->GetByteCodeCount(),
entryPoint->IsNativeCode() ? _u("Jitted") : _u("Interpreted"),
body->GetNativeEntryPointUsed() ? _u("Used") : _u("NotUsed"),
entryPoint->IsNativeCode() ? entryPoint->GetCodeSize() : 0);
});
}
if (body->GetInterpretedCount() == 0)
{
zeroInterpretedFunctions++;
if (body->GetByteCodeCount() > 0)
{
nonZeroBytecodeFunctions++;
}
}
else if (body->GetInterpretedCount() == 1)
{
oneInterpretedFunctions++;
}
// Generate a histogram using interpreted counts.
uint bucket;
uint intrpCount = body->GetInterpretedCount();
if (intrpCount < 100)
{
bucket = intrpCount / bucketSize1;
}
else if (intrpCount < 1000)
{
bucket = size1CutOffbucketId + intrpCount / bucketSize2;
}
else
{
bucket = _countof(totalBuckets) - 1;
}
// Explicitly assume that the bucket count is less than the following counts (which are all equal)
// This is because min will return _countof(totalBuckets) - 1 if the count exceeds _countof(totalBuckets) - 1.
__analysis_assume(bucket < _countof(totalBuckets));
__analysis_assume(bucket < _countof(nativeCodeBuckets));
__analysis_assume(bucket < _countof(usedNativeCodeBuckets));
__analysis_assume(bucket < _countof(rejits));
totalBuckets[bucket]++;
if (isNativeCode)
{
nativeCodeBuckets[bucket]++;
if (body->GetNativeEntryPointUsed())
{
usedNativeCodeBuckets[bucket]++;
}
if (body->HasRejit())
{
rejits[bucket]++;
}
}
body->MapLoopHeaders([&](uint loopNumber, LoopHeader* header)
{
char16 loopBodyName[256];
body->GetLoopBodyName(loopNumber, loopBodyName, _countof(loopBodyName));
header->MapEntryPoints([&](int index, LoopEntryPointInfo * entryPoint)
{
if (entryPoint->IsNativeCode())
{
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
char rejit = index > 0 ? '*' : ' ';
OUTPUT_VERBOSE_STATS(Js::BGJitPhase, _u("%-20s %16s %c, %8d , %10d , %10d, %-10s, %-10s, %10d\n"),
loopBodyName,
body->GetDebugNumberSet(debugStringBuffer),
rejit,
header->interpretCount,
header->GetByteCodeCount(),
header->GetByteCodeCount(),
_u("Jitted"),
entryPoint->IsUsed() ? _u("Used") : _u("NotUsed"),
entryPoint->GetCodeSize());
if (entryPoint->IsUsed())
{
loopJitCodeUsed++;
}
}
});
});
});
}
Output::Print(_u("** SpeculativelyJitted: %6d FunctionsJitted: %6d JittedUsed: %6d Usage:%f ByteCodesJitted: %6d JitCodeUsed: %6d Usage: %f \n"),
speculativeJitCount, funcJITCount, funcJitCodeUsed, ((float)(funcJitCodeUsed) / funcJITCount) * 100, bytecodeJITCount, jitCodeUsed, ((float)(jitCodeUsed) / bytecodeJITCount) * 100);
Output::Print(_u("** LoopJITCount: %6d LoopJitCodeUsed: %6d Usage: %f\n"),
loopJITCount, loopJitCodeUsed, ((float)loopJitCodeUsed / loopJITCount) * 100);
Output::Print(_u("** TotalInterpretedCalls: %6d MaxFuncInterp: %6d InterpretedHighPri: %6d \n"),
interpretedCount, maxFuncInterpret, interpretedCallsHighPri);
Output::Print(_u("** ZeroInterpretedFunctions: %6d OneInterpretedFunctions: %6d ZeroInterpretedWithNonZeroBytecode: %6d \n "), zeroInterpretedFunctions, oneInterpretedFunctions, nonZeroBytecodeFunctions);
Output::Print(_u("** %-24s : %-10s %-10s %-10s %-10s %-10s\n"), _u("InterpretedCounts"), _u("Total"), _u("NativeCode"), _u("Used"), _u("Usage"), _u("Rejits"));
uint low = 0;
uint high = 0;
for (uint i = 0; i < _countof(totalBuckets); i++)
{
low = high;
if (i <= size1CutOffbucketId)
{
high = low + bucketSize1;
}
else if (i < (_countof(totalBuckets) - 1))
{
high = low + bucketSize2; }
else
{
high = 100000;
}
Output::Print(_u("** %10d - %10d : %10d %10d %10d %7.2f %10d\n"), low, high, totalBuckets[i], nativeCodeBuckets[i], usedNativeCodeBuckets[i], ((float)usedNativeCodeBuckets[i] / nativeCodeBuckets[i]) * 100, rejits[i]);
}
Output::Print(_u("\n\n"));
}
#endif
#ifdef REJIT_STATS
if (PHASE_STATS1(Js::ReJITPhase))
{
uint totalBailouts = 0;
uint totalRejits = 0;
WCHAR buf[256];
// Dump bailout data.
Output::Print(_u("%-40s %6s\n"), _u("Bailout Reason,"), _u("Count"));
bailoutReasonCounts->Map([&totalBailouts](uint kind, uint val) {
WCHAR buf[256];
totalBailouts += val;
if (val != 0)
{
swprintf_s(buf, _u("%S,"), GetBailOutKindName((IR::BailOutKind)kind));
Output::Print(_u("%-40s %6d\n"), buf, val);
}
});
Output::Print(_u("%-40s %6d\n"), _u("TOTAL,"), totalBailouts);
Output::Print(_u("\n\n"));
// Dump rejit data.
Output::Print(_u("%-40s %6s\n"), _u("Rejit Reason,"), _u("Count"));
for (uint i = 0; i < NumRejitReasons; ++i)
{
totalRejits += rejitReasonCounts[i];
if (rejitReasonCounts[i] != 0)
{
swprintf_s(buf, _u("%S,"), RejitReasonNames[i]);
Output::Print(_u("%-40s %6d\n"), buf, rejitReasonCounts[i]);
}
}
Output::Print(_u("%-40s %6d\n"), _u("TOTAL,"), totalRejits);
Output::Print(_u("\n\n"));
// If in verbose mode, dump data for each FunctionBody
if (Configuration::Global.flags.Verbose && rejitStatsMap != NULL)
{
// Aggregated data
Output::Print(_u("%-30s %14s %14s\n"), _u("Function (#),"), _u("Bailout Count,"), _u("Rejit Count"));
rejitStatsMap->Map([](Js::FunctionBody const *body, RejitStats *stats, RecyclerWeakReference<const Js::FunctionBody> const*) {
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
for (uint i = 0; i < NumRejitReasons; ++i)
stats->m_totalRejits += stats->m_rejitReasonCounts[i];
stats->m_bailoutReasonCounts->Map([stats](uint kind, uint val) {
stats->m_totalBailouts += val;
});
WCHAR buf[256];
swprintf_s(buf, _u("%s (%s),"), body->GetExternalDisplayName(), (const_cast<Js::FunctionBody*>(body))->GetDebugNumberSet(debugStringBuffer)); //TODO Kount
Output::Print(_u("%-30s %14d, %14d\n"), buf, stats->m_totalBailouts, stats->m_totalRejits);
});
Output::Print(_u("\n\n"));
// Per FunctionBody data
rejitStatsMap->Map([](Js::FunctionBody const *body, RejitStats *stats, RecyclerWeakReference<const Js::FunctionBody> const *) {
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
WCHAR buf[256];
swprintf_s(buf, _u("%s (%s),"), body->GetExternalDisplayName(), (const_cast<Js::FunctionBody*>(body))->GetDebugNumberSet(debugStringBuffer)); //TODO Kount
Output::Print(_u("%-30s\n\n"), buf);
// Dump bailout data
if (stats->m_totalBailouts != 0)
{
Output::Print(_u("%10sBailouts:\n"), _u(""));
stats->m_bailoutReasonCounts->Map([](uint kind, uint val) {
if (val != 0)
{
WCHAR buf[256];
swprintf_s(buf, _u("%S,"), GetBailOutKindName((IR::BailOutKind)kind));
Output::Print(_u("%10s%-40s %6d\n"), _u(""), buf, val);
}
});
}
Output::Print(_u("\n"));
// Dump rejit data.
if (stats->m_totalRejits != 0)
{
Output::Print(_u("%10sRejits:\n"), _u(""));
for (uint i = 0; i < NumRejitReasons; ++i)
{
if (stats->m_rejitReasonCounts[i] != 0)
{
swprintf_s(buf, _u("%S,"), RejitReasonNames[i]);
Output::Print(_u("%10s%-40s %6d\n"), _u(""), buf, stats->m_rejitReasonCounts[i]);
}
}
Output::Print(_u("\n\n"));
}
});
}
}
#endif
#ifdef FIELD_ACCESS_STATS
if (PHASE_STATS1(Js::ObjTypeSpecPhase))
{
FieldAccessStats globalStats;
if (this->fieldAccessStatsByFunctionNumber != nullptr)
{
this->fieldAccessStatsByFunctionNumber->Map([&globalStats](uint functionNumber, FieldAccessStatsEntry* entry)
{
FieldAccessStats functionStats;
entry->stats.Map([&functionStats](FieldAccessStatsPtr entryPointStats)
{
functionStats.Add(entryPointStats);
});
if (PHASE_VERBOSE_STATS1(Js::ObjTypeSpecPhase))
{
FunctionBody* functionBody = entry->functionBodyWeakRef->Get();
const char16* functionName = functionBody != nullptr ? functionBody->GetDisplayName() : _u("<unknown>");
Output::Print(_u("FieldAccessStats: function %s (#%u): inline cache stats:\n"), functionName, functionNumber);
Output::Print(_u(" overall: total %u, no profile info %u\n"), functionStats.totalInlineCacheCount, functionStats.noInfoInlineCacheCount);
Output::Print(_u(" mono: total %u, empty %u, cloned %u\n"),
functionStats.monoInlineCacheCount, functionStats.emptyMonoInlineCacheCount, functionStats.clonedMonoInlineCacheCount);
Output::Print(_u(" poly: total %u (high %u, low %u), null %u, empty %u, ignored %u, disabled %u, equivalent %u, non-equivalent %u, cloned %u\n"),
functionStats.polyInlineCacheCount, functionStats.highUtilPolyInlineCacheCount, functionStats.lowUtilPolyInlineCacheCount,
functionStats.nullPolyInlineCacheCount, functionStats.emptyPolyInlineCacheCount, functionStats.ignoredPolyInlineCacheCount, functionStats.disabledPolyInlineCacheCount,
functionStats.equivPolyInlineCacheCount, functionStats.nonEquivPolyInlineCacheCount, functionStats.clonedPolyInlineCacheCount);
}
globalStats.Add(&functionStats);
});
}
Output::Print(_u("FieldAccessStats: totals\n"));
Output::Print(_u(" overall: total %u, no profile info %u\n"), globalStats.totalInlineCacheCount, globalStats.noInfoInlineCacheCount);
Output::Print(_u(" mono: total %u, empty %u, cloned %u\n"),
globalStats.monoInlineCacheCount, globalStats.emptyMonoInlineCacheCount, globalStats.clonedMonoInlineCacheCount);
Output::Print(_u(" poly: total %u (high %u, low %u), null %u, empty %u, ignored %u, disabled %u, equivalent %u, non-equivalent %u, cloned %u\n"),
globalStats.polyInlineCacheCount, globalStats.highUtilPolyInlineCacheCount, globalStats.lowUtilPolyInlineCacheCount,
globalStats.nullPolyInlineCacheCount, globalStats.emptyPolyInlineCacheCount, globalStats.ignoredPolyInlineCacheCount, globalStats.disabledPolyInlineCacheCount,
globalStats.equivPolyInlineCacheCount, globalStats.nonEquivPolyInlineCacheCount, globalStats.clonedPolyInlineCacheCount);
}
#endif
#ifdef MISSING_PROPERTY_STATS
if (PHASE_STATS1(Js::MissingPropertyCachePhase))
{
Output::Print(_u("MissingPropertyStats: hits = %d, misses = %d, cache attempts = %d.\n"),
this->missingPropertyHits, this->missingPropertyMisses, this->missingPropertyCacheAttempts);
}
#endif
#ifdef INLINE_CACHE_STATS
if (PHASE_STATS1(Js::PolymorphicInlineCachePhase))
{
Output::Print(_u("%s,%s,%s,%s,%s,%s,%s,%s,%s\n"), _u("Function"), _u("Property"), _u("Kind"), _u("Accesses"), _u("Misses"), _u("Miss Rate"), _u("Collisions"), _u("Collision Rate"), _u("Slot Count"));
cacheDataMap->Map([this](Js::PolymorphicInlineCache const *cache, CacheData *data) {
char16 debugStringBuffer[MAX_FUNCTION_BODY_DEBUG_STRING_SIZE];
uint total = data->hits + data->misses;
char16 const *propName = this->threadContext->GetPropertyName(data->propertyId)->GetBuffer();
wchar funcName[1024];
swprintf_s(funcName, _u("%s (%s)"), cache->functionBody->GetExternalDisplayName(), cache->functionBody->GetDebugNumberSet(debugStringBuffer));
Output::Print(_u("%s,%s,%s,%d,%d,%f,%d,%f,%d\n"),
funcName,
propName,
data->isGetCache ? _u("get") : _u("set"),
total,
data->misses,
static_cast<float>(data->misses) / total,
data->collisions,
static_cast<float>(data->collisions) / total,
cache->GetSize()
);
});
}
#endif
#if ENABLE_REGEX_CONFIG_OPTIONS
if (regexStatsDatabase != 0)
regexStatsDatabase->Print(GetRegexDebugWriter());
#endif
OUTPUT_STATS(Js::EmitterPhase, _u("Script Context: 0x%p Url: %s\n"), this, this->url);
OUTPUT_STATS(Js::EmitterPhase, _u(" Total thread committed code size = %d\n"), this->GetThreadContext()->GetCodeSize());
OUTPUT_STATS(Js::ParsePhase, _u("Script Context: 0x%p Url: %s\n"), this, this->url);
OUTPUT_STATS(Js::ParsePhase, _u(" Total ThreadContext source size %d\n"), this->GetThreadContext()->GetSourceSize());
#endif
#ifdef ENABLE_BASIC_TELEMETRY
if (this->telemetry != nullptr)
{
// If an exception (e.g. out-of-memory) happens during InitializeAllocations then `this->telemetry` will be null and the Close method will still be called, hence this guard expression.
this->telemetry->OutputPrint();
}
#endif
Output::Flush();
}
void ScriptContext::SetNextPendingClose(ScriptContext * nextPendingClose) {
this->nextPendingClose = nextPendingClose;
}
#ifdef ENABLE_MUTATION_BREAKPOINT
bool ScriptContext::HasMutationBreakpoints()
{
if (this->GetDebugContext() != nullptr && this->GetDebugContext()->GetProbeContainer() != nullptr)
{
return this->GetDebugContext()->GetProbeContainer()->HasMutationBreakpoints();
}
return false;
}
void ScriptContext::InsertMutationBreakpoint(Js::MutationBreakpoint *mutationBreakpoint)
{
this->GetDebugContext()->GetProbeContainer()->InsertMutationBreakpoint(mutationBreakpoint);
}
#endif
#ifdef REJIT_STATS
void ScriptContext::LogDataForFunctionBody(Js::FunctionBody *body, uint idx, bool isRejit)
{
if (rejitStatsMap == NULL)
{
rejitStatsMap = RecyclerNew(this->recycler, RejitStatsMap, this->recycler);
BindReference(rejitStatsMap);
}
RejitStats *stats = NULL;
if (!rejitStatsMap->TryGetValue(body, &stats))
{
stats = Anew(GeneralAllocator(), RejitStats, this);
rejitStatsMap->Item(body, stats);
}
if (isRejit)
{
stats->m_rejitReasonCounts[idx]++;
}
else
{
if (!stats->m_bailoutReasonCounts->ContainsKey(idx))
{
stats->m_bailoutReasonCounts->Item(idx, 1);
}
else
{
uint val = stats->m_bailoutReasonCounts->Item(idx);
++val;
stats->m_bailoutReasonCounts->Item(idx, val);
}
}
}
void ScriptContext::LogRejit(Js::FunctionBody *body, uint reason)
{
Assert(reason < NumRejitReasons);
rejitReasonCounts[reason]++;
if (Js::Configuration::Global.flags.Verbose)
{
LogDataForFunctionBody(body, reason, true);
}
}
void ScriptContext::LogBailout(Js::FunctionBody *body, uint kind)
{
if (!bailoutReasonCounts->ContainsKey(kind))
{
bailoutReasonCounts->Item(kind, 1);
}
else
{
uint val = bailoutReasonCounts->Item(kind);
++val;
bailoutReasonCounts->Item(kind, val);
}
if (Js::Configuration::Global.flags.Verbose)
{
LogDataForFunctionBody(body, kind, false);
}
}
#endif
#ifdef ENABLE_BASIC_TELEMETRY
ScriptContextTelemetry& ScriptContext::GetTelemetry()
{
return *this->telemetry;
}
bool ScriptContext::HasTelemetry()
{
return this->telemetry != nullptr;
}
#endif
bool ScriptContext::IsScriptContextInNonDebugMode() const
{
if (this->debugContext != nullptr)
{
return this->GetDebugContext()->IsDebugContextInNonDebugMode();
}
return true;
}
bool ScriptContext::IsScriptContextInDebugMode() const
{
if (this->debugContext != nullptr)
{
return this->GetDebugContext()->IsDebugContextInDebugMode();
}
return false;
}
bool ScriptContext::IsScriptContextInSourceRundownOrDebugMode() const
{
if (this->debugContext != nullptr)
{
return this->GetDebugContext()->IsDebugContextInSourceRundownOrDebugMode();
}
return false;
}
bool ScriptContext::IsIntlEnabled()
{
if (GetConfig()->IsIntlEnabled())
{
#ifdef ENABLE_GLOBALIZATION
// This will try to load globalization dlls if not already loaded.
Js::DelayLoadWindowsGlobalization* globLibrary = GetThreadContext()->GetWindowsGlobalizationLibrary();
return globLibrary->HasGlobalizationDllLoaded();
#endif
}
return false;
}
#ifdef INLINE_CACHE_STATS
void ScriptContext::LogCacheUsage(Js::PolymorphicInlineCache *cache, bool isGetter, Js::PropertyId propertyId, bool hit, bool collision)
{
if (cacheDataMap == NULL)
{
cacheDataMap = RecyclerNew(this->recycler, CacheDataMap, this->recycler);
BindReference(cacheDataMap);
}
CacheData *data = NULL;
if (!cacheDataMap->TryGetValue(cache, &data))
{
data = Anew(GeneralAllocator(), CacheData);
cacheDataMap->Item(cache, data);
data->isGetCache = isGetter;
data->propertyId = propertyId;
}
Assert(data->isGetCache == isGetter);
Assert(data->propertyId == propertyId);
if (hit)
{
data->hits++;
}
else
{
data->misses++;
}
if (collision)
{
data->collisions++;
}
}
#endif
#ifdef FIELD_ACCESS_STATS
void ScriptContext::RecordFieldAccessStats(FunctionBody* functionBody, FieldAccessStatsPtr fieldAccessStats)
{
Assert(fieldAccessStats != nullptr);
if (!PHASE_STATS1(Js::ObjTypeSpecPhase))
{
return;
}
FieldAccessStatsEntry* entry;
if (!this->fieldAccessStatsByFunctionNumber->TryGetValue(functionBody->GetFunctionNumber(), &entry))
{
RecyclerWeakReference<FunctionBody>* functionBodyWeakRef;
this->recycler->FindOrCreateWeakReferenceHandle(functionBody, &functionBodyWeakRef);
entry = RecyclerNew(this->recycler, FieldAccessStatsEntry, functionBodyWeakRef, this->recycler);
this->fieldAccessStatsByFunctionNumber->AddNew(functionBody->GetFunctionNumber(), entry);
}
entry->stats.Prepend(fieldAccessStats);
}
#endif
#ifdef MISSING_PROPERTY_STATS
void ScriptContext::RecordMissingPropertyMiss()
{
this->missingPropertyMisses++;
}
void ScriptContext::RecordMissingPropertyHit()
{
this->missingPropertyHits++;
}
void ScriptContext::RecordMissingPropertyCacheAttempt()
{
this->missingPropertyCacheAttempts++;
}
#endif
bool ScriptContext::IsIntConstPropertyOnGlobalObject(Js::PropertyId propId)
{
return intConstPropsOnGlobalObject->ContainsKey(propId);
}
void ScriptContext::TrackIntConstPropertyOnGlobalObject(Js::PropertyId propertyId)
{
intConstPropsOnGlobalObject->AddNew(propertyId);
}
bool ScriptContext::IsIntConstPropertyOnGlobalUserObject(Js::PropertyId propertyId)
{
return intConstPropsOnGlobalUserObject->ContainsKey(propertyId) != NULL;
}
void ScriptContext::TrackIntConstPropertyOnGlobalUserObject(Js::PropertyId propertyId)
{
intConstPropsOnGlobalUserObject->AddNew(propertyId);
}
void ScriptContext::AddCalleeSourceInfoToList(Utf8SourceInfo* sourceInfo)
{
Assert(sourceInfo);
RecyclerWeakReference<Js::Utf8SourceInfo>* sourceInfoWeakRef = nullptr;
this->GetRecycler()->FindOrCreateWeakReferenceHandle(sourceInfo, &sourceInfoWeakRef);
Assert(sourceInfoWeakRef);
if (!calleeUtf8SourceInfoList)
{
Recycler *recycler = this->GetRecycler();
calleeUtf8SourceInfoList.Root(RecyclerNew(recycler, CalleeSourceList, recycler), recycler);
}
if (!calleeUtf8SourceInfoList->Contains(sourceInfoWeakRef))
{
calleeUtf8SourceInfoList->Add(sourceInfoWeakRef);
}
}
#ifdef ENABLE_JS_ETW
void ScriptContext::EmitStackTraceEvent(__in UINT64 operationID, __in USHORT maxFrameCount, bool emitV2AsyncStackEvent)
{
// If call root level is zero, there is no EntryExitRecord and the stack walk will fail.
if (GetThreadContext()->GetCallRootLevel() == 0)
{
return;
}
Assert(EventEnabledJSCRIPT_STACKTRACE() || EventEnabledJSCRIPT_ASYNCCAUSALITY_STACKTRACE_V2() || PHASE_TRACE1(Js::StackFramesEventPhase));
BEGIN_TEMP_ALLOCATOR(tempAllocator, this, _u("StackTraceEvent"))
{
JsUtil::List<StackFrameInfo, ArenaAllocator> stackFrames(tempAllocator);
Js::JavascriptStackWalker walker(this);
unsigned short nameBufferLength = 0;
Js::StringBuilder<ArenaAllocator> nameBuffer(tempAllocator);
nameBuffer.Reset();
OUTPUT_TRACE(Js::StackFramesEventPhase, _u("\nPosting stack trace via ETW:\n"));
ushort frameCount = walker.WalkUntil((ushort)maxFrameCount, [&](Js::JavascriptFunction* function, ushort frameIndex) -> bool
{
ULONG lineNumber = 0;
LONG columnNumber = 0;
UINT32 methodIdOrNameId = 0;
UINT8 isFrameIndex = 0; // FALSE
const WCHAR* name = nullptr;
if (function->IsScriptFunction() && !function->IsLibraryCode())
{
Js::FunctionBody * functionBody = function->GetFunctionBody();
functionBody->GetLineCharOffset(walker.GetByteCodeOffset(), &lineNumber, &columnNumber);
methodIdOrNameId = EtwTrace::GetFunctionId(functionBody);
name = functionBody->GetExternalDisplayName();
}
else
{
if (function->IsScriptFunction())
{
name = function->GetFunctionBody()->GetExternalDisplayName();
}
else
{
name = walker.GetCurrentNativeLibraryEntryName();
}
ushort nameLen = ProcessNameAndGetLength(&nameBuffer, name);
methodIdOrNameId = nameBufferLength;
// Keep track of the current length of the buffer. The next nameIndex will be at this position (+1 for each '\\', '\"', and ';' character added above).
nameBufferLength += nameLen;
isFrameIndex = 1; // TRUE;
}
StackFrameInfo frame((DWORD_PTR)function->GetScriptContext(),
(UINT32)lineNumber,
(UINT32)columnNumber,
methodIdOrNameId,
isFrameIndex);
OUTPUT_TRACE(Js::StackFramesEventPhase, _u("Frame : (%s : %u) (%s), LineNumber : %u, ColumnNumber : %u\n"),
(isFrameIndex == 1) ? (_u("NameBufferIndex")) : (_u("MethodID")),
methodIdOrNameId,
name,
lineNumber,
columnNumber);
stackFrames.Add(frame);
return false;
});
Assert(frameCount == (ushort)stackFrames.Count());
if (frameCount > 0) // No need to emit event if there are no script frames.
{
auto nameBufferString = nameBuffer.Detach();
if (nameBufferLength > 0)
{
// Account for the terminating null character.
nameBufferLength++;
}
if (emitV2AsyncStackEvent)
{
JS_ETW(EventWriteJSCRIPT_ASYNCCAUSALITY_STACKTRACE_V2(operationID, frameCount, nameBufferLength, sizeof(StackFrameInfo), &stackFrames.Item(0), nameBufferString));
}
else
{
JS_ETW(EventWriteJSCRIPT_STACKTRACE(operationID, frameCount, nameBufferLength, sizeof(StackFrameInfo), &stackFrames.Item(0), nameBufferString));
}
}
}
END_TEMP_ALLOCATOR(tempAllocator, this);
OUTPUT_FLUSH();
}
#endif
// Info: Append sourceString to stringBuilder after escaping charToEscape with escapeChar.
// "SomeBadly\0Formed\0String" => "SomeBadly\\\0Formed\\\0String"
// Parameters: stringBuilder - The Js::StringBuilder to which we should append sourceString.
// sourceString - The string we want to escape and append to stringBuilder.
// sourceStringLen - Length of sourceString.
// escapeChar - Char to use for escaping.
// charToEscape - The char which we should escape with escapeChar.
// Returns: Count of chars written to stringBuilder.
charcount_t ScriptContext::AppendWithEscapeCharacters(Js::StringBuilder<ArenaAllocator>* stringBuilder, const WCHAR* sourceString, charcount_t sourceStringLen, WCHAR escapeChar, WCHAR charToEscape)
{
const WCHAR* charToEscapePtr = wcschr(sourceString, charToEscape);
charcount_t charsPadding = 0;
// Only escape characters if sourceString contains one.
if (charToEscapePtr)
{
charcount_t charsWritten = 0;
charcount_t charsToAppend = 0;
while (charToEscapePtr)
{
charsToAppend = static_cast<charcount_t>(charToEscapePtr - sourceString) - charsWritten;
stringBuilder->Append(sourceString + charsWritten, charsToAppend);
stringBuilder->Append(escapeChar);
stringBuilder->Append(charToEscape);
// Keep track of this extra escapeChar character so we can update the buffer length correctly below.
charsPadding++;
// charsWritten is a count of the chars from sourceString which have been written - not count of chars Appended to stringBuilder.
charsWritten += charsToAppend + 1;
// Find next charToEscape.
charToEscapePtr++;
charToEscapePtr = wcschr(charToEscapePtr, charToEscape);
}
// Append the final part of the string if there is any left after the final charToEscape.
if (charsWritten != sourceStringLen)
{
charsToAppend = sourceStringLen - charsWritten;
stringBuilder->Append(sourceString + charsWritten, charsToAppend);
}
}
else
{
stringBuilder->AppendSz(sourceString);
}
return sourceStringLen + charsPadding;
}
/*static*/
ushort ScriptContext::ProcessNameAndGetLength(Js::StringBuilder<ArenaAllocator>* nameBuffer, const WCHAR* name)
{
Assert(nameBuffer);
Assert(name);
ushort nameLen = (ushort)wcslen(name);
// Surround each function name with quotes and escape any quote characters in the function name itself with '\\'.
nameBuffer->Append('\"');
// Adjust nameLen based on any escape characters we added to escape the '\"' in name.
nameLen = (unsigned short)AppendWithEscapeCharacters(nameBuffer, name, nameLen, '\\', '\"');
nameBuffer->AppendCppLiteral(_u("\";"));
// Add 3 padding characters here - one for initial '\"' character, too.
nameLen += 3;
return nameLen;
}
} // End namespace Js