Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
ab639f35
Commit
ab639f35
authored
Feb 13, 2009
by
Ingo Molnar
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'core/percpu' into x86/core
parents
f8a6b2b9
58105ef1
Changes
51
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
51 changed files
with
949 additions
and
612 deletions
+949
-612
Makefile
Makefile
+2
-1
arch/ia64/include/asm/percpu.h
arch/ia64/include/asm/percpu.h
+2
-2
arch/ia64/include/asm/uv/uv.h
arch/ia64/include/asm/uv/uv.h
+13
-0
arch/x86/Kconfig
arch/x86/Kconfig
+4
-1
arch/x86/Makefile
arch/x86/Makefile
+10
-7
arch/x86/include/asm/a.out-core.h
arch/x86/include/asm/a.out-core.h
+1
-1
arch/x86/include/asm/elf.h
arch/x86/include/asm/elf.h
+13
-2
arch/x86/include/asm/mmu_context.h
arch/x86/include/asm/mmu_context.h
+1
-1
arch/x86/include/asm/percpu.h
arch/x86/include/asm/percpu.h
+22
-0
arch/x86/include/asm/processor.h
arch/x86/include/asm/processor.h
+6
-0
arch/x86/include/asm/ptrace.h
arch/x86/include/asm/ptrace.h
+2
-2
arch/x86/include/asm/segment.h
arch/x86/include/asm/segment.h
+8
-1
arch/x86/include/asm/stackprotector.h
arch/x86/include/asm/stackprotector.h
+91
-5
arch/x86/include/asm/syscalls.h
arch/x86/include/asm/syscalls.h
+10
-10
arch/x86/include/asm/system.h
arch/x86/include/asm/system.h
+38
-0
arch/x86/include/asm/traps.h
arch/x86/include/asm/traps.h
+1
-1
arch/x86/include/asm/uaccess.h
arch/x86/include/asm/uaccess.h
+26
-7
arch/x86/include/asm/uv/uv.h
arch/x86/include/asm/uv/uv.h
+3
-0
arch/x86/kernel/asm-offsets_32.c
arch/x86/kernel/asm-offsets_32.c
+1
-0
arch/x86/kernel/cpu/common.c
arch/x86/kernel/cpu/common.c
+12
-11
arch/x86/kernel/entry_32.S
arch/x86/kernel/entry_32.S
+283
-160
arch/x86/kernel/head_32.S
arch/x86/kernel/head_32.S
+20
-1
arch/x86/kernel/head_64.S
arch/x86/kernel/head_64.S
+2
-19
arch/x86/kernel/ioport.c
arch/x86/kernel/ioport.c
+1
-2
arch/x86/kernel/process_32.c
arch/x86/kernel/process_32.c
+32
-21
arch/x86/kernel/process_64.c
arch/x86/kernel/process_64.c
+5
-6
arch/x86/kernel/ptrace.c
arch/x86/kernel/ptrace.c
+7
-12
arch/x86/kernel/setup_percpu.c
arch/x86/kernel/setup_percpu.c
+2
-0
arch/x86/kernel/signal.c
arch/x86/kernel/signal.c
+20
-55
arch/x86/kernel/syscall_table_32.S
arch/x86/kernel/syscall_table_32.S
+10
-10
arch/x86/kernel/traps.c
arch/x86/kernel/traps.c
+5
-4
arch/x86/kernel/vm86_32.c
arch/x86/kernel/vm86_32.c
+10
-10
arch/x86/kernel/vmlinux_64.lds.S
arch/x86/kernel/vmlinux_64.lds.S
+8
-0
arch/x86/lguest/boot.c
arch/x86/lguest/boot.c
+1
-1
arch/x86/math-emu/get_address.c
arch/x86/math-emu/get_address.c
+2
-4
arch/x86/mm/numa_64.c
arch/x86/mm/numa_64.c
+2
-2
arch/x86/vdso/Makefile
arch/x86/vdso/Makefile
+1
-1
arch/x86/xen/enlighten.c
arch/x86/xen/enlighten.c
+9
-8
arch/x86/xen/xen-asm.S
arch/x86/xen/xen-asm.S
+40
-38
arch/x86/xen/xen-asm_32.S
arch/x86/xen/xen-asm_32.S
+124
-114
arch/x86/xen/xen-asm_64.S
arch/x86/xen/xen-asm_64.S
+55
-52
drivers/misc/sgi-gru/gru.h
drivers/misc/sgi-gru/gru.h
+0
-2
drivers/misc/sgi-gru/grufile.c
drivers/misc/sgi-gru/grufile.c
+3
-15
drivers/misc/sgi-xp/xp.h
drivers/misc/sgi-xp/xp.h
+8
-14
include/linux/elfcore.h
include/linux/elfcore.h
+9
-0
include/linux/percpu.h
include/linux/percpu.h
+7
-1
kernel/kexec.c
kernel/kexec.c
+1
-1
kernel/panic.c
kernel/panic.c
+0
-4
scripts/gcc-x86_32-has-stack-protector.sh
scripts/gcc-x86_32-has-stack-protector.sh
+8
-0
scripts/gcc-x86_64-has-stack-protector.sh
scripts/gcc-x86_64-has-stack-protector.sh
+4
-2
scripts/mod/modpost.c
scripts/mod/modpost.c
+4
-1
No files found.
Makefile
View file @
ab639f35
...
@@ -532,8 +532,9 @@ KBUILD_CFLAGS += $(call cc-option,-Wframe-larger-than=${CONFIG_FRAME_WARN})
...
@@ -532,8 +532,9 @@ KBUILD_CFLAGS += $(call cc-option,-Wframe-larger-than=${CONFIG_FRAME_WARN})
endif
endif
# Force gcc to behave correct even for buggy distributions
# Force gcc to behave correct even for buggy distributions
# Arch Makefiles may override this setting
ifndef
CONFIG_CC_STACKPROTECTOR
KBUILD_CFLAGS
+=
$(
call
cc-option,
-fno-stack-protector
)
KBUILD_CFLAGS
+=
$(
call
cc-option,
-fno-stack-protector
)
endif
ifdef
CONFIG_FRAME_POINTER
ifdef
CONFIG_FRAME_POINTER
KBUILD_CFLAGS
+=
-fno-omit-frame-pointer
-fno-optimize-sibling-calls
KBUILD_CFLAGS
+=
-fno-omit-frame-pointer
-fno-optimize-sibling-calls
...
...
arch/ia64/include/asm/percpu.h
View file @
ab639f35
...
@@ -27,12 +27,12 @@ extern void *per_cpu_init(void);
...
@@ -27,12 +27,12 @@ extern void *per_cpu_init(void);
#else
/* ! SMP */
#else
/* ! SMP */
#define PER_CPU_ATTRIBUTES __attribute__((__section__(".data.percpu")))
#define per_cpu_init() (__phys_per_cpu_start)
#define per_cpu_init() (__phys_per_cpu_start)
#endif
/* SMP */
#endif
/* SMP */
#define PER_CPU_BASE_SECTION ".data.percpu"
/*
/*
* Be extremely careful when taking the address of this variable! Due to virtual
* Be extremely careful when taking the address of this variable! Due to virtual
* remapping, it is different from the canonical address returned by __get_cpu_var(var)!
* remapping, it is different from the canonical address returned by __get_cpu_var(var)!
...
...
arch/ia64/include/asm/uv/uv.h
0 → 100644
View file @
ab639f35
#ifndef _ASM_IA64_UV_UV_H
#define _ASM_IA64_UV_UV_H
#include <asm/system.h>
#include <asm/sn/simulator.h>
static
inline
int
is_uv_system
(
void
)
{
/* temporary support for running on hardware simulator */
return
IS_MEDUSA
()
||
ia64_platform_is
(
"uv"
);
}
#endif
/* _ASM_IA64_UV_UV_H */
arch/x86/Kconfig
View file @
ab639f35
...
@@ -194,6 +194,10 @@ config X86_TRAMPOLINE
...
@@ -194,6 +194,10 @@ config X86_TRAMPOLINE
depends on SMP || (64BIT && ACPI_SLEEP)
depends on SMP || (64BIT && ACPI_SLEEP)
default y
default y
config X86_32_LAZY_GS
def_bool y
depends on X86_32 && !CC_STACKPROTECTOR
config KTIME_SCALAR
config KTIME_SCALAR
def_bool X86_32
def_bool X86_32
source "init/Kconfig"
source "init/Kconfig"
...
@@ -1339,7 +1343,6 @@ config CC_STACKPROTECTOR_ALL
...
@@ -1339,7 +1343,6 @@ config CC_STACKPROTECTOR_ALL
config CC_STACKPROTECTOR
config CC_STACKPROTECTOR
bool "Enable -fstack-protector buffer overflow detection (EXPERIMENTAL)"
bool "Enable -fstack-protector buffer overflow detection (EXPERIMENTAL)"
depends on X86_64
select CC_STACKPROTECTOR_ALL
select CC_STACKPROTECTOR_ALL
---help---
---help---
This option turns on the -fstack-protector GCC feature. This
This option turns on the -fstack-protector GCC feature. This
...
...
arch/x86/Makefile
View file @
ab639f35
...
@@ -70,14 +70,17 @@ else
...
@@ -70,14 +70,17 @@ else
# this works around some issues with generating unwind tables in older gccs
# this works around some issues with generating unwind tables in older gccs
# newer gccs do it by default
# newer gccs do it by default
KBUILD_CFLAGS
+=
-maccumulate-outgoing-args
KBUILD_CFLAGS
+=
-maccumulate-outgoing-args
endif
stackp
:=
$(CONFIG_SHELL)
$(srctree)
/scripts/gcc-x86_64-has-stack-protector.sh
ifdef
CONFIG_CC_STACKPROTECTOR
stackp-$(CONFIG_CC_STACKPROTECTOR)
:=
$(
shell
$(stackp)
\
cc_has_sp
:=
$(srctree)
/scripts/gcc-x86_
$(BITS)
-has-stack-protector
.sh
"
$(CC)
"
"-fstack-protector -DGCC_HAS_SP"
)
ifeq
($(shell $(CONFIG_SHELL) $(cc_has_sp) $(CC)),y)
stackp-$(CONFIG_CC_STACKPROTECTOR_ALL)
+=
$(
shell
$(stackp)
\
stackp-y
:=
-fstack-protector
"
$(CC)
"
-fstack-protector-all
)
stackp-$(CONFIG_CC_STACKPROTECTOR_ALL)
+=
-fstack-protector-all
KBUILD_CFLAGS
+=
$
(
stackp-y
)
KBUILD_CFLAGS
+=
$
(
stackp-y
)
else
$(warning
stack
protector
enabled
but
no
compiler
support)
endif
endif
endif
# Stackpointer is addressed different for 32 bit and 64 bit x86
# Stackpointer is addressed different for 32 bit and 64 bit x86
...
...
arch/x86/include/asm/a.out-core.h
View file @
ab639f35
...
@@ -55,7 +55,7 @@ static inline void aout_dump_thread(struct pt_regs *regs, struct user *dump)
...
@@ -55,7 +55,7 @@ static inline void aout_dump_thread(struct pt_regs *regs, struct user *dump)
dump
->
regs
.
ds
=
(
u16
)
regs
->
ds
;
dump
->
regs
.
ds
=
(
u16
)
regs
->
ds
;
dump
->
regs
.
es
=
(
u16
)
regs
->
es
;
dump
->
regs
.
es
=
(
u16
)
regs
->
es
;
dump
->
regs
.
fs
=
(
u16
)
regs
->
fs
;
dump
->
regs
.
fs
=
(
u16
)
regs
->
fs
;
savesegment
(
gs
,
dump
->
regs
.
gs
);
dump
->
regs
.
gs
=
get_user_gs
(
re
gs
);
dump
->
regs
.
orig_ax
=
regs
->
orig_ax
;
dump
->
regs
.
orig_ax
=
regs
->
orig_ax
;
dump
->
regs
.
ip
=
regs
->
ip
;
dump
->
regs
.
ip
=
regs
->
ip
;
dump
->
regs
.
cs
=
(
u16
)
regs
->
cs
;
dump
->
regs
.
cs
=
(
u16
)
regs
->
cs
;
...
...
arch/x86/include/asm/elf.h
View file @
ab639f35
...
@@ -112,7 +112,7 @@ extern unsigned int vdso_enabled;
...
@@ -112,7 +112,7 @@ extern unsigned int vdso_enabled;
* now struct_user_regs, they are different)
* now struct_user_regs, they are different)
*/
*/
#define ELF_CORE_COPY_REGS(pr_reg, regs) \
#define ELF_CORE_COPY_REGS
_COMMON
(pr_reg, regs) \
do { \
do { \
pr_reg[0] = regs->bx; \
pr_reg[0] = regs->bx; \
pr_reg[1] = regs->cx; \
pr_reg[1] = regs->cx; \
...
@@ -124,7 +124,6 @@ do { \
...
@@ -124,7 +124,6 @@ do { \
pr_reg[7] = regs->ds & 0xffff; \
pr_reg[7] = regs->ds & 0xffff; \
pr_reg[8] = regs->es & 0xffff; \
pr_reg[8] = regs->es & 0xffff; \
pr_reg[9] = regs->fs & 0xffff; \
pr_reg[9] = regs->fs & 0xffff; \
savesegment(gs, pr_reg[10]); \
pr_reg[11] = regs->orig_ax; \
pr_reg[11] = regs->orig_ax; \
pr_reg[12] = regs->ip; \
pr_reg[12] = regs->ip; \
pr_reg[13] = regs->cs & 0xffff; \
pr_reg[13] = regs->cs & 0xffff; \
...
@@ -133,6 +132,18 @@ do { \
...
@@ -133,6 +132,18 @@ do { \
pr_reg[16] = regs->ss & 0xffff; \
pr_reg[16] = regs->ss & 0xffff; \
} while (0);
} while (0);
#define ELF_CORE_COPY_REGS(pr_reg, regs) \
do { \
ELF_CORE_COPY_REGS_COMMON(pr_reg, regs);\
pr_reg[10] = get_user_gs(regs); \
} while (0);
#define ELF_CORE_COPY_KERNEL_REGS(pr_reg, regs) \
do { \
ELF_CORE_COPY_REGS_COMMON(pr_reg, regs);\
savesegment(gs, pr_reg[10]); \
} while (0);
#define ELF_PLATFORM (utsname()->machine)
#define ELF_PLATFORM (utsname()->machine)
#define set_personality_64bit() do { } while (0)
#define set_personality_64bit() do { } while (0)
...
...
arch/x86/include/asm/mmu_context.h
View file @
ab639f35
...
@@ -79,7 +79,7 @@ do { \
...
@@ -79,7 +79,7 @@ do { \
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
#define deactivate_mm(tsk, mm) \
#define deactivate_mm(tsk, mm) \
do { \
do { \
l
oadsegment(gs,
0); \
l
azy_load_gs(
0); \
} while (0)
} while (0)
#else
#else
#define deactivate_mm(tsk, mm) \
#define deactivate_mm(tsk, mm) \
...
...
arch/x86/include/asm/percpu.h
View file @
ab639f35
...
@@ -34,6 +34,12 @@
...
@@ -34,6 +34,12 @@
#define PER_CPU_VAR(var) per_cpu__##var
#define PER_CPU_VAR(var) per_cpu__##var
#endif
/* SMP */
#endif
/* SMP */
#ifdef CONFIG_X86_64_SMP
#define INIT_PER_CPU_VAR(var) init_per_cpu__##var
#else
#define INIT_PER_CPU_VAR(var) per_cpu__##var
#endif
#else
/* ...!ASSEMBLY */
#else
/* ...!ASSEMBLY */
#include <linux/stringify.h>
#include <linux/stringify.h>
...
@@ -45,6 +51,22 @@
...
@@ -45,6 +51,22 @@
#define __percpu_arg(x) "%" #x
#define __percpu_arg(x) "%" #x
#endif
#endif
/*
* Initialized pointers to per-cpu variables needed for the boot
* processor need to use these macros to get the proper address
* offset from __per_cpu_load on SMP.
*
* There also must be an entry in vmlinux_64.lds.S
*/
#define DECLARE_INIT_PER_CPU(var) \
extern typeof(per_cpu_var(var)) init_per_cpu_var(var)
#ifdef CONFIG_X86_64_SMP
#define init_per_cpu_var(var) init_per_cpu__##var
#else
#define init_per_cpu_var(var) per_cpu_var(var)
#endif
/* For arch-specific code, we can use direct single-insn ops (they
/* For arch-specific code, we can use direct single-insn ops (they
* don't give an lvalue though). */
* don't give an lvalue though). */
extern
void
__bad_percpu_size
(
void
);
extern
void
__bad_percpu_size
(
void
);
...
...
arch/x86/include/asm/processor.h
View file @
ab639f35
...
@@ -393,8 +393,14 @@ union irq_stack_union {
...
@@ -393,8 +393,14 @@ union irq_stack_union {
};
};
DECLARE_PER_CPU
(
union
irq_stack_union
,
irq_stack_union
);
DECLARE_PER_CPU
(
union
irq_stack_union
,
irq_stack_union
);
DECLARE_INIT_PER_CPU
(
irq_stack_union
);
DECLARE_PER_CPU
(
char
*
,
irq_stack_ptr
);
DECLARE_PER_CPU
(
char
*
,
irq_stack_ptr
);
#else
/* X86_64 */
#ifdef CONFIG_CC_STACKPROTECTOR
DECLARE_PER_CPU
(
unsigned
long
,
stack_canary
);
#endif
#endif
#endif
/* X86_64 */
extern
void
print_cpu_info
(
struct
cpuinfo_x86
*
);
extern
void
print_cpu_info
(
struct
cpuinfo_x86
*
);
extern
unsigned
int
xstate_size
;
extern
unsigned
int
xstate_size
;
...
...
arch/x86/include/asm/ptrace.h
View file @
ab639f35
...
@@ -28,7 +28,7 @@ struct pt_regs {
...
@@ -28,7 +28,7 @@ struct pt_regs {
int
xds
;
int
xds
;
int
xes
;
int
xes
;
int
xfs
;
int
xfs
;
/* int gs; */
int
xgs
;
long
orig_eax
;
long
orig_eax
;
long
eip
;
long
eip
;
int
xcs
;
int
xcs
;
...
@@ -50,7 +50,7 @@ struct pt_regs {
...
@@ -50,7 +50,7 @@ struct pt_regs {
unsigned
long
ds
;
unsigned
long
ds
;
unsigned
long
es
;
unsigned
long
es
;
unsigned
long
fs
;
unsigned
long
fs
;
/* int gs; */
unsigned
long
gs
;
unsigned
long
orig_ax
;
unsigned
long
orig_ax
;
unsigned
long
ip
;
unsigned
long
ip
;
unsigned
long
cs
;
unsigned
long
cs
;
...
...
arch/x86/include/asm/segment.h
View file @
ab639f35
...
@@ -61,7 +61,7 @@
...
@@ -61,7 +61,7 @@
*
*
* 26 - ESPFIX small SS
* 26 - ESPFIX small SS
* 27 - per-cpu [ offset to per-cpu data area ]
* 27 - per-cpu [ offset to per-cpu data area ]
* 28 -
unused
* 28 -
stack_canary-20 [ for stack protector ]
* 29 - unused
* 29 - unused
* 30 - unused
* 30 - unused
* 31 - TSS for double fault handler
* 31 - TSS for double fault handler
...
@@ -95,6 +95,13 @@
...
@@ -95,6 +95,13 @@
#define __KERNEL_PERCPU 0
#define __KERNEL_PERCPU 0
#endif
#endif
#define GDT_ENTRY_STACK_CANARY (GDT_ENTRY_KERNEL_BASE + 16)
#ifdef CONFIG_CC_STACKPROTECTOR
#define __KERNEL_STACK_CANARY (GDT_ENTRY_STACK_CANARY * 8)
#else
#define __KERNEL_STACK_CANARY 0
#endif
#define GDT_ENTRY_DOUBLEFAULT_TSS 31
#define GDT_ENTRY_DOUBLEFAULT_TSS 31
/*
/*
...
...
arch/x86/include/asm/stackprotector.h
View file @
ab639f35
/*
* GCC stack protector support.
*
* Stack protector works by putting predefined pattern at the start of
* the stack frame and verifying that it hasn't been overwritten when
* returning from the function. The pattern is called stack canary
* and unfortunately gcc requires it to be at a fixed offset from %gs.
* On x86_64, the offset is 40 bytes and on x86_32 20 bytes. x86_64
* and x86_32 use segment registers differently and thus handles this
* requirement differently.
*
* On x86_64, %gs is shared by percpu area and stack canary. All
* percpu symbols are zero based and %gs points to the base of percpu
* area. The first occupant of the percpu area is always
* irq_stack_union which contains stack_canary at offset 40. Userland
* %gs is always saved and restored on kernel entry and exit using
* swapgs, so stack protector doesn't add any complexity there.
*
* On x86_32, it's slightly more complicated. As in x86_64, %gs is
* used for userland TLS. Unfortunately, some processors are much
* slower at loading segment registers with different value when
* entering and leaving the kernel, so the kernel uses %fs for percpu
* area and manages %gs lazily so that %gs is switched only when
* necessary, usually during task switch.
*
* As gcc requires the stack canary at %gs:20, %gs can't be managed
* lazily if stack protector is enabled, so the kernel saves and
* restores userland %gs on kernel entry and exit. This behavior is
* controlled by CONFIG_X86_32_LAZY_GS and accessors are defined in
* system.h to hide the details.
*/
#ifndef _ASM_STACKPROTECTOR_H
#ifndef _ASM_STACKPROTECTOR_H
#define _ASM_STACKPROTECTOR_H 1
#define _ASM_STACKPROTECTOR_H 1
#ifdef CONFIG_CC_STACKPROTECTOR
#include <asm/tsc.h>
#include <asm/tsc.h>
#include <asm/processor.h>
#include <asm/processor.h>
#include <asm/percpu.h>
#include <asm/system.h>
#include <asm/desc.h>
#include <linux/random.h>
/*
* 24 byte read-only segment initializer for stack canary. Linker
* can't handle the address bit shifting. Address will be set in
* head_32 for boot CPU and setup_per_cpu_areas() for others.
*/
#define GDT_STACK_CANARY_INIT \
[GDT_ENTRY_STACK_CANARY] = { { { 0x00000018, 0x00409000 } } },
/*
/*
* Initialize the stackprotector canary value.
* Initialize the stackprotector canary value.
...
@@ -15,12 +61,9 @@ static __always_inline void boot_init_stack_canary(void)
...
@@ -15,12 +61,9 @@ static __always_inline void boot_init_stack_canary(void)
u64
canary
;
u64
canary
;
u64
tsc
;
u64
tsc
;
/*
#ifdef CONFIG_X86_64
* Build time only check to make sure the stack_canary is at
* offset 40 in the pda; this is a gcc ABI requirement
*/
BUILD_BUG_ON
(
offsetof
(
union
irq_stack_union
,
stack_canary
)
!=
40
);
BUILD_BUG_ON
(
offsetof
(
union
irq_stack_union
,
stack_canary
)
!=
40
);
#endif
/*
/*
* We both use the random pool and the current TSC as a source
* We both use the random pool and the current TSC as a source
* of randomness. The TSC only matters for very early init,
* of randomness. The TSC only matters for very early init,
...
@@ -32,7 +75,50 @@ static __always_inline void boot_init_stack_canary(void)
...
@@ -32,7 +75,50 @@ static __always_inline void boot_init_stack_canary(void)
canary
+=
tsc
+
(
tsc
<<
32UL
);
canary
+=
tsc
+
(
tsc
<<
32UL
);
current
->
stack_canary
=
canary
;
current
->
stack_canary
=
canary
;
#ifdef CONFIG_X86_64
percpu_write
(
irq_stack_union
.
stack_canary
,
canary
);
percpu_write
(
irq_stack_union
.
stack_canary
,
canary
);
#else
percpu_write
(
stack_canary
,
canary
);
#endif
}
}
static
inline
void
setup_stack_canary_segment
(
int
cpu
)
{
#ifdef CONFIG_X86_32
unsigned
long
canary
=
(
unsigned
long
)
&
per_cpu
(
stack_canary
,
cpu
)
-
20
;
struct
desc_struct
*
gdt_table
=
get_cpu_gdt_table
(
cpu
);
struct
desc_struct
desc
;
desc
=
gdt_table
[
GDT_ENTRY_STACK_CANARY
];
desc
.
base0
=
canary
&
0xffff
;
desc
.
base1
=
(
canary
>>
16
)
&
0xff
;
desc
.
base2
=
(
canary
>>
24
)
&
0xff
;
write_gdt_entry
(
gdt_table
,
GDT_ENTRY_STACK_CANARY
,
&
desc
,
DESCTYPE_S
);
#endif
}
static
inline
void
load_stack_canary_segment
(
void
)
{
#ifdef CONFIG_X86_32
asm
(
"mov %0, %%gs"
:
:
"r"
(
__KERNEL_STACK_CANARY
)
:
"memory"
);
#endif
}
#else
/* CC_STACKPROTECTOR */
#define GDT_STACK_CANARY_INIT
/* dummy boot_init_stack_canary() is defined in linux/stackprotector.h */
static
inline
void
setup_stack_canary_segment
(
int
cpu
)
{
}
static
inline
void
load_stack_canary_segment
(
void
)
{
#ifdef CONFIG_X86_32
asm
volatile
(
"mov %0, %%gs"
:
:
"r"
(
0
));
#endif
#endif
}
#endif
/* CC_STACKPROTECTOR */
#endif
/* _ASM_STACKPROTECTOR_H */
arch/x86/include/asm/syscalls.h
View file @
ab639f35
...
@@ -29,21 +29,21 @@ asmlinkage int sys_get_thread_area(struct user_desc __user *);
...
@@ -29,21 +29,21 @@ asmlinkage int sys_get_thread_area(struct user_desc __user *);
/* X86_32 only */
/* X86_32 only */
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
/* kernel/process_32.c */
/* kernel/process_32.c */
asmlinkage
int
sys_fork
(
struct
pt_regs
);
int
sys_fork
(
struct
pt_regs
*
);
asmlinkage
int
sys_clone
(
struct
pt_regs
);
int
sys_clone
(
struct
pt_regs
*
);
asmlinkage
int
sys_vfork
(
struct
pt_regs
);
int
sys_vfork
(
struct
pt_regs
*
);
asmlinkage
int
sys_execve
(
struct
pt_regs
);
int
sys_execve
(
struct
pt_regs
*
);
/* kernel/signal_32.c */
/* kernel/signal_32.c */
asmlinkage
int
sys_sigsuspend
(
int
,
int
,
old_sigset_t
);
asmlinkage
int
sys_sigsuspend
(
int
,
int
,
old_sigset_t
);
asmlinkage
int
sys_sigaction
(
int
,
const
struct
old_sigaction
__user
*
,
asmlinkage
int
sys_sigaction
(
int
,
const
struct
old_sigaction
__user
*
,
struct
old_sigaction
__user
*
);
struct
old_sigaction
__user
*
);
asmlinkage
int
sys_sigaltstack
(
unsigned
long
);
int
sys_sigaltstack
(
struct
pt_regs
*
);
asmlinkage
unsigned
long
sys_sigreturn
(
unsigned
long
);
unsigned
long
sys_sigreturn
(
struct
pt_regs
*
);
asmlinkage
int
sys_rt_sigreturn
(
unsigned
long
);
long
sys_rt_sigreturn
(
struct
pt_regs
*
);
/* kernel/ioport.c */
/* kernel/ioport.c */
asmlinkage
long
sys_iopl
(
unsigned
long
);
long
sys_iopl
(
struct
pt_regs
*
);
/* kernel/sys_i386_32.c */
/* kernel/sys_i386_32.c */
asmlinkage
long
sys_mmap2
(
unsigned
long
,
unsigned
long
,
unsigned
long
,
asmlinkage
long
sys_mmap2
(
unsigned
long
,
unsigned
long
,
unsigned
long
,
...
@@ -59,8 +59,8 @@ struct oldold_utsname;
...
@@ -59,8 +59,8 @@ struct oldold_utsname;
asmlinkage
int
sys_olduname
(
struct
oldold_utsname
__user
*
);
asmlinkage
int
sys_olduname
(
struct
oldold_utsname
__user
*
);
/* kernel/vm86_32.c */
/* kernel/vm86_32.c */
asmlinkage
int
sys_vm86old
(
struct
pt_regs
);
int
sys_vm86old
(
struct
pt_regs
*
);
asmlinkage
int
sys_vm86
(
struct
pt_regs
);
int
sys_vm86
(
struct
pt_regs
*
);
#else
/* CONFIG_X86_32 */
#else
/* CONFIG_X86_32 */
...
...
arch/x86/include/asm/system.h
View file @
ab639f35
...
@@ -23,6 +23,20 @@ struct task_struct *__switch_to(struct task_struct *prev,
...
@@ -23,6 +23,20 @@ struct task_struct *__switch_to(struct task_struct *prev,
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
#ifdef CONFIG_CC_STACKPROTECTOR
#define __switch_canary \
"movl %P[task_canary](%[next]), %%ebx\n\t" \
"movl %%ebx, "__percpu_arg([stack_canary])"\n\t"
#define __switch_canary_oparam \
, [stack_canary] "=m" (per_cpu_var(stack_canary))
#define __switch_canary_iparam \
, [task_canary] "i" (offsetof(struct task_struct, stack_canary))
#else
/* CC_STACKPROTECTOR */
#define __switch_canary
#define __switch_canary_oparam
#define __switch_canary_iparam
#endif
/* CC_STACKPROTECTOR */
/*
/*
* Saving eflags is important. It switches not only IOPL between tasks,
* Saving eflags is important. It switches not only IOPL between tasks,
* it also protects other tasks from NT leaking through sysenter etc.
* it also protects other tasks from NT leaking through sysenter etc.
...
@@ -44,6 +58,7 @@ do { \
...
@@ -44,6 +58,7 @@ do { \
"movl %[next_sp],%%esp\n\t"
/* restore ESP */
\
"movl %[next_sp],%%esp\n\t"
/* restore ESP */
\
"movl $1f,%[prev_ip]\n\t"
/* save EIP */
\
"movl $1f,%[prev_ip]\n\t"
/* save EIP */
\
"pushl %[next_ip]\n\t"
/* restore EIP */
\
"pushl %[next_ip]\n\t"
/* restore EIP */
\
__switch_canary \
"jmp __switch_to\n"
/* regparm call */
\
"jmp __switch_to\n"
/* regparm call */
\
"1:\t" \
"1:\t" \
"popl %%ebp\n\t"
/* restore EBP */
\
"popl %%ebp\n\t"
/* restore EBP */
\
...
@@ -58,6 +73,8 @@ do { \
...
@@ -58,6 +73,8 @@ do { \
"=b" (ebx), "=c" (ecx), "=d" (edx), \
"=b" (ebx), "=c" (ecx), "=d" (edx), \
"=S" (esi), "=D" (edi) \
"=S" (esi), "=D" (edi) \
\
\
__switch_canary_oparam \
\
/* input parameters: */
\
/* input parameters: */
\
: [next_sp] "m" (next->thread.sp), \
: [next_sp] "m" (next->thread.sp), \
[next_ip] "m" (next->thread.ip), \
[next_ip] "m" (next->thread.ip), \
...
@@ -66,6 +83,8 @@ do { \
...
@@ -66,6 +83,8 @@ do { \
[prev] "a" (prev), \
[prev] "a" (prev), \
[next] "d" (next) \
[next] "d" (next) \
\
\
__switch_canary_iparam \
\
:
/* reloaded segment registers */
\
:
/* reloaded segment registers */
\
"memory"); \
"memory"); \
} while (0)
} while (0)
...
@@ -182,6 +201,25 @@ extern void native_load_gs_index(unsigned);
...
@@ -182,6 +201,25 @@ extern void native_load_gs_index(unsigned);
#define savesegment(seg, value) \
#define savesegment(seg, value) \
asm("mov %%" #seg ",%0":"=r" (value) : : "memory")
asm("mov %%" #seg ",%0":"=r" (value) : : "memory")
/*
* x86_32 user gs accessors.
*/
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32_LAZY_GS
#define get_user_gs(regs) (u16)({unsigned long v; savesegment(gs, v); v;})
#define set_user_gs(regs, v) loadsegment(gs, (unsigned long)(v))
#define task_user_gs(tsk) ((tsk)->thread.gs)
#define lazy_save_gs(v) savesegment(gs, (v))
#define lazy_load_gs(v) loadsegment(gs, (v))
#else
/* X86_32_LAZY_GS */
#define get_user_gs(regs) (u16)((regs)->gs)
#define set_user_gs(regs, v) do { (regs)->gs = (v); } while (0)
#define task_user_gs(tsk) (task_pt_regs(tsk)->gs)
#define lazy_save_gs(v) do { } while (0)
#define lazy_load_gs(v) do { } while (0)
#endif
/* X86_32_LAZY_GS */
#endif
/* X86_32 */
static
inline
unsigned
long
get_limit
(
unsigned
long
segment
)
static
inline
unsigned
long
get_limit
(
unsigned
long
segment
)
{
{
unsigned
long
__limit
;
unsigned
long
__limit
;
...
...
arch/x86/include/asm/traps.h
View file @
ab639f35
...
@@ -41,7 +41,7 @@ dotraplinkage void do_int3(struct pt_regs *, long);
...
@@ -41,7 +41,7 @@ dotraplinkage void do_int3(struct pt_regs *, long);
dotraplinkage
void
do_overflow
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_overflow
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_bounds
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_bounds
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_invalid_op
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_invalid_op
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_device_not_available
(
struct
pt_regs
);
dotraplinkage
void
do_device_not_available
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_coprocessor_segment_overrun
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_coprocessor_segment_overrun
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_invalid_TSS
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_invalid_TSS
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_segment_not_present
(
struct
pt_regs
*
,
long
);
dotraplinkage
void
do_segment_not_present
(
struct
pt_regs
*
,
long
);
...
...
arch/x86/include/asm/uaccess.h
View file @
ab639f35
...
@@ -186,7 +186,7 @@ extern int __get_user_bad(void);
...
@@ -186,7 +186,7 @@ extern int __get_user_bad(void);
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
#define __put_user_asm_u64(x, addr, err
)
\
#define __put_user_asm_u64(x, addr, err
, errret)
\
asm volatile("1: movl %%eax,0(%2)\n" \
asm volatile("1: movl %%eax,0(%2)\n" \
"2: movl %%edx,4(%2)\n" \
"2: movl %%edx,4(%2)\n" \
"3:\n" \
"3:\n" \
...
@@ -197,7 +197,7 @@ extern int __get_user_bad(void);
...
@@ -197,7 +197,7 @@ extern int __get_user_bad(void);
_ASM_EXTABLE(1b, 4b) \
_ASM_EXTABLE(1b, 4b) \
_ASM_EXTABLE(2b, 4b) \
_ASM_EXTABLE(2b, 4b) \
: "=r" (err) \
: "=r" (err) \
: "A" (x), "r" (addr), "i" (
-EFAULT
), "0" (err))
: "A" (x), "r" (addr), "i" (
errret
), "0" (err))
#define __put_user_asm_ex_u64(x, addr) \
#define __put_user_asm_ex_u64(x, addr) \
asm volatile("1: movl %%eax,0(%1)\n" \
asm volatile("1: movl %%eax,0(%1)\n" \
...
@@ -211,8 +211,8 @@ extern int __get_user_bad(void);
...
@@ -211,8 +211,8 @@ extern int __get_user_bad(void);
asm volatile("call __put_user_8" : "=a" (__ret_pu) \
asm volatile("call __put_user_8" : "=a" (__ret_pu) \
: "A" ((typeof(*(ptr)))(x)), "c" (ptr) : "ebx")
: "A" ((typeof(*(ptr)))(x)), "c" (ptr) : "ebx")
#else
#else
#define __put_user_asm_u64(x, ptr, retval) \
#define __put_user_asm_u64(x, ptr, retval
, errret
) \
__put_user_asm(x, ptr, retval, "q", "", "Zr",
-EFAULT
)
__put_user_asm(x, ptr, retval, "q", "", "Zr",
errret
)
#define __put_user_asm_ex_u64(x, addr) \
#define __put_user_asm_ex_u64(x, addr) \
__put_user_asm_ex(x, addr, "q", "", "Zr")
__put_user_asm_ex(x, addr, "q", "", "Zr")
#define __put_user_x8(x, ptr, __ret_pu) __put_user_x(8, x, ptr, __ret_pu)
#define __put_user_x8(x, ptr, __ret_pu) __put_user_x(8, x, ptr, __ret_pu)
...
@@ -289,7 +289,8 @@ do { \
...
@@ -289,7 +289,8 @@ do { \
__put_user_asm(x, ptr, retval, "l", "k", "ir", errret); \
__put_user_asm(x, ptr, retval, "l", "k", "ir", errret); \
break; \
break; \
case 8: \
case 8: \
__put_user_asm_u64((__typeof__(*ptr))(x), ptr, retval); \
__put_user_asm_u64((__typeof__(*ptr))(x), ptr, retval, \
errret); \
break; \
break; \
default: \
default: \
__put_user_bad(); \
__put_user_bad(); \
...
@@ -525,8 +526,6 @@ struct __large_struct { unsigned long buf[100]; };
...
@@ -525,8 +526,6 @@ struct __large_struct { unsigned long buf[100]; };
*/
*/
#define get_user_try uaccess_try
#define get_user_try uaccess_try
#define get_user_catch(err) uaccess_catch(err)
#define get_user_catch(err) uaccess_catch(err)
#define put_user_try uaccess_try
#define put_user_catch(err) uaccess_catch(err)
#define get_user_ex(x, ptr) do { \
#define get_user_ex(x, ptr) do { \
unsigned long __gue_val; \
unsigned long __gue_val; \
...
@@ -534,9 +533,29 @@ struct __large_struct { unsigned long buf[100]; };
...
@@ -534,9 +533,29 @@ struct __large_struct { unsigned long buf[100]; };
(x) = (__force __typeof__(*(ptr)))__gue_val; \
(x) = (__force __typeof__(*(ptr)))__gue_val; \
} while (0)
} while (0)
#ifdef CONFIG_X86_WP_WORKS_OK
#define put_user_try uaccess_try
#define put_user_catch(err) uaccess_catch(err)
#define put_user_ex(x, ptr) \
#define put_user_ex(x, ptr) \
__put_user_size_ex((__typeof__(*(ptr)))(x), (ptr), sizeof(*(ptr)))
__put_user_size_ex((__typeof__(*(ptr)))(x), (ptr), sizeof(*(ptr)))
#else
/* !CONFIG_X86_WP_WORKS_OK */
#define put_user_try do { \
int __uaccess_err = 0;
#define put_user_catch(err) \
(err) |= __uaccess_err; \
} while (0)
#define put_user_ex(x, ptr) do { \
__uaccess_err |= __put_user(x, ptr); \
} while (0)
#endif
/* CONFIG_X86_WP_WORKS_OK */
/*
/*
* movsl can be slow when source and dest are not both 8-byte aligned
* movsl can be slow when source and dest are not both 8-byte aligned
*/
*/
...
...
arch/x86/include/asm/uv/uv.h
View file @
ab639f35
...
@@ -3,6 +3,9 @@
...
@@ -3,6 +3,9 @@
enum
uv_system_type
{
UV_NONE
,
UV_LEGACY_APIC
,
UV_X2APIC
,
UV_NON_UNIQUE_APIC
};
enum
uv_system_type
{
UV_NONE
,
UV_LEGACY_APIC
,
UV_X2APIC
,
UV_NON_UNIQUE_APIC
};
struct
cpumask
;
struct
mm_struct
;
#ifdef CONFIG_X86_UV
#ifdef CONFIG_X86_UV
extern
enum
uv_system_type
get_uv_system_type
(
void
);
extern
enum
uv_system_type
get_uv_system_type
(
void
);
...
...
arch/x86/kernel/asm-offsets_32.c
View file @
ab639f35
...
@@ -75,6 +75,7 @@ void foo(void)
...
@@ -75,6 +75,7 @@ void foo(void)
OFFSET
(
PT_DS
,
pt_regs
,
ds
);
OFFSET
(
PT_DS
,
pt_regs
,
ds
);
OFFSET
(
PT_ES
,
pt_regs
,
es
);
OFFSET
(
PT_ES
,
pt_regs
,
es
);
OFFSET
(
PT_FS
,
pt_regs
,
fs
);
OFFSET
(
PT_FS
,
pt_regs
,
fs
);
OFFSET
(
PT_GS
,
pt_regs
,
gs
);
OFFSET
(
PT_ORIG_EAX
,
pt_regs
,
orig_ax
);
OFFSET
(
PT_ORIG_EAX
,
pt_regs
,
orig_ax
);
OFFSET
(
PT_EIP
,
pt_regs
,
ip
);
OFFSET
(
PT_EIP
,
pt_regs
,
ip
);
OFFSET
(
PT_CS
,
pt_regs
,
cs
);
OFFSET
(
PT_CS
,
pt_regs
,
cs
);
...
...
arch/x86/kernel/cpu/common.c
View file @
ab639f35
...
@@ -39,6 +39,7 @@
...
@@ -39,6 +39,7 @@
#include <asm/sections.h>
#include <asm/sections.h>
#include <asm/setup.h>
#include <asm/setup.h>
#include <asm/hypervisor.h>
#include <asm/hypervisor.h>
#include <asm/stackprotector.h>
#include "cpu.h"
#include "cpu.h"
...
@@ -122,6 +123,7 @@ DEFINE_PER_CPU_PAGE_ALIGNED(struct gdt_page, gdt_page) = { .gdt = {
...
@@ -122,6 +123,7 @@ DEFINE_PER_CPU_PAGE_ALIGNED(struct gdt_page, gdt_page) = { .gdt = {
[
GDT_ENTRY_ESPFIX_SS
]
=
{
{
{
0x00000000
,
0x00c09200
}
}
},
[
GDT_ENTRY_ESPFIX_SS
]
=
{
{
{
0x00000000
,
0x00c09200
}
}
},
[
GDT_ENTRY_PERCPU
]
=
{
{
{
0x0000ffff
,
0x00cf9200
}
}
},
[
GDT_ENTRY_PERCPU
]
=
{
{
{
0x0000ffff
,
0x00cf9200
}
}
},
GDT_STACK_CANARY_INIT
#endif
#endif
}
};
}
};
EXPORT_PER_CPU_SYMBOL_GPL
(
gdt_page
);
EXPORT_PER_CPU_SYMBOL_GPL
(
gdt_page
);
...
@@ -304,6 +306,7 @@ void load_percpu_segment(int cpu)
...
@@ -304,6 +306,7 @@ void load_percpu_segment(int cpu)
loadsegment
(
gs
,
0
);
loadsegment
(
gs
,
0
);
wrmsrl
(
MSR_GS_BASE
,
(
unsigned
long
)
per_cpu
(
irq_stack_union
.
gs_base
,
cpu
));
wrmsrl
(
MSR_GS_BASE
,
(
unsigned
long
)
per_cpu
(
irq_stack_union
.
gs_base
,
cpu
));
#endif
#endif
load_stack_canary_segment
();
}
}
/* Current gdt points %fs at the "master" per-cpu area: after this,
/* Current gdt points %fs at the "master" per-cpu area: after this,
...
@@ -938,12 +941,8 @@ struct desc_ptr idt_descr = { 256 * 16 - 1, (unsigned long) idt_table };
...
@@ -938,12 +941,8 @@ struct desc_ptr idt_descr = { 256 * 16 - 1, (unsigned long) idt_table };
DEFINE_PER_CPU_FIRST
(
union
irq_stack_union
,
DEFINE_PER_CPU_FIRST
(
union
irq_stack_union
,
irq_stack_union
)
__aligned
(
PAGE_SIZE
);
irq_stack_union
)
__aligned
(
PAGE_SIZE
);
#ifdef CONFIG_SMP
DEFINE_PER_CPU
(
char
*
,
irq_stack_ptr
);
/* will be set during per cpu init */
#else
DEFINE_PER_CPU
(
char
*
,
irq_stack_ptr
)
=
DEFINE_PER_CPU
(
char
*
,
irq_stack_ptr
)
=
per_cpu_var
(
irq_stack_union
.
irq_stack
)
+
IRQ_STACK_SIZE
-
64
;
init_per_cpu_var
(
irq_stack_union
.
irq_stack
)
+
IRQ_STACK_SIZE
-
64
;
#endif
DEFINE_PER_CPU
(
unsigned
long
,
kernel_stack
)
=
DEFINE_PER_CPU
(
unsigned
long
,
kernel_stack
)
=
(
unsigned
long
)
&
init_thread_union
-
KERNEL_STACK_OFFSET
+
THREAD_SIZE
;
(
unsigned
long
)
&
init_thread_union
-
KERNEL_STACK_OFFSET
+
THREAD_SIZE
;
...
@@ -986,16 +985,21 @@ unsigned long kernel_eflags;
...
@@ -986,16 +985,21 @@ unsigned long kernel_eflags;
*/
*/
DEFINE_PER_CPU
(
struct
orig_ist
,
orig_ist
);
DEFINE_PER_CPU
(
struct
orig_ist
,
orig_ist
);
#else
#else
/* x86_64 */
/* Make sure %fs is initialized properly in idle threads */
#ifdef CONFIG_CC_STACKPROTECTOR
DEFINE_PER_CPU
(
unsigned
long
,
stack_canary
);
#endif
/* Make sure %fs and %gs are initialized properly in idle threads */
struct
pt_regs
*
__cpuinit
idle_regs
(
struct
pt_regs
*
regs
)
struct
pt_regs
*
__cpuinit
idle_regs
(
struct
pt_regs
*
regs
)
{
{
memset
(
regs
,
0
,
sizeof
(
struct
pt_regs
));
memset
(
regs
,
0
,
sizeof
(
struct
pt_regs
));
regs
->
fs
=
__KERNEL_PERCPU
;
regs
->
fs
=
__KERNEL_PERCPU
;
regs
->
gs
=
__KERNEL_STACK_CANARY
;
return
regs
;
return
regs
;
}
}
#endif
#endif
/* x86_64 */
/*
/*
* cpu_init() initializes state that is per-CPU. Some data is already
* cpu_init() initializes state that is per-CPU. Some data is already
...
@@ -1157,9 +1161,6 @@ void __cpuinit cpu_init(void)
...
@@ -1157,9 +1161,6 @@ void __cpuinit cpu_init(void)
__set_tss_desc
(
cpu
,
GDT_ENTRY_DOUBLEFAULT_TSS
,
&
doublefault_tss
);
__set_tss_desc
(
cpu
,
GDT_ENTRY_DOUBLEFAULT_TSS
,
&
doublefault_tss
);
#endif
#endif
/* Clear %gs. */
asm
volatile
(
"mov %0, %%gs"
:
:
"r"
(
0
));
/* Clear all 6 debug registers: */
/* Clear all 6 debug registers: */
set_debugreg
(
0
,
0
);
set_debugreg
(
0
,
0
);
set_debugreg
(
0
,
1
);
set_debugreg
(
0
,
1
);
...
...
arch/x86/kernel/entry_32.S
View file @
ab639f35
This diff is collapsed.
Click to expand it.
arch/x86/kernel/head_32.S
View file @
ab639f35
...
@@ -19,6 +19,7 @@
...
@@ -19,6 +19,7 @@
#include <asm/asm-offsets.h>
#include <asm/asm-offsets.h>
#include <asm/setup.h>
#include <asm/setup.h>
#include <asm/processor-flags.h>
#include <asm/processor-flags.h>
#include <asm/percpu.h>
/*
Physical
address
*/
/*
Physical
address
*/
#define pa(X) ((X) - __PAGE_OFFSET)
#define pa(X) ((X) - __PAGE_OFFSET)
...
@@ -437,8 +438,26 @@ is386: movl $2,%ecx # set MP
...
@@ -437,8 +438,26 @@ is386: movl $2,%ecx # set MP
movl
$
(
__KERNEL_PERCPU
),
%
eax
movl
$
(
__KERNEL_PERCPU
),
%
eax
movl
%
eax
,%
fs
#
set
this
cpu
's percpu
movl
%
eax
,%
fs
#
set
this
cpu
's percpu
xorl
%
eax
,%
eax
#
Clear
GS
and
LDT
#ifdef CONFIG_CC_STACKPROTECTOR
/
*
*
The
linker
can
't handle this by relocation. Manually set
*
base
address
in
stack
canary
segment
descriptor
.
*/
cmpb
$
0
,
ready
jne
1
f
movl
$per_cpu__gdt_page
,%
eax
movl
$per_cpu__stack_canary
,%
ecx
subl
$
20
,
%
ecx
movw
%
cx
,
8
*
GDT_ENTRY_STACK_CANARY
+
2
(%
eax
)
shrl
$
16
,
%
ecx
movb
%
cl
,
8
*
GDT_ENTRY_STACK_CANARY
+
4
(%
eax
)
movb
%
ch
,
8
*
GDT_ENTRY_STACK_CANARY
+
7
(%
eax
)
1
:
#endif
movl
$
(
__KERNEL_STACK_CANARY
),%
eax
movl
%
eax
,%
gs
movl
%
eax
,%
gs
xorl
%
eax
,%
eax
#
Clear
LDT
lldt
%
ax
lldt
%
ax
cld
#
gcc2
wants
the
direction
flag
cleared
at
all
times
cld
#
gcc2
wants
the
direction
flag
cleared
at
all
times
...
...
arch/x86/kernel/head_64.S
View file @
ab639f35
...
@@ -205,19 +205,6 @@ ENTRY(secondary_startup_64)
...
@@ -205,19 +205,6 @@ ENTRY(secondary_startup_64)
pushq
$
0
pushq
$
0
popfq
popfq
#ifdef CONFIG_SMP
/
*
*
Fix
up
static
pointers
that
need
__per_cpu_load
added
.
The
assembler
*
is
unable
to
do
this
directly
.
This
is
only
needed
for
the
boot
cpu
.
*
These
values
are
set
up
with
the
correct
base
addresses
by
C
code
for
*
secondary
cpus
.
*/
movq
initial_gs
(%
rip
),
%
rax
cmpl
$
0
,
per_cpu__cpu_number
(%
rax
)
jne
1
f
addq
%
rax
,
early_gdt_descr_base
(%
rip
)
1
:
#endif
/
*
/
*
*
We
must
switch
to
a
new
descriptor
in
kernel
space
for
the
GDT
*
We
must
switch
to
a
new
descriptor
in
kernel
space
for
the
GDT
*
because
soon
the
kernel
won
't have access anymore to the userspace
*
because
soon
the
kernel
won
't have access anymore to the userspace
...
@@ -275,11 +262,7 @@ ENTRY(secondary_startup_64)
...
@@ -275,11 +262,7 @@ ENTRY(secondary_startup_64)
ENTRY
(
initial_code
)
ENTRY
(
initial_code
)
.
quad
x86_64_start_kernel
.
quad
x86_64_start_kernel
ENTRY
(
initial_gs
)
ENTRY
(
initial_gs
)
#ifdef CONFIG_SMP
.
quad
INIT_PER_CPU_VAR
(
irq_stack_union
)
.
quad
__per_cpu_load
#else
.
quad
PER_CPU_VAR
(
irq_stack_union
)
#endif
__FINITDATA
__FINITDATA
ENTRY
(
stack_start
)
ENTRY
(
stack_start
)
...
@@ -425,7 +408,7 @@ NEXT_PAGE(level2_spare_pgt)
...
@@ -425,7 +408,7 @@ NEXT_PAGE(level2_spare_pgt)
early_gdt_descr
:
early_gdt_descr
:
.
word
GDT_ENTRIES
*
8
-
1
.
word
GDT_ENTRIES
*
8
-
1
early_gdt_descr_base
:
early_gdt_descr_base
:
.
quad
per_cpu__gdt_page
.
quad
INIT_PER_CPU_VAR
(
gdt_page
)
ENTRY
(
phys_base
)
ENTRY
(
phys_base
)
/
*
This
must
match
the
first
entry
in
level2_kernel_pgt
*/
/
*
This
must
match
the
first
entry
in
level2_kernel_pgt
*/
...
...
arch/x86/kernel/ioport.c
View file @
ab639f35
...
@@ -131,9 +131,8 @@ static int do_iopl(unsigned int level, struct pt_regs *regs)
...
@@ -131,9 +131,8 @@ static int do_iopl(unsigned int level, struct pt_regs *regs)
}
}
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
asmlinkage
long
sys_iopl
(
unsigned
long
regsp
)
long
sys_iopl
(
struct
pt_regs
*
regs
)
{
{
struct
pt_regs
*
regs
=
(
struct
pt_regs
*
)
&
regsp
;
unsigned
int
level
=
regs
->
bx
;
unsigned
int
level
=
regs
->
bx
;
struct
thread_struct
*
t
=
&
current
->
thread
;
struct
thread_struct
*
t
=
&
current
->
thread
;
int
rc
;
int
rc
;
...
...
arch/x86/kernel/process_32.c
View file @
ab639f35
...
@@ -11,6 +11,7 @@
...
@@ -11,6 +11,7 @@
#include <stdarg.h>
#include <stdarg.h>
#include <linux/stackprotector.h>
#include <linux/cpu.h>
#include <linux/cpu.h>
#include <linux/errno.h>
#include <linux/errno.h>
#include <linux/sched.h>
#include <linux/sched.h>
...
@@ -91,6 +92,15 @@ void cpu_idle(void)
...
@@ -91,6 +92,15 @@ void cpu_idle(void)
{
{
int
cpu
=
smp_processor_id
();
int
cpu
=
smp_processor_id
();
/*
* If we're the non-boot CPU, nothing set the stack canary up
* for us. CPU0 already has it initialized but no harm in
* doing it again. This is a good place for updating it, as
* we wont ever return from this function (so the invalid
* canaries already on the stack wont ever trigger).
*/
boot_init_stack_canary
();
current_thread_info
()
->
status
|=
TS_POLLING
;
current_thread_info
()
->
status
|=
TS_POLLING
;
/* endless idle loop with no priority at all */
/* endless idle loop with no priority at all */
...
@@ -131,7 +141,7 @@ void __show_regs(struct pt_regs *regs, int all)
...
@@ -131,7 +141,7 @@ void __show_regs(struct pt_regs *regs, int all)
if
(
user_mode_vm
(
regs
))
{
if
(
user_mode_vm
(
regs
))
{
sp
=
regs
->
sp
;
sp
=
regs
->
sp
;
ss
=
regs
->
ss
&
0xffff
;
ss
=
regs
->
ss
&
0xffff
;
savesegment
(
gs
,
gs
);
gs
=
get_user_gs
(
re
gs
);
}
else
{
}
else
{
sp
=
(
unsigned
long
)
(
&
regs
->
sp
);
sp
=
(
unsigned
long
)
(
&
regs
->
sp
);
savesegment
(
ss
,
ss
);
savesegment
(
ss
,
ss
);
...
@@ -212,6 +222,7 @@ int kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)
...
@@ -212,6 +222,7 @@ int kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)
regs
.
ds
=
__USER_DS
;
regs
.
ds
=
__USER_DS
;
regs
.
es
=
__USER_DS
;
regs
.
es
=
__USER_DS
;
regs
.
fs
=
__KERNEL_PERCPU
;
regs
.
fs
=
__KERNEL_PERCPU
;
regs
.
gs
=
__KERNEL_STACK_CANARY
;
regs
.
orig_ax
=
-
1
;
regs
.
orig_ax
=
-
1
;
regs
.
ip
=
(
unsigned
long
)
kernel_thread_helper
;
regs
.
ip
=
(
unsigned
long
)
kernel_thread_helper
;
regs
.
cs
=
__KERNEL_CS
|
get_kernel_rpl
();
regs
.
cs
=
__KERNEL_CS
|
get_kernel_rpl
();
...
@@ -304,7 +315,7 @@ int copy_thread(int nr, unsigned long clone_flags, unsigned long sp,
...
@@ -304,7 +315,7 @@ int copy_thread(int nr, unsigned long clone_flags, unsigned long sp,
p
->
thread
.
ip
=
(
unsigned
long
)
ret_from_fork
;
p
->
thread
.
ip
=
(
unsigned
long
)
ret_from_fork
;
savesegment
(
gs
,
p
->
thread
.
gs
);
task_user_gs
(
p
)
=
get_user_gs
(
re
gs
);
tsk
=
current
;
tsk
=
current
;
if
(
unlikely
(
test_tsk_thread_flag
(
tsk
,
TIF_IO_BITMAP
)))
{
if
(
unlikely
(
test_tsk_thread_flag
(
tsk
,
TIF_IO_BITMAP
)))
{
...
@@ -342,7 +353,7 @@ int copy_thread(int nr, unsigned long clone_flags, unsigned long sp,
...
@@ -342,7 +353,7 @@ int copy_thread(int nr, unsigned long clone_flags, unsigned long sp,
void
void
start_thread
(
struct
pt_regs
*
regs
,
unsigned
long
new_ip
,
unsigned
long
new_sp
)
start_thread
(
struct
pt_regs
*
regs
,
unsigned
long
new_ip
,
unsigned
long
new_sp
)
{
{
__asm__
(
"movl %0, %%gs"
:
:
"r"
(
0
)
);
set_user_gs
(
regs
,
0
);
regs
->
fs
=
0
;
regs
->
fs
=
0
;
set_fs
(
USER_DS
);
set_fs
(
USER_DS
);
regs
->
ds
=
__USER_DS
;
regs
->
ds
=
__USER_DS
;
...
@@ -539,7 +550,7 @@ __switch_to(struct task_struct *prev_p, struct task_struct *next_p)
...
@@ -539,7 +550,7 @@ __switch_to(struct task_struct *prev_p, struct task_struct *next_p)
* used %fs or %gs (it does not today), or if the kernel is
* used %fs or %gs (it does not today), or if the kernel is
* running inside of a hypervisor layer.
* running inside of a hypervisor layer.
*/
*/
savesegment
(
gs
,
prev
->
gs
);
lazy_save_gs
(
prev
->
gs
);
/*
/*
* Load the per-thread Thread-Local Storage descriptor.
* Load the per-thread Thread-Local Storage descriptor.
...
@@ -585,31 +596,31 @@ __switch_to(struct task_struct *prev_p, struct task_struct *next_p)
...
@@ -585,31 +596,31 @@ __switch_to(struct task_struct *prev_p, struct task_struct *next_p)
* Restore %gs if needed (which is common)
* Restore %gs if needed (which is common)
*/
*/
if
(
prev
->
gs
|
next
->
gs
)
if
(
prev
->
gs
|
next
->
gs
)
l
oadsegment
(
gs
,
next
->
gs
);
l
azy_load_gs
(
next
->
gs
);
percpu_write
(
current_task
,
next_p
);
percpu_write
(
current_task
,
next_p
);
return
prev_p
;
return
prev_p
;
}
}
asmlinkage
int
sys_fork
(
struct
pt_regs
regs
)
int
sys_fork
(
struct
pt_regs
*
regs
)
{
{
return
do_fork
(
SIGCHLD
,
regs
.
sp
,
&
regs
,
0
,
NULL
,
NULL
);
return
do_fork
(
SIGCHLD
,
regs
->
sp
,
regs
,
0
,
NULL
,
NULL
);
}
}
asmlinkage
int
sys_clone
(
struct
pt_regs
regs
)
int
sys_clone
(
struct
pt_regs
*
regs
)
{
{
unsigned
long
clone_flags
;
unsigned
long
clone_flags
;
unsigned
long
newsp
;
unsigned
long
newsp
;
int
__user
*
parent_tidptr
,
*
child_tidptr
;
int
__user
*
parent_tidptr
,
*
child_tidptr
;
clone_flags
=
regs
.
bx
;
clone_flags
=
regs
->
bx
;
newsp
=
regs
.
cx
;
newsp
=
regs
->
cx
;
parent_tidptr
=
(
int
__user
*
)
regs
.
dx
;
parent_tidptr
=
(
int
__user
*
)
regs
->
dx
;
child_tidptr
=
(
int
__user
*
)
regs
.
di
;
child_tidptr
=
(
int
__user
*
)
regs
->
di
;
if
(
!
newsp
)
if
(
!
newsp
)
newsp
=
regs
.
sp
;
newsp
=
regs
->
sp
;
return
do_fork
(
clone_flags
,
newsp
,
&
regs
,
0
,
parent_tidptr
,
child_tidptr
);
return
do_fork
(
clone_flags
,
newsp
,
regs
,
0
,
parent_tidptr
,
child_tidptr
);
}
}
/*
/*
...
@@ -622,27 +633,27 @@ asmlinkage int sys_clone(struct pt_regs regs)
...
@@ -622,27 +633,27 @@ asmlinkage int sys_clone(struct pt_regs regs)
* do not have enough call-clobbered registers to hold all
* do not have enough call-clobbered registers to hold all
* the information you need.
* the information you need.
*/
*/
asmlinkage
int
sys_vfork
(
struct
pt_regs
regs
)
int
sys_vfork
(
struct
pt_regs
*
regs
)
{
{
return
do_fork
(
CLONE_VFORK
|
CLONE_VM
|
SIGCHLD
,
regs
.
sp
,
&
regs
,
0
,
NULL
,
NULL
);
return
do_fork
(
CLONE_VFORK
|
CLONE_VM
|
SIGCHLD
,
regs
->
sp
,
regs
,
0
,
NULL
,
NULL
);
}
}
/*
/*
* sys_execve() executes a new program.
* sys_execve() executes a new program.
*/
*/
asmlinkage
int
sys_execve
(
struct
pt_regs
regs
)
int
sys_execve
(
struct
pt_regs
*
regs
)
{
{
int
error
;
int
error
;
char
*
filename
;
char
*
filename
;
filename
=
getname
((
char
__user
*
)
regs
.
bx
);
filename
=
getname
((
char
__user
*
)
regs
->
bx
);
error
=
PTR_ERR
(
filename
);
error
=
PTR_ERR
(
filename
);
if
(
IS_ERR
(
filename
))
if
(
IS_ERR
(
filename
))
goto
out
;
goto
out
;
error
=
do_execve
(
filename
,
error
=
do_execve
(
filename
,
(
char
__user
*
__user
*
)
regs
.
cx
,
(
char
__user
*
__user
*
)
regs
->
cx
,
(
char
__user
*
__user
*
)
regs
.
dx
,
(
char
__user
*
__user
*
)
regs
->
dx
,
&
regs
);
regs
);
if
(
error
==
0
)
{
if
(
error
==
0
)
{
/* Make sure we don't return using sysenter.. */
/* Make sure we don't return using sysenter.. */
set_thread_flag
(
TIF_IRET
);
set_thread_flag
(
TIF_IRET
);
...
...
arch/x86/kernel/process_64.c
View file @
ab639f35
...
@@ -120,12 +120,11 @@ void cpu_idle(void)
...
@@ -120,12 +120,11 @@ void cpu_idle(void)
current_thread_info
()
->
status
|=
TS_POLLING
;
current_thread_info
()
->
status
|=
TS_POLLING
;
/*
/*
* If we're the non-boot CPU, nothing set the PDA stack
* If we're the non-boot CPU, nothing set the stack canary up
* canary up for us - and if we are the boot CPU we have
* for us. CPU0 already has it initialized but no harm in
* a 0 stack canary. This is a good place for updating
* doing it again. This is a good place for updating it, as
* it, as we wont ever return from this function (so the
* we wont ever return from this function (so the invalid
* invalid canaries already on the stack wont ever
* canaries already on the stack wont ever trigger).
* trigger):
*/
*/
boot_init_stack_canary
();
boot_init_stack_canary
();
...
...
arch/x86/kernel/ptrace.c
View file @
ab639f35
...
@@ -75,10 +75,7 @@ static inline bool invalid_selector(u16 value)
...
@@ -75,10 +75,7 @@ static inline bool invalid_selector(u16 value)
static
unsigned
long
*
pt_regs_access
(
struct
pt_regs
*
regs
,
unsigned
long
regno
)
static
unsigned
long
*
pt_regs_access
(
struct
pt_regs
*
regs
,
unsigned
long
regno
)
{
{
BUILD_BUG_ON
(
offsetof
(
struct
pt_regs
,
bx
)
!=
0
);
BUILD_BUG_ON
(
offsetof
(
struct
pt_regs
,
bx
)
!=
0
);
regno
>>=
2
;
return
&
regs
->
bx
+
(
regno
>>
2
);
if
(
regno
>
FS
)
--
regno
;
return
&
regs
->
bx
+
regno
;
}
}
static
u16
get_segment_reg
(
struct
task_struct
*
task
,
unsigned
long
offset
)
static
u16
get_segment_reg
(
struct
task_struct
*
task
,
unsigned
long
offset
)
...
@@ -90,9 +87,10 @@ static u16 get_segment_reg(struct task_struct *task, unsigned long offset)
...
@@ -90,9 +87,10 @@ static u16 get_segment_reg(struct task_struct *task, unsigned long offset)
if
(
offset
!=
offsetof
(
struct
user_regs_struct
,
gs
))
if
(
offset
!=
offsetof
(
struct
user_regs_struct
,
gs
))
retval
=
*
pt_regs_access
(
task_pt_regs
(
task
),
offset
);
retval
=
*
pt_regs_access
(
task_pt_regs
(
task
),
offset
);
else
{
else
{
retval
=
task
->
thread
.
gs
;
if
(
task
==
current
)
if
(
task
==
current
)
savesegment
(
gs
,
retval
);
retval
=
get_user_gs
(
task_pt_regs
(
task
));
else
retval
=
task_user_gs
(
task
);
}
}
return
retval
;
return
retval
;
}
}
...
@@ -126,13 +124,10 @@ static int set_segment_reg(struct task_struct *task,
...
@@ -126,13 +124,10 @@ static int set_segment_reg(struct task_struct *task,
break
;
break
;
case
offsetof
(
struct
user_regs_struct
,
gs
):
case
offsetof
(
struct
user_regs_struct
,
gs
):
task
->
thread
.
gs
=
value
;
if
(
task
==
current
)
if
(
task
==
current
)
/*
set_user_gs
(
task_pt_regs
(
task
),
value
);
* The user-mode %gs is not affected by
else
* kernel entry, so we must update the CPU.
task_user_gs
(
task
)
=
value
;
*/
loadsegment
(
gs
,
value
);
}
}
return
0
;
return
0
;
...
...
arch/x86/kernel/setup_percpu.c
View file @
ab639f35
...
@@ -16,6 +16,7 @@
...
@@ -16,6 +16,7 @@
#include <asm/proto.h>
#include <asm/proto.h>
#include <asm/cpumask.h>
#include <asm/cpumask.h>
#include <asm/cpu.h>
#include <asm/cpu.h>
#include <asm/stackprotector.h>
#ifdef CONFIG_DEBUG_PER_CPU_MAPS
#ifdef CONFIG_DEBUG_PER_CPU_MAPS
# define DBG(x...) printk(KERN_DEBUG x)
# define DBG(x...) printk(KERN_DEBUG x)
...
@@ -95,6 +96,7 @@ void __init setup_per_cpu_areas(void)
...
@@ -95,6 +96,7 @@ void __init setup_per_cpu_areas(void)
per_cpu
(
this_cpu_off
,
cpu
)
=
per_cpu_offset
(
cpu
);
per_cpu
(
this_cpu_off
,
cpu
)
=
per_cpu_offset
(
cpu
);
per_cpu
(
cpu_number
,
cpu
)
=
cpu
;
per_cpu
(
cpu_number
,
cpu
)
=
cpu
;
setup_percpu_segment
(
cpu
);
setup_percpu_segment
(
cpu
);
setup_stack_canary_segment
(
cpu
);
/*
/*
* Copy data used in early init routines from the
* Copy data used in early init routines from the
* initial arrays to the per cpu data areas. These
* initial arrays to the per cpu data areas. These
...
...
arch/x86/kernel/signal.c
View file @
ab639f35
...
@@ -50,27 +50,23 @@
...
@@ -50,27 +50,23 @@
# define FIX_EFLAGS __FIX_EFLAGS
# define FIX_EFLAGS __FIX_EFLAGS
#endif
#endif
#define COPY(x)
{
\
#define COPY(x)
do {
\
get_user_ex(regs->x, &sc->x); \
get_user_ex(regs->x, &sc->x);
\
}
}
while (0)
#define
COPY_SEG(seg)
{ \
#define
GET_SEG(seg) (
{ \
unsigned short tmp;
\
unsigned short tmp;
\
get_user_ex(tmp, &sc->seg);
\
get_user_ex(tmp, &sc->seg);
\
regs->seg = tmp;
\
tmp;
\
}
}
)
#define COPY_SEG_CPL3(seg) { \
#define COPY_SEG(seg) do { \
unsigned short tmp; \
regs->seg = GET_SEG(seg); \
get_user_ex(tmp, &sc->seg); \
} while (0)
regs->seg = tmp | 3; \
}
#define GET_SEG(seg) { \
#define COPY_SEG_CPL3(seg) do { \
unsigned short tmp; \
regs->seg = GET_SEG(seg) | 3; \
get_user_ex(tmp, &sc->seg); \
} while (0)
loadsegment(seg, tmp); \
}
static
int
static
int
restore_sigcontext
(
struct
pt_regs
*
regs
,
struct
sigcontext
__user
*
sc
,
restore_sigcontext
(
struct
pt_regs
*
regs
,
struct
sigcontext
__user
*
sc
,
...
@@ -86,7 +82,7 @@ restore_sigcontext(struct pt_regs *regs, struct sigcontext __user *sc,
...
@@ -86,7 +82,7 @@ restore_sigcontext(struct pt_regs *regs, struct sigcontext __user *sc,
get_user_try
{
get_user_try
{
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
GET_SEG
(
gs
);
set_user_gs
(
regs
,
GET_SEG
(
gs
)
);
COPY_SEG
(
fs
);
COPY_SEG
(
fs
);
COPY_SEG
(
es
);
COPY_SEG
(
es
);
COPY_SEG
(
ds
);
COPY_SEG
(
ds
);
...
@@ -138,12 +134,7 @@ setup_sigcontext(struct sigcontext __user *sc, void __user *fpstate,
...
@@ -138,12 +134,7 @@ setup_sigcontext(struct sigcontext __user *sc, void __user *fpstate,
put_user_try
{
put_user_try
{
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
{
put_user_ex
(
get_user_gs
(
regs
),
(
unsigned
int
__user
*
)
&
sc
->
gs
);
unsigned
int
tmp
;
savesegment
(
gs
,
tmp
);
put_user_ex
(
tmp
,
(
unsigned
int
__user
*
)
&
sc
->
gs
);
}
put_user_ex
(
regs
->
fs
,
(
unsigned
int
__user
*
)
&
sc
->
fs
);
put_user_ex
(
regs
->
fs
,
(
unsigned
int
__user
*
)
&
sc
->
fs
);
put_user_ex
(
regs
->
es
,
(
unsigned
int
__user
*
)
&
sc
->
es
);
put_user_ex
(
regs
->
es
,
(
unsigned
int
__user
*
)
&
sc
->
es
);
put_user_ex
(
regs
->
ds
,
(
unsigned
int
__user
*
)
&
sc
->
ds
);
put_user_ex
(
regs
->
ds
,
(
unsigned
int
__user
*
)
&
sc
->
ds
);
...
@@ -558,14 +549,9 @@ sys_sigaction(int sig, const struct old_sigaction __user *act,
...
@@ -558,14 +549,9 @@ sys_sigaction(int sig, const struct old_sigaction __user *act,
#endif
/* CONFIG_X86_32 */
#endif
/* CONFIG_X86_32 */
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
asmlinkage
int
sys_sigaltstack
(
unsigned
long
bx
)
int
sys_sigaltstack
(
struct
pt_regs
*
regs
)
{
{
/*
const
stack_t
__user
*
uss
=
(
const
stack_t
__user
*
)
regs
->
bx
;
* This is needed to make gcc realize it doesn't own the
* "struct pt_regs"
*/
struct
pt_regs
*
regs
=
(
struct
pt_regs
*
)
&
bx
;
const
stack_t
__user
*
uss
=
(
const
stack_t
__user
*
)
bx
;
stack_t
__user
*
uoss
=
(
stack_t
__user
*
)
regs
->
cx
;
stack_t
__user
*
uoss
=
(
stack_t
__user
*
)
regs
->
cx
;
return
do_sigaltstack
(
uss
,
uoss
,
regs
->
sp
);
return
do_sigaltstack
(
uss
,
uoss
,
regs
->
sp
);
...
@@ -583,14 +569,12 @@ sys_sigaltstack(const stack_t __user *uss, stack_t __user *uoss,
...
@@ -583,14 +569,12 @@ sys_sigaltstack(const stack_t __user *uss, stack_t __user *uoss,
* Do a signal return; undo the signal stack.
* Do a signal return; undo the signal stack.
*/
*/
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
asmlinkage
unsigned
long
sys_sigreturn
(
unsigned
long
__unused
)
unsigned
long
sys_sigreturn
(
struct
pt_regs
*
regs
)
{
{
struct
sigframe
__user
*
frame
;
struct
sigframe
__user
*
frame
;
struct
pt_regs
*
regs
;
unsigned
long
ax
;
unsigned
long
ax
;
sigset_t
set
;
sigset_t
set
;
regs
=
(
struct
pt_regs
*
)
&
__unused
;
frame
=
(
struct
sigframe
__user
*
)(
regs
->
sp
-
8
);
frame
=
(
struct
sigframe
__user
*
)(
regs
->
sp
-
8
);
if
(
!
access_ok
(
VERIFY_READ
,
frame
,
sizeof
(
*
frame
)))
if
(
!
access_ok
(
VERIFY_READ
,
frame
,
sizeof
(
*
frame
)))
...
@@ -617,7 +601,7 @@ asmlinkage unsigned long sys_sigreturn(unsigned long __unused)
...
@@ -617,7 +601,7 @@ asmlinkage unsigned long sys_sigreturn(unsigned long __unused)
}
}
#endif
/* CONFIG_X86_32 */
#endif
/* CONFIG_X86_32 */
static
long
do
_rt_sigreturn
(
struct
pt_regs
*
regs
)
long
sys
_rt_sigreturn
(
struct
pt_regs
*
regs
)
{
{
struct
rt_sigframe
__user
*
frame
;
struct
rt_sigframe
__user
*
frame
;
unsigned
long
ax
;
unsigned
long
ax
;
...
@@ -648,25 +632,6 @@ static long do_rt_sigreturn(struct pt_regs *regs)
...
@@ -648,25 +632,6 @@ static long do_rt_sigreturn(struct pt_regs *regs)
return
0
;
return
0
;
}
}
#ifdef CONFIG_X86_32
/*
* Note: do not pass in pt_regs directly as with tail-call optimization
* GCC will incorrectly stomp on the caller's frame and corrupt user-space
* register state:
*/
asmlinkage
int
sys_rt_sigreturn
(
unsigned
long
__unused
)
{
struct
pt_regs
*
regs
=
(
struct
pt_regs
*
)
&
__unused
;
return
do_rt_sigreturn
(
regs
);
}
#else
/* !CONFIG_X86_32 */
asmlinkage
long
sys_rt_sigreturn
(
struct
pt_regs
*
regs
)
{
return
do_rt_sigreturn
(
regs
);
}
#endif
/* CONFIG_X86_32 */
/*
/*
* OK, we're invoking a handler:
* OK, we're invoking a handler:
*/
*/
...
...
arch/x86/kernel/syscall_table_32.S
View file @
ab639f35
ENTRY
(
sys_call_table
)
ENTRY
(
sys_call_table
)
.
long
sys_restart_syscall
/*
0
-
old
"setup()"
system
call
,
used
for
restarting
*/
.
long
sys_restart_syscall
/*
0
-
old
"setup()"
system
call
,
used
for
restarting
*/
.
long
sys_exit
.
long
sys_exit
.
long
sy
s_fork
.
long
ptreg
s_fork
.
long
sys_read
.
long
sys_read
.
long
sys_write
.
long
sys_write
.
long
sys_open
/*
5
*/
.
long
sys_open
/*
5
*/
...
@@ -10,7 +10,7 @@ ENTRY(sys_call_table)
...
@@ -10,7 +10,7 @@ ENTRY(sys_call_table)
.
long
sys_creat
.
long
sys_creat
.
long
sys_link
.
long
sys_link
.
long
sys_unlink
/*
10
*/
.
long
sys_unlink
/*
10
*/
.
long
sy
s_execve
.
long
ptreg
s_execve
.
long
sys_chdir
.
long
sys_chdir
.
long
sys_time
.
long
sys_time
.
long
sys_mknod
.
long
sys_mknod
...
@@ -109,17 +109,17 @@ ENTRY(sys_call_table)
...
@@ -109,17 +109,17 @@ ENTRY(sys_call_table)
.
long
sys_newlstat
.
long
sys_newlstat
.
long
sys_newfstat
.
long
sys_newfstat
.
long
sys_uname
.
long
sys_uname
.
long
sys_iopl
/*
110
*/
.
long
ptregs_iopl
/*
110
*/
.
long
sys_vhangup
.
long
sys_vhangup
.
long
sys_ni_syscall
/*
old
"idle"
system
call
*/
.
long
sys_ni_syscall
/*
old
"idle"
system
call
*/
.
long
sy
s_vm86old
.
long
ptreg
s_vm86old
.
long
sys_wait4
.
long
sys_wait4
.
long
sys_swapoff
/*
115
*/
.
long
sys_swapoff
/*
115
*/
.
long
sys_sysinfo
.
long
sys_sysinfo
.
long
sys_ipc
.
long
sys_ipc
.
long
sys_fsync
.
long
sys_fsync
.
long
sy
s_sigreturn
.
long
ptreg
s_sigreturn
.
long
sys_clone
/*
120
*/
.
long
ptregs_clone
/*
120
*/
.
long
sys_setdomainname
.
long
sys_setdomainname
.
long
sys_newuname
.
long
sys_newuname
.
long
sys_modify_ldt
.
long
sys_modify_ldt
...
@@ -165,14 +165,14 @@ ENTRY(sys_call_table)
...
@@ -165,14 +165,14 @@ ENTRY(sys_call_table)
.
long
sys_mremap
.
long
sys_mremap
.
long
sys_setresuid16
.
long
sys_setresuid16
.
long
sys_getresuid16
/*
165
*/
.
long
sys_getresuid16
/*
165
*/
.
long
sy
s_vm86
.
long
ptreg
s_vm86
.
long
sys_ni_syscall
/*
Old
sys_query_module
*/
.
long
sys_ni_syscall
/*
Old
sys_query_module
*/
.
long
sys_poll
.
long
sys_poll
.
long
sys_nfsservctl
.
long
sys_nfsservctl
.
long
sys_setresgid16
/*
170
*/
.
long
sys_setresgid16
/*
170
*/
.
long
sys_getresgid16
.
long
sys_getresgid16
.
long
sys_prctl
.
long
sys_prctl
.
long
sy
s_rt_sigreturn
.
long
ptreg
s_rt_sigreturn
.
long
sys_rt_sigaction
.
long
sys_rt_sigaction
.
long
sys_rt_sigprocmask
/*
175
*/
.
long
sys_rt_sigprocmask
/*
175
*/
.
long
sys_rt_sigpending
.
long
sys_rt_sigpending
...
@@ -185,11 +185,11 @@ ENTRY(sys_call_table)
...
@@ -185,11 +185,11 @@ ENTRY(sys_call_table)
.
long
sys_getcwd
.
long
sys_getcwd
.
long
sys_capget
.
long
sys_capget
.
long
sys_capset
/*
185
*/
.
long
sys_capset
/*
185
*/
.
long
sy
s_sigaltstack
.
long
ptreg
s_sigaltstack
.
long
sys_sendfile
.
long
sys_sendfile
.
long
sys_ni_syscall
/*
reserved
for
streams1
*/
.
long
sys_ni_syscall
/*
reserved
for
streams1
*/
.
long
sys_ni_syscall
/*
reserved
for
streams2
*/
.
long
sys_ni_syscall
/*
reserved
for
streams2
*/
.
long
sys_vfork
/*
190
*/
.
long
ptregs_vfork
/*
190
*/
.
long
sys_getrlimit
.
long
sys_getrlimit
.
long
sys_mmap2
.
long
sys_mmap2
.
long
sys_truncate64
.
long
sys_truncate64
...
...
arch/x86/kernel/traps.c
View file @
ab639f35
...
@@ -905,19 +905,20 @@ void math_emulate(struct math_emu_info *info)
...
@@ -905,19 +905,20 @@ void math_emulate(struct math_emu_info *info)
}
}
#endif
/* CONFIG_MATH_EMULATION */
#endif
/* CONFIG_MATH_EMULATION */
dotraplinkage
void
__kprobes
do_device_not_available
(
struct
pt_regs
regs
)
dotraplinkage
void
__kprobes
do_device_not_available
(
struct
pt_regs
*
regs
,
long
error_code
)
{
{
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
if
(
read_cr0
()
&
X86_CR0_EM
)
{
if
(
read_cr0
()
&
X86_CR0_EM
)
{
struct
math_emu_info
info
=
{
};
struct
math_emu_info
info
=
{
};
conditional_sti
(
&
regs
);
conditional_sti
(
regs
);
info
.
regs
=
&
regs
;
info
.
regs
=
regs
;
math_emulate
(
&
info
);
math_emulate
(
&
info
);
}
else
{
}
else
{
math_state_restore
();
/* interrupts still off */
math_state_restore
();
/* interrupts still off */
conditional_sti
(
&
regs
);
conditional_sti
(
regs
);
}
}
#else
#else
math_state_restore
();
math_state_restore
();
...
...
arch/x86/kernel/vm86_32.c
View file @
ab639f35
...
@@ -158,7 +158,7 @@ struct pt_regs *save_v86_state(struct kernel_vm86_regs *regs)
...
@@ -158,7 +158,7 @@ struct pt_regs *save_v86_state(struct kernel_vm86_regs *regs)
ret
=
KVM86
->
regs32
;
ret
=
KVM86
->
regs32
;
ret
->
fs
=
current
->
thread
.
saved_fs
;
ret
->
fs
=
current
->
thread
.
saved_fs
;
loadsegment
(
gs
,
current
->
thread
.
saved_gs
);
set_user_gs
(
ret
,
current
->
thread
.
saved_gs
);
return
ret
;
return
ret
;
}
}
...
@@ -197,9 +197,9 @@ static void mark_screen_rdonly(struct mm_struct *mm)
...
@@ -197,9 +197,9 @@ static void mark_screen_rdonly(struct mm_struct *mm)
static
int
do_vm86_irq_handling
(
int
subfunction
,
int
irqnumber
);
static
int
do_vm86_irq_handling
(
int
subfunction
,
int
irqnumber
);
static
void
do_sys_vm86
(
struct
kernel_vm86_struct
*
info
,
struct
task_struct
*
tsk
);
static
void
do_sys_vm86
(
struct
kernel_vm86_struct
*
info
,
struct
task_struct
*
tsk
);
asmlinkage
int
sys_vm86old
(
struct
pt_regs
regs
)
int
sys_vm86old
(
struct
pt_regs
*
regs
)
{
{
struct
vm86_struct
__user
*
v86
=
(
struct
vm86_struct
__user
*
)
regs
.
bx
;
struct
vm86_struct
__user
*
v86
=
(
struct
vm86_struct
__user
*
)
regs
->
bx
;
struct
kernel_vm86_struct
info
;
/* declare this _on top_,
struct
kernel_vm86_struct
info
;
/* declare this _on top_,
* this avoids wasting of stack space.
* this avoids wasting of stack space.
* This remains on the stack until we
* This remains on the stack until we
...
@@ -218,7 +218,7 @@ asmlinkage int sys_vm86old(struct pt_regs regs)
...
@@ -218,7 +218,7 @@ asmlinkage int sys_vm86old(struct pt_regs regs)
if
(
tmp
)
if
(
tmp
)
goto
out
;
goto
out
;
memset
(
&
info
.
vm86plus
,
0
,
(
int
)
&
info
.
regs32
-
(
int
)
&
info
.
vm86plus
);
memset
(
&
info
.
vm86plus
,
0
,
(
int
)
&
info
.
regs32
-
(
int
)
&
info
.
vm86plus
);
info
.
regs32
=
&
regs
;
info
.
regs32
=
regs
;
tsk
->
thread
.
vm86_info
=
v86
;
tsk
->
thread
.
vm86_info
=
v86
;
do_sys_vm86
(
&
info
,
tsk
);
do_sys_vm86
(
&
info
,
tsk
);
ret
=
0
;
/* we never return here */
ret
=
0
;
/* we never return here */
...
@@ -227,7 +227,7 @@ asmlinkage int sys_vm86old(struct pt_regs regs)
...
@@ -227,7 +227,7 @@ asmlinkage int sys_vm86old(struct pt_regs regs)
}
}
asmlinkage
int
sys_vm86
(
struct
pt_regs
regs
)
int
sys_vm86
(
struct
pt_regs
*
regs
)
{
{
struct
kernel_vm86_struct
info
;
/* declare this _on top_,
struct
kernel_vm86_struct
info
;
/* declare this _on top_,
* this avoids wasting of stack space.
* this avoids wasting of stack space.
...
@@ -239,12 +239,12 @@ asmlinkage int sys_vm86(struct pt_regs regs)
...
@@ -239,12 +239,12 @@ asmlinkage int sys_vm86(struct pt_regs regs)
struct
vm86plus_struct
__user
*
v86
;
struct
vm86plus_struct
__user
*
v86
;
tsk
=
current
;
tsk
=
current
;
switch
(
regs
.
bx
)
{
switch
(
regs
->
bx
)
{
case
VM86_REQUEST_IRQ
:
case
VM86_REQUEST_IRQ
:
case
VM86_FREE_IRQ
:
case
VM86_FREE_IRQ
:
case
VM86_GET_IRQ_BITS
:
case
VM86_GET_IRQ_BITS
:
case
VM86_GET_AND_RESET_IRQ
:
case
VM86_GET_AND_RESET_IRQ
:
ret
=
do_vm86_irq_handling
(
regs
.
bx
,
(
int
)
regs
.
cx
);
ret
=
do_vm86_irq_handling
(
regs
->
bx
,
(
int
)
regs
->
cx
);
goto
out
;
goto
out
;
case
VM86_PLUS_INSTALL_CHECK
:
case
VM86_PLUS_INSTALL_CHECK
:
/*
/*
...
@@ -261,14 +261,14 @@ asmlinkage int sys_vm86(struct pt_regs regs)
...
@@ -261,14 +261,14 @@ asmlinkage int sys_vm86(struct pt_regs regs)
ret
=
-
EPERM
;
ret
=
-
EPERM
;
if
(
tsk
->
thread
.
saved_sp0
)
if
(
tsk
->
thread
.
saved_sp0
)
goto
out
;
goto
out
;
v86
=
(
struct
vm86plus_struct
__user
*
)
regs
.
cx
;
v86
=
(
struct
vm86plus_struct
__user
*
)
regs
->
cx
;
tmp
=
copy_vm86_regs_from_user
(
&
info
.
regs
,
&
v86
->
regs
,
tmp
=
copy_vm86_regs_from_user
(
&
info
.
regs
,
&
v86
->
regs
,
offsetof
(
struct
kernel_vm86_struct
,
regs32
)
-
offsetof
(
struct
kernel_vm86_struct
,
regs32
)
-
sizeof
(
info
.
regs
));
sizeof
(
info
.
regs
));
ret
=
-
EFAULT
;
ret
=
-
EFAULT
;
if
(
tmp
)
if
(
tmp
)
goto
out
;
goto
out
;
info
.
regs32
=
&
regs
;
info
.
regs32
=
regs
;
info
.
vm86plus
.
is_vm86pus
=
1
;
info
.
vm86plus
.
is_vm86pus
=
1
;
tsk
->
thread
.
vm86_info
=
(
struct
vm86_struct
__user
*
)
v86
;
tsk
->
thread
.
vm86_info
=
(
struct
vm86_struct
__user
*
)
v86
;
do_sys_vm86
(
&
info
,
tsk
);
do_sys_vm86
(
&
info
,
tsk
);
...
@@ -323,7 +323,7 @@ static void do_sys_vm86(struct kernel_vm86_struct *info, struct task_struct *tsk
...
@@ -323,7 +323,7 @@ static void do_sys_vm86(struct kernel_vm86_struct *info, struct task_struct *tsk
info
->
regs32
->
ax
=
0
;
info
->
regs32
->
ax
=
0
;
tsk
->
thread
.
saved_sp0
=
tsk
->
thread
.
sp0
;
tsk
->
thread
.
saved_sp0
=
tsk
->
thread
.
sp0
;
tsk
->
thread
.
saved_fs
=
info
->
regs32
->
fs
;
tsk
->
thread
.
saved_fs
=
info
->
regs32
->
fs
;
savesegment
(
gs
,
tsk
->
thread
.
saved_gs
);
tsk
->
thread
.
saved_gs
=
get_user_gs
(
info
->
regs32
);
tss
=
&
per_cpu
(
init_tss
,
get_cpu
());
tss
=
&
per_cpu
(
init_tss
,
get_cpu
());
tsk
->
thread
.
sp0
=
(
unsigned
long
)
&
info
->
VM86_TSS_ESP0
;
tsk
->
thread
.
sp0
=
(
unsigned
long
)
&
info
->
VM86_TSS_ESP0
;
...
...
arch/x86/kernel/vmlinux_64.lds.S
View file @
ab639f35
...
@@ -257,6 +257,14 @@ SECTIONS
...
@@ -257,6 +257,14 @@ SECTIONS
DWARF_DEBUG
DWARF_DEBUG
}
}
/
*
*
Per
-
cpu
symbols
which
need
to
be
offset
from
__per_cpu_load
*
for
the
boot
processor
.
*/
#define INIT_PER_CPU(x) init_per_cpu__##x = per_cpu__##x + __per_cpu_load
INIT_PER_CPU
(
gdt_page
)
;
INIT_PER_CPU
(
irq_stack_union
)
;
/*
/*
*
Build
-
time
check
on
the
image
size
:
*
Build
-
time
check
on
the
image
size
:
*/
*/
...
...
arch/x86/lguest/boot.c
View file @
ab639f35
...
@@ -283,7 +283,7 @@ static void lguest_load_tls(struct thread_struct *t, unsigned int cpu)
...
@@ -283,7 +283,7 @@ static void lguest_load_tls(struct thread_struct *t, unsigned int cpu)
/* There's one problem which normal hardware doesn't have: the Host
/* There's one problem which normal hardware doesn't have: the Host
* can't handle us removing entries we're currently using. So we clear
* can't handle us removing entries we're currently using. So we clear
* the GS register here: if it's needed it'll be reloaded anyway. */
* the GS register here: if it's needed it'll be reloaded anyway. */
l
oadsegment
(
gs
,
0
);
l
azy_load_gs
(
0
);
lazy_hcall
(
LHCALL_LOAD_TLS
,
__pa
(
&
t
->
tls_array
),
cpu
,
0
);
lazy_hcall
(
LHCALL_LOAD_TLS
,
__pa
(
&
t
->
tls_array
),
cpu
,
0
);
}
}
...
...
arch/x86/math-emu/get_address.c
View file @
ab639f35
...
@@ -150,11 +150,9 @@ static long pm_address(u_char FPU_modrm, u_char segment,
...
@@ -150,11 +150,9 @@ static long pm_address(u_char FPU_modrm, u_char segment,
#endif
/* PARANOID */
#endif
/* PARANOID */
switch
(
segment
)
{
switch
(
segment
)
{
/* gs isn't used by the kernel, so it still has its
user-space value. */
case
PREFIX_GS_
-
1
:
case
PREFIX_GS_
-
1
:
/*
N.B. - movl %seg, mem is a 2 byte write regardless of prefix
*/
/*
user gs handling can be lazy, use special accessors
*/
savesegment
(
gs
,
addr
->
selector
);
addr
->
selector
=
get_user_gs
(
FPU_info
->
regs
);
break
;
break
;
default:
default:
addr
->
selector
=
PM_REG_
(
segment
);
addr
->
selector
=
PM_REG_
(
segment
);
...
...
arch/x86/mm/numa_64.c
View file @
ab639f35
...
@@ -702,7 +702,7 @@ void __cpuinit numa_set_node(int cpu, int node)
...
@@ -702,7 +702,7 @@ void __cpuinit numa_set_node(int cpu, int node)
}
}
#ifdef CONFIG_DEBUG_PER_CPU_MAPS
#ifdef CONFIG_DEBUG_PER_CPU_MAPS
if
(
cpu
>=
nr_cpu_ids
||
!
per_cpu_offset
(
cpu
))
{
if
(
cpu
>=
nr_cpu_ids
||
!
cpu_possible
(
cpu
))
{
printk
(
KERN_ERR
"numa_set_node: invalid cpu# (%d)
\n
"
,
cpu
);
printk
(
KERN_ERR
"numa_set_node: invalid cpu# (%d)
\n
"
,
cpu
);
dump_stack
();
dump_stack
();
return
;
return
;
...
@@ -790,7 +790,7 @@ int early_cpu_to_node(int cpu)
...
@@ -790,7 +790,7 @@ int early_cpu_to_node(int cpu)
if
(
early_per_cpu_ptr
(
x86_cpu_to_node_map
))
if
(
early_per_cpu_ptr
(
x86_cpu_to_node_map
))
return
early_per_cpu_ptr
(
x86_cpu_to_node_map
)[
cpu
];
return
early_per_cpu_ptr
(
x86_cpu_to_node_map
)[
cpu
];
if
(
!
per_cpu_offset
(
cpu
))
{
if
(
!
cpu_possible
(
cpu
))
{
printk
(
KERN_WARNING
printk
(
KERN_WARNING
"early_cpu_to_node(%d): no per_cpu area!
\n
"
,
cpu
);
"early_cpu_to_node(%d): no per_cpu area!
\n
"
,
cpu
);
dump_stack
();
dump_stack
();
...
...
arch/x86/vdso/Makefile
View file @
ab639f35
...
@@ -38,7 +38,7 @@ $(obj)/%.so: $(obj)/%.so.dbg FORCE
...
@@ -38,7 +38,7 @@ $(obj)/%.so: $(obj)/%.so.dbg FORCE
$(
call
if_changed,objcopy
)
$(
call
if_changed,objcopy
)
CFL
:=
$(PROFILING)
-mcmodel
=
small
-fPIC
-O2
-fasynchronous-unwind-tables
-m64
\
CFL
:=
$(PROFILING)
-mcmodel
=
small
-fPIC
-O2
-fasynchronous-unwind-tables
-m64
\
$(
filter
-g
%,
$(KBUILD_CFLAGS)
)
$(
filter
-g
%,
$(KBUILD_CFLAGS)
)
$(
call
cc-option,
-fno-stack-protector
)
$(vobjs)
:
KBUILD_CFLAGS += $(CFL)
$(vobjs)
:
KBUILD_CFLAGS += $(CFL)
...
...
arch/x86/xen/enlighten.c
View file @
ab639f35
...
@@ -323,13 +323,14 @@ static void load_TLS_descriptor(struct thread_struct *t,
...
@@ -323,13 +323,14 @@ static void load_TLS_descriptor(struct thread_struct *t,
static
void
xen_load_tls
(
struct
thread_struct
*
t
,
unsigned
int
cpu
)
static
void
xen_load_tls
(
struct
thread_struct
*
t
,
unsigned
int
cpu
)
{
{
/*
/*
* XXX sleazy hack: If we're being called in a lazy-cpu zone,
* XXX sleazy hack: If we're being called in a lazy-cpu zone
* it means we're in a context switch, and %gs has just been
* and lazy gs handling is enabled, it means we're in a
* saved. This means we can zero it out to prevent faults on
* context switch, and %gs has just been saved. This means we
* exit from the hypervisor if the next process has no %gs.
* can zero it out to prevent faults on exit from the
* Either way, it has been saved, and the new value will get
* hypervisor if the next process has no %gs. Either way, it
* loaded properly. This will go away as soon as Xen has been
* has been saved, and the new value will get loaded properly.
* modified to not save/restore %gs for normal hypercalls.
* This will go away as soon as Xen has been modified to not
* save/restore %gs for normal hypercalls.
*
*
* On x86_64, this hack is not used for %gs, because gs points
* On x86_64, this hack is not used for %gs, because gs points
* to KERNEL_GS_BASE (and uses it for PDA references), so we
* to KERNEL_GS_BASE (and uses it for PDA references), so we
...
@@ -341,7 +342,7 @@ static void xen_load_tls(struct thread_struct *t, unsigned int cpu)
...
@@ -341,7 +342,7 @@ static void xen_load_tls(struct thread_struct *t, unsigned int cpu)
*/
*/
if
(
paravirt_get_lazy_mode
()
==
PARAVIRT_LAZY_CPU
)
{
if
(
paravirt_get_lazy_mode
()
==
PARAVIRT_LAZY_CPU
)
{
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
l
oadsegment
(
gs
,
0
);
l
azy_load_gs
(
0
);
#else
#else
loadsegment
(
fs
,
0
);
loadsegment
(
fs
,
0
);
#endif
#endif
...
...
arch/x86/xen/xen-asm.S
View file @
ab639f35
/*
/*
Asm
versions
of
Xen
pv
-
ops
,
suitable
for
either
direct
use
or
inlining
.
*
Asm
versions
of
Xen
pv
-
ops
,
suitable
for
either
direct
use
or
The
inline
versions
are
the
same
as
the
direct
-
use
versions
,
with
th
e
*
inlining
.
The
inline
versions
are
the
same
as
the
direct
-
us
e
pre
-
and
post
-
amble
chopped
off
.
*
versions
,
with
the
pre
-
and
post
-
amble
chopped
off
.
*
This
code
is
encoded
for
size
rather
than
absolute
efficiency
,
*
This
code
is
encoded
for
size
rather
than
absolute
efficiency
,
with
with
a
view
to
being
able
to
inline
as
much
as
possible
.
*
a
view
to
being
able
to
inline
as
much
as
possible
.
*
We
only
bother
with
direct
forms
(
ie
,
vcpu
in
percpu
data
)
of
*
We
only
bother
with
direct
forms
(
ie
,
vcpu
in
percpu
data
)
of
the
the
operations
here
; the indirect forms are better handled in
*
operations
here
; the indirect forms are better handled in C, since
C
,
since
they
're generally too large to inline anyway.
*
they
're generally too large to inline anyway.
*/
*/
#include <asm/asm-offsets.h>
#include <asm/asm-offsets.h>
...
@@ -18,17 +18,19 @@
...
@@ -18,17 +18,19 @@
#include "xen-asm.h"
#include "xen-asm.h"
/*
/*
Enable
events
.
This
clears
the
event
mask
and
tests
the
pending
*
Enable
events
.
This
clears
the
event
mask
and
tests
the
pending
event
status
with
one
and
operation
.
If
there
are
pending
*
event
status
with
one
and
operation
.
If
there
are
pending
events
,
events
,
then
enter
the
hypervisor
to
get
them
handled
.
*
then
enter
the
hypervisor
to
get
them
handled
.
*/
*/
ENTRY
(
xen_irq_enable_direct
)
ENTRY
(
xen_irq_enable_direct
)
/
*
Unmask
events
*/
/
*
Unmask
events
*/
movb
$
0
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
movb
$
0
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
/
*
Preempt
here
doesn
't matter because that will deal with
/
*
any
pending
interrupts
.
The
pending
check
may
end
up
being
*
Preempt
here
doesn
't matter because that will deal with any
run
on
the
wrong
CPU
,
but
that
doesn
't hurt. */
*
pending
interrupts
.
The
pending
check
may
end
up
being
run
*
on
the
wrong
CPU
,
but
that
doesn
't hurt.
*/
/
*
Test
for
pending
*/
/
*
Test
for
pending
*/
testb
$
0xff
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_pending
testb
$
0xff
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_pending
...
@@ -43,8 +45,8 @@ ENDPATCH(xen_irq_enable_direct)
...
@@ -43,8 +45,8 @@ ENDPATCH(xen_irq_enable_direct)
/*
/*
Disabling
events
is
simply
a
matter
of
making
the
event
mask
*
Disabling
events
is
simply
a
matter
of
making
the
event
mask
non
-
zero
.
*
non
-
zero
.
*/
*/
ENTRY
(
xen_irq_disable_direct
)
ENTRY
(
xen_irq_disable_direct
)
movb
$
1
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
movb
$
1
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
...
@@ -54,18 +56,18 @@ ENDPATCH(xen_irq_disable_direct)
...
@@ -54,18 +56,18 @@ ENDPATCH(xen_irq_disable_direct)
RELOC
(
xen_irq_disable_direct
,
0
)
RELOC
(
xen_irq_disable_direct
,
0
)
/*
/*
(
xen_
)
save_fl
is
used
to
get
the
current
interrupt
enable
status
.
*
(
xen_
)
save_fl
is
used
to
get
the
current
interrupt
enable
status
.
Callers
expect
the
status
to
be
in
X86_EFLAGS_IF
,
and
other
bits
*
Callers
expect
the
status
to
be
in
X86_EFLAGS_IF
,
and
other
bits
may
be
set
in
the
return
value
.
We
take
advantage
of
this
by
*
may
be
set
in
the
return
value
.
We
take
advantage
of
this
by
making
sure
that
X86_EFLAGS_IF
has
the
right
value
(
and
other
bits
*
making
sure
that
X86_EFLAGS_IF
has
the
right
value
(
and
other
bits
in
that
byte
are
0
),
but
other
bits
in
the
return
value
are
*
in
that
byte
are
0
),
but
other
bits
in
the
return
value
are
undefined
.
We
need
to
toggle
the
state
of
the
bit
,
because
*
undefined
.
We
need
to
toggle
the
state
of
the
bit
,
because
Xen
and
Xen
and
x86
use
opposite
senses
(
mask
vs
enable
)
.
*
x86
use
opposite
senses
(
mask
vs
enable
)
.
*/
*/
ENTRY
(
xen_save_fl_direct
)
ENTRY
(
xen_save_fl_direct
)
testb
$
0xff
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
testb
$
0xff
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
setz
%
ah
setz
%
ah
addb
%
ah
,%
ah
addb
%
ah
,
%
ah
ENDPATCH
(
xen_save_fl_direct
)
ENDPATCH
(
xen_save_fl_direct
)
ret
ret
ENDPROC
(
xen_save_fl_direct
)
ENDPROC
(
xen_save_fl_direct
)
...
@@ -73,12 +75,11 @@ ENDPATCH(xen_save_fl_direct)
...
@@ -73,12 +75,11 @@ ENDPATCH(xen_save_fl_direct)
/*
/*
In
principle
the
caller
should
be
passing
us
a
value
return
*
In
principle
the
caller
should
be
passing
us
a
value
return
from
from
xen_save_fl_direct
,
but
for
robustness
sake
we
test
only
*
xen_save_fl_direct
,
but
for
robustness
sake
we
test
only
the
the
X86_EFLAGS_IF
flag
rather
than
the
whole
byte
.
After
*
X86_EFLAGS_IF
flag
rather
than
the
whole
byte
.
After
setting
the
setting
the
interrupt
mask
state
,
it
checks
for
unmasked
*
interrupt
mask
state
,
it
checks
for
unmasked
pending
events
and
pending
events
and
enters
the
hypervisor
to
get
them
delivered
*
enters
the
hypervisor
to
get
them
delivered
if
so
.
if
so
.
*/
*/
ENTRY
(
xen_restore_fl_direct
)
ENTRY
(
xen_restore_fl_direct
)
#ifdef CONFIG_X86_64
#ifdef CONFIG_X86_64
...
@@ -87,9 +88,11 @@ ENTRY(xen_restore_fl_direct)
...
@@ -87,9 +88,11 @@ ENTRY(xen_restore_fl_direct)
testb
$X86_EFLAGS_IF
>>
8
,
%
ah
testb
$X86_EFLAGS_IF
>>
8
,
%
ah
#endif
#endif
setz
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
setz
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_mask
/
*
Preempt
here
doesn
't matter because that will deal with
/
*
any
pending
interrupts
.
The
pending
check
may
end
up
being
*
Preempt
here
doesn
't matter because that will deal with any
run
on
the
wrong
CPU
,
but
that
doesn
't hurt. */
*
pending
interrupts
.
The
pending
check
may
end
up
being
run
*
on
the
wrong
CPU
,
but
that
doesn
't hurt.
*/
/
*
check
for
unmasked
and
pending
*/
/
*
check
for
unmasked
and
pending
*/
cmpw
$
0x0001
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_pending
cmpw
$
0x0001
,
PER_CPU_VAR
(
xen_vcpu_info
)
+
XEN_vcpu_info_pending
...
@@ -103,8 +106,8 @@ ENDPATCH(xen_restore_fl_direct)
...
@@ -103,8 +106,8 @@ ENDPATCH(xen_restore_fl_direct)
/*
/*
Force
an
event
check
by
making
a
hypercall
,
*
Force
an
event
check
by
making
a
hypercall
,
but
preserve
regs
but
preserve
regs
before
making
the
call
.
*
before
making
the
call
.
*/
*/
check_events
:
check_events
:
#ifdef CONFIG_X86_32
#ifdef CONFIG_X86_32
...
@@ -137,4 +140,3 @@ check_events:
...
@@ -137,4 +140,3 @@ check_events:
pop
%
rax
pop
%
rax
#endif
#endif
ret
ret
arch/x86/xen/xen-asm_32.S
View file @
ab639f35
This diff is collapsed.
Click to expand it.
arch/x86/xen/xen-asm_64.S
View file @
ab639f35
/*
/*
Asm
versions
of
Xen
pv
-
ops
,
suitable
for
either
direct
use
or
inlining
.
*
Asm
versions
of
Xen
pv
-
ops
,
suitable
for
either
direct
use
or
The
inline
versions
are
the
same
as
the
direct
-
use
versions
,
with
th
e
*
inlining
.
The
inline
versions
are
the
same
as
the
direct
-
us
e
pre
-
and
post
-
amble
chopped
off
.
*
versions
,
with
the
pre
-
and
post
-
amble
chopped
off
.
*
This
code
is
encoded
for
size
rather
than
absolute
efficiency
,
*
This
code
is
encoded
for
size
rather
than
absolute
efficiency
,
with
with
a
view
to
being
able
to
inline
as
much
as
possible
.
*
a
view
to
being
able
to
inline
as
much
as
possible
.
*
We
only
bother
with
direct
forms
(
ie
,
vcpu
in
pda
)
of
the
operations
*
We
only
bother
with
direct
forms
(
ie
,
vcpu
in
pda
)
of
the
here
; the indirect forms are better handled in C, since they'r
e
*
operations
here
; the indirect forms are better handled in C, sinc
e
generally
too
large
to
inline
anyway
.
*
they
're
generally too large to inline anyway.
*/
*/
#include <asm/errno.h>
#include <asm/errno.h>
...
@@ -21,25 +21,25 @@
...
@@ -21,25 +21,25 @@
#include "xen-asm.h"
#include "xen-asm.h"
ENTRY
(
xen_adjust_exception_frame
)
ENTRY
(
xen_adjust_exception_frame
)
mov
8
+
0
(%
rsp
),%
rcx
mov
8
+
0
(%
rsp
),
%
rcx
mov
8
+
8
(%
rsp
),%
r11
mov
8
+
8
(%
rsp
),
%
r11
ret
$
16
ret
$
16
hypercall_iret
=
hypercall_page
+
__HYPERVISOR_iret
*
32
hypercall_iret
=
hypercall_page
+
__HYPERVISOR_iret
*
32
/*
/*
Xen64
iret
frame
:
*
Xen64
iret
frame
:
*
ss
*
ss
rsp
*
rsp
rflags
*
rflags
cs
*
cs
rip
<--
standard
iret
frame
*
rip
<--
standard
iret
frame
*
flags
*
flags
*
rcx
}
*
rcx
}
r11
}<--
pushed
by
hypercall
page
*
r11
}<--
pushed
by
hypercall
page
rsp
->
rax
}
*
rsp
->
rax
}
*/
*/
ENTRY
(
xen_iret
)
ENTRY
(
xen_iret
)
pushq
$
0
pushq
$
0
...
@@ -48,8 +48,8 @@ ENDPATCH(xen_iret)
...
@@ -48,8 +48,8 @@ ENDPATCH(xen_iret)
RELOC
(
xen_iret
,
1
b
+1)
RELOC
(
xen_iret
,
1
b
+1)
/*
/*
sysexit
is
not
used
for
64
-
bit
processes
,
so
it
's
*
sysexit
is
not
used
for
64
-
bit
processes
,
so
it
's only ever used to
only
ever
used
to
return
to
32
-
bit
compat
userspace
.
*
return
to
32
-
bit
compat
userspace
.
*/
*/
ENTRY
(
xen_sysexit
)
ENTRY
(
xen_sysexit
)
pushq
$
__USER32_DS
pushq
$
__USER32_DS
...
@@ -64,10 +64,12 @@ ENDPATCH(xen_sysexit)
...
@@ -64,10 +64,12 @@ ENDPATCH(xen_sysexit)
RELOC
(
xen_sysexit
,
1
b
+1)
RELOC
(
xen_sysexit
,
1
b
+1)
ENTRY
(
xen_sysret64
)
ENTRY
(
xen_sysret64
)
/
*
We
're already on the usermode stack at this point, but still
/
*
with
the
kernel
gs
,
so
we
can
easily
switch
back
*/
*
We
're already on the usermode stack at this point, but
*
still
with
the
kernel
gs
,
so
we
can
easily
switch
back
*/
movq
%
rsp
,
PER_CPU_VAR
(
old_rsp
)
movq
%
rsp
,
PER_CPU_VAR
(
old_rsp
)
movq
PER_CPU_VAR
(
kernel_stack
),%
rsp
movq
PER_CPU_VAR
(
kernel_stack
),
%
rsp
pushq
$
__USER_DS
pushq
$
__USER_DS
pushq
PER_CPU_VAR
(
old_rsp
)
pushq
PER_CPU_VAR
(
old_rsp
)
...
@@ -81,8 +83,10 @@ ENDPATCH(xen_sysret64)
...
@@ -81,8 +83,10 @@ ENDPATCH(xen_sysret64)
RELOC
(
xen_sysret64
,
1
b
+1)
RELOC
(
xen_sysret64
,
1
b
+1)
ENTRY
(
xen_sysret32
)
ENTRY
(
xen_sysret32
)
/
*
We
're already on the usermode stack at this point, but still
/
*
with
the
kernel
gs
,
so
we
can
easily
switch
back
*/
*
We
're already on the usermode stack at this point, but
*
still
with
the
kernel
gs
,
so
we
can
easily
switch
back
*/
movq
%
rsp
,
PER_CPU_VAR
(
old_rsp
)
movq
%
rsp
,
PER_CPU_VAR
(
old_rsp
)
movq
PER_CPU_VAR
(
kernel_stack
),
%
rsp
movq
PER_CPU_VAR
(
kernel_stack
),
%
rsp
...
@@ -98,28 +102,27 @@ ENDPATCH(xen_sysret32)
...
@@ -98,28 +102,27 @@ ENDPATCH(xen_sysret32)
RELOC
(
xen_sysret32
,
1
b
+1)
RELOC
(
xen_sysret32
,
1
b
+1)
/*
/*
Xen
handles
syscall
callbacks
much
like
ordinary
exceptions
,
*
Xen
handles
syscall
callbacks
much
like
ordinary
exceptions
,
which
which
means
we
have
:
*
means
we
have
:
-
kernel
gs
*
-
kernel
gs
-
kernel
rsp
*
-
kernel
rsp
-
an
iret
-
like
stack
frame
on
the
stack
(
including
rcx
and
r11
)
:
*
-
an
iret
-
like
stack
frame
on
the
stack
(
including
rcx
and
r11
)
:
ss
*
ss
rsp
*
rsp
rflags
*
rflags
cs
*
cs
rip
*
rip
r11
*
r11
rsp
->
rcx
*
rsp
->
rcx
*
In
all
the
entrypoints
,
we
undo
all
that
to
make
it
look
*
In
all
the
entrypoints
,
we
undo
all
that
to
make
it
look
like
a
like
a
CPU
-
generated
syscall
/
sysenter
and
jump
to
the
normal
*
CPU
-
generated
syscall
/
sysenter
and
jump
to
the
normal
entrypoint
.
entrypoint
.
*/
*/
.
macro
undo_xen_syscall
.
macro
undo_xen_syscall
mov
0
*
8
(%
rsp
),%
rcx
mov
0
*
8
(%
rsp
),
%
rcx
mov
1
*
8
(%
rsp
),%
r11
mov
1
*
8
(%
rsp
),
%
r11
mov
5
*
8
(%
rsp
),%
rsp
mov
5
*
8
(%
rsp
),
%
rsp
.
endm
.
endm
/*
Normal
64
-
bit
system
call
target
*/
/*
Normal
64
-
bit
system
call
target
*/
...
@@ -146,7 +149,7 @@ ENDPROC(xen_sysenter_target)
...
@@ -146,7 +149,7 @@ ENDPROC(xen_sysenter_target)
ENTRY
(
xen_syscall32_target
)
ENTRY
(
xen_syscall32_target
)
ENTRY
(
xen_sysenter_target
)
ENTRY
(
xen_sysenter_target
)
lea
16
(%
rsp
),
%
rsp
/*
strip
%
rcx
,%
r11
*/
lea
16
(%
rsp
),
%
rsp
/*
strip
%
rcx
,
%
r11
*/
mov
$
-
ENOSYS
,
%
rax
mov
$
-
ENOSYS
,
%
rax
pushq
$VGCF_in_syscall
pushq
$VGCF_in_syscall
jmp
hypercall_iret
jmp
hypercall_iret
...
...
drivers/misc/sgi-gru/gru.h
View file @
ab639f35
...
@@ -19,8 +19,6 @@
...
@@ -19,8 +19,6 @@
#ifndef __GRU_H__
#ifndef __GRU_H__
#define __GRU_H__
#define __GRU_H__
#include <asm/uv/uv.h>
/*
/*
* GRU architectural definitions
* GRU architectural definitions
*/
*/
...
...
drivers/misc/sgi-gru/grufile.c
View file @
ab639f35
...
@@ -36,23 +36,11 @@
...
@@ -36,23 +36,11 @@
#include <linux/interrupt.h>
#include <linux/interrupt.h>
#include <linux/proc_fs.h>
#include <linux/proc_fs.h>
#include <linux/uaccess.h>
#include <linux/uaccess.h>
#include <asm/uv/uv.h>
#include "gru.h"
#include "gru.h"
#include "grulib.h"
#include "grulib.h"
#include "grutables.h"
#include "grutables.h"
#if defined CONFIG_X86_64
#include <asm/genapic.h>
#include <asm/irq.h>
#define IS_UV() is_uv_system()
#elif defined CONFIG_IA64
#include <asm/system.h>
#include <asm/sn/simulator.h>
/* temp support for running on hardware simulator */
#define IS_UV() IS_MEDUSA() || ia64_platform_is("uv")
#else
#define IS_UV() 0
#endif
#include <asm/uv/uv_hub.h>
#include <asm/uv/uv_hub.h>
#include <asm/uv/uv_mmrs.h>
#include <asm/uv/uv_mmrs.h>
...
@@ -381,7 +369,7 @@ static int __init gru_init(void)
...
@@ -381,7 +369,7 @@ static int __init gru_init(void)
char
id
[
10
];
char
id
[
10
];
void
*
gru_start_vaddr
;
void
*
gru_start_vaddr
;
if
(
!
IS_UV
())
if
(
!
is_uv_system
())
return
0
;
return
0
;
#if defined CONFIG_IA64
#if defined CONFIG_IA64
...
@@ -451,7 +439,7 @@ static void __exit gru_exit(void)
...
@@ -451,7 +439,7 @@ static void __exit gru_exit(void)
int
order
=
get_order
(
sizeof
(
struct
gru_state
)
*
int
order
=
get_order
(
sizeof
(
struct
gru_state
)
*
GRU_CHIPLETS_PER_BLADE
);
GRU_CHIPLETS_PER_BLADE
);
if
(
!
IS_UV
())
if
(
!
is_uv_system
())
return
;
return
;
for
(
i
=
0
;
i
<
GRU_CHIPLETS_PER_BLADE
;
i
++
)
for
(
i
=
0
;
i
<
GRU_CHIPLETS_PER_BLADE
;
i
++
)
...
...
drivers/misc/sgi-xp/xp.h
View file @
ab639f35
...
@@ -15,21 +15,19 @@
...
@@ -15,21 +15,19 @@
#include <linux/mutex.h>
#include <linux/mutex.h>
#if defined CONFIG_X86_UV || defined CONFIG_IA64_SGI_UV
#include <asm/uv/uv.h>
#include <asm/uv/uv.h>
#define is_uv() is_uv_system()
#endif
#ifndef is_uv
#define is_uv() 0
#endif
#if
def
CONFIG_IA64
#if
defined
CONFIG_IA64
#include <asm/system.h>
#include <asm/system.h>
#include <asm/sn/arch.h>
/* defines is_shub1() and is_shub2() */
#include <asm/sn/arch.h>
/* defines is_shub1() and is_shub2() */
#define is_shub() ia64_platform_is("sn2")
#define is_shub() ia64_platform_is("sn2")
#ifdef CONFIG_IA64_SGI_UV
#define is_uv() ia64_platform_is("uv")
#else
#define is_uv() 0
#endif
#endif
#ifdef CONFIG_X86_64
#include <asm/genapic.h>
#define is_uv() is_uv_system()
#endif
#endif
#ifndef is_shub1
#ifndef is_shub1
...
@@ -44,10 +42,6 @@
...
@@ -44,10 +42,6 @@
#define is_shub() 0
#define is_shub() 0
#endif
#endif
#ifndef is_uv
#define is_uv() 0
#endif
#ifdef USE_DBUG_ON
#ifdef USE_DBUG_ON
#define DBUG_ON(condition) BUG_ON(condition)
#define DBUG_ON(condition) BUG_ON(condition)
#else
#else
...
...
include/linux/elfcore.h
View file @
ab639f35
...
@@ -111,6 +111,15 @@ static inline void elf_core_copy_regs(elf_gregset_t *elfregs, struct pt_regs *re
...
@@ -111,6 +111,15 @@ static inline void elf_core_copy_regs(elf_gregset_t *elfregs, struct pt_regs *re
#endif
#endif
}
}
static
inline
void
elf_core_copy_kernel_regs
(
elf_gregset_t
*
elfregs
,
struct
pt_regs
*
regs
)
{
#ifdef ELF_CORE_COPY_KERNEL_REGS
ELF_CORE_COPY_KERNEL_REGS
((
*
elfregs
),
regs
);
#else
elf_core_copy_regs
(
elfregs
,
regs
);
#endif
}
static
inline
int
elf_core_copy_task_regs
(
struct
task_struct
*
t
,
elf_gregset_t
*
elfregs
)
static
inline
int
elf_core_copy_task_regs
(
struct
task_struct
*
t
,
elf_gregset_t
*
elfregs
)
{
{
#ifdef ELF_CORE_COPY_TASK_REGS
#ifdef ELF_CORE_COPY_TASK_REGS
...
...
include/linux/percpu.h
View file @
ab639f35
...
@@ -8,8 +8,15 @@
...
@@ -8,8 +8,15 @@
#include <asm/percpu.h>
#include <asm/percpu.h>
#ifndef PER_CPU_BASE_SECTION
#ifdef CONFIG_SMP
#ifdef CONFIG_SMP
#define PER_CPU_BASE_SECTION ".data.percpu"
#define PER_CPU_BASE_SECTION ".data.percpu"
#else
#define PER_CPU_BASE_SECTION ".data"
#endif
#endif
#ifdef CONFIG_SMP
#ifdef MODULE
#ifdef MODULE
#define PER_CPU_SHARED_ALIGNED_SECTION ""
#define PER_CPU_SHARED_ALIGNED_SECTION ""
...
@@ -20,7 +27,6 @@
...
@@ -20,7 +27,6 @@
#else
#else
#define PER_CPU_BASE_SECTION ".data"
#define PER_CPU_SHARED_ALIGNED_SECTION ""
#define PER_CPU_SHARED_ALIGNED_SECTION ""
#define PER_CPU_FIRST_SECTION ""
#define PER_CPU_FIRST_SECTION ""
...
...
kernel/kexec.c
View file @
ab639f35
...
@@ -1130,7 +1130,7 @@ void crash_save_cpu(struct pt_regs *regs, int cpu)
...
@@ -1130,7 +1130,7 @@ void crash_save_cpu(struct pt_regs *regs, int cpu)
return
;
return
;
memset
(
&
prstatus
,
0
,
sizeof
(
prstatus
));
memset
(
&
prstatus
,
0
,
sizeof
(
prstatus
));
prstatus
.
pr_pid
=
current
->
pid
;
prstatus
.
pr_pid
=
current
->
pid
;
elf_core_copy_regs
(
&
prstatus
.
pr_reg
,
regs
);
elf_core_copy_
kernel_
regs
(
&
prstatus
.
pr_reg
,
regs
);
buf
=
append_elf_note
(
buf
,
KEXEC_CORE_NOTE_NAME
,
NT_PRSTATUS
,
buf
=
append_elf_note
(
buf
,
KEXEC_CORE_NOTE_NAME
,
NT_PRSTATUS
,
&
prstatus
,
sizeof
(
prstatus
));
&
prstatus
,
sizeof
(
prstatus
));
final_note
(
buf
);
final_note
(
buf
);
...
...
kernel/panic.c
View file @
ab639f35
...
@@ -359,10 +359,6 @@ EXPORT_SYMBOL(warn_slowpath);
...
@@ -359,10 +359,6 @@ EXPORT_SYMBOL(warn_slowpath);
#ifdef CONFIG_CC_STACKPROTECTOR
#ifdef CONFIG_CC_STACKPROTECTOR
#ifndef GCC_HAS_SP
#warning You have selected the CONFIG_CC_STACKPROTECTOR option, but the gcc used does not support this.
#endif
/*
/*
* Called when gcc's -fstack-protector feature is used, and
* Called when gcc's -fstack-protector feature is used, and
* gcc detects corruption of the on-stack canary value
* gcc detects corruption of the on-stack canary value
...
...
scripts/gcc-x86_32-has-stack-protector.sh
0 → 100644
View file @
ab639f35
#!/bin/sh
echo
"int foo(void) { char X[200]; return 3; }"
|
$*
-S
-xc
-c
-O0
-fstack-protector
-
-o
- 2> /dev/null |
grep
-q
"%gs"
if
[
"
$?
"
-eq
"0"
]
;
then
echo
y
else
echo
n
fi
scripts/gcc-x86_64-has-stack-protector.sh
View file @
ab639f35
#!/bin/sh
#!/bin/sh
echo
"int foo(void) { char X[200]; return 3; }"
|
$
1
-S
-xc
-c
-O0
-mcmodel
=
kernel
-fstack-protector
-
-o
- 2> /dev/null |
grep
-q
"%gs"
echo
"int foo(void) { char X[200]; return 3; }"
|
$
*
-S
-xc
-c
-O0
-mcmodel
=
kernel
-fstack-protector
-
-o
- 2> /dev/null |
grep
-q
"%gs"
if
[
"
$?
"
-eq
"0"
]
;
then
if
[
"
$?
"
-eq
"0"
]
;
then
echo
$2
echo
y
else
echo
n
fi
fi
scripts/mod/modpost.c
View file @
ab639f35
...
@@ -415,8 +415,9 @@ static int parse_elf(struct elf_info *info, const char *filename)
...
@@ -415,8 +415,9 @@ static int parse_elf(struct elf_info *info, const char *filename)
const
char
*
secstrings
const
char
*
secstrings
=
(
void
*
)
hdr
+
sechdrs
[
hdr
->
e_shstrndx
].
sh_offset
;
=
(
void
*
)
hdr
+
sechdrs
[
hdr
->
e_shstrndx
].
sh_offset
;
const
char
*
secname
;
const
char
*
secname
;
int
nobits
=
sechdrs
[
i
].
sh_type
==
SHT_NOBITS
;
if
(
sechdrs
[
i
].
sh_offset
>
info
->
size
)
{
if
(
!
nobits
&&
sechdrs
[
i
].
sh_offset
>
info
->
size
)
{
fatal
(
"%s is truncated. sechdrs[i].sh_offset=%lu > "
fatal
(
"%s is truncated. sechdrs[i].sh_offset=%lu > "
"sizeof(*hrd)=%zu
\n
"
,
filename
,
"sizeof(*hrd)=%zu
\n
"
,
filename
,
(
unsigned
long
)
sechdrs
[
i
].
sh_offset
,
(
unsigned
long
)
sechdrs
[
i
].
sh_offset
,
...
@@ -425,6 +426,8 @@ static int parse_elf(struct elf_info *info, const char *filename)
...
@@ -425,6 +426,8 @@ static int parse_elf(struct elf_info *info, const char *filename)
}
}
secname
=
secstrings
+
sechdrs
[
i
].
sh_name
;
secname
=
secstrings
+
sechdrs
[
i
].
sh_name
;
if
(
strcmp
(
secname
,
".modinfo"
)
==
0
)
{
if
(
strcmp
(
secname
,
".modinfo"
)
==
0
)
{
if
(
nobits
)
fatal
(
"%s has NOBITS .modinfo
\n
"
,
filename
);
info
->
modinfo
=
(
void
*
)
hdr
+
sechdrs
[
i
].
sh_offset
;
info
->
modinfo
=
(
void
*
)
hdr
+
sechdrs
[
i
].
sh_offset
;
info
->
modinfo_len
=
sechdrs
[
i
].
sh_size
;
info
->
modinfo_len
=
sechdrs
[
i
].
sh_size
;
}
else
if
(
strcmp
(
secname
,
"__ksymtab"
)
==
0
)
}
else
if
(
strcmp
(
secname
,
"__ksymtab"
)
==
0
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment