summaryrefslogtreecommitdiff
path: root/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp
diff options
context:
space:
mode:
authorLorry Tar Creator <lorry-tar-importer@lorry>2015-10-15 09:45:50 +0000
committerLorry Tar Creator <lorry-tar-importer@lorry>2015-10-15 09:45:50 +0000
commite15dd966d523731101f70ccf768bba12435a0208 (patch)
treeae9cb828a24ded2585a41af3f21411523b47897d /Source/JavaScriptCore/profiler/ProfilerDatabase.cpp
downloadWebKitGtk-tarball-e15dd966d523731101f70ccf768bba12435a0208.tar.gz
webkitgtk-2.10.2webkitgtk-2.10.2
Diffstat (limited to 'Source/JavaScriptCore/profiler/ProfilerDatabase.cpp')
-rw-r--r--Source/JavaScriptCore/profiler/ProfilerDatabase.cpp183
1 files changed, 183 insertions, 0 deletions
diff --git a/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp b/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp
new file mode 100644
index 000000000..fc952c0c2
--- /dev/null
+++ b/Source/JavaScriptCore/profiler/ProfilerDatabase.cpp
@@ -0,0 +1,183 @@
+/*
+ * 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 "JSCInlines.h"
+
+namespace JSC { namespace Profiler {
+
+static std::atomic<int> databaseCounter;
+
+static StaticLock registrationLock;
+static std::atomic<int> didRegisterAtExit;
+static Database* firstDatabase;
+
+Database::Database(VM& vm)
+ : m_databaseID(++databaseCounter)
+ , m_vm(vm)
+ , m_shouldSaveAtExit(false)
+ , m_nextRegisteredDatabase(0)
+{
+}
+
+Database::~Database()
+{
+ if (m_shouldSaveAtExit) {
+ removeDatabaseFromAtExit();
+ performAtExitSave();
+ }
+}
+
+Bytecodes* Database::ensureBytecodesFor(CodeBlock* codeBlock)
+{
+ LockHolder locker(m_lock);
+
+ 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)
+{
+ LockHolder locker(m_lock);
+
+ m_bytecodesMap.remove(codeBlock);
+}
+
+void Database::addCompilation(PassRefPtr<Compilation> compilation)
+{
+ ASSERT(!isCompilationThread());
+
+ m_compilations.append(compilation);
+}
+
+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
+{
+ auto 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 (++didRegisterAtExit == 1)
+ atexit(atExitCallback);
+
+ LockHolder holder(registrationLock);
+ m_nextRegisteredDatabase = firstDatabase;
+ firstDatabase = this;
+}
+
+void Database::removeDatabaseFromAtExit()
+{
+ LockHolder holder(registrationLock);
+ 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()
+{
+ LockHolder holder(registrationLock);
+ 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
+