mirror of
https://github.com/team-infusion-developers/android_hardware_samsung.git
synced 2024-11-06 21:55:41 +00:00
audio: Correctly handle out of memory conditions
Change-Id: Id7e0ab1085d3e38c1eaa7905a17726e9d1a68cc9
This commit is contained in:
parent
dc15cec22a
commit
56204f6d19
2 changed files with 70 additions and 7 deletions
|
@ -356,6 +356,7 @@ static int mixer_init(struct audio_device *adev)
|
||||||
char mixer_path[PATH_MAX];
|
char mixer_path[PATH_MAX];
|
||||||
struct mixer_card *mixer_card;
|
struct mixer_card *mixer_card;
|
||||||
struct listnode *node;
|
struct listnode *node;
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
list_init(&adev->mixer_list);
|
list_init(&adev->mixer_list);
|
||||||
|
|
||||||
|
@ -369,6 +370,7 @@ static int mixer_init(struct audio_device *adev)
|
||||||
if (++retry_num > RETRY_NUMBER) {
|
if (++retry_num > RETRY_NUMBER) {
|
||||||
ALOGE("%s unable to open the mixer for--card %d, aborting.",
|
ALOGE("%s unable to open the mixer for--card %d, aborting.",
|
||||||
__func__, card);
|
__func__, card);
|
||||||
|
ret = -ENODEV;
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
usleep(RETRY_US);
|
usleep(RETRY_US);
|
||||||
|
@ -380,9 +382,14 @@ static int mixer_init(struct audio_device *adev)
|
||||||
if (!audio_route) {
|
if (!audio_route) {
|
||||||
ALOGE("%s: Failed to init audio route controls for card %d, aborting.",
|
ALOGE("%s: Failed to init audio route controls for card %d, aborting.",
|
||||||
__func__, card);
|
__func__, card);
|
||||||
|
ret = -ENODEV;
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
mixer_card = calloc(1, sizeof(struct mixer_card));
|
mixer_card = calloc(1, sizeof(struct mixer_card));
|
||||||
|
if (mixer_card == NULL) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
mixer_card->card = card;
|
mixer_card->card = card;
|
||||||
mixer_card->mixer = mixer;
|
mixer_card->mixer = mixer;
|
||||||
mixer_card->audio_route = audio_route;
|
mixer_card->audio_route = audio_route;
|
||||||
|
@ -394,7 +401,7 @@ static int mixer_init(struct audio_device *adev)
|
||||||
|
|
||||||
error:
|
error:
|
||||||
free_mixer_list(adev);
|
free_mixer_list(adev);
|
||||||
return -ENODEV;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *get_snd_device_name(snd_device_t snd_device)
|
static const char *get_snd_device_name(snd_device_t snd_device)
|
||||||
|
@ -1178,6 +1185,9 @@ static int get_hw_echo_reference(struct stream_in *in)
|
||||||
}
|
}
|
||||||
|
|
||||||
ref_device = (struct pcm_device *)calloc(1, sizeof(struct pcm_device));
|
ref_device = (struct pcm_device *)calloc(1, sizeof(struct pcm_device));
|
||||||
|
if (ref_device == NULL) {
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
ref_device->pcm_profile = ref_pcm_profile;
|
ref_device->pcm_profile = ref_pcm_profile;
|
||||||
|
|
||||||
ALOGV("%s: ref_device rate:%d, ch:%d", __func__, ref_pcm_profile->config.rate, ref_pcm_profile->config.channels);
|
ALOGV("%s: ref_device rate:%d, ch:%d", __func__, ref_pcm_profile->config.rate, ref_pcm_profile->config.channels);
|
||||||
|
@ -1932,6 +1942,10 @@ static int start_input_stream(struct stream_in *in)
|
||||||
}
|
}
|
||||||
|
|
||||||
uc_info = (struct audio_usecase *)calloc(1, sizeof(struct audio_usecase));
|
uc_info = (struct audio_usecase *)calloc(1, sizeof(struct audio_usecase));
|
||||||
|
if (uc_info == NULL) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto error_config;
|
||||||
|
}
|
||||||
uc_info->id = in->usecase;
|
uc_info->id = in->usecase;
|
||||||
uc_info->type = PCM_CAPTURE;
|
uc_info->type = PCM_CAPTURE;
|
||||||
uc_info->stream = (struct audio_stream *)in;
|
uc_info->stream = (struct audio_stream *)in;
|
||||||
|
@ -1940,6 +1954,12 @@ static int start_input_stream(struct stream_in *in)
|
||||||
uc_info->out_snd_device = SND_DEVICE_NONE;
|
uc_info->out_snd_device = SND_DEVICE_NONE;
|
||||||
|
|
||||||
pcm_device = (struct pcm_device *)calloc(1, sizeof(struct pcm_device));
|
pcm_device = (struct pcm_device *)calloc(1, sizeof(struct pcm_device));
|
||||||
|
if (pcm_device == NULL) {
|
||||||
|
free(uc_info);
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto error_config;
|
||||||
|
}
|
||||||
|
|
||||||
pcm_device->pcm_profile = pcm_profile;
|
pcm_device->pcm_profile = pcm_profile;
|
||||||
list_init(&in->pcm_dev_list);
|
list_init(&in->pcm_dev_list);
|
||||||
list_add_tail(&in->pcm_dev_list, &pcm_device->stream_list_node);
|
list_add_tail(&in->pcm_dev_list, &pcm_device->stream_list_node);
|
||||||
|
@ -2119,6 +2139,9 @@ static int uc_select_pcm_devices(struct audio_usecase *usecase)
|
||||||
|
|
||||||
while ((pcm_profile = get_pcm_device(usecase->type, devices)) != NULL) {
|
while ((pcm_profile = get_pcm_device(usecase->type, devices)) != NULL) {
|
||||||
pcm_device = calloc(1, sizeof(struct pcm_device));
|
pcm_device = calloc(1, sizeof(struct pcm_device));
|
||||||
|
if (pcm_device == NULL) {
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
pcm_device->pcm_profile = pcm_profile;
|
pcm_device->pcm_profile = pcm_profile;
|
||||||
list_add_tail(&out->pcm_dev_list, &pcm_device->stream_list_node);
|
list_add_tail(&out->pcm_dev_list, &pcm_device->stream_list_node);
|
||||||
mixer_card = uc_get_mixer_for_card(usecase, pcm_profile->card);
|
mixer_card = uc_get_mixer_for_card(usecase, pcm_profile->card);
|
||||||
|
@ -2234,12 +2257,16 @@ int disable_output_path_l(struct stream_out *out)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void enable_output_path_l(struct stream_out *out)
|
int enable_output_path_l(struct stream_out *out)
|
||||||
{
|
{
|
||||||
struct audio_device *adev = out->dev;
|
struct audio_device *adev = out->dev;
|
||||||
struct audio_usecase *uc_info;
|
struct audio_usecase *uc_info;
|
||||||
|
|
||||||
uc_info = (struct audio_usecase *)calloc(1, sizeof(struct audio_usecase));
|
uc_info = (struct audio_usecase *)calloc(1, sizeof(struct audio_usecase));
|
||||||
|
if (uc_info == NULL) {
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
uc_info->id = out->usecase;
|
uc_info->id = out->usecase;
|
||||||
uc_info->type = PCM_PLAYBACK;
|
uc_info->type = PCM_PLAYBACK;
|
||||||
uc_info->stream = (struct audio_stream *)out;
|
uc_info->stream = (struct audio_stream *)out;
|
||||||
|
@ -2250,6 +2277,8 @@ void enable_output_path_l(struct stream_out *out)
|
||||||
|
|
||||||
list_add_tail(&adev->usecase_list, &uc_info->adev_list_node);
|
list_add_tail(&adev->usecase_list, &uc_info->adev_list_node);
|
||||||
select_devices(adev, out->usecase);
|
select_devices(adev, out->usecase);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int stop_output_stream(struct stream_out *out)
|
static int stop_output_stream(struct stream_out *out)
|
||||||
|
@ -2278,7 +2307,10 @@ static int start_output_stream(struct stream_out *out)
|
||||||
ALOGV("%s: enter: usecase(%d: %s) devices(%#x) channels(%d)",
|
ALOGV("%s: enter: usecase(%d: %s) devices(%#x) channels(%d)",
|
||||||
__func__, out->usecase, use_case_table[out->usecase], out->devices, out->config.channels);
|
__func__, out->usecase, use_case_table[out->usecase], out->devices, out->config.channels);
|
||||||
|
|
||||||
enable_output_path_l(out);
|
ret = enable_output_path_l(out);
|
||||||
|
if (ret != 0) {
|
||||||
|
goto error_config;
|
||||||
|
}
|
||||||
|
|
||||||
if (out->usecase != USECASE_AUDIO_PLAYBACK_OFFLOAD) {
|
if (out->usecase != USECASE_AUDIO_PLAYBACK_OFFLOAD) {
|
||||||
out->compr = NULL;
|
out->compr = NULL;
|
||||||
|
@ -2346,10 +2378,16 @@ static int stop_voice_call(struct audio_device *adev)
|
||||||
static int start_voice_call(struct audio_device *adev)
|
static int start_voice_call(struct audio_device *adev)
|
||||||
{
|
{
|
||||||
struct audio_usecase *uc_info;
|
struct audio_usecase *uc_info;
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
ALOGV("%s: enter", __func__);
|
ALOGV("%s: enter", __func__);
|
||||||
|
|
||||||
uc_info = (struct audio_usecase *)calloc(1, sizeof(struct audio_usecase));
|
uc_info = (struct audio_usecase *)calloc(1, sizeof(struct audio_usecase));
|
||||||
|
if (uc_info == NULL) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto exit;
|
||||||
|
}
|
||||||
|
|
||||||
uc_info->id = USECASE_VOICE_CALL;
|
uc_info->id = USECASE_VOICE_CALL;
|
||||||
uc_info->type = VOICE_CALL;
|
uc_info->type = VOICE_CALL;
|
||||||
uc_info->stream = (struct audio_stream *)adev->primary_output;
|
uc_info->stream = (struct audio_stream *)adev->primary_output;
|
||||||
|
@ -2370,8 +2408,9 @@ static int start_voice_call(struct audio_device *adev)
|
||||||
set_voice_volume_l(adev, adev->voice_volume);
|
set_voice_volume_l(adev, adev->voice_volume);
|
||||||
|
|
||||||
adev->in_call = true;
|
adev->in_call = true;
|
||||||
|
exit:
|
||||||
ALOGV("%s: exit", __func__);
|
ALOGV("%s: exit", __func__);
|
||||||
return 0;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int check_input_parameters(uint32_t sample_rate,
|
static int check_input_parameters(uint32_t sample_rate,
|
||||||
|
@ -3569,13 +3608,17 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
|
||||||
{
|
{
|
||||||
struct audio_device *adev = (struct audio_device *)dev;
|
struct audio_device *adev = (struct audio_device *)dev;
|
||||||
struct stream_out *out;
|
struct stream_out *out;
|
||||||
int i, ret;
|
int i, ret = 0;
|
||||||
struct pcm_device_profile *pcm_profile;
|
struct pcm_device_profile *pcm_profile;
|
||||||
|
|
||||||
ALOGV("%s: enter: sample_rate(%d) channel_mask(%#x) devices(%#x) flags(%#x)",
|
ALOGV("%s: enter: sample_rate(%d) channel_mask(%#x) devices(%#x) flags(%#x)",
|
||||||
__func__, config->sample_rate, config->channel_mask, devices, flags);
|
__func__, config->sample_rate, config->channel_mask, devices, flags);
|
||||||
*stream_out = NULL;
|
*stream_out = NULL;
|
||||||
out = (struct stream_out *)calloc(1, sizeof(struct stream_out));
|
out = (struct stream_out *)calloc(1, sizeof(struct stream_out));
|
||||||
|
if (out == NULL) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto error_config;
|
||||||
|
}
|
||||||
|
|
||||||
if (devices == AUDIO_DEVICE_NONE)
|
if (devices == AUDIO_DEVICE_NONE)
|
||||||
devices = AUDIO_DEVICE_OUT_SPEAKER;
|
devices = AUDIO_DEVICE_OUT_SPEAKER;
|
||||||
|
@ -3612,6 +3655,10 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
|
||||||
|
|
||||||
out->compr_config.codec = (struct snd_codec *)
|
out->compr_config.codec = (struct snd_codec *)
|
||||||
calloc(1, sizeof(struct snd_codec));
|
calloc(1, sizeof(struct snd_codec));
|
||||||
|
if (out->compr_config.codec == NULL) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto error_open;
|
||||||
|
}
|
||||||
|
|
||||||
out->usecase = USECASE_AUDIO_PLAYBACK_OFFLOAD;
|
out->usecase = USECASE_AUDIO_PLAYBACK_OFFLOAD;
|
||||||
if (config->offload_info.channel_mask)
|
if (config->offload_info.channel_mask)
|
||||||
|
@ -3718,6 +3765,7 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
|
||||||
error_open:
|
error_open:
|
||||||
free(out);
|
free(out);
|
||||||
*stream_out = NULL;
|
*stream_out = NULL;
|
||||||
|
error_config:
|
||||||
ALOGV("%s: exit: ret %d", __func__, ret);
|
ALOGV("%s: exit: ret %d", __func__, ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -3962,6 +4010,9 @@ static int adev_open_input_stream(struct audio_hw_device *dev,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
in = (struct stream_in *)calloc(1, sizeof(struct stream_in));
|
in = (struct stream_in *)calloc(1, sizeof(struct stream_in));
|
||||||
|
if (in == NULL) {
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
in->stream.common.get_sample_rate = in_get_sample_rate;
|
in->stream.common.get_sample_rate = in_get_sample_rate;
|
||||||
in->stream.common.set_sample_rate = in_set_sample_rate;
|
in->stream.common.set_sample_rate = in_set_sample_rate;
|
||||||
|
@ -4109,7 +4160,12 @@ static int adev_open(const hw_module_t *module, const char *name,
|
||||||
ALOGV("%s: enter", __func__);
|
ALOGV("%s: enter", __func__);
|
||||||
if (strcmp(name, AUDIO_HARDWARE_INTERFACE) != 0) return -EINVAL;
|
if (strcmp(name, AUDIO_HARDWARE_INTERFACE) != 0) return -EINVAL;
|
||||||
|
|
||||||
|
*device = NULL;
|
||||||
|
|
||||||
adev = calloc(1, sizeof(struct audio_device));
|
adev = calloc(1, sizeof(struct audio_device));
|
||||||
|
if (adev == NULL) {
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
adev->device.common.tag = HARDWARE_DEVICE_TAG;
|
adev->device.common.tag = HARDWARE_DEVICE_TAG;
|
||||||
adev->device.common.version = AUDIO_DEVICE_API_VERSION_2_0;
|
adev->device.common.version = AUDIO_DEVICE_API_VERSION_2_0;
|
||||||
|
@ -4143,6 +4199,10 @@ static int adev_open(const hw_module_t *module, const char *name,
|
||||||
adev->in_call = false;
|
adev->in_call = false;
|
||||||
/* adev->cur_hdmi_channels = 0; by calloc() */
|
/* adev->cur_hdmi_channels = 0; by calloc() */
|
||||||
adev->snd_dev_ref_cnt = calloc(SND_DEVICE_MAX, sizeof(int));
|
adev->snd_dev_ref_cnt = calloc(SND_DEVICE_MAX, sizeof(int));
|
||||||
|
if (adev->snd_dev_ref_cnt == NULL) {
|
||||||
|
free(adev);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
adev->dualmic_config = DUALMIC_CONFIG_NONE;
|
adev->dualmic_config = DUALMIC_CONFIG_NONE;
|
||||||
adev->ns_in_voice_rec = false;
|
adev->ns_in_voice_rec = false;
|
||||||
|
|
|
@ -53,7 +53,7 @@ int disable_snd_device(struct audio_device *adev,
|
||||||
struct audio_usecase *uc_info,
|
struct audio_usecase *uc_info,
|
||||||
snd_device_t snd_device,
|
snd_device_t snd_device,
|
||||||
bool update_mixer);
|
bool update_mixer);
|
||||||
void enable_output_path_l(struct stream_out *out);
|
int enable_output_path_l(struct stream_out *out);
|
||||||
int disable_output_path_l(struct stream_out *out);
|
int disable_output_path_l(struct stream_out *out);
|
||||||
|
|
||||||
/* must be called with out->lock locked */
|
/* must be called with out->lock locked */
|
||||||
|
@ -268,8 +268,11 @@ ssize_t out_write_offload(struct audio_stream_out *stream, const void *buffer,
|
||||||
if (out->offload_state == OFFLOAD_STATE_PAUSED_FLUSHED) {
|
if (out->offload_state == OFFLOAD_STATE_PAUSED_FLUSHED) {
|
||||||
ALOGV("start offload write from pause state");
|
ALOGV("start offload write from pause state");
|
||||||
pthread_mutex_lock(&adev->lock);
|
pthread_mutex_lock(&adev->lock);
|
||||||
enable_output_path_l(out);
|
ret = enable_output_path_l(out);
|
||||||
pthread_mutex_unlock(&adev->lock);
|
pthread_mutex_unlock(&adev->lock);
|
||||||
|
if (ret != 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (out->send_new_metadata) {
|
if (out->send_new_metadata) {
|
||||||
|
|
Loading…
Reference in a new issue