mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-09-04 20:19:47 +08:00
Core
----
- Introduce a config option to tweak MAX_SKB_FRAGS. Increasing the
default value allows for better BIG TCP performances.
- Reduce compound page head access for zero-copy data transfers.
- RPS/RFS improvements, avoiding unneeded NET_RX_SOFTIRQ when possible.
- Threaded NAPI improvements, adding defer skb free support and unneeded
softirq avoidance.
- Address dst_entry reference count scalability issues, via false
sharing avoidance and optimize refcount tracking.
- Add lockless accesses annotation to sk_err[_soft].
- Optimize again the skb struct layout.
- Extends the skb drop reasons to make it usable by multiple
subsystems.
- Better const qualifier awareness for socket casts.
BPF
---
- Add skb and XDP typed dynptrs which allow BPF programs for more
ergonomic and less brittle iteration through data and variable-sized
accesses.
- Add a new BPF netfilter program type and minimal support to hook
BPF programs to netfilter hooks such as prerouting or forward.
- Add more precise memory usage reporting for all BPF map types.
- Adds support for using {FOU,GUE} encap with an ipip device operating
in collect_md mode and add a set of BPF kfuncs for controlling encap
params.
- Allow BPF programs to detect at load time whether a particular kfunc
exists or not, and also add support for this in light skeleton.
- Bigger batch of BPF verifier improvements to prepare for upcoming BPF
open-coded iterators allowing for less restrictive looping capabilities.
- Rework RCU enforcement in the verifier, add kptr_rcu and enforce BPF
programs to NULL-check before passing such pointers into kfunc.
- Add support for kptrs in percpu hashmaps, percpu LRU hashmaps and in
local storage maps.
- Enable RCU semantics for task BPF kptrs and allow referenced kptr
tasks to be stored in BPF maps.
- Add support for refcounted local kptrs to the verifier for allowing
shared ownership, useful for adding a node to both the BPF list and
rbtree.
- Add BPF verifier support for ST instructions in convert_ctx_access()
which will help new -mcpu=v4 clang flag to start emitting them.
- Add ARM32 USDT support to libbpf.
- Improve bpftool's visual program dump which produces the control
flow graph in a DOT format by adding C source inline annotations.
Protocols
---------
- IPv4: Allow adding to IPv4 address a 'protocol' tag. Such value
indicates the provenance of the IP address.
- IPv6: optimize route lookup, dropping unneeded R/W lock acquisition.
- Add the handshake upcall mechanism, allowing the user-space
to implement generic TLS handshake on kernel's behalf.
- Bridge: support per-{Port, VLAN} neighbor suppression, increasing
resilience to nodes failures.
- SCTP: add support for Fair Capacity and Weighted Fair Queueing
schedulers.
- MPTCP: delay first subflow allocation up to its first usage. This
will allow for later better LSM interaction.
- xfrm: Remove inner/outer modes from input/output path. These are
not needed anymore.
- WiFi:
- reduced neighbor report (RNR) handling for AP mode
- HW timestamping support
- support for randomized auth/deauth TA for PASN privacy
- per-link debugfs for multi-link
- TC offload support for mac80211 drivers
- mac80211 mesh fast-xmit and fast-rx support
- enable Wi-Fi 7 (EHT) mesh support
Netfilter
---------
- Add nf_tables 'brouting' support, to force a packet to be routed
instead of being bridged.
- Update bridge netfilter and ovs conntrack helpers to handle
IPv6 Jumbo packets properly, i.e. fetch the packet length
from hop-by-hop extension header. This is needed for BIT TCP
support.
- The iptables 32bit compat interface isn't compiled in by default
anymore.
- Move ip(6)tables builtin icmp matches to the udptcp one.
This has the advantage that icmp/icmpv6 match doesn't load the
iptables/ip6tables modules anymore when iptables-nft is used.
- Extended netlink error report for netdevice in flowtables and
netdev/chains. Allow for incrementally add/delete devices to netdev
basechain. Allow to create netdev chain without device.
Driver API
----------
- Remove redundant Device Control Error Reporting Enable, as PCI core
has already error reporting enabled at enumeration time.
- Move Multicast DB netlink handlers to core, allowing devices other
then bridge to use them.
- Allow the page_pool to directly recycle the pages from safely
localized NAPI.
- Implement lockless TX queue stop/wake combo macros, allowing for
further code de-duplication and sanitization.
- Add YNL support for user headers and struct attrs.
- Add partial YNL specification for devlink.
- Add partial YNL specification for ethtool.
- Add tc-mqprio and tc-taprio support for preemptible traffic classes.
- Add tx push buf len param to ethtool, specifies the maximum number
of bytes of a transmitted packet a driver can push directly to the
underlying device.
- Add basic LED support for switch/phy.
- Add NAPI documentation, stop relaying on external links.
- Convert dsa_master_ioctl() to netdev notifier. This is a preparatory
work to make the hardware timestamping layer selectable by user
space.
- Add transceiver support and improve the error messages for CAN-FD
controllers.
New hardware / drivers
----------------------
- Ethernet:
- AMD/Pensando core device support
- MediaTek MT7981 SoC
- MediaTek MT7988 SoC
- Broadcom BCM53134 embedded switch
- Texas Instruments CPSW9G ethernet switch
- Qualcomm EMAC3 DWMAC ethernet
- StarFive JH7110 SoC
- NXP CBTX ethernet PHY
- WiFi:
- Apple M1 Pro/Max devices
- RealTek rtl8710bu/rtl8188gu
- RealTek rtl8822bs, rtl8822cs and rtl8821cs SDIO chipset
- Bluetooth:
- Realtek RTL8821CS, RTL8851B, RTL8852BS
- Mediatek MT7663, MT7922
- NXP w8997
- Actions Semi ATS2851
- QTI WCN6855
- Marvell 88W8997
- Can:
- STMicroelectronics bxcan stm32f429
Drivers
-------
- Ethernet NICs:
- Intel (1G, icg):
- add tracking and reporting of QBV config errors.
- add support for configuring max SDU for each Tx queue.
- Intel (100G, ice):
- refactor mailbox overflow detection to support Scalable IOV
- GNSS interface optimization
- Intel (i40e):
- support XDP multi-buffer
- nVidia/Mellanox:
- add the support for linux bridge multicast offload
- enable TC offload for egress and engress MACVLAN over bond
- add support for VxLAN GBP encap/decap flows offload
- extend packet offload to fully support libreswan
- support tunnel mode in mlx5 IPsec packet offload
- extend XDP multi-buffer support
- support MACsec VLAN offload
- add support for dynamic msix vectors allocation
- drop RX page_cache and fully use page_pool
- implement thermal zone to report NIC temperature
- Netronome/Corigine:
- add support for multi-zone conntrack offload
- Solarflare/Xilinx:
- support offloading TC VLAN push/pop actions to the MAE
- support TC decap rules
- support unicast PTP
- Other NICs:
- Broadcom (bnxt): enforce software based freq adjustments only
on shared PHC NIC
- RealTek (r8169): refactor to addess ASPM issues during NAPI poll.
- Micrel (lan8841): add support for PTP_PF_PEROUT
- Cadence (macb): enable PTP unicast
- Engleder (tsnep): add XDP socket zero-copy support
- virtio-net: implement exact header length guest feature
- veth: add page_pool support for page recycling
- vxlan: add MDB data path support
- gve: add XDP support for GQI-QPL format
- geneve: accept every ethertype
- macvlan: allow some packets to bypass broadcast queue
- mana: add support for jumbo frame
- Ethernet high-speed switches:
- Microchip (sparx5): Add support for TC flower templates.
- Ethernet embedded switches:
- Broadcom (b54):
- configure 6318 and 63268 RGMII ports
- Marvell (mv88e6xxx):
- faster C45 bus scan
- Microchip:
- lan966x:
- add support for IS1 VCAP
- better TX/RX from/to CPU performances
- ksz9477: add ETS Qdisc support
- ksz8: enhance static MAC table operations and error handling
- sama7g5: add PTP capability
- NXP (ocelot):
- add support for external ports
- add support for preemptible traffic classes
- Texas Instruments:
- add CPSWxG SGMII support for J7200 and J721E
- Intel WiFi (iwlwifi):
- preparation for Wi-Fi 7 EHT and multi-link support
- EHT (Wi-Fi 7) sniffer support
- hardware timestamping support for some devices/firwmares
- TX beacon protection on newer hardware
- Qualcomm 802.11ax WiFi (ath11k):
- MU-MIMO parameters support
- ack signal support for management packets
- RealTek WiFi (rtw88):
- SDIO bus support
- better support for some SDIO devices
(e.g. MAC address from efuse)
- RealTek WiFi (rtw89):
- HW scan support for 8852b
- better support for 6 GHz scanning
- support for various newer firmware APIs
- framework firmware backwards compatibility
- MediaTek WiFi (mt76):
- P2P support
- mesh A-MSDU support
- EHT (Wi-Fi 7) support
- coredump support
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
-----BEGIN PGP SIGNATURE-----
iQJGBAABCAAwFiEEg1AjqC77wbdLX2LbKSR5jcyPE6QFAmRI/mUSHHBhYmVuaUBy
ZWRoYXQuY29tAAoJECkkeY3MjxOkgO0QAJGxpuN67YgYV0BIM+/atWKEEexJYG7B
9MMpU4jMO3EW/pUS5t7VRsBLUybLYVPmqCZoHodObDfnu59jiPOegb6SikJv/ZwJ
Zw62PVk5MvDnQjlu4e6kDcGwkplteN08TlgI+a49BUTedpdFitrxHAYGW8f2fRO6
cK2XSld+ZucMoym5vRwf8yWS1BwdxnslPMxDJ+/8ZbWBZv44qAnG2vMB/kIx7ObC
Vel/4m6MzTwVsLYBsRvcwMVbNNlZ9GuhztlTzEbfGA4ZhTadIAMgb5VTWXB84Ws7
Aic5wTdli+q+x6/2cxhbyeoVuB9HHObYmLBAciGg4GNljP5rnQBY3X3+KVZ/x9TI
HQB7CmhxmAZVrO9pLARFV+ECrMTH2/dy3NyrZ7uYQ3WPOXJi8hJZjOTO/eeEGL7C
eTjdz0dZBWIBK2gON/6s4nExXVQUTEF2ZsPi52jTTClKjfe5pz/ddeFQIWaY1DTm
pInEiWPAvd28JyiFmhFNHsuIBCjX/Zqe2JuMfMBeBibDAC09o/OGdKJYUI15AiRf
F46Pdb7use/puqfrYW44kSAfaPYoBiE+hj1RdeQfen35xD9HVE4vdnLNeuhRlFF9
aQfyIRHYQofkumRDr5f8JEY66cl9NiKQ4IVW1xxQfYDNdC6wQqREPG1md7rJVMrJ
vP7ugFnttneg
=ITVa
-----END PGP SIGNATURE-----
Merge tag 'net-next-6.4' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next
Pull networking updates from Paolo Abeni:
"Core:
- Introduce a config option to tweak MAX_SKB_FRAGS. Increasing the
default value allows for better BIG TCP performances
- Reduce compound page head access for zero-copy data transfers
- RPS/RFS improvements, avoiding unneeded NET_RX_SOFTIRQ when
possible
- Threaded NAPI improvements, adding defer skb free support and
unneeded softirq avoidance
- Address dst_entry reference count scalability issues, via false
sharing avoidance and optimize refcount tracking
- Add lockless accesses annotation to sk_err[_soft]
- Optimize again the skb struct layout
- Extends the skb drop reasons to make it usable by multiple
subsystems
- Better const qualifier awareness for socket casts
BPF:
- Add skb and XDP typed dynptrs which allow BPF programs for more
ergonomic and less brittle iteration through data and
variable-sized accesses
- Add a new BPF netfilter program type and minimal support to hook
BPF programs to netfilter hooks such as prerouting or forward
- Add more precise memory usage reporting for all BPF map types
- Adds support for using {FOU,GUE} encap with an ipip device
operating in collect_md mode and add a set of BPF kfuncs for
controlling encap params
- Allow BPF programs to detect at load time whether a particular
kfunc exists or not, and also add support for this in light
skeleton
- Bigger batch of BPF verifier improvements to prepare for upcoming
BPF open-coded iterators allowing for less restrictive looping
capabilities
- Rework RCU enforcement in the verifier, add kptr_rcu and enforce
BPF programs to NULL-check before passing such pointers into kfunc
- Add support for kptrs in percpu hashmaps, percpu LRU hashmaps and
in local storage maps
- Enable RCU semantics for task BPF kptrs and allow referenced kptr
tasks to be stored in BPF maps
- Add support for refcounted local kptrs to the verifier for allowing
shared ownership, useful for adding a node to both the BPF list and
rbtree
- Add BPF verifier support for ST instructions in
convert_ctx_access() which will help new -mcpu=v4 clang flag to
start emitting them
- Add ARM32 USDT support to libbpf
- Improve bpftool's visual program dump which produces the control
flow graph in a DOT format by adding C source inline annotations
Protocols:
- IPv4: Allow adding to IPv4 address a 'protocol' tag. Such value
indicates the provenance of the IP address
- IPv6: optimize route lookup, dropping unneeded R/W lock acquisition
- Add the handshake upcall mechanism, allowing the user-space to
implement generic TLS handshake on kernel's behalf
- Bridge: support per-{Port, VLAN} neighbor suppression, increasing
resilience to nodes failures
- SCTP: add support for Fair Capacity and Weighted Fair Queueing
schedulers
- MPTCP: delay first subflow allocation up to its first usage. This
will allow for later better LSM interaction
- xfrm: Remove inner/outer modes from input/output path. These are
not needed anymore
- WiFi:
- reduced neighbor report (RNR) handling for AP mode
- HW timestamping support
- support for randomized auth/deauth TA for PASN privacy
- per-link debugfs for multi-link
- TC offload support for mac80211 drivers
- mac80211 mesh fast-xmit and fast-rx support
- enable Wi-Fi 7 (EHT) mesh support
Netfilter:
- Add nf_tables 'brouting' support, to force a packet to be routed
instead of being bridged
- Update bridge netfilter and ovs conntrack helpers to handle IPv6
Jumbo packets properly, i.e. fetch the packet length from
hop-by-hop extension header. This is needed for BIT TCP support
- The iptables 32bit compat interface isn't compiled in by default
anymore
- Move ip(6)tables builtin icmp matches to the udptcp one. This has
the advantage that icmp/icmpv6 match doesn't load the
iptables/ip6tables modules anymore when iptables-nft is used
- Extended netlink error report for netdevice in flowtables and
netdev/chains. Allow for incrementally add/delete devices to netdev
basechain. Allow to create netdev chain without device
Driver API:
- Remove redundant Device Control Error Reporting Enable, as PCI core
has already error reporting enabled at enumeration time
- Move Multicast DB netlink handlers to core, allowing devices other
then bridge to use them
- Allow the page_pool to directly recycle the pages from safely
localized NAPI
- Implement lockless TX queue stop/wake combo macros, allowing for
further code de-duplication and sanitization
- Add YNL support for user headers and struct attrs
- Add partial YNL specification for devlink
- Add partial YNL specification for ethtool
- Add tc-mqprio and tc-taprio support for preemptible traffic classes
- Add tx push buf len param to ethtool, specifies the maximum number
of bytes of a transmitted packet a driver can push directly to the
underlying device
- Add basic LED support for switch/phy
- Add NAPI documentation, stop relaying on external links
- Convert dsa_master_ioctl() to netdev notifier. This is a
preparatory work to make the hardware timestamping layer selectable
by user space
- Add transceiver support and improve the error messages for CAN-FD
controllers
New hardware / drivers:
- Ethernet:
- AMD/Pensando core device support
- MediaTek MT7981 SoC
- MediaTek MT7988 SoC
- Broadcom BCM53134 embedded switch
- Texas Instruments CPSW9G ethernet switch
- Qualcomm EMAC3 DWMAC ethernet
- StarFive JH7110 SoC
- NXP CBTX ethernet PHY
- WiFi:
- Apple M1 Pro/Max devices
- RealTek rtl8710bu/rtl8188gu
- RealTek rtl8822bs, rtl8822cs and rtl8821cs SDIO chipset
- Bluetooth:
- Realtek RTL8821CS, RTL8851B, RTL8852BS
- Mediatek MT7663, MT7922
- NXP w8997
- Actions Semi ATS2851
- QTI WCN6855
- Marvell 88W8997
- Can:
- STMicroelectronics bxcan stm32f429
Drivers:
- Ethernet NICs:
- Intel (1G, icg):
- add tracking and reporting of QBV config errors
- add support for configuring max SDU for each Tx queue
- Intel (100G, ice):
- refactor mailbox overflow detection to support Scalable IOV
- GNSS interface optimization
- Intel (i40e):
- support XDP multi-buffer
- nVidia/Mellanox:
- add the support for linux bridge multicast offload
- enable TC offload for egress and engress MACVLAN over bond
- add support for VxLAN GBP encap/decap flows offload
- extend packet offload to fully support libreswan
- support tunnel mode in mlx5 IPsec packet offload
- extend XDP multi-buffer support
- support MACsec VLAN offload
- add support for dynamic msix vectors allocation
- drop RX page_cache and fully use page_pool
- implement thermal zone to report NIC temperature
- Netronome/Corigine:
- add support for multi-zone conntrack offload
- Solarflare/Xilinx:
- support offloading TC VLAN push/pop actions to the MAE
- support TC decap rules
- support unicast PTP
- Other NICs:
- Broadcom (bnxt): enforce software based freq adjustments only on
shared PHC NIC
- RealTek (r8169): refactor to addess ASPM issues during NAPI poll
- Micrel (lan8841): add support for PTP_PF_PEROUT
- Cadence (macb): enable PTP unicast
- Engleder (tsnep): add XDP socket zero-copy support
- virtio-net: implement exact header length guest feature
- veth: add page_pool support for page recycling
- vxlan: add MDB data path support
- gve: add XDP support for GQI-QPL format
- geneve: accept every ethertype
- macvlan: allow some packets to bypass broadcast queue
- mana: add support for jumbo frame
- Ethernet high-speed switches:
- Microchip (sparx5): Add support for TC flower templates
- Ethernet embedded switches:
- Broadcom (b54):
- configure 6318 and 63268 RGMII ports
- Marvell (mv88e6xxx):
- faster C45 bus scan
- Microchip:
- lan966x:
- add support for IS1 VCAP
- better TX/RX from/to CPU performances
- ksz9477: add ETS Qdisc support
- ksz8: enhance static MAC table operations and error handling
- sama7g5: add PTP capability
- NXP (ocelot):
- add support for external ports
- add support for preemptible traffic classes
- Texas Instruments:
- add CPSWxG SGMII support for J7200 and J721E
- Intel WiFi (iwlwifi):
- preparation for Wi-Fi 7 EHT and multi-link support
- EHT (Wi-Fi 7) sniffer support
- hardware timestamping support for some devices/firwmares
- TX beacon protection on newer hardware
- Qualcomm 802.11ax WiFi (ath11k):
- MU-MIMO parameters support
- ack signal support for management packets
- RealTek WiFi (rtw88):
- SDIO bus support
- better support for some SDIO devices (e.g. MAC address from
efuse)
- RealTek WiFi (rtw89):
- HW scan support for 8852b
- better support for 6 GHz scanning
- support for various newer firmware APIs
- framework firmware backwards compatibility
- MediaTek WiFi (mt76):
- P2P support
- mesh A-MSDU support
- EHT (Wi-Fi 7) support
- coredump support"
* tag 'net-next-6.4' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next: (2078 commits)
net: phy: hide the PHYLIB_LEDS knob
net: phy: marvell-88x2222: remove unnecessary (void*) conversions
tcp/udp: Fix memleaks of sk and zerocopy skbs with TX timestamp.
net: amd: Fix link leak when verifying config failed
net: phy: marvell: Fix inconsistent indenting in led_blink_set
lan966x: Don't use xdp_frame when action is XDP_TX
tsnep: Add XDP socket zero-copy TX support
tsnep: Add XDP socket zero-copy RX support
tsnep: Move skb receive action to separate function
tsnep: Add functions for queue enable/disable
tsnep: Rework TX/RX queue initialization
tsnep: Replace modulo operation with mask
net: phy: dp83867: Add led_brightness_set support
net: phy: Fix reading LED reg property
drivers: nfc: nfcsim: remove return value check of `dev_dir`
net: phy: dp83867: Remove unnecessary (void*) conversions
net: ethtool: coalesce: try to make user settings stick twice
net: mana: Check if netdev/napi_alloc_frag returns single page
net: mana: Rename mana_refill_rxoob and remove some empty lines
net: veth: add page_pool stats
...
862 lines
21 KiB
C
862 lines
21 KiB
C
// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
|
|
/*
|
|
* Copyright (C) 2012-2014, 2019-2022 Intel Corporation
|
|
* Copyright (C) 2013-2014 Intel Mobile Communications GmbH
|
|
* Copyright (C) 2015-2016 Intel Deutschland GmbH
|
|
*/
|
|
#include <linux/sort.h>
|
|
|
|
#include "mvm.h"
|
|
|
|
#define IWL_MVM_TEMP_NOTIF_WAIT_TIMEOUT HZ
|
|
|
|
void iwl_mvm_enter_ctkill(struct iwl_mvm *mvm)
|
|
{
|
|
struct iwl_mvm_tt_mgmt *tt = &mvm->thermal_throttle;
|
|
u32 duration = tt->params.ct_kill_duration;
|
|
|
|
if (test_bit(IWL_MVM_STATUS_HW_CTKILL, &mvm->status))
|
|
return;
|
|
|
|
IWL_ERR(mvm, "Enter CT Kill\n");
|
|
iwl_mvm_set_hw_ctkill_state(mvm, true);
|
|
|
|
if (!iwl_mvm_is_tt_in_fw(mvm)) {
|
|
tt->throttle = false;
|
|
tt->dynamic_smps = false;
|
|
}
|
|
|
|
/* Don't schedule an exit work if we're in test mode, since
|
|
* the temperature will not change unless we manually set it
|
|
* again (or disable testing).
|
|
*/
|
|
if (!mvm->temperature_test)
|
|
schedule_delayed_work(&tt->ct_kill_exit,
|
|
round_jiffies_relative(duration * HZ));
|
|
}
|
|
|
|
static void iwl_mvm_exit_ctkill(struct iwl_mvm *mvm)
|
|
{
|
|
if (!test_bit(IWL_MVM_STATUS_HW_CTKILL, &mvm->status))
|
|
return;
|
|
|
|
IWL_ERR(mvm, "Exit CT Kill\n");
|
|
iwl_mvm_set_hw_ctkill_state(mvm, false);
|
|
}
|
|
|
|
static void iwl_mvm_tt_temp_changed(struct iwl_mvm *mvm, u32 temp)
|
|
{
|
|
/* ignore the notification if we are in test mode */
|
|
if (mvm->temperature_test)
|
|
return;
|
|
|
|
if (mvm->temperature == temp)
|
|
return;
|
|
|
|
mvm->temperature = temp;
|
|
iwl_mvm_tt_handler(mvm);
|
|
}
|
|
|
|
static int iwl_mvm_temp_notif_parse(struct iwl_mvm *mvm,
|
|
struct iwl_rx_packet *pkt)
|
|
{
|
|
struct iwl_dts_measurement_notif_v1 *notif_v1;
|
|
int len = iwl_rx_packet_payload_len(pkt);
|
|
int temp;
|
|
|
|
/* we can use notif_v1 only, because v2 only adds an additional
|
|
* parameter, which is not used in this function.
|
|
*/
|
|
if (WARN_ON_ONCE(len < sizeof(*notif_v1))) {
|
|
IWL_ERR(mvm, "Invalid DTS_MEASUREMENT_NOTIFICATION\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
notif_v1 = (void *)pkt->data;
|
|
|
|
temp = le32_to_cpu(notif_v1->temp);
|
|
|
|
/* shouldn't be negative, but since it's s32, make sure it isn't */
|
|
if (WARN_ON_ONCE(temp < 0))
|
|
temp = 0;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "DTS_MEASUREMENT_NOTIFICATION - %d\n", temp);
|
|
|
|
return temp;
|
|
}
|
|
|
|
static bool iwl_mvm_temp_notif_wait(struct iwl_notif_wait_data *notif_wait,
|
|
struct iwl_rx_packet *pkt, void *data)
|
|
{
|
|
struct iwl_mvm *mvm =
|
|
container_of(notif_wait, struct iwl_mvm, notif_wait);
|
|
int *temp = data;
|
|
int ret;
|
|
|
|
ret = iwl_mvm_temp_notif_parse(mvm, pkt);
|
|
if (ret < 0)
|
|
return true;
|
|
|
|
*temp = ret;
|
|
|
|
return true;
|
|
}
|
|
|
|
void iwl_mvm_temp_notif(struct iwl_mvm *mvm, struct iwl_rx_cmd_buffer *rxb)
|
|
{
|
|
struct iwl_rx_packet *pkt = rxb_addr(rxb);
|
|
struct iwl_dts_measurement_notif_v2 *notif_v2;
|
|
int len = iwl_rx_packet_payload_len(pkt);
|
|
int temp;
|
|
u32 ths_crossed;
|
|
|
|
/* the notification is handled synchronously in ctkill, so skip here */
|
|
if (test_bit(IWL_MVM_STATUS_HW_CTKILL, &mvm->status))
|
|
return;
|
|
|
|
temp = iwl_mvm_temp_notif_parse(mvm, pkt);
|
|
|
|
if (!iwl_mvm_is_tt_in_fw(mvm)) {
|
|
if (temp >= 0)
|
|
iwl_mvm_tt_temp_changed(mvm, temp);
|
|
return;
|
|
}
|
|
|
|
if (WARN_ON_ONCE(len < sizeof(*notif_v2))) {
|
|
IWL_ERR(mvm, "Invalid DTS_MEASUREMENT_NOTIFICATION\n");
|
|
return;
|
|
}
|
|
|
|
notif_v2 = (void *)pkt->data;
|
|
ths_crossed = le32_to_cpu(notif_v2->threshold_idx);
|
|
|
|
/* 0xFF in ths_crossed means the notification is not related
|
|
* to a trip, so we can ignore it here.
|
|
*/
|
|
if (ths_crossed == 0xFF)
|
|
return;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "Temp = %d Threshold crossed = %d\n",
|
|
temp, ths_crossed);
|
|
|
|
#ifdef CONFIG_THERMAL
|
|
if (WARN_ON(ths_crossed >= IWL_MAX_DTS_TRIPS))
|
|
return;
|
|
|
|
if (mvm->tz_device.tzone) {
|
|
struct iwl_mvm_thermal_device *tz_dev = &mvm->tz_device;
|
|
|
|
thermal_zone_device_update(tz_dev->tzone,
|
|
THERMAL_TRIP_VIOLATED);
|
|
}
|
|
#endif /* CONFIG_THERMAL */
|
|
}
|
|
|
|
void iwl_mvm_ct_kill_notif(struct iwl_mvm *mvm, struct iwl_rx_cmd_buffer *rxb)
|
|
{
|
|
struct iwl_rx_packet *pkt = rxb_addr(rxb);
|
|
struct ct_kill_notif *notif;
|
|
|
|
notif = (struct ct_kill_notif *)pkt->data;
|
|
IWL_DEBUG_TEMP(mvm, "CT Kill notification temperature = %d\n",
|
|
notif->temperature);
|
|
if (iwl_fw_lookup_notif_ver(mvm->fw, PHY_OPS_GROUP,
|
|
CT_KILL_NOTIFICATION, 0) > 1)
|
|
IWL_DEBUG_TEMP(mvm,
|
|
"CT kill notification DTS bitmap = 0x%x, Scheme = %d\n",
|
|
notif->dts, notif->scheme);
|
|
|
|
iwl_mvm_enter_ctkill(mvm);
|
|
}
|
|
|
|
/*
|
|
* send the DTS_MEASUREMENT_TRIGGER command with or without waiting for a
|
|
* response. If we get a response then the measurement is stored in 'temp'
|
|
*/
|
|
static int iwl_mvm_send_temp_cmd(struct iwl_mvm *mvm, bool response, s32 *temp)
|
|
{
|
|
struct iwl_host_cmd cmd = {};
|
|
struct iwl_dts_measurement_cmd dts_cmd = {
|
|
.flags = cpu_to_le32(DTS_TRIGGER_CMD_FLAGS_TEMP),
|
|
};
|
|
struct iwl_ext_dts_measurement_cmd ext_cmd = {
|
|
.control_mode = cpu_to_le32(DTS_DIRECT_WITHOUT_MEASURE),
|
|
};
|
|
struct iwl_dts_measurement_resp *resp;
|
|
void *cmd_ptr;
|
|
int ret;
|
|
u32 cmd_flags = 0;
|
|
u16 len;
|
|
|
|
/* Check which command format is used (regular/extended) */
|
|
if (fw_has_capa(&mvm->fw->ucode_capa,
|
|
IWL_UCODE_TLV_CAPA_EXTENDED_DTS_MEASURE)) {
|
|
len = sizeof(ext_cmd);
|
|
cmd_ptr = &ext_cmd;
|
|
} else {
|
|
len = sizeof(dts_cmd);
|
|
cmd_ptr = &dts_cmd;
|
|
}
|
|
/* The command version where we get a response is zero length */
|
|
if (response) {
|
|
cmd_flags = CMD_WANT_SKB;
|
|
len = 0;
|
|
}
|
|
|
|
cmd.id = WIDE_ID(PHY_OPS_GROUP, CMD_DTS_MEASUREMENT_TRIGGER_WIDE);
|
|
cmd.len[0] = len;
|
|
cmd.flags = cmd_flags;
|
|
cmd.data[0] = cmd_ptr;
|
|
|
|
IWL_DEBUG_TEMP(mvm,
|
|
"Sending temperature measurement command - %s response\n",
|
|
response ? "with" : "without");
|
|
ret = iwl_mvm_send_cmd(mvm, &cmd);
|
|
|
|
if (ret) {
|
|
IWL_ERR(mvm,
|
|
"Failed to send the temperature measurement command (err=%d)\n",
|
|
ret);
|
|
return ret;
|
|
}
|
|
|
|
if (response) {
|
|
resp = (void *)cmd.resp_pkt->data;
|
|
*temp = le32_to_cpu(resp->temp);
|
|
IWL_DEBUG_TEMP(mvm,
|
|
"Got temperature measurement response: temp=%d\n",
|
|
*temp);
|
|
iwl_free_resp(&cmd);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int iwl_mvm_get_temp(struct iwl_mvm *mvm, s32 *temp)
|
|
{
|
|
struct iwl_notification_wait wait_temp_notif;
|
|
static u16 temp_notif[] = { WIDE_ID(PHY_OPS_GROUP,
|
|
DTS_MEASUREMENT_NOTIF_WIDE) };
|
|
int ret;
|
|
u8 cmd_ver;
|
|
|
|
/*
|
|
* If command version is 1 we send the command and immediately get
|
|
* a response. For older versions we send the command and wait for a
|
|
* notification (no command TLV for previous versions).
|
|
*/
|
|
cmd_ver = iwl_fw_lookup_cmd_ver(mvm->fw,
|
|
WIDE_ID(PHY_OPS_GROUP, CMD_DTS_MEASUREMENT_TRIGGER_WIDE),
|
|
IWL_FW_CMD_VER_UNKNOWN);
|
|
if (cmd_ver == 1)
|
|
return iwl_mvm_send_temp_cmd(mvm, true, temp);
|
|
|
|
lockdep_assert_held(&mvm->mutex);
|
|
|
|
iwl_init_notification_wait(&mvm->notif_wait, &wait_temp_notif,
|
|
temp_notif, ARRAY_SIZE(temp_notif),
|
|
iwl_mvm_temp_notif_wait, temp);
|
|
|
|
ret = iwl_mvm_send_temp_cmd(mvm, false, temp);
|
|
if (ret) {
|
|
iwl_remove_notification(&mvm->notif_wait, &wait_temp_notif);
|
|
return ret;
|
|
}
|
|
|
|
ret = iwl_wait_notification(&mvm->notif_wait, &wait_temp_notif,
|
|
IWL_MVM_TEMP_NOTIF_WAIT_TIMEOUT);
|
|
if (ret)
|
|
IWL_WARN(mvm, "Getting the temperature timed out\n");
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void check_exit_ctkill(struct work_struct *work)
|
|
{
|
|
struct iwl_mvm_tt_mgmt *tt;
|
|
struct iwl_mvm *mvm;
|
|
u32 duration;
|
|
s32 temp;
|
|
int ret;
|
|
|
|
tt = container_of(work, struct iwl_mvm_tt_mgmt, ct_kill_exit.work);
|
|
mvm = container_of(tt, struct iwl_mvm, thermal_throttle);
|
|
|
|
if (iwl_mvm_is_tt_in_fw(mvm)) {
|
|
iwl_mvm_exit_ctkill(mvm);
|
|
|
|
return;
|
|
}
|
|
|
|
duration = tt->params.ct_kill_duration;
|
|
|
|
flush_work(&mvm->roc_done_wk);
|
|
|
|
mutex_lock(&mvm->mutex);
|
|
|
|
if (__iwl_mvm_mac_start(mvm))
|
|
goto reschedule;
|
|
|
|
ret = iwl_mvm_get_temp(mvm, &temp);
|
|
|
|
__iwl_mvm_mac_stop(mvm);
|
|
|
|
if (ret)
|
|
goto reschedule;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "NIC temperature: %d\n", temp);
|
|
|
|
if (temp <= tt->params.ct_kill_exit) {
|
|
mutex_unlock(&mvm->mutex);
|
|
iwl_mvm_exit_ctkill(mvm);
|
|
return;
|
|
}
|
|
|
|
reschedule:
|
|
mutex_unlock(&mvm->mutex);
|
|
schedule_delayed_work(&mvm->thermal_throttle.ct_kill_exit,
|
|
round_jiffies(duration * HZ));
|
|
}
|
|
|
|
static void iwl_mvm_tt_smps_iterator(void *_data, u8 *mac,
|
|
struct ieee80211_vif *vif)
|
|
{
|
|
struct iwl_mvm *mvm = _data;
|
|
enum ieee80211_smps_mode smps_mode;
|
|
|
|
lockdep_assert_held(&mvm->mutex);
|
|
|
|
if (mvm->thermal_throttle.dynamic_smps)
|
|
smps_mode = IEEE80211_SMPS_DYNAMIC;
|
|
else
|
|
smps_mode = IEEE80211_SMPS_AUTOMATIC;
|
|
|
|
if (vif->type != NL80211_IFTYPE_STATION)
|
|
return;
|
|
|
|
iwl_mvm_update_smps(mvm, vif, IWL_MVM_SMPS_REQ_TT, smps_mode, 0);
|
|
}
|
|
|
|
static void iwl_mvm_tt_tx_protection(struct iwl_mvm *mvm, bool enable)
|
|
{
|
|
struct iwl_mvm_sta *mvmsta;
|
|
int i, err;
|
|
|
|
for (i = 0; i < mvm->fw->ucode_capa.num_stations; i++) {
|
|
mvmsta = iwl_mvm_sta_from_staid_protected(mvm, i);
|
|
if (!mvmsta)
|
|
continue;
|
|
|
|
if (enable == mvmsta->tt_tx_protection)
|
|
continue;
|
|
err = iwl_mvm_tx_protection(mvm, mvmsta, enable);
|
|
if (err) {
|
|
IWL_ERR(mvm, "Failed to %s Tx protection\n",
|
|
enable ? "enable" : "disable");
|
|
} else {
|
|
IWL_DEBUG_TEMP(mvm, "%s Tx protection\n",
|
|
enable ? "Enable" : "Disable");
|
|
mvmsta->tt_tx_protection = enable;
|
|
}
|
|
}
|
|
}
|
|
|
|
void iwl_mvm_tt_tx_backoff(struct iwl_mvm *mvm, u32 backoff)
|
|
{
|
|
struct iwl_host_cmd cmd = {
|
|
.id = REPLY_THERMAL_MNG_BACKOFF,
|
|
.len = { sizeof(u32), },
|
|
.data = { &backoff, },
|
|
};
|
|
|
|
backoff = max(backoff, mvm->thermal_throttle.min_backoff);
|
|
|
|
if (iwl_mvm_send_cmd(mvm, &cmd) == 0) {
|
|
IWL_DEBUG_TEMP(mvm, "Set Thermal Tx backoff to: %u\n",
|
|
backoff);
|
|
mvm->thermal_throttle.tx_backoff = backoff;
|
|
} else {
|
|
IWL_ERR(mvm, "Failed to change Thermal Tx backoff\n");
|
|
}
|
|
}
|
|
|
|
void iwl_mvm_tt_handler(struct iwl_mvm *mvm)
|
|
{
|
|
struct iwl_tt_params *params = &mvm->thermal_throttle.params;
|
|
struct iwl_mvm_tt_mgmt *tt = &mvm->thermal_throttle;
|
|
s32 temperature = mvm->temperature;
|
|
bool throttle_enable = false;
|
|
int i;
|
|
u32 tx_backoff;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "NIC temperature: %d\n", mvm->temperature);
|
|
|
|
if (params->support_ct_kill && temperature >= params->ct_kill_entry) {
|
|
iwl_mvm_enter_ctkill(mvm);
|
|
return;
|
|
}
|
|
|
|
if (params->support_ct_kill &&
|
|
temperature <= params->ct_kill_exit) {
|
|
iwl_mvm_exit_ctkill(mvm);
|
|
return;
|
|
}
|
|
|
|
if (params->support_dynamic_smps) {
|
|
if (!tt->dynamic_smps &&
|
|
temperature >= params->dynamic_smps_entry) {
|
|
IWL_DEBUG_TEMP(mvm, "Enable dynamic SMPS\n");
|
|
tt->dynamic_smps = true;
|
|
ieee80211_iterate_active_interfaces_atomic(
|
|
mvm->hw, IEEE80211_IFACE_ITER_NORMAL,
|
|
iwl_mvm_tt_smps_iterator, mvm);
|
|
throttle_enable = true;
|
|
} else if (tt->dynamic_smps &&
|
|
temperature <= params->dynamic_smps_exit) {
|
|
IWL_DEBUG_TEMP(mvm, "Disable dynamic SMPS\n");
|
|
tt->dynamic_smps = false;
|
|
ieee80211_iterate_active_interfaces_atomic(
|
|
mvm->hw, IEEE80211_IFACE_ITER_NORMAL,
|
|
iwl_mvm_tt_smps_iterator, mvm);
|
|
}
|
|
}
|
|
|
|
if (params->support_tx_protection) {
|
|
if (temperature >= params->tx_protection_entry) {
|
|
iwl_mvm_tt_tx_protection(mvm, true);
|
|
throttle_enable = true;
|
|
} else if (temperature <= params->tx_protection_exit) {
|
|
iwl_mvm_tt_tx_protection(mvm, false);
|
|
}
|
|
}
|
|
|
|
if (params->support_tx_backoff) {
|
|
tx_backoff = tt->min_backoff;
|
|
for (i = 0; i < TT_TX_BACKOFF_SIZE; i++) {
|
|
if (temperature < params->tx_backoff[i].temperature)
|
|
break;
|
|
tx_backoff = max(tt->min_backoff,
|
|
params->tx_backoff[i].backoff);
|
|
}
|
|
if (tx_backoff != tt->min_backoff)
|
|
throttle_enable = true;
|
|
if (tt->tx_backoff != tx_backoff)
|
|
iwl_mvm_tt_tx_backoff(mvm, tx_backoff);
|
|
}
|
|
|
|
if (!tt->throttle && throttle_enable) {
|
|
IWL_WARN(mvm,
|
|
"Due to high temperature thermal throttling initiated\n");
|
|
tt->throttle = true;
|
|
} else if (tt->throttle && !tt->dynamic_smps &&
|
|
tt->tx_backoff == tt->min_backoff &&
|
|
temperature <= params->tx_protection_exit) {
|
|
IWL_WARN(mvm,
|
|
"Temperature is back to normal thermal throttling stopped\n");
|
|
tt->throttle = false;
|
|
}
|
|
}
|
|
|
|
static const struct iwl_tt_params iwl_mvm_default_tt_params = {
|
|
.ct_kill_entry = 118,
|
|
.ct_kill_exit = 96,
|
|
.ct_kill_duration = 5,
|
|
.dynamic_smps_entry = 114,
|
|
.dynamic_smps_exit = 110,
|
|
.tx_protection_entry = 114,
|
|
.tx_protection_exit = 108,
|
|
.tx_backoff = {
|
|
{.temperature = 112, .backoff = 200},
|
|
{.temperature = 113, .backoff = 600},
|
|
{.temperature = 114, .backoff = 1200},
|
|
{.temperature = 115, .backoff = 2000},
|
|
{.temperature = 116, .backoff = 4000},
|
|
{.temperature = 117, .backoff = 10000},
|
|
},
|
|
.support_ct_kill = true,
|
|
.support_dynamic_smps = true,
|
|
.support_tx_protection = true,
|
|
.support_tx_backoff = true,
|
|
};
|
|
|
|
/* budget in mWatt */
|
|
static const u32 iwl_mvm_cdev_budgets[] = {
|
|
2400, /* cooling state 0 */
|
|
2000, /* cooling state 1 */
|
|
1800, /* cooling state 2 */
|
|
1600, /* cooling state 3 */
|
|
1400, /* cooling state 4 */
|
|
1200, /* cooling state 5 */
|
|
1000, /* cooling state 6 */
|
|
900, /* cooling state 7 */
|
|
800, /* cooling state 8 */
|
|
700, /* cooling state 9 */
|
|
650, /* cooling state 10 */
|
|
600, /* cooling state 11 */
|
|
550, /* cooling state 12 */
|
|
500, /* cooling state 13 */
|
|
450, /* cooling state 14 */
|
|
400, /* cooling state 15 */
|
|
350, /* cooling state 16 */
|
|
300, /* cooling state 17 */
|
|
250, /* cooling state 18 */
|
|
200, /* cooling state 19 */
|
|
150, /* cooling state 20 */
|
|
};
|
|
|
|
int iwl_mvm_ctdp_command(struct iwl_mvm *mvm, u32 op, u32 state)
|
|
{
|
|
struct iwl_mvm_ctdp_cmd cmd = {
|
|
.operation = cpu_to_le32(op),
|
|
.budget = cpu_to_le32(iwl_mvm_cdev_budgets[state]),
|
|
.window_size = 0,
|
|
};
|
|
int ret;
|
|
u32 status;
|
|
|
|
lockdep_assert_held(&mvm->mutex);
|
|
|
|
status = 0;
|
|
ret = iwl_mvm_send_cmd_pdu_status(mvm, WIDE_ID(PHY_OPS_GROUP,
|
|
CTDP_CONFIG_CMD),
|
|
sizeof(cmd), &cmd, &status);
|
|
|
|
if (ret) {
|
|
IWL_ERR(mvm, "cTDP command failed (err=%d)\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
switch (op) {
|
|
case CTDP_CMD_OPERATION_START:
|
|
#ifdef CONFIG_THERMAL
|
|
mvm->cooling_dev.cur_state = state;
|
|
#endif /* CONFIG_THERMAL */
|
|
break;
|
|
case CTDP_CMD_OPERATION_REPORT:
|
|
IWL_DEBUG_TEMP(mvm, "cTDP avg energy in mWatt = %d\n", status);
|
|
/* when the function is called with CTDP_CMD_OPERATION_REPORT
|
|
* option the function should return the average budget value
|
|
* that is received from the FW.
|
|
* The budget can't be less or equal to 0, so it's possible
|
|
* to distinguish between error values and budgets.
|
|
*/
|
|
return status;
|
|
case CTDP_CMD_OPERATION_STOP:
|
|
IWL_DEBUG_TEMP(mvm, "cTDP stopped successfully\n");
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_THERMAL
|
|
static int compare_temps(const void *a, const void *b)
|
|
{
|
|
return ((s16)le16_to_cpu(*(__le16 *)a) -
|
|
(s16)le16_to_cpu(*(__le16 *)b));
|
|
}
|
|
#endif
|
|
|
|
int iwl_mvm_send_temp_report_ths_cmd(struct iwl_mvm *mvm)
|
|
{
|
|
struct temp_report_ths_cmd cmd = {0};
|
|
int ret;
|
|
#ifdef CONFIG_THERMAL
|
|
int i, j, idx = 0;
|
|
|
|
lockdep_assert_held(&mvm->mutex);
|
|
|
|
if (!mvm->tz_device.tzone)
|
|
goto send;
|
|
|
|
/* The driver holds array of temperature trips that are unsorted
|
|
* and uncompressed, the FW should get it compressed and sorted
|
|
*/
|
|
|
|
/* compress trips to cmd array, remove uninitialized values*/
|
|
for (i = 0; i < IWL_MAX_DTS_TRIPS; i++) {
|
|
if (mvm->tz_device.trips[i].temperature != INT_MIN) {
|
|
cmd.thresholds[idx++] =
|
|
cpu_to_le16((s16)(mvm->tz_device.trips[i].temperature / 1000));
|
|
}
|
|
}
|
|
cmd.num_temps = cpu_to_le32(idx);
|
|
|
|
if (!idx)
|
|
goto send;
|
|
|
|
/*sort cmd array*/
|
|
sort(cmd.thresholds, idx, sizeof(s16), compare_temps, NULL);
|
|
|
|
/* we should save the indexes of trips because we sort
|
|
* and compress the orginal array
|
|
*/
|
|
for (i = 0; i < idx; i++) {
|
|
for (j = 0; j < IWL_MAX_DTS_TRIPS; j++) {
|
|
if ((int)(le16_to_cpu(cmd.thresholds[i]) * 1000) ==
|
|
mvm->tz_device.trips[j].temperature)
|
|
mvm->tz_device.fw_trips_index[i] = j;
|
|
}
|
|
}
|
|
|
|
send:
|
|
#endif
|
|
ret = iwl_mvm_send_cmd_pdu(mvm, WIDE_ID(PHY_OPS_GROUP,
|
|
TEMP_REPORTING_THRESHOLDS_CMD),
|
|
0, sizeof(cmd), &cmd);
|
|
if (ret)
|
|
IWL_ERR(mvm, "TEMP_REPORT_THS_CMD command failed (err=%d)\n",
|
|
ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
#ifdef CONFIG_THERMAL
|
|
static int iwl_mvm_tzone_get_temp(struct thermal_zone_device *device,
|
|
int *temperature)
|
|
{
|
|
struct iwl_mvm *mvm = thermal_zone_device_priv(device);
|
|
int ret;
|
|
int temp;
|
|
|
|
mutex_lock(&mvm->mutex);
|
|
|
|
if (!iwl_mvm_firmware_running(mvm) ||
|
|
mvm->fwrt.cur_fw_img != IWL_UCODE_REGULAR) {
|
|
ret = -ENODATA;
|
|
goto out;
|
|
}
|
|
|
|
ret = iwl_mvm_get_temp(mvm, &temp);
|
|
if (ret)
|
|
goto out;
|
|
|
|
*temperature = temp * 1000;
|
|
|
|
out:
|
|
mutex_unlock(&mvm->mutex);
|
|
return ret;
|
|
}
|
|
|
|
static int iwl_mvm_tzone_set_trip_temp(struct thermal_zone_device *device,
|
|
int trip, int temp)
|
|
{
|
|
struct iwl_mvm *mvm = thermal_zone_device_priv(device);
|
|
struct iwl_mvm_thermal_device *tzone;
|
|
int ret;
|
|
|
|
mutex_lock(&mvm->mutex);
|
|
|
|
if (!iwl_mvm_firmware_running(mvm) ||
|
|
mvm->fwrt.cur_fw_img != IWL_UCODE_REGULAR) {
|
|
ret = -EIO;
|
|
goto out;
|
|
}
|
|
|
|
if ((temp / 1000) > S16_MAX) {
|
|
ret = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
tzone = &mvm->tz_device;
|
|
if (!tzone) {
|
|
ret = -EIO;
|
|
goto out;
|
|
}
|
|
|
|
ret = iwl_mvm_send_temp_report_ths_cmd(mvm);
|
|
out:
|
|
mutex_unlock(&mvm->mutex);
|
|
return ret;
|
|
}
|
|
|
|
static struct thermal_zone_device_ops tzone_ops = {
|
|
.get_temp = iwl_mvm_tzone_get_temp,
|
|
.set_trip_temp = iwl_mvm_tzone_set_trip_temp,
|
|
};
|
|
|
|
/* make all trips writable */
|
|
#define IWL_WRITABLE_TRIPS_MSK (BIT(IWL_MAX_DTS_TRIPS) - 1)
|
|
|
|
static void iwl_mvm_thermal_zone_register(struct iwl_mvm *mvm)
|
|
{
|
|
int i, ret;
|
|
char name[16];
|
|
static atomic_t counter = ATOMIC_INIT(0);
|
|
|
|
if (!iwl_mvm_is_tt_in_fw(mvm)) {
|
|
mvm->tz_device.tzone = NULL;
|
|
|
|
return;
|
|
}
|
|
|
|
BUILD_BUG_ON(ARRAY_SIZE(name) >= THERMAL_NAME_LENGTH);
|
|
|
|
sprintf(name, "iwlwifi_%u", atomic_inc_return(&counter) & 0xFF);
|
|
mvm->tz_device.tzone = thermal_zone_device_register_with_trips(name,
|
|
mvm->tz_device.trips,
|
|
IWL_MAX_DTS_TRIPS,
|
|
IWL_WRITABLE_TRIPS_MSK,
|
|
mvm, &tzone_ops,
|
|
NULL, 0, 0);
|
|
if (IS_ERR(mvm->tz_device.tzone)) {
|
|
IWL_DEBUG_TEMP(mvm,
|
|
"Failed to register to thermal zone (err = %ld)\n",
|
|
PTR_ERR(mvm->tz_device.tzone));
|
|
mvm->tz_device.tzone = NULL;
|
|
return;
|
|
}
|
|
|
|
ret = thermal_zone_device_enable(mvm->tz_device.tzone);
|
|
if (ret) {
|
|
IWL_DEBUG_TEMP(mvm, "Failed to enable thermal zone\n");
|
|
thermal_zone_device_unregister(mvm->tz_device.tzone);
|
|
return;
|
|
}
|
|
|
|
/* 0 is a valid temperature,
|
|
* so initialize the array with S16_MIN which invalid temperature
|
|
*/
|
|
for (i = 0 ; i < IWL_MAX_DTS_TRIPS; i++) {
|
|
mvm->tz_device.trips[i].temperature = INT_MIN;
|
|
mvm->tz_device.trips[i].type = THERMAL_TRIP_PASSIVE;
|
|
}
|
|
}
|
|
|
|
static int iwl_mvm_tcool_get_max_state(struct thermal_cooling_device *cdev,
|
|
unsigned long *state)
|
|
{
|
|
*state = ARRAY_SIZE(iwl_mvm_cdev_budgets) - 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int iwl_mvm_tcool_get_cur_state(struct thermal_cooling_device *cdev,
|
|
unsigned long *state)
|
|
{
|
|
struct iwl_mvm *mvm = (struct iwl_mvm *)(cdev->devdata);
|
|
|
|
*state = mvm->cooling_dev.cur_state;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int iwl_mvm_tcool_set_cur_state(struct thermal_cooling_device *cdev,
|
|
unsigned long new_state)
|
|
{
|
|
struct iwl_mvm *mvm = (struct iwl_mvm *)(cdev->devdata);
|
|
int ret;
|
|
|
|
mutex_lock(&mvm->mutex);
|
|
|
|
if (!iwl_mvm_firmware_running(mvm) ||
|
|
mvm->fwrt.cur_fw_img != IWL_UCODE_REGULAR) {
|
|
ret = -EIO;
|
|
goto unlock;
|
|
}
|
|
|
|
if (new_state >= ARRAY_SIZE(iwl_mvm_cdev_budgets)) {
|
|
ret = -EINVAL;
|
|
goto unlock;
|
|
}
|
|
|
|
ret = iwl_mvm_ctdp_command(mvm, CTDP_CMD_OPERATION_START,
|
|
new_state);
|
|
|
|
unlock:
|
|
mutex_unlock(&mvm->mutex);
|
|
return ret;
|
|
}
|
|
|
|
static const struct thermal_cooling_device_ops tcooling_ops = {
|
|
.get_max_state = iwl_mvm_tcool_get_max_state,
|
|
.get_cur_state = iwl_mvm_tcool_get_cur_state,
|
|
.set_cur_state = iwl_mvm_tcool_set_cur_state,
|
|
};
|
|
|
|
static void iwl_mvm_cooling_device_register(struct iwl_mvm *mvm)
|
|
{
|
|
char name[] = "iwlwifi";
|
|
|
|
if (!iwl_mvm_is_ctdp_supported(mvm))
|
|
return;
|
|
|
|
BUILD_BUG_ON(ARRAY_SIZE(name) >= THERMAL_NAME_LENGTH);
|
|
|
|
mvm->cooling_dev.cdev =
|
|
thermal_cooling_device_register(name,
|
|
mvm,
|
|
&tcooling_ops);
|
|
|
|
if (IS_ERR(mvm->cooling_dev.cdev)) {
|
|
IWL_DEBUG_TEMP(mvm,
|
|
"Failed to register to cooling device (err = %ld)\n",
|
|
PTR_ERR(mvm->cooling_dev.cdev));
|
|
mvm->cooling_dev.cdev = NULL;
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void iwl_mvm_thermal_zone_unregister(struct iwl_mvm *mvm)
|
|
{
|
|
if (!iwl_mvm_is_tt_in_fw(mvm) || !mvm->tz_device.tzone)
|
|
return;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "Thermal zone device unregister\n");
|
|
if (mvm->tz_device.tzone) {
|
|
thermal_zone_device_unregister(mvm->tz_device.tzone);
|
|
mvm->tz_device.tzone = NULL;
|
|
}
|
|
}
|
|
|
|
static void iwl_mvm_cooling_device_unregister(struct iwl_mvm *mvm)
|
|
{
|
|
if (!iwl_mvm_is_ctdp_supported(mvm) || !mvm->cooling_dev.cdev)
|
|
return;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "Cooling device unregister\n");
|
|
if (mvm->cooling_dev.cdev) {
|
|
thermal_cooling_device_unregister(mvm->cooling_dev.cdev);
|
|
mvm->cooling_dev.cdev = NULL;
|
|
}
|
|
}
|
|
#endif /* CONFIG_THERMAL */
|
|
|
|
void iwl_mvm_thermal_initialize(struct iwl_mvm *mvm, u32 min_backoff)
|
|
{
|
|
struct iwl_mvm_tt_mgmt *tt = &mvm->thermal_throttle;
|
|
|
|
IWL_DEBUG_TEMP(mvm, "Initialize Thermal Throttling\n");
|
|
|
|
if (mvm->cfg->thermal_params)
|
|
tt->params = *mvm->cfg->thermal_params;
|
|
else
|
|
tt->params = iwl_mvm_default_tt_params;
|
|
|
|
tt->throttle = false;
|
|
tt->dynamic_smps = false;
|
|
tt->min_backoff = min_backoff;
|
|
INIT_DELAYED_WORK(&tt->ct_kill_exit, check_exit_ctkill);
|
|
|
|
#ifdef CONFIG_THERMAL
|
|
iwl_mvm_cooling_device_register(mvm);
|
|
iwl_mvm_thermal_zone_register(mvm);
|
|
#endif
|
|
mvm->init_status |= IWL_MVM_INIT_STATUS_THERMAL_INIT_COMPLETE;
|
|
}
|
|
|
|
void iwl_mvm_thermal_exit(struct iwl_mvm *mvm)
|
|
{
|
|
if (!(mvm->init_status & IWL_MVM_INIT_STATUS_THERMAL_INIT_COMPLETE))
|
|
return;
|
|
|
|
cancel_delayed_work_sync(&mvm->thermal_throttle.ct_kill_exit);
|
|
IWL_DEBUG_TEMP(mvm, "Exit Thermal Throttling\n");
|
|
|
|
#ifdef CONFIG_THERMAL
|
|
iwl_mvm_cooling_device_unregister(mvm);
|
|
iwl_mvm_thermal_zone_unregister(mvm);
|
|
#endif
|
|
mvm->init_status &= ~IWL_MVM_INIT_STATUS_THERMAL_INIT_COMPLETE;
|
|
}
|