mirror of https://gitee.com/openkylin/linux.git
x86/compat: Adjust in_compat_syscall() to generic code under !COMPAT
The result of in_compat_syscall() can be pictured as:
x86 platform:
---------------------------------------------------
| Arch\syscall | 64-bit | ia32 | x32 |
|-------------------------------------------------|
| x86_64 | false | true | true |
|-------------------------------------------------|
| i686 | | <true> | |
---------------------------------------------------
Other platforms:
-------------------------------------------
| Arch\syscall | 64-bit | compat |
|-----------------------------------------|
| 64-bit | false | true |
|-----------------------------------------|
| 32-bit(?) | | <false> |
-------------------------------------------
As seen, the result of in_compat_syscall() on generic 32-bit platform
differs from i686.
There is no reason for in_compat_syscall() == true on native i686. It also
easy to misread code if the result on native 32-bit platform differs
between arches.
Because of that non arch-specific code has many places with:
if (IS_ENABLED(CONFIG_COMPAT) && in_compat_syscall())
in different variations.
It looks-like the only non-x86 code which uses in_compat_syscall() not
under CONFIG_COMPAT guard is in amd/amdkfd. But according to the commit
a18069c132
("amdkfd: Disable support for 32-bit user processes"), it
actually should be disabled on native i686.
Rename in_compat_syscall() to in_32bit_syscall() for x86-specific code
and make in_compat_syscall() false under !CONFIG_COMPAT.
A follow on patch will clean up generic users which were forced to check
IS_ENABLED(CONFIG_COMPAT) with in_compat_syscall().
Signed-off-by: Dmitry Safonov <dima@arista.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Reviewed-by: Andy Lutomirski <luto@kernel.org>
Cc: Dmitry Safonov <0x7f454c46@gmail.com>
Cc: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: John Stultz <john.stultz@linaro.org>
Cc: "Kirill A. Shutemov" <kirill.shutemov@linux.intel.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: Steffen Klassert <steffen.klassert@secunet.com>
Cc: Stephen Boyd <sboyd@kernel.org>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: linux-efi@vger.kernel.org
Cc: netdev@vger.kernel.org
Link: https://lkml.kernel.org/r/20181012134253.23266-2-dima@arista.com
This commit is contained in:
parent
c6ee7a548e
commit
a846446b19
|
@ -217,11 +217,18 @@ static inline bool in_x32_syscall(void)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool in_compat_syscall(void)
|
static inline bool in_32bit_syscall(void)
|
||||||
{
|
{
|
||||||
return in_ia32_syscall() || in_x32_syscall();
|
return in_ia32_syscall() || in_x32_syscall();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_COMPAT
|
||||||
|
static inline bool in_compat_syscall(void)
|
||||||
|
{
|
||||||
|
return in_32bit_syscall();
|
||||||
|
}
|
||||||
#define in_compat_syscall in_compat_syscall /* override the generic impl */
|
#define in_compat_syscall in_compat_syscall /* override the generic impl */
|
||||||
|
#endif
|
||||||
|
|
||||||
struct compat_siginfo;
|
struct compat_siginfo;
|
||||||
int __copy_siginfo_to_user32(struct compat_siginfo __user *to,
|
int __copy_siginfo_to_user32(struct compat_siginfo __user *to,
|
||||||
|
|
|
@ -76,9 +76,7 @@ static inline bool arch_syscall_match_sym_name(const char *sym, const char *name
|
||||||
#define ARCH_TRACE_IGNORE_COMPAT_SYSCALLS 1
|
#define ARCH_TRACE_IGNORE_COMPAT_SYSCALLS 1
|
||||||
static inline bool arch_trace_is_compat_syscall(struct pt_regs *regs)
|
static inline bool arch_trace_is_compat_syscall(struct pt_regs *regs)
|
||||||
{
|
{
|
||||||
if (in_compat_syscall())
|
return in_32bit_syscall();
|
||||||
return true;
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
#endif /* CONFIG_FTRACE_SYSCALLS && CONFIG_IA32_EMULATION */
|
#endif /* CONFIG_FTRACE_SYSCALLS && CONFIG_IA32_EMULATION */
|
||||||
#endif /* !COMPILE_OFFSETS */
|
#endif /* !COMPILE_OFFSETS */
|
||||||
|
|
|
@ -701,10 +701,10 @@ static void __set_personality_x32(void)
|
||||||
current->mm->context.ia32_compat = TIF_X32;
|
current->mm->context.ia32_compat = TIF_X32;
|
||||||
current->personality &= ~READ_IMPLIES_EXEC;
|
current->personality &= ~READ_IMPLIES_EXEC;
|
||||||
/*
|
/*
|
||||||
* in_compat_syscall() uses the presence of the x32 syscall bit
|
* in_32bit_syscall() uses the presence of the x32 syscall bit
|
||||||
* flag to determine compat status. The x86 mmap() code relies on
|
* flag to determine compat status. The x86 mmap() code relies on
|
||||||
* the syscall bitness so set x32 syscall bit right here to make
|
* the syscall bitness so set x32 syscall bit right here to make
|
||||||
* in_compat_syscall() work during exec().
|
* in_32bit_syscall() work during exec().
|
||||||
*
|
*
|
||||||
* Pretend to come from a x32 execve.
|
* Pretend to come from a x32 execve.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -105,7 +105,7 @@ SYSCALL_DEFINE6(mmap, unsigned long, addr, unsigned long, len,
|
||||||
static void find_start_end(unsigned long addr, unsigned long flags,
|
static void find_start_end(unsigned long addr, unsigned long flags,
|
||||||
unsigned long *begin, unsigned long *end)
|
unsigned long *begin, unsigned long *end)
|
||||||
{
|
{
|
||||||
if (!in_compat_syscall() && (flags & MAP_32BIT)) {
|
if (!in_32bit_syscall() && (flags & MAP_32BIT)) {
|
||||||
/* This is usually used needed to map code in small
|
/* This is usually used needed to map code in small
|
||||||
model, so it needs to be in the first 31bit. Limit
|
model, so it needs to be in the first 31bit. Limit
|
||||||
it to that. This means we need to move the
|
it to that. This means we need to move the
|
||||||
|
@ -122,7 +122,7 @@ static void find_start_end(unsigned long addr, unsigned long flags,
|
||||||
}
|
}
|
||||||
|
|
||||||
*begin = get_mmap_base(1);
|
*begin = get_mmap_base(1);
|
||||||
if (in_compat_syscall())
|
if (in_32bit_syscall())
|
||||||
*end = task_size_32bit();
|
*end = task_size_32bit();
|
||||||
else
|
else
|
||||||
*end = task_size_64bit(addr > DEFAULT_MAP_WINDOW);
|
*end = task_size_64bit(addr > DEFAULT_MAP_WINDOW);
|
||||||
|
@ -193,7 +193,7 @@ arch_get_unmapped_area_topdown(struct file *filp, const unsigned long addr0,
|
||||||
return addr;
|
return addr;
|
||||||
|
|
||||||
/* for MAP_32BIT mappings we force the legacy mmap base */
|
/* for MAP_32BIT mappings we force the legacy mmap base */
|
||||||
if (!in_compat_syscall() && (flags & MAP_32BIT))
|
if (!in_32bit_syscall() && (flags & MAP_32BIT))
|
||||||
goto bottomup;
|
goto bottomup;
|
||||||
|
|
||||||
/* requesting a specific address */
|
/* requesting a specific address */
|
||||||
|
@ -217,9 +217,10 @@ arch_get_unmapped_area_topdown(struct file *filp, const unsigned long addr0,
|
||||||
* If hint address is above DEFAULT_MAP_WINDOW, look for unmapped area
|
* If hint address is above DEFAULT_MAP_WINDOW, look for unmapped area
|
||||||
* in the full address space.
|
* in the full address space.
|
||||||
*
|
*
|
||||||
* !in_compat_syscall() check to avoid high addresses for x32.
|
* !in_32bit_syscall() check to avoid high addresses for x32
|
||||||
|
* (and make it no op on native i386).
|
||||||
*/
|
*/
|
||||||
if (addr > DEFAULT_MAP_WINDOW && !in_compat_syscall())
|
if (addr > DEFAULT_MAP_WINDOW && !in_32bit_syscall())
|
||||||
info.high_limit += TASK_SIZE_MAX - DEFAULT_MAP_WINDOW;
|
info.high_limit += TASK_SIZE_MAX - DEFAULT_MAP_WINDOW;
|
||||||
|
|
||||||
info.align_mask = 0;
|
info.align_mask = 0;
|
||||||
|
|
|
@ -92,7 +92,7 @@ static unsigned long hugetlb_get_unmapped_area_bottomup(struct file *file,
|
||||||
* If hint address is above DEFAULT_MAP_WINDOW, look for unmapped area
|
* If hint address is above DEFAULT_MAP_WINDOW, look for unmapped area
|
||||||
* in the full address space.
|
* in the full address space.
|
||||||
*/
|
*/
|
||||||
info.high_limit = in_compat_syscall() ?
|
info.high_limit = in_32bit_syscall() ?
|
||||||
task_size_32bit() : task_size_64bit(addr > DEFAULT_MAP_WINDOW);
|
task_size_32bit() : task_size_64bit(addr > DEFAULT_MAP_WINDOW);
|
||||||
|
|
||||||
info.align_mask = PAGE_MASK & ~huge_page_mask(h);
|
info.align_mask = PAGE_MASK & ~huge_page_mask(h);
|
||||||
|
@ -116,7 +116,7 @@ static unsigned long hugetlb_get_unmapped_area_topdown(struct file *file,
|
||||||
* If hint address is above DEFAULT_MAP_WINDOW, look for unmapped area
|
* If hint address is above DEFAULT_MAP_WINDOW, look for unmapped area
|
||||||
* in the full address space.
|
* in the full address space.
|
||||||
*/
|
*/
|
||||||
if (addr > DEFAULT_MAP_WINDOW && !in_compat_syscall())
|
if (addr > DEFAULT_MAP_WINDOW && !in_32bit_syscall())
|
||||||
info.high_limit += TASK_SIZE_MAX - DEFAULT_MAP_WINDOW;
|
info.high_limit += TASK_SIZE_MAX - DEFAULT_MAP_WINDOW;
|
||||||
|
|
||||||
info.align_mask = PAGE_MASK & ~huge_page_mask(h);
|
info.align_mask = PAGE_MASK & ~huge_page_mask(h);
|
||||||
|
|
|
@ -166,7 +166,7 @@ unsigned long get_mmap_base(int is_legacy)
|
||||||
struct mm_struct *mm = current->mm;
|
struct mm_struct *mm = current->mm;
|
||||||
|
|
||||||
#ifdef CONFIG_HAVE_ARCH_COMPAT_MMAP_BASES
|
#ifdef CONFIG_HAVE_ARCH_COMPAT_MMAP_BASES
|
||||||
if (in_compat_syscall()) {
|
if (in_32bit_syscall()) {
|
||||||
return is_legacy ? mm->mmap_compat_legacy_base
|
return is_legacy ? mm->mmap_compat_legacy_base
|
||||||
: mm->mmap_compat_base;
|
: mm->mmap_compat_base;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1029,9 +1029,9 @@ int kcompat_sys_fstatfs64(unsigned int fd, compat_size_t sz,
|
||||||
#else /* !CONFIG_COMPAT */
|
#else /* !CONFIG_COMPAT */
|
||||||
|
|
||||||
#define is_compat_task() (0)
|
#define is_compat_task() (0)
|
||||||
#ifndef in_compat_syscall
|
/* Ensure no one redefines in_compat_syscall() under !CONFIG_COMPAT */
|
||||||
|
#define in_compat_syscall in_compat_syscall
|
||||||
static inline bool in_compat_syscall(void) { return false; }
|
static inline bool in_compat_syscall(void) { return false; }
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif /* CONFIG_COMPAT */
|
#endif /* CONFIG_COMPAT */
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue