xref: /dflybsd-src/sys/dev/netif/xe/if_xe_pccard.c (revision dcb4b80df71308df08858318df7f148fde3b3e1b)
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