1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
|
/** $Id: DevE1000Phy.h $ */
/** @file
* DevE1000Phy - Intel 82540EM Ethernet Controller Internal PHY Emulation, Header.
*/
/*
* Copyright (C) 2007-2023 Oracle and/or its affiliates.
*
* This file is part of VirtualBox base platform packages, as
* available from https://www.virtualbox.org.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation, in version 3 of the
* License.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see <https://www.gnu.org/licenses>.
*
* SPDX-License-Identifier: GPL-3.0-only
*/
#ifndef VBOX_INCLUDED_SRC_Network_DevE1000Phy_h
#define VBOX_INCLUDED_SRC_Network_DevE1000Phy_h
#ifndef RT_WITHOUT_PRAGMA_ONCE
# pragma once
#endif
#include <VBox/types.h>
#define PHY_EPID_M881000 0xC50
#define PHY_EPID_M881011 0xC24
#define PCTRL_SPDSELM 0x0040
#define PCTRL_DUPMOD 0x0100
#define PCTRL_ANEG 0x1000
#define PCTRL_SPDSELL 0x2000
#define PCTRL_RESET 0x8000
#define PSTATUS_LNKSTAT 0x0004
#define PSTATUS_NEGCOMP 0x0020
/*
* Speed: 1000 Mb/s
* Duplex: full
* Page received
* Resolved
* Link up
* Receive Pause Enable
*/
#define PSSTAT_LINK_ALL 0xBC08
#define PSSTAT_LINK 0x0400
namespace Phy
{
/**
* Indices of memory-mapped registers in register table
*/
enum enmRegIdx
{
PCTRL_IDX,
PSTATUS_IDX,
PID_IDX,
EPID_IDX,
ANA_IDX,
LPA_IDX,
ANE_IDX,
NPT_IDX,
LPN_IDX,
GCON_IDX,
GSTATUS_IDX,
EPSTATUS_IDX,
PSCON_IDX,
PSSTAT_IDX,
PINTE_IDX,
PINTS_IDX,
EPSCON1_IDX,
PREC_IDX,
EPSCON2_IDX,
R30PS_IDX,
R30AW_IDX,
NUM_OF_PHY_REGS
};
/**
* Emulation state of PHY.
*/
struct Phy_st
{
/** Network controller instance this PHY is attached to. */
int iInstance;
/** Register storage. */
uint16_t au16Regs[NUM_OF_PHY_REGS];
/** Current state of serial MDIO interface. */
uint16_t u16State;
/** Current state of serial MDIO interface. */
uint16_t u16Acc;
/** Number of bits remaining to be shifted into/out of accumulator. */
uint16_t u16Cnt;
/** PHY register offset selected for MDIO operation. */
uint16_t u16RegAdr;
};
}
#define MDIO_IDLE 0
#define MDIO_ST 1
#define MDIO_OP_ADR 2
#define MDIO_TA_RD 3
#define MDIO_TA_WR 4
#define MDIO_READ 5
#define MDIO_WRITE 6
#define MDIO_READ_OP 2
#define MDIO_WRITE_OP 1
/* External callback declaration */
void e1kPhyLinkResetCallback(PPDMDEVINS pDevIns);
typedef struct Phy::Phy_st PHY;
typedef PHY *PPHY;
/* Interface *****************************************************************/
namespace Phy
{
/** Initialize PHY. */
void init(PPHY pPhy, int iNICInstance, uint16_t u16EPid);
/** Read PHY register at specified address. */
uint16_t readRegister(PPHY pPhy, uint32_t u32Address, PPDMDEVINS pDevIns);
/** Write to PHY register at specified address. */
void writeRegister(PPHY pPhy, uint32_t u32Address, uint16_t u16Value, PPDMDEVINS pDevIns);
/** Read the value on MDIO pin. */
bool readMDIO(PPHY pPhy);
/** Set the value of MDIO pin. */
void writeMDIO(PPHY pPhy, bool fPin, PPDMDEVINS pDevIns);
/** Hardware reset. */
void hardReset(PPHY pPhy);
/** Query link status. */
bool isLinkUp(PPHY pPhy);
/** Set link status. */
void setLinkStatus(PPHY pPhy, bool fLinkIsUp);
/** Save PHY state. */
int saveState(struct PDMDEVHLPR3 const *pHlp, PSSMHANDLE pSSM, PPHY pPhy);
/** Restore previously saved PHY state. */
int loadState(struct PDMDEVHLPR3 const *pHlp, PSSMHANDLE pSSM, PPHY pPhy);
}
#endif /* !VBOX_INCLUDED_SRC_Network_DevE1000Phy_h */
|