mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-10-31 22:25:38 -04:00
129 lines
2.5 KiB
C
129 lines
2.5 KiB
C
|
|
/* Spa HFP Codecs */
|
||
|
|
/* SPDX-FileCopyrightText: Copyright © 2025 Pauli Virtanen */
|
||
|
|
/* SPDX-License-Identifier: MIT */
|
||
|
|
|
||
|
|
#ifndef SPA_BLUEZ5_HFP_H2_H
|
||
|
|
#define SPA_BLUEZ5_HFP_H2_H
|
||
|
|
|
||
|
|
#define H2_PACKET_SIZE 60
|
||
|
|
|
||
|
|
struct h2_reader {
|
||
|
|
uint8_t buf[H2_PACKET_SIZE];
|
||
|
|
uint8_t pos;
|
||
|
|
bool msbc;
|
||
|
|
uint16_t seq;
|
||
|
|
bool started;
|
||
|
|
};
|
||
|
|
|
||
|
|
struct h2_writer {
|
||
|
|
uint8_t seq;
|
||
|
|
};
|
||
|
|
|
||
|
|
static inline void h2_reader_init(struct h2_reader *this, bool msbc)
|
||
|
|
{
|
||
|
|
this->pos = 0;
|
||
|
|
this->msbc = msbc;
|
||
|
|
this->seq = 0;
|
||
|
|
this->started = false;
|
||
|
|
}
|
||
|
|
|
||
|
|
static inline void h2_reader_append_byte(struct h2_reader *this, uint8_t byte)
|
||
|
|
{
|
||
|
|
/* Parse H2 sync header */
|
||
|
|
if (this->pos == 0) {
|
||
|
|
if (byte != 0x01) {
|
||
|
|
this->pos = 0;
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
} else if (this->pos == 1) {
|
||
|
|
if (!((byte & 0x0F) == 0x08 &&
|
||
|
|
((byte >> 4) & 1) == ((byte >> 5) & 1) &&
|
||
|
|
((byte >> 6) & 1) == ((byte >> 7) & 1))) {
|
||
|
|
this->pos = 0;
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
} else if (this->msbc) {
|
||
|
|
/* Beginning of MSBC frame: SYNCWORD + 2 nul bytes */
|
||
|
|
if (this->pos == 2) {
|
||
|
|
if (byte != 0xAD) {
|
||
|
|
this->pos = 0;
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
else if (this->pos == 3) {
|
||
|
|
if (byte != 0x00) {
|
||
|
|
this->pos = 0;
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
else if (this->pos == 4) {
|
||
|
|
if (byte != 0x00) {
|
||
|
|
this->pos = 0;
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
if (this->pos >= H2_PACKET_SIZE) {
|
||
|
|
/* Packet completed. Reset. */
|
||
|
|
this->pos = 0;
|
||
|
|
h2_reader_append_byte(this, byte);
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
this->buf[this->pos] = byte;
|
||
|
|
++this->pos;
|
||
|
|
}
|
||
|
|
|
||
|
|
static inline void *h2_reader_read(struct h2_reader *this, const uint8_t *src, size_t src_size, size_t *consumed, size_t *avail)
|
||
|
|
{
|
||
|
|
int seq;
|
||
|
|
size_t i;
|
||
|
|
|
||
|
|
for (i = 0; i < src_size && this->pos < H2_PACKET_SIZE; ++i)
|
||
|
|
h2_reader_append_byte(this, src[i]);
|
||
|
|
|
||
|
|
*consumed = i;
|
||
|
|
*avail = 0;
|
||
|
|
|
||
|
|
if (this->pos < H2_PACKET_SIZE)
|
||
|
|
return NULL;
|
||
|
|
|
||
|
|
this->pos = 0;
|
||
|
|
|
||
|
|
seq = ((this->buf[1] >> 4) & 1) | ((this->buf[1] >> 6) & 2);
|
||
|
|
if (!this->started) {
|
||
|
|
this->seq = seq;
|
||
|
|
this->started = true;
|
||
|
|
}
|
||
|
|
|
||
|
|
this->seq++;
|
||
|
|
while (seq != this->seq % 4)
|
||
|
|
this->seq++;
|
||
|
|
|
||
|
|
*avail = H2_PACKET_SIZE - 2;
|
||
|
|
return &this->buf[2];
|
||
|
|
}
|
||
|
|
|
||
|
|
static inline void h2_write(uint8_t *buf, uint8_t seq)
|
||
|
|
{
|
||
|
|
static const uint8_t sntable[4] = { 0x08, 0x38, 0xc8, 0xf8 };
|
||
|
|
|
||
|
|
buf[0] = 0x01;
|
||
|
|
buf[1] = sntable[seq % 4];
|
||
|
|
buf[59] = 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
static inline bool is_zero_packet(const uint8_t *data, size_t size)
|
||
|
|
{
|
||
|
|
size_t i;
|
||
|
|
|
||
|
|
for (i = 0; i < size; ++i)
|
||
|
|
if (data[i])
|
||
|
|
return false;
|
||
|
|
|
||
|
|
return true;
|
||
|
|
}
|
||
|
|
|
||
|
|
#endif
|