Thread (8 messages) 8 messages, 2 authors, 2006-10-31

Re: [patch 3/3] Add tsi108 On Chip Ethernet device driver support

From: Zang Roy-r61911 <hidden>
Date: 2006-09-29 07:36:55
Also in: lkml

Possibly related (same subject, not in this thread)

On Thu, 2006-09-21 at 12:46, Jeff Garzik wrote:
Zang Roy-r61911 wrote:
quoted
+struct tsi108_prv_data {
+     void  __iomem *regs;    /* Base of normal regs */
+     void  __iomem *phyregs; /* Base of register bank used for PHY
access */
quoted
+     
+     int phy;                /* Index of PHY for this interface */
+     int irq_num;
+     int id;
+
+     struct timer_list timer;/* Timer that triggers the check phy
function */
quoted
+     int rxtail;             /* Next entry in rxring to read */
+     int rxhead;             /* Next entry in rxring to give a new
buffer */
quoted
+     int rxfree;             /* Number of free, allocated RX
buffers */
quoted
+
+     int rxpending;          /* Non-zero if there are still
descriptors
quoted
+                              * to be processed from a previous
descriptor
quoted
+                              * interrupt condition that has been
cleared */
quoted
+
+     int txtail;             /* Next TX descriptor to check status
on */
quoted
+     int txhead;             /* Next TX descriptor to use */
most of these should be unsigned, to prevent bugs.

quoted
+     /* Number of free TX descriptors.  This could be calculated
from
quoted
+      * rxhead and rxtail if one descriptor were left unused to
disambiguate
quoted
+      * full and empty conditions, but it's simpler to just keep
track
quoted
+      * explicitly. */
+
+     int txfree;
+
+     int phy_ok;             /* The PHY is currently powered on. */
+
+     /* PHY status (duplex is 1 for half, 2 for full,
+      * so that the default 0 indicates that neither has
+      * yet been configured). */
+
+     int link_up;
+     int speed;
+     int duplex;
+
+     tx_desc *txring;
+     rx_desc *rxring;
+     struct sk_buff *txskbs[TSI108_TXRING_LEN];
+     struct sk_buff *rxskbs[TSI108_RXRING_LEN];
+
+     dma_addr_t txdma, rxdma;
+
+     /* txlock nests in misclock and phy_lock */
+
+     spinlock_t txlock, misclock;
+
+     /* stats is used to hold the upper bits of each hardware
counter,
quoted
+      * and tmpstats is used to hold the full values for returning
+      * to the caller of get_stats().  They must be separate in
case
quoted
+      * an overflow interrupt occurs before the stats are consumed.
+      */
+
+     struct net_device_stats stats;
+     struct net_device_stats tmpstats;
+
+     /* These stats are kept separate in hardware, thus require
individual
quoted
+      * fields for handling carry.  They are combined in get_stats.
+      */
+
+     unsigned long rx_fcs;   /* Add to rx_frame_errors */
+     unsigned long rx_short_fcs;     /* Add to rx_frame_errors */
+     unsigned long rx_long_fcs;      /* Add to rx_frame_errors */
+     unsigned long rx_underruns;     /* Add to rx_length_errors */
+     unsigned long rx_overruns;      /* Add to rx_length_errors */
+
+     unsigned long tx_coll_abort;    /* Add to
tx_aborted_errors/collisions */
quoted
+     unsigned long tx_pause_drop;    /* Add to tx_aborted_errors */
+
+     unsigned long mc_hash[16];
+};
+
+/* Structure for a device driver */
+
+static struct platform_driver tsi_eth_driver = {
+     .probe = tsi108_init_one,
+     .remove = tsi108_ether_remove,
+     .driver = {
+             .name = "tsi-ethernet",
+     },
+};
+
+static void tsi108_timed_checker(unsigned long dev_ptr);
+
+static void dump_eth_one(struct net_device *dev)
+{
+     struct tsi108_prv_data *data = netdev_priv(dev);
+
+     printk("Dumping %s...\n", dev->name);
+     printk("intstat %x intmask %x phy_ok %d"
+            " link %d speed %d duplex %d\n",
+            TSI108_ETH_READ_REG(TSI108_EC_INTSTAT),
+            TSI108_ETH_READ_REG(TSI108_EC_INTMASK), data->phy_ok,
+            data->link_up, data->speed, data->duplex);
+
+     printk("TX: head %d, tail %d, free %d, stat %x, estat %x, err
%x\n",
quoted
+            data->txhead, data->txtail, data->txfree,
+            TSI108_ETH_READ_REG(TSI108_EC_TXSTAT),
+            TSI108_ETH_READ_REG(TSI108_EC_TXESTAT),
+            TSI108_ETH_READ_REG(TSI108_EC_TXERR));
+
+     printk("RX: head %d, tail %d, free %d, stat %x,"
+            " estat %x, err %x, pending %d\n\n",
+            data->rxhead, data->rxtail, data->rxfree,
+            TSI108_ETH_READ_REG(TSI108_EC_RXSTAT),
+            TSI108_ETH_READ_REG(TSI108_EC_RXESTAT),
+            TSI108_ETH_READ_REG(TSI108_EC_RXERR), data->rxpending);
+}
+
+/* Synchronization is needed between the thread and up/down events.
+ * Note that the PHY is accessed through the same registers for
both
quoted
+ * interfaces, so this can't be made interface-specific.
+ */
+
+static DEFINE_SPINLOCK(phy_lock);
you should have a chip structure, that contains two structs (one for 
each interface/port)
Could you interpret the chip structure in more detail?
Need I create two net_device struct for each port?
Thanks.
Roy
Keyboard shortcuts
hback out one level
jnext message in thread
kprevious message in thread
ldrill in
Escclose help / fold thread tree
?toggle this help