aboutsummaryrefslogtreecommitdiffstats
path: root/etc/patch-trace-template.py
blob: 84400c8e02e5074500c53f4d7501f1d8ea7df24f (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
#!/usr/bin/env python

# Copyright 2013 The Servo Project Developers. See the COPYRIGHT
# file at the top-level directory of this distribution.
#
# Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
# http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
# <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
# option. This file may not be copied, modified, or distributed
# except according to those terms.

import re
import subprocess
import sys
from xml.etree import ElementTree
from xml.etree.ElementTree import Element

if len(sys.argv) < 3:
    sys.stderr.write("""usage: patch-trace-template.py components/profile_traits/time.rs path/to/my.tracetemplate

`time.rs` is usually located in `components/profile_traits/time.rs`.
Trace templates are typically located in `~/Library/Application Support/Instruments/Templates`.
The supplied trace template must contain the "Points of Interest" instrument.
Output is written to standard output and should typically be piped to a new `.tracetemplate` file.

Example:
    patch-trace-template.py \\
    components/profile_traits/time.rs \\
        ~/Library/Application\\ Support/Instruments/Templates/MyTemplate.tracetemplate > \\
        ~/Library/Application\\ Support/Instruments/Templates/MyServoTemplate.tracetemplate
""")
    sys.exit(0)

rust_source = open(sys.argv[1], 'r')
lines = iter(rust_source)
for line in lines:
    if line.lstrip().startswith("pub enum ProfilerCategory"):
        break

code_pairs = []
regex = re.compile(r"\s*(\w+)\s*=\s*([0-9]+|0x[a-fA-F0-9]+),?\s*$")
for line in lines:
    if line.lstrip().startswith("}"):
        break

    match = regex.match(line)
    if match is None:
        continue
    code_pairs.append((match.group(2), match.group(1)))

xml = subprocess.check_output(["plutil", "-convert", "xml1", "-o", "-", sys.argv[2]])
plist = ElementTree.ElementTree(ElementTree.fromstring(xml))

elems = iter(plist.findall("./dict/*"))
for elem in elems:
    if elem.tag != 'key' or elem.text != '$objects':
        continue
    array = elems.next()
    break

elems = iter(array.findall("./*"))
for elem in elems:
    if elem.tag != 'string' or elem.text != 'kdebugIntervalRule':
        continue
    dictionary = elems.next()
    break

elems = iter(dictionary.findall("./*"))
for elem in elems:
    if elem.tag != 'key' or elem.text != 'NS.objects':
        continue
    objects_array = elems.next()
    break

child_count = sum(1 for _ in iter(array.findall("./*")))

for code_pair in code_pairs:
    number_index = child_count
    integer = Element('integer')
    integer.text = str(int(code_pair[0], 0))
    array.append(integer)
    child_count += 1

    string_index = child_count
    string = Element('string')
    string.text = code_pair[1]
    array.append(string)
    child_count += 1

    dictionary = Element('dict')
    key = Element('key')
    key.text = "CF$UID"
    dictionary.append(key)
    integer = Element('integer')
    integer.text = str(number_index)
    dictionary.append(integer)
    objects_array.append(dictionary)

    dictionary = Element('dict')
    key = Element('key')
    key.text = "CF$UID"
    dictionary.append(key)
    integer = Element('integer')
    integer.text = str(string_index)
    dictionary.append(integer)
    objects_array.append(dictionary)

plist.write(sys.stdout, encoding='utf-8', xml_declaration=True)