diff options
| author | Ingo Molnar <mingo@kernel.org> | 2014-12-12 03:09:03 -0500 |
|---|---|---|
| committer | Ingo Molnar <mingo@kernel.org> | 2014-12-12 03:09:03 -0500 |
| commit | 3459f0d78ffe27a1b341c22eb158b622eaaea3fc (patch) | |
| tree | 715b0575eec541d0181876ad367ca5480cdcecf3 /tools/perf/scripts/python | |
| parent | 9fc81d87420d0d3fd62d5e5529972c0ad9eab9cc (diff) | |
| parent | bee2782f30f66898be3f74ad02e4d1f87a969694 (diff) | |
Merge branch 'linus' into perf/urgent, to pick up the upstream merged bits
Signed-off-by: Ingo Molnar <mingo@kernel.org>
Diffstat (limited to 'tools/perf/scripts/python')
3 files changed, 481 insertions, 0 deletions
diff --git a/tools/perf/scripts/python/bin/export-to-postgresql-record b/tools/perf/scripts/python/bin/export-to-postgresql-record new file mode 100644 index 000000000000..221d66e05713 --- /dev/null +++ b/tools/perf/scripts/python/bin/export-to-postgresql-record | |||
| @@ -0,0 +1,8 @@ | |||
| 1 | #!/bin/bash | ||
| 2 | |||
| 3 | # | ||
| 4 | # export perf data to a postgresql database. Can cover | ||
| 5 | # perf ip samples (excluding the tracepoints). No special | ||
| 6 | # record requirements, just record what you want to export. | ||
| 7 | # | ||
| 8 | perf record $@ | ||
diff --git a/tools/perf/scripts/python/bin/export-to-postgresql-report b/tools/perf/scripts/python/bin/export-to-postgresql-report new file mode 100644 index 000000000000..cd335b6e2a01 --- /dev/null +++ b/tools/perf/scripts/python/bin/export-to-postgresql-report | |||
| @@ -0,0 +1,29 @@ | |||
| 1 | #!/bin/bash | ||
| 2 | # description: export perf data to a postgresql database | ||
| 3 | # args: [database name] [columns] [calls] | ||
| 4 | n_args=0 | ||
| 5 | for i in "$@" | ||
| 6 | do | ||
| 7 | if expr match "$i" "-" > /dev/null ; then | ||
| 8 | break | ||
| 9 | fi | ||
| 10 | n_args=$(( $n_args + 1 )) | ||
| 11 | done | ||
| 12 | if [ "$n_args" -gt 3 ] ; then | ||
| 13 | echo "usage: export-to-postgresql-report [database name] [columns] [calls]" | ||
| 14 | exit | ||
| 15 | fi | ||
| 16 | if [ "$n_args" -gt 2 ] ; then | ||
| 17 | dbname=$1 | ||
| 18 | columns=$2 | ||
| 19 | calls=$3 | ||
| 20 | shift 3 | ||
| 21 | elif [ "$n_args" -gt 1 ] ; then | ||
| 22 | dbname=$1 | ||
| 23 | columns=$2 | ||
| 24 | shift 2 | ||
| 25 | elif [ "$n_args" -gt 0 ] ; then | ||
| 26 | dbname=$1 | ||
| 27 | shift | ||
| 28 | fi | ||
| 29 | perf script $@ -s "$PERF_EXEC_PATH"/scripts/python/export-to-postgresql.py $dbname $columns $calls | ||
diff --git a/tools/perf/scripts/python/export-to-postgresql.py b/tools/perf/scripts/python/export-to-postgresql.py new file mode 100644 index 000000000000..4cdafd880074 --- /dev/null +++ b/tools/perf/scripts/python/export-to-postgresql.py | |||
| @@ -0,0 +1,444 @@ | |||
| 1 | # export-to-postgresql.py: export perf data to a postgresql database | ||
| 2 | # Copyright (c) 2014, Intel Corporation. | ||
| 3 | # | ||
| 4 | # This program is free software; you can redistribute it and/or modify it | ||
| 5 | # under the terms and conditions of the GNU General Public License, | ||
| 6 | # version 2, as published by the Free Software Foundation. | ||
| 7 | # | ||
| 8 | # This program is distributed in the hope it will be useful, but WITHOUT | ||
| 9 | # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
| 10 | # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
| 11 | # more details. | ||
| 12 | |||
| 13 | import os | ||
| 14 | import sys | ||
| 15 | import struct | ||
| 16 | import datetime | ||
| 17 | |||
| 18 | from PySide.QtSql import * | ||
| 19 | |||
| 20 | # Need to access PostgreSQL C library directly to use COPY FROM STDIN | ||
| 21 | from ctypes import * | ||
| 22 | libpq = CDLL("libpq.so.5") | ||
| 23 | PQconnectdb = libpq.PQconnectdb | ||
| 24 | PQconnectdb.restype = c_void_p | ||
| 25 | PQfinish = libpq.PQfinish | ||
| 26 | PQstatus = libpq.PQstatus | ||
| 27 | PQexec = libpq.PQexec | ||
| 28 | PQexec.restype = c_void_p | ||
| 29 | PQresultStatus = libpq.PQresultStatus | ||
| 30 | PQputCopyData = libpq.PQputCopyData | ||
| 31 | PQputCopyData.argtypes = [ c_void_p, c_void_p, c_int ] | ||
| 32 | PQputCopyEnd = libpq.PQputCopyEnd | ||
| 33 | PQputCopyEnd.argtypes = [ c_void_p, c_void_p ] | ||
| 34 | |||
| 35 | sys.path.append(os.environ['PERF_EXEC_PATH'] + \ | ||
| 36 | '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') | ||
| 37 | |||
| 38 | # These perf imports are not used at present | ||
| 39 | #from perf_trace_context import * | ||
| 40 | #from Core import * | ||
| 41 | |||
| 42 | perf_db_export_mode = True | ||
| 43 | perf_db_export_calls = False | ||
| 44 | |||
| 45 | def usage(): | ||
| 46 | print >> sys.stderr, "Usage is: export-to-postgresql.py <database name> [<columns>] [<calls>]" | ||
| 47 | print >> sys.stderr, "where: columns 'all' or 'branches'" | ||
| 48 | print >> sys.stderr, " calls 'calls' => create calls table" | ||
| 49 | raise Exception("Too few arguments") | ||
| 50 | |||
| 51 | if (len(sys.argv) < 2): | ||
| 52 | usage() | ||
| 53 | |||
| 54 | dbname = sys.argv[1] | ||
| 55 | |||
| 56 | if (len(sys.argv) >= 3): | ||
| 57 | columns = sys.argv[2] | ||
| 58 | else: | ||
| 59 | columns = "all" | ||
| 60 | |||
| 61 | if columns not in ("all", "branches"): | ||
| 62 | usage() | ||
| 63 | |||
| 64 | branches = (columns == "branches") | ||
| 65 | |||
| 66 | if (len(sys.argv) >= 4): | ||
| 67 | if (sys.argv[3] == "calls"): | ||
| 68 | perf_db_export_calls = True | ||
| 69 | else: | ||
| 70 | usage() | ||
| 71 | |||
| 72 | output_dir_name = os.getcwd() + "/" + dbname + "-perf-data" | ||
| 73 | os.mkdir(output_dir_name) | ||
| 74 | |||
| 75 | def do_query(q, s): | ||
| 76 | if (q.exec_(s)): | ||
| 77 | return | ||
| 78 | raise Exception("Query failed: " + q.lastError().text()) | ||
| 79 | |||
| 80 | print datetime.datetime.today(), "Creating database..." | ||
| 81 | |||
| 82 | db = QSqlDatabase.addDatabase('QPSQL') | ||
| 83 | query = QSqlQuery(db) | ||
| 84 | db.setDatabaseName('postgres') | ||
| 85 | db.open() | ||
| 86 | try: | ||
| 87 | do_query(query, 'CREATE DATABASE ' + dbname) | ||
| 88 | except: | ||
| 89 | os.rmdir(output_dir_name) | ||
| 90 | raise | ||
| 91 | query.finish() | ||
| 92 | query.clear() | ||
| 93 | db.close() | ||
| 94 | |||
| 95 | db.setDatabaseName(dbname) | ||
| 96 | db.open() | ||
| 97 | |||
| 98 | query = QSqlQuery(db) | ||
| 99 | do_query(query, 'SET client_min_messages TO WARNING') | ||
| 100 | |||
| 101 | do_query(query, 'CREATE TABLE selected_events (' | ||
| 102 | 'id bigint NOT NULL,' | ||
| 103 | 'name varchar(80))') | ||
| 104 | do_query(query, 'CREATE TABLE machines (' | ||
| 105 | 'id bigint NOT NULL,' | ||
| 106 | 'pid integer,' | ||
| 107 | 'root_dir varchar(4096))') | ||
| 108 | do_query(query, 'CREATE TABLE threads (' | ||
| 109 | 'id bigint NOT NULL,' | ||
| 110 | 'machine_id bigint,' | ||
| 111 | 'process_id bigint,' | ||
| 112 | 'pid integer,' | ||
| 113 | 'tid integer)') | ||
| 114 | do_query(query, 'CREATE TABLE comms (' | ||
| 115 | 'id bigint NOT NULL,' | ||
| 116 | 'comm varchar(16))') | ||
| 117 | do_query(query, 'CREATE TABLE comm_threads (' | ||
| 118 | 'id bigint NOT NULL,' | ||
| 119 | 'comm_id bigint,' | ||
| 120 | 'thread_id bigint)') | ||
| 121 | do_query(query, 'CREATE TABLE dsos (' | ||
| 122 | 'id bigint NOT NULL,' | ||
| 123 | 'machine_id bigint,' | ||
| 124 | 'short_name varchar(256),' | ||
| 125 | 'long_name varchar(4096),' | ||
| 126 | 'build_id varchar(64))') | ||
| 127 | do_query(query, 'CREATE TABLE symbols (' | ||
| 128 | 'id bigint NOT NULL,' | ||
| 129 | 'dso_id bigint,' | ||
| 130 | 'sym_start bigint,' | ||
| 131 | 'sym_end bigint,' | ||
| 132 | 'binding integer,' | ||
| 133 | 'name varchar(2048))') | ||
| 134 | do_query(query, 'CREATE TABLE branch_types (' | ||
| 135 | 'id integer NOT NULL,' | ||
| 136 | 'name varchar(80))') | ||
| 137 | |||
| 138 | if branches: | ||
| 139 | do_query(query, 'CREATE TABLE samples (' | ||
| 140 | 'id bigint NOT NULL,' | ||
| 141 | 'evsel_id bigint,' | ||
| 142 | 'machine_id bigint,' | ||
| 143 | 'thread_id bigint,' | ||
| 144 | 'comm_id bigint,' | ||
| 145 | 'dso_id bigint,' | ||
| 146 | 'symbol_id bigint,' | ||
| 147 | 'sym_offset bigint,' | ||
| 148 | 'ip bigint,' | ||
| 149 | 'time bigint,' | ||
| 150 | 'cpu integer,' | ||
| 151 | 'to_dso_id bigint,' | ||
| 152 | 'to_symbol_id bigint,' | ||
| 153 | 'to_sym_offset bigint,' | ||
| 154 | 'to_ip bigint,' | ||
| 155 | 'branch_type integer,' | ||
| 156 | 'in_tx boolean)') | ||
| 157 | else: | ||
| 158 | do_query(query, 'CREATE TABLE samples (' | ||
| 159 | 'id bigint NOT NULL,' | ||
| 160 | 'evsel_id bigint,' | ||
| 161 | 'machine_id bigint,' | ||
| 162 | 'thread_id bigint,' | ||
| 163 | 'comm_id bigint,' | ||
| 164 | 'dso_id bigint,' | ||
| 165 | 'symbol_id bigint,' | ||
| 166 | 'sym_offset bigint,' | ||
| 167 | 'ip bigint,' | ||
| 168 | 'time bigint,' | ||
| 169 | 'cpu integer,' | ||
| 170 | 'to_dso_id bigint,' | ||
| 171 | 'to_symbol_id bigint,' | ||
| 172 | 'to_sym_offset bigint,' | ||
| 173 | 'to_ip bigint,' | ||
| 174 | 'period bigint,' | ||
| 175 | 'weight bigint,' | ||
| 176 | 'transaction bigint,' | ||
| 177 | 'data_src bigint,' | ||
| 178 | 'branch_type integer,' | ||
| 179 | 'in_tx boolean)') | ||
| 180 | |||
| 181 | if perf_db_export_calls: | ||
| 182 | do_query(query, 'CREATE TABLE call_paths (' | ||
| 183 | 'id bigint NOT NULL,' | ||
| 184 | 'parent_id bigint,' | ||
| 185 | 'symbol_id bigint,' | ||
| 186 | 'ip bigint)') | ||
| 187 | do_query(query, 'CREATE TABLE calls (' | ||
| 188 | 'id bigint NOT NULL,' | ||
| 189 | 'thread_id bigint,' | ||
| 190 | 'comm_id bigint,' | ||
| 191 | 'call_path_id bigint,' | ||
| 192 | 'call_time bigint,' | ||
| 193 | 'return_time bigint,' | ||
| 194 | 'branch_count bigint,' | ||
| 195 | 'call_id bigint,' | ||
| 196 | 'return_id bigint,' | ||
| 197 | 'parent_call_path_id bigint,' | ||
| 198 | 'flags integer)') | ||
| 199 | |||
| 200 | do_query(query, 'CREATE VIEW samples_view AS ' | ||
| 201 | 'SELECT ' | ||
| 202 | 'id,' | ||
| 203 | 'time,' | ||
| 204 | 'cpu,' | ||
| 205 | '(SELECT pid FROM threads WHERE id = thread_id) AS pid,' | ||
| 206 | '(SELECT tid FROM threads WHERE id = thread_id) AS tid,' | ||
| 207 | '(SELECT comm FROM comms WHERE id = comm_id) AS command,' | ||
| 208 | '(SELECT name FROM selected_events WHERE id = evsel_id) AS event,' | ||
| 209 | 'to_hex(ip) AS ip_hex,' | ||
| 210 | '(SELECT name FROM symbols WHERE id = symbol_id) AS symbol,' | ||
| 211 | 'sym_offset,' | ||
| 212 | '(SELECT short_name FROM dsos WHERE id = dso_id) AS dso_short_name,' | ||
| 213 | 'to_hex(to_ip) AS to_ip_hex,' | ||
| 214 | '(SELECT name FROM symbols WHERE id = to_symbol_id) AS to_symbol,' | ||
| 215 | 'to_sym_offset,' | ||
| 216 | '(SELECT short_name FROM dsos WHERE id = to_dso_id) AS to_dso_short_name,' | ||
| 217 | '(SELECT name FROM branch_types WHERE id = branch_type) AS branch_type_name,' | ||
| 218 | 'in_tx' | ||
| 219 | ' FROM samples') | ||
| 220 | |||
| 221 | |||
| 222 | file_header = struct.pack("!11sii", "PGCOPY\n\377\r\n\0", 0, 0) | ||
| 223 | file_trailer = "\377\377" | ||
| 224 | |||
| 225 | def open_output_file(file_name): | ||
| 226 | path_name = output_dir_name + "/" + file_name | ||
| 227 | file = open(path_name, "w+") | ||
| 228 | file.write(file_header) | ||
| 229 | return file | ||
| 230 | |||
| 231 | def close_output_file(file): | ||
| 232 | file.write(file_trailer) | ||
| 233 | file.close() | ||
| 234 | |||
| 235 | def copy_output_file_direct(file, table_name): | ||
| 236 | close_output_file(file) | ||
| 237 | sql = "COPY " + table_name + " FROM '" + file.name + "' (FORMAT 'binary')" | ||
| 238 | do_query(query, sql) | ||
| 239 | |||
| 240 | # Use COPY FROM STDIN because security may prevent postgres from accessing the files directly | ||
| 241 | def copy_output_file(file, table_name): | ||
| 242 | conn = PQconnectdb("dbname = " + dbname) | ||
| 243 | if (PQstatus(conn)): | ||
| 244 | raise Exception("COPY FROM STDIN PQconnectdb failed") | ||
| 245 | file.write(file_trailer) | ||
| 246 | file.seek(0) | ||
| 247 | sql = "COPY " + table_name + " FROM STDIN (FORMAT 'binary')" | ||
| 248 | res = PQexec(conn, sql) | ||
| 249 | if (PQresultStatus(res) != 4): | ||
| 250 | raise Exception("COPY FROM STDIN PQexec failed") | ||
| 251 | data = file.read(65536) | ||
| 252 | while (len(data)): | ||
| 253 | ret = PQputCopyData(conn, data, len(data)) | ||
| 254 | if (ret != 1): | ||
| 255 | raise Exception("COPY FROM STDIN PQputCopyData failed, error " + str(ret)) | ||
| 256 | data = file.read(65536) | ||
| 257 | ret = PQputCopyEnd(conn, None) | ||
| 258 | if (ret != 1): | ||
| 259 | raise Exception("COPY FROM STDIN PQputCopyEnd failed, error " + str(ret)) | ||
| 260 | PQfinish(conn) | ||
| 261 | |||
| 262 | def remove_output_file(file): | ||
| 263 | name = file.name | ||
| 264 | file.close() | ||
| 265 | os.unlink(name) | ||
| 266 | |||
| 267 | evsel_file = open_output_file("evsel_table.bin") | ||
| 268 | machine_file = open_output_file("machine_table.bin") | ||
| 269 | thread_file = open_output_file("thread_table.bin") | ||
| 270 | comm_file = open_output_file("comm_table.bin") | ||
| 271 | comm_thread_file = open_output_file("comm_thread_table.bin") | ||
| 272 | dso_file = open_output_file("dso_table.bin") | ||
| 273 | symbol_file = open_output_file("symbol_table.bin") | ||
| 274 | branch_type_file = open_output_file("branch_type_table.bin") | ||
| 275 | sample_file = open_output_file("sample_table.bin") | ||
| 276 | if perf_db_export_calls: | ||
| 277 | call_path_file = open_output_file("call_path_table.bin") | ||
| 278 | call_file = open_output_file("call_table.bin") | ||
| 279 | |||
| 280 | def trace_begin(): | ||
| 281 | print datetime.datetime.today(), "Writing to intermediate files..." | ||
| 282 | # id == 0 means unknown. It is easier to create records for them than replace the zeroes with NULLs | ||
| 283 | evsel_table(0, "unknown") | ||
| 284 | machine_table(0, 0, "unknown") | ||
| 285 | thread_table(0, 0, 0, -1, -1) | ||
| 286 | comm_table(0, "unknown") | ||
| 287 | dso_table(0, 0, "unknown", "unknown", "") | ||
| 288 | symbol_table(0, 0, 0, 0, 0, "unknown") | ||
| 289 | sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0) | ||
| 290 | if perf_db_export_calls: | ||
| 291 | call_path_table(0, 0, 0, 0) | ||
| 292 | |||
| 293 | unhandled_count = 0 | ||
| 294 | |||
| 295 | def trace_end(): | ||
| 296 | print datetime.datetime.today(), "Copying to database..." | ||
| 297 | copy_output_file(evsel_file, "selected_events") | ||
| 298 | copy_output_file(machine_file, "machines") | ||
| 299 | copy_output_file(thread_file, "threads") | ||
| 300 | copy_output_file(comm_file, "comms") | ||
| 301 | copy_output_file(comm_thread_file, "comm_threads") | ||
| 302 | copy_output_file(dso_file, "dsos") | ||
| 303 | copy_output_file(symbol_file, "symbols") | ||
| 304 | copy_output_file(branch_type_file, "branch_types") | ||
| 305 | copy_output_file(sample_file, "samples") | ||
| 306 | if perf_db_export_calls: | ||
| 307 | copy_output_file(call_path_file, "call_paths") | ||
| 308 | copy_output_file(call_file, "calls") | ||
| 309 | |||
| 310 | print datetime.datetime.today(), "Removing intermediate files..." | ||
| 311 | remove_output_file(evsel_file) | ||
| 312 | remove_output_file(machine_file) | ||
| 313 | remove_output_file(thread_file) | ||
| 314 | remove_output_file(comm_file) | ||
| 315 | remove_output_file(comm_thread_file) | ||
| 316 | remove_output_file(dso_file) | ||
| 317 | remove_output_file(symbol_file) | ||
| 318 | remove_output_file(branch_type_file) | ||
| 319 | remove_output_file(sample_file) | ||
| 320 | if perf_db_export_calls: | ||
| 321 | remove_output_file(call_path_file) | ||
| 322 | remove_output_file(call_file) | ||
| 323 | os.rmdir(output_dir_name) | ||
| 324 | print datetime.datetime.today(), "Adding primary keys" | ||
| 325 | do_query(query, 'ALTER TABLE selected_events ADD PRIMARY KEY (id)') | ||
| 326 | do_query(query, 'ALTER TABLE machines ADD PRIMARY KEY (id)') | ||
| 327 | do_query(query, 'ALTER TABLE threads ADD PRIMARY KEY (id)') | ||
| 328 | do_query(query, 'ALTER TABLE comms ADD PRIMARY KEY (id)') | ||
| 329 | do_query(query, 'ALTER TABLE comm_threads ADD PRIMARY KEY (id)') | ||
| 330 | do_query(query, 'ALTER TABLE dsos ADD PRIMARY KEY (id)') | ||
| 331 | do_query(query, 'ALTER TABLE symbols ADD PRIMARY KEY (id)') | ||
| 332 | do_query(query, 'ALTER TABLE branch_types ADD PRIMARY KEY (id)') | ||
| 333 | do_query(query, 'ALTER TABLE samples ADD PRIMARY KEY (id)') | ||
| 334 | if perf_db_export_calls: | ||
| 335 | do_query(query, 'ALTER TABLE call_paths ADD PRIMARY KEY (id)') | ||
| 336 | do_query(query, 'ALTER TABLE calls ADD PRIMARY KEY (id)') | ||
| 337 | |||
| 338 | print datetime.datetime.today(), "Adding foreign keys" | ||
| 339 | do_query(query, 'ALTER TABLE threads ' | ||
| 340 | 'ADD CONSTRAINT machinefk FOREIGN KEY (machine_id) REFERENCES machines (id),' | ||
| 341 | 'ADD CONSTRAINT processfk FOREIGN KEY (process_id) REFERENCES threads (id)') | ||
| 342 | do_query(query, 'ALTER TABLE comm_threads ' | ||
| 343 | 'ADD CONSTRAINT commfk FOREIGN KEY (comm_id) REFERENCES comms (id),' | ||
| 344 | 'ADD CONSTRAINT threadfk FOREIGN KEY (thread_id) REFERENCES threads (id)') | ||
| 345 | do_query(query, 'ALTER TABLE dsos ' | ||
| 346 | 'ADD CONSTRAINT machinefk FOREIGN KEY (machine_id) REFERENCES machines (id)') | ||
| 347 | do_query(query, 'ALTER TABLE symbols ' | ||
| 348 | 'ADD CONSTRAINT dsofk FOREIGN KEY (dso_id) REFERENCES dsos (id)') | ||
| 349 | do_query(query, 'ALTER TABLE samples ' | ||
| 350 | 'ADD CONSTRAINT evselfk FOREIGN KEY (evsel_id) REFERENCES selected_events (id),' | ||
| 351 | 'ADD CONSTRAINT machinefk FOREIGN KEY (machine_id) REFERENCES machines (id),' | ||
| 352 | 'ADD CONSTRAINT threadfk FOREIGN KEY (thread_id) REFERENCES threads (id),' | ||
| 353 | 'ADD CONSTRAINT commfk FOREIGN KEY (comm_id) REFERENCES comms (id),' | ||
| 354 | 'ADD CONSTRAINT dsofk FOREIGN KEY (dso_id) REFERENCES dsos (id),' | ||
| 355 | 'ADD CONSTRAINT symbolfk FOREIGN KEY (symbol_id) REFERENCES symbols (id),' | ||
| 356 | 'ADD CONSTRAINT todsofk FOREIGN KEY (to_dso_id) REFERENCES dsos (id),' | ||
| 357 | 'ADD CONSTRAINT tosymbolfk FOREIGN KEY (to_symbol_id) REFERENCES symbols (id)') | ||
| 358 | if perf_db_export_calls: | ||
| 359 | do_query(query, 'ALTER TABLE call_paths ' | ||
| 360 | 'ADD CONSTRAINT parentfk FOREIGN KEY (parent_id) REFERENCES call_paths (id),' | ||
| 361 | 'ADD CONSTRAINT symbolfk FOREIGN KEY (symbol_id) REFERENCES symbols (id)') | ||
| 362 | do_query(query, 'ALTER TABLE calls ' | ||
| 363 | 'ADD CONSTRAINT threadfk FOREIGN KEY (thread_id) REFERENCES threads (id),' | ||
| 364 | 'ADD CONSTRAINT commfk FOREIGN KEY (comm_id) REFERENCES comms (id),' | ||
| 365 | 'ADD CONSTRAINT call_pathfk FOREIGN KEY (call_path_id) REFERENCES call_paths (id),' | ||
| 366 | 'ADD CONSTRAINT callfk FOREIGN KEY (call_id) REFERENCES samples (id),' | ||
| 367 | 'ADD CONSTRAINT returnfk FOREIGN KEY (return_id) REFERENCES samples (id),' | ||
| 368 | 'ADD CONSTRAINT parent_call_pathfk FOREIGN KEY (parent_call_path_id) REFERENCES call_paths (id)') | ||
| 369 | do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)') | ||
| 370 | |||
| 371 | if (unhandled_count): | ||
| 372 | print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events" | ||
| 373 | print datetime.datetime.today(), "Done" | ||
| 374 | |||
| 375 | def trace_unhandled(event_name, context, event_fields_dict): | ||
| 376 | global unhandled_count | ||
| 377 | unhandled_count += 1 | ||
| 378 | |||
| 379 | def sched__sched_switch(*x): | ||
| 380 | pass | ||
| 381 | |||
| 382 | def evsel_table(evsel_id, evsel_name, *x): | ||
| 383 | n = len(evsel_name) | ||
| 384 | fmt = "!hiqi" + str(n) + "s" | ||
| 385 | value = struct.pack(fmt, 2, 8, evsel_id, n, evsel_name) | ||
| 386 | evsel_file.write(value) | ||
| 387 | |||
| 388 | def machine_table(machine_id, pid, root_dir, *x): | ||
| 389 | n = len(root_dir) | ||
| 390 | fmt = "!hiqiii" + str(n) + "s" | ||
| 391 | value = struct.pack(fmt, 3, 8, machine_id, 4, pid, n, root_dir) | ||
| 392 | machine_file.write(value) | ||
| 393 | |||
| 394 | def thread_table(thread_id, machine_id, process_id, pid, tid, *x): | ||
| 395 | value = struct.pack("!hiqiqiqiiii", 5, 8, thread_id, 8, machine_id, 8, process_id, 4, pid, 4, tid) | ||
| 396 | thread_file.write(value) | ||
| 397 | |||
| 398 | def comm_table(comm_id, comm_str, *x): | ||
| 399 | n = len(comm_str) | ||
| 400 | fmt = "!hiqi" + str(n) + "s" | ||
| 401 | value = struct.pack(fmt, 2, 8, comm_id, n, comm_str) | ||
| 402 | comm_file.write(value) | ||
| 403 | |||
| 404 | def comm_thread_table(comm_thread_id, comm_id, thread_id, *x): | ||
| 405 | fmt = "!hiqiqiq" | ||
| 406 | value = struct.pack(fmt, 3, 8, comm_thread_id, 8, comm_id, 8, thread_id) | ||
| 407 | comm_thread_file.write(value) | ||
| 408 | |||
| 409 | def dso_table(dso_id, machine_id, short_name, long_name, build_id, *x): | ||
| 410 | n1 = len(short_name) | ||
| 411 | n2 = len(long_name) | ||
| 412 | n3 = len(build_id) | ||
| 413 | fmt = "!hiqiqi" + str(n1) + "si" + str(n2) + "si" + str(n3) + "s" | ||
| 414 | value = struct.pack(fmt, 5, 8, dso_id, 8, machine_id, n1, short_name, n2, long_name, n3, build_id) | ||
| 415 | dso_file.write(value) | ||
| 416 | |||
| 417 | def symbol_table(symbol_id, dso_id, sym_start, sym_end, binding, symbol_name, *x): | ||
| 418 | n = len(symbol_name) | ||
| 419 | fmt = "!hiqiqiqiqiii" + str(n) + "s" | ||
| 420 | value = struct.pack(fmt, 6, 8, symbol_id, 8, dso_id, 8, sym_start, 8, sym_end, 4, binding, n, symbol_name) | ||
| 421 | symbol_file.write(value) | ||
| 422 | |||
| 423 | def branch_type_table(branch_type, name, *x): | ||
| 424 | n = len(name) | ||
| 425 | fmt = "!hiii" + str(n) + "s" | ||
| 426 | value = struct.pack(fmt, 2, 4, branch_type, n, name) | ||
| 427 | branch_type_file.write(value) | ||
| 428 | |||
| 429 | def sample_table(sample_id, evsel_id, machine_id, thread_id, comm_id, dso_id, symbol_id, sym_offset, ip, time, cpu, to_dso_id, to_symbol_id, to_sym_offset, to_ip, period, weight, transaction, data_src, branch_type, in_tx, *x): | ||
| 430 | if branches: | ||
| 431 | value = struct.pack("!hiqiqiqiqiqiqiqiqiqiqiiiqiqiqiqiiiB", 17, 8, sample_id, 8, evsel_id, 8, machine_id, 8, thread_id, 8, comm_id, 8, dso_id, 8, symbol_id, 8, sym_offset, 8, ip, 8, time, 4, cpu, 8, to_dso_id, 8, to_symbol_id, 8, to_sym_offset, 8, to_ip, 4, branch_type, 1, in_tx) | ||
| 432 | else: | ||
| 433 | value = struct.pack("!hiqiqiqiqiqiqiqiqiqiqiiiqiqiqiqiqiqiqiqiiiB", 21, 8, sample_id, 8, evsel_id, 8, machine_id, 8, thread_id, 8, comm_id, 8, dso_id, 8, symbol_id, 8, sym_offset, 8, ip, 8, time, 4, cpu, 8, to_dso_id, 8, to_symbol_id, 8, to_sym_offset, 8, to_ip, 8, period, 8, weight, 8, transaction, 8, data_src, 4, branch_type, 1, in_tx) | ||
| 434 | sample_file.write(value) | ||
| 435 | |||
| 436 | def call_path_table(cp_id, parent_id, symbol_id, ip, *x): | ||
| 437 | fmt = "!hiqiqiqiq" | ||
| 438 | value = struct.pack(fmt, 4, 8, cp_id, 8, parent_id, 8, symbol_id, 8, ip) | ||
| 439 | call_path_file.write(value) | ||
| 440 | |||
| 441 | def call_return_table(cr_id, thread_id, comm_id, call_path_id, call_time, return_time, branch_count, call_id, return_id, parent_call_path_id, flags, *x): | ||
| 442 | fmt = "!hiqiqiqiqiqiqiqiqiqiqii" | ||
| 443 | value = struct.pack(fmt, 11, 8, cr_id, 8, thread_id, 8, comm_id, 8, call_path_id, 8, call_time, 8, return_time, 8, branch_count, 8, call_id, 8, return_id, 8, parent_call_path_id, 4, flags) | ||
| 444 | call_file.write(value) | ||
