Commit 77b1a7f7 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'mm-nonmm-stable-2023-06-24-19-23' of...

Merge tag 'mm-nonmm-stable-2023-06-24-19-23' of git://git.kernel.org/pub/scm/linux/kernel/git/akpm/mm

Pull non-mm updates from Andrew Morton:

 - Arnd Bergmann has fixed a bunch of -Wmissing-prototypes in top-level
   directories

 - Douglas Anderson has added a new "buddy" mode to the hardlockup
   detector. It permits the detector to work on architectures which
   cannot provide the required interrupts, by having CPUs periodically
   perform checks on other CPUs

 - Zhen Lei has enhanced kexec's ability to support two crash regions

 - Petr Mladek has done a lot of cleanup on the hard lockup detector's
   Kconfig entries

 - And the usual bunch of singleton patches in various places

* tag 'mm-nonmm-stable-2023-06-24-19-23' of git://git.kernel.org/pub/scm/linux/kernel/git/akpm/mm: (72 commits)
  kernel/time/posix-stubs.c: remove duplicated include
  ocfs2: remove redundant assignment to variable bit_off
  watchdog/hardlockup: fix typo in config HARDLOCKUP_DETECTOR_PREFER_BUDDY
  powerpc: move arch_trigger_cpumask_backtrace from nmi.h to irq.h
  devres: show which resource was invalid in __devm_ioremap_resource()
  watchdog/hardlockup: define HARDLOCKUP_DETECTOR_ARCH
  watchdog/sparc64: define HARDLOCKUP_DETECTOR_SPARC64
  watchdog/hardlockup: make HAVE_NMI_WATCHDOG sparc64-specific
  watchdog/hardlockup: declare arch_touch_nmi_watchdog() only in linux/nmi.h
  watchdog/hardlockup: make the config checks more straightforward
  watchdog/hardlockup: sort hardlockup detector related config values a logical way
  watchdog/hardlockup: move SMP barriers from common code to buddy code
  watchdog/buddy: simplify the dependency for HARDLOCKUP_DETECTOR_PREFER_BUDDY
  watchdog/buddy: don't copy the cpumask in watchdog_next_cpu()
  watchdog/buddy: cleanup how watchdog_buddy_check_hardlockup() is called
  watchdog/hardlockup: remove softlockup comment in touch_nmi_watchdog()
  watchdog/hardlockup: in watchdog_hardlockup_check() use cpumask_copy()
  watchdog/hardlockup: don't use raw_cpu_ptr() in watchdog_hardlockup_kick()
  watchdog/hardlockup: HAVE_NMI_WATCHDOG must implement watchdog_hardlockup_probe()
  watchdog/hardlockup: keep kernel.nmi_watchdog sysctl as 0444 if probe fails
  ...
parents 6e17c6de 4afc9a40
...@@ -426,20 +426,14 @@ config HAVE_HARDLOCKUP_DETECTOR_PERF ...@@ -426,20 +426,14 @@ config HAVE_HARDLOCKUP_DETECTOR_PERF
The arch chooses to use the generic perf-NMI-based hardlockup The arch chooses to use the generic perf-NMI-based hardlockup
detector. Must define HAVE_PERF_EVENTS_NMI. detector. Must define HAVE_PERF_EVENTS_NMI.
config HAVE_NMI_WATCHDOG
depends on HAVE_NMI
bool
help
The arch provides a low level NMI watchdog. It provides
asm/nmi.h, and defines its own arch_touch_nmi_watchdog().
config HAVE_HARDLOCKUP_DETECTOR_ARCH config HAVE_HARDLOCKUP_DETECTOR_ARCH
bool bool
select HAVE_NMI_WATCHDOG
help help
The arch chooses to provide its own hardlockup detector, which is The arch provides its own hardlockup detector implementation instead
a superset of the HAVE_NMI_WATCHDOG. It also conforms to config of the generic ones.
interfaces and parameters provided by hardlockup detector subsystem.
It uses the same command line parameters, and sysctl interface,
as the generic hardlockup detectors.
config HAVE_PERF_REGS config HAVE_PERF_REGS
bool bool
......
...@@ -27,7 +27,6 @@ struct irqaction; ...@@ -27,7 +27,6 @@ struct irqaction;
struct pt_regs; struct pt_regs;
void handle_IRQ(unsigned int, struct pt_regs *); void handle_IRQ(unsigned int, struct pt_regs *);
void init_IRQ(void);
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
#include <linux/cpumask.h> #include <linux/cpumask.h>
......
...@@ -204,6 +204,8 @@ config ARM64 ...@@ -204,6 +204,8 @@ config ARM64
select HAVE_FUNCTION_ERROR_INJECTION select HAVE_FUNCTION_ERROR_INJECTION
select HAVE_FUNCTION_GRAPH_TRACER select HAVE_FUNCTION_GRAPH_TRACER
select HAVE_GCC_PLUGINS select HAVE_GCC_PLUGINS
select HAVE_HARDLOCKUP_DETECTOR_PERF if PERF_EVENTS && \
HW_PERF_EVENTS && HAVE_PERF_EVENTS_NMI
select HAVE_HW_BREAKPOINT if PERF_EVENTS select HAVE_HW_BREAKPOINT if PERF_EVENTS
select HAVE_IOREMAP_PROT select HAVE_IOREMAP_PROT
select HAVE_IRQ_TIME_ACCOUNTING select HAVE_IRQ_TIME_ACCOUNTING
...@@ -211,6 +213,7 @@ config ARM64 ...@@ -211,6 +213,7 @@ config ARM64
select HAVE_MOD_ARCH_SPECIFIC select HAVE_MOD_ARCH_SPECIFIC
select HAVE_NMI select HAVE_NMI
select HAVE_PERF_EVENTS select HAVE_PERF_EVENTS
select HAVE_PERF_EVENTS_NMI if ARM64_PSEUDO_NMI
select HAVE_PERF_REGS select HAVE_PERF_REGS
select HAVE_PERF_USER_STACK_DUMP select HAVE_PERF_USER_STACK_DUMP
select HAVE_PREEMPT_DYNAMIC_KEY select HAVE_PREEMPT_DYNAMIC_KEY
......
...@@ -55,10 +55,6 @@ struct thread_info { ...@@ -55,10 +55,6 @@ struct thread_info {
void arch_setup_new_exec(void); void arch_setup_new_exec(void);
#define arch_setup_new_exec arch_setup_new_exec #define arch_setup_new_exec arch_setup_new_exec
void arch_release_task_struct(struct task_struct *tsk);
int arch_dup_task_struct(struct task_struct *dst,
struct task_struct *src);
#endif #endif
#define TIF_SIGPENDING 0 /* signal pending */ #define TIF_SIGPENDING 0 /* signal pending */
......
...@@ -44,6 +44,7 @@ obj-$(CONFIG_KUSER_HELPERS) += kuser32.o ...@@ -44,6 +44,7 @@ obj-$(CONFIG_KUSER_HELPERS) += kuser32.o
obj-$(CONFIG_FUNCTION_TRACER) += ftrace.o entry-ftrace.o obj-$(CONFIG_FUNCTION_TRACER) += ftrace.o entry-ftrace.o
obj-$(CONFIG_MODULES) += module.o module-plts.o obj-$(CONFIG_MODULES) += module.o module-plts.o
obj-$(CONFIG_PERF_EVENTS) += perf_regs.o perf_callchain.o obj-$(CONFIG_PERF_EVENTS) += perf_regs.o perf_callchain.o
obj-$(CONFIG_HARDLOCKUP_DETECTOR_PERF) += watchdog_hld.o
obj-$(CONFIG_HAVE_HW_BREAKPOINT) += hw_breakpoint.o obj-$(CONFIG_HAVE_HW_BREAKPOINT) += hw_breakpoint.o
obj-$(CONFIG_CPU_PM) += sleep.o suspend.o obj-$(CONFIG_CPU_PM) += sleep.o suspend.o
obj-$(CONFIG_CPU_IDLE) += cpuidle.o obj-$(CONFIG_CPU_IDLE) += cpuidle.o
......
// SPDX-License-Identifier: GPL-2.0
#include <linux/nmi.h>
#include <linux/cpufreq.h>
#include <linux/perf/arm_pmu.h>
/*
* Safe maximum CPU frequency in case a particular platform doesn't implement
* cpufreq driver. Although, architecture doesn't put any restrictions on
* maximum frequency but 5 GHz seems to be safe maximum given the available
* Arm CPUs in the market which are clocked much less than 5 GHz. On the other
* hand, we can't make it much higher as it would lead to a large hard-lockup
* detection timeout on parts which are running slower (eg. 1GHz on
* Developerbox) and doesn't possess a cpufreq driver.
*/
#define SAFE_MAX_CPU_FREQ 5000000000UL // 5 GHz
u64 hw_nmi_get_sample_period(int watchdog_thresh)
{
unsigned int cpu = smp_processor_id();
unsigned long max_cpu_freq;
max_cpu_freq = cpufreq_get_hw_max_freq(cpu) * 1000UL;
if (!max_cpu_freq)
max_cpu_freq = SAFE_MAX_CPU_FREQ;
return (u64)max_cpu_freq * watchdog_thresh;
}
bool __init arch_perf_nmi_is_available(void)
{
/*
* hardlockup_detector_perf_init() will success even if Pseudo-NMI turns off,
* however, the pmu interrupts will act like a normal interrupt instead of
* NMI and the hardlockup detector would be broken.
*/
return arm_pmu_irq_is_nmi();
}
...@@ -16,8 +16,6 @@ extern char *klimit; ...@@ -16,8 +16,6 @@ extern char *klimit;
extern void mmu_reset(void); extern void mmu_reset(void);
void time_init(void);
void init_IRQ(void);
void machine_early_init(const char *cmdline, unsigned int ram, void machine_early_init(const char *cmdline, unsigned int ram,
unsigned int fdt, unsigned int msr, unsigned int tlb0, unsigned int fdt, unsigned int msr, unsigned int tlb0,
unsigned int tlb1); unsigned int tlb1);
......
...@@ -17,9 +17,6 @@ ...@@ -17,9 +17,6 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/string.h> #include <linux/string.h>
typedef long intptr_t;
/* /*
* Constants * Constants
*/ */
......
...@@ -19,7 +19,6 @@ ...@@ -19,7 +19,6 @@
#define IRQ_STACK_SIZE THREAD_SIZE #define IRQ_STACK_SIZE THREAD_SIZE
#define IRQ_STACK_START (IRQ_STACK_SIZE - 16) #define IRQ_STACK_START (IRQ_STACK_SIZE - 16)
extern void __init init_IRQ(void);
extern void *irq_stack[NR_CPUS]; extern void *irq_stack[NR_CPUS];
/* /*
......
...@@ -271,7 +271,6 @@ void arch_send_call_function_single_ipi(int cpu) ...@@ -271,7 +271,6 @@ void arch_send_call_function_single_ipi(int cpu)
static void static void
smp_cpu_init(int cpunum) smp_cpu_init(int cpunum)
{ {
extern void init_IRQ(void); /* arch/parisc/kernel/irq.c */
extern void start_cpu_itimer(void); /* arch/parisc/kernel/time.c */ extern void start_cpu_itimer(void); /* arch/parisc/kernel/time.c */
/* Set modes and Enable floating point coprocessor */ /* Set modes and Enable floating point coprocessor */
......
...@@ -90,8 +90,7 @@ config NMI_IPI ...@@ -90,8 +90,7 @@ config NMI_IPI
config PPC_WATCHDOG config PPC_WATCHDOG
bool bool
depends on HARDLOCKUP_DETECTOR depends on HARDLOCKUP_DETECTOR_ARCH
depends on HAVE_HARDLOCKUP_DETECTOR_ARCH
default y default y
help help
This is a placeholder when the powerpc hardlockup detector This is a placeholder when the powerpc hardlockup detector
...@@ -240,7 +239,7 @@ config PPC ...@@ -240,7 +239,7 @@ config PPC
select HAVE_GCC_PLUGINS if GCC_VERSION >= 50200 # plugin support on gcc <= 5.1 is buggy on PPC select HAVE_GCC_PLUGINS if GCC_VERSION >= 50200 # plugin support on gcc <= 5.1 is buggy on PPC
select HAVE_GENERIC_VDSO select HAVE_GENERIC_VDSO
select HAVE_HARDLOCKUP_DETECTOR_ARCH if PPC_BOOK3S_64 && SMP select HAVE_HARDLOCKUP_DETECTOR_ARCH if PPC_BOOK3S_64 && SMP
select HAVE_HARDLOCKUP_DETECTOR_PERF if PERF_EVENTS && HAVE_PERF_EVENTS_NMI && !HAVE_HARDLOCKUP_DETECTOR_ARCH select HAVE_HARDLOCKUP_DETECTOR_PERF if PERF_EVENTS && HAVE_PERF_EVENTS_NMI
select HAVE_HW_BREAKPOINT if PERF_EVENTS && (PPC_BOOK3S || PPC_8xx) select HAVE_HW_BREAKPOINT if PERF_EVENTS && (PPC_BOOK3S || PPC_8xx)
select HAVE_IOREMAP_PROT select HAVE_IOREMAP_PROT
select HAVE_IRQ_TIME_ACCOUNTING select HAVE_IRQ_TIME_ACCOUNTING
......
...@@ -50,9 +50,14 @@ extern void *hardirq_ctx[NR_CPUS]; ...@@ -50,9 +50,14 @@ extern void *hardirq_ctx[NR_CPUS];
extern void *softirq_ctx[NR_CPUS]; extern void *softirq_ctx[NR_CPUS];
void __do_IRQ(struct pt_regs *regs); void __do_IRQ(struct pt_regs *regs);
extern void __init init_IRQ(void);
int irq_choose_cpu(const struct cpumask *mask); int irq_choose_cpu(const struct cpumask *mask);
#if defined(CONFIG_PPC_BOOK3S_64) && defined(CONFIG_NMI_IPI)
extern void arch_trigger_cpumask_backtrace(const cpumask_t *mask,
bool exclude_self);
#define arch_trigger_cpumask_backtrace arch_trigger_cpumask_backtrace
#endif
#endif /* _ASM_IRQ_H */ #endif /* _ASM_IRQ_H */
#endif /* __KERNEL__ */ #endif /* __KERNEL__ */
...@@ -3,18 +3,10 @@ ...@@ -3,18 +3,10 @@
#define _ASM_NMI_H #define _ASM_NMI_H
#ifdef CONFIG_PPC_WATCHDOG #ifdef CONFIG_PPC_WATCHDOG
extern void arch_touch_nmi_watchdog(void);
long soft_nmi_interrupt(struct pt_regs *regs); long soft_nmi_interrupt(struct pt_regs *regs);
void watchdog_nmi_set_timeout_pct(u64 pct); void watchdog_hardlockup_set_timeout_pct(u64 pct);
#else #else
static inline void arch_touch_nmi_watchdog(void) {} static inline void watchdog_hardlockup_set_timeout_pct(u64 pct) {}
static inline void watchdog_nmi_set_timeout_pct(u64 pct) {}
#endif
#ifdef CONFIG_NMI_IPI
extern void arch_trigger_cpumask_backtrace(const cpumask_t *mask,
bool exclude_self);
#define arch_trigger_cpumask_backtrace arch_trigger_cpumask_backtrace
#endif #endif
extern void hv_nmi_check_nonrecoverable(struct pt_regs *regs); extern void hv_nmi_check_nonrecoverable(struct pt_regs *regs);
......
...@@ -438,7 +438,7 @@ static enum hrtimer_restart watchdog_timer_fn(struct hrtimer *hrtimer) ...@@ -438,7 +438,7 @@ static enum hrtimer_restart watchdog_timer_fn(struct hrtimer *hrtimer)
{ {
int cpu = smp_processor_id(); int cpu = smp_processor_id();
if (!(watchdog_enabled & NMI_WATCHDOG_ENABLED)) if (!(watchdog_enabled & WATCHDOG_HARDLOCKUP_ENABLED))
return HRTIMER_NORESTART; return HRTIMER_NORESTART;
if (!cpumask_test_cpu(cpu, &watchdog_cpumask)) if (!cpumask_test_cpu(cpu, &watchdog_cpumask))
...@@ -479,7 +479,7 @@ static void start_watchdog(void *arg) ...@@ -479,7 +479,7 @@ static void start_watchdog(void *arg)
return; return;
} }
if (!(watchdog_enabled & NMI_WATCHDOG_ENABLED)) if (!(watchdog_enabled & WATCHDOG_HARDLOCKUP_ENABLED))
return; return;
if (!cpumask_test_cpu(cpu, &watchdog_cpumask)) if (!cpumask_test_cpu(cpu, &watchdog_cpumask))
...@@ -546,7 +546,7 @@ static void watchdog_calc_timeouts(void) ...@@ -546,7 +546,7 @@ static void watchdog_calc_timeouts(void)
wd_timer_period_ms = watchdog_thresh * 1000 * 2 / 5; wd_timer_period_ms = watchdog_thresh * 1000 * 2 / 5;
} }
void watchdog_nmi_stop(void) void watchdog_hardlockup_stop(void)
{ {
int cpu; int cpu;
...@@ -554,7 +554,7 @@ void watchdog_nmi_stop(void) ...@@ -554,7 +554,7 @@ void watchdog_nmi_stop(void)
stop_watchdog_on_cpu(cpu); stop_watchdog_on_cpu(cpu);
} }
void watchdog_nmi_start(void) void watchdog_hardlockup_start(void)
{ {
int cpu; int cpu;
...@@ -566,7 +566,7 @@ void watchdog_nmi_start(void) ...@@ -566,7 +566,7 @@ void watchdog_nmi_start(void)
/* /*
* Invoked from core watchdog init. * Invoked from core watchdog init.
*/ */
int __init watchdog_nmi_probe(void) int __init watchdog_hardlockup_probe(void)
{ {
int err; int err;
...@@ -582,7 +582,7 @@ int __init watchdog_nmi_probe(void) ...@@ -582,7 +582,7 @@ int __init watchdog_nmi_probe(void)
} }
#ifdef CONFIG_PPC_PSERIES #ifdef CONFIG_PPC_PSERIES
void watchdog_nmi_set_timeout_pct(u64 pct) void watchdog_hardlockup_set_timeout_pct(u64 pct)
{ {
pr_info("Set the NMI watchdog timeout factor to %llu%%\n", pct); pr_info("Set the NMI watchdog timeout factor to %llu%%\n", pct);
WRITE_ONCE(wd_timeout_pct, pct); WRITE_ONCE(wd_timeout_pct, pct);
......
...@@ -750,7 +750,7 @@ static int pseries_migrate_partition(u64 handle) ...@@ -750,7 +750,7 @@ static int pseries_migrate_partition(u64 handle)
goto out; goto out;
if (factor) if (factor)
watchdog_nmi_set_timeout_pct(factor); watchdog_hardlockup_set_timeout_pct(factor);
ret = pseries_suspend(handle); ret = pseries_suspend(handle);
if (ret == 0) { if (ret == 0) {
...@@ -766,7 +766,7 @@ static int pseries_migrate_partition(u64 handle) ...@@ -766,7 +766,7 @@ static int pseries_migrate_partition(u64 handle)
pseries_cancel_migration(handle, ret); pseries_cancel_migration(handle, ret);
if (factor) if (factor)
watchdog_nmi_set_timeout_pct(0); watchdog_hardlockup_set_timeout_pct(0);
out: out:
vas_migration_handler(VAS_RESUME); vas_migration_handler(VAS_RESUME);
......
...@@ -16,6 +16,4 @@ void riscv_set_intc_hwnode_fn(struct fwnode_handle *(*fn)(void)); ...@@ -16,6 +16,4 @@ void riscv_set_intc_hwnode_fn(struct fwnode_handle *(*fn)(void));
struct fwnode_handle *riscv_get_intc_hwnode(void); struct fwnode_handle *riscv_get_intc_hwnode(void);
extern void __init init_IRQ(void);
#endif /* _ASM_RISCV_IRQ_H */ #endif /* _ASM_RISCV_IRQ_H */
...@@ -88,6 +88,4 @@ static inline int read_current_timer(unsigned long *timer_val) ...@@ -88,6 +88,4 @@ static inline int read_current_timer(unsigned long *timer_val)
return 0; return 0;
} }
extern void time_init(void);
#endif /* _ASM_RISCV_TIMEX_H */ #endif /* _ASM_RISCV_TIMEX_H */
...@@ -52,9 +52,6 @@ struct thread_info { ...@@ -52,9 +52,6 @@ struct thread_info {
struct task_struct; struct task_struct;
void arch_release_task_struct(struct task_struct *tsk);
int arch_dup_task_struct(struct task_struct *dst, struct task_struct *src);
void arch_setup_new_exec(void); void arch_setup_new_exec(void);
#define arch_setup_new_exec arch_setup_new_exec #define arch_setup_new_exec arch_setup_new_exec
......
...@@ -34,14 +34,12 @@ void kernel_stack_overflow(struct pt_regs * regs); ...@@ -34,14 +34,12 @@ void kernel_stack_overflow(struct pt_regs * regs);
void handle_signal32(struct ksignal *ksig, sigset_t *oldset, void handle_signal32(struct ksignal *ksig, sigset_t *oldset,
struct pt_regs *regs); struct pt_regs *regs);
void __init init_IRQ(void);
void do_io_irq(struct pt_regs *regs); void do_io_irq(struct pt_regs *regs);
void do_ext_irq(struct pt_regs *regs); void do_ext_irq(struct pt_regs *regs);
void do_restart(void *arg); void do_restart(void *arg);
void __init startup_init(void); void __init startup_init(void);
void die(struct pt_regs *regs, const char *str); void die(struct pt_regs *regs, const char *str);
int setup_profiling_timer(unsigned int multiplier); int setup_profiling_timer(unsigned int multiplier);
void __init time_init(void);
unsigned long prepare_ftrace_return(unsigned long parent, unsigned long sp, unsigned long ip); unsigned long prepare_ftrace_return(unsigned long parent, unsigned long sp, unsigned long ip);
struct s390_mmap_arg_struct; struct s390_mmap_arg_struct;
......
...@@ -22,7 +22,6 @@ extern unsigned short *irq_mask_register; ...@@ -22,7 +22,6 @@ extern unsigned short *irq_mask_register;
/* /*
* PINT IRQs * PINT IRQs
*/ */
void init_IRQ_pint(void);
void make_imask_irq(unsigned int irq); void make_imask_irq(unsigned int irq);
static inline int generic_irq_demux(int irq) static inline int generic_irq_demux(int irq)
......
...@@ -2,8 +2,6 @@ ...@@ -2,8 +2,6 @@
#ifndef _ASM_RTC_H #ifndef _ASM_RTC_H
#define _ASM_RTC_H #define _ASM_RTC_H
void time_init(void);
#define RTC_CAP_4_DIGIT_YEAR (1 << 0) #define RTC_CAP_4_DIGIT_YEAR (1 << 0)
struct sh_rtc_platform_info { struct sh_rtc_platform_info {
......
...@@ -84,9 +84,6 @@ static inline struct thread_info *current_thread_info(void) ...@@ -84,9 +84,6 @@ static inline struct thread_info *current_thread_info(void)
#define THREAD_SIZE_ORDER (THREAD_SHIFT - PAGE_SHIFT) #define THREAD_SIZE_ORDER (THREAD_SHIFT - PAGE_SHIFT)
extern void arch_task_cache_init(void);
extern int arch_dup_task_struct(struct task_struct *dst, struct task_struct *src);
extern void arch_release_task_struct(struct task_struct *tsk);
extern void init_thread_xstate(void); extern void init_thread_xstate(void);
#endif /* __ASSEMBLY__ */ #endif /* __ASSEMBLY__ */
......
...@@ -33,7 +33,7 @@ config SPARC ...@@ -33,7 +33,7 @@ config SPARC
select ARCH_WANT_IPC_PARSE_VERSION select ARCH_WANT_IPC_PARSE_VERSION
select GENERIC_PCI_IOMAP select GENERIC_PCI_IOMAP
select HAS_IOPORT select HAS_IOPORT
select HAVE_NMI_WATCHDOG if SPARC64 select HAVE_HARDLOCKUP_DETECTOR_SPARC64 if SPARC64
select HAVE_CBPF_JIT if SPARC32 select HAVE_CBPF_JIT if SPARC32
select HAVE_EBPF_JIT if SPARC64 select HAVE_EBPF_JIT if SPARC64
select HAVE_DEBUG_BUGVERBOSE select HAVE_DEBUG_BUGVERBOSE
......
...@@ -14,3 +14,17 @@ config FRAME_POINTER ...@@ -14,3 +14,17 @@ config FRAME_POINTER
bool bool
depends on MCOUNT depends on MCOUNT
default y default y
config HAVE_HARDLOCKUP_DETECTOR_SPARC64
bool
depends on HAVE_NMI
select HARDLOCKUP_DETECTOR_SPARC64
help
Sparc64 hardlockup detector is the last one developed before adding
the common infrastructure for handling hardlockup detectors. It is
always built. It does _not_ use the common command line parameters
and sysctl interface, except for /proc/sys/kernel/nmi_watchdog.
config HARDLOCKUP_DETECTOR_SPARC64
bool
depends on HAVE_HARDLOCKUP_DETECTOR_SPARC64
...@@ -17,7 +17,6 @@ ...@@ -17,7 +17,6 @@
#define irq_canonicalize(irq) (irq) #define irq_canonicalize(irq) (irq)
void __init init_IRQ(void);
void __init sun4d_init_sbi_irq(void); void __init sun4d_init_sbi_irq(void);
#define NO_IRQ 0xffffffff #define NO_IRQ 0xffffffff
......
...@@ -61,7 +61,6 @@ void sun4u_destroy_msi(unsigned int irq); ...@@ -61,7 +61,6 @@ void sun4u_destroy_msi(unsigned int irq);
unsigned int irq_alloc(unsigned int dev_handle, unsigned int dev_ino); unsigned int irq_alloc(unsigned int dev_handle, unsigned int dev_ino);
void irq_free(unsigned int irq); void irq_free(unsigned int irq);
void __init init_IRQ(void);
void fixup_irqs(void); void fixup_irqs(void);
static inline void set_softint(unsigned long bits) static inline void set_softint(unsigned long bits)
......
...@@ -8,7 +8,6 @@ void nmi_adjust_hz(unsigned int new_hz); ...@@ -8,7 +8,6 @@ void nmi_adjust_hz(unsigned int new_hz);
extern atomic_t nmi_active; extern atomic_t nmi_active;
void arch_touch_nmi_watchdog(void);
void start_nmi_watchdog(void *unused); void start_nmi_watchdog(void *unused);
void stop_nmi_watchdog(void *unused); void stop_nmi_watchdog(void *unused);
......
...@@ -34,7 +34,6 @@ extern struct sparc64_tick_ops *tick_ops; ...@@ -34,7 +34,6 @@ extern struct sparc64_tick_ops *tick_ops;
unsigned long sparc64_get_clock_tick(unsigned int cpu); unsigned long sparc64_get_clock_tick(unsigned int cpu);
void setup_sparc64_timer(void); void setup_sparc64_timer(void);
void __init time_init(void);
#define TICK_PRIV_BIT BIT(63) #define TICK_PRIV_BIT BIT(63)
#define TICKCMP_IRQ_BIT BIT(63) #define TICKCMP_IRQ_BIT BIT(63)
......
...@@ -91,7 +91,6 @@ extern int static_irq_count; ...@@ -91,7 +91,6 @@ extern int static_irq_count;
extern spinlock_t irq_action_lock; extern spinlock_t irq_action_lock;
void unexpected_irq(int irq, void *dev_id, struct pt_regs * regs); void unexpected_irq(int irq, void *dev_id, struct pt_regs * regs);
void init_IRQ(void);
/* sun4m_irq.c */ /* sun4m_irq.c */
void sun4m_init_IRQ(void); void sun4m_init_IRQ(void);
......
...@@ -65,6 +65,11 @@ void arch_touch_nmi_watchdog(void) ...@@ -65,6 +65,11 @@ void arch_touch_nmi_watchdog(void)
} }
EXPORT_SYMBOL(arch_touch_nmi_watchdog); EXPORT_SYMBOL(arch_touch_nmi_watchdog);
int __init watchdog_hardlockup_probe(void)
{
return 0;
}
static void die_nmi(const char *str, struct pt_regs *regs, int do_panic) static void die_nmi(const char *str, struct pt_regs *regs, int do_panic)
{ {
int this_cpu = smp_processor_id(); int this_cpu = smp_processor_id();
...@@ -282,11 +287,11 @@ __setup("nmi_watchdog=", setup_nmi_watchdog); ...@@ -282,11 +287,11 @@ __setup("nmi_watchdog=", setup_nmi_watchdog);
* sparc specific NMI watchdog enable function. * sparc specific NMI watchdog enable function.
* Enables watchdog if it is not enabled already. * Enables watchdog if it is not enabled already.
*/ */
int watchdog_nmi_enable(unsigned int cpu) void watchdog_hardlockup_enable(unsigned int cpu)
{ {
if (atomic_read(&nmi_active) == -1) { if (atomic_read(&nmi_active) == -1) {
pr_warn("NMI watchdog cannot be enabled or disabled\n"); pr_warn("NMI watchdog cannot be enabled or disabled\n");
return -1; return;
} }
/* /*
...@@ -295,17 +300,15 @@ int watchdog_nmi_enable(unsigned int cpu) ...@@ -295,17 +300,15 @@ int watchdog_nmi_enable(unsigned int cpu)
* process first. * process first.
*/ */
if (!nmi_init_done) if (!nmi_init_done)
return 0; return;
smp_call_function_single(cpu, start_nmi_watchdog, NULL, 1); smp_call_function_single(cpu, start_nmi_watchdog, NULL, 1);
return 0;
} }
/* /*
* sparc specific NMI watchdog disable function. * sparc specific NMI watchdog disable function.
* Disables watchdog if it is not disabled already. * Disables watchdog if it is not disabled already.
*/ */
void watchdog_nmi_disable(unsigned int cpu) void watchdog_hardlockup_disable(unsigned int cpu)
{ {
if (atomic_read(&nmi_active) == -1) if (atomic_read(&nmi_active) == -1)
pr_warn_once("NMI watchdog cannot be enabled or disabled\n"); pr_warn_once("NMI watchdog cannot be enabled or disabled\n");
......
...@@ -40,8 +40,6 @@ extern void __handle_irq(struct irq_desc *desc, struct pt_regs *regs); ...@@ -40,8 +40,6 @@ extern void __handle_irq(struct irq_desc *desc, struct pt_regs *regs);
extern void init_ISA_irqs(void); extern void init_ISA_irqs(void);
extern void __init init_IRQ(void);
#ifdef CONFIG_X86_LOCAL_APIC #ifdef CONFIG_X86_LOCAL_APIC
void arch_trigger_cpumask_backtrace(const struct cpumask *mask, void arch_trigger_cpumask_backtrace(const struct cpumask *mask,
bool exclude_self); bool exclude_self);
......
...@@ -232,9 +232,6 @@ static inline int arch_within_stack_frames(const void * const stack, ...@@ -232,9 +232,6 @@ static inline int arch_within_stack_frames(const void * const stack,
current_thread_info()->status & TS_COMPAT) current_thread_info()->status & TS_COMPAT)
#endif #endif
extern void arch_task_cache_init(void);
extern int arch_dup_task_struct(struct task_struct *dst, struct task_struct *src);
extern void arch_release_task_struct(struct task_struct *tsk);
extern void arch_setup_new_exec(void); extern void arch_setup_new_exec(void);
#define arch_setup_new_exec arch_setup_new_exec #define arch_setup_new_exec arch_setup_new_exec
#endif /* !__ASSEMBLY__ */ #endif /* !__ASSEMBLY__ */
......
...@@ -6,7 +6,6 @@ ...@@ -6,7 +6,6 @@
#include <asm/mc146818rtc.h> #include <asm/mc146818rtc.h>
extern void hpet_time_init(void); extern void hpet_time_init(void);
extern void time_init(void);
extern bool pit_timer_init(void); extern bool pit_timer_init(void);
extern bool tsc_clocksource_watchdog_disabled(void); extern bool tsc_clocksource_watchdog_disabled(void);
......
...@@ -32,7 +32,6 @@ extern struct system_counterval_t convert_art_ns_to_tsc(u64 art_ns); ...@@ -32,7 +32,6 @@ extern struct system_counterval_t convert_art_ns_to_tsc(u64 art_ns);
extern void tsc_early_init(void); extern void tsc_early_init(void);
extern void tsc_init(void); extern void tsc_init(void);
extern unsigned long calibrate_delay_is_known(void);
extern void mark_tsc_unstable(char *reason); extern void mark_tsc_unstable(char *reason);
extern int unsynchronized_tsc(void); extern int unsynchronized_tsc(void);
extern int check_tsc_unstable(void); extern int check_tsc_unstable(void);
......
...@@ -689,6 +689,11 @@ static int armpmu_get_cpu_irq(struct arm_pmu *pmu, int cpu) ...@@ -689,6 +689,11 @@ static int armpmu_get_cpu_irq(struct arm_pmu *pmu, int cpu)
return per_cpu(hw_events->irq, cpu); return per_cpu(hw_events->irq, cpu);
} }
bool arm_pmu_irq_is_nmi(void)
{
return has_nmi;
}
/* /*
* PMU hardware loses all context when a CPU goes offline. * PMU hardware loses all context when a CPU goes offline.
* When a CPU is hotplugged back in, since some hardware registers are * When a CPU is hotplugged back in, since some hardware registers are
......
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/sched_clock.h> #include <linux/sched_clock.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/nmi.h>
#include <asm/arm_pmuv3.h> #include <asm/arm_pmuv3.h>
...@@ -1363,10 +1364,17 @@ static struct platform_driver armv8_pmu_driver = { ...@@ -1363,10 +1364,17 @@ static struct platform_driver armv8_pmu_driver = {
static int __init armv8_pmu_driver_init(void) static int __init armv8_pmu_driver_init(void)
{ {
int ret;
if (acpi_disabled) if (acpi_disabled)
return platform_driver_register(&armv8_pmu_driver); ret = platform_driver_register(&armv8_pmu_driver);
else else
return arm_pmu_acpi_probe(armv8_pmuv3_pmu_init); ret = arm_pmu_acpi_probe(armv8_pmuv3_pmu_init);
if (!ret)
lockup_detector_retry_init();
return ret;
} }
device_initcall(armv8_pmu_driver_init) device_initcall(armv8_pmu_driver_init)
......
...@@ -973,7 +973,7 @@ static int ocfs2_sync_local_to_main(struct ocfs2_super *osb, ...@@ -973,7 +973,7 @@ static int ocfs2_sync_local_to_main(struct ocfs2_super *osb,
la_start_blk = ocfs2_clusters_to_blocks(osb->sb, la_start_blk = ocfs2_clusters_to_blocks(osb->sb,
le32_to_cpu(la->la_bm_off)); le32_to_cpu(la->la_bm_off));
bitmap = la->la_bitmap; bitmap = la->la_bitmap;
start = count = bit_off = 0; start = count = 0;
left = le32_to_cpu(alloc->id1.bitmap1.i_total); left = le32_to_cpu(alloc->id1.bitmap1.i_total);
while ((bit_off = ocfs2_find_next_zero_bit(bitmap, left, start)) while ((bit_off = ocfs2_find_next_zero_bit(bitmap, left, start))
......
...@@ -1315,8 +1315,6 @@ DEFINE_OCFS2_FILE_OPS(ocfs2_sync_file); ...@@ -1315,8 +1315,6 @@ DEFINE_OCFS2_FILE_OPS(ocfs2_sync_file);
DEFINE_OCFS2_FILE_OPS(ocfs2_file_write_iter); DEFINE_OCFS2_FILE_OPS(ocfs2_file_write_iter);
DEFINE_OCFS2_FILE_OPS(ocfs2_file_splice_write);
DEFINE_OCFS2_FILE_OPS(ocfs2_file_read_iter); DEFINE_OCFS2_FILE_OPS(ocfs2_file_read_iter);
DEFINE_OCFS2_FILE_OPS(ocfs2_file_splice_read); DEFINE_OCFS2_FILE_OPS(ocfs2_file_splice_read);
......
...@@ -811,7 +811,7 @@ static int ocfs2_local_free_info(struct super_block *sb, int type) ...@@ -811,7 +811,7 @@ static int ocfs2_local_free_info(struct super_block *sb, int type)
struct ocfs2_quota_chunk *chunk; struct ocfs2_quota_chunk *chunk;
struct ocfs2_local_disk_chunk *dchunk; struct ocfs2_local_disk_chunk *dchunk;
int mark_clean = 1, len; int mark_clean = 1, len;
int status; int status = 0;
iput(oinfo->dqi_gqinode); iput(oinfo->dqi_gqinode);
ocfs2_simple_drop_lockres(OCFS2_SB(sb), &oinfo->dqi_gqlock); ocfs2_simple_drop_lockres(OCFS2_SB(sb), &oinfo->dqi_gqlock);
...@@ -853,17 +853,14 @@ static int ocfs2_local_free_info(struct super_block *sb, int type) ...@@ -853,17 +853,14 @@ static int ocfs2_local_free_info(struct super_block *sb, int type)
oinfo->dqi_libh, oinfo->dqi_libh,
olq_update_info, olq_update_info,
info); info);
if (status < 0) { if (status < 0)
mlog_errno(status); mlog_errno(status);
goto out;
}
out: out:
ocfs2_inode_unlock(sb_dqopt(sb)->files[type], 1); ocfs2_inode_unlock(sb_dqopt(sb)->files[type], 1);
brelse(oinfo->dqi_libh); brelse(oinfo->dqi_libh);
brelse(oinfo->dqi_lqi_bh); brelse(oinfo->dqi_lqi_bh);
kfree(oinfo); kfree(oinfo);
return 0; return status;
} }
static void olq_set_dquot(struct buffer_head *bh, void *private) static void olq_set_dquot(struct buffer_head *bh, void *private)
......
...@@ -419,7 +419,7 @@ static ssize_t read_kcore_iter(struct kiocb *iocb, struct iov_iter *iter) ...@@ -419,7 +419,7 @@ static ssize_t read_kcore_iter(struct kiocb *iocb, struct iov_iter *iter)
char *notes; char *notes;
size_t i = 0; size_t i = 0;
strlcpy(prpsinfo.pr_psargs, saved_command_line, strscpy(prpsinfo.pr_psargs, saved_command_line,
sizeof(prpsinfo.pr_psargs)); sizeof(prpsinfo.pr_psargs));
notes = kzalloc(notes_len, GFP_KERNEL); notes = kzalloc(notes_len, GFP_KERNEL);
......
...@@ -17,8 +17,8 @@ ...@@ -17,8 +17,8 @@
#include <linux/fs.h> #include <linux/fs.h>
#include <linux/vfs.h> #include <linux/vfs.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/pagemap.h>
#include <linux/string.h> #include <linux/string.h>
#include <linux/buffer_head.h>
#include <linux/bio.h> #include <linux/bio.h>
#include "squashfs_fs.h" #include "squashfs_fs.h"
...@@ -76,10 +76,101 @@ static int copy_bio_to_actor(struct bio *bio, ...@@ -76,10 +76,101 @@ static int copy_bio_to_actor(struct bio *bio,
return copied_bytes; return copied_bytes;
} }
static int squashfs_bio_read_cached(struct bio *fullbio,
struct address_space *cache_mapping, u64 index, int length,
u64 read_start, u64 read_end, int page_count)
{
struct page *head_to_cache = NULL, *tail_to_cache = NULL;
struct block_device *bdev = fullbio->bi_bdev;
int start_idx = 0, end_idx = 0;
struct bvec_iter_all iter_all;
struct bio *bio = NULL;
struct bio_vec *bv;
int idx = 0;
int err = 0;
bio_for_each_segment_all(bv, fullbio, iter_all) {
struct page *page = bv->bv_page;
if (page->mapping == cache_mapping) {
idx++;
continue;
}
/*
* We only use this when the device block size is the same as
* the page size, so read_start and read_end cover full pages.
*
* Compare these to the original required index and length to
* only cache pages which were requested partially, since these
* are the ones which are likely to be needed when reading
* adjacent blocks.
*/
if (idx == 0 && index != read_start)
head_to_cache = page;
else if (idx == page_count - 1 && index + length != read_end)
tail_to_cache = page;
if (!bio || idx != end_idx) {
struct bio *new = bio_alloc_clone(bdev, fullbio,
GFP_NOIO, &fs_bio_set);
if (bio) {
bio_trim(bio, start_idx * PAGE_SECTORS,
(end_idx - start_idx) * PAGE_SECTORS);
bio_chain(bio, new);
submit_bio(bio);
}
bio = new;
start_idx = idx;
}
idx++;
end_idx = idx;
}
if (bio) {
bio_trim(bio, start_idx * PAGE_SECTORS,
(end_idx - start_idx) * PAGE_SECTORS);
err = submit_bio_wait(bio);
bio_put(bio);
}
if (err)
return err;
if (head_to_cache) {
int ret = add_to_page_cache_lru(head_to_cache, cache_mapping,
read_start >> PAGE_SHIFT,
GFP_NOIO);
if (!ret) {
SetPageUptodate(head_to_cache);
unlock_page(head_to_cache);
}
}
if (tail_to_cache) {
int ret = add_to_page_cache_lru(tail_to_cache, cache_mapping,
(read_end >> PAGE_SHIFT) - 1,
GFP_NOIO);
if (!ret) {
SetPageUptodate(tail_to_cache);
unlock_page(tail_to_cache);
}
}
return 0;
}
static int squashfs_bio_read(struct super_block *sb, u64 index, int length, static int squashfs_bio_read(struct super_block *sb, u64 index, int length,
struct bio **biop, int *block_offset) struct bio **biop, int *block_offset)
{ {
struct squashfs_sb_info *msblk = sb->s_fs_info; struct squashfs_sb_info *msblk = sb->s_fs_info;
struct address_space *cache_mapping = msblk->cache_mapping;
const u64 read_start = round_down(index, msblk->devblksize); const u64 read_start = round_down(index, msblk->devblksize);
const sector_t block = read_start >> msblk->devblksize_log2; const sector_t block = read_start >> msblk->devblksize_log2;
const u64 read_end = round_up(index + length, msblk->devblksize); const u64 read_end = round_up(index + length, msblk->devblksize);
...@@ -99,21 +190,34 @@ static int squashfs_bio_read(struct super_block *sb, u64 index, int length, ...@@ -99,21 +190,34 @@ static int squashfs_bio_read(struct super_block *sb, u64 index, int length,
for (i = 0; i < page_count; ++i) { for (i = 0; i < page_count; ++i) {
unsigned int len = unsigned int len =
min_t(unsigned int, PAGE_SIZE - offset, total_len); min_t(unsigned int, PAGE_SIZE - offset, total_len);
struct page *page = alloc_page(GFP_NOIO); struct page *page = NULL;
if (cache_mapping)
page = find_get_page(cache_mapping,
(read_start >> PAGE_SHIFT) + i);
if (!page)
page = alloc_page(GFP_NOIO);
if (!page) { if (!page) {
error = -ENOMEM; error = -ENOMEM;
goto out_free_bio; goto out_free_bio;
} }
if (!bio_add_page(bio, page, len, offset)) {
error = -EIO; /*
goto out_free_bio; * Use the __ version to avoid merging since we need each page
} * to be separate when we check for and avoid cached pages.
*/
__bio_add_page(bio, page, len, offset);
offset = 0; offset = 0;
total_len -= len; total_len -= len;
} }
error = submit_bio_wait(bio); if (cache_mapping)
error = squashfs_bio_read_cached(bio, cache_mapping, index,
length, read_start, read_end,
page_count);
else
error = submit_bio_wait(bio);
if (error) if (error)
goto out_free_bio; goto out_free_bio;
......
...@@ -11,7 +11,6 @@ ...@@ -11,7 +11,6 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/mutex.h> #include <linux/mutex.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/buffer_head.h>
#include "squashfs_fs.h" #include "squashfs_fs.h"
#include "squashfs_fs_sb.h" #include "squashfs_fs_sb.h"
......
...@@ -7,7 +7,6 @@ ...@@ -7,7 +7,6 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/percpu.h> #include <linux/percpu.h>
#include <linux/buffer_head.h>
#include <linux/local_lock.h> #include <linux/local_lock.h>
#include "squashfs_fs.h" #include "squashfs_fs.h"
......
...@@ -47,6 +47,7 @@ struct squashfs_sb_info { ...@@ -47,6 +47,7 @@ struct squashfs_sb_info {
struct squashfs_cache *block_cache; struct squashfs_cache *block_cache;
struct squashfs_cache *fragment_cache; struct squashfs_cache *fragment_cache;
struct squashfs_cache *read_page; struct squashfs_cache *read_page;
struct address_space *cache_mapping;
int next_meta_index; int next_meta_index;
__le64 *id_table; __le64 *id_table;
__le64 *fragment_index; __le64 *fragment_index;
......
...@@ -329,6 +329,19 @@ static int squashfs_fill_super(struct super_block *sb, struct fs_context *fc) ...@@ -329,6 +329,19 @@ static int squashfs_fill_super(struct super_block *sb, struct fs_context *fc)
goto failed_mount; goto failed_mount;
} }
if (msblk->devblksize == PAGE_SIZE) {
struct inode *cache = new_inode(sb);
if (cache == NULL)
goto failed_mount;
set_nlink(cache, 1);
cache->i_size = OFFSET_MAX;
mapping_set_gfp_mask(cache->i_mapping, GFP_NOFS);
msblk->cache_mapping = cache->i_mapping;
}
msblk->stream = squashfs_decompressor_setup(sb, flags); msblk->stream = squashfs_decompressor_setup(sb, flags);
if (IS_ERR(msblk->stream)) { if (IS_ERR(msblk->stream)) {
err = PTR_ERR(msblk->stream); err = PTR_ERR(msblk->stream);
...@@ -454,6 +467,8 @@ static int squashfs_fill_super(struct super_block *sb, struct fs_context *fc) ...@@ -454,6 +467,8 @@ static int squashfs_fill_super(struct super_block *sb, struct fs_context *fc)
squashfs_cache_delete(msblk->block_cache); squashfs_cache_delete(msblk->block_cache);
squashfs_cache_delete(msblk->fragment_cache); squashfs_cache_delete(msblk->fragment_cache);
squashfs_cache_delete(msblk->read_page); squashfs_cache_delete(msblk->read_page);
if (msblk->cache_mapping)
iput(msblk->cache_mapping->host);
msblk->thread_ops->destroy(msblk); msblk->thread_ops->destroy(msblk);
kfree(msblk->inode_lookup_table); kfree(msblk->inode_lookup_table);
kfree(msblk->fragment_index); kfree(msblk->fragment_index);
...@@ -572,6 +587,8 @@ static void squashfs_put_super(struct super_block *sb) ...@@ -572,6 +587,8 @@ static void squashfs_put_super(struct super_block *sb)
squashfs_cache_delete(sbi->block_cache); squashfs_cache_delete(sbi->block_cache);
squashfs_cache_delete(sbi->fragment_cache); squashfs_cache_delete(sbi->fragment_cache);
squashfs_cache_delete(sbi->read_page); squashfs_cache_delete(sbi->read_page);
if (sbi->cache_mapping)
iput(sbi->cache_mapping->host);
sbi->thread_ops->destroy(sbi); sbi->thread_ops->destroy(sbi);
kfree(sbi->id_table); kfree(sbi->id_table);
kfree(sbi->fragment_index); kfree(sbi->fragment_index);
......
...@@ -87,10 +87,12 @@ struct bug_entry { ...@@ -87,10 +87,12 @@ struct bug_entry {
* *
* Use the versions with printk format strings to provide better diagnostics. * Use the versions with printk format strings to provide better diagnostics.
*/ */
#ifndef __WARN_FLAGS
extern __printf(4, 5) extern __printf(4, 5)
void warn_slowpath_fmt(const char *file, const int line, unsigned taint, void warn_slowpath_fmt(const char *file, const int line, unsigned taint,
const char *fmt, ...); const char *fmt, ...);
extern __printf(1, 2) void __warn_printk(const char *fmt, ...);
#ifndef __WARN_FLAGS
#define __WARN() __WARN_printf(TAINT_WARN, NULL) #define __WARN() __WARN_printf(TAINT_WARN, NULL)
#define __WARN_printf(taint, arg...) do { \ #define __WARN_printf(taint, arg...) do { \
instrumentation_begin(); \ instrumentation_begin(); \
...@@ -98,7 +100,6 @@ void warn_slowpath_fmt(const char *file, const int line, unsigned taint, ...@@ -98,7 +100,6 @@ void warn_slowpath_fmt(const char *file, const int line, unsigned taint,
instrumentation_end(); \ instrumentation_end(); \
} while (0) } while (0)
#else #else
extern __printf(1, 2) void __warn_printk(const char *fmt, ...);
#define __WARN() __WARN_FLAGS(BUGFLAG_TAINT(TAINT_WARN)) #define __WARN() __WARN_FLAGS(BUGFLAG_TAINT(TAINT_WARN))
#define __WARN_printf(taint, arg...) do { \ #define __WARN_printf(taint, arg...) do { \
instrumentation_begin(); \ instrumentation_begin(); \
......
...@@ -712,7 +712,6 @@ int acpi_match_platform_list(const struct acpi_platform_list *plat); ...@@ -712,7 +712,6 @@ int acpi_match_platform_list(const struct acpi_platform_list *plat);
extern void acpi_early_init(void); extern void acpi_early_init(void);
extern void acpi_subsystem_init(void); extern void acpi_subsystem_init(void);
extern void arch_post_acpi_subsys_init(void);
extern int acpi_nvs_register(__u64 start, __u64 size); extern int acpi_nvs_register(__u64 start, __u64 size);
...@@ -1084,6 +1083,8 @@ static inline bool acpi_sleep_state_supported(u8 sleep_state) ...@@ -1084,6 +1083,8 @@ static inline bool acpi_sleep_state_supported(u8 sleep_state)
#endif /* !CONFIG_ACPI */ #endif /* !CONFIG_ACPI */
extern void arch_post_acpi_subsys_init(void);
#ifdef CONFIG_ACPI_HOTPLUG_IOAPIC #ifdef CONFIG_ACPI_HOTPLUG_IOAPIC
int acpi_ioapic_add(acpi_handle root); int acpi_ioapic_add(acpi_handle root);
#else #else
......
...@@ -56,6 +56,7 @@ static inline void ndelay(unsigned long x) ...@@ -56,6 +56,7 @@ static inline void ndelay(unsigned long x)
extern unsigned long lpj_fine; extern unsigned long lpj_fine;
void calibrate_delay(void); void calibrate_delay(void);
unsigned long calibrate_delay_is_known(void);
void __attribute__((weak)) calibration_delay_done(void); void __attribute__((weak)) calibration_delay_done(void);
void msleep(unsigned int msecs); void msleep(unsigned int msecs);
unsigned long msleep_interruptible(unsigned int msecs); unsigned long msleep_interruptible(unsigned int msecs);
......
...@@ -152,6 +152,24 @@ extern unsigned int reset_devices; ...@@ -152,6 +152,24 @@ extern unsigned int reset_devices;
void setup_arch(char **); void setup_arch(char **);
void prepare_namespace(void); void prepare_namespace(void);
void __init init_rootfs(void); void __init init_rootfs(void);
void init_IRQ(void);
void time_init(void);
void mem_encrypt_init(void);
void poking_init(void);
void pgtable_cache_init(void);
extern initcall_entry_t __initcall_start[];
extern initcall_entry_t __initcall0_start[];
extern initcall_entry_t __initcall1_start[];
extern initcall_entry_t __initcall2_start[];
extern initcall_entry_t __initcall3_start[];
extern initcall_entry_t __initcall4_start[];
extern initcall_entry_t __initcall5_start[];
extern initcall_entry_t __initcall6_start[];
extern initcall_entry_t __initcall7_start[];
extern initcall_entry_t __initcall_end[];
extern struct file_system_type rootfs_fs_type; extern struct file_system_type rootfs_fs_type;
#if defined(CONFIG_STRICT_KERNEL_RWX) || defined(CONFIG_STRICT_MODULE_RWX) #if defined(CONFIG_STRICT_KERNEL_RWX) || defined(CONFIG_STRICT_MODULE_RWX)
...@@ -309,6 +327,8 @@ struct obs_kernel_param { ...@@ -309,6 +327,8 @@ struct obs_kernel_param {
int early; int early;
}; };
extern const struct obs_kernel_param __setup_start[], __setup_end[];
/* /*
* Only for really core code. See moduleparam.h for the normal way. * Only for really core code. See moduleparam.h for the normal way.
* *
......
...@@ -72,6 +72,23 @@ static inline void kcov_remote_stop_softirq(void) ...@@ -72,6 +72,23 @@ static inline void kcov_remote_stop_softirq(void)
kcov_remote_stop(); kcov_remote_stop();
} }
#ifdef CONFIG_64BIT
typedef unsigned long kcov_u64;
#else
typedef unsigned long long kcov_u64;
#endif
void __sanitizer_cov_trace_pc(void);
void __sanitizer_cov_trace_cmp1(u8 arg1, u8 arg2);
void __sanitizer_cov_trace_cmp2(u16 arg1, u16 arg2);
void __sanitizer_cov_trace_cmp4(u32 arg1, u32 arg2);
void __sanitizer_cov_trace_cmp8(kcov_u64 arg1, kcov_u64 arg2);
void __sanitizer_cov_trace_const_cmp1(u8 arg1, u8 arg2);
void __sanitizer_cov_trace_const_cmp2(u16 arg1, u16 arg2);
void __sanitizer_cov_trace_const_cmp4(u32 arg1, u32 arg2);
void __sanitizer_cov_trace_const_cmp8(kcov_u64 arg1, kcov_u64 arg2);
void __sanitizer_cov_trace_switch(kcov_u64 val, void *cases);
#else #else
static inline void kcov_task_init(struct task_struct *t) {} static inline void kcov_task_init(struct task_struct *t) {}
......
...@@ -118,17 +118,17 @@ __STRUCT_FRACT(s32) ...@@ -118,17 +118,17 @@ __STRUCT_FRACT(s32)
__STRUCT_FRACT(u32) __STRUCT_FRACT(u32)
#undef __STRUCT_FRACT #undef __STRUCT_FRACT
/* /* Calculate "x * n / d" without unnecessary overflow or loss of precision. */
* Multiplies an integer by a fraction, while avoiding unnecessary #define mult_frac(x, n, d) \
* overflow or loss of precision. ({ \
*/ typeof(x) x_ = (x); \
#define mult_frac(x, numer, denom)( \ typeof(n) n_ = (n); \
{ \ typeof(d) d_ = (d); \
typeof(x) quot = (x) / (denom); \ \
typeof(x) rem = (x) % (denom); \ typeof(x_) q = x_ / d_; \
(quot * (numer)) + ((rem * (numer)) / (denom)); \ typeof(x_) r = x_ % d_; \
} \ q * n_ + r * n_ / d_; \
) })
#define sector_div(a, b) do_div(a, b) #define sector_div(a, b) do_div(a, b)
......
...@@ -3491,13 +3491,12 @@ static inline bool debug_pagealloc_enabled_static(void) ...@@ -3491,13 +3491,12 @@ static inline bool debug_pagealloc_enabled_static(void)
return static_branch_unlikely(&_debug_pagealloc_enabled); return static_branch_unlikely(&_debug_pagealloc_enabled);
} }
#ifdef CONFIG_DEBUG_PAGEALLOC
/* /*
* To support DEBUG_PAGEALLOC architecture must ensure that * To support DEBUG_PAGEALLOC architecture must ensure that
* __kernel_map_pages() never fails * __kernel_map_pages() never fails
*/ */
extern void __kernel_map_pages(struct page *page, int numpages, int enable); extern void __kernel_map_pages(struct page *page, int numpages, int enable);
#ifdef CONFIG_DEBUG_PAGEALLOC
static inline void debug_pagealloc_map_pages(struct page *page, int numpages) static inline void debug_pagealloc_map_pages(struct page *page, int numpages)
{ {
if (debug_pagealloc_enabled_static()) if (debug_pagealloc_enabled_static())
......
...@@ -123,4 +123,6 @@ extern int iterate_mounts(int (*)(struct vfsmount *, void *), void *, ...@@ -123,4 +123,6 @@ extern int iterate_mounts(int (*)(struct vfsmount *, void *), void *,
struct vfsmount *); struct vfsmount *);
extern void kern_unmount_array(struct vfsmount *mnt[], unsigned int num); extern void kern_unmount_array(struct vfsmount *mnt[], unsigned int num);
extern int cifs_root_data(char **dev, char **opts);
#endif /* _LINUX_MOUNT_H */ #endif /* _LINUX_MOUNT_H */
...@@ -7,19 +7,19 @@ ...@@ -7,19 +7,19 @@
#include <linux/sched.h> #include <linux/sched.h>
#include <asm/irq.h> #include <asm/irq.h>
#if defined(CONFIG_HAVE_NMI_WATCHDOG)
/* Arch specific watchdogs might need to share extra watchdog-related APIs. */
#if defined(CONFIG_HARDLOCKUP_DETECTOR_ARCH) || defined(CONFIG_HARDLOCKUP_DETECTOR_SPARC64)
#include <asm/nmi.h> #include <asm/nmi.h>
#endif #endif
#ifdef CONFIG_LOCKUP_DETECTOR #ifdef CONFIG_LOCKUP_DETECTOR
void lockup_detector_init(void); void lockup_detector_init(void);
void lockup_detector_retry_init(void);
void lockup_detector_soft_poweroff(void); void lockup_detector_soft_poweroff(void);
void lockup_detector_cleanup(void); void lockup_detector_cleanup(void);
bool is_hardlockup(void);
extern int watchdog_user_enabled; extern int watchdog_user_enabled;
extern int nmi_watchdog_user_enabled;
extern int soft_watchdog_user_enabled;
extern int watchdog_thresh; extern int watchdog_thresh;
extern unsigned long watchdog_enabled; extern unsigned long watchdog_enabled;
...@@ -35,6 +35,7 @@ extern int sysctl_hardlockup_all_cpu_backtrace; ...@@ -35,6 +35,7 @@ extern int sysctl_hardlockup_all_cpu_backtrace;
#else /* CONFIG_LOCKUP_DETECTOR */ #else /* CONFIG_LOCKUP_DETECTOR */
static inline void lockup_detector_init(void) { } static inline void lockup_detector_init(void) { }
static inline void lockup_detector_retry_init(void) { }
static inline void lockup_detector_soft_poweroff(void) { } static inline void lockup_detector_soft_poweroff(void) { }
static inline void lockup_detector_cleanup(void) { } static inline void lockup_detector_cleanup(void) { }
#endif /* !CONFIG_LOCKUP_DETECTOR */ #endif /* !CONFIG_LOCKUP_DETECTOR */
...@@ -69,17 +70,17 @@ static inline void reset_hung_task_detector(void) { } ...@@ -69,17 +70,17 @@ static inline void reset_hung_task_detector(void) { }
* 'watchdog_enabled' variable. Each lockup detector has its dedicated bit - * 'watchdog_enabled' variable. Each lockup detector has its dedicated bit -
* bit 0 for the hard lockup detector and bit 1 for the soft lockup detector. * bit 0 for the hard lockup detector and bit 1 for the soft lockup detector.
* *
* 'watchdog_user_enabled', 'nmi_watchdog_user_enabled' and * 'watchdog_user_enabled', 'watchdog_hardlockup_user_enabled' and
* 'soft_watchdog_user_enabled' are variables that are only used as an * 'watchdog_softlockup_user_enabled' are variables that are only used as an
* 'interface' between the parameters in /proc/sys/kernel and the internal * 'interface' between the parameters in /proc/sys/kernel and the internal
* state bits in 'watchdog_enabled'. The 'watchdog_thresh' variable is * state bits in 'watchdog_enabled'. The 'watchdog_thresh' variable is
* handled differently because its value is not boolean, and the lockup * handled differently because its value is not boolean, and the lockup
* detectors are 'suspended' while 'watchdog_thresh' is equal zero. * detectors are 'suspended' while 'watchdog_thresh' is equal zero.
*/ */
#define NMI_WATCHDOG_ENABLED_BIT 0 #define WATCHDOG_HARDLOCKUP_ENABLED_BIT 0
#define SOFT_WATCHDOG_ENABLED_BIT 1 #define WATCHDOG_SOFTOCKUP_ENABLED_BIT 1
#define NMI_WATCHDOG_ENABLED (1 << NMI_WATCHDOG_ENABLED_BIT) #define WATCHDOG_HARDLOCKUP_ENABLED (1 << WATCHDOG_HARDLOCKUP_ENABLED_BIT)
#define SOFT_WATCHDOG_ENABLED (1 << SOFT_WATCHDOG_ENABLED_BIT) #define WATCHDOG_SOFTOCKUP_ENABLED (1 << WATCHDOG_SOFTOCKUP_ENABLED_BIT)
#if defined(CONFIG_HARDLOCKUP_DETECTOR) #if defined(CONFIG_HARDLOCKUP_DETECTOR)
extern void hardlockup_detector_disable(void); extern void hardlockup_detector_disable(void);
...@@ -88,52 +89,63 @@ extern unsigned int hardlockup_panic; ...@@ -88,52 +89,63 @@ extern unsigned int hardlockup_panic;
static inline void hardlockup_detector_disable(void) {} static inline void hardlockup_detector_disable(void) {}
#endif #endif
#if defined(CONFIG_HAVE_NMI_WATCHDOG) || defined(CONFIG_HARDLOCKUP_DETECTOR) /* Sparc64 has special implemetantion that is always enabled. */
# define NMI_WATCHDOG_SYSCTL_PERM 0644 #if defined(CONFIG_HARDLOCKUP_DETECTOR) || defined(CONFIG_HARDLOCKUP_DETECTOR_SPARC64)
void arch_touch_nmi_watchdog(void);
#else #else
# define NMI_WATCHDOG_SYSCTL_PERM 0444 static inline void arch_touch_nmi_watchdog(void) { }
#endif
#if defined(CONFIG_HARDLOCKUP_DETECTOR_COUNTS_HRTIMER)
void watchdog_hardlockup_touch_cpu(unsigned int cpu);
void watchdog_hardlockup_check(unsigned int cpu, struct pt_regs *regs);
#endif #endif
#if defined(CONFIG_HARDLOCKUP_DETECTOR_PERF) #if defined(CONFIG_HARDLOCKUP_DETECTOR_PERF)
extern void arch_touch_nmi_watchdog(void);
extern void hardlockup_detector_perf_stop(void); extern void hardlockup_detector_perf_stop(void);
extern void hardlockup_detector_perf_restart(void); extern void hardlockup_detector_perf_restart(void);
extern void hardlockup_detector_perf_disable(void);
extern void hardlockup_detector_perf_enable(void);
extern void hardlockup_detector_perf_cleanup(void); extern void hardlockup_detector_perf_cleanup(void);
extern int hardlockup_detector_perf_init(void);
#else #else
static inline void hardlockup_detector_perf_stop(void) { } static inline void hardlockup_detector_perf_stop(void) { }
static inline void hardlockup_detector_perf_restart(void) { } static inline void hardlockup_detector_perf_restart(void) { }
static inline void hardlockup_detector_perf_disable(void) { }
static inline void hardlockup_detector_perf_enable(void) { }
static inline void hardlockup_detector_perf_cleanup(void) { } static inline void hardlockup_detector_perf_cleanup(void) { }
# if !defined(CONFIG_HAVE_NMI_WATCHDOG)
static inline int hardlockup_detector_perf_init(void) { return -ENODEV; }
static inline void arch_touch_nmi_watchdog(void) {}
# else
static inline int hardlockup_detector_perf_init(void) { return 0; }
# endif
#endif #endif
void watchdog_nmi_stop(void); void watchdog_hardlockup_stop(void);
void watchdog_nmi_start(void); void watchdog_hardlockup_start(void);
int watchdog_nmi_probe(void); int watchdog_hardlockup_probe(void);
int watchdog_nmi_enable(unsigned int cpu); void watchdog_hardlockup_enable(unsigned int cpu);
void watchdog_nmi_disable(unsigned int cpu); void watchdog_hardlockup_disable(unsigned int cpu);
void lockup_detector_reconfigure(void); void lockup_detector_reconfigure(void);
#ifdef CONFIG_HARDLOCKUP_DETECTOR_BUDDY
void watchdog_buddy_check_hardlockup(int hrtimer_interrupts);
#else
static inline void watchdog_buddy_check_hardlockup(int hrtimer_interrupts) {}
#endif
/** /**
* touch_nmi_watchdog - restart NMI watchdog timeout. * touch_nmi_watchdog - manually reset the hardlockup watchdog timeout.
* *
* If the architecture supports the NMI watchdog, touch_nmi_watchdog() * If we support detecting hardlockups, touch_nmi_watchdog() may be
* may be used to reset the timeout - for code which intentionally * used to pet the watchdog (reset the timeout) - for code which
* disables interrupts for a long time. This call is stateless. * intentionally disables interrupts for a long time. This call is stateless.
*
* Though this function has "nmi" in the name, the hardlockup watchdog might
* not be backed by NMIs. This function will likely be renamed to
* touch_hardlockup_watchdog() in the future.
*/ */
static inline void touch_nmi_watchdog(void) static inline void touch_nmi_watchdog(void)
{ {
/*
* Pass on to the hardlockup detector selected via CONFIG_. Note that
* the hardlockup detector may not be arch-specific nor using NMIs
* and the arch_touch_nmi_watchdog() function will likely be renamed
* in the future.
*/
arch_touch_nmi_watchdog(); arch_touch_nmi_watchdog();
touch_softlockup_watchdog(); touch_softlockup_watchdog();
} }
...@@ -194,10 +206,11 @@ static inline bool trigger_single_cpu_backtrace(int cpu) ...@@ -194,10 +206,11 @@ static inline bool trigger_single_cpu_backtrace(int cpu)
#ifdef CONFIG_HARDLOCKUP_DETECTOR_PERF #ifdef CONFIG_HARDLOCKUP_DETECTOR_PERF
u64 hw_nmi_get_sample_period(int watchdog_thresh); u64 hw_nmi_get_sample_period(int watchdog_thresh);
bool arch_perf_nmi_is_available(void);
#endif #endif
#if defined(CONFIG_HARDLOCKUP_CHECK_TIMESTAMP) && \ #if defined(CONFIG_HARDLOCKUP_CHECK_TIMESTAMP) && \
defined(CONFIG_HARDLOCKUP_DETECTOR) defined(CONFIG_HARDLOCKUP_DETECTOR_PERF)
void watchdog_update_hrtimer_threshold(u64 period); void watchdog_update_hrtimer_threshold(u64 period);
#else #else
static inline void watchdog_update_hrtimer_threshold(u64 period) { } static inline void watchdog_update_hrtimer_threshold(u64 period) { }
......
...@@ -32,6 +32,9 @@ extern int sysctl_panic_on_stackoverflow; ...@@ -32,6 +32,9 @@ extern int sysctl_panic_on_stackoverflow;
extern bool crash_kexec_post_notifiers; extern bool crash_kexec_post_notifiers;
extern void __stack_chk_fail(void);
void abort(void);
/* /*
* panic_cpu is used for synchronizing panic() and crash_kexec() execution. It * panic_cpu is used for synchronizing panic() and crash_kexec() execution. It
* holds a CPU number which is executing panic() currently. A value of * holds a CPU number which is executing panic() currently. A value of
......
...@@ -103,12 +103,10 @@ extern void __init pcpu_free_alloc_info(struct pcpu_alloc_info *ai); ...@@ -103,12 +103,10 @@ extern void __init pcpu_free_alloc_info(struct pcpu_alloc_info *ai);
extern void __init pcpu_setup_first_chunk(const struct pcpu_alloc_info *ai, extern void __init pcpu_setup_first_chunk(const struct pcpu_alloc_info *ai,
void *base_addr); void *base_addr);
#ifdef CONFIG_NEED_PER_CPU_EMBED_FIRST_CHUNK
extern int __init pcpu_embed_first_chunk(size_t reserved_size, size_t dyn_size, extern int __init pcpu_embed_first_chunk(size_t reserved_size, size_t dyn_size,
size_t atom_size, size_t atom_size,
pcpu_fc_cpu_distance_fn_t cpu_distance_fn, pcpu_fc_cpu_distance_fn_t cpu_distance_fn,
pcpu_fc_cpu_to_node_fn_t cpu_to_nd_fn); pcpu_fc_cpu_to_node_fn_t cpu_to_nd_fn);
#endif
#ifdef CONFIG_NEED_PER_CPU_PAGE_FIRST_CHUNK #ifdef CONFIG_NEED_PER_CPU_PAGE_FIRST_CHUNK
void __init pcpu_populate_pte(unsigned long addr); void __init pcpu_populate_pte(unsigned long addr);
......
...@@ -173,6 +173,8 @@ void kvm_host_pmu_init(struct arm_pmu *pmu); ...@@ -173,6 +173,8 @@ void kvm_host_pmu_init(struct arm_pmu *pmu);
#define kvm_host_pmu_init(x) do { } while(0) #define kvm_host_pmu_init(x) do { } while(0)
#endif #endif
bool arm_pmu_irq_is_nmi(void);
/* Internal functions only for core arm_pmu code */ /* Internal functions only for core arm_pmu code */
struct arm_pmu *armpmu_alloc(void); struct arm_pmu *armpmu_alloc(void);
void armpmu_free(struct arm_pmu *pmu); void armpmu_free(struct arm_pmu *pmu);
......
...@@ -256,6 +256,11 @@ check_copy_size(const void *addr, size_t bytes, bool is_source) ...@@ -256,6 +256,11 @@ check_copy_size(const void *addr, size_t bytes, bool is_source)
static inline void arch_setup_new_exec(void) { } static inline void arch_setup_new_exec(void) { }
#endif #endif
void arch_task_cache_init(void); /* for CONFIG_SH */
void arch_release_task_struct(struct task_struct *tsk);
int arch_dup_task_struct(struct task_struct *dst,
struct task_struct *src);
#endif /* __KERNEL__ */ #endif /* __KERNEL__ */
#endif /* _LINUX_THREAD_INFO_H */ #endif /* _LINUX_THREAD_INFO_H */
...@@ -40,6 +40,7 @@ typedef __kernel_uid16_t uid16_t; ...@@ -40,6 +40,7 @@ typedef __kernel_uid16_t uid16_t;
typedef __kernel_gid16_t gid16_t; typedef __kernel_gid16_t gid16_t;
typedef unsigned long uintptr_t; typedef unsigned long uintptr_t;
typedef long intptr_t;
#ifdef CONFIG_HAVE_UID16 #ifdef CONFIG_HAVE_UID16
/* This is defined by include/asm-{arch}/posix_types.h */ /* This is defined by include/asm-{arch}/posix_types.h */
......
...@@ -192,8 +192,19 @@ void __init mount_root_generic(char *name, char *pretty_name, int flags) ...@@ -192,8 +192,19 @@ void __init mount_root_generic(char *name, char *pretty_name, int flags)
printk("VFS: Cannot open root device \"%s\" or %s: error %d\n", printk("VFS: Cannot open root device \"%s\" or %s: error %d\n",
pretty_name, b, err); pretty_name, b, err);
printk("Please append a correct \"root=\" boot option; here are the available partitions:\n"); printk("Please append a correct \"root=\" boot option; here are the available partitions:\n");
printk_all_partitions(); printk_all_partitions();
if (root_fs_names)
num_fs = list_bdev_fs_names(fs_names, PAGE_SIZE);
if (!num_fs)
pr_err("Can't find any bdev filesystem to be used for mount!\n");
else {
pr_err("List of all bdev filesystems:\n");
for (i = 0, p = fs_names; i < num_fs; i++, p += strlen(p)+1)
pr_err(" %s", p);
pr_err("\n");
}
panic("VFS: Unable to mount root fs on %s", b); panic("VFS: Unable to mount root fs on %s", b);
} }
if (!(flags & SB_RDONLY)) { if (!(flags & SB_RDONLY)) {
...@@ -256,8 +267,6 @@ static inline void mount_nfs_root(void) ...@@ -256,8 +267,6 @@ static inline void mount_nfs_root(void)
#ifdef CONFIG_CIFS_ROOT #ifdef CONFIG_CIFS_ROOT
extern int cifs_root_data(char **dev, char **opts);
#define CIFSROOT_TIMEOUT_MIN 5 #define CIFSROOT_TIMEOUT_MIN 5
#define CIFSROOT_TIMEOUT_MAX 30 #define CIFSROOT_TIMEOUT_MAX 30
#define CIFSROOT_RETRY_MAX 5 #define CIFSROOT_RETRY_MAX 5
......
...@@ -113,10 +113,6 @@ ...@@ -113,10 +113,6 @@
static int kernel_init(void *); static int kernel_init(void *);
extern void init_IRQ(void);
extern void radix_tree_init(void);
extern void maple_tree_init(void);
/* /*
* Debug helper: via this flag we know that we are in 'early bootup code' * Debug helper: via this flag we know that we are in 'early bootup code'
* where only the boot processor is running with IRQ disabled. This means * where only the boot processor is running with IRQ disabled. This means
...@@ -135,7 +131,6 @@ EXPORT_SYMBOL(system_state); ...@@ -135,7 +131,6 @@ EXPORT_SYMBOL(system_state);
#define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT #define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT
#define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT #define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT
extern void time_init(void);
/* Default late time init is NULL. archs can override this later. */ /* Default late time init is NULL. archs can override this later. */
void (*__initdata late_time_init)(void); void (*__initdata late_time_init)(void);
...@@ -194,8 +189,6 @@ static const char *argv_init[MAX_INIT_ARGS+2] = { "init", NULL, }; ...@@ -194,8 +189,6 @@ static const char *argv_init[MAX_INIT_ARGS+2] = { "init", NULL, };
const char *envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, }; const char *envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, };
static const char *panic_later, *panic_param; static const char *panic_later, *panic_param;
extern const struct obs_kernel_param __setup_start[], __setup_end[];
static bool __init obsolete_checksetup(char *line) static bool __init obsolete_checksetup(char *line)
{ {
const struct obs_kernel_param *p; const struct obs_kernel_param *p;
...@@ -1256,17 +1249,6 @@ int __init_or_module do_one_initcall(initcall_t fn) ...@@ -1256,17 +1249,6 @@ int __init_or_module do_one_initcall(initcall_t fn)
} }
extern initcall_entry_t __initcall_start[];
extern initcall_entry_t __initcall0_start[];
extern initcall_entry_t __initcall1_start[];
extern initcall_entry_t __initcall2_start[];
extern initcall_entry_t __initcall3_start[];
extern initcall_entry_t __initcall4_start[];
extern initcall_entry_t __initcall5_start[];
extern initcall_entry_t __initcall6_start[];
extern initcall_entry_t __initcall7_start[];
extern initcall_entry_t __initcall_end[];
static initcall_entry_t *initcall_levels[] __initdata = { static initcall_entry_t *initcall_levels[] __initdata = {
__initcall0_start, __initcall0_start,
__initcall1_start, __initcall1_start,
......
...@@ -91,7 +91,8 @@ obj-$(CONFIG_FAIL_FUNCTION) += fail_function.o ...@@ -91,7 +91,8 @@ obj-$(CONFIG_FAIL_FUNCTION) += fail_function.o
obj-$(CONFIG_KGDB) += debug/ obj-$(CONFIG_KGDB) += debug/
obj-$(CONFIG_DETECT_HUNG_TASK) += hung_task.o obj-$(CONFIG_DETECT_HUNG_TASK) += hung_task.o
obj-$(CONFIG_LOCKUP_DETECTOR) += watchdog.o obj-$(CONFIG_LOCKUP_DETECTOR) += watchdog.o
obj-$(CONFIG_HARDLOCKUP_DETECTOR_PERF) += watchdog_hld.o obj-$(CONFIG_HARDLOCKUP_DETECTOR_BUDDY) += watchdog_buddy.o
obj-$(CONFIG_HARDLOCKUP_DETECTOR_PERF) += watchdog_perf.o
obj-$(CONFIG_SECCOMP) += seccomp.o obj-$(CONFIG_SECCOMP) += seccomp.o
obj-$(CONFIG_RELAY) += relay.o obj-$(CONFIG_RELAY) += relay.o
obj-$(CONFIG_SYSCTL) += utsname_sysctl.o obj-$(CONFIG_SYSCTL) += utsname_sysctl.o
......
...@@ -252,23 +252,19 @@ static int memcg_charge_kernel_stack(struct vm_struct *vm) ...@@ -252,23 +252,19 @@ static int memcg_charge_kernel_stack(struct vm_struct *vm)
{ {
int i; int i;
int ret; int ret;
int nr_charged = 0;
BUILD_BUG_ON(IS_ENABLED(CONFIG_VMAP_STACK) && PAGE_SIZE % 1024 != 0);
BUG_ON(vm->nr_pages != THREAD_SIZE / PAGE_SIZE); BUG_ON(vm->nr_pages != THREAD_SIZE / PAGE_SIZE);
for (i = 0; i < THREAD_SIZE / PAGE_SIZE; i++) { for (i = 0; i < THREAD_SIZE / PAGE_SIZE; i++) {
ret = memcg_kmem_charge_page(vm->pages[i], GFP_KERNEL, 0); ret = memcg_kmem_charge_page(vm->pages[i], GFP_KERNEL, 0);
if (ret) if (ret)
goto err; goto err;
nr_charged++;
} }
return 0; return 0;
err: err:
/* for (i = 0; i < nr_charged; i++)
* If memcg_kmem_charge_page() fails, page's memory cgroup pointer is
* NULL, and memcg_kmem_uncharge_page() in free_thread_stack() will
* ignore this page.
*/
for (i = 0; i < THREAD_SIZE / PAGE_SIZE; i++)
memcg_kmem_uncharge_page(vm->pages[i], 0); memcg_kmem_uncharge_page(vm->pages[i], 0);
return ret; return ret;
} }
......
...@@ -279,7 +279,7 @@ void notrace __sanitizer_cov_trace_cmp4(u32 arg1, u32 arg2) ...@@ -279,7 +279,7 @@ void notrace __sanitizer_cov_trace_cmp4(u32 arg1, u32 arg2)
} }
EXPORT_SYMBOL(__sanitizer_cov_trace_cmp4); EXPORT_SYMBOL(__sanitizer_cov_trace_cmp4);
void notrace __sanitizer_cov_trace_cmp8(u64 arg1, u64 arg2) void notrace __sanitizer_cov_trace_cmp8(kcov_u64 arg1, kcov_u64 arg2)
{ {
write_comp_data(KCOV_CMP_SIZE(3), arg1, arg2, _RET_IP_); write_comp_data(KCOV_CMP_SIZE(3), arg1, arg2, _RET_IP_);
} }
...@@ -306,16 +306,17 @@ void notrace __sanitizer_cov_trace_const_cmp4(u32 arg1, u32 arg2) ...@@ -306,16 +306,17 @@ void notrace __sanitizer_cov_trace_const_cmp4(u32 arg1, u32 arg2)
} }
EXPORT_SYMBOL(__sanitizer_cov_trace_const_cmp4); EXPORT_SYMBOL(__sanitizer_cov_trace_const_cmp4);
void notrace __sanitizer_cov_trace_const_cmp8(u64 arg1, u64 arg2) void notrace __sanitizer_cov_trace_const_cmp8(kcov_u64 arg1, kcov_u64 arg2)
{ {
write_comp_data(KCOV_CMP_SIZE(3) | KCOV_CMP_CONST, arg1, arg2, write_comp_data(KCOV_CMP_SIZE(3) | KCOV_CMP_CONST, arg1, arg2,
_RET_IP_); _RET_IP_);
} }
EXPORT_SYMBOL(__sanitizer_cov_trace_const_cmp8); EXPORT_SYMBOL(__sanitizer_cov_trace_const_cmp8);
void notrace __sanitizer_cov_trace_switch(u64 val, u64 *cases) void notrace __sanitizer_cov_trace_switch(kcov_u64 val, void *arg)
{ {
u64 i; u64 i;
u64 *cases = arg;
u64 count = cases[0]; u64 count = cases[0];
u64 size = cases[1]; u64 size = cases[1];
u64 type = KCOV_CMP_CONST; u64 type = KCOV_CMP_CONST;
......
...@@ -1091,6 +1091,11 @@ __bpf_kfunc void crash_kexec(struct pt_regs *regs) ...@@ -1091,6 +1091,11 @@ __bpf_kfunc void crash_kexec(struct pt_regs *regs)
} }
} }
static inline resource_size_t crash_resource_size(const struct resource *res)
{
return !res->end ? 0 : resource_size(res);
}
ssize_t crash_get_memory_size(void) ssize_t crash_get_memory_size(void)
{ {
ssize_t size = 0; ssize_t size = 0;
...@@ -1098,19 +1103,45 @@ ssize_t crash_get_memory_size(void) ...@@ -1098,19 +1103,45 @@ ssize_t crash_get_memory_size(void)
if (!kexec_trylock()) if (!kexec_trylock())
return -EBUSY; return -EBUSY;
if (crashk_res.end != crashk_res.start) size += crash_resource_size(&crashk_res);
size = resource_size(&crashk_res); size += crash_resource_size(&crashk_low_res);
kexec_unlock(); kexec_unlock();
return size; return size;
} }
static int __crash_shrink_memory(struct resource *old_res,
unsigned long new_size)
{
struct resource *ram_res;
ram_res = kzalloc(sizeof(*ram_res), GFP_KERNEL);
if (!ram_res)
return -ENOMEM;
ram_res->start = old_res->start + new_size;
ram_res->end = old_res->end;
ram_res->flags = IORESOURCE_BUSY | IORESOURCE_SYSTEM_RAM;
ram_res->name = "System RAM";
if (!new_size) {
release_resource(old_res);
old_res->start = 0;
old_res->end = 0;
} else {
crashk_res.end = ram_res->start - 1;
}
crash_free_reserved_phys_range(ram_res->start, ram_res->end);
insert_resource(&iomem_resource, ram_res);
return 0;
}
int crash_shrink_memory(unsigned long new_size) int crash_shrink_memory(unsigned long new_size)
{ {
int ret = 0; int ret = 0;
unsigned long start, end; unsigned long old_size, low_size;
unsigned long old_size;
struct resource *ram_res;
if (!kexec_trylock()) if (!kexec_trylock())
return -EBUSY; return -EBUSY;
...@@ -1119,36 +1150,42 @@ int crash_shrink_memory(unsigned long new_size) ...@@ -1119,36 +1150,42 @@ int crash_shrink_memory(unsigned long new_size)
ret = -ENOENT; ret = -ENOENT;
goto unlock; goto unlock;
} }
start = crashk_res.start;
end = crashk_res.end; low_size = crash_resource_size(&crashk_low_res);
old_size = (end == 0) ? 0 : end - start + 1; old_size = crash_resource_size(&crashk_res) + low_size;
new_size = roundup(new_size, KEXEC_CRASH_MEM_ALIGN);
if (new_size >= old_size) { if (new_size >= old_size) {
ret = (new_size == old_size) ? 0 : -EINVAL; ret = (new_size == old_size) ? 0 : -EINVAL;
goto unlock; goto unlock;
} }
ram_res = kzalloc(sizeof(*ram_res), GFP_KERNEL); /*
if (!ram_res) { * (low_size > new_size) implies that low_size is greater than zero.
ret = -ENOMEM; * This also means that if low_size is zero, the else branch is taken.
goto unlock; *
} * If low_size is greater than 0, (low_size > new_size) indicates that
* crashk_low_res also needs to be shrunken. Otherwise, only crashk_res
start = roundup(start, KEXEC_CRASH_MEM_ALIGN); * needs to be shrunken.
end = roundup(start + new_size, KEXEC_CRASH_MEM_ALIGN); */
if (low_size > new_size) {
crash_free_reserved_phys_range(end, crashk_res.end); ret = __crash_shrink_memory(&crashk_res, 0);
if (ret)
if ((start == end) && (crashk_res.parent != NULL)) goto unlock;
release_resource(&crashk_res);
ram_res->start = end;
ram_res->end = crashk_res.end;
ram_res->flags = IORESOURCE_BUSY | IORESOURCE_SYSTEM_RAM;
ram_res->name = "System RAM";
crashk_res.end = end - 1; ret = __crash_shrink_memory(&crashk_low_res, new_size);
} else {
ret = __crash_shrink_memory(&crashk_res, new_size - low_size);
}
insert_resource(&iomem_resource, ram_res); /* Swap crashk_res and crashk_low_res if needed */
if (!crashk_res.end && crashk_low_res.end) {
crashk_res.start = crashk_low_res.start;
crashk_res.end = crashk_low_res.end;
release_resource(&crashk_low_res);
crashk_low_res.start = 0;
crashk_low_res.end = 0;
insert_resource(&iomem_resource, &crashk_res);
}
unlock: unlock:
kexec_unlock(); kexec_unlock();
......
...@@ -867,6 +867,7 @@ static int kexec_purgatory_setup_sechdrs(struct purgatory_info *pi, ...@@ -867,6 +867,7 @@ static int kexec_purgatory_setup_sechdrs(struct purgatory_info *pi,
{ {
unsigned long bss_addr; unsigned long bss_addr;
unsigned long offset; unsigned long offset;
size_t sechdrs_size;
Elf_Shdr *sechdrs; Elf_Shdr *sechdrs;
int i; int i;
...@@ -874,11 +875,11 @@ static int kexec_purgatory_setup_sechdrs(struct purgatory_info *pi, ...@@ -874,11 +875,11 @@ static int kexec_purgatory_setup_sechdrs(struct purgatory_info *pi,
* The section headers in kexec_purgatory are read-only. In order to * The section headers in kexec_purgatory are read-only. In order to
* have them modifiable make a temporary copy. * have them modifiable make a temporary copy.
*/ */
sechdrs = vzalloc(array_size(sizeof(Elf_Shdr), pi->ehdr->e_shnum)); sechdrs_size = array_size(sizeof(Elf_Shdr), pi->ehdr->e_shnum);
sechdrs = vzalloc(sechdrs_size);
if (!sechdrs) if (!sechdrs)
return -ENOMEM; return -ENOMEM;
memcpy(sechdrs, (void *)pi->ehdr + pi->ehdr->e_shoff, memcpy(sechdrs, (void *)pi->ehdr + pi->ehdr->e_shoff, sechdrs_size);
pi->ehdr->e_shnum * sizeof(Elf_Shdr));
pi->sechdrs = sechdrs; pi->sechdrs = sechdrs;
offset = 0; offset = 0;
......
...@@ -322,10 +322,10 @@ void __noreturn kthread_exit(long result) ...@@ -322,10 +322,10 @@ void __noreturn kthread_exit(long result)
* @comp: Completion to complete * @comp: Completion to complete
* @code: The integer value to return to kthread_stop(). * @code: The integer value to return to kthread_stop().
* *
* If present complete @comp and the reuturn code to kthread_stop(). * If present, complete @comp and then return code to kthread_stop().
* *
* A kernel thread whose module may be removed after the completion of * A kernel thread whose module may be removed after the completion of
* @comp can use this function exit safely. * @comp can use this function to exit safely.
* *
* Does not return. * Does not return.
*/ */
......
...@@ -57,4 +57,8 @@ static inline void __lockevent_add(enum lock_events event, int inc) ...@@ -57,4 +57,8 @@ static inline void __lockevent_add(enum lock_events event, int inc)
#define lockevent_cond_inc(ev, c) #define lockevent_cond_inc(ev, c)
#endif /* CONFIG_LOCK_EVENT_COUNTS */ #endif /* CONFIG_LOCK_EVENT_COUNTS */
ssize_t lockevent_read(struct file *file, char __user *user_buf,
size_t count, loff_t *ppos);
#endif /* __LOCKING_LOCK_EVENTS_H */ #endif /* __LOCKING_LOCK_EVENTS_H */
...@@ -684,6 +684,7 @@ void __warn(const char *file, int line, void *caller, unsigned taint, ...@@ -684,6 +684,7 @@ void __warn(const char *file, int line, void *caller, unsigned taint,
add_taint(taint, LOCKDEP_STILL_OK); add_taint(taint, LOCKDEP_STILL_OK);
} }
#ifdef CONFIG_BUG
#ifndef __WARN_FLAGS #ifndef __WARN_FLAGS
void warn_slowpath_fmt(const char *file, int line, unsigned taint, void warn_slowpath_fmt(const char *file, int line, unsigned taint,
const char *fmt, ...) const char *fmt, ...)
...@@ -722,8 +723,6 @@ void __warn_printk(const char *fmt, ...) ...@@ -722,8 +723,6 @@ void __warn_printk(const char *fmt, ...)
EXPORT_SYMBOL(__warn_printk); EXPORT_SYMBOL(__warn_printk);
#endif #endif
#ifdef CONFIG_BUG
/* Support resetting WARN*_ONCE state */ /* Support resetting WARN*_ONCE state */
static int clear_warn_once_set(void *data, u64 val) static int clear_warn_once_set(void *data, u64 val)
......
This diff is collapsed.
// SPDX-License-Identifier: GPL-2.0
#include <linux/cpu.h>
#include <linux/cpumask.h>
#include <linux/kernel.h>
#include <linux/nmi.h>
#include <linux/percpu-defs.h>
static cpumask_t __read_mostly watchdog_cpus;
static unsigned int watchdog_next_cpu(unsigned int cpu)
{
unsigned int next_cpu;
next_cpu = cpumask_next(cpu, &watchdog_cpus);
if (next_cpu >= nr_cpu_ids)
next_cpu = cpumask_first(&watchdog_cpus);
if (next_cpu == cpu)
return nr_cpu_ids;
return next_cpu;
}
int __init watchdog_hardlockup_probe(void)
{
return 0;
}
void watchdog_hardlockup_enable(unsigned int cpu)
{
unsigned int next_cpu;
/*
* The new CPU will be marked online before the hrtimer interrupt
* gets a chance to run on it. If another CPU tests for a
* hardlockup on the new CPU before it has run its the hrtimer
* interrupt, it will get a false positive. Touch the watchdog on
* the new CPU to delay the check for at least 3 sampling periods
* to guarantee one hrtimer has run on the new CPU.
*/
watchdog_hardlockup_touch_cpu(cpu);
/*
* We are going to check the next CPU. Our watchdog_hrtimer
* need not be zero if the CPU has already been online earlier.
* Touch the watchdog on the next CPU to avoid false positive
* if we try to check it in less then 3 interrupts.
*/
next_cpu = watchdog_next_cpu(cpu);
if (next_cpu < nr_cpu_ids)
watchdog_hardlockup_touch_cpu(next_cpu);
/*
* Makes sure that watchdog is touched on this CPU before
* other CPUs could see it in watchdog_cpus. The counter
* part is in watchdog_buddy_check_hardlockup().
*/
smp_wmb();
cpumask_set_cpu(cpu, &watchdog_cpus);
}
void watchdog_hardlockup_disable(unsigned int cpu)
{
unsigned int next_cpu = watchdog_next_cpu(cpu);
/*
* Offlining this CPU will cause the CPU before this one to start
* checking the one after this one. If this CPU just finished checking
* the next CPU and updating hrtimer_interrupts_saved, and then the
* previous CPU checks it within one sample period, it will trigger a
* false positive. Touch the watchdog on the next CPU to prevent it.
*/
if (next_cpu < nr_cpu_ids)
watchdog_hardlockup_touch_cpu(next_cpu);
/*
* Makes sure that watchdog is touched on the next CPU before
* this CPU disappear in watchdog_cpus. The counter part is in
* watchdog_buddy_check_hardlockup().
*/
smp_wmb();
cpumask_clear_cpu(cpu, &watchdog_cpus);
}
void watchdog_buddy_check_hardlockup(int hrtimer_interrupts)
{
unsigned int next_cpu;
/*
* Test for hardlockups every 3 samples. The sample period is
* watchdog_thresh * 2 / 5, so 3 samples gets us back to slightly over
* watchdog_thresh (over by 20%).
*/
if (hrtimer_interrupts % 3 != 0)
return;
/* check for a hardlockup on the next CPU */
next_cpu = watchdog_next_cpu(smp_processor_id());
if (next_cpu >= nr_cpu_ids)
return;
/*
* Make sure that the watchdog was touched on next CPU when
* watchdog_next_cpu() returned another one because of
* a change in watchdog_hardlockup_enable()/disable().
*/
smp_rmb();
watchdog_hardlockup_check(next_cpu, NULL);
}
// SPDX-License-Identifier: GPL-2.0 // SPDX-License-Identifier: GPL-2.0
/* /*
* Detect hard lockups on a system * Detect hard lockups on a system using perf
* *
* started by Don Zickus, Copyright (C) 2010 Red Hat, Inc. * started by Don Zickus, Copyright (C) 2010 Red Hat, Inc.
* *
...@@ -20,28 +20,12 @@ ...@@ -20,28 +20,12 @@
#include <asm/irq_regs.h> #include <asm/irq_regs.h>
#include <linux/perf_event.h> #include <linux/perf_event.h>
static DEFINE_PER_CPU(bool, hard_watchdog_warn);
static DEFINE_PER_CPU(bool, watchdog_nmi_touch);
static DEFINE_PER_CPU(struct perf_event *, watchdog_ev); static DEFINE_PER_CPU(struct perf_event *, watchdog_ev);
static DEFINE_PER_CPU(struct perf_event *, dead_event); static DEFINE_PER_CPU(struct perf_event *, dead_event);
static struct cpumask dead_events_mask; static struct cpumask dead_events_mask;
static unsigned long hardlockup_allcpu_dumped;
static atomic_t watchdog_cpus = ATOMIC_INIT(0); static atomic_t watchdog_cpus = ATOMIC_INIT(0);
notrace void arch_touch_nmi_watchdog(void)
{
/*
* Using __raw here because some code paths have
* preemption enabled. If preemption is enabled
* then interrupts should be enabled too, in which
* case we shouldn't have to worry about the watchdog
* going off.
*/
raw_cpu_write(watchdog_nmi_touch, true);
}
EXPORT_SYMBOL(arch_touch_nmi_watchdog);
#ifdef CONFIG_HARDLOCKUP_CHECK_TIMESTAMP #ifdef CONFIG_HARDLOCKUP_CHECK_TIMESTAMP
static DEFINE_PER_CPU(ktime_t, last_timestamp); static DEFINE_PER_CPU(ktime_t, last_timestamp);
static DEFINE_PER_CPU(unsigned int, nmi_rearmed); static DEFINE_PER_CPU(unsigned int, nmi_rearmed);
...@@ -114,61 +98,24 @@ static void watchdog_overflow_callback(struct perf_event *event, ...@@ -114,61 +98,24 @@ static void watchdog_overflow_callback(struct perf_event *event,
/* Ensure the watchdog never gets throttled */ /* Ensure the watchdog never gets throttled */
event->hw.interrupts = 0; event->hw.interrupts = 0;
if (__this_cpu_read(watchdog_nmi_touch) == true) {
__this_cpu_write(watchdog_nmi_touch, false);
return;
}
if (!watchdog_check_timestamp()) if (!watchdog_check_timestamp())
return; return;
/* check for a hardlockup watchdog_hardlockup_check(smp_processor_id(), regs);
* This is done by making sure our timer interrupt
* is incrementing. The timer interrupt should have
* fired multiple times before we overflow'd. If it hasn't
* then this is a good indication the cpu is stuck
*/
if (is_hardlockup()) {
int this_cpu = smp_processor_id();
/* only print hardlockups once */
if (__this_cpu_read(hard_watchdog_warn) == true)
return;
pr_emerg("Watchdog detected hard LOCKUP on cpu %d\n",
this_cpu);
print_modules();
print_irqtrace_events(current);
if (regs)
show_regs(regs);
else
dump_stack();
/*
* Perform all-CPU dump only once to avoid multiple hardlockups
* generating interleaving traces
*/
if (sysctl_hardlockup_all_cpu_backtrace &&
!test_and_set_bit(0, &hardlockup_allcpu_dumped))
trigger_allbutself_cpu_backtrace();
if (hardlockup_panic)
nmi_panic(regs, "Hard LOCKUP");
__this_cpu_write(hard_watchdog_warn, true);
return;
}
__this_cpu_write(hard_watchdog_warn, false);
return;
} }
static int hardlockup_detector_event_create(void) static int hardlockup_detector_event_create(void)
{ {
unsigned int cpu = smp_processor_id(); unsigned int cpu;
struct perf_event_attr *wd_attr; struct perf_event_attr *wd_attr;
struct perf_event *evt; struct perf_event *evt;
/*
* Preemption is not disabled because memory will be allocated.
* Ensure CPU-locality by calling this in per-CPU kthread.
*/
WARN_ON(!is_percpu_thread());
cpu = raw_smp_processor_id();
wd_attr = &wd_hw_attr; wd_attr = &wd_hw_attr;
wd_attr->sample_period = hw_nmi_get_sample_period(watchdog_thresh); wd_attr->sample_period = hw_nmi_get_sample_period(watchdog_thresh);
...@@ -185,10 +132,14 @@ static int hardlockup_detector_event_create(void) ...@@ -185,10 +132,14 @@ static int hardlockup_detector_event_create(void)
} }
/** /**
* hardlockup_detector_perf_enable - Enable the local event * watchdog_hardlockup_enable - Enable the local event
*
* @cpu: The CPU to enable hard lockup on.
*/ */
void hardlockup_detector_perf_enable(void) void watchdog_hardlockup_enable(unsigned int cpu)
{ {
WARN_ON_ONCE(cpu != smp_processor_id());
if (hardlockup_detector_event_create()) if (hardlockup_detector_event_create())
return; return;
...@@ -200,12 +151,16 @@ void hardlockup_detector_perf_enable(void) ...@@ -200,12 +151,16 @@ void hardlockup_detector_perf_enable(void)
} }
/** /**
* hardlockup_detector_perf_disable - Disable the local event * watchdog_hardlockup_disable - Disable the local event
*
* @cpu: The CPU to enable hard lockup on.
*/ */
void hardlockup_detector_perf_disable(void) void watchdog_hardlockup_disable(unsigned int cpu)
{ {
struct perf_event *event = this_cpu_read(watchdog_ev); struct perf_event *event = this_cpu_read(watchdog_ev);
WARN_ON_ONCE(cpu != smp_processor_id());
if (event) { if (event) {
perf_event_disable(event); perf_event_disable(event);
this_cpu_write(watchdog_ev, NULL); this_cpu_write(watchdog_ev, NULL);
...@@ -268,7 +223,7 @@ void __init hardlockup_detector_perf_restart(void) ...@@ -268,7 +223,7 @@ void __init hardlockup_detector_perf_restart(void)
lockdep_assert_cpus_held(); lockdep_assert_cpus_held();
if (!(watchdog_enabled & NMI_WATCHDOG_ENABLED)) if (!(watchdog_enabled & WATCHDOG_HARDLOCKUP_ENABLED))
return; return;
for_each_online_cpu(cpu) { for_each_online_cpu(cpu) {
...@@ -279,12 +234,22 @@ void __init hardlockup_detector_perf_restart(void) ...@@ -279,12 +234,22 @@ void __init hardlockup_detector_perf_restart(void)
} }
} }
bool __weak __init arch_perf_nmi_is_available(void)
{
return true;
}
/** /**
* hardlockup_detector_perf_init - Probe whether NMI event is available at all * watchdog_hardlockup_probe - Probe whether NMI event is available at all
*/ */
int __init hardlockup_detector_perf_init(void) int __init watchdog_hardlockup_probe(void)
{ {
int ret = hardlockup_detector_event_create(); int ret;
if (!arch_perf_nmi_is_available())
return -ENODEV;
ret = hardlockup_detector_event_create();
if (ret) { if (ret) {
pr_info("Perf NMI watchdog permanently disabled\n"); pr_info("Perf NMI watchdog permanently disabled\n");
......
...@@ -1035,27 +1035,30 @@ config BOOTPARAM_SOFTLOCKUP_PANIC ...@@ -1035,27 +1035,30 @@ config BOOTPARAM_SOFTLOCKUP_PANIC
Say N if unsure. Say N if unsure.
config HARDLOCKUP_DETECTOR_PERF config HAVE_HARDLOCKUP_DETECTOR_BUDDY
bool bool
select SOFTLOCKUP_DETECTOR depends on SMP
default y
# #
# Enables a timestamp based low pass filter to compensate for perf based # Global switch whether to build a hardlockup detector at all. It is available
# hard lockup detection which runs too fast due to turbo modes. # only when the architecture supports at least one implementation. There are
# two exceptions. The hardlockup detector is never enabled on:
# #
config HARDLOCKUP_CHECK_TIMESTAMP # s390: it reported many false positives there
bool
# #
# arch/ can define HAVE_HARDLOCKUP_DETECTOR_ARCH to provide their own hard # sparc64: has a custom implementation which is not using the common
# lockup detector rather than the perf based detector. # hardlockup command line options and sysctl interface.
# #
config HARDLOCKUP_DETECTOR config HARDLOCKUP_DETECTOR
bool "Detect Hard Lockups" bool "Detect Hard Lockups"
depends on DEBUG_KERNEL && !S390 depends on DEBUG_KERNEL && !S390 && !HARDLOCKUP_DETECTOR_SPARC64
depends on HAVE_HARDLOCKUP_DETECTOR_PERF || HAVE_HARDLOCKUP_DETECTOR_ARCH depends on HAVE_HARDLOCKUP_DETECTOR_PERF || HAVE_HARDLOCKUP_DETECTOR_BUDDY || HAVE_HARDLOCKUP_DETECTOR_ARCH
imply HARDLOCKUP_DETECTOR_PERF
imply HARDLOCKUP_DETECTOR_BUDDY
imply HARDLOCKUP_DETECTOR_ARCH
select LOCKUP_DETECTOR select LOCKUP_DETECTOR
select HARDLOCKUP_DETECTOR_PERF if HAVE_HARDLOCKUP_DETECTOR_PERF
help help
Say Y here to enable the kernel to act as a watchdog to detect Say Y here to enable the kernel to act as a watchdog to detect
hard lockups. hard lockups.
...@@ -1065,6 +1068,63 @@ config HARDLOCKUP_DETECTOR ...@@ -1065,6 +1068,63 @@ config HARDLOCKUP_DETECTOR
chance to run. The current stack trace is displayed upon detection chance to run. The current stack trace is displayed upon detection
and the system will stay locked up. and the system will stay locked up.
#
# Note that arch-specific variants are always preferred.
#
config HARDLOCKUP_DETECTOR_PREFER_BUDDY
bool "Prefer the buddy CPU hardlockup detector"
depends on HARDLOCKUP_DETECTOR
depends on HAVE_HARDLOCKUP_DETECTOR_PERF && HAVE_HARDLOCKUP_DETECTOR_BUDDY
depends on !HAVE_HARDLOCKUP_DETECTOR_ARCH
help
Say Y here to prefer the buddy hardlockup detector over the perf one.
With the buddy detector, each CPU uses its softlockup hrtimer
to check that the next CPU is processing hrtimer interrupts by
verifying that a counter is increasing.
This hardlockup detector is useful on systems that don't have
an arch-specific hardlockup detector or if resources needed
for the hardlockup detector are better used for other things.
config HARDLOCKUP_DETECTOR_PERF
bool
depends on HARDLOCKUP_DETECTOR
depends on HAVE_HARDLOCKUP_DETECTOR_PERF && !HARDLOCKUP_DETECTOR_PREFER_BUDDY
depends on !HAVE_HARDLOCKUP_DETECTOR_ARCH
select HARDLOCKUP_DETECTOR_COUNTS_HRTIMER
config HARDLOCKUP_DETECTOR_BUDDY
bool
depends on HARDLOCKUP_DETECTOR
depends on HAVE_HARDLOCKUP_DETECTOR_BUDDY
depends on !HAVE_HARDLOCKUP_DETECTOR_PERF || HARDLOCKUP_DETECTOR_PREFER_BUDDY
depends on !HAVE_HARDLOCKUP_DETECTOR_ARCH
select HARDLOCKUP_DETECTOR_COUNTS_HRTIMER
config HARDLOCKUP_DETECTOR_ARCH
bool
depends on HARDLOCKUP_DETECTOR
depends on HAVE_HARDLOCKUP_DETECTOR_ARCH
help
The arch-specific implementation of the hardlockup detector will
be used.
#
# Both the "perf" and "buddy" hardlockup detectors count hrtimer
# interrupts. This config enables functions managing this common code.
#
config HARDLOCKUP_DETECTOR_COUNTS_HRTIMER
bool
select SOFTLOCKUP_DETECTOR
#
# Enables a timestamp based low pass filter to compensate for perf based
# hard lockup detection which runs too fast due to turbo modes.
#
config HARDLOCKUP_CHECK_TIMESTAMP
bool
config BOOTPARAM_HARDLOCKUP_PANIC config BOOTPARAM_HARDLOCKUP_PANIC
bool "Panic (Reboot) On Hard Lockups" bool "Panic (Reboot) On Hard Lockups"
depends on HARDLOCKUP_DETECTOR depends on HARDLOCKUP_DETECTOR
......
...@@ -39,7 +39,7 @@ static long INIT nofill(void *buffer, unsigned long len) ...@@ -39,7 +39,7 @@ static long INIT nofill(void *buffer, unsigned long len)
} }
/* Included from initramfs et al code */ /* Included from initramfs et al code */
STATIC int INIT __gunzip(unsigned char *buf, long len, static int INIT __gunzip(unsigned char *buf, long len,
long (*fill)(void*, unsigned long), long (*fill)(void*, unsigned long),
long (*flush)(void*, unsigned long), long (*flush)(void*, unsigned long),
unsigned char *out_buf, long out_len, unsigned char *out_buf, long out_len,
......
...@@ -102,6 +102,8 @@ ...@@ -102,6 +102,8 @@
*/ */
#ifdef STATIC #ifdef STATIC
# define XZ_PREBOOT # define XZ_PREBOOT
#else
#include <linux/decompress/unxz.h>
#endif #endif
#ifdef __KERNEL__ #ifdef __KERNEL__
# include <linux/decompress/mm.h> # include <linux/decompress/mm.h>
......
...@@ -69,6 +69,8 @@ ...@@ -69,6 +69,8 @@
# define UNZSTD_PREBOOT # define UNZSTD_PREBOOT
# include "xxhash.c" # include "xxhash.c"
# include "zstd/decompress_sources.h" # include "zstd/decompress_sources.h"
#else
#include <linux/decompress/unzstd.h>
#endif #endif
#include <linux/decompress/mm.h> #include <linux/decompress/mm.h>
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include <linux/mm.h> #include <linux/mm.h>
#include <linux/ioport.h> #include <linux/ioport.h>
#include <linux/io.h>
/* /*
* devmem_is_allowed() checks to see if /dev/mem access to a certain address * devmem_is_allowed() checks to see if /dev/mem access to a certain address
......
...@@ -129,7 +129,7 @@ __devm_ioremap_resource(struct device *dev, const struct resource *res, ...@@ -129,7 +129,7 @@ __devm_ioremap_resource(struct device *dev, const struct resource *res,
BUG_ON(!dev); BUG_ON(!dev);
if (!res || resource_type(res) != IORESOURCE_MEM) { if (!res || resource_type(res) != IORESOURCE_MEM) {
dev_err(dev, "invalid resource\n"); dev_err(dev, "invalid resource %pR\n", res);
return IOMEM_ERR_PTR(-EINVAL); return IOMEM_ERR_PTR(-EINVAL);
} }
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include <kunit/test.h> #include <kunit/test.h>
#include "string-stream.h" #include "string-stream.h"
#include "debugfs.h"
#define KUNIT_DEBUGFS_ROOT "kunit" #define KUNIT_DEBUGFS_ROOT "kunit"
#define KUNIT_DEBUGFS_RESULTS "results" #define KUNIT_DEBUGFS_RESULTS "results"
......
...@@ -105,21 +105,3 @@ static uint64_t ZSTD_div64(uint64_t dividend, uint32_t divisor) { ...@@ -105,21 +105,3 @@ static uint64_t ZSTD_div64(uint64_t dividend, uint32_t divisor) {
#endif /* ZSTD_DEPS_IO */ #endif /* ZSTD_DEPS_IO */
#endif /* ZSTD_DEPS_NEED_IO */ #endif /* ZSTD_DEPS_NEED_IO */
/*
* Only requested when MSAN is enabled.
* Need:
* intptr_t
*/
#ifdef ZSTD_DEPS_NEED_STDINT
#ifndef ZSTD_DEPS_STDINT
#define ZSTD_DEPS_STDINT
/*
* The Linux Kernel doesn't provide intptr_t, only uintptr_t, which
* is an unsigned long.
*/
typedef long intptr_t;
#endif /* ZSTD_DEPS_STDINT */
#endif /* ZSTD_DEPS_NEED_STDINT */
...@@ -701,7 +701,7 @@ static int fill_subsection_map(unsigned long pfn, unsigned long nr_pages) ...@@ -701,7 +701,7 @@ static int fill_subsection_map(unsigned long pfn, unsigned long nr_pages)
return rc; return rc;
} }
#else #else
struct page * __meminit populate_section_memmap(unsigned long pfn, static struct page * __meminit populate_section_memmap(unsigned long pfn,
unsigned long nr_pages, int nid, struct vmem_altmap *altmap, unsigned long nr_pages, int nid, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap) struct dev_pagemap *pgmap)
{ {
......
...@@ -155,6 +155,7 @@ aquired||acquired ...@@ -155,6 +155,7 @@ aquired||acquired
aquisition||acquisition aquisition||acquisition
arbitary||arbitrary arbitary||arbitrary
architechture||architecture architechture||architecture
archtecture||architecture
arguement||argument arguement||argument
arguements||arguments arguements||arguments
arithmatic||arithmetic arithmatic||arithmetic
...@@ -279,6 +280,7 @@ cant'||can't ...@@ -279,6 +280,7 @@ cant'||can't
canot||cannot canot||cannot
cann't||can't cann't||can't
cannnot||cannot cannnot||cannot
capabiity||capability
capabilites||capabilities capabilites||capabilities
capabilties||capabilities capabilties||capabilities
capabilty||capability capabilty||capability
...@@ -426,6 +428,7 @@ cotrol||control ...@@ -426,6 +428,7 @@ cotrol||control
cound||could cound||could
couter||counter couter||counter
coutner||counter coutner||counter
creationg||creating
cryptocraphic||cryptographic cryptocraphic||cryptographic
cummulative||cumulative cummulative||cumulative
cunter||counter cunter||counter
...@@ -492,6 +495,7 @@ destorys||destroys ...@@ -492,6 +495,7 @@ destorys||destroys
destroied||destroyed destroied||destroyed
detabase||database detabase||database
deteced||detected deteced||detected
detecion||detection
detectt||detect detectt||detect
detroyed||destroyed detroyed||destroyed
develope||develop develope||develop
...@@ -513,6 +517,7 @@ diferent||different ...@@ -513,6 +517,7 @@ diferent||different
differrence||difference differrence||difference
diffrent||different diffrent||different
differenciate||differentiate differenciate||differentiate
diffreential||differential
diffrentiate||differentiate diffrentiate||differentiate
difinition||definition difinition||definition
digial||digital digial||digital
...@@ -617,6 +622,7 @@ evalute||evaluate ...@@ -617,6 +622,7 @@ evalute||evaluate
evalutes||evaluates evalutes||evaluates
evalution||evaluation evalution||evaluation
excecutable||executable excecutable||executable
excceed||exceed
exceded||exceeded exceded||exceeded
exceds||exceeds exceds||exceeds
exceeed||exceed exceeed||exceed
...@@ -632,6 +638,7 @@ existant||existent ...@@ -632,6 +638,7 @@ existant||existent
exixt||exist exixt||exist
exsits||exists exsits||exists
exlcude||exclude exlcude||exclude
exlcuding||excluding
exlcusive||exclusive exlcusive||exclusive
exlusive||exclusive exlusive||exclusive
exmaple||example exmaple||example
...@@ -726,6 +733,8 @@ generiously||generously ...@@ -726,6 +733,8 @@ generiously||generously
genereate||generate genereate||generate
genereted||generated genereted||generated
genric||generic genric||generic
gerenal||general
geting||getting
globel||global globel||global
grabing||grabbing grabing||grabbing
grahical||graphical grahical||graphical
...@@ -899,6 +908,7 @@ iteraions||iterations ...@@ -899,6 +908,7 @@ iteraions||iterations
iternations||iterations iternations||iterations
itertation||iteration itertation||iteration
itslef||itself itslef||itself
ivalid||invalid
jave||java jave||java
jeffies||jiffies jeffies||jiffies
jumpimng||jumping jumpimng||jumping
...@@ -977,6 +987,7 @@ microprocesspr||microprocessor ...@@ -977,6 +987,7 @@ microprocesspr||microprocessor
migrateable||migratable migrateable||migratable
millenium||millennium millenium||millennium
milliseonds||milliseconds milliseonds||milliseconds
minimim||minimum
minium||minimum minium||minimum
minimam||minimum minimam||minimum
minimun||minimum minimun||minimum
...@@ -1042,6 +1053,7 @@ notifed||notified ...@@ -1042,6 +1053,7 @@ notifed||notified
notity||notify notity||notify
nubmer||number nubmer||number
numebr||number numebr||number
numer||number
numner||number numner||number
nunber||number nunber||number
obtaion||obtain obtaion||obtain
...@@ -1061,6 +1073,7 @@ offet||offset ...@@ -1061,6 +1073,7 @@ offet||offset
offlaod||offload offlaod||offload
offloded||offloaded offloded||offloaded
offseting||offsetting offseting||offsetting
oflload||offload
omited||omitted omited||omitted
omiting||omitting omiting||omitting
omitt||omit omitt||omit
...@@ -1105,6 +1118,7 @@ pakage||package ...@@ -1105,6 +1118,7 @@ pakage||package
paket||packet paket||packet
pallette||palette pallette||palette
paln||plan paln||plan
palne||plane
paramameters||parameters paramameters||parameters
paramaters||parameters paramaters||parameters
paramater||parameter paramater||parameter
...@@ -1181,12 +1195,14 @@ previsously||previously ...@@ -1181,12 +1195,14 @@ previsously||previously
primative||primitive primative||primitive
princliple||principle princliple||principle
priorty||priority priorty||priority
priting||printing
privilaged||privileged privilaged||privileged
privilage||privilege privilage||privilege
priviledge||privilege priviledge||privilege
priviledges||privileges priviledges||privileges
privleges||privileges privleges||privileges
probaly||probably probaly||probably
probabalistic||probabilistic
procceed||proceed procceed||proceed
proccesors||processors proccesors||processors
procesed||processed procesed||processed
...@@ -1460,6 +1476,7 @@ submited||submitted ...@@ -1460,6 +1476,7 @@ submited||submitted
submition||submission submition||submission
succeded||succeeded succeded||succeeded
suceed||succeed suceed||succeed
succesfuly||successfully
succesfully||successfully succesfully||successfully
succesful||successful succesful||successful
successed||succeeded successed||succeeded
...@@ -1503,6 +1520,7 @@ symetric||symmetric ...@@ -1503,6 +1520,7 @@ symetric||symmetric
synax||syntax synax||syntax
synchonized||synchronized synchonized||synchronized
sychronization||synchronization sychronization||synchronization
sychronously||synchronously
synchronuously||synchronously synchronuously||synchronously
syncronize||synchronize syncronize||synchronize
syncronized||synchronized syncronized||synchronized
...@@ -1532,6 +1550,7 @@ threee||three ...@@ -1532,6 +1550,7 @@ threee||three
threshhold||threshold threshhold||threshold
thresold||threshold thresold||threshold
throught||through throught||through
tansition||transition
trackling||tracking trackling||tracking
troughput||throughput troughput||throughput
trys||tries trys||tries
...@@ -1611,6 +1630,7 @@ unneccessary||unnecessary ...@@ -1611,6 +1630,7 @@ unneccessary||unnecessary
unnecesary||unnecessary unnecesary||unnecessary
unneedingly||unnecessarily unneedingly||unnecessarily
unnsupported||unsupported unnsupported||unsupported
unuspported||unsupported
unmached||unmatched unmached||unmatched
unprecise||imprecise unprecise||imprecise
unpriviledged||unprivileged unpriviledged||unprivileged
...@@ -1657,6 +1677,7 @@ verfication||verification ...@@ -1657,6 +1677,7 @@ verfication||verification
veriosn||version veriosn||version
verisons||versions verisons||versions
verison||version verison||version
veritical||vertical
verson||version verson||version
vicefersa||vice-versa vicefersa||vice-versa
virtal||virtual virtal||virtual
...@@ -1677,6 +1698,7 @@ whenver||whenever ...@@ -1677,6 +1698,7 @@ whenver||whenever
wheter||whether wheter||whether
whe||when whe||when
wierd||weird wierd||weird
wihout||without
wiil||will wiil||will
wirte||write wirte||write
withing||within withing||within
......
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