1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker * Copyright 2022 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker *
4*4d7e907cSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker *
8*4d7e907cSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker *
10*4d7e907cSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker */
16*4d7e907cSAndroid Build Coastguard Worker
17*4d7e907cSAndroid Build Coastguard Worker #include "hci_packetizer.h"
18*4d7e907cSAndroid Build Coastguard Worker
19*4d7e907cSAndroid Build Coastguard Worker #define LOG_TAG "android.hardware.bluetooth.hci-packetizer"
20*4d7e907cSAndroid Build Coastguard Worker #include "log/log.h"
21*4d7e907cSAndroid Build Coastguard Worker
22*4d7e907cSAndroid Build Coastguard Worker namespace android::hardware::bluetooth::hci {
23*4d7e907cSAndroid Build Coastguard Worker
24*4d7e907cSAndroid Build Coastguard Worker namespace {
25*4d7e907cSAndroid Build Coastguard Worker
26*4d7e907cSAndroid Build Coastguard Worker const size_t header_size_for_type[] = {0,
27*4d7e907cSAndroid Build Coastguard Worker kCommandHeaderSize,
28*4d7e907cSAndroid Build Coastguard Worker kAclHeaderSize,
29*4d7e907cSAndroid Build Coastguard Worker kScoHeaderSize,
30*4d7e907cSAndroid Build Coastguard Worker kEventHeaderSize,
31*4d7e907cSAndroid Build Coastguard Worker kIsoHeaderSize};
32*4d7e907cSAndroid Build Coastguard Worker const size_t packet_length_offset_for_type[] = {0,
33*4d7e907cSAndroid Build Coastguard Worker kCommandLengthOffset,
34*4d7e907cSAndroid Build Coastguard Worker kAclLengthOffset,
35*4d7e907cSAndroid Build Coastguard Worker kScoLengthOffset,
36*4d7e907cSAndroid Build Coastguard Worker kEventLengthOffset,
37*4d7e907cSAndroid Build Coastguard Worker kIsoLengthOffset};
38*4d7e907cSAndroid Build Coastguard Worker
HciGetPacketLengthForType(PacketType type,const std::vector<uint8_t> & header)39*4d7e907cSAndroid Build Coastguard Worker size_t HciGetPacketLengthForType(PacketType type,
40*4d7e907cSAndroid Build Coastguard Worker const std::vector<uint8_t>& header) {
41*4d7e907cSAndroid Build Coastguard Worker size_t offset = packet_length_offset_for_type[static_cast<uint8_t>(type)];
42*4d7e907cSAndroid Build Coastguard Worker if (type != PacketType::ACL_DATA && type != PacketType::ISO_DATA) {
43*4d7e907cSAndroid Build Coastguard Worker return header[offset];
44*4d7e907cSAndroid Build Coastguard Worker }
45*4d7e907cSAndroid Build Coastguard Worker return (((header[offset + 1]) << 8) | header[offset]);
46*4d7e907cSAndroid Build Coastguard Worker }
47*4d7e907cSAndroid Build Coastguard Worker
48*4d7e907cSAndroid Build Coastguard Worker } // namespace
49*4d7e907cSAndroid Build Coastguard Worker
GetPacket() const50*4d7e907cSAndroid Build Coastguard Worker const std::vector<uint8_t>& HciPacketizer::GetPacket() const { return packet_; }
51*4d7e907cSAndroid Build Coastguard Worker
OnDataReady(PacketType packet_type,const std::vector<uint8_t> & buffer,size_t * offset)52*4d7e907cSAndroid Build Coastguard Worker bool HciPacketizer::OnDataReady(PacketType packet_type,
53*4d7e907cSAndroid Build Coastguard Worker const std::vector<uint8_t>& buffer,
54*4d7e907cSAndroid Build Coastguard Worker size_t* offset) {
55*4d7e907cSAndroid Build Coastguard Worker bool packet_completed = false;
56*4d7e907cSAndroid Build Coastguard Worker size_t bytes_available = buffer.size() - *offset;
57*4d7e907cSAndroid Build Coastguard Worker
58*4d7e907cSAndroid Build Coastguard Worker switch (state_) {
59*4d7e907cSAndroid Build Coastguard Worker case HCI_HEADER: {
60*4d7e907cSAndroid Build Coastguard Worker size_t header_size =
61*4d7e907cSAndroid Build Coastguard Worker header_size_for_type[static_cast<size_t>(packet_type)];
62*4d7e907cSAndroid Build Coastguard Worker if (bytes_remaining_ == 0) {
63*4d7e907cSAndroid Build Coastguard Worker bytes_remaining_ = header_size;
64*4d7e907cSAndroid Build Coastguard Worker packet_.clear();
65*4d7e907cSAndroid Build Coastguard Worker }
66*4d7e907cSAndroid Build Coastguard Worker
67*4d7e907cSAndroid Build Coastguard Worker size_t bytes_to_copy = std::min(bytes_remaining_, bytes_available);
68*4d7e907cSAndroid Build Coastguard Worker packet_.insert(packet_.end(), buffer.begin() + *offset,
69*4d7e907cSAndroid Build Coastguard Worker buffer.begin() + *offset + bytes_to_copy);
70*4d7e907cSAndroid Build Coastguard Worker bytes_remaining_ -= bytes_to_copy;
71*4d7e907cSAndroid Build Coastguard Worker bytes_available -= bytes_to_copy;
72*4d7e907cSAndroid Build Coastguard Worker *offset += bytes_to_copy;
73*4d7e907cSAndroid Build Coastguard Worker
74*4d7e907cSAndroid Build Coastguard Worker if (bytes_remaining_ == 0) {
75*4d7e907cSAndroid Build Coastguard Worker bytes_remaining_ = HciGetPacketLengthForType(packet_type, packet_);
76*4d7e907cSAndroid Build Coastguard Worker if (bytes_remaining_ > 0) {
77*4d7e907cSAndroid Build Coastguard Worker state_ = HCI_PAYLOAD;
78*4d7e907cSAndroid Build Coastguard Worker if (bytes_available > 0) {
79*4d7e907cSAndroid Build Coastguard Worker packet_completed = OnDataReady(packet_type, buffer, offset);
80*4d7e907cSAndroid Build Coastguard Worker }
81*4d7e907cSAndroid Build Coastguard Worker } else {
82*4d7e907cSAndroid Build Coastguard Worker packet_completed = true;
83*4d7e907cSAndroid Build Coastguard Worker }
84*4d7e907cSAndroid Build Coastguard Worker }
85*4d7e907cSAndroid Build Coastguard Worker break;
86*4d7e907cSAndroid Build Coastguard Worker }
87*4d7e907cSAndroid Build Coastguard Worker
88*4d7e907cSAndroid Build Coastguard Worker case HCI_PAYLOAD: {
89*4d7e907cSAndroid Build Coastguard Worker size_t bytes_to_copy = std::min(bytes_remaining_, bytes_available);
90*4d7e907cSAndroid Build Coastguard Worker packet_.insert(packet_.end(), buffer.begin() + *offset,
91*4d7e907cSAndroid Build Coastguard Worker buffer.begin() + *offset + bytes_to_copy);
92*4d7e907cSAndroid Build Coastguard Worker bytes_remaining_ -= bytes_to_copy;
93*4d7e907cSAndroid Build Coastguard Worker *offset += bytes_to_copy;
94*4d7e907cSAndroid Build Coastguard Worker if (bytes_remaining_ == 0) {
95*4d7e907cSAndroid Build Coastguard Worker state_ = HCI_HEADER;
96*4d7e907cSAndroid Build Coastguard Worker packet_completed = true;
97*4d7e907cSAndroid Build Coastguard Worker }
98*4d7e907cSAndroid Build Coastguard Worker break;
99*4d7e907cSAndroid Build Coastguard Worker }
100*4d7e907cSAndroid Build Coastguard Worker }
101*4d7e907cSAndroid Build Coastguard Worker
102*4d7e907cSAndroid Build Coastguard Worker return packet_completed;
103*4d7e907cSAndroid Build Coastguard Worker }
104*4d7e907cSAndroid Build Coastguard Worker
105*4d7e907cSAndroid Build Coastguard Worker } // namespace android::hardware::bluetooth::hci
106