Commit ed1cd6de authored by Christophe Leroy's avatar Christophe Leroy Committed by Michael Ellerman

powerpc: Activate CONFIG_THREAD_INFO_IN_TASK

This patch activates CONFIG_THREAD_INFO_IN_TASK which
moves the thread_info into task_struct.

Moving thread_info into task_struct has the following advantages:
  - It protects thread_info from corruption in the case of stack
    overflows.
  - Its address is harder to determine if stack addresses are leaked,
    making a number of attacks more difficult.

This has the following consequences:
  - thread_info is now located at the beginning of task_struct.
  - The 'cpu' field is now in task_struct, and only exists when
    CONFIG_SMP is active.
  - thread_info doesn't have anymore the 'task' field.

This patch:
  - Removes all recopy of thread_info struct when the stack changes.
  - Changes the CURRENT_THREAD_INFO() macro to point to current.
  - Selects CONFIG_THREAD_INFO_IN_TASK.
  - Modifies raw_smp_processor_id() to get ->cpu from current without
    including linux/sched.h to avoid circular inclusion and without
    including asm/asm-offsets.h to avoid symbol names duplication
    between ASM constants and C constants.
  - Modifies klp_init_thread_info() to take a task_struct pointer
    argument.
Signed-off-by: default avatarChristophe Leroy <christophe.leroy@c-s.fr>
Reviewed-by: default avatarNicholas Piggin <npiggin@gmail.com>
[mpe: Add task_stack.h to livepatch.h to fix build fails]
Signed-off-by: default avatarMichael Ellerman <mpe@ellerman.id.au>
parent 7aef3766
...@@ -238,6 +238,7 @@ config PPC ...@@ -238,6 +238,7 @@ config PPC
select RTC_LIB select RTC_LIB
select SPARSE_IRQ select SPARSE_IRQ
select SYSCTL_EXCEPTION_TRACE select SYSCTL_EXCEPTION_TRACE
select THREAD_INFO_IN_TASK
select VIRT_TO_BUS if !PPC64 select VIRT_TO_BUS if !PPC64
# #
# Please keep this list sorted alphabetically. # Please keep this list sorted alphabetically.
......
...@@ -427,6 +427,13 @@ else ...@@ -427,6 +427,13 @@ else
endif endif
endif endif
ifdef CONFIG_SMP
prepare: task_cpu_prepare
task_cpu_prepare: prepare0
$(eval KBUILD_CFLAGS += -D_TASK_CPU=$(shell awk '{if ($$2 == "TI_CPU") print $$3;}' include/generated/asm-offsets.h))
endif
# Check toolchain versions: # Check toolchain versions:
# - gcc-4.6 is the minimum kernel-wide version so nothing required. # - gcc-4.6 is the minimum kernel-wide version so nothing required.
checkbin: checkbin:
......
...@@ -51,9 +51,6 @@ struct pt_regs; ...@@ -51,9 +51,6 @@ struct pt_regs;
extern struct thread_info *critirq_ctx[NR_CPUS]; extern struct thread_info *critirq_ctx[NR_CPUS];
extern struct thread_info *dbgirq_ctx[NR_CPUS]; extern struct thread_info *dbgirq_ctx[NR_CPUS];
extern struct thread_info *mcheckirq_ctx[NR_CPUS]; extern struct thread_info *mcheckirq_ctx[NR_CPUS];
extern void exc_lvl_ctx_init(void);
#else
#define exc_lvl_ctx_init()
#endif #endif
/* /*
...@@ -62,7 +59,6 @@ extern void exc_lvl_ctx_init(void); ...@@ -62,7 +59,6 @@ extern void exc_lvl_ctx_init(void);
extern struct thread_info *hardirq_ctx[NR_CPUS]; extern struct thread_info *hardirq_ctx[NR_CPUS];
extern struct thread_info *softirq_ctx[NR_CPUS]; extern struct thread_info *softirq_ctx[NR_CPUS];
extern void irq_ctx_init(void);
void call_do_softirq(void *sp); void call_do_softirq(void *sp);
void call_do_irq(struct pt_regs *regs, void *sp); void call_do_irq(struct pt_regs *regs, void *sp);
extern void do_IRQ(struct pt_regs *regs); extern void do_IRQ(struct pt_regs *regs);
......
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/ftrace.h> #include <linux/ftrace.h>
#include <linux/sched/task_stack.h>
#ifdef CONFIG_LIVEPATCH #ifdef CONFIG_LIVEPATCH
static inline int klp_check_compiler_support(void) static inline int klp_check_compiler_support(void)
...@@ -43,13 +44,13 @@ static inline unsigned long klp_get_ftrace_location(unsigned long faddr) ...@@ -43,13 +44,13 @@ static inline unsigned long klp_get_ftrace_location(unsigned long faddr)
return ftrace_location_range(faddr, faddr + 16); return ftrace_location_range(faddr, faddr + 16);
} }
static inline void klp_init_thread_info(struct thread_info *ti) static inline void klp_init_thread_info(struct task_struct *p)
{ {
/* + 1 to account for STACK_END_MAGIC */ /* + 1 to account for STACK_END_MAGIC */
ti->livepatch_sp = (unsigned long *)(ti + 1) + 1; task_thread_info(p)->livepatch_sp = end_of_stack(p) + 1;
} }
#else #else
static void klp_init_thread_info(struct thread_info *ti) { } static inline void klp_init_thread_info(struct task_struct *p) { }
#endif /* CONFIG_LIVEPATCH */ #endif /* CONFIG_LIVEPATCH */
#endif /* _ASM_POWERPC_LIVEPATCH_H */ #endif /* _ASM_POWERPC_LIVEPATCH_H */
...@@ -83,7 +83,22 @@ int is_cpu_dead(unsigned int cpu); ...@@ -83,7 +83,22 @@ int is_cpu_dead(unsigned int cpu);
/* 32-bit */ /* 32-bit */
extern int smp_hw_index[]; extern int smp_hw_index[];
#define raw_smp_processor_id() (current_thread_info()->cpu) /*
* This is particularly ugly: it appears we can't actually get the definition
* of task_struct here, but we need access to the CPU this task is running on.
* Instead of using task_struct we're using _TASK_CPU which is extracted from
* asm-offsets.h by kbuild to get the current processor ID.
*
* This also needs to be safeguarded when building asm-offsets.s because at
* that time _TASK_CPU is not defined yet. It could have been guarded by
* _TASK_CPU itself, but we want the build to fail if _TASK_CPU is missing
* when building something else than asm-offsets.s
*/
#ifdef GENERATING_ASM_OFFSETS
#define raw_smp_processor_id() (0)
#else
#define raw_smp_processor_id() (*(unsigned int *)((void *)current + _TASK_CPU))
#endif
#define hard_smp_processor_id() (smp_hw_index[smp_processor_id()]) #define hard_smp_processor_id() (smp_hw_index[smp_processor_id()])
static inline int get_hard_smp_processor_id(int cpu) static inline int get_hard_smp_processor_id(int cpu)
......
...@@ -18,9 +18,9 @@ ...@@ -18,9 +18,9 @@
#define THREAD_SIZE (1 << THREAD_SHIFT) #define THREAD_SIZE (1 << THREAD_SHIFT)
#ifdef CONFIG_PPC64 #ifdef CONFIG_PPC64
#define CURRENT_THREAD_INFO(dest, sp) stringify_in_c(clrrdi dest, sp, THREAD_SHIFT) #define CURRENT_THREAD_INFO(dest, sp) stringify_in_c(ld dest, PACACURRENT(r13))
#else #else
#define CURRENT_THREAD_INFO(dest, sp) stringify_in_c(rlwinm dest, sp, 0, 0, 31-THREAD_SHIFT) #define CURRENT_THREAD_INFO(dest, sp) stringify_in_c(mr dest, r2)
#endif #endif
#ifndef __ASSEMBLY__ #ifndef __ASSEMBLY__
...@@ -34,8 +34,6 @@ ...@@ -34,8 +34,6 @@
* low level task data. * low level task data.
*/ */
struct thread_info { struct thread_info {
struct task_struct *task; /* main task structure */
int cpu; /* cpu we're on */
int preempt_count; /* 0 => preemptable, int preempt_count; /* 0 => preemptable,
<0 => BUG */ <0 => BUG */
unsigned long local_flags; /* private flags for thread */ unsigned long local_flags; /* private flags for thread */
...@@ -58,8 +56,6 @@ struct thread_info { ...@@ -58,8 +56,6 @@ struct thread_info {
*/ */
#define INIT_THREAD_INFO(tsk) \ #define INIT_THREAD_INFO(tsk) \
{ \ { \
.task = &tsk, \
.cpu = 0, \
.preempt_count = INIT_PREEMPT_COUNT, \ .preempt_count = INIT_PREEMPT_COUNT, \
.flags = 0, \ .flags = 0, \
} }
...@@ -67,15 +63,6 @@ struct thread_info { ...@@ -67,15 +63,6 @@ struct thread_info {
#define THREAD_SIZE_ORDER (THREAD_SHIFT - PAGE_SHIFT) #define THREAD_SIZE_ORDER (THREAD_SHIFT - PAGE_SHIFT)
/* how to get the thread information struct from C */ /* how to get the thread information struct from C */
static inline struct thread_info *current_thread_info(void)
{
unsigned long val;
asm (CURRENT_THREAD_INFO(%0,1) : "=r" (val));
return (struct thread_info *)val;
}
extern int arch_dup_task_struct(struct task_struct *dst, struct task_struct *src); extern int arch_dup_task_struct(struct task_struct *dst, struct task_struct *src);
#ifdef CONFIG_PPC_BOOK3S_64 #ifdef CONFIG_PPC_BOOK3S_64
......
...@@ -13,6 +13,8 @@ ...@@ -13,6 +13,8 @@
* 2 of the License, or (at your option) any later version. * 2 of the License, or (at your option) any later version.
*/ */
#define GENERATING_ASM_OFFSETS /* asm/smp.h */
#include <linux/compat.h> #include <linux/compat.h>
#include <linux/signal.h> #include <linux/signal.h>
#include <linux/sched.h> #include <linux/sched.h>
...@@ -97,6 +99,9 @@ int main(void) ...@@ -97,6 +99,9 @@ int main(void)
#endif #endif
#endif /* CONFIG_PPC64 */ #endif /* CONFIG_PPC64 */
OFFSET(TASK_STACK, task_struct, stack); OFFSET(TASK_STACK, task_struct, stack);
#ifdef CONFIG_SMP
OFFSET(TI_CPU, task_struct, cpu);
#endif
#ifdef CONFIG_LIVEPATCH #ifdef CONFIG_LIVEPATCH
OFFSET(TI_livepatch_sp, thread_info, livepatch_sp); OFFSET(TI_livepatch_sp, thread_info, livepatch_sp);
...@@ -164,8 +169,6 @@ int main(void) ...@@ -164,8 +169,6 @@ int main(void)
OFFSET(TI_FLAGS, thread_info, flags); OFFSET(TI_FLAGS, thread_info, flags);
OFFSET(TI_LOCAL_FLAGS, thread_info, local_flags); OFFSET(TI_LOCAL_FLAGS, thread_info, local_flags);
OFFSET(TI_PREEMPT, thread_info, preempt_count); OFFSET(TI_PREEMPT, thread_info, preempt_count);
OFFSET(TI_TASK, thread_info, task);
OFFSET(TI_CPU, thread_info, cpu);
#ifdef CONFIG_PPC64 #ifdef CONFIG_PPC64
OFFSET(DCACHEL1BLOCKSIZE, ppc64_caches, l1d.block_size); OFFSET(DCACHEL1BLOCKSIZE, ppc64_caches, l1d.block_size);
......
...@@ -1165,10 +1165,6 @@ ret_from_debug_exc: ...@@ -1165,10 +1165,6 @@ ret_from_debug_exc:
mfspr r9,SPRN_SPRG_THREAD mfspr r9,SPRN_SPRG_THREAD
lwz r10,SAVED_KSP_LIMIT(r1) lwz r10,SAVED_KSP_LIMIT(r1)
stw r10,KSP_LIMIT(r9) stw r10,KSP_LIMIT(r9)
lwz r9,TASK_STACK-THREAD(r9)
CURRENT_THREAD_INFO(r10, r1)
lwz r10,TI_PREEMPT(r10)
stw r10,TI_PREEMPT(r9)
RESTORE_xSRR(SRR0,SRR1); RESTORE_xSRR(SRR0,SRR1);
RESTORE_xSRR(CSRR0,CSRR1); RESTORE_xSRR(CSRR0,CSRR1);
RESTORE_MMU_REGS; RESTORE_MMU_REGS;
...@@ -1291,10 +1287,13 @@ BEGIN_FTR_SECTION ...@@ -1291,10 +1287,13 @@ BEGIN_FTR_SECTION
END_FTR_SECTION_IFSET(CPU_FTR_601) END_FTR_SECTION_IFSET(CPU_FTR_601)
lwz r3,_TRAP(r1) lwz r3,_TRAP(r1)
andi. r0,r3,1 andi. r0,r3,1
beq 4f beq 5f
SAVE_NVGPRS(r1) SAVE_NVGPRS(r1)
rlwinm r3,r3,0,0,30 rlwinm r3,r3,0,0,30
stw r3,_TRAP(r1) stw r3,_TRAP(r1)
5: mfspr r2,SPRN_SPRG_THREAD
addi r2,r2,-THREAD
tovirt(r2,r2) /* set back r2 to current */
4: addi r3,r1,STACK_FRAME_OVERHEAD 4: addi r3,r1,STACK_FRAME_OVERHEAD
bl unrecoverable_exception bl unrecoverable_exception
/* shouldn't return */ /* shouldn't return */
......
...@@ -77,17 +77,6 @@ special_reg_save: ...@@ -77,17 +77,6 @@ special_reg_save:
andi. r3,r3,MSR_PR andi. r3,r3,MSR_PR
bnelr bnelr
/* Copy info into temporary exception thread info */
ld r11,PACAKSAVE(r13)
CURRENT_THREAD_INFO(r11, r11)
CURRENT_THREAD_INFO(r12, r1)
ld r10,TI_FLAGS(r11)
std r10,TI_FLAGS(r12)
ld r10,TI_PREEMPT(r11)
std r10,TI_PREEMPT(r12)
ld r10,TI_TASK(r11)
std r10,TI_TASK(r12)
/* /*
* Advance to the next TLB exception frame for handler * Advance to the next TLB exception frame for handler
* types that don't do it automatically. * types that don't do it automatically.
......
...@@ -834,9 +834,9 @@ __secondary_start: ...@@ -834,9 +834,9 @@ __secondary_start:
/* get current's stack and current */ /* get current's stack and current */
lis r1,secondary_ti@ha lis r1,secondary_ti@ha
tophys(r1,r1) tophys(r1,r1)
lwz r1,secondary_ti@l(r1) lwz r2,secondary_ti@l(r1)
tophys(r2,r1) tophys(r1,r2)
lwz r2,TI_TASK(r2) lwz r1,TASK_STACK(r1)
/* stack */ /* stack */
addi r1,r1,THREAD_SIZE-STACK_FRAME_OVERHEAD addi r1,r1,THREAD_SIZE-STACK_FRAME_OVERHEAD
......
...@@ -1021,8 +1021,8 @@ _GLOBAL(start_secondary_47x) ...@@ -1021,8 +1021,8 @@ _GLOBAL(start_secondary_47x)
/* Get current's stack and current */ /* Get current's stack and current */
lis r1,secondary_ti@ha lis r1,secondary_ti@ha
lwz r1,secondary_ti@l(r1) lwz r2,secondary_ti@l(r1)
lwz r2,TI_TASK(r1) lwz r1,TASK_STACK(r2)
/* Current stack pointer */ /* Current stack pointer */
addi r1,r1,THREAD_SIZE-STACK_FRAME_OVERHEAD addi r1,r1,THREAD_SIZE-STACK_FRAME_OVERHEAD
......
...@@ -155,13 +155,7 @@ END_BTB_FLUSH_SECTION ...@@ -155,13 +155,7 @@ END_BTB_FLUSH_SECTION
stw r10,GPR11(r11); \ stw r10,GPR11(r11); \
b 2f; \ b 2f; \
/* COMING FROM PRIV MODE */ \ /* COMING FROM PRIV MODE */ \
1: lwz r9,TI_FLAGS-EXC_LVL_FRAME_OVERHEAD(r11); \ 1: mr r11, r8; \
lwz r10,TI_PREEMPT-EXC_LVL_FRAME_OVERHEAD(r11); \
stw r9,TI_FLAGS-EXC_LVL_FRAME_OVERHEAD(r8); \
stw r10,TI_PREEMPT-EXC_LVL_FRAME_OVERHEAD(r8); \
lwz r9,TI_TASK-EXC_LVL_FRAME_OVERHEAD(r11); \
stw r9,TI_TASK-EXC_LVL_FRAME_OVERHEAD(r8); \
mr r11,r8; \
2: mfspr r8,SPRN_SPRG_RSCRATCH_##exc_level; \ 2: mfspr r8,SPRN_SPRG_RSCRATCH_##exc_level; \
stw r12,GPR12(r11); /* save various registers */\ stw r12,GPR12(r11); /* save various registers */\
mflr r10; \ mflr r10; \
......
...@@ -719,8 +719,7 @@ finish_tlb_load: ...@@ -719,8 +719,7 @@ finish_tlb_load:
/* Get the next_tlbcam_idx percpu var */ /* Get the next_tlbcam_idx percpu var */
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
lwz r12, TASK_STACK-THREAD(r12) lwz r15, TI_CPU-THREAD(r12)
lwz r15, TI_CPU(r12)
lis r14, __per_cpu_offset@h lis r14, __per_cpu_offset@h
ori r14, r14, __per_cpu_offset@l ori r14, r14, __per_cpu_offset@l
rlwinm r15, r15, 2, 0, 29 rlwinm r15, r15, 2, 0, 29
...@@ -1093,8 +1092,8 @@ __secondary_start: ...@@ -1093,8 +1092,8 @@ __secondary_start:
/* get current's stack and current */ /* get current's stack and current */
lis r1,secondary_ti@ha lis r1,secondary_ti@ha
lwz r1,secondary_ti@l(r1) lwz r2,secondary_ti@l(r1)
lwz r2,TI_TASK(r1) lwz r1,TASK_STACK(r2)
/* stack */ /* stack */
addi r1,r1,THREAD_SIZE-STACK_FRAME_OVERHEAD addi r1,r1,THREAD_SIZE-STACK_FRAME_OVERHEAD
......
...@@ -673,24 +673,9 @@ void do_IRQ(struct pt_regs *regs) ...@@ -673,24 +673,9 @@ void do_IRQ(struct pt_regs *regs)
set_irq_regs(old_regs); set_irq_regs(old_regs);
return; return;
} }
/* Prepare the thread_info in the irq stack */
irqtp->task = curtp->task;
irqtp->flags = 0;
/* Copy the preempt_count so that the [soft]irq checks work. */
irqtp->preempt_count = curtp->preempt_count;
/* Switch stack and call */ /* Switch stack and call */
call_do_irq(regs, irqtp); call_do_irq(regs, irqtp);
/* Restore stack limit */
irqtp->task = NULL;
/* Copy back updates to the thread_info */
if (irqtp->flags)
set_bits(irqtp->flags, &curtp->flags);
set_irq_regs(old_regs); set_irq_regs(old_regs);
} }
...@@ -698,85 +683,23 @@ void __init init_IRQ(void) ...@@ -698,85 +683,23 @@ void __init init_IRQ(void)
{ {
if (ppc_md.init_IRQ) if (ppc_md.init_IRQ)
ppc_md.init_IRQ(); ppc_md.init_IRQ();
exc_lvl_ctx_init();
irq_ctx_init();
} }
#if defined(CONFIG_BOOKE) || defined(CONFIG_40x) #if defined(CONFIG_BOOKE) || defined(CONFIG_40x)
struct thread_info *critirq_ctx[NR_CPUS] __read_mostly; struct thread_info *critirq_ctx[NR_CPUS] __read_mostly;
struct thread_info *dbgirq_ctx[NR_CPUS] __read_mostly; struct thread_info *dbgirq_ctx[NR_CPUS] __read_mostly;
struct thread_info *mcheckirq_ctx[NR_CPUS] __read_mostly; struct thread_info *mcheckirq_ctx[NR_CPUS] __read_mostly;
void exc_lvl_ctx_init(void)
{
struct thread_info *tp;
int i, cpu_nr;
for_each_possible_cpu(i) {
#ifdef CONFIG_PPC64
cpu_nr = i;
#else
#ifdef CONFIG_SMP
cpu_nr = get_hard_smp_processor_id(i);
#else
cpu_nr = 0;
#endif
#endif
tp = critirq_ctx[cpu_nr];
tp->cpu = cpu_nr;
tp->preempt_count = 0;
#ifdef CONFIG_BOOKE
tp = dbgirq_ctx[cpu_nr];
tp->cpu = cpu_nr;
tp->preempt_count = 0;
tp = mcheckirq_ctx[cpu_nr];
tp->cpu = cpu_nr;
tp->preempt_count = HARDIRQ_OFFSET;
#endif
}
}
#endif #endif
struct thread_info *softirq_ctx[NR_CPUS] __read_mostly; struct thread_info *softirq_ctx[NR_CPUS] __read_mostly;
struct thread_info *hardirq_ctx[NR_CPUS] __read_mostly; struct thread_info *hardirq_ctx[NR_CPUS] __read_mostly;
void irq_ctx_init(void)
{
struct thread_info *tp;
int i;
for_each_possible_cpu(i) {
tp = softirq_ctx[i];
tp->cpu = i;
klp_init_thread_info(tp);
tp = hardirq_ctx[i];
tp->cpu = i;
klp_init_thread_info(tp);
}
}
void do_softirq_own_stack(void) void do_softirq_own_stack(void)
{ {
struct thread_info *curtp, *irqtp; struct thread_info *irqtp;
curtp = current_thread_info();
irqtp = softirq_ctx[smp_processor_id()]; irqtp = softirq_ctx[smp_processor_id()];
irqtp->task = curtp->task;
irqtp->flags = 0;
call_do_softirq(irqtp); call_do_softirq(irqtp);
irqtp->task = NULL;
/* Set any flag that may have been set on the
* alternate stack
*/
if (irqtp->flags)
set_bits(irqtp->flags, &curtp->flags);
} }
irq_hw_number_t virq_to_hw(unsigned int virq) irq_hw_number_t virq_to_hw(unsigned int virq)
......
...@@ -151,41 +151,13 @@ static int kgdb_handle_breakpoint(struct pt_regs *regs) ...@@ -151,41 +151,13 @@ static int kgdb_handle_breakpoint(struct pt_regs *regs)
return 1; return 1;
} }
static DEFINE_PER_CPU(struct thread_info, kgdb_thread_info);
static int kgdb_singlestep(struct pt_regs *regs) static int kgdb_singlestep(struct pt_regs *regs)
{ {
struct thread_info *thread_info, *exception_thread_info;
struct thread_info *backup_current_thread_info =
this_cpu_ptr(&kgdb_thread_info);
if (user_mode(regs)) if (user_mode(regs))
return 0; return 0;
/*
* On Book E and perhaps other processors, singlestep is handled on
* the critical exception stack. This causes current_thread_info()
* to fail, since it it locates the thread_info by masking off
* the low bits of the current stack pointer. We work around
* this issue by copying the thread_info from the kernel stack
* before calling kgdb_handle_exception, and copying it back
* afterwards. On most processors the copy is avoided since
* exception_thread_info == thread_info.
*/
thread_info = (struct thread_info *)(regs->gpr[1] & ~(THREAD_SIZE-1));
exception_thread_info = current_thread_info();
if (thread_info != exception_thread_info) {
/* Save the original current_thread_info. */
memcpy(backup_current_thread_info, exception_thread_info, sizeof *thread_info);
memcpy(exception_thread_info, thread_info, sizeof *thread_info);
}
kgdb_handle_exception(0, SIGTRAP, 0, regs); kgdb_handle_exception(0, SIGTRAP, 0, regs);
if (thread_info != exception_thread_info)
/* Restore current_thread_info lastly. */
memcpy(exception_thread_info, backup_current_thread_info, sizeof *thread_info);
return 1; return 1;
} }
......
...@@ -317,10 +317,8 @@ void default_machine_kexec(struct kimage *image) ...@@ -317,10 +317,8 @@ void default_machine_kexec(struct kimage *image)
* We setup preempt_count to avoid using VMX in memcpy. * We setup preempt_count to avoid using VMX in memcpy.
* XXX: the task struct will likely be invalid once we do the copy! * XXX: the task struct will likely be invalid once we do the copy!
*/ */
kexec_stack.thread_info.task = current_thread_info()->task; current_thread_info()->flags = 0;
kexec_stack.thread_info.flags = 0; current_thread_info()->preempt_count = HARDIRQ_OFFSET;
kexec_stack.thread_info.preempt_count = HARDIRQ_OFFSET;
kexec_stack.thread_info.cpu = current_thread_info()->cpu;
/* We need a static PACA, too; copy this CPU's PACA over and switch to /* We need a static PACA, too; copy this CPU's PACA over and switch to
* it. Also poison per_cpu_offset and NULL lppaca to catch anyone using * it. Also poison per_cpu_offset and NULL lppaca to catch anyone using
......
...@@ -1634,7 +1634,7 @@ int copy_thread(unsigned long clone_flags, unsigned long usp, ...@@ -1634,7 +1634,7 @@ int copy_thread(unsigned long clone_flags, unsigned long usp,
unsigned long sp = (unsigned long)task_stack_page(p) + THREAD_SIZE; unsigned long sp = (unsigned long)task_stack_page(p) + THREAD_SIZE;
struct thread_info *ti = task_thread_info(p); struct thread_info *ti = task_thread_info(p);
klp_init_thread_info(ti); klp_init_thread_info(p);
/* Copy registers */ /* Copy registers */
sp -= sizeof(struct pt_regs); sp -= sizeof(struct pt_regs);
......
...@@ -937,7 +937,7 @@ void __init setup_arch(char **cmdline_p) ...@@ -937,7 +937,7 @@ void __init setup_arch(char **cmdline_p)
/* Reserve large chunks of memory for use by CMA for KVM. */ /* Reserve large chunks of memory for use by CMA for KVM. */
kvm_cma_reserve(); kvm_cma_reserve();
klp_init_thread_info(&init_thread_info); klp_init_thread_info(&init_task);
init_mm.start_code = (unsigned long)_stext; init_mm.start_code = (unsigned long)_stext;
init_mm.end_code = (unsigned long) _etext; init_mm.end_code = (unsigned long) _etext;
......
...@@ -689,24 +689,6 @@ void __init exc_lvl_early_init(void) ...@@ -689,24 +689,6 @@ void __init exc_lvl_early_init(void)
} }
#endif #endif
/*
* Emergency stacks are used for a range of things, from asynchronous
* NMIs (system reset, machine check) to synchronous, process context.
* We set preempt_count to zero, even though that isn't necessarily correct. To
* get the right value we'd need to copy it from the previous thread_info, but
* doing that might fault causing more problems.
* TODO: what to do with accounting?
*/
static void emerg_stack_init_thread_info(struct thread_info *ti, int cpu)
{
ti->task = NULL;
ti->cpu = cpu;
ti->preempt_count = 0;
ti->local_flags = 0;
ti->flags = 0;
klp_init_thread_info(ti);
}
/* /*
* Stack space used when we detect a bad kernel stack pointer, and * Stack space used when we detect a bad kernel stack pointer, and
* early in SMP boots before relocation is enabled. Exclusive emergency * early in SMP boots before relocation is enabled. Exclusive emergency
...@@ -737,18 +719,15 @@ void __init emergency_stack_init(void) ...@@ -737,18 +719,15 @@ void __init emergency_stack_init(void)
struct thread_info *ti; struct thread_info *ti;
ti = alloc_stack(limit, i); ti = alloc_stack(limit, i);
emerg_stack_init_thread_info(ti, i);
paca_ptrs[i]->emergency_sp = (void *)ti + THREAD_SIZE; paca_ptrs[i]->emergency_sp = (void *)ti + THREAD_SIZE;
#ifdef CONFIG_PPC_BOOK3S_64 #ifdef CONFIG_PPC_BOOK3S_64
/* emergency stack for NMI exception handling. */ /* emergency stack for NMI exception handling. */
ti = alloc_stack(limit, i); ti = alloc_stack(limit, i);
emerg_stack_init_thread_info(ti, i);
paca_ptrs[i]->nmi_emergency_sp = (void *)ti + THREAD_SIZE; paca_ptrs[i]->nmi_emergency_sp = (void *)ti + THREAD_SIZE;
/* emergency stack for machine check exception handling. */ /* emergency stack for machine check exception handling. */
ti = alloc_stack(limit, i); ti = alloc_stack(limit, i);
emerg_stack_init_thread_info(ti, i);
paca_ptrs[i]->mc_emergency_sp = (void *)ti + THREAD_SIZE; paca_ptrs[i]->mc_emergency_sp = (void *)ti + THREAD_SIZE;
#endif #endif
} }
......
...@@ -988,7 +988,7 @@ static void cpu_idle_thread_init(unsigned int cpu, struct task_struct *idle) ...@@ -988,7 +988,7 @@ static void cpu_idle_thread_init(unsigned int cpu, struct task_struct *idle)
paca_ptrs[cpu]->kstack = (unsigned long)task_stack_page(idle) + paca_ptrs[cpu]->kstack = (unsigned long)task_stack_page(idle) +
THREAD_SIZE - STACK_FRAME_OVERHEAD; THREAD_SIZE - STACK_FRAME_OVERHEAD;
#endif #endif
ti->cpu = cpu; idle->cpu = cpu;
secondary_ti = current_set[cpu] = ti; secondary_ti = current_set[cpu] = ti;
} }
......
...@@ -106,9 +106,8 @@ DECLARE_LOAD_FUNC(sk_load_byte_msh); ...@@ -106,9 +106,8 @@ DECLARE_LOAD_FUNC(sk_load_byte_msh);
} while (0) } while (0)
#else #else
#define PPC_BPF_LOAD_CPU(r) \ #define PPC_BPF_LOAD_CPU(r) \
do { BUILD_BUG_ON(FIELD_SIZEOF(struct thread_info, cpu) != 4); \ do { BUILD_BUG_ON(FIELD_SIZEOF(struct task_struct, cpu) != 4); \
PPC_LHZ_OFFS(r, (1 & ~(THREAD_SIZE - 1)), \ PPC_LHZ_OFFS(r, 2, offsetof(struct task_struct, cpu)); \
offsetof(struct thread_info, cpu)); \
} while(0) } while(0)
#endif #endif
#else #else
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment