Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
OpenHarmony
kernel_linux
提交
b03ed6f2
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看板
提交
b03ed6f2
编写于
10月 24, 2006
作者:
=
=?utf-8?q?Michel_D=C3=A4nzer?=
提交者:
airlied
12月 07, 2006
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
drm: drawable locking + memory management fixes + copyright
Signed-off-by:
N
Dave Airlie
<
airlied@linux.ie
>
上级
2e54a007
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
155 addition
and
107 deletion
+155
-107
drivers/char/drm/drm_drawable.c
drivers/char/drm/drm_drawable.c
+155
-107
未找到文件。
drivers/char/drm/drm_drawable.c
浏览文件 @
b03ed6f2
...
@@ -12,6 +12,7 @@
...
@@ -12,6 +12,7 @@
*
*
* Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
* Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
* Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
* Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
* Copyright 2006 Tungsten Graphics, Inc., Bismarck, North Dakota.
* All Rights Reserved.
* All Rights Reserved.
*
*
* Permission is hereby granted, free of charge, to any person obtaining a
* Permission is hereby granted, free of charge, to any person obtaining a
...
@@ -36,70 +37,86 @@
...
@@ -36,70 +37,86 @@
#include "drmP.h"
#include "drmP.h"
/** No-op. */
/**
* Allocate drawable ID and memory to store information about it.
*/
int
drm_adddraw
(
DRM_IOCTL_ARGS
)
int
drm_adddraw
(
DRM_IOCTL_ARGS
)
{
{
DRM_DEVICE
;
DRM_DEVICE
;
unsigned
long
irqflags
;
unsigned
long
irqflags
;
int
i
,
j
=
0
;
int
i
,
j
;
u32
*
bitfield
=
dev
->
drw_bitfield
;
unsigned
int
bitfield_length
=
dev
->
drw_bitfield_length
;
drm_drawable_info_t
**
info
=
dev
->
drw_info
;
unsigned
int
info_length
=
dev
->
drw_info_length
;
drm_draw_t
draw
;
drm_draw_t
draw
;
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
for
(
i
=
0
,
j
=
0
;
i
<
bitfield_length
;
i
++
)
{
if
(
bitfield
[
i
]
==
~
0
)
for
(
i
=
0
;
i
<
dev
->
drw_bitfield_length
;
i
++
)
{
u32
bitfield
=
dev
->
drw_bitfield
[
i
];
if
(
bitfield
==
~
0
)
continue
;
continue
;
for
(;
j
<
sizeof
(
bitfield
);
j
++
)
for
(;
j
<
8
*
sizeof
(
*
bitfield
);
j
++
)
if
(
!
(
bitfield
&
(
1
<<
j
)))
if
(
!
(
bitfield
[
i
]
&
(
1
<<
j
)))
goto
done
;
goto
done
;
}
}
done:
done:
if
(
i
==
dev
->
drw_bitfield_length
)
{
if
(
i
==
bitfield_length
)
{
u32
*
new_bitfield
=
drm_realloc
(
dev
->
drw_bitfield
,
i
*
4
,
bitfield_length
++
;
(
i
+
1
)
*
4
,
DRM_MEM_BUFS
);
bitfield
=
drm_alloc
(
bitfield_length
*
sizeof
(
*
bitfield
),
DRM_MEM_BUFS
);
if
(
!
new_
bitfield
)
{
if
(
!
bitfield
)
{
DRM_ERROR
(
"Failed to allocate new drawable bitfield
\n
"
);
DRM_ERROR
(
"Failed to allocate new drawable bitfield
\n
"
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
return
DRM_ERR
(
ENOMEM
);
return
DRM_ERR
(
ENOMEM
);
}
}
if
(
32
*
(
i
+
1
)
>
dev
->
drw_info_length
)
{
if
(
8
*
sizeof
(
*
bitfield
)
*
bitfield_length
>
info_length
)
{
void
*
new_info
=
drm_realloc
(
dev
->
drw_info
,
info_length
+=
8
*
sizeof
(
*
bitfield
);
dev
->
drw_info_length
*
sizeof
(
drm_drawable_info_t
*
),
32
*
(
i
+
1
)
*
sizeof
(
drm_drawable_info_t
*
),
DRM_MEM_BUFS
);
if
(
!
new_info
)
{
info
=
drm_alloc
(
info_length
*
sizeof
(
*
info
),
DRM_MEM_BUFS
);
if
(
!
info
)
{
DRM_ERROR
(
"Failed to allocate new drawable info"
DRM_ERROR
(
"Failed to allocate new drawable info"
" array
\n
"
);
" array
\n
"
);
drm_free
(
new_bitfield
,
(
i
+
1
)
*
4
,
DRM_MEM_BUFS
);
drm_free
(
bitfield
,
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
bitfield_length
*
sizeof
(
*
bitfield
),
DRM_MEM_BUFS
);
return
DRM_ERR
(
ENOMEM
);
return
DRM_ERR
(
ENOMEM
);
}
}
dev
->
drw_info
=
(
drm_drawable_info_t
**
)
new_info
;
}
}
new_bitfield
[
i
]
=
0
;
bitfield
[
i
]
=
0
;
dev
->
drw_bitfield
=
new_bitfield
;
dev
->
drw_bitfield_length
++
;
}
}
dev
->
drw_bitfield
[
i
]
|=
1
<<
j
;
draw
.
handle
=
i
*
8
*
sizeof
(
*
bitfield
)
+
j
;
draw
.
handle
=
i
*
sizeof
(
u32
)
+
j
;
DRM_DEBUG
(
"%d
\n
"
,
draw
.
handle
);
DRM_DEBUG
(
"%d
\n
"
,
draw
.
handle
);
dev
->
drw_info
[
draw
.
handle
]
=
NULL
;
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
bitfield
[
i
]
|=
1
<<
j
;
info
[
draw
.
handle
]
=
NULL
;
if
(
bitfield
!=
dev
->
drw_bitfield
)
{
memcpy
(
bitfield
,
dev
->
drw_bitfield
,
dev
->
drw_bitfield_length
*
sizeof
(
*
bitfield
));
drm_free
(
dev
->
drw_bitfield
,
sizeof
(
*
bitfield
)
*
dev
->
drw_bitfield_length
,
DRM_MEM_BUFS
);
dev
->
drw_bitfield
=
bitfield
;
dev
->
drw_bitfield_length
=
bitfield_length
;
}
if
(
info
!=
dev
->
drw_info
)
{
memcpy
(
info
,
dev
->
drw_info
,
dev
->
drw_info_length
*
sizeof
(
*
info
));
drm_free
(
dev
->
drw_info
,
sizeof
(
*
info
)
*
dev
->
drw_info_length
,
DRM_MEM_BUFS
);
dev
->
drw_info
=
info
;
dev
->
drw_info_length
=
info_length
;
}
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
...
@@ -108,63 +125,85 @@ int drm_adddraw(DRM_IOCTL_ARGS)
...
@@ -108,63 +125,85 @@ int drm_adddraw(DRM_IOCTL_ARGS)
return
0
;
return
0
;
}
}
/** No-op. */
/**
* Free drawable ID and memory to store information about it.
*/
int
drm_rmdraw
(
DRM_IOCTL_ARGS
)
int
drm_rmdraw
(
DRM_IOCTL_ARGS
)
{
{
DRM_DEVICE
;
DRM_DEVICE
;
drm_draw_t
draw
;
drm_draw_t
draw
;
unsigned
int
idx
,
mod
;
unsigned
int
idx
,
shift
;
unsigned
long
irqflags
;
unsigned
long
irqflags
;
u32
*
bitfield
=
dev
->
drw_bitfield
;
unsigned
int
bitfield_length
=
dev
->
drw_bitfield_length
;
drm_drawable_info_t
**
info
=
dev
->
drw_info
;
unsigned
int
info_length
=
dev
->
drw_info_length
;
DRM_COPY_FROM_USER_IOCTL
(
draw
,
(
drm_draw_t
__user
*
)
data
,
DRM_COPY_FROM_USER_IOCTL
(
draw
,
(
drm_draw_t
__user
*
)
data
,
sizeof
(
draw
));
sizeof
(
draw
));
idx
=
draw
.
handle
/
32
;
idx
=
draw
.
handle
/
(
8
*
sizeof
(
*
bitfield
))
;
mod
=
draw
.
handle
%
32
;
shift
=
draw
.
handle
%
(
8
*
sizeof
(
*
bitfield
))
;
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
if
(
idx
>=
bitfield_length
||
!
(
bitfield
[
idx
]
&
(
1
<<
shift
)))
{
if
(
idx
>=
dev
->
drw_bitfield_length
||
!
(
dev
->
drw_bitfield
[
idx
]
&
(
1
<<
mod
)))
{
DRM_DEBUG
(
"No such drawable %d
\n
"
,
draw
.
handle
);
DRM_DEBUG
(
"No such drawable %d
\n
"
,
draw
.
handle
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
return
0
;
return
0
;
}
}
dev
->
drw_bitfield
[
idx
]
&=
~
(
1
<<
mod
);
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
bitfield
[
idx
]
&=
~
(
1
<<
shift
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
if
(
idx
==
(
dev
->
drw_bitfield_length
-
1
))
{
/* Can we shrink the arrays? */
while
(
idx
>=
0
&&
!
dev
->
drw_bitfield
[
idx
])
if
(
idx
==
bitfield_length
-
1
)
{
while
(
idx
>=
0
&&
!
bitfield
[
idx
])
--
idx
;
--
idx
;
if
(
idx
!=
draw
.
handle
/
32
)
{
bitfield_length
=
idx
+
1
;
u32
*
new_bitfield
=
drm_realloc
(
dev
->
drw_bitfield
,
dev
->
drw_bitfield_length
*
4
,
(
idx
+
1
)
*
4
,
DRM_MEM_BUFS
);
if
(
new_bitfield
||
idx
==
-
1
)
{
if
(
idx
!=
draw
.
handle
/
(
8
*
sizeof
(
*
bitfield
)))
dev
->
drw_bitfield
=
new_bitfield
;
bitfield
=
drm_alloc
(
bitfield_length
*
dev
->
drw_bitfield_length
=
idx
+
1
;
sizeof
(
*
bitfield
),
DRM_MEM_BUFS
);
}
if
(
!
bitfield
&&
bitfield_length
)
{
bitfield
=
dev
->
drw_bitfield
;
bitfield_length
=
dev
->
drw_bitfield_length
;
}
}
}
}
if
(
32
*
dev
->
drw_bitfield_length
<
dev
->
drw_info_length
)
{
if
(
bitfield
!=
dev
->
drw_bitfield
)
{
void
*
new_info
=
drm_realloc
(
dev
->
drw_info
,
info_length
=
8
*
sizeof
(
*
bitfield
)
*
bitfield_length
;
dev
->
drw_info_length
*
sizeof
(
drm_drawable_info_t
*
),
info
=
drm_alloc
(
info_length
*
sizeof
(
*
info
),
DRM_MEM_BUFS
);
32
*
dev
->
drw_bitfield_length
*
sizeof
(
drm_drawable_info_t
*
),
if
(
!
info
&&
info_length
)
{
DRM_MEM_BUFS
);
info
=
dev
->
drw_info
;
info_length
=
dev
->
drw_info_length
;
if
(
new_info
||
!
dev
->
drw_bitfield_length
)
{
dev
->
drw_info
=
(
drm_drawable_info_t
**
)
new_info
;
dev
->
drw_info_length
=
32
*
dev
->
drw_bitfield_length
;
}
}
}
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
memcpy
(
bitfield
,
dev
->
drw_bitfield
,
bitfield_length
*
sizeof
(
*
bitfield
));
drm_free
(
dev
->
drw_bitfield
,
sizeof
(
*
bitfield
)
*
dev
->
drw_bitfield_length
,
DRM_MEM_BUFS
);
dev
->
drw_bitfield
=
bitfield
;
dev
->
drw_bitfield_length
=
bitfield_length
;
if
(
info
!=
dev
->
drw_info
)
{
memcpy
(
info
,
dev
->
drw_info
,
info_length
*
sizeof
(
*
info
));
drm_free
(
dev
->
drw_info
,
sizeof
(
*
info
)
*
dev
->
drw_info_length
,
DRM_MEM_BUFS
);
dev
->
drw_info
=
info
;
dev
->
drw_info_length
=
info_length
;
}
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
}
DRM_DEBUG
(
"%d
\n
"
,
draw
.
handle
);
DRM_DEBUG
(
"%d
\n
"
,
draw
.
handle
);
return
0
;
return
0
;
...
@@ -173,24 +212,22 @@ int drm_rmdraw(DRM_IOCTL_ARGS)
...
@@ -173,24 +212,22 @@ int drm_rmdraw(DRM_IOCTL_ARGS)
int
drm_update_drawable_info
(
DRM_IOCTL_ARGS
)
{
int
drm_update_drawable_info
(
DRM_IOCTL_ARGS
)
{
DRM_DEVICE
;
DRM_DEVICE
;
drm_update_draw_t
update
;
drm_update_draw_t
update
;
unsigned
int
id
,
idx
,
mod
;
unsigned
int
id
,
idx
,
shift
;
unsigned
long
irqflags
;
u32
*
bitfield
=
dev
->
drw_bitfield
;
unsigned
long
irqflags
,
bitfield_length
=
dev
->
drw_bitfield_length
;
drm_drawable_info_t
*
info
;
drm_drawable_info_t
*
info
;
void
*
new_data
;
drm_clip_rect_t
*
rects
;
int
err
;
DRM_COPY_FROM_USER_IOCTL
(
update
,
(
drm_update_draw_t
__user
*
)
data
,
DRM_COPY_FROM_USER_IOCTL
(
update
,
(
drm_update_draw_t
__user
*
)
data
,
sizeof
(
update
));
sizeof
(
update
));
id
=
update
.
handle
;
id
=
update
.
handle
;
idx
=
id
/
32
;
idx
=
id
/
(
8
*
sizeof
(
*
bitfield
))
;
mod
=
id
%
32
;
shift
=
id
%
(
8
*
sizeof
(
*
bitfield
))
;
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
if
(
idx
>=
bitfield_length
||
!
(
bitfield
[
idx
]
&
(
1
<<
shift
)))
{
if
(
idx
>=
dev
->
drw_bitfield_length
||
!
(
dev
->
drw_bitfield
[
idx
]
&
(
1
<<
mod
)))
{
DRM_ERROR
(
"No such drawable %d
\n
"
,
update
.
handle
);
DRM_ERROR
(
"No such drawable %d
\n
"
,
update
.
handle
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
return
DRM_ERR
(
EINVAL
);
return
DRM_ERR
(
EINVAL
);
}
}
...
@@ -201,66 +238,77 @@ int drm_update_drawable_info(DRM_IOCTL_ARGS) {
...
@@ -201,66 +238,77 @@ int drm_update_drawable_info(DRM_IOCTL_ARGS) {
if
(
!
info
)
{
if
(
!
info
)
{
DRM_ERROR
(
"Failed to allocate drawable info memory
\n
"
);
DRM_ERROR
(
"Failed to allocate drawable info memory
\n
"
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
return
DRM_ERR
(
ENOMEM
);
return
DRM_ERR
(
ENOMEM
);
}
}
dev
->
drw_info
[
id
]
=
info
;
}
}
switch
(
update
.
type
)
{
switch
(
update
.
type
)
{
case
DRM_DRAWABLE_CLIPRECTS
:
case
DRM_DRAWABLE_CLIPRECTS
:
if
(
update
.
num
!=
info
->
num_rects
)
{
if
(
update
.
num
!=
info
->
num_rects
)
{
new_data
=
drm_alloc
(
update
.
num
*
rects
=
drm_alloc
(
update
.
num
*
sizeof
(
drm_clip_rect_t
),
sizeof
(
drm_clip_rect_t
),
DRM_MEM_BUFS
);
DRM_MEM_BUFS
);
}
else
rects
=
info
->
rects
;
if
(
!
new_data
)
{
DRM_ERROR
(
"Can't allocate cliprect memory
\n
"
);
if
(
update
.
num
&&
!
rects
)
{
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
DRM_ERROR
(
"Failed to allocate cliprect memory
\n
"
);
return
DRM_ERR
(
ENOMEM
);
err
=
DRM_ERR
(
ENOMEM
);
}
goto
error
;
info
->
rects
=
new_data
;
}
}
if
(
DRM_COPY_FROM_USER
(
info
->
rects
,
if
(
update
.
num
&&
DRM_COPY_FROM_USER
(
rects
,
(
drm_clip_rect_t
__user
*
)
(
drm_clip_rect_t
__user
*
)
(
unsigned
long
)
update
.
data
,
(
unsigned
long
)
update
.
data
,
update
.
num
*
sizeof
(
drm_clip_rect_t
)))
{
update
.
num
*
DRM_ERROR
(
"Can't copy cliprects from userspace
\n
"
);
sizeof
(
*
rects
)))
{
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
DRM_ERROR
(
"Failed to copy cliprects from userspace
\n
"
);
return
DRM_ERR
(
EFAULT
);
err
=
DRM_ERR
(
EFAULT
);
goto
error
;
}
}
if
(
update
.
num
!=
info
->
num_rects
)
{
spin_lock_irqsave
(
&
dev
->
drw_lock
,
irqflags
);
if
(
rects
!=
info
->
rects
)
{
drm_free
(
info
->
rects
,
info
->
num_rects
*
drm_free
(
info
->
rects
,
info
->
num_rects
*
sizeof
(
drm_clip_rect_t
),
DRM_MEM_BUFS
);
sizeof
(
drm_clip_rect_t
),
DRM_MEM_BUFS
);
info
->
num_rects
=
update
.
num
;
}
}
info
->
rects
=
rects
;
info
->
num_rects
=
update
.
num
;
dev
->
drw_info
[
id
]
=
info
;
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
DRM_DEBUG
(
"Updated %d cliprects for drawable %d
\n
"
,
DRM_DEBUG
(
"Updated %d cliprects for drawable %d
\n
"
,
info
->
num_rects
,
id
);
info
->
num_rects
,
id
);
break
;
break
;
default:
default:
DRM_ERROR
(
"Invalid update type %d
\n
"
,
update
.
type
);
DRM_ERROR
(
"Invalid update type %d
\n
"
,
update
.
type
);
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
return
DRM_ERR
(
EINVAL
);
return
DRM_ERR
(
EINVAL
);
}
}
spin_unlock_irqrestore
(
&
dev
->
drw_lock
,
irqflags
);
return
0
;
return
0
;
error:
if
(
!
dev
->
drw_info
[
id
])
drm_free
(
info
,
sizeof
(
*
info
),
DRM_MEM_BUFS
);
else
if
(
rects
!=
dev
->
drw_info
[
id
]
->
rects
)
drm_free
(
rects
,
update
.
num
*
sizeof
(
drm_clip_rect_t
),
DRM_MEM_BUFS
);
return
err
;
}
}
/**
/**
* Caller must hold the drawable spinlock!
* Caller must hold the drawable spinlock!
*/
*/
drm_drawable_info_t
*
drm_get_drawable_info
(
drm_device_t
*
dev
,
drm_drawable_t
id
)
{
drm_drawable_info_t
*
drm_get_drawable_info
(
drm_device_t
*
dev
,
drm_drawable_t
id
)
{
unsigned
int
idx
=
id
/
32
,
mod
=
id
%
32
;
u32
*
bitfield
=
dev
->
drw_bitfield
;
unsigned
int
idx
=
id
/
(
8
*
sizeof
(
*
bitfield
));
unsigned
int
shift
=
id
%
(
8
*
sizeof
(
*
bitfield
));
if
(
idx
>=
dev
->
drw_bitfield_length
||
if
(
idx
>=
dev
->
drw_bitfield_length
||
!
(
dev
->
drw_bitfield
[
idx
]
&
(
1
<<
mod
)))
{
!
(
bitfield
[
idx
]
&
(
1
<<
shift
)))
{
DRM_DEBUG
(
"No such drawable %d
\n
"
,
id
);
DRM_DEBUG
(
"No such drawable %d
\n
"
,
id
);
return
NULL
;
return
NULL
;
}
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录