| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052 |
- /*
- * Driver for the built-in ethernet switch of the Atheros AR7240 SoC
- * Copyright (c) 2010 Gabor Juhos <[email protected]>
- * Copyright (c) 2010 Felix Fietkau <[email protected]>
- *
- * This program is free software; you can redistribute it and/or modify it
- * under the terms of the GNU General Public License version 2 as published
- * by the Free Software Foundation.
- *
- */
- #include <linux/etherdevice.h>
- #include <linux/list.h>
- #include <linux/netdevice.h>
- #include <linux/phy.h>
- #include <linux/mii.h>
- #include <linux/bitops.h>
- #include <linux/switch.h>
- #include "ag71xx.h"
- #define BITM(_count) (BIT(_count) - 1)
- #define BITS(_shift, _count) (BITM(_count) << _shift)
- #define AR7240_REG_MASK_CTRL 0x00
- #define AR7240_MASK_CTRL_REVISION_M BITM(8)
- #define AR7240_MASK_CTRL_VERSION_M BITM(8)
- #define AR7240_MASK_CTRL_VERSION_S 8
- #define AR7240_MASK_CTRL_VERSION_AR7240 0x01
- #define AR7240_MASK_CTRL_VERSION_AR934X 0x02
- #define AR7240_MASK_CTRL_SOFT_RESET BIT(31)
- #define AR7240_REG_MAC_ADDR0 0x20
- #define AR7240_REG_MAC_ADDR1 0x24
- #define AR7240_REG_FLOOD_MASK 0x2c
- #define AR7240_FLOOD_MASK_BROAD_TO_CPU BIT(26)
- #define AR7240_REG_GLOBAL_CTRL 0x30
- #define AR7240_GLOBAL_CTRL_MTU_M BITM(12)
- #define AR7240_REG_VTU 0x0040
- #define AR7240_VTU_OP BITM(3)
- #define AR7240_VTU_OP_NOOP 0x0
- #define AR7240_VTU_OP_FLUSH 0x1
- #define AR7240_VTU_OP_LOAD 0x2
- #define AR7240_VTU_OP_PURGE 0x3
- #define AR7240_VTU_OP_REMOVE_PORT 0x4
- #define AR7240_VTU_ACTIVE BIT(3)
- #define AR7240_VTU_FULL BIT(4)
- #define AR7240_VTU_PORT BITS(8, 4)
- #define AR7240_VTU_PORT_S 8
- #define AR7240_VTU_VID BITS(16, 12)
- #define AR7240_VTU_VID_S 16
- #define AR7240_VTU_PRIO BITS(28, 3)
- #define AR7240_VTU_PRIO_S 28
- #define AR7240_VTU_PRIO_EN BIT(31)
- #define AR7240_REG_VTU_DATA 0x0044
- #define AR7240_VTUDATA_MEMBER BITS(0, 10)
- #define AR7240_VTUDATA_VALID BIT(11)
- #define AR7240_REG_ATU 0x50
- #define AR7240_ATU_FLUSH_ALL 0x1
- #define AR7240_REG_AT_CTRL 0x5c
- #define AR7240_AT_CTRL_AGE_TIME BITS(0, 15)
- #define AR7240_AT_CTRL_AGE_EN BIT(17)
- #define AR7240_AT_CTRL_LEARN_CHANGE BIT(18)
- #define AR7240_AT_CTRL_RESERVED BIT(19)
- #define AR7240_AT_CTRL_ARP_EN BIT(20)
- #define AR7240_REG_TAG_PRIORITY 0x70
- #define AR7240_REG_SERVICE_TAG 0x74
- #define AR7240_SERVICE_TAG_M BITM(16)
- #define AR7240_REG_CPU_PORT 0x78
- #define AR7240_MIRROR_PORT_S 4
- #define AR7240_CPU_PORT_EN BIT(8)
- #define AR7240_REG_MIB_FUNCTION0 0x80
- #define AR7240_MIB_TIMER_M BITM(16)
- #define AR7240_MIB_AT_HALF_EN BIT(16)
- #define AR7240_MIB_BUSY BIT(17)
- #define AR7240_MIB_FUNC_S 24
- #define AR7240_MIB_FUNC_NO_OP 0x0
- #define AR7240_MIB_FUNC_FLUSH 0x1
- #define AR7240_MIB_FUNC_CAPTURE 0x3
- #define AR7240_REG_MDIO_CTRL 0x98
- #define AR7240_MDIO_CTRL_DATA_M BITM(16)
- #define AR7240_MDIO_CTRL_REG_ADDR_S 16
- #define AR7240_MDIO_CTRL_PHY_ADDR_S 21
- #define AR7240_MDIO_CTRL_CMD_WRITE 0
- #define AR7240_MDIO_CTRL_CMD_READ BIT(27)
- #define AR7240_MDIO_CTRL_MASTER_EN BIT(30)
- #define AR7240_MDIO_CTRL_BUSY BIT(31)
- #define AR7240_REG_PORT_BASE(_port) (0x100 + (_port) * 0x100)
- #define AR7240_REG_PORT_STATUS(_port) (AR7240_REG_PORT_BASE((_port)) + 0x00)
- #define AR7240_PORT_STATUS_SPEED_S 0
- #define AR7240_PORT_STATUS_SPEED_M BITM(2)
- #define AR7240_PORT_STATUS_SPEED_10 0
- #define AR7240_PORT_STATUS_SPEED_100 1
- #define AR7240_PORT_STATUS_SPEED_1000 2
- #define AR7240_PORT_STATUS_TXMAC BIT(2)
- #define AR7240_PORT_STATUS_RXMAC BIT(3)
- #define AR7240_PORT_STATUS_TXFLOW BIT(4)
- #define AR7240_PORT_STATUS_RXFLOW BIT(5)
- #define AR7240_PORT_STATUS_DUPLEX BIT(6)
- #define AR7240_PORT_STATUS_LINK_UP BIT(8)
- #define AR7240_PORT_STATUS_LINK_AUTO BIT(9)
- #define AR7240_PORT_STATUS_LINK_PAUSE BIT(10)
- #define AR7240_REG_PORT_CTRL(_port) (AR7240_REG_PORT_BASE((_port)) + 0x04)
- #define AR7240_PORT_CTRL_STATE_M BITM(3)
- #define AR7240_PORT_CTRL_STATE_DISABLED 0
- #define AR7240_PORT_CTRL_STATE_BLOCK 1
- #define AR7240_PORT_CTRL_STATE_LISTEN 2
- #define AR7240_PORT_CTRL_STATE_LEARN 3
- #define AR7240_PORT_CTRL_STATE_FORWARD 4
- #define AR7240_PORT_CTRL_LEARN_LOCK BIT(7)
- #define AR7240_PORT_CTRL_VLAN_MODE_S 8
- #define AR7240_PORT_CTRL_VLAN_MODE_KEEP 0
- #define AR7240_PORT_CTRL_VLAN_MODE_STRIP 1
- #define AR7240_PORT_CTRL_VLAN_MODE_ADD 2
- #define AR7240_PORT_CTRL_VLAN_MODE_DOUBLE_TAG 3
- #define AR7240_PORT_CTRL_IGMP_SNOOP BIT(10)
- #define AR7240_PORT_CTRL_HEADER BIT(11)
- #define AR7240_PORT_CTRL_MAC_LOOP BIT(12)
- #define AR7240_PORT_CTRL_SINGLE_VLAN BIT(13)
- #define AR7240_PORT_CTRL_LEARN BIT(14)
- #define AR7240_PORT_CTRL_DOUBLE_TAG BIT(15)
- #define AR7240_PORT_CTRL_MIRROR_TX BIT(16)
- #define AR7240_PORT_CTRL_MIRROR_RX BIT(17)
- #define AR7240_REG_PORT_VLAN(_port) (AR7240_REG_PORT_BASE((_port)) + 0x08)
- #define AR7240_PORT_VLAN_DEFAULT_ID_S 0
- #define AR7240_PORT_VLAN_DEST_PORTS_S 16
- #define AR7240_PORT_VLAN_MODE_S 30
- #define AR7240_PORT_VLAN_MODE_PORT_ONLY 0
- #define AR7240_PORT_VLAN_MODE_PORT_FALLBACK 1
- #define AR7240_PORT_VLAN_MODE_VLAN_ONLY 2
- #define AR7240_PORT_VLAN_MODE_SECURE 3
- #define AR7240_REG_STATS_BASE(_port) (0x20000 + (_port) * 0x100)
- #define AR7240_STATS_RXBROAD 0x00
- #define AR7240_STATS_RXPAUSE 0x04
- #define AR7240_STATS_RXMULTI 0x08
- #define AR7240_STATS_RXFCSERR 0x0c
- #define AR7240_STATS_RXALIGNERR 0x10
- #define AR7240_STATS_RXRUNT 0x14
- #define AR7240_STATS_RXFRAGMENT 0x18
- #define AR7240_STATS_RX64BYTE 0x1c
- #define AR7240_STATS_RX128BYTE 0x20
- #define AR7240_STATS_RX256BYTE 0x24
- #define AR7240_STATS_RX512BYTE 0x28
- #define AR7240_STATS_RX1024BYTE 0x2c
- #define AR7240_STATS_RX1518BYTE 0x30
- #define AR7240_STATS_RXMAXBYTE 0x34
- #define AR7240_STATS_RXTOOLONG 0x38
- #define AR7240_STATS_RXGOODBYTE 0x3c
- #define AR7240_STATS_RXBADBYTE 0x44
- #define AR7240_STATS_RXOVERFLOW 0x4c
- #define AR7240_STATS_FILTERED 0x50
- #define AR7240_STATS_TXBROAD 0x54
- #define AR7240_STATS_TXPAUSE 0x58
- #define AR7240_STATS_TXMULTI 0x5c
- #define AR7240_STATS_TXUNDERRUN 0x60
- #define AR7240_STATS_TX64BYTE 0x64
- #define AR7240_STATS_TX128BYTE 0x68
- #define AR7240_STATS_TX256BYTE 0x6c
- #define AR7240_STATS_TX512BYTE 0x70
- #define AR7240_STATS_TX1024BYTE 0x74
- #define AR7240_STATS_TX1518BYTE 0x78
- #define AR7240_STATS_TXMAXBYTE 0x7c
- #define AR7240_STATS_TXOVERSIZE 0x80
- #define AR7240_STATS_TXBYTE 0x84
- #define AR7240_STATS_TXCOLLISION 0x8c
- #define AR7240_STATS_TXABORTCOL 0x90
- #define AR7240_STATS_TXMULTICOL 0x94
- #define AR7240_STATS_TXSINGLECOL 0x98
- #define AR7240_STATS_TXEXCDEFER 0x9c
- #define AR7240_STATS_TXDEFER 0xa0
- #define AR7240_STATS_TXLATECOL 0xa4
- #define AR7240_PORT_CPU 0
- #define AR7240_NUM_PORTS 6
- #define AR7240_NUM_PHYS 5
- #define AR7240_PHY_ID1 0x004d
- #define AR7240_PHY_ID2 0xd041
- #define AR934X_PHY_ID1 0x004d
- #define AR934X_PHY_ID2 0xd042
- #define AR7240_MAX_VLANS 16
- #define AR934X_REG_OPER_MODE0 0x04
- #define AR934X_OPER_MODE0_MAC_GMII_EN BIT(6)
- #define AR934X_OPER_MODE0_PHY_MII_EN BIT(10)
- #define AR934X_REG_OPER_MODE1 0x08
- #define AR934X_REG_OPER_MODE1_PHY4_MII_EN BIT(28)
- #define AR934X_REG_PORT_BASE(_port) (0x100 + (_port) * 0x100)
- #define AR934X_REG_PORT_VLAN1(_port) (AR934X_REG_PORT_BASE((_port)) + 0x08)
- #define AR934X_PORT_VLAN1_DEFAULT_SVID_S 0
- #define AR934X_PORT_VLAN1_FORCE_DEFAULT_VID_EN BIT(12)
- #define AR934X_PORT_VLAN1_PORT_TLS_MODE BIT(13)
- #define AR934X_PORT_VLAN1_PORT_VLAN_PROP_EN BIT(14)
- #define AR934X_PORT_VLAN1_PORT_CLONE_EN BIT(15)
- #define AR934X_PORT_VLAN1_DEFAULT_CVID_S 16
- #define AR934X_PORT_VLAN1_FORCE_PORT_VLAN_EN BIT(28)
- #define AR934X_PORT_VLAN1_ING_PORT_PRI_S 29
- #define AR934X_REG_PORT_VLAN2(_port) (AR934X_REG_PORT_BASE((_port)) + 0x0c)
- #define AR934X_PORT_VLAN2_PORT_VID_MEM_S 16
- #define AR934X_PORT_VLAN2_8021Q_MODE_S 30
- #define AR934X_PORT_VLAN2_8021Q_MODE_PORT_ONLY 0
- #define AR934X_PORT_VLAN2_8021Q_MODE_PORT_FALLBACK 1
- #define AR934X_PORT_VLAN2_8021Q_MODE_VLAN_ONLY 2
- #define AR934X_PORT_VLAN2_8021Q_MODE_SECURE 3
- #define sw_to_ar7240(_dev) container_of(_dev, struct ar7240sw, swdev)
- struct ar7240sw {
- struct mii_bus *mii_bus;
- struct ag71xx_switch_platform_data *swdata;
- struct switch_dev swdev;
- int num_ports;
- u8 ver;
- bool vlan;
- u16 vlan_id[AR7240_MAX_VLANS];
- u8 vlan_table[AR7240_MAX_VLANS];
- u8 vlan_tagged;
- u16 pvid[AR7240_NUM_PORTS];
- char buf[80];
- };
- struct ar7240sw_hw_stat {
- char string[ETH_GSTRING_LEN];
- int sizeof_stat;
- int reg;
- };
- static DEFINE_MUTEX(reg_mutex);
- static inline int sw_is_ar7240(struct ar7240sw *as)
- {
- return as->ver == AR7240_MASK_CTRL_VERSION_AR7240;
- }
- static inline int sw_is_ar934x(struct ar7240sw *as)
- {
- return as->ver == AR7240_MASK_CTRL_VERSION_AR934X;
- }
- static inline u32 ar7240sw_port_mask(struct ar7240sw *as, int port)
- {
- return BIT(port);
- }
- static inline u32 ar7240sw_port_mask_all(struct ar7240sw *as)
- {
- return BIT(as->swdev.ports) - 1;
- }
- static inline u32 ar7240sw_port_mask_but(struct ar7240sw *as, int port)
- {
- return ar7240sw_port_mask_all(as) & ~BIT(port);
- }
- static inline u16 mk_phy_addr(u32 reg)
- {
- return 0x17 & ((reg >> 4) | 0x10);
- }
- static inline u16 mk_phy_reg(u32 reg)
- {
- return (reg << 1) & 0x1e;
- }
- static inline u16 mk_high_addr(u32 reg)
- {
- return (reg >> 7) & 0x1ff;
- }
- static u32 __ar7240sw_reg_read(struct mii_bus *mii, u32 reg)
- {
- unsigned long flags;
- u16 phy_addr;
- u16 phy_reg;
- u32 hi, lo;
- reg = (reg & 0xfffffffc) >> 2;
- phy_addr = mk_phy_addr(reg);
- phy_reg = mk_phy_reg(reg);
- local_irq_save(flags);
- ag71xx_mdio_mii_write(mii->priv, 0x1f, 0x10, mk_high_addr(reg));
- lo = (u32) ag71xx_mdio_mii_read(mii->priv, phy_addr, phy_reg);
- hi = (u32) ag71xx_mdio_mii_read(mii->priv, phy_addr, phy_reg + 1);
- local_irq_restore(flags);
- return (hi << 16) | lo;
- }
- static void __ar7240sw_reg_write(struct mii_bus *mii, u32 reg, u32 val)
- {
- unsigned long flags;
- u16 phy_addr;
- u16 phy_reg;
- reg = (reg & 0xfffffffc) >> 2;
- phy_addr = mk_phy_addr(reg);
- phy_reg = mk_phy_reg(reg);
- local_irq_save(flags);
- ag71xx_mdio_mii_write(mii->priv, 0x1f, 0x10, mk_high_addr(reg));
- ag71xx_mdio_mii_write(mii->priv, phy_addr, phy_reg + 1, (val >> 16));
- ag71xx_mdio_mii_write(mii->priv, phy_addr, phy_reg, (val & 0xffff));
- local_irq_restore(flags);
- }
- static u32 ar7240sw_reg_read(struct mii_bus *mii, u32 reg_addr)
- {
- u32 ret;
- mutex_lock(®_mutex);
- ret = __ar7240sw_reg_read(mii, reg_addr);
- mutex_unlock(®_mutex);
- return ret;
- }
- static void ar7240sw_reg_write(struct mii_bus *mii, u32 reg_addr, u32 reg_val)
- {
- mutex_lock(®_mutex);
- __ar7240sw_reg_write(mii, reg_addr, reg_val);
- mutex_unlock(®_mutex);
- }
- static u32 ar7240sw_reg_rmw(struct mii_bus *mii, u32 reg, u32 mask, u32 val)
- {
- u32 t;
- mutex_lock(®_mutex);
- t = __ar7240sw_reg_read(mii, reg);
- t &= ~mask;
- t |= val;
- __ar7240sw_reg_write(mii, reg, t);
- mutex_unlock(®_mutex);
- return t;
- }
- static void ar7240sw_reg_set(struct mii_bus *mii, u32 reg, u32 val)
- {
- u32 t;
- mutex_lock(®_mutex);
- t = __ar7240sw_reg_read(mii, reg);
- t |= val;
- __ar7240sw_reg_write(mii, reg, t);
- mutex_unlock(®_mutex);
- }
- static int __ar7240sw_reg_wait(struct mii_bus *mii, u32 reg, u32 mask, u32 val,
- unsigned timeout)
- {
- int i;
- for (i = 0; i < timeout; i++) {
- u32 t;
- t = __ar7240sw_reg_read(mii, reg);
- if ((t & mask) == val)
- return 0;
- msleep(1);
- }
- return -ETIMEDOUT;
- }
- static int ar7240sw_reg_wait(struct mii_bus *mii, u32 reg, u32 mask, u32 val,
- unsigned timeout)
- {
- int ret;
- mutex_lock(®_mutex);
- ret = __ar7240sw_reg_wait(mii, reg, mask, val, timeout);
- mutex_unlock(®_mutex);
- return ret;
- }
- u16 ar7240sw_phy_read(struct mii_bus *mii, unsigned phy_addr,
- unsigned reg_addr)
- {
- u32 t, val = 0xffff;
- int err;
- if (phy_addr >= AR7240_NUM_PHYS)
- return 0xffff;
- mutex_lock(®_mutex);
- t = (reg_addr << AR7240_MDIO_CTRL_REG_ADDR_S) |
- (phy_addr << AR7240_MDIO_CTRL_PHY_ADDR_S) |
- AR7240_MDIO_CTRL_MASTER_EN |
- AR7240_MDIO_CTRL_BUSY |
- AR7240_MDIO_CTRL_CMD_READ;
- __ar7240sw_reg_write(mii, AR7240_REG_MDIO_CTRL, t);
- err = __ar7240sw_reg_wait(mii, AR7240_REG_MDIO_CTRL,
- AR7240_MDIO_CTRL_BUSY, 0, 5);
- if (!err)
- val = __ar7240sw_reg_read(mii, AR7240_REG_MDIO_CTRL);
- mutex_unlock(®_mutex);
- return val & AR7240_MDIO_CTRL_DATA_M;
- }
- int ar7240sw_phy_write(struct mii_bus *mii, unsigned phy_addr,
- unsigned reg_addr, u16 reg_val)
- {
- u32 t;
- int ret;
- if (phy_addr >= AR7240_NUM_PHYS)
- return -EINVAL;
- mutex_lock(®_mutex);
- t = (phy_addr << AR7240_MDIO_CTRL_PHY_ADDR_S) |
- (reg_addr << AR7240_MDIO_CTRL_REG_ADDR_S) |
- AR7240_MDIO_CTRL_MASTER_EN |
- AR7240_MDIO_CTRL_BUSY |
- AR7240_MDIO_CTRL_CMD_WRITE |
- reg_val;
- __ar7240sw_reg_write(mii, AR7240_REG_MDIO_CTRL, t);
- ret = __ar7240sw_reg_wait(mii, AR7240_REG_MDIO_CTRL,
- AR7240_MDIO_CTRL_BUSY, 0, 5);
- mutex_unlock(®_mutex);
- return ret;
- }
- static void ar7240sw_disable_port(struct ar7240sw *as, unsigned port)
- {
- ar7240sw_reg_write(as->mii_bus, AR7240_REG_PORT_CTRL(port),
- AR7240_PORT_CTRL_STATE_DISABLED);
- }
- static void ar7240sw_setup(struct ar7240sw *as)
- {
- struct mii_bus *mii = as->mii_bus;
- /* Enable CPU port, and disable mirror port */
- ar7240sw_reg_write(mii, AR7240_REG_CPU_PORT,
- AR7240_CPU_PORT_EN |
- (15 << AR7240_MIRROR_PORT_S));
- /* Setup TAG priority mapping */
- ar7240sw_reg_write(mii, AR7240_REG_TAG_PRIORITY, 0xfa50);
- /* Enable ARP frame acknowledge, aging, MAC replacing */
- ar7240sw_reg_write(mii, AR7240_REG_AT_CTRL,
- AR7240_AT_CTRL_RESERVED |
- 0x2b /* 5 min age time */ |
- AR7240_AT_CTRL_AGE_EN |
- AR7240_AT_CTRL_ARP_EN |
- AR7240_AT_CTRL_LEARN_CHANGE);
- /* Enable Broadcast frames transmitted to the CPU */
- ar7240sw_reg_set(mii, AR7240_REG_FLOOD_MASK,
- AR7240_FLOOD_MASK_BROAD_TO_CPU);
- /* setup MTU */
- ar7240sw_reg_rmw(mii, AR7240_REG_GLOBAL_CTRL, AR7240_GLOBAL_CTRL_MTU_M,
- 1536);
- /* setup Service TAG */
- ar7240sw_reg_rmw(mii, AR7240_REG_SERVICE_TAG, AR7240_SERVICE_TAG_M, 0);
- }
- static int ar7240sw_reset(struct ar7240sw *as)
- {
- struct mii_bus *mii = as->mii_bus;
- int ret;
- int i;
- /* Set all ports to disabled state. */
- for (i = 0; i < AR7240_NUM_PORTS; i++)
- ar7240sw_disable_port(as, i);
- /* Wait for transmit queues to drain. */
- msleep(2);
- /* Reset the switch. */
- ar7240sw_reg_write(mii, AR7240_REG_MASK_CTRL,
- AR7240_MASK_CTRL_SOFT_RESET);
- ret = ar7240sw_reg_wait(mii, AR7240_REG_MASK_CTRL,
- AR7240_MASK_CTRL_SOFT_RESET, 0, 1000);
- ar7240sw_setup(as);
- return ret;
- }
- static void ar7240sw_setup_port(struct ar7240sw *as, unsigned port, u8 portmask)
- {
- struct mii_bus *mii = as->mii_bus;
- u32 ctrl;
- u32 vid, mode;
- ctrl = AR7240_PORT_CTRL_STATE_FORWARD | AR7240_PORT_CTRL_LEARN |
- AR7240_PORT_CTRL_SINGLE_VLAN;
- if (port == AR7240_PORT_CPU) {
- ar7240sw_reg_write(mii, AR7240_REG_PORT_STATUS(port),
- AR7240_PORT_STATUS_SPEED_1000 |
- AR7240_PORT_STATUS_TXFLOW |
- AR7240_PORT_STATUS_RXFLOW |
- AR7240_PORT_STATUS_TXMAC |
- AR7240_PORT_STATUS_RXMAC |
- AR7240_PORT_STATUS_DUPLEX);
- } else {
- ar7240sw_reg_write(mii, AR7240_REG_PORT_STATUS(port),
- AR7240_PORT_STATUS_LINK_AUTO);
- }
- /* Set the default VID for this port */
- if (as->vlan) {
- vid = as->vlan_id[as->pvid[port]];
- mode = AR7240_PORT_VLAN_MODE_SECURE;
- } else {
- vid = port;
- mode = AR7240_PORT_VLAN_MODE_PORT_ONLY;
- }
- if (as->vlan && (as->vlan_tagged & BIT(port))) {
- ctrl |= AR7240_PORT_CTRL_VLAN_MODE_ADD <<
- AR7240_PORT_CTRL_VLAN_MODE_S;
- } else {
- ctrl |= AR7240_PORT_CTRL_VLAN_MODE_STRIP <<
- AR7240_PORT_CTRL_VLAN_MODE_S;
- }
- if (!portmask) {
- if (port == AR7240_PORT_CPU)
- portmask = ar7240sw_port_mask_but(as, AR7240_PORT_CPU);
- else
- portmask = ar7240sw_port_mask(as, AR7240_PORT_CPU);
- }
- /* allow the port to talk to all other ports, but exclude its
- * own ID to prevent frames from being reflected back to the
- * port that they came from */
- portmask &= ar7240sw_port_mask_but(as, port);
- ar7240sw_reg_write(mii, AR7240_REG_PORT_CTRL(port), ctrl);
- if (sw_is_ar934x(as)) {
- u32 vlan1, vlan2;
- vlan1 = (vid << AR934X_PORT_VLAN1_DEFAULT_CVID_S);
- vlan2 = (portmask << AR934X_PORT_VLAN2_PORT_VID_MEM_S) |
- (mode << AR934X_PORT_VLAN2_8021Q_MODE_S);
- ar7240sw_reg_write(mii, AR934X_REG_PORT_VLAN1(port), vlan1);
- ar7240sw_reg_write(mii, AR934X_REG_PORT_VLAN2(port), vlan2);
- } else {
- u32 vlan;
- vlan = vid | (mode << AR7240_PORT_VLAN_MODE_S) |
- (portmask << AR7240_PORT_VLAN_DEST_PORTS_S);
- ar7240sw_reg_write(mii, AR7240_REG_PORT_VLAN(port), vlan);
- }
- }
- static int ar7240_set_addr(struct ar7240sw *as, u8 *addr)
- {
- struct mii_bus *mii = as->mii_bus;
- u32 t;
- t = (addr[4] << 8) | addr[5];
- ar7240sw_reg_write(mii, AR7240_REG_MAC_ADDR0, t);
- t = (addr[0] << 24) | (addr[1] << 16) | (addr[2] << 8) | addr[3];
- ar7240sw_reg_write(mii, AR7240_REG_MAC_ADDR1, t);
- return 0;
- }
- static int
- ar7240_set_vid(struct switch_dev *dev, const struct switch_attr *attr,
- struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- as->vlan_id[val->port_vlan] = val->value.i;
- return 0;
- }
- static int
- ar7240_get_vid(struct switch_dev *dev, const struct switch_attr *attr,
- struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- val->value.i = as->vlan_id[val->port_vlan];
- return 0;
- }
- static int
- ar7240_set_pvid(struct switch_dev *dev, int port, int vlan)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- /* make sure no invalid PVIDs get set */
- if (vlan >= dev->vlans)
- return -EINVAL;
- as->pvid[port] = vlan;
- return 0;
- }
- static int
- ar7240_get_pvid(struct switch_dev *dev, int port, int *vlan)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- *vlan = as->pvid[port];
- return 0;
- }
- static int
- ar7240_get_ports(struct switch_dev *dev, struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- u8 ports = as->vlan_table[val->port_vlan];
- int i;
- val->len = 0;
- for (i = 0; i < as->swdev.ports; i++) {
- struct switch_port *p;
- if (!(ports & (1 << i)))
- continue;
- p = &val->value.ports[val->len++];
- p->id = i;
- if (as->vlan_tagged & (1 << i))
- p->flags = (1 << SWITCH_PORT_FLAG_TAGGED);
- else
- p->flags = 0;
- }
- return 0;
- }
- static int
- ar7240_set_ports(struct switch_dev *dev, struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- u8 *vt = &as->vlan_table[val->port_vlan];
- int i, j;
- *vt = 0;
- for (i = 0; i < val->len; i++) {
- struct switch_port *p = &val->value.ports[i];
- if (p->flags & (1 << SWITCH_PORT_FLAG_TAGGED))
- as->vlan_tagged |= (1 << p->id);
- else {
- as->vlan_tagged &= ~(1 << p->id);
- as->pvid[p->id] = val->port_vlan;
- /* make sure that an untagged port does not
- * appear in other vlans */
- for (j = 0; j < AR7240_MAX_VLANS; j++) {
- if (j == val->port_vlan)
- continue;
- as->vlan_table[j] &= ~(1 << p->id);
- }
- }
- *vt |= 1 << p->id;
- }
- return 0;
- }
- static int
- ar7240_set_vlan(struct switch_dev *dev, const struct switch_attr *attr,
- struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- as->vlan = !!val->value.i;
- return 0;
- }
- static int
- ar7240_get_vlan(struct switch_dev *dev, const struct switch_attr *attr,
- struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- val->value.i = as->vlan;
- return 0;
- }
- static const char *
- ar7240_speed_str(u32 status)
- {
- u32 speed;
- speed = (status >> AR7240_PORT_STATUS_SPEED_S) &
- AR7240_PORT_STATUS_SPEED_M;
- switch (speed) {
- case AR7240_PORT_STATUS_SPEED_10:
- return "10baseT";
- case AR7240_PORT_STATUS_SPEED_100:
- return "100baseT";
- case AR7240_PORT_STATUS_SPEED_1000:
- return "1000baseT";
- }
- return "unknown";
- }
- static int
- ar7240_port_get_link(struct switch_dev *dev, const struct switch_attr *attr,
- struct switch_val *val)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- struct mii_bus *mii = as->mii_bus;
- u32 len;
- u32 status;
- int port;
- port = val->port_vlan;
- memset(as->buf, '\0', sizeof(as->buf));
- status = ar7240sw_reg_read(mii, AR7240_REG_PORT_STATUS(port));
- if (status & AR7240_PORT_STATUS_LINK_UP) {
- len = snprintf(as->buf, sizeof(as->buf),
- "port:%d link:up speed:%s %s-duplex %s%s%s",
- port,
- ar7240_speed_str(status),
- (status & AR7240_PORT_STATUS_DUPLEX) ?
- "full" : "half",
- (status & AR7240_PORT_STATUS_TXFLOW) ?
- "txflow ": "",
- (status & AR7240_PORT_STATUS_RXFLOW) ?
- "rxflow " : "",
- (status & AR7240_PORT_STATUS_LINK_AUTO) ?
- "auto ": "");
- } else {
- len = snprintf(as->buf, sizeof(as->buf),
- "port:%d link:down", port);
- }
- val->value.s = as->buf;
- val->len = len;
- return 0;
- }
- static void
- ar7240_vtu_op(struct ar7240sw *as, u32 op, u32 val)
- {
- struct mii_bus *mii = as->mii_bus;
- if (ar7240sw_reg_wait(mii, AR7240_REG_VTU, AR7240_VTU_ACTIVE, 0, 5))
- return;
- if ((op & AR7240_VTU_OP) == AR7240_VTU_OP_LOAD) {
- val &= AR7240_VTUDATA_MEMBER;
- val |= AR7240_VTUDATA_VALID;
- ar7240sw_reg_write(mii, AR7240_REG_VTU_DATA, val);
- }
- op |= AR7240_VTU_ACTIVE;
- ar7240sw_reg_write(mii, AR7240_REG_VTU, op);
- }
- static int
- ar7240_hw_apply(struct switch_dev *dev)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- u8 portmask[AR7240_NUM_PORTS];
- int i, j;
- /* flush all vlan translation unit entries */
- ar7240_vtu_op(as, AR7240_VTU_OP_FLUSH, 0);
- memset(portmask, 0, sizeof(portmask));
- if (as->vlan) {
- /* calculate the port destination masks and load vlans
- * into the vlan translation unit */
- for (j = 0; j < AR7240_MAX_VLANS; j++) {
- u8 vp = as->vlan_table[j];
- if (!vp)
- continue;
- for (i = 0; i < as->swdev.ports; i++) {
- u8 mask = (1 << i);
- if (vp & mask)
- portmask[i] |= vp & ~mask;
- }
- ar7240_vtu_op(as,
- AR7240_VTU_OP_LOAD |
- (as->vlan_id[j] << AR7240_VTU_VID_S),
- as->vlan_table[j]);
- }
- } else {
- /* vlan disabled:
- * isolate all ports, but connect them to the cpu port */
- for (i = 0; i < as->swdev.ports; i++) {
- if (i == AR7240_PORT_CPU)
- continue;
- portmask[i] = 1 << AR7240_PORT_CPU;
- portmask[AR7240_PORT_CPU] |= (1 << i);
- }
- }
- /* update the port destination mask registers and tag settings */
- for (i = 0; i < as->swdev.ports; i++)
- ar7240sw_setup_port(as, i, portmask[i]);
- return 0;
- }
- static int
- ar7240_reset_switch(struct switch_dev *dev)
- {
- struct ar7240sw *as = sw_to_ar7240(dev);
- ar7240sw_reset(as);
- return 0;
- }
- static struct switch_attr ar7240_globals[] = {
- {
- .type = SWITCH_TYPE_INT,
- .name = "enable_vlan",
- .description = "Enable VLAN mode",
- .set = ar7240_set_vlan,
- .get = ar7240_get_vlan,
- .max = 1
- },
- };
- static struct switch_attr ar7240_port[] = {
- {
- .type = SWITCH_TYPE_STRING,
- .name = "link",
- .description = "Get port link information",
- .max = 1,
- .set = NULL,
- .get = ar7240_port_get_link,
- },
- };
- static struct switch_attr ar7240_vlan[] = {
- {
- .type = SWITCH_TYPE_INT,
- .name = "vid",
- .description = "VLAN ID",
- .set = ar7240_set_vid,
- .get = ar7240_get_vid,
- .max = 4094,
- },
- };
- static const struct switch_dev_ops ar7240_ops = {
- .attr_global = {
- .attr = ar7240_globals,
- .n_attr = ARRAY_SIZE(ar7240_globals),
- },
- .attr_port = {
- .attr = ar7240_port,
- .n_attr = ARRAY_SIZE(ar7240_port),
- },
- .attr_vlan = {
- .attr = ar7240_vlan,
- .n_attr = ARRAY_SIZE(ar7240_vlan),
- },
- .get_port_pvid = ar7240_get_pvid,
- .set_port_pvid = ar7240_set_pvid,
- .get_vlan_ports = ar7240_get_ports,
- .set_vlan_ports = ar7240_set_ports,
- .apply_config = ar7240_hw_apply,
- .reset_switch = ar7240_reset_switch,
- };
- static struct ar7240sw *ar7240_probe(struct ag71xx *ag)
- {
- struct ag71xx_platform_data *pdata = ag71xx_get_pdata(ag);
- struct mii_bus *mii = ag->mii_bus;
- struct ar7240sw *as;
- struct switch_dev *swdev;
- u32 ctrl;
- u16 phy_id1;
- u16 phy_id2;
- int i;
- phy_id1 = ar7240sw_phy_read(mii, 0, MII_PHYSID1);
- phy_id2 = ar7240sw_phy_read(mii, 0, MII_PHYSID2);
- if ((phy_id1 != AR7240_PHY_ID1 || phy_id2 != AR7240_PHY_ID2) &&
- (phy_id1 != AR934X_PHY_ID1 || phy_id2 != AR934X_PHY_ID2)) {
- pr_err("%s: unknown phy id '%04x:%04x'\n",
- ag->dev->name, phy_id1, phy_id2);
- return NULL;
- }
- as = kzalloc(sizeof(*as), GFP_KERNEL);
- if (!as)
- return NULL;
- as->mii_bus = mii;
- as->swdata = pdata->switch_data;
- swdev = &as->swdev;
- ctrl = ar7240sw_reg_read(mii, AR7240_REG_MASK_CTRL);
- as->ver = (ctrl >> AR7240_MASK_CTRL_VERSION_S) &
- AR7240_MASK_CTRL_VERSION_M;
- if (sw_is_ar7240(as)) {
- swdev->name = "AR7240/AR9330 built-in switch";
- } else if (sw_is_ar934x(as)) {
- swdev->name = "AR934X built-in switch";
- if (pdata->phy_if_mode == PHY_INTERFACE_MODE_GMII) {
- ar7240sw_reg_set(mii, AR934X_REG_OPER_MODE0,
- AR934X_OPER_MODE0_MAC_GMII_EN);
- } else if (pdata->phy_if_mode == PHY_INTERFACE_MODE_MII) {
- ar7240sw_reg_set(mii, AR934X_REG_OPER_MODE0,
- AR934X_OPER_MODE0_PHY_MII_EN);
- } else {
- pr_err("%s: invalid PHY interface mode\n",
- ag->dev->name);
- goto err_free;
- }
- if (as->swdata->phy4_mii_en)
- ar7240sw_reg_set(mii, AR934X_REG_OPER_MODE1,
- AR934X_REG_OPER_MODE1_PHY4_MII_EN);
- } else {
- pr_err("%s: unsupported chip, ctrl=%08x\n",
- ag->dev->name, ctrl);
- goto err_free;
- }
- swdev->ports = AR7240_NUM_PORTS - 1;
- swdev->cpu_port = AR7240_PORT_CPU;
- swdev->vlans = AR7240_MAX_VLANS;
- swdev->ops = &ar7240_ops;
- if (register_switch(&as->swdev, ag->dev) < 0)
- goto err_free;
- pr_info("%s: Found an %s\n", ag->dev->name, swdev->name);
- /* initialize defaults */
- for (i = 0; i < AR7240_MAX_VLANS; i++)
- as->vlan_id[i] = i;
- as->vlan_table[0] = ar7240sw_port_mask_all(as);
- return as;
- err_free:
- kfree(as);
- return NULL;
- }
- static void link_function(struct work_struct *work) {
- struct ag71xx *ag = container_of(work, struct ag71xx, link_work.work);
- unsigned long flags;
- int i;
- int status = 0;
- for (i = 0; i < 4; i++) {
- int link = ar7240sw_phy_read(ag->mii_bus, i, MII_BMSR);
- if(link & BMSR_LSTATUS) {
- status = 1;
- break;
- }
- }
- spin_lock_irqsave(&ag->lock, flags);
- if(status != ag->link) {
- ag->link = status;
- ag71xx_link_adjust(ag);
- }
- spin_unlock_irqrestore(&ag->lock, flags);
- schedule_delayed_work(&ag->link_work, HZ / 2);
- }
- void ag71xx_ar7240_start(struct ag71xx *ag)
- {
- struct ar7240sw *as = ag->phy_priv;
- ar7240sw_reset(as);
- ag->speed = SPEED_1000;
- ag->duplex = 1;
- ar7240_set_addr(as, ag->dev->dev_addr);
- ar7240_hw_apply(&as->swdev);
- schedule_delayed_work(&ag->link_work, HZ / 10);
- }
- void ag71xx_ar7240_stop(struct ag71xx *ag)
- {
- cancel_delayed_work_sync(&ag->link_work);
- }
- int __devinit ag71xx_ar7240_init(struct ag71xx *ag)
- {
- struct ar7240sw *as;
- as = ar7240_probe(ag);
- if (!as)
- return -ENODEV;
- ag->phy_priv = as;
- ar7240sw_reset(as);
- INIT_DELAYED_WORK(&ag->link_work, link_function);
- return 0;
- }
- void ag71xx_ar7240_cleanup(struct ag71xx *ag)
- {
- struct ar7240sw *as = ag->phy_priv;
- if (!as)
- return;
- unregister_switch(&as->swdev);
- kfree(as);
- ag->phy_priv = NULL;
- }
|