diff options
author | Allan Sandfeld Jensen <allan.jensen@digia.com> | 2013-09-13 12:51:20 +0200 |
---|---|---|
committer | The Qt Project <gerrit-noreply@qt-project.org> | 2013-09-19 20:50:05 +0200 |
commit | d441d6f39bb846989d95bcf5caf387b42414718d (patch) | |
tree | e367e64a75991c554930278175d403c072de6bb8 /Source/JavaScriptCore/profiler/ProfilerDatabase.cpp | |
parent | 0060b2994c07842f4c59de64b5e3e430525c4b90 (diff) | |
download | qtwebkit-d441d6f39bb846989d95bcf5caf387b42414718d.tar.gz |
Import Qt5x2 branch of QtWebkit for Qt 5.2
Importing a new snapshot of webkit.
Change-Id: I2d01ad12cdc8af8cb015387641120a9d7ea5f10c
Reviewed-by: Allan Sandfeld Jensen <allan.jensen@digia.com>
Diffstat (limited to 'Source/JavaScriptCore/profiler/ProfilerDatabase.cpp')
-rw-r--r-- | Source/JavaScriptCore/profiler/ProfilerDatabase.cpp | 187 |
1 files changed, 187 insertions, 0 deletions
diff --git a/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp b/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp new file mode 100644 index 000000000..a2fffd955 --- /dev/null +++ b/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp @@ -0,0 +1,187 @@ +/* + * Copyright (C) 2012, 2013 Apple Inc. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR + * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, + * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR + * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include "config.h" +#include "ProfilerDatabase.h" + +#include "CodeBlock.h" +#include "JSONObject.h" +#include "ObjectConstructor.h" +#include "Operations.h" + +namespace JSC { namespace Profiler { + +#if COMPILER(MINGW) || COMPILER(MSVC7_OR_LOWER) || OS(WINCE) +static int databaseCounter; +#else +static volatile int databaseCounter; +#endif +static SpinLock registrationLock = SPINLOCK_INITIALIZER; +static int didRegisterAtExit; +static Database* firstDatabase; + +Database::Database(VM& vm) + : m_databaseID(atomicIncrement(&databaseCounter)) + , m_vm(vm) + , m_shouldSaveAtExit(false) + , m_nextRegisteredDatabase(0) +{ +} + +Database::~Database() +{ + if (m_shouldSaveAtExit) { + removeDatabaseFromAtExit(); + performAtExitSave(); + } +} + +Bytecodes* Database::ensureBytecodesFor(CodeBlock* codeBlock) +{ + codeBlock = codeBlock->baselineVersion(); + + HashMap<CodeBlock*, Bytecodes*>::iterator iter = m_bytecodesMap.find(codeBlock); + if (iter != m_bytecodesMap.end()) + return iter->value; + + m_bytecodes.append(Bytecodes(m_bytecodes.size(), codeBlock)); + Bytecodes* result = &m_bytecodes.last(); + + m_bytecodesMap.add(codeBlock, result); + + return result; +} + +void Database::notifyDestruction(CodeBlock* codeBlock) +{ + m_bytecodesMap.remove(codeBlock); +} + +PassRefPtr<Compilation> Database::newCompilation(Bytecodes* bytecodes, CompilationKind kind) +{ + RefPtr<Compilation> compilation = adoptRef(new Compilation(bytecodes, kind)); + m_compilations.append(compilation); + return compilation.release(); +} + +PassRefPtr<Compilation> Database::newCompilation(CodeBlock* codeBlock, CompilationKind kind) +{ + return newCompilation(ensureBytecodesFor(codeBlock), kind); +} + +JSValue Database::toJS(ExecState* exec) const +{ + JSObject* result = constructEmptyObject(exec); + + JSArray* bytecodes = constructEmptyArray(exec, 0); + for (unsigned i = 0; i < m_bytecodes.size(); ++i) + bytecodes->putDirectIndex(exec, i, m_bytecodes[i].toJS(exec)); + result->putDirect(exec->vm(), exec->propertyNames().bytecodes, bytecodes); + + JSArray* compilations = constructEmptyArray(exec, 0); + for (unsigned i = 0; i < m_compilations.size(); ++i) + compilations->putDirectIndex(exec, i, m_compilations[i]->toJS(exec)); + result->putDirect(exec->vm(), exec->propertyNames().compilations, compilations); + + return result; +} + +String Database::toJSON() const +{ + JSGlobalObject* globalObject = JSGlobalObject::create( + m_vm, JSGlobalObject::createStructure(m_vm, jsNull())); + + return JSONStringify(globalObject->globalExec(), toJS(globalObject->globalExec()), 0); +} + +bool Database::save(const char* filename) const +{ + OwnPtr<FilePrintStream> out = FilePrintStream::open(filename, "w"); + if (!out) + return false; + + out->print(toJSON()); + return true; +} + +void Database::registerToSaveAtExit(const char* filename) +{ + m_atExitSaveFilename = filename; + + if (m_shouldSaveAtExit) + return; + + addDatabaseToAtExit(); + m_shouldSaveAtExit = true; +} + +void Database::addDatabaseToAtExit() +{ + if (atomicIncrement(&didRegisterAtExit) == 1) + atexit(atExitCallback); + + TCMalloc_SpinLockHolder holder(®istrationLock); + m_nextRegisteredDatabase = firstDatabase; + firstDatabase = this; +} + +void Database::removeDatabaseFromAtExit() +{ + TCMalloc_SpinLockHolder holder(®istrationLock); + for (Database** current = &firstDatabase; *current; current = &(*current)->m_nextRegisteredDatabase) { + if (*current != this) + continue; + *current = m_nextRegisteredDatabase; + m_nextRegisteredDatabase = 0; + m_shouldSaveAtExit = false; + break; + } +} + +void Database::performAtExitSave() const +{ + save(m_atExitSaveFilename.data()); +} + +Database* Database::removeFirstAtExitDatabase() +{ + TCMalloc_SpinLockHolder holder(®istrationLock); + Database* result = firstDatabase; + if (result) { + firstDatabase = result->m_nextRegisteredDatabase; + result->m_nextRegisteredDatabase = 0; + result->m_shouldSaveAtExit = false; + } + return result; +} + +void Database::atExitCallback() +{ + while (Database* database = removeFirstAtExitDatabase()) + database->performAtExitSave(); +} + +} } // namespace JSC::Profiler + |