Commit 09779c6d authored by Nicolas Pitre's avatar Nicolas Pitre Committed by Jeff Garzik

[PATCH] smc91x: allow for dynamic bus access configs

All accessor's different methods are now selected with C code and unused
ones statically optimized away at compile time instead of being selected
with #if's and #ifdef's.  This has many advantages such as allowing the
compiler to validate the syntax of the whole code, making it cleaner and
easier to understand, and ultimately allowing people to define
configuration symbols in terms of variables if they really want to
dynamically support multiple bus configurations at the same time (with
the unavoidable performance cost).
Signed-off-by: default avatarNicolas Pitre <nico@cam.org>
Signed-off-by: default avatarJeff Garzik <jeff@garzik.org>
parent ac62ef04
...@@ -215,15 +215,12 @@ struct smc_local { ...@@ -215,15 +215,12 @@ struct smc_local {
spinlock_t lock; spinlock_t lock;
#ifdef SMC_CAN_USE_DATACS
u32 __iomem *datacs;
#endif
#ifdef SMC_USE_PXA_DMA #ifdef SMC_USE_PXA_DMA
/* DMA needs the physical address of the chip */ /* DMA needs the physical address of the chip */
u_long physaddr; u_long physaddr;
#endif #endif
void __iomem *base; void __iomem *base;
void __iomem *datacs;
}; };
#if SMC_DEBUG > 0 #if SMC_DEBUG > 0
...@@ -2104,9 +2101,8 @@ static int smc_enable_device(struct platform_device *pdev) ...@@ -2104,9 +2101,8 @@ static int smc_enable_device(struct platform_device *pdev)
* Set the appropriate byte/word mode. * Set the appropriate byte/word mode.
*/ */
ecsr = readb(addr + (ECSR << SMC_IO_SHIFT)) & ~ECSR_IOIS8; ecsr = readb(addr + (ECSR << SMC_IO_SHIFT)) & ~ECSR_IOIS8;
#ifndef SMC_CAN_USE_16BIT if (!SMC_CAN_USE_16BIT)
ecsr |= ECSR_IOIS8; ecsr |= ECSR_IOIS8;
#endif
writeb(ecsr, addr + (ECSR << SMC_IO_SHIFT)); writeb(ecsr, addr + (ECSR << SMC_IO_SHIFT));
local_irq_restore(flags); local_irq_restore(flags);
...@@ -2143,40 +2139,39 @@ static void smc_release_attrib(struct platform_device *pdev) ...@@ -2143,40 +2139,39 @@ static void smc_release_attrib(struct platform_device *pdev)
release_mem_region(res->start, ATTRIB_SIZE); release_mem_region(res->start, ATTRIB_SIZE);
} }
#ifdef SMC_CAN_USE_DATACS static inline void smc_request_datacs(struct platform_device *pdev, struct net_device *ndev)
static void smc_request_datacs(struct platform_device *pdev, struct net_device *ndev)
{ {
struct resource * res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "smc91x-data32"); if (SMC_CAN_USE_DATACS) {
struct smc_local *lp = netdev_priv(ndev); struct resource * res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "smc91x-data32");
struct smc_local *lp = netdev_priv(ndev);
if (!res) if (!res)
return; return;
if(!request_mem_region(res->start, SMC_DATA_EXTENT, CARDNAME)) { if(!request_mem_region(res->start, SMC_DATA_EXTENT, CARDNAME)) {
printk(KERN_INFO "%s: failed to request datacs memory region.\n", CARDNAME); printk(KERN_INFO "%s: failed to request datacs memory region.\n", CARDNAME);
return; return;
} }
lp->datacs = ioremap(res->start, SMC_DATA_EXTENT); lp->datacs = ioremap(res->start, SMC_DATA_EXTENT);
}
} }
static void smc_release_datacs(struct platform_device *pdev, struct net_device *ndev) static void smc_release_datacs(struct platform_device *pdev, struct net_device *ndev)
{ {
struct smc_local *lp = netdev_priv(ndev); if (SMC_CAN_USE_DATACS) {
struct resource * res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "smc91x-data32"); struct smc_local *lp = netdev_priv(ndev);
struct resource * res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "smc91x-data32");
if (lp->datacs) if (lp->datacs)
iounmap(lp->datacs); iounmap(lp->datacs);
lp->datacs = NULL; lp->datacs = NULL;
if (res) if (res)
release_mem_region(res->start, SMC_DATA_EXTENT); release_mem_region(res->start, SMC_DATA_EXTENT);
}
} }
#else
static void smc_request_datacs(struct platform_device *pdev, struct net_device *ndev) {}
static void smc_release_datacs(struct platform_device *pdev, struct net_device *ndev) {}
#endif
/* /*
* smc_init(void) * smc_init(void)
......
...@@ -275,7 +275,10 @@ SMC_outw(u16 val, void __iomem *ioaddr, int reg) ...@@ -275,7 +275,10 @@ SMC_outw(u16 val, void __iomem *ioaddr, int reg)
#define SMC_insw(a,r,p,l) readsw ((void*) ((a) + (r)), p, l) #define SMC_insw(a,r,p,l) readsw ((void*) ((a) + (r)), p, l)
#define SMC_outw(v,a,r) ({ writew ((v), (a) + (r)); LPD7A40X_IOBARRIER; }) #define SMC_outw(v,a,r) ({ writew ((v), (a) + (r)); LPD7A40X_IOBARRIER; })
static inline void SMC_outsw (unsigned long a, int r, unsigned char* p, int l) #define SMC_outsw LPD7A40X_SMC_outsw
static inline void LPD7A40X_SMC_outsw(unsigned long a, int r,
unsigned char* p, int l)
{ {
unsigned short* ps = (unsigned short*) p; unsigned short* ps = (unsigned short*) p;
while (l-- > 0) { while (l-- > 0) {
...@@ -342,10 +345,6 @@ static inline void SMC_outsw (unsigned long a, int r, unsigned char* p, int l) ...@@ -342,10 +345,6 @@ static inline void SMC_outsw (unsigned long a, int r, unsigned char* p, int l)
#endif #endif
#ifndef SMC_IRQ_FLAGS
#define SMC_IRQ_FLAGS SA_TRIGGER_RISING
#endif
#ifdef SMC_USE_PXA_DMA #ifdef SMC_USE_PXA_DMA
/* /*
* Let's use the DMA engine on the XScale PXA2xx for RX packets. This is * Let's use the DMA engine on the XScale PXA2xx for RX packets. This is
...@@ -441,10 +440,85 @@ smc_pxa_dma_irq(int dma, void *dummy, struct pt_regs *regs) ...@@ -441,10 +440,85 @@ smc_pxa_dma_irq(int dma, void *dummy, struct pt_regs *regs)
#endif /* SMC_USE_PXA_DMA */ #endif /* SMC_USE_PXA_DMA */
/* Because of bank switching, the LAN91x uses only 16 I/O ports */ /*
* Everything a particular hardware setup needs should have been defined
* at this point. Add stubs for the undefined cases, mainly to avoid
* compilation warnings since they'll be optimized away, or to prevent buggy
* use of them.
*/
#if ! SMC_CAN_USE_32BIT
#define SMC_inl(ioaddr, reg) ({ BUG(); 0; })
#define SMC_outl(x, ioaddr, reg) BUG()
#define SMC_insl(a, r, p, l) BUG()
#define SMC_outsl(a, r, p, l) BUG()
#endif
#if !defined(SMC_insl) || !defined(SMC_outsl)
#define SMC_insl(a, r, p, l) BUG()
#define SMC_outsl(a, r, p, l) BUG()
#endif
#if ! SMC_CAN_USE_16BIT
/*
* Any 16-bit access is performed with two 8-bit accesses if the hardware
* can't do it directly. Most registers are 16-bit so those are mandatory.
*/
#define SMC_outw(x, ioaddr, reg) \
do { \
unsigned int __val16 = (x); \
SMC_outb( __val16, ioaddr, reg ); \
SMC_outb( __val16 >> 8, ioaddr, reg + (1 << SMC_IO_SHIFT));\
} while (0)
#define SMC_inw(ioaddr, reg) \
({ \
unsigned int __val16; \
__val16 = SMC_inb( ioaddr, reg ); \
__val16 |= SMC_inb( ioaddr, reg + (1 << SMC_IO_SHIFT)) << 8; \
__val16; \
})
#define SMC_insw(a, r, p, l) BUG()
#define SMC_outsw(a, r, p, l) BUG()
#endif
#if !defined(SMC_insw) || !defined(SMC_outsw)
#define SMC_insw(a, r, p, l) BUG()
#define SMC_outsw(a, r, p, l) BUG()
#endif
#if ! SMC_CAN_USE_8BIT
#define SMC_inb(ioaddr, reg) ({ BUG(); 0; })
#define SMC_outb(x, ioaddr, reg) BUG()
#define SMC_insb(a, r, p, l) BUG()
#define SMC_outsb(a, r, p, l) BUG()
#endif
#if !defined(SMC_insb) || !defined(SMC_outsb)
#define SMC_insb(a, r, p, l) BUG()
#define SMC_outsb(a, r, p, l) BUG()
#endif
#ifndef SMC_CAN_USE_DATACS
#define SMC_CAN_USE_DATACS 0
#endif
#ifndef SMC_IO_SHIFT #ifndef SMC_IO_SHIFT
#define SMC_IO_SHIFT 0 #define SMC_IO_SHIFT 0
#endif #endif
#ifndef SMC_IRQ_FLAGS
#define SMC_IRQ_FLAGS SA_TRIGGER_RISING
#endif
#ifndef SMC_INTERRUPT_PREAMBLE
#define SMC_INTERRUPT_PREAMBLE
#endif
/* Because of bank switching, the LAN91x uses only 16 I/O ports */
#define SMC_IO_EXTENT (16 << SMC_IO_SHIFT) #define SMC_IO_EXTENT (16 << SMC_IO_SHIFT)
#define SMC_DATA_EXTENT (4) #define SMC_DATA_EXTENT (4)
...@@ -817,6 +891,11 @@ static const char * chip_ids[ 16 ] = { ...@@ -817,6 +891,11 @@ static const char * chip_ids[ 16 ] = {
* Note: the following macros do *not* select the bank -- this must * Note: the following macros do *not* select the bank -- this must
* be done separately as needed in the main code. The SMC_REG() macro * be done separately as needed in the main code. The SMC_REG() macro
* only uses the bank argument for debugging purposes (when enabled). * only uses the bank argument for debugging purposes (when enabled).
*
* Note: despite inline functions being safer, everything leading to this
* should preferably be macros to let BUG() display the line number in
* the core source code since we're interested in the top call site
* not in any inline function location.
*/ */
#if SMC_DEBUG > 0 #if SMC_DEBUG > 0
...@@ -834,62 +913,142 @@ static const char * chip_ids[ 16 ] = { ...@@ -834,62 +913,142 @@ static const char * chip_ids[ 16 ] = {
#define SMC_REG(reg, bank) (reg<<SMC_IO_SHIFT) #define SMC_REG(reg, bank) (reg<<SMC_IO_SHIFT)
#endif #endif
#if SMC_CAN_USE_8BIT /*
#define SMC_GET_PN() SMC_inb( ioaddr, PN_REG ) * Hack Alert: Some setups just can't write 8 or 16 bits reliably when not
#define SMC_SET_PN(x) SMC_outb( x, ioaddr, PN_REG ) * aligned to a 32 bit boundary. I tell you that does exist!
#define SMC_GET_AR() SMC_inb( ioaddr, AR_REG ) * Fortunately the affected register accesses can be easily worked around
#define SMC_GET_TXFIFO() SMC_inb( ioaddr, TXFIFO_REG ) * since we can write zeroes to the preceeding 16 bits without adverse
#define SMC_GET_RXFIFO() SMC_inb( ioaddr, RXFIFO_REG ) * effects and use a 32-bit access.
#define SMC_GET_INT() SMC_inb( ioaddr, INT_REG ) *
#define SMC_ACK_INT(x) SMC_outb( x, ioaddr, INT_REG ) * Enforce it on any 32-bit capable setup for now.
#define SMC_GET_INT_MASK() SMC_inb( ioaddr, IM_REG ) */
#define SMC_SET_INT_MASK(x) SMC_outb( x, ioaddr, IM_REG ) #define SMC_MUST_ALIGN_WRITE SMC_CAN_USE_32BIT
#else
#define SMC_GET_PN() (SMC_inw( ioaddr, PN_REG ) & 0xFF) #define SMC_GET_PN() \
#define SMC_SET_PN(x) SMC_outw( x, ioaddr, PN_REG ) ( SMC_CAN_USE_8BIT ? (SMC_inb(ioaddr, PN_REG)) \
#define SMC_GET_AR() (SMC_inw( ioaddr, PN_REG ) >> 8) : (SMC_inw(ioaddr, PN_REG) & 0xFF) )
#define SMC_GET_TXFIFO() (SMC_inw( ioaddr, TXFIFO_REG ) & 0xFF)
#define SMC_GET_RXFIFO() (SMC_inw( ioaddr, TXFIFO_REG ) >> 8) #define SMC_SET_PN(x) \
#define SMC_GET_INT() (SMC_inw( ioaddr, INT_REG ) & 0xFF) do { \
if (SMC_MUST_ALIGN_WRITE) \
SMC_outl((x)<<16, ioaddr, SMC_REG(0, 2)); \
else if (SMC_CAN_USE_8BIT) \
SMC_outb(x, ioaddr, PN_REG); \
else \
SMC_outw(x, ioaddr, PN_REG); \
} while (0)
#define SMC_GET_AR() \
( SMC_CAN_USE_8BIT ? (SMC_inb(ioaddr, AR_REG)) \
: (SMC_inw(ioaddr, PN_REG) >> 8) )
#define SMC_GET_TXFIFO() \
( SMC_CAN_USE_8BIT ? (SMC_inb(ioaddr, TXFIFO_REG)) \
: (SMC_inw(ioaddr, TXFIFO_REG) & 0xFF) )
#define SMC_GET_RXFIFO() \
( SMC_CAN_USE_8BIT ? (SMC_inb(ioaddr, RXFIFO_REG)) \
: (SMC_inw(ioaddr, TXFIFO_REG) >> 8) )
#define SMC_GET_INT() \
( SMC_CAN_USE_8BIT ? (SMC_inb(ioaddr, INT_REG)) \
: (SMC_inw(ioaddr, INT_REG) & 0xFF) )
#define SMC_ACK_INT(x) \ #define SMC_ACK_INT(x) \
do { \ do { \
unsigned long __flags; \ if (SMC_CAN_USE_8BIT) \
int __mask; \ SMC_outb(x, ioaddr, INT_REG); \
local_irq_save(__flags); \ else { \
__mask = SMC_inw( ioaddr, INT_REG ) & ~0xff; \ unsigned long __flags; \
SMC_outw( __mask | (x), ioaddr, INT_REG ); \ int __mask; \
local_irq_restore(__flags); \ local_irq_save(__flags); \
__mask = SMC_inw( ioaddr, INT_REG ) & ~0xff; \
SMC_outw( __mask | (x), ioaddr, INT_REG ); \
local_irq_restore(__flags); \
} \
} while (0)
#define SMC_GET_INT_MASK() \
( SMC_CAN_USE_8BIT ? (SMC_inb(ioaddr, IM_REG)) \
: (SMC_inw( ioaddr, INT_REG ) >> 8) )
#define SMC_SET_INT_MASK(x) \
do { \
if (SMC_CAN_USE_8BIT) \
SMC_outb(x, ioaddr, IM_REG); \
else \
SMC_outw((x) << 8, ioaddr, INT_REG); \
} while (0)
#define SMC_CURRENT_BANK() SMC_inw(ioaddr, BANK_SELECT)
#define SMC_SELECT_BANK(x) \
do { \
if (SMC_MUST_ALIGN_WRITE) \
SMC_outl((x)<<16, ioaddr, 12<<SMC_IO_SHIFT); \
else \
SMC_outw(x, ioaddr, BANK_SELECT); \
} while (0)
#define SMC_GET_BASE() SMC_inw(ioaddr, BASE_REG)
#define SMC_SET_BASE(x) SMC_outw(x, ioaddr, BASE_REG)
#define SMC_GET_CONFIG() SMC_inw(ioaddr, CONFIG_REG)
#define SMC_SET_CONFIG(x) SMC_outw(x, ioaddr, CONFIG_REG)
#define SMC_GET_COUNTER() SMC_inw(ioaddr, COUNTER_REG)
#define SMC_GET_CTL() SMC_inw(ioaddr, CTL_REG)
#define SMC_SET_CTL(x) SMC_outw(x, ioaddr, CTL_REG)
#define SMC_GET_MII() SMC_inw(ioaddr, MII_REG)
#define SMC_SET_MII(x) SMC_outw(x, ioaddr, MII_REG)
#define SMC_GET_MIR() SMC_inw(ioaddr, MIR_REG)
#define SMC_SET_MIR(x) SMC_outw(x, ioaddr, MIR_REG)
#define SMC_GET_MMU_CMD() SMC_inw(ioaddr, MMU_CMD_REG)
#define SMC_SET_MMU_CMD(x) SMC_outw(x, ioaddr, MMU_CMD_REG)
#define SMC_GET_FIFO() SMC_inw(ioaddr, FIFO_REG)
#define SMC_GET_PTR() SMC_inw(ioaddr, PTR_REG)
#define SMC_SET_PTR(x) \
do { \
if (SMC_MUST_ALIGN_WRITE) \
SMC_outl((x)<<16, ioaddr, SMC_REG(4, 2)); \
else \
SMC_outw(x, ioaddr, PTR_REG); \
} while (0) } while (0)
#define SMC_GET_INT_MASK() (SMC_inw( ioaddr, INT_REG ) >> 8)
#define SMC_SET_INT_MASK(x) SMC_outw( (x) << 8, ioaddr, INT_REG )
#endif
#define SMC_CURRENT_BANK() SMC_inw( ioaddr, BANK_SELECT ) #define SMC_GET_EPH_STATUS() SMC_inw(ioaddr, EPH_STATUS_REG)
#define SMC_SELECT_BANK(x) SMC_outw( x, ioaddr, BANK_SELECT )
#define SMC_GET_BASE() SMC_inw( ioaddr, BASE_REG ) #define SMC_GET_RCR() SMC_inw(ioaddr, RCR_REG)
#define SMC_SET_BASE(x) SMC_outw( x, ioaddr, BASE_REG )
#define SMC_GET_CONFIG() SMC_inw( ioaddr, CONFIG_REG ) #define SMC_SET_RCR(x) SMC_outw(x, ioaddr, RCR_REG)
#define SMC_SET_CONFIG(x) SMC_outw( x, ioaddr, CONFIG_REG )
#define SMC_GET_COUNTER() SMC_inw( ioaddr, COUNTER_REG ) #define SMC_GET_REV() SMC_inw(ioaddr, REV_REG)
#define SMC_GET_CTL() SMC_inw( ioaddr, CTL_REG )
#define SMC_SET_CTL(x) SMC_outw( x, ioaddr, CTL_REG ) #define SMC_GET_RPC() SMC_inw(ioaddr, RPC_REG)
#define SMC_GET_MII() SMC_inw( ioaddr, MII_REG )
#define SMC_SET_MII(x) SMC_outw( x, ioaddr, MII_REG ) #define SMC_SET_RPC(x) \
#define SMC_GET_MIR() SMC_inw( ioaddr, MIR_REG ) do { \
#define SMC_SET_MIR(x) SMC_outw( x, ioaddr, MIR_REG ) if (SMC_MUST_ALIGN_WRITE) \
#define SMC_GET_MMU_CMD() SMC_inw( ioaddr, MMU_CMD_REG ) SMC_outl((x)<<16, ioaddr, SMC_REG(8, 0)); \
#define SMC_SET_MMU_CMD(x) SMC_outw( x, ioaddr, MMU_CMD_REG ) else \
#define SMC_GET_FIFO() SMC_inw( ioaddr, FIFO_REG ) SMC_outw(x, ioaddr, RPC_REG); \
#define SMC_GET_PTR() SMC_inw( ioaddr, PTR_REG ) } while (0)
#define SMC_SET_PTR(x) SMC_outw( x, ioaddr, PTR_REG )
#define SMC_GET_EPH_STATUS() SMC_inw( ioaddr, EPH_STATUS_REG ) #define SMC_GET_TCR() SMC_inw(ioaddr, TCR_REG)
#define SMC_GET_RCR() SMC_inw( ioaddr, RCR_REG )
#define SMC_SET_RCR(x) SMC_outw( x, ioaddr, RCR_REG ) #define SMC_SET_TCR(x) SMC_outw(x, ioaddr, TCR_REG)
#define SMC_GET_REV() SMC_inw( ioaddr, REV_REG )
#define SMC_GET_RPC() SMC_inw( ioaddr, RPC_REG )
#define SMC_SET_RPC(x) SMC_outw( x, ioaddr, RPC_REG )
#define SMC_GET_TCR() SMC_inw( ioaddr, TCR_REG )
#define SMC_SET_TCR(x) SMC_outw( x, ioaddr, TCR_REG )
#ifndef SMC_GET_MAC_ADDR #ifndef SMC_GET_MAC_ADDR
#define SMC_GET_MAC_ADDR(addr) \ #define SMC_GET_MAC_ADDR(addr) \
...@@ -920,151 +1079,84 @@ static const char * chip_ids[ 16 ] = { ...@@ -920,151 +1079,84 @@ static const char * chip_ids[ 16 ] = {
SMC_outw( mt[6] | (mt[7] << 8), ioaddr, MCAST_REG4 ); \ SMC_outw( mt[6] | (mt[7] << 8), ioaddr, MCAST_REG4 ); \
} while (0) } while (0)
#if SMC_CAN_USE_32BIT
/*
* Some setups just can't write 8 or 16 bits reliably when not aligned
* to a 32 bit boundary. I tell you that exists!
* We re-do the ones here that can be easily worked around if they can have
* their low parts written to 0 without adverse effects.
*/
#undef SMC_SELECT_BANK
#define SMC_SELECT_BANK(x) SMC_outl( (x)<<16, ioaddr, 12<<SMC_IO_SHIFT )
#undef SMC_SET_RPC
#define SMC_SET_RPC(x) SMC_outl( (x)<<16, ioaddr, SMC_REG(8, 0) )
#undef SMC_SET_PN
#define SMC_SET_PN(x) SMC_outl( (x)<<16, ioaddr, SMC_REG(0, 2) )
#undef SMC_SET_PTR
#define SMC_SET_PTR(x) SMC_outl( (x)<<16, ioaddr, SMC_REG(4, 2) )
#endif
#if SMC_CAN_USE_32BIT
#define SMC_PUT_PKT_HDR(status, length) \
SMC_outl( (status) | (length) << 16, ioaddr, DATA_REG )
#define SMC_GET_PKT_HDR(status, length) \
do { \
unsigned int __val = SMC_inl( ioaddr, DATA_REG ); \
(status) = __val & 0xffff; \
(length) = __val >> 16; \
} while (0)
#else
#define SMC_PUT_PKT_HDR(status, length) \ #define SMC_PUT_PKT_HDR(status, length) \
do { \ do { \
SMC_outw( status, ioaddr, DATA_REG ); \ if (SMC_CAN_USE_32BIT) \
SMC_outw( length, ioaddr, DATA_REG ); \ SMC_outl((status) | (length)<<16, ioaddr, DATA_REG); \
} while (0) else { \
#define SMC_GET_PKT_HDR(status, length) \ SMC_outw(status, ioaddr, DATA_REG); \
do { \ SMC_outw(length, ioaddr, DATA_REG); \
(status) = SMC_inw( ioaddr, DATA_REG ); \ } \
(length) = SMC_inw( ioaddr, DATA_REG ); \
} while (0) } while (0)
#endif
#if SMC_CAN_USE_32BIT #define SMC_GET_PKT_HDR(status, length) \
#define _SMC_PUSH_DATA(p, l) \
do { \ do { \
char *__ptr = (p); \ if (SMC_CAN_USE_32BIT) { \
int __len = (l); \ unsigned int __val = SMC_inl(ioaddr, DATA_REG); \
if (__len >= 2 && (unsigned long)__ptr & 2) { \ (status) = __val & 0xffff; \
__len -= 2; \ (length) = __val >> 16; \
SMC_outw( *(u16 *)__ptr, ioaddr, DATA_REG ); \ } else { \
__ptr += 2; \ (status) = SMC_inw(ioaddr, DATA_REG); \
} \ (length) = SMC_inw(ioaddr, DATA_REG); \
SMC_outsl( ioaddr, DATA_REG, __ptr, __len >> 2); \
if (__len & 2) { \
__ptr += (__len & ~3); \
SMC_outw( *((u16 *)__ptr), ioaddr, DATA_REG ); \
} \ } \
} while (0) } while (0)
#define _SMC_PULL_DATA(p, l) \
do { \
char *__ptr = (p); \
int __len = (l); \
if ((unsigned long)__ptr & 2) { \
/* \
* We want 32bit alignment here. \
* Since some buses perform a full 32bit \
* fetch even for 16bit data we can't use \
* SMC_inw() here. Back both source (on chip \
* and destination) pointers of 2 bytes. \
*/ \
__ptr -= 2; \
__len += 2; \
SMC_SET_PTR( 2|PTR_READ|PTR_RCV|PTR_AUTOINC ); \
} \
__len += 2; \
SMC_insl( ioaddr, DATA_REG, __ptr, __len >> 2); \
} while (0)
#elif SMC_CAN_USE_16BIT
#define _SMC_PUSH_DATA(p, l) SMC_outsw( ioaddr, DATA_REG, p, (l) >> 1 )
#define _SMC_PULL_DATA(p, l) SMC_insw ( ioaddr, DATA_REG, p, (l) >> 1 )
#elif SMC_CAN_USE_8BIT
#define _SMC_PUSH_DATA(p, l) SMC_outsb( ioaddr, DATA_REG, p, l )
#define _SMC_PULL_DATA(p, l) SMC_insb ( ioaddr, DATA_REG, p, l )
#endif
#if ! SMC_CAN_USE_16BIT #define SMC_PUSH_DATA(p, l) \
#define SMC_outw(x, ioaddr, reg) \
do { \ do { \
unsigned int __val16 = (x); \ if (SMC_CAN_USE_32BIT) { \
SMC_outb( __val16, ioaddr, reg ); \ void *__ptr = (p); \
SMC_outb( __val16 >> 8, ioaddr, reg + (1 << SMC_IO_SHIFT));\ int __len = (l); \
void *__ioaddr = ioaddr; \
if (__len >= 2 && (unsigned long)__ptr & 2) { \
__len -= 2; \
SMC_outw(*(u16 *)__ptr, ioaddr, DATA_REG); \
__ptr += 2; \
} \
if (SMC_CAN_USE_DATACS && lp->datacs) \
__ioaddr = lp->datacs; \
SMC_outsl(__ioaddr, DATA_REG, __ptr, __len>>2); \
if (__len & 2) { \
__ptr += (__len & ~3); \
SMC_outw(*((u16 *)__ptr), ioaddr, DATA_REG); \
} \
} else if (SMC_CAN_USE_16BIT) \
SMC_outsw(ioaddr, DATA_REG, p, (l) >> 1); \
else if (SMC_CAN_USE_8BIT) \
SMC_outsb(ioaddr, DATA_REG, p, l); \
} while (0) } while (0)
#define SMC_inw(ioaddr, reg) \
({ \
unsigned int __val16; \
__val16 = SMC_inb( ioaddr, reg ); \
__val16 |= SMC_inb( ioaddr, reg + (1 << SMC_IO_SHIFT)) << 8; \
__val16; \
})
#endif
#ifdef SMC_CAN_USE_DATACS
#define SMC_PUSH_DATA(p, l) \
if ( lp->datacs ) { \
unsigned char *__ptr = (p); \
int __len = (l); \
if (__len >= 2 && (unsigned long)__ptr & 2) { \
__len -= 2; \
SMC_outw( *((u16 *)__ptr), ioaddr, DATA_REG ); \
__ptr += 2; \
} \
outsl(lp->datacs, __ptr, __len >> 2); \
if (__len & 2) { \
__ptr += (__len & ~3); \
SMC_outw( *((u16 *)__ptr), ioaddr, DATA_REG ); \
} \
} else { \
_SMC_PUSH_DATA(p, l); \
}
#define SMC_PULL_DATA(p, l) \ #define SMC_PULL_DATA(p, l) \
if ( lp->datacs ) { \ do { \
unsigned char *__ptr = (p); \ if (SMC_CAN_USE_32BIT) { \
int __len = (l); \ void *__ptr = (p); \
if ((unsigned long)__ptr & 2) { \ int __len = (l); \
/* \ void *__ioaddr = ioaddr; \
* We want 32bit alignment here. \ if ((unsigned long)__ptr & 2) { \
* Since some buses perform a full 32bit \ /* \
* fetch even for 16bit data we can't use \ * We want 32bit alignment here. \
* SMC_inw() here. Back both source (on chip \ * Since some buses perform a full \
* and destination) pointers of 2 bytes. \ * 32bit fetch even for 16bit data \
*/ \ * we can't use SMC_inw() here. \
__ptr -= 2; \ * Back both source (on-chip) and \
* destination pointers of 2 bytes. \
* This is possible since the call to \
* SMC_GET_PKT_HDR() already advanced \
* the source pointer of 4 bytes, and \
* the skb_reserve(skb, 2) advanced \
* the destination pointer of 2 bytes. \
*/ \
__ptr -= 2; \
__len += 2; \
SMC_SET_PTR(2|PTR_READ|PTR_RCV|PTR_AUTOINC); \
} \
if (SMC_CAN_USE_DATACS && lp->datacs) \
__ioaddr = lp->datacs; \
__len += 2; \ __len += 2; \
SMC_SET_PTR( 2|PTR_READ|PTR_RCV|PTR_AUTOINC ); \ SMC_insl(__ioaddr, DATA_REG, __ptr, __len>>2); \
} \ } else if (SMC_CAN_USE_16BIT) \
__len += 2; \ SMC_insw(ioaddr, DATA_REG, p, (l) >> 1); \
insl( lp->datacs, __ptr, __len >> 2); \ else if (SMC_CAN_USE_8BIT) \
} else { \ SMC_insb(ioaddr, DATA_REG, p, l); \
_SMC_PULL_DATA(p, l); \ } while (0)
}
#else
#define SMC_PUSH_DATA(p, l) _SMC_PUSH_DATA(p, l)
#define SMC_PULL_DATA(p, l) _SMC_PULL_DATA(p, l)
#endif
#if !defined (SMC_INTERRUPT_PREAMBLE)
# define SMC_INTERRUPT_PREAMBLE
#endif
#endif /* _SMC91X_H_ */ #endif /* _SMC91X_H_ */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment