diff options
author | Beth Flanagan <elizabeth.flanagan@intel.com> | 2011-08-03 12:31:08 -0700 |
---|---|---|
committer | Richard Purdie <richard.purdie@linuxfoundation.org> | 2011-08-05 17:15:59 +0100 |
commit | b525fc13063a6d681a14c871580efaab458dd370 (patch) | |
tree | a1d3f303afd07053032f65ca2e2d62a9e6df6ebb | |
parent | 17e8d8558dbabe2540860ecf1232214dfdf90ff0 (diff) | |
download | poky-b525fc13063a6d681a14c871580efaab458dd370.tar.gz |
buildstats.bbclass: Adding disk io measurement
This commit adds disk-io statistics functionality on a build
and per-task basis.
It pulls measurements for whatever partition TMPDIR exists on.
This data could be off if SSTATE_DIR and DL_DIR exist on
a different partition/volume.
Notes on what this pulls:
ReadsComp: Total number of reads complete
ReadsMerged: Total number of adjacent reads merged
SectRead: Total number of sectors read
TimeReads: Total number of m/s spent reading
WritesComp: Total number of writes completed
SectWrite: Total number of sectors written
TimeWrite: Total number of m/s spent writing
IOinProgress: Total amount of IO in progress at the time of
we look at /proc/diskstats.
TimeIO: Total number of m/s spent doing IO
WTimeIO: Weighted time doing I/O. From iostats.txt:
"This field is incremented at each I/O start, I/O completion, I/O
merge, or read of these stats by the number of I/Os in progress
(field 9) times the number of milliseconds spent doing I/O since the
last update of this field. This can provide an easy measure of both
I/O completion time and the backlog that may be accumulating."
(From OE-Core rev: eeebcebf0d695358e72f3aed753f66cddd5e0e61)
Signed-off-by: Beth Flanagan <elizabeth.flanagan@intel.com>
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
-rw-r--r-- | meta/classes/buildstats.bbclass | 220 |
1 files changed, 142 insertions, 78 deletions
diff --git a/meta/classes/buildstats.bbclass b/meta/classes/buildstats.bbclass index 8077035367..939e3af603 100644 --- a/meta/classes/buildstats.bbclass +++ b/meta/classes/buildstats.bbclass | |||
@@ -1,5 +1,6 @@ | |||
1 | BUILDSTATS_BASE = ${TMPDIR}/buildstats/ | 1 | BUILDSTATS_BASE = ${TMPDIR}/buildstats/ |
2 | BNFILE = ${BUILDSTATS_BASE}/.buildname | 2 | BNFILE = ${BUILDSTATS_BASE}/.buildname |
3 | DEVFILE = ${BUILDSTATS_BASE}/.device | ||
3 | 4 | ||
4 | ################################################################################ | 5 | ################################################################################ |
5 | # Build statistics gathering. | 6 | # Build statistics gathering. |
@@ -19,9 +20,79 @@ def get_cputime(): | |||
19 | fields = open("/proc/stat", "r").readline().rstrip().split()[1:] | 20 | fields = open("/proc/stat", "r").readline().rstrip().split()[1:] |
20 | return sum(int(field) for field in fields) | 21 | return sum(int(field) for field in fields) |
21 | 22 | ||
23 | def set_bn(e): | ||
24 | bn = e.getPkgs()[0] + "-" + bb.data.getVar('MACHINE',e.data, True) | ||
25 | try: | ||
26 | os.remove(bb.data.getVar('BNFILE', e.data, True)) | ||
27 | except: | ||
28 | pass | ||
29 | file = open(bb.data.getVar('BNFILE', e.data, True), "w") | ||
30 | file.write(os.path.join(bn, bb.data.getVar('BUILDNAME', e.data, True))) | ||
31 | file.close() | ||
32 | |||
33 | def get_bn(e): | ||
34 | file = open(bb.data.getVar('BNFILE', e.data, True)) | ||
35 | bn = file.readline() | ||
36 | file.close() | ||
37 | return bn | ||
38 | |||
39 | def set_device(e): | ||
40 | tmpdir = bb.data.getVar('TMPDIR', e.data, True) | ||
41 | try: | ||
42 | os.remove(bb.data.getVar('DEVFILE', e.data, True)) | ||
43 | except: | ||
44 | pass | ||
45 | ############################################################################ | ||
46 | # We look for the volume TMPDIR lives on. To do all disks would make little | ||
47 | # sense and not give us any particularly useful data. In theory we could do | ||
48 | # something like stick DL_DIR on a different partition and this would | ||
49 | # throw stats gathering off. The same goes with SSTATE_DIR. However, let's | ||
50 | # get the basics in here and work on the cornercases later. | ||
51 | ############################################################################ | ||
52 | device=os.stat(tmpdir) | ||
53 | majordev=os.major(device.st_dev) | ||
54 | minordev=os.minor(device.st_dev) | ||
55 | for line in open("/proc/diskstats", "r"): | ||
56 | if majordev == int(line.split()[0]) and minordev == int(line.split()[1]): | ||
57 | rdev=line.split()[2] | ||
58 | file = open(bb.data.getVar('DEVFILE', e.data, True), "w") | ||
59 | file.write(rdev) | ||
60 | file.close() | ||
61 | |||
62 | def get_device(e): | ||
63 | file = open(bb.data.getVar('DEVFILE', e.data, True)) | ||
64 | device = file.readline() | ||
65 | file.close() | ||
66 | return device | ||
67 | |||
68 | def get_diskstats(dev): | ||
69 | import itertools | ||
70 | ############################################################################ | ||
71 | # For info on what these are, see kernel doc file iostats.txt | ||
72 | ############################################################################ | ||
73 | DSTAT_KEYS = ['ReadsComp', 'ReadsMerged', 'SectRead', 'TimeReads', 'WritesComp', 'SectWrite', 'TimeWrite', 'IOinProgress', 'TimeIO', 'WTimeIO'] | ||
74 | for x in open("/proc/diskstats", "r"): | ||
75 | if dev in x: | ||
76 | diskstats_val = x.rstrip().split()[4:] | ||
77 | diskstats = dict(itertools.izip(DSTAT_KEYS, diskstats_val)) | ||
78 | return diskstats | ||
79 | |||
80 | def set_diskdata(var, dev, data): | ||
81 | data.setVar(var, get_diskstats(dev)) | ||
82 | |||
83 | def get_diskdata(var, dev, data): | ||
84 | olddiskdata = data.getVar(var, False) | ||
85 | diskdata = {} | ||
86 | if olddiskdata is None: | ||
87 | return | ||
88 | newdiskdata = get_diskstats(dev) | ||
89 | for key in olddiskdata.iterkeys(): | ||
90 | diskdata["Start"+key] = str(int(olddiskdata[key])) | ||
91 | diskdata["End"+key] = str(int(newdiskdata[key])) | ||
92 | return diskdata | ||
93 | |||
22 | def set_timedata(var, data): | 94 | def set_timedata(var, data): |
23 | import time | 95 | import time |
24 | |||
25 | time = time.time() | 96 | time = time.time() |
26 | cputime = get_cputime() | 97 | cputime = get_cputime() |
27 | proctime = get_process_cputime(os.getpid()) | 98 | proctime = get_process_cputime(os.getpid()) |
@@ -42,26 +113,35 @@ def get_timedata(var, data): | |||
42 | cpuperc = None | 113 | cpuperc = None |
43 | return timediff, cpuperc | 114 | return timediff, cpuperc |
44 | 115 | ||
45 | ############################################## | 116 | def write_task_data(status, logfile, dev, e): |
46 | # We need to set the buildname to a file since | 117 | bn = get_bn(e) |
47 | # BUILDNAME changes throughout a build | 118 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) |
48 | ############################################## | 119 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) |
49 | 120 | file = open(os.path.join(logfile), "a") | |
50 | def set_bn(e): | 121 | timedata = get_timedata("__timedata_task", e.data) |
51 | bn = e.getPkgs()[0] + "-" + bb.data.getVar('MACHINE',e.data, True) | 122 | if timedata: |
52 | try: | 123 | elapsedtime, cpu = timedata |
53 | os.remove(bb.data.getVar('BNFILE',e.data, True)) | 124 | file.write(bb.data.expand("${PF}: %s: Elapsed time: %0.2f seconds \n" % |
54 | except: | 125 | (e.task, elapsedtime), e.data)) |
55 | pass | 126 | if cpu: |
56 | file = open(bb.data.getVar('BNFILE',e.data, True), "w") | 127 | file.write("CPU usage: %0.1f%% \n" % cpu) |
57 | file.write(os.path.join(bn, bb.data.getVar('BUILDNAME', e.data, True))) | 128 | ############################################################################ |
58 | file.close() | 129 | # Here we gather up disk data. In an effort to avoid lying with stats |
59 | 130 | # I do a bare minimum of analysis of collected data. | |
60 | def get_bn(e): | 131 | # The simple fact is, doing disk io collection on a per process basis |
61 | file = open(bb.data.getVar('BNFILE',e.data, True)) | 132 | # without effecting build time would be difficult. |
62 | bn = file.readline() | 133 | # For the best information, running things with BB_TOTAL_THREADS = "1" |
134 | # would return accurate per task results. | ||
135 | ############################################################################ | ||
136 | diskdata = get_diskdata("__diskdata_task", dev, e.data) | ||
137 | if diskdata: | ||
138 | for key in sorted(diskdata.iterkeys()): | ||
139 | file.write(key + ": " + diskdata[key] + "\n") | ||
140 | if status is "passed": | ||
141 | file.write("Status: PASSED") | ||
142 | else: | ||
143 | file.write("Status: FAILED") | ||
63 | file.close() | 144 | file.close() |
64 | return bn | ||
65 | 145 | ||
66 | python run_buildstats () { | 146 | python run_buildstats () { |
67 | import bb.build | 147 | import bb.build |
@@ -69,22 +149,27 @@ python run_buildstats () { | |||
69 | import bb.data | 149 | import bb.data |
70 | import time, subprocess, platform | 150 | import time, subprocess, platform |
71 | 151 | ||
152 | |||
72 | if isinstance(e, bb.event.BuildStarted): | 153 | if isinstance(e, bb.event.BuildStarted): |
73 | ############################################## | 154 | ######################################################################## |
74 | # at first pass make the buildstats heriarchy and then | 155 | # at first pass make the buildstats heriarchy and then |
75 | # set the buildname | 156 | # set the buildname |
76 | ############################################## | 157 | ######################################################################## |
77 | try: | 158 | try: |
78 | bb.mkdirhier(bb.data.getVar('BUILDSTATS_BASE', e.data, True)) | 159 | bb.mkdirhier(bb.data.getVar('BUILDSTATS_BASE', e.data, True)) |
79 | except: | 160 | except: |
80 | pass | 161 | pass |
81 | set_bn(e) | 162 | set_bn(e) |
82 | bn = get_bn(e) | 163 | bn = get_bn(e) |
164 | set_device(e) | ||
165 | device = get_device(e) | ||
166 | |||
83 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) | 167 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) |
84 | try: | 168 | try: |
85 | bb.mkdirhier(bsdir) | 169 | bb.mkdirhier(bsdir) |
86 | except: | 170 | except: |
87 | pass | 171 | pass |
172 | set_diskdata("__diskdata_build", device, e.data) | ||
88 | set_timedata("__timedata_build", e.data) | 173 | set_timedata("__timedata_build", e.data) |
89 | build_time = os.path.join(bsdir, "build_stats") | 174 | build_time = os.path.join(bsdir, "build_stats") |
90 | # write start of build into build_time | 175 | # write start of build into build_time |
@@ -99,27 +184,36 @@ python run_buildstats () { | |||
99 | file.close() | 184 | file.close() |
100 | 185 | ||
101 | elif isinstance(e, bb.event.BuildCompleted): | 186 | elif isinstance(e, bb.event.BuildCompleted): |
102 | bn=get_bn(e) | 187 | bn = get_bn(e) |
103 | timedata = get_timedata("__timedata_build", e.data) | 188 | dev = get_device(e) |
104 | if not timedata: | ||
105 | return | ||
106 | time, cpu = timedata | ||
107 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) | 189 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) |
190 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) | ||
108 | build_time = os.path.join(bsdir, "build_stats") | 191 | build_time = os.path.join(bsdir, "build_stats") |
109 | # write end of build and cpu used into build_time | ||
110 | file = open(build_time, "a") | 192 | file = open(build_time, "a") |
111 | file.write("Elapsed time: %0.2f seconds \n" % (time)) | 193 | ######################################################################## |
112 | if cpu: | 194 | # Write build statistics for the build |
113 | file.write("CPU usage: %0.1f%% \n" % cpu) | 195 | ######################################################################## |
196 | timedata = get_timedata("__timedata_build", e.data) | ||
197 | if timedata: | ||
198 | time, cpu = timedata | ||
199 | # write end of build and cpu used into build_time | ||
200 | file = open(build_time, "a") | ||
201 | file.write("Elapsed time: %0.2f seconds \n" % (time)) | ||
202 | if cpu: | ||
203 | file.write("CPU usage: %0.1f%% \n" % cpu) | ||
204 | diskio = get_diskdata("__diskdata_build", dev, e.data) | ||
205 | if diskio: | ||
206 | for key in sorted(diskio.iterkeys()): | ||
207 | file.write(key + ": " + diskio[key] + "\n") | ||
114 | file.close() | 208 | file.close() |
115 | 209 | ||
116 | |||
117 | if isinstance(e, bb.build.TaskStarted): | 210 | if isinstance(e, bb.build.TaskStarted): |
118 | bn=get_bn(e) | 211 | bn = get_bn(e) |
119 | set_timedata("__timedata_task", e.data) | 212 | device = get_device(e) |
120 | |||
121 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) | 213 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) |
122 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) | 214 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) |
215 | set_diskdata("__diskdata_task", device, e.data) | ||
216 | set_timedata("__timedata_task", e.data) | ||
123 | try: | 217 | try: |
124 | bb.mkdirhier(taskdir) | 218 | bb.mkdirhier(taskdir) |
125 | except: | 219 | except: |
@@ -131,29 +225,13 @@ python run_buildstats () { | |||
131 | file.close() | 225 | file.close() |
132 | 226 | ||
133 | elif isinstance(e, bb.build.TaskSucceeded): | 227 | elif isinstance(e, bb.build.TaskSucceeded): |
134 | bn=get_bn(e) | 228 | bn = get_bn(e) |
135 | timedata = get_timedata("__timedata_task", e.data) | 229 | device = get_device(e) |
136 | if not timedata: | ||
137 | return | ||
138 | elapsedtime, cpu = timedata | ||
139 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) | 230 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) |
140 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) | 231 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) |
141 | file = open(os.path.join(taskdir, e.task), "a") | 232 | write_task_data("passed", os.path.join(taskdir, e.task), device, e) |
142 | file.write(bb.data.expand("${PF}: %s: Elapsed time: %0.2f seconds \n" % | ||
143 | (e.task, elapsedtime), e.data)) | ||
144 | file.write("Ended: %0.2f \n" % time.time()) | ||
145 | if cpu: | ||
146 | file.write("CPU usage: %0.1f%% \n" % cpu) | ||
147 | |||
148 | file.write("Status: PASSED") | ||
149 | file.close() | ||
150 | |||
151 | ############################################## | ||
152 | # Alot of metric gathering occurs here. | ||
153 | # Reminder: I stripped out some in process stuff here | ||
154 | ############################################## | ||
155 | |||
156 | if e.task == "do_rootfs": | 233 | if e.task == "do_rootfs": |
234 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) | ||
157 | bs=os.path.join(bsdir, "build_stats") | 235 | bs=os.path.join(bsdir, "build_stats") |
158 | file = open(bs,"a") | 236 | file = open(bs,"a") |
159 | rootfs = bb.data.getVar('IMAGE_ROOTFS', e.data, True) | 237 | rootfs = bb.data.getVar('IMAGE_ROOTFS', e.data, True) |
@@ -162,35 +240,21 @@ python run_buildstats () { | |||
162 | file.close() | 240 | file.close() |
163 | 241 | ||
164 | elif isinstance(e, bb.build.TaskFailed): | 242 | elif isinstance(e, bb.build.TaskFailed): |
165 | bn=get_bn(e) | 243 | bn = get_bn(e) |
166 | timedata = get_timedata("__timedata_task", e.data) | 244 | device = get_device(e) |
167 | if not timedata: | ||
168 | return | ||
169 | time, cpu = timedata | ||
170 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) | 245 | bsdir = os.path.join(bb.data.getVar('BUILDSTATS_BASE', e.data, True), bn) |
171 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) | 246 | taskdir = os.path.join(bsdir, bb.data.expand("${PF}", e.data)) |
172 | ############################################## | 247 | write_task_data("failed", os.path.join(taskdir, e.task), device, e) |
173 | # If the task fails dump the regular data. | 248 | ######################################################################## |
174 | # fgrep -R "FAILED" <bsdir> | 249 | # Lets make things easier and tell people where the build failed in |
175 | # will grep all the events that failed. | 250 | # build_status. We do this here because BuildCompleted triggers no |
176 | ############################################## | 251 | # matter what the status of the build actually is |
177 | file = open(os.path.join(taskdir, e.task), "a") | 252 | ######################################################################## |
178 | file.write(bb.data.expand("${PF}: %s: Elapsed time: %0.2f seconds \n" % | ||
179 | (e.task, time), e.data)) | ||
180 | if cpu: | ||
181 | file.write("CPU usage: %0.1f%% \n" % cpu) | ||
182 | file.write("Status: FAILED") | ||
183 | file.close() | ||
184 | ############################################## | ||
185 | # Lets make things easier and tell people where the build failed in build_status | ||
186 | # We do this here because BuildCompleted triggers no matter what the status of the | ||
187 | # build actually is | ||
188 | ############################################## | ||
189 | build_status = os.path.join(bsdir, "build_stats") | 253 | build_status = os.path.join(bsdir, "build_stats") |
190 | file = open(build_status,"a") | 254 | file = open(build_status,"a") |
191 | file.write(bb.data.expand("Failed at: ${PF} at task: %s \n" % e.task, e.data)) | 255 | file.write(bb.data.expand("Failed at: ${PF} at task: %s \n" % e.task, e.data)) |
192 | file.close() | 256 | file.close() |
193 | 257 | ||
194 | } | 258 | } |
195 | 259 | ||
196 | addhandler run_buildstats | 260 | addhandler run_buildstats |