diff options
-rw-r--r-- | block/blk-cgroup.c | 52 | ||||
-rw-r--r-- | block/blk-cgroup.h | 60 | ||||
-rw-r--r-- | block/blk-throttle.c | 37 | ||||
-rw-r--r-- | block/cfq-iosched.c | 46 |
4 files changed, 112 insertions, 83 deletions
diff --git a/block/blk-cgroup.c b/block/blk-cgroup.c index 997570329517..3d495528a765 100644 --- a/block/blk-cgroup.c +++ b/block/blk-cgroup.c | |||
@@ -58,11 +58,6 @@ static bool blkcg_policy_enabled(struct request_queue *q, | |||
58 | return pol && test_bit(pol->plid, q->blkcg_pols); | 58 | return pol && test_bit(pol->plid, q->blkcg_pols); |
59 | } | 59 | } |
60 | 60 | ||
61 | static size_t blkg_pd_size(const struct blkcg_policy *pol) | ||
62 | { | ||
63 | return sizeof(struct blkg_policy_data) + pol->pdata_size; | ||
64 | } | ||
65 | |||
66 | /** | 61 | /** |
67 | * blkg_free - free a blkg | 62 | * blkg_free - free a blkg |
68 | * @blkg: blkg to free | 63 | * @blkg: blkg to free |
@@ -122,7 +117,7 @@ static struct blkcg_gq *blkg_alloc(struct blkcg *blkcg, struct request_queue *q) | |||
122 | continue; | 117 | continue; |
123 | 118 | ||
124 | /* alloc per-policy data and attach it to blkg */ | 119 | /* alloc per-policy data and attach it to blkg */ |
125 | pd = kzalloc_node(blkg_pd_size(pol), GFP_ATOMIC, q->node); | 120 | pd = kzalloc_node(pol->pd_size, GFP_ATOMIC, q->node); |
126 | if (!pd) { | 121 | if (!pd) { |
127 | blkg_free(blkg); | 122 | blkg_free(blkg); |
128 | return NULL; | 123 | return NULL; |
@@ -346,7 +341,8 @@ static const char *blkg_dev_name(struct blkcg_gq *blkg) | |||
346 | * cftype->read_seq_string method. | 341 | * cftype->read_seq_string method. |
347 | */ | 342 | */ |
348 | void blkcg_print_blkgs(struct seq_file *sf, struct blkcg *blkcg, | 343 | void blkcg_print_blkgs(struct seq_file *sf, struct blkcg *blkcg, |
349 | u64 (*prfill)(struct seq_file *, void *, int), | 344 | u64 (*prfill)(struct seq_file *, |
345 | struct blkg_policy_data *, int), | ||
350 | const struct blkcg_policy *pol, int data, | 346 | const struct blkcg_policy *pol, int data, |
351 | bool show_total) | 347 | bool show_total) |
352 | { | 348 | { |
@@ -357,7 +353,7 @@ void blkcg_print_blkgs(struct seq_file *sf, struct blkcg *blkcg, | |||
357 | spin_lock_irq(&blkcg->lock); | 353 | spin_lock_irq(&blkcg->lock); |
358 | hlist_for_each_entry(blkg, n, &blkcg->blkg_list, blkcg_node) | 354 | hlist_for_each_entry(blkg, n, &blkcg->blkg_list, blkcg_node) |
359 | if (blkcg_policy_enabled(blkg->q, pol)) | 355 | if (blkcg_policy_enabled(blkg->q, pol)) |
360 | total += prfill(sf, blkg->pd[pol->plid]->pdata, data); | 356 | total += prfill(sf, blkg->pd[pol->plid], data); |
361 | spin_unlock_irq(&blkcg->lock); | 357 | spin_unlock_irq(&blkcg->lock); |
362 | 358 | ||
363 | if (show_total) | 359 | if (show_total) |
@@ -368,14 +364,14 @@ EXPORT_SYMBOL_GPL(blkcg_print_blkgs); | |||
368 | /** | 364 | /** |
369 | * __blkg_prfill_u64 - prfill helper for a single u64 value | 365 | * __blkg_prfill_u64 - prfill helper for a single u64 value |
370 | * @sf: seq_file to print to | 366 | * @sf: seq_file to print to |
371 | * @pdata: policy private data of interest | 367 | * @pd: policy private data of interest |
372 | * @v: value to print | 368 | * @v: value to print |
373 | * | 369 | * |
374 | * Print @v to @sf for the device assocaited with @pdata. | 370 | * Print @v to @sf for the device assocaited with @pd. |
375 | */ | 371 | */ |
376 | u64 __blkg_prfill_u64(struct seq_file *sf, void *pdata, u64 v) | 372 | u64 __blkg_prfill_u64(struct seq_file *sf, struct blkg_policy_data *pd, u64 v) |
377 | { | 373 | { |
378 | const char *dname = blkg_dev_name(pdata_to_blkg(pdata)); | 374 | const char *dname = blkg_dev_name(pd->blkg); |
379 | 375 | ||
380 | if (!dname) | 376 | if (!dname) |
381 | return 0; | 377 | return 0; |
@@ -388,12 +384,12 @@ EXPORT_SYMBOL_GPL(__blkg_prfill_u64); | |||
388 | /** | 384 | /** |
389 | * __blkg_prfill_rwstat - prfill helper for a blkg_rwstat | 385 | * __blkg_prfill_rwstat - prfill helper for a blkg_rwstat |
390 | * @sf: seq_file to print to | 386 | * @sf: seq_file to print to |
391 | * @pdata: policy private data of interest | 387 | * @pd: policy private data of interest |
392 | * @rwstat: rwstat to print | 388 | * @rwstat: rwstat to print |
393 | * | 389 | * |
394 | * Print @rwstat to @sf for the device assocaited with @pdata. | 390 | * Print @rwstat to @sf for the device assocaited with @pd. |
395 | */ | 391 | */ |
396 | u64 __blkg_prfill_rwstat(struct seq_file *sf, void *pdata, | 392 | u64 __blkg_prfill_rwstat(struct seq_file *sf, struct blkg_policy_data *pd, |
397 | const struct blkg_rwstat *rwstat) | 393 | const struct blkg_rwstat *rwstat) |
398 | { | 394 | { |
399 | static const char *rwstr[] = { | 395 | static const char *rwstr[] = { |
@@ -402,7 +398,7 @@ u64 __blkg_prfill_rwstat(struct seq_file *sf, void *pdata, | |||
402 | [BLKG_RWSTAT_SYNC] = "Sync", | 398 | [BLKG_RWSTAT_SYNC] = "Sync", |
403 | [BLKG_RWSTAT_ASYNC] = "Async", | 399 | [BLKG_RWSTAT_ASYNC] = "Async", |
404 | }; | 400 | }; |
405 | const char *dname = blkg_dev_name(pdata_to_blkg(pdata)); | 401 | const char *dname = blkg_dev_name(pd->blkg); |
406 | u64 v; | 402 | u64 v; |
407 | int i; | 403 | int i; |
408 | 404 | ||
@@ -421,30 +417,31 @@ u64 __blkg_prfill_rwstat(struct seq_file *sf, void *pdata, | |||
421 | /** | 417 | /** |
422 | * blkg_prfill_stat - prfill callback for blkg_stat | 418 | * blkg_prfill_stat - prfill callback for blkg_stat |
423 | * @sf: seq_file to print to | 419 | * @sf: seq_file to print to |
424 | * @pdata: policy private data of interest | 420 | * @pd: policy private data of interest |
425 | * @off: offset to the blkg_stat in @pdata | 421 | * @off: offset to the blkg_stat in @pd |
426 | * | 422 | * |
427 | * prfill callback for printing a blkg_stat. | 423 | * prfill callback for printing a blkg_stat. |
428 | */ | 424 | */ |
429 | u64 blkg_prfill_stat(struct seq_file *sf, void *pdata, int off) | 425 | u64 blkg_prfill_stat(struct seq_file *sf, struct blkg_policy_data *pd, int off) |
430 | { | 426 | { |
431 | return __blkg_prfill_u64(sf, pdata, blkg_stat_read(pdata + off)); | 427 | return __blkg_prfill_u64(sf, pd, blkg_stat_read((void *)pd + off)); |
432 | } | 428 | } |
433 | EXPORT_SYMBOL_GPL(blkg_prfill_stat); | 429 | EXPORT_SYMBOL_GPL(blkg_prfill_stat); |
434 | 430 | ||
435 | /** | 431 | /** |
436 | * blkg_prfill_rwstat - prfill callback for blkg_rwstat | 432 | * blkg_prfill_rwstat - prfill callback for blkg_rwstat |
437 | * @sf: seq_file to print to | 433 | * @sf: seq_file to print to |
438 | * @pdata: policy private data of interest | 434 | * @pd: policy private data of interest |
439 | * @off: offset to the blkg_rwstat in @pdata | 435 | * @off: offset to the blkg_rwstat in @pd |
440 | * | 436 | * |
441 | * prfill callback for printing a blkg_rwstat. | 437 | * prfill callback for printing a blkg_rwstat. |
442 | */ | 438 | */ |
443 | u64 blkg_prfill_rwstat(struct seq_file *sf, void *pdata, int off) | 439 | u64 blkg_prfill_rwstat(struct seq_file *sf, struct blkg_policy_data *pd, |
440 | int off) | ||
444 | { | 441 | { |
445 | struct blkg_rwstat rwstat = blkg_rwstat_read(pdata + off); | 442 | struct blkg_rwstat rwstat = blkg_rwstat_read((void *)pd + off); |
446 | 443 | ||
447 | return __blkg_prfill_rwstat(sf, pdata, &rwstat); | 444 | return __blkg_prfill_rwstat(sf, pd, &rwstat); |
448 | } | 445 | } |
449 | EXPORT_SYMBOL_GPL(blkg_prfill_rwstat); | 446 | EXPORT_SYMBOL_GPL(blkg_prfill_rwstat); |
450 | 447 | ||
@@ -733,7 +730,7 @@ int blkcg_activate_policy(struct request_queue *q, | |||
733 | 730 | ||
734 | /* allocate policy_data for all existing blkgs */ | 731 | /* allocate policy_data for all existing blkgs */ |
735 | while (cnt--) { | 732 | while (cnt--) { |
736 | pd = kzalloc_node(blkg_pd_size(pol), GFP_KERNEL, q->node); | 733 | pd = kzalloc_node(pol->pd_size, GFP_KERNEL, q->node); |
737 | if (!pd) { | 734 | if (!pd) { |
738 | ret = -ENOMEM; | 735 | ret = -ENOMEM; |
739 | goto out_free; | 736 | goto out_free; |
@@ -832,6 +829,9 @@ int blkcg_policy_register(struct blkcg_policy *pol) | |||
832 | { | 829 | { |
833 | int i, ret; | 830 | int i, ret; |
834 | 831 | ||
832 | if (WARN_ON(pol->pd_size < sizeof(struct blkg_policy_data))) | ||
833 | return -EINVAL; | ||
834 | |||
835 | mutex_lock(&blkcg_pol_mutex); | 835 | mutex_lock(&blkcg_pol_mutex); |
836 | 836 | ||
837 | /* find an empty slot */ | 837 | /* find an empty slot */ |
diff --git a/block/blk-cgroup.h b/block/blk-cgroup.h index a443b84d2c16..18b021e1c05f 100644 --- a/block/blk-cgroup.h +++ b/block/blk-cgroup.h | |||
@@ -59,16 +59,25 @@ struct blkg_rwstat { | |||
59 | uint64_t cnt[BLKG_RWSTAT_NR]; | 59 | uint64_t cnt[BLKG_RWSTAT_NR]; |
60 | }; | 60 | }; |
61 | 61 | ||
62 | /* per-blkg per-policy data */ | 62 | /* |
63 | * A blkcg_gq (blkg) is association between a block cgroup (blkcg) and a | ||
64 | * request_queue (q). This is used by blkcg policies which need to track | ||
65 | * information per blkcg - q pair. | ||
66 | * | ||
67 | * There can be multiple active blkcg policies and each has its private | ||
68 | * data on each blkg, the size of which is determined by | ||
69 | * blkcg_policy->pd_size. blkcg core allocates and frees such areas | ||
70 | * together with blkg and invokes pd_init/exit_fn() methods. | ||
71 | * | ||
72 | * Such private data must embed struct blkg_policy_data (pd) at the | ||
73 | * beginning and pd_size can't be smaller than pd. | ||
74 | */ | ||
63 | struct blkg_policy_data { | 75 | struct blkg_policy_data { |
64 | /* the blkg this per-policy data belongs to */ | 76 | /* the blkg this per-policy data belongs to */ |
65 | struct blkcg_gq *blkg; | 77 | struct blkcg_gq *blkg; |
66 | 78 | ||
67 | /* used during policy activation */ | 79 | /* used during policy activation */ |
68 | struct list_head alloc_node; | 80 | struct list_head alloc_node; |
69 | |||
70 | /* pol->pdata_size bytes of private data used by policy impl */ | ||
71 | char pdata[] __aligned(__alignof__(unsigned long long)); | ||
72 | }; | 81 | }; |
73 | 82 | ||
74 | /* association between a blk cgroup and a request queue */ | 83 | /* association between a blk cgroup and a request queue */ |
@@ -100,7 +109,7 @@ struct blkcg_policy { | |||
100 | struct blkcg_policy_ops ops; | 109 | struct blkcg_policy_ops ops; |
101 | int plid; | 110 | int plid; |
102 | /* policy specific private data size */ | 111 | /* policy specific private data size */ |
103 | size_t pdata_size; | 112 | size_t pd_size; |
104 | /* cgroup files for the policy */ | 113 | /* cgroup files for the policy */ |
105 | struct cftype *cftypes; | 114 | struct cftype *cftypes; |
106 | }; | 115 | }; |
@@ -125,14 +134,16 @@ void blkcg_deactivate_policy(struct request_queue *q, | |||
125 | const struct blkcg_policy *pol); | 134 | const struct blkcg_policy *pol); |
126 | 135 | ||
127 | void blkcg_print_blkgs(struct seq_file *sf, struct blkcg *blkcg, | 136 | void blkcg_print_blkgs(struct seq_file *sf, struct blkcg *blkcg, |
128 | u64 (*prfill)(struct seq_file *, void *, int), | 137 | u64 (*prfill)(struct seq_file *, |
138 | struct blkg_policy_data *, int), | ||
129 | const struct blkcg_policy *pol, int data, | 139 | const struct blkcg_policy *pol, int data, |
130 | bool show_total); | 140 | bool show_total); |
131 | u64 __blkg_prfill_u64(struct seq_file *sf, void *pdata, u64 v); | 141 | u64 __blkg_prfill_u64(struct seq_file *sf, struct blkg_policy_data *pd, u64 v); |
132 | u64 __blkg_prfill_rwstat(struct seq_file *sf, void *pdata, | 142 | u64 __blkg_prfill_rwstat(struct seq_file *sf, struct blkg_policy_data *pd, |
133 | const struct blkg_rwstat *rwstat); | 143 | const struct blkg_rwstat *rwstat); |
134 | u64 blkg_prfill_stat(struct seq_file *sf, void *pdata, int off); | 144 | u64 blkg_prfill_stat(struct seq_file *sf, struct blkg_policy_data *pd, int off); |
135 | u64 blkg_prfill_rwstat(struct seq_file *sf, void *pdata, int off); | 145 | u64 blkg_prfill_rwstat(struct seq_file *sf, struct blkg_policy_data *pd, |
146 | int off); | ||
136 | 147 | ||
137 | struct blkg_conf_ctx { | 148 | struct blkg_conf_ctx { |
138 | struct gendisk *disk; | 149 | struct gendisk *disk; |
@@ -152,26 +163,21 @@ void blkg_conf_finish(struct blkg_conf_ctx *ctx); | |||
152 | * | 163 | * |
153 | * Return pointer to private data associated with the @blkg-@pol pair. | 164 | * Return pointer to private data associated with the @blkg-@pol pair. |
154 | */ | 165 | */ |
155 | static inline void *blkg_to_pdata(struct blkcg_gq *blkg, | 166 | static inline struct blkg_policy_data *blkg_to_pd(struct blkcg_gq *blkg, |
156 | struct blkcg_policy *pol) | 167 | struct blkcg_policy *pol) |
157 | { | 168 | { |
158 | return blkg ? blkg->pd[pol->plid]->pdata : NULL; | 169 | return blkg ? blkg->pd[pol->plid] : NULL; |
159 | } | 170 | } |
160 | 171 | ||
161 | /** | 172 | /** |
162 | * pdata_to_blkg - get blkg associated with policy private data | 173 | * pdata_to_blkg - get blkg associated with policy private data |
163 | * @pdata: policy private data of interest | 174 | * @pd: policy private data of interest |
164 | * | 175 | * |
165 | * @pdata is policy private data. Determine the blkg it's associated with. | 176 | * @pd is policy private data. Determine the blkg it's associated with. |
166 | */ | 177 | */ |
167 | static inline struct blkcg_gq *pdata_to_blkg(void *pdata) | 178 | static inline struct blkcg_gq *pd_to_blkg(struct blkg_policy_data *pd) |
168 | { | 179 | { |
169 | if (pdata) { | 180 | return pd ? pd->blkg : NULL; |
170 | struct blkg_policy_data *pd = | ||
171 | container_of(pdata, struct blkg_policy_data, pdata); | ||
172 | return pd->blkg; | ||
173 | } | ||
174 | return NULL; | ||
175 | } | 181 | } |
176 | 182 | ||
177 | /** | 183 | /** |
@@ -342,6 +348,9 @@ static inline void blkg_rwstat_reset(struct blkg_rwstat *rwstat) | |||
342 | 348 | ||
343 | struct cgroup; | 349 | struct cgroup; |
344 | 350 | ||
351 | struct blkg_policy_data { | ||
352 | }; | ||
353 | |||
345 | struct blkcg_gq { | 354 | struct blkcg_gq { |
346 | }; | 355 | }; |
347 | 356 | ||
@@ -361,10 +370,9 @@ static inline int blkcg_activate_policy(struct request_queue *q, | |||
361 | static inline void blkcg_deactivate_policy(struct request_queue *q, | 370 | static inline void blkcg_deactivate_policy(struct request_queue *q, |
362 | const struct blkcg_policy *pol) { } | 371 | const struct blkcg_policy *pol) { } |
363 | 372 | ||
364 | static inline void *blkg_to_pdata(struct blkcg_gq *blkg, | 373 | static inline struct blkg_policy_data *blkg_to_pd(struct blkcg_gq *blkg, |
365 | struct blkcg_policy *pol) { return NULL; } | 374 | struct blkcg_policy *pol) { return NULL; } |
366 | static inline struct blkcg_gq *pdata_to_blkg(void *pdata, | 375 | static inline struct blkcg_gq *pd_to_blkg(struct blkg_policy_data *pd) { return NULL; } |
367 | struct blkcg_policy *pol) { return NULL; } | ||
368 | static inline char *blkg_path(struct blkcg_gq *blkg) { return NULL; } | 376 | static inline char *blkg_path(struct blkcg_gq *blkg) { return NULL; } |
369 | static inline void blkg_get(struct blkcg_gq *blkg) { } | 377 | static inline void blkg_get(struct blkcg_gq *blkg) { } |
370 | static inline void blkg_put(struct blkcg_gq *blkg) { } | 378 | static inline void blkg_put(struct blkcg_gq *blkg) { } |
diff --git a/block/blk-throttle.c b/block/blk-throttle.c index 00c7eff66ecf..6a0a17a83862 100644 --- a/block/blk-throttle.c +++ b/block/blk-throttle.c | |||
@@ -49,6 +49,9 @@ struct tg_stats_cpu { | |||
49 | }; | 49 | }; |
50 | 50 | ||
51 | struct throtl_grp { | 51 | struct throtl_grp { |
52 | /* must be the first member */ | ||
53 | struct blkg_policy_data pd; | ||
54 | |||
52 | /* active throtl group service_tree member */ | 55 | /* active throtl group service_tree member */ |
53 | struct rb_node rb_node; | 56 | struct rb_node rb_node; |
54 | 57 | ||
@@ -120,14 +123,19 @@ static LIST_HEAD(tg_stats_alloc_list); | |||
120 | static void tg_stats_alloc_fn(struct work_struct *); | 123 | static void tg_stats_alloc_fn(struct work_struct *); |
121 | static DECLARE_DELAYED_WORK(tg_stats_alloc_work, tg_stats_alloc_fn); | 124 | static DECLARE_DELAYED_WORK(tg_stats_alloc_work, tg_stats_alloc_fn); |
122 | 125 | ||
126 | static inline struct throtl_grp *pd_to_tg(struct blkg_policy_data *pd) | ||
127 | { | ||
128 | return pd ? container_of(pd, struct throtl_grp, pd) : NULL; | ||
129 | } | ||
130 | |||
123 | static inline struct throtl_grp *blkg_to_tg(struct blkcg_gq *blkg) | 131 | static inline struct throtl_grp *blkg_to_tg(struct blkcg_gq *blkg) |
124 | { | 132 | { |
125 | return blkg_to_pdata(blkg, &blkcg_policy_throtl); | 133 | return pd_to_tg(blkg_to_pd(blkg, &blkcg_policy_throtl)); |
126 | } | 134 | } |
127 | 135 | ||
128 | static inline struct blkcg_gq *tg_to_blkg(struct throtl_grp *tg) | 136 | static inline struct blkcg_gq *tg_to_blkg(struct throtl_grp *tg) |
129 | { | 137 | { |
130 | return pdata_to_blkg(tg); | 138 | return pd_to_blkg(&tg->pd); |
131 | } | 139 | } |
132 | 140 | ||
133 | static inline struct throtl_grp *td_root_tg(struct throtl_data *td) | 141 | static inline struct throtl_grp *td_root_tg(struct throtl_data *td) |
@@ -931,9 +939,10 @@ throtl_schedule_delayed_work(struct throtl_data *td, unsigned long delay) | |||
931 | } | 939 | } |
932 | } | 940 | } |
933 | 941 | ||
934 | static u64 tg_prfill_cpu_rwstat(struct seq_file *sf, void *pdata, int off) | 942 | static u64 tg_prfill_cpu_rwstat(struct seq_file *sf, |
943 | struct blkg_policy_data *pd, int off) | ||
935 | { | 944 | { |
936 | struct throtl_grp *tg = pdata; | 945 | struct throtl_grp *tg = pd_to_tg(pd); |
937 | struct blkg_rwstat rwstat = { }, tmp; | 946 | struct blkg_rwstat rwstat = { }, tmp; |
938 | int i, cpu; | 947 | int i, cpu; |
939 | 948 | ||
@@ -945,7 +954,7 @@ static u64 tg_prfill_cpu_rwstat(struct seq_file *sf, void *pdata, int off) | |||
945 | rwstat.cnt[i] += tmp.cnt[i]; | 954 | rwstat.cnt[i] += tmp.cnt[i]; |
946 | } | 955 | } |
947 | 956 | ||
948 | return __blkg_prfill_rwstat(sf, pdata, &rwstat); | 957 | return __blkg_prfill_rwstat(sf, pd, &rwstat); |
949 | } | 958 | } |
950 | 959 | ||
951 | static int tg_print_cpu_rwstat(struct cgroup *cgrp, struct cftype *cft, | 960 | static int tg_print_cpu_rwstat(struct cgroup *cgrp, struct cftype *cft, |
@@ -958,22 +967,26 @@ static int tg_print_cpu_rwstat(struct cgroup *cgrp, struct cftype *cft, | |||
958 | return 0; | 967 | return 0; |
959 | } | 968 | } |
960 | 969 | ||
961 | static u64 tg_prfill_conf_u64(struct seq_file *sf, void *pdata, int off) | 970 | static u64 tg_prfill_conf_u64(struct seq_file *sf, struct blkg_policy_data *pd, |
971 | int off) | ||
962 | { | 972 | { |
963 | u64 v = *(u64 *)(pdata + off); | 973 | struct throtl_grp *tg = pd_to_tg(pd); |
974 | u64 v = *(u64 *)((void *)tg + off); | ||
964 | 975 | ||
965 | if (v == -1) | 976 | if (v == -1) |
966 | return 0; | 977 | return 0; |
967 | return __blkg_prfill_u64(sf, pdata, v); | 978 | return __blkg_prfill_u64(sf, pd, v); |
968 | } | 979 | } |
969 | 980 | ||
970 | static u64 tg_prfill_conf_uint(struct seq_file *sf, void *pdata, int off) | 981 | static u64 tg_prfill_conf_uint(struct seq_file *sf, struct blkg_policy_data *pd, |
982 | int off) | ||
971 | { | 983 | { |
972 | unsigned int v = *(unsigned int *)(pdata + off); | 984 | struct throtl_grp *tg = pd_to_tg(pd); |
985 | unsigned int v = *(unsigned int *)((void *)tg + off); | ||
973 | 986 | ||
974 | if (v == -1) | 987 | if (v == -1) |
975 | return 0; | 988 | return 0; |
976 | return __blkg_prfill_u64(sf, pdata, v); | 989 | return __blkg_prfill_u64(sf, pd, v); |
977 | } | 990 | } |
978 | 991 | ||
979 | static int tg_print_conf_u64(struct cgroup *cgrp, struct cftype *cft, | 992 | static int tg_print_conf_u64(struct cgroup *cgrp, struct cftype *cft, |
@@ -1092,7 +1105,7 @@ static struct blkcg_policy blkcg_policy_throtl = { | |||
1092 | .pd_exit_fn = throtl_pd_exit, | 1105 | .pd_exit_fn = throtl_pd_exit, |
1093 | .pd_reset_stats_fn = throtl_pd_reset_stats, | 1106 | .pd_reset_stats_fn = throtl_pd_reset_stats, |
1094 | }, | 1107 | }, |
1095 | .pdata_size = sizeof(struct throtl_grp), | 1108 | .pd_size = sizeof(struct throtl_grp), |
1096 | .cftypes = throtl_files, | 1109 | .cftypes = throtl_files, |
1097 | }; | 1110 | }; |
1098 | 1111 | ||
diff --git a/block/cfq-iosched.c b/block/cfq-iosched.c index 792218281d91..7865cc38ea77 100644 --- a/block/cfq-iosched.c +++ b/block/cfq-iosched.c | |||
@@ -217,6 +217,9 @@ struct cfqg_stats { | |||
217 | 217 | ||
218 | /* This is per cgroup per device grouping structure */ | 218 | /* This is per cgroup per device grouping structure */ |
219 | struct cfq_group { | 219 | struct cfq_group { |
220 | /* must be the first member */ | ||
221 | struct blkg_policy_data pd; | ||
222 | |||
220 | /* group service_tree member */ | 223 | /* group service_tree member */ |
221 | struct rb_node rb_node; | 224 | struct rb_node rb_node; |
222 | 225 | ||
@@ -409,6 +412,21 @@ CFQ_CFQQ_FNS(deep); | |||
409 | CFQ_CFQQ_FNS(wait_busy); | 412 | CFQ_CFQQ_FNS(wait_busy); |
410 | #undef CFQ_CFQQ_FNS | 413 | #undef CFQ_CFQQ_FNS |
411 | 414 | ||
415 | static inline struct cfq_group *pd_to_cfqg(struct blkg_policy_data *pd) | ||
416 | { | ||
417 | return pd ? container_of(pd, struct cfq_group, pd) : NULL; | ||
418 | } | ||
419 | |||
420 | static inline struct cfq_group *blkg_to_cfqg(struct blkcg_gq *blkg) | ||
421 | { | ||
422 | return pd_to_cfqg(blkg_to_pd(blkg, &blkcg_policy_cfq)); | ||
423 | } | ||
424 | |||
425 | static inline struct blkcg_gq *cfqg_to_blkg(struct cfq_group *cfqg) | ||
426 | { | ||
427 | return pd_to_blkg(&cfqg->pd); | ||
428 | } | ||
429 | |||
412 | #if defined(CONFIG_CFQ_GROUP_IOSCHED) && defined(CONFIG_DEBUG_BLK_CGROUP) | 430 | #if defined(CONFIG_CFQ_GROUP_IOSCHED) && defined(CONFIG_DEBUG_BLK_CGROUP) |
413 | 431 | ||
414 | /* cfqg stats flags */ | 432 | /* cfqg stats flags */ |
@@ -553,16 +571,6 @@ static inline void cfqg_stats_update_avg_queue_size(struct cfq_group *cfqg) { } | |||
553 | 571 | ||
554 | #ifdef CONFIG_CFQ_GROUP_IOSCHED | 572 | #ifdef CONFIG_CFQ_GROUP_IOSCHED |
555 | 573 | ||
556 | static inline struct cfq_group *blkg_to_cfqg(struct blkcg_gq *blkg) | ||
557 | { | ||
558 | return blkg_to_pdata(blkg, &blkcg_policy_cfq); | ||
559 | } | ||
560 | |||
561 | static inline struct blkcg_gq *cfqg_to_blkg(struct cfq_group *cfqg) | ||
562 | { | ||
563 | return pdata_to_blkg(cfqg); | ||
564 | } | ||
565 | |||
566 | static inline void cfqg_get(struct cfq_group *cfqg) | 574 | static inline void cfqg_get(struct cfq_group *cfqg) |
567 | { | 575 | { |
568 | return blkg_get(cfqg_to_blkg(cfqg)); | 576 | return blkg_get(cfqg_to_blkg(cfqg)); |
@@ -662,8 +670,6 @@ static void cfq_pd_reset_stats(struct blkcg_gq *blkg) | |||
662 | 670 | ||
663 | #else /* CONFIG_CFQ_GROUP_IOSCHED */ | 671 | #else /* CONFIG_CFQ_GROUP_IOSCHED */ |
664 | 672 | ||
665 | static inline struct cfq_group *blkg_to_cfqg(struct blkcg_gq *blkg) { return NULL; } | ||
666 | static inline struct blkcg_gq *cfqg_to_blkg(struct cfq_group *cfqg) { return NULL; } | ||
667 | static inline void cfqg_get(struct cfq_group *cfqg) { } | 673 | static inline void cfqg_get(struct cfq_group *cfqg) { } |
668 | static inline void cfqg_put(struct cfq_group *cfqg) { } | 674 | static inline void cfqg_put(struct cfq_group *cfqg) { } |
669 | 675 | ||
@@ -1374,13 +1380,14 @@ static void cfq_link_cfqq_cfqg(struct cfq_queue *cfqq, struct cfq_group *cfqg) | |||
1374 | cfqg_get(cfqg); | 1380 | cfqg_get(cfqg); |
1375 | } | 1381 | } |
1376 | 1382 | ||
1377 | static u64 cfqg_prfill_weight_device(struct seq_file *sf, void *pdata, int off) | 1383 | static u64 cfqg_prfill_weight_device(struct seq_file *sf, |
1384 | struct blkg_policy_data *pd, int off) | ||
1378 | { | 1385 | { |
1379 | struct cfq_group *cfqg = pdata; | 1386 | struct cfq_group *cfqg = pd_to_cfqg(pd); |
1380 | 1387 | ||
1381 | if (!cfqg->dev_weight) | 1388 | if (!cfqg->dev_weight) |
1382 | return 0; | 1389 | return 0; |
1383 | return __blkg_prfill_u64(sf, pdata, cfqg->dev_weight); | 1390 | return __blkg_prfill_u64(sf, pd, cfqg->dev_weight); |
1384 | } | 1391 | } |
1385 | 1392 | ||
1386 | static int cfqg_print_weight_device(struct cgroup *cgrp, struct cftype *cft, | 1393 | static int cfqg_print_weight_device(struct cgroup *cgrp, struct cftype *cft, |
@@ -1467,9 +1474,10 @@ static int cfqg_print_rwstat(struct cgroup *cgrp, struct cftype *cft, | |||
1467 | } | 1474 | } |
1468 | 1475 | ||
1469 | #ifdef CONFIG_DEBUG_BLK_CGROUP | 1476 | #ifdef CONFIG_DEBUG_BLK_CGROUP |
1470 | static u64 cfqg_prfill_avg_queue_size(struct seq_file *sf, void *pdata, int off) | 1477 | static u64 cfqg_prfill_avg_queue_size(struct seq_file *sf, |
1478 | struct blkg_policy_data *pd, int off) | ||
1471 | { | 1479 | { |
1472 | struct cfq_group *cfqg = pdata; | 1480 | struct cfq_group *cfqg = pd_to_cfqg(pd); |
1473 | u64 samples = blkg_stat_read(&cfqg->stats.avg_queue_size_samples); | 1481 | u64 samples = blkg_stat_read(&cfqg->stats.avg_queue_size_samples); |
1474 | u64 v = 0; | 1482 | u64 v = 0; |
1475 | 1483 | ||
@@ -1477,7 +1485,7 @@ static u64 cfqg_prfill_avg_queue_size(struct seq_file *sf, void *pdata, int off) | |||
1477 | v = blkg_stat_read(&cfqg->stats.avg_queue_size_sum); | 1485 | v = blkg_stat_read(&cfqg->stats.avg_queue_size_sum); |
1478 | do_div(v, samples); | 1486 | do_div(v, samples); |
1479 | } | 1487 | } |
1480 | __blkg_prfill_u64(sf, pdata, v); | 1488 | __blkg_prfill_u64(sf, pd, v); |
1481 | return 0; | 1489 | return 0; |
1482 | } | 1490 | } |
1483 | 1491 | ||
@@ -4161,7 +4169,7 @@ static struct blkcg_policy blkcg_policy_cfq = { | |||
4161 | .pd_init_fn = cfq_pd_init, | 4169 | .pd_init_fn = cfq_pd_init, |
4162 | .pd_reset_stats_fn = cfq_pd_reset_stats, | 4170 | .pd_reset_stats_fn = cfq_pd_reset_stats, |
4163 | }, | 4171 | }, |
4164 | .pdata_size = sizeof(struct cfq_group), | 4172 | .pd_size = sizeof(struct cfq_group), |
4165 | .cftypes = cfq_blkcg_files, | 4173 | .cftypes = cfq_blkcg_files, |
4166 | }; | 4174 | }; |
4167 | #endif | 4175 | #endif |