1
6
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
38
39
40
41
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
65
75
76
77
81
90
91
92
96
97
98
99
100
101
102
103
104
105
106
107
108
109
113
119
120
121
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
144
151
152
153
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
239
240
241
242
243
244
264
265
266
267
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
355
356
357
358
359
360
361
362
363
364
365
366
367
368
371
374
375
376
377
378
379
380
381
382
383
384
385
386
389
390
/* ... */
#include <string.h>
#include <stdlib.h>
#include <sys/cdefs.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "esp_log.h"
#include "esp_check.h"
#include "esp_eth_phy_802_3.h"8 includes
static const char *TAG = "lan87xx";
#define LAN87XX_PHY_RESET_ASSERTION_TIME_US 150
#define LAN8710A_MODEL_NUM 0x0F
#define LAN8720A_MODEL_NUM 0x0F
#define LAN8740A_MODEL_NUM 0x11
#define LAN8741A_MODEL_NUM 0x12
#define LAN8742A_MODEL_NUM 0x136 defines
static const uint8_t supported_models[] = {
LAN8710A_MODEL_NUM,
#if (LAN8710A_MODEL_NUM != LAN8720A_MODEL_NUM)
LAN8720A_MODEL_NUM,
#endif
LAN8740A_MODEL_NUM,
LAN8741A_MODEL_NUM,
LAN8742A_MODEL_NUM
}{...};
/* ... */
typedef union {
struct {
uint32_t reserved1 : 1;
uint32_t energy_is_on : 1;
uint32_t reserved2 : 4;
uint32_t en_alternate_interrupt : 1;
uint32_t reserved3 : 2;
uint32_t en_far_loopback : 1;
uint32_t reserved4 : 3;
uint32_t en_energy_detect_powerdown : 1;
uint32_t reserved5 : 2;
}{ ... };
uint32_t val;
}{ ... } mcsr_reg_t;
#define ETH_PHY_MCSR_REG_ADDR (0x11)
/* ... */
typedef union {
struct {
uint32_t phy_addr : 5;
uint32_t mode : 3;
uint32_t reserved_1 : 6;
uint32_t mii_mode : 1;
uint32_t reserved_2 : 1;
}{ ... };
uint32_t val;
}{ ... } smr_reg_t;
#define ETH_PHY_SMR_REG_ADDR (0x12)
/* ... */
typedef union {
struct {
uint32_t tdr_pattern_low : 6;
uint32_t tdr_pattern_high : 6;
uint32_t tdr_line_break_counter : 3;
uint32_t tdr_delay_in : 1;
}{ ... };
uint32_t val;
}{ ... } tdr_pattern_reg_t;
#define EHT_PHY_TDRPD_REG_ADDR (0x18)
/* ... */
typedef union {
struct {
uint32_t tdr_channel_length : 8;
uint32_t tdr_channel_status : 1;
uint32_t tdr_channel_cable_type : 2;
uint32_t reserved : 3;
uint32_t tdr_a2d_filter_enable: 1;
uint32_t tdr_enable : 1;
}{ ... };
uint32_t val;
}{ ... } tdr_control_reg_t;
#define EHT_PHY_TDRC_REG_ADDR (0x19)
/* ... */
typedef union {
struct {
uint32_t symbol_err_count : 16;
}{ ... };
uint32_t val;
}{ ... } secr_reg_t;
#define EHT_PHY_SECR_REG_ADDR (0x1A)
/* ... */
typedef union {
struct {
uint32_t reserved1 : 4;
uint32_t base10_t_polarity : 1;
uint32_t reserved2 : 6;
uint32_t dis_sqe : 1;
uint32_t reserved3 : 1;
uint32_t select_channel : 1;
uint32_t reserved4 : 1;
uint32_t auto_mdix_ctrl : 1;
}{ ... };
uint32_t val;
}{ ... } scsir_reg_t;
#define ETH_PHY_CSIR_REG_ADDR (0x1B)
/* ... */
typedef union {
struct {
uint32_t reserved : 12;
uint32_t cable_length : 4;
}{ ... };
uint32_t val;
}{ ... } cbln_reg_t;
#define EHT_PHY_CBLN_REG_ADDR (0x1C)
/* ... */
typedef union {
struct {
uint32_t reserved1 : 1;
uint32_t auto_nego_page_received : 1;
uint32_t parallel_detect_fault : 1;
uint32_t auto_nego_lp_acknowledge : 1;
uint32_t link_down : 1;
uint32_t remote_fault_detect : 1;
uint32_t auto_nego_complete : 1;
uint32_t energy_on_generate : 1;
uint32_t wake_on_lan : 1;
uint32_t reserved2 : 7;
}{ ... };
uint32_t val;
}{ ... } isfr_reg_t;
#define ETH_PHY_ISR_REG_ADDR (0x1D)
/* ... */
typedef union {
struct {
uint32_t reserved1 : 1;
uint32_t auto_nego_page_received : 1;
uint32_t parallel_detect_fault : 1;
uint32_t auto_nego_lp_acknowledge : 1;
uint32_t link_down : 1;
uint32_t remote_fault_detect : 1;
uint32_t auto_nego_complete : 1;
uint32_t energy_on_generate : 1;
uint32_t wake_on_lan : 1;
uint32_t reserved2 : 7;
}{ ... };
uint32_t val;
}{ ... } imr_reg_t;
#define ETH_PHY_IMR_REG_ADDR (0x1E)
/* ... */
typedef union {
struct {
uint32_t reserved1 : 2;
uint32_t speed_indication : 3;
uint32_t reserved2 : 1;
uint32_t enable_4b5b : 1;
uint32_t reserved3 : 5;
uint32_t auto_nego_done : 1;
uint32_t reserved4 : 3;
}{ ... };
uint32_t val;
}{ ... } pscsr_reg_t;
#define ETH_PHY_PSCSR_REG_ADDR (0x1F)
typedef struct {
phy_802_3_t phy_802_3;
}{ ... } phy_lan87xx_t;
static esp_err_t lan87xx_update_link_duplex_speed(phy_lan87xx_t *lan87xx)
{
esp_err_t ret = ESP_OK;
esp_eth_mediator_t *eth = lan87xx->phy_802_3.eth;
uint32_t addr = lan87xx->phy_802_3.addr;
eth_speed_t speed = ETH_SPEED_10M;
eth_duplex_t duplex = ETH_DUPLEX_HALF;
bmsr_reg_t bmsr;
bmcr_reg_t bmcr;
pscsr_reg_t pscsr;
uint32_t peer_pause_ability = false;
anlpar_reg_t anlpar;
ESP_GOTO_ON_ERROR(eth->phy_reg_read(eth, addr, ETH_PHY_ANLPAR_REG_ADDR, &(anlpar.val)), err, TAG, "read ANLPAR failed");
ESP_GOTO_ON_ERROR(eth->phy_reg_read(eth, addr, ETH_PHY_BMSR_REG_ADDR, &(bmsr.val)), err, TAG, "read BMSR failed");
ESP_GOTO_ON_ERROR(eth->phy_reg_read(eth, addr, ETH_PHY_BMCR_REG_ADDR, &(bmcr.val)), err, TAG, "read BMCR failed");
eth_link_t link;
if(bmcr.en_loopback) {
link = ETH_LINK_UP;
}{...} else {
link = bmsr.link_status ? ETH_LINK_UP : ETH_LINK_DOWN;
}{...}
if (lan87xx->phy_802_3.link_status != link) {
if (link == ETH_LINK_UP) {
ESP_GOTO_ON_ERROR(eth->phy_reg_read(eth, addr, ETH_PHY_PSCSR_REG_ADDR, &(pscsr.val)), err, TAG, "read PSCSR failed");
switch (pscsr.speed_indication) {
case 1:
speed = ETH_SPEED_10M;
duplex = ETH_DUPLEX_HALF;
break;...
case 2:
speed = ETH_SPEED_100M;
duplex = ETH_DUPLEX_HALF;
break;...
case 5:
speed = ETH_SPEED_10M;
duplex = ETH_DUPLEX_FULL;
break;...
case 6:
speed = ETH_SPEED_100M;
duplex = ETH_DUPLEX_FULL;
break;...
default:
break;...
}{...}
ESP_GOTO_ON_ERROR(eth->on_state_changed(eth, ETH_STATE_SPEED, (void *)speed), err, TAG, "change speed failed");
ESP_GOTO_ON_ERROR(eth->on_state_changed(eth, ETH_STATE_DUPLEX, (void *)duplex), err, TAG, "change duplex failed");
if (duplex == ETH_DUPLEX_FULL && anlpar.symmetric_pause) {
peer_pause_ability = 1;
}{...} else {
peer_pause_ability = 0;
}{...}
ESP_GOTO_ON_ERROR(eth->on_state_changed(eth, ETH_STATE_PAUSE, (void *)peer_pause_ability), err, TAG, "change pause ability failed");
}{...}
ESP_GOTO_ON_ERROR(eth->on_state_changed(eth, ETH_STATE_LINK, (void *)link), err, TAG, "change link failed");
lan87xx->phy_802_3.link_status = link;
}{...}
return ESP_OK;
err:
return ret;
}{ ... }
static esp_err_t lan87xx_get_link(esp_eth_phy_t *phy)
{
esp_err_t ret = ESP_OK;
phy_lan87xx_t *lan87xx = __containerof(esp_eth_phy_into_phy_802_3(phy), phy_lan87xx_t, phy_802_3);
ESP_GOTO_ON_ERROR(lan87xx_update_link_duplex_speed(lan87xx), err, TAG, "update link duplex speed failed");
return ESP_OK;
err:
return ret;
}{ ... }
static esp_err_t lan87xx_autonego_ctrl(esp_eth_phy_t *phy, eth_phy_autoneg_cmd_t cmd, bool *autonego_en_stat)
{
esp_err_t ret = ESP_OK;
phy_802_3_t *phy_802_3 = esp_eth_phy_into_phy_802_3(phy);
esp_eth_mediator_t *eth = phy_802_3->eth;
if (cmd == ESP_ETH_PHY_AUTONEGO_EN) {
bmcr_reg_t bmcr;
ESP_GOTO_ON_ERROR(eth->phy_reg_read(eth, phy_802_3->addr, ETH_PHY_BMCR_REG_ADDR, &(bmcr.val)), err, TAG, "read BMCR failed");
ESP_GOTO_ON_FALSE(bmcr.en_loopback == 0, ESP_ERR_INVALID_STATE, err, TAG, "Autonegotiation can't be enabled while in loopback operation");
}{...}
return esp_eth_phy_802_3_autonego_ctrl(phy_802_3, cmd, autonego_en_stat);
err:
return ret;
}{ ... }
static esp_err_t lan87xx_loopback(esp_eth_phy_t *phy, bool enable)
{
esp_err_t ret = ESP_OK;
phy_802_3_t *phy_802_3 = esp_eth_phy_into_phy_802_3(phy);
bool auto_nego_en;
ESP_GOTO_ON_ERROR(lan87xx_autonego_ctrl(phy, ESP_ETH_PHY_AUTONEGO_G_STAT, &auto_nego_en), err, TAG, "get status of autonegotiation failed");
ESP_GOTO_ON_FALSE(!(auto_nego_en && enable), ESP_ERR_INVALID_STATE, err, TAG, "Unable to set loopback while autonegotiation is enabled. Disable it to use loopback");
return esp_eth_phy_802_3_loopback(phy_802_3, enable);
err:
return ret;
}{ ... }
static esp_err_t lan87xx_set_speed(esp_eth_phy_t *phy, eth_speed_t speed)
{
esp_err_t ret = ESP_OK;
phy_802_3_t *phy_802_3 = esp_eth_phy_into_phy_802_3(phy);
/* ... */
ESP_GOTO_ON_ERROR(esp_eth_phy_802_3_set_speed(phy_802_3, speed), err, TAG, "set speed failed");
vTaskDelay(pdMS_TO_TICKS(10));
err:
return ret;
}{ ... }
static esp_err_t lan87xx_init(esp_eth_phy_t *phy)
{
esp_err_t ret = ESP_OK;
phy_802_3_t *phy_802_3 = esp_eth_phy_into_phy_802_3(phy);
ESP_GOTO_ON_ERROR(esp_eth_phy_802_3_basic_phy_init(phy_802_3), err, TAG, "failed to init PHY");
uint32_t oui;
uint8_t model;
ESP_GOTO_ON_ERROR(esp_eth_phy_802_3_read_oui(phy_802_3, &oui), err, TAG, "read OUI failed");
ESP_GOTO_ON_ERROR(esp_eth_phy_802_3_read_manufac_info(phy_802_3, &model, NULL), err, TAG, "read manufacturer's info failed");
ESP_GOTO_ON_FALSE(oui == 0x1F0, ESP_FAIL, err, TAG, "wrong chip OUI");
bool supported_model = false;
for (unsigned int i = 0; i < sizeof(supported_models); i++) {
if (model == supported_models[i]) {
supported_model = true;
break;
}{...}
}{...}
ESP_GOTO_ON_FALSE(supported_model, ESP_FAIL, err, TAG, "unsupported chip model");
return ESP_OK;
err:
return ret;
}{ ... }
esp_eth_phy_t *esp_eth_phy_new_lan87xx(const eth_phy_config_t *config)
{
esp_eth_phy_t *ret = NULL;
phy_lan87xx_t *lan87xx = calloc(1, sizeof(phy_lan87xx_t));
ESP_GOTO_ON_FALSE(lan87xx, NULL, err, TAG, "calloc lan87xx failed");
eth_phy_config_t lan87xx_config = *config;
if (config->hw_reset_assert_time_us == 0) {
lan87xx_config.hw_reset_assert_time_us = LAN87XX_PHY_RESET_ASSERTION_TIME_US;
}{...}
if (config->post_hw_reset_delay_ms == 0) {
lan87xx_config.post_hw_reset_delay_ms = ESP_ETH_NO_POST_HW_RESET_DELAY;
}{...}
ESP_GOTO_ON_FALSE(esp_eth_phy_802_3_obj_config_init(&lan87xx->phy_802_3, &lan87xx_config) == ESP_OK,
NULL, err, TAG, "configuration initialization of PHY 802.3 failed");
lan87xx->phy_802_3.parent.init = lan87xx_init;
lan87xx->phy_802_3.parent.get_link = lan87xx_get_link;
lan87xx->phy_802_3.parent.autonego_ctrl = lan87xx_autonego_ctrl;
lan87xx->phy_802_3.parent.loopback = lan87xx_loopback;
lan87xx->phy_802_3.parent.set_speed = lan87xx_set_speed;
return &lan87xx->phy_802_3.parent;
err:
if (lan87xx != NULL) {
free(lan87xx);
}{...}
return ret;
}{ ... }