mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-03 19:55:31 +00:00
wireless-next patches for v6.4
Most likely the last -next pull request for v6.4. We have changes all over. rtw88 now supports SDIO bus and iwlwifi continues to work on Wi-Fi 7 support. Not much stack changes this time. Major changes: cfg80211/mac80211 * fix some Fine Time Measurement (FTM) frames not being bufferable * flush frames before key removal to avoid potential unencrypted transmission depending on the hardware design iwlwifi * preparation for Wi-Fi 7 EHT and multi-link support rtw88 * SDIO bus support * RTL8822BS, RTL8822CS and RTL8821CS SDIO chipset support rtw89 * framework firmware backwards compatibility brcmfmac * Cypress 43439 SDIO support mt76 * mt7921 P2P support * mt7996 mesh A-MSDU support * mt7996 EHT support * mt7996 coredump support wcn36xx * support for pronto v3 hardware ath11k * PCIe DeviceTree bindings * WCN6750: enable SAR support ath10k * convert DeviceTree bindings to YAML -----BEGIN PGP SIGNATURE----- iQFFBAABCgAvFiEEiBjanGPFTz4PRfLobhckVSbrbZsFAmRCaTURHGt2YWxvQGtl cm5lbC5vcmcACgkQbhckVSbrbZvcRwf+NcLS4HbmqGZhBxl2LZVZ6AFCBM4ijDlO pxdMiC4UxT+UApY1/9YXo0VS97M7paDJH+R/g1HcTvvKURHCmsdhYHm+R1MH+/uD r8RfvJg4VtNnlUpsJh9jxt+e697KP15M7DF0sFlQzdIoTUl13Hp7YhI76zunAbAN u1FBcVVJiCcJWbLolMzqAeBMUWUEG+GtHF6Zn5kChVU/p1nmwJMPUG3Qvb61a7Yc BM1pQX8jQ8PBj+VrGPGvqX0BOdbxq0evauYScq2oTOhQ1fzTNWOsI1yI7AwApptR itwQ2t1UK/C/EWpvWIBSd0nit1uwSx0Zsu/nSZlbKbrvIFwd5XnfwQ== =Irrd -----END PGP SIGNATURE----- Merge tag 'wireless-next-2023-04-21' of git://git.kernel.org/pub/scm/linux/kernel/git/wireless/wireless-next Kalle Valo says: ==================== wireless-next patches for v6.4 Most likely the last -next pull request for v6.4. We have changes all over. rtw88 now supports SDIO bus and iwlwifi continues to work on Wi-Fi 7 support. Not much stack changes this time. Major changes: cfg80211/mac80211 - fix some Fine Time Measurement (FTM) frames not being bufferable - flush frames before key removal to avoid potential unencrypted transmission depending on the hardware design iwlwifi - preparation for Wi-Fi 7 EHT and multi-link support rtw88 - SDIO bus support - RTL8822BS, RTL8822CS and RTL8821CS SDIO chipset support rtw89 - framework firmware backwards compatibility brcmfmac - Cypress 43439 SDIO support mt76 - mt7921 P2P support - mt7996 mesh A-MSDU support - mt7996 EHT support - mt7996 coredump support wcn36xx - support for pronto v3 hardware ath11k - PCIe DeviceTree bindings - WCN6750: enable SAR support ath10k - convert DeviceTree bindings to YAML * tag 'wireless-next-2023-04-21' of git://git.kernel.org/pub/scm/linux/kernel/git/wireless/wireless-next: (261 commits) wifi: rtw88: Update spelling in main.h wifi: airo: remove ISA_DMA_API dependency wifi: rtl8xxxu: Simplify setting the initial gain wifi: rtl8xxxu: Add rtl8xxxu_write{8,16,32}_{set,clear} wifi: rtl8xxxu: Don't print the vendor/product/serial wifi: rtw88: Fix memory leak in rtw88_usb wifi: rtw88: call rtw8821c_switch_rf_set() according to chip variant wifi: rtw88: set pkg_type correctly for specific rtw8821c variants wifi: rtw88: rtw8821c: Fix rfe_option field width wifi: rtw88: usb: fix priority queue to endpoint mapping wifi: rtw88: 8822c: add iface combination wifi: rtw88: handle station mode concurrent scan with AP mode wifi: rtw88: prevent scan abort with other VIFs wifi: rtw88: refine reserved page flow for AP mode wifi: rtw88: disallow PS during AP mode wifi: rtw88: 8822c: extend reserved page number wifi: rtw88: add port switch for AP mode wifi: rtw88: add bitmap for dynamic port settings wifi: rtw89: mac: use regular int as return type of DLE buffer request wifi: mac80211: remove return value check of debugfs_create_dir() ... ==================== Link: https://lore.kernel.org/r/20230421104726.800BCC433D2@smtp.kernel.org Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
ca28896580
@ -111,6 +111,11 @@ properties:
|
||||
$ref: /schemas/leds/common.yaml#
|
||||
additionalProperties: false
|
||||
properties:
|
||||
led-active-low:
|
||||
description:
|
||||
LED is enabled with ground signal.
|
||||
type: boolean
|
||||
|
||||
led-sources:
|
||||
maxItems: 1
|
||||
|
||||
|
@ -1,215 +0,0 @@
|
||||
* Qualcomm Atheros ath10k wireless devices
|
||||
|
||||
Required properties:
|
||||
- compatible: Should be one of the following:
|
||||
* "qcom,ath10k"
|
||||
* "qcom,ipq4019-wifi"
|
||||
* "qcom,wcn3990-wifi"
|
||||
|
||||
PCI based devices uses compatible string "qcom,ath10k" and takes calibration
|
||||
data along with board specific data via "qcom,ath10k-calibration-data".
|
||||
Rest of the properties are not applicable for PCI based devices.
|
||||
|
||||
AHB based devices (i.e. ipq4019) uses compatible string "qcom,ipq4019-wifi"
|
||||
and also uses most of the properties defined in this doc (except
|
||||
"qcom,ath10k-calibration-data"). It uses "qcom,ath10k-pre-calibration-data"
|
||||
to carry pre calibration data.
|
||||
|
||||
In general, entry "qcom,ath10k-pre-calibration-data" and
|
||||
"qcom,ath10k-calibration-data" conflict with each other and only one
|
||||
can be provided per device.
|
||||
|
||||
SNOC based devices (i.e. wcn3990) uses compatible string "qcom,wcn3990-wifi".
|
||||
|
||||
- reg: Address and length of the register set for the device.
|
||||
- reg-names: Must include the list of following reg names,
|
||||
"membase"
|
||||
- interrupts: reference to the list of 17 interrupt numbers for "qcom,ipq4019-wifi"
|
||||
compatible target.
|
||||
reference to the list of 12 interrupt numbers for "qcom,wcn3990-wifi"
|
||||
compatible target.
|
||||
Must contain interrupt-names property per entry for
|
||||
"qcom,ath10k", "qcom,ipq4019-wifi" compatible targets.
|
||||
|
||||
- interrupt-names: Must include the entries for MSI interrupt
|
||||
names ("msi0" to "msi15") and legacy interrupt
|
||||
name ("legacy") for "qcom,ath10k", "qcom,ipq4019-wifi"
|
||||
compatible targets.
|
||||
|
||||
Optional properties:
|
||||
- resets: Must contain an entry for each entry in reset-names.
|
||||
See ../reset/reseti.txt for details.
|
||||
- reset-names: Must include the list of following reset names,
|
||||
"wifi_cpu_init"
|
||||
"wifi_radio_srif"
|
||||
"wifi_radio_warm"
|
||||
"wifi_radio_cold"
|
||||
"wifi_core_warm"
|
||||
"wifi_core_cold"
|
||||
- clocks: List of clock specifiers, must contain an entry for each required
|
||||
entry in clock-names.
|
||||
- clock-names: Should contain the clock names "wifi_wcss_cmd", "wifi_wcss_ref",
|
||||
"wifi_wcss_rtc" for "qcom,ipq4019-wifi" compatible target and
|
||||
"cxo_ref_clk_pin" and optionally "qdss" for "qcom,wcn3990-wifi"
|
||||
compatible target.
|
||||
- qcom,msi_addr: MSI interrupt address.
|
||||
- qcom,msi_base: Base value to add before writing MSI data into
|
||||
MSI address register.
|
||||
- qcom,ath10k-calibration-variant: string to search for in the board-2.bin
|
||||
variant list with the same bus and device
|
||||
specific ids
|
||||
- qcom,ath10k-calibration-data : calibration data + board specific data
|
||||
as an array, the length can vary between
|
||||
hw versions.
|
||||
- qcom,ath10k-pre-calibration-data : pre calibration data as an array,
|
||||
the length can vary between hw versions.
|
||||
- <supply-name>-supply: handle to the regulator device tree node
|
||||
optional "supply-name" are "vdd-0.8-cx-mx",
|
||||
"vdd-1.8-xo", "vdd-1.3-rfa", "vdd-3.3-ch0",
|
||||
and "vdd-3.3-ch1".
|
||||
- memory-region:
|
||||
Usage: optional
|
||||
Value type: <phandle>
|
||||
Definition: reference to the reserved-memory for the msa region
|
||||
used by the wifi firmware running in Q6.
|
||||
- iommus:
|
||||
Usage: optional
|
||||
Value type: <prop-encoded-array>
|
||||
Definition: A list of phandle and IOMMU specifier pairs.
|
||||
- ext-fem-name:
|
||||
Usage: Optional
|
||||
Value type: string
|
||||
Definition: Name of external front end module used. Some valid FEM names
|
||||
for example: "microsemi-lx5586", "sky85703-11"
|
||||
and "sky85803" etc.
|
||||
- qcom,snoc-host-cap-8bit-quirk:
|
||||
Usage: Optional
|
||||
Value type: <empty>
|
||||
Definition: Quirk specifying that the firmware expects the 8bit version
|
||||
of the host capability QMI request
|
||||
- qcom,xo-cal-data: xo cal offset to be configured in xo trim register.
|
||||
|
||||
- qcom,msa-fixed-perm: Boolean context flag to disable SCM call for statically
|
||||
mapped msa region.
|
||||
|
||||
- qcom,coexist-support : should contain eithr "0" or "1" to indicate coex
|
||||
support by the hardware.
|
||||
- qcom,coexist-gpio-pin : gpio pin number information to support coex
|
||||
which will be used by wifi firmware.
|
||||
|
||||
* Subnodes
|
||||
The ath10k wifi node can contain one optional firmware subnode.
|
||||
Firmware subnode is needed when the platform does not have TustZone.
|
||||
The firmware subnode must have:
|
||||
|
||||
- iommus:
|
||||
Usage: required
|
||||
Value type: <prop-encoded-array>
|
||||
Definition: A list of phandle and IOMMU specifier pairs.
|
||||
|
||||
|
||||
Example (to supply PCI based wifi block details):
|
||||
|
||||
In this example, the node is defined as child node of the PCI controller.
|
||||
|
||||
pci {
|
||||
pcie@0 {
|
||||
reg = <0 0 0 0 0>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
device_type = "pci";
|
||||
|
||||
wifi@0,0 {
|
||||
reg = <0 0 0 0 0>;
|
||||
qcom,ath10k-calibration-data = [ 01 02 03 ... ];
|
||||
ext-fem-name = "microsemi-lx5586";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
Example (to supply ipq4019 SoC wifi block details):
|
||||
|
||||
wifi0: wifi@a000000 {
|
||||
compatible = "qcom,ipq4019-wifi";
|
||||
reg = <0xa000000 0x200000>;
|
||||
resets = <&gcc WIFI0_CPU_INIT_RESET>,
|
||||
<&gcc WIFI0_RADIO_SRIF_RESET>,
|
||||
<&gcc WIFI0_RADIO_WARM_RESET>,
|
||||
<&gcc WIFI0_RADIO_COLD_RESET>,
|
||||
<&gcc WIFI0_CORE_WARM_RESET>,
|
||||
<&gcc WIFI0_CORE_COLD_RESET>;
|
||||
reset-names = "wifi_cpu_init",
|
||||
"wifi_radio_srif",
|
||||
"wifi_radio_warm",
|
||||
"wifi_radio_cold",
|
||||
"wifi_core_warm",
|
||||
"wifi_core_cold";
|
||||
clocks = <&gcc GCC_WCSS2G_CLK>,
|
||||
<&gcc GCC_WCSS2G_REF_CLK>,
|
||||
<&gcc GCC_WCSS2G_RTC_CLK>;
|
||||
clock-names = "wifi_wcss_cmd",
|
||||
"wifi_wcss_ref",
|
||||
"wifi_wcss_rtc";
|
||||
interrupts = <0 0x20 0x1>,
|
||||
<0 0x21 0x1>,
|
||||
<0 0x22 0x1>,
|
||||
<0 0x23 0x1>,
|
||||
<0 0x24 0x1>,
|
||||
<0 0x25 0x1>,
|
||||
<0 0x26 0x1>,
|
||||
<0 0x27 0x1>,
|
||||
<0 0x28 0x1>,
|
||||
<0 0x29 0x1>,
|
||||
<0 0x2a 0x1>,
|
||||
<0 0x2b 0x1>,
|
||||
<0 0x2c 0x1>,
|
||||
<0 0x2d 0x1>,
|
||||
<0 0x2e 0x1>,
|
||||
<0 0x2f 0x1>,
|
||||
<0 0xa8 0x0>;
|
||||
interrupt-names = "msi0", "msi1", "msi2", "msi3",
|
||||
"msi4", "msi5", "msi6", "msi7",
|
||||
"msi8", "msi9", "msi10", "msi11",
|
||||
"msi12", "msi13", "msi14", "msi15",
|
||||
"legacy";
|
||||
qcom,msi_addr = <0x0b006040>;
|
||||
qcom,msi_base = <0x40>;
|
||||
qcom,ath10k-pre-calibration-data = [ 01 02 03 ... ];
|
||||
qcom,coexist-support = <1>;
|
||||
qcom,coexist-gpio-pin = <0x33>;
|
||||
};
|
||||
|
||||
Example (to supply wcn3990 SoC wifi block details):
|
||||
|
||||
wifi@18000000 {
|
||||
compatible = "qcom,wcn3990-wifi";
|
||||
reg = <0x18800000 0x800000>;
|
||||
reg-names = "membase";
|
||||
clocks = <&clock_gcc clk_rf_clk2_pin>;
|
||||
clock-names = "cxo_ref_clk_pin";
|
||||
interrupts =
|
||||
<GIC_SPI 414 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 415 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 416 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 417 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 418 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 419 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 420 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 421 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 422 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 423 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 424 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 425 IRQ_TYPE_LEVEL_HIGH>;
|
||||
vdd-0.8-cx-mx-supply = <&pm8998_l5>;
|
||||
vdd-1.8-xo-supply = <&vreg_l7a_1p8>;
|
||||
vdd-1.3-rfa-supply = <&vreg_l17a_1p3>;
|
||||
vdd-3.3-ch0-supply = <&vreg_l25a_3p3>;
|
||||
vdd-3.3-ch1-supply = <&vreg_l26a_3p3>;
|
||||
memory-region = <&wifi_msa_mem>;
|
||||
iommus = <&apps_smmu 0x0040 0x1>;
|
||||
qcom,msa-fixed-perm;
|
||||
wifi-firmware {
|
||||
iommus = <&apps_iommu 0xc22 0x1>;
|
||||
};
|
||||
};
|
358
Documentation/devicetree/bindings/net/wireless/qcom,ath10k.yaml
Normal file
358
Documentation/devicetree/bindings/net/wireless/qcom,ath10k.yaml
Normal file
@ -0,0 +1,358 @@
|
||||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/net/wireless/qcom,ath10k.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Qualcomm Technologies ath10k wireless devices
|
||||
|
||||
maintainers:
|
||||
- Kalle Valo <kvalo@kernel.org>
|
||||
|
||||
description:
|
||||
Qualcomm Technologies, Inc. IEEE 802.11ac devices.
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- qcom,ath10k # SDIO-based devices
|
||||
- qcom,ipq4019-wifi
|
||||
- qcom,wcn3990-wifi # SNoC-based devices
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
reg-names:
|
||||
items:
|
||||
- const: membase
|
||||
|
||||
interrupts:
|
||||
minItems: 12
|
||||
maxItems: 17
|
||||
|
||||
interrupt-names:
|
||||
minItems: 12
|
||||
maxItems: 17
|
||||
|
||||
memory-region:
|
||||
maxItems: 1
|
||||
description:
|
||||
Reference to the MSA memory region used by the Wi-Fi firmware
|
||||
running on the Q6 core.
|
||||
|
||||
iommus:
|
||||
minItems: 1
|
||||
maxItems: 2
|
||||
|
||||
clocks:
|
||||
minItems: 1
|
||||
maxItems: 3
|
||||
|
||||
clock-names:
|
||||
minItems: 1
|
||||
maxItems: 3
|
||||
|
||||
resets:
|
||||
maxItems: 6
|
||||
|
||||
reset-names:
|
||||
items:
|
||||
- const: wifi_cpu_init
|
||||
- const: wifi_radio_srif
|
||||
- const: wifi_radio_warm
|
||||
- const: wifi_radio_cold
|
||||
- const: wifi_core_warm
|
||||
- const: wifi_core_cold
|
||||
|
||||
ext-fem-name:
|
||||
$ref: /schemas/types.yaml#/definitions/string
|
||||
description: Name of external front end module used.
|
||||
enum:
|
||||
- microsemi-lx5586
|
||||
- sky85703-11
|
||||
- sky85803
|
||||
|
||||
wifi-firmware:
|
||||
type: object
|
||||
additionalProperties: false
|
||||
description: |
|
||||
The ath10k Wi-Fi node can contain one optional firmware subnode.
|
||||
Firmware subnode is needed when the platform does not have Trustzone.
|
||||
properties:
|
||||
iommus:
|
||||
maxItems: 1
|
||||
required:
|
||||
- iommus
|
||||
|
||||
qcom,ath10k-calibration-data:
|
||||
$ref: /schemas/types.yaml#/definitions/uint8-array
|
||||
description:
|
||||
Calibration data + board-specific data as a byte array. The length
|
||||
can vary between hardware versions.
|
||||
|
||||
qcom,ath10k-calibration-variant:
|
||||
$ref: /schemas/types.yaml#/definitions/string
|
||||
description:
|
||||
Unique variant identifier of the calibration data in board-2.bin
|
||||
for designs with colliding bus and device specific ids
|
||||
|
||||
qcom,ath10k-pre-calibration-data:
|
||||
$ref: /schemas/types.yaml#/definitions/uint8-array
|
||||
description:
|
||||
Pre-calibration data as a byte array. The length can vary between
|
||||
hardware versions.
|
||||
|
||||
qcom,coexist-support:
|
||||
$ref: /schemas/types.yaml#/definitions/uint8
|
||||
enum: [0, 1]
|
||||
description:
|
||||
Indicate coex support by the hardware.
|
||||
|
||||
qcom,coexist-gpio-pin:
|
||||
$ref: /schemas/types.yaml#/definitions/uint32
|
||||
description:
|
||||
COEX GPIO number provided to the Wi-Fi firmware.
|
||||
|
||||
qcom,msa-fixed-perm:
|
||||
type: boolean
|
||||
description:
|
||||
Whether to skip executing an SCM call that reassigns the memory
|
||||
region ownership.
|
||||
|
||||
qcom,smem-states:
|
||||
$ref: /schemas/types.yaml#/definitions/phandle-array
|
||||
description: State bits used by the AP to signal the WLAN Q6.
|
||||
items:
|
||||
- description: Signal bits used to enable/disable low power mode
|
||||
on WCN in the case of WoW (Wake on Wireless).
|
||||
|
||||
qcom,smem-state-names:
|
||||
description: The names of the state bits used for SMP2P output.
|
||||
items:
|
||||
- const: wlan-smp2p-out
|
||||
|
||||
qcom,snoc-host-cap-8bit-quirk:
|
||||
type: boolean
|
||||
description:
|
||||
Quirk specifying that the firmware expects the 8bit version
|
||||
of the host capability QMI request
|
||||
|
||||
qcom,xo-cal-data:
|
||||
$ref: /schemas/types.yaml#/definitions/uint32
|
||||
description:
|
||||
XO cal offset to be configured in XO trim register.
|
||||
|
||||
vdd-0.8-cx-mx-supply:
|
||||
description: Main logic power rail
|
||||
|
||||
vdd-1.8-xo-supply:
|
||||
description: Crystal oscillator supply
|
||||
|
||||
vdd-1.3-rfa-supply:
|
||||
description: RFA supply
|
||||
|
||||
vdd-3.3-ch0-supply:
|
||||
description: Primary Wi-Fi antenna supply
|
||||
|
||||
vdd-3.3-ch1-supply:
|
||||
description: Secondary Wi-Fi antenna supply
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
allOf:
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,ipq4019-wifi
|
||||
then:
|
||||
properties:
|
||||
interrupts:
|
||||
minItems: 17
|
||||
maxItems: 17
|
||||
|
||||
interrupt-names:
|
||||
items:
|
||||
- const: msi0
|
||||
- const: msi1
|
||||
- const: msi2
|
||||
- const: msi3
|
||||
- const: msi4
|
||||
- const: msi5
|
||||
- const: msi6
|
||||
- const: msi7
|
||||
- const: msi8
|
||||
- const: msi9
|
||||
- const: msi10
|
||||
- const: msi11
|
||||
- const: msi12
|
||||
- const: msi13
|
||||
- const: msi14
|
||||
- const: msi15
|
||||
- const: legacy
|
||||
|
||||
clocks:
|
||||
items:
|
||||
- description: Wi-Fi command clock
|
||||
- description: Wi-Fi reference clock
|
||||
- description: Wi-Fi RTC clock
|
||||
|
||||
clock-names:
|
||||
items:
|
||||
- const: wifi_wcss_cmd
|
||||
- const: wifi_wcss_ref
|
||||
- const: wifi_wcss_rtc
|
||||
|
||||
required:
|
||||
- clocks
|
||||
- clock-names
|
||||
- interrupts
|
||||
- interrupt-names
|
||||
- resets
|
||||
- reset-names
|
||||
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,wcn3990-wifi
|
||||
|
||||
then:
|
||||
properties:
|
||||
clocks:
|
||||
minItems: 1
|
||||
items:
|
||||
- description: XO reference clock
|
||||
- description: Qualcomm Debug Subsystem clock
|
||||
|
||||
clock-names:
|
||||
minItems: 1
|
||||
items:
|
||||
- const: cxo_ref_clk_pin
|
||||
- const: qdss
|
||||
|
||||
interrupts:
|
||||
items:
|
||||
- description: CE0
|
||||
- description: CE1
|
||||
- description: CE2
|
||||
- description: CE3
|
||||
- description: CE4
|
||||
- description: CE5
|
||||
- description: CE6
|
||||
- description: CE7
|
||||
- description: CE8
|
||||
- description: CE9
|
||||
- description: CE10
|
||||
- description: CE11
|
||||
|
||||
interrupt-names: false
|
||||
|
||||
required:
|
||||
- interrupts
|
||||
|
||||
examples:
|
||||
# SNoC
|
||||
- |
|
||||
#include <dt-bindings/clock/qcom,rpmcc.h>
|
||||
#include <dt-bindings/interrupt-controller/arm-gic.h>
|
||||
|
||||
wifi@18800000 {
|
||||
compatible = "qcom,wcn3990-wifi";
|
||||
reg = <0x18800000 0x800000>;
|
||||
reg-names = "membase";
|
||||
memory-region = <&wlan_msa_mem>;
|
||||
clocks = <&rpmcc RPM_SMD_RF_CLK2_PIN>;
|
||||
clock-names = "cxo_ref_clk_pin";
|
||||
interrupts = <GIC_SPI 413 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 414 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 415 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 416 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 417 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 418 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 420 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 421 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 422 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 423 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 424 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<GIC_SPI 425 IRQ_TYPE_LEVEL_HIGH>;
|
||||
iommus = <&anoc2_smmu 0x1900>,
|
||||
<&anoc2_smmu 0x1901>;
|
||||
qcom,snoc-host-cap-8bit-quirk;
|
||||
vdd-0.8-cx-mx-supply = <&vreg_l5a_0p8>;
|
||||
vdd-1.8-xo-supply = <&vreg_l7a_1p8>;
|
||||
vdd-1.3-rfa-supply = <&vreg_l17a_1p3>;
|
||||
vdd-3.3-ch0-supply = <&vreg_l25a_3p3>;
|
||||
vdd-3.3-ch1-supply = <&vreg_l23a_3p3>;
|
||||
|
||||
wifi-firmware {
|
||||
iommus = <&apps_smmu 0x1c02 0x1>;
|
||||
};
|
||||
};
|
||||
|
||||
# AHB
|
||||
- |
|
||||
#include <dt-bindings/clock/qcom,gcc-ipq4019.h>
|
||||
|
||||
wifi@a000000 {
|
||||
compatible = "qcom,ipq4019-wifi";
|
||||
reg = <0xa000000 0x200000>;
|
||||
resets = <&gcc WIFI0_CPU_INIT_RESET>,
|
||||
<&gcc WIFI0_RADIO_SRIF_RESET>,
|
||||
<&gcc WIFI0_RADIO_WARM_RESET>,
|
||||
<&gcc WIFI0_RADIO_COLD_RESET>,
|
||||
<&gcc WIFI0_CORE_WARM_RESET>,
|
||||
<&gcc WIFI0_CORE_COLD_RESET>;
|
||||
reset-names = "wifi_cpu_init",
|
||||
"wifi_radio_srif",
|
||||
"wifi_radio_warm",
|
||||
"wifi_radio_cold",
|
||||
"wifi_core_warm",
|
||||
"wifi_core_cold";
|
||||
clocks = <&gcc GCC_WCSS2G_CLK>,
|
||||
<&gcc GCC_WCSS2G_REF_CLK>,
|
||||
<&gcc GCC_WCSS2G_RTC_CLK>;
|
||||
clock-names = "wifi_wcss_cmd",
|
||||
"wifi_wcss_ref",
|
||||
"wifi_wcss_rtc";
|
||||
interrupts = <GIC_SPI 32 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 33 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 34 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 35 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 36 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 37 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 38 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 39 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 40 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 41 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 42 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 43 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 44 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 45 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 46 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 47 IRQ_TYPE_EDGE_RISING>,
|
||||
<GIC_SPI 168 IRQ_TYPE_LEVEL_HIGH>;
|
||||
interrupt-names = "msi0",
|
||||
"msi1",
|
||||
"msi2",
|
||||
"msi3",
|
||||
"msi4",
|
||||
"msi5",
|
||||
"msi6",
|
||||
"msi7",
|
||||
"msi8",
|
||||
"msi9",
|
||||
"msi10",
|
||||
"msi11",
|
||||
"msi12",
|
||||
"msi13",
|
||||
"msi14",
|
||||
"msi15",
|
||||
"legacy";
|
||||
};
|
@ -0,0 +1,58 @@
|
||||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
# Copyright (c) 2023 Linaro Limited
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/net/wireless/qcom,ath11k-pci.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Qualcomm Technologies ath11k wireless devices (PCIe)
|
||||
|
||||
maintainers:
|
||||
- Kalle Valo <kvalo@kernel.org>
|
||||
|
||||
description: |
|
||||
Qualcomm Technologies IEEE 802.11ax PCIe devices
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- pci17cb,1103 # WCN6855
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
qcom,ath11k-calibration-variant:
|
||||
$ref: /schemas/types.yaml#/definitions/string
|
||||
description: |
|
||||
string to uniquely identify variant of the calibration data for designs
|
||||
with colliding bus and device ids
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
pcie {
|
||||
#address-cells = <3>;
|
||||
#size-cells = <2>;
|
||||
|
||||
pcie@0 {
|
||||
device_type = "pci";
|
||||
reg = <0x0 0x0 0x0 0x0 0x0>;
|
||||
#address-cells = <3>;
|
||||
#size-cells = <2>;
|
||||
ranges;
|
||||
|
||||
bus-range = <0x01 0xff>;
|
||||
|
||||
wifi@0 {
|
||||
compatible = "pci17cb,1103";
|
||||
reg = <0x10000 0x0 0x0 0x0 0x0>;
|
||||
|
||||
qcom,ath11k-calibration-variant = "LE_X13S";
|
||||
};
|
||||
};
|
||||
};
|
@ -17241,7 +17241,7 @@ S: Supported
|
||||
W: https://wireless.wiki.kernel.org/en/users/Drivers/ath10k
|
||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/kvalo/ath.git
|
||||
F: drivers/net/wireless/ath/ath10k/
|
||||
F: Documentation/devicetree/bindings/net/wireless/qcom,ath10k.txt
|
||||
F: Documentation/devicetree/bindings/net/wireless/qcom,ath10k.yaml
|
||||
|
||||
QUALCOMM ATHEROS ATH11K WIRELESS DRIVER
|
||||
M: Kalle Valo <kvalo@kernel.org>
|
||||
|
@ -14,6 +14,7 @@
|
||||
#include <linux/slab.h>
|
||||
#include <linux/of_address.h>
|
||||
#include <linux/of_irq.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/of_platform.h>
|
||||
|
||||
MODULE_DESCRIPTION("Broadcom's specific AMBA driver");
|
||||
|
@ -84,13 +84,6 @@ ath10k_set_ring_byte(unsigned int offset,
|
||||
return ((offset << addr_map->lsb) & addr_map->mask);
|
||||
}
|
||||
|
||||
static inline unsigned int
|
||||
ath10k_get_ring_byte(unsigned int offset,
|
||||
struct ath10k_hw_ce_regs_addr_map *addr_map)
|
||||
{
|
||||
return ((offset & addr_map->mask) >> (addr_map->lsb));
|
||||
}
|
||||
|
||||
static inline u32 ath10k_ce_read32(struct ath10k *ar, u32 offset)
|
||||
{
|
||||
struct ath10k_ce *ce = ath10k_ce_priv(ar);
|
||||
|
@ -3406,15 +3406,12 @@ static int ath10k_pci_claim(struct ath10k *ar)
|
||||
if (!ar_pci->mem) {
|
||||
ath10k_err(ar, "failed to iomap BAR%d\n", BAR_NUM);
|
||||
ret = -EIO;
|
||||
goto err_master;
|
||||
goto err_region;
|
||||
}
|
||||
|
||||
ath10k_dbg(ar, ATH10K_DBG_BOOT, "boot pci_mem 0x%pK\n", ar_pci->mem);
|
||||
return 0;
|
||||
|
||||
err_master:
|
||||
pci_clear_master(pdev);
|
||||
|
||||
err_region:
|
||||
pci_release_region(pdev, BAR_NUM);
|
||||
|
||||
@ -3431,7 +3428,6 @@ static void ath10k_pci_release(struct ath10k *ar)
|
||||
|
||||
pci_iounmap(pdev, ar_pci->mem);
|
||||
pci_release_region(pdev, BAR_NUM);
|
||||
pci_clear_master(pdev);
|
||||
pci_disable_device(pdev);
|
||||
}
|
||||
|
||||
|
@ -1078,6 +1078,12 @@ static int ath11k_ahb_fw_resource_deinit(struct ath11k_base *ab)
|
||||
struct iommu_domain *iommu;
|
||||
size_t unmapped_size;
|
||||
|
||||
/* Chipsets not requiring MSA would have not initialized
|
||||
* MSA resources, return success in such cases.
|
||||
*/
|
||||
if (!ab->hw_params.fixed_fw_mem)
|
||||
return 0;
|
||||
|
||||
if (ab_ahb->fw.use_tz)
|
||||
return 0;
|
||||
|
||||
|
@ -116,7 +116,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tcl_ring_retry = true,
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.ftm_responder = true,
|
||||
},
|
||||
{
|
||||
.hw_rev = ATH11K_HW_IPQ6018_HW10,
|
||||
@ -199,7 +198,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.support_fw_mac_sequence = false,
|
||||
.ftm_responder = true,
|
||||
},
|
||||
{
|
||||
.name = "qca6390 hw2.0",
|
||||
@ -284,7 +282,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.support_fw_mac_sequence = true,
|
||||
.ftm_responder = false,
|
||||
},
|
||||
{
|
||||
.name = "qcn9074 hw1.0",
|
||||
@ -366,7 +363,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.support_fw_mac_sequence = false,
|
||||
.ftm_responder = true,
|
||||
},
|
||||
{
|
||||
.name = "wcn6855 hw2.0",
|
||||
@ -451,7 +447,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.support_fw_mac_sequence = true,
|
||||
.ftm_responder = false,
|
||||
},
|
||||
{
|
||||
.name = "wcn6855 hw2.1",
|
||||
@ -534,7 +529,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.support_fw_mac_sequence = true,
|
||||
.ftm_responder = false,
|
||||
},
|
||||
{
|
||||
.name = "wcn6750 hw1.0",
|
||||
@ -599,7 +593,7 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.current_cc_support = true,
|
||||
.dbr_debug_support = false,
|
||||
.global_reset = false,
|
||||
.bios_sar_capa = NULL,
|
||||
.bios_sar_capa = &ath11k_hw_sar_capa_wcn6855,
|
||||
.m3_fw_support = false,
|
||||
.fixed_bdf_addr = false,
|
||||
.fixed_mem_region = false,
|
||||
@ -615,7 +609,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE_WCN6750,
|
||||
.smp2p_wow_exit = true,
|
||||
.support_fw_mac_sequence = true,
|
||||
.ftm_responder = false,
|
||||
},
|
||||
{
|
||||
.hw_rev = ATH11K_HW_IPQ5018_HW10,
|
||||
@ -695,7 +688,6 @@ static const struct ath11k_hw_params ath11k_hw_params[] = {
|
||||
.tx_ring_size = DP_TCL_DATA_RING_SIZE,
|
||||
.smp2p_wow_exit = false,
|
||||
.support_fw_mac_sequence = false,
|
||||
.ftm_responder = true,
|
||||
},
|
||||
};
|
||||
|
||||
|
@ -26,13 +26,13 @@ int ath11k_dbring_validate_buffer(struct ath11k *ar, void *buffer, u32 size)
|
||||
static void ath11k_dbring_fill_magic_value(struct ath11k *ar,
|
||||
void *buffer, u32 size)
|
||||
{
|
||||
u32 *temp;
|
||||
int idx;
|
||||
/* memset32 function fills buffer payload with the ATH11K_DB_MAGIC_VALUE
|
||||
* and the variable size is expected to be the number of u32 values
|
||||
* to be stored, not the number of bytes.
|
||||
*/
|
||||
size = size / sizeof(u32);
|
||||
|
||||
size = size >> 2;
|
||||
|
||||
for (idx = 0, temp = buffer; idx < size; idx++, temp++)
|
||||
*temp++ = ATH11K_DB_MAGIC_VALUE;
|
||||
memset32(buffer, ATH11K_DB_MAGIC_VALUE, size);
|
||||
}
|
||||
|
||||
static int ath11k_dbring_bufs_replenish(struct ath11k *ar,
|
||||
|
@ -143,7 +143,8 @@ enum htt_tx_pdev_underrun_enum {
|
||||
/* Bytes stored in little endian order */
|
||||
/* Length should be multiple of DWORD */
|
||||
struct htt_stats_string_tlv {
|
||||
u32 data[0]; /* Can be variable length */
|
||||
/* Can be variable length */
|
||||
DECLARE_FLEX_ARRAY(u32, data);
|
||||
} __packed;
|
||||
|
||||
#define HTT_STATS_MAC_ID GENMASK(7, 0)
|
||||
@ -205,27 +206,32 @@ struct htt_tx_pdev_stats_cmn_tlv {
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_pdev_stats_urrn_tlv_v {
|
||||
u32 urrn_stats[0]; /* HTT_TX_PDEV_MAX_URRN_STATS */
|
||||
/* HTT_TX_PDEV_MAX_URRN_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, urrn_stats);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_pdev_stats_flush_tlv_v {
|
||||
u32 flush_errs[0]; /* HTT_TX_PDEV_MAX_FLUSH_REASON_STATS */
|
||||
/* HTT_TX_PDEV_MAX_FLUSH_REASON_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, flush_errs);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_pdev_stats_sifs_tlv_v {
|
||||
u32 sifs_status[0]; /* HTT_TX_PDEV_MAX_SIFS_BURST_STATS */
|
||||
/* HTT_TX_PDEV_MAX_SIFS_BURST_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, sifs_status);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_pdev_stats_phy_err_tlv_v {
|
||||
u32 phy_errs[0]; /* HTT_TX_PDEV_MAX_PHY_ERR_STATS */
|
||||
/* HTT_TX_PDEV_MAX_PHY_ERR_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, phy_errs);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_pdev_stats_sifs_hist_tlv_v {
|
||||
u32 sifs_hist_status[0]; /* HTT_TX_PDEV_SIFS_BURST_HIST_STATS */
|
||||
/* HTT_TX_PDEV_SIFS_BURST_HIST_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, sifs_hist_status);
|
||||
};
|
||||
|
||||
struct htt_tx_pdev_stats_tx_ppdu_stats_tlv_v {
|
||||
@ -590,20 +596,20 @@ struct htt_tx_hwq_difs_latency_stats_tlv_v {
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_hwq_cmd_result_stats_tlv_v {
|
||||
/* Histogram of sched cmd result */
|
||||
u32 cmd_result[0]; /* HTT_TX_HWQ_MAX_CMD_RESULT_STATS */
|
||||
/* Histogram of sched cmd result, HTT_TX_HWQ_MAX_CMD_RESULT_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, cmd_result);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_hwq_cmd_stall_stats_tlv_v {
|
||||
/* Histogram of various pause conitions */
|
||||
u32 cmd_stall_status[0]; /* HTT_TX_HWQ_MAX_CMD_STALL_STATS */
|
||||
/* Histogram of various pause conitions, HTT_TX_HWQ_MAX_CMD_STALL_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, cmd_stall_status);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_hwq_fes_result_stats_tlv_v {
|
||||
/* Histogram of number of user fes result */
|
||||
u32 fes_result[0]; /* HTT_TX_HWQ_MAX_FES_RESULT_STATS */
|
||||
/* Histogram of number of user fes result, HTT_TX_HWQ_MAX_FES_RESULT_STATS */
|
||||
DECLARE_FLEX_ARRAY(u32, fes_result);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size
|
||||
@ -635,8 +641,8 @@ struct htt_tx_hwq_tried_mpdu_cnt_hist_tlv_v {
|
||||
* #define WAL_TXOP_USED_HISTOGRAM_INTERVAL 1000 ( 1 ms )
|
||||
*/
|
||||
struct htt_tx_hwq_txop_used_cnt_hist_tlv_v {
|
||||
/* Histogram of txop used cnt */
|
||||
u32 txop_used_cnt_hist[0]; /* HTT_TX_HWQ_TXOP_USED_CNT_HIST */
|
||||
/* Histogram of txop used cnt, HTT_TX_HWQ_TXOP_USED_CNT_HIST */
|
||||
DECLARE_FLEX_ARRAY(u32, txop_used_cnt_hist);
|
||||
};
|
||||
|
||||
/* == TX SELFGEN STATS == */
|
||||
@ -804,17 +810,20 @@ struct htt_tx_pdev_mpdu_stats_tlv {
|
||||
/* == TX SCHED STATS == */
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_sched_txq_cmd_posted_tlv_v {
|
||||
u32 sched_cmd_posted[0]; /* HTT_TX_PDEV_SCHED_TX_MODE_MAX */
|
||||
/* HTT_TX_PDEV_SCHED_TX_MODE_MAX */
|
||||
DECLARE_FLEX_ARRAY(u32, sched_cmd_posted);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_sched_txq_cmd_reaped_tlv_v {
|
||||
u32 sched_cmd_reaped[0]; /* HTT_TX_PDEV_SCHED_TX_MODE_MAX */
|
||||
/* HTT_TX_PDEV_SCHED_TX_MODE_MAX */
|
||||
DECLARE_FLEX_ARRAY(u32, sched_cmd_reaped);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_sched_txq_sched_order_su_tlv_v {
|
||||
u32 sched_order_su[0]; /* HTT_TX_PDEV_NUM_SCHED_ORDER_LOG */
|
||||
/* HTT_TX_PDEV_NUM_SCHED_ORDER_LOG */
|
||||
DECLARE_FLEX_ARRAY(u32, sched_order_su);
|
||||
};
|
||||
|
||||
enum htt_sched_txq_sched_ineligibility_tlv_enum {
|
||||
@ -842,7 +851,7 @@ enum htt_sched_txq_sched_ineligibility_tlv_enum {
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_sched_txq_sched_ineligibility_tlv_v {
|
||||
/* indexed by htt_sched_txq_sched_ineligibility_tlv_enum */
|
||||
u32 sched_ineligibility[0];
|
||||
DECLARE_FLEX_ARRAY(u32, sched_ineligibility);
|
||||
};
|
||||
|
||||
#define HTT_TX_PDEV_STATS_SCHED_PER_TXQ_MAC_ID GENMASK(7, 0)
|
||||
@ -888,18 +897,20 @@ struct htt_stats_tx_sched_cmn_tlv {
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_tqm_gen_mpdu_stats_tlv_v {
|
||||
u32 gen_mpdu_end_reason[0]; /* HTT_TX_TQM_MAX_GEN_MPDU_END_REASON */
|
||||
/* HTT_TX_TQM_MAX_GEN_MPDU_END_REASON */
|
||||
DECLARE_FLEX_ARRAY(u32, gen_mpdu_end_reason);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_tqm_list_mpdu_stats_tlv_v {
|
||||
u32 list_mpdu_end_reason[0]; /* HTT_TX_TQM_MAX_LIST_MPDU_END_REASON */
|
||||
/* HTT_TX_TQM_MAX_LIST_MPDU_END_REASON */
|
||||
DECLARE_FLEX_ARRAY(u32, list_mpdu_end_reason);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_tx_tqm_list_mpdu_cnt_tlv_v {
|
||||
u32 list_mpdu_cnt_hist[0];
|
||||
/* HTT_TX_TQM_MAX_LIST_MPDU_CNT_HISTOGRAM_BINS */
|
||||
/* HTT_TX_TQM_MAX_LIST_MPDU_CNT_HISTOGRAM_BINS */
|
||||
DECLARE_FLEX_ARRAY(u32, list_mpdu_cnt_hist);
|
||||
};
|
||||
|
||||
struct htt_tx_tqm_pdev_stats_tlv_v {
|
||||
@ -1098,7 +1109,7 @@ struct htt_tx_de_compl_stats_tlv {
|
||||
* ENTRIES_PER_BIN_COUNT)
|
||||
*/
|
||||
struct htt_tx_de_fw2wbm_ring_full_hist_tlv {
|
||||
u32 fw2wbm_ring_full_hist[0];
|
||||
DECLARE_FLEX_ARRAY(u32, fw2wbm_ring_full_hist);
|
||||
};
|
||||
|
||||
struct htt_tx_de_cmn_stats_tlv {
|
||||
@ -1151,7 +1162,7 @@ struct htt_ring_if_cmn_tlv {
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_sfm_client_user_tlv_v {
|
||||
/* Number of DWORDS used per user and per client */
|
||||
u32 dwords_used_by_user_n[0];
|
||||
DECLARE_FLEX_ARRAY(u32, dwords_used_by_user_n);
|
||||
};
|
||||
|
||||
struct htt_sfm_client_tlv {
|
||||
@ -1436,12 +1447,14 @@ struct htt_rx_soc_fw_stats_tlv {
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_rx_soc_fw_refill_ring_empty_tlv_v {
|
||||
u32 refill_ring_empty_cnt[0]; /* HTT_RX_STATS_REFILL_MAX_RING */
|
||||
/* HTT_RX_STATS_REFILL_MAX_RING */
|
||||
DECLARE_FLEX_ARRAY(u32, refill_ring_empty_cnt);
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_rx_soc_fw_refill_ring_num_refill_tlv_v {
|
||||
u32 refill_ring_num_refill[0]; /* HTT_RX_STATS_REFILL_MAX_RING */
|
||||
/* HTT_RX_STATS_REFILL_MAX_RING */
|
||||
DECLARE_FLEX_ARRAY(u32, refill_ring_num_refill);
|
||||
};
|
||||
|
||||
/* RXDMA error code from WBM released packets */
|
||||
@ -1473,7 +1486,7 @@ enum htt_rx_rxdma_error_code_enum {
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_rx_soc_fw_refill_ring_num_rxdma_err_tlv_v {
|
||||
u32 rxdma_err[0]; /* HTT_RX_RXDMA_MAX_ERR_CODE */
|
||||
DECLARE_FLEX_ARRAY(u32, rxdma_err); /* HTT_RX_RXDMA_MAX_ERR_CODE */
|
||||
};
|
||||
|
||||
/* REO error code from WBM released packets */
|
||||
@ -1505,7 +1518,7 @@ enum htt_rx_reo_error_code_enum {
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_rx_soc_fw_refill_ring_num_reo_err_tlv_v {
|
||||
u32 reo_err[0]; /* HTT_RX_REO_MAX_ERR_CODE */
|
||||
DECLARE_FLEX_ARRAY(u32, reo_err); /* HTT_RX_REO_MAX_ERR_CODE */
|
||||
};
|
||||
|
||||
/* == RX PDEV STATS == */
|
||||
@ -1622,13 +1635,13 @@ struct htt_rx_pdev_fw_stats_phy_err_tlv {
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_rx_pdev_fw_ring_mpdu_err_tlv_v {
|
||||
/* Num error MPDU for each RxDMA error type */
|
||||
u32 fw_ring_mpdu_err[0]; /* HTT_RX_STATS_RXDMA_MAX_ERR */
|
||||
DECLARE_FLEX_ARRAY(u32, fw_ring_mpdu_err); /* HTT_RX_STATS_RXDMA_MAX_ERR */
|
||||
};
|
||||
|
||||
/* NOTE: Variable length TLV, use length spec to infer array size */
|
||||
struct htt_rx_pdev_fw_mpdu_drop_tlv_v {
|
||||
/* Num MPDU dropped */
|
||||
u32 fw_mpdu_drop[0]; /* HTT_RX_STATS_FW_DROP_REASON_MAX */
|
||||
DECLARE_FLEX_ARRAY(u32, fw_mpdu_drop); /* HTT_RX_STATS_FW_DROP_REASON_MAX */
|
||||
};
|
||||
|
||||
#define HTT_PDEV_CCA_STATS_TX_FRAME_INFO_PRESENT (0x1)
|
||||
|
@ -36,6 +36,7 @@ void ath11k_dp_peer_cleanup(struct ath11k *ar, int vdev_id, const u8 *addr)
|
||||
}
|
||||
|
||||
ath11k_peer_rx_tid_cleanup(ar, peer);
|
||||
peer->dp_setup_done = false;
|
||||
crypto_free_shash(peer->tfm_mmic);
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
}
|
||||
@ -72,7 +73,8 @@ int ath11k_dp_peer_setup(struct ath11k *ar, int vdev_id, const u8 *addr)
|
||||
ret = ath11k_peer_rx_frag_setup(ar, addr, vdev_id);
|
||||
if (ret) {
|
||||
ath11k_warn(ab, "failed to setup rx defrag context\n");
|
||||
return ret;
|
||||
tid--;
|
||||
goto peer_clean;
|
||||
}
|
||||
|
||||
/* TODO: Setup other peer specific resource used in data path */
|
||||
|
@ -214,7 +214,7 @@ struct ath11k_pdev_dp {
|
||||
#define DP_REO_REINJECT_RING_SIZE 32
|
||||
#define DP_RX_RELEASE_RING_SIZE 1024
|
||||
#define DP_REO_EXCEPTION_RING_SIZE 128
|
||||
#define DP_REO_CMD_RING_SIZE 128
|
||||
#define DP_REO_CMD_RING_SIZE 256
|
||||
#define DP_REO_STATUS_RING_SIZE 2048
|
||||
#define DP_RXDMA_BUF_RING_SIZE 4096
|
||||
#define DP_RXDMA_REFILL_RING_SIZE 2048
|
||||
@ -303,12 +303,16 @@ struct ath11k_dp {
|
||||
|
||||
#define HTT_TX_WBM_COMP_STATUS_OFFSET 8
|
||||
|
||||
#define HTT_INVALID_PEER_ID 0xffff
|
||||
|
||||
/* HTT tx completion is overlaid in wbm_release_ring */
|
||||
#define HTT_TX_WBM_COMP_INFO0_STATUS GENMASK(12, 9)
|
||||
#define HTT_TX_WBM_COMP_INFO0_REINJECT_REASON GENMASK(16, 13)
|
||||
#define HTT_TX_WBM_COMP_INFO0_REINJECT_REASON GENMASK(16, 13)
|
||||
|
||||
#define HTT_TX_WBM_COMP_INFO1_ACK_RSSI GENMASK(31, 24)
|
||||
#define HTT_TX_WBM_COMP_INFO2_SW_PEER_ID GENMASK(15, 0)
|
||||
#define HTT_TX_WBM_COMP_INFO2_VALID BIT(21)
|
||||
|
||||
struct htt_tx_wbm_completion {
|
||||
u32 info0;
|
||||
|
@ -389,10 +389,10 @@ int ath11k_dp_rxbufs_replenish(struct ath11k_base *ab, int mac_id,
|
||||
goto fail_free_skb;
|
||||
|
||||
spin_lock_bh(&rx_ring->idr_lock);
|
||||
buf_id = idr_alloc(&rx_ring->bufs_idr, skb, 0,
|
||||
rx_ring->bufs_max * 3, GFP_ATOMIC);
|
||||
buf_id = idr_alloc(&rx_ring->bufs_idr, skb, 1,
|
||||
(rx_ring->bufs_max * 3) + 1, GFP_ATOMIC);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
if (buf_id < 0)
|
||||
if (buf_id <= 0)
|
||||
goto fail_dma_unmap;
|
||||
|
||||
desc = ath11k_hal_srng_src_get_next_entry(ab, srng);
|
||||
@ -435,7 +435,6 @@ int ath11k_dp_rxbufs_replenish(struct ath11k_base *ab, int mac_id,
|
||||
static int ath11k_dp_rxdma_buf_ring_free(struct ath11k *ar,
|
||||
struct dp_rxdma_ring *rx_ring)
|
||||
{
|
||||
struct ath11k_pdev_dp *dp = &ar->dp;
|
||||
struct sk_buff *skb;
|
||||
int buf_id;
|
||||
|
||||
@ -453,28 +452,6 @@ static int ath11k_dp_rxdma_buf_ring_free(struct ath11k *ar,
|
||||
idr_destroy(&rx_ring->bufs_idr);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
|
||||
/* if rxdma1_enable is false, mon_status_refill_ring
|
||||
* isn't setup, so don't clean.
|
||||
*/
|
||||
if (!ar->ab->hw_params.rxdma1_enable)
|
||||
return 0;
|
||||
|
||||
rx_ring = &dp->rx_mon_status_refill_ring[0];
|
||||
|
||||
spin_lock_bh(&rx_ring->idr_lock);
|
||||
idr_for_each_entry(&rx_ring->bufs_idr, skb, buf_id) {
|
||||
idr_remove(&rx_ring->bufs_idr, buf_id);
|
||||
/* XXX: Understand where internal driver does this dma_unmap
|
||||
* of rxdma_buffer.
|
||||
*/
|
||||
dma_unmap_single(ar->ab->dev, ATH11K_SKB_RXCB(skb)->paddr,
|
||||
skb->len + skb_tailroom(skb), DMA_BIDIRECTIONAL);
|
||||
dev_kfree_skb_any(skb);
|
||||
}
|
||||
|
||||
idr_destroy(&rx_ring->bufs_idr);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -691,13 +668,18 @@ void ath11k_dp_reo_cmd_list_cleanup(struct ath11k_base *ab)
|
||||
struct ath11k_dp *dp = &ab->dp;
|
||||
struct dp_reo_cmd *cmd, *tmp;
|
||||
struct dp_reo_cache_flush_elem *cmd_cache, *tmp_cache;
|
||||
struct dp_rx_tid *rx_tid;
|
||||
|
||||
spin_lock_bh(&dp->reo_cmd_lock);
|
||||
list_for_each_entry_safe(cmd, tmp, &dp->reo_cmd_list, list) {
|
||||
list_del(&cmd->list);
|
||||
dma_unmap_single(ab->dev, cmd->data.paddr,
|
||||
cmd->data.size, DMA_BIDIRECTIONAL);
|
||||
kfree(cmd->data.vaddr);
|
||||
rx_tid = &cmd->data;
|
||||
if (rx_tid->vaddr) {
|
||||
dma_unmap_single(ab->dev, rx_tid->paddr,
|
||||
rx_tid->size, DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
}
|
||||
kfree(cmd);
|
||||
}
|
||||
|
||||
@ -705,9 +687,13 @@ void ath11k_dp_reo_cmd_list_cleanup(struct ath11k_base *ab)
|
||||
&dp->reo_cmd_cache_flush_list, list) {
|
||||
list_del(&cmd_cache->list);
|
||||
dp->reo_cmd_cache_flush_count--;
|
||||
dma_unmap_single(ab->dev, cmd_cache->data.paddr,
|
||||
cmd_cache->data.size, DMA_BIDIRECTIONAL);
|
||||
kfree(cmd_cache->data.vaddr);
|
||||
rx_tid = &cmd_cache->data;
|
||||
if (rx_tid->vaddr) {
|
||||
dma_unmap_single(ab->dev, rx_tid->paddr,
|
||||
rx_tid->size, DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
}
|
||||
kfree(cmd_cache);
|
||||
}
|
||||
spin_unlock_bh(&dp->reo_cmd_lock);
|
||||
@ -721,10 +707,12 @@ static void ath11k_dp_reo_cmd_free(struct ath11k_dp *dp, void *ctx,
|
||||
if (status != HAL_REO_CMD_SUCCESS)
|
||||
ath11k_warn(dp->ab, "failed to flush rx tid hw desc, tid %d status %d\n",
|
||||
rx_tid->tid, status);
|
||||
|
||||
dma_unmap_single(dp->ab->dev, rx_tid->paddr, rx_tid->size,
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
if (rx_tid->vaddr) {
|
||||
dma_unmap_single(dp->ab->dev, rx_tid->paddr, rx_tid->size,
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
static void ath11k_dp_reo_cache_flush(struct ath11k_base *ab,
|
||||
@ -763,6 +751,7 @@ static void ath11k_dp_reo_cache_flush(struct ath11k_base *ab,
|
||||
dma_unmap_single(ab->dev, rx_tid->paddr, rx_tid->size,
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
@ -815,6 +804,7 @@ static void ath11k_dp_rx_tid_del_func(struct ath11k_dp *dp, void *ctx,
|
||||
dma_unmap_single(ab->dev, rx_tid->paddr, rx_tid->size,
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
}
|
||||
|
||||
void ath11k_peer_rx_tid_delete(struct ath11k *ar,
|
||||
@ -827,6 +817,8 @@ void ath11k_peer_rx_tid_delete(struct ath11k *ar,
|
||||
if (!rx_tid->active)
|
||||
return;
|
||||
|
||||
rx_tid->active = false;
|
||||
|
||||
cmd.flag = HAL_REO_CMD_FLG_NEED_STATUS;
|
||||
cmd.addr_lo = lower_32_bits(rx_tid->paddr);
|
||||
cmd.addr_hi = upper_32_bits(rx_tid->paddr);
|
||||
@ -841,9 +833,11 @@ void ath11k_peer_rx_tid_delete(struct ath11k *ar,
|
||||
dma_unmap_single(ar->ab->dev, rx_tid->paddr, rx_tid->size,
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
}
|
||||
|
||||
rx_tid->active = false;
|
||||
rx_tid->paddr = 0;
|
||||
rx_tid->size = 0;
|
||||
}
|
||||
|
||||
static int ath11k_dp_rx_link_desc_return(struct ath11k_base *ab,
|
||||
@ -990,6 +984,7 @@ static void ath11k_dp_rx_tid_mem_free(struct ath11k_base *ab,
|
||||
dma_unmap_single(ab->dev, rx_tid->paddr, rx_tid->size,
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
|
||||
rx_tid->active = false;
|
||||
|
||||
@ -1014,7 +1009,8 @@ int ath11k_peer_rx_tid_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id,
|
||||
|
||||
peer = ath11k_peer_find(ab, vdev_id, peer_mac);
|
||||
if (!peer) {
|
||||
ath11k_warn(ab, "failed to find the peer to set up rx tid\n");
|
||||
ath11k_warn(ab, "failed to find the peer %pM to set up rx tid\n",
|
||||
peer_mac);
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
return -ENOENT;
|
||||
}
|
||||
@ -1027,7 +1023,8 @@ int ath11k_peer_rx_tid_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id,
|
||||
ba_win_sz, ssn, true);
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
if (ret) {
|
||||
ath11k_warn(ab, "failed to update reo for rx tid %d\n", tid);
|
||||
ath11k_warn(ab, "failed to update reo for peer %pM rx tid %d\n: %d",
|
||||
peer_mac, tid, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@ -1035,8 +1032,8 @@ int ath11k_peer_rx_tid_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id,
|
||||
peer_mac, paddr,
|
||||
tid, 1, ba_win_sz);
|
||||
if (ret)
|
||||
ath11k_warn(ab, "failed to send wmi command to update rx reorder queue, tid :%d (%d)\n",
|
||||
tid, ret);
|
||||
ath11k_warn(ab, "failed to send wmi rx reorder queue for peer %pM tid %d: %d\n",
|
||||
peer_mac, tid, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@ -1069,6 +1066,8 @@ int ath11k_peer_rx_tid_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id,
|
||||
ret = dma_mapping_error(ab->dev, paddr);
|
||||
if (ret) {
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
ath11k_warn(ab, "failed to setup dma map for peer %pM rx tid %d: %d\n",
|
||||
peer_mac, tid, ret);
|
||||
goto err_mem_free;
|
||||
}
|
||||
|
||||
@ -1082,15 +1081,16 @@ int ath11k_peer_rx_tid_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id,
|
||||
ret = ath11k_wmi_peer_rx_reorder_queue_setup(ar, vdev_id, peer_mac,
|
||||
paddr, tid, 1, ba_win_sz);
|
||||
if (ret) {
|
||||
ath11k_warn(ar->ab, "failed to setup rx reorder queue, tid :%d (%d)\n",
|
||||
tid, ret);
|
||||
ath11k_warn(ar->ab, "failed to setup rx reorder queue for peer %pM tid %d: %d\n",
|
||||
peer_mac, tid, ret);
|
||||
ath11k_dp_rx_tid_mem_free(ab, peer_mac, vdev_id, tid);
|
||||
}
|
||||
|
||||
return ret;
|
||||
|
||||
err_mem_free:
|
||||
kfree(vaddr);
|
||||
kfree(rx_tid->vaddr);
|
||||
rx_tid->vaddr = NULL;
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -2665,6 +2665,9 @@ int ath11k_dp_process_rx(struct ath11k_base *ab, int ring_id,
|
||||
cookie);
|
||||
mac_id = FIELD_GET(DP_RXDMA_BUF_COOKIE_PDEV_ID, cookie);
|
||||
|
||||
if (unlikely(buf_id == 0))
|
||||
continue;
|
||||
|
||||
ar = ab->pdevs[mac_id].ar;
|
||||
rx_ring = &ar->dp.rx_refill_buf_ring;
|
||||
spin_lock_bh(&rx_ring->idr_lock);
|
||||
@ -3029,39 +3032,51 @@ static int ath11k_dp_rx_reap_mon_status_ring(struct ath11k_base *ab, int mac_id,
|
||||
|
||||
spin_lock_bh(&rx_ring->idr_lock);
|
||||
skb = idr_find(&rx_ring->bufs_idr, buf_id);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
|
||||
if (!skb) {
|
||||
ath11k_warn(ab, "rx monitor status with invalid buf_id %d\n",
|
||||
buf_id);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
pmon->buf_state = DP_MON_STATUS_REPLINISH;
|
||||
goto move_next;
|
||||
}
|
||||
|
||||
idr_remove(&rx_ring->bufs_idr, buf_id);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
|
||||
rxcb = ATH11K_SKB_RXCB(skb);
|
||||
|
||||
dma_unmap_single(ab->dev, rxcb->paddr,
|
||||
skb->len + skb_tailroom(skb),
|
||||
DMA_FROM_DEVICE);
|
||||
dma_sync_single_for_cpu(ab->dev, rxcb->paddr,
|
||||
skb->len + skb_tailroom(skb),
|
||||
DMA_FROM_DEVICE);
|
||||
|
||||
tlv = (struct hal_tlv_hdr *)skb->data;
|
||||
if (FIELD_GET(HAL_TLV_HDR_TAG, tlv->tl) !=
|
||||
HAL_RX_STATUS_BUFFER_DONE) {
|
||||
ath11k_warn(ab, "mon status DONE not set %lx\n",
|
||||
ath11k_warn(ab, "mon status DONE not set %lx, buf_id %d\n",
|
||||
FIELD_GET(HAL_TLV_HDR_TAG,
|
||||
tlv->tl));
|
||||
dev_kfree_skb_any(skb);
|
||||
tlv->tl), buf_id);
|
||||
/* If done status is missing, hold onto status
|
||||
* ring until status is done for this status
|
||||
* ring buffer.
|
||||
* Keep HP in mon_status_ring unchanged,
|
||||
* and break from here.
|
||||
* Check status for same buffer for next time
|
||||
*/
|
||||
pmon->buf_state = DP_MON_STATUS_NO_DMA;
|
||||
goto move_next;
|
||||
break;
|
||||
}
|
||||
|
||||
spin_lock_bh(&rx_ring->idr_lock);
|
||||
idr_remove(&rx_ring->bufs_idr, buf_id);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
if (ab->hw_params.full_monitor_mode) {
|
||||
ath11k_dp_rx_mon_update_status_buf_state(pmon, tlv);
|
||||
if (paddr == pmon->mon_status_paddr)
|
||||
pmon->buf_state = DP_MON_STATUS_MATCH;
|
||||
}
|
||||
|
||||
dma_unmap_single(ab->dev, rxcb->paddr,
|
||||
skb->len + skb_tailroom(skb),
|
||||
DMA_FROM_DEVICE);
|
||||
|
||||
__skb_queue_tail(skb_list, skb);
|
||||
} else {
|
||||
pmon->buf_state = DP_MON_STATUS_REPLINISH;
|
||||
@ -3117,8 +3132,11 @@ int ath11k_peer_rx_frag_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id
|
||||
int i;
|
||||
|
||||
tfm = crypto_alloc_shash("michael_mic", 0, 0);
|
||||
if (IS_ERR(tfm))
|
||||
if (IS_ERR(tfm)) {
|
||||
ath11k_warn(ab, "failed to allocate michael_mic shash: %ld\n",
|
||||
PTR_ERR(tfm));
|
||||
return PTR_ERR(tfm);
|
||||
}
|
||||
|
||||
spin_lock_bh(&ab->base_lock);
|
||||
|
||||
@ -3138,6 +3156,7 @@ int ath11k_peer_rx_frag_setup(struct ath11k *ar, const u8 *peer_mac, int vdev_id
|
||||
}
|
||||
|
||||
peer->tfm_mmic = tfm;
|
||||
peer->dp_setup_done = true;
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
|
||||
return 0;
|
||||
@ -3583,6 +3602,13 @@ static int ath11k_dp_rx_frag_h_mpdu(struct ath11k *ar,
|
||||
ret = -ENOENT;
|
||||
goto out_unlock;
|
||||
}
|
||||
if (!peer->dp_setup_done) {
|
||||
ath11k_warn(ab, "The peer %pM [%d] has uninitialized datapath\n",
|
||||
peer->addr, peer_id);
|
||||
ret = -ENOENT;
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
rx_tid = &peer->rx_tid[tid];
|
||||
|
||||
if ((!skb_queue_empty(&rx_tid->rx_frags) && seqno != rx_tid->cur_sn) ||
|
||||
@ -3598,7 +3624,7 @@ static int ath11k_dp_rx_frag_h_mpdu(struct ath11k *ar,
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
if (frag_no > __fls(rx_tid->rx_frag_bitmap))
|
||||
if (!rx_tid->rx_frag_bitmap || (frag_no > __fls(rx_tid->rx_frag_bitmap)))
|
||||
__skb_queue_tail(&rx_tid->rx_frags, msdu);
|
||||
else
|
||||
ath11k_dp_rx_h_sort_frags(ar, &rx_tid->rx_frags, msdu);
|
||||
|
@ -316,10 +316,12 @@ ath11k_dp_tx_htt_tx_complete_buf(struct ath11k_base *ab,
|
||||
struct dp_tx_ring *tx_ring,
|
||||
struct ath11k_dp_htt_wbm_tx_status *ts)
|
||||
{
|
||||
struct ieee80211_tx_status status = { 0 };
|
||||
struct sk_buff *msdu;
|
||||
struct ieee80211_tx_info *info;
|
||||
struct ath11k_skb_cb *skb_cb;
|
||||
struct ath11k *ar;
|
||||
struct ath11k_peer *peer;
|
||||
|
||||
spin_lock(&tx_ring->tx_idr_lock);
|
||||
msdu = idr_remove(&tx_ring->txbuf_idr, ts->msdu_id);
|
||||
@ -341,6 +343,11 @@ ath11k_dp_tx_htt_tx_complete_buf(struct ath11k_base *ab,
|
||||
|
||||
dma_unmap_single(ab->dev, skb_cb->paddr, msdu->len, DMA_TO_DEVICE);
|
||||
|
||||
if (!skb_cb->vif) {
|
||||
dev_kfree_skb_any(msdu);
|
||||
return;
|
||||
}
|
||||
|
||||
memset(&info->status, 0, sizeof(info->status));
|
||||
|
||||
if (ts->acked) {
|
||||
@ -355,7 +362,23 @@ ath11k_dp_tx_htt_tx_complete_buf(struct ath11k_base *ab,
|
||||
}
|
||||
}
|
||||
|
||||
ieee80211_tx_status(ar->hw, msdu);
|
||||
spin_lock_bh(&ab->base_lock);
|
||||
peer = ath11k_peer_find_by_id(ab, ts->peer_id);
|
||||
if (!peer || !peer->sta) {
|
||||
ath11k_dbg(ab, ATH11K_DBG_DATA,
|
||||
"dp_tx: failed to find the peer with peer_id %d\n",
|
||||
ts->peer_id);
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
dev_kfree_skb_any(msdu);
|
||||
return;
|
||||
}
|
||||
spin_unlock_bh(&ab->base_lock);
|
||||
|
||||
status.sta = peer->sta;
|
||||
status.info = info;
|
||||
status.skb = msdu;
|
||||
|
||||
ieee80211_tx_status_ext(ar->hw, &status);
|
||||
}
|
||||
|
||||
static void
|
||||
@ -379,7 +402,15 @@ ath11k_dp_tx_process_htt_tx_complete(struct ath11k_base *ab,
|
||||
ts.msdu_id = msdu_id;
|
||||
ts.ack_rssi = FIELD_GET(HTT_TX_WBM_COMP_INFO1_ACK_RSSI,
|
||||
status_desc->info1);
|
||||
|
||||
if (FIELD_GET(HTT_TX_WBM_COMP_INFO2_VALID, status_desc->info2))
|
||||
ts.peer_id = FIELD_GET(HTT_TX_WBM_COMP_INFO2_SW_PEER_ID,
|
||||
status_desc->info2);
|
||||
else
|
||||
ts.peer_id = HTT_INVALID_PEER_ID;
|
||||
|
||||
ath11k_dp_tx_htt_tx_complete_buf(ab, tx_ring, &ts);
|
||||
|
||||
break;
|
||||
case HAL_WBM_REL_HTT_TX_COMP_STATUS_REINJ:
|
||||
case HAL_WBM_REL_HTT_TX_COMP_STATUS_INSPECT:
|
||||
|
@ -13,6 +13,7 @@ struct ath11k_dp_htt_wbm_tx_status {
|
||||
u32 msdu_id;
|
||||
bool acked;
|
||||
int ack_rssi;
|
||||
u16 peer_id;
|
||||
};
|
||||
|
||||
void ath11k_dp_tx_update_txcompl(struct ath11k *ar, struct hal_tx_status *ts);
|
||||
|
@ -865,6 +865,12 @@ ath11k_hal_rx_populate_mu_user_info(void *rx_tlv, struct hal_rx_mon_ppdu_info *p
|
||||
ath11k_hal_rx_populate_byte_count(rx_tlv, ppdu_info, rx_user_status);
|
||||
}
|
||||
|
||||
static u16 ath11k_hal_rx_mpduinfo_get_peerid(struct ath11k_base *ab,
|
||||
struct hal_rx_mpdu_info *mpdu_info)
|
||||
{
|
||||
return ab->hw_params.hw_ops->mpdu_info_get_peerid(mpdu_info);
|
||||
}
|
||||
|
||||
static enum hal_rx_mon_status
|
||||
ath11k_hal_rx_parse_mon_status_tlv(struct ath11k_base *ab,
|
||||
struct hal_rx_mon_ppdu_info *ppdu_info,
|
||||
@ -1023,7 +1029,7 @@ ath11k_hal_rx_parse_mon_status_tlv(struct ath11k_base *ab,
|
||||
info1 = __le32_to_cpu(vht_sig->info1);
|
||||
|
||||
ppdu_info->ldpc = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_SU_MU_CODING,
|
||||
info0);
|
||||
info1);
|
||||
ppdu_info->mcs = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_MCS,
|
||||
info1);
|
||||
gi_setting = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_GI_SETTING,
|
||||
@ -1446,7 +1452,7 @@ ath11k_hal_rx_parse_mon_status_tlv(struct ath11k_base *ab,
|
||||
* PHYRX_OTHER_RECEIVE_INFO TLV.
|
||||
*/
|
||||
ppdu_info->rssi_comb =
|
||||
FIELD_GET(HAL_RX_PHYRX_RSSI_LEGACY_INFO_INFO1_RSSI_COMB,
|
||||
FIELD_GET(HAL_RX_PHYRX_RSSI_LEGACY_INFO_INFO0_RSSI_COMB,
|
||||
__le32_to_cpu(rssi->info0));
|
||||
|
||||
if (db2dbm) {
|
||||
@ -1459,9 +1465,11 @@ ath11k_hal_rx_parse_mon_status_tlv(struct ath11k_base *ab,
|
||||
break;
|
||||
}
|
||||
case HAL_RX_MPDU_START: {
|
||||
struct hal_rx_mpdu_info *mpdu_info =
|
||||
(struct hal_rx_mpdu_info *)tlv_data;
|
||||
u16 peer_id;
|
||||
|
||||
peer_id = ab->hw_params.hw_ops->mpdu_info_get_peerid(tlv_data);
|
||||
peer_id = ath11k_hal_rx_mpduinfo_get_peerid(ab, mpdu_info);
|
||||
if (peer_id)
|
||||
ppdu_info->peer_id = peer_id;
|
||||
break;
|
||||
|
@ -385,7 +385,7 @@ struct hal_rx_he_sig_b2_ofdma_info {
|
||||
__le32 info0;
|
||||
} __packed;
|
||||
|
||||
#define HAL_RX_PHYRX_RSSI_LEGACY_INFO_INFO1_RSSI_COMB GENMASK(15, 8)
|
||||
#define HAL_RX_PHYRX_RSSI_LEGACY_INFO_INFO0_RSSI_COMB GENMASK(15, 8)
|
||||
|
||||
#define HAL_RX_PHYRX_RSSI_PREAMBLE_PRI20 GENMASK(7, 0)
|
||||
|
||||
@ -405,7 +405,7 @@ struct hal_rx_phyrx_rssi_legacy_info {
|
||||
#define HAL_RX_MPDU_INFO_INFO0_PEERID_WCN6855 GENMASK(15, 0)
|
||||
#define HAL_RX_MPDU_INFO_INFO1_MPDU_LEN GENMASK(13, 0)
|
||||
|
||||
struct hal_rx_mpdu_info {
|
||||
struct hal_rx_mpdu_info_ipq8074 {
|
||||
__le32 rsvd0;
|
||||
__le32 info0;
|
||||
__le32 rsvd1[11];
|
||||
@ -413,12 +413,28 @@ struct hal_rx_mpdu_info {
|
||||
__le32 rsvd2[9];
|
||||
} __packed;
|
||||
|
||||
struct hal_rx_mpdu_info_qcn9074 {
|
||||
__le32 rsvd0[10];
|
||||
__le32 info0;
|
||||
__le32 rsvd1[2];
|
||||
__le32 info1;
|
||||
__le32 rsvd2[9];
|
||||
} __packed;
|
||||
|
||||
struct hal_rx_mpdu_info_wcn6855 {
|
||||
__le32 rsvd0[8];
|
||||
__le32 info0;
|
||||
__le32 rsvd1[14];
|
||||
} __packed;
|
||||
|
||||
struct hal_rx_mpdu_info {
|
||||
union {
|
||||
struct hal_rx_mpdu_info_ipq8074 ipq8074;
|
||||
struct hal_rx_mpdu_info_qcn9074 qcn9074;
|
||||
struct hal_rx_mpdu_info_wcn6855 wcn6855;
|
||||
} u;
|
||||
} __packed;
|
||||
|
||||
#define HAL_RX_PPDU_END_DURATION GENMASK(23, 0)
|
||||
struct hal_rx_ppdu_end_duration {
|
||||
__le32 rsvd0[9];
|
||||
|
@ -835,26 +835,35 @@ static void ath11k_hw_ipq5018_reo_setup(struct ath11k_base *ab)
|
||||
ring_hash_map);
|
||||
}
|
||||
|
||||
static u16 ath11k_hw_ipq8074_mpdu_info_get_peerid(u8 *tlv_data)
|
||||
static u16
|
||||
ath11k_hw_ipq8074_mpdu_info_get_peerid(struct hal_rx_mpdu_info *mpdu_info)
|
||||
{
|
||||
u16 peer_id = 0;
|
||||
struct hal_rx_mpdu_info *mpdu_info =
|
||||
(struct hal_rx_mpdu_info *)tlv_data;
|
||||
|
||||
peer_id = FIELD_GET(HAL_RX_MPDU_INFO_INFO0_PEERID,
|
||||
__le32_to_cpu(mpdu_info->info0));
|
||||
__le32_to_cpu(mpdu_info->u.ipq8074.info0));
|
||||
|
||||
return peer_id;
|
||||
}
|
||||
|
||||
static u16 ath11k_hw_wcn6855_mpdu_info_get_peerid(u8 *tlv_data)
|
||||
static u16
|
||||
ath11k_hw_qcn9074_mpdu_info_get_peerid(struct hal_rx_mpdu_info *mpdu_info)
|
||||
{
|
||||
u16 peer_id = 0;
|
||||
|
||||
peer_id = FIELD_GET(HAL_RX_MPDU_INFO_INFO0_PEERID,
|
||||
__le32_to_cpu(mpdu_info->u.qcn9074.info0));
|
||||
|
||||
return peer_id;
|
||||
}
|
||||
|
||||
static u16
|
||||
ath11k_hw_wcn6855_mpdu_info_get_peerid(struct hal_rx_mpdu_info *mpdu_info)
|
||||
{
|
||||
u16 peer_id = 0;
|
||||
struct hal_rx_mpdu_info_wcn6855 *mpdu_info =
|
||||
(struct hal_rx_mpdu_info_wcn6855 *)tlv_data;
|
||||
|
||||
peer_id = FIELD_GET(HAL_RX_MPDU_INFO_INFO0_PEERID_WCN6855,
|
||||
__le32_to_cpu(mpdu_info->info0));
|
||||
__le32_to_cpu(mpdu_info->u.wcn6855.info0));
|
||||
return peer_id;
|
||||
}
|
||||
|
||||
@ -1042,7 +1051,7 @@ const struct ath11k_hw_ops qcn9074_ops = {
|
||||
.rx_desc_get_attention = ath11k_hw_qcn9074_rx_desc_get_attention,
|
||||
.rx_desc_get_msdu_payload = ath11k_hw_qcn9074_rx_desc_get_msdu_payload,
|
||||
.reo_setup = ath11k_hw_ipq8074_reo_setup,
|
||||
.mpdu_info_get_peerid = ath11k_hw_ipq8074_mpdu_info_get_peerid,
|
||||
.mpdu_info_get_peerid = ath11k_hw_qcn9074_mpdu_info_get_peerid,
|
||||
.rx_desc_mac_addr2_valid = ath11k_hw_ipq9074_rx_desc_mac_addr2_valid,
|
||||
.rx_desc_mpdu_start_addr2 = ath11k_hw_ipq9074_rx_desc_mpdu_start_addr2,
|
||||
.get_ring_selector = ath11k_hw_ipq8074_get_tcl_ring_selector,
|
||||
@ -1224,6 +1233,7 @@ const struct ath11k_hw_ring_mask ath11k_hw_ring_mask_ipq8074 = {
|
||||
ATH11K_RX_WBM_REL_RING_MASK_0,
|
||||
},
|
||||
.reo_status = {
|
||||
0, 0, 0,
|
||||
ATH11K_REO_STATUS_RING_MASK_0,
|
||||
},
|
||||
.rxdma2host = {
|
||||
|
@ -224,7 +224,6 @@ struct ath11k_hw_params {
|
||||
u32 tx_ring_size;
|
||||
bool smp2p_wow_exit;
|
||||
bool support_fw_mac_sequence;
|
||||
bool ftm_responder;
|
||||
};
|
||||
|
||||
struct ath11k_hw_ops {
|
||||
@ -264,7 +263,7 @@ struct ath11k_hw_ops {
|
||||
struct rx_attention *(*rx_desc_get_attention)(struct hal_rx_desc *desc);
|
||||
u8 *(*rx_desc_get_msdu_payload)(struct hal_rx_desc *desc);
|
||||
void (*reo_setup)(struct ath11k_base *ab);
|
||||
u16 (*mpdu_info_get_peerid)(u8 *tlv_data);
|
||||
u16 (*mpdu_info_get_peerid)(struct hal_rx_mpdu_info *mpdu_info);
|
||||
bool (*rx_desc_mac_addr2_valid)(struct hal_rx_desc *desc);
|
||||
u8* (*rx_desc_mpdu_start_addr2)(struct hal_rx_desc *desc);
|
||||
u32 (*get_ring_selector)(struct sk_buff *skb);
|
||||
|
@ -3538,7 +3538,7 @@ static void ath11k_mac_op_bss_info_changed(struct ieee80211_hw *hw,
|
||||
|
||||
if (changed & BSS_CHANGED_FTM_RESPONDER &&
|
||||
arvif->ftm_responder != info->ftm_responder &&
|
||||
ar->ab->hw_params.ftm_responder &&
|
||||
test_bit(WMI_TLV_SERVICE_RTT, ar->ab->wmi_ab.svc_map) &&
|
||||
(vif->type == NL80211_IFTYPE_AP ||
|
||||
vif->type == NL80211_IFTYPE_MESH_POINT)) {
|
||||
arvif->ftm_responder = info->ftm_responder;
|
||||
@ -3755,6 +3755,18 @@ static int ath11k_mac_op_hw_scan(struct ieee80211_hw *hw,
|
||||
int i;
|
||||
u32 scan_timeout;
|
||||
|
||||
/* Firmwares advertising the support of triggering 11D algorithm
|
||||
* on the scan results of a regular scan expects driver to send
|
||||
* WMI_11D_SCAN_START_CMDID before sending WMI_START_SCAN_CMDID.
|
||||
* With this feature, separate 11D scan can be avoided since
|
||||
* regdomain can be determined with the scan results of the
|
||||
* regular scan.
|
||||
*/
|
||||
if (ar->state_11d == ATH11K_11D_PREPARING &&
|
||||
test_bit(WMI_TLV_SERVICE_SUPPORT_11D_FOR_HOST_SCAN,
|
||||
ar->ab->wmi_ab.svc_map))
|
||||
ath11k_mac_11d_scan_start(ar, arvif->vdev_id);
|
||||
|
||||
mutex_lock(&ar->conf_mutex);
|
||||
|
||||
spin_lock_bh(&ar->data_lock);
|
||||
@ -3819,8 +3831,29 @@ static int ath11k_mac_op_hw_scan(struct ieee80211_hw *hw,
|
||||
goto exit;
|
||||
}
|
||||
|
||||
for (i = 0; i < arg->num_chan; i++)
|
||||
arg->chan_list[i] = req->channels[i]->center_freq;
|
||||
for (i = 0; i < arg->num_chan; i++) {
|
||||
if (test_bit(WMI_TLV_SERVICE_SCAN_CONFIG_PER_CHANNEL,
|
||||
ar->ab->wmi_ab.svc_map)) {
|
||||
arg->chan_list[i] =
|
||||
u32_encode_bits(req->channels[i]->center_freq,
|
||||
WMI_SCAN_CONFIG_PER_CHANNEL_MASK);
|
||||
|
||||
/* If NL80211_SCAN_FLAG_COLOCATED_6GHZ is set in scan
|
||||
* flags, then scan all PSC channels in 6 GHz band and
|
||||
* those non-PSC channels where RNR IE is found during
|
||||
* the legacy 2.4/5 GHz scan.
|
||||
* If NL80211_SCAN_FLAG_COLOCATED_6GHZ is not set,
|
||||
* then all channels in 6 GHz will be scanned.
|
||||
*/
|
||||
if (req->channels[i]->band == NL80211_BAND_6GHZ &&
|
||||
req->flags & NL80211_SCAN_FLAG_COLOCATED_6GHZ &&
|
||||
!cfg80211_channel_is_psc(req->channels[i]))
|
||||
arg->chan_list[i] |=
|
||||
WMI_SCAN_CH_FLAG_SCAN_ONLY_IF_RNR_FOUND;
|
||||
} else {
|
||||
arg->chan_list[i] = req->channels[i]->center_freq;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (req->flags & NL80211_SCAN_FLAG_RANDOM_ADDR) {
|
||||
@ -5552,10 +5585,6 @@ static int ath11k_mac_copy_he_cap(struct ath11k *ar,
|
||||
|
||||
he_cap_elem->mac_cap_info[1] &=
|
||||
IEEE80211_HE_MAC_CAP1_TF_MAC_PAD_DUR_MASK;
|
||||
he_cap_elem->phy_cap_info[0] &=
|
||||
~IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G;
|
||||
he_cap_elem->phy_cap_info[0] &=
|
||||
~IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G;
|
||||
|
||||
he_cap_elem->phy_cap_info[5] &=
|
||||
~IEEE80211_HE_PHY_CAP5_BEAMFORMEE_NUM_SND_DIM_UNDER_80MHZ_MASK;
|
||||
@ -6652,6 +6681,11 @@ static void ath11k_mac_op_remove_interface(struct ieee80211_hw *hw,
|
||||
ath11k_dbg(ab, ATH11K_DBG_MAC, "mac remove interface (vdev %d)\n",
|
||||
arvif->vdev_id);
|
||||
|
||||
ret = ath11k_spectral_vif_stop(arvif);
|
||||
if (ret)
|
||||
ath11k_warn(ab, "failed to stop spectral for vdev %i: %d\n",
|
||||
arvif->vdev_id, ret);
|
||||
|
||||
if (arvif->vdev_type == WMI_VDEV_TYPE_STA)
|
||||
ath11k_mac_11d_scan_stop(ar);
|
||||
|
||||
@ -9213,7 +9247,7 @@ static int __ath11k_mac_register(struct ath11k *ar)
|
||||
wiphy_ext_feature_set(ar->hw->wiphy,
|
||||
NL80211_EXT_FEATURE_SET_SCAN_DWELL);
|
||||
|
||||
if (ab->hw_params.ftm_responder)
|
||||
if (test_bit(WMI_TLV_SERVICE_RTT, ar->ab->wmi_ab.svc_map))
|
||||
wiphy_ext_feature_set(ar->hw->wiphy,
|
||||
NL80211_EXT_FEATURE_ENABLE_FTM_RESPONDER);
|
||||
|
||||
|
@ -540,7 +540,7 @@ static int ath11k_pci_claim(struct ath11k_pci *ab_pci, struct pci_dev *pdev)
|
||||
if (!ab->mem) {
|
||||
ath11k_err(ab, "failed to map pci bar %d\n", ATH11K_PCI_BAR_NUM);
|
||||
ret = -EIO;
|
||||
goto clear_master;
|
||||
goto release_region;
|
||||
}
|
||||
|
||||
ab->mem_ce = ab->mem;
|
||||
@ -548,8 +548,6 @@ static int ath11k_pci_claim(struct ath11k_pci *ab_pci, struct pci_dev *pdev)
|
||||
ath11k_dbg(ab, ATH11K_DBG_BOOT, "boot pci_mem 0x%pK\n", ab->mem);
|
||||
return 0;
|
||||
|
||||
clear_master:
|
||||
pci_clear_master(pdev);
|
||||
release_region:
|
||||
pci_release_region(pdev, ATH11K_PCI_BAR_NUM);
|
||||
disable_device:
|
||||
@ -565,7 +563,6 @@ static void ath11k_pci_free_region(struct ath11k_pci *ab_pci)
|
||||
|
||||
pci_iounmap(pci_dev, ab->mem);
|
||||
ab->mem = NULL;
|
||||
pci_clear_master(pci_dev);
|
||||
pci_release_region(pci_dev, ATH11K_PCI_BAR_NUM);
|
||||
if (pci_is_enabled(pci_dev))
|
||||
pci_disable_device(pci_dev);
|
||||
@ -1039,7 +1036,8 @@ module_exit(ath11k_pci_exit);
|
||||
MODULE_DESCRIPTION("Driver support for Qualcomm Technologies 802.11ax WLAN PCIe devices");
|
||||
MODULE_LICENSE("Dual BSD/GPL");
|
||||
|
||||
/* QCA639x 2.0 firmware files */
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/QCA6390/hw2.0/" ATH11K_BOARD_API2_FILE);
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/QCA6390/hw2.0/" ATH11K_AMSS_FILE);
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/QCA6390/hw2.0/" ATH11K_M3_FILE);
|
||||
/* firmware files */
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/QCA6390/hw2.0/*");
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/QCN9074/hw1.0/*");
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/WCN6855/hw2.0/*");
|
||||
MODULE_FIRMWARE(ATH11K_FW_DIR "/WCN6855/hw2.1/*");
|
||||
|
@ -35,6 +35,7 @@ struct ath11k_peer {
|
||||
u16 sec_type;
|
||||
u16 sec_type_grp;
|
||||
bool is_authorized;
|
||||
bool dp_setup_done;
|
||||
};
|
||||
|
||||
void ath11k_peer_unmap_event(struct ath11k_base *ab, u16 peer_id);
|
||||
|
@ -82,6 +82,12 @@ struct wmi_tlv_fw_stats_parse {
|
||||
bool chain_rssi_done;
|
||||
};
|
||||
|
||||
struct wmi_tlv_mgmt_rx_parse {
|
||||
const struct wmi_mgmt_rx_hdr *fixed;
|
||||
const u8 *frame_buf;
|
||||
bool frame_buf_done;
|
||||
};
|
||||
|
||||
static const struct wmi_tlv_policy wmi_tlv_policies[] = {
|
||||
[WMI_TAG_ARRAY_BYTE]
|
||||
= { .min_len = 0 },
|
||||
@ -865,7 +871,8 @@ static void ath11k_wmi_put_wmi_channel(struct wmi_channel *chan,
|
||||
|
||||
chan->band_center_freq2 = arg->channel.band_center_freq1;
|
||||
|
||||
} else if (arg->channel.mode == MODE_11AC_VHT80_80) {
|
||||
} else if ((arg->channel.mode == MODE_11AC_VHT80_80) ||
|
||||
(arg->channel.mode == MODE_11AX_HE80_80)) {
|
||||
chan->band_center_freq2 = arg->channel.band_center_freq2;
|
||||
} else {
|
||||
chan->band_center_freq2 = 0;
|
||||
@ -5633,28 +5640,49 @@ static int ath11k_pull_vdev_stopped_param_tlv(struct ath11k_base *ab, struct sk_
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ath11k_wmi_tlv_mgmt_rx_parse(struct ath11k_base *ab,
|
||||
u16 tag, u16 len,
|
||||
const void *ptr, void *data)
|
||||
{
|
||||
struct wmi_tlv_mgmt_rx_parse *parse = data;
|
||||
|
||||
switch (tag) {
|
||||
case WMI_TAG_MGMT_RX_HDR:
|
||||
parse->fixed = ptr;
|
||||
break;
|
||||
case WMI_TAG_ARRAY_BYTE:
|
||||
if (!parse->frame_buf_done) {
|
||||
parse->frame_buf = ptr;
|
||||
parse->frame_buf_done = true;
|
||||
}
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ath11k_pull_mgmt_rx_params_tlv(struct ath11k_base *ab,
|
||||
struct sk_buff *skb,
|
||||
struct mgmt_rx_event_params *hdr)
|
||||
{
|
||||
const void **tb;
|
||||
struct wmi_tlv_mgmt_rx_parse parse = { };
|
||||
const struct wmi_mgmt_rx_hdr *ev;
|
||||
const u8 *frame;
|
||||
int ret;
|
||||
|
||||
tb = ath11k_wmi_tlv_parse_alloc(ab, skb->data, skb->len, GFP_ATOMIC);
|
||||
if (IS_ERR(tb)) {
|
||||
ret = PTR_ERR(tb);
|
||||
ath11k_warn(ab, "failed to parse tlv: %d\n", ret);
|
||||
ret = ath11k_wmi_tlv_iter(ab, skb->data, skb->len,
|
||||
ath11k_wmi_tlv_mgmt_rx_parse,
|
||||
&parse);
|
||||
if (ret) {
|
||||
ath11k_warn(ab, "failed to parse mgmt rx tlv %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ev = tb[WMI_TAG_MGMT_RX_HDR];
|
||||
frame = tb[WMI_TAG_ARRAY_BYTE];
|
||||
ev = parse.fixed;
|
||||
frame = parse.frame_buf;
|
||||
|
||||
if (!ev || !frame) {
|
||||
ath11k_warn(ab, "failed to fetch mgmt rx hdr");
|
||||
kfree(tb);
|
||||
return -EPROTO;
|
||||
}
|
||||
|
||||
@ -5673,7 +5701,6 @@ static int ath11k_pull_mgmt_rx_params_tlv(struct ath11k_base *ab,
|
||||
|
||||
if (skb->len < (frame - skb->data) + hdr->buf_len) {
|
||||
ath11k_warn(ab, "invalid length in mgmt rx hdr ev");
|
||||
kfree(tb);
|
||||
return -EPROTO;
|
||||
}
|
||||
|
||||
@ -5685,7 +5712,6 @@ static int ath11k_pull_mgmt_rx_params_tlv(struct ath11k_base *ab,
|
||||
|
||||
ath11k_ce_byte_swap(skb->data, hdr->buf_len);
|
||||
|
||||
kfree(tb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -2100,8 +2100,10 @@ enum wmi_tlv_service {
|
||||
|
||||
/* The second 128 bits */
|
||||
WMI_MAX_EXT_SERVICE = 256,
|
||||
WMI_TLV_SERVICE_SCAN_CONFIG_PER_CHANNEL = 265,
|
||||
WMI_TLV_SERVICE_REG_CC_EXT_EVENT_SUPPORT = 281,
|
||||
WMI_TLV_SERVICE_BIOS_SAR_SUPPORT = 326,
|
||||
WMI_TLV_SERVICE_SUPPORT_11D_FOR_HOST_SCAN = 357,
|
||||
|
||||
/* The third 128 bits */
|
||||
WMI_MAX_EXT2_SERVICE = 384
|
||||
@ -3249,6 +3251,9 @@ struct wmi_start_scan_cmd {
|
||||
#define WMI_SCAN_DWELL_MODE_SHIFT 21
|
||||
#define WMI_SCAN_FLAG_EXT_PASSIVE_SCAN_START_TIME_ENHANCE 0x00000800
|
||||
|
||||
#define WMI_SCAN_CONFIG_PER_CHANNEL_MASK GENMASK(19, 0)
|
||||
#define WMI_SCAN_CH_FLAG_SCAN_ONLY_IF_RNR_FOUND BIT(20)
|
||||
|
||||
enum {
|
||||
WMI_SCAN_DWELL_MODE_DEFAULT = 0,
|
||||
WMI_SCAN_DWELL_MODE_CONSERVATIVE = 1,
|
||||
|
@ -395,6 +395,7 @@ struct ath12k_sta {
|
||||
u8 rssi_comb;
|
||||
struct ath12k_rx_peer_stats *rx_stats;
|
||||
struct ath12k_wbm_tx_stats *wbm_tx_stats;
|
||||
u32 bw_prev;
|
||||
};
|
||||
|
||||
#define ATH12K_MIN_5G_FREQ 4150
|
||||
|
@ -196,7 +196,8 @@ static void ath12k_dp_rxdesc_set_msdu_len(struct ath12k_base *ab,
|
||||
static bool ath12k_dp_rx_h_is_mcbc(struct ath12k_base *ab,
|
||||
struct hal_rx_desc *desc)
|
||||
{
|
||||
return ab->hw_params->hal_ops->rx_desc_is_mcbc(desc);
|
||||
return (ath12k_dp_rx_h_first_msdu(ab, desc) &&
|
||||
ab->hw_params->hal_ops->rx_desc_is_mcbc(desc));
|
||||
}
|
||||
|
||||
static bool ath12k_dp_rxdesc_mac_addr2_valid(struct ath12k_base *ab,
|
||||
@ -3047,10 +3048,14 @@ static int ath12k_dp_rx_h_defrag_reo_reinject(struct ath12k *ar,
|
||||
reo_ent_ring->rx_mpdu_info.peer_meta_data =
|
||||
reo_dest_ring->rx_mpdu_info.peer_meta_data;
|
||||
|
||||
reo_ent_ring->queue_addr_lo = cpu_to_le32(lower_32_bits(rx_tid->paddr));
|
||||
reo_ent_ring->info0 = le32_encode_bits(upper_32_bits(rx_tid->paddr),
|
||||
HAL_REO_ENTR_RING_INFO0_QUEUE_ADDR_HI) |
|
||||
le32_encode_bits(dst_ind, HAL_REO_ENTR_RING_INFO0_DEST_IND);
|
||||
/* Firmware expects physical address to be filled in queue_addr_lo in
|
||||
* the MLO scenario and in case of non MLO peer meta data needs to be
|
||||
* filled.
|
||||
* TODO: Need to handle for MLO scenario.
|
||||
*/
|
||||
reo_ent_ring->queue_addr_lo = reo_dest_ring->rx_mpdu_info.peer_meta_data;
|
||||
reo_ent_ring->info0 = le32_encode_bits(dst_ind,
|
||||
HAL_REO_ENTR_RING_INFO0_DEST_IND);
|
||||
|
||||
reo_ent_ring->info1 = le32_encode_bits(rx_tid->cur_sn,
|
||||
HAL_REO_ENTR_RING_INFO1_MPDU_SEQ_NUM);
|
||||
|
@ -13,6 +13,10 @@ static enum hal_tcl_encap_type
|
||||
ath12k_dp_tx_get_encap_type(struct ath12k_vif *arvif, struct sk_buff *skb)
|
||||
{
|
||||
struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb);
|
||||
struct ath12k_base *ab = arvif->ar->ab;
|
||||
|
||||
if (test_bit(ATH12K_FLAG_RAW_MODE, &ab->dev_flags))
|
||||
return HAL_TCL_ENCAP_TYPE_RAW;
|
||||
|
||||
if (tx_info->flags & IEEE80211_TX_CTL_HW_80211_ENCAP)
|
||||
return HAL_TCL_ENCAP_TYPE_ETHERNET;
|
||||
|
@ -944,7 +944,7 @@ static const struct ath12k_hw_params ath12k_hw_params[] = {
|
||||
.interface_modes = BIT(NL80211_IFTYPE_STATION),
|
||||
.supports_monitor = false,
|
||||
|
||||
.idle_ps = false,
|
||||
.idle_ps = true,
|
||||
.download_calib = false,
|
||||
.supports_suspend = false,
|
||||
.tcl_ring_retry = false,
|
||||
|
@ -3220,10 +3220,11 @@ static void ath12k_sta_rc_update_wk(struct work_struct *wk)
|
||||
enum nl80211_band band;
|
||||
const u8 *ht_mcs_mask;
|
||||
const u16 *vht_mcs_mask;
|
||||
u32 changed, bw, nss, smps;
|
||||
u32 changed, bw, nss, smps, bw_prev;
|
||||
int err, num_vht_rates;
|
||||
const struct cfg80211_bitrate_mask *mask;
|
||||
struct ath12k_wmi_peer_assoc_arg peer_arg;
|
||||
enum wmi_phy_mode peer_phymode;
|
||||
|
||||
arsta = container_of(wk, struct ath12k_sta, update_wk);
|
||||
sta = container_of((void *)arsta, struct ieee80211_sta, drv_priv);
|
||||
@ -3243,6 +3244,7 @@ static void ath12k_sta_rc_update_wk(struct work_struct *wk)
|
||||
arsta->changed = 0;
|
||||
|
||||
bw = arsta->bw;
|
||||
bw_prev = arsta->bw_prev;
|
||||
nss = arsta->nss;
|
||||
smps = arsta->smps;
|
||||
|
||||
@ -3255,11 +3257,53 @@ static void ath12k_sta_rc_update_wk(struct work_struct *wk)
|
||||
ath12k_mac_max_vht_nss(vht_mcs_mask)));
|
||||
|
||||
if (changed & IEEE80211_RC_BW_CHANGED) {
|
||||
err = ath12k_wmi_set_peer_param(ar, sta->addr, arvif->vdev_id,
|
||||
WMI_PEER_CHWIDTH, bw);
|
||||
if (err)
|
||||
ath12k_warn(ar->ab, "failed to update STA %pM peer bw %d: %d\n",
|
||||
sta->addr, bw, err);
|
||||
ath12k_peer_assoc_h_phymode(ar, arvif->vif, sta, &peer_arg);
|
||||
peer_phymode = peer_arg.peer_phymode;
|
||||
|
||||
if (bw > bw_prev) {
|
||||
/* Phymode shows maximum supported channel width, if we
|
||||
* upgrade bandwidth then due to sanity check of firmware,
|
||||
* we have to send WMI_PEER_PHYMODE followed by
|
||||
* WMI_PEER_CHWIDTH
|
||||
*/
|
||||
ath12k_dbg(ar->ab, ATH12K_DBG_MAC, "mac bandwidth upgrade for sta %pM new %d old %d\n",
|
||||
sta->addr, bw, bw_prev);
|
||||
err = ath12k_wmi_set_peer_param(ar, sta->addr,
|
||||
arvif->vdev_id, WMI_PEER_PHYMODE,
|
||||
peer_phymode);
|
||||
if (err) {
|
||||
ath12k_warn(ar->ab, "failed to update STA %pM to peer phymode %d: %d\n",
|
||||
sta->addr, peer_phymode, err);
|
||||
goto err_rc_bw_changed;
|
||||
}
|
||||
err = ath12k_wmi_set_peer_param(ar, sta->addr,
|
||||
arvif->vdev_id, WMI_PEER_CHWIDTH,
|
||||
bw);
|
||||
if (err)
|
||||
ath12k_warn(ar->ab, "failed to update STA %pM to peer bandwidth %d: %d\n",
|
||||
sta->addr, bw, err);
|
||||
} else {
|
||||
/* When we downgrade bandwidth this will conflict with phymode
|
||||
* and cause to trigger firmware crash. In this case we send
|
||||
* WMI_PEER_CHWIDTH followed by WMI_PEER_PHYMODE
|
||||
*/
|
||||
ath12k_dbg(ar->ab, ATH12K_DBG_MAC, "mac bandwidth downgrade for sta %pM new %d old %d\n",
|
||||
sta->addr, bw, bw_prev);
|
||||
err = ath12k_wmi_set_peer_param(ar, sta->addr,
|
||||
arvif->vdev_id, WMI_PEER_CHWIDTH,
|
||||
bw);
|
||||
if (err) {
|
||||
ath12k_warn(ar->ab, "failed to update STA %pM peer to bandwidth %d: %d\n",
|
||||
sta->addr, bw, err);
|
||||
goto err_rc_bw_changed;
|
||||
}
|
||||
err = ath12k_wmi_set_peer_param(ar, sta->addr,
|
||||
arvif->vdev_id, WMI_PEER_PHYMODE,
|
||||
peer_phymode);
|
||||
if (err)
|
||||
ath12k_warn(ar->ab, "failed to update STA %pM to peer phymode %d: %d\n",
|
||||
sta->addr, peer_phymode, err);
|
||||
}
|
||||
}
|
||||
|
||||
if (changed & IEEE80211_RC_NSS_CHANGED) {
|
||||
@ -3321,7 +3365,7 @@ static void ath12k_sta_rc_update_wk(struct work_struct *wk)
|
||||
sta->addr, arvif->vdev_id);
|
||||
}
|
||||
}
|
||||
|
||||
err_rc_bw_changed:
|
||||
mutex_unlock(&ar->conf_mutex);
|
||||
}
|
||||
|
||||
@ -3433,6 +3477,34 @@ static int ath12k_mac_station_add(struct ath12k *ar,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static u32 ath12k_mac_ieee80211_sta_bw_to_wmi(struct ath12k *ar,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
u32 bw = WMI_PEER_CHWIDTH_20MHZ;
|
||||
|
||||
switch (sta->deflink.bandwidth) {
|
||||
case IEEE80211_STA_RX_BW_20:
|
||||
bw = WMI_PEER_CHWIDTH_20MHZ;
|
||||
break;
|
||||
case IEEE80211_STA_RX_BW_40:
|
||||
bw = WMI_PEER_CHWIDTH_40MHZ;
|
||||
break;
|
||||
case IEEE80211_STA_RX_BW_80:
|
||||
bw = WMI_PEER_CHWIDTH_80MHZ;
|
||||
break;
|
||||
case IEEE80211_STA_RX_BW_160:
|
||||
bw = WMI_PEER_CHWIDTH_160MHZ;
|
||||
break;
|
||||
default:
|
||||
ath12k_warn(ar->ab, "Invalid bandwidth %d in rc update for %pM\n",
|
||||
sta->deflink.bandwidth, sta->addr);
|
||||
bw = WMI_PEER_CHWIDTH_20MHZ;
|
||||
break;
|
||||
}
|
||||
|
||||
return bw;
|
||||
}
|
||||
|
||||
static int ath12k_mac_op_sta_state(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
@ -3498,6 +3570,13 @@ static int ath12k_mac_op_sta_state(struct ieee80211_hw *hw,
|
||||
if (ret)
|
||||
ath12k_warn(ar->ab, "Failed to associate station: %pM\n",
|
||||
sta->addr);
|
||||
|
||||
spin_lock_bh(&ar->data_lock);
|
||||
|
||||
arsta->bw = ath12k_mac_ieee80211_sta_bw_to_wmi(ar, sta);
|
||||
arsta->bw_prev = sta->deflink.bandwidth;
|
||||
|
||||
spin_unlock_bh(&ar->data_lock);
|
||||
} else if (old_state == IEEE80211_STA_ASSOC &&
|
||||
new_state == IEEE80211_STA_AUTHORIZED) {
|
||||
spin_lock_bh(&ar->ab->base_lock);
|
||||
@ -3607,28 +3686,8 @@ static void ath12k_mac_op_sta_rc_update(struct ieee80211_hw *hw,
|
||||
spin_lock_bh(&ar->data_lock);
|
||||
|
||||
if (changed & IEEE80211_RC_BW_CHANGED) {
|
||||
bw = WMI_PEER_CHWIDTH_20MHZ;
|
||||
|
||||
switch (sta->deflink.bandwidth) {
|
||||
case IEEE80211_STA_RX_BW_20:
|
||||
bw = WMI_PEER_CHWIDTH_20MHZ;
|
||||
break;
|
||||
case IEEE80211_STA_RX_BW_40:
|
||||
bw = WMI_PEER_CHWIDTH_40MHZ;
|
||||
break;
|
||||
case IEEE80211_STA_RX_BW_80:
|
||||
bw = WMI_PEER_CHWIDTH_80MHZ;
|
||||
break;
|
||||
case IEEE80211_STA_RX_BW_160:
|
||||
bw = WMI_PEER_CHWIDTH_160MHZ;
|
||||
break;
|
||||
default:
|
||||
ath12k_warn(ar->ab, "Invalid bandwidth %d in rc update for %pM\n",
|
||||
sta->deflink.bandwidth, sta->addr);
|
||||
bw = WMI_PEER_CHWIDTH_20MHZ;
|
||||
break;
|
||||
}
|
||||
|
||||
bw = ath12k_mac_ieee80211_sta_bw_to_wmi(ar, sta);
|
||||
arsta->bw_prev = arsta->bw;
|
||||
arsta->bw = bw;
|
||||
}
|
||||
|
||||
|
@ -755,14 +755,12 @@ static int ath12k_pci_claim(struct ath12k_pci *ab_pci, struct pci_dev *pdev)
|
||||
if (!ab->mem) {
|
||||
ath12k_err(ab, "failed to map pci bar %d\n", ATH12K_PCI_BAR_NUM);
|
||||
ret = -EIO;
|
||||
goto clear_master;
|
||||
goto release_region;
|
||||
}
|
||||
|
||||
ath12k_dbg(ab, ATH12K_DBG_BOOT, "boot pci_mem 0x%pK\n", ab->mem);
|
||||
return 0;
|
||||
|
||||
clear_master:
|
||||
pci_clear_master(pdev);
|
||||
release_region:
|
||||
pci_release_region(pdev, ATH12K_PCI_BAR_NUM);
|
||||
disable_device:
|
||||
@ -778,7 +776,6 @@ static void ath12k_pci_free_region(struct ath12k_pci *ab_pci)
|
||||
|
||||
pci_iounmap(pci_dev, ab->mem);
|
||||
ab->mem = NULL;
|
||||
pci_clear_master(pci_dev);
|
||||
pci_release_region(pci_dev, ATH12K_PCI_BAR_NUM);
|
||||
if (pci_is_enabled(pci_dev))
|
||||
pci_disable_device(pci_dev);
|
||||
@ -1223,7 +1220,8 @@ static int ath12k_pci_probe(struct pci_dev *pdev,
|
||||
dev_err(&pdev->dev,
|
||||
"Unknown hardware version found for QCN9274: 0x%x\n",
|
||||
soc_hw_version_major);
|
||||
return -EOPNOTSUPP;
|
||||
ret = -EOPNOTSUPP;
|
||||
goto err_pci_free_region;
|
||||
}
|
||||
break;
|
||||
case WCN7850_DEVICE_ID:
|
||||
|
@ -2991,7 +2991,7 @@ static void ath12k_qmi_driver_event_work(struct work_struct *work)
|
||||
spin_unlock(&qmi->event_lock);
|
||||
|
||||
if (test_bit(ATH12K_FLAG_UNREGISTERING, &ab->dev_flags))
|
||||
return;
|
||||
goto skip;
|
||||
|
||||
switch (event->type) {
|
||||
case ATH12K_QMI_EVENT_SERVER_ARRIVE:
|
||||
@ -3032,6 +3032,8 @@ static void ath12k_qmi_driver_event_work(struct work_struct *work)
|
||||
ath12k_warn(ab, "invalid event type: %d", event->type);
|
||||
break;
|
||||
}
|
||||
|
||||
skip:
|
||||
kfree(event);
|
||||
spin_lock(&qmi->event_lock);
|
||||
}
|
||||
|
@ -2438,6 +2438,9 @@ int ath12k_wmi_send_scan_chan_list_cmd(struct ath12k *ar,
|
||||
if (channel_arg->psc_channel)
|
||||
chan_info->info |= cpu_to_le32(WMI_CHAN_INFO_PSC);
|
||||
|
||||
if (channel_arg->dfs_set)
|
||||
chan_info->info |= cpu_to_le32(WMI_CHAN_INFO_DFS);
|
||||
|
||||
chan_info->info |= le32_encode_bits(channel_arg->phy_mode,
|
||||
WMI_CHAN_INFO_MODE);
|
||||
*reg1 |= le32_encode_bits(channel_arg->minpower,
|
||||
@ -4934,6 +4937,9 @@ static int freq_to_idx(struct ath12k *ar, int freq)
|
||||
int band, ch, idx = 0;
|
||||
|
||||
for (band = NL80211_BAND_2GHZ; band < NUM_NL80211_BANDS; band++) {
|
||||
if (!ar->mac.sbands[band].channels)
|
||||
continue;
|
||||
|
||||
sband = ar->hw->wiphy->bands[band];
|
||||
if (!sband)
|
||||
continue;
|
||||
|
@ -42,8 +42,6 @@ static const struct usb_device_id ath9k_hif_usb_ids[] = {
|
||||
|
||||
{ USB_DEVICE(0x0cf3, 0x7015),
|
||||
.driver_info = AR9287_USB }, /* Atheros */
|
||||
{ USB_DEVICE(0x1668, 0x1200),
|
||||
.driver_info = AR9287_USB }, /* Verizon */
|
||||
|
||||
{ USB_DEVICE(0x0cf3, 0x7010),
|
||||
.driver_info = AR9280_USB }, /* Atheros */
|
||||
|
@ -34,6 +34,12 @@
|
||||
#define NUM_SYMBOLS_PER_USEC(_usec) (_usec >> 2)
|
||||
#define NUM_SYMBOLS_PER_USEC_HALFGI(_usec) (((_usec*5)-4)/18)
|
||||
|
||||
/* Shifts in ar5008_phy.c and ar9003_phy.c are equal for all revisions */
|
||||
#define ATH9K_PWRTBL_11NA_OFDM_SHIFT 0
|
||||
#define ATH9K_PWRTBL_11NG_OFDM_SHIFT 4
|
||||
#define ATH9K_PWRTBL_11NA_HT_SHIFT 8
|
||||
#define ATH9K_PWRTBL_11NG_HT_SHIFT 12
|
||||
|
||||
|
||||
static u16 bits_per_symbol[][2] = {
|
||||
/* 20MHz 40MHz */
|
||||
@ -1169,13 +1175,14 @@ void ath_update_max_aggr_framelen(struct ath_softc *sc, int queue, int txop)
|
||||
}
|
||||
|
||||
static u8 ath_get_rate_txpower(struct ath_softc *sc, struct ath_buf *bf,
|
||||
u8 rateidx, bool is_40, bool is_cck)
|
||||
u8 rateidx, bool is_40, bool is_cck, bool is_mcs)
|
||||
{
|
||||
u8 max_power;
|
||||
struct sk_buff *skb;
|
||||
struct ath_frame_info *fi;
|
||||
struct ieee80211_tx_info *info;
|
||||
struct ath_hw *ah = sc->sc_ah;
|
||||
bool is_2ghz, is_5ghz, use_stbc;
|
||||
|
||||
if (sc->tx99_state || !ah->tpc_enabled)
|
||||
return MAX_RATE_POWER;
|
||||
@ -1184,6 +1191,19 @@ static u8 ath_get_rate_txpower(struct ath_softc *sc, struct ath_buf *bf,
|
||||
fi = get_frame_info(skb);
|
||||
info = IEEE80211_SKB_CB(skb);
|
||||
|
||||
is_2ghz = info->band == NL80211_BAND_2GHZ;
|
||||
is_5ghz = info->band == NL80211_BAND_5GHZ;
|
||||
use_stbc = is_mcs && rateidx < 8 && (info->flags &
|
||||
IEEE80211_TX_CTL_STBC);
|
||||
|
||||
if (is_mcs)
|
||||
rateidx += is_5ghz ? ATH9K_PWRTBL_11NA_HT_SHIFT
|
||||
: ATH9K_PWRTBL_11NG_HT_SHIFT;
|
||||
else if (is_2ghz && !is_cck)
|
||||
rateidx += ATH9K_PWRTBL_11NG_OFDM_SHIFT;
|
||||
else
|
||||
rateidx += ATH9K_PWRTBL_11NA_OFDM_SHIFT;
|
||||
|
||||
if (!AR_SREV_9300_20_OR_LATER(ah)) {
|
||||
int txpower = fi->tx_power;
|
||||
|
||||
@ -1193,10 +1213,8 @@ static u8 ath_get_rate_txpower(struct ath_softc *sc, struct ath_buf *bf,
|
||||
u16 eeprom_rev = ah->eep_ops->get_eeprom_rev(ah);
|
||||
|
||||
if (eeprom_rev >= AR5416_EEP_MINOR_VER_2) {
|
||||
bool is_2ghz;
|
||||
struct modal_eep_header *pmodal;
|
||||
|
||||
is_2ghz = info->band == NL80211_BAND_2GHZ;
|
||||
pmodal = &eep->modalHeader[is_2ghz];
|
||||
power_ht40delta = pmodal->ht40PowerIncForPdadc;
|
||||
} else {
|
||||
@ -1229,7 +1247,7 @@ static u8 ath_get_rate_txpower(struct ath_softc *sc, struct ath_buf *bf,
|
||||
if (!max_power && !AR_SREV_9280_20_OR_LATER(ah))
|
||||
max_power = 1;
|
||||
} else if (!bf->bf_state.bfs_paprd) {
|
||||
if (rateidx < 8 && (info->flags & IEEE80211_TX_CTL_STBC))
|
||||
if (use_stbc)
|
||||
max_power = min_t(u8, ah->tx_power_stbc[rateidx],
|
||||
fi->tx_power);
|
||||
else
|
||||
@ -1319,7 +1337,7 @@ static void ath_buf_set_rate(struct ath_softc *sc, struct ath_buf *bf,
|
||||
}
|
||||
|
||||
info->txpower[i] = ath_get_rate_txpower(sc, bf, rix,
|
||||
is_40, false);
|
||||
is_40, false, true);
|
||||
continue;
|
||||
}
|
||||
|
||||
@ -1350,7 +1368,7 @@ static void ath_buf_set_rate(struct ath_softc *sc, struct ath_buf *bf,
|
||||
|
||||
is_cck = IS_CCK_RATE(info->rates[i].Rate);
|
||||
info->txpower[i] = ath_get_rate_txpower(sc, bf, rix, false,
|
||||
is_cck);
|
||||
is_cck, false);
|
||||
}
|
||||
|
||||
/* For AR5416 - RTS cannot be followed by a frame larger than 8K */
|
||||
|
@ -120,7 +120,7 @@ struct carl9170_cmd *carl9170_cmd_buf(struct ar9170 *ar,
|
||||
{
|
||||
struct carl9170_cmd *tmp;
|
||||
|
||||
tmp = kzalloc(sizeof(struct carl9170_cmd_head) + len, GFP_ATOMIC);
|
||||
tmp = kzalloc(sizeof(*tmp), GFP_ATOMIC);
|
||||
if (tmp) {
|
||||
tmp->hdr.cmd = cmd;
|
||||
tmp->hdr.len = len;
|
||||
|
@ -320,9 +320,9 @@ struct carl9170_rsp {
|
||||
struct carl9170_u32_list rreg_res;
|
||||
struct carl9170_u32_list echo;
|
||||
#ifdef __CARL9170FW__
|
||||
struct carl9170_tx_status tx_status[0];
|
||||
DECLARE_FLEX_ARRAY(struct carl9170_tx_status, tx_status);
|
||||
#endif /* __CARL9170FW__ */
|
||||
struct _carl9170_tx_status _tx_status[0];
|
||||
DECLARE_FLEX_ARRAY(struct _carl9170_tx_status, _tx_status);
|
||||
struct carl9170_gpio gpio;
|
||||
struct carl9170_tsf_rsp tsf;
|
||||
struct carl9170_psm psm;
|
||||
|
@ -112,8 +112,8 @@ int wcn36xx_dxe_alloc_ctl_blks(struct wcn36xx *wcn)
|
||||
wcn->dxe_rx_l_ch.desc_num = WCN36XX_DXE_CH_DESC_NUMB_RX_L;
|
||||
wcn->dxe_rx_h_ch.desc_num = WCN36XX_DXE_CH_DESC_NUMB_RX_H;
|
||||
|
||||
wcn->dxe_tx_l_ch.dxe_wq = WCN36XX_DXE_WQ_TX_L;
|
||||
wcn->dxe_tx_h_ch.dxe_wq = WCN36XX_DXE_WQ_TX_H;
|
||||
wcn->dxe_tx_l_ch.dxe_wq = WCN36XX_DXE_WQ_TX_L(wcn);
|
||||
wcn->dxe_tx_h_ch.dxe_wq = WCN36XX_DXE_WQ_TX_H(wcn);
|
||||
|
||||
wcn->dxe_tx_l_ch.ctrl_bd = WCN36XX_DXE_CTRL_TX_L_BD;
|
||||
wcn->dxe_tx_h_ch.ctrl_bd = WCN36XX_DXE_CTRL_TX_H_BD;
|
||||
@ -165,8 +165,9 @@ void wcn36xx_dxe_free_ctl_blks(struct wcn36xx *wcn)
|
||||
wcn36xx_dxe_free_ctl_block(&wcn->dxe_rx_h_ch);
|
||||
}
|
||||
|
||||
static int wcn36xx_dxe_init_descs(struct device *dev, struct wcn36xx_dxe_ch *wcn_ch)
|
||||
static int wcn36xx_dxe_init_descs(struct wcn36xx *wcn, struct wcn36xx_dxe_ch *wcn_ch)
|
||||
{
|
||||
struct device *dev = wcn->dev;
|
||||
struct wcn36xx_dxe_desc *cur_dxe = NULL;
|
||||
struct wcn36xx_dxe_desc *prev_dxe = NULL;
|
||||
struct wcn36xx_dxe_ctl *cur_ctl = NULL;
|
||||
@ -190,11 +191,11 @@ static int wcn36xx_dxe_init_descs(struct device *dev, struct wcn36xx_dxe_ch *wcn
|
||||
switch (wcn_ch->ch_type) {
|
||||
case WCN36XX_DXE_CH_TX_L:
|
||||
cur_dxe->ctrl = WCN36XX_DXE_CTRL_TX_L;
|
||||
cur_dxe->dst_addr_l = WCN36XX_DXE_WQ_TX_L;
|
||||
cur_dxe->dst_addr_l = WCN36XX_DXE_WQ_TX_L(wcn);
|
||||
break;
|
||||
case WCN36XX_DXE_CH_TX_H:
|
||||
cur_dxe->ctrl = WCN36XX_DXE_CTRL_TX_H;
|
||||
cur_dxe->dst_addr_l = WCN36XX_DXE_WQ_TX_H;
|
||||
cur_dxe->dst_addr_l = WCN36XX_DXE_WQ_TX_H(wcn);
|
||||
break;
|
||||
case WCN36XX_DXE_CH_RX_L:
|
||||
cur_dxe->ctrl = WCN36XX_DXE_CTRL_RX_L;
|
||||
@ -914,7 +915,7 @@ int wcn36xx_dxe_init(struct wcn36xx *wcn)
|
||||
/***************************************/
|
||||
/* Init descriptors for TX LOW channel */
|
||||
/***************************************/
|
||||
ret = wcn36xx_dxe_init_descs(wcn->dev, &wcn->dxe_tx_l_ch);
|
||||
ret = wcn36xx_dxe_init_descs(wcn, &wcn->dxe_tx_l_ch);
|
||||
if (ret) {
|
||||
dev_err(wcn->dev, "Error allocating descriptor\n");
|
||||
return ret;
|
||||
@ -928,14 +929,14 @@ int wcn36xx_dxe_init(struct wcn36xx *wcn)
|
||||
/* Program DMA destination addr for TX LOW */
|
||||
wcn36xx_dxe_write_register(wcn,
|
||||
WCN36XX_DXE_CH_DEST_ADDR_TX_L,
|
||||
WCN36XX_DXE_WQ_TX_L);
|
||||
WCN36XX_DXE_WQ_TX_L(wcn));
|
||||
|
||||
wcn36xx_dxe_read_register(wcn, WCN36XX_DXE_REG_CH_EN, ®_data);
|
||||
|
||||
/***************************************/
|
||||
/* Init descriptors for TX HIGH channel */
|
||||
/***************************************/
|
||||
ret = wcn36xx_dxe_init_descs(wcn->dev, &wcn->dxe_tx_h_ch);
|
||||
ret = wcn36xx_dxe_init_descs(wcn, &wcn->dxe_tx_h_ch);
|
||||
if (ret) {
|
||||
dev_err(wcn->dev, "Error allocating descriptor\n");
|
||||
goto out_err_txh_ch;
|
||||
@ -950,14 +951,14 @@ int wcn36xx_dxe_init(struct wcn36xx *wcn)
|
||||
/* Program DMA destination addr for TX HIGH */
|
||||
wcn36xx_dxe_write_register(wcn,
|
||||
WCN36XX_DXE_CH_DEST_ADDR_TX_H,
|
||||
WCN36XX_DXE_WQ_TX_H);
|
||||
WCN36XX_DXE_WQ_TX_H(wcn));
|
||||
|
||||
wcn36xx_dxe_read_register(wcn, WCN36XX_DXE_REG_CH_EN, ®_data);
|
||||
|
||||
/***************************************/
|
||||
/* Init descriptors for RX LOW channel */
|
||||
/***************************************/
|
||||
ret = wcn36xx_dxe_init_descs(wcn->dev, &wcn->dxe_rx_l_ch);
|
||||
ret = wcn36xx_dxe_init_descs(wcn, &wcn->dxe_rx_l_ch);
|
||||
if (ret) {
|
||||
dev_err(wcn->dev, "Error allocating descriptor\n");
|
||||
goto out_err_rxl_ch;
|
||||
@ -988,7 +989,7 @@ int wcn36xx_dxe_init(struct wcn36xx *wcn)
|
||||
/***************************************/
|
||||
/* Init descriptors for RX HIGH channel */
|
||||
/***************************************/
|
||||
ret = wcn36xx_dxe_init_descs(wcn->dev, &wcn->dxe_rx_h_ch);
|
||||
ret = wcn36xx_dxe_init_descs(wcn, &wcn->dxe_rx_h_ch);
|
||||
if (ret) {
|
||||
dev_err(wcn->dev, "Error allocating descriptor\n");
|
||||
goto out_err_rxh_ch;
|
||||
|
@ -135,8 +135,8 @@ H2H_TEST_RX_TX = DMA2
|
||||
WCN36xx_DXE_CTRL_ENDIANNESS)
|
||||
|
||||
/* TODO This must calculated properly but not hardcoded */
|
||||
#define WCN36XX_DXE_WQ_TX_L 0x17
|
||||
#define WCN36XX_DXE_WQ_TX_H 0x17
|
||||
#define WCN36XX_DXE_WQ_TX_L(wcn) ((wcn)->is_pronto_v3 ? 0x6 : 0x17)
|
||||
#define WCN36XX_DXE_WQ_TX_H(wcn) ((wcn)->is_pronto_v3 ? 0x6 : 0x17)
|
||||
#define WCN36XX_DXE_WQ_RX_L 0xB
|
||||
#define WCN36XX_DXE_WQ_RX_H 0x4
|
||||
|
||||
|
@ -1508,6 +1508,7 @@ static int wcn36xx_platform_get_resources(struct wcn36xx *wcn,
|
||||
}
|
||||
|
||||
wcn->is_pronto = !!of_device_is_compatible(mmio_node, "qcom,pronto");
|
||||
wcn->is_pronto_v3 = !!of_device_is_compatible(mmio_node, "qcom,pronto-v3-pil");
|
||||
|
||||
/* Map the CCU memory */
|
||||
index = of_property_match_string(mmio_node, "reg-names", "ccu");
|
||||
|
@ -217,6 +217,7 @@ struct wcn36xx {
|
||||
u8 fw_major;
|
||||
u32 fw_feat_caps[WCN36XX_HAL_CAPS_SIZE];
|
||||
bool is_pronto;
|
||||
bool is_pronto_v3;
|
||||
|
||||
/* extra byte for the NULL termination */
|
||||
u8 crm_version[WCN36XX_HAL_VERSION_LENGTH + 1];
|
||||
|
@ -965,6 +965,12 @@ int brcmf_sdiod_probe(struct brcmf_sdio_dev *sdiodev)
|
||||
.driver_data = BRCMF_FWVENDOR_ ## fw_vend \
|
||||
}
|
||||
|
||||
#define CYW_SDIO_DEVICE(dev_id, fw_vend) \
|
||||
{ \
|
||||
SDIO_DEVICE(SDIO_VENDOR_ID_CYPRESS, dev_id), \
|
||||
.driver_data = BRCMF_FWVENDOR_ ## fw_vend \
|
||||
}
|
||||
|
||||
/* devices we support, null terminated */
|
||||
static const struct sdio_device_id brcmf_sdmmc_ids[] = {
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_43143, WCC),
|
||||
@ -979,6 +985,7 @@ static const struct sdio_device_id brcmf_sdmmc_ids[] = {
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_4335_4339, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_4339, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_43430, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_43439, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_4345, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_43455, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_4354, WCC),
|
||||
@ -986,9 +993,9 @@ static const struct sdio_device_id brcmf_sdmmc_ids[] = {
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_4359, WCC),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_CYPRESS_4373, CYW),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_CYPRESS_43012, CYW),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_CYPRESS_43439, CYW),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_CYPRESS_43752, CYW),
|
||||
BRCMF_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_CYPRESS_89359, CYW),
|
||||
CYW_SDIO_DEVICE(SDIO_DEVICE_ID_BROADCOM_CYPRESS_43439, CYW),
|
||||
{ /* end: all zeroes */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(sdio, brcmf_sdmmc_ids);
|
||||
|
@ -14,7 +14,7 @@ if WLAN_VENDOR_CISCO
|
||||
|
||||
config AIRO
|
||||
tristate "Cisco/Aironet 34X/35X/4500/4800 ISA and PCI cards"
|
||||
depends on CFG80211 && ISA_DMA_API && (PCI || BROKEN)
|
||||
depends on CFG80211 && (PCI || BROKEN)
|
||||
select WIRELESS_EXT
|
||||
select CRYPTO
|
||||
select CRYPTO_SKCIPHER
|
||||
|
@ -10,7 +10,7 @@
|
||||
#include "fw/api/txq.h"
|
||||
|
||||
/* Highest firmware API version supported */
|
||||
#define IWL_22000_UCODE_API_MAX 75
|
||||
#define IWL_22000_UCODE_API_MAX 78
|
||||
|
||||
/* Lowest firmware API version supported */
|
||||
#define IWL_22000_UCODE_API_MIN 39
|
||||
@ -50,7 +50,13 @@
|
||||
#define IWL_MA_A_GF4_A_FW_PRE "iwlwifi-ma-a0-gf4-a0-"
|
||||
#define IWL_MA_A_MR_A_FW_PRE "iwlwifi-ma-a0-mr-a0-"
|
||||
#define IWL_MA_A_FM_A_FW_PRE "iwlwifi-ma-a0-fm-a0-"
|
||||
#define IWL_MA_B_HR_B_FW_PRE "iwlwifi-ma-b0-hr-b0-"
|
||||
#define IWL_MA_B_GF_A_FW_PRE "iwlwifi-ma-b0-gf-a0-"
|
||||
#define IWL_MA_B_GF4_A_FW_PRE "iwlwifi-ma-b0-gf4-a0-"
|
||||
#define IWL_MA_B_MR_A_FW_PRE "iwlwifi-ma-b0-mr-a0-"
|
||||
#define IWL_MA_B_FM_A_FW_PRE "iwlwifi-ma-b0-fm-a0-"
|
||||
#define IWL_SNJ_A_MR_A_FW_PRE "iwlwifi-SoSnj-a0-mr-a0-"
|
||||
#define IWL_BZ_A_HR_A_FW_PRE "iwlwifi-bz-a0-hr-b0-"
|
||||
#define IWL_BZ_A_HR_B_FW_PRE "iwlwifi-bz-a0-hr-b0-"
|
||||
#define IWL_BZ_A_GF_A_FW_PRE "iwlwifi-bz-a0-gf-a0-"
|
||||
#define IWL_BZ_A_GF4_A_FW_PRE "iwlwifi-bz-a0-gf4-a0-"
|
||||
@ -69,7 +75,9 @@
|
||||
#define IWL_BNJ_B_GF_A_FW_PRE "iwlwifi-BzBnj-b0-gf-a0-"
|
||||
#define IWL_BNJ_A_GF4_A_FW_PRE "iwlwifi-BzBnj-a0-gf4-a0-"
|
||||
#define IWL_BNJ_B_GF4_A_FW_PRE "iwlwifi-BzBnj-b0-gf4-a0-"
|
||||
#define IWL_BNJ_A_HR_A_FW_PRE "iwlwifi-BzBnj-a0-hr-b0-"
|
||||
#define IWL_BNJ_A_HR_B_FW_PRE "iwlwifi-BzBnj-a0-hr-b0-"
|
||||
#define IWL_BNJ_B_HR_A_FW_PRE "iwlwifi-BzBnj-b0-hr-b0-"
|
||||
#define IWL_BNJ_B_HR_B_FW_PRE "iwlwifi-BzBnj-b0-hr-b0-"
|
||||
#define IWL_BNJ_B_FM_B_FW_PRE "iwlwifi-BzBnj-b0-fm-b0-"
|
||||
|
||||
@ -116,8 +124,20 @@
|
||||
IWL_MA_A_MR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_MA_A_FM_A_FW_MODULE_FIRMWARE(api) \
|
||||
IWL_MA_A_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_MA_B_HR_B_FW_MODULE_FIRMWARE(api) \
|
||||
IWL_MA_B_HR_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_MA_B_GF_A_FW_MODULE_FIRMWARE(api) \
|
||||
IWL_MA_B_GF_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_MA_B_GF4_A_FW_MODULE_FIRMWARE(api) \
|
||||
IWL_MA_B_GF4_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_MA_B_MR_A_FW_MODULE_FIRMWARE(api) \
|
||||
IWL_MA_B_MR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_MA_B_FM_A_FW_MODULE_FIRMWARE(api) \
|
||||
IWL_MA_B_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_SNJ_A_MR_A_MODULE_FIRMWARE(api) \
|
||||
IWL_SNJ_A_MR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_HR_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_HR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_HR_B_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_HR_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_GF_A_MODULE_FIRMWARE(api) \
|
||||
@ -127,17 +147,17 @@
|
||||
#define IWL_BZ_A_MR_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_MR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_FM_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
IWL_BZ_A_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_FM4_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_FM4_A_FW_PRE __stringify(api) ".ucode"
|
||||
IWL_BZ_A_FM4_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_FM_B_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_FM_B_FW_PRE __stringify(api) ".ucode"
|
||||
IWL_BZ_A_FM_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BZ_A_FM4_B_MODULE_FIRMWARE(api) \
|
||||
IWL_BZ_A_FM4_B_FW_PRE __stringify(api) ".ucode"
|
||||
IWL_BZ_A_FM4_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_GL_A_FM_A_MODULE_FIRMWARE(api) \
|
||||
IWL_GL_A_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
IWL_GL_A_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_GL_B_FM_B_MODULE_FIRMWARE(api) \
|
||||
IWL_GL_B_FM_B_FW_PRE __stringify(api) ".ucode"
|
||||
IWL_GL_B_FM_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_A_FM_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BNJ_A_FM_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_A_FM4_A_MODULE_FIRMWARE(api) \
|
||||
@ -152,8 +172,12 @@
|
||||
IWL_BNJ_A_GF4_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_B_GF4_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BNJ_B_GF4_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_A_HR_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BNJ_A_HR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_A_HR_B_MODULE_FIRMWARE(api) \
|
||||
IWL_BNJ_A_HR_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_B_HR_A_MODULE_FIRMWARE(api) \
|
||||
IWL_BNJ_B_HR_A_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_B_HR_B_MODULE_FIRMWARE(api) \
|
||||
IWL_BNJ_B_HR_B_FW_PRE __stringify(api) ".ucode"
|
||||
#define IWL_BNJ_B_FM_B_MODULE_FIRMWARE(api) \
|
||||
@ -882,6 +906,41 @@ const struct iwl_cfg iwl_cfg_ma_a0_ms_a0 = {
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_ma_b0_fm_a0 = {
|
||||
.fw_name_pre = IWL_MA_B_FM_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_AX210,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_ma_b0_hr_b0 = {
|
||||
.fw_name_pre = IWL_MA_B_HR_B_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_AX210,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_ma_b0_gf_a0 = {
|
||||
.fw_name_pre = IWL_MA_B_GF_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_AX210,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_ma_b0_gf4_a0 = {
|
||||
.fw_name_pre = IWL_MA_B_GF4_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_AX210,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_ma_b0_mr_a0 = {
|
||||
.fw_name_pre = IWL_MA_B_MR_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_AX210,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_so_a0_ms_a0 = {
|
||||
.fw_name_pre = IWL_SO_A_MR_A_FW_PRE,
|
||||
.uhb_supported = false,
|
||||
@ -928,6 +987,14 @@ const struct iwl_cfg iwl_cfg_quz_a0_hr_b0 = {
|
||||
.num_rbds = IWL_NUM_RBDS_22000_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_bz_a0_hr_a0 = {
|
||||
.fw_name_pre = IWL_BZ_A_HR_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_BZ,
|
||||
.features = IWL_TX_CSUM_NETIF_FLAGS_BZ | NETIF_F_RXCSUM,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_bz_a0_hr_b0 = {
|
||||
.fw_name_pre = IWL_BZ_A_HR_B_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
@ -1072,6 +1139,14 @@ const struct iwl_cfg iwl_cfg_bnj_b0_gf4_a0 = {
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_bnj_a0_hr_a0 = {
|
||||
.fw_name_pre = IWL_BNJ_A_HR_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_BZ,
|
||||
.features = IWL_TX_CSUM_NETIF_FLAGS | NETIF_F_RXCSUM,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_bnj_a0_hr_b0 = {
|
||||
.fw_name_pre = IWL_BNJ_A_HR_B_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
@ -1080,6 +1155,14 @@ const struct iwl_cfg iwl_cfg_bnj_a0_hr_b0 = {
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_bnj_b0_hr_a0 = {
|
||||
.fw_name_pre = IWL_BNJ_B_HR_A_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
IWL_DEVICE_BZ,
|
||||
.features = IWL_TX_CSUM_NETIF_FLAGS | NETIF_F_RXCSUM,
|
||||
.num_rbds = IWL_NUM_RBDS_AX210_HE,
|
||||
};
|
||||
|
||||
const struct iwl_cfg iwl_cfg_bnj_b0_hr_b0 = {
|
||||
.fw_name_pre = IWL_BNJ_B_HR_B_FW_PRE,
|
||||
.uhb_supported = true,
|
||||
@ -1116,7 +1199,13 @@ MODULE_FIRMWARE(IWL_MA_A_GF_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_A_GF4_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_A_MR_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_A_FM_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_B_HR_B_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_B_GF_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_B_GF4_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_B_MR_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_MA_B_FM_A_FW_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_SNJ_A_MR_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BZ_A_HR_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BZ_A_HR_B_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BZ_A_GF_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BZ_A_GF4_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
@ -1132,6 +1221,7 @@ MODULE_FIRMWARE(IWL_BNJ_B_GF_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BNJ_A_GF4_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BNJ_B_GF4_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BNJ_A_HR_B_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BNJ_B_HR_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BNJ_B_HR_B_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BZ_A_FM4_A_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
MODULE_FIRMWARE(IWL_BZ_A_FM4_B_MODULE_FIRMWARE(IWL_22000_UCODE_API_MAX));
|
||||
|
@ -1081,6 +1081,7 @@ static int iwlagn_send_sta_key(struct iwl_priv *priv,
|
||||
{
|
||||
__le16 key_flags;
|
||||
struct iwl_addsta_cmd sta_cmd;
|
||||
size_t to_copy;
|
||||
int i;
|
||||
|
||||
spin_lock_bh(&priv->sta_lock);
|
||||
@ -1100,7 +1101,9 @@ static int iwlagn_send_sta_key(struct iwl_priv *priv,
|
||||
sta_cmd.key.tkip_rx_tsc_byte2 = tkip_iv32;
|
||||
for (i = 0; i < 5; i++)
|
||||
sta_cmd.key.tkip_rx_ttak[i] = cpu_to_le16(tkip_p1k[i]);
|
||||
memcpy(sta_cmd.key.key, keyconf->key, keyconf->keylen);
|
||||
/* keyconf may contain MIC rx/tx keys which iwl does not use */
|
||||
to_copy = min_t(size_t, sizeof(sta_cmd.key.key), keyconf->keylen);
|
||||
memcpy(sta_cmd.key.key, keyconf->key, to_copy);
|
||||
break;
|
||||
case WLAN_CIPHER_SUITE_WEP104:
|
||||
key_flags |= STA_KEY_FLG_KEY_SIZE_MSK;
|
||||
|
@ -1006,8 +1006,10 @@ int iwl_acpi_get_ppag_table(struct iwl_fw_runtime *fwrt)
|
||||
union acpi_object *wifi_pkg, *data, *flags;
|
||||
int i, j, ret, tbl_rev, num_sub_bands = 0;
|
||||
int idx = 2;
|
||||
u8 cmd_ver;
|
||||
|
||||
fwrt->ppag_flags = 0;
|
||||
fwrt->ppag_table_valid = false;
|
||||
|
||||
data = iwl_acpi_get_object(fwrt->dev, ACPI_PPAG_METHOD);
|
||||
if (IS_ERR(data))
|
||||
@ -1054,8 +1056,15 @@ int iwl_acpi_get_ppag_table(struct iwl_fw_runtime *fwrt)
|
||||
}
|
||||
|
||||
fwrt->ppag_flags = flags->integer.value & ACPI_PPAG_MASK;
|
||||
|
||||
if (!fwrt->ppag_flags) {
|
||||
cmd_ver = iwl_fw_lookup_cmd_ver(fwrt->fw,
|
||||
WIDE_ID(PHY_OPS_GROUP,
|
||||
PER_PLATFORM_ANT_GAIN_CMD),
|
||||
IWL_FW_CMD_VER_UNKNOWN);
|
||||
if (cmd_ver == IWL_FW_CMD_VER_UNKNOWN) {
|
||||
ret = -EINVAL;
|
||||
goto out_free;
|
||||
}
|
||||
if (!fwrt->ppag_flags && cmd_ver <= 3) {
|
||||
ret = 0;
|
||||
goto out_free;
|
||||
}
|
||||
@ -1076,21 +1085,22 @@ int iwl_acpi_get_ppag_table(struct iwl_fw_runtime *fwrt)
|
||||
}
|
||||
|
||||
fwrt->ppag_chains[i].subbands[j] = ent->integer.value;
|
||||
|
||||
/* from ver 4 the fw deals with out of range values */
|
||||
if (cmd_ver >= 4)
|
||||
continue;
|
||||
if ((j == 0 &&
|
||||
(fwrt->ppag_chains[i].subbands[j] > ACPI_PPAG_MAX_LB ||
|
||||
fwrt->ppag_chains[i].subbands[j] < ACPI_PPAG_MIN_LB)) ||
|
||||
(j != 0 &&
|
||||
(fwrt->ppag_chains[i].subbands[j] > ACPI_PPAG_MAX_HB ||
|
||||
fwrt->ppag_chains[i].subbands[j] < ACPI_PPAG_MIN_HB))) {
|
||||
fwrt->ppag_flags = 0;
|
||||
ret = -EINVAL;
|
||||
goto out_free;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
fwrt->ppag_table_valid = true;
|
||||
ret = 0;
|
||||
|
||||
out_free:
|
||||
@ -1115,19 +1125,22 @@ int iwl_read_ppag_table(struct iwl_fw_runtime *fwrt, union iwl_ppag_table_cmd *c
|
||||
IWL_DEBUG_RADIO(fwrt,
|
||||
"PPAG capability not supported by FW, command not sent.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
if (!fwrt->ppag_flags) {
|
||||
IWL_DEBUG_RADIO(fwrt, "PPAG not enabled, command not sent.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
cmd_ver = iwl_fw_lookup_cmd_ver(fwrt->fw,
|
||||
WIDE_ID(PHY_OPS_GROUP,
|
||||
PER_PLATFORM_ANT_GAIN_CMD),
|
||||
IWL_FW_CMD_VER_UNKNOWN);
|
||||
if (!fwrt->ppag_table_valid || (cmd_ver <= 3 && !fwrt->ppag_flags)) {
|
||||
IWL_DEBUG_RADIO(fwrt, "PPAG not enabled, command not sent.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* The 'flags' field is the same in v1 and in v2 so we can just
|
||||
* use v1 to access it.
|
||||
*/
|
||||
cmd->v1.flags = cpu_to_le32(fwrt->ppag_flags);
|
||||
cmd_ver = iwl_fw_lookup_cmd_ver(fwrt->fw,
|
||||
WIDE_ID(PHY_OPS_GROUP, PER_PLATFORM_ANT_GAIN_CMD),
|
||||
IWL_FW_CMD_VER_UNKNOWN);
|
||||
|
||||
if (cmd_ver == 1) {
|
||||
num_sub_bands = IWL_NUM_SUB_BANDS_V1;
|
||||
gain = cmd->v1.gain[0];
|
||||
@ -1138,7 +1151,7 @@ int iwl_read_ppag_table(struct iwl_fw_runtime *fwrt, union iwl_ppag_table_cmd *c
|
||||
fwrt->ppag_ver);
|
||||
cmd->v1.flags &= cpu_to_le32(IWL_PPAG_ETSI_MASK);
|
||||
}
|
||||
} else if (cmd_ver == 2 || cmd_ver == 3) {
|
||||
} else if (cmd_ver >= 2 && cmd_ver <= 4) {
|
||||
num_sub_bands = IWL_NUM_SUB_BANDS_V2;
|
||||
gain = cmd->v2.gain[0];
|
||||
*cmd_size = sizeof(cmd->v2);
|
||||
|
@ -767,7 +767,7 @@ struct iwl_wowlan_status_v12 {
|
||||
} __packed; /* WOWLAN_STATUSES_RSP_API_S_VER_12 */
|
||||
|
||||
/**
|
||||
* struct iwl_wowlan_info_notif - WoWLAN information notification
|
||||
* struct iwl_wowlan_info_notif_v1 - WoWLAN information notification
|
||||
* @gtk: GTK data
|
||||
* @igtk: IGTK data
|
||||
* @replay_ctr: GTK rekey replay counter
|
||||
@ -785,7 +785,7 @@ struct iwl_wowlan_status_v12 {
|
||||
* @station_id: station id
|
||||
* @reserved2: reserved
|
||||
*/
|
||||
struct iwl_wowlan_info_notif {
|
||||
struct iwl_wowlan_info_notif_v1 {
|
||||
struct iwl_wowlan_gtk_status_v3 gtk[WOWLAN_GTK_KEYS_NUM];
|
||||
struct iwl_wowlan_igtk_status igtk[WOWLAN_IGTK_KEYS_NUM];
|
||||
__le64 replay_ctr;
|
||||
@ -803,6 +803,39 @@ struct iwl_wowlan_info_notif {
|
||||
u8 reserved2[2];
|
||||
} __packed; /* WOWLAN_INFO_NTFY_API_S_VER_1 */
|
||||
|
||||
/**
|
||||
* struct iwl_wowlan_info_notif - WoWLAN information notification
|
||||
* @gtk: GTK data
|
||||
* @igtk: IGTK data
|
||||
* @replay_ctr: GTK rekey replay counter
|
||||
* @pattern_number: number of the matched patterns
|
||||
* @reserved1: reserved
|
||||
* @qos_seq_ctr: QoS sequence counters to use next
|
||||
* @wakeup_reasons: wakeup reasons, see &enum iwl_wowlan_wakeup_reason
|
||||
* @num_of_gtk_rekeys: number of GTK rekeys
|
||||
* @transmitted_ndps: number of transmitted neighbor discovery packets
|
||||
* @received_beacons: number of received beacons
|
||||
* @tid_tear_down: bit mask of tids whose BA sessions were closed
|
||||
* in suspend state
|
||||
* @station_id: station id
|
||||
* @reserved2: reserved
|
||||
*/
|
||||
struct iwl_wowlan_info_notif {
|
||||
struct iwl_wowlan_gtk_status_v3 gtk[WOWLAN_GTK_KEYS_NUM];
|
||||
struct iwl_wowlan_igtk_status igtk[WOWLAN_IGTK_KEYS_NUM];
|
||||
__le64 replay_ctr;
|
||||
__le16 pattern_number;
|
||||
__le16 reserved1;
|
||||
__le16 qos_seq_ctr[8];
|
||||
__le32 wakeup_reasons;
|
||||
__le32 num_of_gtk_rekeys;
|
||||
__le32 transmitted_ndps;
|
||||
__le32 received_beacons;
|
||||
u8 tid_tear_down;
|
||||
u8 station_id;
|
||||
u8 reserved2[2];
|
||||
} __packed; /* WOWLAN_INFO_NTFY_API_S_VER_2 */
|
||||
|
||||
/**
|
||||
* struct iwl_wowlan_wake_pkt_notif - WoWLAN wake packet notification
|
||||
* @wake_packet_length: wakeup packet length
|
||||
|
@ -223,15 +223,6 @@ struct iwl_mac_client_data {
|
||||
__le32 ctwin;
|
||||
} __packed; /* MAC_CONTEXT_CONFIG_CLIENT_DATA_API_S_VER_1 */
|
||||
|
||||
/**
|
||||
* struct iwl_mac_go_ibss_data - configuration data for GO and IBSS MAC context
|
||||
*
|
||||
* @beacon_template: beacon template ID
|
||||
*/
|
||||
struct iwl_mac_go_ibss_data {
|
||||
__le32 beacon_template;
|
||||
} __packed; /* MAC_CONTEXT_CONFIG_GO_IBSS_DATA_API_S_VER_1 */
|
||||
|
||||
/**
|
||||
* struct iwl_mac_p2p_dev_data - configuration data for P2P device MAC context
|
||||
*
|
||||
@ -278,6 +269,7 @@ enum iwl_mac_config_filter_flags {
|
||||
* @reserved_for_local_mld_addr: reserved
|
||||
* @filter_flags: combination of &enum iwl_mac_config_filter_flags
|
||||
* @he_support: does this MAC support HE
|
||||
* @he_ap_support: HE AP enabled, "pseudo HE", no trigger frame handling
|
||||
* @eht_support: does this MAC support EHT. Requires he_support
|
||||
* @nic_not_ack_enabled: mark that the NIC doesn't support receiving
|
||||
* ACK-enabled AGG, (i.e. both BACK and non-BACK frames in single AGG).
|
||||
@ -296,13 +288,13 @@ struct iwl_mac_config_cmd {
|
||||
u8 local_mld_addr[6];
|
||||
__le16 reserved_for_local_mld_addr;
|
||||
__le32 filter_flags;
|
||||
__le32 he_support;
|
||||
__le16 he_support;
|
||||
__le16 he_ap_support;
|
||||
__le32 eht_support;
|
||||
__le32 nic_not_ack_enabled;
|
||||
/* MAC_CONTEXT_CONFIG_SPECIFIC_DATA_API_U_VER_1 */
|
||||
union {
|
||||
struct iwl_mac_client_data client;
|
||||
struct iwl_mac_go_ibss_data go_ibss;
|
||||
struct iwl_mac_p2p_dev_data p2p_dev;
|
||||
};
|
||||
} __packed; /* MAC_CONTEXT_CONFIG_CMD_API_S_VER_1 */
|
||||
|
@ -709,10 +709,13 @@ enum iwl_umac_scan_general_flags_v2 {
|
||||
* should be aware of a P2P GO operation on the 2GHz band.
|
||||
* @IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_RESPECT_P2P_GO_HB: scan event scheduling
|
||||
* should be aware of a P2P GO operation on the 5GHz or 6GHz band.
|
||||
* @IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_DONT_TOGGLE_ANT: don't toggle between
|
||||
* valid antennas, and use the same antenna as in previous scan
|
||||
*/
|
||||
enum iwl_umac_scan_general_params_flags2 {
|
||||
IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_RESPECT_P2P_GO_LB = BIT(0),
|
||||
IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_RESPECT_P2P_GO_HB = BIT(1),
|
||||
IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_DONT_TOGGLE_ANT = BIT(2),
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1038,7 +1038,7 @@ iwl_dump_ini_prph_mac_iter(struct iwl_fw_runtime *fwrt,
|
||||
range->range_data_size = reg->dev_addr.size;
|
||||
for (i = 0; i < le32_to_cpu(reg->dev_addr.size); i += 4) {
|
||||
prph_val = iwl_read_prph(fwrt->trans, addr + i);
|
||||
if (prph_val == 0x5a5a5a5a)
|
||||
if ((prph_val & ~0xf) == 0xa5a5a5a0)
|
||||
return -EBUSY;
|
||||
*val++ = cpu_to_le32(prph_val);
|
||||
}
|
||||
@ -1388,13 +1388,13 @@ static void iwl_ini_get_rxf_data(struct iwl_fw_runtime *fwrt,
|
||||
if (!data)
|
||||
return;
|
||||
|
||||
memset(data, 0, sizeof(*data));
|
||||
|
||||
/* make sure only one bit is set in only one fid */
|
||||
if (WARN_ONCE(hweight_long(fid1) + hweight_long(fid2) != 1,
|
||||
"fid1=%x, fid2=%x\n", fid1, fid2))
|
||||
return;
|
||||
|
||||
memset(data, 0, sizeof(*data));
|
||||
|
||||
if (fid1) {
|
||||
fifo_idx = ffs(fid1) - 1;
|
||||
if (WARN_ONCE(fifo_idx >= MAX_NUM_LMAC, "fifo_idx=%d\n",
|
||||
@ -1562,7 +1562,7 @@ iwl_dump_ini_dbgi_sram_iter(struct iwl_fw_runtime *fwrt,
|
||||
prph_data = iwl_read_prph_no_grab(fwrt->trans, (i % 2) ?
|
||||
DBGI_SRAM_TARGET_ACCESS_RDATA_MSB :
|
||||
DBGI_SRAM_TARGET_ACCESS_RDATA_LSB);
|
||||
if (prph_data == 0x5a5a5a5a) {
|
||||
if ((prph_data & ~0xf) == 0xa5a5a5a0) {
|
||||
iwl_trans_release_nic_access(fwrt->trans);
|
||||
return -EBUSY;
|
||||
}
|
||||
@ -2345,6 +2345,8 @@ static u32 iwl_dump_ini_file_name_info(struct iwl_fw_runtime *fwrt,
|
||||
/* add the dump file name extension tlv to the list */
|
||||
list_add_tail(&entry->list, list);
|
||||
|
||||
fwrt->trans->dbg.dump_file_name_ext_valid = false;
|
||||
|
||||
return entry->size;
|
||||
}
|
||||
|
||||
|
@ -317,8 +317,10 @@ static void *iwl_dbgfs_fw_info_seq_next(struct seq_file *seq,
|
||||
const struct iwl_fw *fw = priv->fwrt->fw;
|
||||
|
||||
*pos = ++state->pos;
|
||||
if (*pos >= fw->ucode_capa.n_cmd_versions)
|
||||
if (*pos >= fw->ucode_capa.n_cmd_versions) {
|
||||
kfree(state);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return state;
|
||||
}
|
||||
|
@ -484,6 +484,9 @@ static void iwl_fwrt_dump_fseq_regs(struct iwl_fw_runtime *fwrt)
|
||||
|
||||
void iwl_fwrt_dump_error_logs(struct iwl_fw_runtime *fwrt)
|
||||
{
|
||||
struct iwl_pc_data *pc_data;
|
||||
u8 count;
|
||||
|
||||
if (!test_bit(STATUS_DEVICE_ENABLED, &fwrt->trans->status)) {
|
||||
IWL_ERR(fwrt,
|
||||
"DEVICE_ENABLED bit is not set. Aborting dump.\n");
|
||||
@ -502,6 +505,14 @@ void iwl_fwrt_dump_error_logs(struct iwl_fw_runtime *fwrt)
|
||||
iwl_fwrt_dump_rcm_error_log(fwrt, 1);
|
||||
iwl_fwrt_dump_iml_error_log(fwrt);
|
||||
iwl_fwrt_dump_fseq_regs(fwrt);
|
||||
if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000) {
|
||||
pc_data = fwrt->trans->dbg.pc_data;
|
||||
for (count = 0; count < fwrt->trans->dbg.num_pc;
|
||||
count++, pc_data++)
|
||||
IWL_ERR(fwrt, "%s: 0x%x\n",
|
||||
pc_data->pc_name,
|
||||
pc_data->pc_address);
|
||||
}
|
||||
|
||||
if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) {
|
||||
u32 scratch = iwl_read32(fwrt->trans, CSR_FUNC_SCRATCH);
|
||||
|
@ -101,8 +101,10 @@ enum iwl_ucode_tlv_type {
|
||||
|
||||
IWL_UCODE_TLV_SEC_TABLE_ADDR = 66,
|
||||
IWL_UCODE_TLV_D3_KEK_KCK_ADDR = 67,
|
||||
IWL_UCODE_TLV_CURRENT_PC = 68,
|
||||
|
||||
IWL_UCODE_TLV_FW_NUM_STATIONS = IWL_UCODE_TLV_CONST_BASE + 0,
|
||||
IWL_UCODE_TLV_FW_NUM_BEACONS = IWL_UCODE_TLV_CONST_BASE + 2,
|
||||
|
||||
IWL_UCODE_TLV_TYPE_DEBUG_INFO = IWL_UCODE_TLV_DEBUG_BASE + 0,
|
||||
IWL_UCODE_TLV_TYPE_BUFFER_ALLOCATION = IWL_UCODE_TLV_DEBUG_BASE + 1,
|
||||
@ -458,6 +460,8 @@ enum iwl_ucode_tlv_capa {
|
||||
IWL_UCODE_TLV_CAPA_SYNCED_TIME = (__force iwl_ucode_tlv_capa_t)106,
|
||||
IWL_UCODE_TLV_CAPA_TIME_SYNC_BOTH_FTM_TM = (__force iwl_ucode_tlv_capa_t)108,
|
||||
IWL_UCODE_TLV_CAPA_BIGTK_TX_SUPPORT = (__force iwl_ucode_tlv_capa_t)109,
|
||||
IWL_UCODE_TLV_CAPA_MLD_API_SUPPORT = (__force iwl_ucode_tlv_capa_t)110,
|
||||
IWL_UCODE_TLV_CAPA_SCAN_DONT_TOGGLE_ANT = (__force iwl_ucode_tlv_capa_t)111,
|
||||
|
||||
#ifdef __CHECKER__
|
||||
/* sparse says it cannot increment the previous enum member */
|
||||
|
@ -51,6 +51,7 @@ struct iwl_ucode_capabilities {
|
||||
u32 error_log_addr;
|
||||
u32 error_log_size;
|
||||
u32 num_stations;
|
||||
u32 num_beacons;
|
||||
unsigned long _api[BITS_TO_LONGS(NUM_IWL_UCODE_TLV_API)];
|
||||
unsigned long _capa[BITS_TO_LONGS(NUM_IWL_UCODE_TLV_CAPA)];
|
||||
|
||||
|
@ -165,6 +165,7 @@ struct iwl_fw_runtime {
|
||||
struct iwl_ppag_chain ppag_chains[IWL_NUM_CHAIN_LIMITS];
|
||||
u32 ppag_flags;
|
||||
u32 ppag_ver;
|
||||
bool ppag_table_valid;
|
||||
struct iwl_sar_offset_mapping_cmd sgom_table;
|
||||
bool sgom_enabled;
|
||||
u8 reduced_power_flags;
|
||||
|
@ -222,7 +222,7 @@ void *iwl_uefi_get_reduced_power(struct iwl_trans *trans, size_t *len)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
status = efi.get_variable(IWL_UEFI_REDUCED_POWER_NAME, &IWL_EFI_VAR_GUID,
|
||||
NULL, &package_size, data);
|
||||
NULL, &package_size, package);
|
||||
if (status != EFI_SUCCESS) {
|
||||
IWL_DEBUG_FW(trans,
|
||||
"Reduced Power UEFI variable not found 0x%lx (len %lu)\n",
|
||||
|
@ -469,6 +469,7 @@ struct iwl_dev_info {
|
||||
u16 mac_type;
|
||||
u16 rf_type;
|
||||
u8 mac_step;
|
||||
u8 rf_step;
|
||||
u8 rf_id;
|
||||
u8 no_160;
|
||||
u8 cores;
|
||||
@ -639,11 +640,17 @@ extern const struct iwl_cfg iwl_cfg_ma_a0_gf4_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_a0_mr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_a0_ms_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_a0_fm_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_b0_hr_b0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_b0_gf_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_b0_gf4_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_b0_mr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_ma_b0_fm_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_snj_a0_mr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_snj_a0_ms_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_so_a0_hr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_so_a0_ms_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_quz_a0_hr_b0;
|
||||
extern const struct iwl_cfg iwl_cfg_bz_a0_hr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bz_a0_hr_b0;
|
||||
extern const struct iwl_cfg iwl_cfg_bz_a0_gf_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bz_a0_gf4_a0;
|
||||
@ -661,7 +668,9 @@ extern const struct iwl_cfg iwl_cfg_bnj_a0_gf_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_b0_gf_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_a0_gf4_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_b0_gf4_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_a0_hr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_a0_hr_b0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_b0_hr_a0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_b0_hr_b0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_b0_fm_b0;
|
||||
extern const struct iwl_cfg iwl_cfg_bnj_b0_fm4_b0;
|
||||
|
@ -102,6 +102,8 @@
|
||||
#define CSR_LTR_LONG_VAL_AD_SNOOP_VAL 0x000003ff
|
||||
#define CSR_LTR_LONG_VAL_AD_SCALE_USEC 2
|
||||
|
||||
#define CSR_LTR_LAST_MSG (CSR_BASE + 0x0DC)
|
||||
|
||||
/* GIO Chicken Bits (PCI Express bus link power management) */
|
||||
#define CSR_GIO_CHICKEN_BITS (CSR_BASE+0x100)
|
||||
|
||||
@ -309,6 +311,8 @@ enum {
|
||||
SILICON_A_STEP = 0,
|
||||
SILICON_B_STEP,
|
||||
SILICON_C_STEP,
|
||||
SILICON_D_STEP,
|
||||
SILICON_E_STEP,
|
||||
SILICON_Z_STEP = 0xf,
|
||||
};
|
||||
|
||||
|
@ -138,6 +138,12 @@ static int iwl_dbg_tlv_alloc_buf_alloc(struct iwl_trans *trans,
|
||||
alloc_id != IWL_FW_INI_ALLOCATION_ID_DBGC1)
|
||||
goto err;
|
||||
|
||||
if (buf_location == IWL_FW_INI_LOCATION_DRAM_PATH &&
|
||||
alloc->req_size == 0) {
|
||||
IWL_ERR(trans, "WRT: Invalid DRAM buffer allocation requested size (0)\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
trans->dbg.fw_mon_cfg[alloc_id] = *alloc;
|
||||
|
||||
return 0;
|
||||
@ -797,7 +803,7 @@ static void iwl_dbg_tlv_update_drams(struct iwl_fw_runtime *fwrt)
|
||||
if (!ret)
|
||||
dram_alloc = true;
|
||||
else
|
||||
IWL_WARN(fwrt,
|
||||
IWL_INFO(fwrt,
|
||||
"WRT: Failed to set DRAM buffer for alloc id %d, ret=%d\n",
|
||||
i, ret);
|
||||
}
|
||||
|
@ -1,6 +1,6 @@
|
||||
// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
|
||||
/*
|
||||
* Copyright (C) 2005-2011, 2021 Intel Corporation
|
||||
* Copyright (C) 2005-2011, 2021-2022 Intel Corporation
|
||||
*/
|
||||
#include <linux/device.h>
|
||||
#include <linux/interrupt.h>
|
||||
@ -57,6 +57,7 @@ void __iwl_err(struct device *dev, enum iwl_err_mode mode, const char *fmt, ...)
|
||||
default:
|
||||
break;
|
||||
}
|
||||
vaf.va = &args;
|
||||
trace_iwlwifi_err(&vaf);
|
||||
va_end(args);
|
||||
}
|
||||
|
@ -127,6 +127,7 @@ static void iwl_dealloc_ucode(struct iwl_drv *drv)
|
||||
kfree(drv->fw.iml);
|
||||
kfree(drv->fw.ucode_capa.cmd_versions);
|
||||
kfree(drv->fw.phy_integration_ver);
|
||||
kfree(drv->trans->dbg.pc_data);
|
||||
|
||||
for (i = 0; i < IWL_UCODE_TYPE_MAX; i++)
|
||||
iwl_free_fw_img(drv, drv->fw.img + i);
|
||||
@ -894,7 +895,7 @@ static int iwl_parse_tlv_firmware(struct iwl_drv *drv,
|
||||
drv->fw.img[IWL_UCODE_WOWLAN].is_dual_cpus =
|
||||
true;
|
||||
} else if ((num_of_cpus > 2) || (num_of_cpus < 1)) {
|
||||
IWL_ERR(drv, "Driver support upto 2 CPUs\n");
|
||||
IWL_ERR(drv, "Driver support up to 2 CPUs\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
break;
|
||||
@ -1154,6 +1155,12 @@ static int iwl_parse_tlv_firmware(struct iwl_drv *drv,
|
||||
capa->num_stations =
|
||||
le32_to_cpup((const __le32 *)tlv_data);
|
||||
break;
|
||||
case IWL_UCODE_TLV_FW_NUM_BEACONS:
|
||||
if (tlv_len != sizeof(u32))
|
||||
goto invalid_tlv_len;
|
||||
capa->num_beacons =
|
||||
le32_to_cpup((const __le32 *)tlv_data);
|
||||
break;
|
||||
case IWL_UCODE_TLV_UMAC_DEBUG_ADDRS: {
|
||||
const struct iwl_umac_debug_addrs *dbg_ptrs =
|
||||
(const void *)tlv_data;
|
||||
@ -1232,6 +1239,14 @@ static int iwl_parse_tlv_firmware(struct iwl_drv *drv,
|
||||
iwl_drv_set_dump_exclude(drv, tlv_type,
|
||||
tlv_data, tlv_len);
|
||||
break;
|
||||
case IWL_UCODE_TLV_CURRENT_PC:
|
||||
if (tlv_len < sizeof(struct iwl_pc_data))
|
||||
goto invalid_tlv_len;
|
||||
drv->trans->dbg.num_pc =
|
||||
tlv_len / sizeof(struct iwl_pc_data);
|
||||
drv->trans->dbg.pc_data =
|
||||
kmemdup(tlv_data, tlv_len, GFP_KERNEL);
|
||||
break;
|
||||
default:
|
||||
IWL_DEBUG_INFO(drv, "unknown TLV: %d\n", tlv_type);
|
||||
break;
|
||||
@ -1406,6 +1421,7 @@ static void iwl_req_fw_callback(const struct firmware *ucode_raw, void *context)
|
||||
IWL_DEFAULT_STANDARD_PHY_CALIBRATE_TBL_SIZE;
|
||||
fw->ucode_capa.n_scan_channels = IWL_DEFAULT_SCAN_CHANNELS;
|
||||
fw->ucode_capa.num_stations = IWL_MVM_STATION_COUNT_MAX;
|
||||
fw->ucode_capa.num_beacons = 1;
|
||||
/* dump all fw memory areas by default */
|
||||
fw->dbg.dump_mask = 0xffffffff;
|
||||
|
||||
|
@ -47,13 +47,12 @@ struct iwl_nvm_data {
|
||||
struct ieee80211_supported_band bands[NUM_NL80211_BANDS];
|
||||
|
||||
/*
|
||||
* iftype data for low (2.4 GHz) and high (5 and 6 GHz) bands,
|
||||
* we can use the same for 5 and 6 GHz bands because they have
|
||||
* the same data
|
||||
* iftype data for low (2.4 GHz) high (5 GHz) and uhb (6 GHz) bands
|
||||
*/
|
||||
struct {
|
||||
struct ieee80211_sband_iftype_data low[2];
|
||||
struct ieee80211_sband_iftype_data high[2];
|
||||
struct ieee80211_sband_iftype_data uhb[2];
|
||||
} iftd;
|
||||
|
||||
struct ieee80211_channel channels[];
|
||||
|
@ -860,7 +860,10 @@ iwl_nvm_fixup_sband_iftd(struct iwl_trans *trans,
|
||||
/* Advertise an A-MPDU exponent extension based on
|
||||
* operating band
|
||||
*/
|
||||
if (sband->band != NL80211_BAND_2GHZ)
|
||||
if (sband->band == NL80211_BAND_6GHZ && iftype_data->eht_cap.has_eht)
|
||||
iftype_data->he_cap.he_cap_elem.mac_cap_info[3] |=
|
||||
IEEE80211_HE_MAC_CAP3_MAX_AMPDU_LEN_EXP_EXT_2;
|
||||
else if (sband->band != NL80211_BAND_2GHZ)
|
||||
iftype_data->he_cap.he_cap_elem.mac_cap_info[3] |=
|
||||
IEEE80211_HE_MAC_CAP3_MAX_AMPDU_LEN_EXP_EXT_1;
|
||||
else
|
||||
@ -876,16 +879,13 @@ iwl_nvm_fixup_sband_iftd(struct iwl_trans *trans,
|
||||
IEEE80211_EHT_MAC_CAP0_MAX_MPDU_LEN_MASK);
|
||||
break;
|
||||
case NL80211_BAND_6GHZ:
|
||||
if (!is_ap || iwlwifi_mod_params.nvm_file)
|
||||
iftype_data->eht_cap.eht_cap_elem.phy_cap_info[0] |=
|
||||
IEEE80211_EHT_PHY_CAP0_320MHZ_IN_6GHZ;
|
||||
iftype_data->eht_cap.eht_cap_elem.phy_cap_info[0] |=
|
||||
IEEE80211_EHT_PHY_CAP0_320MHZ_IN_6GHZ;
|
||||
fallthrough;
|
||||
case NL80211_BAND_5GHZ:
|
||||
iftype_data->he_cap.he_cap_elem.phy_cap_info[0] |=
|
||||
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_80MHZ_IN_5G;
|
||||
if (!is_ap || iwlwifi_mod_params.nvm_file)
|
||||
iftype_data->he_cap.he_cap_elem.phy_cap_info[0] |=
|
||||
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G;
|
||||
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_80MHZ_IN_5G |
|
||||
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G;
|
||||
break;
|
||||
default:
|
||||
WARN_ON(1);
|
||||
@ -938,6 +938,10 @@ iwl_nvm_fixup_sband_iftd(struct iwl_trans *trans,
|
||||
}
|
||||
}
|
||||
|
||||
if (trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_AX210 && !is_ap)
|
||||
iftype_data->he_cap.he_cap_elem.phy_cap_info[2] |=
|
||||
IEEE80211_HE_PHY_CAP2_UL_MU_FULL_MU_MIMO;
|
||||
|
||||
switch (CSR_HW_RFID_TYPE(trans->hw_rf_id)) {
|
||||
case IWL_CFG_RF_TYPE_GF:
|
||||
case IWL_CFG_RF_TYPE_MR:
|
||||
@ -999,15 +1003,18 @@ static void iwl_init_he_hw_capab(struct iwl_trans *trans,
|
||||
|
||||
BUILD_BUG_ON(sizeof(data->iftd.low) != sizeof(iwl_he_eht_capa));
|
||||
BUILD_BUG_ON(sizeof(data->iftd.high) != sizeof(iwl_he_eht_capa));
|
||||
BUILD_BUG_ON(sizeof(data->iftd.uhb) != sizeof(iwl_he_eht_capa));
|
||||
|
||||
switch (sband->band) {
|
||||
case NL80211_BAND_2GHZ:
|
||||
iftype_data = data->iftd.low;
|
||||
break;
|
||||
case NL80211_BAND_5GHZ:
|
||||
case NL80211_BAND_6GHZ:
|
||||
iftype_data = data->iftd.high;
|
||||
break;
|
||||
case NL80211_BAND_6GHZ:
|
||||
iftype_data = data->iftd.uhb;
|
||||
break;
|
||||
default:
|
||||
WARN_ON(1);
|
||||
return;
|
||||
|
@ -350,6 +350,11 @@
|
||||
#define WFPM_OTP_CFG1_ADDR 0x00a03098
|
||||
#define WFPM_OTP_CFG1_IS_JACKET_BIT BIT(4)
|
||||
#define WFPM_OTP_CFG1_IS_CDB_BIT BIT(5)
|
||||
#define WFPM_OTP_BZ_BNJ_JACKET_BIT 5
|
||||
#define WFPM_OTP_BZ_BNJ_CDB_BIT 4
|
||||
#define WFPM_OTP_CFG1_IS_JACKET(_val) (((_val) & 0x00000020) >> WFPM_OTP_BZ_BNJ_JACKET_BIT)
|
||||
#define WFPM_OTP_CFG1_IS_CDB(_val) (((_val) & 0x00000010) >> WFPM_OTP_BZ_BNJ_CDB_BIT)
|
||||
|
||||
|
||||
#define WFPM_GP2 0xA030B4
|
||||
|
||||
@ -445,6 +450,8 @@ enum {
|
||||
#define REG_CRF_ID_TYPE_GF_TC 0xF08
|
||||
#define REG_CRF_ID_TYPE_MR 0x810
|
||||
#define REG_CRF_ID_TYPE_FM 0x910
|
||||
#define REG_CRF_ID_TYPE_FMI 0x930
|
||||
#define REG_CRF_ID_TYPE_FMR 0x900
|
||||
|
||||
#define HPM_DEBUG 0xA03440
|
||||
#define PERSISTENCE_BIT BIT(12)
|
||||
|
@ -748,6 +748,18 @@ struct iwl_imr_data {
|
||||
__le64 imr_base_addr;
|
||||
};
|
||||
|
||||
#define IWL_TRANS_CURRENT_PC_NAME_MAX_BYTES 32
|
||||
|
||||
/**
|
||||
* struct iwl_pc_data - program counter details
|
||||
* @pc_name: cpu name
|
||||
* @pc_address: cpu program counter
|
||||
*/
|
||||
struct iwl_pc_data {
|
||||
u8 pc_name[IWL_TRANS_CURRENT_PC_NAME_MAX_BYTES];
|
||||
u32 pc_address;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct iwl_trans_debug - transport debug related data
|
||||
*
|
||||
@ -777,6 +789,8 @@ struct iwl_imr_data {
|
||||
* @ucode_preset: preset based on ucode
|
||||
* @dump_file_name_ext: dump file name extension
|
||||
* @dump_file_name_ext_valid: dump file name extension if valid or not
|
||||
* @num_pc: number of program counter for cpu
|
||||
* @pc_data: details of the program counter
|
||||
*/
|
||||
struct iwl_trans_debug {
|
||||
u8 n_dest_reg;
|
||||
@ -817,6 +831,8 @@ struct iwl_trans_debug {
|
||||
struct iwl_imr_data imr_data;
|
||||
u8 dump_file_name_ext[IWL_FW_INI_MAX_NAME];
|
||||
bool dump_file_name_ext_valid;
|
||||
u32 num_pc;
|
||||
struct iwl_pc_data *pc_data;
|
||||
};
|
||||
|
||||
struct iwl_dma_ptr {
|
||||
@ -981,7 +997,7 @@ struct iwl_trans_txqs {
|
||||
* 0 indicates that frag SKBs (NETIF_F_SG) aren't supported.
|
||||
* @hw_rf_id a u32 with the device RF ID
|
||||
* @hw_crf_id a u32 with the device CRF ID
|
||||
* @hw_cdb_id a u32 with the device CDB ID
|
||||
* @hw_wfpm_id a u32 with the device wfpm ID
|
||||
* @hw_id: a u32 with the ID of the device / sub-device.
|
||||
* Set during transport allocation.
|
||||
* @hw_id_str: a string with info about HW ID. Set during transport allocation.
|
||||
@ -1024,7 +1040,8 @@ struct iwl_trans {
|
||||
u32 hw_rev_step;
|
||||
u32 hw_rf_id;
|
||||
u32 hw_crf_id;
|
||||
u32 hw_cdb_id;
|
||||
u32 hw_cnv_id;
|
||||
u32 hw_wfpm_id;
|
||||
u32 hw_id;
|
||||
char hw_id_str[52];
|
||||
u32 sku_id[3];
|
||||
|
@ -1,6 +1,6 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/*
|
||||
* Copyright (C) 2021 Intel Corporation
|
||||
* Copyright (C) 2021 - 2022 Intel Corporation
|
||||
*/
|
||||
|
||||
#ifndef __iwl_mei_h__
|
||||
@ -301,7 +301,7 @@ struct iwl_mei_colloc_info {
|
||||
struct iwl_mei_ops {
|
||||
void (*me_conn_status)(void *priv,
|
||||
const struct iwl_mei_conn_info *conn_info);
|
||||
void (*rfkill)(void *priv, bool blocked);
|
||||
void (*rfkill)(void *priv, bool blocked, bool csme_taking_ownership);
|
||||
void (*roaming_forbidden)(void *priv, bool forbidden);
|
||||
void (*sap_connected)(void *priv);
|
||||
void (*nic_stolen)(void *priv);
|
||||
|
@ -31,6 +31,11 @@ MODULE_LICENSE("GPL");
|
||||
#define MEI_WLAN_UUID UUID_LE(0x13280904, 0x7792, 0x4fcb, \
|
||||
0xa1, 0xaa, 0x5e, 0x70, 0xcb, 0xb1, 0xe8, 0x65)
|
||||
|
||||
/* After CSME takes ownership, it won't release it for 60 seconds to avoid
|
||||
* frequent ownership transitions.
|
||||
*/
|
||||
#define MEI_OWNERSHIP_RETAKE_TIMEOUT_MS msecs_to_jiffies(60000)
|
||||
|
||||
/*
|
||||
* Since iwlwifi calls iwlmei without any context, hold a pointer to the
|
||||
* mei_cl_device structure here.
|
||||
@ -156,6 +161,8 @@ struct iwl_mei_filters {
|
||||
* accessed without the mutex.
|
||||
* @netdev_work: used to defer registering and unregistering of the netdev to
|
||||
* avoid taking the rtnl lock in the SAP messages handlers.
|
||||
* @ownership_dwork: used to re-ask for NIC ownership after ownership was taken
|
||||
* by CSME or when a previous ownership request failed.
|
||||
* @sap_seq_no: the sequence number for the SAP messages
|
||||
* @seq_no: the sequence number for the SAP messages
|
||||
* @dbgfs_dir: the debugfs dir entry
|
||||
@ -179,6 +186,7 @@ struct iwl_mei {
|
||||
bool pldr_active;
|
||||
spinlock_t data_q_lock;
|
||||
struct work_struct netdev_work;
|
||||
struct delayed_work ownership_dwork;
|
||||
|
||||
atomic_t sap_seq_no;
|
||||
atomic_t seq_no;
|
||||
@ -716,7 +724,7 @@ iwl_mei_handle_conn_status(struct mei_cl_device *cldev,
|
||||
status->link_prot_state);
|
||||
else
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv,
|
||||
status->link_prot_state);
|
||||
status->link_prot_state, false);
|
||||
}
|
||||
|
||||
static void iwl_mei_set_init_conf(struct iwl_mei *mei)
|
||||
@ -788,7 +796,7 @@ static void iwl_mei_handle_amt_state(struct mei_cl_device *cldev,
|
||||
if (mei->amt_enabled)
|
||||
iwl_mei_set_init_conf(mei);
|
||||
else if (iwl_mei_cache.ops)
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, false);
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, false, false);
|
||||
|
||||
schedule_work(&mei->netdev_work);
|
||||
|
||||
@ -829,10 +837,12 @@ static void iwl_mei_handle_csme_taking_ownership(struct mei_cl_device *cldev,
|
||||
*/
|
||||
mei->csme_taking_ownership = true;
|
||||
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, true);
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, true, true);
|
||||
} else {
|
||||
iwl_mei_send_sap_msg(cldev,
|
||||
SAP_MSG_NOTIF_CSME_OWNERSHIP_CONFIRMED);
|
||||
schedule_delayed_work(&mei->ownership_dwork,
|
||||
MEI_OWNERSHIP_RETAKE_TIMEOUT_MS);
|
||||
}
|
||||
}
|
||||
|
||||
@ -882,7 +892,7 @@ static void iwl_mei_handle_rx_host_own_req(struct mei_cl_device *cldev,
|
||||
|
||||
/* We can now start the connection, unblock rfkill */
|
||||
if (iwl_mei_cache.ops)
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, false);
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, false, false);
|
||||
}
|
||||
|
||||
static void iwl_mei_handle_pldr_ack(struct mei_cl_device *cldev,
|
||||
@ -1447,7 +1457,13 @@ int iwl_mei_get_ownership(void)
|
||||
|
||||
ret = wait_event_timeout(mei->get_ownership_wq,
|
||||
mei->got_ownership, HZ / 2);
|
||||
return (!ret) ? -ETIMEDOUT : 0;
|
||||
if (!ret) {
|
||||
schedule_delayed_work(&mei->ownership_dwork,
|
||||
MEI_OWNERSHIP_RETAKE_TIMEOUT_MS);
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
return 0;
|
||||
out:
|
||||
mutex_unlock(&iwl_mei_mutex);
|
||||
return ret;
|
||||
@ -1738,6 +1754,8 @@ void iwl_mei_device_state(bool up)
|
||||
iwl_mei_send_sap_msg(mei->cldev,
|
||||
SAP_MSG_NOTIF_CSME_OWNERSHIP_CONFIRMED);
|
||||
mei->csme_taking_ownership = false;
|
||||
schedule_delayed_work(&mei->ownership_dwork,
|
||||
MEI_OWNERSHIP_RETAKE_TIMEOUT_MS);
|
||||
out:
|
||||
mutex_unlock(&iwl_mei_mutex);
|
||||
}
|
||||
@ -1773,7 +1791,8 @@ int iwl_mei_register(void *priv, const struct iwl_mei_ops *ops)
|
||||
if (iwl_mei_is_connected()) {
|
||||
if (mei->amt_enabled)
|
||||
iwl_mei_send_sap_msg(mei->cldev,
|
||||
SAP_MSG_NOTIF_WIFIDR_UP);
|
||||
SAP_MSG_NOTIF_WIFIDR_UP,
|
||||
false);
|
||||
ops->rfkill(priv, mei->link_prot_state);
|
||||
}
|
||||
}
|
||||
@ -1894,6 +1913,11 @@ static void iwl_mei_dbgfs_unregister(struct iwl_mei *mei) {}
|
||||
|
||||
#endif /* CONFIG_DEBUG_FS */
|
||||
|
||||
static void iwl_mei_ownership_dwork(struct work_struct *wk)
|
||||
{
|
||||
iwl_mei_get_ownership();
|
||||
}
|
||||
|
||||
#define ALLOC_SHARED_MEM_RETRY_MAX_NUM 3
|
||||
|
||||
/*
|
||||
@ -1923,6 +1947,7 @@ static int iwl_mei_probe(struct mei_cl_device *cldev,
|
||||
init_waitqueue_head(&mei->pldr_wq);
|
||||
spin_lock_init(&mei->data_q_lock);
|
||||
INIT_WORK(&mei->netdev_work, iwl_mei_netdev_work);
|
||||
INIT_DELAYED_WORK(&mei->ownership_dwork, iwl_mei_ownership_dwork);
|
||||
|
||||
mei_cldev_set_drvdata(cldev, mei);
|
||||
mei->cldev = cldev;
|
||||
@ -2087,7 +2112,7 @@ static void iwl_mei_remove(struct mei_cl_device *cldev)
|
||||
spin_unlock_bh(&mei->data_q_lock);
|
||||
|
||||
if (iwl_mei_cache.ops)
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, false);
|
||||
iwl_mei_cache.ops->rfkill(iwl_mei_cache.priv, false, false);
|
||||
|
||||
/*
|
||||
* mei_cldev_disable will return only after all the MEI Rx is done.
|
||||
@ -2105,6 +2130,7 @@ static void iwl_mei_remove(struct mei_cl_device *cldev)
|
||||
cancel_work_sync(&mei->send_csa_msg_wk);
|
||||
cancel_delayed_work_sync(&mei->csa_throttle_end_wk);
|
||||
cancel_work_sync(&mei->netdev_work);
|
||||
cancel_delayed_work_sync(&mei->ownership_dwork);
|
||||
|
||||
/*
|
||||
* If someone waits for the ownership, let him know that we are going
|
||||
|
@ -564,6 +564,7 @@ static void iwl_mvm_wowlan_get_tkip_data(struct ieee80211_hw *hw,
|
||||
}
|
||||
|
||||
for (i = 0; i < IWL_NUM_RSC; i++) {
|
||||
ieee80211_get_key_rx_seq(key, i, &seq);
|
||||
/* wrapping isn't allowed, AP must rekey */
|
||||
if (seq.tkip.iv32 > cur_rx_iv32)
|
||||
cur_rx_iv32 = seq.tkip.iv32;
|
||||
@ -2017,6 +2018,12 @@ static void iwl_mvm_parse_wowlan_info_notif(struct iwl_mvm *mvm,
|
||||
{
|
||||
u32 i;
|
||||
|
||||
if (!data) {
|
||||
IWL_ERR(mvm, "iwl_wowlan_info_notif data is NULL\n");
|
||||
status = NULL;
|
||||
return;
|
||||
}
|
||||
|
||||
if (len < sizeof(*data)) {
|
||||
IWL_ERR(mvm, "Invalid WoWLAN info notification!\n");
|
||||
status = NULL;
|
||||
@ -2705,10 +2712,15 @@ static bool iwl_mvm_wait_d3_notif(struct iwl_notif_wait_data *notif_wait,
|
||||
struct iwl_d3_data *d3_data = data;
|
||||
u32 len;
|
||||
int ret;
|
||||
int wowlan_info_ver = iwl_fw_lookup_notif_ver(mvm->fw,
|
||||
PROT_OFFLOAD_GROUP,
|
||||
WOWLAN_INFO_NOTIFICATION,
|
||||
IWL_FW_CMD_VER_UNKNOWN);
|
||||
|
||||
|
||||
switch (WIDE_ID(pkt->hdr.group_id, pkt->hdr.cmd)) {
|
||||
case WIDE_ID(PROT_OFFLOAD_GROUP, WOWLAN_INFO_NOTIFICATION): {
|
||||
struct iwl_wowlan_info_notif *notif = (void *)pkt->data;
|
||||
struct iwl_wowlan_info_notif *notif;
|
||||
|
||||
if (d3_data->notif_received & IWL_D3_NOTIF_WOWLAN_INFO) {
|
||||
/* We might get two notifications due to dual bss */
|
||||
@ -2717,10 +2729,32 @@ static bool iwl_mvm_wait_d3_notif(struct iwl_notif_wait_data *notif_wait,
|
||||
break;
|
||||
}
|
||||
|
||||
if (wowlan_info_ver < 2) {
|
||||
struct iwl_wowlan_info_notif_v1 *notif_v1 = (void *)pkt->data;
|
||||
|
||||
notif = kmemdup(notif_v1,
|
||||
offsetofend(struct iwl_wowlan_info_notif,
|
||||
received_beacons),
|
||||
GFP_ATOMIC);
|
||||
|
||||
if (!notif)
|
||||
return false;
|
||||
|
||||
notif->tid_tear_down = notif_v1->tid_tear_down;
|
||||
notif->station_id = notif_v1->station_id;
|
||||
|
||||
} else {
|
||||
notif = (void *)pkt->data;
|
||||
}
|
||||
|
||||
d3_data->notif_received |= IWL_D3_NOTIF_WOWLAN_INFO;
|
||||
len = iwl_rx_packet_payload_len(pkt);
|
||||
iwl_mvm_parse_wowlan_info_notif(mvm, notif, d3_data->status,
|
||||
len);
|
||||
|
||||
if (wowlan_info_ver < 2)
|
||||
kfree(notif);
|
||||
|
||||
if (d3_data->status &&
|
||||
d3_data->status->wakeup_reasons & IWL_WOWLAN_WAKEUP_REASON_HAS_WAKEUP_PKT)
|
||||
/* We are supposed to get also wake packet notif */
|
||||
|
@ -340,6 +340,26 @@ static ssize_t iwl_dbgfs_sar_geo_profile_read(struct file *file,
|
||||
|
||||
return simple_read_from_buffer(user_buf, count, ppos, buf, pos);
|
||||
}
|
||||
|
||||
static ssize_t iwl_dbgfs_wifi_6e_enable_read(struct file *file,
|
||||
char __user *user_buf,
|
||||
size_t count, loff_t *ppos)
|
||||
{
|
||||
struct iwl_mvm *mvm = file->private_data;
|
||||
int err, pos;
|
||||
char buf[12];
|
||||
u32 value;
|
||||
|
||||
err = iwl_acpi_get_dsm_u32(mvm->fwrt.dev, 0,
|
||||
DSM_FUNC_ENABLE_6E,
|
||||
&iwl_guid, &value);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
pos = sprintf(buf, "0x%08x\n", value);
|
||||
|
||||
return simple_read_from_buffer(user_buf, count, ppos, buf, pos);
|
||||
}
|
||||
#endif
|
||||
|
||||
static ssize_t iwl_dbgfs_stations_read(struct file *file, char __user *user_buf,
|
||||
@ -1898,6 +1918,7 @@ MVM_DEBUGFS_READ_FILE_OPS(uapsd_noagg_bssids);
|
||||
|
||||
#ifdef CONFIG_ACPI
|
||||
MVM_DEBUGFS_READ_FILE_OPS(sar_geo_profile);
|
||||
MVM_DEBUGFS_READ_FILE_OPS(wifi_6e_enable);
|
||||
#endif
|
||||
|
||||
MVM_DEBUGFS_READ_WRITE_STA_FILE_OPS(amsdu_len, 16);
|
||||
@ -1940,6 +1961,11 @@ static ssize_t iwl_dbgfs_mem_read(struct file *file, char __user *user_buf,
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
if (iwl_rx_packet_payload_len(hcmd.resp_pkt) < sizeof(*rsp)) {
|
||||
ret = -EIO;
|
||||
goto out;
|
||||
}
|
||||
|
||||
rsp = (void *)hcmd.resp_pkt->data;
|
||||
if (le32_to_cpu(rsp->status) != DEBUG_MEM_STATUS_SUCCESS) {
|
||||
ret = -ENXIO;
|
||||
@ -2016,6 +2042,11 @@ static ssize_t iwl_dbgfs_mem_write(struct file *file,
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
if (iwl_rx_packet_payload_len(hcmd.resp_pkt) < sizeof(*rsp)) {
|
||||
ret = -EIO;
|
||||
goto out;
|
||||
}
|
||||
|
||||
rsp = (void *)hcmd.resp_pkt->data;
|
||||
if (rsp->status != DEBUG_MEM_STATUS_SUCCESS) {
|
||||
ret = -ENXIO;
|
||||
@ -2092,6 +2123,7 @@ void iwl_mvm_dbgfs_register(struct iwl_mvm *mvm)
|
||||
MVM_DEBUGFS_ADD_FILE(tas_get_status, mvm->debugfs_dir, 0400);
|
||||
#ifdef CONFIG_ACPI
|
||||
MVM_DEBUGFS_ADD_FILE(sar_geo_profile, mvm->debugfs_dir, 0400);
|
||||
MVM_DEBUGFS_ADD_FILE(wifi_6e_enable, mvm->debugfs_dir, 0400);
|
||||
#endif
|
||||
MVM_DEBUGFS_ADD_FILE(he_sniffer_params, mvm->debugfs_dir, 0600);
|
||||
|
||||
|
@ -25,6 +25,10 @@ struct iwl_mvm_smooth_entry {
|
||||
u64 host_time;
|
||||
};
|
||||
|
||||
enum iwl_mvm_pasn_flags {
|
||||
IWL_MVM_PASN_FLAG_HAS_HLTK = BIT(0),
|
||||
};
|
||||
|
||||
struct iwl_mvm_ftm_pasn_entry {
|
||||
struct list_head list;
|
||||
u8 addr[ETH_ALEN];
|
||||
@ -33,6 +37,7 @@ struct iwl_mvm_ftm_pasn_entry {
|
||||
u8 cipher;
|
||||
u8 tx_pn[IEEE80211_CCMP_PN_LEN];
|
||||
u8 rx_pn[IEEE80211_CCMP_PN_LEN];
|
||||
u32 flags;
|
||||
};
|
||||
|
||||
int iwl_mvm_ftm_add_pasn_sta(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
@ -79,14 +84,24 @@ int iwl_mvm_ftm_add_pasn_sta(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
if (tk_len != expected_tk_len || hltk_len != sizeof(pasn->hltk)) {
|
||||
if (tk_len != expected_tk_len ||
|
||||
(hltk_len && hltk_len != sizeof(pasn->hltk))) {
|
||||
IWL_ERR(mvm, "Invalid key length: tk_len=%u hltk_len=%u\n",
|
||||
tk_len, hltk_len);
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (!expected_tk_len && !hltk_len) {
|
||||
IWL_ERR(mvm, "TK and HLTK not set\n");
|
||||
goto out;
|
||||
}
|
||||
|
||||
memcpy(pasn->addr, addr, sizeof(pasn->addr));
|
||||
memcpy(pasn->hltk, hltk, sizeof(pasn->hltk));
|
||||
|
||||
if (hltk_len) {
|
||||
memcpy(pasn->hltk, hltk, sizeof(pasn->hltk));
|
||||
pasn->flags |= IWL_MVM_PASN_FLAG_HAS_HLTK;
|
||||
}
|
||||
|
||||
if (tk && tk_len)
|
||||
memcpy(pasn->tk, tk, sizeof(pasn->tk));
|
||||
@ -691,7 +706,11 @@ iwl_mvm_ftm_set_secured_ranging(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
continue;
|
||||
|
||||
target->cipher = entry->cipher;
|
||||
memcpy(target->hltk, entry->hltk, sizeof(target->hltk));
|
||||
|
||||
if (entry->flags & IWL_MVM_PASN_FLAG_HAS_HLTK)
|
||||
memcpy(target->hltk, entry->hltk, sizeof(target->hltk));
|
||||
else
|
||||
memset(target->hltk, 0, sizeof(target->hltk));
|
||||
|
||||
if (vif->cfg.assoc &&
|
||||
!memcmp(vif->bss_conf.bssid, target->bssid,
|
||||
|
@ -317,6 +317,8 @@ int iwl_mvm_ftm_respoder_add_pasn_sta(struct iwl_mvm *mvm,
|
||||
.addr = addr,
|
||||
.hltk = hltk,
|
||||
};
|
||||
struct iwl_mvm_pasn_hltk_data *hltk_data_ptr = NULL;
|
||||
|
||||
u8 cmd_ver = iwl_fw_lookup_cmd_ver(mvm->fw,
|
||||
WIDE_ID(LOCATION_GROUP, TOF_RESPONDER_DYN_CONFIG_CMD),
|
||||
2);
|
||||
@ -328,12 +330,21 @@ int iwl_mvm_ftm_respoder_add_pasn_sta(struct iwl_mvm *mvm,
|
||||
return -ENOTSUPP;
|
||||
}
|
||||
|
||||
hltk_data.cipher = iwl_mvm_cipher_to_location_cipher(cipher);
|
||||
if (hltk_data.cipher == IWL_LOCATION_CIPHER_INVALID) {
|
||||
IWL_ERR(mvm, "invalid cipher: %u\n", cipher);
|
||||
if ((!hltk || !hltk_len) && (!tk || !tk_len)) {
|
||||
IWL_ERR(mvm, "TK and HLTK not set\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (hltk && hltk_len) {
|
||||
hltk_data.cipher = iwl_mvm_cipher_to_location_cipher(cipher);
|
||||
if (hltk_data.cipher == IWL_LOCATION_CIPHER_INVALID) {
|
||||
IWL_ERR(mvm, "invalid cipher: %u\n", cipher);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
hltk_data_ptr = &hltk_data;
|
||||
}
|
||||
|
||||
if (tk && tk_len) {
|
||||
sta = kzalloc(sizeof(*sta), GFP_KERNEL);
|
||||
if (!sta)
|
||||
@ -350,7 +361,7 @@ int iwl_mvm_ftm_respoder_add_pasn_sta(struct iwl_mvm *mvm,
|
||||
list_add_tail(&sta->list, &mvm->resp_pasn_list);
|
||||
}
|
||||
|
||||
ret = iwl_mvm_ftm_responder_dyn_cfg_v3(mvm, vif, NULL, &hltk_data);
|
||||
ret = iwl_mvm_ftm_responder_dyn_cfg_v3(mvm, vif, NULL, hltk_data_ptr);
|
||||
if (ret && sta)
|
||||
iwl_mvm_resp_del_pasn_sta(mvm, vif, sta);
|
||||
|
||||
|
@ -321,6 +321,8 @@ static int iwl_mvm_load_ucode_wait_alive(struct iwl_mvm *mvm,
|
||||
static const u16 alive_cmd[] = { UCODE_ALIVE_NTFY };
|
||||
bool run_in_rfkill =
|
||||
ucode_type == IWL_UCODE_INIT || iwl_mvm_has_unified_ucode(mvm);
|
||||
u8 count;
|
||||
struct iwl_pc_data *pc_data;
|
||||
|
||||
if (ucode_type == IWL_UCODE_REGULAR &&
|
||||
iwl_fw_dbg_conf_usniffer(mvm->fw, FW_DBG_START_FROM_ALIVE) &&
|
||||
@ -393,6 +395,14 @@ static int iwl_mvm_load_ucode_wait_alive(struct iwl_mvm *mvm,
|
||||
|
||||
/* LMAC/UMAC PC info */
|
||||
if (trans->trans_cfg->device_family >=
|
||||
IWL_DEVICE_FAMILY_22000) {
|
||||
pc_data = trans->dbg.pc_data;
|
||||
for (count = 0; count < trans->dbg.num_pc;
|
||||
count++, pc_data++)
|
||||
IWL_ERR(mvm, "%s: 0x%x\n",
|
||||
pc_data->pc_name,
|
||||
pc_data->pc_address);
|
||||
} else if (trans->trans_cfg->device_family >=
|
||||
IWL_DEVICE_FAMILY_9000) {
|
||||
IWL_ERR(mvm, "UMAC PC: 0x%x\n",
|
||||
iwl_read_umac_prph(trans,
|
||||
@ -467,111 +477,6 @@ static int iwl_mvm_load_ucode_wait_alive(struct iwl_mvm *mvm,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int iwl_run_unified_mvm_ucode(struct iwl_mvm *mvm)
|
||||
{
|
||||
struct iwl_notification_wait init_wait;
|
||||
struct iwl_nvm_access_complete_cmd nvm_complete = {};
|
||||
struct iwl_init_extended_cfg_cmd init_cfg = {
|
||||
.init_flags = cpu_to_le32(BIT(IWL_INIT_NVM)),
|
||||
};
|
||||
static const u16 init_complete[] = {
|
||||
INIT_COMPLETE_NOTIF,
|
||||
};
|
||||
int ret;
|
||||
|
||||
if (mvm->trans->cfg->tx_with_siso_diversity)
|
||||
init_cfg.init_flags |= cpu_to_le32(BIT(IWL_INIT_PHY));
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
mvm->rfkill_safe_init_done = false;
|
||||
|
||||
iwl_init_notification_wait(&mvm->notif_wait,
|
||||
&init_wait,
|
||||
init_complete,
|
||||
ARRAY_SIZE(init_complete),
|
||||
iwl_wait_init_complete,
|
||||
NULL);
|
||||
|
||||
iwl_dbg_tlv_time_point(&mvm->fwrt, IWL_FW_INI_TIME_POINT_EARLY, NULL);
|
||||
|
||||
/* Will also start the device */
|
||||
ret = iwl_mvm_load_ucode_wait_alive(mvm, IWL_UCODE_REGULAR);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to start RT ucode: %d\n", ret);
|
||||
goto error;
|
||||
}
|
||||
iwl_dbg_tlv_time_point(&mvm->fwrt, IWL_FW_INI_TIME_POINT_AFTER_ALIVE,
|
||||
NULL);
|
||||
|
||||
/* Send init config command to mark that we are sending NVM access
|
||||
* commands
|
||||
*/
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, WIDE_ID(SYSTEM_GROUP,
|
||||
INIT_EXTENDED_CFG_CMD),
|
||||
CMD_SEND_IN_RFKILL,
|
||||
sizeof(init_cfg), &init_cfg);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to run init config command: %d\n",
|
||||
ret);
|
||||
goto error;
|
||||
}
|
||||
|
||||
/* Load NVM to NIC if needed */
|
||||
if (mvm->nvm_file_name) {
|
||||
ret = iwl_read_external_nvm(mvm->trans, mvm->nvm_file_name,
|
||||
mvm->nvm_sections);
|
||||
if (ret)
|
||||
goto error;
|
||||
ret = iwl_mvm_load_nvm_to_nic(mvm);
|
||||
if (ret)
|
||||
goto error;
|
||||
}
|
||||
|
||||
if (IWL_MVM_PARSE_NVM && !mvm->nvm_data) {
|
||||
ret = iwl_nvm_init(mvm);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to read NVM: %d\n", ret);
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, WIDE_ID(REGULATORY_AND_NVM_GROUP,
|
||||
NVM_ACCESS_COMPLETE),
|
||||
CMD_SEND_IN_RFKILL,
|
||||
sizeof(nvm_complete), &nvm_complete);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to run complete NVM access: %d\n",
|
||||
ret);
|
||||
goto error;
|
||||
}
|
||||
|
||||
/* We wait for the INIT complete notification */
|
||||
ret = iwl_wait_notification(&mvm->notif_wait, &init_wait,
|
||||
MVM_UCODE_ALIVE_TIMEOUT);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* Read the NVM only at driver load time, no need to do this twice */
|
||||
if (!IWL_MVM_PARSE_NVM && !mvm->nvm_data) {
|
||||
mvm->nvm_data = iwl_get_nvm(mvm->trans, mvm->fw);
|
||||
if (IS_ERR(mvm->nvm_data)) {
|
||||
ret = PTR_ERR(mvm->nvm_data);
|
||||
mvm->nvm_data = NULL;
|
||||
IWL_ERR(mvm, "Failed to read NVM: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
mvm->rfkill_safe_init_done = true;
|
||||
|
||||
return 0;
|
||||
|
||||
error:
|
||||
iwl_remove_notification(&mvm->notif_wait, &init_wait);
|
||||
return ret;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_ACPI
|
||||
static void iwl_mvm_phy_filter_init(struct iwl_mvm *mvm,
|
||||
struct iwl_phy_specific_cfg *phy_filters)
|
||||
@ -698,6 +603,118 @@ static int iwl_send_phy_cfg_cmd(struct iwl_mvm *mvm)
|
||||
return iwl_mvm_send_cmd_pdu(mvm, cmd_id, 0, cmd_size, &phy_cfg_cmd);
|
||||
}
|
||||
|
||||
static int iwl_run_unified_mvm_ucode(struct iwl_mvm *mvm)
|
||||
{
|
||||
struct iwl_notification_wait init_wait;
|
||||
struct iwl_nvm_access_complete_cmd nvm_complete = {};
|
||||
struct iwl_init_extended_cfg_cmd init_cfg = {
|
||||
.init_flags = cpu_to_le32(BIT(IWL_INIT_NVM)),
|
||||
};
|
||||
static const u16 init_complete[] = {
|
||||
INIT_COMPLETE_NOTIF,
|
||||
};
|
||||
int ret;
|
||||
|
||||
if (mvm->trans->cfg->tx_with_siso_diversity)
|
||||
init_cfg.init_flags |= cpu_to_le32(BIT(IWL_INIT_PHY));
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
mvm->rfkill_safe_init_done = false;
|
||||
|
||||
iwl_init_notification_wait(&mvm->notif_wait,
|
||||
&init_wait,
|
||||
init_complete,
|
||||
ARRAY_SIZE(init_complete),
|
||||
iwl_wait_init_complete,
|
||||
NULL);
|
||||
|
||||
iwl_dbg_tlv_time_point(&mvm->fwrt, IWL_FW_INI_TIME_POINT_EARLY, NULL);
|
||||
|
||||
/* Will also start the device */
|
||||
ret = iwl_mvm_load_ucode_wait_alive(mvm, IWL_UCODE_REGULAR);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to start RT ucode: %d\n", ret);
|
||||
goto error;
|
||||
}
|
||||
iwl_dbg_tlv_time_point(&mvm->fwrt, IWL_FW_INI_TIME_POINT_AFTER_ALIVE,
|
||||
NULL);
|
||||
|
||||
/* Send init config command to mark that we are sending NVM access
|
||||
* commands
|
||||
*/
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, WIDE_ID(SYSTEM_GROUP,
|
||||
INIT_EXTENDED_CFG_CMD),
|
||||
CMD_SEND_IN_RFKILL,
|
||||
sizeof(init_cfg), &init_cfg);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to run init config command: %d\n",
|
||||
ret);
|
||||
goto error;
|
||||
}
|
||||
|
||||
/* Load NVM to NIC if needed */
|
||||
if (mvm->nvm_file_name) {
|
||||
ret = iwl_read_external_nvm(mvm->trans, mvm->nvm_file_name,
|
||||
mvm->nvm_sections);
|
||||
if (ret)
|
||||
goto error;
|
||||
ret = iwl_mvm_load_nvm_to_nic(mvm);
|
||||
if (ret)
|
||||
goto error;
|
||||
}
|
||||
|
||||
if (IWL_MVM_PARSE_NVM && !mvm->nvm_data) {
|
||||
ret = iwl_nvm_init(mvm);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to read NVM: %d\n", ret);
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, WIDE_ID(REGULATORY_AND_NVM_GROUP,
|
||||
NVM_ACCESS_COMPLETE),
|
||||
CMD_SEND_IN_RFKILL,
|
||||
sizeof(nvm_complete), &nvm_complete);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to run complete NVM access: %d\n",
|
||||
ret);
|
||||
goto error;
|
||||
}
|
||||
|
||||
ret = iwl_send_phy_cfg_cmd(mvm);
|
||||
if (ret) {
|
||||
IWL_ERR(mvm, "Failed to run PHY configuration: %d\n",
|
||||
ret);
|
||||
goto error;
|
||||
}
|
||||
|
||||
/* We wait for the INIT complete notification */
|
||||
ret = iwl_wait_notification(&mvm->notif_wait, &init_wait,
|
||||
MVM_UCODE_ALIVE_TIMEOUT);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* Read the NVM only at driver load time, no need to do this twice */
|
||||
if (!IWL_MVM_PARSE_NVM && !mvm->nvm_data) {
|
||||
mvm->nvm_data = iwl_get_nvm(mvm->trans, mvm->fw);
|
||||
if (IS_ERR(mvm->nvm_data)) {
|
||||
ret = PTR_ERR(mvm->nvm_data);
|
||||
mvm->nvm_data = NULL;
|
||||
IWL_ERR(mvm, "Failed to read NVM: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
mvm->rfkill_safe_init_done = true;
|
||||
|
||||
return 0;
|
||||
|
||||
error:
|
||||
iwl_remove_notification(&mvm->notif_wait, &init_wait);
|
||||
return ret;
|
||||
}
|
||||
|
||||
int iwl_run_init_mvm_ucode(struct iwl_mvm *mvm)
|
||||
{
|
||||
struct iwl_notification_wait calib_wait;
|
||||
@ -1538,12 +1555,11 @@ int iwl_mvm_up(struct iwl_mvm *mvm)
|
||||
ret = iwl_send_phy_db_data(mvm->phy_db);
|
||||
if (ret)
|
||||
goto error;
|
||||
ret = iwl_send_phy_cfg_cmd(mvm);
|
||||
if (ret)
|
||||
goto error;
|
||||
}
|
||||
|
||||
ret = iwl_send_phy_cfg_cmd(mvm);
|
||||
if (ret)
|
||||
goto error;
|
||||
|
||||
ret = iwl_mvm_send_bt_init_conf(mvm);
|
||||
if (ret)
|
||||
goto error;
|
||||
@ -1711,8 +1727,6 @@ int iwl_mvm_up(struct iwl_mvm *mvm)
|
||||
iwl_mvm_tas_init(mvm);
|
||||
iwl_mvm_leds_sync(mvm);
|
||||
|
||||
iwl_mvm_ftm_initiator_smooth_config(mvm);
|
||||
|
||||
if (fw_has_capa(&mvm->fw->ucode_capa,
|
||||
IWL_UCODE_TLV_CAPA_RFIM_SUPPORT)) {
|
||||
if (iwl_mvm_eval_dsm_rfi(mvm) == DSM_VALUE_RFI_ENABLE)
|
||||
|
@ -225,16 +225,20 @@ int iwl_mvm_mac_ctxt_init(struct iwl_mvm *mvm, struct ieee80211_vif *vif)
|
||||
* that we should share it with another interface.
|
||||
*/
|
||||
|
||||
/* Currently, MAC ID 0 should be used only for the managed/IBSS vif */
|
||||
switch (vif->type) {
|
||||
case NL80211_IFTYPE_ADHOC:
|
||||
break;
|
||||
case NL80211_IFTYPE_STATION:
|
||||
if (!vif->p2p)
|
||||
/* MAC ID 0 should be used only for the managed/IBSS vif with non-MLO
|
||||
* FW API
|
||||
*/
|
||||
if (!mvm->mld_api_is_used) {
|
||||
switch (vif->type) {
|
||||
case NL80211_IFTYPE_ADHOC:
|
||||
break;
|
||||
fallthrough;
|
||||
default:
|
||||
__clear_bit(0, data.available_mac_ids);
|
||||
case NL80211_IFTYPE_STATION:
|
||||
if (!vif->p2p)
|
||||
break;
|
||||
fallthrough;
|
||||
default:
|
||||
__clear_bit(0, data.available_mac_ids);
|
||||
}
|
||||
}
|
||||
|
||||
ieee80211_iterate_active_interfaces_atomic(
|
||||
@ -870,17 +874,44 @@ static u32 iwl_mvm_find_ie_offset(u8 *beacon, u8 eid, u32 frame_size)
|
||||
return ie - beacon;
|
||||
}
|
||||
|
||||
static u8 iwl_mvm_mac_ctxt_get_lowest_rate(struct iwl_mvm *mvm,
|
||||
struct ieee80211_tx_info *info,
|
||||
struct ieee80211_vif *vif)
|
||||
u8 iwl_mvm_mac_ctxt_get_lowest_rate(struct iwl_mvm *mvm,
|
||||
struct ieee80211_tx_info *info,
|
||||
struct ieee80211_vif *vif)
|
||||
{
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct ieee80211_supported_band *sband;
|
||||
unsigned long basic = vif->bss_conf.basic_rates;
|
||||
u16 lowest_cck = IWL_RATE_COUNT, lowest_ofdm = IWL_RATE_COUNT;
|
||||
u32 link_id = u32_get_bits(info->control.flags,
|
||||
IEEE80211_TX_CTRL_MLO_LINK);
|
||||
u8 band = info->band;
|
||||
u8 rate;
|
||||
u32 i;
|
||||
|
||||
sband = mvm->hw->wiphy->bands[info->band];
|
||||
if (link_id == IEEE80211_LINK_UNSPECIFIED && vif->valid_links) {
|
||||
for (i = 0; i < ARRAY_SIZE(mvmvif->link); i++) {
|
||||
if (!mvmvif->link[i])
|
||||
continue;
|
||||
/* shouldn't do this when >1 link is active */
|
||||
WARN_ON_ONCE(link_id != IEEE80211_LINK_UNSPECIFIED);
|
||||
link_id = i;
|
||||
}
|
||||
}
|
||||
|
||||
if (link_id < IEEE80211_LINK_UNSPECIFIED) {
|
||||
struct ieee80211_bss_conf *link_conf;
|
||||
|
||||
rcu_read_lock();
|
||||
link_conf = rcu_dereference(vif->link_conf[link_id]);
|
||||
if (link_conf) {
|
||||
basic = link_conf->basic_rates;
|
||||
if (link_conf->chandef.chan)
|
||||
band = link_conf->chandef.chan->band;
|
||||
}
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
sband = mvm->hw->wiphy->bands[band];
|
||||
for_each_set_bit(i, &basic, BITS_PER_LONG) {
|
||||
u16 hw = sband->bitrates[i].hw_value;
|
||||
|
||||
@ -892,7 +923,9 @@ static u8 iwl_mvm_mac_ctxt_get_lowest_rate(struct iwl_mvm *mvm,
|
||||
}
|
||||
}
|
||||
|
||||
if (info->band == NL80211_BAND_2GHZ && !vif->p2p) {
|
||||
if (band == NL80211_BAND_2GHZ && !vif->p2p &&
|
||||
vif->type != NL80211_IFTYPE_P2P_DEVICE &&
|
||||
!(info->flags & IEEE80211_TX_CTL_NO_CCK_RATE)) {
|
||||
if (lowest_cck != IWL_RATE_COUNT)
|
||||
rate = lowest_cck;
|
||||
else if (lowest_ofdm != IWL_RATE_COUNT)
|
||||
@ -1102,10 +1135,10 @@ static int iwl_mvm_mac_ctxt_send_beacon_v9(struct iwl_mvm *mvm,
|
||||
sizeof(beacon_cmd));
|
||||
}
|
||||
|
||||
int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct sk_buff *beacon,
|
||||
struct ieee80211_bss_conf *link_conf)
|
||||
static int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct sk_buff *beacon,
|
||||
struct ieee80211_bss_conf *link_conf)
|
||||
{
|
||||
if (WARN_ON(!beacon))
|
||||
return -EINVAL;
|
||||
|
@ -227,14 +227,18 @@ int iwl_mvm_init_fw_regd(struct iwl_mvm *mvm)
|
||||
static const u8 he_if_types_ext_capa_sta[] = {
|
||||
[0] = WLAN_EXT_CAPA1_EXT_CHANNEL_SWITCHING,
|
||||
[2] = WLAN_EXT_CAPA3_MULTI_BSSID_SUPPORT,
|
||||
[7] = WLAN_EXT_CAPA8_OPMODE_NOTIF,
|
||||
[7] = WLAN_EXT_CAPA8_OPMODE_NOTIF |
|
||||
WLAN_EXT_CAPA8_MAX_MSDU_IN_AMSDU_LSB,
|
||||
[8] = WLAN_EXT_CAPA9_MAX_MSDU_IN_AMSDU_MSB,
|
||||
};
|
||||
|
||||
static const u8 tm_if_types_ext_capa_sta[] = {
|
||||
[0] = WLAN_EXT_CAPA1_EXT_CHANNEL_SWITCHING,
|
||||
[2] = WLAN_EXT_CAPA3_MULTI_BSSID_SUPPORT |
|
||||
WLAN_EXT_CAPA3_TIMING_MEASUREMENT_SUPPORT,
|
||||
[7] = WLAN_EXT_CAPA8_OPMODE_NOTIF,
|
||||
[7] = WLAN_EXT_CAPA8_OPMODE_NOTIF |
|
||||
WLAN_EXT_CAPA8_MAX_MSDU_IN_AMSDU_LSB,
|
||||
[8] = WLAN_EXT_CAPA9_MAX_MSDU_IN_AMSDU_MSB,
|
||||
[9] = WLAN_EXT_CAPA10_TWT_REQUESTER_SUPPORT,
|
||||
};
|
||||
|
||||
@ -301,6 +305,11 @@ int iwl_mvm_mac_setup_register(struct iwl_mvm *mvm)
|
||||
ieee80211_hw_set(hw, BUFF_MMPDU_TXQ);
|
||||
ieee80211_hw_set(hw, STA_MMPDU_TXQ);
|
||||
|
||||
/* Set this early since we need to have it for the check below */
|
||||
if (mvm->mld_api_is_used &&
|
||||
mvm->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ)
|
||||
hw->wiphy->flags |= WIPHY_FLAG_SUPPORTS_MLO;
|
||||
|
||||
/* With MLD FW API, it tracks timing by itself,
|
||||
* no need for any timing from the host
|
||||
*/
|
||||
@ -409,6 +418,8 @@ int iwl_mvm_mac_setup_register(struct iwl_mvm *mvm)
|
||||
|
||||
wiphy_ext_feature_set(hw->wiphy,
|
||||
NL80211_EXT_FEATURE_BEACON_RATE_LEGACY);
|
||||
wiphy_ext_feature_set(hw->wiphy,
|
||||
NL80211_EXT_FEATURE_SCAN_MIN_PREQ_CONTENT);
|
||||
|
||||
if (fw_has_capa(&mvm->fw->ucode_capa,
|
||||
IWL_UCODE_TLV_CAPA_FTM_CALIBRATED)) {
|
||||
@ -1061,6 +1072,9 @@ static void iwl_mvm_restart_cleanup(struct iwl_mvm *mvm)
|
||||
mvm->rx_ba_sessions = 0;
|
||||
mvm->fwrt.dump.conf = FW_DBG_INVALID;
|
||||
mvm->monitor_on = false;
|
||||
#ifdef CONFIG_IWLWIFI_DEBUGFS
|
||||
mvm->beacon_inject_active = false;
|
||||
#endif
|
||||
|
||||
/* keep statistics ticking */
|
||||
iwl_mvm_accu_radio_stats(mvm);
|
||||
@ -3305,8 +3319,8 @@ void iwl_mvm_sta_pre_rcu_remove(struct ieee80211_hw *hw,
|
||||
lockdep_is_held(&mvm->mutex));
|
||||
sta_id = link_sta->sta_id;
|
||||
if (sta == rcu_access_pointer(mvm->fw_id_to_mac_id[sta_id])) {
|
||||
rcu_assign_pointer(mvm->fw_id_to_mac_id[sta_id],
|
||||
ERR_PTR(-ENOENT));
|
||||
RCU_INIT_POINTER(mvm->fw_id_to_mac_id[sta_id],
|
||||
ERR_PTR(-ENOENT));
|
||||
RCU_INIT_POINTER(mvm->fw_id_to_link_sta[sta_id], NULL);
|
||||
}
|
||||
}
|
||||
@ -3568,24 +3582,22 @@ static int iwl_mvm_mac_sta_state(struct ieee80211_hw *hw,
|
||||
*/
|
||||
static void iwl_mvm_rs_rate_init_all_links(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
bool update)
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
unsigned int link_id;
|
||||
|
||||
for_each_mvm_vif_valid_link(mvmvif, link_id) {
|
||||
struct ieee80211_bss_conf *conf =
|
||||
link_conf_dereference_protected(vif, link_id);
|
||||
link_conf_dereference_check(vif, link_id);
|
||||
struct ieee80211_link_sta *link_sta =
|
||||
link_sta_dereference_protected(sta, link_id);
|
||||
link_sta_dereference_check(sta, link_id);
|
||||
|
||||
if (!conf || !link_sta || !mvmvif->link[link_id]->phy_ctxt)
|
||||
continue;
|
||||
|
||||
iwl_mvm_rs_rate_init(mvm, sta, conf, link_sta,
|
||||
mvmvif->link[link_id]->phy_ctxt->channel->band,
|
||||
update);
|
||||
iwl_mvm_rs_rate_init(mvm, vif, sta, conf, link_sta,
|
||||
mvmvif->link[link_id]->phy_ctxt->channel->band);
|
||||
}
|
||||
}
|
||||
|
||||
@ -3662,6 +3674,7 @@ iwl_mvm_sta_state_notexist_to_none(struct iwl_mvm *mvm,
|
||||
struct ieee80211_sta *sta,
|
||||
struct iwl_mvm_sta_state_ops *callbacks)
|
||||
{
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
unsigned int i;
|
||||
int ret;
|
||||
|
||||
@ -3697,6 +3710,9 @@ iwl_mvm_sta_state_notexist_to_none(struct iwl_mvm *mvm,
|
||||
}
|
||||
ieee80211_sta_recalc_aggregates(sta);
|
||||
|
||||
if (vif->type == NL80211_IFTYPE_STATION && !sta->tdls)
|
||||
mvmvif->ap_sta = sta;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -3753,7 +3769,7 @@ iwl_mvm_sta_state_auth_to_assoc(struct ieee80211_hw *hw,
|
||||
}
|
||||
|
||||
out:
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta, false);
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta);
|
||||
|
||||
return callbacks->update_sta(mvm, vif, sta);
|
||||
}
|
||||
@ -3786,7 +3802,9 @@ iwl_mvm_sta_state_assoc_to_authorized(struct iwl_mvm *mvm,
|
||||
iwl_mvm_mei_host_associated(mvm, vif, mvm_sta);
|
||||
}
|
||||
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta, true);
|
||||
mvm_sta->authorized = true;
|
||||
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -3798,14 +3816,17 @@ iwl_mvm_sta_state_authorized_to_assoc(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_sta_state_ops *callbacks)
|
||||
{
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct iwl_mvm_sta *mvmsta = iwl_mvm_sta_from_mac80211(sta);
|
||||
int ret;
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
mvmsta->authorized = false;
|
||||
|
||||
/* once we move into assoc state, need to update rate scale to
|
||||
* disable using wide bandwidth
|
||||
*/
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta, false);
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta);
|
||||
|
||||
if (!sta->tdls) {
|
||||
/* Set this but don't call iwl_mvm_mac_ctxt_changed()
|
||||
@ -3924,8 +3945,10 @@ int iwl_mvm_mac_sta_state_common(struct ieee80211_hw *hw,
|
||||
ret = 0;
|
||||
} else if (old_state == IEEE80211_STA_NONE &&
|
||||
new_state == IEEE80211_STA_NOTEXIST) {
|
||||
if (vif->type == NL80211_IFTYPE_STATION && !sta->tdls)
|
||||
if (vif->type == NL80211_IFTYPE_STATION && !sta->tdls) {
|
||||
iwl_mvm_stop_session_protection(mvm, vif);
|
||||
mvmvif->ap_sta = NULL;
|
||||
}
|
||||
ret = callbacks->rm_sta(mvm, vif, sta);
|
||||
if (sta->tdls) {
|
||||
iwl_mvm_recalc_tdls_state(mvm, vif, false);
|
||||
@ -3968,15 +3991,11 @@ void iwl_mvm_sta_rc_update(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta, u32 changed)
|
||||
{
|
||||
struct iwl_mvm *mvm = IWL_MAC80211_GET_MVM(hw);
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
|
||||
if (changed & (IEEE80211_RC_BW_CHANGED |
|
||||
IEEE80211_RC_SUPP_RATES_CHANGED |
|
||||
IEEE80211_RC_NSS_CHANGED))
|
||||
iwl_mvm_rs_rate_init(mvm, sta,
|
||||
&vif->bss_conf, &sta->deflink,
|
||||
mvmvif->deflink.phy_ctxt->channel->band,
|
||||
true);
|
||||
iwl_mvm_rs_rate_init_all_links(mvm, vif, sta);
|
||||
|
||||
if (vif->type == NL80211_IFTYPE_STATION &&
|
||||
changed & IEEE80211_RC_NSS_CHANGED)
|
||||
@ -4094,7 +4113,7 @@ static int __iwl_mvm_mac_set_key(struct ieee80211_hw *hw,
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct iwl_mvm *mvm = IWL_MAC80211_GET_MVM(hw);
|
||||
struct iwl_mvm_sta *mvmsta = NULL;
|
||||
struct iwl_mvm_key_pn *ptk_pn;
|
||||
struct iwl_mvm_key_pn *ptk_pn = NULL;
|
||||
int keyidx = key->keyidx;
|
||||
u32 sec_key_id = WIDE_ID(DATA_PATH_GROUP, SEC_KEY_CMD);
|
||||
u8 sec_key_ver = iwl_fw_lookup_cmd_ver(mvm->fw, sec_key_id, 0);
|
||||
@ -4252,6 +4271,10 @@ static int __iwl_mvm_mac_set_key(struct ieee80211_hw *hw,
|
||||
if (ret) {
|
||||
IWL_WARN(mvm, "set key failed\n");
|
||||
key->hw_key_idx = STA_KEY_IDX_INVALID;
|
||||
if (ptk_pn) {
|
||||
RCU_INIT_POINTER(mvmsta->ptk_pn[keyidx], NULL);
|
||||
kfree(ptk_pn);
|
||||
}
|
||||
/*
|
||||
* can't add key for RX, but we don't need it
|
||||
* in the device for TX so still return 0,
|
||||
@ -5588,6 +5611,7 @@ void iwl_mvm_mac_flush(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct iwl_mvm_vif *mvmvif;
|
||||
struct iwl_mvm_sta *mvmsta;
|
||||
struct ieee80211_sta *sta;
|
||||
bool ap_sta_done = false;
|
||||
int i;
|
||||
u32 msk = 0;
|
||||
|
||||
@ -5616,8 +5640,14 @@ void iwl_mvm_mac_flush(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
if (mvmsta->vif != vif)
|
||||
continue;
|
||||
|
||||
if (sta == mvmvif->ap_sta) {
|
||||
if (ap_sta_done)
|
||||
continue;
|
||||
ap_sta_done = true;
|
||||
}
|
||||
|
||||
/* make sure only TDLS peers or the AP are flushed */
|
||||
WARN_ON_ONCE(i != mvmvif->deflink.ap_sta_id && !sta->tdls);
|
||||
WARN_ON_ONCE(sta != mvmvif->ap_sta && !sta->tdls);
|
||||
|
||||
if (drop) {
|
||||
if (iwl_mvm_flush_sta(mvm, mvmsta, false))
|
||||
@ -6129,7 +6159,7 @@ static bool iwl_mvm_mac_can_aggregate(struct ieee80211_hw *hw,
|
||||
{
|
||||
struct iwl_mvm *mvm = IWL_MAC80211_GET_MVM(hw);
|
||||
|
||||
if (mvm->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ)
|
||||
if (iwl_mvm_has_new_tx_csum(mvm))
|
||||
return iwl_mvm_tx_csum_bz(mvm, head, true) ==
|
||||
iwl_mvm_tx_csum_bz(mvm, skb, true);
|
||||
|
||||
|
@ -14,23 +14,41 @@ static u32 iwl_mvm_get_sec_sta_mask(struct iwl_mvm *mvm,
|
||||
struct ieee80211_key_conf *keyconf)
|
||||
{
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct iwl_mvm_vif_link_info *link_info = &mvmvif->deflink;
|
||||
|
||||
if (vif->type == NL80211_IFTYPE_AP &&
|
||||
!(keyconf->flags & IEEE80211_KEY_FLAG_PAIRWISE))
|
||||
return BIT(mvmvif->deflink.mcast_sta.sta_id);
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
if (sta) {
|
||||
struct iwl_mvm_sta *mvmsta = iwl_mvm_sta_from_mac80211(sta);
|
||||
|
||||
return BIT(mvmsta->deflink.sta_id);
|
||||
if (keyconf->link_id >= 0) {
|
||||
link_info = mvmvif->link[keyconf->link_id];
|
||||
if (!link_info)
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (vif->type == NL80211_IFTYPE_STATION &&
|
||||
mvmvif->deflink.ap_sta_id != IWL_MVM_INVALID_STA)
|
||||
return BIT(mvmvif->deflink.ap_sta_id);
|
||||
/* AP group keys are per link and should be on the mcast STA */
|
||||
if (vif->type == NL80211_IFTYPE_AP &&
|
||||
!(keyconf->flags & IEEE80211_KEY_FLAG_PAIRWISE))
|
||||
return BIT(link_info->mcast_sta.sta_id);
|
||||
|
||||
/* invalid */
|
||||
return 0;
|
||||
/* for client mode use the AP STA also for group keys */
|
||||
if (!sta && vif->type == NL80211_IFTYPE_STATION)
|
||||
sta = mvmvif->ap_sta;
|
||||
|
||||
/* During remove the STA was removed and the group keys come later
|
||||
* (which sounds like a bad sequence, but remember that to mac80211 the
|
||||
* group keys have no sta pointer), so we don't have a STA now.
|
||||
* Since this happens for group keys only, just use the link_info as
|
||||
* the group keys are per link; make sure that is the case by checking
|
||||
* we do have a link_id or are not doing MLO.
|
||||
* Of course the same can be done during add as well, but we must do
|
||||
* it during remove, since we don't have the mvmvif->ap_sta pointer.
|
||||
*/
|
||||
if (!sta && (keyconf->link_id >= 0 || !vif->valid_links))
|
||||
return BIT(link_info->ap_sta_id);
|
||||
|
||||
/* STA should be non-NULL now, but iwl_mvm_sta_fw_id_mask() checks */
|
||||
|
||||
/* pass link_id to filter by it if not -1 (GTK on client) */
|
||||
return iwl_mvm_sta_fw_id_mask(mvm, sta, keyconf->link_id);
|
||||
}
|
||||
|
||||
static u32 iwl_mvm_get_sec_flags(struct iwl_mvm *mvm,
|
||||
@ -41,6 +59,8 @@ static u32 iwl_mvm_get_sec_flags(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
u32 flags = 0;
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
if (!(keyconf->flags & IEEE80211_KEY_FLAG_PAIRWISE))
|
||||
flags |= IWL_SEC_KEY_FLAG_MCAST_KEY;
|
||||
|
||||
@ -68,22 +88,68 @@ static u32 iwl_mvm_get_sec_flags(struct iwl_mvm *mvm,
|
||||
break;
|
||||
}
|
||||
|
||||
rcu_read_lock();
|
||||
if (!sta && vif->type == NL80211_IFTYPE_STATION &&
|
||||
mvmvif->deflink.ap_sta_id != IWL_MVM_INVALID_STA) {
|
||||
u8 sta_id = mvmvif->deflink.ap_sta_id;
|
||||
|
||||
sta = rcu_dereference_check(mvm->fw_id_to_mac_id[sta_id],
|
||||
lockdep_is_held(&mvm->mutex));
|
||||
}
|
||||
if (!sta && vif->type == NL80211_IFTYPE_STATION)
|
||||
sta = mvmvif->ap_sta;
|
||||
|
||||
if (!IS_ERR_OR_NULL(sta) && sta->mfp)
|
||||
flags |= IWL_SEC_KEY_FLAG_MFP;
|
||||
rcu_read_unlock();
|
||||
|
||||
return flags;
|
||||
}
|
||||
|
||||
struct iwl_mvm_sta_key_update_data {
|
||||
struct ieee80211_sta *sta;
|
||||
u32 old_sta_mask;
|
||||
u32 new_sta_mask;
|
||||
int err;
|
||||
};
|
||||
|
||||
static void iwl_mvm_mld_update_sta_key(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
struct ieee80211_key_conf *key,
|
||||
void *_data)
|
||||
{
|
||||
u32 cmd_id = WIDE_ID(DATA_PATH_GROUP, SEC_KEY_CMD);
|
||||
struct iwl_mvm_sta_key_update_data *data = _data;
|
||||
struct iwl_mvm *mvm = IWL_MAC80211_GET_MVM(hw);
|
||||
struct iwl_sec_key_cmd cmd = {
|
||||
.action = cpu_to_le32(FW_CTXT_ACTION_MODIFY),
|
||||
.u.modify.old_sta_mask = cpu_to_le32(data->old_sta_mask),
|
||||
.u.modify.new_sta_mask = cpu_to_le32(data->new_sta_mask),
|
||||
.u.modify.key_id = cpu_to_le32(key->keyidx),
|
||||
.u.modify.key_flags =
|
||||
cpu_to_le32(iwl_mvm_get_sec_flags(mvm, vif, sta, key)),
|
||||
};
|
||||
int err;
|
||||
|
||||
/* only need to do this for pairwise keys (link_id == -1) */
|
||||
if (sta != data->sta || key->link_id >= 0)
|
||||
return;
|
||||
|
||||
err = iwl_mvm_send_cmd_pdu(mvm, cmd_id, CMD_ASYNC, sizeof(cmd), &cmd);
|
||||
|
||||
if (err)
|
||||
data->err = err;
|
||||
}
|
||||
|
||||
int iwl_mvm_mld_update_sta_keys(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
u32 old_sta_mask,
|
||||
u32 new_sta_mask)
|
||||
{
|
||||
struct iwl_mvm_sta_key_update_data data = {
|
||||
.sta = sta,
|
||||
.old_sta_mask = old_sta_mask,
|
||||
.new_sta_mask = new_sta_mask,
|
||||
};
|
||||
|
||||
ieee80211_iter_keys_rcu(mvm->hw, vif, iwl_mvm_mld_update_sta_key,
|
||||
&data);
|
||||
return data.err;
|
||||
}
|
||||
|
||||
static int __iwl_mvm_sec_key_del(struct iwl_mvm *mvm, u32 sta_mask,
|
||||
u32 key_flags, u32 keyidx, u32 flags)
|
||||
{
|
||||
@ -118,6 +184,9 @@ int iwl_mvm_sec_key_add(struct iwl_mvm *mvm,
|
||||
if (WARN_ON(keyconf->keylen > sizeof(cmd.u.add.key)))
|
||||
return -EINVAL;
|
||||
|
||||
if (WARN_ON(!sta_mask))
|
||||
return -EINVAL;
|
||||
|
||||
if (keyconf->cipher == WLAN_CIPHER_SUITE_WEP40 ||
|
||||
keyconf->cipher == WLAN_CIPHER_SUITE_WEP104)
|
||||
memcpy(cmd.u.add.key + IWL_SEC_WEP_KEY_OFFSET, keyconf->key,
|
||||
@ -164,6 +233,9 @@ static int _iwl_mvm_sec_key_del(struct iwl_mvm *mvm,
|
||||
u32 key_flags = iwl_mvm_get_sec_flags(mvm, vif, sta, keyconf);
|
||||
int ret;
|
||||
|
||||
if (WARN_ON(!sta_mask))
|
||||
return -EINVAL;
|
||||
|
||||
ret = __iwl_mvm_sec_key_del(mvm, sta_mask, key_flags, keyconf->keyidx,
|
||||
flags);
|
||||
if (ret)
|
||||
|
@ -4,6 +4,16 @@
|
||||
*/
|
||||
#include "mvm.h"
|
||||
|
||||
static void iwl_mvm_mld_set_he_support(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct iwl_mac_config_cmd *cmd)
|
||||
{
|
||||
if (vif->type == NL80211_IFTYPE_AP)
|
||||
cmd->he_ap_support = cpu_to_le16(1);
|
||||
else
|
||||
cmd->he_support = cpu_to_le16(1);
|
||||
}
|
||||
|
||||
static void iwl_mvm_mld_mac_ctxt_cmd_common(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct iwl_mac_config_cmd *cmd,
|
||||
@ -41,7 +51,7 @@ static void iwl_mvm_mld_mac_ctxt_cmd_common(struct iwl_mvm *mvm,
|
||||
* and enable both when we have MLO.
|
||||
*/
|
||||
if (vif->valid_links) {
|
||||
cmd->he_support = cpu_to_le32(1);
|
||||
iwl_mvm_mld_set_he_support(mvm, vif, cmd);
|
||||
cmd->eht_support = cpu_to_le32(1);
|
||||
return;
|
||||
}
|
||||
@ -53,7 +63,7 @@ static void iwl_mvm_mld_mac_ctxt_cmd_common(struct iwl_mvm *mvm,
|
||||
continue;
|
||||
|
||||
if (link_conf->he_support)
|
||||
cmd->he_support = cpu_to_le32(1);
|
||||
iwl_mvm_mld_set_he_support(mvm, vif, cmd);
|
||||
|
||||
/* it's not reasonable to have EHT without HE and FW API doesn't
|
||||
* support it. Ignore EHT in this case.
|
||||
@ -157,7 +167,6 @@ static int iwl_mvm_mld_mac_ctxt_cmd_ibss(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
u32 action)
|
||||
{
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct iwl_mac_config_cmd cmd = {};
|
||||
|
||||
WARN_ON(vif->type != NL80211_IFTYPE_ADHOC);
|
||||
@ -168,9 +177,6 @@ static int iwl_mvm_mld_mac_ctxt_cmd_ibss(struct iwl_mvm *mvm,
|
||||
MAC_CFG_FILTER_ACCEPT_PROBE_REQ |
|
||||
MAC_CFG_FILTER_ACCEPT_GRP);
|
||||
|
||||
/* TODO: Assumes that the beacon id == mac context id */
|
||||
cmd.go_ibss.beacon_template = cpu_to_le32(mvmvif->id);
|
||||
|
||||
return iwl_mvm_mld_mac_ctxt_send_cmd(mvm, &cmd);
|
||||
}
|
||||
|
||||
@ -210,9 +216,6 @@ static int iwl_mvm_mld_mac_ctxt_cmd_ap_go(struct iwl_mvm *mvm,
|
||||
MAC_CFG_FILTER_ACCEPT_PROBE_REQ,
|
||||
MAC_CFG_FILTER_ACCEPT_BEACON);
|
||||
|
||||
/* TODO: Assume that the beacon id == mac context id */
|
||||
cmd.go_ibss.beacon_template = cpu_to_le32(mvmvif->id);
|
||||
|
||||
return iwl_mvm_mld_mac_ctxt_send_cmd(mvm, &cmd);
|
||||
}
|
||||
|
||||
|
@ -256,6 +256,30 @@ __iwl_mvm_mld_assign_vif_chanctx(struct iwl_mvm *mvm,
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
/* Initialize rate control for the AP station, since we might be
|
||||
* doing a link switch here - we cannot initialize it before since
|
||||
* this needs the phy context assigned (and in FW?), and we cannot
|
||||
* do it later because it needs to be initialized as soon as we're
|
||||
* able to TX on the link, i.e. when active.
|
||||
*
|
||||
* Firmware restart isn't quite correct yet for MLO, but we don't
|
||||
* need to do it in that case anyway since it will happen from the
|
||||
* normal station state callback.
|
||||
*/
|
||||
if (mvmvif->ap_sta &&
|
||||
!test_bit(IWL_MVM_STATUS_IN_HW_RESTART, &mvm->status)) {
|
||||
struct ieee80211_link_sta *link_sta;
|
||||
|
||||
rcu_read_lock();
|
||||
link_sta = rcu_dereference(mvmvif->ap_sta->link[link_id]);
|
||||
|
||||
if (!WARN_ON_ONCE(!link_sta))
|
||||
iwl_mvm_rs_rate_init(mvm, vif, mvmvif->ap_sta,
|
||||
link_conf, link_sta,
|
||||
phy_ctxt->channel->band);
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
/* then activate */
|
||||
ret = iwl_mvm_link_changed(mvm, vif, link_conf,
|
||||
LINK_CONTEXT_MODIFY_ACTIVE |
|
||||
@ -882,7 +906,10 @@ iwl_mvm_mld_change_vif_links(struct ieee80211_hw *hw,
|
||||
n_active++;
|
||||
}
|
||||
|
||||
if (n_active > 1)
|
||||
if (vif->type == NL80211_IFTYPE_AP &&
|
||||
n_active > mvm->fw->ucode_capa.num_beacons)
|
||||
return -EOPNOTSUPP;
|
||||
else if (n_active > 1)
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
|
@ -4,6 +4,43 @@
|
||||
*/
|
||||
#include "mvm.h"
|
||||
#include "time-sync.h"
|
||||
#include "sta.h"
|
||||
|
||||
u32 iwl_mvm_sta_fw_id_mask(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
int filter_link_id)
|
||||
{
|
||||
struct iwl_mvm_sta *mvmsta;
|
||||
unsigned int link_id;
|
||||
u32 result = 0;
|
||||
|
||||
if (!sta)
|
||||
return 0;
|
||||
|
||||
mvmsta = iwl_mvm_sta_from_mac80211(sta);
|
||||
|
||||
/* it's easy when the STA is not an MLD */
|
||||
if (!sta->valid_links)
|
||||
return BIT(mvmsta->deflink.sta_id);
|
||||
|
||||
/* but if it is an MLD, get the mask of all the FW STAs it has ... */
|
||||
for (link_id = 0; link_id < ARRAY_SIZE(mvmsta->link); link_id++) {
|
||||
struct iwl_mvm_link_sta *link_sta;
|
||||
|
||||
/* unless we have a specific link in mind */
|
||||
if (filter_link_id >= 0 && link_id != filter_link_id)
|
||||
continue;
|
||||
|
||||
link_sta =
|
||||
rcu_dereference_check(mvmsta->link[link_id],
|
||||
lockdep_is_held(&mvm->mutex));
|
||||
if (!link_sta)
|
||||
continue;
|
||||
|
||||
result |= BIT(link_sta->sta_id);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static int iwl_mvm_mld_send_sta_cmd(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_sta_cfg_cmd *cmd)
|
||||
@ -262,19 +299,22 @@ int iwl_mvm_mld_add_aux_sta(struct iwl_mvm *mvm, u32 lmac_id)
|
||||
IWL_MAX_TID_COUNT, NULL);
|
||||
}
|
||||
|
||||
static int iwl_mvm_mld_disable_txq(struct iwl_mvm *mvm, int sta_id,
|
||||
static int iwl_mvm_mld_disable_txq(struct iwl_mvm *mvm, u32 sta_mask,
|
||||
u16 *queueptr, u8 tid)
|
||||
{
|
||||
int queue = *queueptr;
|
||||
int ret = 0;
|
||||
|
||||
if (tid == IWL_MAX_TID_COUNT)
|
||||
tid = IWL_MGMT_TID;
|
||||
|
||||
if (mvm->sta_remove_requires_queue_remove) {
|
||||
u32 cmd_id = WIDE_ID(DATA_PATH_GROUP,
|
||||
SCD_QUEUE_CONFIG_CMD);
|
||||
struct iwl_scd_queue_cfg_cmd remove_cmd = {
|
||||
.operation = cpu_to_le32(IWL_SCD_QUEUE_REMOVE),
|
||||
.u.remove.tid = cpu_to_le32(tid),
|
||||
.u.remove.sta_mask = cpu_to_le32(BIT(sta_id)),
|
||||
.u.remove.sta_mask = cpu_to_le32(sta_mask),
|
||||
};
|
||||
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, cmd_id, 0,
|
||||
@ -304,7 +344,7 @@ static int iwl_mvm_mld_rm_int_sta(struct iwl_mvm *mvm,
|
||||
if (flush)
|
||||
iwl_mvm_flush_sta(mvm, int_sta, true);
|
||||
|
||||
iwl_mvm_mld_disable_txq(mvm, int_sta->sta_id, queuptr, tid);
|
||||
iwl_mvm_mld_disable_txq(mvm, BIT(int_sta->sta_id), queuptr, tid);
|
||||
|
||||
ret = iwl_mvm_mld_rm_sta_from_fw(mvm, int_sta->sta_id);
|
||||
if (ret)
|
||||
@ -387,7 +427,6 @@ static int iwl_mvm_mld_cfg_sta(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
struct iwl_mvm_sta_cfg_cmd cmd = {
|
||||
.sta_id = cpu_to_le32(mvm_link_sta->sta_id),
|
||||
.station_type = cpu_to_le32(mvm_sta->sta_type),
|
||||
.mfp = cpu_to_le32(sta->mfp),
|
||||
};
|
||||
u32 agg_size = 0, mpdu_dens = 0;
|
||||
|
||||
@ -720,6 +759,7 @@ static void iwl_mvm_mld_disable_sta_queues(struct iwl_mvm *mvm,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct iwl_mvm_sta *mvm_sta = iwl_mvm_sta_from_mac80211(sta);
|
||||
u32 sta_mask = iwl_mvm_sta_fw_id_mask(mvm, sta, -1);
|
||||
int i;
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
@ -728,7 +768,7 @@ static void iwl_mvm_mld_disable_sta_queues(struct iwl_mvm *mvm,
|
||||
if (mvm_sta->tid_data[i].txq_id == IWL_MVM_INVALID_QUEUE)
|
||||
continue;
|
||||
|
||||
iwl_mvm_mld_disable_txq(mvm, mvm_sta->deflink.sta_id,
|
||||
iwl_mvm_mld_disable_txq(mvm, sta_mask,
|
||||
&mvm_sta->tid_data[i].txq_id, i);
|
||||
mvm_sta->tid_data[i].txq_id = IWL_MVM_INVALID_QUEUE;
|
||||
}
|
||||
@ -870,11 +910,12 @@ void iwl_mvm_mld_modify_all_sta_disable_tx(struct iwl_mvm *mvm,
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
static int iwl_mvm_mld_update_sta_queue(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_sta *mvm_sta,
|
||||
u32 old_sta_mask,
|
||||
u32 new_sta_mask)
|
||||
static int iwl_mvm_mld_update_sta_queues(struct iwl_mvm *mvm,
|
||||
struct ieee80211_sta *sta,
|
||||
u32 old_sta_mask,
|
||||
u32 new_sta_mask)
|
||||
{
|
||||
struct iwl_mvm_sta *mvm_sta = iwl_mvm_sta_from_mac80211(sta);
|
||||
struct iwl_scd_queue_cfg_cmd cmd = {
|
||||
.operation = cpu_to_le32(IWL_SCD_QUEUE_MODIFY),
|
||||
.u.modify.old_sta_mask = cpu_to_le32(old_sta_mask),
|
||||
@ -910,6 +951,70 @@ static int iwl_mvm_mld_update_sta_queue(struct iwl_mvm *mvm,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int iwl_mvm_mld_update_sta_baids(struct iwl_mvm *mvm,
|
||||
u32 old_sta_mask,
|
||||
u32 new_sta_mask)
|
||||
{
|
||||
struct iwl_rx_baid_cfg_cmd cmd = {
|
||||
.action = cpu_to_le32(IWL_RX_BAID_ACTION_MODIFY),
|
||||
.modify.old_sta_id_mask = cpu_to_le32(old_sta_mask),
|
||||
.modify.new_sta_id_mask = cpu_to_le32(new_sta_mask),
|
||||
};
|
||||
u32 cmd_id = WIDE_ID(DATA_PATH_GROUP, RX_BAID_ALLOCATION_CONFIG_CMD);
|
||||
int baid;
|
||||
|
||||
BUILD_BUG_ON(sizeof(struct iwl_rx_baid_cfg_resp) != sizeof(baid));
|
||||
|
||||
for (baid = 0; baid < ARRAY_SIZE(mvm->baid_map); baid++) {
|
||||
struct iwl_mvm_baid_data *data;
|
||||
int ret;
|
||||
|
||||
data = rcu_dereference_protected(mvm->baid_map[baid],
|
||||
lockdep_is_held(&mvm->mutex));
|
||||
if (!data)
|
||||
continue;
|
||||
|
||||
if (!(data->sta_mask & old_sta_mask))
|
||||
continue;
|
||||
|
||||
WARN_ONCE(data->sta_mask != old_sta_mask,
|
||||
"BAID data for %d corrupted - expected 0x%x found 0x%x\n",
|
||||
baid, old_sta_mask, data->sta_mask);
|
||||
|
||||
cmd.modify.tid = cpu_to_le32(data->tid);
|
||||
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, cmd_id, 0, sizeof(cmd), &cmd);
|
||||
data->sta_mask = new_sta_mask;
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int iwl_mvm_mld_update_sta_resources(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
u32 old_sta_mask,
|
||||
u32 new_sta_mask)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = iwl_mvm_mld_update_sta_queues(mvm, sta,
|
||||
old_sta_mask,
|
||||
new_sta_mask);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = iwl_mvm_mld_update_sta_keys(mvm, vif, sta,
|
||||
old_sta_mask,
|
||||
new_sta_mask);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return iwl_mvm_mld_update_sta_baids(mvm, old_sta_mask, new_sta_mask);
|
||||
}
|
||||
|
||||
int iwl_mvm_mld_update_sta_links(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
@ -946,9 +1051,10 @@ int iwl_mvm_mld_update_sta_links(struct iwl_mvm *mvm,
|
||||
}
|
||||
|
||||
if (sta_mask_to_rem) {
|
||||
ret = iwl_mvm_mld_update_sta_queue(mvm, mvm_sta,
|
||||
current_sta_mask,
|
||||
current_sta_mask & ~sta_mask_to_rem);
|
||||
ret = iwl_mvm_mld_update_sta_resources(mvm, vif, sta,
|
||||
current_sta_mask,
|
||||
current_sta_mask &
|
||||
~sta_mask_to_rem);
|
||||
if (WARN_ON(ret))
|
||||
goto err;
|
||||
|
||||
@ -1020,12 +1126,15 @@ int iwl_mvm_mld_update_sta_links(struct iwl_mvm *mvm,
|
||||
goto err;
|
||||
|
||||
link_sta_added_to_fw |= BIT(link_id);
|
||||
|
||||
iwl_mvm_rs_add_sta_link(mvm, mvm_sta_link);
|
||||
}
|
||||
|
||||
if (sta_mask_added) {
|
||||
ret = iwl_mvm_mld_update_sta_queue(mvm, mvm_sta,
|
||||
current_sta_mask,
|
||||
current_sta_mask | sta_mask_added);
|
||||
ret = iwl_mvm_mld_update_sta_resources(mvm, vif, sta,
|
||||
current_sta_mask,
|
||||
current_sta_mask |
|
||||
sta_mask_added);
|
||||
if (WARN_ON(ret))
|
||||
goto err;
|
||||
}
|
||||
|
@ -434,6 +434,8 @@ struct iwl_mvm_vif {
|
||||
/* TCP Checksum Offload */
|
||||
netdev_features_t features;
|
||||
|
||||
struct ieee80211_sta *ap_sta;
|
||||
|
||||
/* we can only have 2 GTK + 2 IGTK active at a time */
|
||||
struct ieee80211_key_conf *ap_early_keys[4];
|
||||
|
||||
@ -678,7 +680,7 @@ __aligned(roundup_pow_of_two(sizeof(struct _iwl_mvm_reorder_buf_entry)))
|
||||
|
||||
/**
|
||||
* struct iwl_mvm_baid_data - BA session data
|
||||
* @sta_id: station id
|
||||
* @sta_mask: current station mask for the BAID
|
||||
* @tid: tid of the session
|
||||
* @baid baid of the session
|
||||
* @timeout: the timeout set in the addba request
|
||||
@ -692,7 +694,7 @@ __aligned(roundup_pow_of_two(sizeof(struct _iwl_mvm_reorder_buf_entry)))
|
||||
*/
|
||||
struct iwl_mvm_baid_data {
|
||||
struct rcu_head rcu_head;
|
||||
u8 sta_id;
|
||||
u32 sta_mask;
|
||||
u8 tid;
|
||||
u8 baid;
|
||||
u16 timeout;
|
||||
@ -824,6 +826,12 @@ struct iwl_time_sync_data {
|
||||
bool active;
|
||||
};
|
||||
|
||||
struct iwl_mei_scan_filter {
|
||||
bool is_mei_limited_scan;
|
||||
struct sk_buff_head scan_res;
|
||||
struct work_struct scan_work;
|
||||
};
|
||||
|
||||
struct iwl_mvm {
|
||||
/* for logger access */
|
||||
struct device *dev;
|
||||
@ -1175,6 +1183,8 @@ struct iwl_mvm {
|
||||
bool pldr_sync;
|
||||
|
||||
struct iwl_time_sync_data time_sync;
|
||||
|
||||
struct iwl_mei_scan_filter mei_scan_filter;
|
||||
};
|
||||
|
||||
/* Extract MVM priv from op_mode and _hw */
|
||||
@ -1401,24 +1411,8 @@ static inline bool iwl_mvm_has_new_rx_api(struct iwl_mvm *mvm)
|
||||
|
||||
static inline bool iwl_mvm_has_mld_api(const struct iwl_fw *fw)
|
||||
{
|
||||
return (iwl_fw_lookup_cmd_ver(fw, LINK_CONFIG_CMD,
|
||||
IWL_FW_CMD_VER_UNKNOWN) !=
|
||||
IWL_FW_CMD_VER_UNKNOWN) &&
|
||||
(iwl_fw_lookup_cmd_ver(fw, MAC_CONFIG_CMD,
|
||||
IWL_FW_CMD_VER_UNKNOWN) !=
|
||||
IWL_FW_CMD_VER_UNKNOWN) &&
|
||||
(iwl_fw_lookup_cmd_ver(fw, STA_CONFIG_CMD,
|
||||
IWL_FW_CMD_VER_UNKNOWN) !=
|
||||
IWL_FW_CMD_VER_UNKNOWN) &&
|
||||
(iwl_fw_lookup_cmd_ver(fw, AUX_STA_CMD,
|
||||
IWL_FW_CMD_VER_UNKNOWN) !=
|
||||
IWL_FW_CMD_VER_UNKNOWN) &&
|
||||
(iwl_fw_lookup_cmd_ver(fw, STA_REMOVE_CMD,
|
||||
IWL_FW_CMD_VER_UNKNOWN) !=
|
||||
IWL_FW_CMD_VER_UNKNOWN) &&
|
||||
(iwl_fw_lookup_cmd_ver(fw, STA_DISABLE_TX_CMD,
|
||||
IWL_FW_CMD_VER_UNKNOWN) !=
|
||||
IWL_FW_CMD_VER_UNKNOWN);
|
||||
return fw_has_capa(&fw->ucode_capa,
|
||||
IWL_UCODE_TLV_CAPA_MLD_API_SUPPORT);
|
||||
}
|
||||
|
||||
static inline bool iwl_mvm_has_new_tx_api(struct iwl_mvm *mvm)
|
||||
@ -1522,6 +1516,19 @@ static inline bool iwl_mvm_is_ctdp_supported(struct iwl_mvm *mvm)
|
||||
IWL_UCODE_TLV_CAPA_CTDP_SUPPORT);
|
||||
}
|
||||
|
||||
static inline bool iwl_mvm_has_new_tx_csum(struct iwl_mvm *mvm)
|
||||
{
|
||||
if (mvm->trans->trans_cfg->device_family < IWL_DEVICE_FAMILY_BZ)
|
||||
return false;
|
||||
|
||||
if (mvm->trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_BZ &&
|
||||
CSR_HW_REV_TYPE(mvm->trans->hw_rev) == IWL_CFG_MAC_TYPE_GL &&
|
||||
mvm->trans->hw_rev_step <= SILICON_B_STEP)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
extern const u8 iwl_mvm_ac_to_tx_fifo[];
|
||||
extern const u8 iwl_mvm_ac_to_gen2_tx_fifo[];
|
||||
|
||||
@ -1781,16 +1788,15 @@ int iwl_mvm_mac_ctxt_remove(struct iwl_mvm *mvm, struct ieee80211_vif *vif);
|
||||
int iwl_mvm_mac_ctxt_beacon_changed(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_bss_conf *link_conf);
|
||||
int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct sk_buff *beacon,
|
||||
struct ieee80211_bss_conf *link_conf);
|
||||
int iwl_mvm_mac_ctxt_send_beacon_cmd(struct iwl_mvm *mvm,
|
||||
struct sk_buff *beacon,
|
||||
void *data, int len);
|
||||
u8 iwl_mvm_mac_ctxt_get_beacon_rate(struct iwl_mvm *mvm,
|
||||
struct ieee80211_tx_info *info,
|
||||
struct ieee80211_vif *vif);
|
||||
u8 iwl_mvm_mac_ctxt_get_lowest_rate(struct iwl_mvm *mvm,
|
||||
struct ieee80211_tx_info *info,
|
||||
struct ieee80211_vif *vif);
|
||||
u16 iwl_mvm_mac_ctxt_get_beacon_flags(const struct iwl_fw *fw,
|
||||
u8 rate_idx);
|
||||
void iwl_mvm_mac_ctxt_set_tim(struct iwl_mvm *mvm,
|
||||
@ -2306,6 +2312,7 @@ void iwl_mvm_event_frame_timeout_callback(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
const struct ieee80211_sta *sta,
|
||||
u16 tid);
|
||||
void iwl_mvm_mei_scan_filter_init(struct iwl_mei_scan_filter *mei_scan_filter);
|
||||
|
||||
void iwl_mvm_ptp_init(struct iwl_mvm *mvm);
|
||||
void iwl_mvm_ptp_remove(struct iwl_mvm *mvm);
|
||||
@ -2334,6 +2341,11 @@ void iwl_mvm_sec_key_remove_ap(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct iwl_mvm_vif_link_info *link,
|
||||
unsigned int link_id);
|
||||
int iwl_mvm_mld_update_sta_keys(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
u32 old_sta_mask,
|
||||
u32 new_sta_mask);
|
||||
|
||||
int iwl_rfi_send_config_cmd(struct iwl_mvm *mvm,
|
||||
struct iwl_rfi_lut_entry *rfi_table);
|
||||
@ -2512,6 +2524,22 @@ static inline void iwl_mvm_mei_set_sw_rfkill_state(struct iwl_mvm *mvm)
|
||||
sw_rfkill);
|
||||
}
|
||||
|
||||
static inline bool iwl_mvm_mei_filter_scan(struct iwl_mvm *mvm,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
struct ieee80211_mgmt *mgmt = (void *)skb->data;
|
||||
|
||||
if (mvm->mei_scan_filter.is_mei_limited_scan &&
|
||||
(ieee80211_is_probe_resp(mgmt->frame_control) ||
|
||||
ieee80211_is_beacon(mgmt->frame_control))) {
|
||||
skb_queue_tail(&mvm->mei_scan_filter.scan_res, skb);
|
||||
schedule_work(&mvm->mei_scan_filter.scan_work);
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void iwl_mvm_send_roaming_forbidden_event(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
bool forbidden);
|
||||
|
@ -472,8 +472,8 @@ static const struct iwl_hcmd_names iwl_mvm_legacy_names[] = {
|
||||
HCMD_NAME(SCAN_OFFLOAD_PROFILES_QUERY_CMD),
|
||||
HCMD_NAME(BT_COEX_UPDATE_REDUCED_TXP),
|
||||
HCMD_NAME(BT_COEX_CI),
|
||||
HCMD_NAME(WNM_80211V_TIMING_MEASUREMENT_CONFIRM_NOTIFICATION),
|
||||
HCMD_NAME(WNM_80211V_TIMING_MEASUREMENT_NOTIFICATION),
|
||||
HCMD_NAME(WNM_80211V_TIMING_MEASUREMENT_CONFIRM_NOTIFICATION),
|
||||
HCMD_NAME(PHY_CONFIGURATION_CMD),
|
||||
HCMD_NAME(CALIB_RES_NOTIF_PHY_DB),
|
||||
HCMD_NAME(PHY_DB_CMD),
|
||||
@ -1020,10 +1020,14 @@ static void iwl_mvm_me_conn_status(void *priv, const struct iwl_mei_conn_info *c
|
||||
kfree_rcu(prev_conn_info, rcu_head);
|
||||
}
|
||||
|
||||
static void iwl_mvm_mei_rfkill(void *priv, bool blocked)
|
||||
static void iwl_mvm_mei_rfkill(void *priv, bool blocked,
|
||||
bool csme_taking_ownership)
|
||||
{
|
||||
struct iwl_mvm *mvm = priv;
|
||||
|
||||
if (blocked && !csme_taking_ownership)
|
||||
return;
|
||||
|
||||
mvm->mei_rfkill_blocked = blocked;
|
||||
if (!mvm->hw_registered)
|
||||
return;
|
||||
@ -1367,12 +1371,16 @@ iwl_op_mode_mvm_start(struct iwl_trans *trans, const struct iwl_cfg *cfg,
|
||||
else
|
||||
memset(&mvm->rx_stats, 0, sizeof(struct mvm_statistics_rx));
|
||||
|
||||
iwl_mvm_ftm_initiator_smooth_config(mvm);
|
||||
|
||||
iwl_mvm_init_time_sync(&mvm->time_sync);
|
||||
|
||||
mvm->debugfs_dir = dbgfs_dir;
|
||||
|
||||
mvm->mei_registered = !iwl_mei_register(mvm, &mei_ops);
|
||||
|
||||
iwl_mvm_mei_scan_filter_init(&mvm->mei_scan_filter);
|
||||
|
||||
if (iwl_mvm_start_get_nvm(mvm)) {
|
||||
/*
|
||||
* Getting NVM failed while CSME is the owner, but we are
|
||||
|
@ -61,12 +61,14 @@ static u8 rs_fw_sgi_cw_support(struct ieee80211_link_sta *link_sta)
|
||||
}
|
||||
|
||||
static u16 rs_fw_get_config_flags(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
struct ieee80211_supported_band *sband)
|
||||
{
|
||||
struct ieee80211_sta_ht_cap *ht_cap = &link_sta->ht_cap;
|
||||
struct ieee80211_sta_vht_cap *vht_cap = &link_sta->vht_cap;
|
||||
struct ieee80211_sta_he_cap *he_cap = &link_sta->he_cap;
|
||||
const struct ieee80211_sta_he_cap *sband_he_cap;
|
||||
bool vht_ena = vht_cap->vht_supported;
|
||||
u16 flags = 0;
|
||||
|
||||
@ -92,17 +94,19 @@ static u16 rs_fw_get_config_flags(struct iwl_mvm *mvm,
|
||||
IEEE80211_HE_PHY_CAP1_LDPC_CODING_IN_PAYLOAD))
|
||||
flags |= IWL_TLC_MNG_CFG_FLAGS_LDPC_MSK;
|
||||
|
||||
if (sband->iftype_data && sband->iftype_data->he_cap.has_he &&
|
||||
!(sband->iftype_data->he_cap.he_cap_elem.phy_cap_info[1] &
|
||||
IEEE80211_HE_PHY_CAP1_LDPC_CODING_IN_PAYLOAD))
|
||||
sband_he_cap = ieee80211_get_he_iftype_cap(sband,
|
||||
ieee80211_vif_type_p2p(vif));
|
||||
if (sband_he_cap &&
|
||||
!(sband_he_cap->he_cap_elem.phy_cap_info[1] &
|
||||
IEEE80211_HE_PHY_CAP1_LDPC_CODING_IN_PAYLOAD))
|
||||
flags &= ~IWL_TLC_MNG_CFG_FLAGS_LDPC_MSK;
|
||||
|
||||
if (he_cap->has_he &&
|
||||
(he_cap->he_cap_elem.phy_cap_info[3] &
|
||||
IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_RX_MASK &&
|
||||
sband->iftype_data &&
|
||||
sband->iftype_data->he_cap.he_cap_elem.phy_cap_info[3] &
|
||||
IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_TX_MASK))
|
||||
sband_he_cap &&
|
||||
sband_he_cap->he_cap_elem.phy_cap_info[3] &
|
||||
IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_TX_MASK))
|
||||
flags |= IWL_TLC_MNG_CFG_FLAGS_HE_DCM_NSS_1_MSK;
|
||||
|
||||
return flags;
|
||||
@ -283,7 +287,8 @@ rs_fw_rs_mcs2eht_mcs(enum IWL_TLC_MCS_PER_BW bw,
|
||||
}
|
||||
|
||||
static void
|
||||
rs_fw_eht_set_enabled_rates(const struct ieee80211_link_sta *link_sta,
|
||||
rs_fw_eht_set_enabled_rates(struct ieee80211_vif *vif,
|
||||
const struct ieee80211_link_sta *link_sta,
|
||||
struct ieee80211_supported_band *sband,
|
||||
struct iwl_tlc_config_cmd_v4 *cmd)
|
||||
{
|
||||
@ -299,7 +304,8 @@ rs_fw_eht_set_enabled_rates(const struct ieee80211_link_sta *link_sta,
|
||||
struct ieee80211_eht_mcs_nss_supp_20mhz_only mcs_tx_20;
|
||||
|
||||
/* peer is 20Mhz only */
|
||||
if (!(link_sta->he_cap.he_cap_elem.phy_cap_info[0] &
|
||||
if (vif->type == NL80211_IFTYPE_AP &&
|
||||
!(link_sta->he_cap.he_cap_elem.phy_cap_info[0] &
|
||||
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_MASK_ALL)) {
|
||||
mcs_rx_20 = eht_rx_mcs->only_20mhz;
|
||||
} else {
|
||||
@ -361,7 +367,8 @@ rs_fw_eht_set_enabled_rates(const struct ieee80211_link_sta *link_sta,
|
||||
sizeof(cmd->ht_rates[IWL_TLC_NSS_2]));
|
||||
}
|
||||
|
||||
static void rs_fw_set_supp_rates(struct ieee80211_link_sta *link_sta,
|
||||
static void rs_fw_set_supp_rates(struct ieee80211_vif *vif,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
struct ieee80211_supported_band *sband,
|
||||
struct iwl_tlc_config_cmd_v4 *cmd)
|
||||
{
|
||||
@ -383,7 +390,7 @@ static void rs_fw_set_supp_rates(struct ieee80211_link_sta *link_sta,
|
||||
/* HT/VHT rates */
|
||||
if (link_sta->eht_cap.has_eht) {
|
||||
cmd->mode = IWL_TLC_MNG_MODE_EHT;
|
||||
rs_fw_eht_set_enabled_rates(link_sta, sband, cmd);
|
||||
rs_fw_eht_set_enabled_rates(vif, link_sta, sband, cmd);
|
||||
} else if (he_cap->has_he) {
|
||||
cmd->mode = IWL_TLC_MNG_MODE_HE;
|
||||
rs_fw_he_set_enabled_rates(link_sta, sband, cmd);
|
||||
@ -557,10 +564,12 @@ u16 rs_fw_get_max_amsdu_len(struct ieee80211_sta *sta,
|
||||
return 0;
|
||||
}
|
||||
|
||||
void rs_fw_rate_init(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
struct ieee80211_bss_conf *link_conf,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
enum nl80211_band band, bool update)
|
||||
void iwl_mvm_rs_fw_rate_init(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
struct ieee80211_bss_conf *link_conf,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
enum nl80211_band band)
|
||||
{
|
||||
struct ieee80211_hw *hw = mvm->hw;
|
||||
struct iwl_mvm_sta *mvmsta = iwl_mvm_sta_from_mac80211(sta);
|
||||
@ -570,10 +579,9 @@ void rs_fw_rate_init(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
struct iwl_mvm_link_sta *mvm_link_sta;
|
||||
struct iwl_lq_sta_rs_fw *lq_sta;
|
||||
struct iwl_tlc_config_cmd_v4 cfg_cmd = {
|
||||
.max_ch_width = update ?
|
||||
rs_fw_bw_from_sta_bw(link_sta) :
|
||||
IWL_TLC_MNG_CH_WIDTH_20MHZ,
|
||||
.flags = cpu_to_le16(rs_fw_get_config_flags(mvm, link_sta,
|
||||
.max_ch_width = mvmsta->authorized ?
|
||||
rs_fw_bw_from_sta_bw(link_sta) : IWL_TLC_MNG_CH_WIDTH_20MHZ,
|
||||
.flags = cpu_to_le16(rs_fw_get_config_flags(mvm, vif, link_sta,
|
||||
sband)),
|
||||
.chains = rs_fw_set_active_chains(iwl_mvm_get_valid_tx_ant(mvm)),
|
||||
.sgi_ch_width_supp = rs_fw_sgi_cw_support(link_sta),
|
||||
@ -601,7 +609,7 @@ void rs_fw_rate_init(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
#ifdef CONFIG_IWLWIFI_DEBUGFS
|
||||
iwl_mvm_reset_frame_stats(mvm);
|
||||
#endif
|
||||
rs_fw_set_supp_rates(link_sta, sband, &cfg_cmd);
|
||||
rs_fw_set_supp_rates(vif, link_sta, sband, &cfg_cmd);
|
||||
|
||||
/*
|
||||
* since TLC offload works with one mode we can assume
|
||||
@ -671,6 +679,26 @@ int rs_fw_tx_protection(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvmsta,
|
||||
return 0;
|
||||
}
|
||||
|
||||
void iwl_mvm_rs_add_sta_link(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_link_sta *link_sta)
|
||||
{
|
||||
struct iwl_lq_sta_rs_fw *lq_sta;
|
||||
|
||||
lq_sta = &link_sta->lq_sta.rs_fw;
|
||||
|
||||
lq_sta->pers.drv = mvm;
|
||||
lq_sta->pers.sta_id = link_sta->sta_id;
|
||||
lq_sta->pers.chains = 0;
|
||||
memset(lq_sta->pers.chain_signal, 0,
|
||||
sizeof(lq_sta->pers.chain_signal));
|
||||
lq_sta->pers.last_rssi = S8_MIN;
|
||||
lq_sta->last_rate_n_flags = 0;
|
||||
|
||||
#ifdef CONFIG_MAC80211_DEBUGFS
|
||||
lq_sta->pers.dbg_fixed_rate = 0;
|
||||
#endif
|
||||
}
|
||||
|
||||
void iwl_mvm_rs_add_sta(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvmsta)
|
||||
{
|
||||
unsigned int link_id;
|
||||
@ -678,25 +706,12 @@ void iwl_mvm_rs_add_sta(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvmsta)
|
||||
IWL_DEBUG_RATE(mvm, "create station rate scale window\n");
|
||||
|
||||
for (link_id = 0; link_id < ARRAY_SIZE(mvmsta->link); link_id++) {
|
||||
struct iwl_lq_sta_rs_fw *lq_sta;
|
||||
struct iwl_mvm_link_sta *link =
|
||||
rcu_dereference_protected(mvmsta->link[link_id],
|
||||
lockdep_is_held(&mvm->mutex));
|
||||
if (!link)
|
||||
continue;
|
||||
|
||||
lq_sta = &link->lq_sta.rs_fw;
|
||||
|
||||
lq_sta->pers.drv = mvm;
|
||||
lq_sta->pers.sta_id = link->sta_id;
|
||||
lq_sta->pers.chains = 0;
|
||||
memset(lq_sta->pers.chain_signal, 0,
|
||||
sizeof(lq_sta->pers.chain_signal));
|
||||
lq_sta->pers.last_rssi = S8_MIN;
|
||||
lq_sta->last_rate_n_flags = 0;
|
||||
|
||||
#ifdef CONFIG_MAC80211_DEBUGFS
|
||||
lq_sta->pers.dbg_fixed_rate = 0;
|
||||
#endif
|
||||
iwl_mvm_rs_add_sta_link(mvm, link);
|
||||
}
|
||||
}
|
||||
|
@ -3015,9 +3015,9 @@ static void rs_drv_rate_update(void *mvm_r,
|
||||
for (tid = 0; tid < IWL_MAX_TID_COUNT; tid++)
|
||||
ieee80211_stop_tx_ba_session(sta, tid);
|
||||
|
||||
iwl_mvm_rs_rate_init(mvm, sta,
|
||||
iwl_mvm_rs_rate_init(mvm, mvmsta->vif, sta,
|
||||
&mvmsta->vif->bss_conf, &sta->deflink,
|
||||
sband->band, true);
|
||||
sband->band);
|
||||
}
|
||||
|
||||
static void __iwl_mvm_rs_tx_status(struct iwl_mvm *mvm,
|
||||
@ -4101,13 +4101,16 @@ static const struct rate_control_ops rs_mvm_ops_drv = {
|
||||
.capa = RATE_CTRL_CAPA_VHT_EXT_NSS_BW,
|
||||
};
|
||||
|
||||
void iwl_mvm_rs_rate_init(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
void iwl_mvm_rs_rate_init(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
struct ieee80211_bss_conf *link_conf,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
enum nl80211_band band, bool update)
|
||||
enum nl80211_band band)
|
||||
{
|
||||
if (iwl_mvm_has_tlc_offload(mvm)) {
|
||||
rs_fw_rate_init(mvm, sta, link_conf, link_sta, band, update);
|
||||
iwl_mvm_rs_fw_rate_init(mvm, vif, sta, link_conf,
|
||||
link_sta, band);
|
||||
} else {
|
||||
struct iwl_mvm_sta *mvmsta = iwl_mvm_sta_from_mac80211(sta);
|
||||
|
||||
|
@ -394,10 +394,12 @@ struct iwl_lq_sta {
|
||||
((_c) << RS_DRV_DATA_LQ_COLOR_POS)))
|
||||
|
||||
/* Initialize station's rate scaling information after adding station */
|
||||
void iwl_mvm_rs_rate_init(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
void iwl_mvm_rs_rate_init(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
struct ieee80211_bss_conf *link_conf,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
enum nl80211_band band, bool update);
|
||||
enum nl80211_band band);
|
||||
|
||||
/* Notify RS about Tx status */
|
||||
void iwl_mvm_rs_tx_status(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
@ -432,11 +434,18 @@ int iwl_mvm_tx_protection(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvmsta,
|
||||
void iwl_mvm_reset_frame_stats(struct iwl_mvm *mvm);
|
||||
#endif
|
||||
|
||||
struct iwl_mvm_link_sta;
|
||||
|
||||
void iwl_mvm_rs_add_sta(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvmsta);
|
||||
void rs_fw_rate_init(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
struct ieee80211_bss_conf *link_conf,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
enum nl80211_band band, bool update);
|
||||
void iwl_mvm_rs_add_sta_link(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_link_sta *link_sta);
|
||||
|
||||
void iwl_mvm_rs_fw_rate_init(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
struct ieee80211_bss_conf *link_conf,
|
||||
struct ieee80211_link_sta *link_sta,
|
||||
enum nl80211_band band);
|
||||
int rs_fw_tx_protection(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvmsta,
|
||||
bool enable);
|
||||
void iwl_mvm_tlc_update_notif(struct iwl_mvm *mvm,
|
||||
|
@ -383,9 +383,10 @@ void iwl_mvm_rx_rx_mpdu(struct iwl_mvm *mvm, struct napi_struct *napi,
|
||||
* Don't even try to decrypt a MCAST frame that was received
|
||||
* before the managed vif is authorized, we'd fail anyway.
|
||||
*/
|
||||
if (vif->type == NL80211_IFTYPE_STATION &&
|
||||
if (is_multicast_ether_addr(hdr->addr1) &&
|
||||
vif->type == NL80211_IFTYPE_STATION &&
|
||||
!mvmvif->authorized &&
|
||||
is_multicast_ether_addr(hdr->addr1)) {
|
||||
ieee80211_has_protected(hdr->frame_control)) {
|
||||
IWL_DEBUG_DROP(mvm, "MCAST before the vif is authorized\n");
|
||||
kfree_skb(skb);
|
||||
rcu_read_unlock();
|
||||
|
@ -106,28 +106,12 @@ static int iwl_mvm_create_skb(struct iwl_mvm *mvm, struct sk_buff *skb,
|
||||
|
||||
/*
|
||||
* For non monitor interface strip the bytes the RADA might not have
|
||||
* removed. As monitor interface cannot exist with other interfaces
|
||||
* this removal is safe.
|
||||
* removed (it might be disabled, e.g. for mgmt frames). As a monitor
|
||||
* interface cannot exist with other interfaces, this removal is safe
|
||||
* and sufficient, in monitor mode there's no decryption being done.
|
||||
*/
|
||||
if (mic_crc_len && !ieee80211_hw_check(mvm->hw, RX_INCLUDES_FCS)) {
|
||||
u32 pkt_flags = le32_to_cpu(pkt->len_n_flags);
|
||||
|
||||
/*
|
||||
* If RADA was not enabled then decryption was not performed so
|
||||
* the MIC cannot be removed.
|
||||
*/
|
||||
if (!(pkt_flags & FH_RSCSR_RADA_EN)) {
|
||||
if (WARN_ON(crypt_len > mic_crc_len))
|
||||
return -EINVAL;
|
||||
|
||||
mic_crc_len -= crypt_len;
|
||||
}
|
||||
|
||||
if (WARN_ON(mic_crc_len > len))
|
||||
return -EINVAL;
|
||||
|
||||
if (len > mic_crc_len && !ieee80211_hw_check(mvm->hw, RX_INCLUDES_FCS))
|
||||
len -= mic_crc_len;
|
||||
}
|
||||
|
||||
/* If frame is small enough to fit in skb->head, pull it completely.
|
||||
* If not, only pull ieee80211_hdr (including crypto if present, and
|
||||
@ -172,8 +156,7 @@ static int iwl_mvm_create_skb(struct iwl_mvm *mvm, struct sk_buff *skb,
|
||||
* Starting from Bz hardware, it calculates starting directly after
|
||||
* the MAC header, so that matches mac80211's expectation.
|
||||
*/
|
||||
if (skb->ip_summed == CHECKSUM_COMPLETE &&
|
||||
mvm->trans->trans_cfg->device_family < IWL_DEVICE_FAMILY_BZ) {
|
||||
if (skb->ip_summed == CHECKSUM_COMPLETE) {
|
||||
struct {
|
||||
u8 hdr[6];
|
||||
__be16 type;
|
||||
@ -188,7 +171,7 @@ static int iwl_mvm_create_skb(struct iwl_mvm *mvm, struct sk_buff *skb,
|
||||
shdr->type != htons(ETH_P_PAE) &&
|
||||
shdr->type != htons(ETH_P_TDLS))))
|
||||
skb->ip_summed = CHECKSUM_NONE;
|
||||
else
|
||||
else if (mvm->trans->trans_cfg->device_family < IWL_DEVICE_FAMILY_BZ)
|
||||
/* mac80211 assumes full CSUM including SNAP header */
|
||||
skb_postpush_rcsum(skb, shdr, sizeof(*shdr));
|
||||
}
|
||||
@ -412,9 +395,7 @@ static int iwl_mvm_rx_crypto(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
if (!(status & IWL_RX_MPDU_STATUS_MIC_OK))
|
||||
return -1;
|
||||
|
||||
stats->flag |= RX_FLAG_DECRYPTED;
|
||||
if (pkt_flags & FH_RSCSR_RADA_EN)
|
||||
stats->flag |= RX_FLAG_MIC_STRIPPED;
|
||||
stats->flag |= RX_FLAG_DECRYPTED | RX_FLAG_MIC_STRIPPED;
|
||||
*crypt_len = IEEE80211_CCMP_HDR_LEN;
|
||||
return 0;
|
||||
case IWL_RX_MPDU_STATUS_SEC_TKIP:
|
||||
@ -706,7 +687,7 @@ void iwl_mvm_reorder_timer_expired(struct timer_list *t)
|
||||
if (expired) {
|
||||
struct ieee80211_sta *sta;
|
||||
struct iwl_mvm_sta *mvmsta;
|
||||
u8 sta_id = baid_data->sta_id;
|
||||
u8 sta_id = ffs(baid_data->sta_mask) - 1;
|
||||
|
||||
rcu_read_lock();
|
||||
sta = rcu_dereference(buf->mvm->fw_id_to_mac_id[sta_id]);
|
||||
@ -741,6 +722,7 @@ static void iwl_mvm_del_ba(struct iwl_mvm *mvm, int queue,
|
||||
struct ieee80211_sta *sta;
|
||||
struct iwl_mvm_reorder_buffer *reorder_buf;
|
||||
u8 baid = data->baid;
|
||||
u32 sta_id;
|
||||
|
||||
if (WARN_ONCE(baid >= IWL_MAX_BAID, "invalid BAID: %x\n", baid))
|
||||
return;
|
||||
@ -751,7 +733,9 @@ static void iwl_mvm_del_ba(struct iwl_mvm *mvm, int queue,
|
||||
if (WARN_ON_ONCE(!ba_data))
|
||||
goto out;
|
||||
|
||||
sta = rcu_dereference(mvm->fw_id_to_mac_id[ba_data->sta_id]);
|
||||
/* pick any STA ID to find the pointer */
|
||||
sta_id = ffs(ba_data->sta_mask) - 1;
|
||||
sta = rcu_dereference(mvm->fw_id_to_mac_id[sta_id]);
|
||||
if (WARN_ON_ONCE(IS_ERR_OR_NULL(sta)))
|
||||
goto out;
|
||||
|
||||
@ -778,6 +762,7 @@ static void iwl_mvm_release_frames_from_notif(struct iwl_mvm *mvm,
|
||||
struct ieee80211_sta *sta;
|
||||
struct iwl_mvm_reorder_buffer *reorder_buf;
|
||||
struct iwl_mvm_baid_data *ba_data;
|
||||
u32 sta_id;
|
||||
|
||||
IWL_DEBUG_HT(mvm, "Frame release notification for BAID %u, NSSN %d\n",
|
||||
baid, nssn);
|
||||
@ -795,7 +780,9 @@ static void iwl_mvm_release_frames_from_notif(struct iwl_mvm *mvm,
|
||||
goto out;
|
||||
}
|
||||
|
||||
sta = rcu_dereference(mvm->fw_id_to_mac_id[ba_data->sta_id]);
|
||||
/* pick any STA ID to find the pointer */
|
||||
sta_id = ffs(ba_data->sta_mask) - 1;
|
||||
sta = rcu_dereference(mvm->fw_id_to_mac_id[sta_id]);
|
||||
if (WARN_ON_ONCE(IS_ERR_OR_NULL(sta)))
|
||||
goto out;
|
||||
|
||||
@ -936,7 +923,6 @@ static bool iwl_mvm_reorder(struct iwl_mvm *mvm,
|
||||
{
|
||||
struct ieee80211_rx_status *rx_status = IEEE80211_SKB_RXCB(skb);
|
||||
struct ieee80211_hdr *hdr = (void *)skb_mac_header(skb);
|
||||
struct iwl_mvm_sta *mvm_sta;
|
||||
struct iwl_mvm_baid_data *baid_data;
|
||||
struct iwl_mvm_reorder_buffer *buffer;
|
||||
struct sk_buff *tail;
|
||||
@ -948,6 +934,7 @@ static bool iwl_mvm_reorder(struct iwl_mvm *mvm,
|
||||
u8 sub_frame_idx = desc->amsdu_info &
|
||||
IWL_RX_MPDU_AMSDU_SUBFRAME_IDX_MASK;
|
||||
struct iwl_mvm_reorder_buf_entry *entries;
|
||||
u32 sta_mask;
|
||||
int index;
|
||||
u16 nssn, sn;
|
||||
u8 baid;
|
||||
@ -970,8 +957,6 @@ static bool iwl_mvm_reorder(struct iwl_mvm *mvm,
|
||||
"Got valid BAID without a valid station assigned\n"))
|
||||
return false;
|
||||
|
||||
mvm_sta = iwl_mvm_sta_from_mac80211(sta);
|
||||
|
||||
/* not a data packet or a bar */
|
||||
if (!ieee80211_is_back_req(hdr->frame_control) &&
|
||||
(!ieee80211_is_data_qos(hdr->frame_control) ||
|
||||
@ -989,11 +974,14 @@ static bool iwl_mvm_reorder(struct iwl_mvm *mvm,
|
||||
return false;
|
||||
}
|
||||
|
||||
rcu_read_lock();
|
||||
sta_mask = iwl_mvm_sta_fw_id_mask(mvm, sta, -1);
|
||||
rcu_read_unlock();
|
||||
|
||||
if (WARN(tid != baid_data->tid ||
|
||||
mvm_sta->deflink.sta_id != baid_data->sta_id,
|
||||
"baid 0x%x is mapped to sta:%d tid:%d, but was received for sta:%d tid:%d\n",
|
||||
baid, baid_data->sta_id, baid_data->tid, mvm_sta->deflink.sta_id,
|
||||
tid))
|
||||
!(sta_mask & baid_data->sta_mask),
|
||||
"baid 0x%x is mapped to sta_mask:0x%x tid:%d, but was received for sta_mask:0x%x tid:%d\n",
|
||||
baid, baid_data->sta_mask, baid_data->tid, sta_mask, tid))
|
||||
return false;
|
||||
|
||||
nssn = reorder & IWL_RX_MPDU_REORDER_NSSN_MASK;
|
||||
@ -2600,10 +2588,10 @@ void iwl_mvm_rx_mpdu_mq(struct iwl_mvm *mvm, struct napi_struct *napi,
|
||||
}
|
||||
|
||||
if (!iwl_mvm_reorder(mvm, napi, queue, sta, skb, desc) &&
|
||||
likely(!iwl_mvm_time_sync_frame(mvm, skb, hdr->addr2)))
|
||||
likely(!iwl_mvm_time_sync_frame(mvm, skb, hdr->addr2)) &&
|
||||
likely(!iwl_mvm_mei_filter_scan(mvm, skb)))
|
||||
iwl_mvm_pass_packet_to_mac80211(mvm, napi, skb, queue, sta,
|
||||
link_sta);
|
||||
|
||||
out:
|
||||
rcu_read_unlock();
|
||||
}
|
||||
@ -2777,9 +2765,10 @@ void iwl_mvm_rx_bar_frame_release(struct iwl_mvm *mvm, struct napi_struct *napi,
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (WARN(tid != baid_data->tid || sta_id != baid_data->sta_id,
|
||||
"baid 0x%x is mapped to sta:%d tid:%d, but BAR release received for sta:%d tid:%d\n",
|
||||
baid, baid_data->sta_id, baid_data->tid, sta_id,
|
||||
if (WARN(tid != baid_data->tid || sta_id > IWL_MVM_STATION_COUNT_MAX ||
|
||||
!(baid_data->sta_mask & BIT(sta_id)),
|
||||
"baid 0x%x is mapped to sta_mask:0x%x tid:%d, but BAR release received for sta:%d tid:%d\n",
|
||||
baid, baid_data->sta_mask, baid_data->tid, sta_id,
|
||||
tid))
|
||||
goto out;
|
||||
|
||||
|
@ -45,6 +45,9 @@
|
||||
/* minimal number of 2GHz and 5GHz channels in the regular scan request */
|
||||
#define IWL_MVM_6GHZ_PASSIVE_SCAN_MIN_CHANS 4
|
||||
|
||||
/* Number of iterations on the channel for mei filtered scan */
|
||||
#define IWL_MEI_SCAN_NUM_ITER 5U
|
||||
|
||||
struct iwl_mvm_scan_timing_params {
|
||||
u32 suspend_time;
|
||||
u32 max_out_time;
|
||||
@ -98,6 +101,7 @@ struct iwl_mvm_scan_params {
|
||||
bool scan_6ghz;
|
||||
bool enable_6ghz_passive;
|
||||
bool respect_p2p_go, respect_p2p_go_hb;
|
||||
u8 bssid[ETH_ALEN] __aligned(2);
|
||||
};
|
||||
|
||||
static inline void *iwl_mvm_get_scan_req_umac_data(struct iwl_mvm *mvm)
|
||||
@ -240,8 +244,9 @@ iwl_mvm_scan_type _iwl_mvm_get_scan_type(struct iwl_mvm *mvm,
|
||||
* set all scan requests as fast-balance scan
|
||||
*/
|
||||
if (vif && vif->type == NL80211_IFTYPE_STATION &&
|
||||
vif->bss_conf.dtim_period < 220 &&
|
||||
data.is_dcm_with_p2p_go)
|
||||
data.is_dcm_with_p2p_go &&
|
||||
((vif->bss_conf.beacon_int *
|
||||
vif->bss_conf.dtim_period) < 220))
|
||||
return IWL_SCAN_TYPE_FAST_BALANCE;
|
||||
}
|
||||
|
||||
@ -759,7 +764,7 @@ iwl_mvm_build_scan_probe(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
|
||||
frame->frame_control = cpu_to_le16(IEEE80211_STYPE_PROBE_REQ);
|
||||
eth_broadcast_addr(frame->da);
|
||||
eth_broadcast_addr(frame->bssid);
|
||||
ether_addr_copy(frame->bssid, params->bssid);
|
||||
frame->seq_ctrl = 0;
|
||||
|
||||
pos = frame->u.probe_req.variable;
|
||||
@ -2081,6 +2086,11 @@ static u8 iwl_mvm_scan_umac_flags2(struct iwl_mvm *mvm,
|
||||
IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_RESPECT_P2P_GO_HB;
|
||||
}
|
||||
|
||||
if (params->scan_6ghz &&
|
||||
fw_has_capa(&mvm->fw->ucode_capa,
|
||||
IWL_UCODE_TLV_CAPA_SCAN_DONT_TOGGLE_ANT))
|
||||
flags |= IWL_UMAC_SCAN_GEN_PARAMS_FLAGS2_DONT_TOGGLE_ANT;
|
||||
|
||||
return flags;
|
||||
}
|
||||
|
||||
@ -2292,7 +2302,7 @@ iwl_mvm_scan_umac_fill_general_p_v11(struct iwl_mvm *mvm,
|
||||
|
||||
iwl_mvm_scan_umac_dwell_v11(mvm, gp, params);
|
||||
|
||||
IWL_DEBUG_SCAN(mvm, "Gerenal: flags=0x%x, flags2=0x%x\n",
|
||||
IWL_DEBUG_SCAN(mvm, "General: flags=0x%x, flags2=0x%x\n",
|
||||
gen_flags, gen_flags2);
|
||||
|
||||
gp->flags = cpu_to_le16(gen_flags);
|
||||
@ -2617,6 +2627,89 @@ static const struct iwl_scan_umac_handler iwl_scan_umac_handlers[] = {
|
||||
IWL_SCAN_UMAC_HANDLER(12),
|
||||
};
|
||||
|
||||
static void iwl_mvm_mei_scan_work(struct work_struct *wk)
|
||||
{
|
||||
struct iwl_mei_scan_filter *scan_filter =
|
||||
container_of(wk, struct iwl_mei_scan_filter, scan_work);
|
||||
struct iwl_mvm *mvm =
|
||||
container_of(scan_filter, struct iwl_mvm, mei_scan_filter);
|
||||
struct iwl_mvm_csme_conn_info *info;
|
||||
struct sk_buff *skb;
|
||||
u8 bssid[ETH_ALEN];
|
||||
|
||||
mutex_lock(&mvm->mutex);
|
||||
info = iwl_mvm_get_csme_conn_info(mvm);
|
||||
memcpy(bssid, info->conn_info.bssid, ETH_ALEN);
|
||||
mutex_unlock(&mvm->mutex);
|
||||
|
||||
while ((skb = skb_dequeue(&scan_filter->scan_res))) {
|
||||
struct ieee80211_mgmt *mgmt = (void *)skb->data;
|
||||
|
||||
if (!memcmp(mgmt->bssid, bssid, ETH_ALEN))
|
||||
ieee80211_rx_irqsafe(mvm->hw, skb);
|
||||
else
|
||||
kfree_skb(skb);
|
||||
}
|
||||
}
|
||||
|
||||
void iwl_mvm_mei_scan_filter_init(struct iwl_mei_scan_filter *mei_scan_filter)
|
||||
{
|
||||
skb_queue_head_init(&mei_scan_filter->scan_res);
|
||||
INIT_WORK(&mei_scan_filter->scan_work, iwl_mvm_mei_scan_work);
|
||||
}
|
||||
|
||||
/* In case CSME is connected and has link protection set, this function will
|
||||
* override the scan request to scan only the associated channel and only for
|
||||
* the associated SSID.
|
||||
*/
|
||||
static void iwl_mvm_mei_limited_scan(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_scan_params *params)
|
||||
{
|
||||
struct iwl_mvm_csme_conn_info *info = iwl_mvm_get_csme_conn_info(mvm);
|
||||
struct iwl_mei_conn_info *conn_info;
|
||||
struct ieee80211_channel *chan;
|
||||
int scan_iters, i;
|
||||
|
||||
if (!info) {
|
||||
IWL_DEBUG_SCAN(mvm, "mei_limited_scan: no connection info\n");
|
||||
return;
|
||||
}
|
||||
|
||||
conn_info = &info->conn_info;
|
||||
if (!info->conn_info.lp_state || !info->conn_info.ssid_len)
|
||||
return;
|
||||
|
||||
if (!params->n_channels || !params->n_ssids)
|
||||
return;
|
||||
|
||||
mvm->mei_scan_filter.is_mei_limited_scan = true;
|
||||
|
||||
chan = ieee80211_get_channel(mvm->hw->wiphy,
|
||||
ieee80211_channel_to_frequency(conn_info->channel,
|
||||
conn_info->band));
|
||||
if (!chan) {
|
||||
IWL_DEBUG_SCAN(mvm,
|
||||
"Failed to get CSME channel (chan=%u band=%u)\n",
|
||||
conn_info->channel, conn_info->band);
|
||||
return;
|
||||
}
|
||||
|
||||
/* The mei filtered scan must find the AP, otherwise CSME will
|
||||
* take the NIC ownership. Add several iterations on the channel to
|
||||
* make the scan more robust.
|
||||
*/
|
||||
scan_iters = min(IWL_MEI_SCAN_NUM_ITER, params->n_channels);
|
||||
params->n_channels = scan_iters;
|
||||
for (i = 0; i < scan_iters; i++)
|
||||
params->channels[i] = chan;
|
||||
|
||||
IWL_DEBUG_SCAN(mvm, "Mei scan: num iterations=%u\n", scan_iters);
|
||||
|
||||
params->n_ssids = 1;
|
||||
params->ssids[0].ssid_len = conn_info->ssid_len;
|
||||
memcpy(params->ssids[0].ssid, conn_info->ssid, conn_info->ssid_len);
|
||||
}
|
||||
|
||||
static int iwl_mvm_build_scan_cmd(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct iwl_host_cmd *hcmd,
|
||||
@ -2629,6 +2722,8 @@ static int iwl_mvm_build_scan_cmd(struct iwl_mvm *mvm,
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
memset(mvm->scan_cmd, 0, mvm->scan_cmd_size);
|
||||
|
||||
iwl_mvm_mei_limited_scan(mvm, params);
|
||||
|
||||
if (!fw_has_capa(&mvm->fw->ucode_capa, IWL_UCODE_TLV_CAPA_UMAC_SCAN)) {
|
||||
hcmd->id = SCAN_OFFLOAD_REQUEST_CMD;
|
||||
|
||||
@ -2795,6 +2890,7 @@ int iwl_mvm_reg_scan_start(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
params.pass_all = true;
|
||||
params.n_match_sets = 0;
|
||||
params.match_sets = NULL;
|
||||
ether_addr_copy(params.bssid, req->bssid);
|
||||
|
||||
params.scan_plans = &scan_plan;
|
||||
params.n_scan_plans = 1;
|
||||
@ -2888,6 +2984,7 @@ int iwl_mvm_sched_scan_start(struct iwl_mvm *mvm,
|
||||
params.pass_all = iwl_mvm_scan_pass_all(mvm, req);
|
||||
params.n_match_sets = req->n_match_sets;
|
||||
params.match_sets = req->match_sets;
|
||||
eth_broadcast_addr(params.bssid);
|
||||
if (!req->n_scan_plans)
|
||||
return -EINVAL;
|
||||
|
||||
@ -2983,6 +3080,8 @@ void iwl_mvm_rx_umac_scan_complete_notif(struct iwl_mvm *mvm,
|
||||
u32 uid = __le32_to_cpu(notif->uid);
|
||||
bool aborted = (notif->status == IWL_SCAN_OFFLOAD_ABORTED);
|
||||
|
||||
mvm->mei_scan_filter.is_mei_limited_scan = false;
|
||||
|
||||
if (WARN_ON(!(mvm->scan_uid_status[uid] & mvm->scan_status)))
|
||||
return;
|
||||
|
||||
|
@ -8,7 +8,7 @@
|
||||
/* For counting bound interfaces */
|
||||
struct iwl_mvm_active_iface_iterator_data {
|
||||
struct ieee80211_vif *ignore_vif;
|
||||
u8 sta_vif_ap_sta_id;
|
||||
struct ieee80211_sta *sta_vif_ap_sta;
|
||||
enum iwl_sf_state sta_vif_state;
|
||||
u32 num_active_macs;
|
||||
};
|
||||
@ -30,7 +30,7 @@ static void iwl_mvm_bound_iface_iterator(void *_data, u8 *mac,
|
||||
data->num_active_macs++;
|
||||
|
||||
if (vif->type == NL80211_IFTYPE_STATION) {
|
||||
data->sta_vif_ap_sta_id = mvmvif->deflink.ap_sta_id;
|
||||
data->sta_vif_ap_sta = mvmvif->ap_sta;
|
||||
if (vif->cfg.assoc)
|
||||
data->sta_vif_state = SF_FULL_ON;
|
||||
else
|
||||
@ -172,13 +172,12 @@ static void iwl_mvm_fill_sf_command(struct iwl_mvm *mvm,
|
||||
}
|
||||
}
|
||||
|
||||
static int iwl_mvm_sf_config(struct iwl_mvm *mvm, u8 sta_id,
|
||||
static int iwl_mvm_sf_config(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
enum iwl_sf_state new_state)
|
||||
{
|
||||
struct iwl_sf_cfg_cmd sf_cmd = {
|
||||
.state = cpu_to_le32(new_state),
|
||||
};
|
||||
struct ieee80211_sta *sta;
|
||||
int ret = 0;
|
||||
|
||||
if (mvm->cfg->disable_dummy_notification)
|
||||
@ -196,20 +195,12 @@ static int iwl_mvm_sf_config(struct iwl_mvm *mvm, u8 sta_id,
|
||||
iwl_mvm_fill_sf_command(mvm, &sf_cmd, NULL);
|
||||
break;
|
||||
case SF_FULL_ON:
|
||||
if (sta_id == IWL_MVM_INVALID_STA) {
|
||||
if (!sta) {
|
||||
IWL_ERR(mvm,
|
||||
"No station: Cannot switch SF to FULL_ON\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
rcu_read_lock();
|
||||
sta = rcu_dereference(mvm->fw_id_to_mac_id[sta_id]);
|
||||
if (IS_ERR_OR_NULL(sta)) {
|
||||
IWL_ERR(mvm, "Invalid station id\n");
|
||||
rcu_read_unlock();
|
||||
return -EINVAL;
|
||||
}
|
||||
iwl_mvm_fill_sf_command(mvm, &sf_cmd, sta);
|
||||
rcu_read_unlock();
|
||||
break;
|
||||
case SF_INIT_OFF:
|
||||
iwl_mvm_fill_sf_command(mvm, &sf_cmd, NULL);
|
||||
@ -237,13 +228,12 @@ int iwl_mvm_sf_update(struct iwl_mvm *mvm, struct ieee80211_vif *changed_vif,
|
||||
bool remove_vif)
|
||||
{
|
||||
enum iwl_sf_state new_state;
|
||||
u8 sta_id = IWL_MVM_INVALID_STA;
|
||||
struct iwl_mvm_vif *mvmvif = NULL;
|
||||
struct iwl_mvm_active_iface_iterator_data data = {
|
||||
.ignore_vif = changed_vif,
|
||||
.sta_vif_state = SF_UNINIT,
|
||||
.sta_vif_ap_sta_id = IWL_MVM_INVALID_STA,
|
||||
};
|
||||
struct ieee80211_sta *sta = NULL;
|
||||
|
||||
/*
|
||||
* Ignore the call if we are in HW Restart flow, or if the handled
|
||||
@ -273,7 +263,7 @@ int iwl_mvm_sf_update(struct iwl_mvm *mvm, struct ieee80211_vif *changed_vif,
|
||||
* and we filled the relevant data during iteration
|
||||
*/
|
||||
new_state = data.sta_vif_state;
|
||||
sta_id = data.sta_vif_ap_sta_id;
|
||||
sta = data.sta_vif_ap_sta;
|
||||
} else {
|
||||
if (WARN_ON(!changed_vif))
|
||||
return -EINVAL;
|
||||
@ -282,7 +272,7 @@ int iwl_mvm_sf_update(struct iwl_mvm *mvm, struct ieee80211_vif *changed_vif,
|
||||
} else if (changed_vif->cfg.assoc &&
|
||||
changed_vif->bss_conf.dtim_period) {
|
||||
mvmvif = iwl_mvm_vif_from_mac80211(changed_vif);
|
||||
sta_id = mvmvif->deflink.ap_sta_id;
|
||||
sta = mvmvif->ap_sta;
|
||||
new_state = SF_FULL_ON;
|
||||
} else {
|
||||
new_state = SF_INIT_OFF;
|
||||
@ -294,8 +284,5 @@ int iwl_mvm_sf_update(struct iwl_mvm *mvm, struct ieee80211_vif *changed_vif,
|
||||
new_state = SF_UNINIT;
|
||||
}
|
||||
|
||||
/* For MLO it's ok to use deflink->sta_id as it's needed only to get
|
||||
* a pointer to mac80211 sta
|
||||
*/
|
||||
return iwl_mvm_sf_config(mvm, sta_id, new_state);
|
||||
return iwl_mvm_sf_config(mvm, sta, new_state);
|
||||
}
|
||||
|
@ -251,6 +251,7 @@ static void iwl_mvm_rx_agg_session_expired(struct timer_list *t)
|
||||
struct ieee80211_sta *sta;
|
||||
struct iwl_mvm_sta *mvm_sta;
|
||||
unsigned long timeout;
|
||||
unsigned int sta_id;
|
||||
|
||||
rcu_read_lock();
|
||||
|
||||
@ -269,7 +270,8 @@ static void iwl_mvm_rx_agg_session_expired(struct timer_list *t)
|
||||
}
|
||||
|
||||
/* Timer expired */
|
||||
sta = rcu_dereference(ba_data->mvm->fw_id_to_mac_id[ba_data->sta_id]);
|
||||
sta_id = ffs(ba_data->sta_mask) - 1; /* don't care which one */
|
||||
sta = rcu_dereference(ba_data->mvm->fw_id_to_mac_id[sta_id]);
|
||||
|
||||
/*
|
||||
* sta should be valid unless the following happens:
|
||||
@ -356,10 +358,14 @@ static int iwl_mvm_disable_txq(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
SCD_QUEUE_CONFIG_CMD);
|
||||
struct iwl_scd_queue_cfg_cmd remove_cmd = {
|
||||
.operation = cpu_to_le32(IWL_SCD_QUEUE_REMOVE),
|
||||
.u.remove.tid = cpu_to_le32(tid),
|
||||
.u.remove.sta_mask = cpu_to_le32(BIT(sta_id)),
|
||||
};
|
||||
|
||||
if (tid == IWL_MAX_TID_COUNT)
|
||||
tid = IWL_MGMT_TID;
|
||||
|
||||
remove_cmd.u.remove.tid = cpu_to_le32(tid);
|
||||
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm, cmd_id, 0,
|
||||
sizeof(remove_cmd),
|
||||
&remove_cmd);
|
||||
@ -2146,7 +2152,7 @@ int iwl_mvm_allocate_int_sta(struct iwl_mvm *mvm,
|
||||
sta->type = type;
|
||||
|
||||
/* put a non-NULL value so iterating over the stations won't stop */
|
||||
rcu_assign_pointer(mvm->fw_id_to_mac_id[sta->sta_id], ERR_PTR(-EINVAL));
|
||||
RCU_INIT_POINTER(mvm->fw_id_to_mac_id[sta->sta_id], ERR_PTR(-EINVAL));
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -2227,11 +2233,13 @@ static int iwl_mvm_add_int_sta_with_queue(struct iwl_mvm *mvm, int macidx,
|
||||
int iwl_mvm_add_aux_sta(struct iwl_mvm *mvm, u32 lmac_id)
|
||||
{
|
||||
int ret;
|
||||
u32 qmask = mvm->aux_queue == IWL_MVM_INVALID_QUEUE ? 0 :
|
||||
BIT(mvm->aux_queue);
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
/* Allocate aux station and assign to it the aux queue */
|
||||
ret = iwl_mvm_allocate_int_sta(mvm, &mvm->aux_sta, BIT(mvm->aux_queue),
|
||||
ret = iwl_mvm_allocate_int_sta(mvm, &mvm->aux_sta, qmask,
|
||||
NL80211_IFTYPE_UNSPECIFIED,
|
||||
IWL_STA_AUX_ACTIVITY);
|
||||
if (ret)
|
||||
@ -2750,10 +2758,11 @@ static void iwl_mvm_init_reorder_buffer(struct iwl_mvm *mvm,
|
||||
}
|
||||
|
||||
static int iwl_mvm_fw_baid_op_sta(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_sta *mvm_sta,
|
||||
struct ieee80211_sta *sta,
|
||||
bool start, int tid, u16 ssn,
|
||||
u16 buf_size)
|
||||
{
|
||||
struct iwl_mvm_sta *mvm_sta = iwl_mvm_sta_from_mac80211(sta);
|
||||
struct iwl_mvm_add_sta_cmd cmd = {
|
||||
.mac_id_n_color = cpu_to_le32(mvm_sta->mac_id_n_color),
|
||||
.sta_id = mvm_sta->deflink.sta_id,
|
||||
@ -2798,7 +2807,7 @@ static int iwl_mvm_fw_baid_op_sta(struct iwl_mvm *mvm,
|
||||
}
|
||||
|
||||
static int iwl_mvm_fw_baid_op_cmd(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_sta *mvm_sta,
|
||||
struct ieee80211_sta *sta,
|
||||
bool start, int tid, u16 ssn,
|
||||
u16 buf_size, int baid)
|
||||
{
|
||||
@ -2813,7 +2822,7 @@ static int iwl_mvm_fw_baid_op_cmd(struct iwl_mvm *mvm,
|
||||
|
||||
if (start) {
|
||||
cmd.alloc.sta_id_mask =
|
||||
cpu_to_le32(BIT(mvm_sta->deflink.sta_id));
|
||||
cpu_to_le32(iwl_mvm_sta_fw_id_mask(mvm, sta, -1));
|
||||
cmd.alloc.tid = tid;
|
||||
cmd.alloc.ssn = cpu_to_le16(ssn);
|
||||
cmd.alloc.win_size = cpu_to_le16(buf_size);
|
||||
@ -2823,7 +2832,7 @@ static int iwl_mvm_fw_baid_op_cmd(struct iwl_mvm *mvm,
|
||||
BUILD_BUG_ON(sizeof(cmd.remove_v1) > sizeof(cmd.remove));
|
||||
} else {
|
||||
cmd.remove.sta_id_mask =
|
||||
cpu_to_le32(BIT(mvm_sta->deflink.sta_id));
|
||||
cpu_to_le32(iwl_mvm_sta_fw_id_mask(mvm, sta, -1));
|
||||
cmd.remove.tid = cpu_to_le32(tid);
|
||||
}
|
||||
|
||||
@ -2846,16 +2855,16 @@ static int iwl_mvm_fw_baid_op_cmd(struct iwl_mvm *mvm,
|
||||
return baid;
|
||||
}
|
||||
|
||||
static int iwl_mvm_fw_baid_op(struct iwl_mvm *mvm, struct iwl_mvm_sta *mvm_sta,
|
||||
static int iwl_mvm_fw_baid_op(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
bool start, int tid, u16 ssn, u16 buf_size,
|
||||
int baid)
|
||||
{
|
||||
if (fw_has_capa(&mvm->fw->ucode_capa,
|
||||
IWL_UCODE_TLV_CAPA_BAID_ML_SUPPORT))
|
||||
return iwl_mvm_fw_baid_op_cmd(mvm, mvm_sta, start,
|
||||
return iwl_mvm_fw_baid_op_cmd(mvm, sta, start,
|
||||
tid, ssn, buf_size, baid);
|
||||
|
||||
return iwl_mvm_fw_baid_op_sta(mvm, mvm_sta, start,
|
||||
return iwl_mvm_fw_baid_op_sta(mvm, sta, start,
|
||||
tid, ssn, buf_size);
|
||||
}
|
||||
|
||||
@ -2925,7 +2934,7 @@ int iwl_mvm_sta_rx_agg(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
|
||||
/* Don't send command to remove (start=0) BAID during restart */
|
||||
if (start || !test_bit(IWL_MVM_STATUS_IN_HW_RESTART, &mvm->status))
|
||||
baid = iwl_mvm_fw_baid_op(mvm, mvm_sta, start, tid, ssn, buf_size,
|
||||
baid = iwl_mvm_fw_baid_op(mvm, sta, start, tid, ssn, buf_size,
|
||||
baid);
|
||||
|
||||
if (baid < 0) {
|
||||
@ -2947,7 +2956,7 @@ int iwl_mvm_sta_rx_agg(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
iwl_mvm_rx_agg_session_expired, 0);
|
||||
baid_data->mvm = mvm;
|
||||
baid_data->tid = tid;
|
||||
baid_data->sta_id = mvm_sta->deflink.sta_id;
|
||||
baid_data->sta_mask = iwl_mvm_sta_fw_id_mask(mvm, sta, -1);
|
||||
|
||||
mvm_sta->tid_to_baid[tid] = baid;
|
||||
if (timeout)
|
||||
|
@ -356,6 +356,7 @@ struct iwl_mvm_link_sta {
|
||||
* @tid_disable_agg: bitmap: if bit(tid) is set, the fw won't send ampdus for
|
||||
* tid.
|
||||
* @sta_type: station type
|
||||
* @authorized: indicates station is authorized
|
||||
* @sta_state: station state according to enum %ieee80211_sta_state
|
||||
* @bt_reduced_txpower: is reduced tx power enabled for this station
|
||||
* @next_status_eosp: the next reclaimed packet is a PS-Poll response and
|
||||
@ -409,6 +410,7 @@ struct iwl_mvm_sta {
|
||||
enum ieee80211_sta_state sta_state;
|
||||
bool bt_reduced_txpower;
|
||||
bool next_status_eosp;
|
||||
bool authorized;
|
||||
spinlock_t lock;
|
||||
struct iwl_mvm_tid_data tid_data[IWL_MAX_TID_COUNT + 1];
|
||||
u8 tid_to_baid[IWL_MAX_TID_COUNT];
|
||||
@ -642,6 +644,8 @@ int iwl_mvm_mld_update_sta_links(struct iwl_mvm *mvm,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
u16 old_links, u16 new_links);
|
||||
u32 iwl_mvm_sta_fw_id_mask(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
int filter_link_id);
|
||||
|
||||
/* Queues */
|
||||
void iwl_mvm_mld_modify_all_sta_disable_tx(struct iwl_mvm *mvm,
|
||||
|
@ -184,10 +184,7 @@ static u32 iwl_mvm_tx_csum(struct iwl_mvm *mvm, struct sk_buff *skb,
|
||||
struct ieee80211_tx_info *info,
|
||||
bool amsdu)
|
||||
{
|
||||
if (mvm->trans->trans_cfg->device_family < IWL_DEVICE_FAMILY_BZ ||
|
||||
(mvm->trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_BZ &&
|
||||
CSR_HW_REV_TYPE(mvm->trans->hw_rev) == IWL_CFG_MAC_TYPE_GL &&
|
||||
mvm->trans->hw_rev_step == SILICON_A_STEP))
|
||||
if (!iwl_mvm_has_new_tx_csum(mvm))
|
||||
return iwl_mvm_tx_csum_pre_bz(mvm, skb, info, amsdu);
|
||||
return iwl_mvm_tx_csum_bz(mvm, skb, amsdu);
|
||||
}
|
||||
@ -332,22 +329,23 @@ static u32 iwl_mvm_get_tx_rate(struct iwl_mvm *mvm,
|
||||
sta ? iwl_mvm_sta_from_mac80211(sta)->sta_state : -1);
|
||||
|
||||
rate_idx = info->control.rates[0].idx;
|
||||
|
||||
/* For non 2 GHZ band, remap mac80211 rate indices into driver
|
||||
* indices.
|
||||
*/
|
||||
if (info->band != NL80211_BAND_2GHZ ||
|
||||
(info->flags & IEEE80211_TX_CTL_NO_CCK_RATE))
|
||||
rate_idx += IWL_FIRST_OFDM_RATE;
|
||||
|
||||
/* For 2.4 GHZ band, check that there is no need to remap */
|
||||
BUILD_BUG_ON(IWL_FIRST_CCK_RATE != 0);
|
||||
}
|
||||
|
||||
/* if the rate isn't a well known legacy rate, take the lowest one */
|
||||
if (rate_idx < 0 || rate_idx >= IWL_RATE_COUNT_LEGACY)
|
||||
rate_idx = rate_lowest_index(
|
||||
&mvm->nvm_data->bands[info->band], sta);
|
||||
|
||||
/*
|
||||
* For non 2 GHZ band, remap mac80211 rate
|
||||
* indices into driver indices
|
||||
*/
|
||||
if (info->band != NL80211_BAND_2GHZ)
|
||||
rate_idx += IWL_FIRST_OFDM_RATE;
|
||||
|
||||
/* For 2.4 GHZ band, check that there is no need to remap */
|
||||
BUILD_BUG_ON(IWL_FIRST_CCK_RATE != 0);
|
||||
rate_idx = iwl_mvm_mac_ctxt_get_lowest_rate(mvm,
|
||||
info,
|
||||
info->control.vif);
|
||||
|
||||
/* Get PLCP rate for tx_cmd->rate_n_flags */
|
||||
rate_plcp = iwl_mvm_mac80211_idx_to_hwrate(mvm->fw, rate_idx);
|
||||
@ -606,8 +604,9 @@ static void iwl_mvm_skb_prepare_status(struct sk_buff *skb,
|
||||
static int iwl_mvm_get_ctrl_vif_queue(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_vif_link_info *link,
|
||||
struct ieee80211_tx_info *info,
|
||||
struct ieee80211_hdr *hdr)
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
struct ieee80211_hdr *hdr = (void *)skb->data;
|
||||
__le16 fc = hdr->frame_control;
|
||||
|
||||
switch (info->control.vif->type) {
|
||||
@ -624,7 +623,7 @@ static int iwl_mvm_get_ctrl_vif_queue(struct iwl_mvm *mvm,
|
||||
* reason 7 ("Class 3 frame received from nonassociated STA").
|
||||
*/
|
||||
if (ieee80211_is_mgmt(fc) &&
|
||||
(!ieee80211_is_bufferable_mmpdu(fc) ||
|
||||
(!ieee80211_is_bufferable_mmpdu(skb) ||
|
||||
ieee80211_is_deauth(fc) || ieee80211_is_disassoc(fc)))
|
||||
return link->mgmt_queue;
|
||||
|
||||
@ -757,7 +756,7 @@ int iwl_mvm_tx_skb_non_sta(struct iwl_mvm *mvm, struct sk_buff *skb)
|
||||
sta_id = link->mcast_sta.sta_id;
|
||||
|
||||
queue = iwl_mvm_get_ctrl_vif_queue(mvm, link, &info,
|
||||
hdr);
|
||||
skb);
|
||||
} else if (info.control.vif->type == NL80211_IFTYPE_MONITOR) {
|
||||
queue = mvm->snif_queue;
|
||||
sta_id = mvm->snif_sta.sta_id;
|
||||
@ -805,10 +804,11 @@ unsigned int iwl_mvm_max_amsdu_size(struct iwl_mvm *mvm,
|
||||
struct ieee80211_sta *sta, unsigned int tid)
|
||||
{
|
||||
struct iwl_mvm_sta *mvmsta = iwl_mvm_sta_from_mac80211(sta);
|
||||
enum nl80211_band band = mvmsta->vif->bss_conf.chandef.chan->band;
|
||||
u8 ac = tid_to_mac80211_ac[tid];
|
||||
enum nl80211_band band;
|
||||
unsigned int txf;
|
||||
int lmac = iwl_mvm_get_lmac_id(mvm->fw, band);
|
||||
unsigned int val;
|
||||
int lmac;
|
||||
|
||||
/* For HE redirect to trigger based fifos */
|
||||
if (sta->deflink.he_cap.has_he && !WARN_ON(!iwl_mvm_has_new_tx_api(mvm)))
|
||||
@ -822,7 +822,37 @@ unsigned int iwl_mvm_max_amsdu_size(struct iwl_mvm *mvm,
|
||||
* We also want to have the start of the next packet inside the
|
||||
* fifo to be able to send bursts.
|
||||
*/
|
||||
return min_t(unsigned int, mvmsta->max_amsdu_len,
|
||||
val = mvmsta->max_amsdu_len;
|
||||
|
||||
if (hweight16(sta->valid_links) <= 1) {
|
||||
if (sta->valid_links) {
|
||||
struct ieee80211_bss_conf *link_conf;
|
||||
unsigned int link = ffs(sta->valid_links) - 1;
|
||||
|
||||
rcu_read_lock();
|
||||
link_conf = rcu_dereference(mvmsta->vif->link_conf[link]);
|
||||
if (WARN_ON(!link_conf))
|
||||
band = NL80211_BAND_2GHZ;
|
||||
else
|
||||
band = link_conf->chandef.chan->band;
|
||||
rcu_read_unlock();
|
||||
} else {
|
||||
band = mvmsta->vif->bss_conf.chandef.chan->band;
|
||||
}
|
||||
|
||||
lmac = iwl_mvm_get_lmac_id(mvm->fw, band);
|
||||
} else if (fw_has_capa(&mvm->fw->ucode_capa,
|
||||
IWL_UCODE_TLV_CAPA_CDB_SUPPORT)) {
|
||||
/* for real MLO restrict to both LMACs if they exist */
|
||||
lmac = IWL_LMAC_5G_INDEX;
|
||||
val = min_t(unsigned int, val,
|
||||
mvm->fwrt.smem_cfg.lmac[lmac].txfifo_size[txf] - 256);
|
||||
lmac = IWL_LMAC_24G_INDEX;
|
||||
} else {
|
||||
lmac = IWL_LMAC_24G_INDEX;
|
||||
}
|
||||
|
||||
return min_t(unsigned int, val,
|
||||
mvm->fwrt.smem_cfg.lmac[lmac].txfifo_size[txf] - 256);
|
||||
}
|
||||
|
||||
@ -1256,8 +1286,7 @@ int iwl_mvm_tx_skb_sta(struct iwl_mvm *mvm, struct sk_buff *skb,
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (WARN_ON(skb_queue_empty(&mpdus_skbs)))
|
||||
return ret;
|
||||
WARN_ON(skb_queue_empty(&mpdus_skbs));
|
||||
|
||||
while (!skb_queue_empty(&mpdus_skbs)) {
|
||||
skb = __skb_dequeue(&mpdus_skbs);
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -497,6 +497,7 @@ int iwl_pcie_rx_stop(struct iwl_trans *trans);
|
||||
void iwl_pcie_rx_free(struct iwl_trans *trans);
|
||||
void iwl_pcie_free_rbs_pool(struct iwl_trans *trans);
|
||||
void iwl_pcie_rx_init_rxb_lists(struct iwl_rxq *rxq);
|
||||
void iwl_pcie_rx_napi_sync(struct iwl_trans *trans);
|
||||
void iwl_pcie_rxq_alloc_rbs(struct iwl_trans *trans, gfp_t priority,
|
||||
struct iwl_rxq *rxq);
|
||||
|
||||
|
@ -1,6 +1,6 @@
|
||||
// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
|
||||
/*
|
||||
* Copyright (C) 2003-2014, 2018-2022 Intel Corporation
|
||||
* Copyright (C) 2003-2014, 2018-2023 Intel Corporation
|
||||
* Copyright (C) 2013-2015 Intel Mobile Communications GmbH
|
||||
* Copyright (C) 2016-2017 Intel Deutschland GmbH
|
||||
*/
|
||||
@ -1053,6 +1053,22 @@ static int iwl_pcie_napi_poll_msix(struct napi_struct *napi, int budget)
|
||||
return ret;
|
||||
}
|
||||
|
||||
void iwl_pcie_rx_napi_sync(struct iwl_trans *trans)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int i;
|
||||
|
||||
if (unlikely(!trans_pcie->rxq))
|
||||
return;
|
||||
|
||||
for (i = 0; i < trans->num_rx_queues; i++) {
|
||||
struct iwl_rxq *rxq = &trans_pcie->rxq[i];
|
||||
|
||||
if (rxq && rxq->napi.poll)
|
||||
napi_synchronize(&rxq->napi);
|
||||
}
|
||||
}
|
||||
|
||||
static int _iwl_pcie_rx_init(struct iwl_trans *trans)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
@ -156,6 +156,7 @@ void _iwl_trans_pcie_gen2_stop_device(struct iwl_trans *trans)
|
||||
if (test_and_clear_bit(STATUS_DEVICE_ENABLED, &trans->status)) {
|
||||
IWL_DEBUG_INFO(trans,
|
||||
"DEVICE_ENABLED bit was set and is now cleared\n");
|
||||
iwl_pcie_rx_napi_sync(trans);
|
||||
iwl_txq_gen2_tx_free(trans);
|
||||
iwl_pcie_rx_stop(trans);
|
||||
}
|
||||
@ -350,7 +351,7 @@ void iwl_trans_pcie_gen2_fw_alive(struct iwl_trans *trans, u32 scd_addr)
|
||||
mutex_unlock(&trans_pcie->mutex);
|
||||
}
|
||||
|
||||
static void iwl_pcie_set_ltr(struct iwl_trans *trans)
|
||||
static bool iwl_pcie_set_ltr(struct iwl_trans *trans)
|
||||
{
|
||||
u32 ltr_val = CSR_LTR_LONG_VAL_AD_NO_SNOOP_REQ |
|
||||
u32_encode_bits(CSR_LTR_LONG_VAL_AD_SCALE_USEC,
|
||||
@ -371,18 +372,77 @@ static void iwl_pcie_set_ltr(struct iwl_trans *trans)
|
||||
trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_22000) &&
|
||||
!trans->trans_cfg->integrated) {
|
||||
iwl_write32(trans, CSR_LTR_LONG_VAL_AD, ltr_val);
|
||||
} else if (trans->trans_cfg->integrated &&
|
||||
trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_22000) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (trans->trans_cfg->integrated &&
|
||||
trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_22000) {
|
||||
iwl_write_prph(trans, HPM_MAC_LTR_CSR, HPM_MAC_LRT_ENABLE_ALL);
|
||||
iwl_write_prph(trans, HPM_UMAC_LTR, ltr_val);
|
||||
return true;
|
||||
}
|
||||
|
||||
if (trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_AX210) {
|
||||
/* First clear the interrupt, just in case */
|
||||
iwl_write32(trans, CSR_MSIX_HW_INT_CAUSES_AD,
|
||||
MSIX_HW_INT_CAUSES_REG_IML);
|
||||
/* In this case, unfortunately the same ROM bug exists in the
|
||||
* device (not setting LTR correctly), but we don't have control
|
||||
* over the settings from the host due to some hardware security
|
||||
* features. The only workaround we've been able to come up with
|
||||
* so far is to try to keep the CPU and device busy by polling
|
||||
* it and the IML (image loader) completed interrupt.
|
||||
*/
|
||||
return false;
|
||||
}
|
||||
|
||||
/* nothing needs to be done on other devices */
|
||||
return true;
|
||||
}
|
||||
|
||||
static void iwl_pcie_spin_for_iml(struct iwl_trans *trans)
|
||||
{
|
||||
/* in practice, this seems to complete in around 20-30ms at most, wait 100 */
|
||||
#define IML_WAIT_TIMEOUT (HZ / 10)
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
unsigned long end_time = jiffies + IML_WAIT_TIMEOUT;
|
||||
u32 value, loops = 0;
|
||||
bool irq = false;
|
||||
|
||||
if (WARN_ON(!trans_pcie->iml))
|
||||
return;
|
||||
|
||||
value = iwl_read32(trans, CSR_LTR_LAST_MSG);
|
||||
IWL_DEBUG_INFO(trans, "Polling for IML load - CSR_LTR_LAST_MSG=0x%x\n",
|
||||
value);
|
||||
|
||||
while (time_before(jiffies, end_time)) {
|
||||
if (iwl_read32(trans, CSR_MSIX_HW_INT_CAUSES_AD) &
|
||||
MSIX_HW_INT_CAUSES_REG_IML) {
|
||||
irq = true;
|
||||
break;
|
||||
}
|
||||
/* Keep the CPU and device busy. */
|
||||
value = iwl_read32(trans, CSR_LTR_LAST_MSG);
|
||||
loops++;
|
||||
}
|
||||
|
||||
IWL_DEBUG_INFO(trans,
|
||||
"Polled for IML load: irq=%d, loops=%d, CSR_LTR_LAST_MSG=0x%x\n",
|
||||
irq, loops, value);
|
||||
|
||||
/* We don't fail here even if we timed out - maybe we get lucky and the
|
||||
* interrupt comes in later (and we get alive from firmware) and then
|
||||
* we're all happy - but if not we'll fail on alive timeout or get some
|
||||
* other error out.
|
||||
*/
|
||||
}
|
||||
|
||||
int iwl_trans_pcie_gen2_start_fw(struct iwl_trans *trans,
|
||||
const struct fw_img *fw, bool run_in_rfkill)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
bool hw_rfkill;
|
||||
bool hw_rfkill, keep_ram_busy;
|
||||
int ret;
|
||||
|
||||
/* This may fail if AMT took ownership of the device */
|
||||
@ -443,7 +503,7 @@ int iwl_trans_pcie_gen2_start_fw(struct iwl_trans *trans,
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
iwl_pcie_set_ltr(trans);
|
||||
keep_ram_busy = !iwl_pcie_set_ltr(trans);
|
||||
|
||||
if (trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) {
|
||||
iwl_write32(trans, CSR_FUNC_SCRATCH, CSR_FUNC_SCRATCH_INIT_VALUE);
|
||||
@ -455,6 +515,9 @@ int iwl_trans_pcie_gen2_start_fw(struct iwl_trans *trans,
|
||||
iwl_write_prph(trans, UREG_CPU_INIT_RUN, 1);
|
||||
}
|
||||
|
||||
if (keep_ram_busy)
|
||||
iwl_pcie_spin_for_iml(trans);
|
||||
|
||||
/* re-check RF-Kill state since we may have missed the interrupt */
|
||||
hw_rfkill = iwl_pcie_check_hw_rf_kill(trans);
|
||||
if (hw_rfkill && !run_in_rfkill)
|
||||
|
@ -599,7 +599,6 @@ static int iwl_pcie_set_hw_ready(struct iwl_trans *trans)
|
||||
int iwl_pcie_prepare_card_hw(struct iwl_trans *trans)
|
||||
{
|
||||
int ret;
|
||||
int t = 0;
|
||||
int iter;
|
||||
|
||||
IWL_DEBUG_INFO(trans, "iwl_trans_prepare_card_hw enter\n");
|
||||
@ -616,6 +615,8 @@ int iwl_pcie_prepare_card_hw(struct iwl_trans *trans)
|
||||
usleep_range(1000, 2000);
|
||||
|
||||
for (iter = 0; iter < 10; iter++) {
|
||||
int t = 0;
|
||||
|
||||
/* If HW is not ready, prepare the conditions to check again */
|
||||
iwl_set_bit(trans, CSR_HW_IF_CONFIG_REG,
|
||||
CSR_HW_IF_CONFIG_REG_PREPARE);
|
||||
@ -1260,6 +1261,7 @@ static void _iwl_trans_pcie_stop_device(struct iwl_trans *trans)
|
||||
if (test_and_clear_bit(STATUS_DEVICE_ENABLED, &trans->status)) {
|
||||
IWL_DEBUG_INFO(trans,
|
||||
"DEVICE_ENABLED bit was set and is now cleared\n");
|
||||
iwl_pcie_rx_napi_sync(trans);
|
||||
iwl_pcie_tx_stop(trans);
|
||||
iwl_pcie_rx_stop(trans);
|
||||
|
||||
@ -1522,19 +1524,16 @@ static int iwl_pcie_d3_handshake(struct iwl_trans *trans, bool suspend)
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int ret;
|
||||
|
||||
if (trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_AX210) {
|
||||
if (trans->trans_cfg->device_family == IWL_DEVICE_FAMILY_AX210)
|
||||
iwl_write_umac_prph(trans, UREG_DOORBELL_TO_ISR6,
|
||||
suspend ? UREG_DOORBELL_TO_ISR6_SUSPEND :
|
||||
UREG_DOORBELL_TO_ISR6_RESUME);
|
||||
} else if (trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) {
|
||||
else if (trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ)
|
||||
iwl_write32(trans, CSR_IPC_SLEEP_CONTROL,
|
||||
suspend ? CSR_IPC_SLEEP_CONTROL_SUSPEND :
|
||||
CSR_IPC_SLEEP_CONTROL_RESUME);
|
||||
iwl_write_umac_prph(trans, UREG_DOORBELL_TO_ISR6,
|
||||
UREG_DOORBELL_TO_ISR6_SLEEP_CTRL);
|
||||
} else {
|
||||
else
|
||||
return 0;
|
||||
}
|
||||
|
||||
ret = wait_event_timeout(trans_pcie->sx_waitq,
|
||||
trans_pcie->sx_complete, 2 * HZ);
|
||||
@ -2863,7 +2862,7 @@ static bool iwl_write_to_user_buf(char __user *user_buf, ssize_t count,
|
||||
void *buf, ssize_t *size,
|
||||
ssize_t *bytes_copied)
|
||||
{
|
||||
int buf_size_left = count - *bytes_copied;
|
||||
ssize_t buf_size_left = count - *bytes_copied;
|
||||
|
||||
buf_size_left = buf_size_left - (buf_size_left % sizeof(u32));
|
||||
if (*size > buf_size_left)
|
||||
|
@ -1554,14 +1554,18 @@ void iwl_txq_reclaim(struct iwl_trans *trans, int txq_id, int ssn,
|
||||
struct sk_buff_head *skbs)
|
||||
{
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
int tfd_num = iwl_txq_get_cmd_index(txq, ssn);
|
||||
int read_ptr = iwl_txq_get_cmd_index(txq, txq->read_ptr);
|
||||
int last_to_free;
|
||||
int tfd_num, read_ptr, last_to_free;
|
||||
|
||||
/* This function is not meant to release cmd queue*/
|
||||
if (WARN_ON(txq_id == trans->txqs.cmd.q_id))
|
||||
return;
|
||||
|
||||
if (WARN_ON(!txq))
|
||||
return;
|
||||
|
||||
tfd_num = iwl_txq_get_cmd_index(txq, ssn);
|
||||
read_ptr = iwl_txq_get_cmd_index(txq, txq->read_ptr);
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
|
||||
if (!test_bit(txq_id, trans->txqs.queue_used)) {
|
||||
|
@ -402,8 +402,8 @@ mt76_dma_get_buf(struct mt76_dev *dev, struct mt76_queue *q, int idx,
|
||||
*info = le32_to_cpu(desc->info);
|
||||
|
||||
if (mt76_queue_is_wed_rx(q)) {
|
||||
u32 token = FIELD_GET(MT_DMA_CTL_TOKEN,
|
||||
le32_to_cpu(desc->buf1));
|
||||
u32 buf1 = le32_to_cpu(desc->buf1);
|
||||
u32 token = FIELD_GET(MT_DMA_CTL_TOKEN, buf1);
|
||||
struct mt76_txwi_cache *t = mt76_rx_token_release(dev, token);
|
||||
|
||||
if (!t)
|
||||
@ -424,6 +424,8 @@ mt76_dma_get_buf(struct mt76_dev *dev, struct mt76_queue *q, int idx,
|
||||
|
||||
*drop = !!(ctrl & (MT_DMA_CTL_TO_HOST_A |
|
||||
MT_DMA_CTL_DROP));
|
||||
|
||||
*drop |= !!(buf1 & MT_DMA_CTL_WO_DROP);
|
||||
}
|
||||
} else {
|
||||
buf = e->buf;
|
||||
@ -576,7 +578,9 @@ mt76_dma_tx_queue_skb(struct mt76_dev *dev, struct mt76_queue *q,
|
||||
free_skb:
|
||||
status.skb = tx_info.skb;
|
||||
hw = mt76_tx_status_get_hw(dev, tx_info.skb);
|
||||
spin_lock_bh(&dev->rx_lock);
|
||||
ieee80211_tx_status_ext(hw, &status);
|
||||
spin_unlock_bh(&dev->rx_lock);
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -849,7 +853,7 @@ mt76_dma_rx_process(struct mt76_dev *dev, struct mt76_queue *q, int budget)
|
||||
!(dev->drv->rx_check(dev, data, len)))
|
||||
goto free_frag;
|
||||
|
||||
skb = build_skb(data, q->buf_size);
|
||||
skb = napi_build_skb(data, q->buf_size);
|
||||
if (!skb)
|
||||
goto free_frag;
|
||||
|
||||
|
@ -19,6 +19,7 @@
|
||||
#define MT_DMA_CTL_TO_HOST_A BIT(12)
|
||||
#define MT_DMA_CTL_DROP BIT(14)
|
||||
#define MT_DMA_CTL_TOKEN GENMASK(31, 16)
|
||||
#define MT_DMA_CTL_WO_DROP BIT(8)
|
||||
|
||||
#define MT_DMA_PPE_CPU_REASON GENMASK(15, 11)
|
||||
#define MT_DMA_PPE_ENTRY GENMASK(30, 16)
|
||||
|
@ -418,7 +418,8 @@ mt76_phy_init(struct mt76_phy *phy, struct ieee80211_hw *hw)
|
||||
SET_IEEE80211_DEV(hw, dev->dev);
|
||||
SET_IEEE80211_PERM_ADDR(hw, phy->macaddr);
|
||||
|
||||
wiphy->features |= NL80211_FEATURE_ACTIVE_MONITOR;
|
||||
wiphy->features |= NL80211_FEATURE_ACTIVE_MONITOR |
|
||||
NL80211_FEATURE_AP_MODE_CHAN_WIDTH_CHANGE;
|
||||
wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH |
|
||||
WIPHY_FLAG_SUPPORTS_TDLS |
|
||||
WIPHY_FLAG_AP_UAPSD;
|
||||
@ -1066,9 +1067,14 @@ mt76_rx_convert(struct mt76_dev *dev, struct sk_buff *skb,
|
||||
status->enc_flags = mstat.enc_flags;
|
||||
status->encoding = mstat.encoding;
|
||||
status->bw = mstat.bw;
|
||||
status->he_ru = mstat.he_ru;
|
||||
status->he_gi = mstat.he_gi;
|
||||
status->he_dcm = mstat.he_dcm;
|
||||
if (status->encoding == RX_ENC_EHT) {
|
||||
status->eht.ru = mstat.eht.ru;
|
||||
status->eht.gi = mstat.eht.gi;
|
||||
} else {
|
||||
status->he_ru = mstat.he_ru;
|
||||
status->he_gi = mstat.he_gi;
|
||||
status->he_dcm = mstat.he_dcm;
|
||||
}
|
||||
status->rate_idx = mstat.rate_idx;
|
||||
status->nss = mstat.nss;
|
||||
status->band = mstat.band;
|
||||
@ -1303,7 +1309,8 @@ mt76_check_sta(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
if (ps)
|
||||
set_bit(MT_WCID_FLAG_PS, &wcid->flags);
|
||||
|
||||
dev->drv->sta_ps(dev, sta, ps);
|
||||
if (dev->drv->sta_ps)
|
||||
dev->drv->sta_ps(dev, sta, ps);
|
||||
|
||||
if (!ps)
|
||||
clear_bit(MT_WCID_FLAG_PS, &wcid->flags);
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user