Skip site navigation (1)Skip section navigation (2)
Date:      Tue, 11 Oct 2016 02:08:10 +0000 (UTC)
From:      Sepherosa Ziehau <sephe@FreeBSD.org>
To:        src-committers@freebsd.org, svn-src-all@freebsd.org, svn-src-stable@freebsd.org, svn-src-stable-11@freebsd.org
Subject:   svn commit: r307010 - stable/11/sys/compat/linuxkpi/common/include/linux
Message-ID:  <201610110208.u9B28ASC085916@repo.freebsd.org>

next in thread | raw e-mail | index | archive | help
Author: sephe
Date: Tue Oct 11 02:08:09 2016
New Revision: 307010
URL: https://svnweb.freebsd.org/changeset/base/307010

Log:
  MFC 306480
  
      linuxkpi: Fix PCI BAR lazy allocation support.
  
      FreeBSD supports lazy allocation of PCI BAR, that is, when a device
      driver's attach method is invoked, even if the device's PCI BAR
      address wasn't initialized, the invocation of bus_alloc_resource_any()
      (the call chain: pci_alloc_resource() -> pci_alloc_multi_resource() ->
      pci_reserve_map() -> pci_write_bar()) would allocate a proper address
      for the PCI BAR and write this 'lazy allocated' address into the PCI
      BAR.
  
      This model works fine for native FreeBSD device drivers, but _not_ for
      device drivers shared with Linux (e.g. dev/mlx5/mlx5_core/mlx5_main.c
      and ofed/drivers/net/mlx4/main.c.  Both of them use
      pci_request_regions(), which doesn't work properly with the PCI BAR
      lazy allocation, because pci_resource_type() -> _pci_get_rle() always
      returns NULL, so pci_request_regions() doesn't have the opportunity to
      invoke bus_alloc_resource_any().  We now use pci_find_bar() in
      pci_resource_type(), which is able to locate all available PCI BARs
      even if some of them will be lazy allocated.
  
      Submitted by:   Dexuan Cui <decui microsoft com>
      Reviewed by:    hps
      Sponsored by:   Microsoft
      Differential Revision:  https://reviews.freebsd.org/D8071

Modified:
  stable/11/sys/compat/linuxkpi/common/include/linux/pci.h
Directory Properties:
  stable/11/   (props changed)

Modified: stable/11/sys/compat/linuxkpi/common/include/linux/pci.h
==============================================================================
--- stable/11/sys/compat/linuxkpi/common/include/linux/pci.h	Mon Oct 10 23:35:55 2016	(r307009)
+++ stable/11/sys/compat/linuxkpi/common/include/linux/pci.h	Tue Oct 11 02:08:09 2016	(r307010)
@@ -233,11 +233,16 @@ pci_resource_len(struct pci_dev *pdev, i
 static inline int
 pci_resource_type(struct pci_dev *pdev, int bar)
 {
-	struct resource_list_entry *rle;
+	struct pci_map *pm;
 
-	if ((rle = _pci_get_bar(pdev, bar)) == NULL)
+	pm = pci_find_bar(pdev->dev.bsddev, PCIR_BAR(bar));
+	if (!pm)
 		return (-1);
-	return (rle->type);
+
+	if (PCI_BAR_IO(pm->pm_value))
+		return (SYS_RES_IOPORT);
+	else
+		return (SYS_RES_MEMORY);
 }
 
 /*



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?201610110208.u9B28ASC085916>