921a86e0e3
This is a driver for SBE Inc.'s dual port T3/E3 WAN cards. Based on their original GPLed driver. The original driver tarball is now accessible at http://userweb.kernel.org/~chris/SBE_2T3_Linux_2.0c.tgz It needs at least a new generic HDLC setup code (not yet written) before moving to drivers/net/wan. Signed-off-by: Krzysztof Hałasa <khc@pm.waw.pl> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
353 lines
10 KiB
C
353 lines
10 KiB
C
/*
|
|
* SBE 2T3E3 synchronous serial card driver for Linux
|
|
*
|
|
* Copyright (C) 2009-2010 Krzysztof Halasa <khc@pm.waw.pl>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of version 2 of the GNU General Public License
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This code is based on a driver written by SBE Inc.
|
|
*/
|
|
|
|
#include <linux/ip.h>
|
|
#include <asm/system.h>
|
|
#include "2t3e3.h"
|
|
#include "ctrl.h"
|
|
|
|
/* All access to registers done via the 21143 on port 0 must be
|
|
* protected via the card->bootrom_lock. */
|
|
|
|
/* priviate define to be used here only - must be protected by card->bootrom_lock */
|
|
#define cpld_write_nolock(channel, reg, val) \
|
|
bootrom_write((channel), CPLD_MAP_REG(reg, channel), val)
|
|
|
|
u32 cpld_read(struct channel *channel, u32 reg)
|
|
{
|
|
unsigned long flags;
|
|
u32 val;
|
|
|
|
spin_lock_irqsave(&channel->card->bootrom_lock, flags);
|
|
val = bootrom_read((channel), CPLD_MAP_REG(reg, channel));
|
|
spin_unlock_irqrestore(&channel->card->bootrom_lock, flags);
|
|
return val;
|
|
}
|
|
|
|
/****************************************
|
|
* Access via BootROM port
|
|
****************************************/
|
|
|
|
u32 bootrom_read(struct channel *channel, u32 reg)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr;
|
|
u32 result;
|
|
|
|
/* select BootROM address */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_PROGRAMMING_ADDRESS, reg & 0x3FFFF);
|
|
|
|
/* select reading from BootROM */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_BOOT_ROM_SELECT);
|
|
|
|
udelay(2); /* 20 PCI cycles */
|
|
|
|
/* read from BootROM */
|
|
result = dc_read(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT) & 0xff;
|
|
|
|
/* reset CSR9 */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT, 0);
|
|
|
|
return result;
|
|
}
|
|
|
|
void bootrom_write(struct channel *channel, u32 reg, u32 val)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr;
|
|
|
|
/* select BootROM address */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_PROGRAMMING_ADDRESS, reg & 0x3FFFF);
|
|
|
|
/* select writting to BootROM */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_WRITE_OPERATION |
|
|
SBE_2T3E3_21143_VAL_BOOT_ROM_SELECT |
|
|
(val & 0xff));
|
|
|
|
udelay(2); /* 20 PCI cycles */
|
|
|
|
/* reset CSR9 */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT, 0);
|
|
}
|
|
|
|
|
|
/****************************************
|
|
* Access via Serial I/O port
|
|
****************************************/
|
|
|
|
static u32 serialrom_read_bit(struct channel *channel)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr;
|
|
u32 bit;
|
|
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CLOCK |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock high */
|
|
|
|
bit = (dc_read(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT) &
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_DATA_OUT) > 0 ? 1 : 0;
|
|
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
|
|
return bit;
|
|
}
|
|
|
|
static void serialrom_write_bit(struct channel *channel, u32 bit)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr;
|
|
u32 lastbit = -1;
|
|
|
|
bit &= 1;
|
|
|
|
if (bit != lastbit) {
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_WRITE_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT |
|
|
(bit << 2)); /* clock low */
|
|
|
|
lastbit = bit;
|
|
}
|
|
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_WRITE_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CLOCK |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT |
|
|
(bit << 2)); /* clock high */
|
|
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_WRITE_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT |
|
|
(bit << 2)); /* clock low */
|
|
}
|
|
|
|
/****************************************
|
|
* Access to SerialROM (eeprom)
|
|
****************************************/
|
|
|
|
u32 t3e3_eeprom_read_word(struct channel *channel, u32 address)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr;
|
|
u32 i, val;
|
|
unsigned long flags;
|
|
|
|
address &= 0x3f;
|
|
|
|
spin_lock_irqsave(&channel->card->bootrom_lock, flags);
|
|
|
|
/* select correct Serial Chip */
|
|
cpld_write_nolock(channel, SBE_2T3E3_CPLD_REG_SERIAL_CHIP_SELECT,
|
|
SBE_2T3E3_CPLD_VAL_EEPROM_SELECT);
|
|
|
|
/* select reading from Serial I/O Bus */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
|
|
/* select read operation */
|
|
serialrom_write_bit(channel, 0);
|
|
serialrom_write_bit(channel, 1);
|
|
serialrom_write_bit(channel, 1);
|
|
serialrom_write_bit(channel, 0);
|
|
|
|
for (i = 0x20; i; i >>= 1)
|
|
serialrom_write_bit(channel, address & i ? 1 : 0);
|
|
|
|
val = 0;
|
|
for (i = 0x8000; i; i >>= 1)
|
|
val |= (serialrom_read_bit(channel) ? i : 0);
|
|
|
|
/* Reset 21143's CSR9 */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT, 0);
|
|
|
|
/* Unselect Serial Chip */
|
|
cpld_write_nolock(channel, SBE_2T3E3_CPLD_REG_SERIAL_CHIP_SELECT, 0);
|
|
|
|
spin_unlock_irqrestore(&channel->card->bootrom_lock, flags);
|
|
|
|
return ntohs(val);
|
|
}
|
|
|
|
|
|
/****************************************
|
|
* Access to Framer
|
|
****************************************/
|
|
|
|
u32 exar7250_read(struct channel *channel, u32 reg)
|
|
{
|
|
u32 result;
|
|
unsigned long flags;
|
|
|
|
#if 0
|
|
switch (reg) {
|
|
case SBE_2T3E3_FRAMER_REG_OPERATING_MODE:
|
|
return channel->framer_regs[reg];
|
|
break;
|
|
default:
|
|
}
|
|
#endif
|
|
|
|
spin_lock_irqsave(&channel->card->bootrom_lock, flags);
|
|
|
|
result = bootrom_read(channel, cpld_reg_map[SBE_2T3E3_CPLD_REG_FRAMER_BASE_ADDRESS]
|
|
[channel->h.slot] + (t3e3_framer_reg_map[reg] << 2));
|
|
|
|
spin_unlock_irqrestore(&channel->card->bootrom_lock, flags);
|
|
|
|
return result;
|
|
}
|
|
|
|
void exar7250_write(struct channel *channel, u32 reg, u32 val)
|
|
{
|
|
unsigned long flags;
|
|
|
|
val &= 0xff;
|
|
channel->framer_regs[reg] = val;
|
|
|
|
spin_lock_irqsave(&channel->card->bootrom_lock, flags);
|
|
|
|
bootrom_write(channel, cpld_reg_map[SBE_2T3E3_CPLD_REG_FRAMER_BASE_ADDRESS]
|
|
[channel->h.slot] + (t3e3_framer_reg_map[reg] << 2), val);
|
|
|
|
spin_unlock_irqrestore(&channel->card->bootrom_lock, flags);
|
|
}
|
|
|
|
|
|
/****************************************
|
|
* Access to LIU
|
|
****************************************/
|
|
|
|
u32 exar7300_read(struct channel *channel, u32 reg)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr, flags;
|
|
u32 i, val;
|
|
|
|
#if 0
|
|
switch (reg) {
|
|
case SBE_2T3E3_LIU_REG_REG1:
|
|
case SBE_2T3E3_LIU_REG_REG2:
|
|
case SBE_2T3E3_LIU_REG_REG3:
|
|
case SBE_2T3E3_LIU_REG_REG4:
|
|
return channel->liu_regs[reg];
|
|
break;
|
|
default:
|
|
}
|
|
#endif
|
|
|
|
/* select correct Serial Chip */
|
|
|
|
spin_lock_irqsave(&channel->card->bootrom_lock, flags);
|
|
|
|
cpld_write_nolock(channel, SBE_2T3E3_CPLD_REG_SERIAL_CHIP_SELECT,
|
|
cpld_val_map[SBE_2T3E3_CPLD_VAL_LIU_SELECT][channel->h.slot]);
|
|
|
|
/* select reading from Serial I/O Bus */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
|
|
/* select read operation */
|
|
serialrom_write_bit(channel, 1);
|
|
|
|
/* Exar7300 register address is 4 bit long */
|
|
reg = t3e3_liu_reg_map[reg];
|
|
for (i = 0; i < 4; i++, reg >>= 1) /* 4 bits of SerialROM address */
|
|
serialrom_write_bit(channel, reg & 1);
|
|
for (i = 0; i < 3; i++) /* remaining 3 bits of SerialROM address */
|
|
serialrom_write_bit(channel, 0);
|
|
|
|
val = 0; /* Exar7300 register value is 5 bit long */
|
|
for (i = 0; i < 8; i++) /* 8 bits of SerialROM value */
|
|
val += (serialrom_read_bit(channel) << i);
|
|
|
|
/* Reset 21143's CSR9 */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_READ_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT, 0);
|
|
|
|
/* Unselect Serial Chip */
|
|
cpld_write_nolock(channel, SBE_2T3E3_CPLD_REG_SERIAL_CHIP_SELECT, 0);
|
|
|
|
spin_unlock_irqrestore(&channel->card->bootrom_lock, flags);
|
|
|
|
return val;
|
|
}
|
|
|
|
void exar7300_write(struct channel *channel, u32 reg, u32 val)
|
|
{
|
|
unsigned long addr = channel->card->bootrom_addr, flags;
|
|
u32 i;
|
|
|
|
channel->liu_regs[reg] = val;
|
|
|
|
/* select correct Serial Chip */
|
|
|
|
spin_lock_irqsave(&channel->card->bootrom_lock, flags);
|
|
|
|
cpld_write_nolock(channel, SBE_2T3E3_CPLD_REG_SERIAL_CHIP_SELECT,
|
|
cpld_val_map[SBE_2T3E3_CPLD_VAL_LIU_SELECT][channel->h.slot]);
|
|
|
|
/* select writting to Serial I/O Bus */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_WRITE_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
|
|
/* select write operation */
|
|
serialrom_write_bit(channel, 0);
|
|
|
|
/* Exar7300 register address is 4 bit long */
|
|
reg = t3e3_liu_reg_map[reg];
|
|
for (i = 0; i < 4; i++) { /* 4 bits */
|
|
serialrom_write_bit(channel, reg & 1);
|
|
reg >>= 1;
|
|
}
|
|
for (i = 0; i < 3; i++) /* remaining 3 bits of SerialROM address */
|
|
serialrom_write_bit(channel, 0);
|
|
|
|
/* Exar7300 register value is 5 bit long */
|
|
for (i = 0; i < 5; i++) {
|
|
serialrom_write_bit(channel, val & 1);
|
|
val >>= 1;
|
|
}
|
|
for (i = 0; i < 3; i++) /* remaining 3 bits of SerialROM value */
|
|
serialrom_write_bit(channel, 0);
|
|
|
|
/* Reset 21143_CSR9 */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT,
|
|
SBE_2T3E3_21143_VAL_WRITE_OPERATION |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_SELECT |
|
|
SBE_2T3E3_21143_VAL_SERIAL_ROM_CHIP_SELECT); /* clock low */
|
|
dc_write(addr, SBE_2T3E3_21143_REG_BOOT_ROM_SERIAL_ROM_AND_MII_MANAGEMENT, 0);
|
|
|
|
/* Unselect Serial Chip */
|
|
cpld_write_nolock(channel, SBE_2T3E3_CPLD_REG_SERIAL_CHIP_SELECT, 0);
|
|
|
|
spin_unlock_irqrestore(&channel->card->bootrom_lock, flags);
|
|
}
|