Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
libvirt
提交
664085ab
L
libvirt
项目概览
openeuler
/
libvirt
通知
3
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
L
libvirt
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
664085ab
编写于
7月 28, 2009
作者:
D
Daniel P. Berrange
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Fix deadlock in remote driver domain events
* src/remote_internal.c: Release driver lock when dispatching events to callbacks
上级
abe3ee9c
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
59 addition
and
13 deletion
+59
-13
src/remote_internal.c
src/remote_internal.c
+59
-13
未找到文件。
src/remote_internal.c
浏览文件 @
664085ab
...
@@ -167,6 +167,8 @@ struct private_data {
...
@@ -167,6 +167,8 @@ struct private_data {
virDomainEventQueuePtr
domainEvents
;
virDomainEventQueuePtr
domainEvents
;
/* Timer for flushing domainEvents queue */
/* Timer for flushing domainEvents queue */
int
eventFlushTimer
;
int
eventFlushTimer
;
/* Flag if we're in process of dispatching */
int
domainEventDispatching
;
/* Self-pipe to wakeup threads waiting in poll() */
/* Self-pipe to wakeup threads waiting in poll() */
int
wakeupSendFD
;
int
wakeupSendFD
;
...
@@ -6288,18 +6290,26 @@ static int remoteDomainEventDeregister (virConnectPtr conn,
...
@@ -6288,18 +6290,26 @@ static int remoteDomainEventDeregister (virConnectPtr conn,
remoteDriverLock
(
priv
);
remoteDriverLock
(
priv
);
if
(
virDomainEventCallbackListRemove
(
conn
,
priv
->
callbackList
,
if
(
priv
->
domainEventDispatching
)
{
callback
)
<
0
)
{
if
(
virDomainEventCallbackListMarkDelete
(
conn
,
priv
->
callbackList
,
error
(
conn
,
VIR_ERR_RPC
,
_
(
"removing cb fron list"
));
callback
)
<
0
)
{
goto
done
;
error
(
conn
,
VIR_ERR_RPC
,
_
(
"marking cb for deletion"
));
}
if
(
priv
->
callbackList
->
count
==
0
)
{
/* Tell the server when we are the last callback deregistering */
if
(
call
(
conn
,
priv
,
0
,
REMOTE_PROC_DOMAIN_EVENTS_DEREGISTER
,
(
xdrproc_t
)
xdr_void
,
(
char
*
)
NULL
,
(
xdrproc_t
)
xdr_void
,
(
char
*
)
NULL
)
==
-
1
)
goto
done
;
goto
done
;
}
}
else
{
if
(
virDomainEventCallbackListRemove
(
conn
,
priv
->
callbackList
,
callback
)
<
0
)
{
error
(
conn
,
VIR_ERR_RPC
,
_
(
"removing cb from list"
));
goto
done
;
}
if
(
priv
->
callbackList
->
count
==
0
)
{
/* Tell the server when we are the last callback deregistering */
if
(
call
(
conn
,
priv
,
0
,
REMOTE_PROC_DOMAIN_EVENTS_DEREGISTER
,
(
xdrproc_t
)
xdr_void
,
(
char
*
)
NULL
,
(
xdrproc_t
)
xdr_void
,
(
char
*
)
NULL
)
==
-
1
)
goto
done
;
}
}
}
rv
=
0
;
rv
=
0
;
...
@@ -7309,18 +7319,54 @@ done:
...
@@ -7309,18 +7319,54 @@ done:
remoteDriverUnlock
(
priv
);
remoteDriverUnlock
(
priv
);
}
}
static
void
remoteDomainEventDispatchFunc
(
virConnectPtr
conn
,
virDomainEventPtr
event
,
virConnectDomainEventCallback
cb
,
void
*
cbopaque
,
void
*
opaque
)
{
struct
private_data
*
priv
=
opaque
;
/* Drop the lock whle dispatching, for sake of re-entrancy */
remoteDriverUnlock
(
priv
);
virDomainEventDispatchDefaultFunc
(
conn
,
event
,
cb
,
cbopaque
,
NULL
);
remoteDriverLock
(
priv
);
}
void
void
remoteDomainEventQueueFlush
(
int
timer
ATTRIBUTE_UNUSED
,
void
*
opaque
)
remoteDomainEventQueueFlush
(
int
timer
ATTRIBUTE_UNUSED
,
void
*
opaque
)
{
{
virConnectPtr
conn
=
opaque
;
virConnectPtr
conn
=
opaque
;
struct
private_data
*
priv
=
conn
->
privateData
;
struct
private_data
*
priv
=
conn
->
privateData
;
virDomainEventQueue
tempQueue
;
remoteDriverLock
(
priv
);
remoteDriverLock
(
priv
);
virDomainEventQueueDispatch
(
priv
->
domainEvents
,
priv
->
callbackList
,
priv
->
domainEventDispatching
=
1
;
virDomainEventDispatchDefaultFunc
,
NULL
);
/* Copy the queue, so we're reentrant safe */
tempQueue
.
count
=
priv
->
domainEvents
->
count
;
tempQueue
.
events
=
priv
->
domainEvents
->
events
;
priv
->
domainEvents
->
count
=
0
;
priv
->
domainEvents
->
events
=
NULL
;
virDomainEventQueueDispatch
(
&
tempQueue
,
priv
->
callbackList
,
remoteDomainEventDispatchFunc
,
priv
);
virEventUpdateTimeout
(
priv
->
eventFlushTimer
,
-
1
);
virEventUpdateTimeout
(
priv
->
eventFlushTimer
,
-
1
);
/* Purge any deleted callbacks */
virDomainEventCallbackListPurgeMarked
(
priv
->
callbackList
);
if
(
priv
->
callbackList
->
count
==
0
)
{
/* Tell the server when we are the last callback deregistering */
if
(
call
(
conn
,
priv
,
0
,
REMOTE_PROC_DOMAIN_EVENTS_DEREGISTER
,
(
xdrproc_t
)
xdr_void
,
(
char
*
)
NULL
,
(
xdrproc_t
)
xdr_void
,
(
char
*
)
NULL
)
==
-
1
)
VIR_WARN0
(
"Failed to de-register events"
);
}
priv
->
domainEventDispatching
=
0
;
remoteDriverUnlock
(
priv
);
remoteDriverUnlock
(
priv
);
}
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录