Select one of the symbols to view example projects that use it.
 
Outline
#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"
TAG
#define LAN87XX_PHY_RESET_ASSERTION_TIME_US
#define LAN8710A_MODEL_NUM
#define LAN8720A_MODEL_NUM
#define LAN8740A_MODEL_NUM
#define LAN8741A_MODEL_NUM
#define LAN8742A_MODEL_NUM
supported_models
mcsr_reg_t
#define ETH_PHY_MCSR_REG_ADDR
smr_reg_t
#define ETH_PHY_SMR_REG_ADDR
tdr_pattern_reg_t
#define EHT_PHY_TDRPD_REG_ADDR
tdr_control_reg_t
#define EHT_PHY_TDRC_REG_ADDR
secr_reg_t
#define EHT_PHY_SECR_REG_ADDR
scsir_reg_t
#define ETH_PHY_CSIR_REG_ADDR
cbln_reg_t
#define EHT_PHY_CBLN_REG_ADDR
isfr_reg_t
#define ETH_PHY_ISR_REG_ADDR
imr_reg_t
#define ETH_PHY_IMR_REG_ADDR
pscsr_reg_t
#define ETH_PHY_PSCSR_REG_ADDR
phy_lan87xx_t
lan87xx_update_link_duplex_speed(phy_lan87xx_t *)
lan87xx_get_link(esp_eth_phy_t *)
lan87xx_autonego_ctrl(esp_eth_phy_t *, eth_phy_autoneg_cmd_t, bool *)
lan87xx_loopback(esp_eth_phy_t *, bool)
lan87xx_set_speed(esp_eth_phy_t *, eth_speed_t)
lan87xx_init(esp_eth_phy_t *)
esp_eth_phy_new_lan87xx(const eth_phy_config_t *)
Files
loading...
SourceVuESP-IDF Framework and ExamplesESP-IDFcomponents/esp_eth/src/phy/esp_eth_phy_lan87xx.c
 
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
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
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
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * SPDX-FileCopyrightText: 2019-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 *//* ... */ #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"; /* It was observed that assert nRST signal on LAN87xx needs to be a little longer than the minimum specified in datasheet */ #define LAN87XX_PHY_RESET_ASSERTION_TIME_US 150 /***************List of Supported Models***************/ // See Microchip's Application Note AN25.3 summarizing differences among below models #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 }{...}; /***************Vendor Specific Register***************/ /** * @brief MCSR(Mode Control Status Register) * *//* ... */ typedef union { struct { uint32_t reserved1 : 1; /* Reserved */ uint32_t energy_is_on : 1; /* Energy is On */ uint32_t reserved2 : 4; /* Reserved */ uint32_t en_alternate_interrupt : 1; /* Enable Alternate Interrupt Mode */ uint32_t reserved3 : 2; /* Reserved */ uint32_t en_far_loopback : 1; /* Enable Far Loopback Mode */ uint32_t reserved4 : 3; /* Reserved */ uint32_t en_energy_detect_powerdown : 1; /* Enable Energy Detect Power Down */ uint32_t reserved5 : 2; /* Reserved */ }{ ... }; uint32_t val; }{ ... } mcsr_reg_t; #define ETH_PHY_MCSR_REG_ADDR (0x11) /** * @brief SMR(Special Modes Register) * *//* ... */ typedef union { struct { uint32_t phy_addr : 5; /* PHY Address */ uint32_t mode : 3; /* Transceiver Mode of Operation */ uint32_t reserved_1 : 6; /* Reserved */ uint32_t mii_mode : 1; /* Mode of the digital interface (only LAN8710A/LAN8740A/LAN8741A) */ uint32_t reserved_2 : 1; /* Reserved */ }{ ... }; uint32_t val; }{ ... } smr_reg_t; #define ETH_PHY_SMR_REG_ADDR (0x12) /** * @brief Time Domain Reflectometry Patterns/Delay Control Register * Only available in LAN8740A/LAN8742A *//* ... */ typedef union { struct { uint32_t tdr_pattern_low : 6; /* Data pattern sent in TDR mode for the low cycle */ uint32_t tdr_pattern_high : 6; /* Data pattern sent in TDR mode for the high cycle */ uint32_t tdr_line_break_counter : 3; /* Increments of 256ms of break time */ uint32_t tdr_delay_in : 1; /* Line break counter used */ }{ ... }; uint32_t val; }{ ... } tdr_pattern_reg_t; #define EHT_PHY_TDRPD_REG_ADDR (0x18) /** * @brief Time Domain Reflectometry Control/Status Register) * Only available in LAN8740A/LAN8742A *//* ... */ typedef union { struct { uint32_t tdr_channel_length : 8; /* TDR channel length */ uint32_t tdr_channel_status : 1; /* TDR channel status */ uint32_t tdr_channel_cable_type : 2; /* TDR channel cable type */ uint32_t reserved : 3; /* Reserved */ uint32_t tdr_a2d_filter_enable: 1; /* Analog to Digital Filter Enabled */ uint32_t tdr_enable : 1; /* Enable TDR */ }{ ... }; uint32_t val; }{ ... } tdr_control_reg_t; #define EHT_PHY_TDRC_REG_ADDR (0x19) /** * @brief SECR(Symbol Error Counter Register) * *//* ... */ typedef union { struct { uint32_t symbol_err_count : 16; /* Symbol Error Counter */ }{ ... }; uint32_t val; }{ ... } secr_reg_t; #define EHT_PHY_SECR_REG_ADDR (0x1A) /** * @brief CSIR(Control Status Indications Register) * *//* ... */ typedef union { struct { uint32_t reserved1 : 4; /* Reserved */ uint32_t base10_t_polarity : 1; /* Polarity State of 10Base-T */ uint32_t reserved2 : 6; /* Reserved */ uint32_t dis_sqe : 1; /* Disable SQE test(Heartbeat) */ uint32_t reserved3 : 1; /* Reserved */ uint32_t select_channel : 1; /* Manual channel select:MDI(0) or MDIX(1) */ uint32_t reserved4 : 1; /* Reserved */ uint32_t auto_mdix_ctrl : 1; /* Auto-MDIX Control: EN(0) or DE(1) */ }{ ... }; uint32_t val; }{ ... } scsir_reg_t; #define ETH_PHY_CSIR_REG_ADDR (0x1B) /** * @brief Cable Length Register * Only available in LAN8740A/LAN8742A *//* ... */ typedef union { struct { uint32_t reserved : 12; /* Reserved */ uint32_t cable_length : 4; /* Cable length */ }{ ... }; uint32_t val; }{ ... } cbln_reg_t; #define EHT_PHY_CBLN_REG_ADDR (0x1C) /** * @brief ISR(Interrupt Source Register) * *//* ... */ typedef union { struct { uint32_t reserved1 : 1; /* Reserved */ uint32_t auto_nego_page_received : 1; /* Auto-Negotiation Page Received */ uint32_t parallel_detect_fault : 1; /* Parallel Detection Fault */ uint32_t auto_nego_lp_acknowledge : 1; /* Auto-Negotiation LP Acknowledge */ uint32_t link_down : 1; /* Link Down */ uint32_t remote_fault_detect : 1; /* Remote Fault Detect */ uint32_t auto_nego_complete : 1; /* Auto-Negotiation Complete */ uint32_t energy_on_generate : 1; /* ENERGY ON generated */ uint32_t wake_on_lan : 1; /* Wake on Lan (WOL) event detected (only LAN8740A/LAN8742A) */ uint32_t reserved2 : 7; /* Reserved */ }{ ... }; uint32_t val; }{ ... } isfr_reg_t; #define ETH_PHY_ISR_REG_ADDR (0x1D) /** * @brief IMR(Interrupt Mask Register) * *//* ... */ typedef union { struct { uint32_t reserved1 : 1; /* Reserved */ uint32_t auto_nego_page_received : 1; /* Auto-Negotiation Page Received */ uint32_t parallel_detect_fault : 1; /* Parallel Detection Fault */ uint32_t auto_nego_lp_acknowledge : 1; /* Auto-Negotiation LP Acknowledge */ uint32_t link_down : 1; /* Link Down */ uint32_t remote_fault_detect : 1; /* Remote Fault Detect */ uint32_t auto_nego_complete : 1; /* Auto-Negotiation Complete */ uint32_t energy_on_generate : 1; /* ENERGY ON generated */ uint32_t wake_on_lan : 1; /* Wake on Lan (WOL) event detected (only LAN8740A/LAN8742A) */ uint32_t reserved2 : 7; /* Reserved */ }{ ... }; uint32_t val; }{ ... } imr_reg_t; #define ETH_PHY_IMR_REG_ADDR (0x1E) /** * @brief PSCSR(PHY Special Control Status Register) * *//* ... */ typedef union { struct { uint32_t reserved1 : 2; /* Reserved */ uint32_t speed_indication : 3; /* Speed Indication */ uint32_t reserved2 : 1; /* Reserved */ uint32_t enable_4b5b : 1; /* Enable 4B5B encoder (only LAN8740A/LAN8741A) */ uint32_t reserved3 : 5; /* Reserved */ uint32_t auto_nego_done : 1; /* Auto Negotiation Done */ uint32_t reserved4 : 3; /* Reserved */ }{ ... }; 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"); /* link status is forced up because LAN87xx reports link down when loopback is enabled and cable is unplugged */ eth_link_t link; if(bmcr.en_loopback) { link = ETH_LINK_UP; }{...} else { link = bmsr.link_status ? ETH_LINK_UP : ETH_LINK_DOWN; }{...} /* check if link status changed */ if (lan87xx->phy_802_3.link_status != link) { /* when link up, read negotiation result */ 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: //10Base-T half-duplex speed = ETH_SPEED_10M; duplex = ETH_DUPLEX_HALF; break;... case 2: //100Base-TX half-duplex speed = ETH_SPEED_100M; duplex = ETH_DUPLEX_HALF; break;... case 5: //10Base-T full-duplex speed = ETH_SPEED_10M; duplex = ETH_DUPLEX_FULL; break;... case 6: //100Base-TX full-duplex 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 we're in duplex mode, and peer has the flow control ability */ 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); /* Updata information about link, speed, duplex */ 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); /* It was observed that a delay needs to be introduced after setting speed and prior driver's start. Otherwise, the very first read of PHY registers is not valid data (0xFFFF's). *//* ... */ 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); /* Basic PHY init */ ESP_GOTO_ON_ERROR(esp_eth_phy_802_3_basic_phy_init(phy_802_3), err, TAG, "failed to init PHY"); /* Check PHY ID */ 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; // default chip specific configuration 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"); // redefine functions which need to be customized for sake of LAN87xx 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; }{ ... }
Details
Show:
from
Types: Columns:
This file uses the notable symbols shown below. Click anywhere in the file to view more details.