diff mbox

read/write[bwl] endian conversion for big-endian PCI ?

Message ID 49B4E27C.4050108@renesas.com (mailing list archive)
State Not Applicable
Headers show

Commit Message

Takashi Yoshii March 9, 2009, 9:33 a.m. UTC
# Attached patch is not to be checked in to the repository.

I found the patch attached is needed to make a NIC on SH7780LCR
 working in big-endian mode.

Reading the source code drivers/nic/r8169.c, 
what I can guess about the system it has been written for, are

1. Data structures transfered by PCI DMA are little endian.
 It users explicit conversion when accessing descriptors.

2. Data read/written by PCI target transfer are big(=cpu) endian.
 MMIO registers are accessed read/write[bwl] without conversion.
 Or perhaps read/write[bwl]() do the conversion.
 I'v found some of powerPC platforms have such code for PCI space.

Because PCI is defined only for little-endian, endian conversion which
PCI bridge on big-endian hardware provides can vary.
I know I can not help conversion tasks depend on each platforms.

But, I think read/write[bwl]() might have a rule about cross-endian access.
Do they provide endian conversion ?

I could not find the answer in
 Documentation/DocBook/deviceiobook
 Documentation/PCI/
.

Does anybody know the document and/or discussion?

Cheers,
/yoshii


--
To unsubscribe from this list: send the line "unsubscribe linux-sh" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Comments

Paul Mundt March 9, 2009, 5:28 p.m. UTC | #1
On Mon, Mar 09, 2009 at 06:33:48PM +0900, Takashi Yoshii wrote:
> Reading the source code drivers/nic/r8169.c, 
> what I can guess about the system it has been written for, are
> 
> 1. Data structures transfered by PCI DMA are little endian.
>  It users explicit conversion when accessing descriptors.
> 
> 2. Data read/written by PCI target transfer are big(=cpu) endian.
>  MMIO registers are accessed read/write[bwl] without conversion.
>  Or perhaps read/write[bwl]() do the conversion.
>  I'v found some of powerPC platforms have such code for PCI space.
> 
> Because PCI is defined only for little-endian, endian conversion which
> PCI bridge on big-endian hardware provides can vary.
> I know I can not help conversion tasks depend on each platforms.
> 
This is a pretty common problem, we just haven't run in to it much in the
past since almost no one uses any PCI-capable SH part in big-endian mode.

Take a look at the CONFIG_SWAP_IO_SPACE on mips, we should be able to do
something pretty similar without much hassle.
--
To unsubscribe from this list: send the line "unsubscribe linux-sh" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Takashi Yoshii March 10, 2009, 8:39 a.m. UTC | #2
Hi,

> Take a look at the CONFIG_SWAP_IO_SPACE on mips, we should be able to do
> something pretty similar without much hassle.
Reading sources with grep, It is becoming clearer that things
 are not so clearly defined. Anyway, I am reading more, slowly:).
Thank you. 
/yoshii
--
To unsubscribe from this list: send the line "unsubscribe linux-sh" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
diff mbox

Patch

diff --git a/arch/sh/drivers/pci/pci-sh7780.c b/arch/sh/drivers/pci/pci-sh7780.c
index 078dc44..7728ccc 100644
--- a/arch/sh/drivers/pci/pci-sh7780.c
+++ b/arch/sh/drivers/pci/pci-sh7780.c
@@ -154,6 +154,9 @@  int __init sh7780_pcic_init(struct sh4_pci_address_map *map)
 	/* SH7780 init done, set central function init complete */
 	/* use round robin mode to stop a device starving/overruning */
 	word = SH4_PCICR_PREFIX | SH4_PCICR_CFIN | SH4_PCICR_FTO;
+#if defined(CONFIG_CPU_BIG_ENDIAN)
+	word |= SH4_PCICR_BSWP;
+#endif
 	pci_write_reg(word, SH4_PCICR);
 
 	return 1;
diff --git a/drivers/net/r8169.c b/drivers/net/r8169.c
index 0771eb6..de9d72b 100644
--- a/drivers/net/r8169.c
+++ b/drivers/net/r8169.c
@@ -87,11 +87,11 @@  static const int multicast_filter_limit = 32;
 
 /* write/read MMIO register */
 #define RTL_W8(reg, val8)	writeb ((val8), ioaddr + (reg))
-#define RTL_W16(reg, val16)	writew ((val16), ioaddr + (reg))
-#define RTL_W32(reg, val32)	writel ((val32), ioaddr + (reg))
+#define RTL_W16(reg, val16)	writew (cpu_to_le16(val16), ioaddr + (reg))
+#define RTL_W32(reg, val32)	writel (cpu_to_le32(val32), ioaddr + (reg))
 #define RTL_R8(reg)		readb (ioaddr + (reg))
-#define RTL_R16(reg)		readw (ioaddr + (reg))
-#define RTL_R32(reg)		((unsigned long) readl (ioaddr + (reg)))
+#define RTL_R16(reg)		le16_to_cpu(readw (ioaddr + (reg)))
+#define RTL_R32(reg)		((unsigned long) le32_to_cpu(readl (ioaddr + (reg))))
 
 enum mac_version {
 	RTL_GIGA_MAC_VER_01 = 0x01, // 8169