Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
Kernel
提交
45a4a64a
K
Kernel
项目概览
openeuler
/
Kernel
1 年多 前同步成功
通知
8
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
K
Kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
45a4a64a
编写于
7月 09, 2017
作者:
A
Al Viro
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
semctl(): separate all layout-dependent copyin/copyout
Signed-off-by:
N
Al Viro
<
viro@zeniv.linux.org.uk
>
上级
46939168
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
94 addition
and
98 deletion
+94
-98
ipc/sem.c
ipc/sem.c
+94
-98
未找到文件。
ipc/sem.c
浏览文件 @
45a4a64a
...
...
@@ -1177,112 +1177,95 @@ static time_t get_semotime(struct sem_array *sma)
return
res
;
}
static
int
semctl_
nolock
(
struct
ipc_namespace
*
ns
,
int
semid
,
int
cmd
,
int
version
,
void
__user
*
p
)
static
int
semctl_
stat
(
struct
ipc_namespace
*
ns
,
int
semid
,
int
cmd
,
struct
semid64_ds
*
semid64
)
{
int
err
;
struct
sem_array
*
sma
;
int
id
=
0
;
int
err
;
switch
(
cmd
)
{
case
IPC_INFO
:
case
SEM_INFO
:
{
struct
seminfo
seminfo
;
int
max_id
;
err
=
security_sem_semctl
(
NULL
,
cmd
);
if
(
err
)
return
err
;
memset
(
semid64
,
0
,
sizeof
(
*
semid64
));
memset
(
&
seminfo
,
0
,
sizeof
(
seminfo
));
seminfo
.
semmni
=
ns
->
sc_semmni
;
seminfo
.
semmns
=
ns
->
sc_semmns
;
seminfo
.
semmsl
=
ns
->
sc_semmsl
;
seminfo
.
semopm
=
ns
->
sc_semopm
;
seminfo
.
semvmx
=
SEMVMX
;
seminfo
.
semmnu
=
SEMMNU
;
seminfo
.
semmap
=
SEMMAP
;
seminfo
.
semume
=
SEMUME
;
down_read
(
&
sem_ids
(
ns
).
rwsem
);
if
(
cmd
==
SEM_INFO
)
{
seminfo
.
semusz
=
sem_ids
(
ns
).
in_use
;
seminfo
.
semaem
=
ns
->
used_sems
;
}
else
{
seminfo
.
semusz
=
SEMUSZ
;
seminfo
.
semaem
=
SEMAEM
;
rcu_read_lock
();
if
(
cmd
==
SEM_STAT
)
{
sma
=
sem_obtain_object
(
ns
,
semid
);
if
(
IS_ERR
(
sma
))
{
err
=
PTR_ERR
(
sma
);
goto
out_unlock
;
}
id
=
sma
->
sem_perm
.
id
;
}
else
{
sma
=
sem_obtain_object_check
(
ns
,
semid
);
if
(
IS_ERR
(
sma
))
{
err
=
PTR_ERR
(
sma
);
goto
out_unlock
;
}
max_id
=
ipc_get_maxid
(
&
sem_ids
(
ns
));
up_read
(
&
sem_ids
(
ns
).
rwsem
);
if
(
copy_to_user
(
p
,
&
seminfo
,
sizeof
(
struct
seminfo
)))
return
-
EFAULT
;
return
(
max_id
<
0
)
?
0
:
max_id
;
}
case
IPC_STAT
:
case
SEM_STAT
:
{
struct
semid64_ds
tbuf
;
int
id
=
0
;
memset
(
&
tbuf
,
0
,
sizeof
(
tbuf
));
rcu_read_lock
();
if
(
cmd
==
SEM_STAT
)
{
sma
=
sem_obtain_object
(
ns
,
semid
);
if
(
IS_ERR
(
sma
))
{
err
=
PTR_ERR
(
sma
);
goto
out_unlock
;
}
id
=
sma
->
sem_perm
.
id
;
}
else
{
sma
=
sem_obtain_object_check
(
ns
,
semid
);
if
(
IS_ERR
(
sma
))
{
err
=
PTR_ERR
(
sma
);
goto
out_unlock
;
}
}
err
=
-
EACCES
;
if
(
ipcperms
(
ns
,
&
sma
->
sem_perm
,
S_IRUGO
))
goto
out_unlock
;
err
=
-
EACCES
;
if
(
ipcperms
(
ns
,
&
sma
->
sem_perm
,
S_IRUGO
)
)
goto
out_unlock
;
err
=
security_sem_semctl
(
sma
,
cmd
)
;
if
(
err
)
goto
out_unlock
;
err
=
security_sem_semctl
(
sma
,
cmd
);
if
(
err
)
goto
out_unlock
;
kernel_to_ipc64_perm
(
&
sma
->
sem_perm
,
&
semid64
->
sem_perm
);
semid64
->
sem_otime
=
get_semotime
(
sma
);
semid64
->
sem_ctime
=
sma
->
sem_ctime
;
semid64
->
sem_nsems
=
sma
->
sem_nsems
;
rcu_read_unlock
();
return
id
;
kernel_to_ipc64_perm
(
&
sma
->
sem_perm
,
&
tbuf
.
sem_perm
);
tbuf
.
sem_otime
=
get_semotime
(
sma
);
tbuf
.
sem_ctime
=
sma
->
sem_ctime
;
tbuf
.
sem_nsems
=
sma
->
sem_nsems
;
rcu_read_unlock
();
if
(
copy_semid_to_user
(
p
,
&
tbuf
,
version
))
return
-
EFAULT
;
return
id
;
}
default:
return
-
EINVAL
;
}
out_unlock:
rcu_read_unlock
();
return
err
;
}
static
int
semctl_info
(
struct
ipc_namespace
*
ns
,
int
semid
,
int
cmd
,
void
__user
*
p
)
{
struct
seminfo
seminfo
;
int
max_id
;
int
err
;
err
=
security_sem_semctl
(
NULL
,
cmd
);
if
(
err
)
return
err
;
memset
(
&
seminfo
,
0
,
sizeof
(
seminfo
));
seminfo
.
semmni
=
ns
->
sc_semmni
;
seminfo
.
semmns
=
ns
->
sc_semmns
;
seminfo
.
semmsl
=
ns
->
sc_semmsl
;
seminfo
.
semopm
=
ns
->
sc_semopm
;
seminfo
.
semvmx
=
SEMVMX
;
seminfo
.
semmnu
=
SEMMNU
;
seminfo
.
semmap
=
SEMMAP
;
seminfo
.
semume
=
SEMUME
;
down_read
(
&
sem_ids
(
ns
).
rwsem
);
if
(
cmd
==
SEM_INFO
)
{
seminfo
.
semusz
=
sem_ids
(
ns
).
in_use
;
seminfo
.
semaem
=
ns
->
used_sems
;
}
else
{
seminfo
.
semusz
=
SEMUSZ
;
seminfo
.
semaem
=
SEMAEM
;
}
max_id
=
ipc_get_maxid
(
&
sem_ids
(
ns
));
up_read
(
&
sem_ids
(
ns
).
rwsem
);
if
(
copy_to_user
(
p
,
&
seminfo
,
sizeof
(
struct
seminfo
)))
return
-
EFAULT
;
return
(
max_id
<
0
)
?
0
:
max_id
;
}
static
int
semctl_setval
(
struct
ipc_namespace
*
ns
,
int
semid
,
int
semnum
,
unsigned
long
arg
)
int
val
)
{
struct
sem_undo
*
un
;
struct
sem_array
*
sma
;
struct
sem
*
curr
;
int
err
,
val
;
int
err
;
DEFINE_WAKE_Q
(
wake_q
);
#if defined(CONFIG_64BIT) && defined(__BIG_ENDIAN)
/* big-endian 64bit */
val
=
arg
>>
32
;
#else
/* 32bit or little-endian 64bit */
val
=
arg
;
#endif
if
(
val
>
SEMVMX
||
val
<
0
)
return
-
ERANGE
;
...
...
@@ -1531,23 +1514,17 @@ copy_semid_from_user(struct semid64_ds *out, void __user *buf, int version)
* NOTE: no locks must be held, the rwsem is taken inside this function.
*/
static
int
semctl_down
(
struct
ipc_namespace
*
ns
,
int
semid
,
int
cmd
,
int
version
,
void
__user
*
p
)
int
cmd
,
struct
semid64_ds
*
semid64
)
{
struct
sem_array
*
sma
;
int
err
;
struct
semid64_ds
semid64
;
struct
kern_ipc_perm
*
ipcp
;
if
(
cmd
==
IPC_SET
)
{
if
(
copy_semid_from_user
(
&
semid64
,
p
,
version
))
return
-
EFAULT
;
}
down_write
(
&
sem_ids
(
ns
).
rwsem
);
rcu_read_lock
();
ipcp
=
ipcctl_pre_down_nolock
(
ns
,
&
sem_ids
(
ns
),
semid
,
cmd
,
&
semid64
.
sem_perm
,
0
);
&
semid64
->
sem_perm
,
0
);
if
(
IS_ERR
(
ipcp
))
{
err
=
PTR_ERR
(
ipcp
);
goto
out_unlock1
;
...
...
@@ -1567,7 +1544,7 @@ static int semctl_down(struct ipc_namespace *ns, int semid,
goto
out_up
;
case
IPC_SET
:
sem_lock
(
sma
,
NULL
,
-
1
);
err
=
ipc_update_perm
(
&
semid64
.
sem_perm
,
ipcp
);
err
=
ipc_update_perm
(
&
semid64
->
sem_perm
,
ipcp
);
if
(
err
)
goto
out_unlock0
;
sma
->
sem_ctime
=
get_seconds
();
...
...
@@ -1591,6 +1568,8 @@ SYSCALL_DEFINE4(semctl, int, semid, int, semnum, int, cmd, unsigned long, arg)
int
version
;
struct
ipc_namespace
*
ns
;
void
__user
*
p
=
(
void
__user
*
)
arg
;
struct
semid64_ds
semid64
;
int
err
;
if
(
semid
<
0
)
return
-
EINVAL
;
...
...
@@ -1601,9 +1580,15 @@ SYSCALL_DEFINE4(semctl, int, semid, int, semnum, int, cmd, unsigned long, arg)
switch
(
cmd
)
{
case
IPC_INFO
:
case
SEM_INFO
:
return
semctl_info
(
ns
,
semid
,
cmd
,
p
);
case
IPC_STAT
:
case
SEM_STAT
:
return
semctl_nolock
(
ns
,
semid
,
cmd
,
version
,
p
);
err
=
semctl_stat
(
ns
,
semid
,
cmd
,
&
semid64
);
if
(
err
<
0
)
return
err
;
if
(
copy_semid_to_user
(
p
,
&
semid64
,
version
))
err
=
-
EFAULT
;
return
err
;
case
GETALL
:
case
GETVAL
:
case
GETPID
:
...
...
@@ -1611,11 +1596,22 @@ SYSCALL_DEFINE4(semctl, int, semid, int, semnum, int, cmd, unsigned long, arg)
case
GETZCNT
:
case
SETALL
:
return
semctl_main
(
ns
,
semid
,
semnum
,
cmd
,
p
);
case
SETVAL
:
return
semctl_setval
(
ns
,
semid
,
semnum
,
arg
);
case
IPC_RMID
:
case
SETVAL
:
{
int
val
;
#if defined(CONFIG_64BIT) && defined(__BIG_ENDIAN)
/* big-endian 64bit */
val
=
arg
>>
32
;
#else
/* 32bit or little-endian 64bit */
val
=
arg
;
#endif
return
semctl_setval
(
ns
,
semid
,
semnum
,
val
);
}
case
IPC_SET
:
return
semctl_down
(
ns
,
semid
,
cmd
,
version
,
p
);
if
(
copy_semid_from_user
(
&
semid64
,
p
,
version
))
return
-
EFAULT
;
case
IPC_RMID
:
return
semctl_down
(
ns
,
semid
,
cmd
,
&
semid64
);
default:
return
-
EINVAL
;
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录