Merge 12f5c6f5fd66669f9d4b6e03025043a7c2d9559c on remote branch

Change-Id: I2da0192ea08c14207d4e0356cf00be0e4b8a9a6b
diff --git a/sound_trigger_hw.h b/sound_trigger_hw.h
index 7e139c8..d1d725e 100644
--- a/sound_trigger_hw.h
+++ b/sound_trigger_hw.h
@@ -265,6 +265,7 @@
     bool support_dynamic_ec_update;
     bool screen_off;
     bool barge_in_mode;
+    int ec_reset_pending_cnt;
 };
 
 typedef struct sound_trigger_device sound_trigger_device_t;
diff --git a/sound_trigger_platform.c b/sound_trigger_platform.c
index 78dd82c..4fb305a 100644
--- a/sound_trigger_platform.c
+++ b/sound_trigger_platform.c
@@ -187,6 +187,7 @@
 #define ST_PARAM_KEY_SS_SM_TYPE "sm_detection_type"
 #define ST_PARAM_KEY_SS_SM_ID "sm_id"
 #define ST_PARAM_KEY_SS_LIB "module_lib"
+#define ST_PARAM_KEY_SS_DATA_BEFORE_KW_START "data_before_kw_start"
 #define ST_PARAM_KEY_SS_DATA_AFTER_KW_END "data_after_kw_end"
 
 #define ST_BACKEND_PORT_NAME_MAX_SIZE 25
@@ -742,6 +743,7 @@
     stdev->platform_lpi_enable = ST_PLATFORM_LPI_NONE;
     stdev->screen_off = true;
     stdev->support_dynamic_ec_update = true;
+    stdev->ec_reset_pending_cnt = 0;
 
     platform->cpe_fe_to_be_fixed = true;
     platform->bad_mic_channel_index = 0;
@@ -1679,6 +1681,14 @@
         common_params->channel_count = value;
     }
 
+    err = str_parms_get_int(parms, ST_PARAM_KEY_SS_DATA_BEFORE_KW_START, &value);
+    if (err >= 0) {
+        str_parms_del(parms, ST_PARAM_KEY_SS_DATA_BEFORE_KW_START);
+        common_params->data_before_kw_start = value;
+    } else {
+        common_params->data_before_kw_start = VOP_DATA_BEFORE_TRUE_KW_START_MS;
+    }
+
     err = str_parms_get_int(parms, ST_PARAM_KEY_SS_DATA_AFTER_KW_END, &value);
     if (err >= 0) {
         str_parms_del(parms, ST_PARAM_KEY_SS_DATA_AFTER_KW_END);
@@ -5681,6 +5691,12 @@
 
     if (is_ec_profile(profile_type)) {
         event_info.st_ec_ref_enabled = enable;
+        // reset the pending active EC mixer ctls first
+        if (!stdev->audio_ec_enabled && stdev->ec_reset_pending_cnt > 0) {
+            while (stdev->ec_reset_pending_cnt--)
+                audio_route_reset_and_update_path(stdev->audio_route,
+                        my_data->ec_ref_mixer_path);
+        }
         if (enable) {
             stdev->audio_hal_cb(ST_EVENT_UPDATE_ECHO_REF, &event_info);
             strlcpy(my_data->ec_ref_mixer_path, "echo-reference",
@@ -5701,6 +5717,7 @@
                 audio_route_reset_and_update_path(stdev->audio_route,
                         my_data->ec_ref_mixer_path);
             } else {
+                stdev->ec_reset_pending_cnt++;
                 ALOGD("%s: audio hal has already enabled EC", __func__);
             }
         }
diff --git a/st_common_defs.h b/st_common_defs.h
index 2beef66..f9a88ca 100644
--- a/st_common_defs.h
+++ b/st_common_defs.h
@@ -1,6 +1,6 @@
 /* st_common_defs.h
  *
- * Copyright (c) 2016-2019, The Linux Foundation. All rights reserved.
+ * Copyright (c) 2016-2020, The Linux Foundation. All rights reserved.
  *
  * Redistribution and use in source and binary forms, with or without
  * modification, are permitted provided that the following conditions are
@@ -102,6 +102,7 @@
 #define GENERIC_DET_EVENT_HEADER_SIZE (8)
 #define FIRST_STAGE_KW_START_TOLERANCE_MS (300)
 #define FIRST_STAGE_KW_END_TOLERANCE_MS (240)
+#define VOP_DATA_BEFORE_TRUE_KW_START_MS (360)
 #define CNN_DATA_AFTER_TRUE_KW_END_MS (200)
 #define GENERIC_DET_EVENT_USER_LEVEL_OFFSET (17)
 #define GCS_NON_GENERIC_USER_LEVEL_OFFSET (3)
diff --git a/st_hw_session_lsm.c b/st_hw_session_lsm.c
index 929f51b..6ef9673 100644
--- a/st_hw_session_lsm.c
+++ b/st_hw_session_lsm.c
@@ -1395,7 +1395,8 @@
     int status = 0;
     struct listnode *node = NULL, *tmp_node = NULL;
     st_arm_second_stage_t *st_sec_stage = NULL;
-    unsigned int prepend_bytes = 0, cnn_append_bytes = 0, vop_append_bytes = 0;
+    unsigned int cnn_prepend_bytes = 0, vop_prepend_bytes = 0;
+    unsigned int cnn_append_bytes = 0, vop_append_bytes = 0;
     unsigned int kw_duration_bytes = 0;
     bool real_time_check = true;
     uint64_t frame_receive_time = 0, frame_send_time = 0;
@@ -1415,10 +1416,6 @@
     st_buffer_reset(p_lsm_ses->common.buffer);
 
     if (p_lsm_ses->common.enable_second_stage) {
-        prepend_bytes =
-            convert_ms_to_bytes(
-                p_lsm_ses->common.vendor_uuid_info->kw_start_tolerance,
-                &p_lsm_ses->common.config);
         if (p_lsm_ses->common.sthw_cfg.client_req_hist_buf) {
             kw_duration_bytes =
                 convert_ms_to_bytes(
@@ -1446,15 +1443,20 @@
              * detections. Similarly, error tolerance is added to the end of the
              * buffer for generic and non generic detection event usecases.
              */
-            if (p_lsm_ses->common.kw_start_idx > prepend_bytes) {
-                st_sec_stage->ss_session->buf_start =
-                    p_lsm_ses->common.kw_start_idx - prepend_bytes;
-            } else {
-                st_sec_stage->ss_session->buf_start = 0;
-            }
-
             if (st_sec_stage->ss_info->sm_detection_type ==
                 ST_SM_TYPE_KEYWORD_DETECTION) {
+                cnn_prepend_bytes =
+                    convert_ms_to_bytes(
+                        p_lsm_ses->common.vendor_uuid_info->kw_start_tolerance,
+                        &p_lsm_ses->common.config);
+
+                if (p_lsm_ses->common.kw_start_idx > cnn_prepend_bytes) {
+                    st_sec_stage->ss_session->buf_start =
+                        p_lsm_ses->common.kw_start_idx - cnn_prepend_bytes;
+                } else {
+                    st_sec_stage->ss_session->buf_start = 0;
+                }
+
                 cnn_append_bytes =
                     convert_ms_to_bytes(
                         (p_lsm_ses->common.vendor_uuid_info->kw_end_tolerance +
@@ -1479,6 +1481,18 @@
                 st_sec_stage->ss_session->det_status = KEYWORD_DETECTION_PENDING;
             } else if (st_sec_stage->ss_info->sm_detection_type ==
                 ST_SM_TYPE_USER_VERIFICATION) {
+                vop_prepend_bytes =
+                    convert_ms_to_bytes(
+                        st_sec_stage->ss_info->data_before_kw_start,
+                        &p_lsm_ses->common.config);
+
+                if (p_lsm_ses->common.kw_start_idx > vop_prepend_bytes) {
+                    st_sec_stage->ss_session->buf_start =
+                        p_lsm_ses->common.kw_start_idx - vop_prepend_bytes;
+                } else {
+                    st_sec_stage->ss_session->buf_start = 0;
+                }
+
                 vop_append_bytes =
                     convert_ms_to_bytes(
                         p_lsm_ses->common.vendor_uuid_info->kw_end_tolerance,
diff --git a/st_second_stage.h b/st_second_stage.h
index fee487d..6a15e81 100644
--- a/st_second_stage.h
+++ b/st_second_stage.h
@@ -4,7 +4,7 @@
  * abstraction represents a single st second stage session from the st session
  * perspective.
  *
- * Copyright (c) 2018-2019, The Linux Foundation. All rights reserved.
+ * Copyright (c) 2018-2020, The Linux Foundation. All rights reserved.
  *
  * Redistribution and use in source and binary forms, with or without
  * modification, are permitted provided that the following conditions are
@@ -70,6 +70,7 @@
     unsigned int sample_rate;
     unsigned int bit_width;
     unsigned int channel_count;
+    unsigned int data_before_kw_start;
     unsigned int data_after_kw_end;
 }st_second_stage_info_t;
 
diff --git a/st_session.c b/st_session.c
index 9bb1c09..2ae2d81 100644
--- a/st_session.c
+++ b/st_session.c
@@ -2784,7 +2784,7 @@
     return 0;
 }
 
-static inline int prepapre_second_stage_for_client(st_session_t *stc_ses)
+static inline int prepare_second_stage_for_client(st_session_t *stc_ses)
 {
     struct listnode *node = NULL;
     st_arm_second_stage_t *st_sec_stage = NULL;
@@ -4801,18 +4801,34 @@
                  * Note: It is possible that detection event is not sent to
                  * client if second stage is not yet detected during internal
                  * buffering stop, in which case restart is posted from second
-                 * stage thread for further detections.
+                 * stage thread for further detections. Only if the second
+                 * stage detection hasn't be started due to internal buffering
+                 * stop too early, restart session should be explictily issued.
                  */
-                if ((st_ses->current_state == buffering_state_fn) &&
-                    !stc_ses->pending_stop && stc_ses->detection_sent) {
-                    ALOGD("%s:[%d] buffering stopped internally, post c%d stop",
-                        __func__, st_ses->sm_handle,
-                        st_ses->det_stc_ses->sm_handle);
-                    status = hw_session_notifier_enqueue(stc_ses->sm_handle,
-                        ST_SES_EV_DEFERRED_STOP,
-                        ST_SES_DEFERRED_STOP_SS_DELAY_MS);
-                    if (!status)
-                        stc_ses->pending_stop = true;
+                if (st_ses->current_state == buffering_state_fn) {
+                    if (stc_ses->detection_sent) {
+                        if (!stc_ses->pending_stop) {
+                            ALOGD("%s:[%d] buffering stopped internally, post c%d stop",
+                                __func__, st_ses->sm_handle,
+                                st_ses->det_stc_ses->sm_handle);
+                            status = hw_session_notifier_enqueue(stc_ses->sm_handle,
+                                ST_SES_EV_DEFERRED_STOP,
+                                ST_SES_DEFERRED_STOP_SS_DELAY_MS);
+                            if (!status)
+                                stc_ses->pending_stop = true;
+                        }
+                    } else {
+                        list_for_each(node, &stc_ses->second_stage_list) {
+                            st_sec_stage = node_to_item(node, st_arm_second_stage_t,
+                                                        list_node);
+                            if (!st_sec_stage->ss_session->start_processing) {
+                                st_session_ev_t ev = {.ev_id = ST_SES_EV_RESTART,
+                                                      .stc_ses = stc_ses};
+                                DISPATCH_EVENT(st_ses, ev, status);
+                                break;
+                            }
+                        }
+                    }
                 }
                 pthread_mutex_unlock(&st_ses->lock);
             }
@@ -5451,7 +5467,6 @@
                 status = -EINVAL;
                 break;
             }
-            prepapre_second_stage_for_client(stc_ses);
             stc_ses->state = ST_STATE_LOADED;
         } else {
             ALOGE("%s: received unexpected event, client state = %d",
@@ -5465,7 +5480,6 @@
             if (status)
                 ALOGE("%s:[c%d] update sound_model failed %d", __func__,
                     stc_ses->sm_handle, status);
-            stop_second_stage_for_client(stc_ses);
             stc_ses->state = ST_STATE_IDLE;
         } else {
             ALOGE("%s: received unexpected event, client state = %d",
@@ -5558,7 +5572,7 @@
     pthread_mutex_lock(&st_ses->lock);
     DISPATCH_EVENT(st_ses, ev, status);
     if (!status) {
-        prepapre_second_stage_for_client(stc_ses);
+        prepare_second_stage_for_client(stc_ses);
         stc_ses->state = ST_STATE_LOADED;
     }
     pthread_mutex_unlock(&st_ses->lock);