Merge "st-hal: Prevent multiple consecutive device enablements"
diff --git a/sound_trigger_hw.c b/sound_trigger_hw.c
index bc64b76..1eb7f40 100644
--- a/sound_trigger_hw.c
+++ b/sound_trigger_hw.c
@@ -2577,6 +2577,7 @@
     free(stdev->arm_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);
@@ -2687,6 +2688,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;
@@ -2749,6 +2759,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 b7aa451..85b33eb 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 38d8760..24e82f0 100644
--- a/sound_trigger_platform.c
+++ b/sound_trigger_platform.c
@@ -4327,6 +4327,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 307e53a..3686e5e 100644
--- a/sound_trigger_platform.h
+++ b/sound_trigger_platform.h
@@ -652,6 +652,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 f2f1abb..9b41c03 100644
--- a/st_hw_session_lsm.c
+++ b/st_hw_session_lsm.c
@@ -1248,7 +1248,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;
@@ -1291,6 +1291,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);
@@ -1322,15 +1324,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]);
@@ -1346,6 +1356,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) {
@@ -1365,6 +1377,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__);
             }