1 /* 2 * Copyright 2019 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #pragma once 18 19 #include <unordered_map> 20 #include <unordered_set> 21 22 #include "hci/acl_manager.h" 23 #include "l2cap/cid.h" 24 #include "l2cap/classic/security_policy.h" 25 #include "l2cap/internal/ilink.h" 26 #include "l2cap/psm.h" 27 #include "os/handler.h" 28 #include "os/log.h" 29 30 namespace bluetooth { 31 namespace l2cap { 32 33 namespace classic { 34 namespace internal { 35 class DumpsysHelper; 36 } // namespace internal 37 } // namespace classic 38 39 namespace internal { 40 41 class DynamicChannelImpl; 42 43 // Helper class for keeping channels in a Link. It allocates and frees Channel object, and supports querying whether a 44 // channel is in use 45 class DynamicChannelAllocator { 46 public: DynamicChannelAllocator(l2cap::internal::ILink * link,os::Handler * l2cap_handler)47 DynamicChannelAllocator(l2cap::internal::ILink* link, os::Handler* l2cap_handler) 48 : link_(link), l2cap_handler_(l2cap_handler) { 49 log::assert_that(link_ != nullptr, "assert failed: link_ != nullptr"); 50 log::assert_that(l2cap_handler_ != nullptr, "assert failed: l2cap_handler_ != nullptr"); 51 } 52 53 // Allocates a channel. If psm is used, OR the remote cid already exists, return nullptr. 54 // NOTE: The returned DynamicChannelImpl object is still owned by the channel allocator, NOT the client. 55 std::shared_ptr<DynamicChannelImpl> AllocateChannel(Psm psm, Cid remote_cid); 56 57 std::shared_ptr<DynamicChannelImpl> AllocateReservedChannel(Cid reserved_cid, Psm psm, Cid remote_cid); 58 59 // Gives an unused Cid to be used for opening a channel. If a channel is used, call AllocateReservedChannel. If no 60 // longer needed, use FreeChannel. 61 Cid ReserveChannel(); 62 63 // Frees a channel (existing or reserved) 64 void FreeChannel(Cid cid); 65 66 bool IsPsmUsed(Psm psm) const; 67 68 std::shared_ptr<DynamicChannelImpl> FindChannelByCid(Cid cid); 69 std::shared_ptr<DynamicChannelImpl> FindChannelByRemoteCid(Cid cid); 70 71 // Returns number of open, but not reserved channels 72 size_t NumberOfChannels() const; 73 74 void OnAclDisconnected(hci::ErrorCode hci_status); 75 76 private: 77 friend class bluetooth::l2cap::classic::internal::DumpsysHelper; 78 l2cap::internal::ILink* link_; 79 os::Handler* l2cap_handler_; 80 std::unordered_set<Cid> used_cid_; 81 std::unordered_map<Cid, std::shared_ptr<DynamicChannelImpl>> channels_; 82 std::unordered_set<Cid> used_remote_cid_; 83 }; 84 85 } // namespace internal 86 } // namespace l2cap 87 } // namespace bluetooth 88