Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
BaiXuePrincess
Paddle
提交
5f9da86b
P
Paddle
项目概览
BaiXuePrincess
/
Paddle
与 Fork 源项目一致
Fork自
PaddlePaddle / Paddle
通知
1
Star
1
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
P
Paddle
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
未验证
提交
5f9da86b
编写于
3月 29, 2018
作者:
A
Abhinav Arora
提交者:
GitHub
3月 29, 2018
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Fix the order of reads and write from buffered channel (#9423)
* Fix Issue 9388 * Fix typos
上级
9bbd7534
变更
2
显示空白变更内容
内联
并排
Showing
2 changed file
with
77 addition
and
57 deletion
+77
-57
paddle/fluid/framework/channel_impl.h
paddle/fluid/framework/channel_impl.h
+51
-49
paddle/fluid/framework/channel_test.cc
paddle/fluid/framework/channel_test.cc
+26
-8
未找到文件。
paddle/fluid/framework/channel_impl.h
浏览文件 @
5f9da86b
...
@@ -87,6 +87,21 @@ class ChannelImpl : public paddle::framework::Channel<T> {
...
@@ -87,6 +87,21 @@ class ChannelImpl : public paddle::framework::Channel<T> {
return
value
;
return
value
;
}
}
std
::
shared_ptr
<
QueueMessage
>
get_first_message
(
std
::
deque
<
std
::
shared_ptr
<
QueueMessage
>>
&
queue
,
ChannelAction
action
)
{
while
(
!
queue
.
empty
())
{
// Check whether this message was added by Select
// If this was added by Select then execute the callback
// to check if you can execute this message. The callback
// can return false if some other case was executed in Select.
// In that case just discard this QueueMessage and process next.
std
::
shared_ptr
<
QueueMessage
>
m
=
queue
.
front
();
queue
.
pop_front
();
if
(
m
->
callback
==
nullptr
||
m
->
callback
(
action
))
return
m
;
}
return
nullptr
;
}
size_t
cap_
;
size_t
cap_
;
std
::
recursive_mutex
mu_
;
std
::
recursive_mutex
mu_
;
bool
closed_
;
bool
closed_
;
...
@@ -131,37 +146,22 @@ void ChannelImpl<T>::Send(T *item) {
...
@@ -131,37 +146,22 @@ void ChannelImpl<T>::Send(T *item) {
// If there is a receiver, directly pass the value we want
// If there is a receiver, directly pass the value we want
// to send to the receiver, bypassing the channel buffer if any
// to send to the receiver, bypassing the channel buffer if any
if
(
!
recvq
.
empty
())
{
if
(
!
recvq
.
empty
())
{
std
::
shared_ptr
<
QueueMessage
>
m
=
recvq
.
front
();
std
::
shared_ptr
<
QueueMessage
>
m
=
recvq
.
pop_front
();
get_first_message
(
recvq
,
ChannelAction
::
SEND
);
// Do the data transfer
// We will do this data transfer if either of the following
if
(
m
!=
nullptr
)
{
// cases are true
// 1. callback == nullptr // This means it was a regular channel send
// 2. callback returns true
bool
do_send
=
true
;
if
(
m
->
callback
!=
nullptr
)
do_send
=
m
->
callback
(
ChannelAction
::
SEND
);
if
(
do_send
)
*
(
m
->
data
)
=
std
::
move
(
*
item
);
*
(
m
->
data
)
=
std
::
move
(
*
item
);
else
{
m
->
Notify
();
// We cannot do the data transfer because
// this QueueMessage was added by Select
// and some other case was executed.
// So call the Send function again.
// We do not care about notifying other
// because they would have been notified
// by the executed select case.
lock
.
unlock
();
lock
.
unlock
();
Send
(
item
);
send_return
();
send_return
();
return
;
return
;
}
}
else
{
// Wake up the blocked process and unlock
m
->
Notify
();
lock
.
unlock
();
lock
.
unlock
();
Send
(
item
);
send_return
();
send_return
();
return
;
return
;
}
}
}
// Unbuffered channel will always bypass this
// Unbuffered channel will always bypass this
// If buffered channel has space in buffer,
// If buffered channel has space in buffer,
...
@@ -201,32 +201,34 @@ bool ChannelImpl<T>::Receive(T *item) {
...
@@ -201,32 +201,34 @@ bool ChannelImpl<T>::Receive(T *item) {
}
}
// If there is a sender, directly receive the value we want
// If there is a sender, directly receive the value we want
// from the sender, bypassing the channel buffer if any
// from the sender. In case of a buffered channel, read from
// buffer and move front of send queue to the buffer
if
(
!
sendq
.
empty
())
{
if
(
!
sendq
.
empty
())
{
std
::
shared_ptr
<
QueueMessage
>
m
=
sendq
.
front
();
std
::
shared_ptr
<
QueueMessage
>
m
=
sendq
.
pop_front
();
get_first_message
(
sendq
,
ChannelAction
::
RECEIVE
);
// Do the data transfer
if
(
buf_
.
size
()
>
0
)
{
// We will do this data transfer if either of the following
// Case 1 : Channel is Buffered
// cases are true
// Do Data transfer from front of buffer
// 1. callback == nullptr // This means it was a regular channel send
// and add a QueueMessage to the buffer
// 2. callback returns true
*
item
=
std
::
move
(
buf_
.
front
());
bool
do_receive
=
true
;
buf_
.
pop_front
();
if
(
m
->
callback
!=
nullptr
)
// If first message from sendq is not null
do_receive
=
m
->
callback
(
ChannelAction
::
RECEIVE
);
// add it to the buffer and notify it
if
(
do_receive
)
if
(
m
!=
nullptr
)
{
// Copy to buffer
buf_
.
push_back
(
std
::
move
(
*
(
m
->
data
)));
m
->
Notify
();
}
// Ignore if there is no first message
}
else
{
// Case 2: Channel is Unbuffered
// Do data transfer from front of SendQ
// If front is nullptr, then recursively call itself
if
(
m
!=
nullptr
)
{
*
item
=
std
::
move
(
*
(
m
->
data
));
*
item
=
std
::
move
(
*
(
m
->
data
));
else
// We cannot do the data transfer because
// this QueueMessage was added by Select
// and some other case was executed.
// So call the Receive function again.
// We do not care about notifying other
// because they would have been notified
// by the executed select case.
return
recv_return
(
Receive
(
item
));
// Wake up the blocked process and unlock
m
->
Notify
();
m
->
Notify
();
}
else
return
recv_return
(
Receive
(
item
));
}
lock
.
unlock
();
lock
.
unlock
();
return
recv_return
(
true
);
return
recv_return
(
true
);
}
}
...
...
paddle/fluid/framework/channel_test.cc
浏览文件 @
5f9da86b
...
@@ -36,23 +36,25 @@ TEST(Channel, ChannelCapacityTest) {
...
@@ -36,23 +36,25 @@ TEST(Channel, ChannelCapacityTest) {
delete
ch
;
delete
ch
;
}
}
void
RecevingOrderEqualToSendingOrder
(
Channel
<
int
>
*
ch
)
{
void
RecevingOrderEqualToSendingOrder
(
Channel
<
int
>
*
ch
,
int
num_items
)
{
unsigned
sum_send
=
0
;
unsigned
sum_send
=
0
;
std
::
thread
t
([
&
]()
{
std
::
thread
t
([
&
]()
{
for
(
int
i
=
0
;
i
<
5
;
i
++
)
{
for
(
int
i
=
0
;
i
<
num_items
;
i
++
)
{
ch
->
Send
(
&
i
);
ch
->
Send
(
&
i
);
sum_send
+=
i
;
sum_send
+=
i
;
}
}
});
});
for
(
int
i
=
0
;
i
<
5
;
i
++
)
{
std
::
this_thread
::
sleep_for
(
std
::
chrono
::
milliseconds
(
200
));
int
recv
=
999
;
for
(
int
i
=
0
;
i
<
num_items
;
i
++
)
{
int
recv
=
-
1
;
EXPECT_EQ
(
ch
->
Receive
(
&
recv
),
true
);
EXPECT_EQ
(
ch
->
Receive
(
&
recv
),
true
);
EXPECT_EQ
(
recv
,
i
);
EXPECT_EQ
(
recv
,
i
);
}
}
std
::
this_thread
::
sleep_for
(
std
::
chrono
::
milliseconds
(
200
));
std
::
this_thread
::
sleep_for
(
std
::
chrono
::
milliseconds
(
200
));
CloseChannel
(
ch
);
CloseChannel
(
ch
);
t
.
join
();
t
.
join
();
EXPECT_EQ
(
sum_send
,
10U
);
unsigned
expected_sum
=
(
num_items
*
(
num_items
-
1
))
/
2
;
EXPECT_EQ
(
sum_send
,
expected_sum
);
delete
ch
;
delete
ch
;
}
}
...
@@ -185,12 +187,28 @@ TEST(Channel, ConcurrentSendNonConcurrentReceiveWithSufficientBufferSize) {
...
@@ -185,12 +187,28 @@ TEST(Channel, ConcurrentSendNonConcurrentReceiveWithSufficientBufferSize) {
TEST
(
Channel
,
RecevingOrderEqualToSendingOrderWithUnBufferedChannel
)
{
TEST
(
Channel
,
RecevingOrderEqualToSendingOrderWithUnBufferedChannel
)
{
auto
ch
=
MakeChannel
<
int
>
(
0
);
auto
ch
=
MakeChannel
<
int
>
(
0
);
RecevingOrderEqualToSendingOrder
(
ch
);
RecevingOrderEqualToSendingOrder
(
ch
,
20
);
}
TEST
(
Channel
,
RecevingOrderEqualToSendingOrderWithBufferedChannel1
)
{
// Test that Receive Order is same as Send Order when number of items
// sent is less than size of buffer
auto
ch
=
MakeChannel
<
int
>
(
10
);
RecevingOrderEqualToSendingOrder
(
ch
,
5
);
}
TEST
(
Channel
,
RecevingOrderEqualToSendingOrderWithBufferedChannel2
)
{
// Test that Receive Order is same as Send Order when number of items
// sent is equal to size of buffer
auto
ch
=
MakeChannel
<
int
>
(
10
);
RecevingOrderEqualToSendingOrder
(
ch
,
10
);
}
}
TEST
(
Channel
,
RecevingOrderEqualToSendingOrderWithBufferedChannel
)
{
TEST
(
Channel
,
RecevingOrderEqualToSendingOrderWithBufferedChannel3
)
{
// Test that Receive Order is same as Send Order when number of items
// sent is greater than the size of buffer
auto
ch
=
MakeChannel
<
int
>
(
10
);
auto
ch
=
MakeChannel
<
int
>
(
10
);
RecevingOrderEqualToSendingOrder
(
ch
);
RecevingOrderEqualToSendingOrder
(
ch
,
20
);
}
}
void
ChannelCloseUnblocksReceiversTest
(
Channel
<
int
>
*
ch
)
{
void
ChannelCloseUnblocksReceiversTest
(
Channel
<
int
>
*
ch
)
{
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录