Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
6d23c8bc
Commit
6d23c8bc
authored
Nov 10, 2005
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'release' of
git://git.kernel.org/pub/scm/linux/kernel/git/aegl/linux-2.6
parents
985834a1
780d09e8
Changes
19
Show whitespace changes
Inline
Side-by-side
Showing
19 changed files
with
475 additions
and
190 deletions
+475
-190
arch/ia64/kernel/efi.c
arch/ia64/kernel/efi.c
+1
-1
arch/ia64/kernel/kprobes.c
arch/ia64/kernel/kprobes.c
+13
-9
arch/ia64/kernel/mca.c
arch/ia64/kernel/mca.c
+95
-25
arch/ia64/kernel/mca_drv.c
arch/ia64/kernel/mca_drv.c
+13
-4
arch/ia64/kernel/process.c
arch/ia64/kernel/process.c
+6
-0
arch/ia64/kernel/setup.c
arch/ia64/kernel/setup.c
+1
-0
arch/ia64/kernel/signal.c
arch/ia64/kernel/signal.c
+5
-6
arch/ia64/kernel/traps.c
arch/ia64/kernel/traps.c
+20
-24
arch/ia64/mm/discontig.c
arch/ia64/mm/discontig.c
+9
-10
arch/ia64/mm/tlb.c
arch/ia64/mm/tlb.c
+43
-42
arch/ia64/pci/pci.c
arch/ia64/pci/pci.c
+82
-24
arch/ia64/sn/kernel/io_init.c
arch/ia64/sn/kernel/io_init.c
+1
-1
arch/ia64/sn/kernel/xpc.h
arch/ia64/sn/kernel/xpc.h
+1
-1
arch/ia64/sn/kernel/xpc_main.c
arch/ia64/sn/kernel/xpc_main.c
+102
-0
arch/ia64/sn/kernel/xpc_partition.c
arch/ia64/sn/kernel/xpc_partition.c
+4
-4
arch/ia64/sn/pci/tioce_provider.c
arch/ia64/sn/pci/tioce_provider.c
+3
-3
include/asm-ia64/kdebug.h
include/asm-ia64/kdebug.h
+28
-2
include/asm-ia64/mmu_context.h
include/asm-ia64/mmu_context.h
+47
-34
include/asm-ia64/tlbflush.h
include/asm-ia64/tlbflush.h
+1
-0
No files found.
arch/ia64/kernel/efi.c
View file @
6d23c8bc
...
@@ -987,7 +987,7 @@ efi_initialize_iomem_resources(struct resource *code_resource,
...
@@ -987,7 +987,7 @@ efi_initialize_iomem_resources(struct resource *code_resource,
break
;
break
;
}
}
if
((
res
=
k
calloc
(
1
,
sizeof
(
struct
resource
),
GFP_KERNEL
))
==
NULL
)
{
if
((
res
=
k
zalloc
(
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
;
}
}
...
...
arch/ia64/kernel/kprobes.c
View file @
6d23c8bc
...
@@ -739,10 +739,14 @@ int __kprobes kprobe_exceptions_notify(struct notifier_block *self,
...
@@ -739,10 +739,14 @@ int __kprobes kprobe_exceptions_notify(struct notifier_block *self,
switch
(
val
)
{
switch
(
val
)
{
case
DIE_BREAK
:
case
DIE_BREAK
:
/* err is break number from ia64_bad_break() */
if
(
args
->
err
==
0x80200
||
args
->
err
==
0x80300
)
if
(
pre_kprobes_handler
(
args
))
if
(
pre_kprobes_handler
(
args
))
ret
=
NOTIFY_STOP
;
ret
=
NOTIFY_STOP
;
break
;
break
;
case
DIE_SS
:
case
DIE_FAULT
:
/* err is vector number from ia64_fault() */
if
(
args
->
err
==
36
)
if
(
post_kprobes_handler
(
args
->
regs
))
if
(
post_kprobes_handler
(
args
->
regs
))
ret
=
NOTIFY_STOP
;
ret
=
NOTIFY_STOP
;
break
;
break
;
...
...
arch/ia64/kernel/mca.c
View file @
6d23c8bc
...
@@ -51,6 +51,9 @@
...
@@ -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 @@
...
@@ -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 @@
...
@@ -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);
...
@@ -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)
...
@@ -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
*
pt
regs
)
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)
...
@@ -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,
...
@@ -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,
...
@@ -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,
...
@@ -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)
...
@@ -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
(
"
\n
Backtrace 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,
...
@@ -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,
...
@@ -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,
...
@@ -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,
...
@@ -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
(
"
\n
Backtrace of pid %d (%s)
\n
"
,
t
->
pid
,
t
->
comm
);
show_stack
(
t
,
NULL
);
}
while_each_thread
(
g
,
t
);
read_unlock
(
&
tasklist_lock
);
}
printk
(
"
\n
INIT dump complete. Monarch on cpu %d returning to normal service.
\n
"
,
cpu
);
printk
(
"
\n
INIT 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)
...
@@ -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)
...
@@ -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__
);
...
...
arch/ia64/kernel/mca_drv.c
View file @
6d23c8bc
...
@@ -547,9 +547,20 @@ recover_from_processor_error(int platform, slidx_table_t *slidx,
...
@@ -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,
...
@@ -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.
...
...
arch/ia64/kernel/process.c
View file @
6d23c8bc
...
@@ -4,6 +4,9 @@
...
@@ -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 @@
...
@@ -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)
...
@@ -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
();
}
}
...
...
arch/ia64/kernel/setup.c
View file @
6d23c8bc
...
@@ -461,6 +461,7 @@ setup_arch (char **cmdline_p)
...
@@ -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
();
...
...
arch/ia64/kernel/signal.c
View file @
6d23c8bc
...
@@ -387,15 +387,14 @@ setup_frame (int sig, struct k_sigaction *ka, siginfo_t *info, sigset_t *set,
...
@@ -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,
...
@@ -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
);
...
...
arch/ia64/kernel/traps.c
View file @
6d23c8bc
...
@@ -30,17 +30,20 @@ fpswa_interface_t *fpswa_interface;
...
@@ -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)
...
@@ -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)
...
@@ -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)
...
@@ -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)
...
@@ -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,
...
@@ -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"
,
&
regs
,
vector
,
vector
,
SIGTRAP
)
==
NOTIFY_STOP
)
return
;
siginfo
.
si_code
=
TRAP_TRACE
;
ifa
=
0
;
break
;
}
}
if
(
notify_die
(
DIE_FAULT
,
"ia64_fault"
,
&
regs
,
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
;
...
...
arch/ia64/mm/discontig.c
View file @
6d23c8bc
...
@@ -350,14 +350,12 @@ static void __init initialize_pernode_data(void)
...
@@ -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,
...
@@ -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)
...
@@ -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
);
}
}
...
...
arch/ia64/mm/tlb.c
View file @
6d23c8bc
...
@@ -8,6 +8,8 @@
...
@@ -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 @@
...
@@ -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
;
for
(
i
=
0
;
i
<=
ia64_ctx
.
max_ctx
/
BITS_PER_LONG
;
i
++
)
{
flush_bit
=
xchg
(
&
ia64_ctx
.
flushmap
[
i
],
0
);
ia64_ctx
.
bitmap
[
i
]
^=
flush_bit
;
}
if
(
ia64_ctx
.
next
>
max_ctx
)
/* use offset at 300 to skip daemons */
ia64_ctx
.
next
=
300
;
/* skip daemons */
ia64_ctx
.
next
=
find_next_zero_bit
(
ia64_ctx
.
bitmap
,
ia64_ctx
.
limit
=
max_ctx
+
1
;
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:
for_each_process
(
tsk
)
{
if
(
!
tsk
->
mm
)
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
)
if
(
i
!=
cpu
)
per_cpu
(
ia64_need_tlb_flush
,
i
)
=
1
;
per_cpu
(
ia64_need_tlb_flush
,
i
)
=
1
;
}
put_cpu
();
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)
...
@@ -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
...
@@ -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
;
...
...
arch/ia64/pci/pci.c
View file @
6d23c8bc
...
@@ -137,35 +137,98 @@ alloc_pci_controller (int seg)
...
@@ -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
)
sparse
=
1
;
offset
=
(
u64
)
ioremap
(
addr
->
address_translation_offset
,
0
);
mmio_base
=
(
u64
)
ioremap
(
phys_base
,
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)
...
@@ -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)
...
@@ -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)
...
@@ -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
;
...
...
arch/ia64/sn/kernel/io_init.c
View file @
6d23c8bc
...
@@ -349,7 +349,7 @@ void sn_pci_controller_fixup(int segment, int busnum, struct pci_bus *bus)
...
@@ -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
=
k
calloc
(
1
,
sizeof
(
struct
pci_controller
),
GFP_KERNEL
);
controller
=
k
zalloc
(
sizeof
(
struct
pci_controller
),
GFP_KERNEL
);
controller
->
segment
=
segment
;
controller
->
segment
=
segment
;
if
(
!
controller
)
if
(
!
controller
)
BUG
();
BUG
();
...
...
arch/ia64/sn/kernel/xpc.h
View file @
6d23c8bc
...
@@ -163,7 +163,7 @@ struct xpc_vars {
...
@@ -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 runn
ing */
u64
heartbeat_offline
;
/* if 0, heartbeat should be chang
ing */
int
act_nasid
;
int
act_nasid
;
int
act_phys_cpuid
;
int
act_phys_cpuid
;
u64
vars_part_pa
;
u64
vars_part_pa
;
...
...
arch/ia64/sn/kernel/xpc_main.c
View file @
6d23c8bc
...
@@ -57,6 +57,7 @@
...
@@ -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 = {
...
@@ -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)
...
@@ -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)
...
@@ -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)
...
@@ -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)
...
@@ -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)
...
@@ -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
();
...
...
arch/ia64/sn/kernel/xpc_partition.c
View file @
6d23c8bc
...
@@ -436,13 +436,13 @@ xpc_check_remote_hb(void)
...
@@ -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
);
...
...
arch/ia64/sn/pci/tioce_provider.c
View file @
6d23c8bc
...
@@ -218,7 +218,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
...
@@ -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
=
k
calloc
(
1
,
sizeof
(
struct
tioce_dmamap
),
GFP_ATOMIC
);
map
=
k
zalloc
(
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)
...
@@ -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
=
k
calloc
(
1
,
sizeof
(
struct
tioce_kernel
),
GFP_KERNEL
);
tioce_kern
=
k
zalloc
(
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
...
@@ -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
=
k
calloc
(
1
,
sizeof
(
struct
tioce_common
),
GFP_KERNEL
);
tioce_common
=
k
zalloc
(
sizeof
(
struct
tioce_common
),
GFP_KERNEL
);
if
(
!
tioce_common
)
if
(
!
tioce_common
)
return
NULL
;
return
NULL
;
...
...
include/asm-ia64/kdebug.h
View file @
6d23c8bc
...
@@ -22,6 +22,9 @@
...
@@ -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 {
...
@@ -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
,
...
...
include/asm-ia64/mmu_context.h
View file @
6d23c8bc
...
@@ -7,12 +7,13 @@
...
@@ -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 @@
...
@@ -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)
...
@@ -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,12 +65,10 @@ delayed_tlb_flush (void)
...
@@ -60,12 +65,10 @@ 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)
...
@@ -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
))
goto
out
;
spin_lock_irqsave
(
&
ia64_ctx
.
lock
,
flags
);
spin_lock_irqsave
(
&
ia64_ctx
.
lock
,
flags
);
{
/* re-check, now that we've got the lock: */
/* re-check, now that we've got the lock: */
context
=
mm
->
context
;
context
=
mm
->
context
;
if
(
context
==
0
)
{
if
(
context
==
0
)
{
cpus_clear
(
mm
->
cpu_vm_mask
);
cpus_clear
(
mm
->
cpu_vm_mask
);
if
(
ia64_ctx
.
next
>=
ia64_ctx
.
limit
)
if
(
ia64_ctx
.
next
>=
ia64_ctx
.
limit
)
{
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
);
wrap_mmu_context
(
mm
);
mm
->
context
=
context
=
ia64_ctx
.
next
++
;
}
}
mm
->
context
=
context
=
ia64_ctx
.
next
++
;
__set_bit
(
context
,
ia64_ctx
.
bitmap
);
}
}
spin_unlock_irqrestore
(
&
ia64_ctx
.
lock
,
flags
);
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)
...
@@ -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)
...
@@ -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
...
@@ -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
);
...
...
include/asm-ia64/tlbflush.h
View file @
6d23c8bc
...
@@ -51,6 +51,7 @@ flush_tlb_mm (struct mm_struct *mm)
...
@@ -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
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment