Преглед на файлове

qcacld-3.0: Extend pagefault wakeup event to per symbol

Extend the page fault action INI to handle following:
   1) Ignore page fault event
   2) Trigger SSR on page fault threshold
   3) Send blob of data to userspace on page fault threshold

On pagefault wakeup event, save per symbol pagefault timestamp
and once the event count reaches configured threshold within the
configured time interval, either trigger SSR or notify userspace
with pagefault address and count based on INI configuration.

Change-Id: I0a3ece369ad0c7aac676fc91f6863e06a3f4ce8c
CRs-Fixed: 3713813
Vinod Kumar Pirla преди 1 година
родител
ревизия
6d055ec212

+ 23 - 8
components/pmo/core/inc/wlan_pmo_main.h

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2017-2019 The Linux Foundation. All rights reserved.
- * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2023-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -386,21 +386,36 @@ pmo_intersect_packet_filter(struct pmo_psoc_priv_obj *psoc_ctx)
 }
 
 /*
- * pmo_enable_ssr_on_page_fault: Enable/disable ssr on pagefault
- * @psoc: objmgr psoc
+ * pmo_host_action_on_page_fault() - Returns action host will take on page fault
+ * @psoc: PSOC object manager pointer.
  *
- * Return: True if SSR is enabled on pagefault
+ * Returns: Host action on page fault event
  */
-bool pmo_enable_ssr_on_page_fault(struct wlan_objmgr_psoc *psoc);
+enum pmo_page_fault_action
+pmo_host_action_on_page_fault(struct wlan_objmgr_psoc *psoc);
+
+#define pmo_is_host_pagefault_action(_psoc, _action) \
+		(pmo_host_action_on_page_fault(_psoc) == (_action))
+
+static inline bool pmo_no_op_on_page_fault(struct wlan_objmgr_psoc *psoc)
+{
+	return pmo_is_host_pagefault_action(psoc, PMO_PF_HOST_ACTION_NO_OP);
+}
+
+static inline bool pmo_enable_ssr_on_page_fault(struct wlan_objmgr_psoc *psoc)
+{
+	return pmo_is_host_pagefault_action(psoc, PMO_PF_HOST_ACTION_TRIGGER_SSR);
+}
 
 /*
- * pmo_get_max_pagefault_wakeups_for_ssr: get pagefault wakeups for ssr
+ * pmo_get_min_pagefault_wakeups_for_action() - get pagefault wakeups for host
+ * to initiate action
  * @psoc: objmgr psoc
  *
- * Return: SSR interval for pagefault
+ * Return: Min wakeups interval for host action on pagefault
  */
 uint8_t
-pmo_get_max_pagefault_wakeups_for_ssr(struct wlan_objmgr_psoc *psoc);
+pmo_get_min_pagefault_wakeups_for_action(struct wlan_objmgr_psoc *psoc);
 
 /*
  * pmo_get_interval_for_pagefault_wakeup_counts: get ssr interval for pagefault

+ 10 - 11
components/pmo/core/src/wlan_pmo_main.c

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2017-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2021,2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021,2023-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -295,12 +295,10 @@ static void wlan_pmo_init_cfg(struct wlan_objmgr_psoc *psoc,
 			cfg_get(psoc, CFG_DISCONNECT_SAP_TDLS_IN_WOW);
 	wlan_pmo_get_icmp_offload_enable_cfg(psoc, psoc_cfg);
 
-	psoc_cfg->enable_ssr_on_page_fault =
+	psoc_cfg->host_pf_action = cfg_get(psoc, CFG_HOST_ACTION_ON_PAGEFAULT);
+	psoc_cfg->min_pagefault_wakeups_for_action =
 				cfg_get(psoc,
-					CFG_ENABLE_SSR_ON_PAGEFAULT);
-	psoc_cfg->max_pagefault_wakeups_for_ssr =
-				cfg_get(psoc,
-					CFG_MAX_PAGEFAULT_WAKEUPS_FOR_SSR);
+					CFG_MIN_PAGEFAULT_WAKEUPS_FOR_ACTION);
 	psoc_cfg->interval_for_pagefault_wakeup_counts =
 			cfg_get(psoc,
 				CFG_INTERVAL_FOR_PAGEFAULT_WAKEUP_COUNT);
@@ -503,24 +501,25 @@ uint8_t pmo_core_psoc_get_txrx_handle(struct wlan_objmgr_psoc *psoc)
 	return txrx_pdev_id;
 }
 
-bool pmo_enable_ssr_on_page_fault(struct wlan_objmgr_psoc *psoc)
+enum pmo_page_fault_action
+pmo_host_action_on_page_fault(struct wlan_objmgr_psoc *psoc)
 {
 	struct pmo_psoc_priv_obj *pmo_psoc_ctx = pmo_psoc_get_priv(psoc);
 
 	if (!pmo_psoc_ctx)
-		return 0;
+		return PMO_PF_HOST_ACTION_NO_OP;
 
-	return pmo_psoc_ctx->psoc_cfg.enable_ssr_on_page_fault;
+	return pmo_psoc_ctx->psoc_cfg.host_pf_action;
 }
 
-uint8_t pmo_get_max_pagefault_wakeups_for_ssr(struct wlan_objmgr_psoc *psoc)
+uint8_t pmo_get_min_pagefault_wakeups_for_action(struct wlan_objmgr_psoc *psoc)
 {
 	struct pmo_psoc_priv_obj *pmo_psoc_ctx = pmo_psoc_get_priv(psoc);
 
 	if (!pmo_psoc_ctx)
 		return 0;
 
-	return pmo_psoc_ctx->psoc_cfg.max_pagefault_wakeups_for_ssr;
+	return pmo_psoc_ctx->psoc_cfg.min_pagefault_wakeups_for_action;
 }
 
 uint32_t

+ 42 - 38
components/pmo/dispatcher/inc/wlan_pmo_common_cfg.h

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2012-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -636,14 +636,14 @@
 
 /*
  * <ini>
- * enable_ssr_on_page_fault - Enable SSR on pagefault
- * @Min: 0
- * @Max: 1
- * @Default: 0
- *
- * This INI is used to enable/disable SSR when host is woken up with the reason
- * as pagefault.
- * For ex: If enable_ssr_on_page_fault = 1, max_pagefault_wakeups_for_ssr = 30,
+ * action_on_page_fault - Host action on page fault wakeup event
+ * @Min: PMO_PF_HOST_ACTION_NO_OP
+ * @Max: PMO_PF_HOST_ACTION_MAX - 1
+ * @Default: PMO_PF_HOST_ACTION_NO_OP
+ *
+ * This INI is used to determine host behavior on WOW_REASON_PAGE_FAULT wakeup
+ * event.
+ * For ex: If action is to trigger SSR, min_pagefault_wakeups_for_action = 30,
  * interval_for_pagefault_wakeup_counts = 180000 (3 mins) and
  * ssr_frequency_on_pagefault = 3600000 (1hr), in this case host will trigger
  * the SSR if it receives 30 wakeups because of pagefaults in 3 mins, host will
@@ -653,24 +653,28 @@
  *
  * </ini>
  */
-#define CFG_ENABLE_SSR_ON_PAGEFAULT CFG_INI_BOOL( \
-		"enable_ssr_on_page_fault", \
-		0, \
-		"Enable SSR on pagefault")
+#define CFG_HOST_ACTION_ON_PAGEFAULT CFG_INI_UINT( \
+		"action_on_page_fault", \
+		PMO_PF_HOST_ACTION_NO_OP, \
+		PMO_PF_HOST_ACTION_MAX - 1, \
+		PMO_PF_HOST_ACTION_NO_OP, \
+		CFG_VALUE_OR_DEFAULT, \
+		"Host action on FW pagefault event")
 
 /*
  * <ini>
- * max_pagefault_wakeups_for_ssr - Max number of pagefaults wakeups to trigger
- * SSR
- * @Min: 1
+ * min_pagefault_wakeups_for_action - Min number of pagefaults wakeups to
+ * initiate host action.
+ * @Min: 2
  * @Max: 255
  * @Default: 30
  *
- * This ini is used to trigger SSR if fw wakes up host for
- * max_pagefault_wakeups_for_ssr number of times in
- * interval_for_pagefault_wakeup_counts interval. SSR is triggered only once
- * in ssr_frequency_on_pagefault interval.
- * For ex: If enable_ssr_on_page_fault = 1, max_pagefault_wakeups_for_ssr = 30,
+ * This ini is used to get the count of max pagefault wakeups to reach before
+ * host takes action.
+ * If the count is reached within the wakeup time interval host will either
+ * trigger SSR (within the limits of SSR trigger freq) or may notify APPS or
+ * ignore if no action is set.
+ * For ex: If SSR on pagefault = 1, min_pagefault_wakeups_for_action = 30,
  * interval_for_pagefault_wakeup_counts = 180000 (3 mins) and
  * ssr_frequency_on_pagefault = 3600000 (1hr), in this case host will trigger
  * the SSR if it receives 30 wakeups because of pagefaults in 3 mins, host will
@@ -678,9 +682,9 @@
  * trigger next SSR for next 1 hr even if it receives 30 wakeups from fw because
  * of pagefaults. This 1 hr time is getting monitored from last SSR.
  */
-#define CFG_MAX_PAGEFAULT_WAKEUPS_FOR_SSR CFG_INI_UINT( \
-		"max_pagefault_wakeups_for_ssr", \
-		1, \
+#define CFG_MIN_PAGEFAULT_WAKEUPS_FOR_ACTION CFG_INI_UINT( \
+		"min_pagefault_wakeups_for_action", \
+		2, \
 		255, \
 		30, \
 		CFG_VALUE_OR_DEFAULT, \
@@ -689,16 +693,16 @@
 /*
  * <ini>
  * interval_for_pagefault_wakeup_counts - Time in ms in which
- * max_pagefault_wakeups_for_ssr needs to be monitored
- * @Min: 60000
- * @Max: 300000
+ * min_pagefault_wakeups_for_action needs to be monitored
+ * @Min: 60000 (1min)
+ * @Max: 18000000 (5hrs)
  * @Default: 180000 (3 mins)
  *
- * This ini define time in ms in which max_pagefault_wakeups_for_ssr needs to be
- * Monitored. If in interval_for_pagefault_wakeup_counts ms,
- * max_pagefault_wakeups_for_ssr is reached host will trigger the SSR.
+ * This ini define time in ms in which min_pagefault_wakeups_for_host action
+ * needs to be monitored. If in interval_for_pagefault_wakeup_counts ms,
+ * min_pagefault_wakeups_for_action is reached host will trigger the action.
  * SSR is triggered only once in ssr_frequency_on_pagefault interval.
- * For ex: If enable_ssr_on_page_fault = 1, max_pagefault_wakeups_for_ssr = 30,
+ * For ex: If SSR on pagefault = 1, min_pagefault_wakeups_for_action = 30,
  * interval_for_pagefault_wakeup_counts = 180000 (3 mins) and
  * ssr_frequency_on_pagefault = 3600000 (1hr), in this case host will trigger
  * the SSR if it receives 30 wakeups because of pagefaults in 3 mins, host will
@@ -709,10 +713,10 @@
 #define CFG_INTERVAL_FOR_PAGEFAULT_WAKEUP_COUNT CFG_INI_UINT( \
 	"interval_for_pagefault_wakeup_counts", \
 	60000, \
-	300000, \
+	18000000, \
 	180000, \
 	CFG_VALUE_OR_DEFAULT, \
-	"Interval in which max_pagefault_wakeups_for_ssr needs to be monitored")
+	"Interval in which min_pagefault_wakeups_for_ssr needs to be monitored")
 
 /*
  * <ini>
@@ -723,11 +727,11 @@
  * @Default: 3600000 (1 hr)
  *
  * This ini define time in ms in which next SSR needs to be triggered if
- * max_pagefault_wakeups_for_ssr is reached in
+ * min_pagefault_wakeups_for_action is reached in
  * interval_for_pagefault_wakeup_counts time.
  * INIs max_pagefault_wakeups_for_ssr, interval_for_pagefault_wakeup_counts and
  * ssr_frequency_on_pagefault needs to be considered together.
- * For ex: If enable_ssr_on_page_fault = 1, max_pagefault_wakeups_for_ssr = 30,
+ * For ex: If enable_ssr_on_page_fault = 1, min_pagefault_wakeups_for_ssr = 30,
  * interval_for_pagefault_wakeup_counts = 180000 (3 mins) and
  * ssr_frequency_on_pagefault = 3600000 (1hr), in this case host will trigger
  * the SSR if it receives 30 wakeups because of pagefaults in 3 mins, host will
@@ -741,7 +745,7 @@
 	7200000, \
 	3600000, \
 	CFG_VALUE_OR_DEFAULT, \
-	"Interval in which max_pagefault_wakeups_for_ssr needs to be monitored")
+	"Interval in which min_pagefault_wakeups_for_ssr needs to be monitored")
 
 /*
  * <ini>
@@ -790,8 +794,8 @@
 	CFG(CFG_DISCONNECT_SAP_TDLS_IN_WOW) \
 	CFG(CFG_IGMP_VERSION_SUPPORT) \
 	CFG(CFG_ENABLE_ICMP_OFFLOAD) \
-	CFG(CFG_ENABLE_SSR_ON_PAGEFAULT) \
-	CFG(CFG_MAX_PAGEFAULT_WAKEUPS_FOR_SSR) \
+	CFG(CFG_HOST_ACTION_ON_PAGEFAULT) \
+	CFG(CFG_MIN_PAGEFAULT_WAKEUPS_FOR_ACTION) \
 	CFG(CFG_INTERVAL_FOR_PAGEFAULT_WAKEUP_COUNT) \
 	CFG(CFG_SSR_FREQUENCY_ON_PAGEFAULT)
 

+ 23 - 6
components/pmo/dispatcher/inc/wlan_pmo_common_public_struct.h

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2017-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -331,6 +331,23 @@ struct pmo_icmp_offload {
 };
 #endif
 
+/*
+ * enum pmo_page_fault_action - Host action on FW page fault
+ * @PMO_PF_HOST_ACTION_NO_OP: Host will ignore PF wakeup event.
+ * @PMO_PF_HOST_ACTION_TRIGGER_SSR: Host will trigger SSR on PF threshold.
+ * @PMO_PF_HOST_ACTION_NOTIFY_APPS: Host will notify APPS on PF threshold.
+ *
+ * @PMO_PF_HOST_ACTION_MAX: Reserved and invalid value
+ */
+enum pmo_page_fault_action {
+	PMO_PF_HOST_ACTION_NO_OP = 0,
+	PMO_PF_HOST_ACTION_TRIGGER_SSR = 1,
+	PMO_PF_HOST_ACTION_NOTIFY_APPS = 2,
+
+	/* Keep it last */
+	PMO_PF_HOST_ACTION_MAX,
+};
+
 /**
  * struct pmo_psoc_cfg - user configuration required for pmo
  * @ptrn_match_enable_all_vdev: true when pattern match is enable for all vdev
@@ -402,9 +419,9 @@ struct pmo_icmp_offload {
  * @disconnect_sap_tdls_in_wow: sap/p2p_go disconnect or teardown tdls link
  * @is_icmp_offload_enable: true if icmp offload is supported
  *	for psoc else false
- * @enable_ssr_on_page_fault: Enable ssr on pagefault
- * @max_pagefault_wakeups_for_ssr: Maximum number of pagefaults after which host
- * needs to trigger SSR
+ * @host_pf_action: Action to take on page fault
+ * @min_pagefault_wakeups_for_action: Min number of pagefaults after which
+ * host needs to start act upon.
  * @interval_for_pagefault_wakeup_counts: Time in ms in which max pagefault
  * wakeups needs to be monitored.
  * @ssr_frequency_on_pagefault: Time in ms in which SSR needs to be triggered
@@ -490,8 +507,8 @@ struct pmo_psoc_cfg {
 #ifdef WLAN_FEATURE_ICMP_OFFLOAD
 	bool is_icmp_offload_enable;
 #endif
-	bool enable_ssr_on_page_fault;
-	uint8_t max_pagefault_wakeups_for_ssr;
+	enum pmo_page_fault_action host_pf_action;
+	uint8_t min_pagefault_wakeups_for_action;
 	uint32_t interval_for_pagefault_wakeup_counts;
 	uint32_t ssr_frequency_on_pagefault;
 };

+ 20 - 6
components/pmo/dispatcher/inc/wlan_pmo_obj_mgmt_api.h

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2017-2019, 2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2022-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -294,6 +294,14 @@ wlan_pmo_get_sap_mode_bus_suspend(struct wlan_objmgr_psoc *psoc);
 bool
 wlan_pmo_get_go_mode_bus_suspend(struct wlan_objmgr_psoc *psoc);
 
+/*
+ * wlan_pmo_no_op_on_page_fault() - Whether to ignore page fault wakeups
+ * @psoc: PSOC object manager
+ *
+ * Return: true if host has to ignore page fault wakeup events else false.
+ */
+bool wlan_pmo_no_op_on_page_fault(struct wlan_objmgr_psoc *psoc);
+
 /*
  * wlan_pmo_enable_ssr_on_page_fault: Enable/disable ssr on pagefault
  * @psoc: objmgr psoc
@@ -303,13 +311,14 @@ wlan_pmo_get_go_mode_bus_suspend(struct wlan_objmgr_psoc *psoc);
 bool wlan_pmo_enable_ssr_on_page_fault(struct wlan_objmgr_psoc *psoc);
 
 /*
- * wlan_pmo_get_max_pagefault_wakeups_for_ssr: get max pagefault wakeups for ssr
+ * wlan_pmo_get_min_pagefault_wakeups_for_action() - get min pagefault wakeups
+ * for host to initiate action
  * @psoc: objmgr psoc
  *
- * Return: Max pagefault wakeups for SSR
+ * Return: Min pagefault wakeups for action
  */
 uint8_t
-wlan_pmo_get_max_pagefault_wakeups_for_ssr(struct wlan_objmgr_psoc *psoc);
+wlan_pmo_get_min_pagefault_wakeups_for_action(struct wlan_objmgr_psoc *psoc);
 
 /*
  * wlan_pmo_get_interval_for_pagefault_wakeup_counts: get ssr interval for
@@ -508,14 +517,19 @@ wlan_pmo_get_go_mode_bus_suspend(struct wlan_objmgr_psoc *psoc)
 	return false;
 }
 
+static inline bool wlan_pmo_no_op_on_page_fault(struct wlan_objmgr_psoc *psoc)
+{
+	return true;
+}
+
 static inline bool
 wlan_pmo_enable_ssr_on_page_fault(struct wlan_objmgr_psoc *psoc)
 {
-	return 0;
+	return false;
 }
 
 static inline uint8_t
-wlan_pmo_get_max_pagefault_wakeups_for_ssr(struct wlan_objmgr_psoc *psoc)
+wlan_pmo_get_min_pagefault_wakeups_for_action(struct wlan_objmgr_psoc *psoc)
 {
 	return 0;
 }

+ 8 - 3
components/pmo/dispatcher/src/wlan_pmo_obj_mgmt_api.c

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2018-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2023-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -893,15 +893,20 @@ wlan_pmo_get_go_mode_bus_suspend(struct wlan_objmgr_psoc *psoc)
 	return pmo_psoc_ctx->psoc_cfg.is_bus_suspend_enabled_in_go_mode;
 }
 
+bool wlan_pmo_no_op_on_page_fault(struct wlan_objmgr_psoc *psoc)
+{
+	return pmo_no_op_on_page_fault(psoc);
+}
+
 bool wlan_pmo_enable_ssr_on_page_fault(struct wlan_objmgr_psoc *psoc)
 {
 	return pmo_enable_ssr_on_page_fault(psoc);
 }
 
 uint8_t
-wlan_pmo_get_max_pagefault_wakeups_for_ssr(struct wlan_objmgr_psoc *psoc)
+wlan_pmo_get_min_pagefault_wakeups_for_action(struct wlan_objmgr_psoc *psoc)
 {
-	return pmo_get_max_pagefault_wakeups_for_ssr(psoc);
+	return pmo_get_min_pagefault_wakeups_for_action(psoc);
 }
 
 uint32_t

+ 4 - 0
core/hdd/src/wlan_hdd_cfg80211.c

@@ -2259,6 +2259,10 @@ static const struct nl80211_vendor_cmd_info wlan_hdd_cfg80211_vendor_events[] =
 	},
 
 	FEATURE_TX_LATENCY_STATS_EVENTS
+	[QCA_NL80211_VENDOR_SUBCMD_FW_PAGE_FAULT_REPORT_INDEX] = {
+		.vendor_id = QCA_NL80211_VENDOR_ID,
+		.subcmd = QCA_NL80211_VENDOR_SUBCMD_FW_PAGE_FAULT_REPORT,
+	},
 };
 
 /**

+ 53 - 15
core/hdd/src/wlan_hdd_main.c

@@ -17763,37 +17763,75 @@ QDF_STATUS hdd_md_bl_evt_cb(void *ctx, struct sir_md_bl_evt *event)
 }
 #endif /* WLAN_FEATURE_MOTION_DETECTION */
 
-/**
- * hdd_ssr_on_pagefault_cb - Callback to trigger SSR because
- * of host wake up by firmware with reason pagefault
- *
- * Return: None
- */
-static void hdd_ssr_on_pagefault_cb(void)
+static QDF_STATUS hdd_ssr_on_pagefault_cb(struct hdd_context *hdd_ctx)
 {
 	uint32_t ssr_frequency_on_pagefault;
-	struct hdd_context *hdd_ctx = cds_get_context(QDF_MODULE_ID_HDD);
-	qdf_time_t curr_time;
+	qdf_time_t curr_time, ssr_threshold;
 
 	hdd_enter();
 
 	if (!hdd_ctx)
-		return;
+		return QDF_STATUS_E_NULL_VALUE;
 
 	ssr_frequency_on_pagefault =
 		ucfg_pmo_get_ssr_frequency_on_pagefault(hdd_ctx->psoc);
 
-	curr_time = qdf_get_time_of_the_day_ms();
+	curr_time = qdf_get_system_uptime();
+	ssr_threshold = qdf_system_msecs_to_ticks(ssr_frequency_on_pagefault);
 
 	if (!hdd_ctx->last_pagefault_ssr_time ||
-	    (curr_time - hdd_ctx->last_pagefault_ssr_time) >=
-					ssr_frequency_on_pagefault) {
+	    (curr_time - hdd_ctx->last_pagefault_ssr_time) >= ssr_threshold) {
 		hdd_info("curr_time %lu last_pagefault_ssr_time %lu ssr_frequency %d",
 			 curr_time, hdd_ctx->last_pagefault_ssr_time,
-			 ssr_frequency_on_pagefault);
+			 ssr_threshold);
 		hdd_ctx->last_pagefault_ssr_time = curr_time;
 		cds_trigger_recovery(QDF_HOST_WAKEUP_REASON_PAGEFAULT);
+
+		return QDF_STATUS_SUCCESS;
 	}
+
+	return QDF_STATUS_E_AGAIN;
+}
+
+#define FW_PAGE_FAULT_IDX QCA_NL80211_VENDOR_SUBCMD_FW_PAGE_FAULT_REPORT_INDEX
+static QDF_STATUS hdd_send_pagefault_report_to_user(struct hdd_context *hdd_ctx,
+						    void *buf, uint32_t buf_len)
+{
+	struct sk_buff *event_buf;
+	int flags = cds_get_gfp_flags();
+	uint8_t *ev_data = buf;
+	uint16_t event_len = NLMSG_HDRLEN + buf_len;
+
+	event_buf = wlan_cfg80211_vendor_event_alloc(hdd_ctx->wiphy, NULL,
+						     event_len,
+						     FW_PAGE_FAULT_IDX, flags);
+	if (!event_buf) {
+		hdd_err("wlan_cfg80211_vendor_event_alloc failed");
+		return QDF_STATUS_E_NOMEM;
+	}
+
+	if (nla_put(event_buf, QCA_WLAN_VENDOR_ATTR_FW_PAGE_FAULT_REPORT_DATA,
+		    buf_len, ev_data)) {
+		hdd_debug("Failed to fill pagefault blob data");
+		wlan_cfg80211_vendor_free_skb(event_buf);
+		return QDF_STATUS_E_FAILURE;
+	}
+
+	wlan_cfg80211_vendor_event(event_buf, flags);
+	return QDF_STATUS_SUCCESS;
+}
+
+static QDF_STATUS hdd_pagefault_action_cb(void *buf, uint32_t buf_len)
+{
+	struct hdd_context *hdd_ctx = cds_get_context(QDF_MODULE_ID_HDD);
+
+	if (!hdd_ctx)
+		return QDF_STATUS_E_NULL_VALUE;
+
+	if (wlan_pmo_enable_ssr_on_page_fault(hdd_ctx->psoc))
+		return hdd_ssr_on_pagefault_cb(hdd_ctx);
+
+	return hdd_send_pagefault_report_to_user(hdd_ctx, buf, buf_len);
 }
 
 /**
@@ -17904,7 +17942,7 @@ int hdd_register_cb(struct hdd_context *hdd_ctx)
 	sme_async_oem_event_init(mac_handle,
 				 hdd_oem_event_async_cb);
 
-	sme_register_ssr_on_pagefault_cb(mac_handle, hdd_ssr_on_pagefault_cb);
+	sme_register_pagefault_cb(mac_handle, hdd_pagefault_action_cb);
 
 	hdd_exit();
 

+ 7 - 5
core/sme/inc/sme_api.h

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2012-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -762,14 +762,16 @@ QDF_STATUS sme_neighbor_report_request(mac_handle_t mac_handle,
 		tpRrmNeighborRspCallbackInfo callbackInfo);
 
 /**
- * sme_register_ssr_on_pagefault_cb() - Register cb to trigger SSR on pagefault
+ * sme_register_pagefault_cb() - Register cb to handle host action on pagefault
  * @mac_handle: Opaque handle to the global MAC context.
- * @hdd_ssr_on_pagefault_cb: Callback which needs to be registered
+ * @hdd_pagefault_action_cb: Callback which needs to be registered
  *
  * Return: None
  */
-void sme_register_ssr_on_pagefault_cb(mac_handle_t mac_handle,
-				      void (*hdd_ssr_on_pagefault_cb)(void));
+void
+sme_register_pagefault_cb(mac_handle_t mac_handle,
+			  QDF_STATUS (*hdd_pagefault_action_cb)(void *buf,
+								uint32_t buf_len));
 
 /**
  * sme_deregister_ssr_on_pagefault_cb() - Deregister cb to trigger SSR on

+ 2 - 2
core/sme/inc/sme_internal.h

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2011-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -507,7 +507,7 @@ struct sme_context {
 			(const struct oem_data *oem_event_data);
 #endif
 
-	void (*ssr_on_pagefault_cb)(void);
+	QDF_STATUS (*pagefault_action_cb)(void *buf, uint32_t data);
 
 #ifdef MULTI_CLIENT_LL_SUPPORT
 	void (*latency_level_event_handler_cb)

+ 6 - 4
core/sme/src/common/sme_api.c

@@ -4058,8 +4058,10 @@ QDF_STATUS sme_neighbor_report_request(
 	return status;
 }
 
-void sme_register_ssr_on_pagefault_cb(mac_handle_t mac_handle,
-				      void (*hdd_ssr_on_pagefault_cb)(void))
+void
+sme_register_pagefault_cb(mac_handle_t mac_handle,
+			  QDF_STATUS (*hdd_pagefault_action_cb)(void *buf,
+								uint32_t data))
 {
 	QDF_STATUS status;
 	struct mac_context *mac = MAC_CONTEXT(mac_handle);
@@ -4068,7 +4070,7 @@ void sme_register_ssr_on_pagefault_cb(mac_handle_t mac_handle,
 
 	status = sme_acquire_global_lock(&mac->sme);
 	if (QDF_IS_STATUS_SUCCESS(status)) {
-		mac->sme.ssr_on_pagefault_cb = hdd_ssr_on_pagefault_cb;
+		mac->sme.pagefault_action_cb = hdd_pagefault_action_cb;
 		sme_release_global_lock(&mac->sme);
 	}
 
@@ -4084,7 +4086,7 @@ void sme_deregister_ssr_on_pagefault_cb(mac_handle_t mac_handle)
 
 	status = sme_acquire_global_lock(&mac->sme);
 	if (QDF_IS_STATUS_SUCCESS(status)) {
-		mac->sme.ssr_on_pagefault_cb = NULL;
+		mac->sme.pagefault_action_cb = NULL;
 		sme_release_global_lock(&mac->sme);
 	}
 

+ 60 - 6
core/wma/inc/wma.h

@@ -49,6 +49,7 @@
 #include "wmi.h"
 #include "wlan_cm_roam_public_struct.h"
 #include "target_if.h"
+#include <qdf_hang_event_notifier.h>
 
 /* Platform specific configuration for max. no. of fragments */
 #define QCA_OL_11AC_TX_MAX_FRAGS            2
@@ -802,6 +803,63 @@ struct wma_wlm_stats_data {
 };
 #endif
 
+#define WLAN_WMA_MAX_PF_SYM 50
+#define WLAN_WMA_PF_APPS_NOTIFY_BUF_LEN QDF_HANG_EVENT_DATA_SIZE
+#define WLAN_WMA_PF_SYM_LEN 4
+#define WLAN_WMA_PF_SYM_CNT_LEN 1
+#define WLAN_WMA_PF_SYM_FLAGS_LEN 1
+#define WLAN_WMA_PER_PF_SYM_NOTIFY_BUF_LEN  (WLAN_WMA_PF_SYM_LEN + \
+					     WLAN_WMA_PF_SYM_CNT_LEN + \
+					     WLAN_WMA_PF_SYM_FLAGS_LEN)
+
+/*
+ * struct wow_pf_sym - WOW PF wakeup symbol info
+ * @symbol: Address of PF symbol
+ * @count: Count of PF symbol
+ * @flags: Flags associated with @symbol
+ */
+struct wow_pf_sym {
+	uint32_t symbol;
+	uint8_t count;
+	uint8_t flags;
+};
+
+/*
+ * struct wow_pf_wakeup_ev_data - WOW PF wakeup event data
+ * @pf_sym: Array of each unique PF symbol in wakeup event payload
+ * @num_pf_syms: Total unique symbols in event.
+ * @pending_pf_syms: Pending PF symbols to process
+ */
+struct wow_pf_wakeup_ev_data {
+	struct wow_pf_sym *pf_sym;
+	uint8_t num_pf_syms;
+	uint8_t pending_pf_syms;
+};
+
+/**
+ * struct wma_pf_sym - Per symbol PF data in PF symbol history
+ * @pf_sym: PF symbol info
+ * @pf_event_ts: Array of page fault event ts
+ */
+struct wma_pf_sym {
+	struct wow_pf_sym pf_sym;
+	qdf_time_t *pf_ev_ts;
+};
+
+/*
+ * struct wma_pf_sym_hist - System level FW PF symbol history
+ * @wma_pf_sym: Array of symbols in history.
+ * @pf_notify_buf_ptr: Pointer to APPS notify buffer
+ * @pf_notify_buf_len: Current data length of @pf_notify_buf_ptr
+ * @lock: Lock to access PF symbol history
+ */
+struct wma_pf_sym_hist {
+	struct wma_pf_sym wma_pf_sym[WLAN_WMA_MAX_PF_SYM];
+	uint8_t *pf_notify_buf_ptr;
+	uint32_t pf_notify_buf_len;
+	qdf_spinlock_t lock;
+};
+
 /**
  * struct t_wma_handle - wma context
  * @wmi_handle: wmi handle
@@ -920,10 +978,7 @@ struct wma_wlm_stats_data {
  * * @fw_therm_throt_support: FW Supports thermal throttling?
  * @eht_cap: 802.11be capabilities
  * @set_hw_mode_resp_status: Set HW mode response status
- * @pagefault_wakeups_ts: Stores timestamps at which host wakes up by fw
- * because of pagefaults
- * @num_page_fault_wakeups: Stores the number of times host wakes up by fw
- * because of pagefaults
+ * @wma_pf_hist: PF symbol history
  *
  * This structure is the global wma context.  It contains global wma
  * module parameters and handles of other modules.
@@ -1060,8 +1115,7 @@ typedef struct {
 	qdf_wake_lock_t sap_d3_wow_wake_lock;
 	qdf_wake_lock_t go_d3_wow_wake_lock;
 	enum set_hw_mode_status set_hw_mode_resp_status;
-	qdf_time_t *pagefault_wakeups_ts;
-	uint8_t num_page_fault_wakeups;
+	struct wma_pf_sym_hist wma_pf_hist;
 } t_wma_handle, *tp_wma_handle;
 
 /**

+ 382 - 58
core/wma/src/wma_features.c

@@ -3230,39 +3230,357 @@ static void wma_wake_event_log_reason(t_wma_handle *wma,
 	qdf_wma_wow_wakeup_stats_event(wma);
 }
 
-/**
- * wma_wow_wakeup_host_trigger_ssr() - Trigger SSR on host wakeup
- * @handle: wma handle
- * @reason: Host wakeup reason
- *
- * This function triggers SSR if host is woken up by fw with reason as pagefault
- *
- * Return: None
- */
-static void
-wma_wow_wakeup_host_trigger_ssr(t_wma_handle *wma, uint32_t reason)
+static QDF_STATUS wma_wow_pagefault_action_cb(void *buf)
 {
-	uint8_t pagefault_wakeups_for_ssr;
-	uint32_t interval_for_pagefault_wakeup_counts;
-	qdf_time_t curr_time;
 	struct mac_context *mac = cds_get_context(QDF_MODULE_ID_PE);
-	int i;
-	bool ignore_pf = true;
 
-	if (!mac) {
-		wma_debug("NULL mac ptr");
-		return;
+	return mac->sme.pagefault_action_cb(buf, WLAN_WMA_PF_APPS_NOTIFY_BUF_LEN);
+}
+
+static QDF_STATUS
+wma_wow_pagefault_add_sym_to_event(tp_wma_handle wma, struct wow_pf_sym *pf_sym)
+{
+	uint8_t *buf_ptr = wma->wma_pf_hist.pf_notify_buf_ptr;
+	uint32_t buf_len = wma->wma_pf_hist.pf_notify_buf_len;
+
+	if (WLAN_WMA_PF_APPS_NOTIFY_BUF_LEN - buf_len <
+	    WLAN_WMA_PER_PF_SYM_NOTIFY_BUF_LEN) {
+		wma_wow_pagefault_action_cb(buf_ptr);
+		buf_len = 0;
+	}
+
+	/* Mem zero to send buffer with zero padding */
+	if (!buf_len)
+		qdf_mem_zero(buf_ptr, WLAN_WMA_PF_APPS_NOTIFY_BUF_LEN);
+
+	qdf_mem_copy(buf_ptr + buf_len, pf_sym,
+		     WLAN_WMA_PER_PF_SYM_NOTIFY_BUF_LEN);
+	buf_len += WLAN_WMA_PER_PF_SYM_NOTIFY_BUF_LEN;
+
+	wma->wma_pf_hist.pf_notify_buf_len = buf_len;
+
+	return QDF_STATUS_SUCCESS;
+}
+
+static void
+wma_wow_pagefault_add_new_sym_from_event(tp_wma_handle wma,
+					 struct wow_pf_wakeup_ev_data *pf_sym_list,
+					 qdf_time_t cur_time)
+{
+	uint8_t tbl_idx, ev_lst_idx, new_pf_idx, idx2;
+	uint8_t new_idx_cnt, cur_idx_cnt, ev_sym_cnt, pf_th;
+	uint8_t max_sym_count = WLAN_WMA_MAX_PF_SYM;
+	qdf_time_t new_idx_last_ts, cur_idx_last_ts;
+	qdf_time_t new_idx_old_ts, cur_idx_old_ts;
+	struct wma_pf_sym *cur_pf_entry, *new_pf_entry;
+	struct wma_pf_sym_hist *pf_sym_hist = &wma->wma_pf_hist;
+
+	pf_th = wlan_pmo_get_min_pagefault_wakeups_for_action(wma->psoc);
+	for (tbl_idx = 0; tbl_idx < max_sym_count; tbl_idx++) {
+		if (!pf_sym_list->pending_pf_syms)
+			break;
+
+		new_pf_idx = tbl_idx;
+		new_pf_entry = &pf_sym_hist->wma_pf_sym[tbl_idx];
+		new_idx_cnt = new_pf_entry->pf_sym.count;
+		new_idx_last_ts = new_pf_entry->pf_ev_ts[new_idx_cnt - 1];
+		new_idx_old_ts = new_pf_entry->pf_ev_ts[0];
+
+		for (ev_lst_idx = 0; ev_lst_idx < pf_sym_list->num_pf_syms;
+		     ev_lst_idx++) {
+			if (!pf_sym_list->pf_sym[ev_lst_idx].count)
+				continue;
+
+			/* Add the sym that already met threshold to the buf */
+			if (pf_sym_list->pf_sym[ev_lst_idx].count >= pf_th) {
+				wma_wow_pagefault_add_sym_to_event(wma,
+								   pf_sym_list->pf_sym);
+				pf_sym_list->pf_sym[ev_lst_idx].count = 0x0;
+				pf_sym_list->pending_pf_syms--;
+				continue;
+			}
+
+			ev_sym_cnt = pf_sym_list->pf_sym[ev_lst_idx].count;
+
+			/* If current entry is NULL, add the symbol here */
+			if (!new_idx_cnt)
+				goto add_sym;
+
+			/* Replace event if count is equal as current event
+			 * is latest and don't replace symbol from current event
+			 */
+			if (new_idx_cnt > ev_sym_cnt ||
+			    qdf_system_time_after_eq(new_idx_last_ts, cur_time))
+				break;
+
+			for (idx2 = tbl_idx + 1; idx2 < max_sym_count; idx2++) {
+				cur_pf_entry = &pf_sym_hist->wma_pf_sym[idx2];
+				cur_idx_cnt = cur_pf_entry->pf_sym.count;
+				cur_idx_last_ts =
+					cur_pf_entry->pf_ev_ts[cur_idx_cnt - 1];
+				cur_idx_old_ts = cur_pf_entry->pf_ev_ts[0];
+
+				if (cur_idx_cnt > new_idx_cnt)
+					continue;
+
+				/* Don't replace symbol from current event */
+				if (qdf_system_time_after_eq(cur_idx_last_ts,
+							     cur_time)) {
+					continue;
+				}
+
+				if (cur_idx_cnt == new_idx_cnt &&
+				    qdf_system_time_after_eq(cur_idx_old_ts,
+							     new_idx_old_ts)) {
+					continue;
+				}
+
+				new_pf_idx = idx2;
+				new_idx_cnt = cur_idx_cnt;
+				new_idx_last_ts = cur_idx_last_ts;
+				new_idx_old_ts = cur_idx_old_ts;
+			}
+
+add_sym:
+			/* Replace symbol with current event symbol */
+			new_pf_entry = &pf_sym_hist->wma_pf_sym[new_pf_idx];
+			new_pf_entry->pf_sym.symbol =
+					pf_sym_list->pf_sym[ev_lst_idx].symbol;
+			new_pf_entry->pf_sym.count = ev_sym_cnt;
+			for (idx2 = 0; idx2 < ev_sym_cnt; idx2++)
+				new_pf_entry->pf_ev_ts[idx2] = cur_time;
+
+			pf_sym_list->pending_pf_syms--;
+			pf_sym_list->pf_sym[ev_lst_idx].count = 0;
+			break;
+		}
+	}
+}
+
+static void
+wma_wow_pagefault_process_existing_syms(tp_wma_handle wma,
+					struct wma_pf_sym *pf_sym_entry,
+					struct wow_pf_wakeup_ev_data *ev_list,
+					qdf_time_t cur_time)
+{
+	uint8_t ev_idx, add_idx;
+	uint8_t pf_th, *tbl_sym_cnt, ev_sym_cnt;
+
+	pf_th = wlan_pmo_get_min_pagefault_wakeups_for_action(wma->psoc);
+	tbl_sym_cnt = &pf_sym_entry->pf_sym.count;
+
+	for (ev_idx = 0; ev_idx < ev_list->num_pf_syms; ev_idx++) {
+		/* Ignore if all symbols are processed */
+		if (!ev_list->pending_pf_syms)
+			break;
+
+		if (!ev_list->pf_sym[ev_idx].count)
+			continue;
+
+		/* Only process symbol equals current entry in the history */
+		if (ev_list->pf_sym[ev_idx].symbol !=
+		    pf_sym_entry->pf_sym.symbol) {
+			continue;
+		}
+
+		ev_sym_cnt = ev_list->pf_sym[ev_idx].count;
+
+		/* If symbol reaches threshold, then clear the ts data */
+		if (*tbl_sym_cnt + ev_sym_cnt >= pf_th) {
+			qdf_mem_zero(&pf_sym_entry->pf_ev_ts[0],
+				     (*tbl_sym_cnt * sizeof(qdf_time_t)));
+			*tbl_sym_cnt += ev_sym_cnt;
+			wma_wow_pagefault_add_sym_to_event(wma,
+							   &pf_sym_entry->pf_sym);
+			*tbl_sym_cnt = 0x0;
+
+			goto sym_handled;
+		}
+
+		for (add_idx = 0; add_idx < ev_sym_cnt; add_idx++)
+			pf_sym_entry->pf_ev_ts[(*tbl_sym_cnt)++] = cur_time;
+
+sym_handled:
+		ev_list->pending_pf_syms--;
+		ev_list->pf_sym[ev_idx].count = 0;
+		break;
+	}
+}
+
+static void
+wma_wow_pagefault_flush_ageout_entries(struct wma_pf_sym *pf_sym_entry,
+				       qdf_time_t cutoff_time)
+{
+	qdf_time_t entry_ts;
+	uint8_t *cur_pf_count, pf_ts_idx;
+
+	cur_pf_count = &pf_sym_entry->pf_sym.count;
+	/* Find the count of entries which elapsed cutoff time */
+	for (pf_ts_idx = 0; pf_ts_idx < *cur_pf_count; pf_ts_idx++) {
+		entry_ts = pf_sym_entry->pf_ev_ts[pf_ts_idx];
+		if (qdf_system_time_before(cutoff_time, entry_ts))
+			break;
+	}
+
+	/* Remove the entries which elapsed cutoff time */
+	if (pf_ts_idx > 0) {
+		*cur_pf_count -= pf_ts_idx;
+		qdf_mem_copy(&pf_sym_entry->pf_ev_ts[0],
+			     &pf_sym_entry->pf_ev_ts[pf_ts_idx],
+			     (*cur_pf_count * sizeof(qdf_time_t)));
+		qdf_mem_zero(&pf_sym_entry->pf_ev_ts[*cur_pf_count],
+			     pf_ts_idx * sizeof(qdf_time_t));
+	}
+}
+
+static QDF_STATUS
+wma_wow_pagefault_parse_event(struct wlan_objmgr_psoc *psoc,
+			      void *ev, uint32_t ev_len,
+			      struct wow_pf_wakeup_ev_data *pf_sym_list)
+{
+	WMI_WOW_WAKEUP_HOST_EVENTID_param_tlvs *event_param = ev;
+	uint8_t *cur_list_count, *pf_sym_addr, buf_idx, sym_idx, i;
+	uint32_t packet_len, symbol, pf_sym_count;
+	struct wow_pf_sym tmp_pf_sym;
+
+	if (event_param->num_wow_packet_buffer <= sizeof(packet_len)) {
+		wma_err("Invalid wow packet buffer from FW %u",
+			event_param->num_wow_packet_buffer);
+		return QDF_STATUS_E_INVAL;
+	}
+
+	packet_len = *(uint32_t *)event_param->wow_packet_buffer;
+	if (!packet_len) {
+		wma_err("Wake event packet is empty");
+		return QDF_STATUS_E_INVAL;
 	}
 
-	if (WOW_REASON_PAGE_FAULT != reason)
+	if (packet_len >
+	    (event_param->num_wow_packet_buffer - sizeof(packet_len))) {
+		wma_err("Invalid packet_len from firmware, packet_len: %u, num_wow_packet_buffer: %u",
+			packet_len, event_param->num_wow_packet_buffer);
+		return QDF_STATUS_E_INVAL;
+	}
+
+	pf_sym_count = packet_len / sizeof(symbol);
+	/* First 4 bytes following packet len contains UUID */
+	pf_sym_count--;
+
+	pf_sym_list->pf_sym =
+		qdf_mem_malloc(pf_sym_count * sizeof(*pf_sym_list->pf_sym));
+	if (!pf_sym_list->pf_sym)
+		return QDF_STATUS_E_NOMEM;
+
+	/* First 4 bytes of buffer gives length and next 4 bytes for UUID */
+	pf_sym_addr = event_param->wow_packet_buffer + (sizeof(packet_len) * 2);
+
+	cur_list_count = &pf_sym_list->num_pf_syms;
+	for (buf_idx = 0; buf_idx < pf_sym_count; buf_idx++) {
+		symbol = *(uint32_t *)pf_sym_addr;
+
+		/* Ignore invalid symbols */
+		if (!symbol || symbol == (uint32_t)-1)
+			goto iter_next_sym;
+
+		for (sym_idx = 0; sym_idx < *cur_list_count; sym_idx++) {
+			if (pf_sym_list->pf_sym[sym_idx].symbol == symbol) {
+				pf_sym_list->pf_sym[sym_idx].count++;
+				goto iter_next_sym;
+			}
+		}
+
+		pf_sym_list->pf_sym[*cur_list_count].symbol = symbol;
+		pf_sym_list->pf_sym[*cur_list_count].count++;
+		(*cur_list_count)++;
+
+iter_next_sym:
+		pf_sym_addr += sizeof(symbol);
+	}
+
+	pf_sym_list->pending_pf_syms = *cur_list_count;
+
+	/* Reorder to prioritize syms with high frequency */
+	for (sym_idx = 0; sym_idx < *cur_list_count; sym_idx++) {
+		for (i = sym_idx + 1; i < *cur_list_count; i++) {
+			if (pf_sym_list->pf_sym[i].count <=
+			    pf_sym_list->pf_sym[sym_idx].count) {
+				continue;
+			}
+
+			tmp_pf_sym.symbol = pf_sym_list->pf_sym[sym_idx].symbol;
+			tmp_pf_sym.count = pf_sym_list->pf_sym[sym_idx].count;
+
+			pf_sym_list->pf_sym[sym_idx].symbol =
+						pf_sym_list->pf_sym[i].symbol;
+			pf_sym_list->pf_sym[sym_idx].count =
+						pf_sym_list->pf_sym[i].count;
+
+			pf_sym_list->pf_sym[i].symbol = tmp_pf_sym.symbol;
+			pf_sym_list->pf_sym[i].count = tmp_pf_sym.count;
+		}
+	}
+
+	return QDF_STATUS_SUCCESS;
+}
+
+static void wma_wow_pagefault_handle_ssr_action(tp_wma_handle wma)
+{
+	QDF_STATUS status;
+	uint8_t *cur_count, pf_thresh;
+	qdf_time_t cur_time, cutoff_time;
+	uint32_t pf_wakeup_intv;
+	struct wma_pf_sym *pf_sym_entry;
+
+	cur_time = qdf_get_system_uptime();
+	pf_wakeup_intv =
+		wlan_pmo_get_interval_for_pagefault_wakeup_counts(wma->psoc);
+	pf_thresh = wlan_pmo_get_min_pagefault_wakeups_for_action(wma->psoc);
+	cutoff_time = cur_time - qdf_system_msecs_to_ticks(pf_wakeup_intv);
+
+	pf_sym_entry = &wma->wma_pf_hist.wma_pf_sym[0];
+	cur_count = &pf_sym_entry->pf_sym.count;
+	if (cutoff_time < cur_time) {
+		wma_wow_pagefault_flush_ageout_entries(pf_sym_entry,
+						       cutoff_time);
+	}
+
+	pf_sym_entry->pf_ev_ts[(*cur_count)++] = cur_time;
+	if (*cur_count < pf_thresh)
 		return;
 
-	if (!mac->sme.ssr_on_pagefault_cb) {
-		wma_debug("NULL SSR on pagefault cb");
+	/* If SSR threshold condition fails, SSR will not be triggered, so
+	 * save current event and flush oldest entry.
+	 */
+	status = wma_wow_pagefault_action_cb(NULL);
+	if (QDF_IS_STATUS_ERROR(status)) {
+		(*cur_count)--;
+		qdf_mem_copy(&pf_sym_entry->pf_ev_ts[0],
+			     &pf_sym_entry->pf_ev_ts[1],
+			     (*cur_count * sizeof(qdf_time_t)));
+		qdf_mem_zero(&pf_sym_entry->pf_ev_ts[*cur_count],
+			     sizeof(qdf_time_t));
+	}
+}
+
+static void
+wma_wow_wakeup_pagefault_notify(tp_wma_handle wma, void *ev, uint32_t ev_len)
+{
+	QDF_STATUS status;
+	uint32_t pf_wakeup_intv;
+	qdf_time_t cur_time, cutoff_time;
+	struct wma_pf_sym *pf_sym_entry;
+	struct wma_pf_sym_hist *pf_sym_hist = &wma->wma_pf_hist;
+	struct wlan_objmgr_psoc *psoc = wma->psoc;
+	uint8_t pf_tbl_idx;
+	struct wow_pf_wakeup_ev_data pf_sym_list = {0};
+	struct mac_context *mac = cds_get_context(QDF_MODULE_ID_PE);
+
+	if (!mac) {
+		wma_debug("MAC context NULL");
 		return;
 	}
 
-	if (!wlan_pmo_enable_ssr_on_page_fault(wma->psoc))
+	if (wlan_pmo_no_op_on_page_fault(psoc))
 		return;
 
 	if (wmi_get_runtime_pm_inprogress(wma->wmi_handle)) {
@@ -3270,50 +3588,55 @@ wma_wow_wakeup_host_trigger_ssr(t_wma_handle *wma, uint32_t reason)
 		return;
 	}
 
-	pagefault_wakeups_for_ssr =
-			wlan_pmo_get_max_pagefault_wakeups_for_ssr(wma->psoc);
+	if (!mac->sme.pagefault_action_cb) {
+		wma_debug("NULL pagefault action cb");
+		return;
+	}
 
-	interval_for_pagefault_wakeup_counts =
-		wlan_pmo_get_interval_for_pagefault_wakeup_counts(wma->psoc);
+	if (wlan_pmo_enable_ssr_on_page_fault(psoc)) {
+		wma_wow_pagefault_handle_ssr_action(wma);
+		return;
+	}
 
-	curr_time = qdf_get_time_of_the_day_ms();
+	cur_time = qdf_get_system_uptime();
+	pf_wakeup_intv =
+		wlan_pmo_get_interval_for_pagefault_wakeup_counts(psoc);
+	cutoff_time = cur_time - qdf_system_msecs_to_ticks(pf_wakeup_intv);
 
-	for (i = wma->num_page_fault_wakeups - 1; i >= 0; i--) {
-		if (curr_time - wma->pagefault_wakeups_ts[i] >
-		    interval_for_pagefault_wakeup_counts) {
-			if (i == wma->num_page_fault_wakeups - 1) {
-				wma->num_page_fault_wakeups = 0;
-			} else {
-				qdf_mem_copy(&wma->pagefault_wakeups_ts[0],
-					&wma->pagefault_wakeups_ts[i+1],
-					(wma->num_page_fault_wakeups - (i+1)) *
-					sizeof(qdf_time_t));
-				wma->num_page_fault_wakeups -= (i + 1);
-			}
-			ignore_pf = false;
-			break;
-		}
+	status = wma_wow_pagefault_parse_event(psoc, ev, ev_len, &pf_sym_list);
+	if (QDF_IS_STATUS_ERROR(status)) {
+		wma_debug("Failed during page fault payload parse");
+		return;
 	}
 
-	if (wma->num_page_fault_wakeups == pagefault_wakeups_for_ssr) {
-		qdf_mem_copy(&wma->pagefault_wakeups_ts[0],
-			     &wma->pagefault_wakeups_ts[1],
-			     (pagefault_wakeups_for_ssr - 1) *
-			     sizeof(qdf_time_t));
-		wma->num_page_fault_wakeups--;
+	qdf_spinlock_acquire(&pf_sym_hist->lock);
+	for (pf_tbl_idx = 0; pf_tbl_idx < WLAN_WMA_MAX_PF_SYM; pf_tbl_idx++) {
+		pf_sym_entry = &pf_sym_hist->wma_pf_sym[pf_tbl_idx];
+		if (cutoff_time < cur_time) {
+			wma_wow_pagefault_flush_ageout_entries(pf_sym_entry,
+							       cutoff_time);
+		}
+
+		wma_wow_pagefault_process_existing_syms(wma, pf_sym_entry,
+							&pf_sym_list, cur_time);
+
+		if (!pf_sym_list.pending_pf_syms)
+			goto send_event;
 	}
 
-	wma->pagefault_wakeups_ts[wma->num_page_fault_wakeups++] = curr_time;
+	/* Process if any new symbols are present in the event */
+	wma_wow_pagefault_add_new_sym_from_event(wma, &pf_sym_list, cur_time);
 
-	wma_nofl_debug("num pagefault wakeups %d", wma->num_page_fault_wakeups);
+send_event:
+	if (pf_sym_hist->pf_notify_buf_len) {
+		wma_wow_pagefault_action_cb(pf_sym_hist->pf_notify_buf_ptr);
+		pf_sym_hist->pf_notify_buf_len = 0;
+	}
 
-	if (!ignore_pf ||
-	    (wma->num_page_fault_wakeups < pagefault_wakeups_for_ssr))
-		return;
+	qdf_spinlock_release(&pf_sym_hist->lock);
 
-	if (curr_time - wma->pagefault_wakeups_ts[0] <=
-					interval_for_pagefault_wakeup_counts)
-		mac->sme.ssr_on_pagefault_cb();
+	qdf_mem_free(pf_sym_list.pf_sym);
+	pf_sym_list.pf_sym = NULL;
 }
 
 /**
@@ -3348,7 +3671,8 @@ int wma_wow_wakeup_host_event(void *handle, uint8_t *event, uint32_t len)
 	}
 
 	wma_wake_event_log_reason(wma, wake_info);
-	wma_wow_wakeup_host_trigger_ssr(wma, wake_info->wake_reason);
+	if (wake_info->wake_reason == WOW_REASON_PAGE_FAULT)
+		wma_wow_wakeup_pagefault_notify(wma, event, len);
 
 	if (wake_info->wake_reason == WOW_REASON_LOCAL_DATA_UC_DROP)
 		hif_rtpm_set_autosuspend_delay(WOW_LARGE_RX_RTPM_DELAY);

+ 78 - 9
core/wma/src/wma_main.c

@@ -1,6 +1,6 @@
 /*
  * Copyright (c) 2013-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -3426,6 +3426,78 @@ wma_set_exclude_selftx_from_cca_busy_time(bool exclude_selftx_from_cca_busy,
 	cfg->exclude_selftx_from_cca_busy = exclude_selftx_from_cca_busy;
 }
 
+static void wma_deinit_pagefault_wakeup_history(tp_wma_handle wma)
+{
+	struct wma_pf_sym *pf_sym_entry;
+	int8_t idx, max_sym_count = WLAN_WMA_MAX_PF_SYM;
+	bool is_ssr = false;
+
+	if (wlan_pmo_enable_ssr_on_page_fault(wma->psoc)) {
+		is_ssr = true;
+		max_sym_count = 0x1;
+	}
+
+	for (idx = 0; idx < max_sym_count; idx++) {
+		pf_sym_entry = &wma->wma_pf_hist.wma_pf_sym[idx];
+		pf_sym_entry->pf_sym.symbol = 0x0;
+		pf_sym_entry->pf_sym.count = 0x0;
+		qdf_mem_free(pf_sym_entry->pf_ev_ts);
+		pf_sym_entry->pf_ev_ts = NULL;
+	}
+
+	if (!is_ssr) {
+		qdf_mem_free(wma->wma_pf_hist.pf_notify_buf_ptr);
+		wma->wma_pf_hist.pf_notify_buf_ptr = NULL;
+		wma->wma_pf_hist.pf_notify_buf_len = 0x0;
+	}
+	qdf_spinlock_destroy(&wma->wma_pf_hist.lock);
+}
+
+static QDF_STATUS wma_init_pagefault_wakeup_history(tp_wma_handle wma)
+{
+	struct wma_pf_sym *pf_sym_entry;
+	int8_t idx, idx2, max_sym_count = WLAN_WMA_MAX_PF_SYM;
+	uint8_t max_pf_count;
+	bool is_ssr = false;
+
+	if (wlan_pmo_enable_ssr_on_page_fault(wma->psoc)) {
+		is_ssr = true;
+		max_sym_count = 0x1;
+	}
+
+	max_pf_count = wlan_pmo_get_min_pagefault_wakeups_for_action(wma->psoc);
+	for (idx = 0; idx < max_sym_count; idx++) {
+		pf_sym_entry = &wma->wma_pf_hist.wma_pf_sym[idx];
+		pf_sym_entry->pf_sym.symbol = 0x0;
+		pf_sym_entry->pf_sym.count = 0x0;
+		pf_sym_entry->pf_ev_ts = qdf_mem_malloc(max_pf_count *
+							sizeof(qdf_time_t));
+		if (!pf_sym_entry->pf_ev_ts)
+			goto mem_err;
+	}
+
+	if (!is_ssr) {
+		wma->wma_pf_hist.pf_notify_buf_len = 0x0;
+		wma->wma_pf_hist.pf_notify_buf_ptr =
+				qdf_mem_malloc(WLAN_WMA_PF_APPS_NOTIFY_BUF_LEN);
+		if (!wma->wma_pf_hist.pf_notify_buf_ptr)
+			goto mem_err;
+	}
+
+	qdf_spinlock_create(&wma->wma_pf_hist.lock);
+
+	return QDF_STATUS_SUCCESS;
+
+mem_err:
+	for (idx2 = --idx; idx2 >= 0; idx2--) {
+		pf_sym_entry = &wma->wma_pf_hist.wma_pf_sym[idx2];
+		qdf_mem_free(pf_sym_entry->pf_ev_ts);
+		pf_sym_entry->pf_ev_ts = NULL;
+	}
+
+	return QDF_STATUS_E_NOMEM;
+}
+
 /**
  * wma_open() - Allocate wma context and initialize it.
  * @psoc: psoc object
@@ -3531,12 +3603,9 @@ QDF_STATUS wma_open(struct wlan_objmgr_psoc *psoc,
 	}
 	wma_handle->psoc = psoc;
 
-	if (wlan_pmo_enable_ssr_on_page_fault(psoc)) {
-		wma_handle->pagefault_wakeups_ts =
-			qdf_mem_malloc(
-			wlan_pmo_get_max_pagefault_wakeups_for_ssr(psoc) *
-			sizeof(qdf_time_t));
-		if (!wma_handle->pagefault_wakeups_ts)
+	if (!wlan_pmo_no_op_on_page_fault(psoc)) {
+		qdf_status = wma_init_pagefault_wakeup_history(wma_handle);
+		if (QDF_IS_STATUS_ERROR(qdf_status))
 			goto err_wma_handle;
 	}
 
@@ -4949,8 +5018,8 @@ QDF_STATUS wma_close(void)
 	if (wmi_validate_handle(wmi_handle))
 		return QDF_STATUS_E_INVAL;
 
-	if (wlan_pmo_enable_ssr_on_page_fault(wma_handle->psoc))
-		qdf_mem_free(wma_handle->pagefault_wakeups_ts);
+	if (!wlan_pmo_no_op_on_page_fault(wma_handle->psoc))
+		wma_deinit_pagefault_wakeup_history(wma_handle);
 
 	qdf_atomic_set(&wma_handle->sap_num_clients_connected, 0);
 	qdf_atomic_set(&wma_handle->go_num_clients_connected, 0);