| # Copyright 2021-2024 Google LLC |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # https://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| # ----------------------------------------------------------------------------- |
| # Imports |
| # ----------------------------------------------------------------------------- |
| import asyncio |
| import logging |
| import sys |
| import os |
| import secrets |
| import websockets |
| import json |
| |
| from bumble.core import AdvertisingData |
| from bumble.device import Device, AdvertisingParameters, AdvertisingEventProperties |
| from bumble.hci import ( |
| CodecID, |
| CodingFormat, |
| OwnAddressType, |
| ) |
| from bumble.profiles.bap import ( |
| UnicastServerAdvertisingData, |
| CodecSpecificCapabilities, |
| ContextType, |
| AudioLocation, |
| SupportedSamplingFrequency, |
| SupportedFrameDuration, |
| PacRecord, |
| PublishedAudioCapabilitiesService, |
| AudioStreamControlService, |
| ) |
| from bumble.profiles.cap import CommonAudioServiceService |
| from bumble.profiles.csip import CoordinatedSetIdentificationService, SirkType |
| from bumble.profiles.vcp import VolumeControlService |
| |
| from bumble.transport import open_transport_or_link |
| |
| from typing import Optional |
| |
| |
| def dumps_volume_state(volume_setting: int, muted: int, change_counter: int) -> str: |
| return json.dumps( |
| { |
| 'volume_setting': volume_setting, |
| 'muted': muted, |
| 'change_counter': change_counter, |
| } |
| ) |
| |
| |
| # ----------------------------------------------------------------------------- |
| async def main() -> None: |
| if len(sys.argv) < 3: |
| print('Usage: run_vcp_renderer.py <config-file>' '<transport-spec-for-device>') |
| return |
| |
| print('<<< connecting to HCI...') |
| async with await open_transport_or_link(sys.argv[2]) as hci_transport: |
| print('<<< connected') |
| |
| device = Device.from_config_file_with_hci( |
| sys.argv[1], hci_transport.source, hci_transport.sink |
| ) |
| |
| await device.power_on() |
| |
| # Add "placeholder" services to enable Android LEA features. |
| csis = CoordinatedSetIdentificationService( |
| set_identity_resolving_key=secrets.token_bytes(16), |
| set_identity_resolving_key_type=SirkType.PLAINTEXT, |
| ) |
| device.add_service(CommonAudioServiceService(csis)) |
| device.add_service( |
| PublishedAudioCapabilitiesService( |
| supported_source_context=ContextType.PROHIBITED, |
| available_source_context=ContextType.PROHIBITED, |
| supported_sink_context=ContextType.MEDIA, |
| available_sink_context=ContextType.MEDIA, |
| sink_audio_locations=( |
| AudioLocation.FRONT_LEFT | AudioLocation.FRONT_RIGHT |
| ), |
| sink_pac=[ |
| # Codec Capability Setting 48_4 |
| PacRecord( |
| coding_format=CodingFormat(CodecID.LC3), |
| codec_specific_capabilities=CodecSpecificCapabilities( |
| supported_sampling_frequencies=( |
| SupportedSamplingFrequency.FREQ_48000 |
| ), |
| supported_frame_durations=( |
| SupportedFrameDuration.DURATION_10000_US_SUPPORTED |
| ), |
| supported_audio_channel_count=[1], |
| min_octets_per_codec_frame=120, |
| max_octets_per_codec_frame=120, |
| supported_max_codec_frames_per_sdu=1, |
| ), |
| ), |
| ], |
| ) |
| ) |
| device.add_service(AudioStreamControlService(device, sink_ase_id=[1, 2])) |
| |
| vcs = VolumeControlService() |
| device.add_service(vcs) |
| |
| ws: Optional[websockets.WebSocketServerProtocol] = None |
| |
| def on_volume_state(volume_setting: int, muted: int, change_counter: int): |
| if ws: |
| asyncio.create_task( |
| ws.send(dumps_volume_state(volume_setting, muted, change_counter)) |
| ) |
| |
| vcs.on('volume_state', on_volume_state) |
| |
| advertising_data = ( |
| bytes( |
| AdvertisingData( |
| [ |
| ( |
| AdvertisingData.COMPLETE_LOCAL_NAME, |
| bytes('Bumble LE Audio', 'utf-8'), |
| ), |
| ( |
| AdvertisingData.FLAGS, |
| bytes( |
| [ |
| AdvertisingData.LE_GENERAL_DISCOVERABLE_MODE_FLAG |
| | AdvertisingData.BR_EDR_HOST_FLAG |
| | AdvertisingData.BR_EDR_CONTROLLER_FLAG |
| ] |
| ), |
| ), |
| ( |
| AdvertisingData.INCOMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS, |
| bytes(PublishedAudioCapabilitiesService.UUID), |
| ), |
| ] |
| ) |
| ) |
| + csis.get_advertising_data() |
| + bytes(UnicastServerAdvertisingData()) |
| ) |
| |
| await device.create_advertising_set( |
| advertising_parameters=AdvertisingParameters( |
| advertising_event_properties=AdvertisingEventProperties(), |
| own_address_type=OwnAddressType.PUBLIC, |
| ), |
| advertising_data=advertising_data, |
| ) |
| |
| async def serve(websocket: websockets.WebSocketServerProtocol, _path): |
| nonlocal ws |
| await websocket.send( |
| dumps_volume_state(vcs.volume_setting, vcs.muted, vcs.change_counter) |
| ) |
| ws = websocket |
| async for message in websocket: |
| volume_state = json.loads(message) |
| vcs.volume_state_bytes = bytes( |
| [ |
| volume_state['volume_setting'], |
| volume_state['muted'], |
| volume_state['change_counter'], |
| ] |
| ) |
| await device.notify_subscribers( |
| vcs.volume_state, vcs.volume_state_bytes |
| ) |
| ws = None |
| |
| await websockets.serve(serve, 'localhost', 8989) |
| |
| await hci_transport.source.terminated |
| |
| |
| # ----------------------------------------------------------------------------- |
| logging.basicConfig(level=os.environ.get('BUMBLE_LOGLEVEL', 'DEBUG').upper()) |
| asyncio.run(main()) |