Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
cloud-kernel
提交
ebf99093
cloud-kernel
项目概览
openanolis
/
cloud-kernel
1 年多 前同步成功
通知
161
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看板
提交
ebf99093
编写于
6月 04, 2007
作者:
T
Tom Zanussi
提交者:
Jens Axboe
7月 10, 2007
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
splice: relay support
Signed-off-by:
N
Jens Axboe
<
jens.axboe@oracle.com
>
上级
cf8208d0
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
191 addition
and
50 deletion
+191
-50
kernel/relay.c
kernel/relay.c
+191
-50
未找到文件。
kernel/relay.c
浏览文件 @
ebf99093
...
...
@@ -21,6 +21,7 @@
#include <linux/vmalloc.h>
#include <linux/mm.h>
#include <linux/cpu.h>
#include <linux/pipe_fs_i.h>
/* list of open channels, for cpu hotplug */
static
DEFINE_MUTEX
(
relay_channels_mutex
);
...
...
@@ -121,6 +122,7 @@ static void *relay_alloc_buf(struct rchan_buf *buf, size_t *size)
buf
->
page_array
[
i
]
=
alloc_page
(
GFP_KERNEL
);
if
(
unlikely
(
!
buf
->
page_array
[
i
]))
goto
depopulate
;
set_page_private
(
buf
->
page_array
[
i
],
(
unsigned
long
)
buf
);
}
mem
=
vmap
(
buf
->
page_array
,
n_pages
,
VM_MAP
,
PAGE_KERNEL
);
if
(
!
mem
)
...
...
@@ -970,43 +972,6 @@ static int subbuf_read_actor(size_t read_start,
return
ret
;
}
/*
* subbuf_send_actor - send up to one subbuf's worth of data
*/
static
int
subbuf_send_actor
(
size_t
read_start
,
struct
rchan_buf
*
buf
,
size_t
avail
,
read_descriptor_t
*
desc
,
read_actor_t
actor
)
{
unsigned
long
pidx
,
poff
;
unsigned
int
subbuf_pages
;
int
ret
=
0
;
subbuf_pages
=
buf
->
chan
->
alloc_size
>>
PAGE_SHIFT
;
pidx
=
(
read_start
/
PAGE_SIZE
)
%
subbuf_pages
;
poff
=
read_start
&
~
PAGE_MASK
;
while
(
avail
)
{
struct
page
*
p
=
buf
->
page_array
[
pidx
];
unsigned
int
len
;
len
=
PAGE_SIZE
-
poff
;
if
(
len
>
avail
)
len
=
avail
;
len
=
actor
(
desc
,
p
,
poff
,
len
);
if
(
desc
->
error
)
break
;
avail
-=
len
;
ret
+=
len
;
poff
=
0
;
pidx
=
(
pidx
+
1
)
%
subbuf_pages
;
}
return
ret
;
}
typedef
int
(
*
subbuf_actor_t
)
(
size_t
read_start
,
struct
rchan_buf
*
buf
,
size_t
avail
,
...
...
@@ -1067,19 +1032,195 @@ static ssize_t relay_file_read(struct file *filp,
NULL
,
&
desc
);
}
static
ssize_t
relay_file_sendfile
(
struct
file
*
filp
,
loff_t
*
ppos
,
size_t
count
,
read_actor_t
actor
,
void
*
target
)
static
void
relay_pipe_buf_release
(
struct
pipe_inode_info
*
pipe
,
struct
pipe_buffer
*
buf
)
{
read_descriptor_t
desc
;
desc
.
written
=
0
;
desc
.
count
=
count
;
desc
.
arg
.
data
=
target
;
desc
.
error
=
0
;
return
relay_file_read_subbufs
(
filp
,
ppos
,
subbuf_send_actor
,
actor
,
&
desc
);
struct
rchan_buf
*
rbuf
;
rbuf
=
(
struct
rchan_buf
*
)
page_private
(
buf
->
page
);
rbuf
->
bytes_consumed
+=
PAGE_SIZE
;
if
(
rbuf
->
bytes_consumed
==
rbuf
->
chan
->
subbuf_size
)
{
relay_subbufs_consumed
(
rbuf
->
chan
,
rbuf
->
cpu
,
1
);
rbuf
->
bytes_consumed
=
0
;
}
}
static
struct
pipe_buf_operations
relay_pipe_buf_ops
=
{
.
can_merge
=
0
,
.
map
=
generic_pipe_buf_map
,
.
unmap
=
generic_pipe_buf_unmap
,
.
pin
=
generic_pipe_buf_pin
,
.
release
=
relay_pipe_buf_release
,
.
steal
=
generic_pipe_buf_steal
,
.
get
=
generic_pipe_buf_get
,
};
/**
* subbuf_splice_actor - splice up to one subbuf's worth of data
*/
static
int
subbuf_splice_actor
(
struct
file
*
in
,
loff_t
*
ppos
,
struct
pipe_inode_info
*
pipe
,
size_t
len
,
unsigned
int
flags
,
int
*
nonpad_ret
)
{
unsigned
int
pidx
,
poff
;
unsigned
int
subbuf_pages
;
int
ret
=
0
;
int
do_wakeup
=
0
;
struct
rchan_buf
*
rbuf
=
in
->
private_data
;
unsigned
int
subbuf_size
=
rbuf
->
chan
->
subbuf_size
;
size_t
read_start
=
((
size_t
)
*
ppos
)
%
rbuf
->
chan
->
alloc_size
;
size_t
avail
=
subbuf_size
-
read_start
%
subbuf_size
;
size_t
read_subbuf
=
read_start
/
subbuf_size
;
size_t
padding
=
rbuf
->
padding
[
read_subbuf
];
size_t
nonpad_end
=
read_subbuf
*
subbuf_size
+
subbuf_size
-
padding
;
if
(
rbuf
->
subbufs_produced
==
rbuf
->
subbufs_consumed
)
return
0
;
if
(
len
>
avail
)
len
=
avail
;
if
(
pipe
->
inode
)
mutex_lock
(
&
pipe
->
inode
->
i_mutex
);
subbuf_pages
=
rbuf
->
chan
->
alloc_size
>>
PAGE_SHIFT
;
pidx
=
(
read_start
/
PAGE_SIZE
)
%
subbuf_pages
;
poff
=
read_start
&
~
PAGE_MASK
;
for
(;;)
{
unsigned
int
this_len
;
unsigned
int
this_end
;
int
newbuf
=
(
pipe
->
curbuf
+
pipe
->
nrbufs
)
&
(
PIPE_BUFFERS
-
1
);
struct
pipe_buffer
*
buf
=
pipe
->
bufs
+
newbuf
;
if
(
!
pipe
->
readers
)
{
send_sig
(
SIGPIPE
,
current
,
0
);
if
(
!
ret
)
ret
=
-
EPIPE
;
break
;
}
if
(
pipe
->
nrbufs
<
PIPE_BUFFERS
)
{
this_len
=
PAGE_SIZE
-
poff
;
if
(
this_len
>
avail
)
this_len
=
avail
;
buf
->
page
=
rbuf
->
page_array
[
pidx
];
buf
->
offset
=
poff
;
this_end
=
read_start
+
ret
+
this_len
;
if
(
this_end
>
nonpad_end
)
{
if
(
read_start
+
ret
>=
nonpad_end
)
buf
->
len
=
0
;
else
buf
->
len
=
nonpad_end
-
(
read_start
+
ret
);
}
else
buf
->
len
=
this_len
;
*
nonpad_ret
+=
buf
->
len
;
buf
->
ops
=
&
relay_pipe_buf_ops
;
pipe
->
nrbufs
++
;
avail
-=
this_len
;
ret
+=
this_len
;
poff
=
0
;
pidx
=
(
pidx
+
1
)
%
subbuf_pages
;
if
(
pipe
->
inode
)
do_wakeup
=
1
;
if
(
!
avail
)
break
;
if
(
pipe
->
nrbufs
<
PIPE_BUFFERS
)
continue
;
break
;
}
if
(
flags
&
SPLICE_F_NONBLOCK
)
{
if
(
!
ret
)
ret
=
-
EAGAIN
;
break
;
}
if
(
signal_pending
(
current
))
{
if
(
!
ret
)
ret
=
-
ERESTARTSYS
;
break
;
}
if
(
do_wakeup
)
{
smp_mb
();
if
(
waitqueue_active
(
&
pipe
->
wait
))
wake_up_interruptible_sync
(
&
pipe
->
wait
);
kill_fasync
(
&
pipe
->
fasync_readers
,
SIGIO
,
POLL_IN
);
do_wakeup
=
0
;
}
pipe
->
waiting_writers
++
;
pipe_wait
(
pipe
);
pipe
->
waiting_writers
--
;
}
if
(
pipe
->
inode
)
mutex_unlock
(
&
pipe
->
inode
->
i_mutex
);
if
(
do_wakeup
)
{
smp_mb
();
if
(
waitqueue_active
(
&
pipe
->
wait
))
wake_up_interruptible
(
&
pipe
->
wait
);
kill_fasync
(
&
pipe
->
fasync_readers
,
SIGIO
,
POLL_IN
);
}
return
ret
;
}
static
ssize_t
relay_file_splice_read
(
struct
file
*
in
,
loff_t
*
ppos
,
struct
pipe_inode_info
*
pipe
,
size_t
len
,
unsigned
int
flags
)
{
ssize_t
spliced
;
int
ret
;
int
nonpad_ret
=
0
;
ret
=
0
;
spliced
=
0
;
while
(
len
)
{
ret
=
subbuf_splice_actor
(
in
,
ppos
,
pipe
,
len
,
flags
,
&
nonpad_ret
);
if
(
ret
<
0
)
break
;
else
if
(
!
ret
)
{
break
;
if
(
spliced
)
break
;
if
(
flags
&
SPLICE_F_NONBLOCK
)
{
ret
=
-
EAGAIN
;
break
;
}
}
*
ppos
+=
ret
;
if
(
ret
>
len
)
len
=
0
;
else
len
-=
ret
;
spliced
+=
nonpad_ret
;
nonpad_ret
=
0
;
}
if
(
spliced
)
return
spliced
;
return
ret
;
}
const
struct
file_operations
relay_file_operations
=
{
...
...
@@ -1089,7 +1230,7 @@ const struct file_operations relay_file_operations = {
.
read
=
relay_file_read
,
.
llseek
=
no_llseek
,
.
release
=
relay_file_release
,
.
s
endfile
=
relay_file_sendfile
,
.
s
plice_read
=
relay_file_splice_read
,
};
EXPORT_SYMBOL_GPL
(
relay_file_operations
);
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录