From: Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
To: Niklas Cassel <cassel@kernel.org>
Cc: "Krzysztof Wilczyński" <kw@linux.com>,
"Kishon Vijay Abraham I" <kishon@kernel.org>,
"Arnd Bergmann" <arnd@arndb.de>,
"Greg Kroah-Hartman" <gregkh@linuxfoundation.org>,
"Bjorn Helgaas" <bhelgaas@google.com>,
"Damien Le Moal" <dlemoal@kernel.org>,
linux-pci@vger.kernel.org
Subject: Re: [PATCH] misc: pci_endpoint_test: Add consecutive BAR test
Date: Sun, 24 Nov 2024 08:09:22 +0530 [thread overview]
Message-ID: <20241124023922.dpdjublabfnfxrd4@thinkpad> (raw)
In-Reply-To: <20241116032045.2574168-2-cassel@kernel.org>
On Sat, Nov 16, 2024 at 04:20:45AM +0100, Niklas Cassel wrote:
> Add a more advanced BAR test that writes all BARs in one go, and then reads
> them back and verifies that the value matches the BAR number bitwise OR:ed
> with offset, this allows us to verify:
> -The BAR number was what we intended to read.
> -The offset was what we intended to read.
>
> This allows us to detect potential address translation issues on the EP.
>
> Reading back the BAR directly after writing will not allow us to detect the
> case where inbound address translation on the endpoint incorrectly causes
> multiple BARs to be redirected to the same memory region (within the EP).
>
> Signed-off-by: Niklas Cassel <cassel@kernel.org>
Reviewed-by: Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
- Mani
> ---
> drivers/misc/pci_endpoint_test.c | 88 ++++++++++++++++++++++++++++++++
> include/uapi/linux/pcitest.h | 1 +
> tools/pci/pcitest.c | 16 +++++-
> tools/pci/pcitest.sh | 1 +
> 4 files changed, 105 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/misc/pci_endpoint_test.c b/drivers/misc/pci_endpoint_test.c
> index 3aaaf47fa4ee2..34cb54aef3d8b 100644
> --- a/drivers/misc/pci_endpoint_test.c
> +++ b/drivers/misc/pci_endpoint_test.c
> @@ -322,6 +322,91 @@ static bool pci_endpoint_test_bar(struct pci_endpoint_test *test,
> return true;
> }
>
> +static u32 bar_test_pattern_with_offset(enum pci_barno barno, int offset)
> +{
> + u32 val;
> +
> + /* Keep the BAR pattern in the top byte. */
> + val = bar_test_pattern[barno] & 0xff000000;
> + /* Store the (partial) offset in the remaining bytes. */
> + val |= offset & 0x00ffffff;
> +
> + return val;
> +}
> +
> +static bool pci_endpoint_test_bars_write_bar(struct pci_endpoint_test *test,
> + enum pci_barno barno)
> +{
> + struct pci_dev *pdev = test->pdev;
> + int j, size;
> +
> + size = pci_resource_len(pdev, barno);
> +
> + if (barno == test->test_reg_bar)
> + size = 0x4;
> +
> + for (j = 0; j < size; j += 4)
> + writel_relaxed(bar_test_pattern_with_offset(barno, j),
> + test->bar[barno] + j);
> +
> + return true;
> +}
> +
> +static bool pci_endpoint_test_bars_read_bar(struct pci_endpoint_test *test,
> + enum pci_barno barno)
> +{
> + struct pci_dev *pdev = test->pdev;
> + struct device *dev = &pdev->dev;
> + int j, size;
> + u32 val;
> +
> + size = pci_resource_len(pdev, barno);
> +
> + if (barno == test->test_reg_bar)
> + size = 0x4;
> +
> + for (j = 0; j < size; j += 4) {
> + u32 expected = bar_test_pattern_with_offset(barno, j);
> +
> + val = readl_relaxed(test->bar[barno] + j);
> + if (val != expected) {
> + dev_err(dev,
> + "BAR%d incorrect data at offset: %#x, got: %#x expected: %#x\n",
> + barno, j, val, expected);
> + return false;
> + }
> + }
> +
> + return true;
> +}
> +
> +static bool pci_endpoint_test_bars(struct pci_endpoint_test *test)
> +{
> + enum pci_barno bar;
> + bool ret;
> +
> + /* Write all BARs in order (without reading). */
> + for (bar = 0; bar < PCI_STD_NUM_BARS; bar++)
> + if (test->bar[bar])
> + pci_endpoint_test_bars_write_bar(test, bar);
> +
> + /*
> + * Read all BARs in order (without writing).
> + * If there is an address translation issue on the EP, writing one BAR
> + * might have overwritten another BAR. Ensure that this is not the case.
> + * (Reading back the BAR directly after writing can not detect this.)
> + */
> + for (bar = 0; bar < PCI_STD_NUM_BARS; bar++) {
> + if (test->bar[bar]) {
> + ret = pci_endpoint_test_bars_read_bar(test, bar);
> + if (!ret)
> + return ret;
> + }
> + }
> +
> + return true;
> +}
> +
> static bool pci_endpoint_test_intx_irq(struct pci_endpoint_test *test)
> {
> u32 val;
> @@ -768,6 +853,9 @@ static long pci_endpoint_test_ioctl(struct file *file, unsigned int cmd,
> goto ret;
> ret = pci_endpoint_test_bar(test, bar);
> break;
> + case PCITEST_BARS:
> + ret = pci_endpoint_test_bars(test);
> + break;
> case PCITEST_INTX_IRQ:
> ret = pci_endpoint_test_intx_irq(test);
> break;
> diff --git a/include/uapi/linux/pcitest.h b/include/uapi/linux/pcitest.h
> index 94b46b043b536..acd261f498666 100644
> --- a/include/uapi/linux/pcitest.h
> +++ b/include/uapi/linux/pcitest.h
> @@ -20,6 +20,7 @@
> #define PCITEST_MSIX _IOW('P', 0x7, int)
> #define PCITEST_SET_IRQTYPE _IOW('P', 0x8, int)
> #define PCITEST_GET_IRQTYPE _IO('P', 0x9)
> +#define PCITEST_BARS _IO('P', 0xa)
> #define PCITEST_CLEAR_IRQ _IO('P', 0x10)
>
> #define PCITEST_FLAGS_USE_DMA 0x00000001
> diff --git a/tools/pci/pcitest.c b/tools/pci/pcitest.c
> index 470258009ddc2..2ce56ea56202c 100644
> --- a/tools/pci/pcitest.c
> +++ b/tools/pci/pcitest.c
> @@ -22,6 +22,7 @@ static char *irq[] = { "LEGACY", "MSI", "MSI-X" };
> struct pci_test {
> char *device;
> char barnum;
> + bool consecutive_bar_test;
> bool legacyirq;
> unsigned int msinum;
> unsigned int msixnum;
> @@ -57,6 +58,15 @@ static int run_test(struct pci_test *test)
> fprintf(stdout, "%s\n", result[ret]);
> }
>
> + if (test->consecutive_bar_test) {
> + ret = ioctl(fd, PCITEST_BARS);
> + fprintf(stdout, "Consecutive BAR test:\t\t");
> + if (ret < 0)
> + fprintf(stdout, "TEST FAILED\n");
> + else
> + fprintf(stdout, "%s\n", result[ret]);
> + }
> +
> if (test->set_irqtype) {
> ret = ioctl(fd, PCITEST_SET_IRQTYPE, test->irqtype);
> fprintf(stdout, "SET IRQ TYPE TO %s:\t\t", irq[test->irqtype]);
> @@ -172,7 +182,7 @@ int main(int argc, char **argv)
> /* set default endpoint device */
> test->device = "/dev/pci-endpoint-test.0";
>
> - while ((c = getopt(argc, argv, "D:b:m:x:i:deIlhrwcs:")) != EOF)
> + while ((c = getopt(argc, argv, "D:b:Cm:x:i:deIlhrwcs:")) != EOF)
> switch (c) {
> case 'D':
> test->device = optarg;
> @@ -182,6 +192,9 @@ int main(int argc, char **argv)
> if (test->barnum < 0 || test->barnum > 5)
> goto usage;
> continue;
> + case 'C':
> + test->consecutive_bar_test = true;
> + continue;
> case 'l':
> test->legacyirq = true;
> continue;
> @@ -230,6 +243,7 @@ int main(int argc, char **argv)
> "Options:\n"
> "\t-D <dev> PCI endpoint test device {default: /dev/pci-endpoint-test.0}\n"
> "\t-b <bar num> BAR test (bar number between 0..5)\n"
> + "\t-C Consecutive BAR test\n"
> "\t-m <msi num> MSI test (msi number between 1..32)\n"
> "\t-x <msix num> \tMSI-X test (msix number between 1..2048)\n"
> "\t-i <irq type> \tSet IRQ type (0 - Legacy, 1 - MSI, 2 - MSI-X)\n"
> diff --git a/tools/pci/pcitest.sh b/tools/pci/pcitest.sh
> index 75ed48ff29900..770f4d6df34b6 100644
> --- a/tools/pci/pcitest.sh
> +++ b/tools/pci/pcitest.sh
> @@ -11,6 +11,7 @@ do
> pcitest -b $bar
> bar=`expr $bar + 1`
> done
> +pcitest -C
> echo
>
> echo "Interrupt tests"
> --
> 2.47.0
>
--
மணிவண்ணன் சதாசிவம்
next prev parent reply other threads:[~2024-11-24 2:39 UTC|newest]
Thread overview: 4+ messages / expand[flat|nested] mbox.gz Atom feed top
2024-11-16 3:20 [PATCH] misc: pci_endpoint_test: Add consecutive BAR test Niklas Cassel
2024-11-24 2:39 ` Manivannan Sadhasivam [this message]
2024-12-12 8:02 ` Niklas Cassel
2024-12-22 20:55 ` Krzysztof Wilczyński
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=20241124023922.dpdjublabfnfxrd4@thinkpad \
--to=manivannan.sadhasivam@linaro.org \
--cc=arnd@arndb.de \
--cc=bhelgaas@google.com \
--cc=cassel@kernel.org \
--cc=dlemoal@kernel.org \
--cc=gregkh@linuxfoundation.org \
--cc=kishon@kernel.org \
--cc=kw@linux.com \
--cc=linux-pci@vger.kernel.org \
/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