VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DevE1000Phy.cpp@ 98171

Last change on this file since 98171 was 98171, checked in by vboxsync, 20 months ago

Devices/E1000: Made the two testcases build again. bugref:10348

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.5 KB
Line 
1/** $Id: DevE1000Phy.cpp 98171 2023-01-20 20:48:14Z vboxsync $ */
2/** @file
3 * DevE1000Phy - Intel 82540EM Ethernet Controller Internal PHY Emulation.
4 *
5 * Implemented in accordance with the specification:
6 * PCI/PCI-X Family of Gigabit Ethernet Controllers Software Developer's
7 * Manual 82540EP/EM, 82541xx, 82544GC/EI, 82545GM/EM, 82546GB/EB, and
8 * 82547xx
9 *
10 * 317453-002 Revision 3.5
11 */
12
13/*
14 * Copyright (C) 2007-2023 Oracle and/or its affiliates.
15 *
16 * This file is part of VirtualBox base platform packages, as
17 * available from https://www.virtualbox.org.
18 *
19 * This program is free software; you can redistribute it and/or
20 * modify it under the terms of the GNU General Public License
21 * as published by the Free Software Foundation, in version 3 of the
22 * License.
23 *
24 * This program is distributed in the hope that it will be useful, but
25 * WITHOUT ANY WARRANTY; without even the implied warranty of
26 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
27 * General Public License for more details.
28 *
29 * You should have received a copy of the GNU General Public License
30 * along with this program; if not, see <https://www.gnu.org/licenses>.
31 *
32 * SPDX-License-Identifier: GPL-3.0-only
33 */
34
35#define LOG_GROUP LOG_GROUP_DEV_E1000
36
37/** @todo Remove me! For now I want asserts to work in release code. */
38// #ifndef RT_STRICT
39// #define RT_STRICT
40#include <iprt/assert.h>
41// #undef RT_STRICT
42// #endif
43
44#include <iprt/errcore.h>
45#include <VBox/log.h>
46#ifdef IN_RING3
47# include <VBox/vmm/pdmdev.h>
48#endif
49#include "DevE1000Phy.h"
50
51/* Little helpers ************************************************************/
52#ifdef PHY_UNIT_TEST
53# include <stdio.h>
54# define PhyLog(a) printf a
55#else /* PHY_UNIT_TEST */
56# define PhyLog(a) Log(a)
57#endif /* PHY_UNIT_TEST */
58
59#define REG(x) pPhy->au16Regs[x##_IDX]
60
61
62/* Internals */
63namespace Phy {
64#if defined(LOG_ENABLED) || defined(PHY_UNIT_TEST)
65 /** Retrieves state name by id */
66 static const char * getStateName(uint16_t u16State);
67#endif
68 /** Look up register index by address. */
69 static int lookupRegister(uint32_t u32Address);
70 /** Software-triggered reset. */
71 static void softReset(PPHY pPhy, PPDMDEVINS pDevIns);
72
73 /** Read callback. */
74 typedef uint16_t FNREAD(PPHY pPhy, uint32_t index, PPDMDEVINS pDevIns);
75 /** Write callback. */
76 typedef void FNWRITE(PPHY pPhy, uint32_t index, uint16_t u16Value, PPDMDEVINS pDevIns);
77
78 /** @name Generic handlers
79 * @{ */
80 static FNREAD regReadDefault;
81 static FNWRITE regWriteDefault;
82 static FNREAD regReadForbidden;
83 static FNWRITE regWriteForbidden;
84 static FNREAD regReadUnimplemented;
85 static FNWRITE regWriteUnimplemented;
86 /** @} */
87 /** @name Register-specific handlers
88 * @{ */
89 static FNWRITE regWritePCTRL;
90 static FNREAD regReadPSTATUS;
91 static FNREAD regReadGSTATUS;
92 /** @} */
93
94 /**
95 * PHY register map table.
96 *
97 * Override pfnRead and pfnWrite to implement register-specific behavior.
98 */
99 static struct RegMap_st
100 {
101 /** PHY register address. */
102 uint32_t u32Address;
103 /** Read callback. */
104 FNREAD *pfnRead;
105 /** Write callback. */
106 FNWRITE *pfnWrite;
107 /** Abbreviated name. */
108 const char *pszAbbrev;
109 /** Full name. */
110 const char *pszName;
111 } s_regMap[NUM_OF_PHY_REGS] =
112 {
113 /*ra read callback write callback abbrev full name */
114 /*-- ------------------------- -------------------------- ---------- ------------------------------*/
115 { 0, Phy::regReadDefault , Phy::regWritePCTRL , "PCTRL" , "PHY Control" },
116 { 1, Phy::regReadPSTATUS , Phy::regWriteForbidden , "PSTATUS" , "PHY Status" },
117 { 2, Phy::regReadDefault , Phy::regWriteForbidden , "PID" , "PHY Identifier" },
118 { 3, Phy::regReadDefault , Phy::regWriteForbidden , "EPID" , "Extended PHY Identifier" },
119 { 4, Phy::regReadDefault , Phy::regWriteDefault , "ANA" , "Auto-Negotiation Advertisement" },
120 { 5, Phy::regReadDefault , Phy::regWriteForbidden , "LPA" , "Link Partner Ability" },
121 { 6, Phy::regReadUnimplemented, Phy::regWriteForbidden , "ANE" , "Auto-Negotiation Expansion" },
122 { 7, Phy::regReadUnimplemented, Phy::regWriteUnimplemented, "NPT" , "Next Page Transmit" },
123 { 8, Phy::regReadUnimplemented, Phy::regWriteForbidden , "LPN" , "Link Partner Next Page" },
124 { 9, Phy::regReadDefault , Phy::regWriteUnimplemented, "GCON" , "1000BASE-T Control" },
125 { 10, Phy::regReadGSTATUS , Phy::regWriteForbidden , "GSTATUS" , "1000BASE-T Status" },
126 { 15, Phy::regReadUnimplemented, Phy::regWriteForbidden , "EPSTATUS" , "Extended PHY Status" },
127 { 16, Phy::regReadDefault , Phy::regWriteDefault , "PSCON" , "PHY Specific Control" },
128 { 17, Phy::regReadDefault , Phy::regWriteForbidden , "PSSTAT" , "PHY Specific Status" },
129 { 18, Phy::regReadUnimplemented, Phy::regWriteUnimplemented, "PINTE" , "PHY Interrupt Enable" },
130 { 19, Phy::regReadUnimplemented, Phy::regWriteForbidden , "PINTS" , "PHY Interrupt Status" },
131 { 20, Phy::regReadUnimplemented, Phy::regWriteUnimplemented, "EPSCON1" , "Extended PHY Specific Control 1" },
132 { 21, Phy::regReadUnimplemented, Phy::regWriteForbidden , "PREC" , "PHY Receive Error Counter" },
133 { 26, Phy::regReadUnimplemented, Phy::regWriteUnimplemented, "EPSCON2" , "Extended PHY Specific Control 2" },
134 { 29, Phy::regReadForbidden , Phy::regWriteUnimplemented, "R30PS" , "MDI Register 30 Page Select" },
135 { 30, Phy::regReadUnimplemented, Phy::regWriteUnimplemented, "R30AW" , "MDI Register 30 Access Window" }
136 };
137}
138
139/**
140 * Default read handler.
141 *
142 * Fetches register value from the state structure.
143 *
144 * @returns Register value
145 *
146 * @param index Register index in register array.
147 */
148static uint16_t Phy::regReadDefault(PPHY pPhy, uint32_t index, PPDMDEVINS pDevIns)
149{
150 RT_NOREF(pDevIns);
151 AssertReturn(index<Phy::NUM_OF_PHY_REGS, 0);
152 return pPhy->au16Regs[index];
153}
154
155/**
156 * Default write handler.
157 *
158 * Writes the specified register value to the state structure.
159 *
160 * @param index Register index in register array.
161 * @param value The value to store (ignored).
162 */
163static void Phy::regWriteDefault(PPHY pPhy, uint32_t index, uint16_t u16Value, PPDMDEVINS pDevIns)
164{
165 RT_NOREF(pDevIns);
166 AssertReturnVoid(index < NUM_OF_PHY_REGS);
167 pPhy->au16Regs[index] = u16Value;
168}
169
170/**
171 * Read handler for write-only registers.
172 *
173 * Merely reports reads from write-only registers.
174 *
175 * @returns Register value (always 0)
176 *
177 * @param index Register index in register array.
178 */
179static uint16_t Phy::regReadForbidden(PPHY pPhy, uint32_t index, PPDMDEVINS pDevIns)
180{
181 RT_NOREF(pPhy, index, pDevIns);
182 PhyLog(("PHY#%d At %02d read attempted from write-only '%s'\n",
183 pPhy->iInstance, s_regMap[index].u32Address, s_regMap[index].pszName));
184 return 0;
185}
186
187/**
188 * Write handler for read-only registers.
189 *
190 * Merely reports writes to read-only registers.
191 *
192 * @param index Register index in register array.
193 * @param value The value to store (ignored).
194 */
195static void Phy::regWriteForbidden(PPHY pPhy, uint32_t index, uint16_t u16Value, PPDMDEVINS pDevIns)
196{
197 RT_NOREF(pPhy, index, u16Value, pDevIns);
198 PhyLog(("PHY#%d At %02d write attempted to read-only '%s'\n",
199 pPhy->iInstance, s_regMap[index].u32Address, s_regMap[index].pszName));
200}
201
202/**
203 * Read handler for unimplemented registers.
204 *
205 * Merely reports reads from unimplemented registers.
206 *
207 * @returns Register value (always 0)
208 *
209 * @param index Register index in register array.
210 */
211static uint16_t Phy::regReadUnimplemented(PPHY pPhy, uint32_t index, PPDMDEVINS pDevIns)
212{
213 RT_NOREF(pPhy, index, pDevIns);
214 PhyLog(("PHY#%d At %02d read attempted from unimplemented '%s'\n",
215 pPhy->iInstance, s_regMap[index].u32Address, s_regMap[index].pszName));
216 return 0;
217}
218
219/**
220 * Write handler for unimplemented registers.
221 *
222 * Merely reports writes to unimplemented registers.
223 *
224 * @param index Register index in register array.
225 * @param value The value to store (ignored).
226 */
227static void Phy::regWriteUnimplemented(PPHY pPhy, uint32_t index, uint16_t u16Value, PPDMDEVINS pDevIns)
228{
229 RT_NOREF(pPhy, index, u16Value, pDevIns);
230 PhyLog(("PHY#%d At %02d write attempted to unimplemented '%s'\n",
231 pPhy->iInstance, s_regMap[index].u32Address, s_regMap[index].pszName));
232}
233
234
235/**
236 * Search PHY register table for register with matching address.
237 *
238 * @returns Index in the register table or -1 if not found.
239 *
240 * @param u32Address Register address.
241 */
242static int Phy::lookupRegister(uint32_t u32Address)
243{
244 unsigned int index;
245
246 for (index = 0; index < RT_ELEMENTS(s_regMap); index++)
247 {
248 if (s_regMap[index].u32Address == u32Address)
249 {
250 return (int)index;
251 }
252 }
253
254 return -1;
255}
256
257/**
258 * Read PHY register.
259 *
260 * @returns Value of specified PHY register.
261 *
262 * @param u32Address Register address.
263 */
264uint16_t Phy::readRegister(PPHY pPhy, uint32_t u32Address, PPDMDEVINS pDevIns)
265{
266 int index = Phy::lookupRegister(u32Address);
267 uint16_t u16 = 0;
268
269 if (index >= 0)
270 {
271 u16 = s_regMap[index].pfnRead(pPhy, (uint32_t)index, pDevIns);
272 PhyLog(("PHY#%d At %02d read %04X from %s (%s)\n",
273 pPhy->iInstance, s_regMap[index].u32Address, u16,
274 s_regMap[index].pszAbbrev, s_regMap[index].pszName));
275 }
276 else
277 {
278 PhyLog(("PHY#%d read attempted from non-existing register %08x\n",
279 pPhy->iInstance, u32Address));
280 }
281 return u16;
282}
283
284/**
285 * Write to PHY register.
286 *
287 * @param u32Address Register address.
288 * @param u16Value Value to store.
289 */
290void Phy::writeRegister(PPHY pPhy, uint32_t u32Address, uint16_t u16Value, PPDMDEVINS pDevIns)
291{
292 int index = Phy::lookupRegister(u32Address);
293
294 if (index >= 0)
295 {
296 PhyLog(("PHY#%d At %02d write %04X to %s (%s)\n",
297 pPhy->iInstance, s_regMap[index].u32Address, u16Value,
298 s_regMap[index].pszAbbrev, s_regMap[index].pszName));
299 s_regMap[index].pfnWrite(pPhy, (uint32_t)index, u16Value, pDevIns);
300 }
301 else
302 {
303 PhyLog(("PHY#%d write attempted to non-existing register %08x\n",
304 pPhy->iInstance, u32Address));
305 }
306}
307
308/**
309 * PHY constructor.
310 *
311 * Stores E1000 instance internally. Triggers PHY hard reset.
312 *
313 * @param iNICInstance Number of network controller instance this PHY is
314 * attached to.
315 * @param u16EPid Extended PHY Id.
316 */
317void Phy::init(PPHY pPhy, int iNICInstance, uint16_t u16EPid)
318{
319 pPhy->iInstance = iNICInstance;
320 /* The PHY identifier composed of bits 3 through 18 of the OUI */
321 /* (Organizationally Unique Identifier). OUI is 0x05043. */
322 REG(PID) = 0x0141;
323 /* Extended PHY identifier */
324 REG(EPID) = u16EPid;
325 hardReset(pPhy);
326}
327
328/**
329 * Hardware PHY reset.
330 *
331 * Sets all PHY registers to their initial values.
332 */
333void Phy::hardReset(PPHY pPhy)
334{
335 PhyLog(("PHY#%d Hard reset\n", pPhy->iInstance));
336 REG(PCTRL) = PCTRL_SPDSELM | PCTRL_DUPMOD | PCTRL_ANEG;
337 /*
338 * 100 and 10 FD/HD, Extended Status, MF Preamble Suppression,
339 * AUTO NEG AB, EXT CAP
340 */
341 REG(PSTATUS) = 0x7949;
342 REG(ANA) = 0x01E1;
343 /* No flow control by our link partner, all speeds */
344 REG(LPA) = 0x01E0;
345 REG(ANE) = 0x0000;
346 REG(NPT) = 0x2001;
347 REG(LPN) = 0x0000;
348 REG(GCON) = 0x1E00;
349 REG(GSTATUS) = 0x0000;
350 REG(EPSTATUS) = 0x3000;
351 REG(PSCON) = 0x0068;
352 REG(PSSTAT) = 0x0000;
353 REG(PINTE) = 0x0000;
354 REG(PINTS) = 0x0000;
355 REG(EPSCON1) = 0x0D60;
356 REG(PREC) = 0x0000;
357 REG(EPSCON2) = 0x000C;
358 REG(R30PS) = 0x0000;
359 REG(R30AW) = 0x0000;
360
361 pPhy->u16State = MDIO_IDLE;
362}
363
364/**
365 * Software PHY reset.
366 */
367static void Phy::softReset(PPHY pPhy, PPDMDEVINS pDevIns)
368{
369 PhyLog(("PHY#%d Soft reset\n", pPhy->iInstance));
370
371 REG(PCTRL) = REG(PCTRL) & (PCTRL_SPDSELM | PCTRL_DUPMOD | PCTRL_ANEG | PCTRL_SPDSELL);
372 /*
373 * 100 and 10 FD/HD, Extended Status, MF Preamble Suppression,
374 * AUTO NEG AB, EXT CAP
375 */
376 REG(PSTATUS) = 0x7949;
377 REG(PSSTAT) &= 0xe001;
378 PhyLog(("PHY#%d PSTATUS=%04x PSSTAT=%04x\n", pPhy->iInstance, REG(PSTATUS), REG(PSSTAT)));
379
380#ifndef PHY_UNIT_TEST
381 e1kPhyLinkResetCallback(pDevIns);
382#else
383 RT_NOREF(pDevIns);
384#endif
385}
386
387/**
388 * Get the current state of the link.
389 *
390 * @returns true if link is up.
391 */
392bool Phy::isLinkUp(PPHY pPhy)
393{
394 return (REG(PSSTAT) & PSSTAT_LINK) != 0;
395}
396
397/**
398 * Set the current state of the link.
399 *
400 * @remarks Link Status bit in PHY Status register is latched-low and does
401 * not change the state when the link goes up.
402 *
403 * @param fLinkIsUp New state of the link.
404 */
405void Phy::setLinkStatus(PPHY pPhy, bool fLinkIsUp)
406{
407 if (fLinkIsUp)
408 {
409 REG(PSSTAT) |= PSSTAT_LINK_ALL;
410 REG(PSTATUS) |= PSTATUS_NEGCOMP; /* PSTATUS_LNKSTAT is latched low */
411 }
412 else
413 {
414 REG(PSSTAT) &= ~PSSTAT_LINK_ALL;
415 REG(PSTATUS) &= ~(PSTATUS_LNKSTAT | PSTATUS_NEGCOMP);
416 }
417 PhyLog(("PHY#%d setLinkStatus: PSTATUS=%04x PSSTAT=%04x\n", pPhy->iInstance, REG(PSTATUS), REG(PSSTAT)));
418}
419
420#ifdef IN_RING3
421
422/**
423 * Save PHY state.
424 *
425 * @remarks Since PHY is aggregated into E1K it does not currently supports
426 * versioning of its own.
427 *
428 * @returns VBox status code.
429 * @param pHlp Device helper table.
430 * @param pSSM The handle to save the state to.
431 * @param pPhy The pointer to this instance.
432 */
433int Phy::saveState(PCPDMDEVHLPR3 pHlp, PSSMHANDLE pSSM, PPHY pPhy)
434{
435 pHlp->pfnSSMPutMem(pSSM, pPhy->au16Regs, sizeof(pPhy->au16Regs));
436 return VINF_SUCCESS;
437}
438
439/**
440 * Restore previously saved PHY state.
441 *
442 * @remarks Since PHY is aggregated into E1K it does not currently supports
443 * versioning of its own.
444 *
445 * @returns VBox status code.
446 * @param pHlp Device helper table.
447 * @param pSSM The handle to save the state to.
448 * @param pPhy The pointer to this instance.
449 */
450int Phy::loadState(PCPDMDEVHLPR3 pHlp, PSSMHANDLE pSSM, PPHY pPhy)
451{
452 return pHlp->pfnSSMGetMem(pSSM, pPhy->au16Regs, sizeof(pPhy->au16Regs));
453}
454
455#endif /* IN_RING3 */
456
457/* Register-specific handlers ************************************************/
458
459/**
460 * Write handler for PHY Control register.
461 *
462 * Handles reset.
463 *
464 * @param index Register index in register array.
465 * @param value The value to store (ignored).
466 */
467static void Phy::regWritePCTRL(PPHY pPhy, uint32_t index, uint16_t u16Value, PPDMDEVINS pDevIns)
468{
469 if (u16Value & PCTRL_RESET)
470 softReset(pPhy, pDevIns);
471 else
472 regWriteDefault(pPhy, index, u16Value, pDevIns);
473}
474
475/**
476 * Read handler for PHY Status register.
477 *
478 * Handles Latched-Low Link Status bit.
479 *
480 * @returns Register value
481 *
482 * @param index Register index in register array.
483 */
484static uint16_t Phy::regReadPSTATUS(PPHY pPhy, uint32_t index, PPDMDEVINS pDevIns)
485{
486 RT_NOREF(pPhy, index, pDevIns);
487
488 /* Read latched value */
489 uint16_t u16 = REG(PSTATUS);
490 if (REG(PSSTAT) & PSSTAT_LINK)
491 REG(PSTATUS) |= PSTATUS_LNKSTAT;
492 else
493 REG(PSTATUS) &= ~PSTATUS_LNKSTAT;
494 return u16;
495}
496
497/**
498 * Read handler for 1000BASE-T Status register.
499 *
500 * @returns Register value
501 *
502 * @param index Register index in register array.
503 */
504static uint16_t Phy::regReadGSTATUS(PPHY pPhy, uint32_t index, PPDMDEVINS pDevIns)
505{
506 RT_NOREF(pPhy, index, pDevIns);
507
508 /*
509 * - Link partner is capable of 1000BASE-T half duplex
510 * - Link partner is capable of 1000BASE-T full duplex
511 * - Remote receiver OK
512 * - Local receiver OK
513 * - Local PHY config resolved to SLAVE
514 */
515 return 0x3C00;
516}
517
518#if defined(LOG_ENABLED) || defined(PHY_UNIT_TEST)
519static const char * Phy::getStateName(uint16_t u16State)
520{
521 static const char *pcszState[] =
522 {
523 "MDIO_IDLE",
524 "MDIO_ST",
525 "MDIO_OP_ADR",
526 "MDIO_TA_RD",
527 "MDIO_TA_WR",
528 "MDIO_READ",
529 "MDIO_WRITE"
530 };
531
532 return (u16State < RT_ELEMENTS(pcszState)) ? pcszState[u16State] : "<invalid>";
533}
534#endif
535
536bool Phy::readMDIO(PPHY pPhy)
537{
538 bool fPin = false;
539
540 switch (pPhy->u16State)
541 {
542 case MDIO_TA_RD:
543 Assert(pPhy->u16Cnt == 1);
544 fPin = false;
545 pPhy->u16State = MDIO_READ;
546 pPhy->u16Cnt = 16;
547 break;
548 case MDIO_READ:
549 /* Bits are shifted out in MSB to LSB order */
550 fPin = (pPhy->u16Acc & 0x8000) != 0;
551 pPhy->u16Acc <<= 1;
552 if (--pPhy->u16Cnt == 0)
553 pPhy->u16State = MDIO_IDLE;
554 break;
555 default:
556 PhyLog(("PHY#%d WARNING! MDIO pin read in %s state\n", pPhy->iInstance, Phy::getStateName(pPhy->u16State)));
557 pPhy->u16State = MDIO_IDLE;
558 }
559 return fPin;
560}
561
562/** Set the value of MDIO pin. */
563void Phy::writeMDIO(PPHY pPhy, bool fPin, PPDMDEVINS pDevIns)
564{
565 switch (pPhy->u16State)
566 {
567 case MDIO_IDLE:
568 if (!fPin)
569 pPhy->u16State = MDIO_ST;
570 break;
571 case MDIO_ST:
572 if (fPin)
573 {
574 pPhy->u16State = MDIO_OP_ADR;
575 pPhy->u16Cnt = 12; /* OP + PHYADR + REGADR */
576 pPhy->u16Acc = 0;
577 }
578 break;
579 case MDIO_OP_ADR:
580 Assert(pPhy->u16Cnt);
581 /* Shift in 'u16Cnt' bits into accumulator */
582 pPhy->u16Acc <<= 1;
583 if (fPin)
584 pPhy->u16Acc |= 1;
585 if (--pPhy->u16Cnt == 0)
586 {
587 /* Got OP(2) + PHYADR(5) + REGADR(5) */
588 /* Note: A single PHY is supported, ignore PHYADR */
589 switch (pPhy->u16Acc >> 10)
590 {
591 case MDIO_READ_OP:
592 pPhy->u16Acc = readRegister(pPhy, pPhy->u16Acc & 0x1F, pDevIns);
593 pPhy->u16State = MDIO_TA_RD;
594 pPhy->u16Cnt = 1;
595 break;
596 case MDIO_WRITE_OP:
597 pPhy->u16RegAdr = pPhy->u16Acc & 0x1F;
598 pPhy->u16State = MDIO_TA_WR;
599 pPhy->u16Cnt = 2;
600 break;
601 default:
602 PhyLog(("PHY#%d ERROR! Invalid MDIO op: %d\n", pPhy->iInstance, pPhy->u16Acc >> 10));
603 pPhy->u16State = MDIO_IDLE;
604 break;
605 }
606 }
607 break;
608 case MDIO_TA_WR:
609 Assert(pPhy->u16Cnt <= 2);
610 Assert(pPhy->u16Cnt > 0);
611 if (--pPhy->u16Cnt == 0)
612 {
613 pPhy->u16State = MDIO_WRITE;
614 pPhy->u16Cnt = 16;
615 }
616 break;
617 case MDIO_WRITE:
618 Assert(pPhy->u16Cnt);
619 pPhy->u16Acc <<= 1;
620 if (fPin)
621 pPhy->u16Acc |= 1;
622 if (--pPhy->u16Cnt == 0)
623 {
624 writeRegister(pPhy, pPhy->u16RegAdr, pPhy->u16Acc, pDevIns);
625 pPhy->u16State = MDIO_IDLE;
626 }
627 break;
628 default:
629 PhyLog(("PHY#%d ERROR! MDIO pin write in %s state\n", pPhy->iInstance, Phy::getStateName(pPhy->u16State)));
630 pPhy->u16State = MDIO_IDLE;
631 break;
632 }
633}
634
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle
ContactPrivacy/Do Not Sell My InfoTerms of Use