Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 | // SPDX-License-Identifier: GPL-2.0-only /* * Processor thermal device module for registering and processing * power floor. When the hardware reduces the power to the minimum * possible, the power floor is notified via an interrupt. * * Operation: * When user space enables power floor reporting: * - Use mailbox to: * Enable processor thermal device interrupt * * - Current status of power floor is read from offset 0x5B18 * bit 39. * * Two interface functions are provided to call when there is a * thermal device interrupt: * - proc_thermal_power_floor_intr(): * Check if the interrupt is for change in power floor. * Called from interrupt context. * * - proc_thermal_power_floor_intr_callback(): * Callback for interrupt processing in thread context. This involves * sending notification to user space that there is a change in the * power floor status. * * Copyright (c) 2023, Intel Corporation. */ #include <linux/pci.h> #include "processor_thermal_device.h" #define SOC_POWER_FLOOR_STATUS BIT(39) #define SOC_POWER_FLOOR_SHIFT 39 #define SOC_POWER_FLOOR_INT_ENABLE_BIT 31 #define SOC_POWER_FLOOR_INT_ACTIVE BIT(3) int proc_thermal_read_power_floor_status(struct proc_thermal_device *proc_priv) { u64 status = 0; status = readq(proc_priv->mmio_base + SOC_WT_RES_INT_STATUS_OFFSET); return (status & SOC_POWER_FLOOR_STATUS) >> SOC_POWER_FLOOR_SHIFT; } EXPORT_SYMBOL_NS_GPL(proc_thermal_read_power_floor_status, INT340X_THERMAL); static bool enable_state; static DEFINE_MUTEX(pf_lock); int proc_thermal_power_floor_set_state(struct proc_thermal_device *proc_priv, bool enable) { int ret = 0; mutex_lock(&pf_lock); if (enable_state == enable) goto pf_unlock; /* * Time window parameter is not applicable to power floor interrupt configuration. * Hence use -1 for time window. */ ret = processor_thermal_mbox_interrupt_config(to_pci_dev(proc_priv->dev), enable, SOC_POWER_FLOOR_INT_ENABLE_BIT, -1); if (!ret) enable_state = enable; pf_unlock: mutex_unlock(&pf_lock); return ret; } EXPORT_SYMBOL_NS_GPL(proc_thermal_power_floor_set_state, INT340X_THERMAL); bool proc_thermal_power_floor_get_state(struct proc_thermal_device *proc_priv) { return enable_state; } EXPORT_SYMBOL_NS_GPL(proc_thermal_power_floor_get_state, INT340X_THERMAL); /** * proc_thermal_check_power_floor_intr() - Check power floor interrupt. * @proc_priv: Processor thermal device instance. * * Callback to check if the interrupt for power floor is active. * * Context: Called from interrupt context. * * Return: true if power floor is active, false when not active. */ bool proc_thermal_check_power_floor_intr(struct proc_thermal_device *proc_priv) { u64 int_status; int_status = readq(proc_priv->mmio_base + SOC_WT_RES_INT_STATUS_OFFSET); return !!(int_status & SOC_POWER_FLOOR_INT_ACTIVE); } EXPORT_SYMBOL_NS_GPL(proc_thermal_check_power_floor_intr, INT340X_THERMAL); /** * proc_thermal_power_floor_intr_callback() - Process power floor notification * @pdev: PCI device instance * @proc_priv: Processor thermal device instance. * * Check if the power floor interrupt is active, if active send notification to * user space for the attribute "power_limits", so that user can read the attribute * and take action. * * Context: Called from interrupt thread context. * * Return: None. */ void proc_thermal_power_floor_intr_callback(struct pci_dev *pdev, struct proc_thermal_device *proc_priv) { u64 status; status = readq(proc_priv->mmio_base + SOC_WT_RES_INT_STATUS_OFFSET); if (!(status & SOC_POWER_FLOOR_INT_ACTIVE)) return; sysfs_notify(&pdev->dev.kobj, "power_limits", "power_floor_status"); } EXPORT_SYMBOL_NS_GPL(proc_thermal_power_floor_intr_callback, INT340X_THERMAL); MODULE_IMPORT_NS(INT340X_THERMAL); MODULE_LICENSE("GPL"); |