]> git.karo-electronics.de Git - karo-tx-redboot.git/blob - packages/devs/eth/phy/v2_0/src/eth_phy.c
TX53 Release 2011-12-20
[karo-tx-redboot.git] / packages / devs / eth / phy / v2_0 / src / eth_phy.c
1 //==========================================================================
2 //
3 //      dev/eth_phy.c
4 //
5 //      Ethernet transceiver (PHY) support
6 //
7 //==========================================================================
8 //####ECOSGPLCOPYRIGHTBEGIN####
9 // -------------------------------------------
10 // This file is part of eCos, the Embedded Configurable Operating System.
11 // Copyright (C) 2003, 2004 Gary Thomas
12 //
13 // eCos is free software; you can redistribute it and/or modify it under
14 // the terms of the GNU General Public License as published by the Free
15 // Software Foundation; either version 2 or (at your option) any later version.
16 //
17 // eCos is distributed in the hope that it will be useful, but WITHOUT ANY
18 // WARRANTY; without even the implied warranty of MERCHANTABILITY or
19 // FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
20 // for more details.
21 //
22 // You should have received a copy of the GNU General Public License along
23 // with eCos; if not, write to the Free Software Foundation, Inc.,
24 // 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
25 //
26 // As a special exception, if other files instantiate templates or use macros
27 // or inline functions from this file, or you compile this file and link it
28 // with other works to produce a work based on this file, this file does not
29 // by itself cause the resulting work to be covered by the GNU General Public
30 // License. However the source code for this file must still be made available
31 // in accordance with section (3) of the GNU General Public License.
32 //
33 // This exception does not invalidate any other reasons why a work based on
34 // this file might be covered by the GNU General Public License.
35 //
36 // Alternative licenses for eCos may be arranged by contacting Red Hat, Inc.
37 // at http://sources.redhat.com/ecos/ecos-license/
38 // -------------------------------------------
39 //####ECOSGPLCOPYRIGHTEND####
40 //==========================================================================
41 //#####DESCRIPTIONBEGIN####
42 //
43 // Author(s):    gthomas
44 // Contributors:
45 // Date:         2003-08-01
46 // Purpose:
47 // Description:  API support for ethernet PHY
48 //
49 //
50 //####DESCRIPTIONEND####
51 //
52 //==========================================================================
53
54 #include <pkgconf/system.h>
55 #include <pkgconf/io_eth_drivers.h>
56 #include <pkgconf/devs_eth_phy.h>
57 #include <cyg/infra/cyg_type.h>
58
59 #include <cyg/hal/hal_arch.h>
60 #include <cyg/hal/drv_api.h>
61 #include <cyg/hal/hal_if.h>
62 #include <cyg/hal/hal_tables.h>
63
64 #include <cyg/io/eth_phy.h>
65 #include <cyg/io/eth_phy_dev.h>
66
67 // Define table boundaries
68 CYG_HAL_TABLE_BEGIN( __ETH_PHY_TAB__, _eth_phy_devs );
69 CYG_HAL_TABLE_END( __ETH_PHY_TAB_END__, _eth_phy_devs );
70 extern struct _eth_phy_dev_entry __ETH_PHY_TAB__[], __ETH_PHY_TAB_END__;
71
72 // MII interface
73 #define MII_Start                        0x40000000
74 #define MII_Read                         0x20000000
75 #define MII_Write                        0x10000000
76 #define MII_Cmd                          0x30000000
77 #define MII_Phy(phy)             (phy << 23)
78 #define MII_Reg(reg)             (reg << 18)
79 #define MII_TA                           0x00020000
80
81 //
82 // PHY unit access (via MII channel, using bit-level operations)
83 //
84
85 static cyg_uint32
86 phy_cmd(eth_phy_access_t *f, cyg_uint32 cmd)
87 {
88         cyg_uint32  retval;
89         int                     i, off;
90         bool            is_read = (cmd & MII_Cmd) == MII_Read;
91
92         // Set both bits as output
93         f->ops.bit_level_ops.set_dir(1);
94
95         // Preamble
96         for (i = 0; i < 32; i++) {
97                 f->ops.bit_level_ops.set_clock(0);
98                 f->ops.bit_level_ops.set_data(1);
99                 CYGACC_CALL_IF_DELAY_US(1);
100                 f->ops.bit_level_ops.set_clock(1);
101                 CYGACC_CALL_IF_DELAY_US(1);
102         }
103
104         // Command/data
105         for (i = 0, off = 31; i < (is_read ? 14 : 32); i++, --off) {
106                 f->ops.bit_level_ops.set_clock(0);
107                 f->ops.bit_level_ops.set_data((cmd >> off) & 0x00000001);
108                 CYGACC_CALL_IF_DELAY_US(1);
109                 f->ops.bit_level_ops.set_clock(1);
110                 CYGACC_CALL_IF_DELAY_US(1);
111         }
112
113         retval = cmd;
114
115         // If read, fetch data register
116         if (is_read) {
117                 retval >>= 16;
118
119                 f->ops.bit_level_ops.set_clock(0);
120                 f->ops.bit_level_ops.set_dir(0);
121                 CYGACC_CALL_IF_DELAY_US(1);
122                 f->ops.bit_level_ops.set_clock(1);
123                 CYGACC_CALL_IF_DELAY_US(1);
124                 f->ops.bit_level_ops.set_clock(0);
125                 CYGACC_CALL_IF_DELAY_US(1);
126
127                 for (i = 0, off = 15; i < 16; i++, off--) {
128                         f->ops.bit_level_ops.set_clock(1);
129                         retval <<= 1;
130                         retval |= f->ops.bit_level_ops.get_data();
131                         CYGACC_CALL_IF_DELAY_US(1);
132                         f->ops.bit_level_ops.set_clock(0);
133                         CYGACC_CALL_IF_DELAY_US(1);
134                 }
135         }
136
137         // Set both bits as output
138         f->ops.bit_level_ops.set_dir(1);
139
140         // Postamble
141         for (i = 0; i < 32; i++) {
142                 f->ops.bit_level_ops.set_clock(0);
143                 f->ops.bit_level_ops.set_data(1);
144                 CYGACC_CALL_IF_DELAY_US(1);
145                 f->ops.bit_level_ops.set_clock(1);
146                 CYGACC_CALL_IF_DELAY_US(1);
147         }
148
149         return retval;
150 }
151
152 externC bool
153 _eth_phy_init(eth_phy_access_t *f)
154 {
155         int addr;
156         unsigned short state;
157         unsigned long id = 0;
158         struct _eth_phy_dev_entry *dev;
159
160         if (f->init_done) return true;
161         f->init();
162         // Scan to determine PHY address
163         f->init_done = true;
164         for (addr = 0;  addr < 0x20;  addr++) {
165                 if (_eth_phy_read(f, PHY_ID1, addr, &state)) {
166                         id = state << 16;
167                         if (_eth_phy_read(f, PHY_ID2, addr, &state)) {
168                                 id |= state;
169                                 f->phy_addr = addr;
170                                 for (dev = __ETH_PHY_TAB__; dev != &__ETH_PHY_TAB_END__;  dev++) {
171                                         if (dev->id == id) {
172                                                 eth_phy_printf("PHY: %s\n", dev->name);
173                                                 f->dev = dev;
174                                                 return true;
175                                         }
176                                 }
177                         }
178                 }
179         }
180         if (addr >= 0x20) {
181                 // Can't handle this PHY
182                 eth_phy_printf("Unsupported PHY device - id: %lx\n", id);
183         }
184         f->init_done = false;
185         return false;
186 }
187
188 externC void
189 _eth_phy_reset(eth_phy_access_t *f)
190 {
191         if (!f->init_done) {
192                 eth_phy_printf("PHY reset without init on PHY: %p\n", f);
193                 return;
194         }
195         f->init();
196 }
197
198 externC void
199 _eth_phy_write(eth_phy_access_t *f, int reg, int addr, unsigned short data)
200 {
201         if (!f->init_done) {
202                 eth_phy_printf("PHY write without init on PHY: %p\n", f);
203                 return;
204         }
205         if (f->ops_type == PHY_BIT_LEVEL_ACCESS_TYPE) {
206                 phy_cmd(f, MII_Start | MII_Write | MII_Phy(addr) | MII_Reg(reg) | MII_TA | data);
207         } else {
208                 f->ops.reg_level_ops.put_reg(reg, addr, data);
209         }
210 }
211
212 externC bool
213 _eth_phy_read(eth_phy_access_t *f, int reg, int addr, unsigned short *val)
214 {
215         cyg_uint32 ret;
216
217         if (!f->init_done) {
218                 eth_phy_printf("PHY read without init on PHY: %p\n", f);
219                 return false;
220         }
221         if (f->ops_type == PHY_BIT_LEVEL_ACCESS_TYPE) {
222                 ret = phy_cmd(f, MII_Start | MII_Read | MII_Phy(addr) | MII_Reg(reg) | MII_TA);
223                 *val = ret;
224                 return true;
225         } else {
226                 return f->ops.reg_level_ops.get_reg(reg, addr, val);
227         }
228 }
229
230 externC int
231 _eth_phy_cfg(eth_phy_access_t *f, int mode)
232 {
233         int phy_timeout = 5 * 1000;  // Wait 5 seconds max for link to clear
234         bool phy_ok;
235         unsigned short reset_mode, phy_state;
236         int i;
237
238         if (!f->init_done) {
239                 eth_phy_printf("PHY config without init on PHY: %p\n", f);
240                 return 0;
241         }
242
243         // Reset PHY (transceiver)
244         phy_ok = false;
245         _eth_phy_reset(f);
246
247         _eth_phy_write(f, PHY_BMCR, f->phy_addr, PHY_BMCR_RESET);
248         for (i = 0;  i < 5 * 100;  i++) {
249                 phy_ok = _eth_phy_read(f, PHY_BMCR, f->phy_addr, &phy_state);
250                 eth_phy_printf("PHY: %04x\n", phy_state);
251                 if (phy_ok && !(phy_state & PHY_BMCR_RESET)) break;
252                 CYGACC_CALL_IF_DELAY_US(10000);   // 10ms
253         }
254         if (!phy_ok || (phy_state & PHY_BMCR_RESET)) {
255                 eth_phy_printf("Can't get PHY unit to soft reset: %x\n", phy_state);
256                 return 0;
257         }
258
259         reset_mode = PHY_BMCR_RESTART | PHY_BMCR_AUTO_NEG;
260         _eth_phy_write(f, PHY_BMCR, f->phy_addr, reset_mode);
261         while (phy_timeout-- >= 0) {
262                 phy_ok = _eth_phy_read(f, PHY_BMSR, f->phy_addr, &phy_state);
263                 if (phy_ok && (phy_state & PHY_BMSR_AUTO_NEG)) {
264                         break;
265                 } else {
266                         CYGACC_CALL_IF_DELAY_US(10000);   // 10ms
267                 }
268         }
269         if (phy_timeout <= 0) {
270                 eth_phy_printf("**Warning: PHY LINK UP failed: %04x\n", phy_state);
271                 return 0;
272         }
273
274         return _eth_phy_state(f);
275 }
276
277 externC int
278 _eth_phy_state(eth_phy_access_t *f)
279 {
280         int state = 0;
281
282         if (!f->init_done) {
283                 eth_phy_printf("PHY state without init on PHY: %p\n", f);
284                 return 0;
285         }
286         if (f->dev->stat(f, &state)) {
287                 return state;
288         } else {
289                 return 0;
290         }
291         return state;
292 }