Skip to content

Commit 5a5a2d7

Browse files
sarahbodinCQ Bot Account
authored and
CQ Bot Account
committed
pw_bluetooth_sapphire: Set ISO packet sequence number
Calculate and set the ISO packet sequence number and verify in test. Bug: http://b/issues/311639690 Test: bazelisk run --config=fuchsia //pw_bluetooth_sapphire/fuchsia/host/iso:test_pkg Change-Id: I2c91b75901d1bb8871180704c3e3dcbd2b282ebb Reviewed-on: https://pigweed-review.googlesource.com/c/pigweed/pigweed/+/270212 Reviewed-by: Jason Graffius <jgraff@google.com> Docs-Not-Needed: Sarah Bodin <sarahbodin@google.com> Commit-Queue: Sarah Bodin <sarahbodin@google.com> Presubmit-Verified: CQ Bot Account <pigweed-scoped@luci-project-accounts.iam.gserviceaccount.com> Lint: Lint 🤖 <android-build-ayeaye@system.gserviceaccount.com>
1 parent 7bcb789 commit 5a5a2d7

File tree

6 files changed

+267
-19
lines changed

6 files changed

+267
-19
lines changed

pw_bluetooth_sapphire/fuchsia/backends.bazelrc

+1
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ common:fuchsia --@pigweed//pw_async:task_backend=@pigweed//pw_async_fuchsia:task
3232
common:fuchsia --@pigweed//pw_async:fake_dispatcher_backend=@pigweed//pw_async_fuchsia:fake_dispatcher
3333
common:fuchsia --@pigweed//pw_sync:mutex_backend=@pigweed//pw_sync_stl:mutex
3434
common:fuchsia --@pigweed//pw_crypto:aes_backend=@pigweed//pw_crypto:aes_boringssl
35+
common:fuchsia --@pigweed//pw_sync:interrupt_spin_lock_backend=@pigweed//pw_sync_stl:interrupt_spin_lock
3536

3637
# TODO: https://pwbug.dev/374340793 - This is a temporary workaround for now and
3738
# this command line option shouldn't be applied more widely.

pw_bluetooth_sapphire/host/iso/BUILD.bazel

+1
Original file line numberDiff line numberDiff line change
@@ -80,5 +80,6 @@ pw_cc_test(
8080
":iso",
8181
"//pw_bluetooth_sapphire/host/testing",
8282
"//pw_bluetooth_sapphire/host/testing:mock_controller",
83+
"//pw_chrono:simulated_system_clock",
8384
],
8485
)

pw_bluetooth_sapphire/host/iso/BUILD.gn

+1
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,7 @@ pw_test("iso_test") {
6565
":iso",
6666
"$dir_pw_bluetooth_sapphire/host/testing",
6767
"$dir_pw_bluetooth_sapphire/host/testing:mock_controller",
68+
"$dir_pw_chrono:simulated_system_clock",
6869
]
6970
}
7071

pw_bluetooth_sapphire/host/iso/iso_stream.cc

+50-13
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,8 @@ class IsoStreamImpl final : public IsoStream {
9090
CisEstablishedCallback on_established_cb,
9191
hci::CommandChannel::WeakPtr cmd,
9292
pw::Callback<void()> on_closed_cb,
93-
hci::IsoDataChannel* data_channel);
93+
hci::IsoDataChannel* data_channel,
94+
pw::chrono::VirtualSystemClock& clock);
9495

9596
// IsoStream overrides
9697
bool OnCisEstablished(const hci::EventPacket& event) override;
@@ -173,6 +174,11 @@ class IsoStreamImpl final : public IsoStream {
173174

174175
WeakSelf<IsoStreamImpl> weak_self_;
175176

177+
pw::chrono::VirtualSystemClock& clock_;
178+
pw::chrono::SystemClock::time_point reference_time_;
179+
uint16_t next_sdu_sequence_number_ = 0;
180+
uint32_t iso_interval_usec_ = 0;
181+
176182
BT_DISALLOW_COPY_AND_ASSIGN_ALLOW_MOVE(IsoStreamImpl);
177183
};
178184

@@ -182,7 +188,8 @@ IsoStreamImpl::IsoStreamImpl(uint8_t cig_id,
182188
CisEstablishedCallback on_established_cb,
183189
hci::CommandChannel::WeakPtr cmd,
184190
pw::Callback<void()> on_closed_cb,
185-
hci::IsoDataChannel* data_channel)
191+
hci::IsoDataChannel* data_channel,
192+
pw::chrono::VirtualSystemClock& clock)
186193
: IsoStream(),
187194
state_(IsoStreamState::kNotEstablished),
188195
cig_id_(cig_id),
@@ -194,7 +201,8 @@ IsoStreamImpl::IsoStreamImpl(uint8_t cig_id,
194201
on_closed_cb_(std::move(on_closed_cb)),
195202
cmd_(std::move(cmd)),
196203
data_channel_(data_channel),
197-
weak_self_(this) {
204+
weak_self_(this),
205+
clock_(clock) {
198206
PW_CHECK(cmd_.is_alive());
199207
PW_CHECK(data_channel_);
200208

@@ -275,6 +283,11 @@ bool IsoStreamImpl::OnCisEstablished(const hci::EventPacket& event) {
275283

276284
cis_established_cb_(status, GetWeakPtr(), cis_params_);
277285

286+
reference_time_ = clock_.now();
287+
288+
iso_interval_usec_ = cis_params_.iso_interval *
289+
CisEstablishedParameters::kIsoIntervalToMicroseconds;
290+
278291
// Event handled
279292
return true;
280293
}
@@ -528,22 +541,43 @@ std::unique_ptr<IsoDataPacket> IsoStreamImpl::ReadNextQueuedIncomingPacket() {
528541
void IsoStreamImpl::Send(pw::ConstByteSpan data) {
529542
PW_CHECK(data_channel_, "Send called while not registered to a data stream.");
530543
PW_CHECK(data.size() <= std::numeric_limits<uint16_t>::max());
531-
532544
const size_t max_length = data_channel_->buffer_info().max_data_length();
545+
546+
// Calculate the current interval sequence number
547+
auto now = clock_.now();
548+
auto elapsed_time = now - reference_time_;
549+
uint64_t elapsed_usec =
550+
std::chrono::duration_cast<std::chrono::microseconds>(elapsed_time)
551+
.count();
552+
uint16_t interval_sequence_num =
553+
static_cast<uint16_t>(elapsed_usec / iso_interval_usec_);
554+
555+
uint16_t current_sequence_num = next_sdu_sequence_number_;
556+
557+
// Handle missed interval
558+
if (current_sequence_num < interval_sequence_num) {
559+
bt_log(INFO,
560+
"iso",
561+
"Skipped interval: advancing sequence number from %u to current "
562+
"interval %u",
563+
current_sequence_num,
564+
interval_sequence_num);
565+
current_sequence_num = interval_sequence_num;
566+
}
567+
533568
std::optional<SduHeaderInfo> sdu_header = SduHeaderInfo{
534-
// TODO: https://pwbug.dev/393366531 - Implement sequence number.
535-
.packet_sequence_number = 0,
569+
.packet_sequence_number = current_sequence_num,
536570
.iso_sdu_length = static_cast<uint16_t>(data.size()),
537571
};
538572

539573
// Fragmentation loop.
540574
while (!data.empty()) {
541-
size_t length_remaining =
542-
TotalDataLength(/*has_timestamp=*/false,
543-
/*has_sdu_header=*/sdu_header.has_value(),
544-
/*data_size=*/data.size());
545-
// This is the first fragment if we haven't sent the SDU header yet.
575+
// Determine if this is the first fragment of the SDU
546576
const bool is_first = sdu_header.has_value();
577+
578+
size_t length_remaining = TotalDataLength(/*has_timestamp=*/false,
579+
/*has_sdu_header=*/is_first,
580+
/*data_size=*/data.size());
547581
// This is the last fragment if there is sufficient buffer space.
548582
const bool is_last = length_remaining <= max_length;
549583

@@ -567,6 +601,7 @@ void IsoStreamImpl::Send(pw::ConstByteSpan data) {
567601
data_channel_->SendData(BuildPacketForSending(fragment, flag, sdu_header));
568602
sdu_header.reset();
569603
}
604+
next_sdu_sequence_number_ = current_sequence_num + 1;
570605
}
571606

572607
void IsoStreamImpl::Close() { on_closed_cb_(); }
@@ -578,14 +613,16 @@ std::unique_ptr<IsoStream> IsoStream::Create(
578613
CisEstablishedCallback on_established_cb,
579614
hci::CommandChannel::WeakPtr cmd,
580615
pw::Callback<void()> on_closed_cb,
581-
hci::IsoDataChannel* data_channel) {
616+
hci::IsoDataChannel* data_channel,
617+
pw::chrono::VirtualSystemClock& clock) {
582618
return std::make_unique<IsoStreamImpl>(cig_id,
583619
cis_id,
584620
cis_handle,
585621
std::move(on_established_cb),
586622
std::move(cmd),
587623
std::move(on_closed_cb),
588-
data_channel);
624+
data_channel,
625+
clock);
589626
}
590627

591628
} // namespace bt::iso

0 commit comments

Comments
 (0)