From: Gutierrez Asier <gutierrez.asier@huawei-partners.com>
To: Enze Li <lienze@kylinos.cn>, <sj@kernel.org>,
<akpm@linux-foundation.org>
Cc: <damon@lists.linux.dev>, <linux-mm@kvack.org>, <enze.li@gmx.com>
Subject: Re: [RFC PATCH 1/2] mm/damon/core: introduce priority concept for DAMON
Date: Mon, 22 Sep 2025 14:16:31 +0300 [thread overview]
Message-ID: <5a9dd164-8dca-4852-a6ab-bab752a37810@huawei-partners.com> (raw)
In-Reply-To: <20250922101022.362822-2-lienze@kylinos.cn>
Hi,
On 9/22/2025 1:10 PM, Enze Li wrote:
> This patch introduces a priority-based scheme application mechanism to
> DAMON, enhancing its ability to prioritize memory management operations
> for specific processes. Currently, DAMON applies schemes uniformly
> across all monitored processes without regard to their relative
> importance. This change allows users to assign a priority value to each
> target process, influencing the frequency with which schemes are applied.
>
> The core implementation modifies kdamond_apply_schemes to track and apply
> schemes according to user-defined priorities. For instance, if process A
> has priority 50 and process B has priority 30, damon_do_apply_schemes
> will be called 50 times on A for every 30 times on B. This ratio
> ensures that higher-priority processes receive more frequent attention
> from DAMON's operations, which can be critical for performance-sensitive
> workloads or prioritized tasks.
>
> The change maintains the overall fairness and non-starvation properties
> by cycling through targets proportionally. Existing behavior is
> preserved for targets without a priority assigned, ensuring backward
> compatibility. This provides system administrators with finer control
> over how DAMON’s memory optimizations are distributed, making the
> subsystem more adaptable to varied use cases and performance
> requirements.
>
> Signed-off-by: Enze Li <lienze@kylinos.cn>
> ---
> include/linux/damon.h | 2 +
> mm/damon/core.c | 91 ++++++++++++++++++++++++++++++++++++++++++-
> 2 files changed, 91 insertions(+), 2 deletions(-)
>
> diff --git a/include/linux/damon.h b/include/linux/damon.h
> index f13664c62ddd..a6d3ce186fdc 100644
> --- a/include/linux/damon.h
> +++ b/include/linux/damon.h
> @@ -102,6 +102,8 @@ struct damon_target {
> unsigned int nr_regions;
> struct list_head regions_list;
> struct list_head list;
> + unsigned int priority;
> + struct list_head pr_list;
> };
>
> /**
> diff --git a/mm/damon/core.c b/mm/damon/core.c
> index 106ee8b0f2d5..c336914d03cc 100644
> --- a/mm/damon/core.c
> +++ b/mm/damon/core.c
> @@ -15,6 +15,7 @@
> #include <linux/slab.h>
> #include <linux/string.h>
> #include <linux/string_choices.h>
> +#include <linux/list_sort.h>
>
> #define CREATE_TRACE_POINTS
> #include <trace/events/damon.h>
> @@ -28,6 +29,11 @@ static DEFINE_MUTEX(damon_lock);
> static int nr_running_ctxs;
> static bool running_exclusive_ctxs;
>
> +static LIST_HEAD(priority_list);
> +static bool init_priority_list;
> +static int priority_level;
> +static int priority_tick;
What would happen if you have 2 kdamond threads running at the same time?
I guess that you will end up a data race in priority_tick.
> +
> static DEFINE_MUTEX(damon_ops_lock);
> static struct damon_operations damon_registered_ops[NR_DAMON_OPS];
>
> @@ -474,6 +480,7 @@ struct damon_target *damon_new_target(void)
> t->nr_regions = 0;
> INIT_LIST_HEAD(&t->regions_list);
> INIT_LIST_HEAD(&t->list);
> + t->priority = 0;
>
> return t;
> }
> @@ -2154,6 +2161,72 @@ static void damos_adjust_quota(struct damon_ctx *c, struct damos *s)
> quota->min_score = score;
> }
>
> +static int damon_priority_head(void)
> +{
> + struct damon_target *t = list_first_entry_or_null(&priority_list,
> + struct damon_target, pr_list);
Don't we need to check for a null pointer here?
> +
> + return t->priority;
> +}
> +
> +static int damon_priority_find_next(void)
> +{
> + struct damon_target *t;
> +
> + list_for_each_entry(t, &priority_list, pr_list)
> + if (priority_level > t->priority) {
> + priority_level = t->priority;
> + return t->priority;
> + }
> + return damon_priority_head();
> +}
> +
> +static bool kdamond_targets_priority_enabled(struct damon_ctx *c)
> +{
> + struct damon_target *t;
> +
> + damon_for_each_target(t, c)
> + if (t->priority > 0)
> + return true;
> + return false;
> +}
> +
> +static int damon_priority_cmp(void *priv, const struct list_head *a,
> + const struct list_head *b)
> +{
> + struct damon_target *ta = container_of(a, struct damon_target, pr_list);
> + struct damon_target *tb = container_of(b, struct damon_target, pr_list);
> +
> + if (ta->priority < tb->priority)
> + return 1;
> + else
> + return 0;
> +}
> +
> +static bool kdamond_targets_priority_init(struct damon_ctx *c)
> +{
> + struct damon_target *t;
> +
> + damon_for_each_target(t, c)
> + list_add_tail(&t->pr_list, &priority_list);
> +
> + list_for_each_entry(t, &priority_list, pr_list)
> + pr_debug("damon target priority %d %p\n", t->priority, t->pid);
> +
> + list_sort(NULL, &priority_list, damon_priority_cmp);
> +
> + list_for_each_entry(t, &priority_list, pr_list)
> + pr_debug("damon target priority after sort %d %p\n",
> + t->priority, t->pid);
> +
> + init_priority_list = true;
> + priority_level = damon_priority_head();
> + priority_tick = priority_level;
> + pr_debug("damon target priority init level=%d tick=%d\n",
> + priority_level, priority_tick);
> + return true;
> +}
> +
> static void kdamond_apply_schemes(struct damon_ctx *c)
> {
> struct damon_target *t;
> @@ -2162,6 +2235,7 @@ static void kdamond_apply_schemes(struct damon_ctx *c)
> unsigned long sample_interval = c->attrs.sample_interval ?
> c->attrs.sample_interval : 1;
> bool has_schemes_to_apply = false;
> + bool priority_enabled = false;
>
> damon_for_each_scheme(s, c) {
> if (c->passed_sample_intervals < s->next_apply_sis)
> @@ -2178,10 +2252,23 @@ static void kdamond_apply_schemes(struct damon_ctx *c)
> if (!has_schemes_to_apply)
> return;
>
> + priority_enabled = kdamond_targets_priority_enabled(c);
> + if (priority_enabled && !init_priority_list)
> + kdamond_targets_priority_init(c);
What happens if you later add a new target? priority_list is already
initialized, which means that list sorting will not take place and it will
break your logic.
> +
> mutex_lock(&c->walk_control_lock);
> damon_for_each_target(t, c) {
> - damon_for_each_region_safe(r, next_r, t)
> - damon_do_apply_schemes(c, t, r);
> + if (priority_enabled == false ||
> + (priority_enabled == true && t->priority == priority_level &&
> + priority_tick-- > 0)) {
> + if (priority_enabled == true && priority_tick <= 0) {
> + priority_level = damon_priority_find_next();
> + priority_tick = priority_level;
> + }
> + pr_debug("tick() %d(%d)\n", priority_level, priority_tick);
> + damon_for_each_region_safe(r, next_r, t)
> + damon_do_apply_schemes(c, t, r);
> + }
> }
>
> damon_for_each_scheme(s, c) {
--
Asier Gutierrez
Huawei
next prev parent reply other threads:[~2025-09-22 11:16 UTC|newest]
Thread overview: 8+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-09-22 10:10 [RFC PATCH 0/2] DAMON: add priority-based scheme application control Enze Li
2025-09-22 10:10 ` [RFC PATCH 1/2] mm/damon/core: introduce priority concept for DAMON Enze Li
2025-09-22 11:16 ` Gutierrez Asier [this message]
2025-09-26 3:57 ` Enze Li
2025-09-22 10:10 ` [RFC PATCH 2/2] mm/damon/sysfs: add priority support for DAMOS targets Enze Li
2025-09-22 13:01 ` [RFC PATCH 0/2] DAMON: add priority-based scheme application control SeongJae Park
2025-09-26 3:23 ` Enze Li
2025-09-26 17:37 ` SeongJae Park
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=5a9dd164-8dca-4852-a6ab-bab752a37810@huawei-partners.com \
--to=gutierrez.asier@huawei-partners.com \
--cc=akpm@linux-foundation.org \
--cc=damon@lists.linux.dev \
--cc=enze.li@gmx.com \
--cc=lienze@kylinos.cn \
--cc=linux-mm@kvack.org \
--cc=sj@kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox