/* Copyright (c) 2014 The Chromium OS Authors. All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include <sys/time.h> #include <syslog.h> #include "cras_bt_io.h" #include "cras_bt_device.h" #include "cras_utf8.h" #include "cras_iodev.h" #include "cras_iodev_list.h" #include "sfh.h" #include "utlist.h" #define DEFAULT_BT_DEVICE_NAME "BLUETOOTH" /* Extends cras_ionode to hold bluetooth profile information * so that iodevs of different profile(A2DP or HFP/HSP) can be * associated with the same bt_io. * Members: * base - The base class cras_ionode. * profile_dev - Pointer to the profile specific iodev. * profile - The bluetooth profile profile_dev runs on. */ struct bt_node { struct cras_ionode base; struct cras_iodev *profile_dev; unsigned int profile; }; /* The structure represents a virtual input or output device of a * bluetooth audio device, speaker or headset for example. A node * will be added to this virtual iodev for each profile supported * by the bluetooth audio device. * Member: * base - The base class cras_iodev * next_node_id - The index will give to the next node */ struct bt_io { struct cras_iodev base; unsigned int next_node_id; struct cras_bt_device *device; }; /* Returns the active profile specific iodev. */ static struct cras_iodev *active_profile_dev(const struct cras_iodev *iodev) { struct bt_node *active = (struct bt_node *)iodev->active_node; return active->profile_dev; } /* Adds a profile specific iodev to btio. */ static struct cras_ionode *add_profile_dev(struct cras_iodev *bt_iodev, struct cras_iodev *dev, enum cras_bt_device_profile profile) { struct bt_node *n; struct bt_io *btio = (struct bt_io *)bt_iodev; n = (struct bt_node *)calloc(1, sizeof(*n)); if (!n) return NULL; n->base.dev = bt_iodev; n->base.idx = btio->next_node_id++; n->base.type = CRAS_NODE_TYPE_BLUETOOTH; n->base.volume = 100; n->base.stable_id = dev->info.stable_id; n->base.stable_id_new = dev->info.stable_id_new; n->base.max_software_gain = 0; gettimeofday(&n->base.plugged_time, NULL); strcpy(n->base.name, dev->info.name); n->profile_dev = dev; n->profile = profile; cras_iodev_add_node(bt_iodev, &n->base); return &n->base; } /* Forces bt device to switch to use the given profile. Note that if * it has already been open for streaming, the new active profile will * take effect after the related btio(s) are reopened. */ static void bt_switch_to_profile(struct cras_bt_device *device, enum cras_bt_device_profile profile) { switch (profile) { case CRAS_BT_DEVICE_PROFILE_HFP_AUDIOGATEWAY: case CRAS_BT_DEVICE_PROFILE_HSP_AUDIOGATEWAY: cras_bt_device_set_active_profile(device, CRAS_BT_DEVICE_PROFILE_HSP_AUDIOGATEWAY | CRAS_BT_DEVICE_PROFILE_HFP_AUDIOGATEWAY); break; case CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE: cras_bt_device_set_active_profile(device, CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE); break; default: syslog(LOG_ERR, "Unexpect profile %u", profile); break; } } /* Checks if bt device is active for the given profile. */ static int device_using_profile(struct cras_bt_device *device, unsigned int profile) { return cras_bt_device_get_active_profile(device) & profile; } /* Checks if the condition is met to switch to a different profile based * on two rules: * (1) Prefer to use A2DP for output since the audio quality is better. * (2) Must use HFP/HSP for input since A2DP doesn't support audio input. * * If the profile switch happens, return non-zero error code, otherwise * return zero. */ static int open_dev(struct cras_iodev *iodev) { struct bt_io *btio = (struct bt_io *)iodev; /* Force to use HFP if opening input dev. */ if (device_using_profile(btio->device, CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE) && iodev->direction == CRAS_STREAM_INPUT) { bt_switch_to_profile(btio->device, CRAS_BT_DEVICE_PROFILE_HFP_AUDIOGATEWAY); cras_bt_device_switch_profile_enable_dev(btio->device, iodev); return -EAGAIN; } return 0; } static int update_supported_formats(struct cras_iodev *iodev) { struct cras_iodev *dev = active_profile_dev(iodev); int rc, length, i; if (dev->format == NULL) { dev->format = (struct cras_audio_format *) malloc(sizeof(*dev->format)); *dev->format = *iodev->format; } if (dev->update_supported_formats) { rc = dev->update_supported_formats(dev); if (rc) return rc; } /* Fill in the supported rates and channel counts. */ for (length = 0; dev->supported_rates[length]; length++); free(iodev->supported_rates); iodev->supported_rates = (size_t *)malloc( (length + 1) * sizeof(*iodev->supported_rates)); for (i = 0; i < length + 1; i++) iodev->supported_rates[i] = dev->supported_rates[i]; for (length = 0; dev->supported_channel_counts[length]; length++); iodev->supported_channel_counts = (size_t *)malloc( (length + 1) * sizeof(*iodev->supported_channel_counts)); for (i = 0; i < length + 1; i++) iodev->supported_channel_counts[i] = dev->supported_channel_counts[i]; for (length = 0; dev->supported_formats[length]; length++); iodev->supported_formats = (snd_pcm_format_t *)malloc( (length + 1) * sizeof(*iodev->supported_formats)); for (i = 0; i < length + 1; i++) iodev->supported_formats[i] = dev->supported_formats[i]; return 0; } static int configure_dev(struct cras_iodev *iodev) { int rc; struct cras_iodev *dev = active_profile_dev(iodev); if (!dev) return -EINVAL; /* Fill back the format iodev is using. */ *dev->format = *iodev->format; rc = dev->configure_dev(dev); if (rc) { /* Free format here to assure the update_supported_format * callback will be called before any future open_dev call. */ cras_iodev_free_format(iodev); return rc; } iodev->buffer_size = dev->buffer_size; iodev->min_buffer_level = dev->min_buffer_level; return 0; } static int close_dev(struct cras_iodev *iodev) { struct bt_io *btio = (struct bt_io *)iodev; int rc; struct cras_iodev *dev = active_profile_dev(iodev); /* Force back to A2DP if closing HFP. */ if (device_using_profile(btio->device, CRAS_BT_DEVICE_PROFILE_HSP_AUDIOGATEWAY | CRAS_BT_DEVICE_PROFILE_HFP_AUDIOGATEWAY) && iodev->direction == CRAS_STREAM_INPUT && cras_bt_device_has_a2dp(btio->device)) { cras_bt_device_set_active_profile(btio->device, CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE); cras_bt_device_switch_profile(btio->device, iodev); } rc = dev->close_dev(dev); if (rc < 0) return rc; cras_iodev_free_format(iodev); return 0; } static void set_bt_volume(struct cras_iodev *iodev) { struct cras_iodev *dev = active_profile_dev(iodev); if (dev->active_node) dev->active_node->volume = iodev->active_node->volume; /* The parent bt_iodev could set software_volume_needed flag for cases * that software volume provides better experience across profiles * (HFP and A2DP). Otherwise, use the profile specific implementation * to adjust volume. */ if (dev->set_volume && !iodev->software_volume_needed) dev->set_volume(dev); } static int frames_queued(const struct cras_iodev *iodev, struct timespec *tstamp) { struct cras_iodev *dev = active_profile_dev(iodev); if (!dev) return -EINVAL; return dev->frames_queued(dev, tstamp); } static int delay_frames(const struct cras_iodev *iodev) { struct cras_iodev *dev = active_profile_dev(iodev); if (!dev) return -EINVAL; return dev->delay_frames(dev); } static int get_buffer(struct cras_iodev *iodev, struct cras_audio_area **area, unsigned *frames) { struct cras_iodev *dev = active_profile_dev(iodev); if (!dev) return -EINVAL; return dev->get_buffer(dev, area, frames); } static int put_buffer(struct cras_iodev *iodev, unsigned nwritten) { struct cras_iodev *dev = active_profile_dev(iodev); if (!dev) return -EINVAL; return dev->put_buffer(dev, nwritten); } static int flush_buffer(struct cras_iodev *iodev) { struct cras_iodev *dev = active_profile_dev(iodev); if (!dev) return -EINVAL; return dev->flush_buffer(dev); } /* If the first private iodev doesn't match the active profile stored on * device, select to the correct private iodev. */ static void update_active_node(struct cras_iodev *iodev, unsigned node_idx, unsigned dev_enabled) { struct bt_io *btio = (struct bt_io *)iodev; struct cras_ionode *node; struct bt_node *active = (struct bt_node *)iodev->active_node; if (device_using_profile(btio->device, active->profile)) return; /* Switch to the correct dev using active_profile. */ DL_FOREACH(iodev->nodes, node) { struct bt_node *n = (struct bt_node *)node; if (n == active) continue; if (device_using_profile(btio->device, n->profile)) { active->profile = n->profile; active->profile_dev = n->profile_dev; /* Set volume for the new profile. */ set_bt_volume(iodev); } } } struct cras_iodev *cras_bt_io_create(struct cras_bt_device *device, struct cras_iodev *dev, enum cras_bt_device_profile profile) { int err; struct bt_io *btio; struct cras_iodev *iodev; struct cras_ionode *node; struct bt_node *active; if (!dev) return NULL; btio = (struct bt_io *)calloc(1, sizeof(*btio)); if (!btio) goto error; btio->device = device; iodev = &btio->base; iodev->direction = dev->direction; strcpy(iodev->info.name, dev->info.name); iodev->info.stable_id = dev->info.stable_id; iodev->info.stable_id_new = dev->info.stable_id_new; iodev->open_dev = open_dev; iodev->configure_dev = configure_dev; iodev->frames_queued = frames_queued; iodev->delay_frames = delay_frames; iodev->get_buffer = get_buffer; iodev->put_buffer = put_buffer; iodev->flush_buffer = flush_buffer; iodev->close_dev = close_dev; iodev->update_supported_formats = update_supported_formats; iodev->update_active_node = update_active_node; iodev->no_stream = cras_iodev_default_no_stream_playback; /* Input also checks |software_volume_needed| flag for using software * gain. Keep it as false for BT input. * TODO(hychao): after wide band speech mode is supported, consider * enable software gain. */ if (dev->direction == CRAS_STREAM_OUTPUT) { iodev->software_volume_needed = !cras_bt_device_get_use_hardware_volume(device); iodev->set_volume = set_bt_volume; } /* Create the dummy node set to plugged so it's the only node exposed * to UI, and point it to the first profile dev. */ active = (struct bt_node *)calloc(1, sizeof(*active)); if (!active) return NULL; active->base.dev = iodev; active->base.idx = btio->next_node_id++; active->base.type = CRAS_NODE_TYPE_BLUETOOTH; active->base.volume = 100; active->base.plugged = 1; active->base.stable_id = SuperFastHash( cras_bt_device_object_path(device), strlen(cras_bt_device_object_path(device)), strlen(cras_bt_device_object_path(device))); active->base.stable_id_new = active->base.stable_id; active->profile = profile; active->profile_dev = dev; gettimeofday(&active->base.plugged_time, NULL); strcpy(active->base.name, dev->info.name); /* The node name exposed to UI should be a valid UTF8 string. */ if (!is_utf8_string(active->base.name)) strcpy(active->base.name, DEFAULT_BT_DEVICE_NAME); cras_iodev_add_node(iodev, &active->base); node = add_profile_dev(&btio->base, dev, profile); if (node == NULL) goto error; /* Default active profile to a2dp whenever it's allowed. */ if (!cras_bt_device_get_active_profile(device) || (profile == CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE && cras_bt_device_can_switch_to_a2dp(device))) bt_switch_to_profile(device, profile); if (iodev->direction == CRAS_STREAM_OUTPUT) err = cras_iodev_list_add_output(iodev); else err = cras_iodev_list_add_input(iodev); if (err) goto error; cras_iodev_set_active_node(iodev, &active->base); return &btio->base; error: if (btio) free(btio); return NULL; } void cras_bt_io_free_resources(struct cras_iodev *bt_iodev) { struct cras_ionode *node; struct bt_node *n; free(bt_iodev->supported_rates); free(bt_iodev->supported_channel_counts); free(bt_iodev->supported_formats); DL_FOREACH(bt_iodev->nodes, node) { n = (struct bt_node *)node; cras_iodev_rm_node(bt_iodev, node); free(n); } cras_iodev_free_resources(bt_iodev); } void cras_bt_io_destroy(struct cras_iodev *bt_iodev) { int rc; struct bt_io *btio = (struct bt_io *)bt_iodev; if (bt_iodev->direction == CRAS_STREAM_OUTPUT) rc = cras_iodev_list_rm_output(bt_iodev); else rc = cras_iodev_list_rm_input(bt_iodev); if (rc == -EBUSY) return; cras_bt_io_free_resources(bt_iodev); free(btio); } struct cras_ionode *cras_bt_io_get_profile(struct cras_iodev *bt_iodev, enum cras_bt_device_profile profile) { struct cras_ionode *node; DL_FOREACH(bt_iodev->nodes, node) { struct bt_node *n = (struct bt_node *)node; if (n->profile & profile) return node; } return NULL; } int cras_bt_io_append(struct cras_iodev *bt_iodev, struct cras_iodev *dev, enum cras_bt_device_profile profile) { struct cras_ionode *node; struct bt_io *btio = (struct bt_io *)bt_iodev; if (cras_bt_io_get_profile(bt_iodev, profile)) return -EEXIST; node = add_profile_dev(bt_iodev, dev, profile); if (!node) return -ENOMEM; if (profile == CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE && cras_bt_device_can_switch_to_a2dp(btio->device)) { bt_switch_to_profile(btio->device, CRAS_BT_DEVICE_PROFILE_A2DP_SOURCE); cras_bt_device_switch_profile(btio->device, bt_iodev); syslog(LOG_ERR, "Switch to A2DP on append"); } return 0; } int cras_bt_io_on_profile(struct cras_iodev *bt_iodev, enum cras_bt_device_profile profile) { struct bt_node *btnode = (struct bt_node *)bt_iodev->active_node; return !!(profile & btnode->profile); } int cras_bt_io_update_buffer_size(struct cras_iodev *bt_iodev) { struct cras_iodev *dev = active_profile_dev(bt_iodev); if (!dev) return -EINVAL; bt_iodev->buffer_size = dev->buffer_size; return 0; } unsigned int cras_bt_io_try_remove(struct cras_iodev *bt_iodev, struct cras_iodev *dev) { struct cras_ionode *node; struct bt_node *active, *btnode; unsigned int try_profile = 0; active = (struct bt_node *)bt_iodev->active_node; if (active->profile_dev == dev) { DL_FOREACH(bt_iodev->nodes, node) { btnode = (struct bt_node *)node; /* Skip the active node and the node we're trying * to remove. */ if (btnode == active || btnode->profile_dev == dev) continue; try_profile = btnode->profile; break; } } else { try_profile = active->profile; } return try_profile; } int cras_bt_io_remove(struct cras_iodev *bt_iodev, struct cras_iodev *dev) { struct cras_ionode *node; struct bt_node *btnode; DL_FOREACH(bt_iodev->nodes, node) { btnode = (struct bt_node *)node; if (btnode->profile_dev != dev) continue; /* If this is the active node, reset it. Otherwise delete * this node. */ if (node == bt_iodev->active_node) { btnode->profile_dev = NULL; btnode->profile = 0; } else { DL_DELETE(bt_iodev->nodes, node); free(node); } } /* The node of active profile could have been removed, update it. * Return err when fail to locate the active profile dev. */ update_active_node(bt_iodev, 0, 1); btnode = (struct bt_node *)bt_iodev->active_node; if ((btnode->profile == 0) || (btnode->profile_dev == NULL)) return -EINVAL; return 0; }