1*3aa5a3aeSriastradh /* $NetBSD: if_le_oioc.c,v 1.6 2022/02/12 03:24:34 riastradh Exp $ */
2b610ccb0Srumble
3b610ccb0Srumble /*
4b610ccb0Srumble * Copyright (c) 2009 Stephen M. Rumble
5b610ccb0Srumble * All rights reserved.
6b610ccb0Srumble *
7b610ccb0Srumble * Redistribution and use in source and binary forms, with or without
8b610ccb0Srumble * modification, are permitted provided that the following conditions
9b610ccb0Srumble * are met:
10b610ccb0Srumble * 1. Redistributions of source code must retain the above copyright
11b610ccb0Srumble * notice, this list of conditions and the following disclaimer.
12b610ccb0Srumble * 2. The name of the author may not be used to endorse or promote products
13b610ccb0Srumble * derived from this software without specific prior written permission.
14b610ccb0Srumble *
15b610ccb0Srumble * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
16b610ccb0Srumble * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
17b610ccb0Srumble * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
18b610ccb0Srumble * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
19b610ccb0Srumble * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
20b610ccb0Srumble * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21b610ccb0Srumble * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22b610ccb0Srumble * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23b610ccb0Srumble * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
24b610ccb0Srumble * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25b610ccb0Srumble */
26b610ccb0Srumble
27b610ccb0Srumble #include <sys/cdefs.h>
28*3aa5a3aeSriastradh __KERNEL_RCSID(0, "$NetBSD: if_le_oioc.c,v 1.6 2022/02/12 03:24:34 riastradh Exp $");
29b610ccb0Srumble
30b610ccb0Srumble #include "opt_inet.h"
31b610ccb0Srumble
32b610ccb0Srumble #include <sys/param.h>
33b610ccb0Srumble #include <sys/systm.h>
34b610ccb0Srumble #include <sys/mbuf.h>
35b610ccb0Srumble #include <sys/syslog.h>
36b610ccb0Srumble #include <sys/socket.h>
37b610ccb0Srumble #include <sys/device.h>
38b610ccb0Srumble
39b610ccb0Srumble #include <uvm/uvm.h> // for uvm_pglistalloc
40b610ccb0Srumble
41b610ccb0Srumble #include <net/if.h>
42b610ccb0Srumble #include <net/if_ether.h>
43b610ccb0Srumble #include <net/if_media.h>
44b610ccb0Srumble
45b610ccb0Srumble #ifdef INET
46b610ccb0Srumble #include <netinet/in.h>
47b610ccb0Srumble #include <netinet/if_inarp.h>
48b610ccb0Srumble #endif
49b610ccb0Srumble
50b610ccb0Srumble #include <machine/cpu.h>
51b610ccb0Srumble #include <machine/machtype.h>
52b610ccb0Srumble
53b610ccb0Srumble #include <dev/ic/lancereg.h>
54b610ccb0Srumble #include <dev/ic/lancevar.h>
55b610ccb0Srumble #include <dev/ic/am7990reg.h>
56b610ccb0Srumble #include <dev/ic/am7990var.h>
57b610ccb0Srumble
58b610ccb0Srumble #include <dev/arcbios/arcbios.h>
59b610ccb0Srumble #include <dev/arcbios/arcbiosvar.h>
60b610ccb0Srumble
61b610ccb0Srumble #include <sgimips/ioc/oiocvar.h>
62b610ccb0Srumble #include <sgimips/ioc/oiocreg.h>
63b610ccb0Srumble
64b610ccb0Srumble #include <mips/include/cache.h>
65b610ccb0Srumble
66b610ccb0Srumble #ifndef OIOC_LANCE_NPAGES
67b610ccb0Srumble #define OIOC_LANCE_NPAGES 64 /* 256KB */
68b610ccb0Srumble #endif
69b610ccb0Srumble
70b610ccb0Srumble #if OIOC_LANCE_NPAGES > OIOC_ENET_NPGMAPS
71b610ccb0Srumble #error OIOC_LANCE_NPAGES > OIOC_ENET_NPGMAPS (512)
72b610ccb0Srumble #endif
73b610ccb0Srumble
74b610ccb0Srumble /*
75b610ccb0Srumble * Ethernet software status per interface.
76b610ccb0Srumble * The real stuff is in dev/ic/am7990var.h
77b610ccb0Srumble * The really real stuff is in dev/ic/lancevar.h
78b610ccb0Srumble *
79b610ccb0Srumble * Lance is somewhat nasty MI code. We basically get:
80b610ccb0Srumble * struct le_softc {
81b610ccb0Srumble * struct am7990_softc {
82b610ccb0Srumble * struct lance_softc {
83b610ccb0Srumble * device_t sc_dev;
84b610ccb0Srumble * ...
85b610ccb0Srumble * }
86b610ccb0Srumble * }
87b610ccb0Srumble *
88b610ccb0Srumble * bus_space_tag ...
89b610ccb0Srumble * }
90b610ccb0Srumble *
91*3aa5a3aeSriastradh * So, we can cast any three to any other three, plus
92*3aa5a3aeSriastradh * device_private(sc_dev) points back at the top (i.e. to le_softc,
93*3aa5a3aeSriastradh * am7990_softc and lance_softc). Bloody hell!
94b610ccb0Srumble */
95b610ccb0Srumble struct le_softc {
96b610ccb0Srumble struct am7990_softc sc_am7990; /* glue to MI code */
97b610ccb0Srumble
98b610ccb0Srumble bus_space_tag_t sc_st;
99b610ccb0Srumble bus_space_handle_t sc_maph; /* ioc<->lance page map regs */
100b610ccb0Srumble bus_space_handle_t sc_rdph; /* lance rdp */
101b610ccb0Srumble bus_space_handle_t sc_raph; /* lance rap */
102b610ccb0Srumble };
103b610ccb0Srumble
104b610ccb0Srumble static int le_match(device_t, cfdata_t, void *);
105b610ccb0Srumble static void le_attach(device_t, device_t, void *);
106b610ccb0Srumble
107b610ccb0Srumble CFATTACH_DECL_NEW(le, sizeof(struct le_softc),
108b610ccb0Srumble le_match, le_attach, NULL, NULL);
109b610ccb0Srumble
110b610ccb0Srumble #if defined(_KERNEL_OPT)
111b610ccb0Srumble #include "opt_ddb.h"
112b610ccb0Srumble #endif
113b610ccb0Srumble
114b610ccb0Srumble static void lewrcsr(struct lance_softc *, uint16_t, uint16_t);
115b610ccb0Srumble static uint16_t lerdcsr(struct lance_softc *, uint16_t);
116b610ccb0Srumble
117b610ccb0Srumble static void
lewrcsr(struct lance_softc * sc,uint16_t port,uint16_t val)118b610ccb0Srumble lewrcsr(struct lance_softc *sc, uint16_t port, uint16_t val)
119b610ccb0Srumble {
120b610ccb0Srumble struct le_softc *lesc = (struct le_softc *)sc;
121b610ccb0Srumble bus_space_write_2(lesc->sc_st, lesc->sc_raph, 0, port);
122b610ccb0Srumble bus_space_write_2(lesc->sc_st, lesc->sc_rdph, 0, val);
123b610ccb0Srumble }
124b610ccb0Srumble
125b610ccb0Srumble static uint16_t
lerdcsr(struct lance_softc * sc,uint16_t port)126b610ccb0Srumble lerdcsr(struct lance_softc *sc, uint16_t port)
127b610ccb0Srumble {
128b610ccb0Srumble struct le_softc *lesc = (struct le_softc *)sc;
129b610ccb0Srumble bus_space_write_2(lesc->sc_st, lesc->sc_raph, 0, port);
130b610ccb0Srumble return (bus_space_read_2(lesc->sc_st, lesc->sc_rdph, 0));
131b610ccb0Srumble }
132b610ccb0Srumble
133b610ccb0Srumble /*
134b610ccb0Srumble * Always present on IP6 and IP10. IP4? Unknown.
135b610ccb0Srumble */
136b610ccb0Srumble int
le_match(device_t parent,cfdata_t cf,void * aux)137b610ccb0Srumble le_match(device_t parent, cfdata_t cf, void *aux)
138b610ccb0Srumble {
139b610ccb0Srumble struct oioc_attach_args *oa = aux;
140b610ccb0Srumble
141b610ccb0Srumble if (mach_type == MACH_SGI_IP4)
142b610ccb0Srumble return (0);
143b610ccb0Srumble
144b610ccb0Srumble if (strcmp(oa->oa_name, cf->cf_name) == 0)
145b610ccb0Srumble return (1);
146b610ccb0Srumble
147b610ccb0Srumble return (0);
148b610ccb0Srumble }
149b610ccb0Srumble
150b610ccb0Srumble void
le_attach(device_t parent,device_t self,void * aux)151b610ccb0Srumble le_attach(device_t parent, device_t self, void *aux)
152b610ccb0Srumble {
153b610ccb0Srumble struct le_softc *lesc = device_private(self);
154b610ccb0Srumble struct lance_softc *sc = &lesc->sc_am7990.lsc;
155b610ccb0Srumble struct oioc_attach_args *oa = aux;
156b610ccb0Srumble struct pglist mlist;
157b610ccb0Srumble const char *enaddrstr;
1583ce80fd7Stsutsui char enaddr[ETHER_ADDR_LEN];
159b610ccb0Srumble char pbuf[9];
160b610ccb0Srumble int i, error;
161b610ccb0Srumble
162b610ccb0Srumble sc->sc_dev = self;
163b610ccb0Srumble lesc->sc_st = oa->oa_st;
164b610ccb0Srumble
16523347d39Smatt enaddrstr = arcbios_GetEnvironmentVariable("eaddr");
166b610ccb0Srumble if (enaddrstr == NULL) {
167b610ccb0Srumble aprint_error(": failed to obtain MAC address\n");
168b610ccb0Srumble return;
169b610ccb0Srumble }
170b610ccb0Srumble
171b610ccb0Srumble if ((error = bus_space_subregion(oa->oa_st, oa->oa_sh, OIOC_LANCE_RDP,
172b610ccb0Srumble OIOC_LANCE_RDP_SIZE, &lesc->sc_rdph)) != 0) {
173b610ccb0Srumble printf(": unable to map rdp reg, error=%d\n", error);
174b610ccb0Srumble goto fail_0;
175b610ccb0Srumble }
176b610ccb0Srumble
177b610ccb0Srumble if ((error = bus_space_subregion(oa->oa_st, oa->oa_sh, OIOC_LANCE_RAP,
178b610ccb0Srumble OIOC_LANCE_RAP_SIZE, &lesc->sc_raph)) != 0) {
179b610ccb0Srumble printf(": unable to map rap reg, error=%d\n", error);
180b610ccb0Srumble goto fail_1;
181b610ccb0Srumble }
182b610ccb0Srumble
183b610ccb0Srumble if ((error = bus_space_subregion(oa->oa_st, oa->oa_sh,
184b610ccb0Srumble OIOC_ENET_PGMAP_BASE, OIOC_ENET_PGMAP_SIZE, &lesc->sc_maph)) != 0) {
185b610ccb0Srumble printf(": unable to map rap reg, error=%d\n", error);
186b610ccb0Srumble goto fail_2;
187b610ccb0Srumble }
188b610ccb0Srumble
189b610ccb0Srumble /* Allocate a contiguous chunk of physical memory for the le buffer. */
190b610ccb0Srumble error = uvm_pglistalloc(OIOC_LANCE_NPAGES * PAGE_SIZE,
1919b83c0f5Smatt pmap_limits.avail_start, pmap_limits.avail_end, PAGE_SIZE, 0,
1929b83c0f5Smatt &mlist, 1, 0);
193b610ccb0Srumble if (error) {
194b610ccb0Srumble aprint_error(": failed to allocate ioc<->lance buffer space, "
195b610ccb0Srumble "error = %d\n", error);
196b610ccb0Srumble goto fail_3;
197b610ccb0Srumble }
198b610ccb0Srumble
199b610ccb0Srumble /* Use IOC to map the physical memory into the Ethernet chip's space. */
200b610ccb0Srumble for (i = 0; i < OIOC_LANCE_NPAGES; i++) {
201b610ccb0Srumble bus_space_write_2(lesc->sc_st,lesc->sc_maph,
202b610ccb0Srumble OIOC_ENET_PGMAP_OFF(i),
203b610ccb0Srumble (VM_PAGE_TO_PHYS(mlist.tqh_first) >> PAGE_SHIFT) + i);
204b610ccb0Srumble }
205b610ccb0Srumble
206b610ccb0Srumble sc->sc_mem = (void *)MIPS_PHYS_TO_KSEG1(
207b610ccb0Srumble (uint32_t)VM_PAGE_TO_PHYS(mlist.tqh_first));
208b610ccb0Srumble sc->sc_memsize = OIOC_LANCE_NPAGES * PAGE_SIZE;
209b610ccb0Srumble sc->sc_addr = 0;
210b610ccb0Srumble sc->sc_conf3 = LE_C3_BSWP;
211b610ccb0Srumble
2123ce80fd7Stsutsui ether_aton_r(enaddr, sizeof(enaddr), enaddrstr);
213b610ccb0Srumble memcpy(sc->sc_enaddr, enaddr, sizeof(sc->sc_enaddr));
214b610ccb0Srumble
215b610ccb0Srumble if (cpu_intr_establish(oa->oa_irq, IPL_NET, am7990_intr, sc) == NULL) {
216b610ccb0Srumble aprint_error(": failed to establish interrupt %d\n",oa->oa_irq);
217b610ccb0Srumble goto fail_4;
218b610ccb0Srumble }
219b610ccb0Srumble
220b610ccb0Srumble sc->sc_copytodesc = lance_copytobuf_contig;
221b610ccb0Srumble sc->sc_copyfromdesc = lance_copyfrombuf_contig;
222b610ccb0Srumble sc->sc_copytobuf = lance_copytobuf_contig;
223b610ccb0Srumble sc->sc_copyfrombuf = lance_copyfrombuf_contig;
224b610ccb0Srumble sc->sc_zerobuf = lance_zerobuf_contig;
225b610ccb0Srumble
226b610ccb0Srumble sc->sc_rdcsr = lerdcsr;
227b610ccb0Srumble sc->sc_wrcsr = lewrcsr;
228b610ccb0Srumble sc->sc_hwinit = NULL;
229b610ccb0Srumble
230b610ccb0Srumble format_bytes(pbuf, sizeof(pbuf), OIOC_LANCE_NPAGES * PAGE_SIZE);
231b610ccb0Srumble aprint_normal(": main memory used = %s\n", pbuf);
232b610ccb0Srumble aprint_normal("%s", device_xname(self));
233b610ccb0Srumble
234b610ccb0Srumble am7990_config(&lesc->sc_am7990);
235b610ccb0Srumble
236b610ccb0Srumble return;
237b610ccb0Srumble
238b610ccb0Srumble fail_4:
239b610ccb0Srumble uvm_pglistfree(&mlist);
240b610ccb0Srumble fail_3:
241b610ccb0Srumble bus_space_unmap(oa->oa_st, oa->oa_sh, lesc->sc_maph);
242b610ccb0Srumble fail_2:
243b610ccb0Srumble bus_space_unmap(oa->oa_st, oa->oa_sh, lesc->sc_raph);
244b610ccb0Srumble fail_1:
245b610ccb0Srumble bus_space_unmap(oa->oa_st, oa->oa_sh, lesc->sc_rdph);
246b610ccb0Srumble fail_0:
247b610ccb0Srumble return;
248b610ccb0Srumble }
249