1 /* 2 This file is provided under a dual BSD/GPLv2 license. When using or 3 redistributing this file, you may do so under either license. 4 5 GPL LICENSE SUMMARY 6 7 Copyright(c) 2005-2012 Intel Corporation. All rights reserved. 8 9 This program is free software; you can redistribute it and/or modify 10 it under the terms of version 2 of the GNU General Public License as 11 published by the Free Software Foundation. 12 13 This program is distributed in the hope that it will be useful, but 14 WITHOUT ANY WARRANTY; without even the implied warranty of 15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16 General Public License for more details. 17 18 You should have received a copy of the GNU General Public License 19 along with this program; if not, write to the Free Software 20 Foundation, Inc., 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. 21 The full GNU General Public License is included in this distribution 22 in the file called LICENSE.GPL. 23 24 Contact Information: 25 http://software.intel.com/en-us/articles/intel-vtune-amplifier-xe/ 26 27 BSD LICENSE 28 29 Copyright(c) 2005-2012 Intel Corporation. All rights reserved. 30 All rights reserved. 31 32 Redistribution and use in source and binary forms, with or without 33 modification, are permitted provided that the following conditions 34 are met: 35 36 * Redistributions of source code must retain the above copyright 37 notice, this list of conditions and the following disclaimer. 38 * Redistributions in binary form must reproduce the above copyright 39 notice, this list of conditions and the following disclaimer in 40 the documentation and/or other materials provided with the 41 distribution. 42 * Neither the name of Intel Corporation nor the names of its 43 contributors may be used to endorse or promote products derived 44 from this software without specific prior written permission. 45 46 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 47 "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 48 LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 49 A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 50 OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 51 SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 52 LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 53 DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 54 THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 55 (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 56 OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 57 */ 58 59 #include <stdlib.h> 60 #include <string.h> 61 62 #include <list> 63 #include <unordered_map> 64 65 #include "v8-vtune.h" 66 #include "vtune-jit.h" 67 68 namespace vTune { 69 namespace internal { 70 71 72 // This class is used to record the JITted code position info for JIT 73 // code profiling. 74 class JITCodeLineInfo { 75 public: 76 JITCodeLineInfo() { } 77 78 void SetPosition(intptr_t pc, int pos) { 79 AddCodeLineInfo(LineNumInfo(pc, pos)); 80 } 81 82 struct LineNumInfo { 83 LineNumInfo(intptr_t pc, int pos) 84 : pc_(pc), pos_(pos) { } 85 86 intptr_t pc_; 87 int pos_; 88 }; 89 90 std::list<LineNumInfo>* GetLineNumInfo() { 91 return &line_num_info_; 92 } 93 94 private: 95 void AddCodeLineInfo(const LineNumInfo& line_info) { 96 line_num_info_.push_back(line_info); 97 } 98 std::list<LineNumInfo> line_num_info_; 99 }; 100 101 struct SameCodeObjects { 102 bool operator () (void* key1, void* key2) const { 103 return key1 == key2; 104 } 105 }; 106 107 struct HashForCodeObject { 108 uint32_t operator () (void* code) const { 109 static const uintptr_t kGoldenRatio = 2654435761u; 110 uintptr_t hash = reinterpret_cast<uintptr_t>(code); 111 return static_cast<uint32_t>(hash * kGoldenRatio); 112 } 113 }; 114 115 typedef std::unordered_map<void*, void*, HashForCodeObject, SameCodeObjects> 116 JitInfoMap; 117 118 static JitInfoMap* GetEntries() { 119 static JitInfoMap* entries; 120 if (entries == NULL) { 121 entries = new JitInfoMap(); 122 } 123 return entries; 124 } 125 126 static bool IsLineInfoTagged(void* ptr) { 127 return 0 != (reinterpret_cast<intptr_t>(ptr)); 128 } 129 130 static JITCodeLineInfo* UntagLineInfo(void* ptr) { 131 return reinterpret_cast<JITCodeLineInfo*>( 132 reinterpret_cast<intptr_t>(ptr)); 133 } 134 135 // The parameter str is a mixed pattern which contains the 136 // function name and some other info. It comes from all the 137 // Logger::CodeCreateEvent(...) function. This funtion get the 138 // pure function name from the input parameter. 139 static char* GetFunctionNameFromMixedName(const char* str, int length) { 140 int index = 0; 141 int count = 0; 142 char* start_ptr = NULL; 143 144 while (str[index++] != ':' && (index < length)) {} 145 146 if (str[index] == '*' || str[index] == '~' ) index++; 147 if (index >= length) return NULL; 148 149 start_ptr = const_cast<char*>(str + index); 150 151 while (index < length && str[index++] != ' ') { 152 count++; 153 } 154 155 char* result = new char[count + 1]; 156 memcpy(result, start_ptr, count); 157 result[count] = '\0'; 158 159 return result; 160 } 161 162 // The JitCodeEventHandler for Vtune. 163 void VTUNEJITInterface::event_handler(const v8::JitCodeEvent* event) { 164 if (VTUNERUNNING && event != NULL) { 165 switch (event->type) { 166 case v8::JitCodeEvent::CODE_ADDED: { 167 char* temp_file_name = NULL; 168 char* temp_method_name = 169 GetFunctionNameFromMixedName(event->name.str, 170 static_cast<int>(event->name.len)); 171 iJIT_Method_Load jmethod; 172 memset(&jmethod, 0, sizeof jmethod); 173 jmethod.method_id = iJIT_GetNewMethodID(); 174 jmethod.method_load_address = event->code_start; 175 jmethod.method_size = static_cast<unsigned int>(event->code_len); 176 jmethod.method_name = temp_method_name; 177 178 Local<UnboundScript> script = event->script; 179 180 if (*script != NULL) { 181 // Get the source file name and set it to jmethod.source_file_name 182 if ((*script->GetScriptName())->IsString()) { 183 Local<String> script_name = 184 Local<String>::Cast(script->GetScriptName()); 185 temp_file_name = new char[script_name->Utf8Length() + 1]; 186 script_name->WriteUtf8(temp_file_name); 187 jmethod.source_file_name = temp_file_name; 188 } 189 190 JitInfoMap::iterator entry = 191 GetEntries()->find(event->code_start); 192 if (entry != GetEntries()->end() && IsLineInfoTagged(entry->first)) { 193 JITCodeLineInfo* line_info = UntagLineInfo(entry->second); 194 // Get the line_num_info and set it to jmethod.line_number_table 195 std::list<JITCodeLineInfo::LineNumInfo>* vtunelineinfo = 196 line_info->GetLineNumInfo(); 197 198 jmethod.line_number_size = (unsigned int)vtunelineinfo->size(); 199 jmethod.line_number_table = 200 reinterpret_cast<LineNumberInfo*>( 201 malloc(sizeof(LineNumberInfo)*jmethod.line_number_size)); 202 203 std::list<JITCodeLineInfo::LineNumInfo>::iterator Iter; 204 int index = 0; 205 for (Iter = vtunelineinfo->begin(); 206 Iter != vtunelineinfo->end(); 207 Iter++) { 208 jmethod.line_number_table[index].Offset = 209 static_cast<unsigned int>(Iter->pc_); 210 jmethod.line_number_table[index++].LineNumber = 211 script->GetLineNumber(Iter->pos_) + 1; 212 } 213 GetEntries()->erase(event->code_start); 214 } 215 } 216 217 iJIT_NotifyEvent(iJVM_EVENT_TYPE_METHOD_LOAD_FINISHED, 218 reinterpret_cast<void*>(&jmethod)); 219 if (temp_method_name) 220 delete []temp_method_name; 221 if (temp_file_name) 222 delete []temp_file_name; 223 break; 224 } 225 // TODO(chunyang.dai (at) intel.com): code_move will be supported. 226 case v8::JitCodeEvent::CODE_MOVED: 227 break; 228 // Currently the CODE_REMOVED event is not issued. 229 case v8::JitCodeEvent::CODE_REMOVED: 230 break; 231 case v8::JitCodeEvent::CODE_ADD_LINE_POS_INFO: { 232 JITCodeLineInfo* line_info = 233 reinterpret_cast<JITCodeLineInfo*>(event->user_data); 234 if (line_info != NULL) { 235 line_info->SetPosition(static_cast<intptr_t>(event->line_info.offset), 236 static_cast<int>(event->line_info.pos)); 237 } 238 break; 239 } 240 case v8::JitCodeEvent::CODE_START_LINE_INFO_RECORDING: { 241 v8::JitCodeEvent* temp_event = const_cast<v8::JitCodeEvent*>(event); 242 temp_event->user_data = new JITCodeLineInfo(); 243 break; 244 } 245 case v8::JitCodeEvent::CODE_END_LINE_INFO_RECORDING: { 246 GetEntries()->insert(std::pair <void*, void*>(event->code_start, event->user_data)); 247 break; 248 } 249 default: 250 break; 251 } 252 } 253 return; 254 } 255 256 } // namespace internal 257 258 v8::JitCodeEventHandler GetVtuneCodeEventHandler() { 259 v8::V8::SetFlagsFromString("--nocompact_code_space", 260 (int)strlen("--nocompact_code_space")); 261 return vTune::internal::VTUNEJITInterface::event_handler; 262 } 263 264 } // namespace vTune 265