st-hal: Support first stage module version query

Update device API version to 1.3 to support get_extended_properties
API to return hotword module version from ADSP.

Change-Id: I521bde314493edfc09089764a2c5d5ca085d423a
Signed-off-by: Harshal Ahire <hahire@codeaurora.org>
diff --git a/sound_trigger_hw.c b/sound_trigger_hw.c
index 4c727e3..77af18e 100644
--- a/sound_trigger_hw.c
+++ b/sound_trigger_hw.c
@@ -81,6 +81,8 @@
 static pthread_mutex_t stdev_init_lock;
 static struct sound_trigger_device *stdev = NULL;
 
+static struct sound_trigger_properties_extended_1_3 hw_properties_extended;
+
 /* default properties which will later be updated based on platform configuration */
 static struct sound_trigger_properties hw_properties = {
         "QUALCOMM Technologies, Inc", // implementor
@@ -1300,6 +1302,7 @@
 
     memcpy(properties, stdev->hw_properties,
            sizeof(struct sound_trigger_properties));
+    hw_properties_extended.header.version = SOUND_TRIGGER_DEVICE_API_VERSION_1_0;
     return 0;
 }
 
@@ -1518,15 +1521,26 @@
      * configs. So all the values must be checked instead of a memcmp of the
      * whole configs.
      */
+
+    /*
+     * Extra uint32_t is added in memcmp as opaque data config starts after
+     * audio_capabilities as per SOUND_TRIGGER_DEVICE_API_VERSION_1_3
+     */
+
     if ((current_config->capture_handle != new_config->capture_handle) ||
         (current_config->capture_device != new_config->capture_device) ||
         (current_config->capture_requested != new_config->capture_requested) ||
         (current_config->num_phrases != new_config->num_phrases) ||
         (current_config->data_size != new_config->data_size) ||
         (current_config->data_offset != new_config->data_offset) ||
-        memcmp((char *) current_config + current_config->data_offset,
+        (hw_properties_extended.header.version == SOUND_TRIGGER_DEVICE_API_VERSION_1_3 &&
+         memcmp((char *) current_config + current_config->data_offset,
+               (char *) new_config + sizeof(struct sound_trigger_recognition_config) +
+                sizeof(uint32_t), current_config->data_size)) ||
+        (hw_properties_extended.header.version == SOUND_TRIGGER_DEVICE_API_VERSION_1_0 &&
+         memcmp((char *) current_config + current_config->data_offset,
                (char *) new_config + new_config->data_offset,
-               current_config->data_size)) {
+               current_config->data_size))) {
         return false;
     } else {
         for (i = 0; i < current_config->num_phrases; i++) {
@@ -2445,8 +2459,28 @@
         }
 
         memcpy(st_session->rc_config, (char *)config, sizeof(*config));
-        memcpy((char *)st_session->rc_config + st_session->rc_config->data_offset,
-           (char *)config + config->data_offset, config->data_size);
+
+        /*
+         * SOUND_TRIGGER_DEVICE_API_VERSION_1_3 version introduced new strucutre
+         * sound_trigger_recognition_config_extended_1_3 containing header and base,
+         * so the data_offset for opaque data is relative to this new structure,
+         * leading to extra 12 bytes (8 bytes for header and 4 bytes for audio_capabilities)
+         * w.r.t recoginition config structure. So the extra bytes need to be subtracted
+         * from the data_offset passed considering new strucutre. Opaque data starts after
+         * audio_capabilities variable in sound_trigger_recognition_config_extended_1_3
+         */
+        if (hw_properties_extended.header.version == SOUND_TRIGGER_DEVICE_API_VERSION_1_3) {
+            st_session->rc_config->data_offset -= (sizeof(struct sound_trigger_recognition_config_header) +
+                                                   sizeof(uint32_t));
+
+            memcpy((char *)st_session->rc_config + st_session->rc_config->data_offset,
+                    (char *)config + config->data_offset -
+                    sizeof(struct sound_trigger_recognition_config_header),
+                    config->data_size);
+        } else {
+            memcpy((char *)st_session->rc_config + st_session->rc_config->data_offset,
+                   (char *)config + config->data_offset, config->data_size);
+        }
 
         ALOGVV("%s: num_phrases=%d, id=%d", __func__,
                st_session->rc_config->num_phrases,
@@ -2540,6 +2574,21 @@
     return status;
 }
 
+static int stdev_start_recognition_extended
+(
+    const struct sound_trigger_hw_device *dev,
+    sound_model_handle_t sound_model_handle,
+    const struct sound_trigger_recognition_config_header *config,
+    recognition_callback_t callback,
+    void *cookie
+)
+{
+    return stdev_start_recognition(dev, sound_model_handle,
+           &((struct sound_trigger_recognition_config_extended_1_3 *)config)->base,
+           callback, cookie);
+
+}
+
 static int stdev_stop_recognition(const struct sound_trigger_hw_device *dev,
                                  sound_model_handle_t sound_model_handle)
 {
@@ -2633,6 +2682,129 @@
     return 0;
 }
 
+static int stdev_stop_all_recognitions(const struct sound_trigger_hw_device* dev __unused)
+{
+    ALOGV("%s: unsupported API", __func__);
+    return -ENOSYS;
+}
+
+static int stdev_get_parameter(const struct sound_trigger_hw_device *dev __unused,
+    sound_model_handle_t sound_model_handle __unused,
+    sound_trigger_model_parameter_t model_param __unused, int32_t* value __unused)
+{
+    ALOGV("%s: unsupported API", __func__);
+    return -EINVAL;
+}
+
+static int stdev_set_parameter(const struct sound_trigger_hw_device *dev __unused,
+    sound_model_handle_t sound_model_handle __unused,
+    sound_trigger_model_parameter_t model_param __unused, int32_t value __unused)
+{
+    ALOGV("%s: unsupported API", __func__);
+    return -EINVAL;
+}
+
+static int stdev_query_parameter(const struct sound_trigger_hw_device *dev __unused,
+    sound_model_handle_t sound_model_handle __unused,
+    sound_trigger_model_parameter_t  model_param __unused,
+    sound_trigger_model_parameter_range_t* param_range)
+{
+    if (param_range)
+        param_range->is_supported = false;
+    return 0;
+}
+
+static const struct sound_trigger_properties_header*
+stdev_get_properties_extended(const struct sound_trigger_hw_device *dev)
+{
+    struct st_vendor_info *v_info = NULL;
+    struct listnode *v_node = NULL;
+    struct sound_trigger_device *stdev = NULL;
+    sound_model_handle_t handle = 0;
+    st_session_t *st_session = NULL;
+    struct sound_trigger_properties_header *prop_hdr = NULL;
+    int status = 0;
+
+    ALOGI("%s: enter", __func__);
+
+    if (!dev) {
+        ALOGW("%s: invalid sound_trigger_hw_device received", __func__);
+        return NULL;
+    }
+
+    stdev = (struct sound_trigger_device *)dev;
+    prop_hdr = (struct sound_trigger_properties_header *)&hw_properties_extended;
+    status = stdev_get_properties(dev, &hw_properties_extended.base);
+    if (status) {
+        ALOGW("%s: Failed to initialize the stdev properties", __func__);
+        return NULL;
+    }
+    hw_properties_extended.header.size = sizeof(struct sound_trigger_properties_extended_1_3);
+    hw_properties_extended.audio_capabilities = 0;
+    hw_properties_extended.header.version = SOUND_TRIGGER_DEVICE_API_VERSION_1_3;
+
+    /*
+     * The below code is to get hotword algo version from ADSP. In case of any
+     * errors in  getting the version, return base properties as its not harmful
+     * to bail out STHAL feature with version failure.
+     */
+
+    pthread_mutex_lock(&stdev->lock);
+    if (!CHECK_BIT(stdev->hw_type,
+            ST_DEVICE_HW_APE|ST_DEVICE_HW_CPE|ST_DEVICE_HW_ARM)) {
+        status = platform_stdev_get_hw_type(stdev->platform);
+        if (status) {
+            ALOGW("%s: get hw type failed, %d", __func__, status);
+            goto exit_2;
+        }
+    }
+    list_for_each(v_node, &stdev->vendor_uuid_list) {
+        v_info = node_to_item(v_node, struct st_vendor_info, list_node);
+        if (v_info->get_module_version) {
+            st_session = calloc(1, sizeof(st_session_t));
+            if (!st_session) {
+                ALOGW("%s: st_session allocation failed", __func__);
+                goto exit_2;
+            }
+
+            st_session->f_stage_version = ST_MODULE_TYPE_CUSTOM;
+            st_session->vendor_uuid_info = v_info;
+            handle = android_atomic_inc(&stdev->session_id);
+
+            status = st_session_init(st_session, stdev, ST_EXEC_MODE_ADSP, handle);
+            if (status) {
+                ALOGW("%s: failed to initialize st_session with error %d", __func__,
+                      status);
+                goto exit_1;
+            }
+
+            status = st_session_get_module_version(st_session, hw_properties_extended.supported_model_arch);
+            if (status) {
+                ALOGW("%s: failed to get module version with error %d", __func__,
+                      status);
+                goto exit;
+            }
+            ALOGV("%s: version is %s", __func__, hw_properties_extended.supported_model_arch);
+            break;
+        }
+    }
+
+    pthread_mutex_unlock(&stdev->lock);
+    return prop_hdr;
+
+exit:
+    st_session_deinit(st_session);
+
+exit_1:
+    android_atomic_dec(&stdev->session_id);
+    free(st_session);
+    st_session = NULL;
+
+exit_2:
+    pthread_mutex_unlock(&stdev->lock);
+    return prop_hdr;
+}
+
 static int stdev_open(const hw_module_t* module, const char* name,
                      hw_device_t** device)
 {
@@ -2721,7 +2893,7 @@
     }
 
     stdev->device.common.tag = HARDWARE_DEVICE_TAG;
-    stdev->device.common.version = SOUND_TRIGGER_DEVICE_API_VERSION_1_0;
+    stdev->device.common.version = SOUND_TRIGGER_DEVICE_API_VERSION_1_3;
     stdev->device.common.module = (struct hw_module_t *) module;
     stdev->device.common.close = stdev_close;
     stdev->device.get_properties = stdev_get_properties;
@@ -2729,6 +2901,13 @@
     stdev->device.unload_sound_model = stdev_unload_sound_model;
     stdev->device.start_recognition = stdev_start_recognition;
     stdev->device.stop_recognition = stdev_stop_recognition;
+    stdev->device.get_properties_extended = stdev_get_properties_extended;
+    stdev->device.start_recognition_extended = stdev_start_recognition_extended;
+    stdev->device.stop_all_recognitions = stdev_stop_all_recognitions;
+    stdev->device.get_parameter = stdev_get_parameter;
+    stdev->device.set_parameter = stdev_set_parameter;
+    stdev->device.query_parameter = stdev_query_parameter;
+
 #ifdef ST_SUPPORT_GET_MODEL_STATE
     stdev->device.get_model_state = stdev_get_model_state;
 #endif
diff --git a/sound_trigger_platform.c b/sound_trigger_platform.c
index 20c4a7e..cefc47e 100644
--- a/sound_trigger_platform.c
+++ b/sound_trigger_platform.c
@@ -157,6 +157,8 @@
 #define ST_PARAM_KEY_DEDICATED_HEADSET_PATH "dedicated_headset_path"
 #define ST_PARAM_KEY_ENABLE_DEBUG_DUMPS "enable_debug_dumps"
 #define ST_PARAM_KEY_DAM_TOKEN_ID "dam_token_id"
+#define ST_PARAM_KEY_GET_MODULE_VERSION "get_module_version"
+#define ST_PARAM_KEY_VERSION_ID "version_ids"
 
 #ifndef Q6AFE_HWDEP_NODE
 #define Q6AFE_HWDEP_NODE -1
@@ -2187,6 +2189,16 @@
         module_params->param_tag_tracker |= PARAM_LAB_CONTROL_BIT;
     }
 
+    err = str_parms_get_str(parms, ST_PARAM_KEY_VERSION_ID,
+                            str_value, sizeof(str_value));
+    if (err >= 0) {
+        str_parms_del(parms, ST_PARAM_KEY_VERSION_ID);
+        ret = platform_stdev_set_module_param_ids(
+              &module_params->params[VERSION_ID], str_value, false);
+        if (ret)
+            goto err_exit;
+    }
+
     list_add_tail(&lsm_params->module_params_list,
         &module_params->list_node);
     free(kv_pairs);
@@ -2383,6 +2395,16 @@
                 goto err_exit;
             lsm_params->param_tag_tracker |= PARAM_LAB_CONTROL_BIT;
         }
+
+        err = str_parms_get_str(parms, ST_PARAM_KEY_VERSION_ID,
+                                str_value, sizeof(str_value));
+        if (err >= 0) {
+            str_parms_del(parms, ST_PARAM_KEY_VERSION_ID);
+            ret = platform_stdev_set_module_param_ids(
+                &lsm_params->params[VERSION_ID], str_value, is_legacy_params);
+            if (ret)
+                goto err_exit;
+        }
     } else {
         err = str_parms_get_str(parms, ST_PARAM_KEY_PDK5_APP_TYPE,
                             str_value, sizeof(str_value));
@@ -2827,6 +2849,14 @@
             !strncasecmp(str_value, "true", 4) ? true : false;
     }
 
+    err = str_parms_get_str(parms, ST_PARAM_KEY_GET_MODULE_VERSION,
+                            str_value, sizeof(str_value));
+    if (err >= 0) {
+        str_parms_del(parms, ST_PARAM_KEY_GET_MODULE_VERSION);
+        sm_info->get_module_version =
+            !strncasecmp(str_value, "true", 4) ? true : false;
+    }
+
     sm_info->avail_transit_ape_phrases = sm_info->avail_ape_phrases;
     sm_info->avail_transit_ape_users = sm_info->avail_ape_users;
     sm_info->avail_transit_cpe_phrases = sm_info->avail_cpe_phrases;
diff --git a/sound_trigger_platform.h b/sound_trigger_platform.h
index 93654c4..1ffda2a 100644
--- a/sound_trigger_platform.h
+++ b/sound_trigger_platform.h
@@ -102,6 +102,7 @@
 #define PARAM_LAB_CONTROL_BIT (1 << 7)
 #define PARAM_REQUEST_DETECTION_BIT (1 << 8)
 #define PARAM_LAB_DAM_CFG_BIT (1 << 9)
+#define PARAM_GET_MODULE_VERSION (1 << 10)
 #define PARAM_ID_MANDATORY_BITS \
     (PARAM_LOAD_SOUND_MODEL_BIT | PARAM_UNLOAD_SOUND_MODEL_BIT)
 
@@ -213,6 +214,7 @@
     LAB_CONTROL,
     REQUEST_DETECTION,
     LAB_DAM_CFG,
+    VERSION_ID,
     MAX_PARAM_IDS
 };
 
@@ -346,6 +348,7 @@
     int app_type;
     bool is_qcva_uuid;
     bool is_qcmd_uuid;
+    bool get_module_version;
     bool merge_fs_soundmodels;
     unsigned int fwk_mode;
     int sample_rate;
diff --git a/st_hw_session.h b/st_hw_session.h
index 37f1dc9..2c76e00 100644
--- a/st_hw_session.h
+++ b/st_hw_session.h
@@ -154,6 +154,10 @@
 typedef int (*sound_trigger_get_param_data_t)(st_hw_session_t *, const char *,
     void *, size_t, size_t *);
 typedef int (*sound_trigger_send_detection_request_t)(st_hw_session_t *);
+typedef int (*sound_trigger_get_module_version)(st_hw_session_t *, void *, size_t);
+typedef int (*sound_trigger_open_session)(st_hw_session_t *);
+typedef void (*sound_trigger_close_session)(st_hw_session_t *);
+
 
 struct st_session_fptrs {
     sound_trigger_reg_sm_t reg_sm;
@@ -179,6 +183,9 @@
     sound_trigger_enable_device_t enable_device;
     sound_trigger_get_param_data_t get_param_data;
     sound_trigger_send_detection_request_t send_detection_request;
+    sound_trigger_get_module_version get_module_version;
+    sound_trigger_open_session open_session;
+    sound_trigger_close_session close_session;
 };
 
 #endif
diff --git a/st_hw_session_lsm.c b/st_hw_session_lsm.c
index 544d48a..312bd01 100644
--- a/st_hw_session_lsm.c
+++ b/st_hw_session_lsm.c
@@ -73,6 +73,12 @@
 static int ape_start(st_hw_session_t* p_ses);
 static int ape_stop(st_hw_session_t* p_ses);
 static int ape_stop_buffering(st_hw_session_t* p_ses);
+static int ape_open_session(st_hw_session_t* p_ses);
+static void ape_close_session(st_hw_session_t* p_ses);
+#ifdef SNDRV_LSM_GET_MODULE_PARAMS
+static int ape_get_module_version(st_hw_session_t *p_ses, void *param_info_payload,
+                       size_t size);
+#endif
 
 /* Routing layer functions */
 static int route_reg_sm_ape(st_hw_session_t *p_ses,
@@ -134,6 +140,11 @@
     .enable_device = route_enable_device,
     .get_param_data = get_param_data,
     .send_detection_request = send_detection_request,
+    .open_session = ape_open_session,
+    .close_session = ape_close_session,
+#ifdef SNDRV_LSM_GET_MODULE_PARAMS
+    .get_module_version = ape_get_module_version,
+#endif
 };
 
 int pcm_ioctl(struct pcm *pcm, int request, ...)
@@ -190,6 +201,41 @@
     return status;
 }
 
+#ifdef SNDRV_LSM_GET_MODULE_PARAMS
+static void lsm_fill_get_param_info
+(
+    uint32_t param_type,
+    struct lsm_params_get_info *p_info,
+    struct st_module_param_info *mparams,
+    uint16_t stage_idx
+)
+{
+    p_info->param_type = param_type;
+    p_info->module_id = mparams->module_id;
+    p_info->instance_id = mparams->instance_id;
+    p_info->param_id = mparams->param_id;
+    p_info->stage_idx = stage_idx;
+}
+
+static int lsm_get_module_params
+(
+    st_hw_session_lsm_t *p_lsm_ses,
+    struct lsm_params_get_info *lsm_params
+)
+{
+    int status = 0;
+
+    ATRACE_BEGIN("sthal:lsm: pcm_ioctl sndrv_lsm_get_module_params");
+    status = pcm_ioctl(p_lsm_ses->pcm, SNDRV_LSM_GET_MODULE_PARAMS, lsm_params);
+    ATRACE_END();
+
+    if (status)
+        ALOGE("%s: ERROR. SNDRV_LSM_GET_MODULE_PARAMS status(%d)",
+              __func__, status);
+    return status;
+}
+#endif
+
 static void lsm_fill_param_info
 (
     uint32_t param_type,
@@ -2563,6 +2609,144 @@
     return status;
 }
 
+#ifdef SNDRV_LSM_GET_MODULE_PARAMS
+static int ape_open_session(st_hw_session_t *p_ses)
+{
+    st_hw_session_lsm_t *p_lsm_ses = (st_hw_session_lsm_t*)p_ses;
+    struct st_vendor_info *v_info = p_ses->vendor_uuid_info;
+    int status = 0;
+    audio_devices_t capture_device = 0;
+
+    status = platform_get_lsm_usecase(p_ses->stdev->platform, v_info,
+        &p_lsm_ses->lsm_usecase, p_ses->exec_mode, p_ses->lpi_enable,
+        p_ses->f_stage_version);
+
+    if (status) {
+        ALOGE("%s: couldn't get lsm usecase", __func__);
+        return -EINVAL;
+    }
+
+    p_lsm_ses->pcm_id = platform_ape_get_pcm_device_id(
+        p_ses->stdev->platform, &p_ses->use_case_idx);
+    if (p_lsm_ses->pcm_id < 0) {
+        ALOGE("%s: get pcm id failed %d\n",__func__, p_lsm_ses->pcm_id);
+        return -ENODEV;
+    }
+
+    int app_type = (v_info->app_type == 0) ?
+        p_lsm_ses->lsm_usecase.app_type : v_info->app_type;
+
+    capture_device = platform_stdev_get_capture_device(p_ses->stdev->platform);
+    status = platform_stdev_send_calibration(p_ses->stdev->platform,
+                                              capture_device,
+                                              p_ses->exec_mode,
+                                              p_ses->vendor_uuid_info,
+                                              app_type, true,
+                                              ST_SESSION_CAL);
+
+    if (status) {
+        ALOGE("%s: ERROR. sending calibration failed status %d, idx 0",
+              __func__, status);
+        goto error;
+    }
+    p_lsm_ses->num_stages = 1;
+    p_lsm_ses->common.config = stdev_ape_pcm_config;
+    platform_stdev_check_and_update_pcm_config(&p_lsm_ses->common.config,
+                                               v_info);
+
+    ALOGD("%s: opening pcm device=%d", __func__, p_lsm_ses->pcm_id);
+    ALOGV("%s: config: channels=%d rate=%d, period_size=%d, period_cnt=%d, format=%d",
+          __func__, p_lsm_ses->common.config.channels, p_lsm_ses->common.config.rate,
+          p_lsm_ses->common.config.period_size, p_lsm_ses->common.config.period_count,
+          p_lsm_ses->common.config.format);
+
+    ATRACE_BEGIN("sthal:lsm: pcm_open");
+    p_lsm_ses->pcm = pcm_open(p_ses->stdev->snd_card, p_lsm_ses->pcm_id,
+                              PCM_IN, &p_lsm_ses->common.config);
+    ATRACE_END();
+
+    if (!p_lsm_ses->pcm) {
+        ALOGE("%s: ERROR. pcm_open failed", __func__);
+        status = -ENODEV;
+        goto error;
+    }
+    if (!pcm_is_ready(p_lsm_ses->pcm)) {
+        ALOGE("%s: ERROR. pcm_is_ready failed err=%s", __func__,
+              pcm_get_error(p_lsm_ses->pcm));
+        status = -ENODEV;
+        goto error;
+    }
+
+    if (st_hw_check_multi_stage_lsm_support()) {
+        status = lsm_set_session_data_v2(p_ses);
+        if (status)
+            goto error;
+    }
+    return 0;
+
+error:
+    platform_ape_free_pcm_device_id(p_ses->stdev->platform, p_lsm_ses->pcm_id);
+    if (p_lsm_ses->pcm) {
+        pcm_close(p_lsm_ses->pcm);
+        p_lsm_ses->pcm = NULL;
+    }
+    return status;
+}
+
+void ape_close_session(st_hw_session_t *p_ses)
+{
+    st_hw_session_lsm_t *p_lsm_ses = (st_hw_session_lsm_t*)p_ses;
+
+    ATRACE_BEGIN("sthal:lsm: pcm_close");
+    pcm_close(p_lsm_ses->pcm);
+    ATRACE_END();
+    p_lsm_ses->pcm = NULL;
+}
+
+int ape_get_module_version(st_hw_session_t *p_ses, void *param_info_payload,
+                           size_t param_size)
+{
+    struct st_module_param_info *mparams = NULL;
+    struct lsm_params_get_info *get_params;
+    st_hw_session_lsm_t *p_lsm_ses = (st_hw_session_lsm_t*)p_ses;
+    int status = 0;
+    size_t size = 0;
+
+    mparams = p_lsm_ses->lsm_usecase.params;
+    size = sizeof(struct lsm_params_get_info) + param_size;
+    get_params = calloc(1, size);
+    if (!get_params) {
+        ALOGE("%s: ERROR. Can not allocate memory for get params", __func__);
+        return -ENOMEM;
+    }
+
+    get_params->param_size =  param_size;
+    lsm_fill_get_param_info(LSM_GET_CUSTOM_PARAMS, get_params,
+                            &mparams[VERSION_ID],
+                            LSM_STAGE_INDEX_FIRST);
+    status = lsm_get_module_params(p_lsm_ses, get_params);
+    if (status) {
+        ALOGE("%s: ERROR. getting module version. status %d",
+             __func__, status);
+        goto done;
+    }
+    memcpy(param_info_payload, get_params->payload, param_size);
+done:
+    free(get_params);
+    return status;
+}
+#else
+static int ape_open_session(st_hw_session_t* p_ses __unused)
+{
+    return -ENOSYS;
+}
+
+static void ape_close_session(st_hw_session_t* p_ses __unused)
+{
+    return;
+}
+#endif
+
 static int route_reg_sm_ape(st_hw_session_t *p_ses,void *sm_data,
     unsigned int sm_size, uint32_t model_id)
 {
diff --git a/st_hw_session_lsm.h b/st_hw_session_lsm.h
index e713501..676518f 100644
--- a/st_hw_session_lsm.h
+++ b/st_hw_session_lsm.h
@@ -153,6 +153,11 @@
     struct lsm_param_payload common;
 }__packed;
 
+struct version_arch_payload {
+    unsigned int version;
+    char arch[64];
+}__packed;
+
 struct st_hw_session_lsm {
     st_hw_session_t common;
     struct st_lsm_params lsm_usecase;
diff --git a/st_session.c b/st_session.c
index a2d9b03..bbe83f7 100644
--- a/st_session.c
+++ b/st_session.c
@@ -141,6 +141,7 @@
         char *chmix_coeff_str;
         bool enable;
         st_session_getparam_payload_t getparam;
+        char *module_version;
     } payload;
     st_session_t *stc_ses;
 };
@@ -4767,6 +4768,7 @@
     st_session_t *stc_ses = ev->stc_ses;
     st_hw_session_t *hw_ses = st_ses->hw_ses_current;
     struct st_proxy_ses_sm_info_wrapper *p_info = NULL;
+    struct version_arch_payload version_payload;
 
     /* skip parameter check as this is an internal funciton */
     ALOGD("%s:[c%d-%d] handle event id %d", __func__, stc_ses->sm_handle,
@@ -4873,7 +4875,30 @@
             __func__, st_ses->sm_handle);
         status = -EINVAL;
         break;
+    case ST_SES_EV_GET_MODULE_VERSION:
+        /* Open Dummy LSM session for google hotword during bootup */
 
+        status = hw_ses->fptrs->open_session(hw_ses);
+        if (status) {
+            ALOGE("%s: failed to start lsm session with error %d", __func__,
+                  status);
+            break;
+        }
+
+        status = hw_ses->fptrs->get_module_version(hw_ses, &version_payload,
+                                        sizeof(struct version_arch_payload));
+
+        if (status) {
+            ALOGE("%s: failed to get module version %d", __func__,
+                  status);
+            hw_ses->fptrs->close_session(hw_ses);
+            break;
+        }
+        hw_ses->fptrs->close_session(hw_ses);
+        snprintf(ev->payload.module_version, SOUND_TRIGGER_MAX_STRING_LEN, "%d, %s",
+                 version_payload.version, version_payload.arch);
+
+        break;
     default:
         ALOGD("%s:[%d] unhandled event", __func__, st_ses->sm_handle);
         break;
@@ -6807,6 +6832,23 @@
     return status;
 }
 
+int st_session_get_module_version(st_session_t *stc_ses, char version[])
+{
+    int status = 0;
+
+    if (!stc_ses || !stc_ses->hw_proxy_ses)
+        return -EINVAL;
+
+    st_proxy_session_t *st_ses = stc_ses->hw_proxy_ses;
+    st_session_ev_t ev = { .ev_id = ST_SES_EV_GET_MODULE_VERSION, .stc_ses = stc_ses,
+                           .payload.module_version = version};
+
+    pthread_mutex_lock(&st_ses->lock);
+    DISPATCH_EVENT(st_ses, ev, status);
+    pthread_mutex_unlock(&st_ses->lock);
+    return status;
+}
+
 int st_session_init(st_session_t *stc_ses, struct sound_trigger_device *stdev,
     st_exec_mode_t exec_mode, sound_model_handle_t sm_handle)
 {
diff --git a/st_session.h b/st_session.h
index fe0816f..04cdcae 100644
--- a/st_session.h
+++ b/st_session.h
@@ -73,6 +73,7 @@
     ST_SES_EV_GET_PARAM_DATA,
     ST_SES_EV_DEFERRED_STOP,
     ST_SES_EV_REQUEST_DET,
+    ST_SES_EV_GET_MODULE_VERSION,
 } st_session_event_id_t;
 
 struct sound_trigger_device;
@@ -259,6 +260,7 @@
 int st_session_request_detection(st_session_t *st_ses);
 int st_session_update_recongition_config(st_session_t *st_ses);
 int st_session_get_preroll(st_session_t *st_ses);
+int st_session_get_module_version(st_session_t *st_ses, char *version);
 
 int process_detection_event_keyphrase_v2(
     st_proxy_session_t *st_ses, int detect_status,