Hi, On 8/2/22 15:24, Hans de Goede wrote: > Hi, > > On 8/2/22 13:25, Shyam Sundar S K wrote: >> The transition to auto-mode happens when the PMF driver receives >> AMT (Auto Mode transition) event. transition logic will reside in the >> PMF driver but the events would come from other supported drivers[1]. >> >> The thermal parameters would vary between when a performance "on-lap" mode >> is detected and versus when not. The CQL event would get triggered from >> other drivers, so that PMF driver would adjust the system thermal config >> based on the ACPI inputs. >> >> OEMs can control whether or not to enable AMT or CQL via other supported >> drivers[1] but the actual transition logic resides in the AMD PMF driver. >> When an AMT event is received the automatic mode transition RAPL algorithm >> will run. When a CQL event is received an performance "on-lap" mode will >> be enabled and thermal parameters will be adjusted accordingly. >> >> [1] >> Link: https://git.kernel.org/pub/scm/linux/kernel/git/pdx86/platform-drivers-x86.git/commit/?h=review-hans&id=755b249250df1b612d982f3b702c831b26ecdf73 >> >> Cc: Mario Limonciello <mario.limonciello@xxxxxxx> >> Cc: Mark Pearson <markpearson@xxxxxxxxxx> >> Signed-off-by: Shyam Sundar S K <Shyam-sundar.S-k@xxxxxxx> >> --- >> drivers/platform/x86/amd/pmf/acpi.c | 64 ++++++++++++++++++++++++ >> drivers/platform/x86/amd/pmf/auto-mode.c | 38 ++++++++++++++ >> drivers/platform/x86/amd/pmf/core.c | 9 ++++ >> drivers/platform/x86/amd/pmf/pmf.h | 22 ++++++++ >> 4 files changed, 133 insertions(+) >> >> diff --git a/drivers/platform/x86/amd/pmf/acpi.c b/drivers/platform/x86/amd/pmf/acpi.c >> index 200e31033d94..70084807a2e7 100644 >> --- a/drivers/platform/x86/amd/pmf/acpi.c >> +++ b/drivers/platform/x86/amd/pmf/acpi.c >> @@ -11,6 +11,9 @@ >> #include <linux/acpi.h> >> #include "pmf.h" >> >> +#define APMF_CQL_NOTIFICATION 2 >> +#define APMF_AMT_NOTIFICATION 3 >> + >> static union acpi_object *apmf_if_call(struct amd_pmf_dev *pdev, int fn, struct acpi_buffer *param) >> { >> struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; >> @@ -154,6 +157,47 @@ int apmf_get_auto_mode_def(struct amd_pmf_dev *pdev, struct apmf_auto_mode *data >> return apmf_if_call_store_buffer(pdev, APMF_FUNC_AUTO_MODE, data, sizeof(*data)); >> } >> >> +int apmf_get_sbios_requests(struct amd_pmf_dev *pdev, struct apmf_sbios_req *req) >> +{ >> + return apmf_if_call_store_buffer(pdev, APMF_FUNC_SBIOS_REQUESTS, >> + req, sizeof(*req)); >> +} >> + >> +static void apmf_event_handler(acpi_handle handle, u32 event, void *data) >> +{ >> + struct amd_pmf_dev *pmf_dev = data; >> + struct apmf_sbios_req req; >> + int ret; >> + >> + mutex_lock(&pmf_dev->update_mutex); >> + ret = apmf_get_sbios_requests(pmf_dev, &req); >> + if (ret) { >> + dev_err(pmf_dev->dev, "Failed to get SBIOS requests:%d\n", ret); >> + return; >> + } >> + >> + if (req.pending_req & BIT(APMF_AMT_NOTIFICATION)) { >> + dev_dbg(pmf_dev->dev, "AMT is supported and notifications %s\n", >> + req.amt_event ? "Enabled" : "Disabled"); >> + pmf_dev->amt_enabled = !!req.amt_event; >> + } >> + >> + if (pmf_dev->amt_enabled) >> + amd_pmf_handle_amt(pmf_dev); >> + else >> + amd_pmf_reset_amt(pmf_dev); > > AFAICT this should be inside the: > > if (req.pending_req & BIT(APMF_AMT_NOTIFICATION)) { > > block, pmf_dev->amt_enabled only is updated in that block and > re-doing the amd_pmf_handle_amt() / reset_amt() calls when > pmf_dev->amt_enabled is not changed is not necessary. > >> + >> + if (req.pending_req & BIT(APMF_CQL_NOTIFICATION)) { >> + dev_dbg(pmf_dev->dev, "CQL is supported and notifications %s\n", >> + req.cql_event ? "Enabled" : "Disabled"); >> + >> + /* update the target mode information */ >> + if (pmf_dev->amt_enabled) >> + amd_pmf_update_2_cql(pmf_dev, req.cql_event); >> + } >> + mutex_unlock(&pmf_dev->update_mutex); >> +} >> + >> static int apmf_if_verify_interface(struct amd_pmf_dev *pdev) >> { >> struct apmf_verify_interface output; >> @@ -195,12 +239,20 @@ static int apmf_get_system_params(struct amd_pmf_dev *dev) >> >> void apmf_acpi_deinit(struct amd_pmf_dev *pmf_dev) >> { >> + acpi_handle ahandle = ACPI_HANDLE(pmf_dev->dev); >> + >> if (pmf_dev->hb_interval) >> cancel_delayed_work_sync(&pmf_dev->heart_beat); >> + >> + if (is_apmf_func_supported(pmf_dev, APMF_FUNC_AUTO_MODE) && >> + is_apmf_func_supported(pmf_dev, APMF_FUNC_SBIOS_REQUESTS)) >> + acpi_remove_notify_handler(ahandle, ACPI_ALL_NOTIFY, apmf_event_handler); >> } >> >> int apmf_acpi_init(struct amd_pmf_dev *pmf_dev) >> { >> + acpi_handle ahandle = ACPI_HANDLE(pmf_dev->dev); >> + acpi_status status; >> int ret; >> >> ret = apmf_if_verify_interface(pmf_dev); >> @@ -221,6 +273,18 @@ int apmf_acpi_init(struct amd_pmf_dev *pmf_dev) >> schedule_delayed_work(&pmf_dev->heart_beat, 0); >> } >> >> + /* Install the APMF Notify handler */ >> + if (is_apmf_func_supported(pmf_dev, APMF_FUNC_AUTO_MODE) && >> + is_apmf_func_supported(pmf_dev, APMF_FUNC_SBIOS_REQUESTS)) { >> + status = acpi_install_notify_handler(ahandle, >> + ACPI_ALL_NOTIFY, >> + apmf_event_handler, pmf_dev); >> + if (ACPI_FAILURE(status)) { >> + dev_err(pmf_dev->dev, "failed to install notify handler\n"); >> + return -ENODEV; >> + } As mentioned in my latest "Re: [PATCH v2 RESEND 09/11] platform/x86/amd/pmf: Handle AMT and CQL events for Auto mode" email I think you should call apmf_event_handler() manually here once to catch up with possibly missed notifies. >> + } >> + >> out: >> return ret; >> } >> diff --git a/drivers/platform/x86/amd/pmf/auto-mode.c b/drivers/platform/x86/amd/pmf/auto-mode.c >> index 99f5a2396b0b..4e4ec6023525 100644 >> --- a/drivers/platform/x86/amd/pmf/auto-mode.c >> +++ b/drivers/platform/x86/amd/pmf/auto-mode.c >> @@ -108,6 +108,21 @@ void amd_pmf_trans_automode(struct amd_pmf_dev *dev, int socket_power, ktime_t t >> } >> } >> >> +void amd_pmf_update_2_cql(struct amd_pmf_dev *dev, bool is_cql_event) >> +{ >> + int mode = config_store.current_mode; >> + >> + config_store.transition[AUTO_TRANSITION_TO_PERFORMANCE].target_mode = >> + is_cql_event ? AUTO_PERFORMANCE_ON_LAP : AUTO_PERFORMANCE; >> + >> + if ((mode == AUTO_PERFORMANCE || mode == AUTO_PERFORMANCE_ON_LAP) && >> + mode != config_store.transition[AUTO_TRANSITION_TO_PERFORMANCE].target_mode) { >> + mode = config_store.transition[AUTO_TRANSITION_TO_PERFORMANCE].target_mode; >> + amd_pmf_set_automode(dev, mode, NULL); >> + } >> + dev_dbg(dev->dev, "updated CQL thermals\n"); >> +} >> + >> static void amd_pmf_get_power_threshold(void) >> { >> config_store.transition[AUTO_TRANSITION_TO_QUIET].power_threshold = >> @@ -249,6 +264,29 @@ void amd_pmf_load_defaults_auto_mode(struct amd_pmf_dev *dev) >> dev->socket_power_history_idx = -1; >> } >> >> +void amd_pmf_reset_amt(struct amd_pmf_dev *dev) >> +{ >> + /* >> + * OEM BIOS implementation guide says that if the auto mode is enabled >> + * the platform_profile registration shall be done by the OEM driver. >> + * There could be cases where both static slider and auto mode BIOS >> + * functions are enabled, in that case enable static slider updates >> + * only if it advertised as supported. >> + */ >> + >> + if (is_apmf_func_supported(dev, APMF_FUNC_STATIC_SLIDER_GRANULAR)) { >> + u8 mode = amd_pmf_get_pprof_modes(dev); >> + >> + dev_dbg(dev->dev, "resetting AMT thermals\n"); >> + amd_pmf_update_slider(dev, SLIDER_OP_SET, mode, NULL); >> + } >> +} >> + >> +void amd_pmf_handle_amt(struct amd_pmf_dev *dev) >> +{ >> + amd_pmf_set_automode(dev, config_store.current_mode, NULL); >> +} >> + >> void amd_pmf_deinit_auto_mode(struct amd_pmf_dev *dev) >> { >> cancel_delayed_work_sync(&dev->work_buffer); >> diff --git a/drivers/platform/x86/amd/pmf/core.c b/drivers/platform/x86/amd/pmf/core.c >> index 42d803b49d97..4467d682cd11 100644 >> --- a/drivers/platform/x86/amd/pmf/core.c >> +++ b/drivers/platform/x86/amd/pmf/core.c >> @@ -99,6 +99,7 @@ static void amd_pmf_get_metrics(struct work_struct *work) >> ktime_t time_elapsed_ms; >> int socket_power; >> >> + mutex_lock(&dev->update_mutex); >> /* Transfer table contents */ >> memset(dev->buf, 0, sizeof(dev->m_table)); >> amd_pmf_send_cmd(dev, SET_TRANSFER_TABLE, 0, 7, NULL); >> @@ -108,8 +109,14 @@ static void amd_pmf_get_metrics(struct work_struct *work) >> /* Calculate the avg SoC power consumption */ >> socket_power = dev->m_table.apu_power + dev->m_table.dgpu_power; >> >> + if (dev->amt_enabled) { >> + /* Apply the Auto Mode transition */ >> + amd_pmf_trans_automode(dev, socket_power, time_elapsed_ms); >> + } >> + >> dev->start_time = ktime_to_ms(ktime_get()); >> schedule_delayed_work(&dev->work_buffer, msecs_to_jiffies(metrics_table_loop_ms)); >> + mutex_unlock(&dev->update_mutex); >> } >> >> static inline u32 amd_pmf_reg_read(struct amd_pmf_dev *dev, int reg_offset) >> @@ -329,6 +336,7 @@ static int amd_pmf_probe(struct platform_device *pdev) >> amd_pmf_dbgfs_register(dev); >> >> mutex_init(&dev->lock); >> + mutex_init(&dev->update_mutex); >> dev_info(dev->dev, "registered PMF device successfully\n"); >> >> return 0; >> @@ -339,6 +347,7 @@ static int amd_pmf_remove(struct platform_device *pdev) >> struct amd_pmf_dev *dev = platform_get_drvdata(pdev); >> >> mutex_destroy(&dev->lock); >> + mutex_destroy(&dev->update_mutex); >> amd_pmf_deinit_features(dev); >> apmf_acpi_deinit(dev); >> amd_pmf_dbgfs_unregister(dev); >> diff --git a/drivers/platform/x86/amd/pmf/pmf.h b/drivers/platform/x86/amd/pmf/pmf.h >> index 8c92cd6871df..cc88a02b488d 100644 >> --- a/drivers/platform/x86/amd/pmf/pmf.h >> +++ b/drivers/platform/x86/amd/pmf/pmf.h >> @@ -17,6 +17,7 @@ >> /* APMF Functions */ >> #define APMF_FUNC_VERIFY_INTERFACE 0 >> #define APMF_FUNC_GET_SYS_PARAMS 1 >> +#define APMF_FUNC_SBIOS_REQUESTS 2 >> #define APMF_FUNC_SBIOS_HEARTBEAT 4 >> #define APMF_FUNC_AUTO_MODE 5 >> #define APMF_FUNC_SET_FAN_IDX 7 >> @@ -63,6 +64,21 @@ struct apmf_system_params { >> u32 heartbeat_int; >> } __packed; >> >> +struct apmf_sbios_req { >> + u16 size; >> + u32 pending_req; >> + u8 rsd; >> + u8 cql_event; >> + u8 amt_event; >> + u32 fppt; >> + u32 sppt; >> + u32 fppt_apu_only; >> + u32 spl; >> + u32 stt_min_limit; >> + u8 skin_temp_apu; >> + u8 skin_temp_hs2; >> +} __packed; >> + >> struct apmf_fan_idx { >> u16 size; >> u8 fan_ctl_mode; >> @@ -147,6 +163,8 @@ struct amd_pmf_dev { >> ktime_t start_time; >> int socket_power_history[AVG_SAMPLE_SIZE]; >> int socket_power_history_idx; >> + bool amt_enabled; >> + struct mutex update_mutex; /* protects race between ACPI handler and metrics thread */ >> }; >> >> struct apmf_sps_prop_granular { >> @@ -307,5 +325,9 @@ int apmf_get_auto_mode_def(struct amd_pmf_dev *pdev, struct apmf_auto_mode *data >> void amd_pmf_init_auto_mode(struct amd_pmf_dev *dev); >> void amd_pmf_deinit_auto_mode(struct amd_pmf_dev *dev); >> void amd_pmf_trans_automode(struct amd_pmf_dev *dev, int socket_power, ktime_t time_elapsed_ms); >> +int apmf_get_sbios_requests(struct amd_pmf_dev *pdev, struct apmf_sbios_req *req); >> >> +void amd_pmf_update_2_cql(struct amd_pmf_dev *dev, bool is_cql_event); >> +void amd_pmf_reset_amt(struct amd_pmf_dev *dev); >> +void amd_pmf_handle_amt(struct amd_pmf_dev *dev); >> #endif /* PMF_H */ > > Except for the 1 remark this looks good to me. Make that 2 remarks... Regards, Hans