2012-03-05 19:49:31 +08:00
|
|
|
/*
|
|
|
|
* Based on arch/arm/kernel/signal.c
|
|
|
|
*
|
|
|
|
* Copyright (C) 1995-2009 Russell King
|
|
|
|
* Copyright (C) 2012 ARM Ltd.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2014-04-30 17:51:32 +08:00
|
|
|
#include <linux/compat.h>
|
2012-03-05 19:49:31 +08:00
|
|
|
#include <linux/errno.h>
|
2017-06-15 22:03:38 +08:00
|
|
|
#include <linux/kernel.h>
|
2012-03-05 19:49:31 +08:00
|
|
|
#include <linux/signal.h>
|
|
|
|
#include <linux/personality.h>
|
|
|
|
#include <linux/freezer.h>
|
2017-06-15 22:03:39 +08:00
|
|
|
#include <linux/stddef.h>
|
2012-03-05 19:49:31 +08:00
|
|
|
#include <linux/uaccess.h>
|
2017-06-21 01:23:39 +08:00
|
|
|
#include <linux/sizes.h>
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
#include <linux/string.h>
|
2012-03-05 19:49:31 +08:00
|
|
|
#include <linux/tracehook.h>
|
|
|
|
#include <linux/ratelimit.h>
|
2017-06-15 09:12:03 +08:00
|
|
|
#include <linux/syscalls.h>
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
#include <asm/debug-monitors.h>
|
|
|
|
#include <asm/elf.h>
|
|
|
|
#include <asm/cacheflush.h>
|
|
|
|
#include <asm/ucontext.h>
|
|
|
|
#include <asm/unistd.h>
|
|
|
|
#include <asm/fpsimd.h>
|
2017-08-01 22:35:54 +08:00
|
|
|
#include <asm/ptrace.h>
|
2012-03-05 19:49:31 +08:00
|
|
|
#include <asm/signal32.h>
|
|
|
|
#include <asm/vdso.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do a signal return; undo the signal stack. These are aligned to 128-bit.
|
|
|
|
*/
|
|
|
|
struct rt_sigframe {
|
|
|
|
struct siginfo info;
|
|
|
|
struct ucontext uc;
|
2017-06-15 22:03:38 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct frame_record {
|
2012-11-23 20:34:13 +08:00
|
|
|
u64 fp;
|
|
|
|
u64 lr;
|
2012-03-05 19:49:31 +08:00
|
|
|
};
|
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
struct rt_sigframe_user_layout {
|
|
|
|
struct rt_sigframe __user *sigframe;
|
|
|
|
struct frame_record __user *next_frame;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
|
|
|
|
unsigned long size; /* size of allocated sigframe data */
|
|
|
|
unsigned long limit; /* largest allowed size */
|
|
|
|
|
|
|
|
unsigned long fpsimd_offset;
|
|
|
|
unsigned long esr_offset;
|
2017-06-21 01:23:39 +08:00
|
|
|
unsigned long extra_offset;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
unsigned long end_offset;
|
2017-06-15 22:03:38 +08:00
|
|
|
};
|
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
#define BASE_SIGFRAME_SIZE round_up(sizeof(struct rt_sigframe), 16)
|
|
|
|
#define TERMINATOR_SIZE round_up(sizeof(struct _aarch64_ctx), 16)
|
|
|
|
#define EXTRA_CONTEXT_SIZE round_up(sizeof(struct extra_context), 16)
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
static void init_user_layout(struct rt_sigframe_user_layout *user)
|
|
|
|
{
|
2017-06-21 01:23:39 +08:00
|
|
|
const size_t reserved_size =
|
|
|
|
sizeof(user->sigframe->uc.uc_mcontext.__reserved);
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
memset(user, 0, sizeof(*user));
|
|
|
|
user->size = offsetof(struct rt_sigframe, uc.uc_mcontext.__reserved);
|
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
user->limit = user->size + reserved_size;
|
|
|
|
|
|
|
|
user->limit -= TERMINATOR_SIZE;
|
|
|
|
user->limit -= EXTRA_CONTEXT_SIZE;
|
|
|
|
/* Reserve space for extension and terminator ^ */
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static size_t sigframe_size(struct rt_sigframe_user_layout const *user)
|
|
|
|
{
|
|
|
|
return round_up(max(user->size, sizeof(struct rt_sigframe)), 16);
|
|
|
|
}
|
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
/*
|
|
|
|
* Sanity limit on the approximate maximum size of signal frame we'll
|
|
|
|
* try to generate. Stack alignment padding and the frame record are
|
|
|
|
* not taken into account. This limit is not a guarantee and is
|
|
|
|
* NOT ABI.
|
|
|
|
*/
|
|
|
|
#define SIGFRAME_MAXSZ SZ_64K
|
|
|
|
|
|
|
|
static int __sigframe_alloc(struct rt_sigframe_user_layout *user,
|
|
|
|
unsigned long *offset, size_t size, bool extend)
|
|
|
|
{
|
|
|
|
size_t padded_size = round_up(size, 16);
|
|
|
|
|
|
|
|
if (padded_size > user->limit - user->size &&
|
|
|
|
!user->extra_offset &&
|
|
|
|
extend) {
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
user->limit += EXTRA_CONTEXT_SIZE;
|
|
|
|
ret = __sigframe_alloc(user, &user->extra_offset,
|
|
|
|
sizeof(struct extra_context), false);
|
|
|
|
if (ret) {
|
|
|
|
user->limit -= EXTRA_CONTEXT_SIZE;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reserve space for the __reserved[] terminator */
|
|
|
|
user->size += TERMINATOR_SIZE;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allow expansion up to SIGFRAME_MAXSZ, ensuring space for
|
|
|
|
* the terminator:
|
|
|
|
*/
|
|
|
|
user->limit = SIGFRAME_MAXSZ - TERMINATOR_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Still not enough space? Bad luck! */
|
|
|
|
if (padded_size > user->limit - user->size)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
*offset = user->size;
|
|
|
|
user->size += padded_size;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-06-15 22:03:41 +08:00
|
|
|
/*
|
|
|
|
* Allocate space for an optional record of <size> bytes in the user
|
|
|
|
* signal frame. The offset from the signal frame base address to the
|
|
|
|
* allocated block is assigned to *offset.
|
|
|
|
*/
|
|
|
|
static int sigframe_alloc(struct rt_sigframe_user_layout *user,
|
|
|
|
unsigned long *offset, size_t size)
|
|
|
|
{
|
2017-06-21 01:23:39 +08:00
|
|
|
return __sigframe_alloc(user, offset, size, true);
|
|
|
|
}
|
2017-06-15 22:03:41 +08:00
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
/* Allocate the null terminator record and prevent further allocations */
|
|
|
|
static int sigframe_alloc_end(struct rt_sigframe_user_layout *user)
|
|
|
|
{
|
|
|
|
int ret;
|
2017-06-15 22:03:41 +08:00
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
/* Un-reserve the space reserved for the terminator: */
|
|
|
|
user->limit += TERMINATOR_SIZE;
|
|
|
|
|
|
|
|
ret = sigframe_alloc(user, &user->end_offset,
|
|
|
|
sizeof(struct _aarch64_ctx));
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* Prevent further allocation: */
|
|
|
|
user->limit = user->size;
|
2017-06-15 22:03:41 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
static void __user *apply_user_offset(
|
|
|
|
struct rt_sigframe_user_layout const *user, unsigned long offset)
|
|
|
|
{
|
|
|
|
char __user *base = (char __user *)user->sigframe;
|
|
|
|
|
|
|
|
return base + offset;
|
|
|
|
}
|
|
|
|
|
2012-03-05 19:49:31 +08:00
|
|
|
static int preserve_fpsimd_context(struct fpsimd_context __user *ctx)
|
|
|
|
{
|
|
|
|
struct fpsimd_state *fpsimd = ¤t->thread.fpsimd_state;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* dump the hardware registers to the fpsimd_state structure */
|
2014-02-24 22:26:27 +08:00
|
|
|
fpsimd_preserve_current_state();
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
/* copy the FP and status/control registers */
|
|
|
|
err = __copy_to_user(ctx->vregs, fpsimd->vregs, sizeof(fpsimd->vregs));
|
|
|
|
__put_user_error(fpsimd->fpsr, &ctx->fpsr, err);
|
|
|
|
__put_user_error(fpsimd->fpcr, &ctx->fpcr, err);
|
|
|
|
|
|
|
|
/* copy the magic/size information */
|
|
|
|
__put_user_error(FPSIMD_MAGIC, &ctx->head.magic, err);
|
|
|
|
__put_user_error(sizeof(struct fpsimd_context), &ctx->head.size, err);
|
|
|
|
|
|
|
|
return err ? -EFAULT : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int restore_fpsimd_context(struct fpsimd_context __user *ctx)
|
|
|
|
{
|
|
|
|
struct fpsimd_state fpsimd;
|
|
|
|
__u32 magic, size;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/* check the magic/size information */
|
|
|
|
__get_user_error(magic, &ctx->head.magic, err);
|
|
|
|
__get_user_error(size, &ctx->head.size, err);
|
|
|
|
if (err)
|
|
|
|
return -EFAULT;
|
|
|
|
if (magic != FPSIMD_MAGIC || size != sizeof(struct fpsimd_context))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* copy the FP and status/control registers */
|
|
|
|
err = __copy_from_user(fpsimd.vregs, ctx->vregs,
|
|
|
|
sizeof(fpsimd.vregs));
|
|
|
|
__get_user_error(fpsimd.fpsr, &ctx->fpsr, err);
|
|
|
|
__get_user_error(fpsimd.fpcr, &ctx->fpcr, err);
|
|
|
|
|
|
|
|
/* load the hardware registers from the fpsimd_state structure */
|
2014-02-24 22:26:27 +08:00
|
|
|
if (!err)
|
|
|
|
fpsimd_update_current_state(&fpsimd);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
return err ? -EFAULT : 0;
|
|
|
|
}
|
|
|
|
|
2017-06-15 22:03:39 +08:00
|
|
|
struct user_ctxs {
|
|
|
|
struct fpsimd_context __user *fpsimd;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int parse_user_sigframe(struct user_ctxs *user,
|
|
|
|
struct rt_sigframe __user *sf)
|
|
|
|
{
|
|
|
|
struct sigcontext __user *const sc = &sf->uc.uc_mcontext;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
struct _aarch64_ctx __user *head;
|
|
|
|
char __user *base = (char __user *)&sc->__reserved;
|
2017-06-15 22:03:39 +08:00
|
|
|
size_t offset = 0;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
size_t limit = sizeof(sc->__reserved);
|
2017-06-21 01:23:39 +08:00
|
|
|
bool have_extra_context = false;
|
|
|
|
char const __user *const sfp = (char const __user *)sf;
|
2017-06-15 22:03:39 +08:00
|
|
|
|
|
|
|
user->fpsimd = NULL;
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (!IS_ALIGNED((unsigned long)base, 16))
|
|
|
|
goto invalid;
|
|
|
|
|
2017-06-15 22:03:39 +08:00
|
|
|
while (1) {
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
int err = 0;
|
2017-06-15 22:03:39 +08:00
|
|
|
u32 magic, size;
|
2017-06-21 01:23:39 +08:00
|
|
|
char const __user *userp;
|
|
|
|
struct extra_context const __user *extra;
|
|
|
|
u64 extra_datap;
|
|
|
|
u32 extra_size;
|
|
|
|
struct _aarch64_ctx const __user *end;
|
|
|
|
u32 end_magic, end_size;
|
2017-06-15 22:03:39 +08:00
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (limit - offset < sizeof(*head))
|
2017-06-15 22:03:39 +08:00
|
|
|
goto invalid;
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (!IS_ALIGNED(offset, 16))
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
head = (struct _aarch64_ctx __user *)(base + offset);
|
2017-06-15 22:03:39 +08:00
|
|
|
__get_user_error(magic, &head->magic, err);
|
|
|
|
__get_user_error(size, &head->size, err);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (limit - offset < size)
|
|
|
|
goto invalid;
|
|
|
|
|
2017-06-15 22:03:39 +08:00
|
|
|
switch (magic) {
|
|
|
|
case 0:
|
|
|
|
if (size)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
case FPSIMD_MAGIC:
|
|
|
|
if (user->fpsimd)
|
|
|
|
goto invalid;
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (size < sizeof(*user->fpsimd))
|
2017-06-15 22:03:39 +08:00
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
user->fpsimd = (struct fpsimd_context __user *)head;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ESR_MAGIC:
|
|
|
|
/* ignore */
|
|
|
|
break;
|
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
case EXTRA_MAGIC:
|
|
|
|
if (have_extra_context)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
if (size < sizeof(*extra))
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
userp = (char const __user *)head;
|
|
|
|
|
|
|
|
extra = (struct extra_context const __user *)userp;
|
|
|
|
userp += size;
|
|
|
|
|
|
|
|
__get_user_error(extra_datap, &extra->datap, err);
|
|
|
|
__get_user_error(extra_size, &extra->size, err);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
/* Check for the dummy terminator in __reserved[]: */
|
|
|
|
|
|
|
|
if (limit - offset - size < TERMINATOR_SIZE)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
end = (struct _aarch64_ctx const __user *)userp;
|
|
|
|
userp += TERMINATOR_SIZE;
|
|
|
|
|
|
|
|
__get_user_error(end_magic, &end->magic, err);
|
|
|
|
__get_user_error(end_size, &end->size, err);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (end_magic || end_size)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
/* Prevent looping/repeated parsing of extra_context */
|
|
|
|
have_extra_context = true;
|
|
|
|
|
|
|
|
base = (__force void __user *)extra_datap;
|
|
|
|
if (!IS_ALIGNED((unsigned long)base, 16))
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
if (!IS_ALIGNED(extra_size, 16))
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
if (base != userp)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
/* Reject "unreasonably large" frames: */
|
|
|
|
if (extra_size > sfp + SIGFRAME_MAXSZ - userp)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Ignore trailing terminator in __reserved[]
|
|
|
|
* and start parsing extra data:
|
|
|
|
*/
|
|
|
|
offset = 0;
|
|
|
|
limit = extra_size;
|
|
|
|
continue;
|
|
|
|
|
2017-06-15 22:03:39 +08:00
|
|
|
default:
|
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size < sizeof(*head))
|
|
|
|
goto invalid;
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (limit - offset < size)
|
2017-06-15 22:03:39 +08:00
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
offset += size;
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
if (!user->fpsimd)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
invalid:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2012-03-05 19:49:31 +08:00
|
|
|
static int restore_sigframe(struct pt_regs *regs,
|
|
|
|
struct rt_sigframe __user *sf)
|
|
|
|
{
|
|
|
|
sigset_t set;
|
|
|
|
int i, err;
|
2017-06-15 22:03:39 +08:00
|
|
|
struct user_ctxs user;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
err = __copy_from_user(&set, &sf->uc.uc_sigmask, sizeof(set));
|
|
|
|
if (err == 0)
|
|
|
|
set_current_blocked(&set);
|
|
|
|
|
|
|
|
for (i = 0; i < 31; i++)
|
|
|
|
__get_user_error(regs->regs[i], &sf->uc.uc_mcontext.regs[i],
|
|
|
|
err);
|
|
|
|
__get_user_error(regs->sp, &sf->uc.uc_mcontext.sp, err);
|
|
|
|
__get_user_error(regs->pc, &sf->uc.uc_mcontext.pc, err);
|
|
|
|
__get_user_error(regs->pstate, &sf->uc.uc_mcontext.pstate, err);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Avoid sys_rt_sigreturn() restarting.
|
|
|
|
*/
|
2017-08-01 22:35:54 +08:00
|
|
|
forget_syscall(regs);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2016-03-01 22:18:50 +08:00
|
|
|
err |= !valid_user_regs(®s->user_regs, current);
|
2017-06-15 22:03:39 +08:00
|
|
|
if (err == 0)
|
|
|
|
err = parse_user_sigframe(&user, sf);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2017-06-15 22:03:39 +08:00
|
|
|
if (err == 0)
|
|
|
|
err = restore_fpsimd_context(user.fpsimd);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
asmlinkage long sys_rt_sigreturn(struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
struct rt_sigframe __user *frame;
|
|
|
|
|
|
|
|
/* Always make any pending restarted system calls return -EINTR */
|
2015-02-13 07:01:14 +08:00
|
|
|
current->restart_block.fn = do_no_restart_syscall;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Since we stacked the signal on a 128-bit boundary, then 'sp' should
|
|
|
|
* be word aligned here.
|
|
|
|
*/
|
|
|
|
if (regs->sp & 15)
|
|
|
|
goto badframe;
|
|
|
|
|
|
|
|
frame = (struct rt_sigframe __user *)regs->sp;
|
|
|
|
|
|
|
|
if (!access_ok(VERIFY_READ, frame, sizeof (*frame)))
|
|
|
|
goto badframe;
|
|
|
|
|
|
|
|
if (restore_sigframe(regs, frame))
|
|
|
|
goto badframe;
|
|
|
|
|
2012-12-23 14:56:45 +08:00
|
|
|
if (restore_altstack(&frame->uc.uc_stack))
|
2012-03-05 19:49:31 +08:00
|
|
|
goto badframe;
|
|
|
|
|
|
|
|
return regs->regs[0];
|
|
|
|
|
|
|
|
badframe:
|
|
|
|
if (show_unhandled_signals)
|
|
|
|
pr_info_ratelimited("%s[%d]: bad frame in %s: pc=%08llx sp=%08llx\n",
|
|
|
|
current->comm, task_pid_nr(current), __func__,
|
|
|
|
regs->pc, regs->sp);
|
|
|
|
force_sig(SIGSEGV, current);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
/* Determine the layout of optional records in the signal frame */
|
|
|
|
static int setup_sigframe_layout(struct rt_sigframe_user_layout *user)
|
|
|
|
{
|
2017-06-15 22:03:41 +08:00
|
|
|
int err;
|
|
|
|
|
|
|
|
err = sigframe_alloc(user, &user->fpsimd_offset,
|
|
|
|
sizeof(struct fpsimd_context));
|
|
|
|
if (err)
|
|
|
|
return err;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
|
|
|
|
/* fault information, if valid */
|
|
|
|
if (current->thread.fault_code) {
|
2017-06-15 22:03:41 +08:00
|
|
|
err = sigframe_alloc(user, &user->esr_offset,
|
|
|
|
sizeof(struct esr_context));
|
|
|
|
if (err)
|
|
|
|
return err;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
}
|
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
return sigframe_alloc_end(user);
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
static int setup_sigframe(struct rt_sigframe_user_layout *user,
|
2012-03-05 19:49:31 +08:00
|
|
|
struct pt_regs *regs, sigset_t *set)
|
|
|
|
{
|
|
|
|
int i, err = 0;
|
2017-06-15 22:03:38 +08:00
|
|
|
struct rt_sigframe __user *sf = user->sigframe;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2012-11-23 20:34:13 +08:00
|
|
|
/* set up the stack frame for unwinding */
|
2017-06-15 22:03:38 +08:00
|
|
|
__put_user_error(regs->regs[29], &user->next_frame->fp, err);
|
|
|
|
__put_user_error(regs->regs[30], &user->next_frame->lr, err);
|
2012-11-23 20:34:13 +08:00
|
|
|
|
2012-03-05 19:49:31 +08:00
|
|
|
for (i = 0; i < 31; i++)
|
|
|
|
__put_user_error(regs->regs[i], &sf->uc.uc_mcontext.regs[i],
|
|
|
|
err);
|
|
|
|
__put_user_error(regs->sp, &sf->uc.uc_mcontext.sp, err);
|
|
|
|
__put_user_error(regs->pc, &sf->uc.uc_mcontext.pc, err);
|
|
|
|
__put_user_error(regs->pstate, &sf->uc.uc_mcontext.pstate, err);
|
|
|
|
|
|
|
|
__put_user_error(current->thread.fault_address, &sf->uc.uc_mcontext.fault_address, err);
|
|
|
|
|
|
|
|
err |= __copy_to_user(&sf->uc.uc_sigmask, set, sizeof(*set));
|
|
|
|
|
2014-04-04 22:42:16 +08:00
|
|
|
if (err == 0) {
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
struct fpsimd_context __user *fpsimd_ctx =
|
|
|
|
apply_user_offset(user, user->fpsimd_offset);
|
2014-04-04 22:42:16 +08:00
|
|
|
err |= preserve_fpsimd_context(fpsimd_ctx);
|
|
|
|
}
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2013-09-16 22:19:27 +08:00
|
|
|
/* fault information, if valid */
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (err == 0 && user->esr_offset) {
|
|
|
|
struct esr_context __user *esr_ctx =
|
|
|
|
apply_user_offset(user, user->esr_offset);
|
|
|
|
|
2013-09-16 22:19:27 +08:00
|
|
|
__put_user_error(ESR_MAGIC, &esr_ctx->head.magic, err);
|
|
|
|
__put_user_error(sizeof(*esr_ctx), &esr_ctx->head.size, err);
|
|
|
|
__put_user_error(current->thread.fault_code, &esr_ctx->esr, err);
|
|
|
|
}
|
|
|
|
|
2017-06-21 01:23:39 +08:00
|
|
|
if (err == 0 && user->extra_offset) {
|
|
|
|
char __user *sfp = (char __user *)user->sigframe;
|
|
|
|
char __user *userp =
|
|
|
|
apply_user_offset(user, user->extra_offset);
|
|
|
|
|
|
|
|
struct extra_context __user *extra;
|
|
|
|
struct _aarch64_ctx __user *end;
|
|
|
|
u64 extra_datap;
|
|
|
|
u32 extra_size;
|
|
|
|
|
|
|
|
extra = (struct extra_context __user *)userp;
|
|
|
|
userp += EXTRA_CONTEXT_SIZE;
|
|
|
|
|
|
|
|
end = (struct _aarch64_ctx __user *)userp;
|
|
|
|
userp += TERMINATOR_SIZE;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* extra_datap is just written to the signal frame.
|
|
|
|
* The value gets cast back to a void __user *
|
|
|
|
* during sigreturn.
|
|
|
|
*/
|
|
|
|
extra_datap = (__force u64)userp;
|
|
|
|
extra_size = sfp + round_up(user->size, 16) - userp;
|
|
|
|
|
|
|
|
__put_user_error(EXTRA_MAGIC, &extra->head.magic, err);
|
|
|
|
__put_user_error(EXTRA_CONTEXT_SIZE, &extra->head.size, err);
|
|
|
|
__put_user_error(extra_datap, &extra->datap, err);
|
|
|
|
__put_user_error(extra_size, &extra->size, err);
|
|
|
|
|
|
|
|
/* Add the terminator */
|
|
|
|
__put_user_error(0, &end->magic, err);
|
|
|
|
__put_user_error(0, &end->size, err);
|
|
|
|
}
|
|
|
|
|
2012-03-05 19:49:31 +08:00
|
|
|
/* set the "end" magic */
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
if (err == 0) {
|
|
|
|
struct _aarch64_ctx __user *end =
|
|
|
|
apply_user_offset(user, user->end_offset);
|
|
|
|
|
|
|
|
__put_user_error(0, &end->magic, err);
|
|
|
|
__put_user_error(0, &end->size, err);
|
|
|
|
}
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
static int get_sigframe(struct rt_sigframe_user_layout *user,
|
|
|
|
struct ksignal *ksig, struct pt_regs *regs)
|
2012-03-05 19:49:31 +08:00
|
|
|
{
|
|
|
|
unsigned long sp, sp_top;
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
int err;
|
|
|
|
|
|
|
|
init_user_layout(user);
|
|
|
|
err = setup_sigframe_layout(user);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2014-03-05 20:31:20 +08:00
|
|
|
sp = sp_top = sigsp(regs->sp, ksig);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
sp = round_down(sp - sizeof(struct frame_record), 16);
|
|
|
|
user->next_frame = (struct frame_record __user *)sp;
|
|
|
|
|
arm64: signal: factor frame layout and population into separate passes
In preparation for expanding the signal frame, this patch refactors
the signal frame setup code in setup_sigframe() into two separate
passes.
The first pass, setup_sigframe_layout(), determines the size of the
signal frame and its internal layout, including the presence and
location of optional records. The resulting knowledge is used to
allocate and locate the user stack space required for the signal
frame and to determine which optional records to include.
The second pass, setup_sigframe(), is called once the stack frame
is allocated in order to populate it with the necessary context
information.
As a result of these changes, it becomes more natural to represent
locations in the signal frame by a base pointer and an offset,
since the absolute address of each location is not known during the
layout pass. To be more consistent with this logic,
parse_user_sigframe() is refactored to describe signal frame
locations in a similar way.
This change has no effect on the signal ABI, but will make it
easier to expand the signal frame in future patches.
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
2017-06-15 22:03:40 +08:00
|
|
|
sp = round_down(sp, 16) - sigframe_size(user);
|
2017-06-15 22:03:38 +08:00
|
|
|
user->sigframe = (struct rt_sigframe __user *)sp;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that we can actually write to the signal frame.
|
|
|
|
*/
|
2017-06-15 22:03:38 +08:00
|
|
|
if (!access_ok(VERIFY_WRITE, user->sigframe, sp_top - sp))
|
|
|
|
return -EFAULT;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
return 0;
|
2012-03-05 19:49:31 +08:00
|
|
|
}
|
|
|
|
|
2012-11-23 20:34:13 +08:00
|
|
|
static void setup_return(struct pt_regs *regs, struct k_sigaction *ka,
|
2017-06-15 22:03:38 +08:00
|
|
|
struct rt_sigframe_user_layout *user, int usig)
|
2012-03-05 19:49:31 +08:00
|
|
|
{
|
|
|
|
__sigrestore_t sigtramp;
|
|
|
|
|
|
|
|
regs->regs[0] = usig;
|
2017-06-15 22:03:38 +08:00
|
|
|
regs->sp = (unsigned long)user->sigframe;
|
|
|
|
regs->regs[29] = (unsigned long)&user->next_frame->fp;
|
2012-03-05 19:49:31 +08:00
|
|
|
regs->pc = (unsigned long)ka->sa.sa_handler;
|
|
|
|
|
|
|
|
if (ka->sa.sa_flags & SA_RESTORER)
|
|
|
|
sigtramp = ka->sa.sa_restorer;
|
|
|
|
else
|
|
|
|
sigtramp = VDSO_SYMBOL(current->mm->context.vdso, sigtramp);
|
|
|
|
|
|
|
|
regs->regs[30] = (unsigned long)sigtramp;
|
|
|
|
}
|
|
|
|
|
2013-10-07 04:52:44 +08:00
|
|
|
static int setup_rt_frame(int usig, struct ksignal *ksig, sigset_t *set,
|
|
|
|
struct pt_regs *regs)
|
2012-03-05 19:49:31 +08:00
|
|
|
{
|
2017-06-15 22:03:38 +08:00
|
|
|
struct rt_sigframe_user_layout user;
|
2012-03-05 19:49:31 +08:00
|
|
|
struct rt_sigframe __user *frame;
|
|
|
|
int err = 0;
|
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
if (get_sigframe(&user, ksig, regs))
|
2012-03-05 19:49:31 +08:00
|
|
|
return 1;
|
|
|
|
|
2017-06-15 22:03:38 +08:00
|
|
|
frame = user.sigframe;
|
|
|
|
|
2012-03-05 19:49:31 +08:00
|
|
|
__put_user_error(0, &frame->uc.uc_flags, err);
|
|
|
|
__put_user_error(NULL, &frame->uc.uc_link, err);
|
|
|
|
|
2012-12-23 14:56:45 +08:00
|
|
|
err |= __save_altstack(&frame->uc.uc_stack, regs->sp);
|
2017-06-15 22:03:38 +08:00
|
|
|
err |= setup_sigframe(&user, regs, set);
|
2012-11-23 20:34:13 +08:00
|
|
|
if (err == 0) {
|
2017-06-15 22:03:38 +08:00
|
|
|
setup_return(regs, &ksig->ka, &user, usig);
|
2013-10-07 04:52:44 +08:00
|
|
|
if (ksig->ka.sa.sa_flags & SA_SIGINFO) {
|
|
|
|
err |= copy_siginfo_to_user(&frame->info, &ksig->info);
|
2012-11-23 20:34:13 +08:00
|
|
|
regs->regs[1] = (unsigned long)&frame->info;
|
|
|
|
regs->regs[2] = (unsigned long)&frame->uc;
|
|
|
|
}
|
2012-03-05 19:49:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setup_restart_syscall(struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
if (is_compat_task())
|
|
|
|
compat_setup_restart_syscall(regs);
|
|
|
|
else
|
|
|
|
regs->regs[8] = __NR_restart_syscall;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* OK, we're invoking a handler
|
|
|
|
*/
|
2013-10-07 04:52:44 +08:00
|
|
|
static void handle_signal(struct ksignal *ksig, struct pt_regs *regs)
|
2012-03-05 19:49:31 +08:00
|
|
|
{
|
|
|
|
struct task_struct *tsk = current;
|
|
|
|
sigset_t *oldset = sigmask_to_save();
|
2013-10-07 04:52:44 +08:00
|
|
|
int usig = ksig->sig;
|
2012-03-05 19:49:31 +08:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up the stack frame
|
|
|
|
*/
|
|
|
|
if (is_compat_task()) {
|
2013-10-07 04:52:44 +08:00
|
|
|
if (ksig->ka.sa.sa_flags & SA_SIGINFO)
|
|
|
|
ret = compat_setup_rt_frame(usig, ksig, oldset, regs);
|
2012-03-05 19:49:31 +08:00
|
|
|
else
|
2013-10-07 04:52:44 +08:00
|
|
|
ret = compat_setup_frame(usig, ksig, oldset, regs);
|
2012-03-05 19:49:31 +08:00
|
|
|
} else {
|
2013-10-07 04:52:44 +08:00
|
|
|
ret = setup_rt_frame(usig, ksig, oldset, regs);
|
2012-03-05 19:49:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the resulting registers are actually sane.
|
|
|
|
*/
|
2016-03-01 22:18:50 +08:00
|
|
|
ret |= !valid_user_regs(®s->user_regs, current);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Fast forward the stepping logic so we step into the signal
|
|
|
|
* handler.
|
|
|
|
*/
|
2013-10-07 04:52:44 +08:00
|
|
|
if (!ret)
|
|
|
|
user_fastforward_single_step(tsk);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
2013-10-07 04:52:44 +08:00
|
|
|
signal_setup_done(ret, ksig, 0);
|
2012-03-05 19:49:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note that 'init' is a special process: it doesn't get signals it doesn't
|
|
|
|
* want to handle. Thus you cannot kill init even with a SIGKILL even by
|
|
|
|
* mistake.
|
|
|
|
*
|
|
|
|
* Note that we go through the signals twice: once to check the signals that
|
|
|
|
* the kernel can handle, and then we build all the user-level signal handling
|
|
|
|
* stack-frames in one go after that.
|
|
|
|
*/
|
|
|
|
static void do_signal(struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
unsigned long continue_addr = 0, restart_addr = 0;
|
2013-10-07 04:52:44 +08:00
|
|
|
int retval = 0;
|
|
|
|
struct ksignal ksig;
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If we were from a system call, check for system call restarting...
|
|
|
|
*/
|
2017-08-01 22:35:54 +08:00
|
|
|
if (in_syscall(regs)) {
|
2012-03-05 19:49:31 +08:00
|
|
|
continue_addr = regs->pc;
|
|
|
|
restart_addr = continue_addr - (compat_thumb_mode(regs) ? 2 : 4);
|
|
|
|
retval = regs->regs[0];
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Avoid additional syscall restarting via ret_to_user.
|
|
|
|
*/
|
2017-08-01 22:35:54 +08:00
|
|
|
forget_syscall(regs);
|
2012-03-05 19:49:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Prepare for system call restart. We do this here so that a
|
|
|
|
* debugger will see the already changed PC.
|
|
|
|
*/
|
|
|
|
switch (retval) {
|
|
|
|
case -ERESTARTNOHAND:
|
|
|
|
case -ERESTARTSYS:
|
|
|
|
case -ERESTARTNOINTR:
|
|
|
|
case -ERESTART_RESTARTBLOCK:
|
|
|
|
regs->regs[0] = regs->orig_x0;
|
|
|
|
regs->pc = restart_addr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the signal to deliver. When running under ptrace, at this point
|
|
|
|
* the debugger may change all of our registers.
|
|
|
|
*/
|
2013-10-07 04:52:44 +08:00
|
|
|
if (get_signal(&ksig)) {
|
2012-03-05 19:49:31 +08:00
|
|
|
/*
|
|
|
|
* Depending on the signal settings, we may need to revert the
|
|
|
|
* decision to restart the system call, but skip this if a
|
|
|
|
* debugger has chosen to restart at a different PC.
|
|
|
|
*/
|
|
|
|
if (regs->pc == restart_addr &&
|
|
|
|
(retval == -ERESTARTNOHAND ||
|
|
|
|
retval == -ERESTART_RESTARTBLOCK ||
|
|
|
|
(retval == -ERESTARTSYS &&
|
2013-10-07 04:52:44 +08:00
|
|
|
!(ksig.ka.sa.sa_flags & SA_RESTART)))) {
|
2012-03-05 19:49:31 +08:00
|
|
|
regs->regs[0] = -EINTR;
|
|
|
|
regs->pc = continue_addr;
|
|
|
|
}
|
|
|
|
|
2013-10-07 04:52:44 +08:00
|
|
|
handle_signal(&ksig, regs);
|
2012-03-05 19:49:31 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Handle restarting a different system call. As above, if a debugger
|
|
|
|
* has chosen to restart at a different PC, ignore the restart.
|
|
|
|
*/
|
2017-08-01 22:35:54 +08:00
|
|
|
if (in_syscall(regs) && regs->pc == restart_addr) {
|
2012-03-05 19:49:31 +08:00
|
|
|
if (retval == -ERESTART_RESTARTBLOCK)
|
|
|
|
setup_restart_syscall(regs);
|
|
|
|
user_rewind_single_step(current);
|
|
|
|
}
|
|
|
|
|
|
|
|
restore_saved_sigmask();
|
|
|
|
}
|
|
|
|
|
|
|
|
asmlinkage void do_notify_resume(struct pt_regs *regs,
|
|
|
|
unsigned int thread_flags)
|
|
|
|
{
|
2016-07-15 04:48:14 +08:00
|
|
|
/*
|
|
|
|
* The assembly code enters us with IRQs off, but it hasn't
|
|
|
|
* informed the tracing code of that for efficiency reasons.
|
|
|
|
* Update the trace code with the current status.
|
|
|
|
*/
|
|
|
|
trace_hardirqs_off();
|
2017-06-15 09:12:03 +08:00
|
|
|
|
2016-07-15 04:48:14 +08:00
|
|
|
do {
|
2017-09-07 23:30:47 +08:00
|
|
|
/* Check valid user FS if needed */
|
|
|
|
addr_limit_user_check();
|
|
|
|
|
2016-07-15 04:48:14 +08:00
|
|
|
if (thread_flags & _TIF_NEED_RESCHED) {
|
|
|
|
schedule();
|
|
|
|
} else {
|
|
|
|
local_irq_enable();
|
|
|
|
|
2016-11-02 17:10:46 +08:00
|
|
|
if (thread_flags & _TIF_UPROBE)
|
|
|
|
uprobe_notify_resume(regs);
|
|
|
|
|
2016-07-15 04:48:14 +08:00
|
|
|
if (thread_flags & _TIF_SIGPENDING)
|
|
|
|
do_signal(regs);
|
|
|
|
|
|
|
|
if (thread_flags & _TIF_NOTIFY_RESUME) {
|
|
|
|
clear_thread_flag(TIF_NOTIFY_RESUME);
|
|
|
|
tracehook_notify_resume(regs);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (thread_flags & _TIF_FOREIGN_FPSTATE)
|
|
|
|
fpsimd_restore_current_state();
|
|
|
|
}
|
2014-05-08 17:20:23 +08:00
|
|
|
|
2016-07-15 04:48:14 +08:00
|
|
|
local_irq_disable();
|
|
|
|
thread_flags = READ_ONCE(current_thread_info()->flags);
|
|
|
|
} while (thread_flags & _TIF_WORK_MASK);
|
2012-03-05 19:49:31 +08:00
|
|
|
}
|