3 # Copyright (C) 2019 Tejun Heo <tj@kernel.org>
4 # Copyright (C) 2019 Facebook
7 This is a drgn script to monitor the blk-iocost cgroup controller.
8 See the comment at the top of block/blk-iocost.c for more details.
9 For drgn, visit https://github.com/osandov/drgn.
19 from drgn
import container_of
20 from drgn
.helpers
.linux
.list import list_for_each_entry
,list_empty
21 from drgn
.helpers
.linux
.radixtree
import radix_tree_for_each
,radix_tree_lookup
24 parser
= argparse
.ArgumentParser(description
=desc
,
25 formatter_class
=argparse
.RawTextHelpFormatter
)
26 parser
.add_argument('devname', metavar
='DEV',
27 help='Target block device name (e.g. sda)')
28 parser
.add_argument('--cgroup', action
='append', metavar
='REGEX',
29 help='Regex for target cgroups, ')
30 parser
.add_argument('--interval', '-i', metavar
='SECONDS', type=float, default
=1,
31 help='Monitoring interval in seconds')
32 parser
.add_argument('--json', action
='store_true',
33 help='Output in json')
34 args
= parser
.parse_args()
37 print(s
, file=sys
.stderr
, flush
=True)
41 blkcg_root
= prog
['blkcg_root']
42 plid
= prog
['blkcg_policy_iocost'].plid
.value_()
44 err('The kernel does not have iocost enabled')
46 IOC_RUNNING
= prog
['IOC_RUNNING'].value_()
47 NR_USAGE_SLOTS
= prog
['NR_USAGE_SLOTS'].value_()
48 HWEIGHT_WHOLE
= prog
['HWEIGHT_WHOLE'].value_()
49 VTIME_PER_SEC
= prog
['VTIME_PER_SEC'].value_()
50 VTIME_PER_USEC
= prog
['VTIME_PER_USEC'].value_()
51 AUTOP_SSD_FAST
= prog
['AUTOP_SSD_FAST'].value_()
52 AUTOP_SSD_DFL
= prog
['AUTOP_SSD_DFL'].value_()
53 AUTOP_SSD_QD1
= prog
['AUTOP_SSD_QD1'].value_()
54 AUTOP_HDD
= prog
['AUTOP_HDD'].value_()
57 AUTOP_SSD_FAST
: 'ssd_fast',
58 AUTOP_SSD_DFL
: 'ssd_dfl',
59 AUTOP_SSD_QD1
: 'ssd_qd1',
64 def blkcg_name(blkcg
):
65 return blkcg
.css
.cgroup
.kn
.name
.string_().decode('utf-8')
67 def walk(self
, blkcg
, q_id
, parent_path
):
68 if not self
.include_dying
and \
69 not (blkcg
.css
.flags
.value_() & prog
['CSS_ONLINE'].value_()):
72 name
= BlkgIterator
.blkcg_name(blkcg
)
73 path
= parent_path
+ '/' + name
if parent_path
else name
74 blkg
= drgn
.Object(prog
, 'struct blkcg_gq',
75 address
=radix_tree_lookup(blkcg
.blkg_tree
.address_of_(), q_id
))
79 self
.blkgs
.append((path
if path
else '/', blkg
))
81 for c
in list_for_each_entry('struct blkcg',
82 blkcg
.css
.children
.address_of_(), 'css.sibling'):
83 self
.walk(c
, q_id
, path
)
85 def __init__(self
, root_blkcg
, q_id
, include_dying
=False):
86 self
.include_dying
= include_dying
88 self
.walk(root_blkcg
, q_id
, '')
91 return iter(self
.blkgs
)
94 def __init__(self
, ioc
):
97 self
.enabled
= ioc
.enabled
.value_()
98 self
.running
= ioc
.running
.value_() == IOC_RUNNING
99 self
.period_ms
= ioc
.period_us
.value_() / 1_000
100 self
.period_at
= ioc
.period_at
.value_() / 1_000_000
101 self
.vperiod_at
= ioc
.period_at_vtime
.value_() / VTIME_PER_SEC
102 self
.vrate_pct
= ioc
.vtime_rate
.counter
.value_() * 100 / VTIME_PER_USEC
103 self
.busy_level
= ioc
.busy_level
.value_()
104 self
.autop_idx
= ioc
.autop_idx
.value_()
105 self
.user_cost_model
= ioc
.user_cost_model
.value_()
106 self
.user_qos_params
= ioc
.user_qos_params
.value_()
108 if self
.autop_idx
in autop_names
:
109 self
.autop_name
= autop_names
[self
.autop_idx
]
111 self
.autop_name
= '?'
114 return { 'device' : devname
,
116 'enabled' : self
.enabled
,
117 'running' : self
.running
,
118 'period_ms' : self
.period_ms
,
119 'period_at' : self
.period_at
,
120 'period_vtime_at' : self
.vperiod_at
,
121 'busy_level' : self
.busy_level
,
122 'vrate_pct' : self
.vrate_pct
, }
124 def table_preamble_str(self
):
125 state
= ('RUN' if self
.running
else 'IDLE') if self
.enabled
else 'OFF'
126 output
= f
'{devname} {state:4} ' \
127 f
'per={self.period_ms}ms ' \
128 f
'cur_per={self.period_at:.3f}:v{self.vperiod_at:.3f} ' \
129 f
'busy={self.busy_level:+3} ' \
130 f
'vrate={self.vrate_pct:6.2f}% ' \
131 f
'params={self.autop_name}'
132 if self
.user_cost_model
or self
.user_qos_params
:
133 output
+= f
'({"C" if self.user_cost_model else ""}{"Q" if self.user_qos_params else ""})'
136 def table_header_str(self
):
137 return f
'{"":25} active {"weight":>9} {"hweight%":>13} {"inflt%":>6} ' \
138 f
'{"dbt":>3} {"delay":>6} {"usages%"}'
141 def __init__(self
, iocg
):
145 self
.is_active
= not list_empty(iocg
.active_list
.address_of_())
146 self
.weight
= iocg
.weight
.value_()
147 self
.active
= iocg
.active
.value_()
148 self
.inuse
= iocg
.inuse
.value_()
149 self
.hwa_pct
= iocg
.hweight_active
.value_() * 100 / HWEIGHT_WHOLE
150 self
.hwi_pct
= iocg
.hweight_inuse
.value_() * 100 / HWEIGHT_WHOLE
151 self
.address
= iocg
.value_()
153 vdone
= iocg
.done_vtime
.counter
.value_()
154 vtime
= iocg
.vtime
.counter
.value_()
155 vrate
= ioc
.vtime_rate
.counter
.value_()
156 period_vtime
= ioc
.period_us
.value_() * vrate
158 self
.inflight_pct
= (vtime
- vdone
) * 100 / period_vtime
160 self
.inflight_pct
= 0
162 # vdebt used to be an atomic64_t and is now u64, support both
164 self
.debt_ms
= iocg
.abs_vdebt
.counter
.value_() / VTIME_PER_USEC
/ 1000
166 self
.debt_ms
= iocg
.abs_vdebt
.value_() / VTIME_PER_USEC
/ 1000
168 self
.use_delay
= blkg
.use_delay
.counter
.value_()
169 self
.delay_ms
= blkg
.delay_nsec
.counter
.value_() / 1_000_000
171 usage_idx
= iocg
.usage_idx
.value_()
174 for i
in range(NR_USAGE_SLOTS
):
175 usage
= iocg
.usages
[(usage_idx
+ i
) % NR_USAGE_SLOTS
].value_()
176 upct
= usage
* 100 / HWEIGHT_WHOLE
177 self
.usages
.append(upct
)
178 self
.usage
= max(self
.usage
, upct
)
180 def dict(self
, now
, path
):
181 out
= { 'cgroup' : path
,
183 'is_active' : self
.is_active
,
184 'weight' : self
.weight
,
185 'weight_active' : self
.active
,
186 'weight_inuse' : self
.inuse
,
187 'hweight_active_pct' : self
.hwa_pct
,
188 'hweight_inuse_pct' : self
.hwi_pct
,
189 'inflight_pct' : self
.inflight_pct
,
190 'debt_ms' : self
.debt_ms
,
191 'use_delay' : self
.use_delay
,
192 'delay_ms' : self
.delay_ms
,
193 'usage_pct' : self
.usage
,
194 'address' : self
.address
}
195 for i
in range(len(self
.usages
)):
196 out
[f
'usage_pct_{i}'] = str(self
.usages
[i
])
199 def table_row_str(self
, path
):
200 out
= f
'{path[-28:]:28} ' \
201 f
'{"*" if self.is_active else " "} ' \
202 f
'{self.inuse:5}/{self.active:5} ' \
203 f
'{self.hwi_pct:6.2f}/{self.hwa_pct:6.2f} ' \
204 f
'{self.inflight_pct:6.2f} ' \
205 f
'{min(math.ceil(self.debt_ms), 999):3} ' \
206 f
'{min(self.use_delay, 99):2}*'\
207 f
'{min(math.ceil(self.delay_ms), 999):03} '
208 for u
in self
.usages
:
209 out
+= f
'{min(round(u), 999):03d}:'
210 out
= out
.rstrip(':')
214 table_fmt
= not args
.json
215 interval
= args
.interval
216 devname
= args
.devname
223 for r
in args
.cgroup
:
229 filter_re
= re
.compile(re_str
) if re_str
else None
236 for i
, ptr
in radix_tree_for_each(blkcg_root
.blkg_tree
.address_of_()):
237 blkg
= drgn
.Object(prog
, 'struct blkcg_gq', address
=ptr
)
239 if devname
== blkg
.q
.kobj
.parent
.name
.string_().decode('utf-8'):
240 q_id
= blkg
.q
.id.value_()
242 root_iocg
= container_of(blkg
.pd
[plid
], 'struct ioc_gq', 'pd')
249 err(f
'Could not find ioc for {devname}');
254 iocstat
= IocStat(ioc
)
258 output
+= '\n' + iocstat
.table_preamble_str()
259 output
+= '\n' + iocstat
.table_header_str()
261 output
+= json
.dumps(iocstat
.dict(now
))
263 for path
, blkg
in BlkgIterator(blkcg_root
, q_id
):
264 if filter_re
and not filter_re
.match(path
):
266 if not blkg
.pd
[plid
]:
269 iocg
= container_of(blkg
.pd
[plid
], 'struct ioc_gq', 'pd')
270 iocg_stat
= IocgStat(iocg
)
272 if not filter_re
and not iocg_stat
.is_active
:
276 output
+= '\n' + iocg_stat
.table_row_str(path
)
278 output
+= '\n' + json
.dumps(iocg_stat
.dict(now
, path
))