diff options
Diffstat (limited to 'js/src/jit/CacheIRHealth.cpp')
-rw-r--r-- | js/src/jit/CacheIRHealth.cpp | 417 |
1 files changed, 417 insertions, 0 deletions
diff --git a/js/src/jit/CacheIRHealth.cpp b/js/src/jit/CacheIRHealth.cpp new file mode 100644 index 0000000000..5a5ac279b3 --- /dev/null +++ b/js/src/jit/CacheIRHealth.cpp @@ -0,0 +1,417 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- + * vim: set ts=8 sts=2 et sw=2 tw=80: + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ +#ifdef JS_CACHEIR_SPEW + +# include "jit/CacheIRHealth.h" + +# include "mozilla/Maybe.h" + +# include "gc/Zone.h" +# include "jit/BaselineIC.h" +# include "jit/CacheIRCompiler.h" +# include "jit/JitScript.h" +# include "js/ColumnNumber.h" // JS::LimitedColumnNumberOneOrigin +# include "vm/JSScript.h" + +# include "vm/JSObject-inl.h" +# include "vm/Realm-inl.h" + +using namespace js; +using namespace js::jit; + +// TODO: Refine how we assign happiness based on total health score. +CacheIRHealth::Happiness CacheIRHealth::determineStubHappiness( + uint32_t stubHealthScore) { + if (stubHealthScore >= 30) { + return Sad; + } + + if (stubHealthScore >= 20) { + return MediumSad; + } + + if (stubHealthScore >= 10) { + return MediumHappy; + } + + return Happy; +} + +CacheIRHealth::Happiness CacheIRHealth::spewStubHealth( + AutoStructuredSpewer& spew, ICCacheIRStub* stub) { + const CacheIRStubInfo* stubInfo = stub->stubInfo(); + CacheIRReader stubReader(stubInfo); + uint32_t totalStubHealth = 0; + spew->beginListProperty("cacheIROps"); + while (stubReader.more()) { + CacheOp op = stubReader.readOp(); + uint32_t opHealth = CacheIROpHealth[size_t(op)]; + uint32_t argLength = CacheIROpInfos[size_t(op)].argLength; + const char* opName = CacheIROpNames[size_t(op)]; + + spew->beginObject(); + if (opHealth == UINT32_MAX) { + spew->property("unscoredOp", opName); + } else { + spew->property("cacheIROp", opName); + spew->property("opHealth", opHealth); + totalStubHealth += opHealth; + } + spew->endObject(); + + stubReader.skip(argLength); + } + spew->endList(); // cacheIROps + + spew->property("stubHealth", totalStubHealth); + + Happiness stubHappiness = determineStubHappiness(totalStubHealth); + spew->property("stubHappiness", stubHappiness); + + return stubHappiness; +} + +BaseScript* CacheIRHealth::maybeExtractBaseScript(JSContext* cx, Shape* shape) { + TaggedProto taggedProto = shape->base()->proto(); + if (!taggedProto.isObject()) { + return nullptr; + } + Value cval; + JSObject* proto = taggedProto.toObject(); + AutoRealm ar(cx, proto); + if (!GetPropertyPure(cx, proto, NameToId(cx->names().constructor), &cval)) { + return nullptr; + } + if (!IsFunctionObject(cval)) { + return nullptr; + } + JSFunction& jsfun = cval.toObject().as<JSFunction>(); + if (!jsfun.hasBaseScript()) { + return nullptr; + } + return jsfun.baseScript(); +} + +void CacheIRHealth::spewShapeInformation(AutoStructuredSpewer& spew, + JSContext* cx, ICStub* stub) { + bool shapesStarted = false; + const CacheIRStubInfo* stubInfo = stub->toCacheIRStub()->stubInfo(); + size_t offset = 0; + uint32_t fieldIndex = 0; + + while (stubInfo->fieldType(fieldIndex) != StubField::Type::Limit) { + if (stubInfo->fieldType(fieldIndex) == StubField::Type::Shape) { + Shape* shape = reinterpret_cast<Shape*>( + stubInfo->getStubRawWord(stub->toCacheIRStub(), offset)); + if (!shapesStarted) { + shapesStarted = true; + spew->beginListProperty("shapes"); + } + + const PropMap* propMap = + shape->isNative() ? shape->asNative().propMap() : nullptr; + if (propMap) { + spew->beginObject(); + { + if (!propMap->isDictionary()) { + uint32_t mapLength = shape->asNative().propMapLength(); + if (mapLength) { + PropertyKey lastKey = shape->asNative().lastProperty().key(); + if (lastKey.isInt()) { + spew->property("lastProperty", lastKey.toInt()); + } else if (lastKey.isString()) { + JSString* str = lastKey.toString(); + if (str && str->isLinear()) { + spew->property("lastProperty", &str->asLinear()); + } + } else { + MOZ_ASSERT(lastKey.isSymbol()); + JSString* str = lastKey.toSymbol()->description(); + if (str && str->isLinear()) { + spew->property("lastProperty", &str->asLinear()); + } + } + } + spew->property("totalKeys", propMap->approximateEntryCount()); + BaseScript* baseScript = maybeExtractBaseScript(cx, shape); + if (baseScript) { + spew->beginObjectProperty("shapeAllocSite"); + { + spew->property("filename", baseScript->filename()); + spew->property("line", baseScript->lineno()); + spew->property("column", baseScript->column().oneOriginValue()); + } + spew->endObject(); + } + } + } + spew->endObject(); + } + } + offset += StubField::sizeInBytes(stubInfo->fieldType(fieldIndex)); + fieldIndex++; + } + + if (shapesStarted) { + spew->endList(); + } +} + +bool CacheIRHealth::spewNonFallbackICInformation(AutoStructuredSpewer& spew, + JSContext* cx, + ICStub* firstStub, + Happiness* entryHappiness) { + const CacheIRStubInfo* stubInfo = firstStub->toCacheIRStub()->stubInfo(); + Vector<bool, 8, SystemAllocPolicy> sawDistinctValueAtFieldIndex; + + bool sawNonZeroCount = false; + bool sawDifferentCacheIRStubs = false; + ICStub* stub = firstStub; + + spew->beginListProperty("stubs"); + while (stub && !stub->isFallback()) { + spew->beginObject(); + { + Happiness stubHappiness = spewStubHealth(spew, stub->toCacheIRStub()); + if (stubHappiness < *entryHappiness) { + *entryHappiness = stubHappiness; + } + + spewShapeInformation(spew, cx, stub); + + ICStub* nextStub = stub->toCacheIRStub()->next(); + if (!nextStub->isFallback()) { + if (nextStub->enteredCount() > 0) { + // More than one stub has a hit count greater than zero. + // This is sad because we do not Warp transpile in this case. + *entryHappiness = Sad; + sawNonZeroCount = true; + } + + if (nextStub->toCacheIRStub()->stubInfo() != stubInfo) { + sawDifferentCacheIRStubs = true; + } + + // If there are multiple stubs with non zero hit counts and if all + // of the stubs have equivalent CacheIR, then keep track of how many + // distinct stub field values are seen for each field index. + if (sawNonZeroCount && !sawDifferentCacheIRStubs) { + uint32_t fieldIndex = 0; + size_t offset = 0; + + while (stubInfo->fieldType(fieldIndex) != StubField::Type::Limit) { + if (sawDistinctValueAtFieldIndex.length() <= fieldIndex) { + if (!sawDistinctValueAtFieldIndex.append(false)) { + return false; + } + } + + if (StubField::sizeIsWord(stubInfo->fieldType(fieldIndex))) { + uintptr_t firstRaw = + stubInfo->getStubRawWord(firstStub->toCacheIRStub(), offset); + uintptr_t nextRaw = + stubInfo->getStubRawWord(nextStub->toCacheIRStub(), offset); + if (firstRaw != nextRaw) { + sawDistinctValueAtFieldIndex[fieldIndex] = true; + } + } else { + MOZ_ASSERT( + StubField::sizeIsInt64(stubInfo->fieldType(fieldIndex))); + int64_t firstRaw = + stubInfo->getStubRawInt64(firstStub->toCacheIRStub(), offset); + int64_t nextRaw = + stubInfo->getStubRawInt64(nextStub->toCacheIRStub(), offset); + + if (firstRaw != nextRaw) { + sawDistinctValueAtFieldIndex[fieldIndex] = true; + } + } + + offset += StubField::sizeInBytes(stubInfo->fieldType(fieldIndex)); + fieldIndex++; + } + } + } + + spew->property("hitCount", stub->enteredCount()); + stub = nextStub; + } + spew->endObject(); + } + spew->endList(); // stubs + + // If more than one CacheIR stub has an entered count greater than + // zero and all the stubs have equivalent CacheIR, then spew + // the information collected about the stub fields across the IC. + if (sawNonZeroCount && !sawDifferentCacheIRStubs) { + spew->beginListProperty("stubFields"); + for (size_t i = 0; i < sawDistinctValueAtFieldIndex.length(); i++) { + spew->beginObject(); + { + spew->property("fieldType", uint8_t(stubInfo->fieldType(i))); + spew->property("sawDistinctFieldValues", + sawDistinctValueAtFieldIndex[i]); + } + spew->endObject(); + } + spew->endList(); + } + + return true; +} + +bool CacheIRHealth::spewICEntryHealth(AutoStructuredSpewer& spew, JSContext* cx, + HandleScript script, ICEntry* entry, + ICFallbackStub* fallback, jsbytecode* pc, + JSOp op, Happiness* entryHappiness) { + spew->property("op", CodeName(op)); + + // TODO: If a perf issue arises, look into improving the SrcNotes + // API call below. + JS::LimitedColumnNumberOneOrigin column; + spew->property("lineno", PCToLineNumber(script, pc, &column)); + spew->property("column", column.oneOriginValue()); + + ICStub* firstStub = entry->firstStub(); + if (!firstStub->isFallback()) { + if (!spewNonFallbackICInformation(spew, cx, firstStub, entryHappiness)) { + return false; + } + } + + if (fallback->state().mode() != ICState::Mode::Specialized) { + *entryHappiness = Sad; + } + + spew->property("entryHappiness", uint8_t(*entryHappiness)); + + spew->property("mode", uint8_t(fallback->state().mode())); + + spew->property("fallbackCount", fallback->enteredCount()); + + return true; +} + +void CacheIRHealth::spewScriptFinalWarmUpCount(JSContext* cx, + const char* filename, + JSScript* script, + uint32_t warmUpCount) { + AutoStructuredSpewer spew(cx, SpewChannel::CacheIRHealthReport, nullptr); + if (!spew) { + return; + } + + spew->property("filename", filename); + spew->property("line", script->lineno()); + spew->property("column", script->column().oneOriginValue()); + spew->property("finalWarmUpCount", warmUpCount); +} + +static bool addScriptToFinalWarmUpCountMap(JSContext* cx, HandleScript script) { + // Create Zone::scriptFilenameMap if necessary. + JS::Zone* zone = script->zone(); + if (!zone->scriptFinalWarmUpCountMap) { + auto map = MakeUnique<ScriptFinalWarmUpCountMap>(); + if (!map) { + return false; + } + + zone->scriptFinalWarmUpCountMap = std::move(map); + } + + SharedImmutableString sfilename = + SharedImmutableStringsCache::getSingleton().getOrCreate( + script->filename(), strlen(script->filename())); + if (!sfilename) { + ReportOutOfMemory(cx); + return false; + } + + if (!zone->scriptFinalWarmUpCountMap->put( + script, std::make_tuple(uint32_t(0), std::move(sfilename)))) { + ReportOutOfMemory(cx); + return false; + } + + script->setNeedsFinalWarmUpCount(); + return true; +} + +void CacheIRHealth::healthReportForIC(JSContext* cx, ICEntry* entry, + ICFallbackStub* fallback, + HandleScript script, + SpewContext context) { + AutoStructuredSpewer spew(cx, SpewChannel::CacheIRHealthReport, script); + if (!spew) { + return; + } + + if (!addScriptToFinalWarmUpCountMap(cx, script)) { + cx->recoverFromOutOfMemory(); + return; + } + spew->property("spewContext", uint8_t(context)); + + jsbytecode* op = script->offsetToPC(fallback->pcOffset()); + JSOp jsOp = JSOp(*op); + + Happiness entryHappiness = Happy; + if (!spewICEntryHealth(spew, cx, script, entry, fallback, op, jsOp, + &entryHappiness)) { + cx->recoverFromOutOfMemory(); + return; + } + MOZ_ASSERT(entryHappiness == Sad); +} + +void CacheIRHealth::healthReportForScript(JSContext* cx, HandleScript script, + SpewContext context) { + jit::JitScript* jitScript = script->maybeJitScript(); + if (!jitScript) { + return; + } + + AutoStructuredSpewer spew(cx, SpewChannel::CacheIRHealthReport, script); + if (!spew) { + return; + } + + if (!addScriptToFinalWarmUpCountMap(cx, script)) { + cx->recoverFromOutOfMemory(); + return; + } + + spew->property("spewContext", uint8_t(context)); + + spew->beginListProperty("entries"); + + Happiness scriptHappiness = Happy; + + for (size_t i = 0; i < jitScript->numICEntries(); i++) { + ICEntry& entry = jitScript->icEntry(i); + ICFallbackStub* fallback = jitScript->fallbackStub(i); + jsbytecode* pc = script->offsetToPC(fallback->pcOffset()); + JSOp op = JSOp(*pc); + + spew->beginObject(); + Happiness entryHappiness = Happy; + if (!spewICEntryHealth(spew, cx, script, &entry, fallback, pc, op, + &entryHappiness)) { + cx->recoverFromOutOfMemory(); + return; + } + if (entryHappiness < scriptHappiness) { + scriptHappiness = entryHappiness; + } + spew->endObject(); + } + + spew->endList(); // entries + + spew->property("scriptHappiness", uint8_t(scriptHappiness)); +} + +#endif /* JS_CACHEIR_SPEW */ |