[PATCH V3 1/7] drivers/acpi: Add support for Wifi band RF mitigations
Quan, Evan
Evan.Quan at amd.com
Wed Jun 21 01:26:05 UTC 2023
[AMD Official Use Only - General]
> -----Original Message-----
> From: Limonciello, Mario <Mario.Limonciello at amd.com>
> Sent: Monday, June 19, 2023 10:04 AM
> To: Quan, Evan <Evan.Quan at amd.com>
> Cc: linux-kernel at vger.kernel.org; linux-acpi at vger.kernel.org; amd-
> gfx at lists.freedesktop.org; dri-devel at lists.freedesktop.org; linux-
> wireless at vger.kernel.org; rafael at kernel.org; lenb at kernel.org; Deucher,
> Alexander <Alexander.Deucher at amd.com>; Koenig, Christian
> <Christian.Koenig at amd.com>; Pan, Xinhui <Xinhui.Pan at amd.com>;
> airlied at gmail.com; daniel at ffwll.ch; kvalo at kernel.org; nbd at nbd.name;
> lorenzo at kernel.org; ryder.lee at mediatek.com; shayne.chen at mediatek.com;
> sean.wang at mediatek.com; matthias.bgg at gmail.com;
> angelogioacchino.delregno at collabora.com; Lazar, Lijo <Lijo.Lazar at amd.com>
> Subject: Re: [PATCH V3 1/7] drivers/acpi: Add support for Wifi band RF
> mitigations
>
> On 6/16/23 01:57, Evan Quan wrote:
> > From: Mario Limonciello <mario.limonciello at amd.com>
> >
> > Due to electrical and mechanical constraints in certain platform
> > designs there may be likely interference of relatively high-powered
> > harmonics of the (G-)DDR memory clocks with local radio module
> > frequency bands used by Wifi 6/6e/7.
> >
> > To mitigate this, AMD has introduced an ACPI based mechanism that
> > devices can use to notify active use of particular frequencies so that
> > devices can make relative internal adjustments as necessary to avoid
> > this resonance.
> >
> > In order for a device to support this, the expected flow for device
> > driver or subsystems:
> >
> > Drivers/subsystems contributing frequencies:
> >
> > 1) During probe, check `wbrf_supported_producer` to see if WBRF
> supported
> > for the device.
> > 2) If adding frequencies, then call `wbrf_add_exclusion` with the
> > start and end ranges of the frequencies.
> > 3) If removing frequencies, then call `wbrf_remove_exclusion` with
> > start and end ranges of the frequencies.
> >
> > Drivers/subsystems responding to frequencies:
> >
> > 1) During probe, check `wbrf_supported_consumer` to see if WBRF is
> supported
> > for the device.
> > 2) Call the `wbrf_retrieve_exclusions` to retrieve the current
> > exclusions on receiving an ACPI notification for a new frequency
> > change.
> >
> > Signed-off-by: Mario Limonciello <mario.limonciello at amd.com>
> > Co-developed-by: Evan Quan <evan.quan at amd.com>
> > Signed-off-by: Evan Quan <evan.quan at amd.com>
> > --
> > v1->v2:
> > - move those wlan specific implementations to net/mac80211(Mario)
> > ---
> > drivers/acpi/Kconfig | 7 ++
> > drivers/acpi/Makefile | 2 +
> > drivers/acpi/acpi_wbrf.c | 215
> +++++++++++++++++++++++++++++++++++++++
> > include/linux/wbrf.h | 55 ++++++++++
> > 4 files changed, 279 insertions(+)
> > create mode 100644 drivers/acpi/acpi_wbrf.c
> > create mode 100644 include/linux/wbrf.h
> >
> > diff --git a/drivers/acpi/Kconfig b/drivers/acpi/Kconfig index
> > ccbeab9500ec..9ee7c7dcc3e6 100644
> > --- a/drivers/acpi/Kconfig
> > +++ b/drivers/acpi/Kconfig
> > @@ -611,3 +611,10 @@ config X86_PM_TIMER
> >
> > You should nearly always say Y here because many modern
> > systems require this timer.
> > +
> > +config ACPI_WBRF
> > + bool "ACPI Wifi band RF mitigation mechanism"
> > + help
> > + Wifi band RF mitigation mechanism allows multiple drivers from
> > + different domains to notify the frequencies in use so that hardware
> > + can be reconfigured to avoid harmonic conflicts.
> > \ No newline at end of file
>
> There should be a newline at the end of the Kconfig file.
OK, will add that.
Evan
>
> > diff --git a/drivers/acpi/Makefile b/drivers/acpi/Makefile index
> > feb36c0b9446..be173e76aa62 100644
> > --- a/drivers/acpi/Makefile
> > +++ b/drivers/acpi/Makefile
> > @@ -131,3 +131,5 @@ obj-y += dptf/
> > obj-$(CONFIG_ARM64) += arm64/
> >
> > obj-$(CONFIG_ACPI_VIOT) += viot.o
> > +
> > +obj-$(CONFIG_ACPI_WBRF) += acpi_wbrf.o
> > \ No newline at end of file
> > diff --git a/drivers/acpi/acpi_wbrf.c b/drivers/acpi/acpi_wbrf.c new
> > file mode 100644 index 000000000000..8c275998ac29
> > --- /dev/null
> > +++ b/drivers/acpi/acpi_wbrf.c
> > @@ -0,0 +1,215 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * AMD Wifi Band Exclusion Interface
> > + * Copyright (C) 2023 Advanced Micro Devices
> > + *
> > + */
> > +
> > +#include <linux/wbrf.h>
> > +
> > +/* functions */
> > +#define WBRF_RECORD 0x1
> > +#define WBRF_RETRIEVE 0x2
> > +
> > +/* record actions */
> > +#define WBRF_RECORD_ADD 0x0
> > +#define WBRF_RECORD_REMOVE 0x1
> > +
> > +#define WBRF_REVISION 0x1
> > +
> > +static const guid_t wifi_acpi_dsm_guid =
> > + GUID_INIT(0x7b7656cf, 0xdc3d, 0x4c1c,
> > + 0x83, 0xe9, 0x66, 0xe7, 0x21, 0xde, 0x30, 0x70);
> > +
> > +static int wbrf_dsm(struct acpi_device *adev, u8 fn,
> > + union acpi_object *argv4,
> > + union acpi_object **out)
> > +{
> > + union acpi_object *obj;
> > + int rc;
> > +
> > + obj = acpi_evaluate_dsm(adev->handle, &wifi_acpi_dsm_guid,
> > + WBRF_REVISION, fn, argv4);
> > + if (!obj)
> > + return -ENXIO;
> > +
> > + switch (obj->type) {
> > + case ACPI_TYPE_BUFFER:
> > + if (!*out) {
> > + rc = -EINVAL;
> > + break;
> > + }
> > + *out = obj;
> > + return 0;
> > +
> > + case ACPI_TYPE_INTEGER:
> > + rc = obj->integer.value ? -EINVAL : 0;
> > + break;
> > + default:
> > + rc = -EOPNOTSUPP;
> > + }
> > + ACPI_FREE(obj);
> > +
> > + return rc;
> > +}
> > +
> > +static int wbrf_record(struct acpi_device *adev, uint8_t action,
> > + struct wbrf_ranges_in *in)
> > +{
> > + union acpi_object *argv4;
> > + uint32_t num_of_ranges = 0;
> > + uint32_t arg_idx = 0;
> > + uint32_t loop_idx;
> > + int ret;
> > +
> > + if (!in)
> > + return -EINVAL;
> > +
> > + for (loop_idx = 0; loop_idx < ARRAY_SIZE(in->band_list);
> > + loop_idx++)
> > + if (in->band_list[loop_idx].start &&
> > + in->band_list[loop_idx].end)
> > + num_of_ranges++;
> > +
> > + argv4 = kzalloc(sizeof(*argv4) * (2 * num_of_ranges + 2 + 1),
> GFP_KERNEL);
> > + if (!argv4)
> > + return -ENOMEM;
> > +
> > + argv4[arg_idx].package.type = ACPI_TYPE_PACKAGE;
> > + argv4[arg_idx].package.count = 2 + 2 * num_of_ranges;
> > + argv4[arg_idx++].package.elements = &argv4[1];
> > + argv4[arg_idx].integer.type = ACPI_TYPE_INTEGER;
> > + argv4[arg_idx++].integer.value = num_of_ranges;
> > + argv4[arg_idx].integer.type = ACPI_TYPE_INTEGER;
> > + argv4[arg_idx++].integer.value = action;
> > +
> > + for (loop_idx = 0; loop_idx < ARRAY_SIZE(in->band_list);
> > + loop_idx++) {
> > + if (!in->band_list[loop_idx].start ||
> > + !in->band_list[loop_idx].end)
> > + continue;
> > +
> > + argv4[arg_idx].integer.type = ACPI_TYPE_INTEGER;
> > + argv4[arg_idx++].integer.value = in->band_list[loop_idx].start;
> > + argv4[arg_idx].integer.type = ACPI_TYPE_INTEGER;
> > + argv4[arg_idx++].integer.value = in->band_list[loop_idx].end;
> > + }
> > +
> > + ret = wbrf_dsm(adev, WBRF_RECORD, argv4, NULL);
> > +
> > + kfree(argv4);
> > +
> > + return ret;
> > +}
> > +
> > +int wbrf_add_exclusion(struct acpi_device *adev,
> > + struct wbrf_ranges_in *in)
> > +{
> > + return wbrf_record(adev, WBRF_RECORD_ADD, in); }
> > +EXPORT_SYMBOL_GPL(wbrf_add_exclusion);
> > +
> > +int wbrf_remove_exclusion(struct acpi_device *adev,
> > + struct wbrf_ranges_in *in)
> > +{
> > + return wbrf_record(adev, WBRF_RECORD_REMOVE, in); }
> > +EXPORT_SYMBOL_GPL(wbrf_remove_exclusion);
> > +
> > +bool wbrf_supported_producer(struct acpi_device *adev) {
> > + return acpi_check_dsm(adev->handle, &wifi_acpi_dsm_guid,
> > + WBRF_REVISION,
> > + (1ULL << WBRF_RECORD) | (1ULL <<
> WBRF_RETRIEVE)); }
> > +EXPORT_SYMBOL_GPL(wbrf_supported_producer);
> > +
> > +static union acpi_object *
> > +acpi_evaluate_wbrf(acpi_handle handle, u64 rev, u64 func) {
> > + acpi_status ret;
> > + struct acpi_buffer buf = {ACPI_ALLOCATE_BUFFER, NULL};
> > + union acpi_object params[4];
> > + struct acpi_object_list input = {
> > + .count = 4,
> > + .pointer = params,
> > + };
> > +
> > + params[0].type = ACPI_TYPE_INTEGER;
> > + params[0].integer.value = rev;
> > + params[1].type = ACPI_TYPE_INTEGER;
> > + params[1].integer.value = func;
> > + params[2].type = ACPI_TYPE_PACKAGE;
> > + params[2].package.count = 0;
> > + params[2].package.elements = NULL;
> > + params[3].type = ACPI_TYPE_STRING;
> > + params[3].string.length = 0;
> > + params[3].string.pointer= NULL;
> > +
> > + ret = acpi_evaluate_object(handle, "WBRF", &input, &buf);
> > + if (ACPI_SUCCESS(ret))
> > + return (union acpi_object *)buf.pointer;
> > +
> > + if (ret != AE_NOT_FOUND)
> > + acpi_handle_warn(handle,
> > + "failed to evaluate WBRF(0x%x)\n", ret);
> > +
> > + return NULL;
> > +}
> > +
> > +static bool check_acpi_wbrf(acpi_handle handle, u64 rev, u64 funcs) {
> > + int i;
> > + u64 mask = 0;
> > + union acpi_object *obj;
> > +
> > + if (funcs == 0)
> > + return false;
> > +
> > + obj = acpi_evaluate_wbrf(handle, rev, 0);
> > + if (!obj)
> > + return false;
> > +
> > + if (obj->type != ACPI_TYPE_BUFFER)
> > + return false;
> > +
> > + for (i = 0; i < obj->buffer.length && i < 8; i++)
> > + mask |= (((u64)obj->buffer.pointer[i]) << (i * 8));
> > + ACPI_FREE(obj);
> > +
> > + /*
> > + * Bit 0 indicates whether there's support for any functions other than
> > + * function 0.
> > + */
> > + if ((mask & 0x1) && (mask & funcs) == funcs)
> > + return true;
> > +
> > + return false;
> > +}
> > +
> > +bool wbrf_supported_consumer(struct acpi_device *adev) {
> > + return check_acpi_wbrf(adev->handle,
> > + WBRF_REVISION,
> > + 1ULL << WBRF_RETRIEVE);
> > +}
> > +EXPORT_SYMBOL_GPL(wbrf_supported_consumer);
> > +
> > +int wbrf_retrieve_exclusions(struct acpi_device *adev,
> > + struct wbrf_ranges_out *exclusions_out) {
> > + union acpi_object *obj;
> > +
> > + obj = acpi_evaluate_wbrf(adev->handle,
> > + WBRF_REVISION,
> > + WBRF_RETRIEVE);
> > + if (!obj)
> > + return -EINVAL;
> > +
> > + memcpy(exclusions_out, obj->buffer.pointer, obj->buffer.length);
> > +
> > + ACPI_FREE(obj);
> > +
> > + return 0;
> > +}
> > +EXPORT_SYMBOL_GPL(wbrf_retrieve_exclusions);
> > diff --git a/include/linux/wbrf.h b/include/linux/wbrf.h new file mode
> > 100644 index 000000000000..e4c99b69f1d2
> > --- /dev/null
> > +++ b/include/linux/wbrf.h
> > @@ -0,0 +1,55 @@
> > +/* SPDX-License-Identifier: GPL-2.0 */
> > +/*
> > + * AMD Wifi Band Exclusion Interface
> > + * Copyright (C) 2023 Advanced Micro Devices */
> > +
> > +#ifndef _LINUX_WBRF_H
> > +#define _LINUX_WBRF_H
> > +
> > +#include <linux/acpi.h>
> > +
> > +/* Maximum number of wbrf ranges */
> > +#define MAX_NUM_OF_WBRF_RANGES 11 > +
> > +struct exclusion_range {
> > + /* start and end point of the frequency range in Hz */
> > + uint64_t start;
> > + uint64_t end;
> > +};
> > +
> > +struct wbrf_ranges_in {
> > + /* valid entry: `start` and `end` filled with non-zero values */
> > + struct exclusion_range band_list[MAX_NUM_OF_WBRF_RANGES];
> > +};
> > +
> > +struct wbrf_ranges_out {
> > + uint32_t num_of_ranges;
> > + struct exclusion_range band_list[MAX_NUM_OF_WBRF_RANGES];
> > +} __attribute__((packed));
> > +
> > +/**
> > + * APIs needed by drivers/subsystems for contributing frequencies:
> > + * During probe, check `wbrf_supported_producer` to see if WBRF is
> supported.
> > + * If adding frequencies, then call `wbrf_add_exclusion` with the
> > + * start and end points specified for the frequency ranges added.
> > + * If removing frequencies, then call `wbrf_remove_exclusion` with
> > + * start and end points specified for the frequency ranges added.
> > + */
> > +bool wbrf_supported_producer(struct acpi_device *adev); int
> > +wbrf_add_exclusion(struct acpi_device *adev,
> > + struct wbrf_ranges_in *in);
> > +int wbrf_remove_exclusion(struct acpi_device *adev,
> > + struct wbrf_ranges_in *in);
> > +
> > +/**
> > + * APIs needed by drivers/subsystems responding to frequencies:
> > + * During probe, check `wbrf_supported_consumer` to see if WBRF is
> supported.
> > + * When receiving an ACPI notification for some frequencies change,
> > +run
> > + * `wbrf_retrieve_exclusions` to retrieve the latest frequencies ranges.
> > + */
> > +int wbrf_retrieve_exclusions(struct acpi_device *adev,
> > + struct wbrf_ranges_out *out); bool
> > +wbrf_supported_consumer(struct acpi_device *adev);
> > +
> > +#endif /* _LINUX_WBRF_H */
More information about the dri-devel
mailing list