Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
OpenHarmony
kernel_linux
提交
4f70f7a9
K
kernel_linux
项目概览
OpenHarmony
/
kernel_linux
上一次同步 3 年多
通知
13
Star
8
Fork
2
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
K
kernel_linux
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
4f70f7a9
编写于
8月 12, 2008
作者:
D
David S. Miller
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
sparc64: Implement IRQ stacks.
Signed-off-by:
N
David S. Miller
<
davem@davemloft.net
>
上级
e3445682
变更
8
隐藏空白更改
内联
并排
Showing
8 changed file
with
161 addition
and
30 deletion
+161
-30
arch/sparc/include/asm/irq_64.h
arch/sparc/include/asm/irq_64.h
+4
-0
arch/sparc64/kernel/irq.c
arch/sparc64/kernel/irq.c
+52
-0
arch/sparc64/kernel/kstack.h
arch/sparc64/kernel/kstack.h
+58
-0
arch/sparc64/kernel/process.c
arch/sparc64/kernel/process.c
+7
-20
arch/sparc64/kernel/stacktrace.c
arch/sparc64/kernel/stacktrace.c
+4
-6
arch/sparc64/kernel/traps.c
arch/sparc64/kernel/traps.c
+3
-4
arch/sparc64/lib/mcount.S
arch/sparc64/lib/mcount.S
+22
-0
arch/sparc64/mm/init.c
arch/sparc64/mm/init.c
+11
-0
未找到文件。
arch/sparc/include/asm/irq_64.h
浏览文件 @
4f70f7a9
...
...
@@ -93,4 +93,8 @@ static inline unsigned long get_softint(void)
void
__trigger_all_cpu_backtrace
(
void
);
#define trigger_all_cpu_backtrace() __trigger_all_cpu_backtrace()
extern
void
*
hardirq_stack
[
NR_CPUS
];
extern
void
*
softirq_stack
[
NR_CPUS
];
#define __ARCH_HAS_DO_SOFTIRQ
#endif
arch/sparc64/kernel/irq.c
浏览文件 @
4f70f7a9
...
...
@@ -682,10 +682,32 @@ void ack_bad_irq(unsigned int virt_irq)
ino
,
virt_irq
);
}
void
*
hardirq_stack
[
NR_CPUS
];
void
*
softirq_stack
[
NR_CPUS
];
static
__attribute__
((
always_inline
))
void
*
set_hardirq_stack
(
void
)
{
void
*
orig_sp
,
*
sp
=
hardirq_stack
[
smp_processor_id
()];
__asm__
__volatile__
(
"mov %%sp, %0"
:
"=r"
(
orig_sp
));
if
(
orig_sp
<
sp
||
orig_sp
>
(
sp
+
THREAD_SIZE
))
{
sp
+=
THREAD_SIZE
-
192
-
STACK_BIAS
;
__asm__
__volatile__
(
"mov %0, %%sp"
:
:
"r"
(
sp
));
}
return
orig_sp
;
}
static
__attribute__
((
always_inline
))
void
restore_hardirq_stack
(
void
*
orig_sp
)
{
__asm__
__volatile__
(
"mov %0, %%sp"
:
:
"r"
(
orig_sp
));
}
void
handler_irq
(
int
irq
,
struct
pt_regs
*
regs
)
{
unsigned
long
pstate
,
bucket_pa
;
struct
pt_regs
*
old_regs
;
void
*
orig_sp
;
clear_softint
(
1
<<
irq
);
...
...
@@ -703,6 +725,8 @@ void handler_irq(int irq, struct pt_regs *regs)
"i"
(
PSTATE_IE
)
:
"memory"
);
orig_sp
=
set_hardirq_stack
();
while
(
bucket_pa
)
{
struct
irq_desc
*
desc
;
unsigned
long
next_pa
;
...
...
@@ -719,10 +743,38 @@ void handler_irq(int irq, struct pt_regs *regs)
bucket_pa
=
next_pa
;
}
restore_hardirq_stack
(
orig_sp
);
irq_exit
();
set_irq_regs
(
old_regs
);
}
void
do_softirq
(
void
)
{
unsigned
long
flags
;
if
(
in_interrupt
())
return
;
local_irq_save
(
flags
);
if
(
local_softirq_pending
())
{
void
*
orig_sp
,
*
sp
=
softirq_stack
[
smp_processor_id
()];
sp
+=
THREAD_SIZE
-
192
-
STACK_BIAS
;
__asm__
__volatile__
(
"mov %%sp, %0
\n\t
"
"mov %1, %%sp"
:
"=&r"
(
orig_sp
)
:
"r"
(
sp
));
__do_softirq
();
__asm__
__volatile__
(
"mov %0, %%sp"
:
:
"r"
(
orig_sp
));
}
local_irq_restore
(
flags
);
}
#ifdef CONFIG_HOTPLUG_CPU
void
fixup_irqs
(
void
)
{
...
...
arch/sparc64/kernel/kstack.h
0 → 100644
浏览文件 @
4f70f7a9
#ifndef _KSTACK_H
#define _KSTACK_H
#include <linux/thread_info.h>
#include <linux/sched.h>
#include <asm/ptrace.h>
#include <asm/irq.h>
/* SP must be STACK_BIAS adjusted already. */
static
inline
bool
kstack_valid
(
struct
thread_info
*
tp
,
unsigned
long
sp
)
{
unsigned
long
base
=
(
unsigned
long
)
tp
;
if
(
sp
>=
(
base
+
sizeof
(
struct
thread_info
))
&&
sp
<=
(
base
+
THREAD_SIZE
-
sizeof
(
struct
sparc_stackf
)))
return
true
;
base
=
(
unsigned
long
)
hardirq_stack
[
tp
->
cpu
];
if
(
sp
>=
base
&&
sp
<=
(
base
+
THREAD_SIZE
-
sizeof
(
struct
sparc_stackf
)))
return
true
;
base
=
(
unsigned
long
)
softirq_stack
[
tp
->
cpu
];
if
(
sp
>=
base
&&
sp
<=
(
base
+
THREAD_SIZE
-
sizeof
(
struct
sparc_stackf
)))
return
true
;
return
false
;
}
/* Does "regs" point to a valid pt_regs trap frame? */
static
inline
bool
kstack_is_trap_frame
(
struct
thread_info
*
tp
,
struct
pt_regs
*
regs
)
{
unsigned
long
base
=
(
unsigned
long
)
tp
;
unsigned
long
addr
=
(
unsigned
long
)
regs
;
if
(
addr
>=
base
&&
addr
<=
(
base
+
THREAD_SIZE
-
sizeof
(
*
regs
)))
goto
check_magic
;
base
=
(
unsigned
long
)
hardirq_stack
[
tp
->
cpu
];
if
(
addr
>=
base
&&
addr
<=
(
base
+
THREAD_SIZE
-
sizeof
(
*
regs
)))
goto
check_magic
;
base
=
(
unsigned
long
)
softirq_stack
[
tp
->
cpu
];
if
(
addr
>=
base
&&
addr
<=
(
base
+
THREAD_SIZE
-
sizeof
(
*
regs
)))
goto
check_magic
;
return
false
;
check_magic:
if
((
regs
->
magic
&
~
0x1ff
)
==
PT_REGS_MAGIC
)
return
true
;
return
false
;
}
#endif
/* _KSTACK_H */
arch/sparc64/kernel/process.c
浏览文件 @
4f70f7a9
...
...
@@ -52,6 +52,8 @@
#include <asm/irq_regs.h>
#include <asm/smp.h>
#include "kstack.h"
static
void
sparc64_yield
(
int
cpu
)
{
if
(
tlb_type
!=
hypervisor
)
...
...
@@ -235,19 +237,6 @@ void show_regs(struct pt_regs *regs)
struct
global_reg_snapshot
global_reg_snapshot
[
NR_CPUS
];
static
DEFINE_SPINLOCK
(
global_reg_snapshot_lock
);
static
bool
kstack_valid
(
struct
thread_info
*
tp
,
struct
reg_window
*
rw
)
{
unsigned
long
thread_base
,
fp
;
thread_base
=
(
unsigned
long
)
tp
;
fp
=
(
unsigned
long
)
rw
;
if
(
fp
<
(
thread_base
+
sizeof
(
struct
thread_info
))
||
fp
>=
(
thread_base
+
THREAD_SIZE
))
return
false
;
return
true
;
}
static
void
__global_reg_self
(
struct
thread_info
*
tp
,
struct
pt_regs
*
regs
,
int
this_cpu
)
{
...
...
@@ -264,11 +253,11 @@ static void __global_reg_self(struct thread_info *tp, struct pt_regs *regs,
rw
=
(
struct
reg_window
*
)
(
regs
->
u_regs
[
UREG_FP
]
+
STACK_BIAS
);
if
(
kstack_valid
(
tp
,
rw
))
{
if
(
kstack_valid
(
tp
,
(
unsigned
long
)
rw
))
{
global_reg_snapshot
[
this_cpu
].
i7
=
rw
->
ins
[
7
];
rw
=
(
struct
reg_window
*
)
(
rw
->
ins
[
6
]
+
STACK_BIAS
);
if
(
kstack_valid
(
tp
,
rw
))
if
(
kstack_valid
(
tp
,
(
unsigned
long
)
rw
))
global_reg_snapshot
[
this_cpu
].
rpc
=
rw
->
ins
[
7
];
}
}
else
{
...
...
@@ -828,7 +817,7 @@ asmlinkage int sparc_execve(struct pt_regs *regs)
unsigned
long
get_wchan
(
struct
task_struct
*
task
)
{
unsigned
long
pc
,
fp
,
bias
=
0
;
unsigned
long
thread_info_base
;
struct
thread_info
*
tp
;
struct
reg_window
*
rw
;
unsigned
long
ret
=
0
;
int
count
=
0
;
...
...
@@ -837,14 +826,12 @@ unsigned long get_wchan(struct task_struct *task)
task
->
state
==
TASK_RUNNING
)
goto
out
;
t
hread_info_base
=
(
unsigned
long
)
task_stack_page
(
task
);
t
p
=
task_thread_info
(
task
);
bias
=
STACK_BIAS
;
fp
=
task_thread_info
(
task
)
->
ksp
+
bias
;
do
{
/* Bogus frame pointer? */
if
(
fp
<
(
thread_info_base
+
sizeof
(
struct
thread_info
))
||
fp
>=
(
thread_info_base
+
THREAD_SIZE
))
if
(
!
kstack_valid
(
tp
,
fp
))
break
;
rw
=
(
struct
reg_window
*
)
fp
;
pc
=
rw
->
ins
[
7
];
...
...
arch/sparc64/kernel/stacktrace.c
浏览文件 @
4f70f7a9
...
...
@@ -5,6 +5,8 @@
#include <asm/ptrace.h>
#include <asm/stacktrace.h>
#include "kstack.h"
void
save_stack_trace
(
struct
stack_trace
*
trace
)
{
unsigned
long
ksp
,
fp
,
thread_base
;
...
...
@@ -24,17 +26,13 @@ void save_stack_trace(struct stack_trace *trace)
struct
pt_regs
*
regs
;
unsigned
long
pc
;
/* Bogus frame pointer? */
if
(
fp
<
(
thread_base
+
sizeof
(
struct
thread_info
))
||
fp
>
(
thread_base
+
THREAD_SIZE
-
sizeof
(
struct
sparc_stackf
)))
if
(
!
kstack_valid
(
tp
,
fp
))
break
;
sf
=
(
struct
sparc_stackf
*
)
fp
;
regs
=
(
struct
pt_regs
*
)
(
sf
+
1
);
if
(((
unsigned
long
)
regs
<=
(
thread_base
+
THREAD_SIZE
-
sizeof
(
*
regs
)))
&&
(
regs
->
magic
&
~
0x1ff
)
==
PT_REGS_MAGIC
)
{
if
(
kstack_is_trap_frame
(
tp
,
regs
))
{
if
(
!
(
regs
->
tstate
&
TSTATE_PRIV
))
break
;
pc
=
regs
->
tpc
;
...
...
arch/sparc64/kernel/traps.c
浏览文件 @
4f70f7a9
...
...
@@ -39,6 +39,7 @@
#include <asm/prom.h>
#include "entry.h"
#include "kstack.h"
/* When an irrecoverable trap occurs at tl > 0, the trap entry
* code logs the trap state registers at every level in the trap
...
...
@@ -2115,14 +2116,12 @@ void show_stack(struct task_struct *tsk, unsigned long *_ksp)
struct
pt_regs
*
regs
;
unsigned
long
pc
;
/* Bogus frame pointer? */
if
(
fp
<
(
thread_base
+
sizeof
(
struct
thread_info
))
||
fp
>=
(
thread_base
+
THREAD_SIZE
))
if
(
!
kstack_valid
(
tp
,
fp
))
break
;
sf
=
(
struct
sparc_stackf
*
)
fp
;
regs
=
(
struct
pt_regs
*
)
(
sf
+
1
);
if
(
(
regs
->
magic
&
~
0x1ff
)
==
PT_REGS_MAGIC
)
{
if
(
kstack_is_trap_frame
(
tp
,
regs
)
)
{
if
(
!
(
regs
->
tstate
&
TSTATE_PRIV
))
break
;
pc
=
regs
->
tpc
;
...
...
arch/sparc64/lib/mcount.S
浏览文件 @
4f70f7a9
...
...
@@ -49,6 +49,28 @@ mcount:
cmp
%
sp
,
%
g3
bg
,
pt
%
xcc
,
1
f
nop
lduh
[%
g6
+
TI_CPU
],
%
g1
sethi
%
hi
(
hardirq_stack
),
%
g3
or
%
g3
,
%
lo
(
hardirq_stack
),
%
g3
sllx
%
g1
,
3
,
%
g1
ldx
[%
g3
+
%
g1
],
%
g7
sub
%
g7
,
STACK_BIAS
,
%
g7
cmp
%
sp
,
%
g7
bleu
,
pt
%
xcc
,
2
f
sethi
%
hi
(
THREAD_SIZE
),
%
g3
add
%
g7
,
%
g3
,
%
g7
cmp
%
sp
,
%
g7
blu
,
pn
%
xcc
,
1
f
2
:
sethi
%
hi
(
softirq_stack
),
%
g3
or
%
g3
,
%
lo
(
softirq_stack
),
%
g3
ldx
[%
g3
+
%
g1
],
%
g7
cmp
%
sp
,
%
g7
bleu
,
pt
%
xcc
,
2
f
sethi
%
hi
(
THREAD_SIZE
),
%
g3
add
%
g7
,
%
g3
,
%
g7
cmp
%
sp
,
%
g7
blu
,
pn
%
xcc
,
1
f
nop
/
*
If
we
are
already
on
ovstack
,
don
't hop onto it
*
again
,
we
are
already
trying
to
output
the
stack
overflow
*
message
.
...
...
arch/sparc64/mm/init.c
浏览文件 @
4f70f7a9
...
...
@@ -49,6 +49,7 @@
#include <asm/sstate.h>
#include <asm/mdesc.h>
#include <asm/cpudata.h>
#include <asm/irq.h>
#define MAX_PHYS_ADDRESS (1UL << 42UL)
#define KPTE_BITMAP_CHUNK_SZ (256UL * 1024UL * 1024UL)
...
...
@@ -1771,6 +1772,16 @@ void __init paging_init(void)
if
(
tlb_type
==
hypervisor
)
sun4v_mdesc_init
();
/* Once the OF device tree and MDESC have been setup, we know
* the list of possible cpus. Therefore we can allocate the
* IRQ stacks.
*/
for_each_possible_cpu
(
i
)
{
/* XXX Use node local allocations... XXX */
softirq_stack
[
i
]
=
__va
(
lmb_alloc
(
THREAD_SIZE
,
THREAD_SIZE
));
hardirq_stack
[
i
]
=
__va
(
lmb_alloc
(
THREAD_SIZE
,
THREAD_SIZE
));
}
/* Setup bootmem... */
last_valid_pfn
=
end_pfn
=
bootmem_init
(
phys_base
);
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录