Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
3f371438
Commit
3f371438
authored
Jan 12, 2005
by
Russell King
Browse files
Options
Browse Files
Download
Plain Diff
Merge flint.arm.linux.org.uk:/usr/src/bk/linux-2.6-smp
into flint.arm.linux.org.uk:/usr/src/bk/linux-2.6-rmk
parents
08432e0e
d4118bd6
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
157 additions
and
14 deletions
+157
-14
arch/arm/kernel/irq.c
arch/arm/kernel/irq.c
+126
-3
arch/arm/kernel/smp.c
arch/arm/kernel/smp.c
+10
-10
include/asm-arm/cpu.h
include/asm-arm/cpu.h
+0
-1
include/asm-arm/mach/irq.h
include/asm-arm/mach/irq.h
+14
-0
include/asm-arm/smp.h
include/asm-arm/smp.h
+7
-0
No files found.
arch/arm/kernel/irq.c
View file @
3f371438
...
@@ -32,6 +32,7 @@
...
@@ -32,6 +32,7 @@
#include <linux/errno.h>
#include <linux/errno.h>
#include <linux/list.h>
#include <linux/list.h>
#include <linux/kallsyms.h>
#include <linux/kallsyms.h>
#include <linux/proc_fs.h>
#include <asm/irq.h>
#include <asm/irq.h>
#include <asm/system.h>
#include <asm/system.h>
...
@@ -85,6 +86,23 @@ static struct irqdesc bad_irq_desc = {
...
@@ -85,6 +86,23 @@ static struct irqdesc bad_irq_desc = {
.
disable_depth
=
1
,
.
disable_depth
=
1
,
};
};
#ifdef CONFIG_SMP
void
synchronize_irq
(
unsigned
int
irq
)
{
struct
irqdesc
*
desc
=
irq_desc
+
irq
;
while
(
desc
->
running
)
barrier
();
}
EXPORT_SYMBOL
(
synchronize_irq
);
#define smp_set_running(desc) do { desc->running = 1; } while (0)
#define smp_clear_running(desc) do { desc->running = 0; } while (0)
#else
#define smp_set_running(desc) do { } while (0)
#define smp_clear_running(desc) do { } while (0)
#endif
/**
/**
* disable_irq_nosync - disable an irq without waiting
* disable_irq_nosync - disable an irq without waiting
* @irq: Interrupt to disable
* @irq: Interrupt to disable
...
@@ -231,6 +249,9 @@ int show_interrupts(struct seq_file *p, void *v)
...
@@ -231,6 +249,9 @@ int show_interrupts(struct seq_file *p, void *v)
}
else
if
(
i
==
NR_IRQS
)
{
}
else
if
(
i
==
NR_IRQS
)
{
#ifdef CONFIG_ARCH_ACORN
#ifdef CONFIG_ARCH_ACORN
show_fiq_list
(
p
,
v
);
show_fiq_list
(
p
,
v
);
#endif
#ifdef CONFIG_SMP
show_ipi_list
(
p
);
#endif
#endif
seq_printf
(
p
,
"Err: %10lu
\n
"
,
irq_err_count
);
seq_printf
(
p
,
"Err: %10lu
\n
"
,
irq_err_count
);
}
}
...
@@ -329,18 +350,22 @@ void
...
@@ -329,18 +350,22 @@ void
do_simple_IRQ
(
unsigned
int
irq
,
struct
irqdesc
*
desc
,
struct
pt_regs
*
regs
)
do_simple_IRQ
(
unsigned
int
irq
,
struct
irqdesc
*
desc
,
struct
pt_regs
*
regs
)
{
{
struct
irqaction
*
action
;
struct
irqaction
*
action
;
const
int
cpu
=
smp_processor_id
();
const
unsigned
int
cpu
=
smp_processor_id
();
desc
->
triggered
=
1
;
desc
->
triggered
=
1
;
kstat_cpu
(
cpu
).
irqs
[
irq
]
++
;
kstat_cpu
(
cpu
).
irqs
[
irq
]
++
;
smp_set_running
(
desc
);
action
=
desc
->
action
;
action
=
desc
->
action
;
if
(
action
)
{
if
(
action
)
{
int
ret
=
__do_irq
(
irq
,
action
,
regs
);
int
ret
=
__do_irq
(
irq
,
action
,
regs
);
if
(
ret
!=
IRQ_HANDLED
)
if
(
ret
!=
IRQ_HANDLED
)
report_bad_irq
(
irq
,
regs
,
desc
,
ret
);
report_bad_irq
(
irq
,
regs
,
desc
,
ret
);
}
}
smp_clear_running
(
desc
);
}
}
/*
/*
...
@@ -350,7 +375,7 @@ do_simple_IRQ(unsigned int irq, struct irqdesc *desc, struct pt_regs *regs)
...
@@ -350,7 +375,7 @@ do_simple_IRQ(unsigned int irq, struct irqdesc *desc, struct pt_regs *regs)
void
void
do_edge_IRQ
(
unsigned
int
irq
,
struct
irqdesc
*
desc
,
struct
pt_regs
*
regs
)
do_edge_IRQ
(
unsigned
int
irq
,
struct
irqdesc
*
desc
,
struct
pt_regs
*
regs
)
{
{
const
int
cpu
=
smp_processor_id
();
const
unsigned
int
cpu
=
smp_processor_id
();
desc
->
triggered
=
1
;
desc
->
triggered
=
1
;
...
@@ -414,7 +439,7 @@ void
...
@@ -414,7 +439,7 @@ void
do_level_IRQ
(
unsigned
int
irq
,
struct
irqdesc
*
desc
,
struct
pt_regs
*
regs
)
do_level_IRQ
(
unsigned
int
irq
,
struct
irqdesc
*
desc
,
struct
pt_regs
*
regs
)
{
{
struct
irqaction
*
action
;
struct
irqaction
*
action
;
const
int
cpu
=
smp_processor_id
();
const
unsigned
int
cpu
=
smp_processor_id
();
desc
->
triggered
=
1
;
desc
->
triggered
=
1
;
...
@@ -426,6 +451,8 @@ do_level_IRQ(unsigned int irq, struct irqdesc *desc, struct pt_regs *regs)
...
@@ -426,6 +451,8 @@ do_level_IRQ(unsigned int irq, struct irqdesc *desc, struct pt_regs *regs)
if
(
likely
(
!
desc
->
disable_depth
))
{
if
(
likely
(
!
desc
->
disable_depth
))
{
kstat_cpu
(
cpu
).
irqs
[
irq
]
++
;
kstat_cpu
(
cpu
).
irqs
[
irq
]
++
;
smp_set_running
(
desc
);
/*
/*
* Return with this interrupt masked if no action
* Return with this interrupt masked if no action
*/
*/
...
@@ -440,6 +467,8 @@ do_level_IRQ(unsigned int irq, struct irqdesc *desc, struct pt_regs *regs)
...
@@ -440,6 +467,8 @@ do_level_IRQ(unsigned int irq, struct irqdesc *desc, struct pt_regs *regs)
!
check_irq_lock
(
desc
,
irq
,
regs
)))
!
check_irq_lock
(
desc
,
irq
,
regs
)))
desc
->
chip
->
unmask
(
irq
);
desc
->
chip
->
unmask
(
irq
);
}
}
smp_clear_running
(
desc
);
}
}
}
}
...
@@ -878,8 +907,97 @@ int probe_irq_off(unsigned long irqs)
...
@@ -878,8 +907,97 @@ int probe_irq_off(unsigned long irqs)
EXPORT_SYMBOL
(
probe_irq_off
);
EXPORT_SYMBOL
(
probe_irq_off
);
#ifdef CONFIG_SMP
static
void
route_irq
(
struct
irqdesc
*
desc
,
unsigned
int
irq
,
unsigned
int
cpu
)
{
pr_debug
(
"IRQ%u: moving from cpu%u to cpu%u
\n
"
,
irq
,
desc
->
cpu
,
cpu
);
spin_lock_irq
(
&
irq_controller_lock
);
desc
->
cpu
=
cpu
;
desc
->
chip
->
set_cpu
(
desc
,
irq
,
cpu
);
spin_unlock_irq
(
&
irq_controller_lock
);
}
#ifdef CONFIG_PROC_FS
static
int
irq_affinity_read_proc
(
char
*
page
,
char
**
start
,
off_t
off
,
int
count
,
int
*
eof
,
void
*
data
)
{
struct
irqdesc
*
desc
=
irq_desc
+
((
int
)
data
);
int
len
=
cpumask_scnprintf
(
page
,
count
,
desc
->
affinity
);
if
(
count
-
len
<
2
)
return
-
EINVAL
;
page
[
len
++
]
=
'\n'
;
page
[
len
]
=
'\0'
;
return
len
;
}
static
int
irq_affinity_write_proc
(
struct
file
*
file
,
const
char
__user
*
buffer
,
unsigned
long
count
,
void
*
data
)
{
unsigned
int
irq
=
(
unsigned
int
)
data
;
struct
irqdesc
*
desc
=
irq_desc
+
irq
;
cpumask_t
affinity
,
tmp
;
int
ret
=
-
EIO
;
if
(
!
desc
->
chip
->
set_cpu
)
goto
out
;
ret
=
cpumask_parse
(
buffer
,
count
,
affinity
);
if
(
ret
)
goto
out
;
cpus_and
(
tmp
,
affinity
,
cpu_online_map
);
if
(
cpus_empty
(
tmp
))
{
ret
=
-
EINVAL
;
goto
out
;
}
desc
->
affinity
=
affinity
;
route_irq
(
desc
,
irq
,
first_cpu
(
tmp
));
ret
=
count
;
out:
return
ret
;
}
#endif
#endif
void
__init
init_irq_proc
(
void
)
void
__init
init_irq_proc
(
void
)
{
{
#if defined(CONFIG_SMP) && defined(CONFIG_PROC_FS)
struct
proc_dir_entry
*
dir
;
int
irq
;
dir
=
proc_mkdir
(
"irq"
,
0
);
if
(
!
dir
)
return
;
for
(
irq
=
0
;
irq
<
NR_IRQS
;
irq
++
)
{
struct
proc_dir_entry
*
entry
;
struct
irqdesc
*
desc
;
char
name
[
16
];
desc
=
irq_desc
+
irq
;
memset
(
name
,
0
,
sizeof
(
name
));
snprintf
(
name
,
sizeof
(
name
)
-
1
,
"%u"
,
irq
);
desc
->
procdir
=
proc_mkdir
(
name
,
dir
);
if
(
!
desc
->
procdir
)
continue
;
entry
=
create_proc_entry
(
"smp_affinity"
,
0600
,
desc
->
procdir
);
if
(
entry
)
{
entry
->
nlink
=
1
;
entry
->
data
=
(
void
*
)
irq
;
entry
->
read_proc
=
irq_affinity_read_proc
;
entry
->
write_proc
=
irq_affinity_write_proc
;
}
}
#endif
}
}
void
__init
init_IRQ
(
void
)
void
__init
init_IRQ
(
void
)
...
@@ -888,6 +1006,11 @@ void __init init_IRQ(void)
...
@@ -888,6 +1006,11 @@ void __init init_IRQ(void)
extern
void
init_dma
(
void
);
extern
void
init_dma
(
void
);
int
irq
;
int
irq
;
#ifdef CONFIG_SMP
bad_irq_desc
.
affinity
=
CPU_MASK_ALL
;
bad_irq_desc
.
cpu
=
smp_processor_id
();
#endif
for
(
irq
=
0
,
desc
=
irq_desc
;
irq
<
NR_IRQS
;
irq
++
,
desc
++
)
{
for
(
irq
=
0
,
desc
=
irq_desc
;
irq
<
NR_IRQS
;
irq
++
,
desc
++
)
{
*
desc
=
bad_irq_desc
;
*
desc
=
bad_irq_desc
;
INIT_LIST_HEAD
(
&
desc
->
pend
);
INIT_LIST_HEAD
(
&
desc
->
pend
);
...
...
arch/arm/kernel/smp.c
View file @
3f371438
...
@@ -17,16 +17,16 @@
...
@@ -17,16 +17,16 @@
#include <linux/profile.h>
#include <linux/profile.h>
#include <linux/errno.h>
#include <linux/errno.h>
#include <linux/mm.h>
#include <linux/mm.h>
#include <linux/cpu.h>
#include <linux/smp.h>
#include <linux/seq_file.h>
#include <linux/seq_file.h>
#include <asm/atomic.h>
#include <asm/atomic.h>
#include <asm/cacheflush.h>
#include <asm/cpu.h>
#include <asm/cpu.h>
#include <asm/processor.h>
#include <asm/processor.h>
#include <asm/smp.h>
#include <asm/ptrace.h>
#include <asm/cacheflush.h>
#include <asm/tlbflush.h>
#include <asm/tlbflush.h>
#include <asm/ptrace.h>
/*
/*
* bitmask of present and online CPUs.
* bitmask of present and online CPUs.
...
@@ -42,6 +42,7 @@ cpumask_t cpu_online_map;
...
@@ -42,6 +42,7 @@ cpumask_t cpu_online_map;
*/
*/
struct
ipi_data
{
struct
ipi_data
{
spinlock_t
lock
;
spinlock_t
lock
;
unsigned
long
ipi_count
;
unsigned
long
bits
;
unsigned
long
bits
;
};
};
...
@@ -242,12 +243,12 @@ int smp_call_function(void (*func)(void *info), void *info, int retry,
...
@@ -242,12 +243,12 @@ int smp_call_function(void (*func)(void *info), void *info, int retry,
void
show_ipi_list
(
struct
seq_file
*
p
)
void
show_ipi_list
(
struct
seq_file
*
p
)
{
{
int
cpu
;
unsigned
int
cpu
;
seq_p
rintf
(
p
,
"IPI:
"
);
seq_p
uts
(
p
,
"IPI:
"
);
for_each_online_cpu
(
cpu
)
for_each_online_cpu
(
cpu
)
seq_printf
(
p
,
"
%10lu "
,
per_cpu
(
cpu
_data
,
cpu
).
ipi_count
);
seq_printf
(
p
,
"
%10lu"
,
per_cpu
(
ipi
_data
,
cpu
).
ipi_count
);
seq_putc
(
p
,
'\n'
);
seq_putc
(
p
,
'\n'
);
}
}
...
@@ -316,12 +317,11 @@ static void ipi_cpu_stop(unsigned int cpu)
...
@@ -316,12 +317,11 @@ static void ipi_cpu_stop(unsigned int cpu)
void
do_IPI
(
unsigned
int
ipimask
,
struct
pt_regs
*
regs
)
void
do_IPI
(
unsigned
int
ipimask
,
struct
pt_regs
*
regs
)
{
{
unsigned
int
cpu
=
smp_processor_id
();
unsigned
int
cpu
=
smp_processor_id
();
struct
ipi_data
*
ipi
=
&
per_cpu
(
ipi_data
,
cpu
);
per_cpu
(
cpu_data
,
cpu
).
ipi_count
++
;
ipi
->
ipi_count
++
;
if
(
ipimask
&
(
1
<<
0
))
{
if
(
ipimask
&
(
1
<<
0
))
{
struct
ipi_data
*
ipi
=
&
per_cpu
(
ipi_data
,
cpu
);
for
(;;)
{
for
(;;)
{
unsigned
long
msgs
;
unsigned
long
msgs
;
...
...
include/asm-arm/cpu.h
View file @
3f371438
...
@@ -17,7 +17,6 @@ struct cpuinfo_arm {
...
@@ -17,7 +17,6 @@ struct cpuinfo_arm {
struct
cpu
cpu
;
struct
cpu
cpu
;
#ifdef CONFIG_SMP
#ifdef CONFIG_SMP
unsigned
int
loops_per_jiffy
;
unsigned
int
loops_per_jiffy
;
unsigned
long
ipi_count
;
#endif
#endif
};
};
...
...
include/asm-arm/mach/irq.h
View file @
3f371438
...
@@ -47,6 +47,13 @@ struct irqchip {
...
@@ -47,6 +47,13 @@ struct irqchip {
* Set wakeup-enable on the selected IRQ
* Set wakeup-enable on the selected IRQ
*/
*/
int
(
*
wake
)(
unsigned
int
,
unsigned
int
);
int
(
*
wake
)(
unsigned
int
,
unsigned
int
);
#ifdef CONFIG_SMP
/*
* Route an interrupt to a CPU
*/
void
(
*
set_cpu
)(
struct
irqdesc
*
desc
,
unsigned
int
irq
,
unsigned
int
cpu
);
#endif
};
};
struct
irqdesc
{
struct
irqdesc
{
...
@@ -67,6 +74,13 @@ struct irqdesc {
...
@@ -67,6 +74,13 @@ struct irqdesc {
unsigned
int
noautoenable
:
1
;
/* don't automatically enable IRQ */
unsigned
int
noautoenable
:
1
;
/* don't automatically enable IRQ */
unsigned
int
unused
:
25
;
unsigned
int
unused
:
25
;
struct
proc_dir_entry
*
procdir
;
#ifdef CONFIG_SMP
cpumask_t
affinity
;
unsigned
int
cpu
;
#endif
/*
/*
* IRQ lock detection
* IRQ lock detection
*/
*/
...
...
include/asm-arm/smp.h
View file @
3f371438
...
@@ -32,6 +32,13 @@ extern cpumask_t cpu_present_mask;
...
@@ -32,6 +32,13 @@ extern cpumask_t cpu_present_mask;
*/
*/
#define PROC_CHANGE_PENALTY 15
#define PROC_CHANGE_PENALTY 15
struct
seq_file
;
/*
* generate IPI list text
*/
extern
void
show_ipi_list
(
struct
seq_file
*
p
);
/*
/*
* Move global data into per-processor storage.
* Move global data into per-processor storage.
*/
*/
...
...
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