/* Copyright (c) 2013 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 <dbus/dbus.h>
#include <errno.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
#include "audio_thread.h"
#include "cras_dbus.h"
#include "cras_dbus_control.h"
#include "cras_dbus_util.h"
#include "cras_iodev_list.h"
#include "cras_observer.h"
#include "cras_system_state.h"
#include "cras_util.h"
#include "utlist.h"
#define CRAS_CONTROL_INTERFACE "org.chromium.cras.Control"
#define CRAS_ROOT_OBJECT_PATH "/org/chromium/cras"
#define CONTROL_INTROSPECT_XML \
DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE \
"<node>\n" \
" <interface name=\""CRAS_CONTROL_INTERFACE"\">\n" \
" <method name=\"SetOutputVolume\">\n" \
" <arg name=\"volume\" type=\"i\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetOutputNodeVolume\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" <arg name=\"volume\" type=\"i\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SwapLeftRight\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" <arg name=\"swap\" type=\"b\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetOutputMute\">\n" \
" <arg name=\"mute_on\" type=\"b\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetOutputUserMute\">\n" \
" <arg name=\"mute_on\" type=\"b\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetSuspendAudio\">\n" \
" <arg name=\"suspend\" type=\"b\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetInputGain\">\n" \
" <arg name=\"gain\" type=\"i\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetInputNodeGain\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" <arg name=\"gain\" type=\"i\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetInputMute\">\n" \
" <arg name=\"mute_on\" type=\"b\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"GetVolumeState\">\n" \
" <arg name=\"output_volume\" type=\"i\" direction=\"out\"/>\n"\
" <arg name=\"output_mute\" type=\"b\" direction=\"out\"/>\n" \
" <arg name=\"input_gain\" type=\"i\" direction=\"out\"/>\n" \
" <arg name=\"input_mute\" type=\"b\" direction=\"out\"/>\n" \
" <arg name=\"output_user_mute\" type=\"b\" direction=\"out\"/>\n"\
" </method>\n" \
" <method name=\"GetDefaultOutputBufferSize\">\n" \
" <arg name=\"buffer_size\" type=\"i\" direction=\"out\"/>\n" \
" </method>\n" \
" <method name=\"GetNodes\">\n" \
" <arg name=\"nodes\" type=\"a{sv}\" direction=\"out\"/>\n" \
" </method>\n" \
" <method name=\"GetSystemAecSupported\">\n" \
" <arg name=\"supported\" type=\"b\" direction=\"out\"/>\n" \
" </method>\n" \
" <method name=\"SetActiveOutputNode\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetActiveInputNode\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"AddActiveInputNode\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"AddActiveOutputNode\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"RemoveActiveInputNode\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"RemoveActiveOutputNode\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"GetNumberOfActiveStreams\">\n" \
" <arg name=\"num\" type=\"i\" direction=\"out\"/>\n" \
" </method>\n" \
" <method name=\"GetNumberOfActiveOutputStreams\">\n" \
" <arg name=\"num\" type=\"i\" direction=\"out\"/>\n" \
" </method>\n" \
" <method name=\"GetNumberOfActiveInputStreams\">\n" \
" <arg name=\"num\" type=\"i\" direction=\"out\"/>\n" \
" </method>\n" \
" <method name=\"SetGlobalOutputChannelRemix\">\n" \
" <arg name=\"num_channels\" type=\"i\" direction=\"in\"/>\n" \
" <arg name=\"coefficient\" type=\"ad\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"SetHotwordModel\">\n" \
" <arg name=\"node_id\" type=\"t\" direction=\"in\"/>\n" \
" <arg name=\"model_name\" type=\"s\" direction=\"in\"/>\n" \
" </method>\n" \
" <method name=\"IsAudioOutputActive\">\n" \
" <arg name=\"active\" type=\"b\" direction=\"out\"/>\n" \
" </method>\n" \
" </interface>\n" \
" <interface name=\"" DBUS_INTERFACE_INTROSPECTABLE "\">\n" \
" <method name=\"Introspect\">\n" \
" <arg name=\"data\" type=\"s\" direction=\"out\"/>\n" \
" </method>\n" \
" </interface>\n" \
"</node>\n"
struct cras_dbus_control {
DBusConnection *conn;
struct cras_observer_client *observer;
};
static struct cras_dbus_control dbus_control;
/* helper to extract a single argument from a DBus message. */
static int get_single_arg(DBusMessage *message, int dbus_type, void *arg)
{
DBusError dbus_error;
dbus_error_init(&dbus_error);
if (!dbus_message_get_args(message, &dbus_error,
dbus_type, arg,
DBUS_TYPE_INVALID)) {
syslog(LOG_WARNING,
"Bad method received: %s",
dbus_error.message);
dbus_error_free(&dbus_error);
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
return 0;
}
/* Helper to send an empty reply. */
static void send_empty_reply(DBusConnection *conn, DBusMessage *message)
{
DBusMessage *reply;
dbus_uint32_t serial = 0;
reply = dbus_message_new_method_return(message);
if (!reply)
return;
dbus_connection_send(conn, reply, &serial);
dbus_message_unref(reply);
}
/* Helper to send an int32 reply. */
static void send_int32_reply(DBusConnection *conn,
DBusMessage *message,
dbus_int32_t value)
{
DBusMessage *reply;
dbus_uint32_t serial = 0;
reply = dbus_message_new_method_return(message);
if (!reply)
return;
dbus_message_append_args(reply,
DBUS_TYPE_INT32, &value,
DBUS_TYPE_INVALID);
dbus_connection_send(conn, reply, &serial);
dbus_message_unref(reply);
}
/* Handlers for exported DBus method calls. */
static DBusHandlerResult handle_set_output_volume(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
int rc;
dbus_int32_t new_vol;
rc = get_single_arg(message, DBUS_TYPE_INT32, &new_vol);
if (rc)
return rc;
cras_system_set_volume(new_vol);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_output_node_volume(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
dbus_int32_t new_vol;
cras_node_id_t id;
DBusError dbus_error;
dbus_error_init(&dbus_error);
if (!dbus_message_get_args(message, &dbus_error,
DBUS_TYPE_UINT64, &id,
DBUS_TYPE_INT32, &new_vol,
DBUS_TYPE_INVALID)) {
syslog(LOG_WARNING,
"Bad method received: %s",
dbus_error.message);
dbus_error_free(&dbus_error);
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
cras_iodev_list_set_node_attr(id, IONODE_ATTR_VOLUME, new_vol);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_swap_left_right(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
cras_node_id_t id;
dbus_bool_t swap;
DBusError dbus_error;
dbus_error_init(&dbus_error);
if (!dbus_message_get_args(message, &dbus_error,
DBUS_TYPE_UINT64, &id,
DBUS_TYPE_BOOLEAN, &swap,
DBUS_TYPE_INVALID)) {
syslog(LOG_WARNING,
"Bad method received: %s",
dbus_error.message);
dbus_error_free(&dbus_error);
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
cras_iodev_list_set_node_attr(id, IONODE_ATTR_SWAP_LEFT_RIGHT,
swap);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_output_mute(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
int rc;
dbus_bool_t new_mute;
rc = get_single_arg(message, DBUS_TYPE_BOOLEAN, &new_mute);
if (rc)
return rc;
cras_system_set_mute(new_mute);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_output_user_mute(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
int rc;
dbus_bool_t new_mute;
rc = get_single_arg(message, DBUS_TYPE_BOOLEAN, &new_mute);
if (rc)
return rc;
cras_system_set_user_mute(new_mute);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_suspend_audio(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
int rc;
dbus_bool_t suspend;
rc = get_single_arg(message, DBUS_TYPE_BOOLEAN, &suspend);
if (rc)
return rc;
cras_system_set_suspended(suspend);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_input_gain(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
int rc;
dbus_int32_t new_gain;
rc = get_single_arg(message, DBUS_TYPE_INT32, &new_gain);
if (rc)
return rc;
cras_system_set_capture_gain(new_gain);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_input_node_gain(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
dbus_int32_t new_gain;
cras_node_id_t id;
DBusError dbus_error;
dbus_error_init(&dbus_error);
if (!dbus_message_get_args(message, &dbus_error,
DBUS_TYPE_UINT64, &id,
DBUS_TYPE_INT32, &new_gain,
DBUS_TYPE_INVALID)) {
syslog(LOG_WARNING,
"Bad method received: %s",
dbus_error.message);
dbus_error_free(&dbus_error);
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
cras_iodev_list_set_node_attr(id, IONODE_ATTR_CAPTURE_GAIN, new_gain);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_input_mute(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
int rc;
dbus_bool_t new_mute;
rc = get_single_arg(message, DBUS_TYPE_BOOLEAN, &new_mute);
if (rc)
return rc;
cras_system_set_capture_mute(new_mute);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_get_volume_state(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
DBusMessage *reply;
dbus_uint32_t serial = 0;
dbus_int32_t volume;
dbus_bool_t system_muted;
dbus_bool_t user_muted;
dbus_int32_t capture_gain;
dbus_bool_t capture_muted;
reply = dbus_message_new_method_return(message);
volume = cras_system_get_volume();
system_muted = cras_system_get_system_mute();
user_muted = cras_system_get_user_mute();
capture_gain = cras_system_get_capture_gain();
capture_muted = cras_system_get_capture_mute();
dbus_message_append_args(reply,
DBUS_TYPE_INT32, &volume,
DBUS_TYPE_BOOLEAN, &system_muted,
DBUS_TYPE_INT32, &capture_gain,
DBUS_TYPE_BOOLEAN, &capture_muted,
DBUS_TYPE_BOOLEAN, &user_muted,
DBUS_TYPE_INVALID);
dbus_connection_send(conn, reply, &serial);
dbus_message_unref(reply);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_get_default_output_buffer_size(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
DBusMessage *reply;
dbus_uint32_t serial = 0;
dbus_int32_t buffer_size;
reply = dbus_message_new_method_return(message);
buffer_size = cras_system_get_default_output_buffer_size();
dbus_message_append_args(reply,
DBUS_TYPE_INT32, &buffer_size,
DBUS_TYPE_INVALID);
dbus_connection_send(conn, reply, &serial);
dbus_message_unref(reply);
return DBUS_HANDLER_RESULT_HANDLED;
}
/* Appends the information about a node to the dbus message. Returns
* false if not enough memory. */
static dbus_bool_t append_node_dict(DBusMessageIter *iter,
const struct cras_iodev_info *dev,
const struct cras_ionode_info *node,
enum CRAS_STREAM_DIRECTION direction)
{
DBusMessageIter dict;
dbus_bool_t is_input;
dbus_uint64_t id;
const char *dev_name = dev->name;
dbus_uint64_t stable_dev_id = node->stable_id;
dbus_uint64_t stable_dev_id_new = node->stable_id_new;
const char *node_type = node->type;
const char *node_name = node->name;
const char *mic_positions = node->mic_positions;
dbus_bool_t active;
dbus_uint64_t plugged_time = node->plugged_time.tv_sec * 1000000ULL +
node->plugged_time.tv_usec;
dbus_uint64_t node_volume = node->volume;
dbus_int64_t node_capture_gain = node->capture_gain;
char *models, *empty_models = "";
is_input = (direction == CRAS_STREAM_INPUT);
id = node->iodev_idx;
id = (id << 32) | node->ionode_idx;
active = !!node->active;
if (!dbus_message_iter_open_container(iter, DBUS_TYPE_ARRAY, "{sv}",
&dict))
return FALSE;
if (!append_key_value(&dict, "IsInput", DBUS_TYPE_BOOLEAN,
DBUS_TYPE_BOOLEAN_AS_STRING, &is_input))
return FALSE;
if (!append_key_value(&dict, "Id", DBUS_TYPE_UINT64,
DBUS_TYPE_UINT64_AS_STRING, &id))
return FALSE;
if (!append_key_value(&dict, "DeviceName", DBUS_TYPE_STRING,
DBUS_TYPE_STRING_AS_STRING, &dev_name))
return FALSE;
if (!append_key_value(&dict, "StableDeviceId", DBUS_TYPE_UINT64,
DBUS_TYPE_UINT64_AS_STRING, &stable_dev_id))
return FALSE;
if (!append_key_value(&dict, "StableDeviceIdNew", DBUS_TYPE_UINT64,
DBUS_TYPE_UINT64_AS_STRING, &stable_dev_id_new))
return FALSE;
if (!append_key_value(&dict, "Type", DBUS_TYPE_STRING,
DBUS_TYPE_STRING_AS_STRING, &node_type))
return FALSE;
if (!append_key_value(&dict, "Name", DBUS_TYPE_STRING,
DBUS_TYPE_STRING_AS_STRING, &node_name))
return FALSE;
if (!append_key_value(&dict, "MicPositions", DBUS_TYPE_STRING,
DBUS_TYPE_STRING_AS_STRING, &mic_positions))
return FALSE;
if (!append_key_value(&dict, "Active", DBUS_TYPE_BOOLEAN,
DBUS_TYPE_BOOLEAN_AS_STRING, &active))
return FALSE;
if (!append_key_value(&dict, "PluggedTime", DBUS_TYPE_UINT64,
DBUS_TYPE_UINT64_AS_STRING, &plugged_time))
return FALSE;
if (!append_key_value(&dict, "NodeVolume", DBUS_TYPE_UINT64,
DBUS_TYPE_UINT64_AS_STRING, &node_volume))
return FALSE;
if (!append_key_value(&dict, "NodeCaptureGain", DBUS_TYPE_INT64,
DBUS_TYPE_INT64_AS_STRING, &node_capture_gain))
return FALSE;
models = cras_iodev_list_get_hotword_models(id);
if (!append_key_value(&dict, "HotwordModels", DBUS_TYPE_STRING,
DBUS_TYPE_STRING_AS_STRING,
models ? &models : &empty_models)) {
free(models);
return FALSE;
}
free(models);
if (!dbus_message_iter_close_container(iter, &dict))
return FALSE;
return TRUE;
}
/* Appends the information about all nodes in a given direction. Returns false
* if not enough memory. */
static dbus_bool_t append_nodes(enum CRAS_STREAM_DIRECTION direction,
DBusMessageIter *array)
{
const struct cras_iodev_info *devs;
const struct cras_ionode_info *nodes;
int ndevs, nnodes;
int i, j;
if (direction == CRAS_STREAM_OUTPUT) {
ndevs = cras_system_state_get_output_devs(&devs);
nnodes = cras_system_state_get_output_nodes(&nodes);
} else {
ndevs = cras_system_state_get_input_devs(&devs);
nnodes = cras_system_state_get_input_nodes(&nodes);
}
for (i = 0; i < nnodes; i++) {
/* Don't reply unplugged nodes. */
if (!nodes[i].plugged)
continue;
/* Find the device for this node. */
for (j = 0; j < ndevs; j++)
if (devs[j].idx == nodes[i].iodev_idx)
break;
if (j == ndevs)
continue;
/* Send information about this node. */
if (!append_node_dict(array, &devs[j], &nodes[i], direction))
return FALSE;
}
return TRUE;
}
static DBusHandlerResult handle_get_nodes(DBusConnection *conn,
DBusMessage *message,
void *arg)
{
DBusMessage *reply;
DBusMessageIter array;
dbus_uint32_t serial = 0;
reply = dbus_message_new_method_return(message);
dbus_message_iter_init_append(reply, &array);
if (!append_nodes(CRAS_STREAM_OUTPUT, &array))
return DBUS_HANDLER_RESULT_NEED_MEMORY;
if (!append_nodes(CRAS_STREAM_INPUT, &array))
return DBUS_HANDLER_RESULT_NEED_MEMORY;
dbus_connection_send(conn, reply, &serial);
dbus_message_unref(reply);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_get_system_aec_supported(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
DBusMessage *reply;
dbus_uint32_t serial = 0;
dbus_bool_t system_aec_supported;
reply = dbus_message_new_method_return(message);
system_aec_supported = cras_system_get_aec_supported();
dbus_message_append_args(reply,
DBUS_TYPE_BOOLEAN, &system_aec_supported,
DBUS_TYPE_INVALID);
dbus_connection_send(conn, reply, &serial);
dbus_message_unref(reply);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult
handle_set_active_node(DBusConnection *conn,
DBusMessage *message,
void *arg,
enum CRAS_STREAM_DIRECTION direction)
{
int rc;
cras_node_id_t id;
rc = get_single_arg(message, DBUS_TYPE_UINT64, &id);
if (rc)
return rc;
cras_iodev_list_select_node(direction, id);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult
handle_add_active_node(DBusConnection *conn,
DBusMessage *message,
void *arg,
enum CRAS_STREAM_DIRECTION direction)
{
int rc;
cras_node_id_t id;
rc = get_single_arg(message, DBUS_TYPE_UINT64, &id);
if (rc)
return rc;
cras_iodev_list_add_active_node(direction, id);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult
handle_rm_active_node(DBusConnection *conn,
DBusMessage *message,
void *arg,
enum CRAS_STREAM_DIRECTION direction)
{
int rc;
cras_node_id_t id;
rc = get_single_arg(message, DBUS_TYPE_UINT64, &id);
if (rc)
return rc;
cras_iodev_list_rm_active_node(direction, id);
send_empty_reply(conn, message);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_get_num_active_streams(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
send_int32_reply(conn, message, cras_system_state_get_active_streams());
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_get_num_active_streams_use_input_hw(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
dbus_int32_t num = 0;
unsigned i;
for (i = 0; i < CRAS_NUM_DIRECTIONS; i++) {
if (cras_stream_uses_input_hw(i))
num += cras_system_state_get_active_streams_by_direction(i);
}
send_int32_reply(conn, message, num);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_get_num_active_streams_use_output_hw(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
dbus_int32_t num = 0;
unsigned i;
for (i = 0; i < CRAS_NUM_DIRECTIONS; i++) {
if (cras_stream_uses_output_hw(i))
num += cras_system_state_get_active_streams_by_direction(i);
}
send_int32_reply(conn, message, num);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_global_output_channel_remix(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
dbus_int32_t num_channels;
double *coeff_array;
dbus_int32_t count;
DBusError dbus_error;
float *coefficient;
int i;
dbus_error_init(&dbus_error);
if (!dbus_message_get_args(message, &dbus_error,
DBUS_TYPE_INT32, &num_channels,
DBUS_TYPE_ARRAY,
DBUS_TYPE_DOUBLE, &coeff_array, &count,
DBUS_TYPE_INVALID)) {
syslog(LOG_WARNING, "Set global output channel remix error: %s",
dbus_error.message);
dbus_error_free(&dbus_error);
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
coefficient = (float *)calloc(count, sizeof(*coefficient));
if (!coefficient)
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
for (i = 0; i < count; i++)
coefficient[i] = coeff_array[i];
audio_thread_config_global_remix(
cras_iodev_list_get_audio_thread(),
num_channels,
coefficient);
send_empty_reply(conn, message);
free(coefficient);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_set_hotword_model(
DBusConnection *conn,
DBusMessage *message,
void *arg)
{
cras_node_id_t id;
const char *model_name;
DBusError dbus_error;
dbus_int32_t ret;
dbus_error_init(&dbus_error);
if (!dbus_message_get_args(message, &dbus_error,
DBUS_TYPE_UINT64, &id,
DBUS_TYPE_STRING, &model_name,
DBUS_TYPE_INVALID)) {
syslog(LOG_WARNING,
"Bad method received: %s",
dbus_error.message);
dbus_error_free(&dbus_error);
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
ret = cras_iodev_list_set_hotword_model(id, model_name);
send_int32_reply(conn, message, ret);
return DBUS_HANDLER_RESULT_HANDLED;
}
static DBusHandlerResult handle_is_audio_active(
DBusConnection *conn,
DBusMessage *message,
void* arg)
{
dbus_int32_t active = cras_system_state_get_non_empty_status();
send_int32_reply(conn, message, active);
return DBUS_HANDLER_RESULT_HANDLED;
}
/* Handle incoming messages. */
static DBusHandlerResult handle_control_message(DBusConnection *conn,
DBusMessage *message,
void *arg)
{
syslog(LOG_DEBUG, "Control message: %s %s %s",
dbus_message_get_path(message),
dbus_message_get_interface(message),
dbus_message_get_member(message));
if (dbus_message_is_method_call(message,
DBUS_INTERFACE_INTROSPECTABLE,
"Introspect")) {
DBusMessage *reply;
const char *xml = CONTROL_INTROSPECT_XML;
reply = dbus_message_new_method_return(message);
if (!reply)
return DBUS_HANDLER_RESULT_NEED_MEMORY;
if (!dbus_message_append_args(reply,
DBUS_TYPE_STRING, &xml,
DBUS_TYPE_INVALID))
return DBUS_HANDLER_RESULT_NEED_MEMORY;
if (!dbus_connection_send(conn, reply, NULL))
return DBUS_HANDLER_RESULT_NEED_MEMORY;
dbus_message_unref(reply);
return DBUS_HANDLER_RESULT_HANDLED;
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetOutputVolume")) {
return handle_set_output_volume(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetOutputNodeVolume")) {
return handle_set_output_node_volume(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SwapLeftRight")) {
return handle_swap_left_right(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetOutputMute")) {
return handle_set_output_mute(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetOutputUserMute")) {
return handle_set_output_user_mute(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetSuspendAudio")) {
return handle_set_suspend_audio(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetInputGain")) {
return handle_set_input_gain(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetInputNodeGain")) {
return handle_set_input_node_gain(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetInputMute")) {
return handle_set_input_mute(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetVolumeState")) {
return handle_get_volume_state(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetDefaultOutputBufferSize")) {
return handle_get_default_output_buffer_size(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetNodes")) {
return handle_get_nodes(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetSystemAecSupported")) {
return handle_get_system_aec_supported(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetActiveOutputNode")) {
return handle_set_active_node(conn, message, arg,
CRAS_STREAM_OUTPUT);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetActiveInputNode")) {
return handle_set_active_node(conn, message, arg,
CRAS_STREAM_INPUT);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"AddActiveInputNode")) {
return handle_add_active_node(conn, message, arg,
CRAS_STREAM_INPUT);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"AddActiveOutputNode")) {
return handle_add_active_node(conn, message, arg,
CRAS_STREAM_OUTPUT);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"RemoveActiveInputNode")) {
return handle_rm_active_node(conn, message, arg,
CRAS_STREAM_INPUT);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"RemoveActiveOutputNode")) {
return handle_rm_active_node(conn, message, arg,
CRAS_STREAM_OUTPUT);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetNumberOfActiveStreams")) {
return handle_get_num_active_streams(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetNumberOfActiveInputStreams")) {
return handle_get_num_active_streams_use_input_hw(
conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"GetNumberOfActiveOutputStreams")) {
return handle_get_num_active_streams_use_output_hw(
conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetGlobalOutputChannelRemix")) {
return handle_set_global_output_channel_remix(
conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"SetHotwordModel")) {
return handle_set_hotword_model(conn, message, arg);
} else if (dbus_message_is_method_call(message,
CRAS_CONTROL_INTERFACE,
"IsAudioOutputActive")) {
return handle_is_audio_active(conn, message, arg);
}
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}
/* Creates a new DBus message, must be freed with dbus_message_unref. */
static DBusMessage *create_dbus_message(const char *name)
{
DBusMessage *msg;
msg = dbus_message_new_signal(CRAS_ROOT_OBJECT_PATH,
CRAS_CONTROL_INTERFACE,
name);
if (!msg)
syslog(LOG_ERR, "Failed to create signal");
return msg;
}
/* Handlers for system updates that generate DBus signals. */
static void signal_output_volume(void *context, int32_t volume)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("OutputVolumeChanged");
if (!msg)
return;
volume = cras_system_get_volume();
dbus_message_append_args(msg,
DBUS_TYPE_INT32, &volume,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_output_mute(void *context, int muted, int user_muted,
int mute_locked)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("OutputMuteChanged");
if (!msg)
return;
muted = cras_system_get_system_mute();
user_muted = cras_system_get_user_mute();
dbus_message_append_args(msg,
DBUS_TYPE_BOOLEAN, &muted,
DBUS_TYPE_BOOLEAN, &user_muted,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_capture_gain(void *context, int32_t gain)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("InputGainChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_INT32, &gain,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_capture_mute(void *context, int muted, int mute_locked)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("InputMuteChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_BOOLEAN, &muted,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_nodes_changed(void *context)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("NodesChanged");
if (!msg)
return;
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_active_node_changed(void *context,
enum CRAS_STREAM_DIRECTION dir,
cras_node_id_t node_id)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
DBusMessage *msg;
dbus_uint32_t serial = 0;
msg = create_dbus_message((dir == CRAS_STREAM_OUTPUT)
? "ActiveOutputNodeChanged"
: "ActiveInputNodeChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_UINT64, &node_id,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
/* Called by iodev_list when a node volume changes. */
static void signal_node_volume_changed(void *context,
cras_node_id_t node_id,
int32_t volume)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("OutputNodeVolumeChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_UINT64, &node_id,
DBUS_TYPE_INT32, &volume,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_node_capture_gain_changed(void *context,
cras_node_id_t node_id,
int capture_gain)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("InputNodeGainChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_UINT64, &node_id,
DBUS_TYPE_INT32, &capture_gain,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_node_left_right_swapped_changed(void *context,
cras_node_id_t node_id,
int swapped)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("NodeLeftRightSwappedChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_UINT64, &node_id,
DBUS_TYPE_BOOLEAN, &swapped,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_num_active_streams_changed(void *context,
enum CRAS_STREAM_DIRECTION dir,
uint32_t num_active_streams)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
dbus_int32_t num;
msg = create_dbus_message("NumberOfActiveStreamsChanged");
if (!msg)
return;
num = cras_system_state_get_active_streams();
dbus_message_append_args(msg,
DBUS_TYPE_INT32, &num,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_hotword_triggered(void *context,
int64_t tv_sec,
int64_t tv_nsec)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("HotwordTriggered");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_INT64, &tv_sec,
DBUS_TYPE_INT64, &tv_nsec,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
static void signal_non_empty_audio_state_changed(void *context, int non_empty)
{
struct cras_dbus_control *control = (struct cras_dbus_control *)context;
dbus_uint32_t serial = 0;
DBusMessage *msg;
msg = create_dbus_message("AudioOutputActiveStateChanged");
if (!msg)
return;
dbus_message_append_args(msg,
DBUS_TYPE_BOOLEAN, &non_empty,
DBUS_TYPE_INVALID);
dbus_connection_send(control->conn, msg, &serial);
dbus_message_unref(msg);
}
/* Exported Interface */
void cras_dbus_control_start(DBusConnection *conn)
{
static const DBusObjectPathVTable control_vtable = {
.message_function = handle_control_message,
};
DBusError dbus_error;
struct cras_observer_ops observer_ops;
dbus_control.conn = conn;
dbus_connection_ref(dbus_control.conn);
if (!dbus_connection_register_object_path(conn,
CRAS_ROOT_OBJECT_PATH,
&control_vtable,
&dbus_error)) {
syslog(LOG_WARNING,
"Couldn't register CRAS control: %s: %s",
CRAS_ROOT_OBJECT_PATH, dbus_error.message);
dbus_error_free(&dbus_error);
return;
}
memset(&observer_ops, 0, sizeof(observer_ops));
observer_ops.output_volume_changed = signal_output_volume;
observer_ops.output_mute_changed = signal_output_mute;
observer_ops.capture_gain_changed = signal_capture_gain;
observer_ops.capture_mute_changed = signal_capture_mute;
observer_ops.num_active_streams_changed =
signal_num_active_streams_changed;
observer_ops.nodes_changed = signal_nodes_changed;
observer_ops.active_node_changed = signal_active_node_changed;
observer_ops.input_node_gain_changed = signal_node_capture_gain_changed;
observer_ops.output_node_volume_changed = signal_node_volume_changed;
observer_ops.node_left_right_swapped_changed =
signal_node_left_right_swapped_changed;
observer_ops.hotword_triggered = signal_hotword_triggered;
observer_ops.non_empty_audio_state_changed =
signal_non_empty_audio_state_changed;
dbus_control.observer = cras_observer_add(&observer_ops, &dbus_control);
}
void cras_dbus_control_stop()
{
if (!dbus_control.conn)
return;
dbus_connection_unregister_object_path(dbus_control.conn,
CRAS_ROOT_OBJECT_PATH);
dbus_connection_unref(dbus_control.conn);
dbus_control.conn = NULL;
cras_observer_remove(dbus_control.observer);
dbus_control.observer = NULL;
}