2013-03-19 15:59:21

by Christophe Aeschlimann

[permalink] [raw]
Subject: [PATCH] ixp4xx_eth: set the device dma_coherent_mask

Without the mask it is impossible to take the network interface up
since it returns the following error:

> net eth1: coherent DMA mask is unset
> ifconfig: SIOCSIFFLAGS: Cannot allocate memory

Tested on an out-of-tree ixp425 based board.

Signed-off-by: Christophe Aeschlimann <[email protected]>
---
drivers/net/ethernet/xscale/ixp4xx_eth.c | 1 +
1 files changed, 1 insertions(+), 0 deletions(-)

diff --git a/drivers/net/ethernet/xscale/ixp4xx_eth.c b/drivers/net/ethernet/xscale/ixp4xx_eth.c
index 6958a5e..ff23c5c 100644
--- a/drivers/net/ethernet/xscale/ixp4xx_eth.c
+++ b/drivers/net/ethernet/xscale/ixp4xx_eth.c
@@ -1398,6 +1398,7 @@ static int eth_init_one(struct platform_device *pdev)
return -ENOMEM;

SET_NETDEV_DEV(dev, &pdev->dev);
+ dev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
port = netdev_priv(dev);
port->netdev = dev;
port->id = pdev->id;
--
1.7.9


2013-03-19 16:34:01

by David Miller

[permalink] [raw]
Subject: Re: [PATCH] ixp4xx_eth: set the device dma_coherent_mask

From: Christophe Aeschlimann <[email protected]>
Date: Tue, 19 Mar 2013 16:59:25 +0100

> Without the mask it is impossible to take the network interface up
> since it returns the following error:
>
>> net eth1: coherent DMA mask is unset
>> ifconfig: SIOCSIFFLAGS: Cannot allocate memory
>
> Tested on an out-of-tree ixp425 based board.
>
> Signed-off-by: Christophe Aeschlimann <[email protected]>
...
> @@ -1398,6 +1398,7 @@ static int eth_init_one(struct platform_device *pdev)
> return -ENOMEM;
>
> SET_NETDEV_DEV(dev, &pdev->dev);
> + dev->dev.coherent_dma_mask = DMA_BIT_MASK(32);

Hmmm, shouldn't this be the default value, set by the bus layer or
similar?

2013-03-19 17:04:59

by Mugunthan V N

[permalink] [raw]
Subject: Re: [PATCH] ixp4xx_eth: set the device dma_coherent_mask

On 3/19/2013 10:03 PM, David Miller wrote:
> From: Christophe Aeschlimann<[email protected]>
> Date: Tue, 19 Mar 2013 16:59:25 +0100
>
>> >Without the mask it is impossible to take the network interface up
>> >since it returns the following error:
>> >
>>> >>net eth1: coherent DMA mask is unset
>>> >>ifconfig: SIOCSIFFLAGS: Cannot allocate memory
>> >
>> >Tested on an out-of-tree ixp425 based board.
>> >
>> >Signed-off-by: Christophe Aeschlimann<[email protected]>
> ...
>> >@@ -1398,6 +1398,7 @@ static int eth_init_one(struct platform_device *pdev)
>> > return -ENOMEM;
>> >
>> > SET_NETDEV_DEV(dev, &pdev->dev);
>> >+ dev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
> Hmmm, shouldn't this be the default value, set by the bus layer or
> similar?
bus layer or any platform code doesn't init this value. The same issue
applies
to CPSW driver also. Previously the same was done in board or device
file. But
this approach is obsolete now, need to think of how it can be resolved in DT
approach

Regards
Mugunthan V N

2013-03-19 22:45:02

by Krzysztof Halasa

[permalink] [raw]
Subject: Re: [PATCH] ixp4xx_eth: set the device dma_coherent_mask

Christophe Aeschlimann <[email protected]> writes:

> Without the mask it is impossible to take the network interface up
> since it returns the following error:
>
>> net eth1: coherent DMA mask is unset
>> ifconfig: SIOCSIFFLAGS: Cannot allocate memory
>
> +++ b/drivers/net/ethernet/xscale/ixp4xx_eth.c
> @@ -1398,6 +1398,7 @@ static int eth_init_one(struct platform_device *pdev)
> return -ENOMEM;
>
> SET_NETDEV_DEV(dev, &pdev->dev);
> + dev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
> port = netdev_priv(dev);
> port->netdev = dev;
> port->id = pdev->id;

This happens to work but seems to me like a bad "solution". The real
issue is that current ARM coherent_dma_mask (and regular dma_mask)
handling is just broken (at least on platforms with
CONFIG_ARCH_HAS_DMA_SET_COHERENT_MASK). I'm thinking how can it be
fixed.

The problematic parts include the dmabouncer (which takes over DMA
operations and provide services similar to swiotlb), the host DMA and
coherent DMA masks (which are property of the buses) and the device
masks, which should be ANDed with bus masks (but aren't).

What the driver should use is:
err = dma_set_coherent_mask(xxx);

(but the subsystem have to be fixed first).
Also, according to DMA-API, 32-bit is the default.
--
Krzysztof Halasa