3 # Copyright (C) 2020 Roman Gushchin <guro@fb.com>
4 # Copyright (C) 2020 Facebook
10 from drgn
.helpers
.linux
import list_for_each_entry
, list_empty
11 from drgn
.helpers
.linux
import for_each_page
12 from drgn
.helpers
.linux
.cpumask
import for_each_online_cpu
13 from drgn
.helpers
.linux
.percpu
import per_cpu_ptr
14 from drgn
import container_of
, FaultError
, Object
18 This is a drgn script to provide slab statistics for memory cgroups.
19 It supports cgroup v2 and v1 and can emulate memory.kmem.slabinfo
20 interface of cgroup v1.
21 For drgn, visit https://github.com/osandov/drgn.
28 OO_MASK
= ((1 << OO_SHIFT
) - 1)
32 print('slabinfo.py: error: %s' % s
, file=sys
.stderr
, flush
=True)
36 def find_memcg_ids(css
=prog
['root_mem_cgroup'].css
, prefix
=''):
37 if not list_empty(css
.children
.address_of_()):
38 for css
in list_for_each_entry('struct cgroup_subsys_state',
39 css
.children
.address_of_(),
41 name
= prefix
+ '/' + css
.cgroup
.kn
.name
.string_().decode('utf-8')
42 memcg
= container_of(css
, 'struct mem_cgroup', 'css')
43 MEMCGS
[css
.cgroup
.kn
.id.value_()] = memcg
44 find_memcg_ids(css
, name
)
49 return False if s
.memcg_params
.root_cache
else True
50 except AttributeError:
56 return s
.name
.string_().decode('utf-8')
58 return s
.memcg_params
.root_cache
.name
.string_().decode('utf-8')
64 return s
.oo
.x
>> OO_SHIFT
68 return s
.oo
.x
& OO_MASK
71 def count_partial(n
, fn
):
73 for page
in list_for_each_entry('struct page', n
.partial
.address_of_(),
80 return page
.objects
- page
.inuse
83 def slub_get_slabinfo(s
, cfg
):
88 for node
in range(cfg
['nr_nodes']):
90 nr_slabs
+= n
.nr_slabs
.counter
.value_()
91 nr_objs
+= n
.total_objects
.counter
.value_()
92 nr_free
+= count_partial(n
, count_free
)
94 return {'active_objs': nr_objs
- nr_free
,
96 'active_slabs': nr_slabs
,
97 'num_slabs': nr_slabs
,
98 'objects_per_slab': oo_objects(s
),
99 'cache_order': oo_order(s
),
106 def cache_show(s
, cfg
, objs
):
107 if cfg
['allocator'] == 'SLUB':
108 sinfo
= slub_get_slabinfo(s
, cfg
)
110 err('SLAB isn\'t supported yet')
112 if cfg
['shared_slab_pages']:
113 sinfo
['active_objs'] = objs
114 sinfo
['num_objs'] = objs
116 print('%-17s %6lu %6lu %6u %4u %4d'
117 ' : tunables %4u %4u %4u'
118 ' : slabdata %6lu %6lu %6lu' % (
119 cache_name(s
), sinfo
['active_objs'], sinfo
['num_objs'],
120 s
.size
, sinfo
['objects_per_slab'], 1 << sinfo
['cache_order'],
121 sinfo
['limit'], sinfo
['batchcount'], sinfo
['shared'],
122 sinfo
['active_slabs'], sinfo
['num_slabs'],
123 sinfo
['shared_avail']))
126 def detect_kernel_config():
129 cfg
['nr_nodes'] = prog
['nr_online_nodes'].value_()
131 if prog
.type('struct kmem_cache').members
[1][1] == 'flags':
132 cfg
['allocator'] = 'SLUB'
133 elif prog
.type('struct kmem_cache').members
[1][1] == 'batchcount':
134 cfg
['allocator'] = 'SLAB'
136 err('Can\'t determine the slab allocator')
138 cfg
['shared_slab_pages'] = False
140 if prog
.type('struct obj_cgroup'):
141 cfg
['shared_slab_pages'] = True
148 def for_each_slab_page(prog
):
149 PGSlab
= 1 << prog
.constant('PG_slab')
150 PGHead
= 1 << prog
.constant('PG_head')
152 for page
in for_each_page(prog
):
154 if page
.flags
.value_() & PGSlab
:
161 parser
= argparse
.ArgumentParser(description
=DESC
,
163 argparse
.RawTextHelpFormatter
)
164 parser
.add_argument('cgroup', metavar
='CGROUP',
165 help='Target memory cgroup')
166 args
= parser
.parse_args()
169 cgroup_id
= stat(args
.cgroup
).st_ino
171 memcg
= MEMCGS
[cgroup_id
]
173 err('Can\'t find the memory cgroup')
175 cfg
= detect_kernel_config()
177 print('# name <active_objs> <num_objs> <objsize> <objperslab> <pagesperslab>'
178 ' : tunables <limit> <batchcount> <sharedfactor>'
179 ' : slabdata <active_slabs> <num_slabs> <sharedavail>')
181 if cfg
['shared_slab_pages']:
186 # find memcg pointers belonging to the specified cgroup
187 obj_cgroups
.add(memcg
.objcg
.value_())
188 for ptr
in list_for_each_entry('struct obj_cgroup',
189 memcg
.objcg_list
.address_of_(),
191 obj_cgroups
.add(ptr
.value_())
193 # look over all slab pages, belonging to non-root memcgs
194 # and look for objects belonging to the given memory cgroup
195 for page
in for_each_slab_page(prog
):
196 objcg_vec_raw
= page
.obj_cgroups
.value_()
197 if objcg_vec_raw
== 0:
199 cache
= page
.slab_cache
202 addr
= cache
.value_()
204 # clear the lowest bit to get the true obj_cgroups
205 objcg_vec
= Object(prog
, page
.obj_cgroups
.type_
,
206 value
=objcg_vec_raw
& ~
1)
208 if addr
not in stats
:
211 for i
in range(oo_objects(cache
)):
212 if objcg_vec
[i
].value_() in obj_cgroups
:
217 cache_show(caches
[addr
], cfg
, stats
[addr
])
220 for s
in list_for_each_entry('struct kmem_cache',
221 memcg
.kmem_caches
.address_of_(),
222 'memcg_params.kmem_caches_node'):
223 cache_show(s
, cfg
, None)