| /* | 
 |  * Copyright (C) 2012-2019 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 "ProfilerBytecodeSequence.h" | 
 |  | 
 | #include "CodeBlock.h" | 
 | #include "JSCInlines.h" | 
 | #include <wtf/StringPrintStream.h> | 
 |  | 
 | namespace JSC { namespace Profiler { | 
 |  | 
 | BytecodeSequence::BytecodeSequence(CodeBlock* codeBlock) | 
 | { | 
 |     StringPrintStream out; | 
 |      | 
 |     for (unsigned i = 0; i < codeBlock->numberOfArgumentValueProfiles(); ++i) { | 
 |         ConcurrentJSLocker locker(codeBlock->m_lock); | 
 |         CString description = codeBlock->valueProfileForArgument(i).briefDescription(locker); | 
 |         if (!description.length()) | 
 |             continue; | 
 |         out.reset(); | 
 |         out.print("arg", i, ": ", description); | 
 |         m_header.append(out.toCString()); | 
 |     } | 
 |      | 
 |     ICStatusMap statusMap; | 
 |     codeBlock->getICStatusMap(statusMap); | 
 |      | 
 |     for (unsigned bytecodeIndex = 0; bytecodeIndex < codeBlock->instructions().size();) { | 
 |         out.reset(); | 
 |         codeBlock->dumpBytecode(out, bytecodeIndex, statusMap); | 
 |         auto instruction = codeBlock->instructions().at(bytecodeIndex); | 
 |         OpcodeID opcodeID = instruction->opcodeID(); | 
 |         m_sequence.append(Bytecode(bytecodeIndex, opcodeID, out.toCString())); | 
 |         bytecodeIndex += instruction->size(); | 
 |     } | 
 | } | 
 |  | 
 | BytecodeSequence::~BytecodeSequence() | 
 | { | 
 | } | 
 |  | 
 | unsigned BytecodeSequence::indexForBytecodeIndex(unsigned bytecodeIndex) const | 
 | { | 
 |     return binarySearch<Bytecode, unsigned>(m_sequence, m_sequence.size(), bytecodeIndex, getBytecodeIndexForBytecode) - m_sequence.begin(); | 
 | } | 
 |  | 
 | const Bytecode& BytecodeSequence::forBytecodeIndex(unsigned bytecodeIndex) const | 
 | { | 
 |     return at(indexForBytecodeIndex(bytecodeIndex)); | 
 | } | 
 |  | 
 | void BytecodeSequence::addSequenceProperties(JSGlobalObject* globalObject, JSObject* result) const | 
 | { | 
 |     VM& vm = globalObject->vm(); | 
 |     auto scope = DECLARE_THROW_SCOPE(vm); | 
 |     JSArray* header = constructEmptyArray(globalObject, nullptr); | 
 |     RETURN_IF_EXCEPTION(scope, void()); | 
 |     for (unsigned i = 0; i < m_header.size(); ++i) { | 
 |         header->putDirectIndex(globalObject, i, jsString(vm, String::fromUTF8(m_header[i]))); | 
 |         RETURN_IF_EXCEPTION(scope, void()); | 
 |     } | 
 |     result->putDirect(vm, vm.propertyNames->header, header); | 
 |      | 
 |     JSArray* sequence = constructEmptyArray(globalObject, nullptr); | 
 |     RETURN_IF_EXCEPTION(scope, void()); | 
 |     for (unsigned i = 0; i < m_sequence.size(); ++i) { | 
 |         sequence->putDirectIndex(globalObject, i, m_sequence[i].toJS(globalObject)); | 
 |         RETURN_IF_EXCEPTION(scope, void()); | 
 |     } | 
 |     result->putDirect(vm, vm.propertyNames->bytecode, sequence); | 
 | } | 
 |  | 
 | } } // namespace JSC::Profiler | 
 |  |