Blame view

block/blk-mq-sysfs.c 8.31 KB
3dcf60bcb   Christoph Hellwig   block: add SPDX t...
1
  // SPDX-License-Identifier: GPL-2.0
320ae51fe   Jens Axboe   blk-mq: new multi...
2
3
4
5
6
7
8
9
10
11
12
13
  #include <linux/kernel.h>
  #include <linux/module.h>
  #include <linux/backing-dev.h>
  #include <linux/bio.h>
  #include <linux/blkdev.h>
  #include <linux/mm.h>
  #include <linux/init.h>
  #include <linux/slab.h>
  #include <linux/workqueue.h>
  #include <linux/smp.h>
  
  #include <linux/blk-mq.h>
c7e2d94b3   Ming Lei   blk-mq: free hw q...
14
  #include "blk.h"
320ae51fe   Jens Axboe   blk-mq: new multi...
15
16
17
18
19
  #include "blk-mq.h"
  #include "blk-mq-tag.h"
  
  static void blk_mq_sysfs_release(struct kobject *kobj)
  {
1db4909e7   Ming Lei   blk-mq: not embed...
20
21
22
23
24
25
26
27
28
29
30
31
  	struct blk_mq_ctxs *ctxs = container_of(kobj, struct blk_mq_ctxs, kobj);
  
  	free_percpu(ctxs->queue_ctx);
  	kfree(ctxs);
  }
  
  static void blk_mq_ctx_sysfs_release(struct kobject *kobj)
  {
  	struct blk_mq_ctx *ctx = container_of(kobj, struct blk_mq_ctx, kobj);
  
  	/* ctx->ctxs won't be released until all ctx are freed */
  	kobject_put(&ctx->ctxs->kobj);
320ae51fe   Jens Axboe   blk-mq: new multi...
32
  }
6c8b232ef   Ming Lei   blk-mq: make life...
33
34
35
36
  static void blk_mq_hw_sysfs_release(struct kobject *kobj)
  {
  	struct blk_mq_hw_ctx *hctx = container_of(kobj, struct blk_mq_hw_ctx,
  						  kobj);
c7e2d94b3   Ming Lei   blk-mq: free hw q...
37
38
39
40
41
  
  	if (hctx->flags & BLK_MQ_F_BLOCKING)
  		cleanup_srcu_struct(hctx->srcu);
  	blk_free_flush_queue(hctx->fq);
  	sbitmap_free(&hctx->ctx_map);
01388df37   Ming Lei   blk-mq: free hctx...
42
  	free_cpumask_var(hctx->cpumask);
6c8b232ef   Ming Lei   blk-mq: make life...
43
44
45
  	kfree(hctx->ctxs);
  	kfree(hctx);
  }
320ae51fe   Jens Axboe   blk-mq: new multi...
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
  struct blk_mq_ctx_sysfs_entry {
  	struct attribute attr;
  	ssize_t (*show)(struct blk_mq_ctx *, char *);
  	ssize_t (*store)(struct blk_mq_ctx *, const char *, size_t);
  };
  
  struct blk_mq_hw_ctx_sysfs_entry {
  	struct attribute attr;
  	ssize_t (*show)(struct blk_mq_hw_ctx *, char *);
  	ssize_t (*store)(struct blk_mq_hw_ctx *, const char *, size_t);
  };
  
  static ssize_t blk_mq_sysfs_show(struct kobject *kobj, struct attribute *attr,
  				 char *page)
  {
  	struct blk_mq_ctx_sysfs_entry *entry;
  	struct blk_mq_ctx *ctx;
  	struct request_queue *q;
  	ssize_t res;
  
  	entry = container_of(attr, struct blk_mq_ctx_sysfs_entry, attr);
  	ctx = container_of(kobj, struct blk_mq_ctx, kobj);
  	q = ctx->queue;
  
  	if (!entry->show)
  		return -EIO;
320ae51fe   Jens Axboe   blk-mq: new multi...
72
  	mutex_lock(&q->sysfs_lock);
bae85c156   Bart Van Assche   block: Remove "dy...
73
  	res = entry->show(ctx, page);
320ae51fe   Jens Axboe   blk-mq: new multi...
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
  	mutex_unlock(&q->sysfs_lock);
  	return res;
  }
  
  static ssize_t blk_mq_sysfs_store(struct kobject *kobj, struct attribute *attr,
  				  const char *page, size_t length)
  {
  	struct blk_mq_ctx_sysfs_entry *entry;
  	struct blk_mq_ctx *ctx;
  	struct request_queue *q;
  	ssize_t res;
  
  	entry = container_of(attr, struct blk_mq_ctx_sysfs_entry, attr);
  	ctx = container_of(kobj, struct blk_mq_ctx, kobj);
  	q = ctx->queue;
  
  	if (!entry->store)
  		return -EIO;
320ae51fe   Jens Axboe   blk-mq: new multi...
92
  	mutex_lock(&q->sysfs_lock);
bae85c156   Bart Van Assche   block: Remove "dy...
93
  	res = entry->store(ctx, page, length);
320ae51fe   Jens Axboe   blk-mq: new multi...
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
  	mutex_unlock(&q->sysfs_lock);
  	return res;
  }
  
  static ssize_t blk_mq_hw_sysfs_show(struct kobject *kobj,
  				    struct attribute *attr, char *page)
  {
  	struct blk_mq_hw_ctx_sysfs_entry *entry;
  	struct blk_mq_hw_ctx *hctx;
  	struct request_queue *q;
  	ssize_t res;
  
  	entry = container_of(attr, struct blk_mq_hw_ctx_sysfs_entry, attr);
  	hctx = container_of(kobj, struct blk_mq_hw_ctx, kobj);
  	q = hctx->queue;
  
  	if (!entry->show)
  		return -EIO;
320ae51fe   Jens Axboe   blk-mq: new multi...
112
  	mutex_lock(&q->sysfs_lock);
bae85c156   Bart Van Assche   block: Remove "dy...
113
  	res = entry->show(hctx, page);
320ae51fe   Jens Axboe   blk-mq: new multi...
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
  	mutex_unlock(&q->sysfs_lock);
  	return res;
  }
  
  static ssize_t blk_mq_hw_sysfs_store(struct kobject *kobj,
  				     struct attribute *attr, const char *page,
  				     size_t length)
  {
  	struct blk_mq_hw_ctx_sysfs_entry *entry;
  	struct blk_mq_hw_ctx *hctx;
  	struct request_queue *q;
  	ssize_t res;
  
  	entry = container_of(attr, struct blk_mq_hw_ctx_sysfs_entry, attr);
  	hctx = container_of(kobj, struct blk_mq_hw_ctx, kobj);
  	q = hctx->queue;
  
  	if (!entry->store)
  		return -EIO;
320ae51fe   Jens Axboe   blk-mq: new multi...
133
  	mutex_lock(&q->sysfs_lock);
bae85c156   Bart Van Assche   block: Remove "dy...
134
  	res = entry->store(hctx, page, length);
320ae51fe   Jens Axboe   blk-mq: new multi...
135
136
137
  	mutex_unlock(&q->sysfs_lock);
  	return res;
  }
d96b37c0a   Omar Sandoval   blk-mq: move tags...
138
139
  static ssize_t blk_mq_hw_sysfs_nr_tags_show(struct blk_mq_hw_ctx *hctx,
  					    char *page)
bd166ef18   Jens Axboe   blk-mq-sched: add...
140
  {
d96b37c0a   Omar Sandoval   blk-mq: move tags...
141
142
  	return sprintf(page, "%u
  ", hctx->tags->nr_tags);
bd166ef18   Jens Axboe   blk-mq-sched: add...
143
  }
d96b37c0a   Omar Sandoval   blk-mq: move tags...
144
145
  static ssize_t blk_mq_hw_sysfs_nr_reserved_tags_show(struct blk_mq_hw_ctx *hctx,
  						     char *page)
320ae51fe   Jens Axboe   blk-mq: new multi...
146
  {
d96b37c0a   Omar Sandoval   blk-mq: move tags...
147
148
  	return sprintf(page, "%u
  ", hctx->tags->nr_reserved_tags);
320ae51fe   Jens Axboe   blk-mq: new multi...
149
  }
676141e48   Jens Axboe   blk-mq: don't dum...
150
151
  static ssize_t blk_mq_hw_sysfs_cpus_show(struct blk_mq_hw_ctx *hctx, char *page)
  {
8962842ca   Ming Lei   blk-mq: avoid sys...
152
  	const size_t size = PAGE_SIZE - 1;
cb2da43e3   Jens Axboe   blk-mq: simplify ...
153
  	unsigned int i, first = 1;
8962842ca   Ming Lei   blk-mq: avoid sys...
154
  	int ret = 0, pos = 0;
676141e48   Jens Axboe   blk-mq: don't dum...
155

cb2da43e3   Jens Axboe   blk-mq: simplify ...
156
  	for_each_cpu(i, hctx->cpumask) {
676141e48   Jens Axboe   blk-mq: don't dum...
157
  		if (first)
8962842ca   Ming Lei   blk-mq: avoid sys...
158
  			ret = snprintf(pos + page, size - pos, "%u", i);
676141e48   Jens Axboe   blk-mq: don't dum...
159
  		else
8962842ca   Ming Lei   blk-mq: avoid sys...
160
161
162
163
  			ret = snprintf(pos + page, size - pos, ", %u", i);
  
  		if (ret >= size - pos)
  			break;
676141e48   Jens Axboe   blk-mq: don't dum...
164
165
  
  		first = 0;
8962842ca   Ming Lei   blk-mq: avoid sys...
166
  		pos += ret;
676141e48   Jens Axboe   blk-mq: don't dum...
167
  	}
d2c9be89f   Ming Lei   blk-mq: make sure...
168
169
  	ret = snprintf(pos + page, size + 1 - pos, "
  ");
8962842ca   Ming Lei   blk-mq: avoid sys...
170
  	return pos + ret;
676141e48   Jens Axboe   blk-mq: don't dum...
171
  }
d96b37c0a   Omar Sandoval   blk-mq: move tags...
172
  static struct blk_mq_hw_ctx_sysfs_entry blk_mq_hw_sysfs_nr_tags = {
5657a819a   Joe Perches   block drivers/blo...
173
  	.attr = {.name = "nr_tags", .mode = 0444 },
d96b37c0a   Omar Sandoval   blk-mq: move tags...
174
175
176
  	.show = blk_mq_hw_sysfs_nr_tags_show,
  };
  static struct blk_mq_hw_ctx_sysfs_entry blk_mq_hw_sysfs_nr_reserved_tags = {
5657a819a   Joe Perches   block drivers/blo...
177
  	.attr = {.name = "nr_reserved_tags", .mode = 0444 },
d96b37c0a   Omar Sandoval   blk-mq: move tags...
178
179
  	.show = blk_mq_hw_sysfs_nr_reserved_tags_show,
  };
676141e48   Jens Axboe   blk-mq: don't dum...
180
  static struct blk_mq_hw_ctx_sysfs_entry blk_mq_hw_sysfs_cpus = {
5657a819a   Joe Perches   block drivers/blo...
181
  	.attr = {.name = "cpu_list", .mode = 0444 },
676141e48   Jens Axboe   blk-mq: don't dum...
182
183
  	.show = blk_mq_hw_sysfs_cpus_show,
  };
320ae51fe   Jens Axboe   blk-mq: new multi...
184
185
  
  static struct attribute *default_hw_ctx_attrs[] = {
d96b37c0a   Omar Sandoval   blk-mq: move tags...
186
187
  	&blk_mq_hw_sysfs_nr_tags.attr,
  	&blk_mq_hw_sysfs_nr_reserved_tags.attr,
676141e48   Jens Axboe   blk-mq: don't dum...
188
  	&blk_mq_hw_sysfs_cpus.attr,
320ae51fe   Jens Axboe   blk-mq: new multi...
189
190
  	NULL,
  };
800f5aa1e   Kimberly Brown   block: Replace al...
191
  ATTRIBUTE_GROUPS(default_hw_ctx);
320ae51fe   Jens Axboe   blk-mq: new multi...
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
  
  static const struct sysfs_ops blk_mq_sysfs_ops = {
  	.show	= blk_mq_sysfs_show,
  	.store	= blk_mq_sysfs_store,
  };
  
  static const struct sysfs_ops blk_mq_hw_sysfs_ops = {
  	.show	= blk_mq_hw_sysfs_show,
  	.store	= blk_mq_hw_sysfs_store,
  };
  
  static struct kobj_type blk_mq_ktype = {
  	.sysfs_ops	= &blk_mq_sysfs_ops,
  	.release	= blk_mq_sysfs_release,
  };
  
  static struct kobj_type blk_mq_ctx_ktype = {
  	.sysfs_ops	= &blk_mq_sysfs_ops,
1db4909e7   Ming Lei   blk-mq: not embed...
210
  	.release	= blk_mq_ctx_sysfs_release,
320ae51fe   Jens Axboe   blk-mq: new multi...
211
212
213
214
  };
  
  static struct kobj_type blk_mq_hw_ktype = {
  	.sysfs_ops	= &blk_mq_hw_sysfs_ops,
800f5aa1e   Kimberly Brown   block: Replace al...
215
  	.default_groups = default_hw_ctx_groups,
6c8b232ef   Ming Lei   blk-mq: make life...
216
  	.release	= blk_mq_hw_sysfs_release,
320ae51fe   Jens Axboe   blk-mq: new multi...
217
  };
ee3c5db08   Fengguang Wu   blk-mq: blk_mq_un...
218
  static void blk_mq_unregister_hctx(struct blk_mq_hw_ctx *hctx)
67aec14ce   Jens Axboe   blk-mq: make the ...
219
220
221
  {
  	struct blk_mq_ctx *ctx;
  	int i;
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
222
  	if (!hctx->nr_ctx)
67aec14ce   Jens Axboe   blk-mq: make the ...
223
224
225
226
227
228
229
  		return;
  
  	hctx_for_each_ctx(hctx, ctx, i)
  		kobject_del(&ctx->kobj);
  
  	kobject_del(&hctx->kobj);
  }
ee3c5db08   Fengguang Wu   blk-mq: blk_mq_un...
230
  static int blk_mq_register_hctx(struct blk_mq_hw_ctx *hctx)
67aec14ce   Jens Axboe   blk-mq: make the ...
231
232
233
234
  {
  	struct request_queue *q = hctx->queue;
  	struct blk_mq_ctx *ctx;
  	int i, ret;
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
235
  	if (!hctx->nr_ctx)
67aec14ce   Jens Axboe   blk-mq: make the ...
236
  		return 0;
1db4909e7   Ming Lei   blk-mq: not embed...
237
  	ret = kobject_add(&hctx->kobj, q->mq_kobj, "%u", hctx->queue_num);
67aec14ce   Jens Axboe   blk-mq: make the ...
238
239
240
241
242
243
244
245
246
247
248
  	if (ret)
  		return ret;
  
  	hctx_for_each_ctx(hctx, ctx, i) {
  		ret = kobject_add(&ctx->kobj, &hctx->kobj, "cpu%u", ctx->cpu);
  		if (ret)
  			break;
  	}
  
  	return ret;
  }
667257e8b   Mike Snitzer   block: properly p...
249
  void blk_mq_unregister_dev(struct device *dev, struct request_queue *q)
320ae51fe   Jens Axboe   blk-mq: new multi...
250
  {
851573660   Andrey Vagin   block: fix memory...
251
  	struct blk_mq_hw_ctx *hctx;
7ea5fe31c   Ming Lei   blk-mq: make life...
252
  	int i;
851573660   Andrey Vagin   block: fix memory...
253

cecf5d87f   Ming Lei   block: split .sys...
254
  	lockdep_assert_held(&q->sysfs_dir_lock);
2d0364c8c   Bart Van Assche   blk-mq: Register ...
255

6c8b232ef   Ming Lei   blk-mq: make life...
256
  	queue_for_each_hw_ctx(q, hctx, i)
67aec14ce   Jens Axboe   blk-mq: make the ...
257
  		blk_mq_unregister_hctx(hctx);
1db4909e7   Ming Lei   blk-mq: not embed...
258
259
  	kobject_uevent(q->mq_kobj, KOBJ_REMOVE);
  	kobject_del(q->mq_kobj);
b21d5b301   Matias Bjørling   blk-mq: register ...
260
  	kobject_put(&dev->kobj);
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
261
262
  
  	q->mq_sysfs_init_done = false;
c0f3fd2b3   Jens Axboe   blk-mq: fix deadl...
263
  }
868f2f0b7   Keith Busch   blk-mq: dynamic h...
264
265
266
267
  void blk_mq_hctx_kobj_init(struct blk_mq_hw_ctx *hctx)
  {
  	kobject_init(&hctx->kobj, &blk_mq_hw_ktype);
  }
7ea5fe31c   Ming Lei   blk-mq: make life...
268
269
270
271
272
273
274
275
276
  void blk_mq_sysfs_deinit(struct request_queue *q)
  {
  	struct blk_mq_ctx *ctx;
  	int cpu;
  
  	for_each_possible_cpu(cpu) {
  		ctx = per_cpu_ptr(q->queue_ctx, cpu);
  		kobject_put(&ctx->kobj);
  	}
1db4909e7   Ming Lei   blk-mq: not embed...
277
  	kobject_put(q->mq_kobj);
7ea5fe31c   Ming Lei   blk-mq: make life...
278
  }
737f98cfe   Ming Lei   blk-mq: initializ...
279
  void blk_mq_sysfs_init(struct request_queue *q)
67aec14ce   Jens Axboe   blk-mq: make the ...
280
  {
67aec14ce   Jens Axboe   blk-mq: make the ...
281
  	struct blk_mq_ctx *ctx;
897bb0c7f   Thomas Gleixner   blk-mq: Use prope...
282
  	int cpu;
67aec14ce   Jens Axboe   blk-mq: make the ...
283

1db4909e7   Ming Lei   blk-mq: not embed...
284
  	kobject_init(q->mq_kobj, &blk_mq_ktype);
67aec14ce   Jens Axboe   blk-mq: make the ...
285

897bb0c7f   Thomas Gleixner   blk-mq: Use prope...
286
287
  	for_each_possible_cpu(cpu) {
  		ctx = per_cpu_ptr(q->queue_ctx, cpu);
1db4909e7   Ming Lei   blk-mq: not embed...
288
289
  
  		kobject_get(q->mq_kobj);
06a41a99d   Takashi Iwai   blk-mq: Fix unini...
290
  		kobject_init(&ctx->kobj, &blk_mq_ctx_ktype);
897bb0c7f   Thomas Gleixner   blk-mq: Use prope...
291
  	}
67aec14ce   Jens Axboe   blk-mq: make the ...
292
  }
2d0364c8c   Bart Van Assche   blk-mq: Register ...
293
  int __blk_mq_register_dev(struct device *dev, struct request_queue *q)
320ae51fe   Jens Axboe   blk-mq: new multi...
294
  {
320ae51fe   Jens Axboe   blk-mq: new multi...
295
  	struct blk_mq_hw_ctx *hctx;
67aec14ce   Jens Axboe   blk-mq: make the ...
296
  	int ret, i;
320ae51fe   Jens Axboe   blk-mq: new multi...
297

2d0364c8c   Bart Van Assche   blk-mq: Register ...
298
  	WARN_ON_ONCE(!q->kobj.parent);
cecf5d87f   Ming Lei   block: split .sys...
299
  	lockdep_assert_held(&q->sysfs_dir_lock);
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
300

1db4909e7   Ming Lei   blk-mq: not embed...
301
  	ret = kobject_add(q->mq_kobj, kobject_get(&dev->kobj), "%s", "mq");
320ae51fe   Jens Axboe   blk-mq: new multi...
302
  	if (ret < 0)
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
303
  		goto out;
320ae51fe   Jens Axboe   blk-mq: new multi...
304

1db4909e7   Ming Lei   blk-mq: not embed...
305
  	kobject_uevent(q->mq_kobj, KOBJ_ADD);
320ae51fe   Jens Axboe   blk-mq: new multi...
306
307
  
  	queue_for_each_hw_ctx(q, hctx, i) {
67aec14ce   Jens Axboe   blk-mq: make the ...
308
  		ret = blk_mq_register_hctx(hctx);
320ae51fe   Jens Axboe   blk-mq: new multi...
309
  		if (ret)
f05d1ba78   Bart Van Assche   blk-mq: Only unre...
310
  			goto unreg;
320ae51fe   Jens Axboe   blk-mq: new multi...
311
  	}
f05d1ba78   Bart Van Assche   blk-mq: Only unre...
312
  	q->mq_sysfs_init_done = true;
2d0364c8c   Bart Van Assche   blk-mq: Register ...
313

4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
314
  out:
2d0364c8c   Bart Van Assche   blk-mq: Register ...
315
  	return ret;
f05d1ba78   Bart Van Assche   blk-mq: Only unre...
316
317
318
319
  
  unreg:
  	while (--i >= 0)
  		blk_mq_unregister_hctx(q->queue_hw_ctx[i]);
1db4909e7   Ming Lei   blk-mq: not embed...
320
321
  	kobject_uevent(q->mq_kobj, KOBJ_REMOVE);
  	kobject_del(q->mq_kobj);
f05d1ba78   Bart Van Assche   blk-mq: Only unre...
322
323
  	kobject_put(&dev->kobj);
  	return ret;
2d0364c8c   Bart Van Assche   blk-mq: Register ...
324
  }
67aec14ce   Jens Axboe   blk-mq: make the ...
325
326
327
328
  void blk_mq_sysfs_unregister(struct request_queue *q)
  {
  	struct blk_mq_hw_ctx *hctx;
  	int i;
cecf5d87f   Ming Lei   block: split .sys...
329
  	mutex_lock(&q->sysfs_dir_lock);
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
330
  	if (!q->mq_sysfs_init_done)
2d0364c8c   Bart Van Assche   blk-mq: Register ...
331
  		goto unlock;
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
332

67aec14ce   Jens Axboe   blk-mq: make the ...
333
334
  	queue_for_each_hw_ctx(q, hctx, i)
  		blk_mq_unregister_hctx(hctx);
2d0364c8c   Bart Van Assche   blk-mq: Register ...
335
336
  
  unlock:
cecf5d87f   Ming Lei   block: split .sys...
337
  	mutex_unlock(&q->sysfs_dir_lock);
67aec14ce   Jens Axboe   blk-mq: make the ...
338
339
340
341
342
343
  }
  
  int blk_mq_sysfs_register(struct request_queue *q)
  {
  	struct blk_mq_hw_ctx *hctx;
  	int i, ret = 0;
cecf5d87f   Ming Lei   block: split .sys...
344
  	mutex_lock(&q->sysfs_dir_lock);
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
345
  	if (!q->mq_sysfs_init_done)
2d0364c8c   Bart Van Assche   blk-mq: Register ...
346
  		goto unlock;
4593fdbe7   Akinobu Mita   blk-mq: fix sysfs...
347

67aec14ce   Jens Axboe   blk-mq: make the ...
348
349
350
351
352
  	queue_for_each_hw_ctx(q, hctx, i) {
  		ret = blk_mq_register_hctx(hctx);
  		if (ret)
  			break;
  	}
2d0364c8c   Bart Van Assche   blk-mq: Register ...
353
  unlock:
cecf5d87f   Ming Lei   block: split .sys...
354
  	mutex_unlock(&q->sysfs_dir_lock);
2d0364c8c   Bart Van Assche   blk-mq: Register ...
355

67aec14ce   Jens Axboe   blk-mq: make the ...
356
357
  	return ret;
  }