|
@@ -35,6 +35,9 @@
|
|
|
#include "wlan_hdd_cfg80211.h"
|
|
|
#include <qca_vendor.h>
|
|
|
#include "wlan_fwol_ucfg_api.h"
|
|
|
+#include <pld_common.h>
|
|
|
+
|
|
|
+#define DC_OFF_PERCENT_WPPS 50
|
|
|
|
|
|
const struct nla_policy
|
|
|
wlan_hdd_thermal_mitigation_policy
|
|
@@ -44,6 +47,102 @@ const struct nla_policy
|
|
|
.type = NLA_U32},
|
|
|
};
|
|
|
|
|
|
+#ifdef FEATURE_WPSS_THERMAL_MITIGATION
|
|
|
+void
|
|
|
+hdd_thermal_fill_clientid_priority(uint8_t mon_id, uint8_t priority_apps,
|
|
|
+ uint8_t priority_wpps,
|
|
|
+ struct thermal_mitigation_params *params)
|
|
|
+{
|
|
|
+ if (mon_id == THERMAL_MONITOR_APPS) {
|
|
|
+ params->priority = priority_apps;
|
|
|
+ params->client_id = mon_id;
|
|
|
+ hdd_debug("Thermal client:%d priority_apps: %d", mon_id,
|
|
|
+ priority_apps);
|
|
|
+ } else if (mon_id == THERMAL_MONITOR_WPSS) {
|
|
|
+ params->priority = priority_wpps;
|
|
|
+ params->client_id = mon_id;
|
|
|
+ /* currently hardcoded, can be changed based on requirement */
|
|
|
+ params->levelconf[0].dcoffpercent = DC_OFF_PERCENT_WPPS;
|
|
|
+ hdd_debug("Thermal client:%d priority_wpps: %d", mon_id,
|
|
|
+ priority_wpps);
|
|
|
+ }
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+static QDF_STATUS
|
|
|
+hdd_send_thermal_mitigation_val(struct hdd_context *hdd_ctx, uint32_t level,
|
|
|
+ uint8_t mon_id)
|
|
|
+{
|
|
|
+ uint32_t dc, dc_off_percent;
|
|
|
+ uint32_t prio = 0, target_temp = 0;
|
|
|
+ struct wlan_fwol_thermal_temp thermal_temp = {0};
|
|
|
+ QDF_STATUS status;
|
|
|
+ bool enable = true;
|
|
|
+ struct thermal_mitigation_params therm_cfg_params = {0};
|
|
|
+
|
|
|
+ status = ucfg_fwol_get_thermal_temp(hdd_ctx->psoc, &thermal_temp);
|
|
|
+ if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
+ hdd_err_rl("Failed to get fwol thermal obj");
|
|
|
+ return status;
|
|
|
+ }
|
|
|
+
|
|
|
+ switch (level) {
|
|
|
+ case QCA_WLAN_VENDOR_THERMAL_LEVEL_EMERGENCY:
|
|
|
+ dc_off_percent = thermal_temp.throttle_dutycycle_level[5];
|
|
|
+ break;
|
|
|
+ case QCA_WLAN_VENDOR_THERMAL_LEVEL_CRITICAL:
|
|
|
+ dc_off_percent = thermal_temp.throttle_dutycycle_level[4];
|
|
|
+ break;
|
|
|
+ case QCA_WLAN_VENDOR_THERMAL_LEVEL_SEVERE:
|
|
|
+ dc_off_percent = thermal_temp.throttle_dutycycle_level[3];
|
|
|
+ break;
|
|
|
+ case QCA_WLAN_VENDOR_THERMAL_LEVEL_MODERATE:
|
|
|
+ dc_off_percent = thermal_temp.throttle_dutycycle_level[2];
|
|
|
+ break;
|
|
|
+ case QCA_WLAN_VENDOR_THERMAL_LEVEL_LIGHT:
|
|
|
+ dc_off_percent = thermal_temp.throttle_dutycycle_level[1];
|
|
|
+ break;
|
|
|
+ case QCA_WLAN_VENDOR_THERMAL_LEVEL_NONE:
|
|
|
+ enable = false;
|
|
|
+ dc_off_percent = thermal_temp.throttle_dutycycle_level[0];
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ hdd_debug("Invalid thermal state");
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ dc = thermal_temp.thermal_sampling_time;
|
|
|
+ therm_cfg_params.enable = enable;
|
|
|
+ therm_cfg_params.dc = dc;
|
|
|
+ therm_cfg_params.levelconf[0].dcoffpercent = dc_off_percent;
|
|
|
+ therm_cfg_params.levelconf[0].priority = prio;
|
|
|
+ therm_cfg_params.levelconf[0].tmplwm = target_temp;
|
|
|
+ therm_cfg_params.num_thermal_conf = 1;
|
|
|
+ therm_cfg_params.pdev_id = 0;
|
|
|
+
|
|
|
+ hdd_thermal_fill_clientid_priority(mon_id, thermal_temp.priority_apps,
|
|
|
+ thermal_temp.priority_wpps,
|
|
|
+ &therm_cfg_params);
|
|
|
+
|
|
|
+ hdd_debug("dc %d dc_off_per %d", dc, dc_off_percent);
|
|
|
+
|
|
|
+ status = sme_set_thermal_throttle_cfg(hdd_ctx->mac_handle,
|
|
|
+ &therm_cfg_params);
|
|
|
+ if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ hdd_err_rl("Failed to set throttle configuration %d", status);
|
|
|
+
|
|
|
+ else
|
|
|
+ /*
|
|
|
+ * After SSR, the thermal mitigation level is lost.
|
|
|
+ * As SSR is hidden from userland, this command will not come
|
|
|
+ * from userspace after a SSR. To restore this configuration,
|
|
|
+ * save this in hdd context and restore after re-init.
|
|
|
+ */
|
|
|
+ hdd_ctx->dutycycle_off_percent = dc_off_percent;
|
|
|
+
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+
|
|
|
/**
|
|
|
* __wlan_hdd_cfg80211_set_thermal_mitigation_policy() - Set the thermal policy
|
|
|
* @wiphy: Pointer to wireless phy
|
|
@@ -61,10 +160,7 @@ __wlan_hdd_cfg80211_set_thermal_mitigation_policy(struct wiphy *wiphy,
|
|
|
{
|
|
|
struct hdd_context *hdd_ctx = wiphy_priv(wiphy);
|
|
|
struct nlattr *tb[QCA_WLAN_VENDOR_ATTR_THERMAL_CMD_MAX + 1];
|
|
|
- bool enable = true;
|
|
|
- uint32_t dc, dc_off_percent, level, cmd_type;
|
|
|
- uint32_t prio = 0, target_temp = 0;
|
|
|
- struct wlan_fwol_thermal_temp thermal_temp = {0};
|
|
|
+ uint32_t level, cmd_type;
|
|
|
QDF_STATUS status;
|
|
|
|
|
|
hdd_enter();
|
|
@@ -100,60 +196,10 @@ __wlan_hdd_cfg80211_set_thermal_mitigation_policy(struct wiphy *wiphy,
|
|
|
level =
|
|
|
nla_get_u32(tb[QCA_WLAN_VENDOR_ATTR_THERMAL_LEVEL]);
|
|
|
|
|
|
- hdd_debug("thermal mitigation level %d", level);
|
|
|
-
|
|
|
- status = ucfg_fwol_get_thermal_temp(hdd_ctx->psoc, &thermal_temp);
|
|
|
- if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
- hdd_err_rl("Failed to get fwol thermal obj");
|
|
|
- return qdf_status_to_os_return(status);
|
|
|
- }
|
|
|
-
|
|
|
- switch (level) {
|
|
|
- case QCA_WLAN_VENDOR_THERMAL_LEVEL_EMERGENCY:
|
|
|
- dc_off_percent = thermal_temp.throttle_dutycycle_level[5];
|
|
|
- break;
|
|
|
- case QCA_WLAN_VENDOR_THERMAL_LEVEL_CRITICAL:
|
|
|
- dc_off_percent = thermal_temp.throttle_dutycycle_level[4];
|
|
|
- break;
|
|
|
- case QCA_WLAN_VENDOR_THERMAL_LEVEL_SEVERE:
|
|
|
- dc_off_percent = thermal_temp.throttle_dutycycle_level[3];
|
|
|
- break;
|
|
|
- case QCA_WLAN_VENDOR_THERMAL_LEVEL_MODERATE:
|
|
|
- dc_off_percent = thermal_temp.throttle_dutycycle_level[2];
|
|
|
- break;
|
|
|
- case QCA_WLAN_VENDOR_THERMAL_LEVEL_LIGHT:
|
|
|
- dc_off_percent = thermal_temp.throttle_dutycycle_level[1];
|
|
|
- break;
|
|
|
- case QCA_WLAN_VENDOR_THERMAL_LEVEL_NONE:
|
|
|
- enable = false;
|
|
|
- dc_off_percent = thermal_temp.throttle_dutycycle_level[0];
|
|
|
- break;
|
|
|
- default:
|
|
|
- hdd_debug("Invalid thermal state");
|
|
|
- return -EINVAL;
|
|
|
- }
|
|
|
-
|
|
|
- dc = thermal_temp.thermal_sampling_time;
|
|
|
- hdd_debug("dc %d dc_off_per %d", dc, dc_off_percent);
|
|
|
-
|
|
|
- status = sme_set_thermal_throttle_cfg(hdd_ctx->mac_handle,
|
|
|
- enable,
|
|
|
- dc,
|
|
|
- dc_off_percent,
|
|
|
- prio,
|
|
|
- target_temp);
|
|
|
- if (QDF_IS_STATUS_ERROR(status))
|
|
|
- hdd_err_rl("Failed to set throttle configuration %d", status);
|
|
|
- else {
|
|
|
- /*
|
|
|
- * After SSR, the thermal mitigation level is lost.
|
|
|
- * As SSR is hidden from userland, this command will not come
|
|
|
- * from userspace after a SSR. To restore this configuration,
|
|
|
- * save this in hdd context and restore after re-init.
|
|
|
- */
|
|
|
- hdd_ctx->dutycycle_off_percent = dc_off_percent;
|
|
|
- }
|
|
|
-
|
|
|
+ hdd_debug("thermal mitigation level from userspace %d", level);
|
|
|
+ status = hdd_send_thermal_mitigation_val(hdd_ctx, level,
|
|
|
+ THERMAL_MONITOR_APPS);
|
|
|
+ hdd_exit();
|
|
|
return qdf_status_to_os_return(status);
|
|
|
}
|
|
|
|
|
@@ -200,6 +246,7 @@ QDF_STATUS hdd_restore_thermal_mitigation_config(struct hdd_context *hdd_ctx)
|
|
|
uint32_t prio = 0, target_temp = 0;
|
|
|
struct wlan_fwol_thermal_temp thermal_temp = {0};
|
|
|
QDF_STATUS status;
|
|
|
+ struct thermal_mitigation_params therm_cfg_params;
|
|
|
|
|
|
status = ucfg_fwol_get_thermal_temp(hdd_ctx->psoc, &thermal_temp);
|
|
|
if (QDF_IS_STATUS_ERROR(status)) {
|
|
@@ -213,17 +260,85 @@ QDF_STATUS hdd_restore_thermal_mitigation_config(struct hdd_context *hdd_ctx)
|
|
|
if (!dc_off_percent)
|
|
|
enable = false;
|
|
|
|
|
|
+ therm_cfg_params.enable = enable;
|
|
|
+ therm_cfg_params.dc = dc;
|
|
|
+ therm_cfg_params.levelconf[0].dcoffpercent = dc_off_percent;
|
|
|
+ therm_cfg_params.levelconf[0].priority = prio;
|
|
|
+ therm_cfg_params.levelconf[0].tmplwm = target_temp;
|
|
|
+ therm_cfg_params.num_thermal_conf = 1;
|
|
|
+ therm_cfg_params.client_id = THERMAL_MONITOR_APPS;
|
|
|
+ therm_cfg_params.priority = 0;
|
|
|
hdd_debug("dc %d dc_off_per %d enable %d", dc, dc_off_percent, enable);
|
|
|
|
|
|
status = sme_set_thermal_throttle_cfg(hdd_ctx->mac_handle,
|
|
|
- enable,
|
|
|
- dc,
|
|
|
- dc_off_percent,
|
|
|
- prio,
|
|
|
- target_temp);
|
|
|
+ &therm_cfg_params);
|
|
|
if (QDF_IS_STATUS_ERROR(status))
|
|
|
hdd_err_rl("Failed to set throttle configuration %d", status);
|
|
|
|
|
|
return status;
|
|
|
}
|
|
|
|
|
|
+int wlan_hdd_pld_set_thermal_mitigation(struct device *dev, unsigned long state,
|
|
|
+ int mon_id)
|
|
|
+{
|
|
|
+ struct hdd_context *hdd_ctx = cds_get_context(QDF_MODULE_ID_HDD);
|
|
|
+ QDF_STATUS status;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ hdd_enter();
|
|
|
+
|
|
|
+ ret = wlan_hdd_validate_context(hdd_ctx);
|
|
|
+ if (ret)
|
|
|
+ return ret;
|
|
|
+
|
|
|
+ if (hdd_ctx->driver_status == DRIVER_MODULES_CLOSED)
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ status = hdd_send_thermal_mitigation_val(hdd_ctx, state, mon_id);
|
|
|
+
|
|
|
+ hdd_exit();
|
|
|
+ return qdf_status_to_os_return(status);
|
|
|
+}
|
|
|
+
|
|
|
+#ifdef FEATURE_WPSS_THERMAL_MITIGATION
|
|
|
+inline void hdd_thermal_mitigation_register_wpps(struct hdd_context *hdd_ctx,
|
|
|
+ struct device *dev)
|
|
|
+{
|
|
|
+ if (hdd_ctx->multi_client_thermal_mitigation)
|
|
|
+ pld_thermal_register(dev, HDD_THERMAL_STATE_LIGHT,
|
|
|
+ THERMAL_MONITOR_WPSS);
|
|
|
+}
|
|
|
+
|
|
|
+inline void hdd_thermal_mitigation_unregister_wpps(struct hdd_context *hdd_ctx,
|
|
|
+ struct device *dev)
|
|
|
+{
|
|
|
+ if (hdd_ctx->multi_client_thermal_mitigation)
|
|
|
+ pld_thermal_unregister(dev, THERMAL_MONITOR_WPSS);
|
|
|
+}
|
|
|
+#else
|
|
|
+static inline
|
|
|
+void hdd_thermal_mitigation_register_wpps(struct hdd_context *hdd_ctx,
|
|
|
+ struct device *dev)
|
|
|
+{
|
|
|
+}
|
|
|
+
|
|
|
+static inline
|
|
|
+void hdd_thermal_mitigation_unregister_wpps(struct hdd_context *hdd_ctx,
|
|
|
+ struct device *dev)
|
|
|
+{
|
|
|
+}
|
|
|
+#endif
|
|
|
+void hdd_thermal_mitigation_register(struct hdd_context *hdd_ctx,
|
|
|
+ struct device *dev)
|
|
|
+{
|
|
|
+ pld_thermal_register(dev, HDD_THERMAL_STATE_EMERGENCY,
|
|
|
+ THERMAL_MONITOR_APPS);
|
|
|
+ hdd_thermal_mitigation_register_wpps(hdd_ctx, dev);
|
|
|
+}
|
|
|
+
|
|
|
+void hdd_thermal_mitigation_unregister(struct hdd_context *hdd_ctx,
|
|
|
+ struct device *dev)
|
|
|
+{
|
|
|
+ hdd_thermal_mitigation_unregister_wpps(hdd_ctx, dev);
|
|
|
+ pld_thermal_unregister(dev, THERMAL_MONITOR_APPS);
|
|
|
+}
|