* Revert the reverse the addons iteration of r35121.
* The media_addon_server before unloading addons need to be sure the nodes are deleted. For instance, applications could keep references on global nodes, thus preventing deletion. To release all references, the media_addon server uses a new method BMediaRosterEx::ReleaseNodeAll(). * Quit the MediaRoster looper when quitting media_addon server before unloading addons. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@35133 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
d9626569fb
commit
6aee58a4cb
@ -38,6 +38,7 @@ public:
|
|||||||
|
|
||||||
status_t IncrementAddonFlavorInstancesCount(media_addon_id addonid, int32 flavorid);
|
status_t IncrementAddonFlavorInstancesCount(media_addon_id addonid, int32 flavorid);
|
||||||
status_t DecrementAddonFlavorInstancesCount(media_addon_id addonid, int32 flavorid);
|
status_t DecrementAddonFlavorInstancesCount(media_addon_id addonid, int32 flavorid);
|
||||||
|
status_t ReleaseNodeAll(const media_node& node);
|
||||||
|
|
||||||
status_t SetNodeCreator(media_node_id node, team_id creator);
|
status_t SetNodeCreator(media_node_id node, team_id creator);
|
||||||
|
|
||||||
|
@ -53,6 +53,7 @@ enum {
|
|||||||
SERVER_GET_LIVE_NODES,
|
SERVER_GET_LIVE_NODES,
|
||||||
SERVER_GET_NODE_FOR,
|
SERVER_GET_NODE_FOR,
|
||||||
SERVER_RELEASE_NODE,
|
SERVER_RELEASE_NODE,
|
||||||
|
SERVER_RELEASE_NODE_ALL,
|
||||||
SERVER_REGISTER_NODE,
|
SERVER_REGISTER_NODE,
|
||||||
SERVER_UNREGISTER_NODE,
|
SERVER_UNREGISTER_NODE,
|
||||||
SERVER_GET_DORMANT_NODE_FOR,
|
SERVER_GET_DORMANT_NODE_FOR,
|
||||||
|
@ -190,6 +190,40 @@ BMediaRosterEx::DecrementAddonFlavorInstancesCount(media_addon_id addonID,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
status_t
|
||||||
|
BMediaRosterEx::ReleaseNodeAll(const media_node& node)
|
||||||
|
{
|
||||||
|
CALLED();
|
||||||
|
if (IS_INVALID_NODE(node))
|
||||||
|
return B_MEDIA_BAD_NODE;
|
||||||
|
|
||||||
|
if (node.kind & NODE_KIND_NO_REFCOUNTING) {
|
||||||
|
printf("BMediaRoster::ReleaseNodeAll, trying to release reference "
|
||||||
|
"counting disabled timesource, node %ld, port %ld, team %ld\n",
|
||||||
|
node.node, node.port, BPrivate::current_team());
|
||||||
|
return B_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
server_release_node_request request;
|
||||||
|
server_release_node_reply reply;
|
||||||
|
status_t rv;
|
||||||
|
|
||||||
|
request.node = node;
|
||||||
|
request.team = BPrivate::current_team();
|
||||||
|
|
||||||
|
TRACE("BMediaRoster::ReleaseNodeAll, node %ld, port %ld, team %ld\n",
|
||||||
|
node.node, node.port, BPrivate::current_team());
|
||||||
|
|
||||||
|
rv = QueryServer(SERVER_RELEASE_NODE_ALL, &request, sizeof(request), &reply,
|
||||||
|
sizeof(reply));
|
||||||
|
if (rv != B_OK) {
|
||||||
|
ERROR("BMediaRoster::ReleaseNodeAll FAILED, node %ld, port %ld, team "
|
||||||
|
"%ld!\n", node.node, node.port, BPrivate::current_team());
|
||||||
|
}
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
status_t
|
status_t
|
||||||
BMediaRosterEx::SetNodeCreator(media_node_id node, team_id creator)
|
BMediaRosterEx::SetNodeCreator(media_node_id node, team_id creator)
|
||||||
{
|
{
|
||||||
|
@ -280,6 +280,38 @@ NodeManager::ReleaseNodeReference(media_node_id id, team_id team)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
status_t
|
||||||
|
NodeManager::ReleaseNodeAll(media_node_id id)
|
||||||
|
{
|
||||||
|
TRACE("NodeManager::ReleaseNodeAll enter: node %ld, team %ld\n", node.node,
|
||||||
|
team);
|
||||||
|
|
||||||
|
BAutolock _(this);
|
||||||
|
|
||||||
|
NodeMap::iterator found = fNodeMap.find(id);
|
||||||
|
if (found == fNodeMap.end()) {
|
||||||
|
ERROR("NodeManager::ReleaseNodeAll: node %ld not found\n", id);
|
||||||
|
return B_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
registered_node& node = found->second;
|
||||||
|
node.team_ref_count.clear();
|
||||||
|
node.ref_count = 0;
|
||||||
|
|
||||||
|
node_final_release_command command;
|
||||||
|
status_t status = SendToPort(node.port, NODE_FINAL_RELEASE, &command,
|
||||||
|
sizeof(command));
|
||||||
|
if (status != B_OK) {
|
||||||
|
ERROR("NodeManager::ReleaseNodeAll: can't send command to "
|
||||||
|
"node %ld\n", id);
|
||||||
|
// ignore error
|
||||||
|
}
|
||||||
|
|
||||||
|
TRACE("NodeManager::ReleaseNodeAll leave: node %ld\n", id);
|
||||||
|
return B_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
status_t
|
status_t
|
||||||
NodeManager::SetNodeCreator(media_node_id id, team_id creator)
|
NodeManager::SetNodeCreator(media_node_id id, team_id creator)
|
||||||
{
|
{
|
||||||
|
@ -79,6 +79,7 @@ public:
|
|||||||
int32* _flavorID);
|
int32* _flavorID);
|
||||||
status_t ReleaseNodeReference(media_node_id id,
|
status_t ReleaseNodeReference(media_node_id id,
|
||||||
team_id team);
|
team_id team);
|
||||||
|
status_t ReleaseNodeAll(media_node_id id);
|
||||||
status_t GetCloneForID(media_node_id id, team_id team,
|
status_t GetCloneForID(media_node_id id, team_id team,
|
||||||
media_node* node);
|
media_node* node);
|
||||||
status_t GetClone(node_type type, team_id team,
|
status_t GetClone(node_type type, team_id team,
|
||||||
|
@ -437,6 +437,17 @@ ServerApp::_HandleMessage(int32 code, const void* data, size_t size)
|
|||||||
request.SendReply(status, &reply, sizeof(reply));
|
request.SendReply(status, &reply, sizeof(reply));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case SERVER_RELEASE_NODE_ALL:
|
||||||
|
{
|
||||||
|
const server_release_node_request& request
|
||||||
|
= *static_cast<const server_release_node_request*>(data);
|
||||||
|
server_release_node_reply reply;
|
||||||
|
|
||||||
|
status_t status = gNodeManager->ReleaseNodeAll(request.node.node);
|
||||||
|
request.SendReply(status, &reply, sizeof(reply));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case SERVER_REGISTER_NODE:
|
case SERVER_REGISTER_NODE:
|
||||||
{
|
{
|
||||||
|
@ -341,13 +341,15 @@ MediaAddonServer::QuitRequested()
|
|||||||
{
|
{
|
||||||
CALLED();
|
CALLED();
|
||||||
|
|
||||||
InfoMap::reverse_iterator iterator = fInfoMap.rbegin();
|
InfoMap::iterator iterator = fInfoMap.begin();
|
||||||
for (; iterator != fInfoMap.rend(); iterator++) {
|
for (iterator = fInfoMap.begin(); iterator != fInfoMap.end(); iterator++)
|
||||||
AddOnInfo& info = iterator->second;
|
_DestroyInstantiatedFlavors(iterator->second);
|
||||||
|
|
||||||
_DestroyInstantiatedFlavors(info);
|
BMediaRoster::CurrentRoster()->Lock();
|
||||||
_PutAddonIfPossible(info);
|
BMediaRoster::CurrentRoster()->Quit();
|
||||||
}
|
|
||||||
|
for (iterator = fInfoMap.begin(); iterator != fInfoMap.end(); iterator++)
|
||||||
|
_PutAddonIfPossible(iterator->second);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -605,8 +607,8 @@ MediaAddonServer::_DestroyInstantiatedFlavors(AddOnInfo& info)
|
|||||||
{
|
{
|
||||||
printf("MediaAddonServer::_DestroyInstantiatedFlavors addon %ld\n", info.id);
|
printf("MediaAddonServer::_DestroyInstantiatedFlavors addon %ld\n", info.id);
|
||||||
|
|
||||||
NodeVector::reverse_iterator iterator = info.active_flavors.rbegin();
|
NodeVector::iterator iterator = info.active_flavors.begin();
|
||||||
for (; iterator != info.active_flavors.rend(); iterator++) {
|
for (; iterator != info.active_flavors.end(); iterator++) {
|
||||||
media_node& node = *iterator;
|
media_node& node = *iterator;
|
||||||
|
|
||||||
printf("node %ld\n", node.node);
|
printf("node %ld\n", node.node);
|
||||||
@ -680,7 +682,7 @@ MediaAddonServer::_DestroyInstantiatedFlavors(AddOnInfo& info)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fMediaRoster->ReleaseNode(node);
|
MediaRosterEx(fMediaRoster)->ReleaseNodeAll(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
info.active_flavors.clear();
|
info.active_flavors.clear();
|
||||||
|
Loading…
Reference in New Issue
Block a user