1f572e449SJoerg Sonnenberger /*
2f572e449SJoerg Sonnenberger * Copyright (c) 2002 Takeshi Shibagaki
3f572e449SJoerg Sonnenberger * All rights reserved.
4f572e449SJoerg Sonnenberger *
5f572e449SJoerg Sonnenberger * Redistribution and use in source and binary forms, with or without
6f572e449SJoerg Sonnenberger * modification, are permitted provided that the following conditions
7f572e449SJoerg Sonnenberger * are met:
8f572e449SJoerg Sonnenberger * 1. Redistributions of source code must retain the above copyright
9f572e449SJoerg Sonnenberger * notice, this list of conditions and the following disclaimer.
10f572e449SJoerg Sonnenberger * 2. Redistributions in binary form must reproduce the above copyright
11f572e449SJoerg Sonnenberger * notice, this list of conditions and the following disclaimer in the
12f572e449SJoerg Sonnenberger * documentation and/or other materials provided with the distribution.
13f572e449SJoerg Sonnenberger *
14f572e449SJoerg Sonnenberger * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS AS IS'' AND
15f572e449SJoerg Sonnenberger * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16f572e449SJoerg Sonnenberger * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17f572e449SJoerg Sonnenberger * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18f572e449SJoerg Sonnenberger * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19f572e449SJoerg Sonnenberger * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20f572e449SJoerg Sonnenberger * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21f572e449SJoerg Sonnenberger * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22f572e449SJoerg Sonnenberger * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23f572e449SJoerg Sonnenberger * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24f572e449SJoerg Sonnenberger * SUCH DAMAGE.
25f572e449SJoerg Sonnenberger *
26f572e449SJoerg Sonnenberger * xe pccard interface driver
27f572e449SJoerg Sonnenberger *
287261a835SSepherosa Ziehau * $FreeBSD: src/sys/dev/xe/if_xe_pccard.c,v 1.11 2003/10/14 22:51:35 rsm Exp $
29f572e449SJoerg Sonnenberger */
30f572e449SJoerg Sonnenberger
31f572e449SJoerg Sonnenberger #include <sys/param.h>
32f572e449SJoerg Sonnenberger #include <sys/systm.h>
33f572e449SJoerg Sonnenberger #include <sys/kernel.h>
34f572e449SJoerg Sonnenberger #include <sys/socket.h>
35f572e449SJoerg Sonnenberger #include <sys/module.h>
36f572e449SJoerg Sonnenberger #include <sys/bus.h>
37f572e449SJoerg Sonnenberger #include <sys/rman.h>
38f572e449SJoerg Sonnenberger
39f572e449SJoerg Sonnenberger #include <net/ethernet.h>
40f572e449SJoerg Sonnenberger #include <net/if.h>
41f572e449SJoerg Sonnenberger #include <net/if_arp.h>
42f572e449SJoerg Sonnenberger #include <net/if_media.h>
43f572e449SJoerg Sonnenberger #include <net/if_mib.h>
44f572e449SJoerg Sonnenberger
45f572e449SJoerg Sonnenberger #include <bus/pccard/pccardvar.h>
46*dcb4b80dSSascha Wildner
47f572e449SJoerg Sonnenberger #include "card_if.h"
48*dcb4b80dSSascha Wildner #include "pccarddevs.h"
49f572e449SJoerg Sonnenberger
50f572e449SJoerg Sonnenberger #include "if_xereg.h"
51f572e449SJoerg Sonnenberger #include "if_xevar.h"
52f572e449SJoerg Sonnenberger
537261a835SSepherosa Ziehau #define XE_DEBUG
547261a835SSepherosa Ziehau
557261a835SSepherosa Ziehau #ifdef XE_DEBUG
567261a835SSepherosa Ziehau #define DEVPRINTF(level, arg) if (xe_debug >= (level)) device_printf arg
577261a835SSepherosa Ziehau #else
587261a835SSepherosa Ziehau #define DEVPRINTF(level, arg)
597261a835SSepherosa Ziehau #endif
607261a835SSepherosa Ziehau
61f572e449SJoerg Sonnenberger static const struct pccard_product xe_pccard_products[] = {
623aef8050SSepherosa Ziehau PCMCIA_CARD(COMPAQ, CPQ550, 0),
63f572e449SJoerg Sonnenberger PCMCIA_CARD(COMPAQ2, CPQ_10_100, 0),
64f572e449SJoerg Sonnenberger PCMCIA_CARD(INTEL, EEPRO100, 0),
653aef8050SSepherosa Ziehau PCMCIA_CARD(RACORE, ACCTON_EN2226, 0),
66f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CE, 0),
67f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CE2, 0),
68f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CE3, 0),
69f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CEM, 0),
70f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CEM28, 0),
71f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CEM33, 0),
72f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CEM56, 0),
73f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, REM56, 0),
74f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CNW_801, 0),
75f572e449SJoerg Sonnenberger PCMCIA_CARD(XIRCOM, CNW_802, 0),
76f572e449SJoerg Sonnenberger { NULL }
77f572e449SJoerg Sonnenberger };
78f572e449SJoerg Sonnenberger
79f572e449SJoerg Sonnenberger struct xe_vendor {
80f572e449SJoerg Sonnenberger uint32_t vendor_id;
81f572e449SJoerg Sonnenberger const char *vendor_desc;
82f572e449SJoerg Sonnenberger } xe_vendors[] = {
83f572e449SJoerg Sonnenberger { PCMCIA_VENDOR_XIRCOM, "Xircom" },
84f572e449SJoerg Sonnenberger { PCMCIA_VENDOR_COMPAQ, "Compaq" },
85f572e449SJoerg Sonnenberger { PCMCIA_VENDOR_COMPAQ2, "Compaq" },
86f572e449SJoerg Sonnenberger { PCMCIA_VENDOR_INTEL, "Intel" },
87f572e449SJoerg Sonnenberger { 0, "Unknown" }
88f572e449SJoerg Sonnenberger };
89f572e449SJoerg Sonnenberger
90f572e449SJoerg Sonnenberger #define XE_CARD_TYPE_FLAGS_NO 0x0
91f572e449SJoerg Sonnenberger #define XE_CARD_TYPE_FLAGS_CE2 0x1
92f572e449SJoerg Sonnenberger #define XE_CARD_TYPE_FLAGS_MOHAWK 0x2
93f572e449SJoerg Sonnenberger #define XE_CARD_TYPE_FLAGS_DINGO 0x4
94f572e449SJoerg Sonnenberger
957261a835SSepherosa Ziehau #define XE_PROD_UMASK 0x11000f
967261a835SSepherosa Ziehau #define XE_PROD_ETHER_UMASK 0x010000
977261a835SSepherosa Ziehau #define XE_PROD_MODEM_UMASK 0x100000
987261a835SSepherosa Ziehau #define XE_PROD_SINGLE_ID1 0x010001
997261a835SSepherosa Ziehau #define XE_PROD_SINGLE_ID2 0x010002
1007261a835SSepherosa Ziehau #define XE_PROD_SINGLE_ID3 0x010003
1017261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID1 0x110001
1027261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID2 0x110002
1037261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID3 0x110003
1047261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID4 0x110004
1057261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID5 0x110005
1067261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID6 0x110006
1077261a835SSepherosa Ziehau #define XE_PROD_MULTI_ID7 0x110007
108f572e449SJoerg Sonnenberger
109f572e449SJoerg Sonnenberger struct xe_card_type {
110f572e449SJoerg Sonnenberger uint32_t prod_type;
111f572e449SJoerg Sonnenberger const char *card_type_desc;
112f572e449SJoerg Sonnenberger uint32_t flags;
113f572e449SJoerg Sonnenberger } xe_card_types[] = {
114f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID1, "CEM", XE_CARD_TYPE_FLAGS_NO },
115f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID2, "CEM2", XE_CARD_TYPE_FLAGS_CE2 },
116f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID3, "CEM3", XE_CARD_TYPE_FLAGS_CE2 },
117f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID4, "CEM33", XE_CARD_TYPE_FLAGS_CE2 },
118f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID5, "CEM56M", XE_CARD_TYPE_FLAGS_MOHAWK },
119f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID6, "CEM56", XE_CARD_TYPE_FLAGS_MOHAWK |
120f572e449SJoerg Sonnenberger XE_CARD_TYPE_FLAGS_DINGO },
121f572e449SJoerg Sonnenberger { XE_PROD_MULTI_ID7, "CEM56", XE_CARD_TYPE_FLAGS_MOHAWK |
122f572e449SJoerg Sonnenberger XE_CARD_TYPE_FLAGS_DINGO },
123f572e449SJoerg Sonnenberger { XE_PROD_SINGLE_ID1, "CE", XE_CARD_TYPE_FLAGS_NO },
124f572e449SJoerg Sonnenberger { XE_PROD_SINGLE_ID2, "CE2", XE_CARD_TYPE_FLAGS_CE2 },
125f572e449SJoerg Sonnenberger { XE_PROD_SINGLE_ID3, "CE3", XE_CARD_TYPE_FLAGS_MOHAWK },
126f572e449SJoerg Sonnenberger { 0, NULL, -1 }
127f572e449SJoerg Sonnenberger };
128f572e449SJoerg Sonnenberger
129f572e449SJoerg Sonnenberger static struct xe_vendor *xe_vendor_lookup (uint32_t);
130f572e449SJoerg Sonnenberger static struct xe_card_type *xe_card_type_lookup (uint32_t);
131f572e449SJoerg Sonnenberger
1327261a835SSepherosa Ziehau static int xe_cemfix (device_t);
133f572e449SJoerg Sonnenberger static int xe_pccard_probe (device_t);
134f572e449SJoerg Sonnenberger static int xe_pccard_match (device_t);
135f572e449SJoerg Sonnenberger static int xe_pccard_attach(device_t);
136f572e449SJoerg Sonnenberger
137f572e449SJoerg Sonnenberger static device_method_t xe_pccard_methods[] = {
138f572e449SJoerg Sonnenberger /* Device interface */
139f572e449SJoerg Sonnenberger DEVMETHOD(device_probe, pccard_compat_probe),
140f572e449SJoerg Sonnenberger DEVMETHOD(device_attach, pccard_compat_attach),
141f572e449SJoerg Sonnenberger DEVMETHOD(device_detach, xe_detach),
142f572e449SJoerg Sonnenberger
143f572e449SJoerg Sonnenberger /* Card interface */
144f572e449SJoerg Sonnenberger DEVMETHOD(card_compat_match, xe_pccard_match),
145f572e449SJoerg Sonnenberger DEVMETHOD(card_compat_probe, xe_pccard_probe),
146f572e449SJoerg Sonnenberger DEVMETHOD(card_compat_attach, xe_pccard_attach),
147f572e449SJoerg Sonnenberger
148d3c9c58eSSascha Wildner DEVMETHOD_END
149f572e449SJoerg Sonnenberger };
150f572e449SJoerg Sonnenberger
151f572e449SJoerg Sonnenberger static driver_t xe_pccard_driver = {
152f572e449SJoerg Sonnenberger "xe",
153f572e449SJoerg Sonnenberger xe_pccard_methods,
154f572e449SJoerg Sonnenberger sizeof(struct xe_softc),
155f572e449SJoerg Sonnenberger };
156f572e449SJoerg Sonnenberger
157f572e449SJoerg Sonnenberger devclass_t xe_devclass;
158aa2b9d05SSascha Wildner DRIVER_MODULE(xe, pccard, xe_pccard_driver, xe_devclass, NULL, NULL);
159f572e449SJoerg Sonnenberger
160f572e449SJoerg Sonnenberger /*
1617261a835SSepherosa Ziehau * Fixing for CEM2, CEM3 and CEM56/REM56 cards. These need some magic to
1627261a835SSepherosa Ziehau * enable the Ethernet function, which isn't mentioned anywhere in the CIS.
1637261a835SSepherosa Ziehau * Despite the register names, most of this isn't Dingo-specific.
164f572e449SJoerg Sonnenberger */
165f572e449SJoerg Sonnenberger static int
xe_cemfix(device_t dev)1667261a835SSepherosa Ziehau xe_cemfix(device_t dev)
167f572e449SJoerg Sonnenberger {
168f572e449SJoerg Sonnenberger struct xe_softc *sc = device_get_softc(dev);
169f572e449SJoerg Sonnenberger bus_space_tag_t bst;
170f572e449SJoerg Sonnenberger bus_space_handle_t bsh;
171f572e449SJoerg Sonnenberger struct resource *r;
172f572e449SJoerg Sonnenberger int rid;
173f572e449SJoerg Sonnenberger int ioport;
174f572e449SJoerg Sonnenberger
1757261a835SSepherosa Ziehau device_printf(dev, "CEM I/O port 0x%0lx, size 0x%0lx\n",
176f572e449SJoerg Sonnenberger bus_get_resource_start(dev, SYS_RES_IOPORT, sc->port_rid),
177f572e449SJoerg Sonnenberger bus_get_resource_count(dev, SYS_RES_IOPORT, sc->port_rid));
178f572e449SJoerg Sonnenberger
179f572e449SJoerg Sonnenberger rid = 0;
180f572e449SJoerg Sonnenberger r = bus_alloc_resource(dev, SYS_RES_MEMORY, &rid, 0,
181f572e449SJoerg Sonnenberger ~0, 4 << 10, RF_ACTIVE);
182f572e449SJoerg Sonnenberger if (r == NULL) {
183f572e449SJoerg Sonnenberger device_printf(dev, "Can't map in attribute memory\n");
184f572e449SJoerg Sonnenberger return -1;
185f572e449SJoerg Sonnenberger }
186f572e449SJoerg Sonnenberger
187f572e449SJoerg Sonnenberger bsh = rman_get_bushandle(r);
188f572e449SJoerg Sonnenberger bst = rman_get_bustag(r);
189f572e449SJoerg Sonnenberger
190f572e449SJoerg Sonnenberger CARD_SET_RES_FLAGS(device_get_parent(dev), dev, SYS_RES_MEMORY, rid,
191f572e449SJoerg Sonnenberger PCCARD_A_MEM_ATTR);
192f572e449SJoerg Sonnenberger
193f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_ECOR, DINGO_ECOR_IRQ_LEVEL |
194f572e449SJoerg Sonnenberger DINGO_ECOR_INT_ENABLE |
195f572e449SJoerg Sonnenberger DINGO_ECOR_IOB_ENABLE |
196f572e449SJoerg Sonnenberger DINGO_ECOR_ETH_ENABLE);
197f572e449SJoerg Sonnenberger ioport = bus_get_resource_start(dev, SYS_RES_IOPORT, sc->port_rid);
198f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_EBAR0, ioport & 0xff);
199f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_EBAR1, (ioport >> 8) & 0xff);
200f572e449SJoerg Sonnenberger
2017261a835SSepherosa Ziehau if (sc->dingo) {
202f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_DCOR0, DINGO_DCOR0_SF_INT);
2037261a835SSepherosa Ziehau bus_space_write_1(bst, bsh, DINGO_DCOR1,
2047261a835SSepherosa Ziehau DINGO_DCOR1_INT_LEVEL | DINGO_DCOR1_EEDIO);
205f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_DCOR2, 0x00);
206f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_DCOR3, 0x00);
207f572e449SJoerg Sonnenberger bus_space_write_1(bst, bsh, DINGO_DCOR4, 0x00);
2087261a835SSepherosa Ziehau }
209f572e449SJoerg Sonnenberger
210f572e449SJoerg Sonnenberger bus_release_resource(dev, SYS_RES_MEMORY, rid, r);
211f572e449SJoerg Sonnenberger
212f572e449SJoerg Sonnenberger /* success! */
213f572e449SJoerg Sonnenberger return 0;
214f572e449SJoerg Sonnenberger }
215f572e449SJoerg Sonnenberger
216f572e449SJoerg Sonnenberger static struct xe_vendor *
xe_vendor_lookup(uint32_t vendor_id)217f572e449SJoerg Sonnenberger xe_vendor_lookup(uint32_t vendor_id)
218f572e449SJoerg Sonnenberger {
219f572e449SJoerg Sonnenberger struct xe_vendor *v;
220f572e449SJoerg Sonnenberger
221f572e449SJoerg Sonnenberger for (v = xe_vendors; v->vendor_id != 0; ++v) {
222f572e449SJoerg Sonnenberger if(v->vendor_id == vendor_id)
223f572e449SJoerg Sonnenberger break;
224f572e449SJoerg Sonnenberger }
225f572e449SJoerg Sonnenberger return v;
226f572e449SJoerg Sonnenberger }
227f572e449SJoerg Sonnenberger
228f572e449SJoerg Sonnenberger static struct xe_card_type *
xe_card_type_lookup(uint32_t prod)229f572e449SJoerg Sonnenberger xe_card_type_lookup(uint32_t prod)
230f572e449SJoerg Sonnenberger {
231f572e449SJoerg Sonnenberger struct xe_card_type *ct;
232f572e449SJoerg Sonnenberger
233f572e449SJoerg Sonnenberger for (ct = xe_card_types; ct->card_type_desc != NULL; ++ct) {
234f572e449SJoerg Sonnenberger if(ct->prod_type == (prod & XE_PROD_UMASK))
235f572e449SJoerg Sonnenberger return ct;
236f572e449SJoerg Sonnenberger }
237f572e449SJoerg Sonnenberger return NULL;
238f572e449SJoerg Sonnenberger }
239f572e449SJoerg Sonnenberger
240f572e449SJoerg Sonnenberger /*
241f572e449SJoerg Sonnenberger * PCMCIA probe routine.
242f572e449SJoerg Sonnenberger * Identify the device. Called from the bus driver when the card is
243f572e449SJoerg Sonnenberger * inserted or otherwise powers up.
244f572e449SJoerg Sonnenberger */
245f572e449SJoerg Sonnenberger static int
xe_pccard_probe(device_t dev)246f572e449SJoerg Sonnenberger xe_pccard_probe(device_t dev)
247f572e449SJoerg Sonnenberger {
248f572e449SJoerg Sonnenberger struct xe_softc *scp = device_get_softc(dev);
249f572e449SJoerg Sonnenberger uint32_t vendor, product, prod;
250f572e449SJoerg Sonnenberger uint16_t prodext;
2516494cec5SSepherosa Ziehau const uint8_t *ether_addr;
252f572e449SJoerg Sonnenberger const char *cis3_str=NULL;
253f572e449SJoerg Sonnenberger struct xe_vendor *vendor_itm;
254f572e449SJoerg Sonnenberger struct xe_card_type *card_itm;
255f572e449SJoerg Sonnenberger int i;
256f572e449SJoerg Sonnenberger
2577261a835SSepherosa Ziehau #ifdef XE_DEBUG
2587261a835SSepherosa Ziehau const char *vendor_str = NULL;
2597261a835SSepherosa Ziehau const char *product_str = NULL;
2607261a835SSepherosa Ziehau const char *cis4_str = NULL;
2617261a835SSepherosa Ziehau
2627261a835SSepherosa Ziehau vendor = pccard_get_vendor(dev);
2637261a835SSepherosa Ziehau product = pccard_get_product(dev);
2647261a835SSepherosa Ziehau prodext = pccard_get_prodext(dev);
2657261a835SSepherosa Ziehau vendor_str = pccard_get_vendor_str(dev);
2667261a835SSepherosa Ziehau product_str = pccard_get_product_str(dev);
2677261a835SSepherosa Ziehau cis3_str = pccard_get_cis3_str(dev);
2687261a835SSepherosa Ziehau cis4_str = pccard_get_cis4_str(dev);
2697261a835SSepherosa Ziehau
2707261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "pccard_probe\n"));
2717261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "vendor = 0x%04x\n", vendor));
2727261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "product = 0x%04x\n", product));
2737261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "prodext = 0x%02x\n", prodext));
2747261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "vendor_str = %s\n",
2757261a835SSepherosa Ziehau vendor_str == NULL ? "NULL" : vendor_str));
2767261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "product_str = %s\n",
2777261a835SSepherosa Ziehau product_str == NULL ? "NULL" : product_str));
2787261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "cis3_str = %s\n",
2797261a835SSepherosa Ziehau cis3_str == NULL ? "NULL" : cis3_str));
2807261a835SSepherosa Ziehau DEVPRINTF(1, (dev, "cis4_str = %s\n",
2817261a835SSepherosa Ziehau cis4_str == NULL ? "NULL" : cis4_str));
2827261a835SSepherosa Ziehau #endif
2837261a835SSepherosa Ziehau
284f572e449SJoerg Sonnenberger /*
285f572e449SJoerg Sonnenberger * PCCARD_CISTPL_MANFID = 0x20
286f572e449SJoerg Sonnenberger */
287f572e449SJoerg Sonnenberger vendor = pccard_get_vendor(dev);
288f572e449SJoerg Sonnenberger vendor_itm = xe_vendor_lookup(vendor);
289f572e449SJoerg Sonnenberger /*
290f572e449SJoerg Sonnenberger * We always have some vendor here, although
291f572e449SJoerg Sonnenberger * vendor description may be "Unknown".
292f572e449SJoerg Sonnenberger */
293f572e449SJoerg Sonnenberger scp->vendor = vendor_itm->vendor_desc;
294f572e449SJoerg Sonnenberger
295f572e449SJoerg Sonnenberger product = pccard_get_product(dev);
296f572e449SJoerg Sonnenberger prodext = pccard_get_prodext(dev);
297f572e449SJoerg Sonnenberger
298f572e449SJoerg Sonnenberger /*
299f572e449SJoerg Sonnenberger * prod(new) = rev, media, prod(old)
300f572e449SJoerg Sonnenberger * prod(new) = (don't care), (care 0x10 bit), (care 0x0f bit)
301f572e449SJoerg Sonnenberger */
302f572e449SJoerg Sonnenberger prod = (product << 8) | prodext;
303f572e449SJoerg Sonnenberger card_itm = xe_card_type_lookup(prod);
304f572e449SJoerg Sonnenberger if (card_itm == NULL)
305f572e449SJoerg Sonnenberger return ENODEV;
306f572e449SJoerg Sonnenberger
307f572e449SJoerg Sonnenberger scp->card_type = card_itm->card_type_desc;
308f572e449SJoerg Sonnenberger if (card_itm->prod_type & XE_PROD_MODEM_UMASK)
309f572e449SJoerg Sonnenberger scp->modem = 1;
310f572e449SJoerg Sonnenberger
311f572e449SJoerg Sonnenberger for (i = 1; i != XE_CARD_TYPE_FLAGS_DINGO; i = i << 1) {
312f572e449SJoerg Sonnenberger switch(i & card_itm->flags) {
313f572e449SJoerg Sonnenberger case XE_CARD_TYPE_FLAGS_CE2:
314f572e449SJoerg Sonnenberger scp->ce2 = 1;
315f572e449SJoerg Sonnenberger break;
316f572e449SJoerg Sonnenberger case XE_CARD_TYPE_FLAGS_MOHAWK:
317f572e449SJoerg Sonnenberger scp->mohawk = 1;
318f572e449SJoerg Sonnenberger break;
319f572e449SJoerg Sonnenberger case XE_CARD_TYPE_FLAGS_DINGO:
320f572e449SJoerg Sonnenberger scp->dingo = 1;
321f572e449SJoerg Sonnenberger break;
322f572e449SJoerg Sonnenberger }
323f572e449SJoerg Sonnenberger }
324f572e449SJoerg Sonnenberger
325f572e449SJoerg Sonnenberger /*
326f572e449SJoerg Sonnenberger * PCCARD_CISTPL_VERS_1 = 0x15
327f572e449SJoerg Sonnenberger *
328f572e449SJoerg Sonnenberger * Check for certain strange CE2's that look like CE's:
329f572e449SJoerg Sonnenberger * match 3rd version string against "CE2"
330f572e449SJoerg Sonnenberger */
331f572e449SJoerg Sonnenberger cis3_str = pccard_get_cis3_str(dev);
332f572e449SJoerg Sonnenberger if (strcmp(scp->card_type, "CE") == 0)
3337261a835SSepherosa Ziehau if (cis3_str != NULL && strcmp(cis3_str, "PS-CE2-10") == 0)
334f572e449SJoerg Sonnenberger scp->card_type = "CE2";
335f572e449SJoerg Sonnenberger
336f572e449SJoerg Sonnenberger /*
337f572e449SJoerg Sonnenberger * PCCARD_CISTPL_FUNCE = 0x22
338f572e449SJoerg Sonnenberger */
339f572e449SJoerg Sonnenberger ether_addr = pccard_get_ether(dev);
340f572e449SJoerg Sonnenberger bcopy(ether_addr, scp->arpcom.ac_enaddr, ETHER_ADDR_LEN);
341f572e449SJoerg Sonnenberger
342f572e449SJoerg Sonnenberger /* Reject unsupported cards */
343f572e449SJoerg Sonnenberger if (strcmp(scp->card_type, "CE") == 0 ||
344f572e449SJoerg Sonnenberger strcmp(scp->card_type, "CEM") == 0) {
345f572e449SJoerg Sonnenberger device_printf(dev, "Sorry, your %s card is not supported :(\n",
346f572e449SJoerg Sonnenberger scp->card_type);
347f572e449SJoerg Sonnenberger return ENODEV;
348f572e449SJoerg Sonnenberger }
349f572e449SJoerg Sonnenberger
350f572e449SJoerg Sonnenberger /* Success */
351f572e449SJoerg Sonnenberger return 0;
352f572e449SJoerg Sonnenberger }
353f572e449SJoerg Sonnenberger
354f572e449SJoerg Sonnenberger static int
xe_pccard_attach(device_t dev)355f572e449SJoerg Sonnenberger xe_pccard_attach(device_t dev)
356f572e449SJoerg Sonnenberger {
357f572e449SJoerg Sonnenberger struct xe_softc *scp = device_get_softc(dev);
358f572e449SJoerg Sonnenberger int err;
359f572e449SJoerg Sonnenberger
360f572e449SJoerg Sonnenberger if ((err = xe_activate(dev)) != 0)
361f572e449SJoerg Sonnenberger return err;
362f572e449SJoerg Sonnenberger
363f572e449SJoerg Sonnenberger /* Hack RealPorts into submission */
3647261a835SSepherosa Ziehau if (scp->modem && xe_cemfix(dev) < 0) {
3657261a835SSepherosa Ziehau device_printf(dev, "Unable to fix your %s combo card\n",
3667261a835SSepherosa Ziehau scp->card_type);
367f572e449SJoerg Sonnenberger xe_deactivate(dev);
368f572e449SJoerg Sonnenberger return ENODEV;
369f572e449SJoerg Sonnenberger }
370f572e449SJoerg Sonnenberger return xe_attach(dev);
371f572e449SJoerg Sonnenberger }
372f572e449SJoerg Sonnenberger
373f572e449SJoerg Sonnenberger static int
xe_pccard_match(device_t dev)374f572e449SJoerg Sonnenberger xe_pccard_match(device_t dev)
375f572e449SJoerg Sonnenberger {
376f572e449SJoerg Sonnenberger const struct pccard_product *pp;
377f572e449SJoerg Sonnenberger
378f572e449SJoerg Sonnenberger if ((pp = pccard_product_lookup(dev, xe_pccard_products,
379f572e449SJoerg Sonnenberger sizeof(xe_pccard_products[0]), NULL)) != NULL) {
380f572e449SJoerg Sonnenberger if (pp->pp_name != NULL)
381f572e449SJoerg Sonnenberger device_set_desc(dev, pp->pp_name);
382f572e449SJoerg Sonnenberger return 0;
383f572e449SJoerg Sonnenberger }
384f572e449SJoerg Sonnenberger return EIO;
385f572e449SJoerg Sonnenberger }
386