Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 359500
b: refs/heads/master
c: 7918ffb
h: refs/heads/master
v: v3
  • Loading branch information
Tejun Heo committed Jan 9, 2013
1 parent fd1b1c2 commit 8736e56
Show file tree
Hide file tree
Showing 2 changed files with 77 additions and 1 deletion.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: e71357e118bdd4057e3bc020b9d80fecdd08f588
refs/heads/master: 7918ffb5b83e3373206ada84873c674fbddf61cc
76 changes: 76 additions & 0 deletions trunk/block/cfq-iosched.c
Original file line number Diff line number Diff line change
Expand Up @@ -224,6 +224,18 @@ struct cfq_group {
/* group service_tree key */
u64 vdisktime;

/*
* The number of active cfqgs and sum of their weights under this
* cfqg. This covers this cfqg's leaf_weight and all children's
* weights, but does not cover weights of further descendants.
*
* If a cfqg is on the service tree, it's active. An active cfqg
* also activates its parent and contributes to the children_weight
* of the parent.
*/
int nr_active;
unsigned int children_weight;

/*
* There are two weights - (internal) weight is the weight of this
* cfqg against the sibling cfqgs. leaf_weight is the wight of
Expand Down Expand Up @@ -583,6 +595,22 @@ static inline struct cfq_group *blkg_to_cfqg(struct blkcg_gq *blkg)
return pd_to_cfqg(blkg_to_pd(blkg, &blkcg_policy_cfq));
}

/*
* Determine the parent cfqg for weight calculation. Currently, cfqg
* scheduling is flat and the root is the parent of everyone else.
*/
static inline struct cfq_group *cfqg_flat_parent(struct cfq_group *cfqg)
{
struct blkcg_gq *blkg = cfqg_to_blkg(cfqg);
struct cfq_group *root;

while (blkg->parent)
blkg = blkg->parent;
root = blkg_to_cfqg(blkg);

return root != cfqg ? root : NULL;
}

static inline void cfqg_get(struct cfq_group *cfqg)
{
return blkg_get(cfqg_to_blkg(cfqg));
Expand Down Expand Up @@ -683,6 +711,7 @@ static void cfq_pd_reset_stats(struct blkcg_gq *blkg)

#else /* CONFIG_CFQ_GROUP_IOSCHED */

static inline struct cfq_group *cfqg_flat_parent(struct cfq_group *cfqg) { return NULL; }
static inline void cfqg_get(struct cfq_group *cfqg) { }
static inline void cfqg_put(struct cfq_group *cfqg) { }

Expand Down Expand Up @@ -1208,11 +1237,33 @@ cfq_update_group_weight(struct cfq_group *cfqg)
static void
cfq_group_service_tree_add(struct cfq_rb_root *st, struct cfq_group *cfqg)
{
struct cfq_group *pos = cfqg;
bool propagate;

/* add to the service tree */
BUG_ON(!RB_EMPTY_NODE(&cfqg->rb_node));

cfq_update_group_weight(cfqg);
__cfq_group_service_tree_add(st, cfqg);
st->total_weight += cfqg->weight;

/*
* Activate @cfqg and propagate activation upwards until we meet an
* already activated node or reach root.
*/
propagate = !pos->nr_active++;
pos->children_weight += pos->leaf_weight;

while (propagate) {
struct cfq_group *parent = cfqg_flat_parent(pos);

if (!parent)
break;

propagate = !parent->nr_active++;
parent->children_weight += pos->weight;
pos = parent;
}
}

static void
Expand Down Expand Up @@ -1243,6 +1294,31 @@ cfq_group_notify_queue_add(struct cfq_data *cfqd, struct cfq_group *cfqg)
static void
cfq_group_service_tree_del(struct cfq_rb_root *st, struct cfq_group *cfqg)
{
struct cfq_group *pos = cfqg;
bool propagate;

/*
* Undo activation from cfq_group_service_tree_add(). Deactivate
* @cfqg and propagate deactivation upwards.
*/
propagate = !--pos->nr_active;
pos->children_weight -= pos->leaf_weight;

while (propagate) {
struct cfq_group *parent = cfqg_flat_parent(pos);

/* @pos has 0 nr_active at this point */
WARN_ON_ONCE(pos->children_weight);

if (!parent)
break;

propagate = !--parent->nr_active;
parent->children_weight -= pos->weight;
pos = parent;
}

/* remove from the service tree */
st->total_weight -= cfqg->weight;
if (!RB_EMPTY_NODE(&cfqg->rb_node))
cfq_rb_erase(&cfqg->rb_node, st);
Expand Down

0 comments on commit 8736e56

Please sign in to comment.