| /* |
| * CDDL HEADER START |
| * |
| * The contents of this file are subject to the terms of the |
| * Common Development and Distribution License (the "License"). |
| * You may not use this file except in compliance with the License. |
| * |
| * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE |
| * or http://www.opensolaris.org/os/licensing. |
| * See the License for the specific language governing permissions |
| * and limitations under the License. |
| * |
| * When distributing Covered Code, include this CDDL HEADER in each |
| * file and include the License file at usr/src/OPENSOLARIS.LICENSE. |
| * If applicable, add the following below this CDDL HEADER, with the |
| * fields enclosed by brackets "[]" replaced with your own identifying |
| * information: Portions Copyright [yyyy] [name of copyright owner] |
| * |
| * CDDL HEADER END |
| */ |
| /* |
| * Copyright 2010 Sun Microsystems, Inc. All rights reserved. |
| * Use is subject to license terms. |
| */ |
| |
| /* |
| * MII overrides for other PHYs. |
| */ |
| |
| #include <sys/types.h> |
| #include <sys/ddi.h> |
| #include <sys/sunddi.h> |
| #include <sys/mii.h> |
| #include <sys/miiregs.h> |
| #include "miipriv.h" |
| |
| #define OUI(MFG, VEND) { MII_OUI_##MFG, VEND } |
| |
| static const struct { |
| uint32_t oui; |
| const char *vendor; |
| } other_vendors[] = { |
| OUI(ALTIMA, "Altima Communications"), |
| OUI(AMD, "Advanced Micro Devices"), |
| OUI(AMD_2, "Advanced Micro Devices"), |
| OUI(ATTANSIC, "Atheros/Attansic"), |
| OUI(BROADCOM, "Broadcom Corporation"), |
| OUI(BROADCOM_2, "Broadcom Corporation"), |
| OUI(CICADA, "Cicada Semiconductor"), |
| OUI(CICADA_2, "Cicada Semiconductor"), |
| OUI(DAVICOM, "Davicom Semiconductor"), |
| OUI(DAVICOM_2, "Davicom Semiconductor"), |
| OUI(ICPLUS, "IC Plus Corp."), |
| OUI(ICS, "Integrated Circuit Systems"), |
| OUI(LUCENT, "Lucent Technologies"), |
| OUI(INTEL, "Intel"), |
| OUI(MARVELL, "Marvell Technology"), |
| OUI(NATIONAL_SEMI, "National Semiconductor"), |
| OUI(NATIONAL_SEMI_2, "National Semiconductor"), |
| OUI(QUALITY_SEMI, "Quality Semiconductor"), |
| OUI(QUALITY_SEMI_2, "Quality Semiconductor"), |
| { 0, NULL } |
| }; |
| |
| #define ID(MFG, MODEL, DESC) \ |
| { MII_OUI_##MFG, MII_MODEL_##MFG##_##MODEL, DESC } |
| #define IDN(MFG, N, MODEL, DESC) \ |
| { MII_OUI_##MFG##_##N, MII_MODEL_##MFG##_##MODEL, DESC } |
| static const struct { |
| uint32_t oui; |
| uint32_t model; |
| const char *desc; |
| } other_phys[] = { |
| |
| /* |
| * Altima phys are standard compliant. |
| * AMD Am79C874 and Am79C875 phys are work-alikes. |
| */ |
| ID(ALTIMA, AC101, "AC101/Am79C874"), |
| ID(ALTIMA, AC101L, "AC101L"), |
| ID(ALTIMA, AM79C875, "Am79C875"), |
| |
| /* |
| * AMD phys are pretty much standard. |
| */ |
| ID(AMD, AM79C901, "Am79C901"), |
| ID(AMD, AM79C972, "Am79C792"), |
| ID(AMD, AM79C973, "Am79C793"), |
| IDN(AMD, 2, AM79C901, "Am79C901"), |
| IDN(AMD, 2, AM79C972, "Am79C792"), |
| IDN(AMD, 2, AM79C973, "Am79C793"), |
| |
| /* |
| * Davicom phys are standard compliant. |
| */ |
| ID(DAVICOM, DM9101, "DM9101"), |
| ID(DAVICOM, DM9102, "DM9102"), |
| ID(DAVICOM, DM9161, "DM9161"), |
| IDN(DAVICOM, 2, DM9101, "DM9101"), |
| IDN(DAVICOM, 2, DM9102, "DM9102"), |
| |
| /* |
| * IC Plus phy is standard compliant. |
| */ |
| ID(ICPLUS, IP101, "IP101"), |
| |
| /* |
| * ICS phys need double read (bits are latched), have some |
| * faster polling support, and support for automatic power down. |
| * The framework deals with the first, we don't need the second, |
| * and the third is set to default anyway, so we don't need to |
| * use any special handling. |
| */ |
| ID(ICS, ICS1889, "ICS1889"), |
| ID(ICS, ICS1890, "ICS1890"), |
| ID(ICS, ICS1892, "ICS1892"), |
| ID(ICS, ICS1893, "ICS1893"), |
| |
| ID(LUCENT, LU6612, "LU6612"), |
| |
| { 0, 0, NULL }, |
| }; |
| |
| boolean_t |
| phy_other_probe(phy_handle_t *ph) |
| { |
| uint32_t vid = MII_PHY_MFG(ph->phy_id); |
| uint32_t pid = MII_PHY_MODEL(ph->phy_id); |
| |
| if ((ph->phy_id == 0) || (ph->phy_id == 0xffffffffU)) { |
| /* |
| * IDs are technically optional, but all discrete PHYs |
| * should have them. |
| */ |
| ph->phy_vendor = "Internal"; |
| ph->phy_model = "PHY"; |
| } |
| for (int i = 0; other_vendors[i].vendor; i++) { |
| if (vid == other_vendors[i].oui) { |
| ph->phy_vendor = other_vendors[i].vendor; |
| |
| for (int j = 0; other_phys[j].desc; j++) { |
| if (vid == other_phys[j].oui && |
| pid == other_phys[j].model) { |
| ph->phy_model = other_phys[j].desc; |
| return (B_TRUE); |
| } |
| } |
| |
| /* PHY from this vendor isn't known to us */ |
| return (B_FALSE); |
| } |
| } |
| return (B_FALSE); |
| } |