From: Frank Li <Frank.li@nxp.com>
To: Nathan Lynch <nathan.lynch@amd.com>
Cc: Vinod Koul <vkoul@kernel.org>, Frank Li <Frank.Li@kernel.org>,
Bjorn Helgaas <bhelgaas@google.com>,
David Rientjes <rientjes@google.com>,
John.Kariuki@amd.com, Kinsey Ho <kinseyho@google.com>,
Mario Limonciello <mario.limonciello@amd.com>,
PradeepVineshReddy.Kodamati@amd.com,
Shivank Garg <shivankg@amd.com>,
Stephen Bates <Stephen.Bates@amd.com>,
Wei Huang <wei.huang2@amd.com>, Wei Xu <weixugc@google.com>,
dmaengine@vger.kernel.org, linux-kernel@vger.kernel.org,
linux-pci@vger.kernel.org, Jonathan Cameron <jic23@kernel.org>
Subject: Re: [PATCH v2 04/23] dmaengine: sdxi: Feature discovery and initial configuration
Date: Mon, 11 May 2026 17:30:36 -0400 [thread overview]
Message-ID: <agJKfAjCvWOvTLpV@lizhi-Precision-Tower-5810> (raw)
In-Reply-To: <20260511-sdxi-base-v2-4-889cfed17e3f@amd.com>
On Mon, May 11, 2026 at 02:16:16PM -0500, Nathan Lynch wrote:
> Discover via the capability registers the doorbell region stride, the
> maximum supported context ID, the operation groups implemented, and
> limits on buffer and control structure sizes. The driver has the
> option of writing more conservative limits to the ctl2 register, but
> it uses those supplied by the implementation for now.
>
> Introduce device register definitions and associated masks via mmio.h.
>
> Add convenience wrappers which are first used here:
> - sdxi_read64()
> - sdxi_write64()
>
> Report the version of the standard to which the device conforms, e.g.
>
> sdxi 0000:00:03.0: SDXI 1.0 device found
>
> After bus-specific initialization, force the SDXI function to stopped
> state. This is the expected state from reset, but kexec or driver bugs
> can leave a function in other states from which the initialization
> code must be able to recover.
>
> Co-developed-by: Wei Huang <wei.huang2@amd.com>
> Signed-off-by: Wei Huang <wei.huang2@amd.com>
> Signed-off-by: Nathan Lynch <nathan.lynch@amd.com>
> ---
Reviewed-by: Frank Li <Frank.Li@nxp.com>
> drivers/dma/sdxi/device.c | 172 +++++++++++++++++++++++++++++++++++++++++++++-
> drivers/dma/sdxi/mmio.h | 51 ++++++++++++++
> drivers/dma/sdxi/sdxi.h | 19 +++++
> 3 files changed, 241 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/dma/sdxi/device.c b/drivers/dma/sdxi/device.c
> index b718ce04afa0..f9a9944ad892 100644
> --- a/drivers/dma/sdxi/device.c
> +++ b/drivers/dma/sdxi/device.c
> @@ -5,14 +5,180 @@
> * Copyright Advanced Micro Devices, Inc.
> */
>
> +#include <linux/bitfield.h>
> +#include <linux/delay.h>
> #include <linux/device.h>
> +#include <linux/iopoll.h>
> +#include <linux/jiffies.h>
> #include <linux/slab.h>
> +#include <linux/time.h>
>
> +#include "mmio.h"
> #include "sdxi.h"
>
> +enum sdxi_fn_gsv {
> + SDXI_GSV_STOP = 0,
> + SDXI_GSV_INIT = 1,
> + SDXI_GSV_ACTIVE = 2,
> + SDXI_GSV_STOPG_SF = 3,
> + SDXI_GSV_STOPG_HD = 4,
> + SDXI_GSV_ERROR = 5,
> +};
> +
> +static const char *const gsv_strings[] = {
> + [SDXI_GSV_STOP] = "stopped",
> + [SDXI_GSV_INIT] = "initializing",
> + [SDXI_GSV_ACTIVE] = "active",
> + [SDXI_GSV_STOPG_SF] = "soft stopping",
> + [SDXI_GSV_STOPG_HD] = "hard stopping",
> + [SDXI_GSV_ERROR] = "error",
> +};
> +
> +static const char *gsv_str(enum sdxi_fn_gsv gsv)
> +{
> + if ((size_t)gsv < ARRAY_SIZE(gsv_strings))
> + return gsv_strings[(size_t)gsv];
> +
> + WARN_ONCE(1, "unexpected gsv %u\n", gsv);
> +
> + return "unknown";
> +}
> +
> +enum sdxi_fn_gsr {
> + SDXI_GSRV_RESET = 0,
> + SDXI_GSRV_STOP_SF = 1,
> + SDXI_GSRV_STOP_HD = 2,
> + SDXI_GSRV_ACTIVE = 3,
> +};
> +
> +static enum sdxi_fn_gsv sdxi_dev_gsv(const struct sdxi_dev *sdxi)
> +{
> + u64 sts0 = sdxi_read64(sdxi, SDXI_MMIO_STS0);
> + enum sdxi_fn_gsv gsv = FIELD_GET(SDXI_MMIO_STS0_FN_GSV, sts0);
> +
> + switch (gsv) {
> + case SDXI_GSV_STOP ... SDXI_GSV_ERROR:
> + break;
> + default:
> + dev_warn_ratelimited(sdxi->dev, "unknown gsv %u\n", gsv);
> + break;
> + }
> +
> + return gsv;
> +}
> +
> +static const unsigned long gsv_poll_interval_us = USEC_PER_MSEC;
> +static const unsigned long gsv_transition_timeout_us = USEC_PER_SEC;
> +
> +#define sdxi_dev_gsv_poll(sdxi, val, cond) \
> + read_poll_timeout(sdxi_dev_gsv, val, cond, gsv_poll_interval_us, \
> + gsv_transition_timeout_us, false, sdxi)
> +
> +static void sdxi_write_fn_gsr(struct sdxi_dev *sdxi, enum sdxi_fn_gsr cmd)
> +{
> + u64 ctl0 = sdxi_read64(sdxi, SDXI_MMIO_CTL0);
> +
> + FIELD_MODIFY(SDXI_MMIO_CTL0_FN_GSR, &ctl0, cmd);
> + sdxi_write64(sdxi, SDXI_MMIO_CTL0, ctl0);
> +}
> +
> +/* Get the device to the GSV_STOP state. */
> +static int sdxi_dev_stop(struct sdxi_dev *sdxi)
> +{
> + enum sdxi_fn_gsv status = sdxi_dev_gsv(sdxi);
> + int ret;
> +
> + dev_dbg(sdxi->dev, "attempting stop, current state: %s\n",
> + gsv_str(status));
> +
> + switch (status) {
> + case SDXI_GSV_INIT:
> + case SDXI_GSV_ACTIVE:
> + sdxi_write_fn_gsr(sdxi, SDXI_GSRV_STOP_SF);
> + break;
> + case SDXI_GSV_STOPG_SF:
> + sdxi_write_fn_gsr(sdxi, SDXI_GSRV_STOP_HD);
> + break;
> + case SDXI_GSV_STOPG_HD:
> + case SDXI_GSV_ERROR:
> + /*
> + * If hard-stopping, there's nothing to do but wait.
> + * If in error state, the reset is issued below.
> + */
> + break;
> + default:
> + /* Unrecognized state; try a reset. */
> + sdxi_write_fn_gsr(sdxi, SDXI_GSRV_RESET);
> + break;
> + }
> +
> + /* Wait for transition to either stop or error state. */
> + ret = sdxi_dev_gsv_poll(sdxi, status,
> + status == SDXI_GSV_STOP ||
> + status == SDXI_GSV_ERROR);
> +
> + if (ret == 0 && status == SDXI_GSV_ERROR) {
> + sdxi_write_fn_gsr(sdxi, SDXI_GSRV_RESET);
> + ret = sdxi_dev_gsv_poll(sdxi, status, status == SDXI_GSV_STOP);
> + }
> +
> + if (ret) {
> + dev_err(sdxi->dev, "stop timed out, current state: %s\n",
> + gsv_str(status));
> + return ret;
> + }
> +
> + return 0;
> +}
> +
> +/*
> + * See SDXI 1.0 4.1.8 Activation of the SDXI Function by Software.
> + */
> +static int sdxi_fn_activate(struct sdxi_dev *sdxi)
> +{
> + u64 version, cap0, cap1, ctl2;
> + int err;
> +
> + /*
> + * Clear any existing configuration from MMIO_CTL0 and ensure
> + * the function is in GSV_STOP state.
> + */
> + sdxi_write64(sdxi, SDXI_MMIO_CTL0, 0);
> + err = sdxi_dev_stop(sdxi);
> + if (err)
> + return err;
> +
> + version = sdxi_read64(sdxi, SDXI_MMIO_VERSION);
> + dev_info(sdxi->dev, "SDXI %llu.%llu device found\n",
> + FIELD_GET(SDXI_MMIO_VERSION_MAJOR, version),
> + FIELD_GET(SDXI_MMIO_VERSION_MINOR, version));
> +
> + /* Read capabilities and features. */
> + cap0 = sdxi_read64(sdxi, SDXI_MMIO_CAP0);
> + sdxi->db_stride = SZ_4K;
> + sdxi->db_stride *= 1U << FIELD_GET(SDXI_MMIO_CAP0_DB_STRIDE, cap0);
> +
> + cap1 = sdxi_read64(sdxi, SDXI_MMIO_CAP1);
> + sdxi->op_grp_cap = FIELD_GET(SDXI_MMIO_CAP1_OPB_000_CAP, cap1);
> + sdxi->max_cxtid = FIELD_GET(SDXI_MMIO_CAP1_MAX_CXT, cap1);
> +
> + /* Apply our configuration. */
> + ctl2 = FIELD_PREP(SDXI_MMIO_CTL2_MAX_CXT, sdxi->max_cxtid);
> + ctl2 |= FIELD_PREP(SDXI_MMIO_CTL2_MAX_BUFFER,
> + FIELD_GET(SDXI_MMIO_CAP1_MAX_BUFFER, cap1));
> + ctl2 |= FIELD_PREP(SDXI_MMIO_CTL2_MAX_AKEY_SZ,
> + FIELD_GET(SDXI_MMIO_CAP1_MAX_AKEY_SZ, cap1));
> + ctl2 |= FIELD_PREP(SDXI_MMIO_CTL2_OPB_000_AVL,
> + FIELD_GET(SDXI_MMIO_CAP1_OPB_000_CAP, cap1));
> + sdxi_write64(sdxi, SDXI_MMIO_CTL2, ctl2);
> +
> + return 0;
> +}
> +
> int sdxi_register(struct device *dev, const struct sdxi_bus_ops *ops)
> {
> struct sdxi_dev *sdxi;
> + int err;
>
> sdxi = devm_kzalloc(dev, sizeof(*sdxi), GFP_KERNEL);
> if (!sdxi)
> @@ -22,5 +188,9 @@ int sdxi_register(struct device *dev, const struct sdxi_bus_ops *ops)
> sdxi->bus_ops = ops;
> dev_set_drvdata(dev, sdxi);
>
> - return sdxi->bus_ops->init(sdxi);
> + err = sdxi->bus_ops->init(sdxi);
> + if (err)
> + return err;
> +
> + return sdxi_fn_activate(sdxi);
> }
> diff --git a/drivers/dma/sdxi/mmio.h b/drivers/dma/sdxi/mmio.h
> new file mode 100644
> index 000000000000..c9a11c3f2f76
> --- /dev/null
> +++ b/drivers/dma/sdxi/mmio.h
> @@ -0,0 +1,51 @@
> +/* SPDX-License-Identifier: GPL-2.0-only */
> +
> +/*
> + * SDXI MMIO register offsets and layouts.
> + *
> + * Copyright Advanced Micro Devices, Inc.
> + */
> +
> +#ifndef DMA_SDXI_MMIO_H
> +#define DMA_SDXI_MMIO_H
> +
> +#include <linux/bits.h>
> +
> +enum sdxi_reg {
> + /* SDXI 1.0 9.1 General Control and Status Registers */
> + SDXI_MMIO_CTL0 = 0x00000,
> + SDXI_MMIO_CTL2 = 0x00010,
> + SDXI_MMIO_STS0 = 0x00100,
> + SDXI_MMIO_CAP0 = 0x00200,
> + SDXI_MMIO_CAP1 = 0x00208,
> + SDXI_MMIO_VERSION = 0x00210,
> +};
> +
> +/* SDXI 1.0 Table 9-2: MMIO_CTL0 */
> +#define SDXI_MMIO_CTL0_FN_GSR GENMASK_ULL(1, 0)
> +
> +/* SDXI 1.0 Table 9-4: MMIO_CTL2 */
> +#define SDXI_MMIO_CTL2_MAX_BUFFER GENMASK_ULL(3, 0)
> +#define SDXI_MMIO_CTL2_MAX_AKEY_SZ GENMASK_ULL(15, 12)
> +#define SDXI_MMIO_CTL2_MAX_CXT GENMASK_ULL(31, 16)
> +#define SDXI_MMIO_CTL2_OPB_000_AVL GENMASK_ULL(63, 32)
> +
> +/* SDXI 1.0 Table 9-5: MMIO_STS0 */
> +#define SDXI_MMIO_STS0_FN_GSV GENMASK_ULL(2, 0)
> +
> +/* SDXI 1.0 Table 9-6: MMIO_CAP0 */
> +#define SDXI_MMIO_CAP0_SFUNC GENMASK_ULL(15, 0)
> +#define SDXI_MMIO_CAP0_DB_STRIDE GENMASK_ULL(22, 20)
> +#define SDXI_MMIO_CAP0_MAX_DS_RING_SZ GENMASK_ULL(28, 24)
> +
> +/* SDXI 1.0 Table 9-7: MMIO_CAP1 */
> +#define SDXI_MMIO_CAP1_MAX_BUFFER GENMASK_ULL(3, 0)
> +#define SDXI_MMIO_CAP1_MAX_AKEY_SZ GENMASK_ULL(15, 12)
> +#define SDXI_MMIO_CAP1_MAX_CXT GENMASK_ULL(31, 16)
> +#define SDXI_MMIO_CAP1_OPB_000_CAP GENMASK_ULL(63, 32)
> +
> +/* SDXI 1.0 Table 9-8: MMIO_VERSION */
> +#define SDXI_MMIO_VERSION_MINOR GENMASK_ULL(7, 0)
> +#define SDXI_MMIO_VERSION_MAJOR GENMASK_ULL(23, 16)
> +
> +#endif /* DMA_SDXI_MMIO_H */
> diff --git a/drivers/dma/sdxi/sdxi.h b/drivers/dma/sdxi/sdxi.h
> index d4c61ca2f875..84b87066f438 100644
> --- a/drivers/dma/sdxi/sdxi.h
> +++ b/drivers/dma/sdxi/sdxi.h
> @@ -9,8 +9,12 @@
> #define DMA_SDXI_H
>
> #include <linux/compiler_types.h>
> +#include <linux/dev_printk.h>
> +#include <linux/io-64-nonatomic-lo-hi.h>
> #include <linux/types.h>
>
> +#include "mmio.h"
> +
> struct sdxi_dev;
>
> /**
> @@ -30,9 +34,24 @@ struct sdxi_dev {
> void __iomem *ctrl_regs; /* virt addr of ctrl registers */
> void __iomem *dbs; /* virt addr of doorbells */
>
> + /* hardware capabilities (from cap0 & cap1) */
> + u32 db_stride; /* doorbell stride in bytes */
> + u16 max_cxtid; /* Maximum context ID allowed. */
> + u32 op_grp_cap; /* supported operation group cap */
> +
> const struct sdxi_bus_ops *bus_ops;
> };
>
> int sdxi_register(struct device *dev, const struct sdxi_bus_ops *ops);
>
> +static inline u64 sdxi_read64(const struct sdxi_dev *sdxi, enum sdxi_reg reg)
> +{
> + return ioread64(sdxi->ctrl_regs + reg);
> +}
> +
> +static inline void sdxi_write64(struct sdxi_dev *sdxi, enum sdxi_reg reg, u64 val)
> +{
> + iowrite64(val, sdxi->ctrl_regs + reg);
> +}
> +
> #endif /* DMA_SDXI_H */
>
> --
> 2.54.0
>
next prev parent reply other threads:[~2026-05-11 21:30 UTC|newest]
Thread overview: 29+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-05-11 19:16 [PATCH v2 00/23] dmaengine: Smart Data Accelerator Interface (SDXI) basic support Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 01/23] PCI: Add SNIA SDXI accelerator sub-class Nathan Lynch via B4 Relay
2026-05-11 20:48 ` Frank Li
2026-05-11 19:16 ` [PATCH v2 02/23] MAINTAINERS: Add entry for SDXI driver Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 03/23] dmaengine: sdxi: Add PCI initialization Nathan Lynch via B4 Relay
2026-05-11 21:22 ` Frank Li
2026-05-11 19:16 ` [PATCH v2 04/23] dmaengine: sdxi: Feature discovery and initial configuration Nathan Lynch via B4 Relay
2026-05-11 21:30 ` Frank Li [this message]
2026-05-11 19:16 ` [PATCH v2 05/23] dmaengine: sdxi: Configure context tables Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 06/23] dmaengine: sdxi: Allocate DMA pools Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 07/23] dmaengine: sdxi: Allocate administrative context Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 08/23] dmaengine: sdxi: Install " Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 09/23] dmaengine: sdxi: Start functions on probe, stop on remove Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 10/23] dmaengine: sdxi: Complete administrative context jump start Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 11/23] dmaengine: sdxi: Add client context alloc and release APIs Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 12/23] dmaengine: sdxi: Add descriptor ring management Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 13/23] dmaengine: sdxi: Add unit tests for descriptor ring reservations Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 14/23] dmaengine: sdxi: Attach descriptor ring state to contexts Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 15/23] dmaengine: sdxi: Per-context access key (AKey) table entry allocator Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 16/23] dmaengine: sdxi: Generic descriptor manipulation helpers Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 17/23] dmaengine: sdxi: Add completion status block API Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 18/23] dmaengine: sdxi: Encode context start, stop, and sync descriptors Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 19/23] dmaengine: sdxi: Provide context start and stop APIs Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 20/23] dmaengine: sdxi: Encode nop, copy, and interrupt descriptors Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 21/23] dmaengine: sdxi: Add unit tests for descriptor encoding Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 22/23] dmaengine: sdxi: MSI/MSI-X vector allocation and mapping Nathan Lynch via B4 Relay
2026-05-11 19:16 ` [PATCH v2 23/23] dmaengine: sdxi: Add DMA engine provider Nathan Lynch via B4 Relay
2026-05-11 20:47 ` Frank Li
2026-05-11 22:28 ` Lynch, Nathan
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=agJKfAjCvWOvTLpV@lizhi-Precision-Tower-5810 \
--to=frank.li@nxp.com \
--cc=Frank.Li@kernel.org \
--cc=John.Kariuki@amd.com \
--cc=PradeepVineshReddy.Kodamati@amd.com \
--cc=Stephen.Bates@amd.com \
--cc=bhelgaas@google.com \
--cc=dmaengine@vger.kernel.org \
--cc=jic23@kernel.org \
--cc=kinseyho@google.com \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-pci@vger.kernel.org \
--cc=mario.limonciello@amd.com \
--cc=nathan.lynch@amd.com \
--cc=rientjes@google.com \
--cc=shivankg@amd.com \
--cc=vkoul@kernel.org \
--cc=wei.huang2@amd.com \
--cc=weixugc@google.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox