]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - sys/dev/ed/if_ed_rtl80x9.c
FCP-101: Remove de(4).
[FreeBSD/FreeBSD.git] / sys / dev / ed / if_ed_rtl80x9.c
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2003, David Madole
5  * All rights reserved.
6  * Copyright (c) 2005, M. Warner Losh.
7  * All rights reserved.
8  *
9  * Redistribution and use in source and binary forms, with or without
10  * modification, are permitted provided that the following conditions
11  * are met:
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice unmodified, this list of conditions, and the following
14  *    disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
23  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29  * SUCH DAMAGE.
30  *
31  * Based on patches subitted by: David Madole, edited by M. Warner Losh.
32  */
33
34
35 #include <sys/cdefs.h>
36 __FBSDID("$FreeBSD$");
37
38 #include "opt_ed.h"
39
40 #include <sys/param.h>
41 #include <sys/systm.h>
42 #include <sys/sockio.h>
43 #include <sys/mbuf.h>
44 #include <sys/kernel.h>
45 #include <sys/socket.h>
46 #include <sys/syslog.h>
47
48 #include <sys/bus.h>
49
50 #include <machine/bus.h>
51 #include <sys/rman.h>
52 #include <machine/resource.h>
53
54 #include <net/ethernet.h>
55 #include <net/if.h>
56 #include <net/if_var.h>
57 #include <net/if_arp.h>
58 #include <net/if_dl.h>
59 #include <net/if_mib.h>
60 #include <net/if_media.h>
61
62 #include <net/bpf.h>
63
64 #include <dev/ed/if_edreg.h>
65 #include <dev/ed/if_edvar.h>
66 #include <dev/ed/rtl80x9reg.h>
67
68 static int      ed_rtl_set_media(struct ifnet *ifp);
69 static void     ed_rtl_get_media(struct ifnet *ifp, struct ifmediareq *);
70
71 static int
72 ed_rtl80x9_media_ioctl(struct ed_softc *sc, struct ifreq *ifr, u_long command)
73 {
74         return (ifmedia_ioctl(sc->ifp, ifr, &sc->ifmedia, command));
75 }
76
77 int
78 ed_probe_RTL80x9(device_t dev, int port_rid, int flags)
79 {
80         struct ed_softc *sc = device_get_softc(dev);
81         char *ts;
82         int error;
83
84         if ((error = ed_alloc_port(dev, port_rid, ED_NOVELL_IO_PORTS)))
85                 return (error);
86         
87         sc->asic_offset = ED_NOVELL_ASIC_OFFSET;
88         sc->nic_offset  = ED_NOVELL_NIC_OFFSET;
89
90         if (ed_nic_inb(sc, ED_P0_CR) & (ED_CR_PS0 | ED_CR_PS1))
91                 ed_nic_outb(sc, ED_P0_CR, ED_CR_RD2 | ED_CR_STP);
92
93         if (ed_nic_inb(sc, ED_RTL80X9_80X9ID0) != ED_RTL80X9_ID0)
94                 return (ENXIO);
95
96         switch (ed_nic_inb(sc, ED_RTL80X9_80X9ID1)) {
97         case ED_RTL8019_ID1:
98                 sc->chip_type = ED_CHIP_TYPE_RTL8019;
99                 ts = "RTL8019";
100                 break;
101         case ED_RTL8029_ID1:
102                 sc->chip_type = ED_CHIP_TYPE_RTL8029;
103                 ts = "RTL8029";
104                 break;
105         default:
106                 return (ENXIO);
107         }
108
109         if ((error = ed_probe_Novell_generic(dev, flags)))
110                 return (error);
111
112         sc->type_str = ts;
113         sc->sc_media_ioctl = &ed_rtl80x9_media_ioctl;
114         ifmedia_init(&sc->ifmedia, 0, ed_rtl_set_media, ed_rtl_get_media);
115
116         ifmedia_add(&sc->ifmedia, IFM_ETHER | IFM_10_T | IFM_FDX, 0, 0);
117         ifmedia_add(&sc->ifmedia, IFM_ETHER | IFM_10_T, 0, 0);
118         ifmedia_add(&sc->ifmedia, IFM_ETHER | IFM_10_2, 0, 0);
119         ifmedia_add(&sc->ifmedia, IFM_ETHER | IFM_10_5, 0, 0);
120         ifmedia_add(&sc->ifmedia, IFM_ETHER | IFM_AUTO, 0, 0);
121
122         ed_nic_barrier(sc, ED_P0_CR, 1,
123             BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE);
124         ed_nic_outb(sc, ED_P0_CR, ED_CR_RD2 | ED_CR_PAGE_3 | ED_CR_STP);
125         ed_nic_barrier(sc, ED_P0_CR, 1,
126             BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE);
127
128         switch (ed_nic_inb(sc, ED_RTL80X9_CONFIG2) & ED_RTL80X9_CF2_MEDIA) {
129         case ED_RTL80X9_CF2_AUTO:
130                 ifmedia_set(&sc->ifmedia, IFM_ETHER | IFM_AUTO);
131                 break;
132         case ED_RTL80X9_CF2_10_5:
133                 ifmedia_set(&sc->ifmedia, IFM_ETHER | IFM_10_5);
134                 break;
135         case ED_RTL80X9_CF2_10_2:
136                 ifmedia_set(&sc->ifmedia, IFM_ETHER | IFM_10_2);
137                 break;
138         case ED_RTL80X9_CF2_10_T:
139                 ifmedia_set(&sc->ifmedia, IFM_ETHER | IFM_10_T |
140                     ((ed_nic_inb(sc, ED_RTL80X9_CONFIG3)
141                     & ED_RTL80X9_CF3_FUDUP) ? IFM_FDX : 0));
142                 break;
143         }
144         return (0);
145 }
146
147 static int
148 ed_rtl_set_media(struct ifnet *ifp)
149 {
150         struct ed_softc *sc;
151
152         sc = ifp->if_softc;
153         ED_LOCK(sc);
154         ed_nic_barrier(sc, ED_P0_CR, 1,
155             BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE);
156         ed_nic_outb(sc, ED_P0_CR, sc->cr_proto | ED_CR_PAGE_3
157                 | (ed_nic_inb(sc, ED_P0_CR) & (ED_CR_STA | ED_CR_STP)));
158         ed_nic_barrier(sc, ED_P0_CR, 1,
159             BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE);
160
161         switch(IFM_SUBTYPE(sc->ifmedia.ifm_cur->ifm_media)) {
162         case IFM_10_T:
163                 ed_nic_outb(sc, ED_RTL80X9_CONFIG2, ED_RTL80X9_CF2_10_T
164                         | (ed_nic_inb(sc, ED_RTL80X9_CONFIG2)
165                                 & ~ED_RTL80X9_CF2_MEDIA));
166                 break;
167         case IFM_10_2:
168                 ed_nic_outb(sc, ED_RTL80X9_CONFIG2, ED_RTL80X9_CF2_10_2
169                         | (ed_nic_inb(sc, ED_RTL80X9_CONFIG2)
170                                 & ~ED_RTL80X9_CF2_MEDIA));
171                 break;
172         case IFM_10_5:
173                 ed_nic_outb(sc, ED_RTL80X9_CONFIG2, ED_RTL80X9_CF2_10_5
174                         | (ed_nic_inb(sc, ED_RTL80X9_CONFIG2)
175                                 & ~ED_RTL80X9_CF2_MEDIA));
176                 break;
177         case IFM_AUTO:
178                 ed_nic_outb(sc, ED_RTL80X9_CONFIG2, ED_RTL80X9_CF2_AUTO
179                         | (ed_nic_inb(sc, ED_RTL80X9_CONFIG2)
180                                 & ~ED_RTL80X9_CF2_MEDIA));
181                 break;
182         }
183         ed_nic_outb(sc, ED_RTL80X9_CONFIG3,
184                 (sc->ifmedia.ifm_cur->ifm_media & IFM_FDX) ?
185                 (ed_nic_inb(sc, ED_RTL80X9_CONFIG3) | ED_RTL80X9_CF3_FUDUP) :
186                 (ed_nic_inb(sc, ED_RTL80X9_CONFIG3) & ~ED_RTL80X9_CF3_FUDUP));
187
188         ED_UNLOCK(sc);
189         return (0);
190 }
191
192 static void
193 ed_rtl_get_media(struct ifnet *ifp, struct ifmediareq *imr)
194 {
195         struct ed_softc *sc;
196
197         sc = ifp->if_softc;
198         imr->ifm_active = sc->ifmedia.ifm_cur->ifm_media;
199
200
201         if (IFM_SUBTYPE(imr->ifm_active) == IFM_AUTO) {
202                 ED_LOCK(sc);
203                 ed_nic_barrier(sc, ED_P0_CR, 1,
204                     BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE);
205                 ed_nic_outb(sc, ED_P0_CR, sc->cr_proto | ED_CR_PAGE_3 |
206                         (ed_nic_inb(sc, ED_P0_CR) & (ED_CR_STA | ED_CR_STP)));
207                 ed_nic_barrier(sc, ED_P0_CR, 1,
208                     BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE);
209
210                 switch (ed_nic_inb(sc, ED_RTL80X9_CONFIG0)
211                                 & (sc->chip_type == ED_CHIP_TYPE_RTL8029 ? ED_RTL80X9_CF0_BNC
212                                 : (ED_RTL80X9_CF0_AUI | ED_RTL80X9_CF0_BNC))) {
213                 case ED_RTL80X9_CF0_BNC:
214                         imr->ifm_active |= IFM_10_2;
215                         break;
216                 case ED_RTL80X9_CF0_AUI:
217                         imr->ifm_active |= IFM_10_5;
218                         break;
219                 default:
220                         imr->ifm_active |= IFM_10_T;
221                         break;
222                 }
223                 ED_UNLOCK(sc);
224         }
225         imr->ifm_status = 0;
226 }
227