Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
cloud-kernel
提交
24e8341e
cloud-kernel
项目概览
openanolis
/
cloud-kernel
1 年多 前同步成功
通知
163
Star
36
Fork
7
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
10
列表
看板
标记
里程碑
合并请求
2
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
cloud-kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
10
Issue
10
列表
看板
标记
里程碑
合并请求
2
合并请求
2
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
24e8341e
编写于
2月 05, 2014
作者:
B
Ben Skeggs
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
drm/nvc0/fifo: attempt to recover engines from mmu faults
Signed-off-by:
N
Ben Skeggs
<
bskeggs@redhat.com
>
上级
33f8c6d0
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
103 addition
and
2 deletion
+103
-2
drivers/gpu/drm/nouveau/core/engine/fifo/nvc0.c
drivers/gpu/drm/nouveau/core/engine/fifo/nvc0.c
+103
-2
未找到文件。
drivers/gpu/drm/nouveau/core/engine/fifo/nvc0.c
浏览文件 @
24e8341e
...
...
@@ -41,11 +41,16 @@
struct
nvc0_fifo_priv
{
struct
nouveau_fifo
base
;
struct
work_struct
fault
;
u64
mask
;
struct
{
struct
nouveau_gpuobj
*
mem
[
2
];
int
active
;
wait_queue_head_t
wait
;
}
runlist
;
struct
{
struct
nouveau_gpuobj
*
mem
;
struct
nouveau_vma
bar
;
...
...
@@ -359,6 +364,90 @@ nvc0_fifo_cclass = {
* PFIFO engine
******************************************************************************/
static
inline
int
nvc0_fifo_engidx
(
struct
nvc0_fifo_priv
*
priv
,
u32
engn
)
{
switch
(
engn
)
{
case
NVDEV_ENGINE_GR
:
engn
=
0
;
break
;
case
NVDEV_ENGINE_BSP
:
engn
=
1
;
break
;
case
NVDEV_ENGINE_PPP
:
engn
=
2
;
break
;
case
NVDEV_ENGINE_VP
:
engn
=
3
;
break
;
case
NVDEV_ENGINE_COPY0
:
engn
=
4
;
break
;
case
NVDEV_ENGINE_COPY1
:
engn
=
5
;
break
;
default:
return
-
1
;
}
return
engn
;
}
static
inline
struct
nouveau_engine
*
nvc0_fifo_engine
(
struct
nvc0_fifo_priv
*
priv
,
u32
engn
)
{
switch
(
engn
)
{
case
0
:
engn
=
NVDEV_ENGINE_GR
;
break
;
case
1
:
engn
=
NVDEV_ENGINE_BSP
;
break
;
case
2
:
engn
=
NVDEV_ENGINE_PPP
;
break
;
case
3
:
engn
=
NVDEV_ENGINE_VP
;
break
;
case
4
:
engn
=
NVDEV_ENGINE_COPY0
;
break
;
case
5
:
engn
=
NVDEV_ENGINE_COPY1
;
break
;
default:
return
NULL
;
}
return
nouveau_engine
(
priv
,
engn
);
}
static
void
nvc0_fifo_recover_work
(
struct
work_struct
*
work
)
{
struct
nvc0_fifo_priv
*
priv
=
container_of
(
work
,
typeof
(
*
priv
),
fault
);
struct
nouveau_object
*
engine
;
unsigned
long
flags
;
u32
engn
,
engm
=
0
;
u64
mask
,
todo
;
spin_lock_irqsave
(
&
priv
->
base
.
lock
,
flags
);
mask
=
priv
->
mask
;
priv
->
mask
=
0ULL
;
spin_unlock_irqrestore
(
&
priv
->
base
.
lock
,
flags
);
for
(
todo
=
mask
;
engn
=
__ffs64
(
todo
),
todo
;
todo
&=
~
(
1
<<
engn
))
engm
|=
1
<<
nvc0_fifo_engidx
(
priv
,
engn
);
nv_mask
(
priv
,
0x002630
,
engm
,
engm
);
for
(
todo
=
mask
;
engn
=
__ffs64
(
todo
),
todo
;
todo
&=
~
(
1
<<
engn
))
{
if
((
engine
=
(
void
*
)
nouveau_engine
(
priv
,
engn
)))
{
nv_ofuncs
(
engine
)
->
fini
(
engine
,
false
);
WARN_ON
(
nv_ofuncs
(
engine
)
->
init
(
engine
));
}
}
nvc0_fifo_runlist_update
(
priv
);
nv_wr32
(
priv
,
0x00262c
,
engm
);
nv_mask
(
priv
,
0x002630
,
engm
,
0x00000000
);
}
static
void
nvc0_fifo_recover
(
struct
nvc0_fifo_priv
*
priv
,
struct
nouveau_engine
*
engine
,
struct
nvc0_fifo_chan
*
chan
)
{
struct
nouveau_object
*
engobj
=
nv_object
(
engine
);
u32
chid
=
chan
->
base
.
chid
;
unsigned
long
flags
;
nv_error
(
priv
,
"%s engine fault on channel %d, recovering...
\n
"
,
nv_subdev
(
engine
)
->
name
,
chid
);
nv_mask
(
priv
,
0x003004
+
(
chid
*
0x08
),
0x00000001
,
0x00000000
);
chan
->
state
=
KILLED
;
spin_lock_irqsave
(
&
priv
->
base
.
lock
,
flags
);
priv
->
mask
|=
1ULL
<<
nv_engidx
(
engobj
);
spin_unlock_irqrestore
(
&
priv
->
base
.
lock
,
flags
);
schedule_work
(
&
priv
->
fault
);
}
static
int
nvc0_fifo_swmthd
(
struct
nvc0_fifo_priv
*
priv
,
u32
chid
,
u32
mthd
,
u32
data
)
{
...
...
@@ -477,8 +566,8 @@ nvc0_fifo_intr_fault(struct nvc0_fifo_priv *priv, int unit)
u32
write
=
(
stat
&
0x00000080
);
u32
hub
=
(
stat
&
0x00000040
);
u32
reason
=
(
stat
&
0x0000000f
);
struct
nouveau_object
*
engctx
=
NULL
;
struct
nouveau_engine
*
engine
;
struct
nouveau_object
*
engctx
=
NULL
,
*
object
;
struct
nouveau_engine
*
engine
=
NULL
;
const
struct
nouveau_enum
*
er
,
*
eu
,
*
ec
;
char
erunk
[
6
]
=
""
;
char
euunk
[
6
]
=
""
;
...
...
@@ -528,6 +617,16 @@ nvc0_fifo_intr_fault(struct nvc0_fifo_priv *priv, int unit)
ec
?
ec
->
name
:
ecunk
,
(
u64
)
inst
<<
12
,
nouveau_client_name
(
engctx
));
object
=
engctx
;
while
(
object
)
{
switch
(
nv_mclass
(
object
))
{
case
NVC0_CHANNEL_IND_CLASS
:
nvc0_fifo_recover
(
priv
,
engine
,
(
void
*
)
object
);
break
;
}
object
=
object
->
parent
;
}
nouveau_engctx_put
(
engctx
);
}
...
...
@@ -720,6 +819,8 @@ nvc0_fifo_ctor(struct nouveau_object *parent, struct nouveau_object *engine,
if
(
ret
)
return
ret
;
INIT_WORK
(
&
priv
->
fault
,
nvc0_fifo_recover_work
);
ret
=
nouveau_gpuobj_new
(
nv_object
(
priv
),
NULL
,
0x1000
,
0x1000
,
0
,
&
priv
->
runlist
.
mem
[
0
]);
if
(
ret
)
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录