1 /*
   2  * sfe_util.c: general ethernet mac driver framework version 2.6
   3  *
   4  * Copyright (c) 2002-2008 Masayuki Murayama.  All rights reserved.
   5  *
   6  * Redistribution and use in source and binary forms, with or without
   7  * modification, are permitted provided that the following conditions are met:
   8  *
   9  * 1. Redistributions of source code must retain the above copyright notice,
  10  *    this list of conditions and the following disclaimer.
  11  *
  12  * 2. Redistributions in binary form must reproduce the above copyright notice,
  13  *    this list of conditions and the following disclaimer in the documentation
  14  *    and/or other materials provided with the distribution.
  15  *
  16  * 3. Neither the name of the author nor the names of its contributors may be
  17  *    used to endorse or promote products derived from this software without
  18  *    specific prior written permission.
  19  *
  20  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  21  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  22  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
  23  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
  24  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
  25  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
  26  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
  27  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
  28  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  29  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  30  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
  31  * DAMAGE.
  32  */
  33 
  34 /*
  35  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
  36  * Use is subject to license terms.
  37  */
  38 
  39 /*
  40  * System Header files.
  41  */
  42 #include <sys/types.h>
  43 #include <sys/conf.h>
  44 #include <sys/debug.h>
  45 #include <sys/kmem.h>
  46 #include <sys/vtrace.h>
  47 #include <sys/ethernet.h>
  48 #include <sys/modctl.h>
  49 #include <sys/errno.h>
  50 #include <sys/ddi.h>
  51 #include <sys/sunddi.h>
  52 #include <sys/stream.h>           /* required for MBLK* */
  53 #include <sys/strsun.h>           /* required for mionack() */
  54 #include <sys/byteorder.h>
  55 #include <sys/pci.h>
  56 #include <inet/common.h>
  57 #include <inet/led.h>
  58 #include <inet/mi.h>
  59 #include <inet/nd.h>
  60 #include <sys/crc32.h>
  61 
  62 #include <sys/note.h>
  63 
  64 #include "sfe_mii.h"
  65 #include "sfe_util.h"
  66 
  67 
  68 
  69 extern char ident[];
  70 
  71 /* Debugging support */
  72 #ifdef GEM_DEBUG_LEVEL
  73 static int gem_debug = GEM_DEBUG_LEVEL;
  74 #define DPRINTF(n, args)        if (gem_debug > (n)) cmn_err args
  75 #else
  76 #define DPRINTF(n, args)
  77 #undef ASSERT
  78 #define ASSERT(x)
  79 #endif
  80 
  81 #define IOC_LINESIZE    0x40    /* Is it right for amd64? */
  82 
  83 /*
  84  * Useful macros and typedefs
  85  */
  86 #define ROUNDUP(x, a)   (((x) + (a) - 1) & ~((a) - 1))
  87 
  88 #define GET_NET16(p)    ((((uint8_t *)(p))[0] << 8)| ((uint8_t *)(p))[1])
  89 #define GET_ETHERTYPE(p)        GET_NET16(((uint8_t *)(p)) + ETHERADDRL*2)
  90 
  91 #define GET_IPTYPEv4(p) (((uint8_t *)(p))[sizeof (struct ether_header) + 9])
  92 #define GET_IPTYPEv6(p) (((uint8_t *)(p))[sizeof (struct ether_header) + 6])
  93 
  94 
  95 #ifndef INT32_MAX
  96 #define INT32_MAX       0x7fffffff
  97 #endif
  98 
  99 #define VTAG_OFF        (ETHERADDRL*2)
 100 #ifndef VTAG_SIZE
 101 #define VTAG_SIZE       4
 102 #endif
 103 #ifndef VTAG_TPID
 104 #define VTAG_TPID       0x8100U
 105 #endif
 106 
 107 #define GET_TXBUF(dp, sn)       \
 108         &(dp)->tx_buf[SLOT((dp)->tx_slots_base + (sn), (dp)->gc.gc_tx_buf_size)]
 109 
 110 #ifndef offsetof
 111 #define offsetof(t, m)  ((long)&(((t *) 0)->m))
 112 #endif
 113 #define TXFLAG_VTAG(flag)       \
 114         (((flag) & GEM_TXFLAG_VTAG) >> GEM_TXFLAG_VTAG_SHIFT)
 115 
 116 #define MAXPKTBUF(dp)   \
 117         ((dp)->mtu + sizeof (struct ether_header) + VTAG_SIZE + ETHERFCSL)
 118 
 119 #define WATCH_INTERVAL_FAST     drv_usectohz(100*1000)  /* 100mS */
 120 #define BOOLEAN(x)      ((x) != 0)
 121 
 122 /*
 123  * Macros to distinct chip generation.
 124  */
 125 
 126 /*
 127  * Private functions
 128  */
 129 static void gem_mii_start(struct gem_dev *);
 130 static void gem_mii_stop(struct gem_dev *);
 131 
 132 /* local buffer management */
 133 static void gem_nd_setup(struct gem_dev *dp);
 134 static void gem_nd_cleanup(struct gem_dev *dp);
 135 static int gem_alloc_memory(struct gem_dev *);
 136 static void gem_free_memory(struct gem_dev *);
 137 static void gem_init_rx_ring(struct gem_dev *);
 138 static void gem_init_tx_ring(struct gem_dev *);
 139 __INLINE__ static void gem_append_rxbuf(struct gem_dev *, struct rxbuf *);
 140 
 141 static void gem_tx_timeout(struct gem_dev *);
 142 static void gem_mii_link_watcher(struct gem_dev *dp);
 143 static int gem_mac_init(struct gem_dev *dp);
 144 static int gem_mac_start(struct gem_dev *dp);
 145 static int gem_mac_stop(struct gem_dev *dp, uint_t flags);
 146 static void gem_mac_ioctl(struct gem_dev *dp, queue_t *wq, mblk_t *mp);
 147 
 148 static  struct ether_addr       gem_etherbroadcastaddr = {
 149         0xff, 0xff, 0xff, 0xff, 0xff, 0xff
 150 };
 151 
 152 int gem_speed_value[] = {10, 100, 1000};
 153 
 154 /* ============================================================== */
 155 /*
 156  * Misc runtime routines
 157  */
 158 /* ============================================================== */
 159 /*
 160  * Ether CRC calculation according to 21143 data sheet
 161  */
 162 uint32_t
 163 gem_ether_crc_le(const uint8_t *addr, int len)
 164 {
 165         uint32_t        crc;
 166 
 167         CRC32(crc, addr, ETHERADDRL, 0xffffffffU, crc32_table);
 168         return (crc);
 169 }
 170 
 171 uint32_t
 172 gem_ether_crc_be(const uint8_t *addr, int len)
 173 {
 174         int             idx;
 175         int             bit;
 176         uint_t          data;
 177         uint32_t        crc;
 178 #define CRC32_POLY_BE   0x04c11db7
 179 
 180         crc = 0xffffffff;
 181         for (idx = 0; idx < len; idx++) {
 182                 for (data = *addr++, bit = 0; bit < 8; bit++, data >>= 1) {
 183                         crc = (crc << 1)
 184                             ^ ((((crc >> 31) ^ data) & 1) ? CRC32_POLY_BE : 0);
 185                 }
 186         }
 187         return (crc);
 188 #undef  CRC32_POLY_BE
 189 }
 190 
 191 int
 192 gem_prop_get_int(struct gem_dev *dp, char *prop_template, int def_val)
 193 {
 194         char    propname[32];
 195 
 196         (void) sprintf(propname, prop_template, dp->name);
 197 
 198         return (ddi_prop_get_int(DDI_DEV_T_ANY, dp->dip,
 199             DDI_PROP_DONTPASS, propname, def_val));
 200 }
 201 
 202 static int
 203 gem_population(uint32_t x)
 204 {
 205         int     i;
 206         int     cnt;
 207 
 208         cnt = 0;
 209         for (i = 0; i < 32; i++) {
 210                 if (x & (1 << i)) {
 211                         cnt++;
 212                 }
 213         }
 214         return (cnt);
 215 }
 216 
 217 #ifdef GEM_DEBUG_LEVEL
 218 #ifdef GEM_DEBUG_VLAN
 219 static void
 220 gem_dump_packet(struct gem_dev *dp, char *title, mblk_t *mp,
 221     boolean_t check_cksum)
 222 {
 223         char    msg[180];
 224         uint8_t buf[18+20+20];
 225         uint8_t *p;
 226         size_t  offset;
 227         uint_t  ethertype;
 228         uint_t  proto;
 229         uint_t  ipproto = 0;
 230         uint_t  iplen;
 231         uint_t  iphlen;
 232         uint_t  tcplen;
 233         uint_t  udplen;
 234         uint_t  cksum;
 235         int     rest;
 236         int     len;
 237         char    *bp;
 238         mblk_t  *tp;
 239         extern uint_t   ip_cksum(mblk_t *, int, uint32_t);
 240 
 241         msg[0] = 0;
 242         bp = msg;
 243 
 244         rest = sizeof (buf);
 245         offset = 0;
 246         for (tp = mp; tp; tp = tp->b_cont) {
 247                 len = tp->b_wptr - tp->b_rptr;
 248                 len = min(rest, len);
 249                 bcopy(tp->b_rptr, &buf[offset], len);
 250                 rest -= len;
 251                 offset += len;
 252                 if (rest == 0) {
 253                         break;
 254                 }
 255         }
 256 
 257         offset = 0;
 258         p = &buf[offset];
 259 
 260         /* ethernet address */
 261         sprintf(bp,
 262             "ether: %02x:%02x:%02x:%02x:%02x:%02x"
 263             " -> %02x:%02x:%02x:%02x:%02x:%02x",
 264             p[6], p[7], p[8], p[9], p[10], p[11],
 265             p[0], p[1], p[2], p[3], p[4], p[5]);
 266         bp = &msg[strlen(msg)];
 267 
 268         /* vlag tag and etherrtype */
 269         ethertype = GET_ETHERTYPE(p);
 270         if (ethertype == VTAG_TPID) {
 271                 sprintf(bp, " vtag:0x%04x", GET_NET16(&p[14]));
 272                 bp = &msg[strlen(msg)];
 273 
 274                 offset += VTAG_SIZE;
 275                 p = &buf[offset];
 276                 ethertype = GET_ETHERTYPE(p);
 277         }
 278         sprintf(bp, " type:%04x", ethertype);
 279         bp = &msg[strlen(msg)];
 280 
 281         /* ethernet packet length */
 282         sprintf(bp, " mblklen:%d", msgdsize(mp));
 283         bp = &msg[strlen(msg)];
 284         if (mp->b_cont) {
 285                 sprintf(bp, "(");
 286                 bp = &msg[strlen(msg)];
 287                 for (tp = mp; tp; tp = tp->b_cont) {
 288                         if (tp == mp) {
 289                                 sprintf(bp, "%d", tp->b_wptr - tp->b_rptr);
 290                         } else {
 291                                 sprintf(bp, "+%d", tp->b_wptr - tp->b_rptr);
 292                         }
 293                         bp = &msg[strlen(msg)];
 294                 }
 295                 sprintf(bp, ")");
 296                 bp = &msg[strlen(msg)];
 297         }
 298 
 299         if (ethertype != ETHERTYPE_IP) {
 300                 goto x;
 301         }
 302 
 303         /* ip address */
 304         offset += sizeof (struct ether_header);
 305         p = &buf[offset];
 306         ipproto = p[9];
 307         iplen = GET_NET16(&p[2]);
 308         sprintf(bp, ", ip: %d.%d.%d.%d -> %d.%d.%d.%d proto:%d iplen:%d",
 309             p[12], p[13], p[14], p[15],
 310             p[16], p[17], p[18], p[19],
 311             ipproto, iplen);
 312         bp = (void *)&msg[strlen(msg)];
 313 
 314         iphlen = (p[0] & 0xf) * 4;
 315 
 316         /* cksum for psuedo header */
 317         cksum = *(uint16_t *)&p[12];
 318         cksum += *(uint16_t *)&p[14];
 319         cksum += *(uint16_t *)&p[16];
 320         cksum += *(uint16_t *)&p[18];
 321         cksum += BE_16(ipproto);
 322 
 323         /* tcp or udp protocol header */
 324         offset += iphlen;
 325         p = &buf[offset];
 326         if (ipproto == IPPROTO_TCP) {
 327                 tcplen = iplen - iphlen;
 328                 sprintf(bp, ", tcp: len:%d cksum:%x",
 329                     tcplen, GET_NET16(&p[16]));
 330                 bp = (void *)&msg[strlen(msg)];
 331 
 332                 if (check_cksum) {
 333                         cksum += BE_16(tcplen);
 334                         cksum = (uint16_t)ip_cksum(mp, offset, cksum);
 335                         sprintf(bp, " (%s)",
 336                             (cksum == 0 || cksum == 0xffff) ? "ok" : "ng");
 337                         bp = (void *)&msg[strlen(msg)];
 338                 }
 339         } else if (ipproto == IPPROTO_UDP) {
 340                 udplen = GET_NET16(&p[4]);
 341                 sprintf(bp, ", udp: len:%d cksum:%x",
 342                     udplen, GET_NET16(&p[6]));
 343                 bp = (void *)&msg[strlen(msg)];
 344 
 345                 if (GET_NET16(&p[6]) && check_cksum) {
 346                         cksum += *(uint16_t *)&p[4];
 347                         cksum = (uint16_t)ip_cksum(mp, offset, cksum);
 348                         sprintf(bp, " (%s)",
 349                             (cksum == 0 || cksum == 0xffff) ? "ok" : "ng");
 350                         bp = (void *)&msg[strlen(msg)];
 351                 }
 352         }
 353 x:
 354         cmn_err(CE_CONT, "!%s: %s: %s", dp->name, title, msg);
 355 }
 356 #endif /* GEM_DEBUG_VLAN */
 357 #endif /* GEM_DEBUG_LEVEL */
 358 
 359 /* ============================================================== */
 360 /*
 361  * IO cache flush
 362  */
 363 /* ============================================================== */
 364 __INLINE__ void
 365 gem_rx_desc_dma_sync(struct gem_dev *dp, int head, int nslot, int how)
 366 {
 367         int     n;
 368         int     m;
 369         int     rx_desc_unit_shift = dp->gc.gc_rx_desc_unit_shift;
 370 
 371         /* sync active descriptors */
 372         if (rx_desc_unit_shift < 0 || nslot == 0) {
 373                 /* no rx descriptor ring */
 374                 return;
 375         }
 376 
 377         n = dp->gc.gc_rx_ring_size - head;
 378         if ((m = nslot - n) > 0) {
 379                 (void) ddi_dma_sync(dp->desc_dma_handle,
 380                     (off_t)0,
 381                     (size_t)(m << rx_desc_unit_shift),
 382                     how);
 383                 nslot = n;
 384         }
 385 
 386         (void) ddi_dma_sync(dp->desc_dma_handle,
 387             (off_t)(head << rx_desc_unit_shift),
 388             (size_t)(nslot << rx_desc_unit_shift),
 389             how);
 390 }
 391 
 392 __INLINE__ void
 393 gem_tx_desc_dma_sync(struct gem_dev *dp, int head, int nslot, int how)
 394 {
 395         int     n;
 396         int     m;
 397         int     tx_desc_unit_shift = dp->gc.gc_tx_desc_unit_shift;
 398 
 399         /* sync active descriptors */
 400         if (tx_desc_unit_shift < 0 || nslot == 0) {
 401                 /* no tx descriptor ring */
 402                 return;
 403         }
 404 
 405         n = dp->gc.gc_tx_ring_size - head;
 406         if ((m = nslot - n) > 0) {
 407                 (void) ddi_dma_sync(dp->desc_dma_handle,
 408                     (off_t)(dp->tx_ring_dma - dp->rx_ring_dma),
 409                     (size_t)(m << tx_desc_unit_shift),
 410                     how);
 411                 nslot = n;
 412         }
 413 
 414         (void) ddi_dma_sync(dp->desc_dma_handle,
 415             (off_t)((head << tx_desc_unit_shift)
 416             + (dp->tx_ring_dma - dp->rx_ring_dma)),
 417             (size_t)(nslot << tx_desc_unit_shift),
 418             how);
 419 }
 420 
 421 static void
 422 gem_rx_start_default(struct gem_dev *dp, int head, int nslot)
 423 {
 424         gem_rx_desc_dma_sync(dp,
 425             SLOT(head, dp->gc.gc_rx_ring_size), nslot,
 426             DDI_DMA_SYNC_FORDEV);
 427 }
 428 
 429 /* ============================================================== */
 430 /*
 431  * Buffer management
 432  */
 433 /* ============================================================== */
 434 static void
 435 gem_dump_txbuf(struct gem_dev *dp, int level, const char *title)
 436 {
 437         cmn_err(level,
 438             "!%s: %s: tx_active: %d[%d] %d[%d] (+%d), "
 439             "tx_softq: %d[%d] %d[%d] (+%d), "
 440             "tx_free: %d[%d] %d[%d] (+%d), "
 441             "tx_desc: %d[%d] %d[%d] (+%d), "
 442             "intr: %d[%d] (+%d), ",
 443             dp->name, title,
 444             dp->tx_active_head,
 445             SLOT(dp->tx_active_head, dp->gc.gc_tx_buf_size),
 446             dp->tx_active_tail,
 447             SLOT(dp->tx_active_tail, dp->gc.gc_tx_buf_size),
 448             dp->tx_active_tail - dp->tx_active_head,
 449             dp->tx_softq_head,
 450             SLOT(dp->tx_softq_head, dp->gc.gc_tx_buf_size),
 451             dp->tx_softq_tail,
 452             SLOT(dp->tx_softq_tail, dp->gc.gc_tx_buf_size),
 453             dp->tx_softq_tail - dp->tx_softq_head,
 454             dp->tx_free_head,
 455             SLOT(dp->tx_free_head, dp->gc.gc_tx_buf_size),
 456             dp->tx_free_tail,
 457             SLOT(dp->tx_free_tail, dp->gc.gc_tx_buf_size),
 458             dp->tx_free_tail - dp->tx_free_head,
 459             dp->tx_desc_head,
 460             SLOT(dp->tx_desc_head, dp->gc.gc_tx_ring_size),
 461             dp->tx_desc_tail,
 462             SLOT(dp->tx_desc_tail, dp->gc.gc_tx_ring_size),
 463             dp->tx_desc_tail - dp->tx_desc_head,
 464             dp->tx_desc_intr,
 465             SLOT(dp->tx_desc_intr, dp->gc.gc_tx_ring_size),
 466             dp->tx_desc_intr - dp->tx_desc_head);
 467 }
 468 
 469 static void
 470 gem_free_rxbuf(struct rxbuf *rbp)
 471 {
 472         struct gem_dev  *dp;
 473 
 474         dp = rbp->rxb_devp;
 475         ASSERT(mutex_owned(&dp->intrlock));
 476         rbp->rxb_next = dp->rx_buf_freelist;
 477         dp->rx_buf_freelist = rbp;
 478         dp->rx_buf_freecnt++;
 479 }
 480 
 481 /*
 482  * gem_get_rxbuf: supply a receive buffer which have been mapped into
 483  * DMA space.
 484  */
 485 struct rxbuf *
 486 gem_get_rxbuf(struct gem_dev *dp, int cansleep)
 487 {
 488         struct rxbuf            *rbp;
 489         uint_t                  count = 0;
 490         int                     i;
 491         int                     err;
 492 
 493         ASSERT(mutex_owned(&dp->intrlock));
 494 
 495         DPRINTF(3, (CE_CONT, "!gem_get_rxbuf: called freecnt:%d",
 496             dp->rx_buf_freecnt));
 497         /*
 498          * Get rx buffer management structure
 499          */
 500         rbp = dp->rx_buf_freelist;
 501         if (rbp) {
 502                 /* get one from the recycle list */
 503                 ASSERT(dp->rx_buf_freecnt > 0);
 504 
 505                 dp->rx_buf_freelist = rbp->rxb_next;
 506                 dp->rx_buf_freecnt--;
 507                 rbp->rxb_next = NULL;
 508                 return (rbp);
 509         }
 510 
 511         /*
 512          * Allocate a rx buffer management structure
 513          */
 514         rbp = kmem_zalloc(sizeof (*rbp), cansleep ? KM_SLEEP : KM_NOSLEEP);
 515         if (rbp == NULL) {
 516                 /* no memory */
 517                 return (NULL);
 518         }
 519 
 520         /*
 521          * Prepare a back pointer to the device structure which will be
 522          * refered on freeing the buffer later.
 523          */
 524         rbp->rxb_devp = dp;
 525 
 526         /* allocate a dma handle for rx data buffer */
 527         if ((err = ddi_dma_alloc_handle(dp->dip,
 528             &dp->gc.gc_dma_attr_rxbuf,
 529             (cansleep ? DDI_DMA_SLEEP : DDI_DMA_DONTWAIT),
 530             NULL, &rbp->rxb_dh)) != DDI_SUCCESS) {
 531 
 532                 cmn_err(CE_WARN,
 533                     "!%s: %s: ddi_dma_alloc_handle:1 failed, err=%d",
 534                     dp->name, __func__, err);
 535 
 536                 kmem_free(rbp, sizeof (struct rxbuf));
 537                 return (NULL);
 538         }
 539 
 540         /* allocate a bounce buffer for rx */
 541         if ((err = ddi_dma_mem_alloc(rbp->rxb_dh,
 542             ROUNDUP(dp->rx_buf_len, IOC_LINESIZE),
 543             &dp->gc.gc_buf_attr,
 544                 /*
 545                  * if the nic requires a header at the top of receive buffers,
 546                  * it may access the rx buffer randomly.
 547                  */
 548             (dp->gc.gc_rx_header_len > 0)
 549             ? DDI_DMA_CONSISTENT : DDI_DMA_STREAMING,
 550             cansleep ? DDI_DMA_SLEEP : DDI_DMA_DONTWAIT,
 551             NULL,
 552             &rbp->rxb_buf, &rbp->rxb_buf_len,
 553             &rbp->rxb_bah)) != DDI_SUCCESS) {
 554 
 555                 cmn_err(CE_WARN,
 556                     "!%s: %s: ddi_dma_mem_alloc: failed, err=%d",
 557                     dp->name, __func__, err);
 558 
 559                 ddi_dma_free_handle(&rbp->rxb_dh);
 560                 kmem_free(rbp, sizeof (struct rxbuf));
 561                 return (NULL);
 562         }
 563 
 564         /* Mapin the bounce buffer into the DMA space */
 565         if ((err = ddi_dma_addr_bind_handle(rbp->rxb_dh,
 566             NULL, rbp->rxb_buf, dp->rx_buf_len,
 567             ((dp->gc.gc_rx_header_len > 0)
 568             ?(DDI_DMA_RDWR | DDI_DMA_CONSISTENT)
 569             :(DDI_DMA_READ | DDI_DMA_STREAMING)),
 570             cansleep ? DDI_DMA_SLEEP : DDI_DMA_DONTWAIT,
 571             NULL,
 572             rbp->rxb_dmacookie,
 573             &count)) != DDI_DMA_MAPPED) {
 574 
 575                 ASSERT(err != DDI_DMA_INUSE);
 576                 DPRINTF(0, (CE_WARN,
 577                     "!%s: ddi_dma_addr_bind_handle: failed, err=%d",
 578                     dp->name, __func__, err));
 579 
 580                 /*
 581                  * we failed to allocate a dma resource
 582                  * for the rx bounce buffer.
 583                  */
 584                 ddi_dma_mem_free(&rbp->rxb_bah);
 585                 ddi_dma_free_handle(&rbp->rxb_dh);
 586                 kmem_free(rbp, sizeof (struct rxbuf));
 587                 return (NULL);
 588         }
 589 
 590         /* correct the rest of the DMA mapping */
 591         for (i = 1; i < count; i++) {
 592                 ddi_dma_nextcookie(rbp->rxb_dh, &rbp->rxb_dmacookie[i]);
 593         }
 594         rbp->rxb_nfrags = count;
 595 
 596         /* Now we successfully prepared an rx buffer */
 597         dp->rx_buf_allocated++;
 598 
 599         return (rbp);
 600 }
 601 
 602 /* ============================================================== */
 603 /*
 604  * memory resource management
 605  */
 606 /* ============================================================== */
 607 static int
 608 gem_alloc_memory(struct gem_dev *dp)
 609 {
 610         caddr_t                 ring;
 611         caddr_t                 buf;
 612         size_t                  req_size;
 613         size_t                  ring_len;
 614         size_t                  buf_len;
 615         ddi_dma_cookie_t        ring_cookie;
 616         ddi_dma_cookie_t        buf_cookie;
 617         uint_t                  count;
 618         int                     i;
 619         int                     err;
 620         struct txbuf            *tbp;
 621         int                     tx_buf_len;
 622         ddi_dma_attr_t          dma_attr_txbounce;
 623 
 624         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
 625 
 626         dp->desc_dma_handle = NULL;
 627         req_size = dp->rx_desc_size + dp->tx_desc_size + dp->gc.gc_io_area_size;
 628 
 629         if (req_size > 0) {
 630                 /*
 631                  * Alloc RX/TX descriptors and a io area.
 632                  */
 633                 if ((err = ddi_dma_alloc_handle(dp->dip,
 634                     &dp->gc.gc_dma_attr_desc,
 635                     DDI_DMA_SLEEP, NULL,
 636                     &dp->desc_dma_handle)) != DDI_SUCCESS) {
 637                         cmn_err(CE_WARN,
 638                             "!%s: %s: ddi_dma_alloc_handle failed: %d",
 639                             dp->name, __func__, err);
 640                         return (ENOMEM);
 641                 }
 642 
 643                 if ((err = ddi_dma_mem_alloc(dp->desc_dma_handle,
 644                     req_size, &dp->gc.gc_desc_attr,
 645                     DDI_DMA_CONSISTENT, DDI_DMA_SLEEP, NULL,
 646                     &ring, &ring_len,
 647                     &dp->desc_acc_handle)) != DDI_SUCCESS) {
 648                         cmn_err(CE_WARN,
 649                             "!%s: %s: ddi_dma_mem_alloc failed: "
 650                             "ret %d, request size: %d",
 651                             dp->name, __func__, err, (int)req_size);
 652                         ddi_dma_free_handle(&dp->desc_dma_handle);
 653                         return (ENOMEM);
 654                 }
 655 
 656                 if ((err = ddi_dma_addr_bind_handle(dp->desc_dma_handle,
 657                     NULL, ring, ring_len,
 658                     DDI_DMA_RDWR | DDI_DMA_CONSISTENT,
 659                     DDI_DMA_SLEEP, NULL,
 660                     &ring_cookie, &count)) != DDI_SUCCESS) {
 661                         ASSERT(err != DDI_DMA_INUSE);
 662                         cmn_err(CE_WARN,
 663                             "!%s: %s: ddi_dma_addr_bind_handle failed: %d",
 664                             dp->name, __func__, err);
 665                         ddi_dma_mem_free(&dp->desc_acc_handle);
 666                         ddi_dma_free_handle(&dp->desc_dma_handle);
 667                         return (ENOMEM);
 668                 }
 669                 ASSERT(count == 1);
 670 
 671                 /* set base of rx descriptor ring */
 672                 dp->rx_ring = ring;
 673                 dp->rx_ring_dma = ring_cookie.dmac_laddress;
 674 
 675                 /* set base of tx descriptor ring */
 676                 dp->tx_ring = dp->rx_ring + dp->rx_desc_size;
 677                 dp->tx_ring_dma = dp->rx_ring_dma + dp->rx_desc_size;
 678 
 679                 /* set base of io area */
 680                 dp->io_area = dp->tx_ring + dp->tx_desc_size;
 681                 dp->io_area_dma = dp->tx_ring_dma + dp->tx_desc_size;
 682         }
 683 
 684         /*
 685          * Prepare DMA resources for tx packets
 686          */
 687         ASSERT(dp->gc.gc_tx_buf_size > 0);
 688 
 689         /* Special dma attribute for tx bounce buffers */
 690         dma_attr_txbounce = dp->gc.gc_dma_attr_txbuf;
 691         dma_attr_txbounce.dma_attr_sgllen = 1;
 692         dma_attr_txbounce.dma_attr_align =
 693             max(dma_attr_txbounce.dma_attr_align, IOC_LINESIZE);
 694 
 695         /* Size for tx bounce buffers must be max tx packet size. */
 696         tx_buf_len = MAXPKTBUF(dp);
 697         tx_buf_len = ROUNDUP(tx_buf_len, IOC_LINESIZE);
 698 
 699         ASSERT(tx_buf_len >= ETHERMAX+ETHERFCSL);
 700 
 701         for (i = 0, tbp = dp->tx_buf;
 702             i < dp->gc.gc_tx_buf_size; i++, tbp++) {
 703 
 704                 /* setup bounce buffers for tx packets */
 705                 if ((err = ddi_dma_alloc_handle(dp->dip,
 706                     &dma_attr_txbounce,
 707                     DDI_DMA_SLEEP, NULL,
 708                     &tbp->txb_bdh)) != DDI_SUCCESS) {
 709 
 710                         cmn_err(CE_WARN,
 711                     "!%s: %s ddi_dma_alloc_handle for bounce buffer failed:"
 712                             " err=%d, i=%d",
 713                             dp->name, __func__, err, i);
 714                         goto err_alloc_dh;
 715                 }
 716 
 717                 if ((err = ddi_dma_mem_alloc(tbp->txb_bdh,
 718                     tx_buf_len,
 719                     &dp->gc.gc_buf_attr,
 720                     DDI_DMA_STREAMING, DDI_DMA_SLEEP, NULL,
 721                     &buf, &buf_len,
 722                     &tbp->txb_bah)) != DDI_SUCCESS) {
 723                         cmn_err(CE_WARN,
 724                     "!%s: %s: ddi_dma_mem_alloc for bounce buffer failed"
 725                             "ret %d, request size %d",
 726                             dp->name, __func__, err, tx_buf_len);
 727                         ddi_dma_free_handle(&tbp->txb_bdh);
 728                         goto err_alloc_dh;
 729                 }
 730 
 731                 if ((err = ddi_dma_addr_bind_handle(tbp->txb_bdh,
 732                     NULL, buf, buf_len,
 733                     DDI_DMA_WRITE | DDI_DMA_STREAMING,
 734                     DDI_DMA_SLEEP, NULL,
 735                     &buf_cookie, &count)) != DDI_SUCCESS) {
 736                                 ASSERT(err != DDI_DMA_INUSE);
 737                                 cmn_err(CE_WARN,
 738         "!%s: %s: ddi_dma_addr_bind_handle for bounce buffer failed: %d",
 739                                     dp->name, __func__, err);
 740                                 ddi_dma_mem_free(&tbp->txb_bah);
 741                                 ddi_dma_free_handle(&tbp->txb_bdh);
 742                                 goto err_alloc_dh;
 743                 }
 744                 ASSERT(count == 1);
 745                 tbp->txb_buf = buf;
 746                 tbp->txb_buf_dma = buf_cookie.dmac_laddress;
 747         }
 748 
 749         return (0);
 750 
 751 err_alloc_dh:
 752         if (dp->gc.gc_tx_buf_size > 0) {
 753                 while (i-- > 0) {
 754                         (void) ddi_dma_unbind_handle(dp->tx_buf[i].txb_bdh);
 755                         ddi_dma_mem_free(&dp->tx_buf[i].txb_bah);
 756                         ddi_dma_free_handle(&dp->tx_buf[i].txb_bdh);
 757                 }
 758         }
 759 
 760         if (dp->desc_dma_handle) {
 761                 (void) ddi_dma_unbind_handle(dp->desc_dma_handle);
 762                 ddi_dma_mem_free(&dp->desc_acc_handle);
 763                 ddi_dma_free_handle(&dp->desc_dma_handle);
 764                 dp->desc_dma_handle = NULL;
 765         }
 766 
 767         return (ENOMEM);
 768 }
 769 
 770 static void
 771 gem_free_memory(struct gem_dev *dp)
 772 {
 773         int             i;
 774         struct rxbuf    *rbp;
 775         struct txbuf    *tbp;
 776 
 777         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
 778 
 779         /* Free TX/RX descriptors and tx padding buffer */
 780         if (dp->desc_dma_handle) {
 781                 (void) ddi_dma_unbind_handle(dp->desc_dma_handle);
 782                 ddi_dma_mem_free(&dp->desc_acc_handle);
 783                 ddi_dma_free_handle(&dp->desc_dma_handle);
 784                 dp->desc_dma_handle = NULL;
 785         }
 786 
 787         /* Free dma handles for Tx */
 788         for (i = dp->gc.gc_tx_buf_size, tbp = dp->tx_buf; i--; tbp++) {
 789                 /* Free bounce buffer associated to each txbuf */
 790                 (void) ddi_dma_unbind_handle(tbp->txb_bdh);
 791                 ddi_dma_mem_free(&tbp->txb_bah);
 792                 ddi_dma_free_handle(&tbp->txb_bdh);
 793         }
 794 
 795         /* Free rx buffer */
 796         while ((rbp = dp->rx_buf_freelist) != NULL) {
 797 
 798                 ASSERT(dp->rx_buf_freecnt > 0);
 799 
 800                 dp->rx_buf_freelist = rbp->rxb_next;
 801                 dp->rx_buf_freecnt--;
 802 
 803                 /* release DMA mapping */
 804                 ASSERT(rbp->rxb_dh != NULL);
 805 
 806                 /* free dma handles for rx bbuf */
 807                 /* it has dma mapping always */
 808                 ASSERT(rbp->rxb_nfrags > 0);
 809                 (void) ddi_dma_unbind_handle(rbp->rxb_dh);
 810 
 811                 /* free the associated bounce buffer and dma handle */
 812                 ASSERT(rbp->rxb_bah != NULL);
 813                 ddi_dma_mem_free(&rbp->rxb_bah);
 814                 /* free the associated dma handle */
 815                 ddi_dma_free_handle(&rbp->rxb_dh);
 816 
 817                 /* free the base memory of rx buffer management */
 818                 kmem_free(rbp, sizeof (struct rxbuf));
 819         }
 820 }
 821 
 822 /* ============================================================== */
 823 /*
 824  * Rx/Tx descriptor slot management
 825  */
 826 /* ============================================================== */
 827 /*
 828  * Initialize an empty rx ring.
 829  */
 830 static void
 831 gem_init_rx_ring(struct gem_dev *dp)
 832 {
 833         int             i;
 834         int             rx_ring_size = dp->gc.gc_rx_ring_size;
 835 
 836         DPRINTF(1, (CE_CONT, "!%s: %s ring_size:%d, buf_max:%d",
 837             dp->name, __func__,
 838             rx_ring_size, dp->gc.gc_rx_buf_max));
 839 
 840         /* make a physical chain of rx descriptors */
 841         for (i = 0; i < rx_ring_size; i++) {
 842                 (*dp->gc.gc_rx_desc_init)(dp, i);
 843         }
 844         gem_rx_desc_dma_sync(dp, 0, rx_ring_size, DDI_DMA_SYNC_FORDEV);
 845 
 846         dp->rx_active_head = (seqnum_t)0;
 847         dp->rx_active_tail = (seqnum_t)0;
 848 
 849         ASSERT(dp->rx_buf_head == (struct rxbuf *)NULL);
 850         ASSERT(dp->rx_buf_tail == (struct rxbuf *)NULL);
 851 }
 852 
 853 /*
 854  * Prepare rx buffers and put them into the rx buffer/descriptor ring.
 855  */
 856 static void
 857 gem_prepare_rx_buf(struct gem_dev *dp)
 858 {
 859         int             i;
 860         int             nrbuf;
 861         struct rxbuf    *rbp;
 862 
 863         ASSERT(mutex_owned(&dp->intrlock));
 864 
 865         /* Now we have no active buffers in rx ring */
 866 
 867         nrbuf = min(dp->gc.gc_rx_ring_size, dp->gc.gc_rx_buf_max);
 868         for (i = 0; i < nrbuf; i++) {
 869                 if ((rbp = gem_get_rxbuf(dp, B_TRUE)) == NULL) {
 870                         break;
 871                 }
 872                 gem_append_rxbuf(dp, rbp);
 873         }
 874 
 875         gem_rx_desc_dma_sync(dp,
 876             0, dp->gc.gc_rx_ring_size, DDI_DMA_SYNC_FORDEV);
 877 }
 878 
 879 /*
 880  * Reclaim active rx buffers in rx buffer ring.
 881  */
 882 static void
 883 gem_clean_rx_buf(struct gem_dev *dp)
 884 {
 885         int             i;
 886         struct rxbuf    *rbp;
 887         int             rx_ring_size = dp->gc.gc_rx_ring_size;
 888 #ifdef GEM_DEBUG_LEVEL
 889         int             total;
 890 #endif
 891         ASSERT(mutex_owned(&dp->intrlock));
 892 
 893         DPRINTF(2, (CE_CONT, "!%s: %s: %d buffers are free",
 894             dp->name, __func__, dp->rx_buf_freecnt));
 895         /*
 896          * clean up HW descriptors
 897          */
 898         for (i = 0; i < rx_ring_size; i++) {
 899                 (*dp->gc.gc_rx_desc_clean)(dp, i);
 900         }
 901         gem_rx_desc_dma_sync(dp, 0, rx_ring_size, DDI_DMA_SYNC_FORDEV);
 902 
 903 #ifdef GEM_DEBUG_LEVEL
 904         total = 0;
 905 #endif
 906         /*
 907          * Reclaim allocated rx buffers
 908          */
 909         while ((rbp = dp->rx_buf_head) != NULL) {
 910 #ifdef GEM_DEBUG_LEVEL
 911                 total++;
 912 #endif
 913                 /* remove the first one from rx buffer list */
 914                 dp->rx_buf_head = rbp->rxb_next;
 915 
 916                 /* recycle the rxbuf */
 917                 gem_free_rxbuf(rbp);
 918         }
 919         dp->rx_buf_tail = (struct rxbuf *)NULL;
 920 
 921         DPRINTF(2, (CE_CONT,
 922             "!%s: %s: %d buffers freeed, total: %d free",
 923             dp->name, __func__, total, dp->rx_buf_freecnt));
 924 }
 925 
 926 /*
 927  * Initialize an empty transmit buffer/descriptor ring
 928  */
 929 static void
 930 gem_init_tx_ring(struct gem_dev *dp)
 931 {
 932         int             i;
 933         int             tx_buf_size = dp->gc.gc_tx_buf_size;
 934         int             tx_ring_size = dp->gc.gc_tx_ring_size;
 935 
 936         DPRINTF(2, (CE_CONT, "!%s: %s: ring_size:%d, buf_size:%d",
 937             dp->name, __func__,
 938             dp->gc.gc_tx_ring_size, dp->gc.gc_tx_buf_size));
 939 
 940         ASSERT(!dp->mac_active);
 941 
 942         /* initialize active list and free list */
 943         dp->tx_slots_base =
 944             SLOT(dp->tx_slots_base + dp->tx_softq_head, tx_buf_size);
 945         dp->tx_softq_tail -= dp->tx_softq_head;
 946         dp->tx_softq_head = (seqnum_t)0;
 947 
 948         dp->tx_active_head = dp->tx_softq_head;
 949         dp->tx_active_tail = dp->tx_softq_head;
 950 
 951         dp->tx_free_head   = dp->tx_softq_tail;
 952         dp->tx_free_tail   = dp->gc.gc_tx_buf_limit;
 953 
 954         dp->tx_desc_head = (seqnum_t)0;
 955         dp->tx_desc_tail = (seqnum_t)0;
 956         dp->tx_desc_intr = (seqnum_t)0;
 957 
 958         for (i = 0; i < tx_ring_size; i++) {
 959                 (*dp->gc.gc_tx_desc_init)(dp, i);
 960         }
 961         gem_tx_desc_dma_sync(dp, 0, tx_ring_size, DDI_DMA_SYNC_FORDEV);
 962 }
 963 
 964 __INLINE__
 965 static void
 966 gem_txbuf_free_dma_resources(struct txbuf *tbp)
 967 {
 968         if (tbp->txb_mp) {
 969                 freemsg(tbp->txb_mp);
 970                 tbp->txb_mp = NULL;
 971         }
 972         tbp->txb_nfrags = 0;
 973         tbp->txb_flag = 0;
 974 }
 975 #pragma inline(gem_txbuf_free_dma_resources)
 976 
 977 /*
 978  * reclaim active tx buffers and reset positions in tx rings.
 979  */
 980 static void
 981 gem_clean_tx_buf(struct gem_dev *dp)
 982 {
 983         int             i;
 984         seqnum_t        head;
 985         seqnum_t        tail;
 986         seqnum_t        sn;
 987         struct txbuf    *tbp;
 988         int             tx_ring_size = dp->gc.gc_tx_ring_size;
 989 #ifdef GEM_DEBUG_LEVEL
 990         int             err;
 991 #endif
 992 
 993         ASSERT(!dp->mac_active);
 994         ASSERT(dp->tx_busy == 0);
 995         ASSERT(dp->tx_softq_tail == dp->tx_free_head);
 996 
 997         /*
 998          * clean up all HW descriptors
 999          */
1000         for (i = 0; i < tx_ring_size; i++) {
1001                 (*dp->gc.gc_tx_desc_clean)(dp, i);
1002         }
1003         gem_tx_desc_dma_sync(dp, 0, tx_ring_size, DDI_DMA_SYNC_FORDEV);
1004 
1005         /* dequeue all active and loaded buffers */
1006         head = dp->tx_active_head;
1007         tail = dp->tx_softq_tail;
1008 
1009         ASSERT(dp->tx_free_head - head >= 0);
1010         tbp = GET_TXBUF(dp, head);
1011         for (sn = head; sn != tail; sn++) {
1012                 gem_txbuf_free_dma_resources(tbp);
1013                 ASSERT(tbp->txb_mp == NULL);
1014                 dp->stats.errxmt++;
1015                 tbp = tbp->txb_next;
1016         }
1017 
1018 #ifdef GEM_DEBUG_LEVEL
1019         /* ensure no dma resources for tx are not in use now */
1020         err = 0;
1021         while (sn != head + dp->gc.gc_tx_buf_size) {
1022                 if (tbp->txb_mp || tbp->txb_nfrags) {
1023                         DPRINTF(0, (CE_CONT,
1024                             "%s: %s: sn:%d[%d] mp:%p nfrags:%d",
1025                             dp->name, __func__,
1026                             sn, SLOT(sn, dp->gc.gc_tx_buf_size),
1027                             tbp->txb_mp, tbp->txb_nfrags));
1028                         err = 1;
1029                 }
1030                 sn++;
1031                 tbp = tbp->txb_next;
1032         }
1033 
1034         if (err) {
1035                 gem_dump_txbuf(dp, CE_WARN,
1036                     "gem_clean_tx_buf: tbp->txb_mp != NULL");
1037         }
1038 #endif
1039         /* recycle buffers, now no active tx buffers in the ring */
1040         dp->tx_free_tail += tail - head;
1041         ASSERT(dp->tx_free_tail == dp->tx_free_head + dp->gc.gc_tx_buf_limit);
1042 
1043         /* fix positions in tx buffer rings */
1044         dp->tx_active_head = dp->tx_free_head;
1045         dp->tx_active_tail = dp->tx_free_head;
1046         dp->tx_softq_head  = dp->tx_free_head;
1047         dp->tx_softq_tail  = dp->tx_free_head;
1048 }
1049 
1050 /*
1051  * Reclaim transmitted buffers from tx buffer/descriptor ring.
1052  */
1053 __INLINE__ int
1054 gem_reclaim_txbuf(struct gem_dev *dp)
1055 {
1056         struct txbuf    *tbp;
1057         uint_t          txstat;
1058         int             err = GEM_SUCCESS;
1059         seqnum_t        head;
1060         seqnum_t        tail;
1061         seqnum_t        sn;
1062         seqnum_t        desc_head;
1063         int             tx_ring_size = dp->gc.gc_tx_ring_size;
1064         uint_t (*tx_desc_stat)(struct gem_dev *dp,
1065             int slot, int ndesc) = dp->gc.gc_tx_desc_stat;
1066         clock_t         now;
1067 
1068         now = ddi_get_lbolt();
1069         if (now == (clock_t)0) {
1070                 /* make non-zero timestamp */
1071                 now--;
1072         }
1073 
1074         mutex_enter(&dp->xmitlock);
1075 
1076         head = dp->tx_active_head;
1077         tail = dp->tx_active_tail;
1078 
1079 #if GEM_DEBUG_LEVEL > 2
1080         if (head != tail) {
1081                 cmn_err(CE_CONT, "!%s: %s: "
1082                     "testing active_head:%d[%d], active_tail:%d[%d]",
1083                     dp->name, __func__,
1084                     head, SLOT(head, dp->gc.gc_tx_buf_size),
1085                     tail, SLOT(tail, dp->gc.gc_tx_buf_size));
1086         }
1087 #endif
1088 #ifdef DEBUG
1089         if (dp->tx_reclaim_busy == 0) {
1090                 /* check tx buffer management consistency */
1091                 ASSERT(dp->tx_free_tail - dp->tx_active_head
1092                     == dp->gc.gc_tx_buf_limit);
1093                 /* EMPTY */
1094         }
1095 #endif
1096         dp->tx_reclaim_busy++;
1097 
1098         /* sync all active HW descriptors */
1099         gem_tx_desc_dma_sync(dp,
1100             SLOT(dp->tx_desc_head, tx_ring_size),
1101             dp->tx_desc_tail - dp->tx_desc_head,
1102             DDI_DMA_SYNC_FORKERNEL);
1103 
1104         tbp = GET_TXBUF(dp, head);
1105         desc_head = dp->tx_desc_head;
1106         for (sn = head; sn != tail;
1107             dp->tx_active_head = (++sn), tbp = tbp->txb_next) {
1108                 int     ndescs;
1109 
1110                 ASSERT(tbp->txb_desc == desc_head);
1111 
1112                 ndescs = tbp->txb_ndescs;
1113                 if (ndescs == 0) {
1114                         /* skip errored descriptors */
1115                         continue;
1116                 }
1117                 txstat = (*tx_desc_stat)(dp,
1118                     SLOT(tbp->txb_desc, tx_ring_size), ndescs);
1119 
1120                 if (txstat == 0) {
1121                         /* not transmitted yet */
1122                         break;
1123                 }
1124 
1125                 if (!dp->tx_blocked && (tbp->txb_flag & GEM_TXFLAG_INTR)) {
1126                         dp->tx_blocked = now;
1127                 }
1128 
1129                 ASSERT(txstat & (GEM_TX_DONE | GEM_TX_ERR));
1130 
1131                 if (txstat & GEM_TX_ERR) {
1132                         err = GEM_FAILURE;
1133                         cmn_err(CE_WARN, "!%s: tx error at desc %d[%d]",
1134                             dp->name, sn, SLOT(sn, tx_ring_size));
1135                 }
1136 #if GEM_DEBUG_LEVEL > 4
1137                 if (now - tbp->txb_stime >= 50) {
1138                         cmn_err(CE_WARN, "!%s: tx delay while %d mS",
1139                             dp->name, (now - tbp->txb_stime)*10);
1140                 }
1141 #endif
1142                 /* free transmitted descriptors */
1143                 desc_head += ndescs;
1144         }
1145 
1146         if (dp->tx_desc_head != desc_head) {
1147                 /* we have reclaimed one or more tx buffers */
1148                 dp->tx_desc_head = desc_head;
1149 
1150                 /* If we passed the next interrupt position, update it */
1151                 if (desc_head - dp->tx_desc_intr > 0) {
1152                         dp->tx_desc_intr = desc_head;
1153                 }
1154         }
1155         mutex_exit(&dp->xmitlock);
1156 
1157         /* free dma mapping resources associated with transmitted tx buffers */
1158         tbp = GET_TXBUF(dp, head);
1159         tail = sn;
1160 #if GEM_DEBUG_LEVEL > 2
1161         if (head != tail) {
1162                 cmn_err(CE_CONT, "%s: freeing head:%d[%d], tail:%d[%d]",
1163                     __func__,
1164                     head, SLOT(head, dp->gc.gc_tx_buf_size),
1165                     tail, SLOT(tail, dp->gc.gc_tx_buf_size));
1166         }
1167 #endif
1168         for (sn = head; sn != tail; sn++, tbp = tbp->txb_next) {
1169                 gem_txbuf_free_dma_resources(tbp);
1170         }
1171 
1172         /* recycle the tx buffers */
1173         mutex_enter(&dp->xmitlock);
1174         if (--dp->tx_reclaim_busy == 0) {
1175                 /* we are the last thread who can update free tail */
1176 #if GEM_DEBUG_LEVEL > 4
1177                 /* check all resouces have been deallocated */
1178                 sn = dp->tx_free_tail;
1179                 tbp = GET_TXBUF(dp, new_tail);
1180                 while (sn != dp->tx_active_head + dp->gc.gc_tx_buf_limit) {
1181                         if (tbp->txb_nfrags) {
1182                                 /* in use */
1183                                 break;
1184                         }
1185                         ASSERT(tbp->txb_mp == NULL);
1186                         tbp = tbp->txb_next;
1187                         sn++;
1188                 }
1189                 ASSERT(dp->tx_active_head + dp->gc.gc_tx_buf_limit == sn);
1190 #endif
1191                 dp->tx_free_tail =
1192                     dp->tx_active_head + dp->gc.gc_tx_buf_limit;
1193         }
1194         if (!dp->mac_active) {
1195                 /* someone may be waiting for me. */
1196                 cv_broadcast(&dp->tx_drain_cv);
1197         }
1198 #if GEM_DEBUG_LEVEL > 2
1199         cmn_err(CE_CONT, "!%s: %s: called, "
1200             "free_head:%d free_tail:%d(+%d) added:%d",
1201             dp->name, __func__,
1202             dp->tx_free_head, dp->tx_free_tail,
1203             dp->tx_free_tail - dp->tx_free_head, tail - head);
1204 #endif
1205         mutex_exit(&dp->xmitlock);
1206 
1207         return (err);
1208 }
1209 #pragma inline(gem_reclaim_txbuf)
1210 
1211 
1212 /*
1213  * Make tx descriptors in out-of-order manner
1214  */
1215 static void
1216 gem_tx_load_descs_oo(struct gem_dev *dp,
1217         seqnum_t start_slot, seqnum_t end_slot, uint64_t flags)
1218 {
1219         seqnum_t        sn;
1220         struct txbuf    *tbp;
1221         int     tx_ring_size = dp->gc.gc_tx_ring_size;
1222         int     (*tx_desc_write)
1223             (struct gem_dev *dp, int slot,
1224             ddi_dma_cookie_t *dmacookie,
1225             int frags, uint64_t flag) = dp->gc.gc_tx_desc_write;
1226         clock_t now = ddi_get_lbolt();
1227 
1228         sn = start_slot;
1229         tbp = GET_TXBUF(dp, sn);
1230         do {
1231 #if GEM_DEBUG_LEVEL > 1
1232                 if (dp->tx_cnt < 100) {
1233                         dp->tx_cnt++;
1234                         flags |= GEM_TXFLAG_INTR;
1235                 }
1236 #endif
1237                 /* write a tx descriptor */
1238                 tbp->txb_desc = sn;
1239                 tbp->txb_ndescs = (*tx_desc_write)(dp,
1240                     SLOT(sn, tx_ring_size),
1241                     tbp->txb_dmacookie,
1242                     tbp->txb_nfrags, flags | tbp->txb_flag);
1243                 tbp->txb_stime = now;
1244                 ASSERT(tbp->txb_ndescs == 1);
1245 
1246                 flags = 0;
1247                 sn++;
1248                 tbp = tbp->txb_next;
1249         } while (sn != end_slot);
1250 }
1251 
1252 __INLINE__
1253 static size_t
1254 gem_setup_txbuf_copy(struct gem_dev *dp, mblk_t *mp, struct txbuf *tbp)
1255 {
1256         size_t                  min_pkt;
1257         caddr_t                 bp;
1258         size_t                  off;
1259         mblk_t                  *tp;
1260         size_t                  len;
1261         uint64_t                flag;
1262 
1263         ASSERT(tbp->txb_mp == NULL);
1264 
1265         /* we use bounce buffer for the packet */
1266         min_pkt = ETHERMIN;
1267         bp = tbp->txb_buf;
1268         off = 0;
1269         tp = mp;
1270 
1271         flag = tbp->txb_flag;
1272         if (flag & GEM_TXFLAG_SWVTAG) {
1273                 /* need to increase min packet size */
1274                 min_pkt += VTAG_SIZE;
1275                 ASSERT((flag & GEM_TXFLAG_VTAG) == 0);
1276         }
1277 
1278         /* copy the rest */
1279         for (; tp; tp = tp->b_cont) {
1280                 if ((len = (long)tp->b_wptr - (long)tp->b_rptr) > 0) {
1281                         bcopy(tp->b_rptr, &bp[off], len);
1282                         off += len;
1283                 }
1284         }
1285 
1286         if (off < min_pkt &&
1287             (min_pkt > ETHERMIN || !dp->gc.gc_tx_auto_pad)) {
1288                 /*
1289                  * Extend the packet to minimum packet size explicitly.
1290                  * For software vlan packets, we shouldn't use tx autopad
1291                  * function because nics may not be aware of vlan.
1292                  * we must keep 46 octet of payload even if we use vlan.
1293                  */
1294                 bzero(&bp[off], min_pkt - off);
1295                 off = min_pkt;
1296         }
1297 
1298         (void) ddi_dma_sync(tbp->txb_bdh, (off_t)0, off, DDI_DMA_SYNC_FORDEV);
1299 
1300         tbp->txb_dmacookie[0].dmac_laddress = tbp->txb_buf_dma;
1301         tbp->txb_dmacookie[0].dmac_size = off;
1302 
1303         DPRINTF(2, (CE_CONT,
1304             "!%s: %s: copy: addr:0x%llx len:0x%x, vtag:0x%04x, min_pkt:%d",
1305             dp->name, __func__,
1306             tbp->txb_dmacookie[0].dmac_laddress,
1307             tbp->txb_dmacookie[0].dmac_size,
1308             (flag & GEM_TXFLAG_VTAG) >> GEM_TXFLAG_VTAG_SHIFT,
1309             min_pkt));
1310 
1311         /* save misc info */
1312         tbp->txb_mp = mp;
1313         tbp->txb_nfrags = 1;
1314 #ifdef DEBUG_MULTIFRAGS
1315         if (dp->gc.gc_tx_max_frags >= 3 &&
1316             tbp->txb_dmacookie[0].dmac_size > 16*3) {
1317                 tbp->txb_dmacookie[1].dmac_laddress =
1318                     tbp->txb_dmacookie[0].dmac_laddress + 16;
1319                 tbp->txb_dmacookie[2].dmac_laddress =
1320                     tbp->txb_dmacookie[1].dmac_laddress + 16;
1321 
1322                 tbp->txb_dmacookie[2].dmac_size =
1323                     tbp->txb_dmacookie[0].dmac_size - 16*2;
1324                 tbp->txb_dmacookie[1].dmac_size = 16;
1325                 tbp->txb_dmacookie[0].dmac_size = 16;
1326                 tbp->txb_nfrags  = 3;
1327         }
1328 #endif
1329         return (off);
1330 }
1331 #pragma inline(gem_setup_txbuf_copy)
1332 
1333 __INLINE__
1334 static void
1335 gem_tx_start_unit(struct gem_dev *dp)
1336 {
1337         seqnum_t        head;
1338         seqnum_t        tail;
1339         struct txbuf    *tbp_head;
1340         struct txbuf    *tbp_tail;
1341 
1342         /* update HW descriptors from soft queue */
1343         ASSERT(mutex_owned(&dp->xmitlock));
1344         ASSERT(dp->tx_softq_head == dp->tx_active_tail);
1345 
1346         head = dp->tx_softq_head;
1347         tail = dp->tx_softq_tail;
1348 
1349         DPRINTF(1, (CE_CONT,
1350             "%s: %s: called, softq %d %d[+%d], desc %d %d[+%d]",
1351             dp->name, __func__, head, tail, tail - head,
1352             dp->tx_desc_head, dp->tx_desc_tail,
1353             dp->tx_desc_tail - dp->tx_desc_head));
1354 
1355         ASSERT(tail - head > 0);
1356 
1357         dp->tx_desc_tail = tail;
1358 
1359         tbp_head = GET_TXBUF(dp, head);
1360         tbp_tail = GET_TXBUF(dp, tail - 1);
1361 
1362         ASSERT(tbp_tail->txb_desc + tbp_tail->txb_ndescs == dp->tx_desc_tail);
1363 
1364         dp->gc.gc_tx_start(dp,
1365             SLOT(tbp_head->txb_desc, dp->gc.gc_tx_ring_size),
1366             tbp_tail->txb_desc + tbp_tail->txb_ndescs - tbp_head->txb_desc);
1367 
1368         /* advance softq head and active tail */
1369         dp->tx_softq_head = dp->tx_active_tail = tail;
1370 }
1371 #pragma inline(gem_tx_start_unit)
1372 
1373 #ifdef GEM_DEBUG_LEVEL
1374 static int gem_send_cnt[10];
1375 #endif
1376 #define PKT_MIN_SIZE    (sizeof (struct ether_header) + 10 + VTAG_SIZE)
1377 #define EHLEN   (sizeof (struct ether_header))
1378 /*
1379  * check ether packet type and ip protocol
1380  */
1381 static uint64_t
1382 gem_txbuf_options(struct gem_dev *dp, mblk_t *mp, uint8_t *bp)
1383 {
1384         mblk_t          *tp;
1385         ssize_t         len;
1386         uint_t          vtag;
1387         int             off;
1388         uint64_t        flag;
1389 
1390         flag = 0ULL;
1391 
1392         /*
1393          * prepare continuous header of the packet for protocol analysis
1394          */
1395         if ((long)mp->b_wptr - (long)mp->b_rptr < PKT_MIN_SIZE) {
1396                 /* we use work buffer to copy mblk */
1397                 for (tp = mp, off = 0;
1398                     tp && (off < PKT_MIN_SIZE);
1399                     tp = tp->b_cont, off += len) {
1400                         len = (long)tp->b_wptr - (long)tp->b_rptr;
1401                         len = min(len, PKT_MIN_SIZE - off);
1402                         bcopy(tp->b_rptr, &bp[off], len);
1403                 }
1404         } else {
1405                 /* we can use mblk without copy */
1406                 bp = mp->b_rptr;
1407         }
1408 
1409         /* process vlan tag for GLD v3 */
1410         if (GET_NET16(&bp[VTAG_OFF]) == VTAG_TPID) {
1411                 if (dp->misc_flag & GEM_VLAN_HARD) {
1412                         vtag = GET_NET16(&bp[VTAG_OFF + 2]);
1413                         ASSERT(vtag);
1414                         flag |= vtag << GEM_TXFLAG_VTAG_SHIFT;
1415                 } else {
1416                         flag |= GEM_TXFLAG_SWVTAG;
1417                 }
1418         }
1419         return (flag);
1420 }
1421 #undef EHLEN
1422 #undef PKT_MIN_SIZE
1423 /*
1424  * gem_send_common is an exported function because hw depend routines may
1425  * use it for sending control frames like setup frames for 2114x chipset.
1426  */
1427 mblk_t *
1428 gem_send_common(struct gem_dev *dp, mblk_t *mp_head, uint32_t flags)
1429 {
1430         int                     nmblk;
1431         int                     avail;
1432         mblk_t                  *tp;
1433         mblk_t                  *mp;
1434         int                     i;
1435         struct txbuf            *tbp;
1436         seqnum_t                head;
1437         uint64_t                load_flags;
1438         uint64_t                len_total = 0;
1439         uint32_t                bcast = 0;
1440         uint32_t                mcast = 0;
1441 
1442         ASSERT(mp_head != NULL);
1443 
1444         mp = mp_head;
1445         nmblk = 1;
1446         while ((mp = mp->b_next) != NULL) {
1447                 nmblk++;
1448         }
1449 #ifdef GEM_DEBUG_LEVEL
1450         gem_send_cnt[0]++;
1451         gem_send_cnt[min(nmblk, 9)]++;
1452 #endif
1453         /*
1454          * Aquire resources
1455          */
1456         mutex_enter(&dp->xmitlock);
1457         if (dp->mac_suspended) {
1458                 mutex_exit(&dp->xmitlock);
1459                 mp = mp_head;
1460                 while (mp) {
1461                         tp = mp->b_next;
1462                         freemsg(mp);
1463                         mp = tp;
1464                 }
1465                 return (NULL);
1466         }
1467 
1468         if (!dp->mac_active && (flags & GEM_SEND_CTRL) == 0) {
1469                 /* don't send data packets while mac isn't active */
1470                 /* XXX - should we discard packets? */
1471                 mutex_exit(&dp->xmitlock);
1472                 return (mp_head);
1473         }
1474 
1475         /* allocate free slots */
1476         head = dp->tx_free_head;
1477         avail = dp->tx_free_tail - head;
1478 
1479         DPRINTF(2, (CE_CONT,
1480             "!%s: %s: called, free_head:%d free_tail:%d(+%d) req:%d",
1481             dp->name, __func__,
1482             dp->tx_free_head, dp->tx_free_tail, avail, nmblk));
1483 
1484         avail = min(avail, dp->tx_max_packets);
1485 
1486         if (nmblk > avail) {
1487                 if (avail == 0) {
1488                         /* no resources; short cut */
1489                         DPRINTF(2, (CE_CONT, "!%s: no resources", __func__));
1490                         dp->tx_max_packets = max(dp->tx_max_packets - 1, 1);
1491                         goto done;
1492                 }
1493                 nmblk = avail;
1494         }
1495 
1496         dp->tx_free_head = head + nmblk;
1497         load_flags = ((dp->tx_busy++) == 0) ? GEM_TXFLAG_HEAD : 0;
1498 
1499         /* update last interrupt position if tx buffers exhaust.  */
1500         if (nmblk == avail) {
1501                 tbp = GET_TXBUF(dp, head + avail - 1);
1502                 tbp->txb_flag = GEM_TXFLAG_INTR;
1503                 dp->tx_desc_intr = head + avail;
1504         }
1505         mutex_exit(&dp->xmitlock);
1506 
1507         tbp = GET_TXBUF(dp, head);
1508 
1509         for (i = nmblk; i > 0; i--, tbp = tbp->txb_next) {
1510                 uint8_t         *bp;
1511                 uint64_t        txflag;
1512 
1513                 /* remove one from the mblk list */
1514                 ASSERT(mp_head != NULL);
1515                 mp = mp_head;
1516                 mp_head = mp_head->b_next;
1517                 mp->b_next = NULL;
1518 
1519                 /* statistics for non-unicast packets */
1520                 bp = mp->b_rptr;
1521                 if ((bp[0] & 1) && (flags & GEM_SEND_CTRL) == 0) {
1522                         if (bcmp(bp, gem_etherbroadcastaddr.ether_addr_octet,
1523                             ETHERADDRL) == 0) {
1524                                 bcast++;
1525                         } else {
1526                                 mcast++;
1527                         }
1528                 }
1529 
1530                 /* save misc info */
1531                 txflag = tbp->txb_flag;
1532                 txflag |= (flags & GEM_SEND_CTRL) << GEM_TXFLAG_PRIVATE_SHIFT;
1533                 txflag |= gem_txbuf_options(dp, mp, (uint8_t *)tbp->txb_buf);
1534                 tbp->txb_flag = txflag;
1535 
1536                 len_total += gem_setup_txbuf_copy(dp, mp, tbp);
1537         }
1538 
1539         (void) gem_tx_load_descs_oo(dp, head, head + nmblk, load_flags);
1540 
1541         /* Append the tbp at the tail of the active tx buffer list */
1542         mutex_enter(&dp->xmitlock);
1543 
1544         if ((--dp->tx_busy) == 0) {
1545                 /* extend the tail of softq, as new packets have been ready. */
1546                 dp->tx_softq_tail = dp->tx_free_head;
1547 
1548                 if (!dp->mac_active && (flags & GEM_SEND_CTRL) == 0) {
1549                         /*
1550                          * The device status has changed while we are
1551                          * preparing tx buf.
1552                          * As we are the last one that make tx non-busy.
1553                          * wake up someone who may wait for us.
1554                          */
1555                         cv_broadcast(&dp->tx_drain_cv);
1556                 } else {
1557                         ASSERT(dp->tx_softq_tail - dp->tx_softq_head > 0);
1558                         gem_tx_start_unit(dp);
1559                 }
1560         }
1561         dp->stats.obytes += len_total;
1562         dp->stats.opackets += nmblk;
1563         dp->stats.obcast += bcast;
1564         dp->stats.omcast += mcast;
1565 done:
1566         mutex_exit(&dp->xmitlock);
1567 
1568         return (mp_head);
1569 }
1570 
1571 /* ========================================================== */
1572 /*
1573  * error detection and restart routines
1574  */
1575 /* ========================================================== */
1576 int
1577 gem_restart_nic(struct gem_dev *dp, uint_t flags)
1578 {
1579         ASSERT(mutex_owned(&dp->intrlock));
1580 
1581         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
1582 #ifdef GEM_DEBUG_LEVEL
1583 #if GEM_DEBUG_LEVEL > 1
1584         gem_dump_txbuf(dp, CE_CONT, "gem_restart_nic");
1585 #endif
1586 #endif
1587 
1588         if (dp->mac_suspended) {
1589                 /* should we return GEM_FAILURE ? */
1590                 return (GEM_FAILURE);
1591         }
1592 
1593         /*
1594          * We should avoid calling any routines except xxx_chip_reset
1595          * when we are resuming the system.
1596          */
1597         if (dp->mac_active) {
1598                 if (flags & GEM_RESTART_KEEP_BUF) {
1599                         /* stop rx gracefully */
1600                         dp->rxmode &= ~RXMODE_ENABLE;
1601                         (void) (*dp->gc.gc_set_rx_filter)(dp);
1602                 }
1603                 (void) gem_mac_stop(dp, flags);
1604         }
1605 
1606         /* reset the chip. */
1607         if ((*dp->gc.gc_reset_chip)(dp) != GEM_SUCCESS) {
1608                 cmn_err(CE_WARN, "%s: %s: failed to reset chip",
1609                     dp->name, __func__);
1610                 goto err;
1611         }
1612 
1613         if (gem_mac_init(dp) != GEM_SUCCESS) {
1614                 goto err;
1615         }
1616 
1617         /* setup media mode if the link have been up */
1618         if (dp->mii_state == MII_STATE_LINKUP) {
1619                 if ((dp->gc.gc_set_media)(dp) != GEM_SUCCESS) {
1620                         goto err;
1621                 }
1622         }
1623 
1624         /* setup mac address and enable rx filter */
1625         dp->rxmode |= RXMODE_ENABLE;
1626         if ((*dp->gc.gc_set_rx_filter)(dp) != GEM_SUCCESS) {
1627                 goto err;
1628         }
1629 
1630         /*
1631          * XXX - a panic happened because of linkdown.
1632          * We must check mii_state here, because the link can be down just
1633          * before the restart event happen. If the link is down now,
1634          * gem_mac_start() will be called from gem_mii_link_check() when
1635          * the link become up later.
1636          */
1637         if (dp->mii_state == MII_STATE_LINKUP) {
1638                 /* restart the nic */
1639                 ASSERT(!dp->mac_active);
1640                 (void) gem_mac_start(dp);
1641         }
1642         return (GEM_SUCCESS);
1643 err:
1644         return (GEM_FAILURE);
1645 }
1646 
1647 
1648 static void
1649 gem_tx_timeout(struct gem_dev *dp)
1650 {
1651         clock_t         now;
1652         boolean_t       tx_sched;
1653         struct txbuf    *tbp;
1654 
1655         mutex_enter(&dp->intrlock);
1656 
1657         tx_sched = B_FALSE;
1658         now = ddi_get_lbolt();
1659 
1660         mutex_enter(&dp->xmitlock);
1661         if (!dp->mac_active || dp->mii_state != MII_STATE_LINKUP) {
1662                 mutex_exit(&dp->xmitlock);
1663                 goto schedule_next;
1664         }
1665         mutex_exit(&dp->xmitlock);
1666 
1667         /* reclaim transmitted buffers to check the trasmitter hangs or not. */
1668         if (gem_reclaim_txbuf(dp) != GEM_SUCCESS) {
1669                 /* tx error happened, reset transmitter in the chip */
1670                 (void) gem_restart_nic(dp, 0);
1671                 tx_sched = B_TRUE;
1672                 dp->tx_blocked = (clock_t)0;
1673 
1674                 goto schedule_next;
1675         }
1676 
1677         mutex_enter(&dp->xmitlock);
1678         /* check if the transmitter thread is stuck */
1679         if (dp->tx_active_head == dp->tx_active_tail) {
1680                 /* no tx buffer is loaded to the nic */
1681                 if (dp->tx_blocked &&
1682                     now - dp->tx_blocked > dp->gc.gc_tx_timeout_interval) {
1683                         gem_dump_txbuf(dp, CE_WARN,
1684                             "gem_tx_timeout: tx blocked");
1685                         tx_sched = B_TRUE;
1686                         dp->tx_blocked = (clock_t)0;
1687                 }
1688                 mutex_exit(&dp->xmitlock);
1689                 goto schedule_next;
1690         }
1691 
1692         tbp = GET_TXBUF(dp, dp->tx_active_head);
1693         if (now - tbp->txb_stime < dp->gc.gc_tx_timeout) {
1694                 mutex_exit(&dp->xmitlock);
1695                 goto schedule_next;
1696         }
1697         mutex_exit(&dp->xmitlock);
1698 
1699         gem_dump_txbuf(dp, CE_WARN, "gem_tx_timeout: tx timeout");
1700 
1701         /* discard untransmitted packet and restart tx.  */
1702         (void) gem_restart_nic(dp, GEM_RESTART_NOWAIT);
1703         tx_sched = B_TRUE;
1704         dp->tx_blocked = (clock_t)0;
1705 
1706 schedule_next:
1707         mutex_exit(&dp->intrlock);
1708 
1709         /* restart the downstream if needed */
1710         if (tx_sched) {
1711                 mac_tx_update(dp->mh);
1712         }
1713 
1714         DPRINTF(4, (CE_CONT,
1715             "!%s: blocked:%d active_head:%d active_tail:%d desc_intr:%d",
1716             dp->name, BOOLEAN(dp->tx_blocked),
1717             dp->tx_active_head, dp->tx_active_tail, dp->tx_desc_intr));
1718         dp->timeout_id =
1719             timeout((void (*)(void *))gem_tx_timeout,
1720             (void *)dp, dp->gc.gc_tx_timeout_interval);
1721 }
1722 
1723 /* ================================================================== */
1724 /*
1725  * Interrupt handler
1726  */
1727 /* ================================================================== */
1728 __INLINE__
1729 static void
1730 gem_append_rxbuf(struct gem_dev *dp, struct rxbuf *rbp_head)
1731 {
1732         struct rxbuf    *rbp;
1733         seqnum_t        tail;
1734         int             rx_ring_size = dp->gc.gc_rx_ring_size;
1735 
1736         ASSERT(rbp_head != NULL);
1737         ASSERT(mutex_owned(&dp->intrlock));
1738 
1739         DPRINTF(3, (CE_CONT, "!%s: %s: slot_head:%d, slot_tail:%d",
1740             dp->name, __func__, dp->rx_active_head, dp->rx_active_tail));
1741 
1742         /*
1743          * Add new buffers into active rx buffer list
1744          */
1745         if (dp->rx_buf_head == NULL) {
1746                 dp->rx_buf_head = rbp_head;
1747                 ASSERT(dp->rx_buf_tail == NULL);
1748         } else {
1749                 dp->rx_buf_tail->rxb_next = rbp_head;
1750         }
1751 
1752         tail = dp->rx_active_tail;
1753         for (rbp = rbp_head; rbp; rbp = rbp->rxb_next) {
1754                 /* need to notify the tail for the lower layer */
1755                 dp->rx_buf_tail = rbp;
1756 
1757                 dp->gc.gc_rx_desc_write(dp,
1758                     SLOT(tail, rx_ring_size),
1759                     rbp->rxb_dmacookie,
1760                     rbp->rxb_nfrags);
1761 
1762                 dp->rx_active_tail = tail = tail + 1;
1763         }
1764 }
1765 #pragma inline(gem_append_rxbuf)
1766 
1767 mblk_t *
1768 gem_get_packet_default(struct gem_dev *dp, struct rxbuf *rbp, size_t len)
1769 {
1770         int             rx_header_len = dp->gc.gc_rx_header_len;
1771         uint8_t         *bp;
1772         mblk_t          *mp;
1773 
1774         /* allocate a new mblk */
1775         if (mp = allocb(len + VTAG_SIZE, BPRI_MED)) {
1776                 ASSERT(mp->b_next == NULL);
1777                 ASSERT(mp->b_cont == NULL);
1778 
1779                 mp->b_rptr += VTAG_SIZE;
1780                 bp = mp->b_rptr;
1781                 mp->b_wptr = bp + len;
1782 
1783                 /*
1784                  * flush the range of the entire buffer to invalidate
1785                  * all of corresponding dirty entries in iocache.
1786                  */
1787                 (void) ddi_dma_sync(rbp->rxb_dh, rx_header_len,
1788                     0, DDI_DMA_SYNC_FORKERNEL);
1789 
1790                 bcopy(rbp->rxb_buf + rx_header_len, bp, len);
1791         }
1792         return (mp);
1793 }
1794 
1795 #ifdef GEM_DEBUG_LEVEL
1796 uint_t  gem_rx_pkts[17];
1797 #endif
1798 
1799 
1800 int
1801 gem_receive(struct gem_dev *dp)
1802 {
1803         uint64_t        len_total = 0;
1804         struct rxbuf    *rbp;
1805         mblk_t          *mp;
1806         int             cnt = 0;
1807         uint64_t        rxstat;
1808         struct rxbuf    *newbufs;
1809         struct rxbuf    **newbufs_tailp;
1810         mblk_t          *rx_head;
1811         mblk_t          **rx_tailp;
1812         int             rx_ring_size = dp->gc.gc_rx_ring_size;
1813         seqnum_t        active_head;
1814         uint64_t        (*rx_desc_stat)(struct gem_dev *dp,
1815             int slot, int ndesc);
1816         int             ethermin = ETHERMIN;
1817         int             ethermax = dp->mtu + sizeof (struct ether_header);
1818         int             rx_header_len = dp->gc.gc_rx_header_len;
1819 
1820         ASSERT(mutex_owned(&dp->intrlock));
1821 
1822         DPRINTF(3, (CE_CONT, "!%s: gem_receive: rx_buf_head:%p",
1823             dp->name, dp->rx_buf_head));
1824 
1825         rx_desc_stat  = dp->gc.gc_rx_desc_stat;
1826         newbufs_tailp = &newbufs;
1827         rx_tailp = &rx_head;
1828         for (active_head = dp->rx_active_head;
1829             (rbp = dp->rx_buf_head) != NULL; active_head++) {
1830                 int             len;
1831                 if (cnt == 0) {
1832                         cnt = max(dp->poll_pkt_delay*2, 10);
1833                         cnt = min(cnt,
1834                             dp->rx_active_tail - active_head);
1835                         gem_rx_desc_dma_sync(dp,
1836                             SLOT(active_head, rx_ring_size),
1837                             cnt,
1838                             DDI_DMA_SYNC_FORKERNEL);
1839                 }
1840 
1841                 if (rx_header_len > 0) {
1842                         (void) ddi_dma_sync(rbp->rxb_dh, 0,
1843                             rx_header_len, DDI_DMA_SYNC_FORKERNEL);
1844                 }
1845 
1846                 if (((rxstat = (*rx_desc_stat)(dp,
1847                     SLOT(active_head, rx_ring_size),
1848                     rbp->rxb_nfrags))
1849                     & (GEM_RX_DONE | GEM_RX_ERR)) == 0) {
1850                         /* not received yet */
1851                         break;
1852                 }
1853 
1854                 /* Remove the head of the rx buffer list */
1855                 dp->rx_buf_head = rbp->rxb_next;
1856                 cnt--;
1857 
1858 
1859                 if (rxstat & GEM_RX_ERR) {
1860                         goto next;
1861                 }
1862 
1863                 len = rxstat & GEM_RX_LEN;
1864                 DPRINTF(3, (CE_CONT, "!%s: %s: rxstat:0x%llx, len:0x%x",
1865                     dp->name, __func__, rxstat, len));
1866 
1867                 /*
1868                  * Copy the packet
1869                  */
1870                 if ((mp = dp->gc.gc_get_packet(dp, rbp, len)) == NULL) {
1871                         /* no memory, discard the packet */
1872                         dp->stats.norcvbuf++;
1873                         goto next;
1874                 }
1875 
1876                 /*
1877                  * Process VLAN tag
1878                  */
1879                 ethermin = ETHERMIN;
1880                 ethermax = dp->mtu + sizeof (struct ether_header);
1881                 if (GET_NET16(mp->b_rptr + VTAG_OFF) == VTAG_TPID) {
1882                         ethermax += VTAG_SIZE;
1883                 }
1884 
1885                 /* check packet size */
1886                 if (len < ethermin) {
1887                         dp->stats.errrcv++;
1888                         dp->stats.runt++;
1889                         freemsg(mp);
1890                         goto next;
1891                 }
1892 
1893                 if (len > ethermax) {
1894                         dp->stats.errrcv++;
1895                         dp->stats.frame_too_long++;
1896                         freemsg(mp);
1897                         goto next;
1898                 }
1899 
1900                 len_total += len;
1901 
1902 #ifdef GEM_DEBUG_VLAN
1903                 if (GET_ETHERTYPE(mp->b_rptr) == VTAG_TPID) {
1904                         gem_dump_packet(dp, (char *)__func__, mp, B_TRUE);
1905                 }
1906 #endif
1907                 /* append received packet to temporaly rx buffer list */
1908                 *rx_tailp = mp;
1909                 rx_tailp  = &mp->b_next;
1910 
1911                 if (mp->b_rptr[0] & 1) {
1912                         if (bcmp(mp->b_rptr,
1913                             gem_etherbroadcastaddr.ether_addr_octet,
1914                             ETHERADDRL) == 0) {
1915                                 dp->stats.rbcast++;
1916                         } else {
1917                                 dp->stats.rmcast++;
1918                         }
1919                 }
1920 next:
1921                 ASSERT(rbp != NULL);
1922 
1923                 /* append new one to temporal new buffer list */
1924                 *newbufs_tailp = rbp;
1925                 newbufs_tailp  = &rbp->rxb_next;
1926         }
1927 
1928         /* advance rx_active_head */
1929         if ((cnt = active_head - dp->rx_active_head) > 0) {
1930                 dp->stats.rbytes += len_total;
1931                 dp->stats.rpackets += cnt;
1932         }
1933         dp->rx_active_head = active_head;
1934 
1935         /* terminate the working list */
1936         *newbufs_tailp = NULL;
1937         *rx_tailp = NULL;
1938 
1939         if (dp->rx_buf_head == NULL) {
1940                 dp->rx_buf_tail = NULL;
1941         }
1942 
1943         DPRINTF(4, (CE_CONT, "%s: %s: cnt:%d, rx_head:%p",
1944             dp->name, __func__, cnt, rx_head));
1945 
1946         if (newbufs) {
1947                 /*
1948                  * fillfull rx list with new buffers
1949                  */
1950                 seqnum_t        head;
1951 
1952                 /* save current tail */
1953                 head = dp->rx_active_tail;
1954                 gem_append_rxbuf(dp, newbufs);
1955 
1956                 /* call hw depend start routine if we have. */
1957                 dp->gc.gc_rx_start(dp,
1958                     SLOT(head, rx_ring_size), dp->rx_active_tail - head);
1959         }
1960 
1961         if (rx_head) {
1962                 /*
1963                  * send up received packets
1964                  */
1965                 mutex_exit(&dp->intrlock);
1966                 mac_rx(dp->mh, NULL, rx_head);
1967                 mutex_enter(&dp->intrlock);
1968         }
1969 
1970 #ifdef GEM_DEBUG_LEVEL
1971         gem_rx_pkts[min(cnt, sizeof (gem_rx_pkts)/sizeof (uint_t)-1)]++;
1972 #endif
1973         return (cnt);
1974 }
1975 
1976 boolean_t
1977 gem_tx_done(struct gem_dev *dp)
1978 {
1979         boolean_t       tx_sched = B_FALSE;
1980 
1981         if (gem_reclaim_txbuf(dp) != GEM_SUCCESS) {
1982                 (void) gem_restart_nic(dp, GEM_RESTART_KEEP_BUF);
1983                 DPRINTF(2, (CE_CONT, "!%s: gem_tx_done: tx_desc: %d %d",
1984                     dp->name, dp->tx_active_head, dp->tx_active_tail));
1985                 tx_sched = B_TRUE;
1986                 goto x;
1987         }
1988 
1989         mutex_enter(&dp->xmitlock);
1990 
1991         /* XXX - we must not have any packets in soft queue */
1992         ASSERT(dp->tx_softq_head == dp->tx_softq_tail);
1993         /*
1994          * If we won't have chance to get more free tx buffers, and blocked,
1995          * it is worth to reschedule the downstream i.e. tx side.
1996          */
1997         ASSERT(dp->tx_desc_intr - dp->tx_desc_head >= 0);
1998         if (dp->tx_blocked && dp->tx_desc_intr == dp->tx_desc_head) {
1999                 /*
2000                  * As no further tx-done interrupts are scheduled, this
2001                  * is the last chance to kick tx side, which may be
2002                  * blocked now, otherwise the tx side never works again.
2003                  */
2004                 tx_sched = B_TRUE;
2005                 dp->tx_blocked = (clock_t)0;
2006                 dp->tx_max_packets =
2007                     min(dp->tx_max_packets + 2, dp->gc.gc_tx_buf_limit);
2008         }
2009 
2010         mutex_exit(&dp->xmitlock);
2011 
2012         DPRINTF(3, (CE_CONT, "!%s: %s: ret: blocked:%d",
2013             dp->name, __func__, BOOLEAN(dp->tx_blocked)));
2014 x:
2015         return (tx_sched);
2016 }
2017 
2018 static uint_t
2019 gem_intr(struct gem_dev *dp)
2020 {
2021         uint_t          ret;
2022 
2023         mutex_enter(&dp->intrlock);
2024         if (dp->mac_suspended) {
2025                 mutex_exit(&dp->intrlock);
2026                 return (DDI_INTR_UNCLAIMED);
2027         }
2028         dp->intr_busy = B_TRUE;
2029 
2030         ret = (*dp->gc.gc_interrupt)(dp);
2031 
2032         if (ret == DDI_INTR_UNCLAIMED) {
2033                 dp->intr_busy = B_FALSE;
2034                 mutex_exit(&dp->intrlock);
2035                 return (ret);
2036         }
2037 
2038         if (!dp->mac_active) {
2039                 cv_broadcast(&dp->tx_drain_cv);
2040         }
2041 
2042 
2043         dp->stats.intr++;
2044         dp->intr_busy = B_FALSE;
2045 
2046         mutex_exit(&dp->intrlock);
2047 
2048         if (ret & INTR_RESTART_TX) {
2049                 DPRINTF(4, (CE_CONT, "!%s: calling mac_tx_update", dp->name));
2050                 mac_tx_update(dp->mh);
2051                 ret &= ~INTR_RESTART_TX;
2052         }
2053         return (ret);
2054 }
2055 
2056 static void
2057 gem_intr_watcher(struct gem_dev *dp)
2058 {
2059         (void) gem_intr(dp);
2060 
2061         /* schedule next call of tu_intr_watcher */
2062         dp->intr_watcher_id =
2063             timeout((void (*)(void *))gem_intr_watcher, (void *)dp, 1);
2064 }
2065 
2066 /* ======================================================================== */
2067 /*
2068  * MII support routines
2069  */
2070 /* ======================================================================== */
2071 static void
2072 gem_choose_forcedmode(struct gem_dev *dp)
2073 {
2074         /* choose media mode */
2075         if (dp->anadv_1000fdx || dp->anadv_1000hdx) {
2076                 dp->speed = GEM_SPD_1000;
2077                 dp->full_duplex = dp->anadv_1000fdx;
2078         } else if (dp->anadv_100fdx || dp->anadv_100t4) {
2079                 dp->speed = GEM_SPD_100;
2080                 dp->full_duplex = B_TRUE;
2081         } else if (dp->anadv_100hdx) {
2082                 dp->speed = GEM_SPD_100;
2083                 dp->full_duplex = B_FALSE;
2084         } else {
2085                 dp->speed = GEM_SPD_10;
2086                 dp->full_duplex = dp->anadv_10fdx;
2087         }
2088 }
2089 
2090 uint16_t
2091 gem_mii_read(struct gem_dev *dp, uint_t reg)
2092 {
2093         if ((dp->mii_status & MII_STATUS_MFPRMBLSUPR) == 0) {
2094                 (*dp->gc.gc_mii_sync)(dp);
2095         }
2096         return ((*dp->gc.gc_mii_read)(dp, reg));
2097 }
2098 
2099 void
2100 gem_mii_write(struct gem_dev *dp, uint_t reg, uint16_t val)
2101 {
2102         if ((dp->mii_status & MII_STATUS_MFPRMBLSUPR) == 0) {
2103                 (*dp->gc.gc_mii_sync)(dp);
2104         }
2105         (*dp->gc.gc_mii_write)(dp, reg, val);
2106 }
2107 
2108 #define fc_cap_decode(x)        \
2109         ((((x) & MII_ABILITY_PAUSE) ? 1 : 0) |      \
2110         (((x) & MII_ABILITY_ASMPAUSE) ? 2 : 0))
2111 
2112 int
2113 gem_mii_config_default(struct gem_dev *dp)
2114 {
2115         uint16_t        mii_stat;
2116         uint16_t        val;
2117         static uint16_t fc_cap_encode[4] = {
2118                 0, /* none */
2119                 MII_ABILITY_PAUSE, /* symmetric */
2120                 MII_ABILITY_ASMPAUSE, /* tx */
2121                 MII_ABILITY_PAUSE | MII_ABILITY_ASMPAUSE, /* rx-symmetric */
2122         };
2123 
2124         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
2125 
2126         /*
2127          * Configure bits in advertisement register
2128          */
2129         mii_stat = dp->mii_status;
2130 
2131         DPRINTF(1, (CE_CONT, "!%s: %s: MII_STATUS reg:%b",
2132             dp->name, __func__, mii_stat, MII_STATUS_BITS));
2133 
2134         if ((mii_stat & MII_STATUS_ABILITY_TECH) == 0) {
2135                 /* it's funny */
2136                 cmn_err(CE_WARN, "!%s: wrong ability bits: mii_status:%b",
2137                     dp->name, mii_stat, MII_STATUS_BITS);
2138                 return (GEM_FAILURE);
2139         }
2140 
2141         /* Do not change the rest of the ability bits in the advert reg */
2142         val = gem_mii_read(dp, MII_AN_ADVERT) & ~MII_ABILITY_ALL;
2143 
2144         DPRINTF(0, (CE_CONT,
2145             "!%s: %s: 100T4:%d 100F:%d 100H:%d 10F:%d 10H:%d",
2146             dp->name, __func__,
2147             dp->anadv_100t4, dp->anadv_100fdx, dp->anadv_100hdx,
2148             dp->anadv_10fdx, dp->anadv_10hdx));
2149 
2150         if (dp->anadv_100t4) {
2151                 val |= MII_ABILITY_100BASE_T4;
2152         }
2153         if (dp->anadv_100fdx) {
2154                 val |= MII_ABILITY_100BASE_TX_FD;
2155         }
2156         if (dp->anadv_100hdx) {
2157                 val |= MII_ABILITY_100BASE_TX;
2158         }
2159         if (dp->anadv_10fdx) {
2160                 val |= MII_ABILITY_10BASE_T_FD;
2161         }
2162         if (dp->anadv_10hdx) {
2163                 val |= MII_ABILITY_10BASE_T;
2164         }
2165 
2166         /* set flow control capability */
2167         val |= fc_cap_encode[dp->anadv_flow_control];
2168 
2169         DPRINTF(0, (CE_CONT,
2170             "!%s: %s: setting MII_AN_ADVERT reg:%b, mii_mode:%d, fc:%d",
2171             dp->name, __func__, val, MII_ABILITY_BITS, dp->gc.gc_mii_mode,
2172             dp->anadv_flow_control));
2173 
2174         gem_mii_write(dp, MII_AN_ADVERT, val);
2175 
2176         if (mii_stat & MII_STATUS_XSTATUS) {
2177                 /*
2178                  * 1000Base-T GMII support
2179                  */
2180                 if (!dp->anadv_autoneg) {
2181                         /* enable manual configuration */
2182                         val = MII_1000TC_CFG_EN;
2183                 } else {
2184                         val = 0;
2185                         if (dp->anadv_1000fdx) {
2186                                 val |= MII_1000TC_ADV_FULL;
2187                         }
2188                         if (dp->anadv_1000hdx) {
2189                                 val |= MII_1000TC_ADV_HALF;
2190                         }
2191                 }
2192                 DPRINTF(0, (CE_CONT,
2193                     "!%s: %s: setting MII_1000TC reg:%b",
2194                     dp->name, __func__, val, MII_1000TC_BITS));
2195 
2196                 gem_mii_write(dp, MII_1000TC, val);
2197         }
2198 
2199         return (GEM_SUCCESS);
2200 }
2201 
2202 #define GEM_LINKUP(dp)          mac_link_update((dp)->mh, LINK_STATE_UP)
2203 #define GEM_LINKDOWN(dp)        mac_link_update((dp)->mh, LINK_STATE_DOWN)
2204 
2205 static uint8_t gem_fc_result[4 /* my cap */ ][4 /* lp cap */] = {
2206 /*       none   symm    tx      rx/symm */
2207 /* none */
2208         {FLOW_CONTROL_NONE,
2209                 FLOW_CONTROL_NONE,
2210                         FLOW_CONTROL_NONE,
2211                                 FLOW_CONTROL_NONE},
2212 /* sym */
2213         {FLOW_CONTROL_NONE,
2214                 FLOW_CONTROL_SYMMETRIC,
2215                         FLOW_CONTROL_NONE,
2216                                 FLOW_CONTROL_SYMMETRIC},
2217 /* tx */
2218         {FLOW_CONTROL_NONE,
2219                 FLOW_CONTROL_NONE,
2220                         FLOW_CONTROL_NONE,
2221                                 FLOW_CONTROL_TX_PAUSE},
2222 /* rx/symm */
2223         {FLOW_CONTROL_NONE,
2224                 FLOW_CONTROL_SYMMETRIC,
2225                         FLOW_CONTROL_RX_PAUSE,
2226                                 FLOW_CONTROL_SYMMETRIC},
2227 };
2228 
2229 static char *gem_fc_type[] = {
2230         "without",
2231         "with symmetric",
2232         "with tx",
2233         "with rx",
2234 };
2235 
2236 boolean_t
2237 gem_mii_link_check(struct gem_dev *dp)
2238 {
2239         uint16_t        old_mii_state;
2240         boolean_t       tx_sched = B_FALSE;
2241         uint16_t        status;
2242         uint16_t        advert;
2243         uint16_t        lpable;
2244         uint16_t        exp;
2245         uint16_t        ctl1000;
2246         uint16_t        stat1000;
2247         uint16_t        val;
2248         clock_t         now;
2249         clock_t         diff;
2250         int             linkdown_action;
2251         boolean_t       fix_phy = B_FALSE;
2252 
2253         now = ddi_get_lbolt();
2254         old_mii_state = dp->mii_state;
2255 
2256         DPRINTF(3, (CE_CONT, "!%s: %s: time:%d state:%d",
2257             dp->name, __func__, now, dp->mii_state));
2258 
2259         diff = now - dp->mii_last_check;
2260         dp->mii_last_check = now;
2261 
2262         /*
2263          * For NWAM, don't show linkdown state right
2264          * after the system boots
2265          */
2266         if (dp->linkup_delay > 0) {
2267                 if (dp->linkup_delay > diff) {
2268                         dp->linkup_delay -= diff;
2269                 } else {
2270                         /* link up timeout */
2271                         dp->linkup_delay = -1;
2272                 }
2273         }
2274 
2275 next_nowait:
2276         switch (dp->mii_state) {
2277         case MII_STATE_UNKNOWN:
2278                 /* power-up, DP83840 requires 32 sync bits */
2279                 (*dp->gc.gc_mii_sync)(dp);
2280                 goto reset_phy;
2281 
2282         case MII_STATE_RESETTING:
2283                 dp->mii_timer -= diff;
2284                 if (dp->mii_timer > 0) {
2285                         /* don't read phy registers in resetting */
2286                         dp->mii_interval = WATCH_INTERVAL_FAST;
2287                         goto next;
2288                 }
2289 
2290                 /* Timer expired, ensure reset bit is not set */
2291 
2292                 if (dp->mii_status & MII_STATUS_MFPRMBLSUPR) {
2293                         /* some phys need sync bits after reset */
2294                         (*dp->gc.gc_mii_sync)(dp);
2295                 }
2296                 val = gem_mii_read(dp, MII_CONTROL);
2297                 if (val & MII_CONTROL_RESET) {
2298                         cmn_err(CE_NOTE,
2299                             "!%s: time:%ld resetting phy not complete."
2300                             " mii_control:0x%b",
2301                             dp->name, ddi_get_lbolt(),
2302                             val, MII_CONTROL_BITS);
2303                 }
2304 
2305                 /* ensure neither isolated nor pwrdown nor auto-nego mode */
2306                 /* XXX -- this operation is required for NS DP83840A. */
2307                 gem_mii_write(dp, MII_CONTROL, 0);
2308 
2309                 /* As resetting PHY has completed, configure PHY registers */
2310                 if ((*dp->gc.gc_mii_config)(dp) != GEM_SUCCESS) {
2311                         /* we failed to configure PHY. */
2312                         goto reset_phy;
2313                 }
2314 
2315                 /* mii_config may disable autonegatiation */
2316                 gem_choose_forcedmode(dp);
2317 
2318                 dp->mii_lpable = 0;
2319                 dp->mii_advert = 0;
2320                 dp->mii_exp = 0;
2321                 dp->mii_ctl1000 = 0;
2322                 dp->mii_stat1000 = 0;
2323                 dp->flow_control = FLOW_CONTROL_NONE;
2324 
2325                 if (!dp->anadv_autoneg) {
2326                         /* skip auto-negotiation phase */
2327                         dp->mii_state = MII_STATE_MEDIA_SETUP;
2328                         dp->mii_timer = 0;
2329                         dp->mii_interval = 0;
2330                         goto next_nowait;
2331                 }
2332 
2333                 /* Issue auto-negotiation command */
2334                 goto autonego;
2335 
2336         case MII_STATE_AUTONEGOTIATING:
2337                 /*
2338                  * Autonegotiation is in progress
2339                  */
2340                 dp->mii_timer -= diff;
2341                 if (dp->mii_timer -
2342                     (dp->gc.gc_mii_an_timeout
2343                     - dp->gc.gc_mii_an_wait) > 0) {
2344                         /*
2345                          * wait for a while, typically autonegotiation
2346                          * completes in 2.3 - 2.5 sec.
2347                          */
2348                         dp->mii_interval = WATCH_INTERVAL_FAST;
2349                         goto next;
2350                 }
2351 
2352                 /* read PHY status */
2353                 status = gem_mii_read(dp, MII_STATUS);
2354                 DPRINTF(4, (CE_CONT,
2355                     "!%s: %s: called: mii_state:%d MII_STATUS reg:%b",
2356                     dp->name, __func__, dp->mii_state,
2357                     status, MII_STATUS_BITS));
2358 
2359                 if (status & MII_STATUS_REMFAULT) {
2360                         /*
2361                          * The link parnert told me something wrong happend.
2362                          * What do we do ?
2363                          */
2364                         cmn_err(CE_CONT,
2365                             "!%s: auto-negotiation failed: remote fault",
2366                             dp->name);
2367                         goto autonego;
2368                 }
2369 
2370                 if ((status & MII_STATUS_ANDONE) == 0) {
2371                         if (dp->mii_timer <= 0) {
2372                                 /*
2373                                  * Auto-negotiation was timed out,
2374                                  * try again w/o resetting phy.
2375                                  */
2376                                 if (!dp->mii_supress_msg) {
2377                                         cmn_err(CE_WARN,
2378                                     "!%s: auto-negotiation failed: timeout",
2379                                             dp->name);
2380                                         dp->mii_supress_msg = B_TRUE;
2381                                 }
2382                                 goto autonego;
2383                         }
2384                         /*
2385                          * Auto-negotiation is in progress. Wait.
2386                          */
2387                         dp->mii_interval = dp->gc.gc_mii_an_watch_interval;
2388                         goto next;
2389                 }
2390 
2391                 /*
2392                  * Auto-negotiation have completed.
2393                  * Assume linkdown and fall through.
2394                  */
2395                 dp->mii_supress_msg = B_FALSE;
2396                 dp->mii_state = MII_STATE_AN_DONE;
2397                 DPRINTF(0, (CE_CONT,
2398                     "!%s: auto-negotiation completed, MII_STATUS:%b",
2399                     dp->name, status, MII_STATUS_BITS));
2400 
2401                 if (dp->gc.gc_mii_an_delay > 0) {
2402                         dp->mii_timer = dp->gc.gc_mii_an_delay;
2403                         dp->mii_interval = drv_usectohz(20*1000);
2404                         goto next;
2405                 }
2406 
2407                 dp->mii_timer = 0;
2408                 diff = 0;
2409                 goto next_nowait;
2410 
2411         case MII_STATE_AN_DONE:
2412                 /*
2413                  * Auto-negotiation have done. Now we can set up media.
2414                  */
2415                 dp->mii_timer -= diff;
2416                 if (dp->mii_timer > 0) {
2417                         /* wait for a while */
2418                         dp->mii_interval = WATCH_INTERVAL_FAST;
2419                         goto next;
2420                 }
2421 
2422                 /*
2423                  * set up the result of auto negotiation
2424                  */
2425 
2426                 /*
2427                  * Read registers required to determin current
2428                  * duplex mode and media speed.
2429                  */
2430                 if (dp->gc.gc_mii_an_delay > 0) {
2431                         /*
2432                          * As the link watcher context has been suspended,
2433                          * 'status' is invalid. We must status register here
2434                          */
2435                         status = gem_mii_read(dp, MII_STATUS);
2436                 }
2437                 advert = gem_mii_read(dp, MII_AN_ADVERT);
2438                 lpable = gem_mii_read(dp, MII_AN_LPABLE);
2439                 exp = gem_mii_read(dp, MII_AN_EXPANSION);
2440                 if (exp == 0xffff) {
2441                         /* some phys don't have exp register */
2442                         exp = 0;
2443                 }
2444                 ctl1000  = 0;
2445                 stat1000 = 0;
2446                 if (dp->mii_status & MII_STATUS_XSTATUS) {
2447                         ctl1000  = gem_mii_read(dp, MII_1000TC);
2448                         stat1000 = gem_mii_read(dp, MII_1000TS);
2449                 }
2450                 dp->mii_lpable = lpable;
2451                 dp->mii_advert = advert;
2452                 dp->mii_exp = exp;
2453                 dp->mii_ctl1000  = ctl1000;
2454                 dp->mii_stat1000 = stat1000;
2455 
2456                 cmn_err(CE_CONT,
2457                 "!%s: auto-negotiation done, advert:%b, lpable:%b, exp:%b",
2458                     dp->name,
2459                     advert, MII_ABILITY_BITS,
2460                     lpable, MII_ABILITY_BITS,
2461                     exp, MII_AN_EXP_BITS);
2462 
2463                 if (dp->mii_status & MII_STATUS_XSTATUS) {
2464                         cmn_err(CE_CONT,
2465                             "! MII_1000TC:%b, MII_1000TS:%b",
2466                             ctl1000, MII_1000TC_BITS,
2467                             stat1000, MII_1000TS_BITS);
2468                 }
2469 
2470                 if (gem_population(lpable) <= 1 &&
2471                     (exp & MII_AN_EXP_LPCANAN) == 0) {
2472                         if ((advert & MII_ABILITY_TECH) != lpable) {
2473                                 cmn_err(CE_WARN,
2474                                     "!%s: but the link partnar doesn't seem"
2475                                     " to have auto-negotiation capability."
2476                                     " please check the link configuration.",
2477                                     dp->name);
2478                         }
2479                         /*
2480                          * it should be result of parallel detection, which
2481                          * cannot detect duplex mode.
2482                          */
2483                         if (lpable & MII_ABILITY_100BASE_TX) {
2484                                 /*
2485                                  * we prefer full duplex mode for 100Mbps
2486                                  * connection, if we can.
2487                                  */
2488                                 lpable |= advert & MII_ABILITY_100BASE_TX_FD;
2489                         }
2490 
2491                         if ((advert & lpable) == 0 &&
2492                             lpable & MII_ABILITY_10BASE_T) {
2493                                 lpable |= advert & MII_ABILITY_10BASE_T_FD;
2494                         }
2495                         /*
2496                          * as the link partnar isn't auto-negotiatable, use
2497                          * fixed mode temporally.
2498                          */
2499                         fix_phy = B_TRUE;
2500                 } else if (lpable == 0) {
2501                         cmn_err(CE_WARN, "!%s: wrong lpable.", dp->name);
2502                         goto reset_phy;
2503                 }
2504                 /*
2505                  * configure current link mode according to AN priority.
2506                  */
2507                 val = advert & lpable;
2508                 if ((ctl1000 & MII_1000TC_ADV_FULL) &&
2509                     (stat1000 & MII_1000TS_LP_FULL)) {
2510                         /* 1000BaseT & full duplex */
2511                         dp->speed     = GEM_SPD_1000;
2512                         dp->full_duplex  = B_TRUE;
2513                 } else if ((ctl1000 & MII_1000TC_ADV_HALF) &&
2514                     (stat1000 & MII_1000TS_LP_HALF)) {
2515                         /* 1000BaseT & half duplex */
2516                         dp->speed = GEM_SPD_1000;
2517                         dp->full_duplex = B_FALSE;
2518                 } else if (val & MII_ABILITY_100BASE_TX_FD) {
2519                         /* 100BaseTx & full duplex */
2520                         dp->speed = GEM_SPD_100;
2521                         dp->full_duplex = B_TRUE;
2522                 } else if (val & MII_ABILITY_100BASE_T4) {
2523                         /* 100BaseT4 & full duplex */
2524                         dp->speed = GEM_SPD_100;
2525                         dp->full_duplex = B_TRUE;
2526                 } else if (val & MII_ABILITY_100BASE_TX) {
2527                         /* 100BaseTx & half duplex */
2528                         dp->speed     = GEM_SPD_100;
2529                         dp->full_duplex  = B_FALSE;
2530                 } else if (val & MII_ABILITY_10BASE_T_FD) {
2531                         /* 10BaseT & full duplex */
2532                         dp->speed     = GEM_SPD_10;
2533                         dp->full_duplex  = B_TRUE;
2534                 } else if (val & MII_ABILITY_10BASE_T) {
2535                         /* 10BaseT & half duplex */
2536                         dp->speed     = GEM_SPD_10;
2537                         dp->full_duplex  = B_FALSE;
2538                 } else {
2539                         /*
2540                          * It seems that the link partnar doesn't have
2541                          * auto-negotiation capability and our PHY
2542                          * could not report the correct current mode.
2543                          * We guess current mode by mii_control register.
2544                          */
2545                         val = gem_mii_read(dp, MII_CONTROL);
2546 
2547                         /* select 100m full or 10m half */
2548                         dp->speed = (val & MII_CONTROL_100MB) ?
2549                             GEM_SPD_100 : GEM_SPD_10;
2550                         dp->full_duplex = dp->speed != GEM_SPD_10;
2551                         fix_phy = B_TRUE;
2552 
2553                         cmn_err(CE_NOTE,
2554                             "!%s: auto-negotiation done but "
2555                             "common ability not found.\n"
2556                             "PHY state: control:%b advert:%b lpable:%b\n"
2557                             "guessing %d Mbps %s duplex mode",
2558                             dp->name,
2559                             val, MII_CONTROL_BITS,
2560                             advert, MII_ABILITY_BITS,
2561                             lpable, MII_ABILITY_BITS,
2562                             gem_speed_value[dp->speed],
2563                             dp->full_duplex ? "full" : "half");
2564                 }
2565 
2566                 if (dp->full_duplex) {
2567                         dp->flow_control =
2568                             gem_fc_result[fc_cap_decode(advert)]
2569                             [fc_cap_decode(lpable)];
2570                 } else {
2571                         dp->flow_control = FLOW_CONTROL_NONE;
2572                 }
2573                 dp->mii_state = MII_STATE_MEDIA_SETUP;
2574                 /* FALLTHROUGH */
2575 
2576         case MII_STATE_MEDIA_SETUP:
2577                 dp->mii_state = MII_STATE_LINKDOWN;
2578                 dp->mii_timer = dp->gc.gc_mii_linkdown_timeout;
2579                 DPRINTF(2, (CE_CONT, "!%s: setup midia mode done", dp->name));
2580                 dp->mii_supress_msg = B_FALSE;
2581 
2582                 /* use short interval */
2583                 dp->mii_interval = WATCH_INTERVAL_FAST;
2584 
2585                 if ((!dp->anadv_autoneg) ||
2586                     dp->gc.gc_mii_an_oneshot || fix_phy) {
2587 
2588                         /*
2589                          * write specified mode to phy.
2590                          */
2591                         val = gem_mii_read(dp, MII_CONTROL);
2592                         val &= ~(MII_CONTROL_SPEED | MII_CONTROL_FDUPLEX |
2593                             MII_CONTROL_ANE | MII_CONTROL_RSAN);
2594 
2595                         if (dp->full_duplex) {
2596                                 val |= MII_CONTROL_FDUPLEX;
2597                         }
2598 
2599                         switch (dp->speed) {
2600                         case GEM_SPD_1000:
2601                                 val |= MII_CONTROL_1000MB;
2602                                 break;
2603 
2604                         case GEM_SPD_100:
2605                                 val |= MII_CONTROL_100MB;
2606                                 break;
2607 
2608                         default:
2609                                 cmn_err(CE_WARN, "%s: unknown speed:%d",
2610                                     dp->name, dp->speed);
2611                                 /* FALLTHROUGH */
2612                         case GEM_SPD_10:
2613                                 /* for GEM_SPD_10, do nothing */
2614                                 break;
2615                         }
2616 
2617                         if (dp->mii_status & MII_STATUS_XSTATUS) {
2618                                 gem_mii_write(dp,
2619                                     MII_1000TC, MII_1000TC_CFG_EN);
2620                         }
2621                         gem_mii_write(dp, MII_CONTROL, val);
2622                 }
2623 
2624                 if (dp->nic_state >= NIC_STATE_INITIALIZED) {
2625                         /* notify the result of auto-negotiation to mac */
2626                         (*dp->gc.gc_set_media)(dp);
2627                 }
2628 
2629                 if ((void *)dp->gc.gc_mii_tune_phy) {
2630                         /* for built-in sis900 */
2631                         /* XXX - this code should be removed.  */
2632                         (*dp->gc.gc_mii_tune_phy)(dp);
2633                 }
2634 
2635                 goto next_nowait;
2636 
2637         case MII_STATE_LINKDOWN:
2638                 status = gem_mii_read(dp, MII_STATUS);
2639                 if (status & MII_STATUS_LINKUP) {
2640                         /*
2641                          * Link going up
2642                          */
2643                         dp->mii_state = MII_STATE_LINKUP;
2644                         dp->mii_supress_msg = B_FALSE;
2645 
2646                         DPRINTF(0, (CE_CONT,
2647                             "!%s: link up detected: mii_stat:%b",
2648                             dp->name, status, MII_STATUS_BITS));
2649 
2650                         /*
2651                          * MII_CONTROL_100MB and  MII_CONTROL_FDUPLEX are
2652                          * ignored when MII_CONTROL_ANE is set.
2653                          */
2654                         cmn_err(CE_CONT,
2655                             "!%s: Link up: %d Mbps %s duplex %s flow control",
2656                             dp->name,
2657                             gem_speed_value[dp->speed],
2658                             dp->full_duplex ? "full" : "half",
2659                             gem_fc_type[dp->flow_control]);
2660 
2661                         dp->mii_interval = dp->gc.gc_mii_link_watch_interval;
2662 
2663                         /* XXX - we need other timer to watch statictics */
2664                         if (dp->gc.gc_mii_hw_link_detection &&
2665                             dp->nic_state == NIC_STATE_ONLINE) {
2666                                 dp->mii_interval = 0;
2667                         }
2668 
2669                         if (dp->nic_state == NIC_STATE_ONLINE) {
2670                                 if (!dp->mac_active) {
2671                                         (void) gem_mac_start(dp);
2672                                 }
2673                                 tx_sched = B_TRUE;
2674                         }
2675                         goto next;
2676                 }
2677 
2678                 dp->mii_supress_msg = B_TRUE;
2679                 if (dp->anadv_autoneg) {
2680                         dp->mii_timer -= diff;
2681                         if (dp->mii_timer <= 0) {
2682                                 /*
2683                                  * link down timer expired.
2684                                  * need to restart auto-negotiation.
2685                                  */
2686                                 linkdown_action =
2687                                     dp->gc.gc_mii_linkdown_timeout_action;
2688                                 goto restart_autonego;
2689                         }
2690                 }
2691                 /* don't change mii_state */
2692                 break;
2693 
2694         case MII_STATE_LINKUP:
2695                 status = gem_mii_read(dp, MII_STATUS);
2696                 if ((status & MII_STATUS_LINKUP) == 0) {
2697                         /*
2698                          * Link going down
2699                          */
2700                         cmn_err(CE_NOTE,
2701                             "!%s: link down detected: mii_stat:%b",
2702                             dp->name, status, MII_STATUS_BITS);
2703 
2704                         if (dp->nic_state == NIC_STATE_ONLINE &&
2705                             dp->mac_active &&
2706                             dp->gc.gc_mii_stop_mac_on_linkdown) {
2707                                 (void) gem_mac_stop(dp, 0);
2708 
2709                                 if (dp->tx_blocked) {
2710                                         /* drain tx */
2711                                         tx_sched = B_TRUE;
2712                                 }
2713                         }
2714 
2715                         if (dp->anadv_autoneg) {
2716                                 /* need to restart auto-negotiation */
2717                                 linkdown_action = dp->gc.gc_mii_linkdown_action;
2718                                 goto restart_autonego;
2719                         }
2720 
2721                         dp->mii_state = MII_STATE_LINKDOWN;
2722                         dp->mii_timer = dp->gc.gc_mii_linkdown_timeout;
2723 
2724                         if ((void *)dp->gc.gc_mii_tune_phy) {
2725                                 /* for built-in sis900 */
2726                                 (*dp->gc.gc_mii_tune_phy)(dp);
2727                         }
2728                         dp->mii_interval = dp->gc.gc_mii_link_watch_interval;
2729                         goto next;
2730                 }
2731 
2732                 /* don't change mii_state */
2733                 if (dp->gc.gc_mii_hw_link_detection &&
2734                     dp->nic_state == NIC_STATE_ONLINE) {
2735                         dp->mii_interval = 0;
2736                         goto next;
2737                 }
2738                 break;
2739         }
2740         dp->mii_interval = dp->gc.gc_mii_link_watch_interval;
2741         goto next;
2742 
2743         /* Actions on the end of state routine */
2744 
2745 restart_autonego:
2746         switch (linkdown_action) {
2747         case MII_ACTION_RESET:
2748                 if (!dp->mii_supress_msg) {
2749                         cmn_err(CE_CONT, "!%s: resetting PHY", dp->name);
2750                 }
2751                 dp->mii_supress_msg = B_TRUE;
2752                 goto reset_phy;
2753 
2754         case MII_ACTION_NONE:
2755                 dp->mii_supress_msg = B_TRUE;
2756                 if (dp->gc.gc_mii_an_oneshot) {
2757                         goto autonego;
2758                 }
2759                 /* PHY will restart autonego automatically */
2760                 dp->mii_state = MII_STATE_AUTONEGOTIATING;
2761                 dp->mii_timer = dp->gc.gc_mii_an_timeout;
2762                 dp->mii_interval = dp->gc.gc_mii_an_watch_interval;
2763                 goto next;
2764 
2765         case MII_ACTION_RSA:
2766                 if (!dp->mii_supress_msg) {
2767                         cmn_err(CE_CONT, "!%s: restarting auto-negotiation",
2768                             dp->name);
2769                 }
2770                 dp->mii_supress_msg = B_TRUE;
2771                 goto autonego;
2772 
2773         default:
2774                 cmn_err(CE_WARN, "!%s: unknowm linkdown action: %d",
2775                     dp->name, dp->gc.gc_mii_linkdown_action);
2776                 dp->mii_supress_msg = B_TRUE;
2777         }
2778         /* NOTREACHED */
2779 
2780 reset_phy:
2781         if (!dp->mii_supress_msg) {
2782                 cmn_err(CE_CONT, "!%s: resetting PHY", dp->name);
2783         }
2784         dp->mii_state = MII_STATE_RESETTING;
2785         dp->mii_timer = dp->gc.gc_mii_reset_timeout;
2786         if (!dp->gc.gc_mii_dont_reset) {
2787                 gem_mii_write(dp, MII_CONTROL, MII_CONTROL_RESET);
2788         }
2789         dp->mii_interval = WATCH_INTERVAL_FAST;
2790         goto next;
2791 
2792 autonego:
2793         if (!dp->mii_supress_msg) {
2794                 cmn_err(CE_CONT, "!%s: auto-negotiation started", dp->name);
2795         }
2796         dp->mii_state = MII_STATE_AUTONEGOTIATING;
2797         dp->mii_timer = dp->gc.gc_mii_an_timeout;
2798 
2799         /* start/restart auto nego */
2800         val = gem_mii_read(dp, MII_CONTROL) &
2801             ~(MII_CONTROL_ISOLATE | MII_CONTROL_PWRDN | MII_CONTROL_RESET);
2802 
2803         gem_mii_write(dp, MII_CONTROL,
2804             val | MII_CONTROL_RSAN | MII_CONTROL_ANE);
2805 
2806         dp->mii_interval = dp->gc.gc_mii_an_watch_interval;
2807 
2808 next:
2809         if (dp->link_watcher_id == 0 && dp->mii_interval) {
2810                 /* we must schedule next mii_watcher */
2811                 dp->link_watcher_id =
2812                     timeout((void (*)(void *))&gem_mii_link_watcher,
2813                     (void *)dp, dp->mii_interval);
2814         }
2815 
2816         if (old_mii_state != dp->mii_state) {
2817                 /* notify new mii link state */
2818                 if (dp->mii_state == MII_STATE_LINKUP) {
2819                         dp->linkup_delay = 0;
2820                         GEM_LINKUP(dp);
2821                 } else if (dp->linkup_delay <= 0) {
2822                         GEM_LINKDOWN(dp);
2823                 }
2824         } else if (dp->linkup_delay < 0) {
2825                 /* first linkup timeout */
2826                 dp->linkup_delay = 0;
2827                 GEM_LINKDOWN(dp);
2828         }
2829 
2830         return (tx_sched);
2831 }
2832 
2833 static void
2834 gem_mii_link_watcher(struct gem_dev *dp)
2835 {
2836         boolean_t       tx_sched;
2837 
2838         mutex_enter(&dp->intrlock);
2839 
2840         dp->link_watcher_id = 0;
2841         tx_sched = gem_mii_link_check(dp);
2842 #if GEM_DEBUG_LEVEL > 2
2843         if (dp->link_watcher_id == 0) {
2844                 cmn_err(CE_CONT, "%s: link watcher stopped", dp->name);
2845         }
2846 #endif
2847         mutex_exit(&dp->intrlock);
2848 
2849         if (tx_sched) {
2850                 /* kick potentially stopped downstream */
2851                 mac_tx_update(dp->mh);
2852         }
2853 }
2854 
2855 int
2856 gem_mii_probe_default(struct gem_dev *dp)
2857 {
2858         int8_t          phy;
2859         uint16_t        status;
2860         uint16_t        adv;
2861         uint16_t        adv_org;
2862 
2863         DPRINTF(3, (CE_CONT, "!%s: %s: called", dp->name, __func__));
2864 
2865         /*
2866          * Scan PHY
2867          */
2868         /* ensure to send sync bits */
2869         dp->mii_status = 0;
2870 
2871         /* Try default phy first */
2872         if (dp->mii_phy_addr) {
2873                 status = gem_mii_read(dp, MII_STATUS);
2874                 if (status != 0xffff && status != 0) {
2875                         gem_mii_write(dp, MII_CONTROL, 0);
2876                         goto PHY_found;
2877                 }
2878 
2879                 if (dp->mii_phy_addr < 0) {
2880                         cmn_err(CE_NOTE,
2881             "!%s: failed to probe default internal and/or non-MII PHY",
2882                             dp->name);
2883                         return (GEM_FAILURE);
2884                 }
2885 
2886                 cmn_err(CE_NOTE,
2887                     "!%s: failed to probe default MII PHY at %d",
2888                     dp->name, dp->mii_phy_addr);
2889         }
2890 
2891         /* Try all possible address */
2892         for (phy = dp->gc.gc_mii_addr_min; phy < 32; phy++) {
2893                 dp->mii_phy_addr = phy;
2894                 status = gem_mii_read(dp, MII_STATUS);
2895 
2896                 if (status != 0xffff && status != 0) {
2897                         gem_mii_write(dp, MII_CONTROL, 0);
2898                         goto PHY_found;
2899                 }
2900         }
2901 
2902         for (phy = dp->gc.gc_mii_addr_min; phy < 32; phy++) {
2903                 dp->mii_phy_addr = phy;
2904                 gem_mii_write(dp, MII_CONTROL, 0);
2905                 status = gem_mii_read(dp, MII_STATUS);
2906 
2907                 if (status != 0xffff && status != 0) {
2908                         goto PHY_found;
2909                 }
2910         }
2911 
2912         cmn_err(CE_NOTE, "!%s: no MII PHY found", dp->name);
2913         dp->mii_phy_addr = -1;
2914 
2915         return (GEM_FAILURE);
2916 
2917 PHY_found:
2918         dp->mii_status = status;
2919         dp->mii_phy_id  = (gem_mii_read(dp, MII_PHYIDH) << 16) |
2920             gem_mii_read(dp, MII_PHYIDL);
2921 
2922         if (dp->mii_phy_addr < 0) {
2923                 cmn_err(CE_CONT, "!%s: using internal/non-MII PHY(0x%08x)",
2924                     dp->name, dp->mii_phy_id);
2925         } else {
2926                 cmn_err(CE_CONT, "!%s: MII PHY (0x%08x) found at %d",
2927                     dp->name, dp->mii_phy_id, dp->mii_phy_addr);
2928         }
2929 
2930         cmn_err(CE_CONT, "!%s: PHY control:%b, status:%b, advert:%b, lpar:%b",
2931             dp->name,
2932             gem_mii_read(dp, MII_CONTROL), MII_CONTROL_BITS,
2933             status, MII_STATUS_BITS,
2934             gem_mii_read(dp, MII_AN_ADVERT), MII_ABILITY_BITS,
2935             gem_mii_read(dp, MII_AN_LPABLE), MII_ABILITY_BITS);
2936 
2937         dp->mii_xstatus = 0;
2938         if (status & MII_STATUS_XSTATUS) {
2939                 dp->mii_xstatus = gem_mii_read(dp, MII_XSTATUS);
2940 
2941                 cmn_err(CE_CONT, "!%s: xstatus:%b",
2942                     dp->name, dp->mii_xstatus, MII_XSTATUS_BITS);
2943         }
2944 
2945         /* check if the phy can advertize pause abilities */
2946         adv_org = gem_mii_read(dp, MII_AN_ADVERT);
2947 
2948         gem_mii_write(dp, MII_AN_ADVERT,
2949             MII_ABILITY_PAUSE | MII_ABILITY_ASMPAUSE);
2950 
2951         adv = gem_mii_read(dp, MII_AN_ADVERT);
2952 
2953         if ((adv & MII_ABILITY_PAUSE) == 0) {
2954                 dp->gc.gc_flow_control &= ~1;
2955         }
2956 
2957         if ((adv & MII_ABILITY_ASMPAUSE) == 0) {
2958                 dp->gc.gc_flow_control &= ~2;
2959         }
2960 
2961         gem_mii_write(dp, MII_AN_ADVERT, adv_org);
2962 
2963         return (GEM_SUCCESS);
2964 }
2965 
2966 static void
2967 gem_mii_start(struct gem_dev *dp)
2968 {
2969         DPRINTF(3, (CE_CONT, "!%s: %s: called", dp->name, __func__));
2970 
2971         /* make a first call of check link */
2972         dp->mii_state = MII_STATE_UNKNOWN;
2973         dp->mii_last_check = ddi_get_lbolt();
2974         dp->linkup_delay = dp->gc.gc_mii_linkdown_timeout;
2975         (void) gem_mii_link_watcher(dp);
2976 }
2977 
2978 static void
2979 gem_mii_stop(struct gem_dev *dp)
2980 {
2981         DPRINTF(3, (CE_CONT, "!%s: %s: called", dp->name, __func__));
2982 
2983         /* Ensure timer routine stopped */
2984         mutex_enter(&dp->intrlock);
2985         if (dp->link_watcher_id) {
2986                 while (untimeout(dp->link_watcher_id) == -1)
2987                         ;
2988                 dp->link_watcher_id = 0;
2989         }
2990         mutex_exit(&dp->intrlock);
2991 }
2992 
2993 boolean_t
2994 gem_get_mac_addr_conf(struct gem_dev *dp)
2995 {
2996         char            propname[32];
2997         char            *valstr;
2998         uint8_t         mac[ETHERADDRL];
2999         char            *cp;
3000         int             c;
3001         int             i;
3002         int             j;
3003         uint8_t         v;
3004         uint8_t         d;
3005         uint8_t         ored;
3006 
3007         DPRINTF(3, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3008         /*
3009          * Get ethernet address from .conf file
3010          */
3011         (void) sprintf(propname, "mac-addr");
3012         if ((ddi_prop_lookup_string(DDI_DEV_T_ANY, dp->dip,
3013             DDI_PROP_DONTPASS, propname, &valstr)) !=
3014             DDI_PROP_SUCCESS) {
3015                 return (B_FALSE);
3016         }
3017 
3018         if (strlen(valstr) != ETHERADDRL*3-1) {
3019                 goto syntax_err;
3020         }
3021 
3022         cp = valstr;
3023         j  = 0;
3024         ored = 0;
3025         for (;;) {
3026                 v = 0;
3027                 for (i = 0; i < 2; i++) {
3028                         c = *cp++;
3029 
3030                         if (c >= 'a' && c <= 'f') {
3031                                 d = c - 'a' + 10;
3032                         } else if (c >= 'A' && c <= 'F') {
3033                                 d = c - 'A' + 10;
3034                         } else if (c >= '0' && c <= '9') {
3035                                 d = c - '0';
3036                         } else {
3037                                 goto syntax_err;
3038                         }
3039                         v = (v << 4) | d;
3040                 }
3041 
3042                 mac[j++] = v;
3043                 ored |= v;
3044                 if (j == ETHERADDRL) {
3045                         /* done */
3046                         break;
3047                 }
3048 
3049                 c = *cp++;
3050                 if (c != ':') {
3051                         goto syntax_err;
3052                 }
3053         }
3054 
3055         if (ored == 0) {
3056                 goto err;
3057         }
3058         for (i = 0; i < ETHERADDRL; i++) {
3059                 dp->dev_addr.ether_addr_octet[i] = mac[i];
3060         }
3061         ddi_prop_free(valstr);
3062         return (B_TRUE);
3063 
3064 syntax_err:
3065         cmn_err(CE_CONT,
3066             "!%s: read mac addr: trying .conf: syntax err %s",
3067             dp->name, valstr);
3068 err:
3069         ddi_prop_free(valstr);
3070 
3071         return (B_FALSE);
3072 }
3073 
3074 
3075 /* ============================================================== */
3076 /*
3077  * internal start/stop interface
3078  */
3079 /* ============================================================== */
3080 static int
3081 gem_mac_set_rx_filter(struct gem_dev *dp)
3082 {
3083         return ((*dp->gc.gc_set_rx_filter)(dp));
3084 }
3085 
3086 /*
3087  * gem_mac_init: cold start
3088  */
3089 static int
3090 gem_mac_init(struct gem_dev *dp)
3091 {
3092         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3093 
3094         if (dp->mac_suspended) {
3095                 return (GEM_FAILURE);
3096         }
3097 
3098         dp->mac_active = B_FALSE;
3099 
3100         gem_init_rx_ring(dp);
3101         gem_init_tx_ring(dp);
3102 
3103         /* reset transmitter state */
3104         dp->tx_blocked = (clock_t)0;
3105         dp->tx_busy = 0;
3106         dp->tx_reclaim_busy = 0;
3107         dp->tx_max_packets = dp->gc.gc_tx_buf_limit;
3108 
3109         if ((*dp->gc.gc_init_chip)(dp) != GEM_SUCCESS) {
3110                 return (GEM_FAILURE);
3111         }
3112 
3113         gem_prepare_rx_buf(dp);
3114 
3115         return (GEM_SUCCESS);
3116 }
3117 /*
3118  * gem_mac_start: warm start
3119  */
3120 static int
3121 gem_mac_start(struct gem_dev *dp)
3122 {
3123         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3124 
3125         ASSERT(mutex_owned(&dp->intrlock));
3126         ASSERT(dp->nic_state == NIC_STATE_ONLINE);
3127         ASSERT(dp->mii_state ==  MII_STATE_LINKUP);
3128 
3129         /* enable tx and rx */
3130         mutex_enter(&dp->xmitlock);
3131         if (dp->mac_suspended) {
3132                 mutex_exit(&dp->xmitlock);
3133                 return (GEM_FAILURE);
3134         }
3135         dp->mac_active = B_TRUE;
3136         mutex_exit(&dp->xmitlock);
3137 
3138         /* setup rx buffers */
3139         (*dp->gc.gc_rx_start)(dp,
3140             SLOT(dp->rx_active_head, dp->gc.gc_rx_ring_size),
3141             dp->rx_active_tail - dp->rx_active_head);
3142 
3143         if ((*dp->gc.gc_start_chip)(dp) != GEM_SUCCESS) {
3144                 cmn_err(CE_WARN, "%s: %s: start_chip: failed",
3145                     dp->name, __func__);
3146                 return (GEM_FAILURE);
3147         }
3148 
3149         mutex_enter(&dp->xmitlock);
3150 
3151         /* load untranmitted packets to the nic */
3152         ASSERT(dp->tx_softq_tail - dp->tx_softq_head >= 0);
3153         if (dp->tx_softq_tail - dp->tx_softq_head > 0) {
3154                 gem_tx_load_descs_oo(dp,
3155                     dp->tx_softq_head, dp->tx_softq_tail,
3156                     GEM_TXFLAG_HEAD);
3157                 /* issue preloaded tx buffers */
3158                 gem_tx_start_unit(dp);
3159         }
3160 
3161         mutex_exit(&dp->xmitlock);
3162 
3163         return (GEM_SUCCESS);
3164 }
3165 
3166 static int
3167 gem_mac_stop(struct gem_dev *dp, uint_t flags)
3168 {
3169         int             i;
3170         int             wait_time; /* in uS */
3171 #ifdef GEM_DEBUG_LEVEL
3172         clock_t         now;
3173 #endif
3174         int             ret = GEM_SUCCESS;
3175 
3176         DPRINTF(1, (CE_CONT, "!%s: %s: called, rx_buf_free:%d",
3177             dp->name, __func__, dp->rx_buf_freecnt));
3178 
3179         ASSERT(mutex_owned(&dp->intrlock));
3180         ASSERT(!mutex_owned(&dp->xmitlock));
3181 
3182         /*
3183          * Block transmits
3184          */
3185         mutex_enter(&dp->xmitlock);
3186         if (dp->mac_suspended) {
3187                 mutex_exit(&dp->xmitlock);
3188                 return (GEM_SUCCESS);
3189         }
3190         dp->mac_active = B_FALSE;
3191 
3192         while (dp->tx_busy > 0) {
3193                 cv_wait(&dp->tx_drain_cv, &dp->xmitlock);
3194         }
3195         mutex_exit(&dp->xmitlock);
3196 
3197         if ((flags & GEM_RESTART_NOWAIT) == 0) {
3198                 /*
3199                  * Wait for all tx buffers sent.
3200                  */
3201                 wait_time =
3202                     2 * (8 * MAXPKTBUF(dp) / gem_speed_value[dp->speed]) *
3203                     (dp->tx_active_tail - dp->tx_active_head);
3204 
3205                 DPRINTF(0, (CE_CONT, "%s: %s: max drain time: %d uS",
3206                     dp->name, __func__, wait_time));
3207                 i = 0;
3208 #ifdef GEM_DEBUG_LEVEL
3209                 now = ddi_get_lbolt();
3210 #endif
3211                 while (dp->tx_active_tail != dp->tx_active_head) {
3212                         if (i > wait_time) {
3213                                 /* timeout */
3214                                 cmn_err(CE_NOTE, "%s: %s timeout: tx drain",
3215                                     dp->name, __func__);
3216                                 break;
3217                         }
3218                         (void) gem_reclaim_txbuf(dp);
3219                         drv_usecwait(100);
3220                         i += 100;
3221                 }
3222                 DPRINTF(0, (CE_NOTE,
3223                     "!%s: %s: the nic have drained in %d uS, real %d mS",
3224                     dp->name, __func__, i,
3225                     10*((int)(ddi_get_lbolt() - now))));
3226         }
3227 
3228         /*
3229          * Now we can stop the nic safely.
3230          */
3231         if ((*dp->gc.gc_stop_chip)(dp) != GEM_SUCCESS) {
3232                 cmn_err(CE_NOTE, "%s: %s: resetting the chip to stop it",
3233                     dp->name, __func__);
3234                 if ((*dp->gc.gc_reset_chip)(dp) != GEM_SUCCESS) {
3235                         cmn_err(CE_WARN, "%s: %s: failed to reset chip",
3236                             dp->name, __func__);
3237                 }
3238         }
3239 
3240         /*
3241          * Clear all rx buffers
3242          */
3243         if (flags & GEM_RESTART_KEEP_BUF) {
3244                 (void) gem_receive(dp);
3245         }
3246         gem_clean_rx_buf(dp);
3247 
3248         /*
3249          * Update final statistics
3250          */
3251         (*dp->gc.gc_get_stats)(dp);
3252 
3253         /*
3254          * Clear all pended tx packets
3255          */
3256         ASSERT(dp->tx_active_tail == dp->tx_softq_head);
3257         ASSERT(dp->tx_softq_tail == dp->tx_free_head);
3258         if (flags & GEM_RESTART_KEEP_BUF) {
3259                 /* restore active tx buffers */
3260                 dp->tx_active_tail = dp->tx_active_head;
3261                 dp->tx_softq_head  = dp->tx_active_head;
3262         } else {
3263                 gem_clean_tx_buf(dp);
3264         }
3265 
3266         return (ret);
3267 }
3268 
3269 static int
3270 gem_add_multicast(struct gem_dev *dp, const uint8_t *ep)
3271 {
3272         int             cnt;
3273         int             err;
3274 
3275         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3276 
3277         mutex_enter(&dp->intrlock);
3278         if (dp->mac_suspended) {
3279                 mutex_exit(&dp->intrlock);
3280                 return (GEM_FAILURE);
3281         }
3282 
3283         if (dp->mc_count_req++ < GEM_MAXMC) {
3284                 /* append the new address at the end of the mclist */
3285                 cnt = dp->mc_count;
3286                 bcopy(ep, dp->mc_list[cnt].addr.ether_addr_octet,
3287                     ETHERADDRL);
3288                 if (dp->gc.gc_multicast_hash) {
3289                         dp->mc_list[cnt].hash =
3290                             (*dp->gc.gc_multicast_hash)(dp, (uint8_t *)ep);
3291                 }
3292                 dp->mc_count = cnt + 1;
3293         }
3294 
3295         if (dp->mc_count_req != dp->mc_count) {
3296                 /* multicast address list overflow */
3297                 dp->rxmode |= RXMODE_MULTI_OVF;
3298         } else {
3299                 dp->rxmode &= ~RXMODE_MULTI_OVF;
3300         }
3301 
3302         /* tell new multicast list to the hardware */
3303         err = gem_mac_set_rx_filter(dp);
3304 
3305         mutex_exit(&dp->intrlock);
3306 
3307         return (err);
3308 }
3309 
3310 static int
3311 gem_remove_multicast(struct gem_dev *dp, const uint8_t *ep)
3312 {
3313         size_t          len;
3314         int             i;
3315         int             cnt;
3316         int             err;
3317 
3318         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3319 
3320         mutex_enter(&dp->intrlock);
3321         if (dp->mac_suspended) {
3322                 mutex_exit(&dp->intrlock);
3323                 return (GEM_FAILURE);
3324         }
3325 
3326         dp->mc_count_req--;
3327         cnt = dp->mc_count;
3328         for (i = 0; i < cnt; i++) {
3329                 if (bcmp(ep, &dp->mc_list[i].addr, ETHERADDRL)) {
3330                         continue;
3331                 }
3332                 /* shrink the mclist by copying forward */
3333                 len = (cnt - (i + 1)) * sizeof (*dp->mc_list);
3334                 if (len > 0) {
3335                         bcopy(&dp->mc_list[i+1], &dp->mc_list[i], len);
3336                 }
3337                 dp->mc_count--;
3338                 break;
3339         }
3340 
3341         if (dp->mc_count_req != dp->mc_count) {
3342                 /* multicast address list overflow */
3343                 dp->rxmode |= RXMODE_MULTI_OVF;
3344         } else {
3345                 dp->rxmode &= ~RXMODE_MULTI_OVF;
3346         }
3347         /* In gem v2, don't hold xmitlock on calling set_rx_filter */
3348         err = gem_mac_set_rx_filter(dp);
3349 
3350         mutex_exit(&dp->intrlock);
3351 
3352         return (err);
3353 }
3354 
3355 /* ============================================================== */
3356 /*
3357  * ND interface
3358  */
3359 /* ============================================================== */
3360 enum {
3361         PARAM_AUTONEG_CAP,
3362         PARAM_PAUSE_CAP,
3363         PARAM_ASYM_PAUSE_CAP,
3364         PARAM_1000FDX_CAP,
3365         PARAM_1000HDX_CAP,
3366         PARAM_100T4_CAP,
3367         PARAM_100FDX_CAP,
3368         PARAM_100HDX_CAP,
3369         PARAM_10FDX_CAP,
3370         PARAM_10HDX_CAP,
3371 
3372         PARAM_ADV_AUTONEG_CAP,
3373         PARAM_ADV_PAUSE_CAP,
3374         PARAM_ADV_ASYM_PAUSE_CAP,
3375         PARAM_ADV_1000FDX_CAP,
3376         PARAM_ADV_1000HDX_CAP,
3377         PARAM_ADV_100T4_CAP,
3378         PARAM_ADV_100FDX_CAP,
3379         PARAM_ADV_100HDX_CAP,
3380         PARAM_ADV_10FDX_CAP,
3381         PARAM_ADV_10HDX_CAP,
3382 
3383         PARAM_LP_AUTONEG_CAP,
3384         PARAM_LP_PAUSE_CAP,
3385         PARAM_LP_ASYM_PAUSE_CAP,
3386         PARAM_LP_1000FDX_CAP,
3387         PARAM_LP_1000HDX_CAP,
3388         PARAM_LP_100T4_CAP,
3389         PARAM_LP_100FDX_CAP,
3390         PARAM_LP_100HDX_CAP,
3391         PARAM_LP_10FDX_CAP,
3392         PARAM_LP_10HDX_CAP,
3393 
3394         PARAM_LINK_STATUS,
3395         PARAM_LINK_SPEED,
3396         PARAM_LINK_DUPLEX,
3397 
3398         PARAM_LINK_AUTONEG,
3399         PARAM_LINK_RX_PAUSE,
3400         PARAM_LINK_TX_PAUSE,
3401 
3402         PARAM_LOOP_MODE,
3403         PARAM_MSI_CNT,
3404 
3405 #ifdef DEBUG_RESUME
3406         PARAM_RESUME_TEST,
3407 #endif
3408         PARAM_COUNT
3409 };
3410 
3411 enum ioc_reply {
3412         IOC_INVAL = -1,                         /* bad, NAK with EINVAL */
3413         IOC_DONE,                               /* OK, reply sent       */
3414         IOC_ACK,                                /* OK, just send ACK    */
3415         IOC_REPLY,                              /* OK, just send reply  */
3416         IOC_RESTART_ACK,                        /* OK, restart & ACK        */
3417         IOC_RESTART_REPLY                       /* OK, restart & reply      */
3418 };
3419 
3420 struct gem_nd_arg {
3421         struct gem_dev  *dp;
3422         int             item;
3423 };
3424 
3425 static int
3426 gem_param_get(queue_t *q, mblk_t *mp, caddr_t arg, cred_t *credp)
3427 {
3428         struct gem_dev  *dp = ((struct gem_nd_arg *)(void *)arg)->dp;
3429         int             item = ((struct gem_nd_arg *)(void *)arg)->item;
3430         long            val;
3431 
3432         DPRINTF(0, (CE_CONT, "!%s: %s: called, item:%d",
3433             dp->name, __func__, item));
3434 
3435         switch (item) {
3436         case PARAM_AUTONEG_CAP:
3437                 val = BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG);
3438                 DPRINTF(0, (CE_CONT, "autoneg_cap:%d", val));
3439                 break;
3440 
3441         case PARAM_PAUSE_CAP:
3442                 val = BOOLEAN(dp->gc.gc_flow_control & 1);
3443                 break;
3444 
3445         case PARAM_ASYM_PAUSE_CAP:
3446                 val = BOOLEAN(dp->gc.gc_flow_control & 2);
3447                 break;
3448 
3449         case PARAM_1000FDX_CAP:
3450                 val = (dp->mii_xstatus & MII_XSTATUS_1000BASET_FD) ||
3451                     (dp->mii_xstatus & MII_XSTATUS_1000BASEX_FD);
3452                 break;
3453 
3454         case PARAM_1000HDX_CAP:
3455                 val = (dp->mii_xstatus & MII_XSTATUS_1000BASET) ||
3456                     (dp->mii_xstatus & MII_XSTATUS_1000BASEX);
3457                 break;
3458 
3459         case PARAM_100T4_CAP:
3460                 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASE_T4);
3461                 break;
3462 
3463         case PARAM_100FDX_CAP:
3464                 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD);
3465                 break;
3466 
3467         case PARAM_100HDX_CAP:
3468                 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX);
3469                 break;
3470 
3471         case PARAM_10FDX_CAP:
3472                 val = BOOLEAN(dp->mii_status & MII_STATUS_10_FD);
3473                 break;
3474 
3475         case PARAM_10HDX_CAP:
3476                 val = BOOLEAN(dp->mii_status & MII_STATUS_10);
3477                 break;
3478 
3479         case PARAM_ADV_AUTONEG_CAP:
3480                 val = dp->anadv_autoneg;
3481                 break;
3482 
3483         case PARAM_ADV_PAUSE_CAP:
3484                 val = BOOLEAN(dp->anadv_flow_control & 1);
3485                 break;
3486 
3487         case PARAM_ADV_ASYM_PAUSE_CAP:
3488                 val = BOOLEAN(dp->anadv_flow_control & 2);
3489                 break;
3490 
3491         case PARAM_ADV_1000FDX_CAP:
3492                 val = dp->anadv_1000fdx;
3493                 break;
3494 
3495         case PARAM_ADV_1000HDX_CAP:
3496                 val = dp->anadv_1000hdx;
3497                 break;
3498 
3499         case PARAM_ADV_100T4_CAP:
3500                 val = dp->anadv_100t4;
3501                 break;
3502 
3503         case PARAM_ADV_100FDX_CAP:
3504                 val = dp->anadv_100fdx;
3505                 break;
3506 
3507         case PARAM_ADV_100HDX_CAP:
3508                 val = dp->anadv_100hdx;
3509                 break;
3510 
3511         case PARAM_ADV_10FDX_CAP:
3512                 val = dp->anadv_10fdx;
3513                 break;
3514 
3515         case PARAM_ADV_10HDX_CAP:
3516                 val = dp->anadv_10hdx;
3517                 break;
3518 
3519         case PARAM_LP_AUTONEG_CAP:
3520                 val = BOOLEAN(dp->mii_exp & MII_AN_EXP_LPCANAN);
3521                 break;
3522 
3523         case PARAM_LP_PAUSE_CAP:
3524                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_PAUSE);
3525                 break;
3526 
3527         case PARAM_LP_ASYM_PAUSE_CAP:
3528                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_ASMPAUSE);
3529                 break;
3530 
3531         case PARAM_LP_1000FDX_CAP:
3532                 val = BOOLEAN(dp->mii_stat1000 & MII_1000TS_LP_FULL);
3533                 break;
3534 
3535         case PARAM_LP_1000HDX_CAP:
3536                 val = BOOLEAN(dp->mii_stat1000 & MII_1000TS_LP_HALF);
3537                 break;
3538 
3539         case PARAM_LP_100T4_CAP:
3540                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_T4);
3541                 break;
3542 
3543         case PARAM_LP_100FDX_CAP:
3544                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_TX_FD);
3545                 break;
3546 
3547         case PARAM_LP_100HDX_CAP:
3548                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_TX);
3549                 break;
3550 
3551         case PARAM_LP_10FDX_CAP:
3552                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_10BASE_T_FD);
3553                 break;
3554 
3555         case PARAM_LP_10HDX_CAP:
3556                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_10BASE_T);
3557                 break;
3558 
3559         case PARAM_LINK_STATUS:
3560                 val = (dp->mii_state == MII_STATE_LINKUP);
3561                 break;
3562 
3563         case PARAM_LINK_SPEED:
3564                 val = gem_speed_value[dp->speed];
3565                 break;
3566 
3567         case PARAM_LINK_DUPLEX:
3568                 val = 0;
3569                 if (dp->mii_state == MII_STATE_LINKUP) {
3570                         val = dp->full_duplex ? 2 : 1;
3571                 }
3572                 break;
3573 
3574         case PARAM_LINK_AUTONEG:
3575                 val = BOOLEAN(dp->mii_exp & MII_AN_EXP_LPCANAN);
3576                 break;
3577 
3578         case PARAM_LINK_RX_PAUSE:
3579                 val = (dp->flow_control == FLOW_CONTROL_SYMMETRIC) ||
3580                     (dp->flow_control == FLOW_CONTROL_RX_PAUSE);
3581                 break;
3582 
3583         case PARAM_LINK_TX_PAUSE:
3584                 val = (dp->flow_control == FLOW_CONTROL_SYMMETRIC) ||
3585                     (dp->flow_control == FLOW_CONTROL_TX_PAUSE);
3586                 break;
3587 
3588 #ifdef DEBUG_RESUME
3589         case PARAM_RESUME_TEST:
3590                 val = 0;
3591                 break;
3592 #endif
3593         default:
3594                 cmn_err(CE_WARN, "%s: unimplemented ndd control (%d)",
3595                     dp->name, item);
3596                 break;
3597         }
3598 
3599         (void) mi_mpprintf(mp, "%ld", val);
3600 
3601         return (0);
3602 }
3603 
3604 static int
3605 gem_param_set(queue_t *q, mblk_t *mp, char *value, caddr_t arg, cred_t *credp)
3606 {
3607         struct gem_dev  *dp = ((struct gem_nd_arg *)(void *)arg)->dp;
3608         int             item = ((struct gem_nd_arg *)(void *)arg)->item;
3609         long            val;
3610         char            *end;
3611 
3612         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3613         if (ddi_strtol(value, &end, 10, &val)) {
3614                 return (EINVAL);
3615         }
3616         if (end == value) {
3617                 return (EINVAL);
3618         }
3619 
3620         switch (item) {
3621         case PARAM_ADV_AUTONEG_CAP:
3622                 if (val != 0 && val != 1) {
3623                         goto err;
3624                 }
3625                 if (val && (dp->mii_status & MII_STATUS_CANAUTONEG) == 0) {
3626                         goto err;
3627                 }
3628                 dp->anadv_autoneg = (int)val;
3629                 break;
3630 
3631         case PARAM_ADV_PAUSE_CAP:
3632                 if (val != 0 && val != 1) {
3633                         goto err;
3634                 }
3635                 if (val) {
3636                         dp->anadv_flow_control |= 1;
3637                 } else {
3638                         dp->anadv_flow_control &= ~1;
3639                 }
3640                 break;
3641 
3642         case PARAM_ADV_ASYM_PAUSE_CAP:
3643                 if (val != 0 && val != 1) {
3644                         goto err;
3645                 }
3646                 if (val) {
3647                         dp->anadv_flow_control |= 2;
3648                 } else {
3649                         dp->anadv_flow_control &= ~2;
3650                 }
3651                 break;
3652 
3653         case PARAM_ADV_1000FDX_CAP:
3654                 if (val != 0 && val != 1) {
3655                         goto err;
3656                 }
3657                 if (val && (dp->mii_xstatus &
3658                     (MII_XSTATUS_1000BASET_FD |
3659                     MII_XSTATUS_1000BASEX_FD)) == 0) {
3660                         goto err;
3661                 }
3662                 dp->anadv_1000fdx = (int)val;
3663                 break;
3664 
3665         case PARAM_ADV_1000HDX_CAP:
3666                 if (val != 0 && val != 1) {
3667                         goto err;
3668                 }
3669                 if (val && (dp->mii_xstatus &
3670                     (MII_XSTATUS_1000BASET | MII_XSTATUS_1000BASEX)) == 0) {
3671                         goto err;
3672                 }
3673                 dp->anadv_1000hdx = (int)val;
3674                 break;
3675 
3676         case PARAM_ADV_100T4_CAP:
3677                 if (val != 0 && val != 1) {
3678                         goto err;
3679                 }
3680                 if (val && (dp->mii_status & MII_STATUS_100_BASE_T4) == 0) {
3681                         goto err;
3682                 }
3683                 dp->anadv_100t4 = (int)val;
3684                 break;
3685 
3686         case PARAM_ADV_100FDX_CAP:
3687                 if (val != 0 && val != 1) {
3688                         goto err;
3689                 }
3690                 if (val && (dp->mii_status & MII_STATUS_100_BASEX_FD) == 0) {
3691                         goto err;
3692                 }
3693                 dp->anadv_100fdx = (int)val;
3694                 break;
3695 
3696         case PARAM_ADV_100HDX_CAP:
3697                 if (val != 0 && val != 1) {
3698                         goto err;
3699                 }
3700                 if (val && (dp->mii_status & MII_STATUS_100_BASEX) == 0) {
3701                         goto err;
3702                 }
3703                 dp->anadv_100hdx = (int)val;
3704                 break;
3705 
3706         case PARAM_ADV_10FDX_CAP:
3707                 if (val != 0 && val != 1) {
3708                         goto err;
3709                 }
3710                 if (val && (dp->mii_status & MII_STATUS_10_FD) == 0) {
3711                         goto err;
3712                 }
3713                 dp->anadv_10fdx = (int)val;
3714                 break;
3715 
3716         case PARAM_ADV_10HDX_CAP:
3717                 if (val != 0 && val != 1) {
3718                         goto err;
3719                 }
3720                 if (val && (dp->mii_status & MII_STATUS_10) == 0) {
3721                         goto err;
3722                 }
3723                 dp->anadv_10hdx = (int)val;
3724                 break;
3725         }
3726 
3727         /* sync with PHY */
3728         gem_choose_forcedmode(dp);
3729 
3730         dp->mii_state = MII_STATE_UNKNOWN;
3731         if (dp->gc.gc_mii_hw_link_detection && dp->link_watcher_id == 0) {
3732                 /* XXX - Can we ignore the return code ? */
3733                 (void) gem_mii_link_check(dp);
3734         }
3735 
3736         return (0);
3737 err:
3738         return (EINVAL);
3739 }
3740 
3741 static void
3742 gem_nd_load(struct gem_dev *dp, char *name, ndgetf_t gf, ndsetf_t sf, int item)
3743 {
3744         struct gem_nd_arg       *arg;
3745 
3746         ASSERT(item >= 0);
3747         ASSERT(item < PARAM_COUNT);
3748 
3749         arg = &((struct gem_nd_arg *)(void *)dp->nd_arg_p)[item];
3750         arg->dp = dp;
3751         arg->item = item;
3752 
3753         DPRINTF(2, (CE_CONT, "!%s: %s: name:%s, item:%d",
3754             dp->name, __func__, name, item));
3755         (void) nd_load(&dp->nd_data_p, name, gf, sf, (caddr_t)arg);
3756 }
3757 
3758 static void
3759 gem_nd_setup(struct gem_dev *dp)
3760 {
3761         DPRINTF(0, (CE_CONT, "!%s: %s: called, mii_status:0x%b",
3762             dp->name, __func__, dp->mii_status, MII_STATUS_BITS));
3763 
3764         ASSERT(dp->nd_arg_p == NULL);
3765 
3766         dp->nd_arg_p =
3767             kmem_zalloc(sizeof (struct gem_nd_arg) * PARAM_COUNT, KM_SLEEP);
3768 
3769 #define SETFUNC(x)      ((x) ? gem_param_set : NULL)
3770 
3771         gem_nd_load(dp, "autoneg_cap",
3772             gem_param_get, NULL, PARAM_AUTONEG_CAP);
3773         gem_nd_load(dp, "pause_cap",
3774             gem_param_get, NULL, PARAM_PAUSE_CAP);
3775         gem_nd_load(dp, "asym_pause_cap",
3776             gem_param_get, NULL, PARAM_ASYM_PAUSE_CAP);
3777         gem_nd_load(dp, "1000fdx_cap",
3778             gem_param_get, NULL, PARAM_1000FDX_CAP);
3779         gem_nd_load(dp, "1000hdx_cap",
3780             gem_param_get, NULL, PARAM_1000HDX_CAP);
3781         gem_nd_load(dp, "100T4_cap",
3782             gem_param_get, NULL, PARAM_100T4_CAP);
3783         gem_nd_load(dp, "100fdx_cap",
3784             gem_param_get, NULL, PARAM_100FDX_CAP);
3785         gem_nd_load(dp, "100hdx_cap",
3786             gem_param_get, NULL, PARAM_100HDX_CAP);
3787         gem_nd_load(dp, "10fdx_cap",
3788             gem_param_get, NULL, PARAM_10FDX_CAP);
3789         gem_nd_load(dp, "10hdx_cap",
3790             gem_param_get, NULL, PARAM_10HDX_CAP);
3791 
3792         /* Our advertised capabilities */
3793         gem_nd_load(dp, "adv_autoneg_cap", gem_param_get,
3794             SETFUNC(dp->mii_status & MII_STATUS_CANAUTONEG),
3795             PARAM_ADV_AUTONEG_CAP);
3796         gem_nd_load(dp, "adv_pause_cap", gem_param_get,
3797             SETFUNC(dp->gc.gc_flow_control & 1),
3798             PARAM_ADV_PAUSE_CAP);
3799         gem_nd_load(dp, "adv_asym_pause_cap", gem_param_get,
3800             SETFUNC(dp->gc.gc_flow_control & 2),
3801             PARAM_ADV_ASYM_PAUSE_CAP);
3802         gem_nd_load(dp, "adv_1000fdx_cap", gem_param_get,
3803             SETFUNC(dp->mii_xstatus &
3804             (MII_XSTATUS_1000BASEX_FD | MII_XSTATUS_1000BASET_FD)),
3805             PARAM_ADV_1000FDX_CAP);
3806         gem_nd_load(dp, "adv_1000hdx_cap", gem_param_get,
3807             SETFUNC(dp->mii_xstatus &
3808             (MII_XSTATUS_1000BASEX | MII_XSTATUS_1000BASET)),
3809             PARAM_ADV_1000HDX_CAP);
3810         gem_nd_load(dp, "adv_100T4_cap", gem_param_get,
3811             SETFUNC((dp->mii_status & MII_STATUS_100_BASE_T4) &&
3812             !dp->mii_advert_ro),
3813             PARAM_ADV_100T4_CAP);
3814         gem_nd_load(dp, "adv_100fdx_cap", gem_param_get,
3815             SETFUNC((dp->mii_status & MII_STATUS_100_BASEX_FD) &&
3816             !dp->mii_advert_ro),
3817             PARAM_ADV_100FDX_CAP);
3818         gem_nd_load(dp, "adv_100hdx_cap", gem_param_get,
3819             SETFUNC((dp->mii_status & MII_STATUS_100_BASEX) &&
3820             !dp->mii_advert_ro),
3821             PARAM_ADV_100HDX_CAP);
3822         gem_nd_load(dp, "adv_10fdx_cap", gem_param_get,
3823             SETFUNC((dp->mii_status & MII_STATUS_10_FD) &&
3824             !dp->mii_advert_ro),
3825             PARAM_ADV_10FDX_CAP);
3826         gem_nd_load(dp, "adv_10hdx_cap", gem_param_get,
3827             SETFUNC((dp->mii_status & MII_STATUS_10) &&
3828             !dp->mii_advert_ro),
3829             PARAM_ADV_10HDX_CAP);
3830 
3831         /* Partner's advertised capabilities */
3832         gem_nd_load(dp, "lp_autoneg_cap",
3833             gem_param_get, NULL, PARAM_LP_AUTONEG_CAP);
3834         gem_nd_load(dp, "lp_pause_cap",
3835             gem_param_get, NULL, PARAM_LP_PAUSE_CAP);
3836         gem_nd_load(dp, "lp_asym_pause_cap",
3837             gem_param_get, NULL, PARAM_LP_ASYM_PAUSE_CAP);
3838         gem_nd_load(dp, "lp_1000fdx_cap",
3839             gem_param_get, NULL, PARAM_LP_1000FDX_CAP);
3840         gem_nd_load(dp, "lp_1000hdx_cap",
3841             gem_param_get, NULL, PARAM_LP_1000HDX_CAP);
3842         gem_nd_load(dp, "lp_100T4_cap",
3843             gem_param_get, NULL, PARAM_LP_100T4_CAP);
3844         gem_nd_load(dp, "lp_100fdx_cap",
3845             gem_param_get, NULL, PARAM_LP_100FDX_CAP);
3846         gem_nd_load(dp, "lp_100hdx_cap",
3847             gem_param_get, NULL, PARAM_LP_100HDX_CAP);
3848         gem_nd_load(dp, "lp_10fdx_cap",
3849             gem_param_get, NULL, PARAM_LP_10FDX_CAP);
3850         gem_nd_load(dp, "lp_10hdx_cap",
3851             gem_param_get, NULL, PARAM_LP_10HDX_CAP);
3852 
3853         /* Current operating modes */
3854         gem_nd_load(dp, "link_status",
3855             gem_param_get, NULL, PARAM_LINK_STATUS);
3856         gem_nd_load(dp, "link_speed",
3857             gem_param_get, NULL, PARAM_LINK_SPEED);
3858         gem_nd_load(dp, "link_duplex",
3859             gem_param_get, NULL, PARAM_LINK_DUPLEX);
3860         gem_nd_load(dp, "link_autoneg",
3861             gem_param_get, NULL, PARAM_LINK_AUTONEG);
3862         gem_nd_load(dp, "link_rx_pause",
3863             gem_param_get, NULL, PARAM_LINK_RX_PAUSE);
3864         gem_nd_load(dp, "link_tx_pause",
3865             gem_param_get, NULL, PARAM_LINK_TX_PAUSE);
3866 #ifdef DEBUG_RESUME
3867         gem_nd_load(dp, "resume_test",
3868             gem_param_get, NULL, PARAM_RESUME_TEST);
3869 #endif
3870 #undef  SETFUNC
3871 }
3872 
3873 static
3874 enum ioc_reply
3875 gem_nd_ioctl(struct gem_dev *dp, queue_t *wq, mblk_t *mp, struct iocblk *iocp)
3876 {
3877         boolean_t       ok;
3878 
3879         ASSERT(mutex_owned(&dp->intrlock));
3880 
3881         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3882 
3883         switch (iocp->ioc_cmd) {
3884         case ND_GET:
3885                 ok = nd_getset(wq, dp->nd_data_p, mp);
3886                 DPRINTF(0, (CE_CONT,
3887                     "%s: get %s", dp->name, ok ? "OK" : "FAIL"));
3888                 return (ok ? IOC_REPLY : IOC_INVAL);
3889 
3890         case ND_SET:
3891                 ok = nd_getset(wq, dp->nd_data_p, mp);
3892 
3893                 DPRINTF(0, (CE_CONT, "%s: set %s err %d",
3894                     dp->name, ok ? "OK" : "FAIL", iocp->ioc_error));
3895 
3896                 if (!ok) {
3897                         return (IOC_INVAL);
3898                 }
3899 
3900                 if (iocp->ioc_error) {
3901                         return (IOC_REPLY);
3902                 }
3903 
3904                 return (IOC_RESTART_REPLY);
3905         }
3906 
3907         cmn_err(CE_WARN, "%s: invalid cmd 0x%x", dp->name, iocp->ioc_cmd);
3908 
3909         return (IOC_INVAL);
3910 }
3911 
3912 static void
3913 gem_nd_cleanup(struct gem_dev *dp)
3914 {
3915         ASSERT(dp->nd_data_p != NULL);
3916         ASSERT(dp->nd_arg_p != NULL);
3917 
3918         nd_free(&dp->nd_data_p);
3919 
3920         kmem_free(dp->nd_arg_p, sizeof (struct gem_nd_arg) * PARAM_COUNT);
3921         dp->nd_arg_p = NULL;
3922 }
3923 
3924 static void
3925 gem_mac_ioctl(struct gem_dev *dp, queue_t *wq, mblk_t *mp)
3926 {
3927         struct iocblk   *iocp;
3928         enum ioc_reply  status;
3929         int             cmd;
3930 
3931         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
3932 
3933         /*
3934          * Validate the command before bothering with the mutex ...
3935          */
3936         iocp = (void *)mp->b_rptr;
3937         iocp->ioc_error = 0;
3938         cmd = iocp->ioc_cmd;
3939 
3940         DPRINTF(0, (CE_CONT, "%s: %s cmd:0x%x", dp->name, __func__, cmd));
3941 
3942         mutex_enter(&dp->intrlock);
3943         mutex_enter(&dp->xmitlock);
3944 
3945         switch (cmd) {
3946         default:
3947                 _NOTE(NOTREACHED)
3948                 status = IOC_INVAL;
3949                 break;
3950 
3951         case ND_GET:
3952         case ND_SET:
3953                 status = gem_nd_ioctl(dp, wq, mp, iocp);
3954                 break;
3955         }
3956 
3957         mutex_exit(&dp->xmitlock);
3958         mutex_exit(&dp->intrlock);
3959 
3960 #ifdef DEBUG_RESUME
3961         if (cmd == ND_GET)  {
3962                 gem_suspend(dp->dip);
3963                 gem_resume(dp->dip);
3964         }
3965 #endif
3966         /*
3967          * Finally, decide how to reply
3968          */
3969         switch (status) {
3970         default:
3971         case IOC_INVAL:
3972                 /*
3973                  * Error, reply with a NAK and EINVAL or the specified error
3974                  */
3975                 miocnak(wq, mp, 0, iocp->ioc_error == 0 ?
3976                     EINVAL : iocp->ioc_error);
3977                 break;
3978 
3979         case IOC_DONE:
3980                 /*
3981                  * OK, reply already sent
3982                  */
3983                 break;
3984 
3985         case IOC_RESTART_ACK:
3986         case IOC_ACK:
3987                 /*
3988                  * OK, reply with an ACK
3989                  */
3990                 miocack(wq, mp, 0, 0);
3991                 break;
3992 
3993         case IOC_RESTART_REPLY:
3994         case IOC_REPLY:
3995                 /*
3996                  * OK, send prepared reply as ACK or NAK
3997                  */
3998                 mp->b_datap->db_type =
3999                     iocp->ioc_error == 0 ? M_IOCACK : M_IOCNAK;
4000                 qreply(wq, mp);
4001                 break;
4002         }
4003 }
4004 
4005 #ifndef SYS_MAC_H
4006 #define XCVR_UNDEFINED  0
4007 #define XCVR_NONE       1
4008 #define XCVR_10         2
4009 #define XCVR_100T4      3
4010 #define XCVR_100X       4
4011 #define XCVR_100T2      5
4012 #define XCVR_1000X      6
4013 #define XCVR_1000T      7
4014 #endif
4015 static int
4016 gem_mac_xcvr_inuse(struct gem_dev *dp)
4017 {
4018         int     val = XCVR_UNDEFINED;
4019 
4020         if ((dp->mii_status & MII_STATUS_XSTATUS) == 0) {
4021                 if (dp->mii_status & MII_STATUS_100_BASE_T4) {
4022                         val = XCVR_100T4;
4023                 } else if (dp->mii_status &
4024                     (MII_STATUS_100_BASEX_FD |
4025                     MII_STATUS_100_BASEX)) {
4026                         val = XCVR_100X;
4027                 } else if (dp->mii_status &
4028                     (MII_STATUS_100_BASE_T2_FD |
4029                     MII_STATUS_100_BASE_T2)) {
4030                         val = XCVR_100T2;
4031                 } else if (dp->mii_status &
4032                     (MII_STATUS_10_FD | MII_STATUS_10)) {
4033                         val = XCVR_10;
4034                 }
4035         } else if (dp->mii_xstatus &
4036             (MII_XSTATUS_1000BASET_FD | MII_XSTATUS_1000BASET)) {
4037                 val = XCVR_1000T;
4038         } else if (dp->mii_xstatus &
4039             (MII_XSTATUS_1000BASEX_FD | MII_XSTATUS_1000BASEX)) {
4040                 val = XCVR_1000X;
4041         }
4042 
4043         return (val);
4044 }
4045 
4046 /* ============================================================== */
4047 /*
4048  * GLDv3 interface
4049  */
4050 /* ============================================================== */
4051 static int              gem_m_getstat(void *, uint_t, uint64_t *);
4052 static int              gem_m_start(void *);
4053 static void             gem_m_stop(void *);
4054 static int              gem_m_setpromisc(void *, boolean_t);
4055 static int              gem_m_multicst(void *, boolean_t, const uint8_t *);
4056 static int              gem_m_unicst(void *, const uint8_t *);
4057 static mblk_t           *gem_m_tx(void *, mblk_t *);
4058 static void             gem_m_ioctl(void *, queue_t *, mblk_t *);
4059 static boolean_t        gem_m_getcapab(void *, mac_capab_t, void *);
4060 
4061 #define GEM_M_CALLBACK_FLAGS    (MC_IOCTL | MC_GETCAPAB)
4062 
4063 static mac_callbacks_t gem_m_callbacks = {
4064         GEM_M_CALLBACK_FLAGS,
4065         gem_m_getstat,
4066         gem_m_start,
4067         gem_m_stop,
4068         gem_m_setpromisc,
4069         gem_m_multicst,
4070         gem_m_unicst,
4071         gem_m_tx,
4072         NULL,
4073         gem_m_ioctl,
4074         gem_m_getcapab,
4075 };
4076 
4077 static int
4078 gem_m_start(void *arg)
4079 {
4080         int             err = 0;
4081         struct gem_dev *dp = arg;
4082 
4083         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4084 
4085         mutex_enter(&dp->intrlock);
4086         if (dp->mac_suspended) {
4087                 err = EIO;
4088                 goto x;
4089         }
4090         if (gem_mac_init(dp) != GEM_SUCCESS) {
4091                 err = EIO;
4092                 goto x;
4093         }
4094         dp->nic_state = NIC_STATE_INITIALIZED;
4095 
4096         /* reset rx filter state */
4097         dp->mc_count = 0;
4098         dp->mc_count_req = 0;
4099 
4100         /* setup media mode if the link have been up */
4101         if (dp->mii_state == MII_STATE_LINKUP) {
4102                 (dp->gc.gc_set_media)(dp);
4103         }
4104 
4105         /* setup initial rx filter */
4106         bcopy(dp->dev_addr.ether_addr_octet,
4107             dp->cur_addr.ether_addr_octet, ETHERADDRL);
4108         dp->rxmode |= RXMODE_ENABLE;
4109 
4110         if (gem_mac_set_rx_filter(dp) != GEM_SUCCESS) {
4111                 err = EIO;
4112                 goto x;
4113         }
4114 
4115         dp->nic_state = NIC_STATE_ONLINE;
4116         if (dp->mii_state == MII_STATE_LINKUP) {
4117                 if (gem_mac_start(dp) != GEM_SUCCESS) {
4118                         err = EIO;
4119                         goto x;
4120                 }
4121         }
4122 
4123         dp->timeout_id = timeout((void (*)(void *))gem_tx_timeout,
4124             (void *)dp, dp->gc.gc_tx_timeout_interval);
4125         mutex_exit(&dp->intrlock);
4126 
4127         return (0);
4128 x:
4129         dp->nic_state = NIC_STATE_STOPPED;
4130         mutex_exit(&dp->intrlock);
4131         return (err);
4132 }
4133 
4134 static void
4135 gem_m_stop(void *arg)
4136 {
4137         struct gem_dev  *dp = arg;
4138 
4139         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4140 
4141         /* stop rx */
4142         mutex_enter(&dp->intrlock);
4143         if (dp->mac_suspended) {
4144                 mutex_exit(&dp->intrlock);
4145                 return;
4146         }
4147         dp->rxmode &= ~RXMODE_ENABLE;
4148         (void) gem_mac_set_rx_filter(dp);
4149         mutex_exit(&dp->intrlock);
4150 
4151         /* stop tx timeout watcher */
4152         if (dp->timeout_id) {
4153                 while (untimeout(dp->timeout_id) == -1)
4154                         ;
4155                 dp->timeout_id = 0;
4156         }
4157 
4158         /* make the nic state inactive */
4159         mutex_enter(&dp->intrlock);
4160         if (dp->mac_suspended) {
4161                 mutex_exit(&dp->intrlock);
4162                 return;
4163         }
4164         dp->nic_state = NIC_STATE_STOPPED;
4165 
4166         /* we need deassert mac_active due to block interrupt handler */
4167         mutex_enter(&dp->xmitlock);
4168         dp->mac_active = B_FALSE;
4169         mutex_exit(&dp->xmitlock);
4170 
4171         /* block interrupts */
4172         while (dp->intr_busy) {
4173                 cv_wait(&dp->tx_drain_cv, &dp->intrlock);
4174         }
4175         (void) gem_mac_stop(dp, 0);
4176         mutex_exit(&dp->intrlock);
4177 }
4178 
4179 static int
4180 gem_m_multicst(void *arg, boolean_t add, const uint8_t *ep)
4181 {
4182         int             err;
4183         int             ret;
4184         struct gem_dev  *dp = arg;
4185 
4186         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4187 
4188         if (add) {
4189                 ret = gem_add_multicast(dp, ep);
4190         } else {
4191                 ret = gem_remove_multicast(dp, ep);
4192         }
4193 
4194         err = 0;
4195         if (ret != GEM_SUCCESS) {
4196                 err = EIO;
4197         }
4198 
4199         return (err);
4200 }
4201 
4202 static int
4203 gem_m_setpromisc(void *arg, boolean_t on)
4204 {
4205         int             err = 0;        /* no error */
4206         struct gem_dev  *dp = arg;
4207 
4208         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4209 
4210         mutex_enter(&dp->intrlock);
4211         if (dp->mac_suspended) {
4212                 mutex_exit(&dp->intrlock);
4213                 return (EIO);
4214         }
4215         if (on) {
4216                 dp->rxmode |= RXMODE_PROMISC;
4217         } else {
4218                 dp->rxmode &= ~RXMODE_PROMISC;
4219         }
4220 
4221         if (gem_mac_set_rx_filter(dp) != GEM_SUCCESS) {
4222                 err = EIO;
4223         }
4224         mutex_exit(&dp->intrlock);
4225 
4226         return (err);
4227 }
4228 
4229 int
4230 gem_m_getstat(void *arg, uint_t stat, uint64_t *valp)
4231 {
4232         struct gem_dev          *dp = arg;
4233         struct gem_stats        *gstp = &dp->stats;
4234         uint64_t                val = 0;
4235 
4236         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4237 
4238         if (mutex_owned(&dp->intrlock)) {
4239                 if (dp->mac_suspended) {
4240                         return (EIO);
4241                 }
4242         } else {
4243                 mutex_enter(&dp->intrlock);
4244                 if (dp->mac_suspended) {
4245                         mutex_exit(&dp->intrlock);
4246                         return (EIO);
4247                 }
4248                 mutex_exit(&dp->intrlock);
4249         }
4250 
4251         if ((*dp->gc.gc_get_stats)(dp) != GEM_SUCCESS) {
4252                 return (EIO);
4253         }
4254 
4255         switch (stat) {
4256         case MAC_STAT_IFSPEED:
4257                 val = gem_speed_value[dp->speed] *1000000ull;
4258                 break;
4259 
4260         case MAC_STAT_MULTIRCV:
4261                 val = gstp->rmcast;
4262                 break;
4263 
4264         case MAC_STAT_BRDCSTRCV:
4265                 val = gstp->rbcast;
4266                 break;
4267 
4268         case MAC_STAT_MULTIXMT:
4269                 val = gstp->omcast;
4270                 break;
4271 
4272         case MAC_STAT_BRDCSTXMT:
4273                 val = gstp->obcast;
4274                 break;
4275 
4276         case MAC_STAT_NORCVBUF:
4277                 val = gstp->norcvbuf + gstp->missed;
4278                 break;
4279 
4280         case MAC_STAT_IERRORS:
4281                 val = gstp->errrcv;
4282                 break;
4283 
4284         case MAC_STAT_NOXMTBUF:
4285                 val = gstp->noxmtbuf;
4286                 break;
4287 
4288         case MAC_STAT_OERRORS:
4289                 val = gstp->errxmt;
4290                 break;
4291 
4292         case MAC_STAT_COLLISIONS:
4293                 val = gstp->collisions;
4294                 break;
4295 
4296         case MAC_STAT_RBYTES:
4297                 val = gstp->rbytes;
4298                 break;
4299 
4300         case MAC_STAT_IPACKETS:
4301                 val = gstp->rpackets;
4302                 break;
4303 
4304         case MAC_STAT_OBYTES:
4305                 val = gstp->obytes;
4306                 break;
4307 
4308         case MAC_STAT_OPACKETS:
4309                 val = gstp->opackets;
4310                 break;
4311 
4312         case MAC_STAT_UNDERFLOWS:
4313                 val = gstp->underflow;
4314                 break;
4315 
4316         case MAC_STAT_OVERFLOWS:
4317                 val = gstp->overflow;
4318                 break;
4319 
4320         case ETHER_STAT_ALIGN_ERRORS:
4321                 val = gstp->frame;
4322                 break;
4323 
4324         case ETHER_STAT_FCS_ERRORS:
4325                 val = gstp->crc;
4326                 break;
4327 
4328         case ETHER_STAT_FIRST_COLLISIONS:
4329                 val = gstp->first_coll;
4330                 break;
4331 
4332         case ETHER_STAT_MULTI_COLLISIONS:
4333                 val = gstp->multi_coll;
4334                 break;
4335 
4336         case ETHER_STAT_SQE_ERRORS:
4337                 val = gstp->sqe;
4338                 break;
4339 
4340         case ETHER_STAT_DEFER_XMTS:
4341                 val = gstp->defer;
4342                 break;
4343 
4344         case ETHER_STAT_TX_LATE_COLLISIONS:
4345                 val = gstp->xmtlatecoll;
4346                 break;
4347 
4348         case ETHER_STAT_EX_COLLISIONS:
4349                 val = gstp->excoll;
4350                 break;
4351 
4352         case ETHER_STAT_MACXMT_ERRORS:
4353                 val = gstp->xmit_internal_err;
4354                 break;
4355 
4356         case ETHER_STAT_CARRIER_ERRORS:
4357                 val = gstp->nocarrier;
4358                 break;
4359 
4360         case ETHER_STAT_TOOLONG_ERRORS:
4361                 val = gstp->frame_too_long;
4362                 break;
4363 
4364         case ETHER_STAT_MACRCV_ERRORS:
4365                 val = gstp->rcv_internal_err;
4366                 break;
4367 
4368         case ETHER_STAT_XCVR_ADDR:
4369                 val = dp->mii_phy_addr;
4370                 break;
4371 
4372         case ETHER_STAT_XCVR_ID:
4373                 val = dp->mii_phy_id;
4374                 break;
4375 
4376         case ETHER_STAT_XCVR_INUSE:
4377                 val = gem_mac_xcvr_inuse(dp);
4378                 break;
4379 
4380         case ETHER_STAT_CAP_1000FDX:
4381                 val = (dp->mii_xstatus & MII_XSTATUS_1000BASET_FD) ||
4382                     (dp->mii_xstatus & MII_XSTATUS_1000BASEX_FD);
4383                 break;
4384 
4385         case ETHER_STAT_CAP_1000HDX:
4386                 val = (dp->mii_xstatus & MII_XSTATUS_1000BASET) ||
4387                     (dp->mii_xstatus & MII_XSTATUS_1000BASEX);
4388                 break;
4389 
4390         case ETHER_STAT_CAP_100FDX:
4391                 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD);
4392                 break;
4393 
4394         case ETHER_STAT_CAP_100HDX:
4395                 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX);
4396                 break;
4397 
4398         case ETHER_STAT_CAP_10FDX:
4399                 val = BOOLEAN(dp->mii_status & MII_STATUS_10_FD);
4400                 break;
4401 
4402         case ETHER_STAT_CAP_10HDX:
4403                 val = BOOLEAN(dp->mii_status & MII_STATUS_10);
4404                 break;
4405 
4406         case ETHER_STAT_CAP_ASMPAUSE:
4407                 val = BOOLEAN(dp->gc.gc_flow_control & 2);
4408                 break;
4409 
4410         case ETHER_STAT_CAP_PAUSE:
4411                 val = BOOLEAN(dp->gc.gc_flow_control & 1);
4412                 break;
4413 
4414         case ETHER_STAT_CAP_AUTONEG:
4415                 val = BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG);
4416                 break;
4417 
4418         case ETHER_STAT_ADV_CAP_1000FDX:
4419                 val = dp->anadv_1000fdx;
4420                 break;
4421 
4422         case ETHER_STAT_ADV_CAP_1000HDX:
4423                 val = dp->anadv_1000hdx;
4424                 break;
4425 
4426         case ETHER_STAT_ADV_CAP_100FDX:
4427                 val = dp->anadv_100fdx;
4428                 break;
4429 
4430         case ETHER_STAT_ADV_CAP_100HDX:
4431                 val = dp->anadv_100hdx;
4432                 break;
4433 
4434         case ETHER_STAT_ADV_CAP_10FDX:
4435                 val = dp->anadv_10fdx;
4436                 break;
4437 
4438         case ETHER_STAT_ADV_CAP_10HDX:
4439                 val = dp->anadv_10hdx;
4440                 break;
4441 
4442         case ETHER_STAT_ADV_CAP_ASMPAUSE:
4443                 val = BOOLEAN(dp->anadv_flow_control & 2);
4444                 break;
4445 
4446         case ETHER_STAT_ADV_CAP_PAUSE:
4447                 val = BOOLEAN(dp->anadv_flow_control & 1);
4448                 break;
4449 
4450         case ETHER_STAT_ADV_CAP_AUTONEG:
4451                 val = dp->anadv_autoneg;
4452                 break;
4453 
4454         case ETHER_STAT_LP_CAP_1000FDX:
4455                 val = BOOLEAN(dp->mii_stat1000 & MII_1000TS_LP_FULL);
4456                 break;
4457 
4458         case ETHER_STAT_LP_CAP_1000HDX:
4459                 val = BOOLEAN(dp->mii_stat1000 & MII_1000TS_LP_HALF);
4460                 break;
4461 
4462         case ETHER_STAT_LP_CAP_100FDX:
4463                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_TX_FD);
4464                 break;
4465 
4466         case ETHER_STAT_LP_CAP_100HDX:
4467                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_TX);
4468                 break;
4469 
4470         case ETHER_STAT_LP_CAP_10FDX:
4471                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_10BASE_T_FD);
4472                 break;
4473 
4474         case ETHER_STAT_LP_CAP_10HDX:
4475                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_10BASE_T);
4476                 break;
4477 
4478         case ETHER_STAT_LP_CAP_ASMPAUSE:
4479                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_ASMPAUSE);
4480                 break;
4481 
4482         case ETHER_STAT_LP_CAP_PAUSE:
4483                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_PAUSE);
4484                 break;
4485 
4486         case ETHER_STAT_LP_CAP_AUTONEG:
4487                 val = BOOLEAN(dp->mii_exp & MII_AN_EXP_LPCANAN);
4488                 break;
4489 
4490         case ETHER_STAT_LINK_ASMPAUSE:
4491                 val = BOOLEAN(dp->flow_control & 2);
4492                 break;
4493 
4494         case ETHER_STAT_LINK_PAUSE:
4495                 val = BOOLEAN(dp->flow_control & 1);
4496                 break;
4497 
4498         case ETHER_STAT_LINK_AUTONEG:
4499                 val = dp->anadv_autoneg &&
4500                     BOOLEAN(dp->mii_exp & MII_AN_EXP_LPCANAN);
4501                 break;
4502 
4503         case ETHER_STAT_LINK_DUPLEX:
4504                 val = (dp->mii_state == MII_STATE_LINKUP) ?
4505                     (dp->full_duplex ? 2 : 1) : 0;
4506                 break;
4507 
4508         case ETHER_STAT_TOOSHORT_ERRORS:
4509                 val = gstp->runt;
4510                 break;
4511         case ETHER_STAT_LP_REMFAULT:
4512                 val = BOOLEAN(dp->mii_lpable & MII_AN_ADVERT_REMFAULT);
4513                 break;
4514 
4515         case ETHER_STAT_JABBER_ERRORS:
4516                 val = gstp->jabber;
4517                 break;
4518 
4519         case ETHER_STAT_CAP_100T4:
4520                 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASE_T4);
4521                 break;
4522 
4523         case ETHER_STAT_ADV_CAP_100T4:
4524                 val = dp->anadv_100t4;
4525                 break;
4526 
4527         case ETHER_STAT_LP_CAP_100T4:
4528                 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_T4);
4529                 break;
4530 
4531         default:
4532 #if GEM_DEBUG_LEVEL > 2
4533                 cmn_err(CE_WARN,
4534                     "%s: unrecognized parameter value = %d",
4535                     __func__, stat);
4536 #endif
4537                 return (ENOTSUP);
4538         }
4539 
4540         *valp = val;
4541 
4542         return (0);
4543 }
4544 
4545 static int
4546 gem_m_unicst(void *arg, const uint8_t *mac)
4547 {
4548         int             err = 0;
4549         struct gem_dev  *dp = arg;
4550 
4551         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4552 
4553         mutex_enter(&dp->intrlock);
4554         if (dp->mac_suspended) {
4555                 mutex_exit(&dp->intrlock);
4556                 return (EIO);
4557         }
4558         bcopy(mac, dp->cur_addr.ether_addr_octet, ETHERADDRL);
4559         dp->rxmode |= RXMODE_ENABLE;
4560 
4561         if (gem_mac_set_rx_filter(dp) != GEM_SUCCESS) {
4562                 err = EIO;
4563         }
4564         mutex_exit(&dp->intrlock);
4565 
4566         return (err);
4567 }
4568 
4569 /*
4570  * gem_m_tx is used only for sending data packets into ethernet wire.
4571  */
4572 static mblk_t *
4573 gem_m_tx(void *arg, mblk_t *mp)
4574 {
4575         uint32_t        flags = 0;
4576         struct gem_dev  *dp = arg;
4577         mblk_t          *tp;
4578 
4579         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4580 
4581         ASSERT(dp->nic_state == NIC_STATE_ONLINE);
4582         if (dp->mii_state != MII_STATE_LINKUP) {
4583                 /* Some nics hate to send packets when the link is down. */
4584                 while (mp) {
4585                         tp = mp->b_next;
4586                         mp->b_next = NULL;
4587                         freemsg(mp);
4588                         mp = tp;
4589                 }
4590                 return (NULL);
4591         }
4592 
4593         return (gem_send_common(dp, mp, flags));
4594 }
4595 
4596 static void
4597 gem_m_ioctl(void *arg, queue_t *wq, mblk_t *mp)
4598 {
4599         DPRINTF(0, (CE_CONT, "!%s: %s: called",
4600             ((struct gem_dev *)arg)->name, __func__));
4601 
4602         gem_mac_ioctl((struct gem_dev *)arg, wq, mp);
4603 }
4604 
4605 /* ARGSUSED */
4606 static boolean_t
4607 gem_m_getcapab(void *arg, mac_capab_t cap, void *cap_data)
4608 {
4609         return (B_FALSE);
4610 }
4611 
4612 static void
4613 gem_gld3_init(struct gem_dev *dp, mac_register_t *macp)
4614 {
4615         macp->m_type_ident = MAC_PLUGIN_IDENT_ETHER;
4616         macp->m_driver = dp;
4617         macp->m_dip = dp->dip;
4618         macp->m_src_addr = dp->dev_addr.ether_addr_octet;
4619         macp->m_callbacks = &gem_m_callbacks;
4620         macp->m_min_sdu = 0;
4621         macp->m_max_sdu = dp->mtu;
4622 
4623         if (dp->misc_flag & GEM_VLAN) {
4624                 macp->m_margin = VTAG_SIZE;
4625         }
4626 }
4627 
4628 /* ======================================================================== */
4629 /*
4630  * attach/detatch support
4631  */
4632 /* ======================================================================== */
4633 static void
4634 gem_read_conf(struct gem_dev *dp)
4635 {
4636         int     val;
4637 
4638         DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__));
4639 
4640         /*
4641          * Get media mode infomation from .conf file
4642          */
4643         dp->anadv_autoneg = gem_prop_get_int(dp, "adv_autoneg_cap", 1) != 0;
4644         dp->anadv_1000fdx = gem_prop_get_int(dp, "adv_1000fdx_cap", 1) != 0;
4645         dp->anadv_1000hdx = gem_prop_get_int(dp, "adv_1000hdx_cap", 1) != 0;
4646         dp->anadv_100t4   = gem_prop_get_int(dp, "adv_100T4_cap", 1) != 0;
4647         dp->anadv_100fdx  = gem_prop_get_int(dp, "adv_100fdx_cap", 1) != 0;
4648         dp->anadv_100hdx  = gem_prop_get_int(dp, "adv_100hdx_cap", 1) != 0;
4649         dp->anadv_10fdx   = gem_prop_get_int(dp, "adv_10fdx_cap", 1) != 0;
4650         dp->anadv_10hdx   = gem_prop_get_int(dp, "adv_10hdx_cap", 1) != 0;
4651 
4652         if ((ddi_prop_exists(DDI_DEV_T_ANY, dp->dip,
4653             DDI_PROP_DONTPASS, "full-duplex"))) {
4654                 dp->full_duplex = gem_prop_get_int(dp, "full-duplex", 1) != 0;
4655                 dp->anadv_autoneg = B_FALSE;
4656                 if (dp->full_duplex) {
4657                         dp->anadv_1000hdx = B_FALSE;
4658                         dp->anadv_100hdx = B_FALSE;
4659                         dp->anadv_10hdx = B_FALSE;
4660                 } else {
4661                         dp->anadv_1000fdx = B_FALSE;
4662                         dp->anadv_100fdx = B_FALSE;
4663                         dp->anadv_10fdx = B_FALSE;
4664                 }
4665         }
4666 
4667         if ((val = gem_prop_get_int(dp, "speed", 0)) > 0) {
4668                 dp->anadv_autoneg = B_FALSE;
4669                 switch (val) {
4670                 case 1000:
4671                         dp->speed = GEM_SPD_1000;
4672                         dp->anadv_100t4   = B_FALSE;
4673                         dp->anadv_100fdx  = B_FALSE;
4674                         dp->anadv_100hdx  = B_FALSE;
4675                         dp->anadv_10fdx   = B_FALSE;
4676                         dp->anadv_10hdx   = B_FALSE;
4677                         break;
4678                 case 100:
4679                         dp->speed = GEM_SPD_100;
4680                         dp->anadv_1000fdx = B_FALSE;
4681                         dp->anadv_1000hdx = B_FALSE;
4682                         dp->anadv_10fdx   = B_FALSE;
4683                         dp->anadv_10hdx   = B_FALSE;
4684                         break;
4685                 case 10:
4686                         dp->speed = GEM_SPD_10;
4687                         dp->anadv_1000fdx = B_FALSE;
4688                         dp->anadv_1000hdx = B_FALSE;
4689                         dp->anadv_100t4   = B_FALSE;
4690                         dp->anadv_100fdx  = B_FALSE;
4691                         dp->anadv_100hdx  = B_FALSE;
4692                         break;
4693                 default:
4694                         cmn_err(CE_WARN,
4695                             "!%s: property %s: illegal value:%d",
4696                             dp->name, "speed", val);
4697                         dp->anadv_autoneg = B_TRUE;
4698                         break;
4699                 }
4700         }
4701 
4702         val = gem_prop_get_int(dp, "flow-control", dp->gc.gc_flow_control);
4703         if (val > FLOW_CONTROL_RX_PAUSE || val < FLOW_CONTROL_NONE) {
4704                 cmn_err(CE_WARN,
4705                     "!%s: property %s: illegal value:%d",
4706                     dp->name, "flow-control", val);
4707         } else {
4708                 val = min(val, dp->gc.gc_flow_control);
4709         }
4710         dp->anadv_flow_control = val;
4711 
4712         if (gem_prop_get_int(dp, "nointr", 0)) {
4713                 dp->misc_flag |= GEM_NOINTR;
4714                 cmn_err(CE_NOTE, "!%s: polling mode enabled", dp->name);
4715         }
4716 
4717         dp->mtu = gem_prop_get_int(dp, "mtu", dp->mtu);
4718         dp->txthr = gem_prop_get_int(dp, "txthr", dp->txthr);
4719         dp->rxthr = gem_prop_get_int(dp, "rxthr", dp->rxthr);
4720         dp->txmaxdma = gem_prop_get_int(dp, "txmaxdma", dp->txmaxdma);
4721         dp->rxmaxdma = gem_prop_get_int(dp, "rxmaxdma", dp->rxmaxdma);
4722 }
4723 
4724 
4725 /*
4726  * Gem kstat support
4727  */
4728 
4729 #define GEM_LOCAL_DATA_SIZE(gc) \
4730         (sizeof (struct gem_dev) + \
4731         sizeof (struct mcast_addr) * GEM_MAXMC + \
4732         sizeof (struct txbuf) * ((gc)->gc_tx_buf_size) + \
4733         sizeof (void *) * ((gc)->gc_tx_buf_size))
4734 
4735 struct gem_dev *
4736 gem_do_attach(dev_info_t *dip, int port,
4737         struct gem_conf *gc, void *base, ddi_acc_handle_t *regs_handlep,
4738         void *lp, int lmsize)
4739 {
4740         struct gem_dev          *dp;
4741         int                     i;
4742         ddi_iblock_cookie_t     c;
4743         mac_register_t          *macp = NULL;
4744         int                     ret;
4745         int                     unit;
4746         int                     nports;
4747 
4748         unit = ddi_get_instance(dip);
4749         if ((nports = gc->gc_nports) == 0) {
4750                 nports = 1;
4751         }
4752         if (nports == 1) {
4753                 ddi_set_driver_private(dip, NULL);
4754         }
4755 
4756         DPRINTF(2, (CE_CONT, "!gem%d: gem_do_attach: called cmd:ATTACH",
4757             unit));
4758 
4759         /*
4760          * Allocate soft data structure
4761          */
4762         dp = kmem_zalloc(GEM_LOCAL_DATA_SIZE(gc), KM_SLEEP);
4763 
4764         if ((macp = mac_alloc(MAC_VERSION)) == NULL) {
4765                 cmn_err(CE_WARN, "!gem%d: %s: mac_alloc failed",
4766                     unit, __func__);
4767                 return (NULL);
4768         }
4769         /* ddi_set_driver_private(dip, dp); */
4770 
4771         /* link to private area */
4772         dp->private = lp;
4773         dp->priv_size = lmsize;
4774         dp->mc_list = (struct mcast_addr *)&dp[1];
4775 
4776         dp->dip = dip;
4777         (void) sprintf(dp->name, gc->gc_name, nports * unit + port);
4778 
4779         /*
4780          * Get iblock cookie
4781          */
4782         if (ddi_get_iblock_cookie(dip, 0, &c) != DDI_SUCCESS) {
4783                 cmn_err(CE_CONT,
4784                     "!%s: gem_do_attach: ddi_get_iblock_cookie: failed",
4785                     dp->name);
4786                 goto err_free_private;
4787         }
4788         dp->iblock_cookie = c;
4789 
4790         /*
4791          * Initialize mutex's for this device.
4792          */
4793         mutex_init(&dp->intrlock, NULL, MUTEX_DRIVER, (void *)c);
4794         mutex_init(&dp->xmitlock, NULL, MUTEX_DRIVER, (void *)c);
4795         cv_init(&dp->tx_drain_cv, NULL, CV_DRIVER, NULL);
4796 
4797         /*
4798          * configure gem parameter
4799          */
4800         dp->base_addr = base;
4801         dp->regs_handle = *regs_handlep;
4802         dp->gc = *gc;
4803         gc = &dp->gc;
4804         /* patch for simplify dma resource management */
4805         gc->gc_tx_max_frags = 1;
4806         gc->gc_tx_max_descs_per_pkt = 1;
4807         gc->gc_tx_ring_size = gc->gc_tx_buf_size;
4808         gc->gc_tx_ring_limit = gc->gc_tx_buf_limit;
4809         gc->gc_tx_desc_write_oo = B_TRUE;
4810 
4811         gc->gc_nports = nports;      /* fix nports */
4812 
4813         /* fix copy threadsholds */
4814         gc->gc_tx_copy_thresh = max(ETHERMIN, gc->gc_tx_copy_thresh);
4815         gc->gc_rx_copy_thresh = max(ETHERMIN, gc->gc_rx_copy_thresh);
4816 
4817         /* fix rx buffer boundary for iocache line size */
4818         ASSERT(gc->gc_dma_attr_txbuf.dma_attr_align-1 == gc->gc_tx_buf_align);
4819         ASSERT(gc->gc_dma_attr_rxbuf.dma_attr_align-1 == gc->gc_rx_buf_align);
4820         gc->gc_rx_buf_align = max(gc->gc_rx_buf_align, IOC_LINESIZE - 1);
4821         gc->gc_dma_attr_rxbuf.dma_attr_align = gc->gc_rx_buf_align + 1;
4822 
4823         /* fix descriptor boundary for cache line size */
4824         gc->gc_dma_attr_desc.dma_attr_align =
4825             max(gc->gc_dma_attr_desc.dma_attr_align, IOC_LINESIZE);
4826 
4827         /* patch get_packet method */
4828         if (gc->gc_get_packet == NULL) {
4829                 gc->gc_get_packet = &gem_get_packet_default;
4830         }
4831 
4832         /* patch get_rx_start method */
4833         if (gc->gc_rx_start == NULL) {
4834                 gc->gc_rx_start = &gem_rx_start_default;
4835         }
4836 
4837         /* calculate descriptor area */
4838         if (gc->gc_rx_desc_unit_shift >= 0) {
4839                 dp->rx_desc_size =
4840                     ROUNDUP(gc->gc_rx_ring_size << gc->gc_rx_desc_unit_shift,
4841                     gc->gc_dma_attr_desc.dma_attr_align);
4842         }
4843         if (gc->gc_tx_desc_unit_shift >= 0) {
4844                 dp->tx_desc_size =
4845                     ROUNDUP(gc->gc_tx_ring_size << gc->gc_tx_desc_unit_shift,
4846                     gc->gc_dma_attr_desc.dma_attr_align);
4847         }
4848 
4849         dp->mtu = ETHERMTU;
4850         dp->tx_buf = (void *)&dp->mc_list[GEM_MAXMC];
4851         /* link tx buffers */
4852         for (i = 0; i < dp->gc.gc_tx_buf_size; i++) {
4853                 dp->tx_buf[i].txb_next =
4854                     &dp->tx_buf[SLOT(i + 1, dp->gc.gc_tx_buf_size)];
4855         }
4856 
4857         dp->rxmode      = 0;
4858         dp->speed       = GEM_SPD_10;        /* default is 10Mbps */
4859         dp->full_duplex    = B_FALSE;                /* default is half */
4860         dp->flow_control   = FLOW_CONTROL_NONE;
4861         dp->poll_pkt_delay = 8;              /* typical coalease for rx packets */
4862 
4863         /* performance tuning parameters */
4864         dp->txthr    = ETHERMAX;     /* tx fifo threshold */
4865         dp->txmaxdma = 16*4;         /* tx max dma burst size */
4866         dp->rxthr    = 128;          /* rx fifo threshold */
4867         dp->rxmaxdma = 16*4;         /* rx max dma burst size */
4868 
4869         /*
4870          * Get media mode information from .conf file
4871          */
4872         gem_read_conf(dp);
4873 
4874         /* rx_buf_len is required buffer length without padding for alignment */
4875         dp->rx_buf_len = MAXPKTBUF(dp) + dp->gc.gc_rx_header_len;
4876 
4877         /*
4878          * Reset the chip
4879          */
4880         mutex_enter(&dp->intrlock);
4881         dp->nic_state = NIC_STATE_STOPPED;
4882         ret = (*dp->gc.gc_reset_chip)(dp);
4883         mutex_exit(&dp->intrlock);
4884         if (ret != GEM_SUCCESS) {
4885                 goto err_free_regs;
4886         }
4887 
4888         /*
4889          * HW dependant paremeter initialization
4890          */
4891         mutex_enter(&dp->intrlock);
4892         ret = (*dp->gc.gc_attach_chip)(dp);
4893         mutex_exit(&dp->intrlock);
4894         if (ret != GEM_SUCCESS) {
4895                 goto err_free_regs;
4896         }
4897 
4898 #ifdef DEBUG_MULTIFRAGS
4899         dp->gc.gc_tx_copy_thresh = dp->mtu;
4900 #endif
4901         /* allocate tx and rx resources */
4902         if (gem_alloc_memory(dp)) {
4903                 goto err_free_regs;
4904         }
4905 
4906         DPRINTF(0, (CE_CONT,
4907             "!%s: at 0x%x, %02x:%02x:%02x:%02x:%02x:%02x",
4908             dp->name, (long)dp->base_addr,
4909             dp->dev_addr.ether_addr_octet[0],
4910             dp->dev_addr.ether_addr_octet[1],
4911             dp->dev_addr.ether_addr_octet[2],
4912             dp->dev_addr.ether_addr_octet[3],
4913             dp->dev_addr.ether_addr_octet[4],
4914             dp->dev_addr.ether_addr_octet[5]));
4915 
4916         /* copy mac address */
4917         dp->cur_addr = dp->dev_addr;
4918 
4919         gem_gld3_init(dp, macp);
4920 
4921         /* Probe MII phy (scan phy) */
4922         dp->mii_lpable = 0;
4923         dp->mii_advert = 0;
4924         dp->mii_exp = 0;
4925         dp->mii_ctl1000 = 0;
4926         dp->mii_stat1000 = 0;
4927         if ((*dp->gc.gc_mii_probe)(dp) != GEM_SUCCESS) {
4928                 goto err_free_ring;
4929         }
4930 
4931         /* mask unsupported abilities */
4932         dp->anadv_autoneg &= BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG);
4933         dp->anadv_1000fdx &=
4934             BOOLEAN(dp->mii_xstatus &
4935             (MII_XSTATUS_1000BASEX_FD | MII_XSTATUS_1000BASET_FD));
4936         dp->anadv_1000hdx &=
4937             BOOLEAN(dp->mii_xstatus &
4938             (MII_XSTATUS_1000BASEX | MII_XSTATUS_1000BASET));
4939         dp->anadv_100t4  &= BOOLEAN(dp->mii_status & MII_STATUS_100_BASE_T4);
4940         dp->anadv_100fdx &= BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD);
4941         dp->anadv_100hdx &= BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX);
4942         dp->anadv_10fdx  &= BOOLEAN(dp->mii_status & MII_STATUS_10_FD);
4943         dp->anadv_10hdx  &= BOOLEAN(dp->mii_status & MII_STATUS_10);
4944 
4945         gem_choose_forcedmode(dp);
4946 
4947         /* initialize MII phy if required */
4948         if (dp->gc.gc_mii_init) {
4949                 if ((*dp->gc.gc_mii_init)(dp) != GEM_SUCCESS) {
4950                         goto err_free_ring;
4951                 }
4952         }
4953 
4954         /*
4955          * initialize kstats including mii statistics
4956          */
4957         gem_nd_setup(dp);
4958 
4959         /*
4960          * Add interrupt to system.
4961          */
4962         if (ret = mac_register(macp, &dp->mh)) {
4963                 cmn_err(CE_WARN, "!%s: mac_register failed, error:%d",
4964                     dp->name, ret);
4965                 goto err_release_stats;
4966         }
4967         mac_free(macp);
4968         macp = NULL;
4969 
4970         if (dp->misc_flag & GEM_SOFTINTR) {
4971                 if (ddi_add_softintr(dip,
4972                     DDI_SOFTINT_LOW, &dp->soft_id,
4973                     NULL, NULL,
4974                     (uint_t (*)(caddr_t))gem_intr,
4975                     (caddr_t)dp) != DDI_SUCCESS) {
4976                         cmn_err(CE_WARN, "!%s: ddi_add_softintr failed",
4977                             dp->name);
4978                         goto err_unregister;
4979                 }
4980         } else if ((dp->misc_flag & GEM_NOINTR) == 0) {
4981                 if (ddi_add_intr(dip, 0, NULL, NULL,
4982                     (uint_t (*)(caddr_t))gem_intr,
4983                     (caddr_t)dp) != DDI_SUCCESS) {
4984                         cmn_err(CE_WARN, "!%s: ddi_add_intr failed", dp->name);
4985                         goto err_unregister;
4986                 }
4987         } else {
4988                 /*
4989                  * Dont use interrupt.
4990                  * schedule first call of gem_intr_watcher
4991                  */
4992                 dp->intr_watcher_id =
4993                     timeout((void (*)(void *))gem_intr_watcher,
4994                     (void *)dp, drv_usectohz(3*1000000));
4995         }
4996 
4997         /* link this device to dev_info */
4998         dp->next = (struct gem_dev *)ddi_get_driver_private(dip);
4999         dp->port = port;
5000         ddi_set_driver_private(dip, (caddr_t)dp);
5001 
5002         /* reset mii phy and start mii link watcher */
5003         gem_mii_start(dp);
5004 
5005         DPRINTF(2, (CE_CONT, "!gem_do_attach: return: success"));
5006         return (dp);
5007 
5008 err_unregister:
5009         (void) mac_unregister(dp->mh);
5010 err_release_stats:
5011         /* release NDD resources */
5012         gem_nd_cleanup(dp);
5013 
5014 err_free_ring:
5015         gem_free_memory(dp);
5016 err_free_regs:
5017         ddi_regs_map_free(&dp->regs_handle);
5018 err_free_locks:
5019         mutex_destroy(&dp->xmitlock);
5020         mutex_destroy(&dp->intrlock);
5021         cv_destroy(&dp->tx_drain_cv);
5022 err_free_private:
5023         if (macp) {
5024                 mac_free(macp);
5025         }
5026         kmem_free((caddr_t)dp, GEM_LOCAL_DATA_SIZE(gc));
5027 
5028         return (NULL);
5029 }
5030 
5031 int
5032 gem_do_detach(dev_info_t *dip)
5033 {
5034         struct gem_dev  *dp;
5035         struct gem_dev  *tmp;
5036         caddr_t         private;
5037         int             priv_size;
5038         ddi_acc_handle_t        rh;
5039 
5040         dp = GEM_GET_DEV(dip);
5041         if (dp == NULL) {
5042                 return (DDI_SUCCESS);
5043         }
5044 
5045         rh = dp->regs_handle;
5046         private = dp->private;
5047         priv_size = dp->priv_size;
5048 
5049         while (dp) {
5050                 /* unregister with gld v3 */
5051                 if (mac_unregister(dp->mh) != 0) {
5052                         return (DDI_FAILURE);
5053                 }
5054 
5055                 /* ensure any rx buffers are not used */
5056                 if (dp->rx_buf_allocated != dp->rx_buf_freecnt) {
5057                         /* resource is busy */
5058                         cmn_err(CE_PANIC,
5059                             "!%s: %s: rxbuf is busy: allocated:%d, freecnt:%d",
5060                             dp->name, __func__,
5061                             dp->rx_buf_allocated, dp->rx_buf_freecnt);
5062                         /* NOT REACHED */
5063                 }
5064 
5065                 /* stop mii link watcher */
5066                 gem_mii_stop(dp);
5067 
5068                 /* unregister interrupt handler */
5069                 if (dp->misc_flag & GEM_SOFTINTR) {
5070                         ddi_remove_softintr(dp->soft_id);
5071                 } else if ((dp->misc_flag & GEM_NOINTR) == 0) {
5072                         ddi_remove_intr(dip, 0, dp->iblock_cookie);
5073                 } else {
5074                         /* stop interrupt watcher */
5075                         if (dp->intr_watcher_id) {
5076                                 while (untimeout(dp->intr_watcher_id) == -1)
5077                                         ;
5078                                 dp->intr_watcher_id = 0;
5079                         }
5080                 }
5081 
5082                 /* release NDD resources */
5083                 gem_nd_cleanup(dp);
5084                 /* release buffers, descriptors and dma resources */
5085                 gem_free_memory(dp);
5086 
5087                 /* release locks and condition variables */
5088                 mutex_destroy(&dp->xmitlock);
5089                 mutex_destroy(&dp->intrlock);
5090                 cv_destroy(&dp->tx_drain_cv);
5091 
5092                 /* release basic memory resources */
5093                 tmp = dp->next;
5094                 kmem_free((caddr_t)dp, GEM_LOCAL_DATA_SIZE(&dp->gc));
5095                 dp = tmp;
5096         }
5097 
5098         /* release common private memory for the nic */
5099         kmem_free(private, priv_size);
5100 
5101         /* release register mapping resources */
5102         ddi_regs_map_free(&rh);
5103 
5104         DPRINTF(2, (CE_CONT, "!%s%d: gem_do_detach: return: success",
5105             ddi_driver_name(dip), ddi_get_instance(dip)));
5106 
5107         return (DDI_SUCCESS);
5108 }
5109 
5110 int
5111 gem_suspend(dev_info_t *dip)
5112 {
5113         struct gem_dev  *dp;
5114 
5115         /*
5116          * stop the device
5117          */
5118         dp = GEM_GET_DEV(dip);
5119         ASSERT(dp);
5120 
5121         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
5122 
5123         for (; dp; dp = dp->next) {
5124 
5125                 /* stop mii link watcher */
5126                 gem_mii_stop(dp);
5127 
5128                 /* stop interrupt watcher for no-intr mode */
5129                 if (dp->misc_flag & GEM_NOINTR) {
5130                         if (dp->intr_watcher_id) {
5131                                 while (untimeout(dp->intr_watcher_id) == -1)
5132                                         ;
5133                         }
5134                         dp->intr_watcher_id = 0;
5135                 }
5136 
5137                 /* stop tx timeout watcher */
5138                 if (dp->timeout_id) {
5139                         while (untimeout(dp->timeout_id) == -1)
5140                                 ;
5141                         dp->timeout_id = 0;
5142                 }
5143 
5144                 /* make the nic state inactive */
5145                 mutex_enter(&dp->intrlock);
5146                 (void) gem_mac_stop(dp, 0);
5147                 ASSERT(!dp->mac_active);
5148 
5149                 /* no further register access */
5150                 dp->mac_suspended = B_TRUE;
5151                 mutex_exit(&dp->intrlock);
5152         }
5153 
5154         /* XXX - power down the nic */
5155 
5156         return (DDI_SUCCESS);
5157 }
5158 
5159 int
5160 gem_resume(dev_info_t *dip)
5161 {
5162         struct gem_dev  *dp;
5163 
5164         /*
5165          * restart the device
5166          */
5167         dp = GEM_GET_DEV(dip);
5168         ASSERT(dp);
5169 
5170         DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__));
5171 
5172         for (; dp; dp = dp->next) {
5173 
5174                 /*
5175                  * Bring up the nic after power up
5176                  */
5177 
5178                 /* gem_xxx.c layer to setup power management state. */
5179                 ASSERT(!dp->mac_active);
5180 
5181                 /* reset the chip, because we are just after power up. */
5182                 mutex_enter(&dp->intrlock);
5183 
5184                 dp->mac_suspended = B_FALSE;
5185                 dp->nic_state = NIC_STATE_STOPPED;
5186 
5187                 if ((*dp->gc.gc_reset_chip)(dp) != GEM_SUCCESS) {
5188                         cmn_err(CE_WARN, "%s: %s: failed to reset chip",
5189                             dp->name, __func__);
5190                         mutex_exit(&dp->intrlock);
5191                         goto err;
5192                 }
5193                 mutex_exit(&dp->intrlock);
5194 
5195                 /* initialize mii phy because we are just after power up */
5196                 if (dp->gc.gc_mii_init) {
5197                         (void) (*dp->gc.gc_mii_init)(dp);
5198                 }
5199 
5200                 if (dp->misc_flag & GEM_NOINTR) {
5201                         /*
5202                          * schedule first call of gem_intr_watcher
5203                          * instead of interrupts.
5204                          */
5205                         dp->intr_watcher_id =
5206                             timeout((void (*)(void *))gem_intr_watcher,
5207                             (void *)dp, drv_usectohz(3*1000000));
5208                 }
5209 
5210                 /* restart mii link watcher */
5211                 gem_mii_start(dp);
5212 
5213                 /* restart mac */
5214                 mutex_enter(&dp->intrlock);
5215 
5216                 if (gem_mac_init(dp) != GEM_SUCCESS) {
5217                         mutex_exit(&dp->intrlock);
5218                         goto err_reset;
5219                 }
5220                 dp->nic_state = NIC_STATE_INITIALIZED;
5221 
5222                 /* setup media mode if the link have been up */
5223                 if (dp->mii_state == MII_STATE_LINKUP) {
5224                         if ((dp->gc.gc_set_media)(dp) != GEM_SUCCESS) {
5225                                 mutex_exit(&dp->intrlock);
5226                                 goto err_reset;
5227                         }
5228                 }
5229 
5230                 /* enable mac address and rx filter */
5231                 dp->rxmode |= RXMODE_ENABLE;
5232                 if ((*dp->gc.gc_set_rx_filter)(dp) != GEM_SUCCESS) {
5233                         mutex_exit(&dp->intrlock);
5234                         goto err_reset;
5235                 }
5236                 dp->nic_state = NIC_STATE_ONLINE;
5237 
5238                 /* restart tx timeout watcher */
5239                 dp->timeout_id = timeout((void (*)(void *))gem_tx_timeout,
5240                     (void *)dp,
5241                     dp->gc.gc_tx_timeout_interval);
5242 
5243                 /* now the nic is fully functional */
5244                 if (dp->mii_state == MII_STATE_LINKUP) {
5245                         if (gem_mac_start(dp) != GEM_SUCCESS) {
5246                                 mutex_exit(&dp->intrlock);
5247                                 goto err_reset;
5248                         }
5249                 }
5250                 mutex_exit(&dp->intrlock);
5251         }
5252 
5253         return (DDI_SUCCESS);
5254 
5255 err_reset:
5256         if (dp->intr_watcher_id) {
5257                 while (untimeout(dp->intr_watcher_id) == -1)
5258                         ;
5259                 dp->intr_watcher_id = 0;
5260         }
5261         mutex_enter(&dp->intrlock);
5262         (*dp->gc.gc_reset_chip)(dp);
5263         dp->nic_state = NIC_STATE_STOPPED;
5264         mutex_exit(&dp->intrlock);
5265 
5266 err:
5267         return (DDI_FAILURE);
5268 }
5269 
5270 /*
5271  * misc routines for PCI
5272  */
5273 uint8_t
5274 gem_search_pci_cap(dev_info_t *dip,
5275                 ddi_acc_handle_t conf_handle, uint8_t target)
5276 {
5277         uint8_t         pci_cap_ptr;
5278         uint32_t        pci_cap;
5279 
5280         /* search power management capablities */
5281         pci_cap_ptr = pci_config_get8(conf_handle, PCI_CONF_CAP_PTR);
5282         while (pci_cap_ptr) {
5283                 /* read pci capability header */
5284                 pci_cap = pci_config_get32(conf_handle, pci_cap_ptr);
5285                 if ((pci_cap & 0xff) == target) {
5286                         /* found */
5287                         break;
5288                 }
5289                 /* get next_ptr */
5290                 pci_cap_ptr = (pci_cap >> 8) & 0xff;
5291         }
5292         return (pci_cap_ptr);
5293 }
5294 
5295 int
5296 gem_pci_set_power_state(dev_info_t *dip,
5297                 ddi_acc_handle_t conf_handle, uint_t new_mode)
5298 {
5299         uint8_t         pci_cap_ptr;
5300         uint32_t        pmcsr;
5301         uint_t          unit;
5302         const char      *drv_name;
5303 
5304         ASSERT(new_mode < 4);
5305 
5306         unit = ddi_get_instance(dip);
5307         drv_name = ddi_driver_name(dip);
5308 
5309         /* search power management capablities */
5310         pci_cap_ptr = gem_search_pci_cap(dip, conf_handle, PCI_CAP_ID_PM);
5311 
5312         if (pci_cap_ptr == 0) {
5313                 cmn_err(CE_CONT,
5314                     "!%s%d: doesn't have pci power management capability",
5315                     drv_name, unit);
5316                 return (DDI_FAILURE);
5317         }
5318 
5319         /* read power management capabilities */
5320         pmcsr = pci_config_get32(conf_handle, pci_cap_ptr + PCI_PMCSR);
5321 
5322         DPRINTF(0, (CE_CONT,
5323             "!%s%d: pmc found at 0x%x: pmcsr: 0x%08x",
5324             drv_name, unit, pci_cap_ptr, pmcsr));
5325 
5326         /*
5327          * Is the resuested power mode supported?
5328          */
5329         /* not yet */
5330 
5331         /*
5332          * move to new mode
5333          */
5334         pmcsr = (pmcsr & ~PCI_PMCSR_STATE_MASK) | new_mode;
5335         pci_config_put32(conf_handle, pci_cap_ptr + PCI_PMCSR, pmcsr);
5336 
5337         return (DDI_SUCCESS);
5338 }
5339 
5340 /*
5341  * select suitable register for by specified address space or register
5342  * offset in PCI config space
5343  */
5344 int
5345 gem_pci_regs_map_setup(dev_info_t *dip, uint32_t which, uint32_t mask,
5346         struct ddi_device_acc_attr *attrp,
5347         caddr_t *basep, ddi_acc_handle_t *hp)
5348 {
5349         struct pci_phys_spec    *regs;
5350         uint_t          len;
5351         uint_t          unit;
5352         uint_t          n;
5353         uint_t          i;
5354         int             ret;
5355         const char      *drv_name;
5356 
5357         unit = ddi_get_instance(dip);
5358         drv_name = ddi_driver_name(dip);
5359 
5360         /* Search IO-range or memory-range to be mapped */
5361         regs = NULL;
5362         len  = 0;
5363 
5364         if ((ret = ddi_prop_lookup_int_array(
5365             DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
5366             "reg", (void *)&regs, &len)) != DDI_PROP_SUCCESS) {
5367                 cmn_err(CE_WARN,
5368                     "!%s%d: failed to get reg property (ret:%d)",
5369                     drv_name, unit, ret);
5370                 return (DDI_FAILURE);
5371         }
5372         n = len / (sizeof (struct pci_phys_spec) / sizeof (int));
5373 
5374         ASSERT(regs != NULL && len > 0);
5375 
5376 #if GEM_DEBUG_LEVEL > 0
5377         for (i = 0; i < n; i++) {
5378                 cmn_err(CE_CONT,
5379                     "!%s%d: regs[%d]: %08x.%08x.%08x.%08x.%08x",
5380                     drv_name, unit, i,
5381                     regs[i].pci_phys_hi,
5382                     regs[i].pci_phys_mid,
5383                     regs[i].pci_phys_low,
5384                     regs[i].pci_size_hi,
5385                     regs[i].pci_size_low);
5386         }
5387 #endif
5388         for (i = 0; i < n; i++) {
5389                 if ((regs[i].pci_phys_hi & mask) == which) {
5390                         /* it's the requested space */
5391                         ddi_prop_free(regs);
5392                         goto address_range_found;
5393                 }
5394         }
5395         ddi_prop_free(regs);
5396         return (DDI_FAILURE);
5397 
5398 address_range_found:
5399         if ((ret = ddi_regs_map_setup(dip, i, basep, 0, 0, attrp, hp))
5400             != DDI_SUCCESS) {
5401                 cmn_err(CE_CONT,
5402                     "!%s%d: ddi_regs_map_setup failed (ret:%d)",
5403                     drv_name, unit, ret);
5404         }
5405 
5406         return (ret);
5407 }
5408 
5409 void
5410 gem_mod_init(struct dev_ops *dop, char *name)
5411 {
5412         mac_init_ops(dop, name);
5413 }
5414 
5415 void
5416 gem_mod_fini(struct dev_ops *dop)
5417 {
5418         mac_fini_ops(dop);
5419 }