diff options
Diffstat (limited to 'mm/damon/sysfs-schemes.c')
| -rw-r--r-- | mm/damon/sysfs-schemes.c | 504 | 
1 files changed, 443 insertions, 61 deletions
| diff --git a/mm/damon/sysfs-schemes.c b/mm/damon/sysfs-schemes.c index 30ae7518ffbf..74056bcd6a2c 100644 --- a/mm/damon/sysfs-schemes.c +++ b/mm/damon/sysfs-schemes.c @@ -341,16 +341,45 @@ static struct damon_sysfs_scheme_filter *damon_sysfs_scheme_filter_alloc(  	return filter;  } -/* Should match with enum damos_filter_type */ -static const char * const damon_sysfs_scheme_filter_type_strs[] = { -	"anon", -	"active", -	"memcg", -	"young", -	"hugepage_size", -	"unmapped", -	"addr", -	"target", +struct damos_sysfs_filter_type_name { +	enum damos_filter_type type; +	char *name; +}; + +static const struct damos_sysfs_filter_type_name +damos_sysfs_filter_type_names[] = { +	{ +		.type = DAMOS_FILTER_TYPE_ANON, +		.name = "anon", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_ACTIVE, +		.name = "active", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_MEMCG, +		.name = "memcg", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_YOUNG, +		.name = "young", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_HUGEPAGE_SIZE, +		.name = "hugepage_size", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_UNMAPPED, +		.name = "unmapped", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_ADDR, +		.name = "addr", +	}, +	{ +		.type = DAMOS_FILTER_TYPE_TARGET, +		.name = "target", +	},  };  static ssize_t type_show(struct kobject *kobj, @@ -358,9 +387,16 @@ static ssize_t type_show(struct kobject *kobj,  {  	struct damon_sysfs_scheme_filter *filter = container_of(kobj,  			struct damon_sysfs_scheme_filter, kobj); +	int i; -	return sysfs_emit(buf, "%s\n", -			damon_sysfs_scheme_filter_type_strs[filter->type]); +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_filter_type_names); i++) { +		const struct damos_sysfs_filter_type_name *type_name; + +		type_name = &damos_sysfs_filter_type_names[i]; +		if (type_name->type == filter->type) +			return sysfs_emit(buf, "%s\n", type_name->name); +	} +	return -EINVAL;  }  static bool damos_sysfs_scheme_filter_valid_type( @@ -385,16 +421,19 @@ static ssize_t type_store(struct kobject *kobj,  {  	struct damon_sysfs_scheme_filter *filter = container_of(kobj,  			struct damon_sysfs_scheme_filter, kobj); -	enum damos_filter_type type;  	ssize_t ret = -EINVAL; +	int i; + +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_filter_type_names); i++) { +		const struct damos_sysfs_filter_type_name *type_name; -	for (type = 0; type < NR_DAMOS_FILTER_TYPES; type++) { -		if (sysfs_streq(buf, damon_sysfs_scheme_filter_type_strs[ -					type])) { +		type_name = &damos_sysfs_filter_type_names[i]; +		if (sysfs_streq(buf, type_name->name)) {  			if (!damos_sysfs_scheme_filter_valid_type( -						filter->handle_layer, type)) +						filter->handle_layer, +						type_name->type))  				break; -			filter->type = type; +			filter->type = type_name->type;  			ret = count;  			break;  		} @@ -785,10 +824,21 @@ static struct damon_sysfs_watermarks *damon_sysfs_watermarks_alloc(  	return watermarks;  } -/* Should match with enum damos_wmark_metric */ -static const char * const damon_sysfs_wmark_metric_strs[] = { -	"none", -	"free_mem_rate", +struct damos_sysfs_wmark_metric_name { +	enum damos_wmark_metric metric; +	char *name; +}; + +static const struct damos_sysfs_wmark_metric_name +damos_sysfs_wmark_metric_names[] = { +	{ +		.metric = DAMOS_WMARK_NONE, +		.name = "none", +	}, +	{ +		.metric = DAMOS_WMARK_FREE_MEM_RATE, +		.name = "free_mem_rate", +	},  };  static ssize_t metric_show(struct kobject *kobj, struct kobj_attribute *attr, @@ -796,9 +846,16 @@ static ssize_t metric_show(struct kobject *kobj, struct kobj_attribute *attr,  {  	struct damon_sysfs_watermarks *watermarks = container_of(kobj,  			struct damon_sysfs_watermarks, kobj); +	int i; -	return sysfs_emit(buf, "%s\n", -			damon_sysfs_wmark_metric_strs[watermarks->metric]); +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_wmark_metric_names); i++) { +		const struct damos_sysfs_wmark_metric_name *metric_name; + +		metric_name = &damos_sysfs_wmark_metric_names[i]; +		if (metric_name->metric == watermarks->metric) +			return sysfs_emit(buf, "%s\n", metric_name->name); +	} +	return -EINVAL;  }  static ssize_t metric_store(struct kobject *kobj, struct kobj_attribute *attr, @@ -806,11 +863,14 @@ static ssize_t metric_store(struct kobject *kobj, struct kobj_attribute *attr,  {  	struct damon_sysfs_watermarks *watermarks = container_of(kobj,  			struct damon_sysfs_watermarks, kobj); -	enum damos_wmark_metric metric; +	int i; -	for (metric = 0; metric < NR_DAMOS_WMARK_METRICS; metric++) { -		if (sysfs_streq(buf, damon_sysfs_wmark_metric_strs[metric])) { -			watermarks->metric = metric; +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_wmark_metric_names); i++) { +		const struct damos_sysfs_wmark_metric_name *metric_name; + +		metric_name = &damos_sysfs_wmark_metric_names[i]; +		if (sysfs_streq(buf, metric_name->name)) { +			watermarks->metric = metric_name->metric;  			return count;  		}  	} @@ -941,27 +1001,51 @@ struct damos_sysfs_quota_goal {  	int nid;  }; -/* This should match with enum damos_quota_goal_metric */ -static const char * const damos_sysfs_quota_goal_metric_strs[] = { -	"user_input", -	"some_mem_psi_us", -	"node_mem_used_bp", -	"node_mem_free_bp", -}; -  static struct damos_sysfs_quota_goal *damos_sysfs_quota_goal_alloc(void)  {  	return kzalloc(sizeof(struct damos_sysfs_quota_goal), GFP_KERNEL);  } +struct damos_sysfs_qgoal_metric_name { +	enum damos_quota_goal_metric metric; +	char *name; +}; + +static +struct damos_sysfs_qgoal_metric_name damos_sysfs_qgoal_metric_names[] = { +	{ +		.metric = DAMOS_QUOTA_USER_INPUT, +		.name = "user_input", +	}, +	{ +		.metric = DAMOS_QUOTA_SOME_MEM_PSI_US, +		.name = "some_mem_psi_us", +	}, +	{ +		.metric = DAMOS_QUOTA_NODE_MEM_USED_BP, +		.name = "node_mem_used_bp", +	}, +	{ +		.metric = DAMOS_QUOTA_NODE_MEM_FREE_BP, +		.name = "node_mem_free_bp", +	}, +}; +  static ssize_t target_metric_show(struct kobject *kobj,  		struct kobj_attribute *attr, char *buf)  {  	struct damos_sysfs_quota_goal *goal = container_of(kobj,  			struct damos_sysfs_quota_goal, kobj); +	int i; -	return sysfs_emit(buf, "%s\n", -			damos_sysfs_quota_goal_metric_strs[goal->metric]); +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_qgoal_metric_names); i++) { +		struct damos_sysfs_qgoal_metric_name *metric_name; + +		metric_name = &damos_sysfs_qgoal_metric_names[i]; +		if (metric_name->metric == goal->metric) +			return sysfs_emit(buf, "%s\n", metric_name->name); +	} +	return -EINVAL;  }  static ssize_t target_metric_store(struct kobject *kobj, @@ -969,11 +1053,14 @@ static ssize_t target_metric_store(struct kobject *kobj,  {  	struct damos_sysfs_quota_goal *goal = container_of(kobj,  			struct damos_sysfs_quota_goal, kobj); -	enum damos_quota_goal_metric m; +	int i; + +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_qgoal_metric_names); i++) { +		struct damos_sysfs_qgoal_metric_name *metric_name; -	for (m = 0; m < NR_DAMOS_QUOTA_GOAL_METRICS; m++) { -		if (sysfs_streq(buf, damos_sysfs_quota_goal_metric_strs[m])) { -			goal->metric = m; +		metric_name = &damos_sysfs_qgoal_metric_names[i]; +		if (sysfs_streq(buf, metric_name->name)) { +			goal->metric = metric_name->metric;  			return count;  		}  	} @@ -1569,6 +1656,204 @@ static const struct kobj_type damon_sysfs_access_pattern_ktype = {  };  /* + * dest (action destination) directory + */ + +struct damos_sysfs_dest { +	struct kobject kobj; +	unsigned int id; +	unsigned int weight; +}; + +static struct damos_sysfs_dest *damos_sysfs_dest_alloc(void) +{ +	return kzalloc(sizeof(struct damos_sysfs_dest), GFP_KERNEL); +} + +static ssize_t id_show( +		struct kobject *kobj, struct kobj_attribute *attr, char *buf) +{ +	struct damos_sysfs_dest *dest = container_of(kobj, +			struct damos_sysfs_dest, kobj); + +	return sysfs_emit(buf, "%u\n", dest->id); +} + +static ssize_t id_store(struct kobject *kobj, +		struct kobj_attribute *attr, const char *buf, size_t count) +{ +	struct damos_sysfs_dest *dest = container_of(kobj, +			struct damos_sysfs_dest, kobj); +	int err = kstrtouint(buf, 0, &dest->id); + +	return err ? err : count; +} + +static ssize_t weight_show( +		struct kobject *kobj, struct kobj_attribute *attr, char *buf) +{ +	struct damos_sysfs_dest *dest = container_of(kobj, +			struct damos_sysfs_dest, kobj); + +	return sysfs_emit(buf, "%u\n", dest->weight); +} + +static ssize_t weight_store(struct kobject *kobj, +		struct kobj_attribute *attr, const char *buf, size_t count) +{ +	struct damos_sysfs_dest *dest = container_of(kobj, +			struct damos_sysfs_dest, kobj); +	int err = kstrtouint(buf, 0, &dest->weight); + +	return err ? err : count; +} + +static void damos_sysfs_dest_release(struct kobject *kobj) +{ +	struct damos_sysfs_dest *dest = container_of(kobj, +			struct damos_sysfs_dest, kobj); +	kfree(dest); +} + +static struct kobj_attribute damos_sysfs_dest_id_attr = +		__ATTR_RW_MODE(id, 0600); + +static struct kobj_attribute damos_sysfs_dest_weight_attr = +		__ATTR_RW_MODE(weight, 0600); + +static struct attribute *damos_sysfs_dest_attrs[] = { +	&damos_sysfs_dest_id_attr.attr, +	&damos_sysfs_dest_weight_attr.attr, +	NULL, +}; +ATTRIBUTE_GROUPS(damos_sysfs_dest); + +static const struct kobj_type damos_sysfs_dest_ktype = { +	.release = damos_sysfs_dest_release, +	.sysfs_ops = &kobj_sysfs_ops, +	.default_groups = damos_sysfs_dest_groups, +}; + +/* + * dests (action destinations) directory + */ + +struct damos_sysfs_dests { +	struct kobject kobj; +	struct damos_sysfs_dest **dests_arr; +	int nr; +}; + +static struct damos_sysfs_dests * +damos_sysfs_dests_alloc(void) +{ +	return kzalloc(sizeof(struct damos_sysfs_dests), GFP_KERNEL); +} + +static void damos_sysfs_dests_rm_dirs( +		struct damos_sysfs_dests *dests) +{ +	struct damos_sysfs_dest **dests_arr = dests->dests_arr; +	int i; + +	for (i = 0; i < dests->nr; i++) +		kobject_put(&dests_arr[i]->kobj); +	dests->nr = 0; +	kfree(dests_arr); +	dests->dests_arr = NULL; +} + +static int damos_sysfs_dests_add_dirs( +		struct damos_sysfs_dests *dests, int nr_dests) +{ +	struct damos_sysfs_dest **dests_arr, *dest; +	int err, i; + +	damos_sysfs_dests_rm_dirs(dests); +	if (!nr_dests) +		return 0; + +	dests_arr = kmalloc_array(nr_dests, sizeof(*dests_arr), +			GFP_KERNEL | __GFP_NOWARN); +	if (!dests_arr) +		return -ENOMEM; +	dests->dests_arr = dests_arr; + +	for (i = 0; i < nr_dests; i++) { +		dest = damos_sysfs_dest_alloc(); +		if (!dest) { +			damos_sysfs_dests_rm_dirs(dests); +			return -ENOMEM; +		} + +		err = kobject_init_and_add(&dest->kobj, +				&damos_sysfs_dest_ktype, +				&dests->kobj, "%d", i); +		if (err) { +			kobject_put(&dest->kobj); +			damos_sysfs_dests_rm_dirs(dests); +			return err; +		} + +		dests_arr[i] = dest; +		dests->nr++; +	} +	return 0; +} + +static ssize_t nr_dests_show(struct kobject *kobj, +		struct kobj_attribute *attr, char *buf) +{ +	struct damos_sysfs_dests *dests = container_of(kobj, +			struct damos_sysfs_dests, kobj); + +	return sysfs_emit(buf, "%d\n", dests->nr); +} + +static ssize_t nr_dests_store(struct kobject *kobj, +		struct kobj_attribute *attr, const char *buf, size_t count) +{ +	struct damos_sysfs_dests *dests; +	int nr, err = kstrtoint(buf, 0, &nr); + +	if (err) +		return err; +	if (nr < 0) +		return -EINVAL; + +	dests = container_of(kobj, struct damos_sysfs_dests, kobj); + +	if (!mutex_trylock(&damon_sysfs_lock)) +		return -EBUSY; +	err = damos_sysfs_dests_add_dirs(dests, nr); +	mutex_unlock(&damon_sysfs_lock); +	if (err) +		return err; + +	return count; +} + +static void damos_sysfs_dests_release(struct kobject *kobj) +{ +	kfree(container_of(kobj, struct damos_sysfs_dests, kobj)); +} + +static struct kobj_attribute damos_sysfs_dests_nr_attr = +		__ATTR_RW_MODE(nr_dests, 0600); + +static struct attribute *damos_sysfs_dests_attrs[] = { +	&damos_sysfs_dests_nr_attr.attr, +	NULL, +}; +ATTRIBUTE_GROUPS(damos_sysfs_dests); + +static const struct kobj_type damos_sysfs_dests_ktype = { +	.release = damos_sysfs_dests_release, +	.sysfs_ops = &kobj_sysfs_ops, +	.default_groups = damos_sysfs_dests_groups, +}; + +/*   * scheme directory   */ @@ -1585,20 +1870,55 @@ struct damon_sysfs_scheme {  	struct damon_sysfs_stats *stats;  	struct damon_sysfs_scheme_regions *tried_regions;  	int target_nid; +	struct damos_sysfs_dests *dests; +}; + +struct damos_sysfs_action_name { +	enum damos_action action; +	char *name;  }; -/* This should match with enum damos_action */ -static const char * const damon_sysfs_damos_action_strs[] = { -	"willneed", -	"cold", -	"pageout", -	"hugepage", -	"nohugepage", -	"lru_prio", -	"lru_deprio", -	"migrate_hot", -	"migrate_cold", -	"stat", +static struct damos_sysfs_action_name damos_sysfs_action_names[] = { +	{ +		.action = DAMOS_WILLNEED, +		.name = "willneed", +	}, +	{ +		.action = DAMOS_COLD, +		.name = "cold", +	}, +	{ +		.action = DAMOS_PAGEOUT, +		.name = "pageout", +	}, +	{ +		.action = DAMOS_HUGEPAGE, +		.name = "hugepage", +	}, +	{ +		.action = DAMOS_NOHUGEPAGE, +		.name = "nohugepage", +	}, +	{ +		.action = DAMOS_LRU_PRIO, +		.name = "lru_prio", +	}, +	{ +		.action = DAMOS_LRU_DEPRIO, +		.name = "lru_deprio", +	}, +	{ +		.action = DAMOS_MIGRATE_HOT, +		.name = "migrate_hot", +	}, +	{ +		.action = DAMOS_MIGRATE_COLD, +		.name = "migrate_cold", +	}, +	{ +		.action = DAMOS_STAT, +		.name = "stat", +	},  };  static struct damon_sysfs_scheme *damon_sysfs_scheme_alloc( @@ -1641,6 +1961,22 @@ out:  	return err;  } +static int damos_sysfs_set_dests(struct damon_sysfs_scheme *scheme) +{ +	struct damos_sysfs_dests *dests = damos_sysfs_dests_alloc(); +	int err; + +	if (!dests) +		return -ENOMEM; +	err = kobject_init_and_add(&dests->kobj, &damos_sysfs_dests_ktype, +			&scheme->kobj, "dests"); +	if (err) +		kobject_put(&dests->kobj); +	else +		scheme->dests = dests; +	return err; +} +  static int damon_sysfs_scheme_set_quotas(struct damon_sysfs_scheme *scheme)  {  	struct damon_sysfs_quotas *quotas = damon_sysfs_quotas_alloc(); @@ -1773,9 +2109,12 @@ static int damon_sysfs_scheme_add_dirs(struct damon_sysfs_scheme *scheme)  	err = damon_sysfs_scheme_set_access_pattern(scheme);  	if (err)  		return err; -	err = damon_sysfs_scheme_set_quotas(scheme); +	err = damos_sysfs_set_dests(scheme);  	if (err)  		goto put_access_pattern_out; +	err = damon_sysfs_scheme_set_quotas(scheme); +	if (err) +		goto put_dests_out;  	err = damon_sysfs_scheme_set_watermarks(scheme);  	if (err)  		goto put_quotas_access_pattern_out; @@ -1806,6 +2145,9 @@ put_watermarks_quotas_access_pattern_out:  put_quotas_access_pattern_out:  	kobject_put(&scheme->quotas->kobj);  	scheme->quotas = NULL; +put_dests_out: +	kobject_put(&scheme->dests->kobj); +	scheme->dests = NULL;  put_access_pattern_out:  	kobject_put(&scheme->access_pattern->kobj);  	scheme->access_pattern = NULL; @@ -1816,6 +2158,8 @@ static void damon_sysfs_scheme_rm_dirs(struct damon_sysfs_scheme *scheme)  {  	damon_sysfs_access_pattern_rm_dirs(scheme->access_pattern);  	kobject_put(&scheme->access_pattern->kobj); +	kobject_put(&scheme->dests->kobj); +	damos_sysfs_dests_rm_dirs(scheme->dests);  	damon_sysfs_quotas_rm_dirs(scheme->quotas);  	kobject_put(&scheme->quotas->kobj);  	kobject_put(&scheme->watermarks->kobj); @@ -1835,9 +2179,16 @@ static ssize_t action_show(struct kobject *kobj, struct kobj_attribute *attr,  {  	struct damon_sysfs_scheme *scheme = container_of(kobj,  			struct damon_sysfs_scheme, kobj); +	int i; -	return sysfs_emit(buf, "%s\n", -			damon_sysfs_damos_action_strs[scheme->action]); +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_action_names); i++) { +		struct damos_sysfs_action_name *action_name; + +		action_name = &damos_sysfs_action_names[i]; +		if (action_name->action == scheme->action) +			return sysfs_emit(buf, "%s\n", action_name->name); +	} +	return -EINVAL;  }  static ssize_t action_store(struct kobject *kobj, struct kobj_attribute *attr, @@ -1845,11 +2196,14 @@ static ssize_t action_store(struct kobject *kobj, struct kobj_attribute *attr,  {  	struct damon_sysfs_scheme *scheme = container_of(kobj,  			struct damon_sysfs_scheme, kobj); -	enum damos_action action; +	int i; + +	for (i = 0; i < ARRAY_SIZE(damos_sysfs_action_names); i++) { +		struct damos_sysfs_action_name *action_name; -	for (action = 0; action < NR_DAMOS_ACTIONS; action++) { -		if (sysfs_streq(buf, damon_sysfs_damos_action_strs[action])) { -			scheme->action = action; +		action_name = &damos_sysfs_action_names[i]; +		if (sysfs_streq(buf, action_name->name)) { +			scheme->action = action_name->action;  			return count;  		}  	} @@ -2222,6 +2576,29 @@ void damos_sysfs_update_effective_quotas(  	}  } +static int damos_sysfs_add_migrate_dest(struct damos *scheme, +		struct damos_sysfs_dests *sysfs_dests) +{ +	struct damos_migrate_dests *dests = &scheme->migrate_dests; +	int i; + +	dests->node_id_arr = kmalloc_array(sysfs_dests->nr, +			sizeof(*dests->node_id_arr), GFP_KERNEL); +	if (!dests->node_id_arr) +		return -ENOMEM; +	dests->weight_arr = kmalloc_array(sysfs_dests->nr, +			sizeof(*dests->weight_arr), GFP_KERNEL); +	if (!dests->weight_arr) +		/* ->node_id_arr will be freed by scheme destruction */ +		return -ENOMEM; +	for (i = 0; i < sysfs_dests->nr; i++) { +		dests->node_id_arr[i] = sysfs_dests->dests_arr[i]->id; +		dests->weight_arr[i] = sysfs_dests->dests_arr[i]->weight; +	} +	dests->nr_dests = sysfs_dests->nr; +	return 0; +} +  static struct damos *damon_sysfs_mk_scheme(  		struct damon_sysfs_scheme *sysfs_scheme)  { @@ -2284,6 +2661,11 @@ static struct damos *damon_sysfs_mk_scheme(  		damon_destroy_scheme(scheme);  		return NULL;  	} +	err = damos_sysfs_add_migrate_dest(scheme, sysfs_scheme->dests); +	if (err) { +		damon_destroy_scheme(scheme); +		return NULL; +	}  	return scheme;  } | 
