* [PATCH RESEND 0/2]: ixp4xx: Fix 3.7 regression for IXP4xx ethernet driver
@ 2014-03-17 9:54 Simon Kågström
2014-03-17 9:56 ` [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation Simon Kågström
2014-03-17 9:57 ` [PATCH RESEND 2/2] ixp4xx_eth: Setup coherent_dma_mask Simon Kågström
0 siblings, 2 replies; 8+ messages in thread
From: Simon Kågström @ 2014-03-17 9:54 UTC (permalink / raw)
To: linux-arm-kernel
Hi!
These two patches fixes the regression introduced in 3.7 by commit
1a4901177574083c35fafc24c4d151c2a7c7647c,
ixp4xx_eth: avoid calling dma_pool_create() with NULL dev
The patch above is not incorrect itself, but causes the driver to fail
with
[ 33.055473] net eth1: coherent DMA mask is unset
[ 33.055523] net eth1: coherent allocation too big (requested 0x1000 mask 0x0)
The two patches fix this by generalizing and correcting the ixp4xx
dma_set_coherent_mask(), which earlier only dealt with the PCI case.
The second patch simply sets up the mask for the ixp4xx ethernet device.
The patches have been verified on a board with 256MiB memory, one
ipx4xx_eth device and a e100 PCI device, running Linus' tip.
This issue has been discussed before:
https://lkml.org/lkml/2014/1/2/46
http://lists.infradead.org/pipermail/linux-arm-kernel/2013-February/148811.html
This is a resend of the patch series. There was some discussion about it here:
http://www.spinics.net/lists/arm-kernel/msg313082.html
and if I understand Russels comments (as per the documentation),
drivers should setup the coherent DMA mask even if it happens to be the
platform/CPU default.
// Simon
^ permalink raw reply [flat|nested] 8+ messages in thread
* [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation
2014-03-17 9:54 [PATCH RESEND 0/2]: ixp4xx: Fix 3.7 regression for IXP4xx ethernet driver Simon Kågström
@ 2014-03-17 9:56 ` Simon Kågström
2014-03-17 11:29 ` Arnd Bergmann
2014-03-18 7:41 ` Krzysztof Hałasa
2014-03-17 9:57 ` [PATCH RESEND 2/2] ixp4xx_eth: Setup coherent_dma_mask Simon Kågström
1 sibling, 2 replies; 8+ messages in thread
From: Simon Kågström @ 2014-03-17 9:56 UTC (permalink / raw)
To: linux-arm-kernel
Non-PCI devices can use the entire 32-bit range, PCI dittos are
limited to the first 64MiB.
Also actually setup coherent_dma_mask.
Signed-off-by: Simon Kagstrom <simon.kagstrom@netinsight.net>
---
arch/arm/mach-ixp4xx/common-pci.c | 8 --------
arch/arm/mach-ixp4xx/common.c | 11 +++++++++++
2 files changed, 11 insertions(+), 8 deletions(-)
diff --git a/arch/arm/mach-ixp4xx/common-pci.c b/arch/arm/mach-ixp4xx/common-pci.c
index 200970d..b02c764 100644
--- a/arch/arm/mach-ixp4xx/common-pci.c
+++ b/arch/arm/mach-ixp4xx/common-pci.c
@@ -481,14 +481,6 @@ int ixp4xx_setup(int nr, struct pci_sys_data *sys)
return 1;
}
-int dma_set_coherent_mask(struct device *dev, u64 mask)
-{
- if (mask >= SZ_64M - 1)
- return 0;
-
- return -EIO;
-}
-
EXPORT_SYMBOL(ixp4xx_pci_read);
EXPORT_SYMBOL(ixp4xx_pci_write);
EXPORT_SYMBOL(dma_set_coherent_mask);
diff --git a/arch/arm/mach-ixp4xx/common.c b/arch/arm/mach-ixp4xx/common.c
index 6d68aed..b221c78 100644
--- a/arch/arm/mach-ixp4xx/common.c
+++ b/arch/arm/mach-ixp4xx/common.c
@@ -31,6 +31,7 @@
#include <linux/gpio.h>
#include <linux/cpu.h>
#include <linux/sched_clock.h>
+#include <linux/pci.h>
#include <mach/udc.h>
#include <mach/hardware.h>
@@ -578,6 +579,16 @@ void ixp4xx_restart(enum reboot_mode mode, const char *cmd)
}
}
+int dma_set_coherent_mask(struct device *dev, u64 mask)
+{
+ if (dev_is_pci(dev) && mask >= SZ_64M)
+ return -EIO;
+
+ dev->coherent_dma_mask = mask;
+
+ return 0;
+}
+
#ifdef CONFIG_IXP4XX_INDIRECT_PCI
/*
* In the case of using indirect PCI, we simply return the actual PCI
--
1.7.9.6
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [PATCH RESEND 2/2] ixp4xx_eth: Setup coherent_dma_mask
2014-03-17 9:54 [PATCH RESEND 0/2]: ixp4xx: Fix 3.7 regression for IXP4xx ethernet driver Simon Kågström
2014-03-17 9:56 ` [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation Simon Kågström
@ 2014-03-17 9:57 ` Simon Kågström
1 sibling, 0 replies; 8+ messages in thread
From: Simon Kågström @ 2014-03-17 9:57 UTC (permalink / raw)
To: linux-arm-kernel
This driver has been broken since commit 1a4901177574083c35fafc24c4d151c2a7c7647c
ixp4xx_eth: avoid calling dma_pool_create() with NULL dev
in 3.7 and would fail with
[ 33.055473] net eth1: coherent DMA mask is unset
[ 33.055523] net eth1: coherent allocation too big (requested 0x1000 mask 0x0)
Fix by setting up the DMA mask to the entire 32-bit range.
Signed-off-by: Simon Kagstrom <simon.kagstrom@netinsight.net>
---
drivers/net/ethernet/xscale/ixp4xx_eth.c | 4 ++++
1 file changed, 4 insertions(+)
diff --git a/drivers/net/ethernet/xscale/ixp4xx_eth.c b/drivers/net/ethernet/xscale/ixp4xx_eth.c
index 25283f1..e540e51 100644
--- a/drivers/net/ethernet/xscale/ixp4xx_eth.c
+++ b/drivers/net/ethernet/xscale/ixp4xx_eth.c
@@ -1426,6 +1426,10 @@ static int eth_init_one(struct platform_device *pdev)
port->netdev = dev;
port->id = pdev->id;
+ err = dma_set_coherent_mask(&dev->dev, DMA_BIT_MASK(32));
+ if (err < 0)
+ goto err_free;
+
switch (port->id) {
case IXP4XX_ETH_NPEA:
port->regs = (struct eth_regs __iomem *)IXP4XX_EthA_BASE_VIRT;
--
1.7.9.6
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation
2014-03-17 9:56 ` [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation Simon Kågström
@ 2014-03-17 11:29 ` Arnd Bergmann
2014-03-17 11:54 ` Simon Kågström
2014-03-18 7:41 ` Krzysztof Hałasa
1 sibling, 1 reply; 8+ messages in thread
From: Arnd Bergmann @ 2014-03-17 11:29 UTC (permalink / raw)
To: linux-arm-kernel
On Monday 17 March 2014, Simon K?gstr?m wrote:
> diff --git a/arch/arm/mach-ixp4xx/common-pci.c b/arch/arm/mach-ixp4xx/common-pci.c
> index 200970d..b02c764 100644
> --- a/arch/arm/mach-ixp4xx/common-pci.c
> +++ b/arch/arm/mach-ixp4xx/common-pci.c
> @@ -481,14 +481,6 @@ int ixp4xx_setup(int nr, struct pci_sys_data *sys)
> return 1;
> }
>
> -int dma_set_coherent_mask(struct device *dev, u64 mask)
> -{
> - if (mask >= SZ_64M - 1)
> - return 0;
> -
> - return -EIO;
> -}
> -
> EXPORT_SYMBOL(ixp4xx_pci_read);
> EXPORT_SYMBOL(ixp4xx_pci_write);
> EXPORT_SYMBOL(dma_set_coherent_mask);
When you move this function, you should really move the
'EXPORT_SYMBOL' along with it.
Arnd
^ permalink raw reply [flat|nested] 8+ messages in thread
* [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation
2014-03-17 11:29 ` Arnd Bergmann
@ 2014-03-17 11:54 ` Simon Kågström
0 siblings, 0 replies; 8+ messages in thread
From: Simon Kågström @ 2014-03-17 11:54 UTC (permalink / raw)
To: linux-arm-kernel
On Mon, 17 Mar 2014 12:29:38 +0100
Arnd Bergmann <arnd@arndb.de> wrote:
> >
> > -int dma_set_coherent_mask(struct device *dev, u64 mask)
> > -{
> > - if (mask >= SZ_64M - 1)
> > - return 0;
> > -
> > - return -EIO;
> > -}
> > -
> > EXPORT_SYMBOL(ixp4xx_pci_read);
> > EXPORT_SYMBOL(ixp4xx_pci_write);
> > EXPORT_SYMBOL(dma_set_coherent_mask);
>
> When you move this function, you should really move the
> 'EXPORT_SYMBOL' along with it.
Yes, you are right. Will fix and resend the patches!
// Simon
^ permalink raw reply [flat|nested] 8+ messages in thread
* [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation
2014-03-17 9:56 ` [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation Simon Kågström
2014-03-17 11:29 ` Arnd Bergmann
@ 2014-03-18 7:41 ` Krzysztof Hałasa
2014-03-18 8:09 ` Simon Kågström
1 sibling, 1 reply; 8+ messages in thread
From: Krzysztof Hałasa @ 2014-03-18 7:41 UTC (permalink / raw)
To: linux-arm-kernel
Hello,
This patch is incorrect, the functionality is reversed:
Simon K?gstr?m <simon.kagstrom@netinsight.net> writes:
> +++ b/arch/arm/mach-ixp4xx/common-pci.c
> @@ -481,14 +481,6 @@ int ixp4xx_setup(int nr, struct pci_sys_data *sys)
> return 1;
> }
>
> -int dma_set_coherent_mask(struct device *dev, u64 mask)
> -{
> - if (mask >= SZ_64M - 1)
> - return 0;
> -
> - return -EIO;
Originally, coherent masks equal to 64 MiB or wider are accepted.
> --- a/arch/arm/mach-ixp4xx/common.c
> +++ b/arch/arm/mach-ixp4xx/common.c
> +int dma_set_coherent_mask(struct device *dev, u64 mask)
> +{
> + if (dev_is_pci(dev) && mask >= SZ_64M)
> + return -EIO;
Now, you reject them. However, with coherent allocations, big masks
(usually 32-bit) are ok. They have to be internally limited to 64 MiB,
though.
It's a narrow mask (e.g. 16 MiB) which can't be used, because we only
have two memory pools (4 GiB and 64 MiB), and we simply have no means to
request e.g. memory from 16 MiB range. Doesn't matter in practice,
otherwise we would have additional memory pool.
This differs a bit from the streaming mask. For streaming, we still
accept basically everything (because we obviously want to support 32-bit
capable devices) and we limit the masks to 64 MiB internally, but we
then have to use dmabounce or whatever (perhaps swiotlb could be an
option?).
Also, we can limit some heavy duty allocations to 64 MiB (e.g. skbuffs
on routers, with a custom patch). We don't do DAC, IXP4xx address space
is 32-bit anyway. BTW possibility to use some system-wide DMA mask while
creating skbuffs, disk buffers etc. could be an improvement (drivers
would still need to check the addresses with dmabounce).
Believe it or not, the correct patch is the one I'm attaching. One can
add extra *set_masks* (e.g. the new call setting both streaming and
coherent masks) in the drivers, but essentially it must do what this one
does. Also, converting the mask (in the dev struct) from a pointer to a
simple value would IMHO make sense, too.
Also, comparing bit masks with ">=" etc. doesn't look very valid to me.
What if some device wants a mask which isn't a power of 2 - 1?
I'm unable to look at this ATM but I will update the patch to the new
kernel, perhaps soon.
IXP4xx: Fix DMA masks.
Now, devices will have 32-bit default DMA masks (0xFFFFFFFF) as per DMA
API, and the masks will actually work.
Signed-off-by: Krzysztof Ha?asa <khalasa@piap.pl>
diff --git a/arch/arm/mach-ixp4xx/common-pci.c b/arch/arm/mach-ixp4xx/common-pci.c
index 6d6bde3..cefb80b 100644
--- a/arch/arm/mach-ixp4xx/common-pci.c
+++ b/arch/arm/mach-ixp4xx/common-pci.c
@@ -316,32 +316,6 @@ static int abort_handler(unsigned long addr, unsigned int fsr, struct pt_regs *r
}
-static int ixp4xx_needs_bounce(struct device *dev, dma_addr_t dma_addr, size_t size)
-{
- return (dma_addr + size) >= SZ_64M;
-}
-
-/*
- * Setup DMA mask to 64MB on PCI devices. Ignore all other devices.
- */
-static int ixp4xx_pci_platform_notify(struct device *dev)
-{
- if(dev->bus == &pci_bus_type) {
- *dev->dma_mask = SZ_64M - 1;
- dev->coherent_dma_mask = SZ_64M - 1;
- dmabounce_register_dev(dev, 2048, 4096, ixp4xx_needs_bounce);
- }
- return 0;
-}
-
-static int ixp4xx_pci_platform_notify_remove(struct device *dev)
-{
- if(dev->bus == &pci_bus_type) {
- dmabounce_unregister_dev(dev);
- }
- return 0;
-}
-
void __init ixp4xx_pci_preinit(void)
{
unsigned long cpuid = read_cpuid_id();
@@ -475,20 +449,8 @@ int ixp4xx_setup(int nr, struct pci_sys_data *sys)
pci_add_resource_offset(&sys->resources, &res[0], sys->io_offset);
pci_add_resource_offset(&sys->resources, &res[1], sys->mem_offset);
- platform_notify = ixp4xx_pci_platform_notify;
- platform_notify_remove = ixp4xx_pci_platform_notify_remove;
-
return 1;
}
-int dma_set_coherent_mask(struct device *dev, u64 mask)
-{
- if (mask >= SZ_64M - 1)
- return 0;
-
- return -EIO;
-}
-
EXPORT_SYMBOL(ixp4xx_pci_read);
EXPORT_SYMBOL(ixp4xx_pci_write);
-EXPORT_SYMBOL(dma_set_coherent_mask);
diff --git a/arch/arm/mach-ixp4xx/common.c b/arch/arm/mach-ixp4xx/common.c
index a7906eb..0b6d146 100644
--- a/arch/arm/mach-ixp4xx/common.c
+++ b/arch/arm/mach-ixp4xx/common.c
@@ -30,8 +30,8 @@
#include <linux/export.h>
#include <linux/gpio.h>
#include <linux/cpu.h>
+#include <linux/pci.h>
#include <linux/sched_clock.h>
-
#include <mach/udc.h>
#include <mach/hardware.h>
#include <mach/io.h>
@@ -40,7 +40,6 @@
#include <asm/page.h>
#include <asm/irq.h>
#include <asm/system_misc.h>
-
#include <asm/mach/map.h>
#include <asm/mach/irq.h>
#include <asm/mach/time.h>
@@ -578,6 +577,56 @@ void ixp4xx_restart(enum reboot_mode mode, const char *cmd)
}
}
+#ifdef CONFIG_PCI
+static int ixp4xx_needs_bounce(struct device *dev, dma_addr_t dma_addr, size_t size)
+{
+ return (dma_addr + size) >= SZ_64M;
+}
+
+static int ixp4xx_platform_notify_remove(struct device *dev)
+{
+ if (dev->bus == &pci_bus_type)
+ dmabounce_unregister_dev(dev);
+
+ return 0;
+}
+#endif
+
+/*
+ * Setup DMA mask to 64MB on PCI devices and 4 GB on all other things.
+ */
+static int ixp4xx_platform_notify(struct device *dev)
+{
+ dev->dma_mask = &dev->coherent_dma_mask;
+
+#ifdef CONFIG_PCI
+ if (dev_is_pci(dev)) {
+ dev->coherent_dma_mask = DMA_BIT_MASK(28); /* 64 MB */
+ dmabounce_register_dev(dev, 2048, 4096, ixp4xx_needs_bounce);
+ return 0;
+ }
+#endif
+
+ dev->coherent_dma_mask = DMA_BIT_MASK(32);
+ return 0;
+}
+
+int dma_set_coherent_mask(struct device *dev, u64 mask)
+{
+#ifdef CONFIG_PCI
+ if (dev_is_pci(dev))
+ mask &= DMA_BIT_MASK(28); /* 64 MB */
+#endif
+
+ if ((mask & DMA_BIT_MASK(28)) == DMA_BIT_MASK(28)) {
+ dev->coherent_dma_mask = mask;
+ return 0;
+ }
+
+ return -EIO; /* device wanted sub-64MB mask */
+}
+EXPORT_SYMBOL(dma_set_coherent_mask);
+
#ifdef CONFIG_IXP4XX_INDIRECT_PCI
/*
* In the case of using indirect PCI, we simply return the actual PCI
@@ -600,12 +649,16 @@ static void ixp4xx_iounmap(void __iomem *addr)
if (!is_pci_memory((__force u32)addr))
__iounmap(addr);
}
+#endif
void __init ixp4xx_init_early(void)
{
+ platform_notify = ixp4xx_platform_notify;
+#ifdef CONFIG_PCI
+ platform_notify_remove = ixp4xx_platform_notify_remove;
+#endif
+#ifdef CONFIG_IXP4XX_INDIRECT_PCI
arch_ioremap_caller = ixp4xx_ioremap_caller;
arch_iounmap = ixp4xx_iounmap;
-}
-#else
-void __init ixp4xx_init_early(void) {}
#endif
+}
--
Krzysztof Halasa
Research Institute for Automation and Measurements PIAP
Al. Jerozolimskie 202, 02-486 Warsaw, Poland
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation
2014-03-18 7:41 ` Krzysztof Hałasa
@ 2014-03-18 8:09 ` Simon Kågström
2014-03-18 10:23 ` Krzysztof Hałasa
0 siblings, 1 reply; 8+ messages in thread
From: Simon Kågström @ 2014-03-18 8:09 UTC (permalink / raw)
To: linux-arm-kernel
On Tue, 18 Mar 2014 08:41:12 +0100
Krzysztof Ha?asa <khalasa@piap.pl> wrote:
> Believe it or not, the correct patch is the one I'm attaching. One can
> add extra *set_masks* (e.g. the new call setting both streaming and
> coherent masks) in the drivers, but essentially it must do what this one
> does. Also, converting the mask (in the dev struct) from a pointer to a
> simple value would IMHO make sense, too.
OK, you know the hardware better so if you say so!
> I'm unable to look at this ATM but I will update the patch to the new
> kernel, perhaps soon.
Some comments below though:
> #include <asm/mach/map.h>
> #include <asm/mach/irq.h>
> #include <asm/mach/time.h>
> @@ -578,6 +577,56 @@ void ixp4xx_restart(enum reboot_mode mode, const char *cmd)
> }
> }
>
> +#ifdef CONFIG_PCI
To me it seems the conditional compilation is superfluous - dev_is_pci
is used to check for PCI-devices anyway, and the compiler should be
smart enough to remove dead code for the non-PCI case.
> +static int ixp4xx_needs_bounce(struct device *dev, dma_addr_t dma_addr, size_t size)
> +{
> + return (dma_addr + size) >= SZ_64M;
> +}
> +
> +static int ixp4xx_platform_notify_remove(struct device *dev)
> +{
> + if (dev->bus == &pci_bus_type)
if (dev_is_pci(dev))
> + dmabounce_unregister_dev(dev);
> +
> + return 0;
> +}
// Simon
^ permalink raw reply [flat|nested] 8+ messages in thread
* [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation
2014-03-18 8:09 ` Simon Kågström
@ 2014-03-18 10:23 ` Krzysztof Hałasa
0 siblings, 0 replies; 8+ messages in thread
From: Krzysztof Hałasa @ 2014-03-18 10:23 UTC (permalink / raw)
To: linux-arm-kernel
Simon K?gstr?m <simon.kagstrom@netinsight.net> writes:
>> +#ifdef CONFIG_PCI
>
> To me it seems the conditional compilation is superfluous - dev_is_pci
> is used to check for PCI-devices anyway, and the compiler should be
> smart enough to remove dead code for the non-PCI case.
Right.
>> +static int ixp4xx_needs_bounce(struct device *dev, dma_addr_t dma_addr, size_t size)
>> +{
>> + return (dma_addr + size) >= SZ_64M;
>> +}
>> +
>> +static int ixp4xx_platform_notify_remove(struct device *dev)
>> +{
>> + if (dev->bus == &pci_bus_type)
>
> if (dev_is_pci(dev))
Sure.
I will finally fix this, but I want it to at least successfully build
and boot on a couple of boards from different makers before I send
anything to be applied.
Just not now :-(
--
Krzysztof Halasa
Research Institute for Automation and Measurements PIAP
Al. Jerozolimskie 202, 02-486 Warsaw, Poland
^ permalink raw reply [flat|nested] 8+ messages in thread
end of thread, other threads:[~2014-03-18 10:23 UTC | newest]
Thread overview: 8+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-03-17 9:54 [PATCH RESEND 0/2]: ixp4xx: Fix 3.7 regression for IXP4xx ethernet driver Simon Kågström
2014-03-17 9:56 ` [PATCH RESEND 1/2] ARM: ixp4xx: Make dma_set_coherent_mask common, correct implementation Simon Kågström
2014-03-17 11:29 ` Arnd Bergmann
2014-03-17 11:54 ` Simon Kågström
2014-03-18 7:41 ` Krzysztof Hałasa
2014-03-18 8:09 ` Simon Kågström
2014-03-18 10:23 ` Krzysztof Hałasa
2014-03-17 9:57 ` [PATCH RESEND 2/2] ixp4xx_eth: Setup coherent_dma_mask Simon Kågström
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).