1 /* 2 * usbgem.c: General USB to Fast Ethernet mac driver framework 3 * 4 * Copyright (c) 2002-2012 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 2019 Joyent, Inc. 36 * Copyright 2022 Garrett D'Amore 37 */ 38 39 /* 40 * Change log 41 */ 42 43 /* 44 * TODO: 45 * implement DELAYED_START 46 */ 47 48 /* 49 * System Header files. 50 */ 51 #include <sys/types.h> 52 #include <sys/conf.h> 53 #include <sys/debug.h> 54 #include <sys/kmem.h> 55 #include <sys/vtrace.h> 56 #include <sys/ethernet.h> 57 #include <sys/modctl.h> 58 #include <sys/errno.h> 59 #include <sys/ddi.h> 60 #include <sys/sunddi.h> 61 #include <sys/stream.h> /* required for MBLK* */ 62 #include <sys/strsun.h> /* required for mionack() */ 63 #include <sys/byteorder.h> 64 65 #include <sys/usb/usba.h> 66 #include <inet/common.h> 67 #include <inet/led.h> 68 #include <inet/mi.h> 69 #include <inet/nd.h> 70 71 /* supplement definitions */ 72 extern const char *usb_str_cr(usb_cr_t); 73 74 #include <sys/note.h> 75 76 #include "usbgem_mii.h" 77 #include "usbgem.h" 78 79 char ident[] = "usb general ethernet mac driver v" VERSION; 80 81 /* Debugging support */ 82 #ifdef USBGEM_DEBUG_LEVEL 83 static int usbgem_debug = USBGEM_DEBUG_LEVEL; 84 #define DPRINTF(n, args) if (usbgem_debug > (n)) cmn_err args 85 #else 86 #define DPRINTF(n, args) 87 #endif 88 89 /* 90 * Useful macros and typedefs 91 */ 92 #define ROUNDUP(x, a) (((x) + (a) - 1) & ~((a) - 1)) 93 #define DEFAULT_PIPE(dp) ((dp)->reg_data->dev_default_ph) 94 #define VTAG_SIZE 4 95 #define BOOLEAN(x) ((x) != 0) 96 /* 97 * configuration parameters 98 */ 99 #define USBDRV_MAJOR_VER 2 100 #define USBDRV_MINOR_VER 0 101 102 #define ETHERHEADERL (sizeof (struct ether_header)) 103 #define MAXPKTLEN(dp) ((dp)->mtu + ETHERHEADERL) 104 #define MAXPKTBUF(dp) ((dp)->mtu + ETHERHEADERL + ETHERFCSL) 105 106 #define WATCH_INTERVAL_FAST drv_usectohz(100*1000) 107 108 #define STOP_GRACEFUL B_TRUE 109 110 /* 111 * Private functions 112 */ 113 static int usbgem_open_pipes(struct usbgem_dev *dp); 114 static int usbgem_close_pipes(struct usbgem_dev *dp); 115 static void usbgem_intr_cb(usb_pipe_handle_t, usb_intr_req_t *); 116 static void usbgem_bulkin_cb(usb_pipe_handle_t, usb_bulk_req_t *); 117 static void usbgem_bulkout_cb(usb_pipe_handle_t, usb_bulk_req_t *); 118 119 static int usbgem_mii_start(struct usbgem_dev *); 120 static void usbgem_mii_stop(struct usbgem_dev *); 121 122 /* local buffer management */ 123 static int usbgem_init_rx_buf(struct usbgem_dev *); 124 125 /* internal mac interfaces */ 126 static void usbgem_tx_timeout(struct usbgem_dev *); 127 static void usbgem_mii_link_watcher(struct usbgem_dev *); 128 static int usbgem_mac_init(struct usbgem_dev *); 129 static int usbgem_mac_start(struct usbgem_dev *); 130 static int usbgem_mac_stop(struct usbgem_dev *, int, boolean_t); 131 static void usbgem_mac_ioctl(struct usbgem_dev *, queue_t *, mblk_t *); 132 133 int usbgem_speed_value[] = {10, 100, 1000}; 134 135 static int usbgem_ctrl_retry = 5; 136 137 /* usb event support */ 138 static int usbgem_disconnect_cb(dev_info_t *dip); 139 static int usbgem_reconnect_cb(dev_info_t *dip); 140 int usbgem_suspend(dev_info_t *dip); 141 int usbgem_resume(dev_info_t *dip); 142 143 static uint8_t usbgem_bcastaddr[] = { 144 0xff, 0xff, 0xff, 0xff, 0xff, 0xff 145 }; 146 147 extern struct mod_ops mod_miscops; 148 149 static struct modlmisc modlmisc = { 150 &mod_miscops, 151 "usbgem v" VERSION, 152 }; 153 154 static struct modlinkage modlinkage = { 155 MODREV_1, &modlmisc, NULL 156 }; 157 158 /* 159 * _init : done 160 */ 161 int 162 _init(void) 163 { 164 int status; 165 166 DPRINTF(2, (CE_CONT, "!usbgem: _init: called")); 167 status = mod_install(&modlinkage); 168 169 return (status); 170 } 171 172 /* 173 * _fini : done 174 */ 175 int 176 _fini(void) 177 { 178 int status; 179 180 DPRINTF(2, (CE_CONT, "!usbgem: _fini: called")); 181 status = mod_remove(&modlinkage); 182 return (status); 183 } 184 185 int 186 _info(struct modinfo *modinfop) 187 { 188 return (mod_info(&modlinkage, modinfop)); 189 } 190 191 /* ============================================================== */ 192 /* 193 * Ether CRC calculation utilities 194 */ 195 /* ============================================================== */ 196 /* 197 * Ether CRC calculation according to 21143 data sheet 198 */ 199 #define CRC32_POLY_LE 0xedb88320 200 uint32_t 201 usbgem_ether_crc_le(const uint8_t *addr) 202 { 203 int idx; 204 int bit; 205 uint_t data; 206 uint32_t crc = 0xffffffff; 207 208 crc = 0xffffffff; 209 for (idx = 0; idx < ETHERADDRL; idx++) { 210 for (data = *addr++, bit = 0; bit < 8; bit++, data >>= 1) { 211 crc = (crc >> 1) ^ 212 (((crc ^ data) & 1) ? CRC32_POLY_LE : 0); 213 } 214 } 215 return (crc); 216 } 217 218 #define CRC32_POLY_BE 0x04c11db7 219 uint32_t 220 usbgem_ether_crc_be(const uint8_t *addr) 221 { 222 int idx; 223 int bit; 224 uint_t data; 225 uint32_t crc; 226 227 crc = 0xffffffff; 228 for (idx = 0; idx < ETHERADDRL; idx++) { 229 for (data = *addr++, bit = 0; bit < 8; bit++, data >>= 1) { 230 crc = (crc << 1) ^ 231 ((((crc >> 31) ^ data) & 1) ? CRC32_POLY_BE : 0); 232 } 233 } 234 return (crc); 235 } 236 237 int 238 usbgem_prop_get_int(struct usbgem_dev *dp, char *prop_template, int def_val) 239 { 240 char propname[32]; 241 242 (void) sprintf(propname, prop_template, dp->name); 243 244 return (ddi_prop_get_int(DDI_DEV_T_ANY, dp->dip, 245 DDI_PROP_DONTPASS, propname, def_val)); 246 } 247 248 static int 249 usbgem_population(uint32_t x) 250 { 251 int i; 252 int cnt; 253 254 cnt = 0; 255 for (i = 0; i < 32; i++) { 256 if (x & (1 << i)) { 257 cnt++; 258 } 259 } 260 return (cnt); 261 } 262 263 static clock_t 264 usbgem_timestamp_nz() 265 { 266 clock_t now; 267 now = ddi_get_lbolt(); 268 return (now ? now : (clock_t)1); 269 } 270 271 /* ============================================================== */ 272 /* 273 * hardware operations 274 */ 275 /* ============================================================== */ 276 static int 277 usbgem_hal_reset_chip(struct usbgem_dev *dp) 278 { 279 int err; 280 281 sema_p(&dp->hal_op_lock); 282 err = (*dp->ugc.usbgc_reset_chip)(dp); 283 sema_v(&dp->hal_op_lock); 284 return (err); 285 } 286 287 static int 288 usbgem_hal_init_chip(struct usbgem_dev *dp) 289 { 290 int err; 291 292 sema_p(&dp->hal_op_lock); 293 err = (*dp->ugc.usbgc_init_chip)(dp); 294 sema_v(&dp->hal_op_lock); 295 return (err); 296 } 297 298 static int 299 usbgem_hal_attach_chip(struct usbgem_dev *dp) 300 { 301 int err; 302 303 sema_p(&dp->hal_op_lock); 304 err = (*dp->ugc.usbgc_attach_chip)(dp); 305 sema_v(&dp->hal_op_lock); 306 return (err); 307 } 308 309 static int 310 usbgem_hal_set_rx_filter(struct usbgem_dev *dp) 311 { 312 int err; 313 314 sema_p(&dp->hal_op_lock); 315 err = (*dp->ugc.usbgc_set_rx_filter)(dp); 316 sema_v(&dp->hal_op_lock); 317 return (err); 318 } 319 320 static int 321 usbgem_hal_set_media(struct usbgem_dev *dp) 322 { 323 int err; 324 325 sema_p(&dp->hal_op_lock); 326 err = (*dp->ugc.usbgc_set_media)(dp); 327 sema_v(&dp->hal_op_lock); 328 return (err); 329 } 330 331 static int 332 usbgem_hal_start_chip(struct usbgem_dev *dp) 333 { 334 int err; 335 336 sema_p(&dp->hal_op_lock); 337 err = (*dp->ugc.usbgc_start_chip)(dp); 338 sema_v(&dp->hal_op_lock); 339 return (err); 340 } 341 342 static int 343 usbgem_hal_stop_chip(struct usbgem_dev *dp) 344 { 345 int err; 346 347 sema_p(&dp->hal_op_lock); 348 err = (*dp->ugc.usbgc_stop_chip)(dp); 349 sema_v(&dp->hal_op_lock); 350 return (err); 351 } 352 353 static int 354 usbgem_hal_get_stats(struct usbgem_dev *dp) 355 { 356 int err; 357 358 sema_p(&dp->hal_op_lock); 359 err = (*dp->ugc.usbgc_get_stats)(dp); 360 sema_v(&dp->hal_op_lock); 361 return (err); 362 } 363 364 365 /* ============================================================== */ 366 /* 367 * USB pipe management 368 */ 369 /* ============================================================== */ 370 static boolean_t 371 usbgem_rx_start_unit(struct usbgem_dev *dp, usb_bulk_req_t *req) 372 { 373 mblk_t *mp; 374 int err; 375 usb_flags_t flags; 376 377 ASSERT(req); 378 379 mp = allocb(dp->rx_buf_len, BPRI_MED); 380 if (mp == NULL) { 381 cmn_err(CE_WARN, "!%s: %s: failed to allocate mblk", 382 dp->name, __func__); 383 goto err; 384 } 385 386 req->bulk_len = dp->rx_buf_len; 387 req->bulk_data = mp; 388 req->bulk_client_private = (usb_opaque_t)dp; 389 req->bulk_timeout = 0; 390 req->bulk_attributes = USB_ATTRS_SHORT_XFER_OK; 391 req->bulk_cb = usbgem_bulkin_cb; 392 req->bulk_exc_cb = usbgem_bulkin_cb; 393 req->bulk_completion_reason = 0; 394 req->bulk_cb_flags = 0; 395 396 flags = 0; 397 err = usb_pipe_bulk_xfer(dp->bulkin_pipe, req, flags); 398 399 if (err != USB_SUCCESS) { 400 cmn_err(CE_WARN, "%s: failed to bulk_xfer for rx, err:%d", 401 dp->name, err); 402 403 /* free req and mp */ 404 usb_free_bulk_req(req); 405 goto err; 406 } 407 return (B_TRUE); 408 err: 409 return (B_FALSE); 410 } 411 412 /* ============================================================== */ 413 /* 414 * Rx/Tx buffer management 415 */ 416 /* ============================================================== */ 417 static int 418 usbgem_init_rx_buf(struct usbgem_dev *dp) 419 { 420 int i; 421 usb_bulk_req_t *req; 422 423 ASSERT(dp->mac_state == MAC_STATE_ONLINE); 424 425 for (i = 0; i < dp->ugc.usbgc_rx_list_max; i++) { 426 req = usb_alloc_bulk_req(dp->dip, 0, USB_FLAGS_SLEEP); 427 if (req == NULL) { 428 cmn_err(CE_WARN, 429 "!%s: %s: failed to allocate bulkreq for rx", 430 dp->name, __func__); 431 return (USB_FAILURE); 432 } 433 if (!usbgem_rx_start_unit(dp, req)) { 434 return (USB_FAILURE); 435 } 436 mutex_enter(&dp->rxlock); 437 dp->rx_busy_cnt++; 438 mutex_exit(&dp->rxlock); 439 } 440 return (USB_SUCCESS); 441 } 442 443 /* ============================================================== */ 444 /* 445 * memory resource management 446 */ 447 /* ============================================================== */ 448 static int 449 usbgem_free_memory(struct usbgem_dev *dp) 450 { 451 usb_bulk_req_t *req; 452 453 /* free all tx requst structure */ 454 while ((req = dp->tx_free_list) != NULL) { 455 dp->tx_free_list = 456 (usb_bulk_req_t *)req->bulk_client_private; 457 req->bulk_data = NULL; 458 usb_free_bulk_req(req); 459 } 460 return (USB_SUCCESS); 461 } 462 463 static int 464 usbgem_alloc_memory(struct usbgem_dev *dp) 465 { 466 int i; 467 usb_bulk_req_t *req; 468 469 /* allocate tx requests */ 470 dp->tx_free_list = NULL; 471 for (i = 0; i < dp->ugc.usbgc_tx_list_max; i++) { 472 req = usb_alloc_bulk_req(dp->dip, 0, USB_FLAGS_SLEEP); 473 if (req == NULL) { 474 cmn_err(CE_WARN, 475 "%s:%s failed to allocate tx requests", 476 dp->name, __func__); 477 478 /* free partially allocated tx requests */ 479 (void) usbgem_free_memory(dp); 480 return (USB_FAILURE); 481 } 482 483 /* add the new one allocated into tx free list */ 484 req->bulk_client_private = (usb_opaque_t)dp->tx_free_list; 485 dp->tx_free_list = req; 486 } 487 488 return (USB_SUCCESS); 489 } 490 491 /* ========================================================== */ 492 /* 493 * Start transmission. 494 * Return zero on success, 495 */ 496 /* ========================================================== */ 497 498 #ifdef TXTIMEOUT_TEST 499 static int usbgem_send_cnt = 0; 500 #endif 501 502 /* 503 * usbgem_send is used only to send data packet into ethernet line. 504 */ 505 static mblk_t * 506 usbgem_send_common(struct usbgem_dev *dp, mblk_t *mp, uint32_t flags) 507 { 508 int err; 509 mblk_t *new; 510 usb_bulk_req_t *req; 511 int mcast; 512 int bcast; 513 int len; 514 boolean_t intr; 515 usb_flags_t usb_flags = 0; 516 #ifdef USBGEM_DEBUG_LEVEL 517 usb_pipe_state_t p_state; 518 #endif 519 DPRINTF(2, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 520 521 intr = (flags & 1) != 0; 522 len = msgdsize(mp); 523 bcast = 0; 524 mcast = 0; 525 if (mp->b_rptr[0] & 1) { 526 if (bcmp(mp->b_rptr, &usbgem_bcastaddr, ETHERADDRL) == 0) { 527 bcast = 1; 528 } else { 529 mcast = 1; 530 } 531 } 532 new = (*dp->ugc.usbgc_tx_make_packet)(dp, mp); 533 if (new == NULL) { 534 /* 535 * no memory resource. we don't stop downstream, 536 * we just discard the packet. 537 */ 538 DPRINTF(0, (CE_CONT, "!%s: %s: no memory", 539 dp->name, __func__)); 540 freemsg(mp); 541 542 mutex_enter(&dp->txlock); 543 dp->stats.noxmtbuf++; 544 dp->stats.errxmt++; 545 mutex_exit(&dp->txlock); 546 547 return (NULL); 548 } 549 550 ASSERT(new->b_cont == NULL); 551 552 mutex_enter(&dp->txlock); 553 if (dp->tx_free_list == NULL) { 554 /* 555 * no tx free slot 556 */ 557 ASSERT(dp->tx_busy_cnt == dp->ugc.usbgc_tx_list_max); 558 mutex_exit(&dp->txlock); 559 560 DPRINTF(4, (CE_CONT, "!%s: %s: no free slot", 561 dp->name, __func__)); 562 if (new && new != mp) { 563 /* free reallocated message */ 564 freemsg(new); 565 } 566 return (mp); 567 } 568 req = dp->tx_free_list; 569 dp->tx_free_list = (usb_bulk_req_t *)req->bulk_client_private; 570 dp->tx_busy_cnt++; 571 572 if (dp->tx_free_list == NULL) { 573 intr = B_TRUE; 574 } 575 if (intr) { 576 dp->tx_intr_pended++; 577 } 578 DB_TCI(new) = intr; 579 #ifdef USBGEM_DEBUG_LEVEL 580 new->b_datap->db_cksum32 = dp->tx_seq_num; 581 dp->tx_seq_num++; 582 #endif 583 dp->stats.obytes += len; 584 dp->stats.opackets++; 585 if (bcast | mcast) { 586 dp->stats.obcast += bcast; 587 dp->stats.omcast += mcast; 588 } 589 mutex_exit(&dp->txlock); 590 591 DPRINTF(2, (CE_CONT, "!%s: %s: sending", dp->name, __func__)); 592 593 req->bulk_len = (long)new->b_wptr - (long)new->b_rptr; 594 req->bulk_data = new; 595 req->bulk_client_private = (usb_opaque_t)dp; 596 req->bulk_timeout = dp->bulkout_timeout; /* in second */ 597 req->bulk_attributes = 0; 598 req->bulk_cb = usbgem_bulkout_cb; 599 req->bulk_exc_cb = usbgem_bulkout_cb; 600 req->bulk_completion_reason = 0; 601 req->bulk_cb_flags = 0; 602 603 if (intr) { 604 usb_flags = USB_FLAGS_SLEEP; 605 } 606 if ((err = usb_pipe_bulk_xfer(dp->bulkout_pipe, req, usb_flags)) 607 != USB_SUCCESS) { 608 609 /* failed to transfer the packet, discard it. */ 610 freemsg(new); 611 req->bulk_data = NULL; 612 613 /* recycle the request block */ 614 mutex_enter(&dp->txlock); 615 dp->tx_busy_cnt--; 616 req->bulk_client_private = (usb_opaque_t)dp->tx_free_list; 617 dp->tx_free_list = req; 618 mutex_exit(&dp->txlock); 619 620 cmn_err(CE_NOTE, 621 "%s: %s: usb_pipe_bulk_xfer: failed: err:%d", 622 dp->name, __func__, err); 623 624 /* we use another flag to indicate error state. */ 625 if (dp->fatal_error == (clock_t)0) { 626 dp->fatal_error = usbgem_timestamp_nz(); 627 } 628 } else { 629 /* record the start time */ 630 dp->tx_start_time = ddi_get_lbolt(); 631 } 632 633 if (err == USB_SUCCESS && (usb_flags & USB_FLAGS_SLEEP)) { 634 usbgem_bulkout_cb(dp->bulkout_pipe, req); 635 } 636 637 if (new != mp) { 638 freemsg(mp); 639 } 640 return (NULL); 641 } 642 643 int 644 usbgem_restart_nic(struct usbgem_dev *dp) 645 { 646 int ret; 647 int flags = 0; 648 649 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 650 651 ASSERT(dp->mac_state != MAC_STATE_DISCONNECTED); 652 653 /* 654 * ensure to stop the nic 655 */ 656 if (dp->mac_state == MAC_STATE_ONLINE) { 657 (void) usbgem_mac_stop(dp, MAC_STATE_STOPPED, STOP_GRACEFUL); 658 } 659 660 /* now the nic become quiescent, reset the chip */ 661 if (usbgem_hal_reset_chip(dp) != USB_SUCCESS) { 662 cmn_err(CE_WARN, "%s: %s: failed to reset chip", 663 dp->name, __func__); 664 goto err; 665 } 666 667 /* 668 * restore the nic state step by step 669 */ 670 if (dp->nic_state < NIC_STATE_INITIALIZED) { 671 goto done; 672 } 673 674 if (usbgem_mac_init(dp) != USB_SUCCESS) { 675 cmn_err(CE_WARN, "%s: %s: failed to initialize chip", 676 dp->name, __func__); 677 goto err; 678 } 679 680 /* setup mac address and enable rx filter */ 681 sema_p(&dp->rxfilter_lock); 682 dp->rxmode |= RXMODE_ENABLE; 683 ret = usbgem_hal_set_rx_filter(dp); 684 sema_v(&dp->rxfilter_lock); 685 if (ret != USB_SUCCESS) { 686 goto err; 687 } 688 689 /* 690 * update the link state asynchronously 691 */ 692 cv_signal(&dp->link_watcher_wait_cv); 693 694 /* 695 * XXX - a panic happened because of linkdown. 696 * We must check mii_state here, because the link can be down just 697 * before the restart event happen. If the link is down now, 698 * gem_mac_start() will be called from gem_mii_link_check() when 699 * the link become up later. 700 */ 701 if (dp->mii_state == MII_STATE_LINKUP) { 702 if (usbgem_hal_set_media(dp) != USB_SUCCESS) { 703 goto err; 704 } 705 if (dp->nic_state < NIC_STATE_ONLINE) { 706 goto done; 707 } 708 709 (void) usbgem_mac_start(dp); 710 711 } 712 done: 713 return (USB_SUCCESS); 714 err: 715 return (USB_FAILURE); 716 } 717 718 static void 719 usbgem_tx_timeout(struct usbgem_dev *dp) 720 { 721 uint_t rwlock; 722 clock_t now; 723 724 for (; ; ) { 725 mutex_enter(&dp->tx_watcher_lock); 726 (void) cv_timedwait(&dp->tx_watcher_cv, &dp->tx_watcher_lock, 727 dp->tx_watcher_interval + ddi_get_lbolt()); 728 mutex_exit(&dp->tx_watcher_lock); 729 730 if (dp->tx_watcher_stop) { 731 break; 732 } 733 734 now = ddi_get_lbolt(); 735 736 rwlock = RW_READER; 737 again: 738 rw_enter(&dp->dev_state_lock, rwlock); 739 740 if ((dp->mac_state != MAC_STATE_DISCONNECTED && 741 dp->fatal_error && 742 now - dp->fatal_error >= dp->ugc.usbgc_tx_timeout) || 743 (dp->mac_state == MAC_STATE_ONLINE && 744 dp->mii_state == MII_STATE_LINKUP && 745 dp->tx_busy_cnt != 0 && 746 now - dp->tx_start_time >= dp->ugc.usbgc_tx_timeout)) { 747 if (rwlock == RW_READER) { 748 /* 749 * Upgrade dev_state_lock from shared mode 750 * to exclusive mode to restart nic 751 */ 752 rwlock = RW_WRITER; 753 rw_exit(&dp->dev_state_lock); 754 goto again; 755 } 756 cmn_err(CE_WARN, "%s: %s: restarting the nic:" 757 " fatal_error:%ld nic_state:%d" 758 " mac_state:%d starttime:%ld", 759 dp->name, __func__, 760 dp->fatal_error ? now - dp->fatal_error: 0, 761 dp->nic_state, dp->mac_state, 762 dp->tx_busy_cnt ? now - dp->tx_start_time : 0); 763 764 (void) usbgem_restart_nic(dp); 765 } 766 767 rw_exit(&dp->dev_state_lock); 768 } 769 } 770 771 static int 772 usbgem_tx_watcher_start(struct usbgem_dev *dp) 773 { 774 int err; 775 kthread_t *wdth; 776 777 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 778 779 /* make a first call of uwgem_lw_link_check() */ 780 dp->tx_watcher_stop = 0; 781 dp->tx_watcher_interval = drv_usectohz(1000*1000); 782 783 wdth = thread_create(NULL, 0, usbgem_tx_timeout, dp, 0, &p0, 784 TS_RUN, minclsyspri); 785 if (wdth == NULL) { 786 cmn_err(CE_WARN, 787 "!%s: %s: failed to create a tx_watcher thread", 788 dp->name, __func__); 789 return (USB_FAILURE); 790 } 791 dp->tx_watcher_did = wdth->t_did; 792 793 return (USB_SUCCESS); 794 } 795 796 static void 797 usbgem_tx_watcher_stop(struct usbgem_dev *dp) 798 { 799 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 800 if (dp->tx_watcher_did) { 801 /* Ensure timer routine stopped */ 802 dp->tx_watcher_stop = 1; 803 cv_signal(&dp->tx_watcher_cv); 804 thread_join(dp->tx_watcher_did); 805 dp->tx_watcher_did = 0; 806 } 807 } 808 809 /* ================================================================== */ 810 /* 811 * Callback handlers 812 */ 813 /* ================================================================== */ 814 static void 815 usbgem_bulkin_cb(usb_pipe_handle_t pipe, usb_bulk_req_t *req) 816 { 817 mblk_t *newmp; 818 mblk_t *mp; 819 mblk_t *tp; 820 uint64_t len = 0; 821 int pkts = 0; 822 int bcast = 0; 823 int mcast = 0; 824 boolean_t busy; 825 struct usbgem_dev *dp; 826 827 dp = (struct usbgem_dev *)req->bulk_client_private; 828 mp = req->bulk_data; 829 req->bulk_data = NULL; 830 831 DPRINTF(2, (CE_CONT, "!%s: %s: mp:%p, cr:%s(%d)", 832 dp->name, __func__, mp, 833 usb_str_cr(req->bulk_completion_reason), 834 req->bulk_completion_reason)); 835 836 /* 837 * we cannot acquire dev_state_lock because the routine 838 * must be executed during usbgem_mac_stop() to avoid 839 * dead lock. 840 * we use a simle membar operation to get the state correctly. 841 */ 842 membar_consumer(); 843 844 if (req->bulk_completion_reason == USB_CR_OK && 845 dp->nic_state == NIC_STATE_ONLINE) { 846 newmp = (*dp->ugc.usbgc_rx_make_packet)(dp, mp); 847 848 if (newmp != mp) { 849 /* the message has been reallocated, free old one */ 850 freemsg(mp); 851 } 852 853 /* the message may includes one or more ethernet packets */ 854 for (tp = newmp; tp; tp = tp->b_next) { 855 len += (uintptr_t)tp->b_wptr - (uintptr_t)tp->b_rptr; 856 pkts++; 857 if (tp->b_rptr[0] & 1) { 858 if (bcmp(tp->b_rptr, &usbgem_bcastaddr, 859 ETHERADDRL) == 0) { 860 bcast++; 861 } else { 862 mcast++; 863 } 864 } 865 } 866 867 /* send up if it is a valid packet */ 868 mac_rx(dp->mh, NULL, newmp); 869 } else { 870 freemsg(mp); 871 len = 0; 872 } 873 874 mutex_enter(&dp->rxlock); 875 /* update rx_active */ 876 if (dp->rx_active) { 877 dp->rx_active = dp->mac_state == MAC_STATE_ONLINE; 878 } 879 880 dp->stats.rbytes += len; 881 dp->stats.rpackets += pkts; 882 if (bcast | mcast) { 883 dp->stats.rbcast += bcast; 884 dp->stats.rmcast += mcast; 885 } 886 mutex_exit(&dp->rxlock); 887 888 if (dp->rx_active) { 889 /* prepare to receive the next packets */ 890 if (usbgem_rx_start_unit(dp, req)) { 891 /* we successed */ 892 goto done; 893 } 894 cmn_err(CE_WARN, 895 "!%s: %s: failed to fill next rx packet", 896 dp->name, __func__); 897 /* 898 * we use another flag to indicate error state. 899 * if we acquire dev_state_lock for RW_WRITER here, 900 * usbgem_mac_stop() may hang. 901 */ 902 if (dp->fatal_error == (clock_t)0) { 903 dp->fatal_error = usbgem_timestamp_nz(); 904 } 905 } else { 906 /* no need to prepare the next packets */ 907 usb_free_bulk_req(req); 908 } 909 910 mutex_enter(&dp->rxlock); 911 dp->rx_active = B_FALSE; 912 dp->rx_busy_cnt--; 913 if (dp->rx_busy_cnt == 0) { 914 /* wake up someone waits for me */ 915 cv_broadcast(&dp->rx_drain_cv); 916 } 917 mutex_exit(&dp->rxlock); 918 done: 919 ; 920 } 921 922 static void 923 usbgem_bulkout_cb(usb_pipe_handle_t pipe, usb_bulk_req_t *req) 924 { 925 boolean_t intr; 926 boolean_t tx_sched; 927 struct usbgem_dev *dp; 928 929 dp = (struct usbgem_dev *)req->bulk_client_private; 930 tx_sched = B_FALSE; 931 932 DPRINTF(2, (CE_CONT, 933 "!%s: %s: cr:%s(%d) cb_flags:0x%x head:%d tail:%d", 934 dp->name, __func__, 935 usb_str_cr(req->bulk_completion_reason), 936 req->bulk_completion_reason, 937 req->bulk_cb_flags, 938 dp->tx_busy_cnt)); 939 940 /* we have finished to transfer the packet into tx fifo */ 941 intr = DB_TCI(req->bulk_data); 942 freemsg(req->bulk_data); 943 944 if (req->bulk_completion_reason != USB_CR_OK && 945 dp->fatal_error == (clock_t)0) { 946 dp->fatal_error = usbgem_timestamp_nz(); 947 } 948 949 mutex_enter(&dp->txlock); 950 951 if (intr) { 952 ASSERT(dp->tx_intr_pended > 0); 953 /* find the last interrupt we have scheduled */ 954 if (--(dp->tx_intr_pended) == 0) { 955 tx_sched = B_TRUE; 956 } 957 } 958 959 ASSERT(dp->tx_busy_cnt > 0); 960 req->bulk_client_private = (usb_opaque_t)dp->tx_free_list; 961 dp->tx_free_list = req; 962 dp->tx_busy_cnt--; 963 964 #ifdef CONFIG_TX_LIMITER 965 if (tx_sched) { 966 dp->tx_max_packets = 967 min(dp->tx_max_packets + 1, dp->ugc.usbgc_tx_list_max); 968 } 969 #endif 970 if (dp->mac_state != MAC_STATE_ONLINE && dp->tx_busy_cnt == 0) { 971 cv_broadcast(&dp->tx_drain_cv); 972 } 973 974 mutex_exit(&dp->txlock); 975 976 if (tx_sched) { 977 mac_tx_update(dp->mh); 978 } 979 } 980 981 static void 982 usbgem_intr_cb(usb_pipe_handle_t ph, usb_intr_req_t *req) 983 { 984 struct usbgem_dev *dp; 985 986 dp = (struct usbgem_dev *)req->intr_client_private; 987 dp->stats.intr++; 988 989 if (req->intr_completion_reason == USB_CR_OK) { 990 (*dp->ugc.usbgc_interrupt)(dp, req->intr_data); 991 } 992 993 /* free the request and data */ 994 usb_free_intr_req(req); 995 } 996 997 /* ======================================================================== */ 998 /* 999 * MII support routines 1000 */ 1001 /* ======================================================================== */ 1002 static void 1003 usbgem_choose_forcedmode(struct usbgem_dev *dp) 1004 { 1005 /* choose media mode */ 1006 if (dp->anadv_1000fdx || dp->anadv_1000hdx) { 1007 dp->speed = USBGEM_SPD_1000; 1008 dp->full_duplex = dp->anadv_1000fdx; 1009 } else if (dp->anadv_100fdx || dp->anadv_100t4) { 1010 dp->speed = USBGEM_SPD_100; 1011 dp->full_duplex = B_TRUE; 1012 } else if (dp->anadv_100hdx) { 1013 dp->speed = USBGEM_SPD_100; 1014 dp->full_duplex = B_FALSE; 1015 } else { 1016 dp->speed = USBGEM_SPD_10; 1017 dp->full_duplex = dp->anadv_10fdx; 1018 } 1019 } 1020 1021 static uint16_t 1022 usbgem_mii_read(struct usbgem_dev *dp, uint_t reg, int *errp) 1023 { 1024 uint16_t val; 1025 1026 sema_p(&dp->hal_op_lock); 1027 val = (*dp->ugc.usbgc_mii_read)(dp, reg, errp); 1028 sema_v(&dp->hal_op_lock); 1029 1030 return (val); 1031 } 1032 1033 static void 1034 usbgem_mii_write(struct usbgem_dev *dp, uint_t reg, uint16_t val, int *errp) 1035 { 1036 sema_p(&dp->hal_op_lock); 1037 (*dp->ugc.usbgc_mii_write)(dp, reg, val, errp); 1038 sema_v(&dp->hal_op_lock); 1039 } 1040 1041 static int 1042 usbgem_mii_probe(struct usbgem_dev *dp) 1043 { 1044 int err; 1045 1046 err = (*dp->ugc.usbgc_mii_probe)(dp); 1047 return (err); 1048 } 1049 1050 static int 1051 usbgem_mii_init(struct usbgem_dev *dp) 1052 { 1053 int err; 1054 1055 err = (*dp->ugc.usbgc_mii_init)(dp); 1056 return (err); 1057 } 1058 1059 #define fc_cap_decode(x) \ 1060 ((((x) & MII_ABILITY_PAUSE) != 0 ? 1 : 0) | \ 1061 (((x) & MII_ABILITY_ASM_DIR) != 0 ? 2 : 0)) 1062 1063 int 1064 usbgem_mii_config_default(struct usbgem_dev *dp, int *errp) 1065 { 1066 uint16_t mii_stat; 1067 uint16_t val; 1068 1069 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 1070 1071 /* 1072 * Configure bits in advertisement register 1073 */ 1074 mii_stat = dp->mii_status; 1075 1076 DPRINTF(1, (CE_CONT, "!%s: %s: MII_STATUS reg:%b", 1077 dp->name, __func__, mii_stat, MII_STATUS_BITS)); 1078 1079 if ((mii_stat & MII_STATUS_ABILITY_TECH) == 0) { 1080 /* it's funny */ 1081 cmn_err(CE_WARN, "!%s: wrong ability bits: mii_status:%b", 1082 dp->name, mii_stat, MII_STATUS_BITS); 1083 return (USB_FAILURE); 1084 } 1085 1086 /* Do not change the rest of ability bits in advert reg */ 1087 val = usbgem_mii_read(dp, MII_AN_ADVERT, errp) & ~MII_ABILITY_ALL; 1088 if (*errp != USB_SUCCESS) { 1089 goto usberr; 1090 } 1091 1092 DPRINTF(0, (CE_CONT, 1093 "!%s: %s: 100T4:%d 100F:%d 100H:%d 10F:%d 10H:%d", 1094 dp->name, __func__, 1095 dp->anadv_100t4, dp->anadv_100fdx, dp->anadv_100hdx, 1096 dp->anadv_10fdx, dp->anadv_10hdx)); 1097 1098 /* set technology bits */ 1099 if (dp->anadv_100t4) { 1100 val |= MII_ABILITY_100BASE_T4; 1101 } 1102 if (dp->anadv_100fdx) { 1103 val |= MII_ABILITY_100BASE_TX_FD; 1104 } 1105 if (dp->anadv_100hdx) { 1106 val |= MII_ABILITY_100BASE_TX; 1107 } 1108 if (dp->anadv_10fdx) { 1109 val |= MII_ABILITY_10BASE_T_FD; 1110 } 1111 if (dp->anadv_10hdx) { 1112 val |= MII_ABILITY_10BASE_T; 1113 } 1114 1115 /* set flow control capabilities */ 1116 if (dp->anadv_pause) { 1117 val |= MII_ABILITY_PAUSE; 1118 } 1119 if (dp->anadv_asmpause) { 1120 val |= MII_ABILITY_ASM_DIR; 1121 } 1122 1123 DPRINTF(0, (CE_CONT, 1124 "!%s: %s: setting MII_AN_ADVERT reg:%b, pause:%d, asmpause:%d", 1125 dp->name, __func__, val, MII_ABILITY_BITS, 1126 dp->anadv_pause, dp->anadv_asmpause)); 1127 1128 usbgem_mii_write(dp, MII_AN_ADVERT, val, errp); 1129 if (*errp != USB_SUCCESS) { 1130 goto usberr; 1131 } 1132 1133 if (dp->mii_status & MII_STATUS_XSTATUS) { 1134 /* 1135 * 1000Base-T GMII support 1136 */ 1137 if (!dp->anadv_autoneg) { 1138 /* enable manual configuration */ 1139 val = MII_1000TC_CFG_EN; 1140 if (dp->anadv_1000t_ms == 2) { 1141 val |= MII_1000TC_CFG_VAL; 1142 } 1143 } else { 1144 val = 0; 1145 if (dp->anadv_1000fdx) { 1146 val |= MII_1000TC_ADV_FULL; 1147 } 1148 if (dp->anadv_1000hdx) { 1149 val |= MII_1000TC_ADV_HALF; 1150 } 1151 switch (dp->anadv_1000t_ms) { 1152 case 1: 1153 /* slave */ 1154 val |= MII_1000TC_CFG_EN; 1155 break; 1156 1157 case 2: 1158 /* master */ 1159 val |= MII_1000TC_CFG_EN | MII_1000TC_CFG_VAL; 1160 break; 1161 1162 default: 1163 /* auto: do nothing */ 1164 break; 1165 } 1166 } 1167 DPRINTF(0, (CE_CONT, 1168 "!%s: %s: setting MII_1000TC reg:%b", 1169 dp->name, __func__, val, MII_1000TC_BITS)); 1170 1171 usbgem_mii_write(dp, MII_1000TC, val, errp); 1172 if (*errp != USB_SUCCESS) { 1173 goto usberr; 1174 } 1175 } 1176 return (USB_SUCCESS); 1177 1178 usberr: 1179 return (*errp); 1180 } 1181 1182 static char *usbgem_fc_type[] = { 1183 "without", 1184 "with symmetric", 1185 "with tx", 1186 "with rx", 1187 }; 1188 1189 #define USBGEM_LINKUP(dp) mac_link_update((dp)->mh, LINK_STATE_UP) 1190 #define USBGEM_LINKDOWN(dp) mac_link_update((dp)->mh, LINK_STATE_DOWN) 1191 1192 static uint8_t usbgem_fc_result[4 /* my cap */][4 /* lp cap */] = { 1193 /* none symm tx rx/symm */ 1194 /* none */ 1195 {FLOW_CONTROL_NONE, 1196 FLOW_CONTROL_NONE, 1197 FLOW_CONTROL_NONE, 1198 FLOW_CONTROL_NONE}, 1199 /* sym */ 1200 {FLOW_CONTROL_NONE, 1201 FLOW_CONTROL_SYMMETRIC, 1202 FLOW_CONTROL_NONE, 1203 FLOW_CONTROL_SYMMETRIC}, 1204 /* tx */ 1205 {FLOW_CONTROL_NONE, 1206 FLOW_CONTROL_NONE, 1207 FLOW_CONTROL_NONE, 1208 FLOW_CONTROL_TX_PAUSE}, 1209 /* rx/symm */ 1210 {FLOW_CONTROL_NONE, 1211 FLOW_CONTROL_SYMMETRIC, 1212 FLOW_CONTROL_RX_PAUSE, 1213 FLOW_CONTROL_SYMMETRIC}, 1214 }; 1215 1216 static boolean_t 1217 usbgem_mii_link_check(struct usbgem_dev *dp, int *oldstatep, int *newstatep) 1218 { 1219 boolean_t tx_sched = B_FALSE; 1220 uint16_t status; 1221 uint16_t advert; 1222 uint16_t lpable; 1223 uint16_t exp; 1224 uint16_t ctl1000; 1225 uint16_t stat1000; 1226 uint16_t val; 1227 clock_t now; 1228 clock_t diff; 1229 int linkdown_action; 1230 boolean_t fix_phy = B_FALSE; 1231 int err; 1232 uint_t rwlock; 1233 1234 DPRINTF(4, (CE_CONT, "!%s: %s: time:%d state:%d", 1235 dp->name, __func__, ddi_get_lbolt(), dp->mii_state)); 1236 1237 if (dp->mii_state != MII_STATE_LINKUP) { 1238 rwlock = RW_WRITER; 1239 } else { 1240 rwlock = RW_READER; 1241 } 1242 again: 1243 rw_enter(&dp->dev_state_lock, rwlock); 1244 1245 /* save old mii state */ 1246 *oldstatep = dp->mii_state; 1247 1248 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 1249 /* stop periodic execution of the link watcher */ 1250 dp->mii_interval = 0; 1251 tx_sched = B_FALSE; 1252 goto next; 1253 } 1254 1255 now = ddi_get_lbolt(); 1256 diff = now - dp->mii_last_check; 1257 dp->mii_last_check = now; 1258 1259 /* 1260 * For NWAM, don't show linkdown state right 1261 * when the device is attached. 1262 */ 1263 if (dp->linkup_delay > 0) { 1264 if (dp->linkup_delay > diff) { 1265 dp->linkup_delay -= diff; 1266 } else { 1267 /* link up timeout */ 1268 dp->linkup_delay = -1; 1269 } 1270 } 1271 1272 next_nowait: 1273 switch (dp->mii_state) { 1274 case MII_STATE_UNKNOWN: 1275 goto reset_phy; 1276 1277 case MII_STATE_RESETTING: 1278 dp->mii_timer -= diff; 1279 if (dp->mii_timer > 0) { 1280 /* don't read phy registers in resetting */ 1281 dp->mii_interval = WATCH_INTERVAL_FAST; 1282 goto next; 1283 } 1284 1285 val = usbgem_mii_read(dp, MII_CONTROL, &err); 1286 if (err != USB_SUCCESS) { 1287 goto usberr; 1288 } 1289 if (val & MII_CONTROL_RESET) { 1290 cmn_err(CE_NOTE, 1291 "!%s: time:%ld resetting phy not complete." 1292 " mii_control:0x%b", 1293 dp->name, ddi_get_lbolt(), 1294 val, MII_CONTROL_BITS); 1295 } 1296 1297 /* ensure neither isolated nor pwrdown nor auto-nego mode */ 1298 usbgem_mii_write(dp, MII_CONTROL, 0, &err); 1299 if (err != USB_SUCCESS) { 1300 goto usberr; 1301 } 1302 #if USBGEM_DEBUG_LEVEL > 10 1303 val = usbgem_mii_read(dp, MII_CONTROL, &err); 1304 cmn_err(CE_CONT, "!%s: readback control %b", 1305 dp->name, val, MII_CONTROL_BITS); 1306 #endif 1307 /* As resetting PHY has completed, configure PHY registers */ 1308 if ((*dp->ugc.usbgc_mii_config)(dp, &err) != USB_SUCCESS) { 1309 /* we failed to configure PHY */ 1310 goto usberr; 1311 } 1312 1313 /* prepare for forced mode */ 1314 usbgem_choose_forcedmode(dp); 1315 1316 dp->mii_lpable = 0; 1317 dp->mii_advert = 0; 1318 dp->mii_exp = 0; 1319 dp->mii_ctl1000 = 0; 1320 dp->mii_stat1000 = 0; 1321 1322 dp->flow_control = FLOW_CONTROL_NONE; 1323 1324 if (!dp->anadv_autoneg) { 1325 /* skip auto-negotiation phase */ 1326 dp->mii_state = MII_STATE_MEDIA_SETUP; 1327 dp->mii_timer = dp->ugc.usbgc_mii_linkdown_timeout; 1328 goto next_nowait; 1329 } 1330 1331 /* issue an auto-negotiation command */ 1332 goto autonego; 1333 1334 case MII_STATE_AUTONEGOTIATING: 1335 /* 1336 * Autonegotiation in progress 1337 */ 1338 dp->mii_timer -= diff; 1339 if (dp->mii_timer - 1340 (dp->ugc.usbgc_mii_an_timeout - dp->ugc.usbgc_mii_an_wait) 1341 > 0) { 1342 /* wait for minimum time (2.3 - 2.5 sec) */ 1343 dp->mii_interval = WATCH_INTERVAL_FAST; 1344 goto next; 1345 } 1346 1347 /* read PHY status */ 1348 status = usbgem_mii_read(dp, MII_STATUS, &err); 1349 if (err != USB_SUCCESS) { 1350 goto usberr; 1351 } 1352 DPRINTF(4, (CE_CONT, 1353 "!%s: %s: called: mii_state:%d MII_STATUS reg:%b", 1354 dp->name, __func__, dp->mii_state, 1355 status, MII_STATUS_BITS)); 1356 1357 if (status & MII_STATUS_REMFAULT) { 1358 /* 1359 * The link parnert told me something wrong happend. 1360 * What do we do ? 1361 */ 1362 cmn_err(CE_CONT, 1363 "!%s: auto-negotiation failed: remote fault", 1364 dp->name); 1365 goto autonego; 1366 } 1367 1368 if ((status & MII_STATUS_ANDONE) == 0) { 1369 if (dp->mii_timer <= 0) { 1370 /* 1371 * Auto-negotiation has been timed out, 1372 * Reset PHY and try again. 1373 */ 1374 if (!dp->mii_supress_msg) { 1375 cmn_err(CE_WARN, 1376 "!%s: auto-negotiation failed:" 1377 " timeout", 1378 dp->name); 1379 dp->mii_supress_msg = B_TRUE; 1380 } 1381 goto autonego; 1382 } 1383 /* 1384 * Auto-negotiation is in progress. Wait for a while. 1385 */ 1386 dp->mii_interval = dp->ugc.usbgc_mii_an_watch_interval; 1387 goto next; 1388 } 1389 1390 /* 1391 * Auto-negotiation has been completed. Let's go to AN_DONE. 1392 */ 1393 dp->mii_state = MII_STATE_AN_DONE; 1394 dp->mii_supress_msg = B_FALSE; 1395 DPRINTF(0, (CE_CONT, 1396 "!%s: auto-negotiation completed, MII_STATUS:%b", 1397 dp->name, status, MII_STATUS_BITS)); 1398 1399 if (dp->ugc.usbgc_mii_an_delay > 0) { 1400 dp->mii_timer = dp->ugc.usbgc_mii_an_delay; 1401 dp->mii_interval = drv_usectohz(20*1000); 1402 goto next; 1403 } 1404 1405 dp->mii_timer = 0; 1406 diff = 0; 1407 goto next_nowait; 1408 1409 case MII_STATE_AN_DONE: 1410 /* 1411 * Auto-negotiation has done. Now we can set up media. 1412 */ 1413 dp->mii_timer -= diff; 1414 if (dp->mii_timer > 0) { 1415 /* wait for a while */ 1416 dp->mii_interval = WATCH_INTERVAL_FAST; 1417 goto next; 1418 } 1419 1420 /* 1421 * Setup speed and duplex mode according with 1422 * the result of auto negotiation. 1423 */ 1424 1425 /* 1426 * Read registers required to determin current 1427 * duplex mode and media speed. 1428 */ 1429 if (dp->ugc.usbgc_mii_an_delay > 0) { 1430 /* the 'status' variable is not initialized yet */ 1431 status = usbgem_mii_read(dp, MII_STATUS, &err); 1432 if (err != USB_SUCCESS) { 1433 goto usberr; 1434 } 1435 } 1436 advert = usbgem_mii_read(dp, MII_AN_ADVERT, &err); 1437 if (err != USB_SUCCESS) { 1438 goto usberr; 1439 } 1440 lpable = usbgem_mii_read(dp, MII_AN_LPABLE, &err); 1441 if (err != USB_SUCCESS) { 1442 goto usberr; 1443 } 1444 exp = usbgem_mii_read(dp, MII_AN_EXPANSION, &err); 1445 if (err != USB_SUCCESS) { 1446 goto usberr; 1447 } 1448 if (exp == 0xffff) { 1449 /* some phys don't have exp register */ 1450 exp = 0; 1451 } 1452 1453 ctl1000 = 0; 1454 stat1000 = 0; 1455 if (dp->mii_status & MII_STATUS_XSTATUS) { 1456 ctl1000 = usbgem_mii_read(dp, MII_1000TC, &err); 1457 if (err != USB_SUCCESS) { 1458 goto usberr; 1459 } 1460 stat1000 = usbgem_mii_read(dp, MII_1000TS, &err); 1461 if (err != USB_SUCCESS) { 1462 goto usberr; 1463 } 1464 } 1465 dp->mii_lpable = lpable; 1466 dp->mii_advert = advert; 1467 dp->mii_exp = exp; 1468 dp->mii_ctl1000 = ctl1000; 1469 dp->mii_stat1000 = stat1000; 1470 1471 cmn_err(CE_CONT, 1472 "!%s: auto-negotiation done: " 1473 "status:%b, advert:%b, lpable:%b, exp:%b", 1474 dp->name, 1475 status, MII_STATUS_BITS, 1476 advert, MII_ABILITY_BITS, 1477 lpable, MII_ABILITY_BITS, 1478 exp, MII_AN_EXP_BITS); 1479 1480 DPRINTF(0, (CE_CONT, "!%s: MII_STATUS:%b", 1481 dp->name, status, MII_STATUS_BITS)); 1482 1483 if (dp->mii_status & MII_STATUS_XSTATUS) { 1484 cmn_err(CE_CONT, 1485 "! MII_1000TC reg:%b, MII_1000TS reg:%b", 1486 ctl1000, MII_1000TC_BITS, 1487 stat1000, MII_1000TS_BITS); 1488 } 1489 1490 if (usbgem_population(lpable) <= 1 && 1491 (exp & MII_AN_EXP_LPCANAN) == 0) { 1492 if ((advert & MII_ABILITY_TECH) != lpable) { 1493 cmn_err(CE_WARN, 1494 "!%s: but the link partner doesn't seem" 1495 " to have auto-negotiation capability." 1496 " please check the link configuration.", 1497 dp->name); 1498 } 1499 /* 1500 * it should be a result of pararell detection, 1501 * which cannot detect duplex mode. 1502 */ 1503 if ((advert & lpable) == 0 && 1504 lpable & MII_ABILITY_10BASE_T) { 1505 /* no common technology, try 10M half mode */ 1506 lpable |= advert & MII_ABILITY_10BASE_T; 1507 fix_phy = B_TRUE; 1508 } 1509 } else if (lpable == 0) { 1510 cmn_err(CE_WARN, "!%s: wrong lpable.", dp->name); 1511 goto reset_phy; 1512 } 1513 /* 1514 * configure current link mode according to AN priority. 1515 */ 1516 val = advert & lpable; 1517 if ((ctl1000 & MII_1000TC_ADV_FULL) && 1518 (stat1000 & MII_1000TS_LP_FULL)) { 1519 /* 1000BaseT & full duplex */ 1520 dp->speed = USBGEM_SPD_1000; 1521 dp->full_duplex = B_TRUE; 1522 } else if ((ctl1000 & MII_1000TC_ADV_HALF) && 1523 (stat1000 & MII_1000TS_LP_HALF)) { 1524 /* 1000BaseT & half duplex */ 1525 dp->speed = USBGEM_SPD_1000; 1526 dp->full_duplex = B_FALSE; 1527 } else if ((val & MII_ABILITY_100BASE_TX_FD)) { 1528 /* 100BaseTx & fullduplex */ 1529 dp->speed = USBGEM_SPD_100; 1530 dp->full_duplex = B_TRUE; 1531 } else if ((val & MII_ABILITY_100BASE_T4)) { 1532 /* 100BaseTx & fullduplex */ 1533 dp->speed = USBGEM_SPD_100; 1534 dp->full_duplex = B_TRUE; 1535 } else if ((val & MII_ABILITY_100BASE_TX)) { 1536 /* 100BaseTx & half duplex */ 1537 dp->speed = USBGEM_SPD_100; 1538 dp->full_duplex = B_FALSE; 1539 } else if ((val & MII_ABILITY_10BASE_T_FD)) { 1540 /* 10BaseT & full duplex */ 1541 dp->speed = USBGEM_SPD_10; 1542 dp->full_duplex = B_TRUE; 1543 } else if ((val & MII_ABILITY_10BASE_T)) { 1544 /* 10BaseT & half duplex */ 1545 dp->speed = USBGEM_SPD_10; 1546 dp->full_duplex = B_FALSE; 1547 } else { 1548 /* 1549 * the link partner doesn't seem to have 1550 * auto-negotiation capability and our PHY 1551 * could not report current mode correctly. 1552 * We guess current mode by mii_control register. 1553 */ 1554 val = usbgem_mii_read(dp, MII_CONTROL, &err); 1555 if (err != USB_SUCCESS) { 1556 goto usberr; 1557 } 1558 1559 /* select 100m half or 10m half */ 1560 dp->speed = (val & MII_CONTROL_100MB) ? 1561 USBGEM_SPD_100 : USBGEM_SPD_10; 1562 dp->full_duplex = B_FALSE; 1563 fix_phy = B_TRUE; 1564 1565 cmn_err(CE_NOTE, 1566 "!%s: auto-negotiation done but " 1567 "common ability not found.\n" 1568 "PHY state: control:%b advert:%b lpable:%b\n" 1569 "guessing %d Mbps %s duplex mode", 1570 dp->name, 1571 val, MII_CONTROL_BITS, 1572 advert, MII_ABILITY_BITS, 1573 lpable, MII_ABILITY_BITS, 1574 usbgem_speed_value[dp->speed], 1575 dp->full_duplex ? "full" : "half"); 1576 } 1577 1578 if (dp->full_duplex) { 1579 dp->flow_control = 1580 usbgem_fc_result[fc_cap_decode(advert)] 1581 [fc_cap_decode(lpable)]; 1582 } else { 1583 dp->flow_control = FLOW_CONTROL_NONE; 1584 } 1585 dp->mii_state = MII_STATE_MEDIA_SETUP; 1586 dp->mii_timer = dp->ugc.usbgc_mii_linkdown_timeout; 1587 goto next_nowait; 1588 1589 case MII_STATE_MEDIA_SETUP: 1590 DPRINTF(2, (CE_CONT, "!%s: setup midia mode", dp->name)); 1591 1592 /* assume the link state is down */ 1593 dp->mii_state = MII_STATE_LINKDOWN; 1594 dp->mii_supress_msg = B_FALSE; 1595 1596 /* use short interval */ 1597 dp->mii_interval = WATCH_INTERVAL_FAST; 1598 1599 if ((!dp->anadv_autoneg) || 1600 dp->ugc.usbgc_mii_an_oneshot || fix_phy) { 1601 1602 /* 1603 * write the result of auto negotiation back. 1604 */ 1605 val = usbgem_mii_read(dp, MII_CONTROL, &err); 1606 if (err != USB_SUCCESS) { 1607 goto usberr; 1608 } 1609 val &= ~(MII_CONTROL_SPEED | MII_CONTROL_FDUPLEX | 1610 MII_CONTROL_ANE | MII_CONTROL_RSAN); 1611 1612 if (dp->full_duplex) { 1613 val |= MII_CONTROL_FDUPLEX; 1614 } 1615 1616 switch (dp->speed) { 1617 case USBGEM_SPD_1000: 1618 val |= MII_CONTROL_1000MB; 1619 break; 1620 1621 case USBGEM_SPD_100: 1622 val |= MII_CONTROL_100MB; 1623 break; 1624 1625 default: 1626 cmn_err(CE_WARN, "%s: unknown speed:%d", 1627 dp->name, dp->speed); 1628 /* FALLTHROUGH */ 1629 1630 case USBGEM_SPD_10: 1631 /* for USBGEM_SPD_10, do nothing */ 1632 break; 1633 } 1634 1635 if (dp->mii_status & MII_STATUS_XSTATUS) { 1636 usbgem_mii_write(dp, 1637 MII_1000TC, MII_1000TC_CFG_EN, &err); 1638 if (err != USB_SUCCESS) { 1639 goto usberr; 1640 } 1641 } 1642 usbgem_mii_write(dp, MII_CONTROL, val, &err); 1643 if (err != USB_SUCCESS) { 1644 goto usberr; 1645 } 1646 } 1647 /* 1648 * XXX -- nic state should be one of 1649 * NIC_STATE_DISCONNECTED 1650 * NIC_STATE_STOPPED 1651 * NIC_STATE_INITIALIZED 1652 * NIC_STATE_ONLINE 1653 */ 1654 if (dp->nic_state >= NIC_STATE_INITIALIZED) { 1655 /* notify the result of autonegotiation to mac */ 1656 if (usbgem_hal_set_media(dp) != USB_SUCCESS) { 1657 goto usberr; 1658 } 1659 } 1660 goto next_nowait; 1661 1662 case MII_STATE_LINKDOWN: 1663 status = usbgem_mii_read(dp, MII_STATUS, &err); 1664 if (err != USB_SUCCESS) { 1665 goto usberr; 1666 } 1667 if (status & MII_STATUS_LINKUP) { 1668 /* 1669 * Link is going up 1670 */ 1671 dp->mii_state = MII_STATE_LINKUP; 1672 dp->mii_supress_msg = B_FALSE; 1673 1674 DPRINTF(0, (CE_CONT, 1675 "!%s: link up detected: status:%b", 1676 dp->name, status, MII_STATUS_BITS)); 1677 1678 /* 1679 * MII_CONTROL_100MB and MII_CONTROL_FDUPLEX are 1680 * ignored when MII_CONTROL_ANE is set. 1681 */ 1682 cmn_err(CE_CONT, 1683 "!%s: Link up: %d Mbps %s duplex %s flow control", 1684 dp->name, 1685 usbgem_speed_value[dp->speed], 1686 dp->full_duplex ? "full" : "half", 1687 usbgem_fc_type[dp->flow_control]); 1688 1689 dp->mii_interval = 1690 dp->ugc.usbgc_mii_link_watch_interval; 1691 1692 if (dp->ugc.usbgc_mii_hw_link_detection && 1693 dp->nic_state == NIC_STATE_ONLINE) { 1694 dp->mii_interval = 0; 1695 } 1696 1697 if (dp->nic_state == NIC_STATE_ONLINE) { 1698 if (dp->mac_state == MAC_STATE_INITIALIZED) { 1699 (void) usbgem_mac_start(dp); 1700 } 1701 tx_sched = B_TRUE; 1702 } 1703 1704 goto next; 1705 } 1706 1707 dp->mii_supress_msg = B_TRUE; 1708 if (dp->anadv_autoneg) { 1709 dp->mii_timer -= diff; 1710 if (dp->mii_timer <= 0) { 1711 /* 1712 * the link down timer expired. 1713 * need to restart auto-negotiation. 1714 */ 1715 linkdown_action = 1716 dp->ugc.usbgc_mii_linkdown_timeout_action; 1717 goto restart_autonego; 1718 } 1719 } 1720 /* don't change mii_state */ 1721 goto next; 1722 1723 case MII_STATE_LINKUP: 1724 if (rwlock == RW_READER) { 1725 /* first pass, read mii status */ 1726 status = usbgem_mii_read(dp, MII_STATUS, &err); 1727 if (err != USB_SUCCESS) { 1728 goto usberr; 1729 } 1730 } 1731 if ((status & MII_STATUS_LINKUP) == 0) { 1732 /* 1733 * Link is going down 1734 */ 1735 cmn_err(CE_NOTE, 1736 "!%s: link down detected: status:%b", 1737 dp->name, status, MII_STATUS_BITS); 1738 /* 1739 * Acquire exclusive lock to change mii_state 1740 */ 1741 if (rwlock == RW_READER) { 1742 rwlock = RW_WRITER; 1743 rw_exit(&dp->dev_state_lock); 1744 goto again; 1745 } 1746 1747 dp->mii_state = MII_STATE_LINKDOWN; 1748 dp->mii_timer = dp->ugc.usbgc_mii_linkdown_timeout; 1749 1750 /* 1751 * As we may change the state of the device, 1752 * let us acquire exclusive lock for the state. 1753 */ 1754 if (dp->nic_state == NIC_STATE_ONLINE && 1755 dp->mac_state == MAC_STATE_ONLINE && 1756 dp->ugc.usbgc_mii_stop_mac_on_linkdown) { 1757 (void) usbgem_restart_nic(dp); 1758 /* drain tx */ 1759 tx_sched = B_TRUE; 1760 } 1761 1762 if (dp->anadv_autoneg) { 1763 /* need to restart auto-negotiation */ 1764 linkdown_action = 1765 dp->ugc.usbgc_mii_linkdown_action; 1766 goto restart_autonego; 1767 } 1768 /* 1769 * don't use hw link down detection until the link 1770 * status become stable for a while. 1771 */ 1772 dp->mii_interval = 1773 dp->ugc.usbgc_mii_link_watch_interval; 1774 1775 goto next; 1776 } 1777 1778 /* 1779 * still link up, no need to change mii_state 1780 */ 1781 if (dp->ugc.usbgc_mii_hw_link_detection && 1782 dp->nic_state == NIC_STATE_ONLINE) { 1783 /* 1784 * no need to check link status periodicly 1785 * if nic can generate interrupts when link go down. 1786 */ 1787 dp->mii_interval = 0; 1788 } 1789 goto next; 1790 } 1791 /* NOTREACHED */ 1792 cmn_err(CE_PANIC, "!%s: %s: not reached", dp->name, __func__); 1793 1794 /* 1795 * Actions for new state. 1796 */ 1797 restart_autonego: 1798 switch (linkdown_action) { 1799 case MII_ACTION_RESET: 1800 if (!dp->mii_supress_msg) { 1801 cmn_err(CE_CONT, "!%s: resetting PHY", dp->name); 1802 } 1803 dp->mii_supress_msg = B_TRUE; 1804 goto reset_phy; 1805 1806 case MII_ACTION_NONE: 1807 dp->mii_supress_msg = B_TRUE; 1808 if (dp->ugc.usbgc_mii_an_oneshot) { 1809 goto autonego; 1810 } 1811 /* PHY will restart autonego automatically */ 1812 dp->mii_state = MII_STATE_AUTONEGOTIATING; 1813 dp->mii_timer = dp->ugc.usbgc_mii_an_timeout; 1814 dp->mii_interval = dp->ugc.usbgc_mii_an_watch_interval; 1815 goto next; 1816 1817 case MII_ACTION_RSA: 1818 if (!dp->mii_supress_msg) { 1819 cmn_err(CE_CONT, "!%s: restarting auto-negotiation", 1820 dp->name); 1821 } 1822 dp->mii_supress_msg = B_TRUE; 1823 goto autonego; 1824 1825 default: 1826 cmn_err(CE_PANIC, "!%s: unknowm linkdown action: %d", 1827 dp->name, dp->ugc.usbgc_mii_linkdown_action); 1828 } 1829 /* NOTREACHED */ 1830 1831 reset_phy: 1832 if (!dp->mii_supress_msg) { 1833 cmn_err(CE_CONT, "!%s: resetting PHY", dp->name); 1834 } 1835 dp->mii_state = MII_STATE_RESETTING; 1836 dp->mii_timer = dp->ugc.usbgc_mii_reset_timeout; 1837 if (!dp->ugc.usbgc_mii_dont_reset) { 1838 usbgem_mii_write(dp, MII_CONTROL, MII_CONTROL_RESET, &err); 1839 if (err != USB_SUCCESS) { 1840 goto usberr; 1841 } 1842 } 1843 dp->mii_interval = WATCH_INTERVAL_FAST; 1844 goto next; 1845 1846 autonego: 1847 if (!dp->mii_supress_msg) { 1848 cmn_err(CE_CONT, "!%s: auto-negotiation started", dp->name); 1849 } 1850 dp->mii_state = MII_STATE_AUTONEGOTIATING; 1851 dp->mii_timer = dp->ugc.usbgc_mii_an_timeout; 1852 1853 /* start/restart autoneg */ 1854 val = usbgem_mii_read(dp, MII_CONTROL, &err) & 1855 ~(MII_CONTROL_ISOLATE | MII_CONTROL_PWRDN | MII_CONTROL_RESET); 1856 if (err != USB_SUCCESS) { 1857 goto usberr; 1858 } 1859 if (val & MII_CONTROL_ANE) { 1860 val |= MII_CONTROL_RSAN; 1861 } 1862 usbgem_mii_write(dp, MII_CONTROL, 1863 val | dp->ugc.usbgc_mii_an_cmd | MII_CONTROL_ANE, &err); 1864 if (err != USB_SUCCESS) { 1865 goto usberr; 1866 } 1867 1868 dp->mii_interval = dp->ugc.usbgc_mii_an_watch_interval; 1869 goto next; 1870 1871 usberr: 1872 dp->mii_state = MII_STATE_UNKNOWN; 1873 dp->mii_interval = dp->ugc.usbgc_mii_link_watch_interval; 1874 tx_sched = B_TRUE; 1875 1876 next: 1877 *newstatep = dp->mii_state; 1878 rw_exit(&dp->dev_state_lock); 1879 return (tx_sched); 1880 } 1881 1882 static void 1883 usbgem_mii_link_watcher(struct usbgem_dev *dp) 1884 { 1885 int old_mii_state; 1886 int new_mii_state; 1887 boolean_t tx_sched; 1888 1889 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 1890 1891 for (; ; ) { 1892 1893 mutex_enter(&dp->link_watcher_lock); 1894 if (dp->mii_interval) { 1895 (void) cv_timedwait(&dp->link_watcher_wait_cv, 1896 &dp->link_watcher_lock, 1897 dp->mii_interval + ddi_get_lbolt()); 1898 } else { 1899 cv_wait(&dp->link_watcher_wait_cv, 1900 &dp->link_watcher_lock); 1901 } 1902 mutex_exit(&dp->link_watcher_lock); 1903 1904 if (dp->link_watcher_stop) { 1905 break; 1906 } 1907 1908 /* we block callbacks from disconnect/suspend and restart */ 1909 tx_sched = usbgem_mii_link_check(dp, 1910 &old_mii_state, &new_mii_state); 1911 1912 /* 1913 * gld v2 notifier functions are not able to 1914 * be called with any locks in this layer. 1915 */ 1916 if (tx_sched) { 1917 /* kick potentially stopped downstream */ 1918 mac_tx_update(dp->mh); 1919 } 1920 1921 if (old_mii_state != new_mii_state) { 1922 /* notify new mii link state */ 1923 if (new_mii_state == MII_STATE_LINKUP) { 1924 dp->linkup_delay = 0; 1925 USBGEM_LINKUP(dp); 1926 } else if (dp->linkup_delay <= 0) { 1927 USBGEM_LINKDOWN(dp); 1928 } 1929 } else if (dp->linkup_delay < 0) { 1930 /* first linkup timeout */ 1931 dp->linkup_delay = 0; 1932 USBGEM_LINKDOWN(dp); 1933 } 1934 } 1935 1936 thread_exit(); 1937 } 1938 1939 void 1940 usbgem_mii_update_link(struct usbgem_dev *dp) 1941 { 1942 cv_signal(&dp->link_watcher_wait_cv); 1943 } 1944 1945 int 1946 usbgem_mii_probe_default(struct usbgem_dev *dp) 1947 { 1948 int phy; 1949 uint16_t status; 1950 uint16_t xstatus; 1951 int err; 1952 uint16_t adv; 1953 uint16_t adv_org; 1954 1955 DPRINTF(3, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 1956 1957 /* 1958 * Scan PHY 1959 */ 1960 dp->mii_status = 0; 1961 1962 /* Try default phy first */ 1963 if (dp->mii_phy_addr) { 1964 status = usbgem_mii_read(dp, MII_STATUS, &err); 1965 if (err != USB_SUCCESS) { 1966 goto usberr; 1967 } 1968 if (status != 0xffff && status != 0x0000) { 1969 goto PHY_found; 1970 } 1971 1972 if (dp->mii_phy_addr < 0) { 1973 cmn_err(CE_NOTE, 1974 "!%s: failed to probe default internal and/or non-MII PHY", 1975 dp->name); 1976 return (USB_FAILURE); 1977 } 1978 1979 cmn_err(CE_NOTE, 1980 "!%s: failed to probe default MII PHY at %d", 1981 dp->name, dp->mii_phy_addr); 1982 } 1983 1984 /* Try all possible address */ 1985 for (phy = dp->ugc.usbgc_mii_addr_min; phy < 32; phy++) { 1986 dp->mii_phy_addr = phy; 1987 status = usbgem_mii_read(dp, MII_STATUS, &err); 1988 if (err != USB_SUCCESS) { 1989 DPRINTF(0, (CE_CONT, 1990 "!%s: %s: mii_read(status) failed", 1991 dp->name, __func__)); 1992 goto usberr; 1993 } 1994 1995 if (status != 0xffff && status != 0x0000) { 1996 usbgem_mii_write(dp, MII_CONTROL, 0, &err); 1997 if (err != USB_SUCCESS) { 1998 DPRINTF(0, (CE_CONT, 1999 "!%s: %s: mii_write(control) failed", 2000 dp->name, __func__)); 2001 goto usberr; 2002 } 2003 goto PHY_found; 2004 } 2005 } 2006 for (phy = dp->ugc.usbgc_mii_addr_min; phy < 32; phy++) { 2007 dp->mii_phy_addr = phy; 2008 usbgem_mii_write(dp, MII_CONTROL, 0, &err); 2009 if (err != USB_SUCCESS) { 2010 DPRINTF(0, (CE_CONT, 2011 "!%s: %s: mii_write(control) failed", 2012 dp->name, __func__)); 2013 goto usberr; 2014 } 2015 status = usbgem_mii_read(dp, MII_STATUS, &err); 2016 if (err != USB_SUCCESS) { 2017 DPRINTF(0, (CE_CONT, 2018 "!%s: %s: mii_read(status) failed", 2019 dp->name, __func__)); 2020 goto usberr; 2021 } 2022 2023 if (status != 0xffff && status != 0) { 2024 goto PHY_found; 2025 } 2026 } 2027 2028 cmn_err(CE_NOTE, "!%s: no MII PHY found", dp->name); 2029 return (USB_FAILURE); 2030 2031 PHY_found: 2032 dp->mii_status = status; 2033 dp->mii_status_ro = ~status; 2034 dp->mii_phy_id = usbgem_mii_read(dp, MII_PHYIDH, &err) << 16; 2035 if (err != USB_SUCCESS) { 2036 DPRINTF(0, (CE_CONT, 2037 "!%s: %s: mii_read(PHYIDH) failed", 2038 dp->name, __func__)); 2039 goto usberr; 2040 } 2041 dp->mii_phy_id |= usbgem_mii_read(dp, MII_PHYIDL, &err); 2042 if (err != USB_SUCCESS) { 2043 DPRINTF(0, (CE_CONT, 2044 "!%s: %s: mii_read(PHYIDL) failed", 2045 dp->name, __func__)); 2046 goto usberr; 2047 } 2048 2049 if (dp->mii_phy_addr < 0) { 2050 cmn_err(CE_CONT, "!%s: using internal/non-MII PHY(0x%08x)", 2051 dp->name, dp->mii_phy_id); 2052 } else { 2053 cmn_err(CE_CONT, "!%s: MII PHY (0x%08x) found at %d", 2054 dp->name, dp->mii_phy_id, dp->mii_phy_addr); 2055 } 2056 2057 cmn_err(CE_CONT, 2058 "!%s: PHY control:%b, status:%b, advert:%b, lpar:%b, exp:%b", 2059 dp->name, 2060 usbgem_mii_read(dp, MII_CONTROL, &err), MII_CONTROL_BITS, 2061 status, MII_STATUS_BITS, 2062 usbgem_mii_read(dp, MII_AN_ADVERT, &err), MII_ABILITY_BITS, 2063 usbgem_mii_read(dp, MII_AN_LPABLE, &err), MII_ABILITY_BITS, 2064 usbgem_mii_read(dp, MII_AN_EXPANSION, &err), MII_AN_EXP_BITS); 2065 2066 dp->mii_xstatus = 0; 2067 if (status & MII_STATUS_XSTATUS) { 2068 dp->mii_xstatus = usbgem_mii_read(dp, MII_XSTATUS, &err); 2069 2070 cmn_err(CE_CONT, "!%s: xstatus:%b", 2071 dp->name, dp->mii_xstatus, MII_XSTATUS_BITS); 2072 } 2073 dp->mii_xstatus_ro = ~dp->mii_xstatus; 2074 2075 /* check if the phy can advertize pause abilities */ 2076 adv_org = usbgem_mii_read(dp, MII_AN_ADVERT, &err); 2077 if (err != USB_SUCCESS) { 2078 goto usberr; 2079 } 2080 2081 usbgem_mii_write(dp, MII_AN_ADVERT, 2082 MII_ABILITY_PAUSE | MII_ABILITY_ASM_DIR, &err); 2083 if (err != USB_SUCCESS) { 2084 goto usberr; 2085 } 2086 2087 adv = usbgem_mii_read(dp, MII_AN_ADVERT, &err); 2088 if (err != USB_SUCCESS) { 2089 goto usberr; 2090 } 2091 2092 if ((adv & MII_ABILITY_PAUSE) == 0) { 2093 dp->ugc.usbgc_flow_control &= ~1; 2094 } 2095 2096 if ((adv & MII_ABILITY_ASM_DIR) == 0) { 2097 dp->ugc.usbgc_flow_control &= ~2; 2098 } 2099 2100 usbgem_mii_write(dp, MII_AN_ADVERT, adv_org, &err); 2101 if (err != USB_SUCCESS) { 2102 goto usberr; 2103 } 2104 return (USB_SUCCESS); 2105 2106 usberr: 2107 return (USB_FAILURE); 2108 } 2109 2110 int 2111 usbgem_mii_init_default(struct usbgem_dev *dp) 2112 { 2113 /* ENPTY */ 2114 return (USB_SUCCESS); 2115 } 2116 2117 static int 2118 usbgem_mii_start(struct usbgem_dev *dp) 2119 { 2120 int err; 2121 kthread_t *lwth; 2122 2123 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2124 2125 /* make a first call of usbgem_mii_link_check() */ 2126 dp->link_watcher_stop = 0; 2127 dp->mii_state = MII_STATE_UNKNOWN; 2128 dp->mii_interval = drv_usectohz(1000*1000); /* 1sec */ 2129 dp->mii_last_check = ddi_get_lbolt(); 2130 dp->linkup_delay = 600 * drv_usectohz(1000*1000); /* 10 minutes */ 2131 2132 lwth = thread_create(NULL, 0, usbgem_mii_link_watcher, dp, 0, &p0, 2133 TS_RUN, minclsyspri); 2134 if (lwth == NULL) { 2135 cmn_err(CE_WARN, 2136 "!%s: %s: failed to create a link watcher thread", 2137 dp->name, __func__); 2138 return (USB_FAILURE); 2139 } 2140 dp->link_watcher_did = lwth->t_did; 2141 2142 return (USB_SUCCESS); 2143 } 2144 2145 static void 2146 usbgem_mii_stop(struct usbgem_dev *dp) 2147 { 2148 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2149 2150 /* Ensure timer routine stopped */ 2151 dp->link_watcher_stop = 1; 2152 cv_signal(&dp->link_watcher_wait_cv); 2153 thread_join(dp->link_watcher_did); 2154 } 2155 2156 /* ============================================================== */ 2157 /* 2158 * internal mac register operation interface 2159 */ 2160 /* ============================================================== */ 2161 /* 2162 * usbgem_mac_init: cold start 2163 */ 2164 static int 2165 usbgem_mac_init(struct usbgem_dev *dp) 2166 { 2167 int err; 2168 2169 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2170 2171 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 2172 /* pretend we succeeded */ 2173 return (USB_SUCCESS); 2174 } 2175 2176 ASSERT(dp->mac_state == MAC_STATE_STOPPED); 2177 2178 /* reset fatal error timestamp */ 2179 dp->fatal_error = (clock_t)0; 2180 2181 /* reset tx side state */ 2182 mutex_enter(&dp->txlock); 2183 dp->tx_busy_cnt = 0; 2184 dp->tx_max_packets = dp->ugc.usbgc_tx_list_max; 2185 mutex_exit(&dp->txlock); 2186 2187 /* reset rx side state */ 2188 mutex_enter(&dp->rxlock); 2189 dp->rx_busy_cnt = 0; 2190 mutex_exit(&dp->rxlock); 2191 2192 err = usbgem_hal_init_chip(dp); 2193 if (err == USB_SUCCESS) { 2194 dp->mac_state = MAC_STATE_INITIALIZED; 2195 } 2196 2197 return (err); 2198 } 2199 2200 /* 2201 * usbgem_mac_start: warm start 2202 */ 2203 static int 2204 usbgem_mac_start(struct usbgem_dev *dp) 2205 { 2206 int err; 2207 int i; 2208 usb_flags_t flags = 0; 2209 usb_intr_req_t *req; 2210 #ifdef USBGEM_DEBUG_LEVEL 2211 usb_pipe_state_t p_state; 2212 #endif 2213 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2214 2215 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 2216 /* do nothing but don't return failure */ 2217 return (USB_SUCCESS); 2218 } 2219 2220 if (dp->mac_state != MAC_STATE_INITIALIZED) { 2221 /* don't return failer */ 2222 DPRINTF(0, (CE_CONT, 2223 "!%s: %s: mac_state(%d) is not MAC_STATE_INITIALIZED", 2224 dp->name, __func__, dp->mac_state)); 2225 goto x; 2226 } 2227 2228 dp->mac_state = MAC_STATE_ONLINE; 2229 2230 if (usbgem_hal_start_chip(dp) != USB_SUCCESS) { 2231 cmn_err(CE_NOTE, 2232 "!%s: %s: usb error was detected during start_chip", 2233 dp->name, __func__); 2234 goto x; 2235 } 2236 2237 #ifdef USBGEM_DEBUG_LEVEL 2238 usb_pipe_get_state(dp->intr_pipe, &p_state, 0); 2239 ASSERT(p_state == USB_PIPE_STATE_IDLE); 2240 #endif /* USBGEM_DEBUG_LEVEL */ 2241 2242 if (dp->ugc.usbgc_interrupt && dp->intr_pipe) { 2243 2244 /* make a request for interrupt */ 2245 2246 req = usb_alloc_intr_req(dp->dip, 0, USB_FLAGS_SLEEP); 2247 if (req == NULL) { 2248 cmn_err(CE_WARN, "!%s: %s: failed to allocate intreq", 2249 dp->name, __func__); 2250 goto x; 2251 } 2252 req->intr_data = NULL; 2253 req->intr_client_private = (usb_opaque_t)dp; 2254 req->intr_timeout = 0; 2255 req->intr_attributes = 2256 USB_ATTRS_SHORT_XFER_OK | USB_ATTRS_AUTOCLEARING; 2257 req->intr_len = dp->ep_intr->wMaxPacketSize; 2258 req->intr_cb = usbgem_intr_cb; 2259 req->intr_exc_cb = usbgem_intr_cb; 2260 req->intr_completion_reason = 0; 2261 req->intr_cb_flags = 0; 2262 2263 err = usb_pipe_intr_xfer(dp->intr_pipe, req, flags); 2264 if (err != USB_SUCCESS) { 2265 cmn_err(CE_WARN, 2266 "%s: err:%d failed to start polling of intr pipe", 2267 dp->name, err); 2268 goto x; 2269 } 2270 } 2271 2272 /* kick to receive the first packet */ 2273 if (usbgem_init_rx_buf(dp) != USB_SUCCESS) { 2274 goto err_stop_intr; 2275 } 2276 dp->rx_active = B_TRUE; 2277 2278 return (USB_SUCCESS); 2279 2280 err_stop_intr: 2281 /* stop the interrupt pipe */ 2282 DPRINTF(0, (CE_CONT, "!%s: %s: FAULURE", dp->name, __func__)); 2283 if (dp->ugc.usbgc_interrupt && dp->intr_pipe) { 2284 usb_pipe_stop_intr_polling(dp->intr_pipe, USB_FLAGS_SLEEP); 2285 } 2286 x: 2287 ASSERT(dp->mac_state == MAC_STATE_ONLINE); 2288 /* we use another flag to indicate error state. */ 2289 if (dp->fatal_error == (clock_t)0) { 2290 dp->fatal_error = usbgem_timestamp_nz(); 2291 } 2292 return (USB_FAILURE); 2293 } 2294 2295 static int 2296 usbgem_mac_stop(struct usbgem_dev *dp, int new_state, boolean_t graceful) 2297 { 2298 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2299 2300 /* 2301 * we must have writer lock for dev_state_lock 2302 */ 2303 ASSERT(new_state == MAC_STATE_STOPPED || 2304 new_state == MAC_STATE_DISCONNECTED); 2305 2306 /* stop polling interrupt pipe */ 2307 if (dp->ugc.usbgc_interrupt && dp->intr_pipe) { 2308 usb_pipe_stop_intr_polling(dp->intr_pipe, USB_FLAGS_SLEEP); 2309 } 2310 2311 if (new_state == MAC_STATE_STOPPED || graceful) { 2312 /* stop the nic hardware completely */ 2313 if (usbgem_hal_stop_chip(dp) != USB_SUCCESS) { 2314 (void) usbgem_hal_reset_chip(dp); 2315 } 2316 } 2317 2318 /* stop preparing new rx packets and sending new packets */ 2319 dp->mac_state = new_state; 2320 2321 /* other processors must get mac_state correctly after here */ 2322 membar_producer(); 2323 2324 /* cancel all requests we have sent */ 2325 usb_pipe_reset(dp->dip, dp->bulkin_pipe, USB_FLAGS_SLEEP, NULL, 0); 2326 usb_pipe_reset(dp->dip, dp->bulkout_pipe, USB_FLAGS_SLEEP, NULL, 0); 2327 2328 DPRINTF(0, (CE_CONT, 2329 "!%s: %s: rx_busy_cnt:%d tx_busy_cnt:%d", 2330 dp->name, __func__, dp->rx_busy_cnt, dp->tx_busy_cnt)); 2331 2332 /* 2333 * Here all rx packets has been cancelled and their call back 2334 * function has been exeuted, because we called usb_pipe_reset 2335 * synchronously. 2336 * So actually we just ensure rx_busy_cnt == 0. 2337 */ 2338 mutex_enter(&dp->rxlock); 2339 while (dp->rx_busy_cnt > 0) { 2340 cv_wait(&dp->rx_drain_cv, &dp->rxlock); 2341 } 2342 mutex_exit(&dp->rxlock); 2343 2344 DPRINTF(0, (CE_CONT, "!%s: %s: rx_busy_cnt is %d now", 2345 dp->name, __func__, dp->rx_busy_cnt)); 2346 2347 mutex_enter(&dp->txlock); 2348 while (dp->tx_busy_cnt > 0) { 2349 cv_wait(&dp->tx_drain_cv, &dp->txlock); 2350 } 2351 mutex_exit(&dp->txlock); 2352 2353 DPRINTF(0, (CE_CONT, "!%s: %s: tx_busy_cnt is %d now", 2354 dp->name, __func__, dp->tx_busy_cnt)); 2355 2356 return (USB_SUCCESS); 2357 } 2358 2359 static int 2360 usbgem_add_multicast(struct usbgem_dev *dp, const uint8_t *ep) 2361 { 2362 int cnt; 2363 int err; 2364 2365 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2366 2367 sema_p(&dp->rxfilter_lock); 2368 if (dp->mc_count_req++ < USBGEM_MAXMC) { 2369 /* append the new address at the end of the mclist */ 2370 cnt = dp->mc_count; 2371 bcopy(ep, dp->mc_list[cnt].addr.ether_addr_octet, 2372 ETHERADDRL); 2373 if (dp->ugc.usbgc_multicast_hash) { 2374 dp->mc_list[cnt].hash = 2375 (*dp->ugc.usbgc_multicast_hash)(dp, ep); 2376 } 2377 dp->mc_count = cnt + 1; 2378 } 2379 2380 if (dp->mc_count_req != dp->mc_count) { 2381 /* multicast address list overflow */ 2382 dp->rxmode |= RXMODE_MULTI_OVF; 2383 } else { 2384 dp->rxmode &= ~RXMODE_MULTI_OVF; 2385 } 2386 2387 if (dp->mac_state != MAC_STATE_DISCONNECTED) { 2388 /* tell new multicast list to the hardware */ 2389 err = usbgem_hal_set_rx_filter(dp); 2390 } 2391 sema_v(&dp->rxfilter_lock); 2392 2393 return (err); 2394 } 2395 2396 static int 2397 usbgem_remove_multicast(struct usbgem_dev *dp, const uint8_t *ep) 2398 { 2399 size_t len; 2400 int i; 2401 int cnt; 2402 int err; 2403 2404 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2405 2406 sema_p(&dp->rxfilter_lock); 2407 dp->mc_count_req--; 2408 cnt = dp->mc_count; 2409 for (i = 0; i < cnt; i++) { 2410 if (bcmp(ep, &dp->mc_list[i].addr, ETHERADDRL)) { 2411 continue; 2412 } 2413 /* shrink the mclist by copying forward */ 2414 len = (cnt - (i + 1)) * sizeof (*dp->mc_list); 2415 if (len > 0) { 2416 bcopy(&dp->mc_list[i+1], &dp->mc_list[i], len); 2417 } 2418 dp->mc_count--; 2419 break; 2420 } 2421 2422 if (dp->mc_count_req != dp->mc_count) { 2423 /* multicast address list overflow */ 2424 dp->rxmode |= RXMODE_MULTI_OVF; 2425 } else { 2426 dp->rxmode &= ~RXMODE_MULTI_OVF; 2427 } 2428 2429 if (dp->mac_state != MAC_STATE_DISCONNECTED) { 2430 err = usbgem_hal_set_rx_filter(dp); 2431 } 2432 sema_v(&dp->rxfilter_lock); 2433 2434 return (err); 2435 } 2436 2437 2438 /* ============================================================== */ 2439 /* 2440 * ioctl 2441 */ 2442 /* ============================================================== */ 2443 enum ioc_reply { 2444 IOC_INVAL = -1, /* bad, NAK with EINVAL */ 2445 IOC_DONE, /* OK, reply sent */ 2446 IOC_ACK, /* OK, just send ACK */ 2447 IOC_REPLY, /* OK, just send reply */ 2448 IOC_RESTART_ACK, /* OK, restart & ACK */ 2449 IOC_RESTART_REPLY /* OK, restart & reply */ 2450 }; 2451 2452 2453 static int 2454 usbgem_get_def_val(struct usbgem_dev *dp, mac_prop_id_t pr_num, 2455 uint_t pr_valsize, void *pr_val) 2456 { 2457 link_flowctrl_t fl; 2458 int err = 0; 2459 2460 ASSERT(pr_valsize > 0); 2461 switch (pr_num) { 2462 case MAC_PROP_AUTONEG: 2463 *(uint8_t *)pr_val = 2464 BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG); 2465 break; 2466 2467 case MAC_PROP_FLOWCTRL: 2468 if (pr_valsize < sizeof (link_flowctrl_t)) { 2469 return (EINVAL); 2470 } 2471 switch (dp->ugc.usbgc_flow_control) { 2472 case FLOW_CONTROL_NONE: 2473 fl = LINK_FLOWCTRL_NONE; 2474 break; 2475 case FLOW_CONTROL_SYMMETRIC: 2476 fl = LINK_FLOWCTRL_BI; 2477 break; 2478 case FLOW_CONTROL_TX_PAUSE: 2479 fl = LINK_FLOWCTRL_TX; 2480 break; 2481 case FLOW_CONTROL_RX_PAUSE: 2482 fl = LINK_FLOWCTRL_RX; 2483 break; 2484 } 2485 bcopy(&fl, pr_val, sizeof (fl)); 2486 break; 2487 2488 case MAC_PROP_ADV_1000FDX_CAP: 2489 case MAC_PROP_EN_1000FDX_CAP: 2490 *(uint8_t *)pr_val = 2491 (dp->mii_xstatus & MII_XSTATUS_1000BASET_FD) || 2492 (dp->mii_xstatus & MII_XSTATUS_1000BASEX_FD); 2493 break; 2494 2495 case MAC_PROP_ADV_1000HDX_CAP: 2496 case MAC_PROP_EN_1000HDX_CAP: 2497 *(uint8_t *)pr_val = 2498 (dp->mii_xstatus & MII_XSTATUS_1000BASET) || 2499 (dp->mii_xstatus & MII_XSTATUS_1000BASEX); 2500 break; 2501 2502 case MAC_PROP_ADV_100T4_CAP: 2503 case MAC_PROP_EN_100T4_CAP: 2504 *(uint8_t *)pr_val = 2505 BOOLEAN(dp->mii_status & MII_STATUS_100_BASE_T4); 2506 break; 2507 2508 case MAC_PROP_ADV_100FDX_CAP: 2509 case MAC_PROP_EN_100FDX_CAP: 2510 *(uint8_t *)pr_val = 2511 BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD); 2512 break; 2513 2514 case MAC_PROP_ADV_100HDX_CAP: 2515 case MAC_PROP_EN_100HDX_CAP: 2516 *(uint8_t *)pr_val = 2517 BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX); 2518 break; 2519 2520 case MAC_PROP_ADV_10FDX_CAP: 2521 case MAC_PROP_EN_10FDX_CAP: 2522 *(uint8_t *)pr_val = 2523 BOOLEAN(dp->mii_status & MII_STATUS_10_FD); 2524 break; 2525 2526 case MAC_PROP_ADV_10HDX_CAP: 2527 case MAC_PROP_EN_10HDX_CAP: 2528 *(uint8_t *)pr_val = 2529 BOOLEAN(dp->mii_status & MII_STATUS_10); 2530 break; 2531 2532 default: 2533 err = ENOTSUP; 2534 break; 2535 } 2536 return (err); 2537 } 2538 2539 static void 2540 usbgem_m_propinfo(void *arg, const char *pr_name, mac_prop_id_t pr_num, 2541 mac_prop_info_handle_t prh) 2542 { 2543 struct usbgem_dev *dp = arg; 2544 link_flowctrl_t fl; 2545 2546 /* 2547 * By default permissions are read/write unless specified 2548 * otherwise by the driver. 2549 */ 2550 2551 switch (pr_num) { 2552 case MAC_PROP_DUPLEX: 2553 case MAC_PROP_SPEED: 2554 case MAC_PROP_STATUS: 2555 case MAC_PROP_ADV_1000FDX_CAP: 2556 case MAC_PROP_ADV_1000HDX_CAP: 2557 case MAC_PROP_ADV_100FDX_CAP: 2558 case MAC_PROP_ADV_100HDX_CAP: 2559 case MAC_PROP_ADV_10FDX_CAP: 2560 case MAC_PROP_ADV_10HDX_CAP: 2561 case MAC_PROP_ADV_100T4_CAP: 2562 case MAC_PROP_EN_100T4_CAP: 2563 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2564 break; 2565 2566 case MAC_PROP_EN_1000FDX_CAP: 2567 if ((dp->mii_xstatus_ro & MII_XSTATUS_1000BASET_FD) == 0) { 2568 mac_prop_info_set_default_uint8(prh, 2569 BOOLEAN( 2570 dp->mii_xstatus & MII_XSTATUS_1000BASET_FD)); 2571 } else if ((dp->mii_xstatus_ro & MII_XSTATUS_1000BASEX_FD) 2572 == 0) { 2573 mac_prop_info_set_default_uint8(prh, 2574 BOOLEAN( 2575 dp->mii_xstatus & MII_XSTATUS_1000BASEX_FD)); 2576 } else { 2577 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2578 } 2579 break; 2580 2581 case MAC_PROP_EN_1000HDX_CAP: 2582 if ((dp->mii_xstatus_ro & MII_XSTATUS_1000BASET) == 0) { 2583 mac_prop_info_set_default_uint8(prh, 2584 BOOLEAN( 2585 dp->mii_xstatus & MII_XSTATUS_1000BASET)); 2586 } else if ((dp->mii_xstatus_ro & MII_XSTATUS_1000BASEX) == 0) { 2587 mac_prop_info_set_default_uint8(prh, 2588 BOOLEAN( 2589 dp->mii_xstatus & MII_XSTATUS_1000BASEX)); 2590 } else { 2591 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2592 } 2593 break; 2594 2595 case MAC_PROP_EN_100FDX_CAP: 2596 if ((dp->mii_status_ro & MII_STATUS_100_BASEX_FD) == 0) { 2597 mac_prop_info_set_default_uint8(prh, 2598 BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD)); 2599 } else { 2600 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2601 } 2602 break; 2603 2604 case MAC_PROP_EN_100HDX_CAP: 2605 if ((dp->mii_status_ro & MII_STATUS_100_BASEX) == 0) { 2606 mac_prop_info_set_default_uint8(prh, 2607 BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX)); 2608 } else { 2609 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2610 } 2611 break; 2612 2613 case MAC_PROP_EN_10FDX_CAP: 2614 if ((dp->mii_status_ro & MII_STATUS_10_FD) == 0) { 2615 mac_prop_info_set_default_uint8(prh, 2616 BOOLEAN(dp->mii_status & MII_STATUS_10_FD)); 2617 } else { 2618 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2619 } 2620 break; 2621 2622 case MAC_PROP_EN_10HDX_CAP: 2623 if ((dp->mii_status_ro & MII_STATUS_10) == 0) { 2624 mac_prop_info_set_default_uint8(prh, 2625 BOOLEAN(dp->mii_status & MII_STATUS_10)); 2626 } else { 2627 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2628 } 2629 break; 2630 2631 case MAC_PROP_AUTONEG: 2632 if ((dp->mii_status_ro & MII_STATUS_CANAUTONEG) == 0) { 2633 mac_prop_info_set_default_uint8(prh, 2634 BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG)); 2635 } else { 2636 mac_prop_info_set_perm(prh, MAC_PROP_PERM_READ); 2637 } 2638 break; 2639 2640 case MAC_PROP_FLOWCTRL: 2641 switch (dp->ugc.usbgc_flow_control) { 2642 case FLOW_CONTROL_NONE: 2643 fl = LINK_FLOWCTRL_NONE; 2644 break; 2645 case FLOW_CONTROL_SYMMETRIC: 2646 fl = LINK_FLOWCTRL_BI; 2647 break; 2648 case FLOW_CONTROL_TX_PAUSE: 2649 fl = LINK_FLOWCTRL_TX; 2650 break; 2651 case FLOW_CONTROL_RX_PAUSE: 2652 fl = LINK_FLOWCTRL_RX; 2653 break; 2654 } 2655 mac_prop_info_set_default_link_flowctrl(prh, fl); 2656 break; 2657 2658 case MAC_PROP_MTU: 2659 mac_prop_info_set_range_uint32(prh, 2660 dp->ugc.usbgc_min_mtu, dp->ugc.usbgc_max_mtu); 2661 break; 2662 2663 case MAC_PROP_PRIVATE: 2664 break; 2665 } 2666 } 2667 2668 static int 2669 usbgem_m_setprop(void *arg, const char *pr_name, mac_prop_id_t pr_num, 2670 uint_t pr_valsize, const void *pr_val) 2671 { 2672 struct usbgem_dev *dp = arg; 2673 int err = 0; 2674 boolean_t update = B_FALSE; 2675 link_flowctrl_t flowctrl; 2676 uint32_t cur_mtu, new_mtu; 2677 2678 rw_enter(&dp->dev_state_lock, RW_WRITER); 2679 switch (pr_num) { 2680 case MAC_PROP_EN_1000FDX_CAP: 2681 if ((dp->mii_xstatus_ro & MII_XSTATUS_1000BASET_FD) == 0 || 2682 (dp->mii_xstatus_ro & MII_XSTATUS_1000BASEX_FD) == 0) { 2683 if (dp->anadv_1000fdx != *(uint8_t *)pr_val) { 2684 dp->anadv_1000fdx = *(uint8_t *)pr_val; 2685 update = B_TRUE; 2686 } 2687 } else { 2688 err = ENOTSUP; 2689 } 2690 break; 2691 2692 case MAC_PROP_EN_1000HDX_CAP: 2693 if ((dp->mii_xstatus_ro & MII_XSTATUS_1000BASET) == 0 || 2694 (dp->mii_xstatus_ro & MII_XSTATUS_1000BASEX) == 0) { 2695 if (dp->anadv_1000hdx != *(uint8_t *)pr_val) { 2696 dp->anadv_1000hdx = *(uint8_t *)pr_val; 2697 update = B_TRUE; 2698 } 2699 } else { 2700 err = ENOTSUP; 2701 } 2702 break; 2703 2704 case MAC_PROP_EN_100FDX_CAP: 2705 if ((dp->mii_status_ro & MII_STATUS_100_BASEX_FD) == 0) { 2706 if (dp->anadv_100fdx != *(uint8_t *)pr_val) { 2707 dp->anadv_100fdx = *(uint8_t *)pr_val; 2708 update = B_TRUE; 2709 } 2710 } else { 2711 err = ENOTSUP; 2712 } 2713 break; 2714 2715 case MAC_PROP_EN_100HDX_CAP: 2716 if ((dp->mii_status_ro & MII_STATUS_100_BASEX) == 0) { 2717 if (dp->anadv_100hdx != *(uint8_t *)pr_val) { 2718 dp->anadv_100hdx = *(uint8_t *)pr_val; 2719 update = B_TRUE; 2720 } 2721 } else { 2722 err = ENOTSUP; 2723 } 2724 break; 2725 2726 case MAC_PROP_EN_10FDX_CAP: 2727 if ((dp->mii_status_ro & MII_STATUS_10_FD) == 0) { 2728 if (dp->anadv_10fdx != *(uint8_t *)pr_val) { 2729 dp->anadv_10fdx = *(uint8_t *)pr_val; 2730 update = B_TRUE; 2731 } 2732 } else { 2733 err = ENOTSUP; 2734 } 2735 break; 2736 2737 case MAC_PROP_EN_10HDX_CAP: 2738 if ((dp->mii_status_ro & MII_STATUS_10_FD) == 0) { 2739 if (dp->anadv_10hdx != *(uint8_t *)pr_val) { 2740 dp->anadv_10hdx = *(uint8_t *)pr_val; 2741 update = B_TRUE; 2742 } 2743 } else { 2744 err = ENOTSUP; 2745 } 2746 break; 2747 2748 case MAC_PROP_AUTONEG: 2749 if ((dp->mii_status_ro & MII_STATUS_CANAUTONEG) == 0) { 2750 if (dp->anadv_autoneg != *(uint8_t *)pr_val) { 2751 dp->anadv_autoneg = *(uint8_t *)pr_val; 2752 update = B_TRUE; 2753 } 2754 } else { 2755 err = ENOTSUP; 2756 } 2757 break; 2758 2759 case MAC_PROP_FLOWCTRL: 2760 bcopy(pr_val, &flowctrl, sizeof (flowctrl)); 2761 2762 switch (flowctrl) { 2763 default: 2764 err = EINVAL; 2765 break; 2766 2767 case LINK_FLOWCTRL_NONE: 2768 if (dp->flow_control != FLOW_CONTROL_NONE) { 2769 dp->flow_control = FLOW_CONTROL_NONE; 2770 update = B_TRUE; 2771 } 2772 break; 2773 2774 case LINK_FLOWCTRL_RX: 2775 if (dp->flow_control != FLOW_CONTROL_RX_PAUSE) { 2776 dp->flow_control = FLOW_CONTROL_RX_PAUSE; 2777 update = B_TRUE; 2778 } 2779 break; 2780 2781 case LINK_FLOWCTRL_TX: 2782 if (dp->flow_control != FLOW_CONTROL_TX_PAUSE) { 2783 dp->flow_control = FLOW_CONTROL_TX_PAUSE; 2784 update = B_TRUE; 2785 } 2786 break; 2787 2788 case LINK_FLOWCTRL_BI: 2789 if (dp->flow_control != FLOW_CONTROL_SYMMETRIC) { 2790 dp->flow_control = FLOW_CONTROL_SYMMETRIC; 2791 update = B_TRUE; 2792 } 2793 break; 2794 } 2795 break; 2796 2797 case MAC_PROP_ADV_1000FDX_CAP: 2798 case MAC_PROP_ADV_1000HDX_CAP: 2799 case MAC_PROP_ADV_100FDX_CAP: 2800 case MAC_PROP_ADV_100HDX_CAP: 2801 case MAC_PROP_ADV_10FDX_CAP: 2802 case MAC_PROP_ADV_10HDX_CAP: 2803 case MAC_PROP_STATUS: 2804 case MAC_PROP_SPEED: 2805 case MAC_PROP_DUPLEX: 2806 err = ENOTSUP; /* read-only prop. Can't set this. */ 2807 break; 2808 2809 case MAC_PROP_MTU: 2810 bcopy(pr_val, &new_mtu, sizeof (new_mtu)); 2811 if (new_mtu != dp->mtu) { 2812 err = EINVAL; 2813 } 2814 break; 2815 2816 case MAC_PROP_PRIVATE: 2817 err = ENOTSUP; 2818 break; 2819 2820 default: 2821 err = ENOTSUP; 2822 break; 2823 } 2824 2825 if (update) { 2826 /* sync with PHY */ 2827 usbgem_choose_forcedmode(dp); 2828 dp->mii_state = MII_STATE_UNKNOWN; 2829 cv_signal(&dp->link_watcher_wait_cv); 2830 } 2831 rw_exit(&dp->dev_state_lock); 2832 return (err); 2833 } 2834 2835 static int 2836 usbgem_m_getprop(void *arg, const char *pr_name, mac_prop_id_t pr_num, 2837 uint_t pr_valsize, void *pr_val) 2838 { 2839 struct usbgem_dev *dp = arg; 2840 int err = 0; 2841 link_flowctrl_t flowctrl; 2842 uint64_t tmp = 0; 2843 2844 if (pr_valsize == 0) { 2845 return (EINVAL); 2846 } 2847 2848 bzero(pr_val, pr_valsize); 2849 rw_enter(&dp->dev_state_lock, RW_READER); 2850 switch (pr_num) { 2851 case MAC_PROP_DUPLEX: 2852 if (pr_valsize >= sizeof (link_duplex_t)) { 2853 if (dp->mii_state != MII_STATE_LINKUP) { 2854 *(link_duplex_t *)pr_val = LINK_DUPLEX_UNKNOWN; 2855 } else if (dp->full_duplex) { 2856 *(link_duplex_t *)pr_val = LINK_DUPLEX_FULL; 2857 } else { 2858 *(link_duplex_t *)pr_val = LINK_DUPLEX_HALF; 2859 } 2860 } else { 2861 err = EINVAL; 2862 } 2863 break; 2864 case MAC_PROP_SPEED: 2865 if (pr_valsize >= sizeof (uint64_t)) { 2866 switch (dp->speed) { 2867 case USBGEM_SPD_1000: 2868 tmp = 1000000000; 2869 break; 2870 case USBGEM_SPD_100: 2871 tmp = 100000000; 2872 break; 2873 case USBGEM_SPD_10: 2874 tmp = 10000000; 2875 break; 2876 default: 2877 tmp = 0; 2878 } 2879 bcopy(&tmp, pr_val, sizeof (tmp)); 2880 } else { 2881 err = EINVAL; 2882 } 2883 break; 2884 2885 case MAC_PROP_AUTONEG: 2886 *(uint8_t *)pr_val = dp->anadv_autoneg; 2887 break; 2888 2889 case MAC_PROP_FLOWCTRL: 2890 if (pr_valsize >= sizeof (link_flowctrl_t)) { 2891 switch (dp->flow_control) { 2892 case FLOW_CONTROL_NONE: 2893 flowctrl = LINK_FLOWCTRL_NONE; 2894 break; 2895 case FLOW_CONTROL_RX_PAUSE: 2896 flowctrl = LINK_FLOWCTRL_RX; 2897 break; 2898 case FLOW_CONTROL_TX_PAUSE: 2899 flowctrl = LINK_FLOWCTRL_TX; 2900 break; 2901 case FLOW_CONTROL_SYMMETRIC: 2902 flowctrl = LINK_FLOWCTRL_BI; 2903 break; 2904 } 2905 bcopy(&flowctrl, pr_val, sizeof (flowctrl)); 2906 } else { 2907 err = EINVAL; 2908 } 2909 break; 2910 2911 case MAC_PROP_ADV_1000FDX_CAP: 2912 case MAC_PROP_ADV_1000HDX_CAP: 2913 case MAC_PROP_ADV_100FDX_CAP: 2914 case MAC_PROP_ADV_100HDX_CAP: 2915 case MAC_PROP_ADV_10FDX_CAP: 2916 case MAC_PROP_ADV_10HDX_CAP: 2917 case MAC_PROP_ADV_100T4_CAP: 2918 usbgem_get_def_val(dp, pr_num, pr_valsize, pr_val); 2919 break; 2920 2921 case MAC_PROP_EN_1000FDX_CAP: 2922 *(uint8_t *)pr_val = dp->anadv_1000fdx; 2923 break; 2924 2925 case MAC_PROP_EN_1000HDX_CAP: 2926 *(uint8_t *)pr_val = dp->anadv_1000hdx; 2927 break; 2928 2929 case MAC_PROP_EN_100FDX_CAP: 2930 *(uint8_t *)pr_val = dp->anadv_100fdx; 2931 break; 2932 2933 case MAC_PROP_EN_100HDX_CAP: 2934 *(uint8_t *)pr_val = dp->anadv_100hdx; 2935 break; 2936 2937 case MAC_PROP_EN_10FDX_CAP: 2938 *(uint8_t *)pr_val = dp->anadv_10fdx; 2939 break; 2940 2941 case MAC_PROP_EN_10HDX_CAP: 2942 *(uint8_t *)pr_val = dp->anadv_10hdx; 2943 break; 2944 2945 case MAC_PROP_EN_100T4_CAP: 2946 *(uint8_t *)pr_val = dp->anadv_100t4; 2947 break; 2948 2949 case MAC_PROP_PRIVATE: 2950 err = ENOTSUP; 2951 break; 2952 2953 default: 2954 err = ENOTSUP; 2955 break; 2956 } 2957 2958 rw_exit(&dp->dev_state_lock); 2959 return (err); 2960 } 2961 2962 static void 2963 usbgem_mac_ioctl(struct usbgem_dev *dp, queue_t *wq, mblk_t *mp) 2964 { 2965 struct iocblk *iocp; 2966 enum ioc_reply status; 2967 2968 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 2969 2970 /* 2971 * Validate the command before bothering with the mutex ... 2972 */ 2973 iocp = (void *)mp->b_rptr; 2974 iocp->ioc_error = 0; 2975 2976 DPRINTF(1, (CE_CONT, "%s: %s cmd:0x%x", dp->name, __func__, 2977 iocp->ioc_cmd)); 2978 2979 miocnak(wq, mp, 0, EINVAL); 2980 } 2981 2982 static int 2983 usbgem_mac_xcvr_inuse(struct usbgem_dev *dp) 2984 { 2985 int val = XCVR_UNDEFINED; 2986 2987 if ((dp->mii_status & MII_STATUS_XSTATUS) == 0) { 2988 if (dp->mii_status & MII_STATUS_100_BASE_T4) { 2989 val = XCVR_100T4; 2990 } else if (dp->mii_status & 2991 (MII_STATUS_100_BASEX_FD | 2992 MII_STATUS_100_BASEX)) { 2993 val = XCVR_100X; 2994 } else if (dp->mii_status & 2995 (MII_STATUS_100_BASE_T2_FD | 2996 MII_STATUS_100_BASE_T2)) { 2997 val = XCVR_100T2; 2998 } else if (dp->mii_status & 2999 (MII_STATUS_10_FD | MII_STATUS_10)) { 3000 val = XCVR_10; 3001 } 3002 } else if (dp->mii_xstatus & 3003 (MII_XSTATUS_1000BASET_FD | MII_XSTATUS_1000BASET)) { 3004 val = XCVR_1000T; 3005 } else if (dp->mii_xstatus & 3006 (MII_XSTATUS_1000BASEX_FD | MII_XSTATUS_1000BASEX)) { 3007 val = XCVR_1000X; 3008 } 3009 3010 return (val); 3011 } 3012 3013 /* ============================================================== */ 3014 /* 3015 * GLDv3 interface 3016 */ 3017 /* ============================================================== */ 3018 static int usbgem_m_getstat(void *, uint_t, uint64_t *); 3019 static int usbgem_m_start(void *); 3020 static void usbgem_m_stop(void *); 3021 static int usbgem_m_setpromisc(void *, boolean_t); 3022 static int usbgem_m_multicst(void *, boolean_t, const uint8_t *); 3023 static int usbgem_m_unicst(void *, const uint8_t *); 3024 static mblk_t *usbgem_m_tx(void *, mblk_t *); 3025 static void usbgem_m_ioctl(void *, queue_t *, mblk_t *); 3026 static int usbgem_m_setprop(void *, const char *, mac_prop_id_t, 3027 uint_t, const void *); 3028 static int usbgem_m_getprop(void *, const char *, mac_prop_id_t, 3029 uint_t, void *); 3030 3031 static mac_callbacks_t gem_m_callbacks = { 3032 MC_IOCTL | MC_SETPROP | MC_GETPROP | MC_PROPINFO, 3033 usbgem_m_getstat, 3034 usbgem_m_start, 3035 usbgem_m_stop, 3036 usbgem_m_setpromisc, 3037 usbgem_m_multicst, 3038 usbgem_m_unicst, 3039 usbgem_m_tx, 3040 NULL, 3041 usbgem_m_ioctl, 3042 NULL, /* m_getcapab */ 3043 NULL, 3044 NULL, 3045 usbgem_m_setprop, 3046 usbgem_m_getprop, 3047 usbgem_m_propinfo, 3048 }; 3049 3050 static int 3051 usbgem_m_start(void *arg) 3052 { 3053 int ret; 3054 int err; 3055 struct usbgem_dev *dp = arg; 3056 3057 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3058 3059 err = EIO; 3060 3061 rw_enter(&dp->dev_state_lock, RW_WRITER); 3062 dp->nic_state = NIC_STATE_ONLINE; 3063 3064 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 3065 err = 0; 3066 goto x; 3067 } 3068 if (usbgem_mac_init(dp) != USB_SUCCESS) { 3069 goto x; 3070 } 3071 3072 /* initialize rx filter state */ 3073 sema_p(&dp->rxfilter_lock); 3074 dp->mc_count = 0; 3075 dp->mc_count_req = 0; 3076 3077 bcopy(dp->dev_addr.ether_addr_octet, 3078 dp->cur_addr.ether_addr_octet, ETHERADDRL); 3079 dp->rxmode |= RXMODE_ENABLE; 3080 3081 ret = usbgem_hal_set_rx_filter(dp); 3082 sema_v(&dp->rxfilter_lock); 3083 3084 if (ret != USB_SUCCESS) { 3085 goto x; 3086 } 3087 3088 if (dp->mii_state == MII_STATE_LINKUP) { 3089 /* setup media mode if the link have been up */ 3090 if (usbgem_hal_set_media(dp) != USB_SUCCESS) { 3091 goto x; 3092 } 3093 if (usbgem_mac_start(dp) != USB_SUCCESS) { 3094 goto x; 3095 } 3096 } 3097 3098 err = 0; 3099 x: 3100 rw_exit(&dp->dev_state_lock); 3101 return (err); 3102 } 3103 3104 static void 3105 usbgem_m_stop(void *arg) 3106 { 3107 struct usbgem_dev *dp = arg; 3108 3109 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3110 3111 /* stop rx gracefully */ 3112 rw_enter(&dp->dev_state_lock, RW_READER); 3113 sema_p(&dp->rxfilter_lock); 3114 dp->rxmode &= ~RXMODE_ENABLE; 3115 3116 if (dp->mac_state != MAC_STATE_DISCONNECTED) { 3117 (void) usbgem_hal_set_rx_filter(dp); 3118 } 3119 sema_v(&dp->rxfilter_lock); 3120 rw_exit(&dp->dev_state_lock); 3121 3122 /* make the nic state inactive */ 3123 rw_enter(&dp->dev_state_lock, RW_WRITER); 3124 dp->nic_state = NIC_STATE_STOPPED; 3125 3126 /* stop mac completely */ 3127 if (dp->mac_state != MAC_STATE_DISCONNECTED) { 3128 (void) usbgem_mac_stop(dp, MAC_STATE_STOPPED, STOP_GRACEFUL); 3129 } 3130 rw_exit(&dp->dev_state_lock); 3131 } 3132 3133 static int 3134 usbgem_m_multicst(void *arg, boolean_t add, const uint8_t *ep) 3135 { 3136 int err; 3137 int ret; 3138 struct usbgem_dev *dp = arg; 3139 3140 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3141 3142 rw_enter(&dp->dev_state_lock, RW_READER); 3143 if (add) { 3144 ret = usbgem_add_multicast(dp, ep); 3145 } else { 3146 ret = usbgem_remove_multicast(dp, ep); 3147 } 3148 rw_exit(&dp->dev_state_lock); 3149 3150 err = 0; 3151 if (ret != USB_SUCCESS) { 3152 err = EIO; 3153 } 3154 3155 return (err); 3156 } 3157 3158 static int 3159 usbgem_m_setpromisc(void *arg, boolean_t on) 3160 { 3161 int err; 3162 struct usbgem_dev *dp = arg; 3163 3164 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3165 3166 rw_enter(&dp->dev_state_lock, RW_READER); 3167 3168 sema_p(&dp->rxfilter_lock); 3169 if (on) { 3170 dp->rxmode |= RXMODE_PROMISC; 3171 } else { 3172 dp->rxmode &= ~RXMODE_PROMISC; 3173 } 3174 3175 err = 0; 3176 if (dp->mac_state != MAC_STATE_DISCONNECTED) { 3177 if (usbgem_hal_set_rx_filter(dp) != USB_SUCCESS) { 3178 err = EIO; 3179 } 3180 } 3181 sema_v(&dp->rxfilter_lock); 3182 3183 rw_exit(&dp->dev_state_lock); 3184 3185 return (err); 3186 } 3187 3188 int 3189 usbgem_m_getstat(void *arg, uint_t stat, uint64_t *valp) 3190 { 3191 uint64_t val; 3192 struct usbgem_dev *dp = arg; 3193 struct usbgem_stats *gstp = &dp->stats; 3194 3195 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3196 3197 rw_enter(&dp->dev_state_lock, RW_READER); 3198 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 3199 rw_exit(&dp->dev_state_lock); 3200 return (0); 3201 } 3202 3203 (void) usbgem_hal_get_stats(dp); 3204 rw_exit(&dp->dev_state_lock); 3205 3206 switch (stat) { 3207 case MAC_STAT_IFSPEED: 3208 val = usbgem_speed_value[dp->speed] *1000000ull; 3209 break; 3210 3211 case MAC_STAT_MULTIRCV: 3212 val = gstp->rmcast; 3213 break; 3214 3215 case MAC_STAT_BRDCSTRCV: 3216 val = gstp->rbcast; 3217 break; 3218 3219 case MAC_STAT_MULTIXMT: 3220 val = gstp->omcast; 3221 break; 3222 3223 case MAC_STAT_BRDCSTXMT: 3224 val = gstp->obcast; 3225 break; 3226 3227 case MAC_STAT_NORCVBUF: 3228 val = gstp->norcvbuf + gstp->missed; 3229 break; 3230 3231 case MAC_STAT_IERRORS: 3232 val = gstp->errrcv; 3233 break; 3234 3235 case MAC_STAT_NOXMTBUF: 3236 val = gstp->noxmtbuf; 3237 break; 3238 3239 case MAC_STAT_OERRORS: 3240 val = gstp->errxmt; 3241 break; 3242 3243 case MAC_STAT_COLLISIONS: 3244 val = gstp->collisions; 3245 break; 3246 3247 case MAC_STAT_RBYTES: 3248 val = gstp->rbytes; 3249 break; 3250 3251 case MAC_STAT_IPACKETS: 3252 val = gstp->rpackets; 3253 break; 3254 3255 case MAC_STAT_OBYTES: 3256 val = gstp->obytes; 3257 break; 3258 3259 case MAC_STAT_OPACKETS: 3260 val = gstp->opackets; 3261 break; 3262 3263 case MAC_STAT_UNDERFLOWS: 3264 val = gstp->underflow; 3265 break; 3266 3267 case MAC_STAT_OVERFLOWS: 3268 val = gstp->overflow; 3269 break; 3270 3271 case ETHER_STAT_ALIGN_ERRORS: 3272 val = gstp->frame; 3273 break; 3274 3275 case ETHER_STAT_FCS_ERRORS: 3276 val = gstp->crc; 3277 break; 3278 3279 case ETHER_STAT_FIRST_COLLISIONS: 3280 val = gstp->first_coll; 3281 break; 3282 3283 case ETHER_STAT_MULTI_COLLISIONS: 3284 val = gstp->multi_coll; 3285 break; 3286 3287 case ETHER_STAT_SQE_ERRORS: 3288 val = gstp->sqe; 3289 break; 3290 3291 case ETHER_STAT_DEFER_XMTS: 3292 val = gstp->defer; 3293 break; 3294 3295 case ETHER_STAT_TX_LATE_COLLISIONS: 3296 val = gstp->xmtlatecoll; 3297 break; 3298 3299 case ETHER_STAT_EX_COLLISIONS: 3300 val = gstp->excoll; 3301 break; 3302 3303 case ETHER_STAT_MACXMT_ERRORS: 3304 val = gstp->xmit_internal_err; 3305 break; 3306 3307 case ETHER_STAT_CARRIER_ERRORS: 3308 val = gstp->nocarrier; 3309 break; 3310 3311 case ETHER_STAT_TOOLONG_ERRORS: 3312 val = gstp->frame_too_long; 3313 break; 3314 3315 case ETHER_STAT_MACRCV_ERRORS: 3316 val = gstp->rcv_internal_err; 3317 break; 3318 3319 case ETHER_STAT_XCVR_ADDR: 3320 val = dp->mii_phy_addr; 3321 break; 3322 3323 case ETHER_STAT_XCVR_ID: 3324 val = dp->mii_phy_id; 3325 break; 3326 3327 case ETHER_STAT_XCVR_INUSE: 3328 val = usbgem_mac_xcvr_inuse(dp); 3329 break; 3330 3331 case ETHER_STAT_CAP_1000FDX: 3332 val = (dp->mii_xstatus & MII_XSTATUS_1000BASET_FD) || 3333 (dp->mii_xstatus & MII_XSTATUS_1000BASEX_FD); 3334 break; 3335 3336 case ETHER_STAT_CAP_1000HDX: 3337 val = (dp->mii_xstatus & MII_XSTATUS_1000BASET) || 3338 (dp->mii_xstatus & MII_XSTATUS_1000BASEX); 3339 break; 3340 3341 case ETHER_STAT_CAP_100FDX: 3342 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD); 3343 break; 3344 3345 case ETHER_STAT_CAP_100HDX: 3346 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX); 3347 break; 3348 3349 case ETHER_STAT_CAP_10FDX: 3350 val = BOOLEAN(dp->mii_status & MII_STATUS_10_FD); 3351 break; 3352 3353 case ETHER_STAT_CAP_10HDX: 3354 val = BOOLEAN(dp->mii_status & MII_STATUS_10); 3355 break; 3356 3357 case ETHER_STAT_CAP_ASMPAUSE: 3358 val = dp->ugc.usbgc_flow_control > FLOW_CONTROL_SYMMETRIC; 3359 break; 3360 3361 case ETHER_STAT_CAP_PAUSE: 3362 val = dp->ugc.usbgc_flow_control != FLOW_CONTROL_NONE; 3363 break; 3364 3365 case ETHER_STAT_CAP_AUTONEG: 3366 val = BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG); 3367 break; 3368 3369 case ETHER_STAT_ADV_CAP_1000FDX: 3370 val = dp->anadv_1000fdx; 3371 break; 3372 3373 case ETHER_STAT_ADV_CAP_1000HDX: 3374 val = dp->anadv_1000hdx; 3375 break; 3376 3377 case ETHER_STAT_ADV_CAP_100FDX: 3378 val = dp->anadv_100fdx; 3379 break; 3380 3381 case ETHER_STAT_ADV_CAP_100HDX: 3382 val = dp->anadv_100hdx; 3383 break; 3384 3385 case ETHER_STAT_ADV_CAP_10FDX: 3386 val = dp->anadv_10fdx; 3387 break; 3388 3389 case ETHER_STAT_ADV_CAP_10HDX: 3390 val = dp->anadv_10hdx; 3391 break; 3392 3393 case ETHER_STAT_ADV_CAP_ASMPAUSE: 3394 val = dp->anadv_asmpause; 3395 break; 3396 3397 case ETHER_STAT_ADV_CAP_PAUSE: 3398 val = dp->anadv_pause; 3399 break; 3400 3401 case ETHER_STAT_ADV_CAP_AUTONEG: 3402 val = dp->anadv_autoneg; 3403 break; 3404 3405 case ETHER_STAT_LP_CAP_1000FDX: 3406 val = BOOLEAN(dp->mii_stat1000 & MII_1000TS_LP_FULL); 3407 break; 3408 3409 case ETHER_STAT_LP_CAP_1000HDX: 3410 val = BOOLEAN(dp->mii_stat1000 & MII_1000TS_LP_HALF); 3411 break; 3412 3413 case ETHER_STAT_LP_CAP_100FDX: 3414 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_TX_FD); 3415 break; 3416 3417 case ETHER_STAT_LP_CAP_100HDX: 3418 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_TX); 3419 break; 3420 3421 case ETHER_STAT_LP_CAP_10FDX: 3422 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_10BASE_T_FD); 3423 break; 3424 3425 case ETHER_STAT_LP_CAP_10HDX: 3426 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_10BASE_T); 3427 break; 3428 3429 case ETHER_STAT_LP_CAP_ASMPAUSE: 3430 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_ASM_DIR); 3431 break; 3432 3433 case ETHER_STAT_LP_CAP_PAUSE: 3434 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_PAUSE); 3435 break; 3436 3437 case ETHER_STAT_LP_CAP_AUTONEG: 3438 val = BOOLEAN(dp->mii_exp & MII_AN_EXP_LPCANAN); 3439 break; 3440 3441 case ETHER_STAT_LINK_ASMPAUSE: 3442 val = BOOLEAN(dp->flow_control & 2); 3443 break; 3444 3445 case ETHER_STAT_LINK_PAUSE: 3446 val = BOOLEAN(dp->flow_control & 1); 3447 break; 3448 3449 case ETHER_STAT_LINK_AUTONEG: 3450 val = dp->anadv_autoneg && 3451 BOOLEAN(dp->mii_exp & MII_AN_EXP_LPCANAN); 3452 break; 3453 3454 case ETHER_STAT_LINK_DUPLEX: 3455 val = (dp->mii_state == MII_STATE_LINKUP) ? 3456 (dp->full_duplex ? 2 : 1) : 0; 3457 break; 3458 3459 case ETHER_STAT_TOOSHORT_ERRORS: 3460 val = gstp->runt; 3461 break; 3462 #ifdef NEVER /* it doesn't make sense */ 3463 case ETHER_STAT_CAP_REMFAULT: 3464 val = B_TRUE; 3465 break; 3466 3467 case ETHER_STAT_ADV_REMFAULT: 3468 val = dp->anadv_remfault; 3469 break; 3470 #endif 3471 case ETHER_STAT_LP_REMFAULT: 3472 val = BOOLEAN(dp->mii_lpable & MII_AN_ADVERT_REMFAULT); 3473 break; 3474 3475 case ETHER_STAT_JABBER_ERRORS: 3476 val = gstp->jabber; 3477 break; 3478 3479 case ETHER_STAT_CAP_100T4: 3480 val = BOOLEAN(dp->mii_status & MII_STATUS_100_BASE_T4); 3481 break; 3482 3483 case ETHER_STAT_ADV_CAP_100T4: 3484 val = dp->anadv_100t4; 3485 break; 3486 3487 case ETHER_STAT_LP_CAP_100T4: 3488 val = BOOLEAN(dp->mii_lpable & MII_ABILITY_100BASE_T4); 3489 break; 3490 3491 default: 3492 #if GEM_DEBUG_LEVEL > 2 3493 cmn_err(CE_WARN, 3494 "%s: unrecognized parameter value = %d", 3495 __func__, stat); 3496 #endif 3497 *valp = 0; 3498 return (ENOTSUP); 3499 } 3500 3501 *valp = val; 3502 3503 return (0); 3504 } 3505 3506 static int 3507 usbgem_m_unicst(void *arg, const uint8_t *mac) 3508 { 3509 int err; 3510 struct usbgem_dev *dp = arg; 3511 3512 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3513 3514 rw_enter(&dp->dev_state_lock, RW_READER); 3515 3516 sema_p(&dp->rxfilter_lock); 3517 bcopy(mac, dp->cur_addr.ether_addr_octet, ETHERADDRL); 3518 dp->rxmode |= RXMODE_ENABLE; 3519 3520 err = 0; 3521 if (dp->mac_state != MAC_STATE_DISCONNECTED) { 3522 if (usbgem_hal_set_rx_filter(dp) != USB_SUCCESS) { 3523 err = EIO; 3524 } 3525 } 3526 sema_v(&dp->rxfilter_lock); 3527 rw_exit(&dp->dev_state_lock); 3528 3529 return (err); 3530 } 3531 3532 /* 3533 * usbgem_m_tx is used only for sending data packets into ethernet wire. 3534 */ 3535 static mblk_t * 3536 usbgem_m_tx(void *arg, mblk_t *mp_head) 3537 { 3538 int limit; 3539 mblk_t *mp; 3540 mblk_t *nmp; 3541 struct usbgem_dev *dp = arg; 3542 3543 DPRINTF(4, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3544 3545 mp = mp_head; 3546 3547 rw_enter(&dp->dev_state_lock, RW_READER); 3548 3549 if (dp->mii_state != MII_STATE_LINKUP || 3550 dp->mac_state != MAC_STATE_ONLINE) { 3551 /* some nics hate to send packets during the link is down */ 3552 for (; mp; mp = nmp) { 3553 nmp = mp->b_next; 3554 mp->b_next = NULL; 3555 freemsg(mp); 3556 } 3557 goto x; 3558 } 3559 3560 ASSERT(dp->nic_state == NIC_STATE_ONLINE); 3561 3562 limit = dp->tx_max_packets; 3563 for (; limit-- && mp; mp = nmp) { 3564 nmp = mp->b_next; 3565 mp->b_next = NULL; 3566 if (usbgem_send_common(dp, mp, 3567 (limit == 0 && nmp) ? 1 : 0)) { 3568 mp->b_next = nmp; 3569 break; 3570 } 3571 } 3572 #ifdef CONFIG_TX_LIMITER 3573 if (mp == mp_head) { 3574 /* no packets were sent, descrease allocation limit */ 3575 mutex_enter(&dp->txlock); 3576 dp->tx_max_packets = max(dp->tx_max_packets - 1, 1); 3577 mutex_exit(&dp->txlock); 3578 } 3579 #endif 3580 x: 3581 rw_exit(&dp->dev_state_lock); 3582 3583 return (mp); 3584 } 3585 3586 static void 3587 usbgem_m_ioctl(void *arg, queue_t *wq, mblk_t *mp) 3588 { 3589 struct usbgem_dev *dp = arg; 3590 3591 DPRINTF(1, (CE_CONT, "!%s: %s: called", 3592 ((struct usbgem_dev *)arg)->name, __func__)); 3593 3594 rw_enter(&dp->dev_state_lock, RW_READER); 3595 usbgem_mac_ioctl((struct usbgem_dev *)arg, wq, mp); 3596 rw_exit(&dp->dev_state_lock); 3597 } 3598 3599 static void 3600 usbgem_gld3_init(struct usbgem_dev *dp, mac_register_t *macp) 3601 { 3602 macp->m_type_ident = MAC_PLUGIN_IDENT_ETHER; 3603 macp->m_driver = dp; 3604 macp->m_dip = dp->dip; 3605 macp->m_src_addr = dp->dev_addr.ether_addr_octet; 3606 macp->m_callbacks = &gem_m_callbacks; 3607 macp->m_min_sdu = 0; 3608 macp->m_max_sdu = dp->mtu; 3609 3610 if (dp->misc_flag & USBGEM_VLAN) { 3611 macp->m_margin = VTAG_SIZE; 3612 } 3613 } 3614 3615 /* ======================================================================== */ 3616 /* 3617 * .conf interface 3618 */ 3619 /* ======================================================================== */ 3620 void 3621 usbgem_generate_macaddr(struct usbgem_dev *dp, uint8_t *mac) 3622 { 3623 extern char hw_serial[]; 3624 char *hw_serial_p; 3625 int i; 3626 uint64_t val; 3627 uint64_t key; 3628 3629 cmn_err(CE_NOTE, 3630 "!%s: using temp ether address," 3631 " do not use this for long time", 3632 dp->name); 3633 3634 /* prefer a fixed address for DHCP */ 3635 hw_serial_p = &hw_serial[0]; 3636 val = stoi(&hw_serial_p); 3637 3638 key = 0; 3639 for (i = 0; i < USBGEM_NAME_LEN; i++) { 3640 if (dp->name[i] == 0) { 3641 break; 3642 } 3643 key ^= dp->name[i]; 3644 } 3645 key ^= ddi_get_instance(dp->dip); 3646 val ^= key << 32; 3647 3648 /* generate a local address */ 3649 mac[0] = 0x02; 3650 mac[1] = (uint8_t)(val >> 32); 3651 mac[2] = (uint8_t)(val >> 24); 3652 mac[3] = (uint8_t)(val >> 16); 3653 mac[4] = (uint8_t)(val >> 8); 3654 mac[5] = (uint8_t)val; 3655 } 3656 3657 boolean_t 3658 usbgem_get_mac_addr_conf(struct usbgem_dev *dp) 3659 { 3660 char propname[32]; 3661 char *valstr; 3662 uint8_t mac[ETHERADDRL]; 3663 char *cp; 3664 int c; 3665 int i; 3666 int j; 3667 uint8_t v; 3668 uint8_t d; 3669 uint8_t ored; 3670 3671 DPRINTF(3, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3672 /* 3673 * Get ethernet address from .conf file 3674 */ 3675 (void) sprintf(propname, "mac-addr"); 3676 if ((ddi_prop_lookup_string(DDI_DEV_T_ANY, dp->dip, 3677 DDI_PROP_DONTPASS, propname, &valstr)) != DDI_PROP_SUCCESS) { 3678 return (B_FALSE); 3679 } 3680 3681 if (strlen(valstr) != ETHERADDRL*3-1) { 3682 goto syntax_err; 3683 } 3684 3685 cp = valstr; 3686 j = 0; 3687 ored = 0; 3688 for (;;) { 3689 v = 0; 3690 for (i = 0; i < 2; i++) { 3691 c = *cp++; 3692 3693 if (c >= 'a' && c <= 'f') { 3694 d = c - 'a' + 10; 3695 } else if (c >= 'A' && c <= 'F') { 3696 d = c - 'A' + 10; 3697 } else if (c >= '0' && c <= '9') { 3698 d = c - '0'; 3699 } else { 3700 goto syntax_err; 3701 } 3702 v = (v << 4) | d; 3703 } 3704 3705 mac[j++] = v; 3706 ored |= v; 3707 if (j == ETHERADDRL) { 3708 /* done */ 3709 break; 3710 } 3711 3712 c = *cp++; 3713 if (c != ':') { 3714 goto syntax_err; 3715 } 3716 } 3717 3718 if (ored == 0) { 3719 usbgem_generate_macaddr(dp, mac); 3720 } 3721 for (i = 0; i < ETHERADDRL; i++) { 3722 dp->dev_addr.ether_addr_octet[i] = mac[i]; 3723 } 3724 ddi_prop_free(valstr); 3725 return (B_TRUE); 3726 3727 syntax_err: 3728 cmn_err(CE_CONT, 3729 "!%s: read mac addr: trying .conf: syntax err %s", 3730 dp->name, valstr); 3731 ddi_prop_free(valstr); 3732 3733 return (B_FALSE); 3734 } 3735 3736 static void 3737 usbgem_read_conf(struct usbgem_dev *dp) 3738 { 3739 int val; 3740 3741 DPRINTF(1, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 3742 3743 /* 3744 * Get media mode infomation from .conf file 3745 */ 3746 dp->anadv_autoneg = usbgem_prop_get_int(dp, "adv_autoneg_cap", 1) != 0; 3747 dp->anadv_1000fdx = usbgem_prop_get_int(dp, "adv_1000fdx_cap", 1) != 0; 3748 dp->anadv_1000hdx = usbgem_prop_get_int(dp, "adv_1000hdx_cap", 1) != 0; 3749 dp->anadv_100t4 = usbgem_prop_get_int(dp, "adv_100T4_cap", 1) != 0; 3750 dp->anadv_100fdx = usbgem_prop_get_int(dp, "adv_100fdx_cap", 1) != 0; 3751 dp->anadv_100hdx = usbgem_prop_get_int(dp, "adv_100hdx_cap", 1) != 0; 3752 dp->anadv_10fdx = usbgem_prop_get_int(dp, "adv_10fdx_cap", 1) != 0; 3753 dp->anadv_10hdx = usbgem_prop_get_int(dp, "adv_10hdx_cap", 1) != 0; 3754 dp->anadv_1000t_ms = usbgem_prop_get_int(dp, "adv_1000t_ms", 0); 3755 3756 if ((ddi_prop_exists(DDI_DEV_T_ANY, dp->dip, 3757 DDI_PROP_DONTPASS, "full-duplex"))) { 3758 dp->full_duplex = 3759 usbgem_prop_get_int(dp, "full-duplex", 1) != 0; 3760 dp->anadv_autoneg = B_FALSE; 3761 if (dp->full_duplex) { 3762 dp->anadv_1000hdx = B_FALSE; 3763 dp->anadv_100hdx = B_FALSE; 3764 dp->anadv_10hdx = B_FALSE; 3765 } else { 3766 dp->anadv_1000fdx = B_FALSE; 3767 dp->anadv_100fdx = B_FALSE; 3768 dp->anadv_10fdx = B_FALSE; 3769 } 3770 } 3771 3772 if ((val = usbgem_prop_get_int(dp, "speed", 0)) > 0) { 3773 dp->anadv_autoneg = B_FALSE; 3774 switch (val) { 3775 case 1000: 3776 dp->speed = USBGEM_SPD_1000; 3777 dp->anadv_100t4 = B_FALSE; 3778 dp->anadv_100fdx = B_FALSE; 3779 dp->anadv_100hdx = B_FALSE; 3780 dp->anadv_10fdx = B_FALSE; 3781 dp->anadv_10hdx = B_FALSE; 3782 break; 3783 case 100: 3784 dp->speed = USBGEM_SPD_100; 3785 dp->anadv_1000fdx = B_FALSE; 3786 dp->anadv_1000hdx = B_FALSE; 3787 dp->anadv_10fdx = B_FALSE; 3788 dp->anadv_10hdx = B_FALSE; 3789 break; 3790 case 10: 3791 dp->speed = USBGEM_SPD_10; 3792 dp->anadv_1000fdx = B_FALSE; 3793 dp->anadv_1000hdx = B_FALSE; 3794 dp->anadv_100t4 = B_FALSE; 3795 dp->anadv_100fdx = B_FALSE; 3796 dp->anadv_100hdx = B_FALSE; 3797 break; 3798 default: 3799 cmn_err(CE_WARN, 3800 "!%s: property %s: illegal value:%d", 3801 dp->name, "speed", val); 3802 dp->anadv_autoneg = B_TRUE; 3803 break; 3804 } 3805 } 3806 val = usbgem_prop_get_int(dp, 3807 "adv_pause", dp->ugc.usbgc_flow_control & 1); 3808 val |= usbgem_prop_get_int(dp, 3809 "adv_asmpause", BOOLEAN(dp->ugc.usbgc_flow_control & 2)) << 1; 3810 if (val > FLOW_CONTROL_RX_PAUSE || val < FLOW_CONTROL_NONE) { 3811 cmn_err(CE_WARN, 3812 "!%s: property %s: illegal value:%d", 3813 dp->name, "flow-control", val); 3814 } else { 3815 val = min(val, dp->ugc.usbgc_flow_control); 3816 } 3817 dp->anadv_pause = BOOLEAN(val & 1); 3818 dp->anadv_asmpause = BOOLEAN(val & 2); 3819 3820 dp->mtu = usbgem_prop_get_int(dp, "mtu", dp->mtu); 3821 dp->txthr = usbgem_prop_get_int(dp, "txthr", dp->txthr); 3822 dp->rxthr = usbgem_prop_get_int(dp, "rxthr", dp->rxthr); 3823 dp->txmaxdma = usbgem_prop_get_int(dp, "txmaxdma", dp->txmaxdma); 3824 dp->rxmaxdma = usbgem_prop_get_int(dp, "rxmaxdma", dp->rxmaxdma); 3825 #ifdef GEM_CONFIG_POLLING 3826 dp->poll_pkt_delay = 3827 usbgem_prop_get_int(dp, "pkt_delay", dp->poll_pkt_delay); 3828 3829 dp->max_poll_interval[GEM_SPD_10] = 3830 usbgem_prop_get_int(dp, "max_poll_interval_10", 3831 dp->max_poll_interval[GEM_SPD_10]); 3832 dp->max_poll_interval[GEM_SPD_100] = 3833 usbgem_prop_get_int(dp, "max_poll_interval_100", 3834 dp->max_poll_interval[GEM_SPD_100]); 3835 dp->max_poll_interval[GEM_SPD_1000] = 3836 usbgem_prop_get_int(dp, "max_poll_interval_1000", 3837 dp->max_poll_interval[GEM_SPD_1000]); 3838 3839 dp->min_poll_interval[GEM_SPD_10] = 3840 usbgem_prop_get_int(dp, "min_poll_interval_10", 3841 dp->min_poll_interval[GEM_SPD_10]); 3842 dp->min_poll_interval[GEM_SPD_100] = 3843 usbgem_prop_get_int(dp, "min_poll_interval_100", 3844 dp->min_poll_interval[GEM_SPD_100]); 3845 dp->min_poll_interval[GEM_SPD_1000] = 3846 usbgem_prop_get_int(dp, "min_poll_interval_1000", 3847 dp->min_poll_interval[GEM_SPD_1000]); 3848 #endif 3849 } 3850 3851 /* 3852 * attach/detatch/usb support 3853 */ 3854 /* ======================================================================== */ 3855 int 3856 usbgem_ctrl_out(struct usbgem_dev *dp, 3857 uint8_t reqt, uint8_t req, uint16_t val, uint16_t ix, uint16_t len, 3858 void *bp, int size) 3859 { 3860 mblk_t *data; 3861 usb_ctrl_setup_t setup; 3862 usb_cr_t completion_reason; 3863 usb_cb_flags_t cb_flags; 3864 usb_flags_t flags; 3865 int i; 3866 int ret; 3867 3868 DPRINTF(4, (CE_CONT, "!%s: %s " 3869 "reqt:0x%02x req:0x%02x val:0x%04x ix:0x%04x len:0x%02x " 3870 "bp:0x%p nic_state:%d", 3871 dp->name, __func__, reqt, req, val, ix, len, bp, dp->nic_state)); 3872 3873 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 3874 return (USB_PIPE_ERROR); 3875 } 3876 3877 data = NULL; 3878 if (size > 0) { 3879 if ((data = allocb(size, 0)) == NULL) { 3880 return (USB_FAILURE); 3881 } 3882 3883 bcopy(bp, data->b_rptr, size); 3884 data->b_wptr = data->b_rptr + size; 3885 } 3886 3887 setup.bmRequestType = reqt; 3888 setup.bRequest = req; 3889 setup.wValue = val; 3890 setup.wIndex = ix; 3891 setup.wLength = len; 3892 setup.attrs = 0; /* attributes */ 3893 3894 for (i = usbgem_ctrl_retry; i > 0; i--) { 3895 completion_reason = 0; 3896 cb_flags = 0; 3897 3898 ret = usb_pipe_ctrl_xfer_wait(DEFAULT_PIPE(dp), 3899 &setup, &data, &completion_reason, &cb_flags, 0); 3900 3901 if (ret == USB_SUCCESS) { 3902 break; 3903 } 3904 if (i == 1) { 3905 cmn_err(CE_WARN, 3906 "!%s: %s failed: " 3907 "reqt:0x%x req:0x%x val:0x%x ix:0x%x len:0x%x " 3908 "ret:%d cr:%s(%d), cb_flags:0x%x %s", 3909 dp->name, __func__, reqt, req, val, ix, len, 3910 ret, usb_str_cr(completion_reason), 3911 completion_reason, 3912 cb_flags, 3913 (i > 1) ? "retrying..." : "fatal"); 3914 } 3915 } 3916 3917 if (data != NULL) { 3918 freemsg(data); 3919 } 3920 3921 return (ret); 3922 } 3923 3924 int 3925 usbgem_ctrl_in(struct usbgem_dev *dp, 3926 uint8_t reqt, uint8_t req, uint16_t val, uint16_t ix, uint16_t len, 3927 void *bp, int size) 3928 { 3929 mblk_t *data; 3930 usb_ctrl_setup_t setup; 3931 usb_cr_t completion_reason; 3932 usb_cb_flags_t cb_flags; 3933 int i; 3934 int ret; 3935 int reclen; 3936 3937 DPRINTF(4, (CE_CONT, 3938 "!%s: %s:" 3939 " reqt:0x%02x req:0x%02x val:0x%04x ix:0x%04x len:0x%02x" 3940 " bp:x%p mac_state:%d", 3941 dp->name, __func__, reqt, req, val, ix, len, bp, dp->mac_state)); 3942 3943 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 3944 return (USB_PIPE_ERROR); 3945 } 3946 3947 data = NULL; 3948 3949 setup.bmRequestType = reqt; 3950 setup.bRequest = req; 3951 setup.wValue = val; 3952 setup.wIndex = ix; 3953 setup.wLength = len; 3954 setup.attrs = USB_ATTRS_AUTOCLEARING; /* XXX */ 3955 3956 for (i = usbgem_ctrl_retry; i > 0; i--) { 3957 completion_reason = 0; 3958 cb_flags = 0; 3959 ret = usb_pipe_ctrl_xfer_wait(DEFAULT_PIPE(dp), &setup, &data, 3960 &completion_reason, &cb_flags, 0); 3961 3962 if (ret == USB_SUCCESS) { 3963 reclen = msgdsize(data); 3964 bcopy(data->b_rptr, bp, min(reclen, size)); 3965 break; 3966 } 3967 if (i == 1) { 3968 cmn_err(CE_WARN, 3969 "!%s: %s failed: " 3970 "reqt:0x%x req:0x%x val:0x%x ix:0x%x len:0x%x " 3971 "ret:%d cr:%s(%d) cb_flags:0x%x %s", 3972 dp->name, __func__, 3973 reqt, req, val, ix, len, 3974 ret, usb_str_cr(completion_reason), 3975 completion_reason, 3976 cb_flags, 3977 (i > 1) ? "retrying..." : "fatal"); 3978 } 3979 } 3980 3981 if (data) { 3982 freemsg(data); 3983 } 3984 3985 return (ret); 3986 } 3987 3988 int 3989 usbgem_ctrl_out_val(struct usbgem_dev *dp, 3990 uint8_t reqt, uint8_t req, uint16_t val, uint16_t ix, uint16_t len, 3991 uint32_t v) 3992 { 3993 uint8_t buf[4]; 3994 3995 /* convert to little endian from native byte order */ 3996 switch (len) { 3997 case 4: 3998 buf[3] = v >> 24; 3999 buf[2] = v >> 16; 4000 /* FALLTHROUGH */ 4001 case 2: 4002 buf[1] = v >> 8; 4003 /* FALLTHROUGH */ 4004 case 1: 4005 buf[0] = v; 4006 } 4007 4008 return (usbgem_ctrl_out(dp, reqt, req, val, ix, len, buf, len)); 4009 } 4010 4011 int 4012 usbgem_ctrl_in_val(struct usbgem_dev *dp, 4013 uint8_t reqt, uint8_t req, uint16_t val, uint16_t ix, uint16_t len, 4014 void *valp) 4015 { 4016 uint8_t buf[4]; 4017 uint_t v; 4018 int err; 4019 4020 #ifdef SANITY 4021 bzero(buf, sizeof (buf)); 4022 #endif 4023 err = usbgem_ctrl_in(dp, reqt, req, val, ix, len, buf, len); 4024 if (err == USB_SUCCESS) { 4025 v = 0; 4026 switch (len) { 4027 case 4: 4028 v |= buf[3] << 24; 4029 v |= buf[2] << 16; 4030 /* FALLTHROUGH */ 4031 case 2: 4032 v |= buf[1] << 8; 4033 /* FALLTHROUGH */ 4034 case 1: 4035 v |= buf[0]; 4036 } 4037 4038 switch (len) { 4039 case 4: 4040 *(uint32_t *)valp = v; 4041 break; 4042 case 2: 4043 *(uint16_t *)valp = v; 4044 break; 4045 case 1: 4046 *(uint8_t *)valp = v; 4047 break; 4048 } 4049 } 4050 return (err); 4051 } 4052 4053 /* 4054 * Attach / detach / disconnect / reconnect management 4055 */ 4056 static int 4057 usbgem_open_pipes(struct usbgem_dev *dp) 4058 { 4059 int i; 4060 int ret; 4061 int ifnum; 4062 int alt; 4063 usb_client_dev_data_t *reg_data; 4064 usb_ep_data_t *ep_tree_node; 4065 4066 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 4067 4068 ifnum = dp->ugc.usbgc_ifnum; 4069 alt = dp->ugc.usbgc_alt; 4070 4071 ep_tree_node = usb_lookup_ep_data(dp->dip, dp->reg_data, ifnum, alt, 4072 0, USB_EP_ATTR_BULK, USB_EP_DIR_IN); 4073 if (ep_tree_node == NULL) { 4074 cmn_err(CE_WARN, "!%s: %s: ep_bulkin is NULL", 4075 dp->name, __func__); 4076 goto err; 4077 } 4078 dp->ep_bulkin = &ep_tree_node->ep_descr; 4079 4080 ep_tree_node = usb_lookup_ep_data(dp->dip, dp->reg_data, ifnum, alt, 4081 0, USB_EP_ATTR_BULK, USB_EP_DIR_OUT); 4082 if (ep_tree_node == NULL) { 4083 cmn_err(CE_WARN, "!%s: %s: ep_bulkout is NULL", 4084 dp->name, __func__); 4085 goto err; 4086 } 4087 dp->ep_bulkout = &ep_tree_node->ep_descr; 4088 4089 ep_tree_node = usb_lookup_ep_data(dp->dip, dp->reg_data, ifnum, alt, 4090 0, USB_EP_ATTR_INTR, USB_EP_DIR_IN); 4091 if (ep_tree_node) { 4092 dp->ep_intr = &ep_tree_node->ep_descr; 4093 } else { 4094 /* don't care */ 4095 DPRINTF(1, (CE_CONT, "!%s: %s: ep_intr is NULL", 4096 dp->name, __func__)); 4097 dp->ep_intr = NULL; 4098 } 4099 4100 /* XXX -- no need to open default pipe */ 4101 4102 /* open bulk out pipe */ 4103 bzero(&dp->policy_bulkout, sizeof (usb_pipe_policy_t)); 4104 dp->policy_bulkout.pp_max_async_reqs = 1; 4105 4106 if ((ret = usb_pipe_open(dp->dip, 4107 dp->ep_bulkout, &dp->policy_bulkout, USB_FLAGS_SLEEP, 4108 &dp->bulkout_pipe)) != USB_SUCCESS) { 4109 cmn_err(CE_WARN, 4110 "!%s: %s: err:%x: failed to open bulk-out pipe", 4111 dp->name, __func__, ret); 4112 dp->bulkout_pipe = NULL; 4113 goto err; 4114 } 4115 DPRINTF(1, (CE_CONT, "!%s: %s: bulkout_pipe opened successfully", 4116 dp->name, __func__)); 4117 4118 /* open bulk in pipe */ 4119 bzero(&dp->policy_bulkin, sizeof (usb_pipe_policy_t)); 4120 dp->policy_bulkin.pp_max_async_reqs = 1; 4121 if ((ret = usb_pipe_open(dp->dip, 4122 dp->ep_bulkin, &dp->policy_bulkin, USB_FLAGS_SLEEP, 4123 &dp->bulkin_pipe)) != USB_SUCCESS) { 4124 cmn_err(CE_WARN, 4125 "!%s: %s: ret:%x failed to open bulk-in pipe", 4126 dp->name, __func__, ret); 4127 dp->bulkin_pipe = NULL; 4128 goto err; 4129 } 4130 DPRINTF(1, (CE_CONT, "!%s: %s: bulkin_pipe opened successfully", 4131 dp->name, __func__)); 4132 4133 if (dp->ep_intr) { 4134 /* open interrupt pipe */ 4135 bzero(&dp->policy_interrupt, sizeof (usb_pipe_policy_t)); 4136 dp->policy_interrupt.pp_max_async_reqs = 1; 4137 if ((ret = usb_pipe_open(dp->dip, dp->ep_intr, 4138 &dp->policy_interrupt, USB_FLAGS_SLEEP, 4139 &dp->intr_pipe)) != USB_SUCCESS) { 4140 cmn_err(CE_WARN, 4141 "!%s: %s: ret:%x failed to open interrupt pipe", 4142 dp->name, __func__, ret); 4143 dp->intr_pipe = NULL; 4144 goto err; 4145 } 4146 } 4147 DPRINTF(1, (CE_CONT, "!%s: %s: intr_pipe opened successfully", 4148 dp->name, __func__)); 4149 4150 return (USB_SUCCESS); 4151 4152 err: 4153 if (dp->bulkin_pipe) { 4154 usb_pipe_close(dp->dip, 4155 dp->bulkin_pipe, USB_FLAGS_SLEEP, NULL, 0); 4156 dp->bulkin_pipe = NULL; 4157 } 4158 if (dp->bulkout_pipe) { 4159 usb_pipe_close(dp->dip, 4160 dp->bulkout_pipe, USB_FLAGS_SLEEP, NULL, 0); 4161 dp->bulkout_pipe = NULL; 4162 } 4163 if (dp->intr_pipe) { 4164 usb_pipe_close(dp->dip, 4165 dp->intr_pipe, USB_FLAGS_SLEEP, NULL, 0); 4166 dp->intr_pipe = NULL; 4167 } 4168 4169 return (USB_FAILURE); 4170 } 4171 4172 static int 4173 usbgem_close_pipes(struct usbgem_dev *dp) 4174 { 4175 DPRINTF(0, (CE_CONT, "!%s: %s: called", dp->name, __func__)); 4176 4177 if (dp->intr_pipe) { 4178 usb_pipe_close(dp->dip, 4179 dp->intr_pipe, USB_FLAGS_SLEEP, NULL, 0); 4180 dp->intr_pipe = NULL; 4181 } 4182 DPRINTF(1, (CE_CONT, "!%s: %s: 1", dp->name, __func__)); 4183 4184 ASSERT(dp->bulkin_pipe); 4185 usb_pipe_close(dp->dip, dp->bulkin_pipe, USB_FLAGS_SLEEP, NULL, 0); 4186 dp->bulkin_pipe = NULL; 4187 DPRINTF(1, (CE_CONT, "!%s: %s: 2", dp->name, __func__)); 4188 4189 ASSERT(dp->bulkout_pipe); 4190 usb_pipe_close(dp->dip, dp->bulkout_pipe, USB_FLAGS_SLEEP, NULL, 0); 4191 dp->bulkout_pipe = NULL; 4192 DPRINTF(1, (CE_CONT, "!%s: %s: 3", dp->name, __func__)); 4193 4194 return (USB_SUCCESS); 4195 } 4196 4197 #define FREEZE_GRACEFUL (B_TRUE) 4198 #define FREEZE_NO_GRACEFUL (B_FALSE) 4199 static int 4200 usbgem_freeze_device(struct usbgem_dev *dp, boolean_t graceful) 4201 { 4202 DPRINTF(0, (CE_NOTE, "!%s: %s: called", dp->name, __func__)); 4203 4204 /* stop nic activity */ 4205 (void) usbgem_mac_stop(dp, MAC_STATE_DISCONNECTED, graceful); 4206 4207 /* 4208 * Here we free all memory resource allocated, because it will 4209 * cause to panic the system that we free usb_bulk_req objects 4210 * during the usb device is disconnected. 4211 */ 4212 (void) usbgem_free_memory(dp); 4213 4214 return (USB_SUCCESS); 4215 } 4216 4217 static int 4218 usbgem_disconnect_cb(dev_info_t *dip) 4219 { 4220 int ret; 4221 struct usbgem_dev *dp; 4222 4223 dp = USBGEM_GET_DEV(dip); 4224 4225 cmn_err(CE_NOTE, "!%s: the usb device was disconnected (dp=%p)", 4226 dp->name, (void *)dp); 4227 4228 /* start serialize */ 4229 rw_enter(&dp->dev_state_lock, RW_WRITER); 4230 4231 ret = usbgem_freeze_device(dp, 0); 4232 4233 /* end of serialize */ 4234 rw_exit(&dp->dev_state_lock); 4235 4236 return (ret); 4237 } 4238 4239 static int 4240 usbgem_recover_device(struct usbgem_dev *dp) 4241 { 4242 int err; 4243 4244 DPRINTF(0, (CE_NOTE, "!%s: %s: called", dp->name, __func__)); 4245 4246 err = USB_SUCCESS; 4247 4248 /* reinitialize the usb connection */ 4249 usbgem_close_pipes(dp); 4250 if ((err = usbgem_open_pipes(dp)) != USB_SUCCESS) { 4251 goto x; 4252 } 4253 4254 /* initialize nic state */ 4255 dp->mac_state = MAC_STATE_STOPPED; 4256 dp->mii_state = MII_STATE_UNKNOWN; 4257 4258 /* allocate memory resources again */ 4259 if ((err = usbgem_alloc_memory(dp)) != USB_SUCCESS) { 4260 goto x; 4261 } 4262 4263 /* restart nic and recover state */ 4264 (void) usbgem_restart_nic(dp); 4265 4266 usbgem_mii_init(dp); 4267 4268 /* kick potentially stopped house keeping thread */ 4269 cv_signal(&dp->link_watcher_wait_cv); 4270 x: 4271 return (err); 4272 } 4273 4274 static int 4275 usbgem_reconnect_cb(dev_info_t *dip) 4276 { 4277 int err = USB_SUCCESS; 4278 struct usbgem_dev *dp; 4279 4280 dp = USBGEM_GET_DEV(dip); 4281 DPRINTF(0, (CE_CONT, "!%s: dp=%p", ddi_get_name(dip), dp)); 4282 #ifdef notdef 4283 /* check device changes after disconnect */ 4284 if (usb_check_same_device(dp->dip, NULL, USB_LOG_L2, -1, 4285 USB_CHK_BASIC | USB_CHK_CFG, NULL) != USB_SUCCESS) { 4286 cmn_err(CE_CONT, 4287 "!%s: no or different device installed", dp->name); 4288 return (DDI_SUCCESS); 4289 } 4290 #endif 4291 cmn_err(CE_NOTE, "%s: the usb device was reconnected", dp->name); 4292 4293 /* start serialize */ 4294 rw_enter(&dp->dev_state_lock, RW_WRITER); 4295 4296 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 4297 err = usbgem_recover_device(dp); 4298 } 4299 4300 /* end of serialize */ 4301 rw_exit(&dp->dev_state_lock); 4302 4303 return (err == USB_SUCCESS ? DDI_SUCCESS : DDI_FAILURE); 4304 } 4305 4306 int 4307 usbgem_suspend(dev_info_t *dip) 4308 { 4309 int err = USB_SUCCESS; 4310 struct usbgem_dev *dp; 4311 4312 dp = USBGEM_GET_DEV(dip); 4313 4314 DPRINTF(0, (CE_CONT, "!%s: %s: callded", dp->name, __func__)); 4315 4316 /* start serialize */ 4317 rw_enter(&dp->dev_state_lock, RW_WRITER); 4318 4319 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 4320 err = usbgem_freeze_device(dp, STOP_GRACEFUL); 4321 } 4322 4323 /* end of serialize */ 4324 rw_exit(&dp->dev_state_lock); 4325 4326 return (err == USB_SUCCESS ? DDI_SUCCESS : DDI_FAILURE); 4327 } 4328 4329 int 4330 usbgem_resume(dev_info_t *dip) 4331 { 4332 int err = USB_SUCCESS; 4333 struct usbgem_dev *dp; 4334 4335 dp = USBGEM_GET_DEV(dip); 4336 4337 DPRINTF(0, (CE_CONT, "!%s: %s: callded", dp->name, __func__)); 4338 #ifdef notdef 4339 /* check device changes after disconnect */ 4340 if (usb_check_same_device(dp->dip, NULL, USB_LOG_L2, -1, 4341 USB_CHK_BASIC | USB_CHK_CFG, NULL) != USB_SUCCESS) { 4342 cmn_err(CE_CONT, 4343 "!%s: no or different device installed", dp->name); 4344 return (DDI_SUCCESS); 4345 } 4346 #endif 4347 /* start serialize */ 4348 rw_enter(&dp->dev_state_lock, RW_WRITER); 4349 4350 if (dp->mac_state == MAC_STATE_DISCONNECTED) { 4351 err = usbgem_recover_device(dp); 4352 } 4353 4354 /* end of serialize */ 4355 rw_exit(&dp->dev_state_lock); 4356 4357 return (err == USB_SUCCESS ? DDI_SUCCESS : DDI_FAILURE); 4358 } 4359 4360 #define USBGEM_LOCAL_DATA_SIZE(gc) \ 4361 (sizeof (struct usbgem_dev) + USBGEM_MCALLOC) 4362 4363 struct usbgem_dev * 4364 usbgem_do_attach(dev_info_t *dip, 4365 struct usbgem_conf *gc, void *lp, int lmsize) 4366 { 4367 struct usbgem_dev *dp; 4368 int i; 4369 mac_register_t *macp = NULL; 4370 int ret; 4371 int unit; 4372 int err; 4373 4374 unit = ddi_get_instance(dip); 4375 4376 DPRINTF(2, (CE_CONT, "!usbgem%d: %s: called", unit, __func__)); 4377 4378 if ((macp = mac_alloc(MAC_VERSION)) == NULL) { 4379 cmn_err(CE_WARN, "!gem%d: %s: mac_alloc failed", 4380 unit, __func__); 4381 return (NULL); 4382 } 4383 /* 4384 * Allocate soft data structure 4385 */ 4386 dp = kmem_zalloc(USBGEM_LOCAL_DATA_SIZE(gc), KM_SLEEP); 4387 4388 /* link to private area */ 4389 dp->private = lp; 4390 dp->priv_size = lmsize; 4391 dp->mc_list = (struct mcast_addr *)&dp[1]; 4392 4393 dp->dip = dip; 4394 bcopy(gc->usbgc_name, dp->name, USBGEM_NAME_LEN); 4395 4396 /* 4397 * register with usb service 4398 */ 4399 if (usb_client_attach(dip, USBDRV_VERSION, 0) != USB_SUCCESS) { 4400 cmn_err(CE_WARN, 4401 "%s: %s: usb_client_attach failed", 4402 dp->name, __func__); 4403 goto err_free_private; 4404 } 4405 4406 if (usb_get_dev_data(dip, &dp->reg_data, 4407 USB_PARSE_LVL_ALL, 0) != USB_SUCCESS) { 4408 dp->reg_data = NULL; 4409 goto err_unregister_client; 4410 } 4411 #ifdef USBGEM_DEBUG_LEVEL 4412 usb_print_descr_tree(dp->dip, dp->reg_data); 4413 #endif 4414 4415 if (usbgem_open_pipes(dp) != USB_SUCCESS) { 4416 /* failed to open pipes */ 4417 cmn_err(CE_WARN, "!%s: %s: failed to open pipes", 4418 dp->name, __func__); 4419 goto err_unregister_client; 4420 } 4421 4422 /* 4423 * Initialize mutexs and condition variables 4424 */ 4425 mutex_init(&dp->rxlock, NULL, MUTEX_DRIVER, NULL); 4426 mutex_init(&dp->txlock, NULL, MUTEX_DRIVER, NULL); 4427 cv_init(&dp->rx_drain_cv, NULL, CV_DRIVER, NULL); 4428 cv_init(&dp->tx_drain_cv, NULL, CV_DRIVER, NULL); 4429 rw_init(&dp->dev_state_lock, NULL, RW_DRIVER, NULL); 4430 mutex_init(&dp->link_watcher_lock, NULL, MUTEX_DRIVER, NULL); 4431 cv_init(&dp->link_watcher_wait_cv, NULL, CV_DRIVER, NULL); 4432 sema_init(&dp->hal_op_lock, 1, NULL, SEMA_DRIVER, NULL); 4433 sema_init(&dp->rxfilter_lock, 1, NULL, SEMA_DRIVER, NULL); 4434 4435 /* 4436 * Initialize configuration 4437 */ 4438 dp->ugc = *gc; 4439 4440 dp->mtu = ETHERMTU; 4441 dp->rxmode = 0; 4442 dp->speed = USBGEM_SPD_10; /* default is 10Mbps */ 4443 dp->full_duplex = B_FALSE; /* default is half */ 4444 dp->flow_control = FLOW_CONTROL_NONE; 4445 4446 dp->nic_state = NIC_STATE_STOPPED; 4447 dp->mac_state = MAC_STATE_STOPPED; 4448 dp->mii_state = MII_STATE_UNKNOWN; 4449 4450 /* performance tuning parameters */ 4451 dp->txthr = ETHERMAX; /* tx fifo threshoold */ 4452 dp->txmaxdma = 16*4; /* tx max dma burst size */ 4453 dp->rxthr = 128; /* rx fifo threshoold */ 4454 dp->rxmaxdma = 16*4; /* rx max dma burst size */ 4455 4456 /* 4457 * Get media mode infomation from .conf file 4458 */ 4459 usbgem_read_conf(dp); 4460 4461 /* rx_buf_len depend on MTU */ 4462 dp->rx_buf_len = MAXPKTBUF(dp) + dp->ugc.usbgc_rx_header_len; 4463 4464 /* 4465 * Reset the chip 4466 */ 4467 if (usbgem_hal_reset_chip(dp) != USB_SUCCESS) { 4468 cmn_err(CE_WARN, 4469 "!%s: %s: failed to reset the usb device", 4470 dp->name, __func__); 4471 goto err_destroy_locks; 4472 } 4473 4474 /* 4475 * HW dependant paremeter initialization 4476 */ 4477 if (usbgem_hal_attach_chip(dp) != USB_SUCCESS) { 4478 cmn_err(CE_WARN, 4479 "!%s: %s: failed to attach the usb device", 4480 dp->name, __func__); 4481 goto err_destroy_locks; 4482 } 4483 4484 /* allocate resources */ 4485 if (usbgem_alloc_memory(dp) != USB_SUCCESS) { 4486 goto err_destroy_locks; 4487 } 4488 4489 DPRINTF(0, (CE_CONT, 4490 "!%s: %02x:%02x:%02x:%02x:%02x:%02x", 4491 dp->name, 4492 dp->dev_addr.ether_addr_octet[0], 4493 dp->dev_addr.ether_addr_octet[1], 4494 dp->dev_addr.ether_addr_octet[2], 4495 dp->dev_addr.ether_addr_octet[3], 4496 dp->dev_addr.ether_addr_octet[4], 4497 dp->dev_addr.ether_addr_octet[5])); 4498 4499 /* copy mac address */ 4500 dp->cur_addr = dp->dev_addr; 4501 4502 /* pre-calculated tx timeout in second for performance */ 4503 dp->bulkout_timeout = 4504 dp->ugc.usbgc_tx_timeout / drv_usectohz(1000*1000); 4505 4506 usbgem_gld3_init(dp, macp); 4507 4508 /* Probe MII phy (scan phy) */ 4509 dp->mii_lpable = 0; 4510 dp->mii_advert = 0; 4511 dp->mii_exp = 0; 4512 dp->mii_ctl1000 = 0; 4513 dp->mii_stat1000 = 0; 4514 4515 dp->mii_status_ro = 0; 4516 dp->mii_xstatus_ro = 0; 4517 4518 if (usbgem_mii_probe(dp) != USB_SUCCESS) { 4519 cmn_err(CE_WARN, "!%s: %s: mii_probe failed", 4520 dp->name, __func__); 4521 goto err_free_memory; 4522 } 4523 4524 /* mask unsupported abilities */ 4525 dp->anadv_autoneg &= BOOLEAN(dp->mii_status & MII_STATUS_CANAUTONEG); 4526 dp->anadv_1000fdx &= 4527 BOOLEAN(dp->mii_xstatus & 4528 (MII_XSTATUS_1000BASEX_FD | MII_XSTATUS_1000BASET_FD)); 4529 dp->anadv_1000hdx &= 4530 BOOLEAN(dp->mii_xstatus & 4531 (MII_XSTATUS_1000BASEX | MII_XSTATUS_1000BASET)); 4532 dp->anadv_100t4 &= BOOLEAN(dp->mii_status & MII_STATUS_100_BASE_T4); 4533 dp->anadv_100fdx &= BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX_FD); 4534 dp->anadv_100hdx &= BOOLEAN(dp->mii_status & MII_STATUS_100_BASEX); 4535 dp->anadv_10fdx &= BOOLEAN(dp->mii_status & MII_STATUS_10_FD); 4536 dp->anadv_10hdx &= BOOLEAN(dp->mii_status & MII_STATUS_10); 4537 4538 if (usbgem_mii_init(dp) != USB_SUCCESS) { 4539 cmn_err(CE_WARN, "!%s: %s: mii_init failed", 4540 dp->name, __func__); 4541 goto err_free_memory; 4542 } 4543 4544 /* 4545 * Add interrupt to system. 4546 */ 4547 if (ret = mac_register(macp, &dp->mh)) { 4548 cmn_err(CE_WARN, "!%s: mac_register failed, error:%d", 4549 dp->name, ret); 4550 goto err_release_stats; 4551 } 4552 mac_free(macp); 4553 macp = NULL; 4554 4555 if (usb_register_hotplug_cbs(dip, 4556 usbgem_suspend, usbgem_resume) != USB_SUCCESS) { 4557 cmn_err(CE_WARN, 4558 "!%s: %s: failed to register hotplug cbs", 4559 dp->name, __func__); 4560 goto err_unregister_gld; 4561 } 4562 4563 /* reset mii and start mii link watcher */ 4564 if (usbgem_mii_start(dp) != USB_SUCCESS) { 4565 goto err_unregister_hotplug; 4566 } 4567 4568 /* start tx watchdow watcher */ 4569 if (usbgem_tx_watcher_start(dp)) { 4570 goto err_usbgem_mii_stop; 4571 } 4572 4573 ddi_set_driver_private(dip, (caddr_t)dp); 4574 4575 DPRINTF(2, (CE_CONT, "!%s: %s: return: success", dp->name, __func__)); 4576 4577 return (dp); 4578 4579 err_usbgem_mii_stop: 4580 usbgem_mii_stop(dp); 4581 4582 err_unregister_hotplug: 4583 usb_unregister_hotplug_cbs(dip); 4584 4585 err_unregister_gld: 4586 mac_unregister(dp->mh); 4587 4588 err_release_stats: 4589 err_free_memory: 4590 usbgem_free_memory(dp); 4591 4592 err_destroy_locks: 4593 cv_destroy(&dp->tx_drain_cv); 4594 cv_destroy(&dp->rx_drain_cv); 4595 mutex_destroy(&dp->txlock); 4596 mutex_destroy(&dp->rxlock); 4597 rw_destroy(&dp->dev_state_lock); 4598 mutex_destroy(&dp->link_watcher_lock); 4599 cv_destroy(&dp->link_watcher_wait_cv); 4600 sema_destroy(&dp->hal_op_lock); 4601 sema_destroy(&dp->rxfilter_lock); 4602 4603 err_close_pipes: 4604 (void) usbgem_close_pipes(dp); 4605 4606 err_unregister_client: 4607 usb_client_detach(dp->dip, dp->reg_data); 4608 4609 err_free_private: 4610 if (macp) { 4611 mac_free(macp); 4612 } 4613 4614 kmem_free((caddr_t)dp, USBGEM_LOCAL_DATA_SIZE(gc)); 4615 4616 return (NULL); 4617 } 4618 4619 int 4620 usbgem_do_detach(dev_info_t *dip) 4621 { 4622 struct usbgem_dev *dp; 4623 4624 dp = USBGEM_GET_DEV(dip); 4625 4626 /* unregister with gld v3 */ 4627 if (mac_unregister(dp->mh) != DDI_SUCCESS) { 4628 return (DDI_FAILURE); 4629 } 4630 4631 /* unregister with hotplug service */ 4632 usb_unregister_hotplug_cbs(dip); 4633 4634 /* stop tx watchdog watcher */ 4635 usbgem_tx_watcher_stop(dp); 4636 4637 /* stop the link manager */ 4638 usbgem_mii_stop(dp); 4639 4640 /* unregister with usb service */ 4641 (void) usbgem_free_memory(dp); 4642 (void) usbgem_close_pipes(dp); 4643 usb_client_detach(dp->dip, dp->reg_data); 4644 dp->reg_data = NULL; 4645 4646 /* release locks and condition variables */ 4647 mutex_destroy(&dp->txlock); 4648 mutex_destroy(&dp->rxlock); 4649 cv_destroy(&dp->tx_drain_cv); 4650 cv_destroy(&dp->rx_drain_cv); 4651 rw_destroy(&dp->dev_state_lock); 4652 mutex_destroy(&dp->link_watcher_lock); 4653 cv_destroy(&dp->link_watcher_wait_cv); 4654 sema_destroy(&dp->hal_op_lock); 4655 sema_destroy(&dp->rxfilter_lock); 4656 4657 /* release basic memory resources */ 4658 kmem_free((caddr_t)(dp->private), dp->priv_size); 4659 kmem_free((caddr_t)dp, USBGEM_LOCAL_DATA_SIZE(&dp->ugc)); 4660 4661 DPRINTF(2, (CE_CONT, "!%s: %s: return: success", 4662 ddi_driver_name(dip), __func__)); 4663 4664 return (DDI_SUCCESS); 4665 } 4666 4667 int 4668 usbgem_mod_init(struct dev_ops *dop, char *name) 4669 { 4670 major_t major; 4671 major = ddi_name_to_major(name); 4672 if (major == DDI_MAJOR_T_NONE) { 4673 return (DDI_FAILURE); 4674 } 4675 mac_init_ops(dop, name); 4676 4677 return (DDI_SUCCESS); 4678 } 4679 4680 void 4681 usbgem_mod_fini(struct dev_ops *dop) 4682 { 4683 mac_fini_ops(dop); 4684 } 4685 4686 int 4687 usbgem_quiesce(dev_info_t *dip) 4688 { 4689 struct usbgem_dev *dp; 4690 4691 dp = USBGEM_GET_DEV(dip); 4692 4693 ASSERT(dp != NULL); 4694 4695 if (dp->mac_state != MAC_STATE_DISCONNECTED && 4696 dp->mac_state != MAC_STATE_STOPPED) { 4697 if (usbgem_hal_stop_chip(dp) != USB_SUCCESS) { 4698 (void) usbgem_hal_reset_chip(dp); 4699 } 4700 } 4701 4702 /* devo_quiesce() must return DDI_SUCCESS always */ 4703 return (DDI_SUCCESS); 4704 } 4705