public inbox for linux-pci@vger.kernel.org
 help / color / mirror / Atom feed
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
> 

-- 
மணிவண்ணன் சதாசிவம்

  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