diff options
author | Mark Kettenis <kettenis@cvs.openbsd.org> | 2020-10-26 14:34:47 +0000 |
---|---|---|
committer | Mark Kettenis <kettenis@cvs.openbsd.org> | 2020-10-26 14:34:47 +0000 |
commit | ca973f5b2930b380b008f39459be3ccbbae61958 (patch) | |
tree | 521a845a8afc872f82967faf612106d7155e1c33 | |
parent | c11eaa2cf540860449f8805c10c27b52126301e6 (diff) |
Improve I/O functions to match Linux:
- Reverse byte order on big-endian architectures (except on sparc64)
- Add memory barriers
Note that sparc64 is special as we use a little-endian mapping for
PCI bus space, so we shouldn't reverse the byte order there.
ok jsg@
-rw-r--r-- | sys/dev/pci/drm/include/linux/io.h | 109 |
1 files changed, 100 insertions, 9 deletions
diff --git a/sys/dev/pci/drm/include/linux/io.h b/sys/dev/pci/drm/include/linux/io.h index b88352f733a..09c66b0d877 100644 --- a/sys/dev/pci/drm/include/linux/io.h +++ b/sys/dev/pci/drm/include/linux/io.h @@ -8,6 +8,7 @@ #include <sys/memrange.h> /* for MDF_WRITECOMBINE */ #include <linux/types.h> +#include <linux/atomic.h> #include <linux/compiler.h> #include <linux/vmalloc.h> @@ -18,51 +19,141 @@ static inline u8 ioread8(const volatile void __iomem *addr) { - return (*(volatile uint8_t *)addr); + uint8_t val; + + barrier(); + val = *(volatile uint8_t *)addr; + rmb(); + return val; +} + +static inline void +iowrite8(u8 val, volatile void __iomem *addr) +{ + wmb(); + *(volatile uint8_t *)addr = val; } +#ifdef __sparc64__ + +/* + * On sparc64, bus_space_vaddr(9) returns a virtual address that has + * been mapped little-endian, so we should just use the CPU byte + * order. + */ + static inline u16 ioread16(const volatile void __iomem *addr) { - return (*(volatile uint16_t *)addr); + uint16_t val; + + barrier(); + val = *(volatile uint16_t *)addr; + rmb(); + return val; } static inline u32 ioread32(const volatile void __iomem *addr) { - return (*(volatile uint32_t *)addr); + uint32_t val; + + barrier(); + val = *(volatile uint32_t *)addr; + rmb(); + return val; } static inline u64 ioread64(const volatile void __iomem *addr) { - return (*(volatile uint64_t *)addr); -} + uint64_t val; -static inline void -iowrite8(u8 val, volatile void __iomem *addr) -{ - *(volatile uint8_t *)addr = val; + barrier(); + val = *(volatile uint64_t *)addr; + rmb(); + return val; } static inline void iowrite16(u16 val, volatile void __iomem *addr) { + wmb(); *(volatile uint16_t *)addr = val; } static inline void iowrite32(u32 val, volatile void __iomem *addr) { + wmb(); *(volatile uint32_t *)addr = val; } static inline void iowrite64(u64 val, volatile void __iomem *addr) { + wmb(); *(volatile uint64_t *)addr = val; } +#else + +static inline u16 +ioread16(const volatile void __iomem *addr) +{ + uint16_t val; + + barrier(); + val = lemtoh16(addr); + rmb(); + return val; +} + +static inline u32 +ioread32(const volatile void __iomem *addr) +{ + uint32_t val; + + barrier(); + val = lemtoh32(addr); + rmb(); + return val; +} + +static inline u64 +ioread64(const volatile void __iomem *addr) +{ + uint64_t val; + + barrier(); + val = lemtoh64(addr); + rmb(); + return val; +} + +static inline void +iowrite16(u16 val, volatile void __iomem *addr) +{ + wmb(); + htolem16(addr, val); +} + +static inline void +iowrite32(u32 val, volatile void __iomem *addr) +{ + wmb(); + htolem32(addr, val); +} + +static inline void +iowrite64(u64 val, volatile void __iomem *addr) +{ + wmb(); + htolem64(addr, val); +} + +#endif + #define readb(p) ioread8(p) #define writeb(v, p) iowrite8(v, p) #define readw(p) ioread16(p) |