mirror of https://gitee.com/openkylin/linux.git
Merge branch 'release' of git://git.kernel.org/pub/scm/linux/kernel/git/aegl/linux-2.6
This commit is contained in:
commit
6d23c8bc7a
|
@ -987,7 +987,7 @@ efi_initialize_iomem_resources(struct resource *code_resource,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((res = kcalloc(1, sizeof(struct resource), GFP_KERNEL)) == NULL) {
|
if ((res = kzalloc(sizeof(struct resource), GFP_KERNEL)) == NULL) {
|
||||||
printk(KERN_ERR "failed to alocate resource for iomem\n");
|
printk(KERN_ERR "failed to alocate resource for iomem\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -347,7 +347,7 @@ int __kprobes trampoline_probe_handler(struct kprobe *p, struct pt_regs *regs)
|
||||||
((struct fnptr *)kretprobe_trampoline)->ip;
|
((struct fnptr *)kretprobe_trampoline)->ip;
|
||||||
|
|
||||||
spin_lock_irqsave(&kretprobe_lock, flags);
|
spin_lock_irqsave(&kretprobe_lock, flags);
|
||||||
head = kretprobe_inst_table_head(current);
|
head = kretprobe_inst_table_head(current);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* It is possible to have multiple instances associated with a given
|
* It is possible to have multiple instances associated with a given
|
||||||
|
@ -363,9 +363,9 @@ int __kprobes trampoline_probe_handler(struct kprobe *p, struct pt_regs *regs)
|
||||||
* kretprobe_trampoline
|
* kretprobe_trampoline
|
||||||
*/
|
*/
|
||||||
hlist_for_each_entry_safe(ri, node, tmp, head, hlist) {
|
hlist_for_each_entry_safe(ri, node, tmp, head, hlist) {
|
||||||
if (ri->task != current)
|
if (ri->task != current)
|
||||||
/* another task is sharing our hash bucket */
|
/* another task is sharing our hash bucket */
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (ri->rp && ri->rp->handler)
|
if (ri->rp && ri->rp->handler)
|
||||||
ri->rp->handler(ri, regs);
|
ri->rp->handler(ri, regs);
|
||||||
|
@ -394,7 +394,7 @@ int __kprobes trampoline_probe_handler(struct kprobe *p, struct pt_regs *regs)
|
||||||
* kprobe_handler() that we don't want the post_handler
|
* kprobe_handler() that we don't want the post_handler
|
||||||
* to run (and have re-enabled preemption)
|
* to run (and have re-enabled preemption)
|
||||||
*/
|
*/
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Called with kretprobe_lock held */
|
/* Called with kretprobe_lock held */
|
||||||
|
@ -739,12 +739,16 @@ int __kprobes kprobe_exceptions_notify(struct notifier_block *self,
|
||||||
|
|
||||||
switch(val) {
|
switch(val) {
|
||||||
case DIE_BREAK:
|
case DIE_BREAK:
|
||||||
if (pre_kprobes_handler(args))
|
/* err is break number from ia64_bad_break() */
|
||||||
ret = NOTIFY_STOP;
|
if (args->err == 0x80200 || args->err == 0x80300)
|
||||||
|
if (pre_kprobes_handler(args))
|
||||||
|
ret = NOTIFY_STOP;
|
||||||
break;
|
break;
|
||||||
case DIE_SS:
|
case DIE_FAULT:
|
||||||
if (post_kprobes_handler(args->regs))
|
/* err is vector number from ia64_fault() */
|
||||||
ret = NOTIFY_STOP;
|
if (args->err == 36)
|
||||||
|
if (post_kprobes_handler(args->regs))
|
||||||
|
ret = NOTIFY_STOP;
|
||||||
break;
|
break;
|
||||||
case DIE_PAGE_FAULT:
|
case DIE_PAGE_FAULT:
|
||||||
/* kprobe_running() needs smp_processor_id() */
|
/* kprobe_running() needs smp_processor_id() */
|
||||||
|
|
|
@ -51,6 +51,9 @@
|
||||||
*
|
*
|
||||||
* 2005-08-12 Keith Owens <kaos@sgi.com>
|
* 2005-08-12 Keith Owens <kaos@sgi.com>
|
||||||
* Convert MCA/INIT handlers to use per event stacks and SAL/OS state.
|
* Convert MCA/INIT handlers to use per event stacks and SAL/OS state.
|
||||||
|
*
|
||||||
|
* 2005-10-07 Keith Owens <kaos@sgi.com>
|
||||||
|
* Add notify_die() hooks.
|
||||||
*/
|
*/
|
||||||
#include <linux/config.h>
|
#include <linux/config.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
@ -58,7 +61,6 @@
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/irq.h>
|
#include <linux/irq.h>
|
||||||
#include <linux/kallsyms.h>
|
|
||||||
#include <linux/smp_lock.h>
|
#include <linux/smp_lock.h>
|
||||||
#include <linux/bootmem.h>
|
#include <linux/bootmem.h>
|
||||||
#include <linux/acpi.h>
|
#include <linux/acpi.h>
|
||||||
|
@ -69,6 +71,7 @@
|
||||||
#include <linux/workqueue.h>
|
#include <linux/workqueue.h>
|
||||||
|
|
||||||
#include <asm/delay.h>
|
#include <asm/delay.h>
|
||||||
|
#include <asm/kdebug.h>
|
||||||
#include <asm/machvec.h>
|
#include <asm/machvec.h>
|
||||||
#include <asm/meminit.h>
|
#include <asm/meminit.h>
|
||||||
#include <asm/page.h>
|
#include <asm/page.h>
|
||||||
|
@ -132,6 +135,14 @@ extern void salinfo_log_wakeup(int type, u8 *buffer, u64 size, int irqsafe);
|
||||||
|
|
||||||
static int mca_init;
|
static int mca_init;
|
||||||
|
|
||||||
|
|
||||||
|
static void inline
|
||||||
|
ia64_mca_spin(const char *func)
|
||||||
|
{
|
||||||
|
printk(KERN_EMERG "%s: spinning here, not returning to SAL\n", func);
|
||||||
|
while (1)
|
||||||
|
cpu_relax();
|
||||||
|
}
|
||||||
/*
|
/*
|
||||||
* IA64_MCA log support
|
* IA64_MCA log support
|
||||||
*/
|
*/
|
||||||
|
@ -526,13 +537,16 @@ ia64_mca_wakeup_all(void)
|
||||||
* Outputs : None
|
* Outputs : None
|
||||||
*/
|
*/
|
||||||
static irqreturn_t
|
static irqreturn_t
|
||||||
ia64_mca_rendez_int_handler(int rendez_irq, void *arg, struct pt_regs *ptregs)
|
ia64_mca_rendez_int_handler(int rendez_irq, void *arg, struct pt_regs *regs)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int cpu = smp_processor_id();
|
int cpu = smp_processor_id();
|
||||||
|
|
||||||
/* Mask all interrupts */
|
/* Mask all interrupts */
|
||||||
local_irq_save(flags);
|
local_irq_save(flags);
|
||||||
|
if (notify_die(DIE_MCA_RENDZVOUS_ENTER, "MCA", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
|
||||||
ia64_mc_info.imi_rendez_checkin[cpu] = IA64_MCA_RENDEZ_CHECKIN_DONE;
|
ia64_mc_info.imi_rendez_checkin[cpu] = IA64_MCA_RENDEZ_CHECKIN_DONE;
|
||||||
/* Register with the SAL monarch that the slave has
|
/* Register with the SAL monarch that the slave has
|
||||||
|
@ -540,10 +554,18 @@ ia64_mca_rendez_int_handler(int rendez_irq, void *arg, struct pt_regs *ptregs)
|
||||||
*/
|
*/
|
||||||
ia64_sal_mc_rendez();
|
ia64_sal_mc_rendez();
|
||||||
|
|
||||||
|
if (notify_die(DIE_MCA_RENDZVOUS_PROCESS, "MCA", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
|
||||||
/* Wait for the monarch cpu to exit. */
|
/* Wait for the monarch cpu to exit. */
|
||||||
while (monarch_cpu != -1)
|
while (monarch_cpu != -1)
|
||||||
cpu_relax(); /* spin until monarch leaves */
|
cpu_relax(); /* spin until monarch leaves */
|
||||||
|
|
||||||
|
if (notify_die(DIE_MCA_RENDZVOUS_LEAVE, "MCA", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
|
||||||
/* Enable all interrupts */
|
/* Enable all interrupts */
|
||||||
local_irq_restore(flags);
|
local_irq_restore(flags);
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
|
@ -933,6 +955,9 @@ ia64_mca_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
oops_in_progress = 1; /* FIXME: make printk NMI/MCA/INIT safe */
|
oops_in_progress = 1; /* FIXME: make printk NMI/MCA/INIT safe */
|
||||||
previous_current = ia64_mca_modify_original_stack(regs, sw, sos, "MCA");
|
previous_current = ia64_mca_modify_original_stack(regs, sw, sos, "MCA");
|
||||||
monarch_cpu = cpu;
|
monarch_cpu = cpu;
|
||||||
|
if (notify_die(DIE_MCA_MONARCH_ENTER, "MCA", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
ia64_wait_for_slaves(cpu);
|
ia64_wait_for_slaves(cpu);
|
||||||
|
|
||||||
/* Wakeup all the processors which are spinning in the rendezvous loop.
|
/* Wakeup all the processors which are spinning in the rendezvous loop.
|
||||||
|
@ -942,6 +967,9 @@ ia64_mca_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
* spinning in SAL does not work.
|
* spinning in SAL does not work.
|
||||||
*/
|
*/
|
||||||
ia64_mca_wakeup_all();
|
ia64_mca_wakeup_all();
|
||||||
|
if (notify_die(DIE_MCA_MONARCH_PROCESS, "MCA", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
|
||||||
/* Get the MCA error record and log it */
|
/* Get the MCA error record and log it */
|
||||||
ia64_mca_log_sal_error_record(SAL_INFO_TYPE_MCA);
|
ia64_mca_log_sal_error_record(SAL_INFO_TYPE_MCA);
|
||||||
|
@ -960,6 +988,9 @@ ia64_mca_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
ia64_sal_clear_state_info(SAL_INFO_TYPE_MCA);
|
ia64_sal_clear_state_info(SAL_INFO_TYPE_MCA);
|
||||||
sos->os_status = IA64_MCA_CORRECTED;
|
sos->os_status = IA64_MCA_CORRECTED;
|
||||||
}
|
}
|
||||||
|
if (notify_die(DIE_MCA_MONARCH_LEAVE, "MCA", regs, 0, 0, recover)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
|
||||||
set_curr_task(cpu, previous_current);
|
set_curr_task(cpu, previous_current);
|
||||||
monarch_cpu = -1;
|
monarch_cpu = -1;
|
||||||
|
@ -1188,6 +1219,37 @@ ia64_mca_cpe_poll (unsigned long dummy)
|
||||||
|
|
||||||
#endif /* CONFIG_ACPI */
|
#endif /* CONFIG_ACPI */
|
||||||
|
|
||||||
|
static int
|
||||||
|
default_monarch_init_process(struct notifier_block *self, unsigned long val, void *data)
|
||||||
|
{
|
||||||
|
int c;
|
||||||
|
struct task_struct *g, *t;
|
||||||
|
if (val != DIE_INIT_MONARCH_PROCESS)
|
||||||
|
return NOTIFY_DONE;
|
||||||
|
printk(KERN_ERR "Processes interrupted by INIT -");
|
||||||
|
for_each_online_cpu(c) {
|
||||||
|
struct ia64_sal_os_state *s;
|
||||||
|
t = __va(__per_cpu_mca[c] + IA64_MCA_CPU_INIT_STACK_OFFSET);
|
||||||
|
s = (struct ia64_sal_os_state *)((char *)t + MCA_SOS_OFFSET);
|
||||||
|
g = s->prev_task;
|
||||||
|
if (g) {
|
||||||
|
if (g->pid)
|
||||||
|
printk(" %d", g->pid);
|
||||||
|
else
|
||||||
|
printk(" %d (cpu %d task 0x%p)", g->pid, task_cpu(g), g);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
printk("\n\n");
|
||||||
|
if (read_trylock(&tasklist_lock)) {
|
||||||
|
do_each_thread (g, t) {
|
||||||
|
printk("\nBacktrace of pid %d (%s)\n", t->pid, t->comm);
|
||||||
|
show_stack(t, NULL);
|
||||||
|
} while_each_thread (g, t);
|
||||||
|
read_unlock(&tasklist_lock);
|
||||||
|
}
|
||||||
|
return NOTIFY_DONE;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* C portion of the OS INIT handler
|
* C portion of the OS INIT handler
|
||||||
*
|
*
|
||||||
|
@ -1212,8 +1274,7 @@ ia64_init_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
static atomic_t slaves;
|
static atomic_t slaves;
|
||||||
static atomic_t monarchs;
|
static atomic_t monarchs;
|
||||||
task_t *previous_current;
|
task_t *previous_current;
|
||||||
int cpu = smp_processor_id(), c;
|
int cpu = smp_processor_id();
|
||||||
struct task_struct *g, *t;
|
|
||||||
|
|
||||||
oops_in_progress = 1; /* FIXME: make printk NMI/MCA/INIT safe */
|
oops_in_progress = 1; /* FIXME: make printk NMI/MCA/INIT safe */
|
||||||
console_loglevel = 15; /* make sure printks make it to console */
|
console_loglevel = 15; /* make sure printks make it to console */
|
||||||
|
@ -1253,8 +1314,17 @@ ia64_init_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
ia64_mc_info.imi_rendez_checkin[cpu] = IA64_MCA_RENDEZ_CHECKIN_INIT;
|
ia64_mc_info.imi_rendez_checkin[cpu] = IA64_MCA_RENDEZ_CHECKIN_INIT;
|
||||||
while (monarch_cpu == -1)
|
while (monarch_cpu == -1)
|
||||||
cpu_relax(); /* spin until monarch enters */
|
cpu_relax(); /* spin until monarch enters */
|
||||||
|
if (notify_die(DIE_INIT_SLAVE_ENTER, "INIT", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
if (notify_die(DIE_INIT_SLAVE_PROCESS, "INIT", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
while (monarch_cpu != -1)
|
while (monarch_cpu != -1)
|
||||||
cpu_relax(); /* spin until monarch leaves */
|
cpu_relax(); /* spin until monarch leaves */
|
||||||
|
if (notify_die(DIE_INIT_SLAVE_LEAVE, "INIT", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
printk("Slave on cpu %d returning to normal service.\n", cpu);
|
printk("Slave on cpu %d returning to normal service.\n", cpu);
|
||||||
set_curr_task(cpu, previous_current);
|
set_curr_task(cpu, previous_current);
|
||||||
ia64_mc_info.imi_rendez_checkin[cpu] = IA64_MCA_RENDEZ_CHECKIN_NOTDONE;
|
ia64_mc_info.imi_rendez_checkin[cpu] = IA64_MCA_RENDEZ_CHECKIN_NOTDONE;
|
||||||
|
@ -1263,6 +1333,9 @@ ia64_init_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
}
|
}
|
||||||
|
|
||||||
monarch_cpu = cpu;
|
monarch_cpu = cpu;
|
||||||
|
if (notify_die(DIE_INIT_MONARCH_ENTER, "INIT", regs, 0, 0, 0)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
ia64_mca_spin(__FUNCTION__);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Wait for a bit. On some machines (e.g., HP's zx2000 and zx6000, INIT can be
|
* Wait for a bit. On some machines (e.g., HP's zx2000 and zx6000, INIT can be
|
||||||
|
@ -1273,27 +1346,16 @@ ia64_init_handler(struct pt_regs *regs, struct switch_stack *sw,
|
||||||
printk("Delaying for 5 seconds...\n");
|
printk("Delaying for 5 seconds...\n");
|
||||||
udelay(5*1000000);
|
udelay(5*1000000);
|
||||||
ia64_wait_for_slaves(cpu);
|
ia64_wait_for_slaves(cpu);
|
||||||
printk(KERN_ERR "Processes interrupted by INIT -");
|
/* If nobody intercepts DIE_INIT_MONARCH_PROCESS then we drop through
|
||||||
for_each_online_cpu(c) {
|
* to default_monarch_init_process() above and just print all the
|
||||||
struct ia64_sal_os_state *s;
|
* tasks.
|
||||||
t = __va(__per_cpu_mca[c] + IA64_MCA_CPU_INIT_STACK_OFFSET);
|
*/
|
||||||
s = (struct ia64_sal_os_state *)((char *)t + MCA_SOS_OFFSET);
|
if (notify_die(DIE_INIT_MONARCH_PROCESS, "INIT", regs, 0, 0, 0)
|
||||||
g = s->prev_task;
|
== NOTIFY_STOP)
|
||||||
if (g) {
|
ia64_mca_spin(__FUNCTION__);
|
||||||
if (g->pid)
|
if (notify_die(DIE_INIT_MONARCH_LEAVE, "INIT", regs, 0, 0, 0)
|
||||||
printk(" %d", g->pid);
|
== NOTIFY_STOP)
|
||||||
else
|
ia64_mca_spin(__FUNCTION__);
|
||||||
printk(" %d (cpu %d task 0x%p)", g->pid, task_cpu(g), g);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
printk("\n\n");
|
|
||||||
if (read_trylock(&tasklist_lock)) {
|
|
||||||
do_each_thread (g, t) {
|
|
||||||
printk("\nBacktrace of pid %d (%s)\n", t->pid, t->comm);
|
|
||||||
show_stack(t, NULL);
|
|
||||||
} while_each_thread (g, t);
|
|
||||||
read_unlock(&tasklist_lock);
|
|
||||||
}
|
|
||||||
printk("\nINIT dump complete. Monarch on cpu %d returning to normal service.\n", cpu);
|
printk("\nINIT dump complete. Monarch on cpu %d returning to normal service.\n", cpu);
|
||||||
atomic_dec(&monarchs);
|
atomic_dec(&monarchs);
|
||||||
set_curr_task(cpu, previous_current);
|
set_curr_task(cpu, previous_current);
|
||||||
|
@ -1462,6 +1524,10 @@ ia64_mca_init(void)
|
||||||
s64 rc;
|
s64 rc;
|
||||||
struct ia64_sal_retval isrv;
|
struct ia64_sal_retval isrv;
|
||||||
u64 timeout = IA64_MCA_RENDEZ_TIMEOUT; /* platform specific */
|
u64 timeout = IA64_MCA_RENDEZ_TIMEOUT; /* platform specific */
|
||||||
|
static struct notifier_block default_init_monarch_nb = {
|
||||||
|
.notifier_call = default_monarch_init_process,
|
||||||
|
.priority = 0/* we need to notified last */
|
||||||
|
};
|
||||||
|
|
||||||
IA64_MCA_DEBUG("%s: begin\n", __FUNCTION__);
|
IA64_MCA_DEBUG("%s: begin\n", __FUNCTION__);
|
||||||
|
|
||||||
|
@ -1555,6 +1621,10 @@ ia64_mca_init(void)
|
||||||
"(status %ld)\n", rc);
|
"(status %ld)\n", rc);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (register_die_notifier(&default_init_monarch_nb)) {
|
||||||
|
printk(KERN_ERR "Failed to register default monarch INIT process\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
IA64_MCA_DEBUG("%s: registered OS INIT handler with SAL\n", __FUNCTION__);
|
IA64_MCA_DEBUG("%s: registered OS INIT handler with SAL\n", __FUNCTION__);
|
||||||
|
|
||||||
|
|
|
@ -547,9 +547,20 @@ recover_from_processor_error(int platform, slidx_table_t *slidx,
|
||||||
(pal_processor_state_info_t*)peidx_psp(peidx);
|
(pal_processor_state_info_t*)peidx_psp(peidx);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We cannot recover errors with other than bus_check.
|
* Processor recovery status must key off of the PAL recovery
|
||||||
|
* status in the Processor State Parameter.
|
||||||
*/
|
*/
|
||||||
if (psp->cc || psp->rc || psp->uc)
|
|
||||||
|
/*
|
||||||
|
* The machine check is corrected.
|
||||||
|
*/
|
||||||
|
if (psp->cm == 1)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The error was not contained. Software must be reset.
|
||||||
|
*/
|
||||||
|
if (psp->us || psp->ci == 0)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -570,8 +581,6 @@ recover_from_processor_error(int platform, slidx_table_t *slidx,
|
||||||
return 0;
|
return 0;
|
||||||
if (pbci->eb && pbci->bsi > 0)
|
if (pbci->eb && pbci->bsi > 0)
|
||||||
return 0;
|
return 0;
|
||||||
if (psp->ci == 0)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This is a local MCA and estimated as recoverble external bus error.
|
* This is a local MCA and estimated as recoverble external bus error.
|
||||||
|
|
|
@ -4,6 +4,9 @@
|
||||||
* Copyright (C) 1998-2003 Hewlett-Packard Co
|
* Copyright (C) 1998-2003 Hewlett-Packard Co
|
||||||
* David Mosberger-Tang <davidm@hpl.hp.com>
|
* David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
* 04/11/17 Ashok Raj <ashok.raj@intel.com> Added CPU Hotplug Support
|
* 04/11/17 Ashok Raj <ashok.raj@intel.com> Added CPU Hotplug Support
|
||||||
|
*
|
||||||
|
* 2005-10-07 Keith Owens <kaos@sgi.com>
|
||||||
|
* Add notify_die() hooks.
|
||||||
*/
|
*/
|
||||||
#define __KERNEL_SYSCALLS__ /* see <asm/unistd.h> */
|
#define __KERNEL_SYSCALLS__ /* see <asm/unistd.h> */
|
||||||
#include <linux/config.h>
|
#include <linux/config.h>
|
||||||
|
@ -34,6 +37,7 @@
|
||||||
#include <asm/elf.h>
|
#include <asm/elf.h>
|
||||||
#include <asm/ia32.h>
|
#include <asm/ia32.h>
|
||||||
#include <asm/irq.h>
|
#include <asm/irq.h>
|
||||||
|
#include <asm/kdebug.h>
|
||||||
#include <asm/pgalloc.h>
|
#include <asm/pgalloc.h>
|
||||||
#include <asm/processor.h>
|
#include <asm/processor.h>
|
||||||
#include <asm/sal.h>
|
#include <asm/sal.h>
|
||||||
|
@ -808,12 +812,14 @@ cpu_halt (void)
|
||||||
void
|
void
|
||||||
machine_restart (char *restart_cmd)
|
machine_restart (char *restart_cmd)
|
||||||
{
|
{
|
||||||
|
(void) notify_die(DIE_MACHINE_RESTART, restart_cmd, NULL, 0, 0, 0);
|
||||||
(*efi.reset_system)(EFI_RESET_WARM, 0, 0, NULL);
|
(*efi.reset_system)(EFI_RESET_WARM, 0, 0, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
machine_halt (void)
|
machine_halt (void)
|
||||||
{
|
{
|
||||||
|
(void) notify_die(DIE_MACHINE_HALT, "", NULL, 0, 0, 0);
|
||||||
cpu_halt();
|
cpu_halt();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -461,6 +461,7 @@ setup_arch (char **cmdline_p)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
cpu_init(); /* initialize the bootstrap CPU */
|
cpu_init(); /* initialize the bootstrap CPU */
|
||||||
|
mmu_context_init(); /* initialize context_id bitmap */
|
||||||
|
|
||||||
#ifdef CONFIG_ACPI
|
#ifdef CONFIG_ACPI
|
||||||
acpi_boot_init();
|
acpi_boot_init();
|
||||||
|
|
|
@ -387,15 +387,14 @@ setup_frame (int sig, struct k_sigaction *ka, siginfo_t *info, sigset_t *set,
|
||||||
struct sigscratch *scr)
|
struct sigscratch *scr)
|
||||||
{
|
{
|
||||||
extern char __kernel_sigtramp[];
|
extern char __kernel_sigtramp[];
|
||||||
unsigned long tramp_addr, new_rbs = 0;
|
unsigned long tramp_addr, new_rbs = 0, new_sp;
|
||||||
struct sigframe __user *frame;
|
struct sigframe __user *frame;
|
||||||
long err;
|
long err;
|
||||||
|
|
||||||
frame = (void __user *) scr->pt.r12;
|
new_sp = scr->pt.r12;
|
||||||
tramp_addr = (unsigned long) __kernel_sigtramp;
|
tramp_addr = (unsigned long) __kernel_sigtramp;
|
||||||
if ((ka->sa.sa_flags & SA_ONSTACK) && sas_ss_flags((unsigned long) frame) == 0) {
|
if ((ka->sa.sa_flags & SA_ONSTACK) && sas_ss_flags(new_sp) == 0) {
|
||||||
frame = (void __user *) ((current->sas_ss_sp + current->sas_ss_size)
|
new_sp = current->sas_ss_sp + current->sas_ss_size;
|
||||||
& ~(STACK_ALIGN - 1));
|
|
||||||
/*
|
/*
|
||||||
* We need to check for the register stack being on the signal stack
|
* We need to check for the register stack being on the signal stack
|
||||||
* separately, because it's switched separately (memory stack is switched
|
* separately, because it's switched separately (memory stack is switched
|
||||||
|
@ -404,7 +403,7 @@ setup_frame (int sig, struct k_sigaction *ka, siginfo_t *info, sigset_t *set,
|
||||||
if (!rbs_on_sig_stack(scr->pt.ar_bspstore))
|
if (!rbs_on_sig_stack(scr->pt.ar_bspstore))
|
||||||
new_rbs = (current->sas_ss_sp + sizeof(long) - 1) & ~(sizeof(long) - 1);
|
new_rbs = (current->sas_ss_sp + sizeof(long) - 1) & ~(sizeof(long) - 1);
|
||||||
}
|
}
|
||||||
frame = (void __user *) frame - ((sizeof(*frame) + STACK_ALIGN - 1) & ~(STACK_ALIGN - 1));
|
frame = (void __user *) ((new_sp - sizeof(*frame)) & -STACK_ALIGN);
|
||||||
|
|
||||||
if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
|
if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
|
||||||
return force_sigsegv_info(sig, frame);
|
return force_sigsegv_info(sig, frame);
|
||||||
|
|
|
@ -30,17 +30,20 @@ fpswa_interface_t *fpswa_interface;
|
||||||
EXPORT_SYMBOL(fpswa_interface);
|
EXPORT_SYMBOL(fpswa_interface);
|
||||||
|
|
||||||
struct notifier_block *ia64die_chain;
|
struct notifier_block *ia64die_chain;
|
||||||
static DEFINE_SPINLOCK(die_notifier_lock);
|
|
||||||
|
|
||||||
int register_die_notifier(struct notifier_block *nb)
|
int
|
||||||
|
register_die_notifier(struct notifier_block *nb)
|
||||||
{
|
{
|
||||||
int err = 0;
|
return notifier_chain_register(&ia64die_chain, nb);
|
||||||
unsigned long flags;
|
|
||||||
spin_lock_irqsave(&die_notifier_lock, flags);
|
|
||||||
err = notifier_chain_register(&ia64die_chain, nb);
|
|
||||||
spin_unlock_irqrestore(&die_notifier_lock, flags);
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(register_die_notifier);
|
||||||
|
|
||||||
|
int
|
||||||
|
unregister_die_notifier(struct notifier_block *nb)
|
||||||
|
{
|
||||||
|
return notifier_chain_unregister(&ia64die_chain, nb);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(unregister_die_notifier);
|
||||||
|
|
||||||
void __init
|
void __init
|
||||||
trap_init (void)
|
trap_init (void)
|
||||||
|
@ -105,6 +108,7 @@ die (const char *str, struct pt_regs *regs, long err)
|
||||||
if (++die.lock_owner_depth < 3) {
|
if (++die.lock_owner_depth < 3) {
|
||||||
printk("%s[%d]: %s %ld [%d]\n",
|
printk("%s[%d]: %s %ld [%d]\n",
|
||||||
current->comm, current->pid, str, err, ++die_counter);
|
current->comm, current->pid, str, err, ++die_counter);
|
||||||
|
(void) notify_die(DIE_OOPS, (char *)str, regs, err, 255, SIGSEGV);
|
||||||
show_regs(regs);
|
show_regs(regs);
|
||||||
} else
|
} else
|
||||||
printk(KERN_ERR "Recursive die() failure, output suppressed\n");
|
printk(KERN_ERR "Recursive die() failure, output suppressed\n");
|
||||||
|
@ -155,9 +159,8 @@ __kprobes ia64_bad_break (unsigned long break_num, struct pt_regs *regs)
|
||||||
switch (break_num) {
|
switch (break_num) {
|
||||||
case 0: /* unknown error (used by GCC for __builtin_abort()) */
|
case 0: /* unknown error (used by GCC for __builtin_abort()) */
|
||||||
if (notify_die(DIE_BREAK, "break 0", regs, break_num, TRAP_BRKPT, SIGTRAP)
|
if (notify_die(DIE_BREAK, "break 0", regs, break_num, TRAP_BRKPT, SIGTRAP)
|
||||||
== NOTIFY_STOP) {
|
== NOTIFY_STOP)
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
die_if_kernel("bugcheck!", regs, break_num);
|
die_if_kernel("bugcheck!", regs, break_num);
|
||||||
sig = SIGILL; code = ILL_ILLOPC;
|
sig = SIGILL; code = ILL_ILLOPC;
|
||||||
break;
|
break;
|
||||||
|
@ -210,15 +213,6 @@ __kprobes ia64_bad_break (unsigned long break_num, struct pt_regs *regs)
|
||||||
sig = SIGILL; code = __ILL_BNDMOD;
|
sig = SIGILL; code = __ILL_BNDMOD;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x80200:
|
|
||||||
case 0x80300:
|
|
||||||
if (notify_die(DIE_BREAK, "kprobe", regs, break_num, TRAP_BRKPT, SIGTRAP)
|
|
||||||
== NOTIFY_STOP) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
sig = SIGTRAP; code = TRAP_BRKPT;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
if (break_num < 0x40000 || break_num > 0x100000)
|
if (break_num < 0x40000 || break_num > 0x100000)
|
||||||
die_if_kernel("Bad break", regs, break_num);
|
die_if_kernel("Bad break", regs, break_num);
|
||||||
|
@ -226,6 +220,9 @@ __kprobes ia64_bad_break (unsigned long break_num, struct pt_regs *regs)
|
||||||
if (break_num < 0x80000) {
|
if (break_num < 0x80000) {
|
||||||
sig = SIGILL; code = __ILL_BREAK;
|
sig = SIGILL; code = __ILL_BREAK;
|
||||||
} else {
|
} else {
|
||||||
|
if (notify_die(DIE_BREAK, "bad break", regs, break_num, TRAP_BRKPT, SIGTRAP)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
return;
|
||||||
sig = SIGTRAP; code = TRAP_BRKPT;
|
sig = SIGTRAP; code = TRAP_BRKPT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -578,12 +575,11 @@ ia64_fault (unsigned long vector, unsigned long isr, unsigned long ifa,
|
||||||
#endif
|
#endif
|
||||||
break;
|
break;
|
||||||
case 35: siginfo.si_code = TRAP_BRANCH; ifa = 0; break;
|
case 35: siginfo.si_code = TRAP_BRANCH; ifa = 0; break;
|
||||||
case 36:
|
case 36: siginfo.si_code = TRAP_TRACE; ifa = 0; break;
|
||||||
if (notify_die(DIE_SS, "ss", ®s, vector,
|
|
||||||
vector, SIGTRAP) == NOTIFY_STOP)
|
|
||||||
return;
|
|
||||||
siginfo.si_code = TRAP_TRACE; ifa = 0; break;
|
|
||||||
}
|
}
|
||||||
|
if (notify_die(DIE_FAULT, "ia64_fault", ®s, vector, siginfo.si_code, SIGTRAP)
|
||||||
|
== NOTIFY_STOP)
|
||||||
|
return;
|
||||||
siginfo.si_signo = SIGTRAP;
|
siginfo.si_signo = SIGTRAP;
|
||||||
siginfo.si_errno = 0;
|
siginfo.si_errno = 0;
|
||||||
siginfo.si_addr = (void __user *) ifa;
|
siginfo.si_addr = (void __user *) ifa;
|
||||||
|
|
|
@ -350,14 +350,12 @@ static void __init initialize_pernode_data(void)
|
||||||
* for best.
|
* for best.
|
||||||
* @nid: node id
|
* @nid: node id
|
||||||
* @pernodesize: size of this node's pernode data
|
* @pernodesize: size of this node's pernode data
|
||||||
* @align: alignment to use for this node's pernode data
|
|
||||||
*/
|
*/
|
||||||
static void __init *memory_less_node_alloc(int nid, unsigned long pernodesize,
|
static void __init *memory_less_node_alloc(int nid, unsigned long pernodesize)
|
||||||
unsigned long align)
|
|
||||||
{
|
{
|
||||||
void *ptr = NULL;
|
void *ptr = NULL;
|
||||||
u8 best = 0xff;
|
u8 best = 0xff;
|
||||||
int bestnode = -1, node;
|
int bestnode = -1, node, anynode = 0;
|
||||||
|
|
||||||
for_each_online_node(node) {
|
for_each_online_node(node) {
|
||||||
if (node_isset(node, memory_less_mask))
|
if (node_isset(node, memory_less_mask))
|
||||||
|
@ -366,13 +364,15 @@ static void __init *memory_less_node_alloc(int nid, unsigned long pernodesize,
|
||||||
best = node_distance(nid, node);
|
best = node_distance(nid, node);
|
||||||
bestnode = node;
|
bestnode = node;
|
||||||
}
|
}
|
||||||
|
anynode = node;
|
||||||
}
|
}
|
||||||
|
|
||||||
ptr = __alloc_bootmem_node(mem_data[bestnode].pgdat,
|
if (bestnode == -1)
|
||||||
pernodesize, align, __pa(MAX_DMA_ADDRESS));
|
bestnode = anynode;
|
||||||
|
|
||||||
|
ptr = __alloc_bootmem_node(mem_data[bestnode].pgdat, pernodesize,
|
||||||
|
PERCPU_PAGE_SIZE, __pa(MAX_DMA_ADDRESS));
|
||||||
|
|
||||||
if (!ptr)
|
|
||||||
panic("NO memory for memory less node\n");
|
|
||||||
return ptr;
|
return ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -413,8 +413,7 @@ static void __init memory_less_nodes(void)
|
||||||
|
|
||||||
for_each_node_mask(node, memory_less_mask) {
|
for_each_node_mask(node, memory_less_mask) {
|
||||||
pernodesize = compute_pernodesize(node);
|
pernodesize = compute_pernodesize(node);
|
||||||
pernode = memory_less_node_alloc(node, pernodesize,
|
pernode = memory_less_node_alloc(node, pernodesize);
|
||||||
(node) ? (node * PERCPU_PAGE_SIZE) : (1024*1024));
|
|
||||||
fill_pernode(node, __pa(pernode), pernodesize);
|
fill_pernode(node, __pa(pernode), pernodesize);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,8 @@
|
||||||
* Modified RID allocation for SMP
|
* Modified RID allocation for SMP
|
||||||
* Goutham Rao <goutham.rao@intel.com>
|
* Goutham Rao <goutham.rao@intel.com>
|
||||||
* IPI based ptc implementation and A-step IPI implementation.
|
* IPI based ptc implementation and A-step IPI implementation.
|
||||||
|
* Rohit Seth <rohit.seth@intel.com>
|
||||||
|
* Ken Chen <kenneth.w.chen@intel.com>
|
||||||
*/
|
*/
|
||||||
#include <linux/config.h>
|
#include <linux/config.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
|
@ -16,78 +18,75 @@
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/smp.h>
|
#include <linux/smp.h>
|
||||||
#include <linux/mm.h>
|
#include <linux/mm.h>
|
||||||
|
#include <linux/bootmem.h>
|
||||||
|
|
||||||
#include <asm/delay.h>
|
#include <asm/delay.h>
|
||||||
#include <asm/mmu_context.h>
|
#include <asm/mmu_context.h>
|
||||||
#include <asm/pgalloc.h>
|
#include <asm/pgalloc.h>
|
||||||
#include <asm/pal.h>
|
#include <asm/pal.h>
|
||||||
#include <asm/tlbflush.h>
|
#include <asm/tlbflush.h>
|
||||||
|
#include <asm/dma.h>
|
||||||
|
|
||||||
static struct {
|
static struct {
|
||||||
unsigned long mask; /* mask of supported purge page-sizes */
|
unsigned long mask; /* mask of supported purge page-sizes */
|
||||||
unsigned long max_bits; /* log2() of largest supported purge page-size */
|
unsigned long max_bits; /* log2 of largest supported purge page-size */
|
||||||
} purge;
|
} purge;
|
||||||
|
|
||||||
struct ia64_ctx ia64_ctx = {
|
struct ia64_ctx ia64_ctx = {
|
||||||
.lock = SPIN_LOCK_UNLOCKED,
|
.lock = SPIN_LOCK_UNLOCKED,
|
||||||
.next = 1,
|
.next = 1,
|
||||||
.limit = (1 << 15) - 1, /* start out with the safe (architected) limit */
|
|
||||||
.max_ctx = ~0U
|
.max_ctx = ~0U
|
||||||
};
|
};
|
||||||
|
|
||||||
DEFINE_PER_CPU(u8, ia64_need_tlb_flush);
|
DEFINE_PER_CPU(u8, ia64_need_tlb_flush);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Initializes the ia64_ctx.bitmap array based on max_ctx+1.
|
||||||
|
* Called after cpu_init() has setup ia64_ctx.max_ctx based on
|
||||||
|
* maximum RID that is supported by boot CPU.
|
||||||
|
*/
|
||||||
|
void __init
|
||||||
|
mmu_context_init (void)
|
||||||
|
{
|
||||||
|
ia64_ctx.bitmap = alloc_bootmem((ia64_ctx.max_ctx+1)>>3);
|
||||||
|
ia64_ctx.flushmap = alloc_bootmem((ia64_ctx.max_ctx+1)>>3);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Acquire the ia64_ctx.lock before calling this function!
|
* Acquire the ia64_ctx.lock before calling this function!
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
wrap_mmu_context (struct mm_struct *mm)
|
wrap_mmu_context (struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
unsigned long tsk_context, max_ctx = ia64_ctx.max_ctx;
|
int i, cpu;
|
||||||
struct task_struct *tsk;
|
unsigned long flush_bit;
|
||||||
int i;
|
|
||||||
|
|
||||||
if (ia64_ctx.next > max_ctx)
|
for (i=0; i <= ia64_ctx.max_ctx / BITS_PER_LONG; i++) {
|
||||||
ia64_ctx.next = 300; /* skip daemons */
|
flush_bit = xchg(&ia64_ctx.flushmap[i], 0);
|
||||||
ia64_ctx.limit = max_ctx + 1;
|
ia64_ctx.bitmap[i] ^= flush_bit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* use offset at 300 to skip daemons */
|
||||||
|
ia64_ctx.next = find_next_zero_bit(ia64_ctx.bitmap,
|
||||||
|
ia64_ctx.max_ctx, 300);
|
||||||
|
ia64_ctx.limit = find_next_bit(ia64_ctx.bitmap,
|
||||||
|
ia64_ctx.max_ctx, ia64_ctx.next);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Scan all the task's mm->context and set proper safe range
|
* can't call flush_tlb_all() here because of race condition
|
||||||
|
* with O(1) scheduler [EF]
|
||||||
*/
|
*/
|
||||||
|
cpu = get_cpu(); /* prevent preemption/migration */
|
||||||
read_lock(&tasklist_lock);
|
for_each_online_cpu(i)
|
||||||
repeat:
|
if (i != cpu)
|
||||||
for_each_process(tsk) {
|
per_cpu(ia64_need_tlb_flush, i) = 1;
|
||||||
if (!tsk->mm)
|
put_cpu();
|
||||||
continue;
|
|
||||||
tsk_context = tsk->mm->context;
|
|
||||||
if (tsk_context == ia64_ctx.next) {
|
|
||||||
if (++ia64_ctx.next >= ia64_ctx.limit) {
|
|
||||||
/* empty range: reset the range limit and start over */
|
|
||||||
if (ia64_ctx.next > max_ctx)
|
|
||||||
ia64_ctx.next = 300;
|
|
||||||
ia64_ctx.limit = max_ctx + 1;
|
|
||||||
goto repeat;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ((tsk_context > ia64_ctx.next) && (tsk_context < ia64_ctx.limit))
|
|
||||||
ia64_ctx.limit = tsk_context;
|
|
||||||
}
|
|
||||||
read_unlock(&tasklist_lock);
|
|
||||||
/* can't call flush_tlb_all() here because of race condition with O(1) scheduler [EF] */
|
|
||||||
{
|
|
||||||
int cpu = get_cpu(); /* prevent preemption/migration */
|
|
||||||
for_each_online_cpu(i) {
|
|
||||||
if (i != cpu)
|
|
||||||
per_cpu(ia64_need_tlb_flush, i) = 1;
|
|
||||||
}
|
|
||||||
put_cpu();
|
|
||||||
}
|
|
||||||
local_flush_tlb_all();
|
local_flush_tlb_all();
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
ia64_global_tlb_purge (struct mm_struct *mm, unsigned long start, unsigned long end, unsigned long nbits)
|
ia64_global_tlb_purge (struct mm_struct *mm, unsigned long start,
|
||||||
|
unsigned long end, unsigned long nbits)
|
||||||
{
|
{
|
||||||
static DEFINE_SPINLOCK(ptcg_lock);
|
static DEFINE_SPINLOCK(ptcg_lock);
|
||||||
|
|
||||||
|
@ -135,7 +134,8 @@ local_flush_tlb_all (void)
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
flush_tlb_range (struct vm_area_struct *vma, unsigned long start, unsigned long end)
|
flush_tlb_range (struct vm_area_struct *vma, unsigned long start,
|
||||||
|
unsigned long end)
|
||||||
{
|
{
|
||||||
struct mm_struct *mm = vma->vm_mm;
|
struct mm_struct *mm = vma->vm_mm;
|
||||||
unsigned long size = end - start;
|
unsigned long size = end - start;
|
||||||
|
@ -149,7 +149,8 @@ flush_tlb_range (struct vm_area_struct *vma, unsigned long start, unsigned long
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
nbits = ia64_fls(size + 0xfff);
|
nbits = ia64_fls(size + 0xfff);
|
||||||
while (unlikely (((1UL << nbits) & purge.mask) == 0) && (nbits < purge.max_bits))
|
while (unlikely (((1UL << nbits) & purge.mask) == 0) &&
|
||||||
|
(nbits < purge.max_bits))
|
||||||
++nbits;
|
++nbits;
|
||||||
if (nbits > purge.max_bits)
|
if (nbits > purge.max_bits)
|
||||||
nbits = purge.max_bits;
|
nbits = purge.max_bits;
|
||||||
|
@ -191,5 +192,5 @@ ia64_tlb_init (void)
|
||||||
local_cpu_data->ptce_stride[0] = ptce_info.stride[0];
|
local_cpu_data->ptce_stride[0] = ptce_info.stride[0];
|
||||||
local_cpu_data->ptce_stride[1] = ptce_info.stride[1];
|
local_cpu_data->ptce_stride[1] = ptce_info.stride[1];
|
||||||
|
|
||||||
local_flush_tlb_all(); /* nuke left overs from bootstrapping... */
|
local_flush_tlb_all(); /* nuke left overs from bootstrapping... */
|
||||||
}
|
}
|
||||||
|
|
|
@ -95,7 +95,7 @@ pci_sal_write (unsigned int seg, unsigned int bus, unsigned int devfn,
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct pci_raw_ops pci_sal_ops = {
|
static struct pci_raw_ops pci_sal_ops = {
|
||||||
.read = pci_sal_read,
|
.read = pci_sal_read,
|
||||||
.write = pci_sal_write
|
.write = pci_sal_write
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -137,35 +137,98 @@ alloc_pci_controller (int seg)
|
||||||
return controller;
|
return controller;
|
||||||
}
|
}
|
||||||
|
|
||||||
static u64 __devinit
|
struct pci_root_info {
|
||||||
add_io_space (struct acpi_resource_address64 *addr)
|
struct pci_controller *controller;
|
||||||
|
char *name;
|
||||||
|
};
|
||||||
|
|
||||||
|
static unsigned int
|
||||||
|
new_space (u64 phys_base, int sparse)
|
||||||
{
|
{
|
||||||
u64 offset;
|
u64 mmio_base;
|
||||||
int sparse = 0;
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (addr->address_translation_offset == 0)
|
if (phys_base == 0)
|
||||||
return IO_SPACE_BASE(0); /* part of legacy IO space */
|
return 0; /* legacy I/O port space */
|
||||||
|
|
||||||
if (addr->attribute.io.translation_attribute == ACPI_SPARSE_TRANSLATION)
|
mmio_base = (u64) ioremap(phys_base, 0);
|
||||||
sparse = 1;
|
|
||||||
|
|
||||||
offset = (u64) ioremap(addr->address_translation_offset, 0);
|
|
||||||
for (i = 0; i < num_io_spaces; i++)
|
for (i = 0; i < num_io_spaces; i++)
|
||||||
if (io_space[i].mmio_base == offset &&
|
if (io_space[i].mmio_base == mmio_base &&
|
||||||
io_space[i].sparse == sparse)
|
io_space[i].sparse == sparse)
|
||||||
return IO_SPACE_BASE(i);
|
return i;
|
||||||
|
|
||||||
if (num_io_spaces == MAX_IO_SPACES) {
|
if (num_io_spaces == MAX_IO_SPACES) {
|
||||||
printk("Too many IO port spaces\n");
|
printk(KERN_ERR "PCI: Too many IO port spaces "
|
||||||
|
"(MAX_IO_SPACES=%lu)\n", MAX_IO_SPACES);
|
||||||
return ~0;
|
return ~0;
|
||||||
}
|
}
|
||||||
|
|
||||||
i = num_io_spaces++;
|
i = num_io_spaces++;
|
||||||
io_space[i].mmio_base = offset;
|
io_space[i].mmio_base = mmio_base;
|
||||||
io_space[i].sparse = sparse;
|
io_space[i].sparse = sparse;
|
||||||
|
|
||||||
return IO_SPACE_BASE(i);
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u64 __devinit
|
||||||
|
add_io_space (struct pci_root_info *info, struct acpi_resource_address64 *addr)
|
||||||
|
{
|
||||||
|
struct resource *resource;
|
||||||
|
char *name;
|
||||||
|
u64 base, min, max, base_port;
|
||||||
|
unsigned int sparse = 0, space_nr, len;
|
||||||
|
|
||||||
|
resource = kzalloc(sizeof(*resource), GFP_KERNEL);
|
||||||
|
if (!resource) {
|
||||||
|
printk(KERN_ERR "PCI: No memory for %s I/O port space\n",
|
||||||
|
info->name);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
len = strlen(info->name) + 32;
|
||||||
|
name = kzalloc(len, GFP_KERNEL);
|
||||||
|
if (!name) {
|
||||||
|
printk(KERN_ERR "PCI: No memory for %s I/O port space name\n",
|
||||||
|
info->name);
|
||||||
|
goto free_resource;
|
||||||
|
}
|
||||||
|
|
||||||
|
min = addr->min_address_range;
|
||||||
|
max = min + addr->address_length - 1;
|
||||||
|
if (addr->attribute.io.translation_attribute == ACPI_SPARSE_TRANSLATION)
|
||||||
|
sparse = 1;
|
||||||
|
|
||||||
|
space_nr = new_space(addr->address_translation_offset, sparse);
|
||||||
|
if (space_nr == ~0)
|
||||||
|
goto free_name;
|
||||||
|
|
||||||
|
base = __pa(io_space[space_nr].mmio_base);
|
||||||
|
base_port = IO_SPACE_BASE(space_nr);
|
||||||
|
snprintf(name, len, "%s I/O Ports %08lx-%08lx", info->name,
|
||||||
|
base_port + min, base_port + max);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The SDM guarantees the legacy 0-64K space is sparse, but if the
|
||||||
|
* mapping is done by the processor (not the bridge), ACPI may not
|
||||||
|
* mark it as sparse.
|
||||||
|
*/
|
||||||
|
if (space_nr == 0)
|
||||||
|
sparse = 1;
|
||||||
|
|
||||||
|
resource->name = name;
|
||||||
|
resource->flags = IORESOURCE_MEM;
|
||||||
|
resource->start = base + (sparse ? IO_SPACE_SPARSE_ENCODING(min) : min);
|
||||||
|
resource->end = base + (sparse ? IO_SPACE_SPARSE_ENCODING(max) : max);
|
||||||
|
insert_resource(&iomem_resource, resource);
|
||||||
|
|
||||||
|
return base_port;
|
||||||
|
|
||||||
|
free_name:
|
||||||
|
kfree(name);
|
||||||
|
free_resource:
|
||||||
|
kfree(resource);
|
||||||
|
out:
|
||||||
|
return ~0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static acpi_status __devinit resource_to_window(struct acpi_resource *resource,
|
static acpi_status __devinit resource_to_window(struct acpi_resource *resource,
|
||||||
|
@ -205,11 +268,6 @@ count_window (struct acpi_resource *resource, void *data)
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct pci_root_info {
|
|
||||||
struct pci_controller *controller;
|
|
||||||
char *name;
|
|
||||||
};
|
|
||||||
|
|
||||||
static __devinit acpi_status add_window(struct acpi_resource *res, void *data)
|
static __devinit acpi_status add_window(struct acpi_resource *res, void *data)
|
||||||
{
|
{
|
||||||
struct pci_root_info *info = data;
|
struct pci_root_info *info = data;
|
||||||
|
@ -231,7 +289,7 @@ static __devinit acpi_status add_window(struct acpi_resource *res, void *data)
|
||||||
} else if (addr.resource_type == ACPI_IO_RANGE) {
|
} else if (addr.resource_type == ACPI_IO_RANGE) {
|
||||||
flags = IORESOURCE_IO;
|
flags = IORESOURCE_IO;
|
||||||
root = &ioport_resource;
|
root = &ioport_resource;
|
||||||
offset = add_io_space(&addr);
|
offset = add_io_space(info, &addr);
|
||||||
if (offset == ~0)
|
if (offset == ~0)
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
} else
|
} else
|
||||||
|
@ -241,7 +299,7 @@ static __devinit acpi_status add_window(struct acpi_resource *res, void *data)
|
||||||
window->resource.name = info->name;
|
window->resource.name = info->name;
|
||||||
window->resource.flags = flags;
|
window->resource.flags = flags;
|
||||||
window->resource.start = addr.min_address_range + offset;
|
window->resource.start = addr.min_address_range + offset;
|
||||||
window->resource.end = addr.max_address_range + offset;
|
window->resource.end = window->resource.start + addr.address_length - 1;
|
||||||
window->resource.child = NULL;
|
window->resource.child = NULL;
|
||||||
window->offset = offset;
|
window->offset = offset;
|
||||||
|
|
||||||
|
@ -739,7 +797,7 @@ int pci_vector_resources(int last, int nr_released)
|
||||||
{
|
{
|
||||||
int count = nr_released;
|
int count = nr_released;
|
||||||
|
|
||||||
count += (IA64_LAST_DEVICE_VECTOR - last);
|
count += (IA64_LAST_DEVICE_VECTOR - last);
|
||||||
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
|
@ -349,7 +349,7 @@ void sn_pci_controller_fixup(int segment, int busnum, struct pci_bus *bus)
|
||||||
return; /*bus # does not exist */
|
return; /*bus # does not exist */
|
||||||
prom_bussoft_ptr = __va(prom_bussoft_ptr);
|
prom_bussoft_ptr = __va(prom_bussoft_ptr);
|
||||||
|
|
||||||
controller = kcalloc(1,sizeof(struct pci_controller), GFP_KERNEL);
|
controller = kzalloc(sizeof(struct pci_controller), GFP_KERNEL);
|
||||||
controller->segment = segment;
|
controller->segment = segment;
|
||||||
if (!controller)
|
if (!controller)
|
||||||
BUG();
|
BUG();
|
||||||
|
|
|
@ -163,7 +163,7 @@ struct xpc_vars {
|
||||||
u8 version;
|
u8 version;
|
||||||
u64 heartbeat;
|
u64 heartbeat;
|
||||||
u64 heartbeating_to_mask;
|
u64 heartbeating_to_mask;
|
||||||
u64 kdb_status; /* 0 = machine running */
|
u64 heartbeat_offline; /* if 0, heartbeat should be changing */
|
||||||
int act_nasid;
|
int act_nasid;
|
||||||
int act_phys_cpuid;
|
int act_phys_cpuid;
|
||||||
u64 vars_part_pa;
|
u64 vars_part_pa;
|
||||||
|
|
|
@ -57,6 +57,7 @@
|
||||||
#include <linux/reboot.h>
|
#include <linux/reboot.h>
|
||||||
#include <asm/sn/intr.h>
|
#include <asm/sn/intr.h>
|
||||||
#include <asm/sn/sn_sal.h>
|
#include <asm/sn/sn_sal.h>
|
||||||
|
#include <asm/kdebug.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
#include "xpc.h"
|
#include "xpc.h"
|
||||||
|
|
||||||
|
@ -188,6 +189,11 @@ static struct notifier_block xpc_reboot_notifier = {
|
||||||
.notifier_call = xpc_system_reboot,
|
.notifier_call = xpc_system_reboot,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static int xpc_system_die(struct notifier_block *, unsigned long, void *);
|
||||||
|
static struct notifier_block xpc_die_notifier = {
|
||||||
|
.notifier_call = xpc_system_die,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Timer function to enforce the timelimit on the partition disengage request.
|
* Timer function to enforce the timelimit on the partition disengage request.
|
||||||
|
@ -997,6 +1003,9 @@ xpc_do_exit(enum xpc_retval reason)
|
||||||
/* take ourselves off of the reboot_notifier_list */
|
/* take ourselves off of the reboot_notifier_list */
|
||||||
(void) unregister_reboot_notifier(&xpc_reboot_notifier);
|
(void) unregister_reboot_notifier(&xpc_reboot_notifier);
|
||||||
|
|
||||||
|
/* take ourselves off of the die_notifier list */
|
||||||
|
(void) unregister_die_notifier(&xpc_die_notifier);
|
||||||
|
|
||||||
/* close down protections for IPI operations */
|
/* close down protections for IPI operations */
|
||||||
xpc_restrict_IPI_ops();
|
xpc_restrict_IPI_ops();
|
||||||
|
|
||||||
|
@ -1010,6 +1019,63 @@ xpc_do_exit(enum xpc_retval reason)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Called when the system is about to be either restarted or halted.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
xpc_die_disengage(void)
|
||||||
|
{
|
||||||
|
struct xpc_partition *part;
|
||||||
|
partid_t partid;
|
||||||
|
unsigned long engaged;
|
||||||
|
long time, print_time, disengage_request_timeout;
|
||||||
|
|
||||||
|
|
||||||
|
/* keep xpc_hb_checker thread from doing anything (just in case) */
|
||||||
|
xpc_exiting = 1;
|
||||||
|
|
||||||
|
xpc_vars->heartbeating_to_mask = 0; /* indicate we're deactivated */
|
||||||
|
|
||||||
|
for (partid = 1; partid < XP_MAX_PARTITIONS; partid++) {
|
||||||
|
part = &xpc_partitions[partid];
|
||||||
|
|
||||||
|
if (!XPC_SUPPORTS_DISENGAGE_REQUEST(part->
|
||||||
|
remote_vars_version)) {
|
||||||
|
|
||||||
|
/* just in case it was left set by an earlier XPC */
|
||||||
|
xpc_clear_partition_engaged(1UL << partid);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (xpc_partition_engaged(1UL << partid) ||
|
||||||
|
part->act_state != XPC_P_INACTIVE) {
|
||||||
|
xpc_request_partition_disengage(part);
|
||||||
|
xpc_mark_partition_disengaged(part);
|
||||||
|
xpc_IPI_send_disengage(part);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
print_time = rtc_time();
|
||||||
|
disengage_request_timeout = print_time +
|
||||||
|
(xpc_disengage_request_timelimit * sn_rtc_cycles_per_second);
|
||||||
|
|
||||||
|
/* wait for all other partitions to disengage from us */
|
||||||
|
|
||||||
|
while ((engaged = xpc_partition_engaged(-1UL)) &&
|
||||||
|
(time = rtc_time()) < disengage_request_timeout) {
|
||||||
|
|
||||||
|
if (time >= print_time) {
|
||||||
|
dev_info(xpc_part, "waiting for remote partitions to "
|
||||||
|
"disengage, engaged=0x%lx\n", engaged);
|
||||||
|
print_time = time + (XPC_DISENGAGE_PRINTMSG_INTERVAL *
|
||||||
|
sn_rtc_cycles_per_second);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dev_info(xpc_part, "finished waiting for remote partitions to "
|
||||||
|
"disengage, engaged=0x%lx\n", engaged);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This function is called when the system is being rebooted.
|
* This function is called when the system is being rebooted.
|
||||||
*/
|
*/
|
||||||
|
@ -1038,6 +1104,33 @@ xpc_system_reboot(struct notifier_block *nb, unsigned long event, void *unused)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This function is called when the system is being rebooted.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
xpc_system_die(struct notifier_block *nb, unsigned long event, void *unused)
|
||||||
|
{
|
||||||
|
switch (event) {
|
||||||
|
case DIE_MACHINE_RESTART:
|
||||||
|
case DIE_MACHINE_HALT:
|
||||||
|
xpc_die_disengage();
|
||||||
|
break;
|
||||||
|
case DIE_MCA_MONARCH_ENTER:
|
||||||
|
case DIE_INIT_MONARCH_ENTER:
|
||||||
|
xpc_vars->heartbeat++;
|
||||||
|
xpc_vars->heartbeat_offline = 1;
|
||||||
|
break;
|
||||||
|
case DIE_MCA_MONARCH_LEAVE:
|
||||||
|
case DIE_INIT_MONARCH_LEAVE:
|
||||||
|
xpc_vars->heartbeat++;
|
||||||
|
xpc_vars->heartbeat_offline = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NOTIFY_DONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int __init
|
int __init
|
||||||
xpc_init(void)
|
xpc_init(void)
|
||||||
{
|
{
|
||||||
|
@ -1154,6 +1247,12 @@ xpc_init(void)
|
||||||
dev_warn(xpc_part, "can't register reboot notifier\n");
|
dev_warn(xpc_part, "can't register reboot notifier\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* add ourselves to the die_notifier list (i.e., ia64die_chain) */
|
||||||
|
ret = register_die_notifier(&xpc_die_notifier);
|
||||||
|
if (ret != 0) {
|
||||||
|
dev_warn(xpc_part, "can't register die notifier\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Set the beating to other partitions into motion. This is
|
* Set the beating to other partitions into motion. This is
|
||||||
|
@ -1179,6 +1278,9 @@ xpc_init(void)
|
||||||
/* take ourselves off of the reboot_notifier_list */
|
/* take ourselves off of the reboot_notifier_list */
|
||||||
(void) unregister_reboot_notifier(&xpc_reboot_notifier);
|
(void) unregister_reboot_notifier(&xpc_reboot_notifier);
|
||||||
|
|
||||||
|
/* take ourselves off of the die_notifier list */
|
||||||
|
(void) unregister_die_notifier(&xpc_die_notifier);
|
||||||
|
|
||||||
del_timer_sync(&xpc_hb_timer);
|
del_timer_sync(&xpc_hb_timer);
|
||||||
free_irq(SGI_XPC_ACTIVATE, NULL);
|
free_irq(SGI_XPC_ACTIVATE, NULL);
|
||||||
xpc_restrict_IPI_ops();
|
xpc_restrict_IPI_ops();
|
||||||
|
|
|
@ -436,13 +436,13 @@ xpc_check_remote_hb(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_dbg(xpc_part, "partid = %d, heartbeat = %ld, last_heartbeat"
|
dev_dbg(xpc_part, "partid = %d, heartbeat = %ld, last_heartbeat"
|
||||||
" = %ld, kdb_status = %ld, HB_mask = 0x%lx\n", partid,
|
" = %ld, heartbeat_offline = %ld, HB_mask = 0x%lx\n",
|
||||||
remote_vars->heartbeat, part->last_heartbeat,
|
partid, remote_vars->heartbeat, part->last_heartbeat,
|
||||||
remote_vars->kdb_status,
|
remote_vars->heartbeat_offline,
|
||||||
remote_vars->heartbeating_to_mask);
|
remote_vars->heartbeating_to_mask);
|
||||||
|
|
||||||
if (((remote_vars->heartbeat == part->last_heartbeat) &&
|
if (((remote_vars->heartbeat == part->last_heartbeat) &&
|
||||||
(remote_vars->kdb_status == 0)) ||
|
(remote_vars->heartbeat_offline == 0)) ||
|
||||||
!xpc_hb_allowed(sn_partition_id, remote_vars)) {
|
!xpc_hb_allowed(sn_partition_id, remote_vars)) {
|
||||||
|
|
||||||
XPC_DEACTIVATE_PARTITION(part, xpcNoHeartbeat);
|
XPC_DEACTIVATE_PARTITION(part, xpcNoHeartbeat);
|
||||||
|
|
|
@ -218,7 +218,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
if (i > last)
|
if (i > last)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
map = kcalloc(1, sizeof(struct tioce_dmamap), GFP_ATOMIC);
|
map = kzalloc(sizeof(struct tioce_dmamap), GFP_ATOMIC);
|
||||||
if (!map)
|
if (!map)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -555,7 +555,7 @@ tioce_kern_init(struct tioce_common *tioce_common)
|
||||||
struct tioce *tioce_mmr;
|
struct tioce *tioce_mmr;
|
||||||
struct tioce_kernel *tioce_kern;
|
struct tioce_kernel *tioce_kern;
|
||||||
|
|
||||||
tioce_kern = kcalloc(1, sizeof(struct tioce_kernel), GFP_KERNEL);
|
tioce_kern = kzalloc(sizeof(struct tioce_kernel), GFP_KERNEL);
|
||||||
if (!tioce_kern) {
|
if (!tioce_kern) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -727,7 +727,7 @@ tioce_bus_fixup(struct pcibus_bussoft *prom_bussoft, struct pci_controller *cont
|
||||||
* Allocate kernel bus soft and copy from prom.
|
* Allocate kernel bus soft and copy from prom.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
tioce_common = kcalloc(1, sizeof(struct tioce_common), GFP_KERNEL);
|
tioce_common = kzalloc(sizeof(struct tioce_common), GFP_KERNEL);
|
||||||
if (!tioce_common)
|
if (!tioce_common)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
|
|
@ -22,6 +22,9 @@
|
||||||
* 2005-Apr Rusty Lynch <rusty.lynch@intel.com> and Anil S Keshavamurthy
|
* 2005-Apr Rusty Lynch <rusty.lynch@intel.com> and Anil S Keshavamurthy
|
||||||
* <anil.s.keshavamurthy@intel.com> adopted from
|
* <anil.s.keshavamurthy@intel.com> adopted from
|
||||||
* include/asm-x86_64/kdebug.h
|
* include/asm-x86_64/kdebug.h
|
||||||
|
*
|
||||||
|
* 2005-Oct Keith Owens <kaos@sgi.com>. Expand notify_die to cover more
|
||||||
|
* events.
|
||||||
*/
|
*/
|
||||||
#include <linux/notifier.h>
|
#include <linux/notifier.h>
|
||||||
|
|
||||||
|
@ -35,13 +38,36 @@ struct die_args {
|
||||||
int signr;
|
int signr;
|
||||||
};
|
};
|
||||||
|
|
||||||
int register_die_notifier(struct notifier_block *nb);
|
extern int register_die_notifier(struct notifier_block *);
|
||||||
|
extern int unregister_die_notifier(struct notifier_block *);
|
||||||
extern struct notifier_block *ia64die_chain;
|
extern struct notifier_block *ia64die_chain;
|
||||||
|
|
||||||
enum die_val {
|
enum die_val {
|
||||||
DIE_BREAK = 1,
|
DIE_BREAK = 1,
|
||||||
DIE_SS,
|
DIE_FAULT,
|
||||||
|
DIE_OOPS,
|
||||||
DIE_PAGE_FAULT,
|
DIE_PAGE_FAULT,
|
||||||
|
DIE_MACHINE_HALT,
|
||||||
|
DIE_MACHINE_RESTART,
|
||||||
|
DIE_MCA_MONARCH_ENTER,
|
||||||
|
DIE_MCA_MONARCH_PROCESS,
|
||||||
|
DIE_MCA_MONARCH_LEAVE,
|
||||||
|
DIE_MCA_SLAVE_ENTER,
|
||||||
|
DIE_MCA_SLAVE_PROCESS,
|
||||||
|
DIE_MCA_SLAVE_LEAVE,
|
||||||
|
DIE_MCA_RENDZVOUS_ENTER,
|
||||||
|
DIE_MCA_RENDZVOUS_PROCESS,
|
||||||
|
DIE_MCA_RENDZVOUS_LEAVE,
|
||||||
|
DIE_INIT_MONARCH_ENTER,
|
||||||
|
DIE_INIT_MONARCH_PROCESS,
|
||||||
|
DIE_INIT_MONARCH_LEAVE,
|
||||||
|
DIE_INIT_SLAVE_ENTER,
|
||||||
|
DIE_INIT_SLAVE_PROCESS,
|
||||||
|
DIE_INIT_SLAVE_LEAVE,
|
||||||
|
DIE_KDEBUG_ENTER,
|
||||||
|
DIE_KDEBUG_LEAVE,
|
||||||
|
DIE_KDUMP_ENTER,
|
||||||
|
DIE_KDUMP_LEAVE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline int notify_die(enum die_val val, char *str, struct pt_regs *regs,
|
static inline int notify_die(enum die_val val, char *str, struct pt_regs *regs,
|
||||||
|
|
|
@ -7,12 +7,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Routines to manage the allocation of task context numbers. Task context numbers are
|
* Routines to manage the allocation of task context numbers. Task context
|
||||||
* used to reduce or eliminate the need to perform TLB flushes due to context switches.
|
* numbers are used to reduce or eliminate the need to perform TLB flushes
|
||||||
* Context numbers are implemented using ia-64 region ids. Since the IA-64 TLB does not
|
* due to context switches. Context numbers are implemented using ia-64
|
||||||
* consider the region number when performing a TLB lookup, we need to assign a unique
|
* region ids. Since the IA-64 TLB does not consider the region number when
|
||||||
* region id to each region in a process. We use the least significant three bits in a
|
* performing a TLB lookup, we need to assign a unique region id to each
|
||||||
* region id for this purpose.
|
* region in a process. We use the least significant three bits in aregion
|
||||||
|
* id for this purpose.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define IA64_REGION_ID_KERNEL 0 /* the kernel's region id (tlb.c depends on this being 0) */
|
#define IA64_REGION_ID_KERNEL 0 /* the kernel's region id (tlb.c depends on this being 0) */
|
||||||
|
@ -32,13 +33,17 @@
|
||||||
struct ia64_ctx {
|
struct ia64_ctx {
|
||||||
spinlock_t lock;
|
spinlock_t lock;
|
||||||
unsigned int next; /* next context number to use */
|
unsigned int next; /* next context number to use */
|
||||||
unsigned int limit; /* next >= limit => must call wrap_mmu_context() */
|
unsigned int limit; /* available free range */
|
||||||
unsigned int max_ctx; /* max. context value supported by all CPUs */
|
unsigned int max_ctx; /* max. context value supported by all CPUs */
|
||||||
|
/* call wrap_mmu_context when next >= max */
|
||||||
|
unsigned long *bitmap; /* bitmap size is max_ctx+1 */
|
||||||
|
unsigned long *flushmap;/* pending rid to be flushed */
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct ia64_ctx ia64_ctx;
|
extern struct ia64_ctx ia64_ctx;
|
||||||
DECLARE_PER_CPU(u8, ia64_need_tlb_flush);
|
DECLARE_PER_CPU(u8, ia64_need_tlb_flush);
|
||||||
|
|
||||||
|
extern void mmu_context_init (void);
|
||||||
extern void wrap_mmu_context (struct mm_struct *mm);
|
extern void wrap_mmu_context (struct mm_struct *mm);
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
|
@ -47,10 +52,10 @@ enter_lazy_tlb (struct mm_struct *mm, struct task_struct *tsk)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* When the context counter wraps around all TLBs need to be flushed because an old
|
* When the context counter wraps around all TLBs need to be flushed because
|
||||||
* context number might have been reused. This is signalled by the ia64_need_tlb_flush
|
* an old context number might have been reused. This is signalled by the
|
||||||
* per-CPU variable, which is checked in the routine below. Called by activate_mm().
|
* ia64_need_tlb_flush per-CPU variable, which is checked in the routine
|
||||||
* <efocht@ess.nec.de>
|
* below. Called by activate_mm(). <efocht@ess.nec.de>
|
||||||
*/
|
*/
|
||||||
static inline void
|
static inline void
|
||||||
delayed_tlb_flush (void)
|
delayed_tlb_flush (void)
|
||||||
|
@ -60,11 +65,9 @@ delayed_tlb_flush (void)
|
||||||
|
|
||||||
if (unlikely(__ia64_per_cpu_var(ia64_need_tlb_flush))) {
|
if (unlikely(__ia64_per_cpu_var(ia64_need_tlb_flush))) {
|
||||||
spin_lock_irqsave(&ia64_ctx.lock, flags);
|
spin_lock_irqsave(&ia64_ctx.lock, flags);
|
||||||
{
|
if (__ia64_per_cpu_var(ia64_need_tlb_flush)) {
|
||||||
if (__ia64_per_cpu_var(ia64_need_tlb_flush)) {
|
local_flush_tlb_all();
|
||||||
local_flush_tlb_all();
|
__ia64_per_cpu_var(ia64_need_tlb_flush) = 0;
|
||||||
__ia64_per_cpu_var(ia64_need_tlb_flush) = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
spin_unlock_irqrestore(&ia64_ctx.lock, flags);
|
spin_unlock_irqrestore(&ia64_ctx.lock, flags);
|
||||||
}
|
}
|
||||||
|
@ -76,20 +79,27 @@ get_mmu_context (struct mm_struct *mm)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
nv_mm_context_t context = mm->context;
|
nv_mm_context_t context = mm->context;
|
||||||
|
|
||||||
if (unlikely(!context)) {
|
if (likely(context))
|
||||||
spin_lock_irqsave(&ia64_ctx.lock, flags);
|
goto out;
|
||||||
{
|
|
||||||
/* re-check, now that we've got the lock: */
|
spin_lock_irqsave(&ia64_ctx.lock, flags);
|
||||||
context = mm->context;
|
/* re-check, now that we've got the lock: */
|
||||||
if (context == 0) {
|
context = mm->context;
|
||||||
cpus_clear(mm->cpu_vm_mask);
|
if (context == 0) {
|
||||||
if (ia64_ctx.next >= ia64_ctx.limit)
|
cpus_clear(mm->cpu_vm_mask);
|
||||||
wrap_mmu_context(mm);
|
if (ia64_ctx.next >= ia64_ctx.limit) {
|
||||||
mm->context = context = ia64_ctx.next++;
|
ia64_ctx.next = find_next_zero_bit(ia64_ctx.bitmap,
|
||||||
}
|
ia64_ctx.max_ctx, ia64_ctx.next);
|
||||||
|
ia64_ctx.limit = find_next_bit(ia64_ctx.bitmap,
|
||||||
|
ia64_ctx.max_ctx, ia64_ctx.next);
|
||||||
|
if (ia64_ctx.next >= ia64_ctx.max_ctx)
|
||||||
|
wrap_mmu_context(mm);
|
||||||
}
|
}
|
||||||
spin_unlock_irqrestore(&ia64_ctx.lock, flags);
|
mm->context = context = ia64_ctx.next++;
|
||||||
|
__set_bit(context, ia64_ctx.bitmap);
|
||||||
}
|
}
|
||||||
|
spin_unlock_irqrestore(&ia64_ctx.lock, flags);
|
||||||
|
out:
|
||||||
/*
|
/*
|
||||||
* Ensure we're not starting to use "context" before any old
|
* Ensure we're not starting to use "context" before any old
|
||||||
* uses of it are gone from our TLB.
|
* uses of it are gone from our TLB.
|
||||||
|
@ -100,8 +110,8 @@ get_mmu_context (struct mm_struct *mm)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialize context number to some sane value. MM is guaranteed to be a brand-new
|
* Initialize context number to some sane value. MM is guaranteed to be a
|
||||||
* address-space, so no TLB flushing is needed, ever.
|
* brand-new address-space, so no TLB flushing is needed, ever.
|
||||||
*/
|
*/
|
||||||
static inline int
|
static inline int
|
||||||
init_new_context (struct task_struct *p, struct mm_struct *mm)
|
init_new_context (struct task_struct *p, struct mm_struct *mm)
|
||||||
|
@ -162,7 +172,10 @@ activate_context (struct mm_struct *mm)
|
||||||
if (!cpu_isset(smp_processor_id(), mm->cpu_vm_mask))
|
if (!cpu_isset(smp_processor_id(), mm->cpu_vm_mask))
|
||||||
cpu_set(smp_processor_id(), mm->cpu_vm_mask);
|
cpu_set(smp_processor_id(), mm->cpu_vm_mask);
|
||||||
reload_context(context);
|
reload_context(context);
|
||||||
/* in the unlikely event of a TLB-flush by another thread, redo the load: */
|
/*
|
||||||
|
* in the unlikely event of a TLB-flush by another thread,
|
||||||
|
* redo the load.
|
||||||
|
*/
|
||||||
} while (unlikely(context != mm->context));
|
} while (unlikely(context != mm->context));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -175,8 +188,8 @@ static inline void
|
||||||
activate_mm (struct mm_struct *prev, struct mm_struct *next)
|
activate_mm (struct mm_struct *prev, struct mm_struct *next)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* We may get interrupts here, but that's OK because interrupt handlers cannot
|
* We may get interrupts here, but that's OK because interrupt
|
||||||
* touch user-space.
|
* handlers cannot touch user-space.
|
||||||
*/
|
*/
|
||||||
ia64_set_kr(IA64_KR_PT_BASE, __pa(next->pgd));
|
ia64_set_kr(IA64_KR_PT_BASE, __pa(next->pgd));
|
||||||
activate_context(next);
|
activate_context(next);
|
||||||
|
|
|
@ -51,6 +51,7 @@ flush_tlb_mm (struct mm_struct *mm)
|
||||||
if (!mm)
|
if (!mm)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
set_bit(mm->context, ia64_ctx.flushmap);
|
||||||
mm->context = 0;
|
mm->context = 0;
|
||||||
|
|
||||||
if (atomic_read(&mm->mm_users) == 0)
|
if (atomic_read(&mm->mm_users) == 0)
|
||||||
|
|
Loading…
Reference in New Issue