diff options
Diffstat (limited to 'audio')
-rw-r--r-- | audio/audio_hw.c | 179 |
1 files changed, 51 insertions, 128 deletions
diff --git a/audio/audio_hw.c b/audio/audio_hw.c index b431548..2d23e9c 100644 --- a/audio/audio_hw.c +++ b/audio/audio_hw.c @@ -211,96 +211,6 @@ struct route_setting earpiece_switch[] = { }, }; -/* The following four routes deliberately ensure that -the new mixer is enabled before the old are disabled */ -struct route_setting speaker_mm[] = { - { - .ctl_name = MIXER_DL2_MIXER_MULTIMEDIA, - .intval = 1, - }, - { - .ctl_name = MIXER_DL1_MIXER_MULTIMEDIA, - .intval = 0, - }, - { - .ctl_name = MIXER_DL1_MIXER_VOICE, - .intval = 0, - }, - { - .ctl_name = MIXER_DL2_MIXER_VOICE, - .intval = 0, - }, - { - .ctl_name = NULL, - }, -}; - -struct route_setting headset_mm[] = { - { - .ctl_name = MIXER_DL1_MIXER_MULTIMEDIA, - .intval = 1, - }, - { - .ctl_name = MIXER_DL1_MIXER_VOICE, - .intval = 0, - }, - { - .ctl_name = MIXER_DL2_MIXER_MULTIMEDIA, - .intval = 0, - }, - { - .ctl_name = MIXER_DL2_MIXER_VOICE, - .intval = 0, - }, - { - .ctl_name = NULL, - }, -}; - -struct route_setting speaker_vx[] = { - { - .ctl_name = MIXER_DL2_MIXER_VOICE, - .intval = 1, - }, - { - .ctl_name = MIXER_DL1_MIXER_MULTIMEDIA, - .intval = 0, - }, - { - .ctl_name = MIXER_DL1_MIXER_VOICE, - .intval = 0, - }, - { - .ctl_name = MIXER_DL2_MIXER_MULTIMEDIA, - .intval = 0, - }, - { - .ctl_name = NULL, - }, -}; - -struct route_setting headset_vx[] = { - { - .ctl_name = MIXER_DL1_MIXER_VOICE, - .intval = 1, - }, - { - .ctl_name = MIXER_DL1_MIXER_MULTIMEDIA, - .intval = 0, - }, - { - .ctl_name = MIXER_DL2_MIXER_MULTIMEDIA, - .intval = 0, - }, - { - .ctl_name = MIXER_DL2_MIXER_VOICE, - .intval = 0, - }, - { - .ctl_name = NULL, - }, -}; - struct route_setting amic_vx[] = { { .ctl_name = MIXER_MUX_VX0, @@ -319,11 +229,20 @@ struct route_setting amic_vx[] = { }, }; +struct mixer_ctls +{ + struct mixer_ctl *headset_mm; + struct mixer_ctl *headset_vx; + struct mixer_ctl *speaker_mm; + struct mixer_ctl *speaker_vx; +}; + struct tuna_audio_device { struct audio_hw_device device; pthread_mutex_t lock; struct mixer *mixer; + struct mixer_ctls mixer_ctls; int mode; int out_device; struct pcm *pcm_modem_dl; @@ -450,9 +369,6 @@ static void select_mode(struct tuna_audio_device *adev) if (adev->mode == AUDIO_MODE_IN_CALL) { if (!adev->in_call) { set_route_by_array(adev->mixer, amic_vx, 1); - /* force headset voice route otherwise microphone - does not function */ - set_route_by_array(adev->mixer, headset_vx, 1); start_call(adev); adev->in_call = 1; } @@ -465,41 +381,36 @@ static void select_mode(struct tuna_audio_device *adev) } } -/* Note: currently the headset/earpiece route gets priority -over speaker if both are selected as output devices. */ static void select_output_device(struct tuna_audio_device *adev) { - struct mixer_ctl *ctl; + int headset_on; + int speaker_on; + + /* tear down call stream before changing route, + otherwise microphone does not function */ + if (adev->in_call) + end_call(adev); + + headset_on = adev->out_device & AUDIO_DEVICE_OUT_ALL_HEADSET; + speaker_on = adev->out_device & AUDIO_DEVICE_OUT_SPEAKER; + + /* Select output route(s) */ + mixer_ctl_set_value(adev->mixer_ctls.headset_mm, 0, + headset_on && !adev->in_call); + mixer_ctl_set_value(adev->mixer_ctls.headset_vx, 0, + headset_on && adev->in_call); + mixer_ctl_set_value(adev->mixer_ctls.speaker_mm, 0, + speaker_on && !adev->in_call); + mixer_ctl_set_value(adev->mixer_ctls.speaker_vx, 0, + speaker_on && adev->in_call); + + if (adev->out_device & AUDIO_DEVICE_OUT_EARPIECE) + set_route_by_array(adev->mixer, earpiece_switch, 1); + else + set_route_by_array(adev->mixer, earpiece_switch, 0); - /* Select output device */ - if (adev->out_device & AUDIO_DEVICE_OUT_SPEAKER) { - if (adev->in_call) { - /* tear down call stream before changing route, - otherwise microphone does not function */ - end_call(adev); - set_route_by_array(adev->mixer, speaker_vx, 1); - start_call(adev); - } else - set_route_by_array(adev->mixer, speaker_mm, 1); - } else if (adev->out_device & AUDIO_DEVICE_OUT_ALL_HEADSET) { - if (adev->in_call) { - /* tear down call stream before changing route, - otherwise microphone does not function */ - end_call(adev); - set_route_by_array(adev->mixer, headset_vx, 1); - if (adev->out_device & AUDIO_DEVICE_OUT_EARPIECE) - set_route_by_array(adev->mixer, earpiece_switch, 1); - else - set_route_by_array(adev->mixer, earpiece_switch, 0); - start_call(adev); - } else { - set_route_by_array(adev->mixer, headset_mm, 1); - if (adev->out_device & AUDIO_DEVICE_OUT_EARPIECE) - set_route_by_array(adev->mixer, earpiece_switch, 1); - else - set_route_by_array(adev->mixer, earpiece_switch, 0); - } - } + if (adev->in_call) + start_call(adev); } static int start_output_stream(struct tuna_stream_out *out) @@ -683,9 +594,6 @@ static int start_input_stream(struct tuna_stream_in *in) struct tuna_audio_device *adev = in->dev; set_route_by_array(adev->mixer, amic_vx, 1); - /* force headset voice route otherwise microphone - does not function */ - set_route_by_array(adev->mixer, headset_vx, 1); /* this assumes routing is done previously */ in->pcm = pcm_open(0, in->port, PCM_IN, &in->config); @@ -1124,6 +1032,21 @@ static int adev_open(const hw_module_t* module, const char* name, return -ENOMEM; } + adev->mixer_ctls.headset_mm = mixer_get_ctl_by_name(adev->mixer, + MIXER_DL1_MIXER_MULTIMEDIA); + adev->mixer_ctls.headset_vx = mixer_get_ctl_by_name(adev->mixer, + MIXER_DL1_MIXER_VOICE); + adev->mixer_ctls.speaker_mm = mixer_get_ctl_by_name(adev->mixer, + MIXER_DL2_MIXER_MULTIMEDIA); + adev->mixer_ctls.speaker_vx = mixer_get_ctl_by_name(adev->mixer, + MIXER_DL2_MIXER_VOICE); + if (!adev->mixer_ctls.headset_mm || !adev->mixer_ctls.headset_vx || + !adev->mixer_ctls.speaker_mm || !adev->mixer_ctls.speaker_vx) { + mixer_close(adev->mixer); + free(adev); + return -ENOMEM; + } + /* Set the default route before the PCM stream is opened */ set_route_by_array(adev->mixer, defaults, 1); adev->mode = AUDIO_MODE_NORMAL; |