/* * Copyright 2019 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "l2cap/internal/dynamic_channel_allocator.h" #include #include #include "l2cap/cid.h" #include "l2cap/classic/internal/link.h" #include "l2cap/classic/security_policy.h" #include "l2cap/internal/dynamic_channel_impl.h" #include "os/log.h" namespace bluetooth { namespace l2cap { namespace internal { std::shared_ptr DynamicChannelAllocator::AllocateChannel(Psm psm, Cid remote_cid) { if (used_remote_cid_.find(remote_cid) != used_remote_cid_.end()) { log::info("Remote cid 0x{:x} is used", remote_cid); return nullptr; } Cid cid = kFirstDynamicChannel; for (; cid <= kLastDynamicChannel; cid++) { if (used_cid_.find(cid) == used_cid_.end()) break; } if (cid > kLastDynamicChannel) { log::warn("All cid are used"); return nullptr; } auto elem = channels_.try_emplace(cid, std::make_shared(psm, cid, remote_cid, link_, l2cap_handler_)); log::assert_that( elem.second, "Failed to create channel for psm 0x{:x} device {}", psm, ADDRESS_TO_LOGGABLE_CSTR(link_->GetDevice())); log::assert_that(elem.first->second != nullptr, "assert failed: elem.first->second != nullptr"); used_remote_cid_.insert(remote_cid); used_cid_.insert(cid); return elem.first->second; } std::shared_ptr DynamicChannelAllocator::AllocateReservedChannel(Cid reserved_cid, Psm psm, Cid remote_cid) { if (used_remote_cid_.find(remote_cid) != used_remote_cid_.end()) { log::info("Remote cid 0x{:x} is used", remote_cid); return nullptr; } auto elem = channels_.try_emplace( reserved_cid, std::make_shared(psm, reserved_cid, remote_cid, link_, l2cap_handler_)); log::assert_that( elem.second, "Failed to create channel for psm 0x{:x} device {}", psm, ADDRESS_TO_LOGGABLE_CSTR(link_->GetDevice())); log::assert_that(elem.first->second != nullptr, "assert failed: elem.first->second != nullptr"); used_remote_cid_.insert(remote_cid); return elem.first->second; } Cid DynamicChannelAllocator::ReserveChannel() { Cid cid = kFirstDynamicChannel; for (; cid <= kLastDynamicChannel; cid++) { if (used_cid_.find(cid) == used_cid_.end()) break; } if (cid > kLastDynamicChannel) { log::warn("All cid are used"); return kInvalidCid; } used_cid_.insert(cid); return cid; } void DynamicChannelAllocator::FreeChannel(Cid cid) { used_cid_.erase(cid); auto channel = FindChannelByCid(cid); if (channel == nullptr) { log::info("Channel is not in use: cid {}, device {}", cid, link_->GetDevice()); return; } used_remote_cid_.erase(channel->GetRemoteCid()); channels_.erase(cid); } bool DynamicChannelAllocator::IsPsmUsed(Psm psm) const { for (const auto& channel : channels_) { if (channel.second->GetPsm() == psm) { return true; } } return false; } std::shared_ptr DynamicChannelAllocator::FindChannelByCid(Cid cid) { if (channels_.find(cid) == channels_.end()) { log::warn("Can't find cid {}", cid); return nullptr; } return channels_.find(cid)->second; } std::shared_ptr DynamicChannelAllocator::FindChannelByRemoteCid(Cid remote_cid) { for (auto& channel : channels_) { if (channel.second->GetRemoteCid() == remote_cid) { return channel.second; } } return nullptr; } size_t DynamicChannelAllocator::NumberOfChannels() const { return channels_.size(); } void DynamicChannelAllocator::OnAclDisconnected(hci::ErrorCode reason) { for (auto& elem : channels_) { elem.second->OnClosed(reason); } } } // namespace internal } // namespace l2cap } // namespace bluetooth