xref: /btstack/src/le-audio/broadcast_audio_uri_builder.c (revision faddce61d0c9554da1e8ac2e2e6403058d69d66e)
165dc4892SMatthias Ringwald /*
265dc4892SMatthias Ringwald  * Copyright (C) 2024 BlueKitchen GmbH
365dc4892SMatthias Ringwald  *
465dc4892SMatthias Ringwald  * Redistribution and use in source and binary forms, with or without
565dc4892SMatthias Ringwald  * modification, are permitted provided that the following conditions
665dc4892SMatthias Ringwald  * are met:
765dc4892SMatthias Ringwald  *
865dc4892SMatthias Ringwald  * 1. Redistributions of source code must retain the above copyright
965dc4892SMatthias Ringwald  *    notice, this list of conditions and the following disclaimer.
1065dc4892SMatthias Ringwald  * 2. Redistributions in binary form must reproduce the above copyright
1165dc4892SMatthias Ringwald  *    notice, this list of conditions and the following disclaimer in the
1265dc4892SMatthias Ringwald  *    documentation and/or other materials provided with the distribution.
1365dc4892SMatthias Ringwald  * 3. Neither the name of the copyright holders nor the names of
1465dc4892SMatthias Ringwald  *    contributors may be used to endorse or promote products derived
1565dc4892SMatthias Ringwald  *    from this software without specific prior written permission.
1665dc4892SMatthias Ringwald  * 4. Any redistribution, use, or modification is done solely for
1765dc4892SMatthias Ringwald  *    personal benefit and not for any commercial purpose or for
1865dc4892SMatthias Ringwald  *    monetary gain.
1965dc4892SMatthias Ringwald  *
2065dc4892SMatthias Ringwald  * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
2165dc4892SMatthias Ringwald  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
2265dc4892SMatthias Ringwald  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
2365dc4892SMatthias Ringwald  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN
2465dc4892SMatthias Ringwald  * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
2565dc4892SMatthias Ringwald  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
2665dc4892SMatthias Ringwald  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
2765dc4892SMatthias Ringwald  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
2865dc4892SMatthias Ringwald  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
2965dc4892SMatthias Ringwald  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
3065dc4892SMatthias Ringwald  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3165dc4892SMatthias Ringwald  * SUCH DAMAGE.
3265dc4892SMatthias Ringwald  *
3365dc4892SMatthias Ringwald  * Please inquire about commercial licensing options at
3465dc4892SMatthias Ringwald  * [email protected]
3565dc4892SMatthias Ringwald  *
3665dc4892SMatthias Ringwald  */
3765dc4892SMatthias Ringwald 
3865dc4892SMatthias Ringwald #define BTSTACK_FILE__ "broadcast_audio_uri_builder.c"
3965dc4892SMatthias Ringwald 
4065dc4892SMatthias Ringwald #include "le-audio/broadcast_audio_uri_builder.h"
4165dc4892SMatthias Ringwald #include "btstack_util.h"
4265dc4892SMatthias Ringwald #include <stdint.h>
4365dc4892SMatthias Ringwald #include <string.h>
4465dc4892SMatthias Ringwald #include <inttypes.h>
4565dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_string_hexdump(uint8_t * buffer,const uint8_t * data,uint16_t size)4665dc4892SMatthias Ringwald static void broadcast_audio_uri_builder_string_hexdump(uint8_t * buffer, const uint8_t * data, uint16_t size){
4765dc4892SMatthias Ringwald     uint8_t i;
4865dc4892SMatthias Ringwald     for (i = 0; i < size ; i++) {
4965dc4892SMatthias Ringwald         uint8_t byte = data[i];
5065dc4892SMatthias Ringwald         buffer[2*i+0] = char_for_nibble(byte >> 4);
5165dc4892SMatthias Ringwald         buffer[2*i+1] = char_for_nibble(byte & 0x0f);
5265dc4892SMatthias Ringwald     }
5365dc4892SMatthias Ringwald }
5465dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_have_space(const broadcast_audio_uri_builder_t * builder,uint16_t len)5565dc4892SMatthias Ringwald static inline bool broadcast_audio_uri_builder_have_space(const broadcast_audio_uri_builder_t * builder, uint16_t len){
5665dc4892SMatthias Ringwald     return builder->len + len <= builder->size;
5765dc4892SMatthias Ringwald }
5865dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_init(broadcast_audio_uri_builder_t * builder,uint8_t * buffer,uint16_t size)5965dc4892SMatthias Ringwald void broadcast_audio_uri_builder_init(broadcast_audio_uri_builder_t * builder, uint8_t * buffer, uint16_t size){
6065dc4892SMatthias Ringwald     builder->buffer = buffer;
6165dc4892SMatthias Ringwald     builder->size = size;
6265dc4892SMatthias Ringwald     builder->len = 0;
6365dc4892SMatthias Ringwald }
6465dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_get_remaining_space(const broadcast_audio_uri_builder_t * builder)6565dc4892SMatthias Ringwald uint16_t broadcast_audio_uri_builder_get_remaining_space(const broadcast_audio_uri_builder_t * builder){
6665dc4892SMatthias Ringwald     return builder->size - builder->len;
6765dc4892SMatthias Ringwald }
6865dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_get_size(const broadcast_audio_uri_builder_t * builder)6965dc4892SMatthias Ringwald uint16_t broadcast_audio_uri_builder_get_size(const broadcast_audio_uri_builder_t * builder){
7065dc4892SMatthias Ringwald     return builder->len;
7165dc4892SMatthias Ringwald }
7265dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_bytes(broadcast_audio_uri_builder_t * builder,const uint8_t * data,uint16_t len)7365dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_bytes(broadcast_audio_uri_builder_t * builder, const uint8_t * data, uint16_t len){
7465dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_have_space(builder, len);
7565dc4892SMatthias Ringwald     if (ok){
7665dc4892SMatthias Ringwald         memcpy(&builder->buffer[builder->len], data, len);
7765dc4892SMatthias Ringwald         builder->len += len;
7865dc4892SMatthias Ringwald     }
7965dc4892SMatthias Ringwald     return ok;
8065dc4892SMatthias Ringwald }
8165dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_string(broadcast_audio_uri_builder_t * builder,const char * text)8265dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_string(broadcast_audio_uri_builder_t * builder, const char * text){
8365dc4892SMatthias Ringwald     uint16_t len = (uint16_t) strlen(text);
8465dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_bytes(builder, (const uint8_t *) text, len);
8565dc4892SMatthias Ringwald }
8665dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_broadcast_name(broadcast_audio_uri_builder_t * builder,const char * broadcast_name)8765dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_broadcast_name(broadcast_audio_uri_builder_t * builder, const char * broadcast_name){
88*faddce61SMatthias Ringwald     UNUSED(broadcast_name);
8965dc4892SMatthias Ringwald     // TODO: base64
9065dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, "BN:QnJvYWRjYXN0IE5hbWU=;");
9165dc4892SMatthias Ringwald }
9265dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_advertiser_address_type(broadcast_audio_uri_builder_t * builder,bd_addr_type_t advertiser_address_type)9365dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_advertiser_address_type(broadcast_audio_uri_builder_t * builder, bd_addr_type_t advertiser_address_type){
9465dc4892SMatthias Ringwald     char buffer[10];
9565dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "AT:%u;",advertiser_address_type == BD_ADDR_TYPE_LE_RANDOM ? 1 : 0);
9665dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
9765dc4892SMatthias Ringwald }
9865dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_standard_quality(broadcast_audio_uri_builder_t * builder,bool standard_quality)9965dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_standard_quality(broadcast_audio_uri_builder_t * builder, bool standard_quality){
10065dc4892SMatthias Ringwald     char buffer[10];
10165dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "SQ:%u;", (int) standard_quality);
10265dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
10365dc4892SMatthias Ringwald }
10465dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_high_quality(broadcast_audio_uri_builder_t * builder,bool high_quality)10565dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_high_quality(broadcast_audio_uri_builder_t * builder, bool high_quality){
10665dc4892SMatthias Ringwald     char buffer[10];
10765dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "HQ:%u;", (int) high_quality);
10865dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
10965dc4892SMatthias Ringwald }
11065dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_advertiser_address(broadcast_audio_uri_builder_t * builder,bd_addr_t advertiser_address)11165dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_advertiser_address(broadcast_audio_uri_builder_t * builder, bd_addr_t advertiser_address){
11265dc4892SMatthias Ringwald     uint16_t len = builder->len;
11365dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_append_string(builder,"AD:");
11465dc4892SMatthias Ringwald     if (ok){
11565dc4892SMatthias Ringwald         char buffer[13];
11665dc4892SMatthias Ringwald         broadcast_audio_uri_builder_string_hexdump((uint8_t *)buffer, (const uint8_t *) advertiser_address, 6);
11765dc4892SMatthias Ringwald         buffer[12] = ';';
11865dc4892SMatthias Ringwald         broadcast_audio_uri_builder_append_bytes(builder, (const uint8_t *) buffer, sizeof(buffer));
11965dc4892SMatthias Ringwald     } else {
12065dc4892SMatthias Ringwald         builder->len = len;
12165dc4892SMatthias Ringwald     }
12265dc4892SMatthias Ringwald     return ok;
12365dc4892SMatthias Ringwald }
12465dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_broadcast_id(broadcast_audio_uri_builder_t * builder,uint32_t broadcast_id)12565dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_broadcast_id(broadcast_audio_uri_builder_t * builder, uint32_t broadcast_id){
12665dc4892SMatthias Ringwald     uint16_t len = builder->len;
12765dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_append_string(builder,"BI:");
12865dc4892SMatthias Ringwald     if (ok){
12965dc4892SMatthias Ringwald         uint8_t buffer[7];
13065dc4892SMatthias Ringwald         uint8_t big_endian_id[3];
13165dc4892SMatthias Ringwald         big_endian_store_24(big_endian_id, 0, broadcast_id);
13265dc4892SMatthias Ringwald         broadcast_audio_uri_builder_string_hexdump(buffer, big_endian_id, 3);
13365dc4892SMatthias Ringwald         buffer[6] = ';';
13465dc4892SMatthias Ringwald         broadcast_audio_uri_builder_append_bytes(builder, (const uint8_t *) buffer, sizeof(buffer));
13565dc4892SMatthias Ringwald     } else {
13665dc4892SMatthias Ringwald         builder->len = len;
13765dc4892SMatthias Ringwald     }
13865dc4892SMatthias Ringwald     return ok;
13965dc4892SMatthias Ringwald 
14065dc4892SMatthias Ringwald }
14165dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_broadcast_code(broadcast_audio_uri_builder_t * builder,const uint8_t * broadcast_code)14265dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_broadcast_code(broadcast_audio_uri_builder_t * builder, const uint8_t * broadcast_code){
143*faddce61SMatthias Ringwald     UNUSED(broadcast_code);
14465dc4892SMatthias Ringwald     uint16_t len = builder->len;
14565dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_append_string(builder,"BC:");
14665dc4892SMatthias Ringwald     if (ok){
14765dc4892SMatthias Ringwald         uint8_t buffer[25];
14865dc4892SMatthias Ringwald         // TODO: base64
14965dc4892SMatthias Ringwald         memcpy(buffer, "MDEyMzQ1Njc4OWFiY2RlZg==", 24);
15065dc4892SMatthias Ringwald         buffer[24] = ';';
15165dc4892SMatthias Ringwald         broadcast_audio_uri_builder_append_bytes(builder, (const uint8_t *) buffer, sizeof(buffer));
15265dc4892SMatthias Ringwald     } else {
15365dc4892SMatthias Ringwald         builder->len = len;
15465dc4892SMatthias Ringwald     }
15565dc4892SMatthias Ringwald     return ok;
15665dc4892SMatthias Ringwald }
15765dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_vendor_specific(broadcast_audio_uri_builder_t * builder,uint16_t vendor_id,const uint8_t * data,uint16_t data_len)15865dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_vendor_specific(broadcast_audio_uri_builder_t * builder, uint16_t vendor_id, const uint8_t * data, uint16_t data_len){
159*faddce61SMatthias Ringwald     UNUSED(data);
160*faddce61SMatthias Ringwald     UNUSED(data_len);
16165dc4892SMatthias Ringwald     uint16_t len = builder->len;
16265dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_append_string(builder, "VS:");
16365dc4892SMatthias Ringwald     if (ok){
16465dc4892SMatthias Ringwald         uint8_t vendor_id_big_endian_id[2];
16565dc4892SMatthias Ringwald         big_endian_store_16(vendor_id_big_endian_id, 0, vendor_id);
16665dc4892SMatthias Ringwald         uint8_t vendor_id_hex[4];
167892de207SMatthias Ringwald         memset(vendor_id_hex, 0, sizeof(vendor_id_hex));
16865dc4892SMatthias Ringwald         broadcast_audio_uri_builder_string_hexdump(vendor_id_hex, vendor_id_hex, 2);
16965dc4892SMatthias Ringwald         // TODO: base64(vendor_id_hex + data)
17065dc4892SMatthias Ringwald         ok = broadcast_audio_uri_builder_append_string(builder, ";");
17165dc4892SMatthias Ringwald     }
17265dc4892SMatthias Ringwald     if (ok == false){
17365dc4892SMatthias Ringwald         builder->len = len;
17465dc4892SMatthias Ringwald     }
17565dc4892SMatthias Ringwald     return ok;
17665dc4892SMatthias Ringwald }
17765dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_advertising_sid(broadcast_audio_uri_builder_t * builder,uint8_t advertising_sid)17865dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_advertising_sid(broadcast_audio_uri_builder_t * builder, uint8_t advertising_sid){
17965dc4892SMatthias Ringwald     char buffer[10];
18065dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "AS:%02X;", advertising_sid);
18165dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
18265dc4892SMatthias Ringwald }
18365dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_pa_interval(broadcast_audio_uri_builder_t * builder,uint16_t pa_interval)18465dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_pa_interval(broadcast_audio_uri_builder_t * builder, uint16_t pa_interval){
18565dc4892SMatthias Ringwald     char buffer[10];
18665dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "PI:%04" PRIX32 ";", pa_interval);
18765dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
18865dc4892SMatthias Ringwald }
18965dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_num_subgroups(broadcast_audio_uri_builder_t * builder,uint8_t num_subgroups)19065dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_num_subgroups(broadcast_audio_uri_builder_t * builder, uint8_t num_subgroups){
19165dc4892SMatthias Ringwald     char buffer[10];
19265dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "NS:%02X", num_subgroups);
19365dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
19465dc4892SMatthias Ringwald }
19565dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_bis_sync(broadcast_audio_uri_builder_t * builder,uint32_t bis_sync)19665dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_bis_sync(broadcast_audio_uri_builder_t * builder, uint32_t bis_sync){
19765dc4892SMatthias Ringwald     char buffer[10];
19865dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "BS:%04" PRIX32 ";", bis_sync);
19965dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
20065dc4892SMatthias Ringwald }
20165dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_sg_number_of_bises(broadcast_audio_uri_builder_t * builder,uint32_t sg_number_of_bises)20265dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_sg_number_of_bises(broadcast_audio_uri_builder_t * builder, uint32_t sg_number_of_bises){
20365dc4892SMatthias Ringwald     char buffer[10];
20465dc4892SMatthias Ringwald     btstack_snprintf_assert_complete(buffer, sizeof(buffer), "NB:%04" PRIX32 ";", sg_number_of_bises);
20565dc4892SMatthias Ringwald     return broadcast_audio_uri_builder_append_string(builder, buffer);
20665dc4892SMatthias Ringwald }
20765dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_sg_metadata(broadcast_audio_uri_builder_t * builder,const uint8_t * metadata,uint16_t metadata_len)20865dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_sg_metadata(broadcast_audio_uri_builder_t * builder, const uint8_t * metadata, uint16_t metadata_len){
20965dc4892SMatthias Ringwald     uint16_t len = builder->len;
21065dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_append_string(builder, "SM:");
21165dc4892SMatthias Ringwald     if (ok && (metadata_len > 0)) {
21265dc4892SMatthias Ringwald         // TODO: base64(data)
21365dc4892SMatthias Ringwald         ok = broadcast_audio_uri_builder_append_bytes(builder, metadata, metadata_len);
21465dc4892SMatthias Ringwald     }
21565dc4892SMatthias Ringwald     if (ok) {
21665dc4892SMatthias Ringwald         ok = broadcast_audio_uri_builder_append_string(builder, ";");
21765dc4892SMatthias Ringwald     } else {
21865dc4892SMatthias Ringwald         builder->len = len;
21965dc4892SMatthias Ringwald     }
22065dc4892SMatthias Ringwald     return ok;
22165dc4892SMatthias Ringwald }
22265dc4892SMatthias Ringwald 
broadcast_audio_uri_builder_append_public_broadcast_announcement_metadata(broadcast_audio_uri_builder_t * builder,const uint8_t * metadata,uint16_t metadata_len)22365dc4892SMatthias Ringwald bool broadcast_audio_uri_builder_append_public_broadcast_announcement_metadata(broadcast_audio_uri_builder_t * builder, const uint8_t * metadata, uint16_t metadata_len){
22465dc4892SMatthias Ringwald     uint16_t len = builder->len;
22565dc4892SMatthias Ringwald     bool ok = broadcast_audio_uri_builder_append_string(builder, "PM:");
22665dc4892SMatthias Ringwald     if (ok && metadata_len > 0) {
22765dc4892SMatthias Ringwald         // TODO: base64(data)
22865dc4892SMatthias Ringwald         ok = broadcast_audio_uri_builder_append_bytes(builder, metadata, metadata_len);
22965dc4892SMatthias Ringwald     }
23065dc4892SMatthias Ringwald     if (ok) {
23165dc4892SMatthias Ringwald         ok = broadcast_audio_uri_builder_append_string(builder, ";");
23265dc4892SMatthias Ringwald     } else {
23365dc4892SMatthias Ringwald         builder->len = len;
23465dc4892SMatthias Ringwald     }
23565dc4892SMatthias Ringwald     return ok;
23665dc4892SMatthias Ringwald }
237