File: data_types.py

package info (click to toggle)
android-platform-system-extras 8.1.0%2Br23-2
  • links: PTS, VCS
  • area: main
  • in suites: buster
  • size: 12,020 kB
  • sloc: cpp: 38,496; ansic: 16,188; python: 4,363; sh: 4,026; java: 584; xml: 367; asm: 169; makefile: 20
file content (114 lines) | stat: -rw-r--r-- 3,335 bytes parent folder | download
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
109
110
111
112
113
114
#
# Copyright (C) 2016 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#      http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#

class CallSite:
    def __init__(self, ip, method, dso):
        self.ip = ip
        self.method = method
        self.dso = dso



class Thread:
    def __init__(self, tid):
        self.tid = tid
        self.samples = []
        self.flamegraph = {}
        self.num_samples = 0


    def add_callchain(self, callchain, symbol, sample):
        chain = []
        self.num_samples += 1
        for j in range(callchain.nr):
            entry = callchain.entries[callchain.nr - j - 1]
            if entry.ip == 0:
                continue
            chain.append(CallSite(entry.ip, entry.symbol.symbol_name, entry.symbol.dso_name))

        chain.append(CallSite(sample.ip, symbol.symbol_name, symbol.dso_name))
        self.samples.append(chain)


    def collapse_flamegraph(self):
        flamegraph = FlameGraphCallSite("root", "")
        flamegraph.id = 0 # This is used for wasd navigation, 0 = not a valid target.
        self.flamegraph = flamegraph
        for sample in self.samples:
            flamegraph = self.flamegraph
            for callsite in sample:
               flamegraph = flamegraph.get_callsite(callsite.method, callsite.dso)

        # Populate root note.
        for node in self.flamegraph.callsites:
            self.flamegraph.num_samples += node.num_samples


class Process:
    def __init__(self, name, pid):
        self.name = name
        self.pid = pid
        self.threads = {}
        self.cmd = ""
        self.props = {}
        self.args = None
        self.num_samples = 0

    def get_thread(self, tid):
        if (tid not in self.threads.keys()):
            self.threads[tid] = Thread(tid)
        return self.threads[tid]

CALLSITE_COUNTER = 0
def get_callsite_id():
    global CALLSITE_COUNTER
    CALLSITE_COUNTER += 1
    toReturn = CALLSITE_COUNTER
    return toReturn


class FlameGraphCallSite:

    def __init__(self, method, dso):
        self.callsites = []
        self.method = method
        self.dso = dso
        self.num_samples = 0
        self.offset = 0 # Offset allows position nodes in different branches.
        self.id = get_callsite_id()


    def get_callsite(self, name, dso):
        for c in self.callsites:
            if c.equivalent(name, dso):
                c.num_samples += 1
                return c
        callsite = FlameGraphCallSite(name, dso)
        callsite.num_samples = 1
        self.callsites.append(callsite)
        return callsite

    def equivalent(self, method, dso):
        return self.method == method and self.dso == dso


    def get_max_depth(self):
        max = 0
        for c in self.callsites:
            depth = c.get_max_depth()
            if depth > max:
                max = depth
        return max +1