/* WirePlumber * * Copyright © 2023 Collabora Ltd. * @author Ashok Sidipotu * * SPDX-License-Identifier: MIT */ #include #include #include #include #include "audiomixer.h" typedef struct _AudioMixerTest AudioMixerTest; struct _AudioMixerTest { struct audiomixer *am; WpCore *core; const struct mixer_control **ctrls; unsigned int nctrls; GMainLoop *loop; }; static void audio_mixer_clear (AudioMixerTest *self) { audiomixer_free (self->am); g_clear_pointer (&self->loop, g_main_loop_unref); } static gint set_mute (AudioMixerTest *self, gint id) { gint ret = -1; gboolean mute = FALSE; int i; for (i = 0; i < self->nctrls; i++) { const struct mixer_control *ctrl = self->ctrls[i]; if (id == i) { if (g_str_equal ("bass", ctrl->name) || g_str_equal ("treble", ctrl->name)) fprintf (stderr, "mute cannot be applied for %s control\n", ctrl->name); else { /* toggle mute value */ mute = (ctrl->mute) ? FALSE : TRUE; audiomixer_change_mute (self->am, ctrl, mute); ret = TRUE; } break; } } return ret; } static gint set_gain (AudioMixerTest *self, gint id, gfloat gain) { gint ret = -1; int i; for (i = 0; i < self->nctrls; i++) { const struct mixer_control *ctrl = self->ctrls[i]; if(id == i) { if (g_str_equal ("bass", ctrl->name) || g_str_equal ("treble", ctrl->name)) { if (fabs (ctrl->gain - gain) < 0.00001) fprintf (stderr, "gain already at requested level %f\n", ctrl->gain); else { audiomixer_change_gain (self->am, ctrl, gain); ret = TRUE; } } else fprintf (stderr, "gain cannot be applied for %s control\n", ctrl->name); break; } } return ret; } static gint set_volume (AudioMixerTest *self, gint id, double vol) { gint ret = -1; int i; for (i = 0; i < self->nctrls; i++) { const struct mixer_control *ctrl = self->ctrls[i]; if (id == i) { if (g_str_equal ("bass", ctrl->name) || g_str_equal ("treble", ctrl->name)) fprintf (stderr, "volume cannot be applied for %s control\n", ctrl->name); else { if (fabs (ctrl->volume - vol) < 0.00001) fprintf (stderr, "volume is already at requested level %f\n", ctrl->volume); else { audiomixer_change_volume (self->am, ctrl, vol); ret = TRUE; } } break; } } return ret; } static gint set_channels_volume (AudioMixerTest *self, gint id, double lvol, double rvol) { gint ret = -1; int i; gboolean set_left_channel_volume = TRUE; gboolean set_right_channel_volume = TRUE; for (i = 0; i < self->nctrls; i++) { const struct mixer_control *ctrl = self->ctrls[i]; if (id == i) { if (g_str_equal ("bass", ctrl->name) || g_str_equal ("treble", ctrl->name)) fprintf (stderr, "volume cannot be applied for %s control\n", ctrl->name); else { if (fabs (ctrl->lvolume - lvol) < 0.00001) { set_left_channel_volume = FALSE; lvol = ctrl->lvolume; fprintf (stderr, "left volume is already at requested level %f\n", ctrl->lvolume); } if (fabs (ctrl->rvolume - rvol) < 0.00001) { set_right_channel_volume = FALSE; rvol = ctrl->rvolume; fprintf (stderr, "right volume is already at requested level %f\n", ctrl->rvolume); } if (set_left_channel_volume || set_right_channel_volume) { audiomixer_change_channel_volume (self->am, ctrl, lvol, rvol); ret = TRUE; } } break; } } return ret; } static void print_ctrls (AudioMixerTest *self) { const struct mixer_control **ctrls = self->ctrls; unsigned int nctrls = self->nctrls; int i; fprintf (stdout, "\nControls:"); for (i = 0; i < nctrls; i++) { const struct mixer_control *ctrl = ctrls[i]; if (g_str_equal ("bass", ctrl->name) || g_str_equal ("treble", ctrl->name)) fprintf(stdout, "\n%2d. %-25s [gain: %.2f]", i, ctrl->name, ctrl->gain); else fprintf (stdout, "\n%2d. %-25s [vol: %.2f, lvol: %.2f, rvol: %.2f," " mute:%d]", i, ctrl->name, ctrl->volume, ctrl->lvolume, ctrl->rvolume, ctrl->mute); } fprintf (stdout, "\n"); } static void refresh_ctrls (AudioMixerTest *self) { self->ctrls = audiomixer_get_active_controls (self->am, &self->nctrls); print_ctrls (self); } static void show_help (void) { fprintf (stdout, "\n" " -h, --help Show this help\n" " -p, --print-controls prints controls\n" " -i, --id control id (serial#) of the control to apply volume changes to\n" " -v, --set-volume Set volume level for a volume control (all controls except bass and treble)\n" " This option sets volume for all the channels, see -l or -r for individual channels\n" " Examples:\n" " audio-mixer-test -v 0.2 -> sets volume (for all channels)\n" " of the 1st control (default id) to 0.2\n" " audio-mixer-test -i 9 -v 0.2 -> sets volume of the 10th control to 0.2\n" " -l, --left-chan-vol Set left channel volume for a volume control (all controls except bass and treble)\n" " -r, --right-chan-vol Set right channel volume for a volume control (all controls except bass and treble)\n" " Only stereo channels are supported for now, and vol update on both the channels is required\n" " Examples:\n" " audio-mixer-test -l 0.2 -> gives an error as only one channel is updated\n" " audio-mixer-test -l 0.1 -r 0.2 -> sets left channel volume to 0.1 and right channel volume\n" " to 0.2 on the 1st control (default id)\n" " audio-mixer-test -i 9 -l 0.1 -r 0.2 -> sets left channel volume to 0.1 and right channel volume\n" " to 0.2 on the 10th control\n" " -g, --set-gain Set gain level for gain controls like bass and treble\n" " Examples:\n" " audio-mixer-test -i 12 -g 0.8 -> sets gain of the 13th control to 0.8\n" " -m, --set-mute mute/unmute volume controls (all controls except bass and treble)\n" " Examples:\n" " audio-mixer-test -m -> mutes or unmutes the 1st control (default id)\n" " depending on the previous mute state (toggle switch)\n" " audio-mixer-test -i 9 -m -> toggles mute on the 10th control\n"); } static const struct option long_options[] = { { "help", no_argument, NULL, 'h' }, { "print-controls", no_argument, NULL, 'p' }, { "id", required_argument, NULL, 'i' }, { "left-chan-vol", required_argument, NULL, 'l' }, { "right-chan-vol", required_argument, NULL, 'r' }, { "set-volume", required_argument, NULL, 'v' }, { "set-mute", no_argument, NULL, 'm' }, { "set-gain", required_argument, NULL, 'g' }, { NULL, 0, NULL, 0} }; static void mixer_value_change_cb (void *data, unsigned int change_mask, const struct mixer_control *ctrl) { AudioMixerTest *self = (AudioMixerTest *)data; g_main_loop_quit (self->loop); } static void wait_for_change (AudioMixerTest *self) { audiomixer_unlock (self->am); g_main_loop_run (self->loop); audiomixer_lock (self->am); refresh_ctrls (self); } gint main (gint argc, gchar **argv) { AudioMixerTest self = { 0 }; gint c, ret = 0; struct audiomixer_events audiomixer_events = { 0 }; gint id = -1; double vol = 0.0, lvol = 0.0, rvol = 0.0; gfloat gain = 0.0; gboolean set_right_channel_volume = FALSE; gboolean set_left_channel_volume = FALSE; if (argc == 1) { fprintf (stderr, "no options given\n"); show_help (); return 0; } while ((c = getopt_long (argc, argv, "hpi:v:mg:l:r:", long_options, NULL)) != -1) { switch(c) { case 'h': show_help (); return 0; default: break; } } self.loop = g_main_loop_new (NULL, FALSE); self.am = audiomixer_new (); if (!self.am) { fprintf (stderr, "unable to open audiomixer\n"); goto exit; } audiomixer_lock (self.am); ret = audiomixer_ensure_controls (self.am, 3); if (ret < 0) { fprintf (stderr, "ensure controls failed\n"); goto exit; } self.ctrls = audiomixer_get_active_controls (self.am, &self.nctrls); audiomixer_events.value_changed = mixer_value_change_cb; audiomixer_add_event_listener (self.am, &audiomixer_events, (void *)&self); optind = 1; /* reset option scanning */ while ((c = getopt_long (argc, argv, "hpi:v:mg:l:r:", long_options, NULL)) != -1) { switch(c) { case 'p': print_ctrls (&self); break; case 'i': id = atoi (optarg); if (!(id >= 0 && id < self.nctrls)) { ret = -1; fprintf (stderr, "id '%d' is invalid\n", id); } break; case 'v': vol = (double)atof (optarg); if (id == -1) { fprintf (stderr, "control id not given, defaulting it to 0 (Master Playback)\n"); id = 0; } ret = set_volume (&self, id, vol); if (ret != TRUE) fprintf (stderr, "set-volume failed"); else /* wait for volume to be acked */ wait_for_change (&self); break; case 'l': if (id == -1) { fprintf (stderr, "control id not given, defaulting it to 0 (Master Playback)\n"); id = 0; } lvol = (double)atof (optarg); set_left_channel_volume = TRUE; break; case 'r': if (id == -1) { fprintf (stderr, "control id not given, defaulting it to 0 (Master Playback)\n"); id = 0; } rvol = (double)atof (optarg); set_right_channel_volume = TRUE; break; case 'm': if (id == -1) { fprintf (stderr, "control id not given, defaulting it to 0 (Master Playback)\n"); id = 0; } ret = set_mute (&self, id); if (ret != TRUE) fprintf (stderr, "set-mute failed\n"); else /* wait for mute to be acked */ wait_for_change (&self); break; case 'g': gain = atof (optarg); if (id == -1) { fprintf (stderr, "control id not given defaulting it to 11 (bass)\n"); id = 11; /* bass ctrl */ } ret = set_gain (&self, id, gain); if (ret != TRUE) fprintf (stderr, "set-gain failed\n"); else /* wait for gain to be acked */ wait_for_change (&self); break; default: show_help (); break; } } if (set_left_channel_volume && set_right_channel_volume) { ret = set_channels_volume (&self, id, lvol, rvol); if (ret != TRUE) fprintf (stderr, "set_channels_volume failed\n"); else /* wait for volume to be acked */ wait_for_change (&self); } else if (set_left_channel_volume || set_right_channel_volume) { fprintf (stderr, "set volume of both left and right channels\n"); } exit: /* clean up at program exit */ audiomixer_unlock (self.am); audio_mixer_clear (&self); return ret; }