blob: 8452c8b63dbb5a259eebe2f210215c079c69266f [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
2 * Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
pbos@webrtc.org956aa7e2013-05-21 13:52:32 +000011#include "webrtc/voice_engine/channel_manager.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000012
Minyue2013aec2015-05-13 14:14:42 +020013#include "webrtc/common.h"
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000014#include "webrtc/voice_engine/channel.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000015
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000016namespace webrtc {
17namespace voe {
niklase@google.com470e71d2011-07-07 08:21:25 +000018
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000019ChannelOwner::ChannelOwner(class Channel* channel)
20 : channel_ref_(new ChannelRef(channel)) {}
21
22ChannelOwner::ChannelOwner(const ChannelOwner& channel_owner)
23 : channel_ref_(channel_owner.channel_ref_) {
24 ++channel_ref_->ref_count;
niklase@google.com470e71d2011-07-07 08:21:25 +000025}
26
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000027ChannelOwner::~ChannelOwner() {
28 if (--channel_ref_->ref_count == 0)
29 delete channel_ref_;
niklase@google.com470e71d2011-07-07 08:21:25 +000030}
31
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000032ChannelOwner& ChannelOwner::operator=(const ChannelOwner& other) {
33 if (other.channel_ref_ == channel_ref_)
34 return *this;
35
36 if (--channel_ref_->ref_count == 0)
37 delete channel_ref_;
38
39 channel_ref_ = other.channel_ref_;
40 ++channel_ref_->ref_count;
41
42 return *this;
niklase@google.com470e71d2011-07-07 08:21:25 +000043}
44
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000045ChannelOwner::ChannelRef::ChannelRef(class Channel* channel)
46 : channel(channel), ref_count(1) {}
47
minyue@webrtc.orge509f942013-09-12 17:03:00 +000048ChannelManager::ChannelManager(uint32_t instance_id, const Config& config)
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000049 : instance_id_(instance_id),
50 last_channel_id_(-1),
minyue@webrtc.orge509f942013-09-12 17:03:00 +000051 lock_(CriticalSectionWrapper::CreateCriticalSection()),
ivocb04965c2015-09-09 00:09:43 -070052 config_(config),
53 event_log_(RtcEventLog::Create()) {}
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000054
55ChannelOwner ChannelManager::CreateChannel() {
turaj@webrtc.org03f33702013-11-13 00:02:48 +000056 return CreateChannelInternal(config_);
57}
58
59ChannelOwner ChannelManager::CreateChannel(const Config& external_config) {
60 return CreateChannelInternal(external_config);
61}
62
63ChannelOwner ChannelManager::CreateChannelInternal(const Config& config) {
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000064 Channel* channel;
ivocb04965c2015-09-09 00:09:43 -070065 Channel::CreateChannel(channel, ++last_channel_id_, instance_id_,
66 event_log_.get(), config);
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000067 ChannelOwner channel_owner(channel);
68
69 CriticalSectionScoped crit(lock_.get());
70
71 channels_.push_back(channel_owner);
72
73 return channel_owner;
74}
75
76ChannelOwner ChannelManager::GetChannel(int32_t channel_id) {
77 CriticalSectionScoped crit(lock_.get());
78
79 for (size_t i = 0; i < channels_.size(); ++i) {
80 if (channels_[i].channel()->ChannelId() == channel_id)
81 return channels_[i];
82 }
83 return ChannelOwner(NULL);
84}
85
86void ChannelManager::GetAllChannels(std::vector<ChannelOwner>* channels) {
87 CriticalSectionScoped crit(lock_.get());
88
89 *channels = channels_;
90}
91
92void ChannelManager::DestroyChannel(int32_t channel_id) {
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +000093 assert(channel_id >= 0);
pbos@webrtc.org58d76cb2013-08-08 17:32:21 +000094 // Holds a reference to a channel, this is used so that we never delete
95 // Channels while holding a lock, but rather when the method returns.
96 ChannelOwner reference(NULL);
97 {
98 CriticalSectionScoped crit(lock_.get());
Minyue2013aec2015-05-13 14:14:42 +020099 std::vector<ChannelOwner>::iterator to_delete = channels_.end();
100 for (auto it = channels_.begin(); it != channels_.end(); ++it) {
101 Channel* channel = it->channel();
102 // For channels associated with the channel to be deleted, disassociate
103 // with that channel.
104 channel->DisassociateSendChannel(channel_id);
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000105
Minyue2013aec2015-05-13 14:14:42 +0200106 if (channel->ChannelId() == channel_id) {
107 to_delete = it;
pbos@webrtc.org58d76cb2013-08-08 17:32:21 +0000108 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000109 }
Minyue2013aec2015-05-13 14:14:42 +0200110 if (to_delete != channels_.end()) {
111 reference = *to_delete;
112 channels_.erase(to_delete);
113 }
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000114 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000115}
116
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000117void ChannelManager::DestroyAllChannels() {
pbos@webrtc.org58d76cb2013-08-08 17:32:21 +0000118 // Holds references so that Channels are not destroyed while holding this
119 // lock, but rather when the method returns.
120 std::vector<ChannelOwner> references;
121 {
122 CriticalSectionScoped crit(lock_.get());
123 references = channels_;
124 channels_.clear();
125 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000126}
127
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000128size_t ChannelManager::NumOfChannels() const {
129 CriticalSectionScoped crit(lock_.get());
130 return channels_.size();
niklase@google.com470e71d2011-07-07 08:21:25 +0000131}
132
ivocb04965c2015-09-09 00:09:43 -0700133RtcEventLog* ChannelManager::GetEventLog() const {
134 return event_log_.get();
135}
136
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000137ChannelManager::Iterator::Iterator(ChannelManager* channel_manager)
138 : iterator_pos_(0) {
139 channel_manager->GetAllChannels(&channels_);
niklase@google.com470e71d2011-07-07 08:21:25 +0000140}
141
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000142Channel* ChannelManager::Iterator::GetChannel() {
143 if (iterator_pos_ < channels_.size())
144 return channels_[iterator_pos_].channel();
145 return NULL;
niklase@google.com470e71d2011-07-07 08:21:25 +0000146}
147
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000148bool ChannelManager::Iterator::IsValid() {
149 return iterator_pos_ < channels_.size();
niklase@google.com470e71d2011-07-07 08:21:25 +0000150}
151
pbos@webrtc.org676ff1e2013-08-07 17:57:36 +0000152void ChannelManager::Iterator::Increment() {
153 ++iterator_pos_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000154}
155
pbos@webrtc.orgd900e8b2013-07-03 15:12:26 +0000156} // namespace voe
pbos@webrtc.orgd900e8b2013-07-03 15:12:26 +0000157} // namespace webrtc