st-hal: Prevent multiple consecutive device enablements
In capture concurrency usecases, the device disable can be skipped
due to audio hal continuing to need the device. In these usecases,
do not allow sthal to enable the same device again as it throws off
the mixer control reference count.
Change-Id: I8f905901f705f7c6c38b27ed547f2d609efcdf4c
diff --git a/sound_trigger_hw.c b/sound_trigger_hw.c
index f677147..ff46d0e 100644
--- a/sound_trigger_hw.c
+++ b/sound_trigger_hw.c
@@ -2607,6 +2607,7 @@
free(stdev->cpe_pcm_use_cases);
free(stdev->ape_pcm_use_cases);
free(stdev->dev_ref_cnt);
+ free(stdev->dev_enable_cnt);
list_for_each_safe(node, tmp_node, &stdev->sound_model_list) {
st_session = node_to_item(node, st_session_t, list_node);
list_remove(node);
@@ -2728,6 +2729,15 @@
goto exit_1;
}
+ stdev->dev_enable_cnt =
+ calloc(ST_EXEC_MODE_MAX * ST_DEVICE_MAX, sizeof(int));
+
+ if (!stdev->dev_enable_cnt) {
+ ALOGE("%s: ERROR. Mem alloc failed dev enable cnt", __func__);
+ status = -ENOMEM;
+ goto exit_1;
+ }
+
if (hw_session_notifier_init(stdev_session_event_cb) < 0) {
ALOGE("%s: Failed to initialize notifier", __func__);
status = -ENOMEM;
@@ -2790,6 +2800,9 @@
if (stdev->dev_ref_cnt)
free(stdev->dev_ref_cnt);
+ if (stdev->dev_enable_cnt)
+ free(stdev->dev_enable_cnt);
+
if (stdev->arm_pcm_use_cases)
free(stdev->arm_pcm_use_cases);
diff --git a/sound_trigger_hw.h b/sound_trigger_hw.h
index d1d725e..bcfbb69 100644
--- a/sound_trigger_hw.h
+++ b/sound_trigger_hw.h
@@ -178,6 +178,7 @@
pthread_mutex_t ref_cnt_lock;
int *dev_ref_cnt;
struct listnode available_devices;
+ int *dev_enable_cnt;
pthread_t transitions_thread;
pthread_cond_t transitions_cond;
diff --git a/sound_trigger_platform.c b/sound_trigger_platform.c
index 4fb305a..a45f6c6 100644
--- a/sound_trigger_platform.c
+++ b/sound_trigger_platform.c
@@ -4386,6 +4386,30 @@
return my_data->codec_backend_cfg.lpi_enable;
}
+int platform_get_lpi_st_device(int st_device)
+{
+ int lpi_device = st_device;
+
+ switch (st_device) {
+ case ST_DEVICE_HANDSET_DMIC:
+ lpi_device = ST_DEVICE_HANDSET_DMIC_LPI;
+ break;
+ case ST_DEVICE_HANDSET_TMIC:
+ lpi_device = ST_DEVICE_HANDSET_TMIC_LPI;
+ break;
+ case ST_DEVICE_HANDSET_QMIC:
+ lpi_device = ST_DEVICE_HANDSET_QMIC_LPI;
+ break;
+ case ST_DEVICE_HEADSET_MIC:
+ lpi_device = ST_DEVICE_HEADSET_MIC_LPI;
+ break;
+ default:
+ ALOGV("%s: No need to convert device %d", __func__, st_device);
+ }
+
+ return lpi_device;
+}
+
#ifdef SNDRV_IOCTL_HWDEP_VAD_CAL_TYPE
static int platform_stdev_send_hwvad_cal
(
diff --git a/sound_trigger_platform.h b/sound_trigger_platform.h
index 7768e05..5320f07 100644
--- a/sound_trigger_platform.h
+++ b/sound_trigger_platform.h
@@ -671,6 +671,8 @@
bool platform_get_lpi_mode(void *my_data);
+int platform_get_lpi_st_device(int st_device);
+
void platform_stdev_reset_backend_cfg(void *platform);
void platform_stdev_get_lpma_config
diff --git a/st_hw_session_lsm.c b/st_hw_session_lsm.c
index 6ef9673..dd9291f 100644
--- a/st_hw_session_lsm.c
+++ b/st_hw_session_lsm.c
@@ -1964,7 +1964,7 @@
)
{
char st_device_name[DEVICE_NAME_MAX_SIZE] = { 0 };
- int ref_cnt_idx = 0, ref_cnt = 0;
+ int ref_cnt_idx = 0, ref_cnt = 0, ref_enable_idx = 0;
int status = 0;
st_device_t st_device = 0;
audio_devices_t capture_device = 0;
@@ -2007,6 +2007,8 @@
pthread_mutex_lock(&p_ses->stdev->ref_cnt_lock);
ref_cnt_idx = (p_ses->exec_mode * ST_DEVICE_MAX) + st_device;
+ ref_enable_idx = (p_ses->exec_mode * ST_DEVICE_MAX) +
+ platform_get_lpi_st_device(st_device);
ref_cnt = ++(p_ses->stdev->dev_ref_cnt[ref_cnt_idx]);
app_type = platform_stdev_get_device_app_type(p_ses->stdev->platform,
profile_type);
@@ -2038,15 +2040,23 @@
__func__, p_ses->st_device, p_ses->st_device_name);
audio_route_reset_and_update_path(p_ses->stdev->audio_route,
st_device_name);
+ if (0 < p_ses->stdev->dev_enable_cnt[ref_enable_idx])
+ --(p_ses->stdev->dev_enable_cnt[ref_enable_idx]);
}
- ALOGD("%s: enable device (%x) = %s", __func__, st_device,
- st_device_name);
- ATRACE_BEGIN("sthal:lsm: audio_route_apply_and_update_path");
- audio_route_apply_and_update_path(p_ses->stdev->audio_route,
- st_device_name);
- ATRACE_END();
- update_hw_mad_exec_mode(p_ses->exec_mode, profile_type);
+ if (0 == p_ses->stdev->dev_enable_cnt[ref_enable_idx]) {
+ ALOGD("%s: enable device (%x) = %s", __func__, st_device,
+ st_device_name);
+ ATRACE_BEGIN("sthal:lsm: audio_route_apply_and_update_path");
+ audio_route_apply_and_update_path(p_ses->stdev->audio_route,
+ st_device_name);
+ ATRACE_END();
+ update_hw_mad_exec_mode(p_ses->exec_mode, profile_type);
+ ++(p_ses->stdev->dev_enable_cnt[ref_enable_idx]);
+ } else {
+ ALOGD("%s: Device already enabled, no not re-enable",
+ __func__);
+ }
}
} else {
--(p_ses->stdev->dev_ref_cnt[ref_cnt_idx]);
@@ -2062,6 +2072,8 @@
}
ref_cnt_idx = (p_ses->exec_mode * ST_DEVICE_MAX) + p_ses->st_device;
+ ref_enable_idx = (p_ses->exec_mode * ST_DEVICE_MAX) +
+ platform_get_lpi_st_device(p_ses->st_device);
pthread_mutex_lock(&p_ses->stdev->ref_cnt_lock);
ref_cnt = p_ses->stdev->dev_ref_cnt[ref_cnt_idx];
if (0 < ref_cnt) {
@@ -2081,6 +2093,8 @@
p_ses->st_device_name);
ATRACE_END();
update_hw_mad_exec_mode(ST_EXEC_MODE_NONE, profile_type);
+ if (0 < p_ses->stdev->dev_enable_cnt[ref_enable_idx])
+ --(p_ses->stdev->dev_enable_cnt[ref_enable_idx]);
} else {
ALOGD("%s: Non-hwmad device, concurrent capture on, do not disable", __func__);
}