From: Vincenzo Frascino <vincenzo.frasc...@arm.com>

The gcr_user mask is a per thread mask that represents the tags that are
excluded from random generation when the Memory Tagging Extension is
present and an 'irg' instruction is invoked.

gcr_user affects the behavior on EL0 only.

Currently that mask is an include mask and it is controlled by the user
via prctl() while GCR_EL1 accepts an exclude mask.

Convert the include mask into an exclude one to make it easier the
register setting.

Note: This change will affect gcr_kernel (for EL1) introduced with a
future patch.

Signed-off-by: Vincenzo Frascino <vincenzo.frasc...@arm.com>
Signed-off-by: Andrey Konovalov <andreyk...@google.com>
Reviewed-by: Catalin Marinas <catalin.mari...@arm.com>
---
Change-Id: Id15c0b47582fb51594bb26fb8353d78c7d0953c1
---
 arch/arm64/include/asm/processor.h |  2 +-
 arch/arm64/kernel/mte.c            | 29 +++++++++++++++--------------
 2 files changed, 16 insertions(+), 15 deletions(-)

diff --git a/arch/arm64/include/asm/processor.h 
b/arch/arm64/include/asm/processor.h
index fce8cbecd6bc..e8cfc41a92d4 100644
--- a/arch/arm64/include/asm/processor.h
+++ b/arch/arm64/include/asm/processor.h
@@ -154,7 +154,7 @@ struct thread_struct {
 #endif
 #ifdef CONFIG_ARM64_MTE
        u64                     sctlr_tcf0;
-       u64                     gcr_user_incl;
+       u64                     gcr_user_excl;
 #endif
 };
 
diff --git a/arch/arm64/kernel/mte.c b/arch/arm64/kernel/mte.c
index 7f477991a6cf..664c968dc43c 100644
--- a/arch/arm64/kernel/mte.c
+++ b/arch/arm64/kernel/mte.c
@@ -156,23 +156,22 @@ static void set_sctlr_el1_tcf0(u64 tcf0)
        preempt_enable();
 }
 
-static void update_gcr_el1_excl(u64 incl)
+static void update_gcr_el1_excl(u64 excl)
 {
-       u64 excl = ~incl & SYS_GCR_EL1_EXCL_MASK;
 
        /*
-        * Note that 'incl' is an include mask (controlled by the user via
-        * prctl()) while GCR_EL1 accepts an exclude mask.
+        * Note that the mask controlled by the user via prctl() is an
+        * include while GCR_EL1 accepts an exclude mask.
         * No need for ISB since this only affects EL0 currently, implicit
         * with ERET.
         */
        sysreg_clear_set_s(SYS_GCR_EL1, SYS_GCR_EL1_EXCL_MASK, excl);
 }
 
-static void set_gcr_el1_excl(u64 incl)
+static void set_gcr_el1_excl(u64 excl)
 {
-       current->thread.gcr_user_incl = incl;
-       update_gcr_el1_excl(incl);
+       current->thread.gcr_user_excl = excl;
+       update_gcr_el1_excl(excl);
 }
 
 void flush_mte_state(void)
@@ -187,7 +186,7 @@ void flush_mte_state(void)
        /* disable tag checking */
        set_sctlr_el1_tcf0(SCTLR_EL1_TCF0_NONE);
        /* reset tag generation mask */
-       set_gcr_el1_excl(0);
+       set_gcr_el1_excl(SYS_GCR_EL1_EXCL_MASK);
 }
 
 void mte_thread_switch(struct task_struct *next)
@@ -198,7 +197,7 @@ void mte_thread_switch(struct task_struct *next)
        /* avoid expensive SCTLR_EL1 accesses if no change */
        if (current->thread.sctlr_tcf0 != next->thread.sctlr_tcf0)
                update_sctlr_el1_tcf0(next->thread.sctlr_tcf0);
-       update_gcr_el1_excl(next->thread.gcr_user_incl);
+       update_gcr_el1_excl(next->thread.gcr_user_excl);
 }
 
 void mte_suspend_exit(void)
@@ -206,13 +205,14 @@ void mte_suspend_exit(void)
        if (!system_supports_mte())
                return;
 
-       update_gcr_el1_excl(current->thread.gcr_user_incl);
+       update_gcr_el1_excl(current->thread.gcr_user_excl);
 }
 
 long set_mte_ctrl(struct task_struct *task, unsigned long arg)
 {
        u64 tcf0;
-       u64 gcr_incl = (arg & PR_MTE_TAG_MASK) >> PR_MTE_TAG_SHIFT;
+       u64 gcr_excl = ~((arg & PR_MTE_TAG_MASK) >> PR_MTE_TAG_SHIFT) &
+                      SYS_GCR_EL1_EXCL_MASK;
 
        if (!system_supports_mte())
                return 0;
@@ -233,10 +233,10 @@ long set_mte_ctrl(struct task_struct *task, unsigned long 
arg)
 
        if (task != current) {
                task->thread.sctlr_tcf0 = tcf0;
-               task->thread.gcr_user_incl = gcr_incl;
+               task->thread.gcr_user_excl = gcr_excl;
        } else {
                set_sctlr_el1_tcf0(tcf0);
-               set_gcr_el1_excl(gcr_incl);
+               set_gcr_el1_excl(gcr_excl);
        }
 
        return 0;
@@ -245,11 +245,12 @@ long set_mte_ctrl(struct task_struct *task, unsigned long 
arg)
 long get_mte_ctrl(struct task_struct *task)
 {
        unsigned long ret;
+       u64 incl = ~task->thread.gcr_user_excl & SYS_GCR_EL1_EXCL_MASK;
 
        if (!system_supports_mte())
                return 0;
 
-       ret = task->thread.gcr_user_incl << PR_MTE_TAG_SHIFT;
+       ret = incl << PR_MTE_TAG_SHIFT;
 
        switch (task->thread.sctlr_tcf0) {
        case SCTLR_EL1_TCF0_NONE:
-- 
2.29.2.222.g5d2a92d10f8-goog

Reply via email to