diff options
author | Florian Weimer <fweimer@redhat.com> | 2020-02-14 20:55:39 +0100 |
---|---|---|
committer | Florian Weimer <fweimer@redhat.com> | 2020-02-14 20:55:39 +0100 |
commit | a803367bab167f5ec4fde1f0d0ec447707c29520 (patch) | |
tree | 25ee7f81b6b75c2eedb9aafba1dff99ffd554663 /sysdeps/unix/sysv/linux/powerpc/powerpc64/pkey_set.c | |
parent | 2b7dc93f826ee2cab7e6b7e5b6432f7920f0eec0 (diff) | |
download | glibc-a803367bab167f5ec4fde1f0d0ec447707c29520.tar glibc-a803367bab167f5ec4fde1f0d0ec447707c29520.tar.gz glibc-a803367bab167f5ec4fde1f0d0ec447707c29520.tar.bz2 glibc-a803367bab167f5ec4fde1f0d0ec447707c29520.zip |
powerpc64: Add memory protection key support [BZ #23202]
The 32-bit protection key behavior is somewhat unclear on 32-bit powerpc,
so this change is restricted to the 64-bit variants.
Flag translation is needed because of hardware differences between the
POWER implementation (read and write flags) and the Intel implementation
(write and read+write flags).
Diffstat (limited to 'sysdeps/unix/sysv/linux/powerpc/powerpc64/pkey_set.c')
-rw-r--r-- | sysdeps/unix/sysv/linux/powerpc/powerpc64/pkey_set.c | 48 |
1 files changed, 48 insertions, 0 deletions
diff --git a/sysdeps/unix/sysv/linux/powerpc/powerpc64/pkey_set.c b/sysdeps/unix/sysv/linux/powerpc/powerpc64/pkey_set.c new file mode 100644 index 0000000000..20b372ee29 --- /dev/null +++ b/sysdeps/unix/sysv/linux/powerpc/powerpc64/pkey_set.c @@ -0,0 +1,48 @@ +/* Changing the per-thread memory protection key, powerpc64 version. + Copyright (C) 2017-2020 Free Software Foundation, Inc. + This file is part of the GNU C Library. + + The GNU C Library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + The GNU C Library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with the GNU C Library; if not, see + <http://www.gnu.org/licenses/>. */ + +#include <arch-pkey.h> +#include <errno.h> +#include <sys/mman.h> + +int +pkey_set (int key, unsigned int rights) +{ + if (key < 0 || key > PKEY_MAX || rights > 3) + { + __set_errno (EINVAL); + return -1; + } + + /* Translate to AMR bit values. */ + unsigned long int bits; + if (rights & PKEY_DISABLE_ACCESS) + /* The PKEY_DISABLE_WRITE bit does not matter. */ + bits = PKEY_AMR_READ | PKEY_AMR_WRITE; + else if (rights == PKEY_DISABLE_WRITE) + bits = PKEY_AMR_WRITE; + else + bits = 0; + + unsigned int index = pkey_index (key); + unsigned long int mask = 3UL << index; + unsigned long int amr = pkey_read (); + amr = (amr & ~mask) | (bits << index); + pkey_write (amr); + return 0; +} |