o Merge in changes to the NetBSD sources:
pcmciavar.h 1.9->1.12
1.12, enami, minor coding nits
1.11, augustss, (pcmcia_devinfo NRFB)
1.10, cgd, add generic lookup routines
pcmcia.c 1.14->1.23
1.23, drochner, (probe code printing, NRFB)
1.22, augustss, KNF
1.21, uch, (hpcmips tweaks NRFB)
1.20, chopps, remove bogus debug
1.19, enami, minor coding nits
1.18, augustss, (pcmcia_devinfo NRFB)
1.17, nathanw, LP64 printf fixes
1.16, cgd, add generic lookup routines
1.15, aymeric, printf fixes
NRFB == not relevant to freebsd
o Expand the pccard matching routines to include the ability to match
against the CIS strings since our current driver database is based on
that.
o Add lots more ivars to get the information necessary to snag these values.
2000-09-22 01:15:26 +00:00
|
|
|
/* $NetBSD: pcmcia.c,v 1.23 2000/07/28 19:17:02 drochner Exp $ */
|
1999-10-26 06:52:31 +00:00
|
|
|
/* $FreeBSD$ */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 1997 Marc Horowitz. All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by Marc Horowitz.
|
|
|
|
* 4. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/malloc.h>
|
|
|
|
#include <sys/module.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/queue.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
#include <sys/bus.h>
|
|
|
|
#include <machine/bus.h>
|
|
|
|
#include <sys/rman.h>
|
|
|
|
#include <machine/resource.h>
|
|
|
|
|
|
|
|
#include <dev/pccard/pccardreg.h>
|
|
|
|
#include <dev/pccard/pccardvar.h>
|
|
|
|
|
2000-01-06 07:30:28 +00:00
|
|
|
#include "power_if.h"
|
2000-04-13 06:42:58 +00:00
|
|
|
#include "card_if.h"
|
2000-01-06 07:30:28 +00:00
|
|
|
|
|
|
|
#define PCCARDDEBUG
|
|
|
|
|
1999-10-26 06:52:31 +00:00
|
|
|
#ifdef PCCARDDEBUG
|
2000-01-06 07:30:28 +00:00
|
|
|
int pccard_debug = 1;
|
1999-10-26 06:52:31 +00:00
|
|
|
#define DPRINTF(arg) if (pccard_debug) printf arg
|
2000-01-06 07:30:28 +00:00
|
|
|
#define DEVPRINTF(arg) if (pccard_debug) device_printf arg
|
2000-10-19 22:36:41 +00:00
|
|
|
#define PRVERBOSE(arg) printf arg
|
|
|
|
#define DEVPRVERBOSE(arg) device_printf arg
|
1999-10-26 06:52:31 +00:00
|
|
|
#else
|
|
|
|
#define DPRINTF(arg)
|
2000-01-06 07:30:28 +00:00
|
|
|
#define DEVPRINTF(arg)
|
2000-10-19 22:36:41 +00:00
|
|
|
#define PRVERBOSE(arg) if (bootverbose) printf arg
|
|
|
|
#define DEVPRVERBOSE(arg) if (bootverbose) device_printf arg
|
1999-10-26 06:52:31 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef PCCARDVERBOSE
|
|
|
|
int pccard_verbose = 1;
|
|
|
|
#else
|
|
|
|
int pccard_verbose = 0;
|
|
|
|
#endif
|
|
|
|
|
1999-12-07 06:44:38 +00:00
|
|
|
int pccard_print(void *, const char *);
|
1999-10-26 06:52:31 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
pccard_ccr_read(pf, ccr)
|
|
|
|
struct pccard_function *pf;
|
|
|
|
int ccr;
|
|
|
|
{
|
|
|
|
return (bus_space_read_1(pf->pf_ccrt, pf->pf_ccrh,
|
|
|
|
pf->pf_ccr_offset + ccr));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
pccard_ccr_write(pf, ccr, val)
|
|
|
|
struct pccard_function *pf;
|
|
|
|
int ccr;
|
|
|
|
int val;
|
|
|
|
{
|
|
|
|
|
|
|
|
if ((pf->ccr_mask) & (1 << (ccr / 2))) {
|
|
|
|
bus_space_write_1(pf->pf_ccrt, pf->pf_ccrh,
|
|
|
|
pf->pf_ccr_offset + ccr, val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
static int
|
|
|
|
pccard_attach_card(device_t dev)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
2000-08-19 19:22:04 +00:00
|
|
|
struct pccard_softc *sc = PCCARD_SOFTC(dev);
|
1999-10-26 06:52:31 +00:00
|
|
|
struct pccard_function *pf;
|
2000-09-16 06:52:20 +00:00
|
|
|
struct pccard_ivar *ivar;
|
2000-06-18 05:28:59 +00:00
|
|
|
device_t child;
|
1999-10-26 06:52:31 +00:00
|
|
|
int attached;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* this is here so that when socket_enable calls gettype, trt happens
|
|
|
|
*/
|
|
|
|
STAILQ_INIT(&sc->card.pf_head);
|
|
|
|
|
2000-01-06 07:30:28 +00:00
|
|
|
DEVPRINTF((dev, "chip_socket_enable\n"));
|
|
|
|
POWER_ENABLE_SOCKET(device_get_parent(dev), dev);
|
1999-10-26 06:52:31 +00:00
|
|
|
|
2000-01-06 07:30:28 +00:00
|
|
|
DEVPRINTF((dev, "read_cis\n"));
|
1999-10-26 06:52:31 +00:00
|
|
|
pccard_read_cis(sc);
|
|
|
|
|
2000-01-06 07:30:28 +00:00
|
|
|
DEVPRINTF((dev, "check_cis_quirks\n"));
|
1999-10-26 06:52:31 +00:00
|
|
|
pccard_check_cis_quirks(dev);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bail now if the card has no functions, or if there was an error in
|
|
|
|
* the cis.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (sc->card.error)
|
|
|
|
return (1);
|
|
|
|
if (STAILQ_EMPTY(&sc->card.pf_head))
|
|
|
|
return (1);
|
|
|
|
|
2000-06-18 05:28:59 +00:00
|
|
|
if (1)
|
1999-10-26 06:52:31 +00:00
|
|
|
pccard_print_cis(dev);
|
|
|
|
|
|
|
|
attached = 0;
|
|
|
|
|
2000-01-06 07:30:28 +00:00
|
|
|
DEVPRINTF((dev, "functions scanning\n"));
|
1999-10-26 06:52:31 +00:00
|
|
|
STAILQ_FOREACH(pf, &sc->card.pf_head, pf_list) {
|
|
|
|
if (STAILQ_EMPTY(&pf->cfe_head))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
pf->sc = sc;
|
|
|
|
pf->cfe = NULL;
|
2000-08-22 04:25:57 +00:00
|
|
|
pf->dev = NULL;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
2000-08-26 00:07:40 +00:00
|
|
|
#if 0
|
2000-08-19 19:22:04 +00:00
|
|
|
DEVPRINTF((dev, "chip_socket_disable\n"));
|
|
|
|
POWER_DISABLE_SOCKET(device_get_parent(dev), dev);
|
2000-08-26 00:07:40 +00:00
|
|
|
#endif
|
1999-10-26 06:52:31 +00:00
|
|
|
STAILQ_FOREACH(pf, &sc->card.pf_head, pf_list) {
|
|
|
|
if (STAILQ_EMPTY(&pf->cfe_head))
|
|
|
|
continue;
|
2000-06-18 05:28:59 +00:00
|
|
|
/*
|
|
|
|
* In NetBSD, the drivers are responsible for activating
|
|
|
|
* each function of a card. I think that in FreeBSD we
|
|
|
|
* want to activate them enough for the usual bus_*_resource
|
|
|
|
* routines will do the right thing. This many mean a
|
|
|
|
* departure from the current NetBSD model.
|
|
|
|
*
|
|
|
|
* This could get really ugly for multifunction cards. But
|
|
|
|
* it might also just fall out of the FreeBSD resource model.
|
|
|
|
*
|
|
|
|
*/
|
2000-10-29 16:29:05 +00:00
|
|
|
ivar = malloc(sizeof(struct pccard_ivar), M_DEVBUF,
|
|
|
|
M_WAITOK | M_ZERO);
|
2000-06-18 05:28:59 +00:00
|
|
|
child = device_add_child(dev, NULL, -1);
|
2000-09-16 06:52:20 +00:00
|
|
|
device_set_ivars(child, ivar);
|
2000-10-09 03:40:15 +00:00
|
|
|
ivar->fcn = pf;
|
2000-10-16 07:51:12 +00:00
|
|
|
pf->dev = child;
|
2000-10-15 17:21:50 +00:00
|
|
|
/*
|
|
|
|
* XXX We might want to move the next two lines into
|
|
|
|
* XXX the pccard interface layer. For the moment, this
|
|
|
|
* XXX is OK, but some drivers want to pick the config
|
|
|
|
* XXX entry to use as well as some address tweaks (mostly
|
|
|
|
* XXX due to bugs in decode logic that makes some
|
|
|
|
* XXX addresses illegal or broken).
|
|
|
|
*/
|
2000-09-16 06:52:20 +00:00
|
|
|
pccard_function_init(pf);
|
2000-10-19 22:36:41 +00:00
|
|
|
if (pccard_function_enable(pf) == 0 &&
|
|
|
|
device_probe_and_attach(child) == 0) {
|
1999-10-26 06:52:31 +00:00
|
|
|
attached++;
|
|
|
|
|
2000-01-06 07:30:28 +00:00
|
|
|
DEVPRINTF((sc->dev, "function %d CCR at %d "
|
2000-10-15 17:21:50 +00:00
|
|
|
"offset %x: %x %x %x %x, %x %x %x %x, %x\n",
|
|
|
|
pf->number, pf->pf_ccr_window, pf->pf_ccr_offset,
|
|
|
|
pccard_ccr_read(pf, 0x00),
|
1999-10-26 06:52:31 +00:00
|
|
|
pccard_ccr_read(pf, 0x02), pccard_ccr_read(pf, 0x04),
|
|
|
|
pccard_ccr_read(pf, 0x06), pccard_ccr_read(pf, 0x0A),
|
|
|
|
pccard_ccr_read(pf, 0x0C), pccard_ccr_read(pf, 0x0E),
|
|
|
|
pccard_ccr_read(pf, 0x10), pccard_ccr_read(pf, 0x12)));
|
2000-10-15 17:21:50 +00:00
|
|
|
} else {
|
|
|
|
device_delete_child(dev, child);
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
}
|
2000-06-18 05:28:59 +00:00
|
|
|
return 0;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
static int
|
|
|
|
pccard_detach_card(device_t dev, int flags)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
2000-08-19 19:22:04 +00:00
|
|
|
struct pccard_softc *sc = PCCARD_SOFTC(dev);
|
1999-10-26 06:52:31 +00:00
|
|
|
struct pccard_function *pf;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We are running on either the PCCARD socket's event thread
|
|
|
|
* or in user context detaching a device by user request.
|
|
|
|
*/
|
|
|
|
STAILQ_FOREACH(pf, &sc->card.pf_head, pf_list) {
|
|
|
|
if (STAILQ_FIRST(&pf->cfe_head) == NULL)
|
|
|
|
continue;
|
2000-06-18 05:28:59 +00:00
|
|
|
|
|
|
|
pccard_function_disable(pf);
|
2000-08-22 04:25:57 +00:00
|
|
|
if (pf->dev)
|
|
|
|
device_delete_child(dev, pf->dev);
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
2000-04-13 06:42:58 +00:00
|
|
|
return 0;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
o Merge in changes to the NetBSD sources:
pcmciavar.h 1.9->1.12
1.12, enami, minor coding nits
1.11, augustss, (pcmcia_devinfo NRFB)
1.10, cgd, add generic lookup routines
pcmcia.c 1.14->1.23
1.23, drochner, (probe code printing, NRFB)
1.22, augustss, KNF
1.21, uch, (hpcmips tweaks NRFB)
1.20, chopps, remove bogus debug
1.19, enami, minor coding nits
1.18, augustss, (pcmcia_devinfo NRFB)
1.17, nathanw, LP64 printf fixes
1.16, cgd, add generic lookup routines
1.15, aymeric, printf fixes
NRFB == not relevant to freebsd
o Expand the pccard matching routines to include the ability to match
against the CIS strings since our current driver database is based on
that.
o Add lots more ivars to get the information necessary to snag these values.
2000-09-22 01:15:26 +00:00
|
|
|
const struct pccard_product *
|
|
|
|
pccard_product_lookup(device_t dev, const struct pccard_product *tab,
|
|
|
|
size_t ent_size, pccard_product_match_fn matchfn)
|
|
|
|
{
|
|
|
|
const struct pccard_product *ent;
|
|
|
|
int matches;
|
|
|
|
u_int32_t fcn;
|
|
|
|
u_int32_t vendor;
|
|
|
|
u_int32_t prod;
|
|
|
|
char *vendorstr;
|
|
|
|
char *prodstr;
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (sizeof *ent > ent_size)
|
|
|
|
panic("pccard_product_lookup: bogus ent_size %ld",
|
|
|
|
(long) ent_size);
|
|
|
|
#endif
|
|
|
|
if (pccard_get_vendor(dev, &vendor))
|
|
|
|
return (NULL);
|
|
|
|
if (pccard_get_product(dev, &prod))
|
|
|
|
return (NULL);
|
|
|
|
if (pccard_get_function_number(dev, &fcn))
|
|
|
|
return (NULL);
|
|
|
|
if (pccard_get_vendor_str(dev, &vendorstr))
|
|
|
|
return (NULL);
|
|
|
|
if (pccard_get_product_str(dev, &prodstr))
|
|
|
|
return (NULL);
|
|
|
|
for (ent = tab; ent->pp_name != NULL;
|
|
|
|
ent = (const struct pccard_product *)
|
|
|
|
((const char *) ent + ent_size)) {
|
|
|
|
matches = 1;
|
|
|
|
if (matches && ent->pp_vendor != PCCARD_VENDOR_ANY &&
|
|
|
|
vendor != ent->pp_vendor)
|
|
|
|
matches = 0;
|
|
|
|
if (matches && ent->pp_product != PCCARD_PRODUCT_ANY &&
|
|
|
|
prod != ent->pp_product)
|
|
|
|
matches = 0;
|
|
|
|
if (matches && fcn != ent->pp_expfunc)
|
|
|
|
matches = 0;
|
|
|
|
if (matches && ent->pp_vendor_str &&
|
|
|
|
strcmp(ent->pp_vendor_str, vendorstr) != 0)
|
|
|
|
matches = 0;
|
|
|
|
if (matches && ent->pp_product_str &&
|
|
|
|
strcmp(ent->pp_product_str, prodstr) != 0)
|
|
|
|
matches = 0;
|
|
|
|
if (matchfn != NULL)
|
|
|
|
matches = (*matchfn)(dev, ent, matches);
|
|
|
|
if (matches)
|
|
|
|
return (ent);
|
|
|
|
}
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
static int
|
|
|
|
pccard_card_gettype(device_t dev, int *type)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
2000-08-19 19:22:04 +00:00
|
|
|
struct pccard_softc *sc = PCCARD_SOFTC(dev);
|
1999-10-26 06:52:31 +00:00
|
|
|
struct pccard_function *pf;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* set the iftype to memory if this card has no functions (not yet
|
|
|
|
* probed), or only one function, and that is not initialized yet or
|
|
|
|
* that is memory.
|
|
|
|
*/
|
|
|
|
pf = STAILQ_FIRST(&sc->card.pf_head);
|
|
|
|
if (pf == NULL ||
|
|
|
|
(STAILQ_NEXT(pf, pf_list) == NULL &&
|
|
|
|
(pf->cfe == NULL || pf->cfe->iftype == PCCARD_IFTYPE_MEMORY)))
|
2000-04-13 06:42:58 +00:00
|
|
|
*type = PCCARD_IFTYPE_MEMORY;
|
1999-10-26 06:52:31 +00:00
|
|
|
else
|
2000-04-13 06:42:58 +00:00
|
|
|
*type = PCCARD_IFTYPE_IO;
|
|
|
|
return 0;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize a PCCARD function. May be called as long as the function is
|
|
|
|
* disabled.
|
|
|
|
*/
|
|
|
|
void
|
2000-09-16 06:52:20 +00:00
|
|
|
pccard_function_init(struct pccard_function *pf)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
2000-09-16 06:52:20 +00:00
|
|
|
struct pccard_config_entry *cfe;
|
2000-10-16 07:51:12 +00:00
|
|
|
int i;
|
2000-10-17 06:29:21 +00:00
|
|
|
struct pccard_ivar *devi = PCCARD_IVAR(pf->dev);
|
|
|
|
struct resource_list *rl = &devi->resources;
|
|
|
|
struct resource *r = 0;
|
|
|
|
device_t bus;
|
2000-10-22 04:46:56 +00:00
|
|
|
int start;
|
|
|
|
int end;
|
2000-09-16 06:52:20 +00:00
|
|
|
|
1999-10-26 06:52:31 +00:00
|
|
|
if (pf->pf_flags & PFF_ENABLED)
|
|
|
|
panic("pccard_function_init: function is enabled");
|
|
|
|
|
2000-10-17 06:29:21 +00:00
|
|
|
bus = device_get_parent(pf->dev);
|
1999-10-26 06:52:31 +00:00
|
|
|
/* Remember which configuration entry we are using. */
|
2000-10-15 17:21:50 +00:00
|
|
|
for (cfe = STAILQ_FIRST(&pf->cfe_head); cfe != NULL;
|
|
|
|
cfe = STAILQ_NEXT(cfe, cfe_list)) {
|
2000-10-16 07:51:12 +00:00
|
|
|
for (i = 0; i < cfe->num_iospace; i++)
|
|
|
|
cfe->iores[i] = NULL;
|
|
|
|
cfe->irqres = NULL;
|
|
|
|
for (i = 0; i < cfe->num_iospace; i++) {
|
2000-10-22 04:46:56 +00:00
|
|
|
start = cfe->iospace[i].start;
|
|
|
|
if (start)
|
|
|
|
end = start + cfe->iospace[i].length - 1;
|
|
|
|
else
|
|
|
|
end = ~0;
|
2000-10-16 07:51:12 +00:00
|
|
|
cfe->iorid[i] = i;
|
2000-10-17 06:29:21 +00:00
|
|
|
r = cfe->iores[i] = bus_alloc_resource(bus,
|
2000-10-22 04:46:56 +00:00
|
|
|
SYS_RES_IOPORT, &cfe->iorid[i], start, end,
|
|
|
|
cfe->iospace[i].length,
|
|
|
|
rman_make_alignment_flags(cfe->iospace[i].length));
|
2000-10-16 07:51:12 +00:00
|
|
|
if (cfe->iores[i] == 0)
|
|
|
|
goto not_this_one;
|
2000-10-17 06:29:21 +00:00
|
|
|
resource_list_add(rl, SYS_RES_IOPORT, cfe->iorid[i],
|
|
|
|
rman_get_start(r), rman_get_end(r),
|
|
|
|
cfe->iospace[i].length);
|
2000-10-16 07:51:12 +00:00
|
|
|
}
|
|
|
|
if (cfe->num_memspace > 0) {
|
|
|
|
goto not_this_one;
|
|
|
|
}
|
|
|
|
if (cfe->irqmask) {
|
|
|
|
cfe->irqrid = 0;
|
2000-10-20 20:27:22 +00:00
|
|
|
r = cfe->irqres = bus_alloc_resource(bus, SYS_RES_IRQ,
|
2000-10-22 04:46:56 +00:00
|
|
|
&cfe->irqrid, 0, ~0, 1, 0);
|
2000-10-16 07:51:12 +00:00
|
|
|
if (cfe->irqres == 0)
|
|
|
|
goto not_this_one;
|
2000-10-17 06:29:21 +00:00
|
|
|
resource_list_add(rl, SYS_RES_IRQ, cfe->irqrid,
|
|
|
|
rman_get_start(r), rman_get_end(r), 1);
|
2000-10-16 07:51:12 +00:00
|
|
|
}
|
|
|
|
/* XXX Don't know how to deal with maxtwins */
|
|
|
|
/* If we get to here, we've allocated all we need */
|
2000-10-15 17:21:50 +00:00
|
|
|
pf->cfe = cfe;
|
2000-10-16 07:51:12 +00:00
|
|
|
break;
|
|
|
|
not_this_one:;
|
2000-10-22 04:46:56 +00:00
|
|
|
DEVPRVERBOSE((bus, "Allocation failed for cfe %d\n",
|
|
|
|
cfe->number));
|
2000-10-19 22:36:41 +00:00
|
|
|
/*
|
|
|
|
* Release resources that we partially allocated
|
|
|
|
* from this config entry.
|
|
|
|
*/
|
2000-10-16 07:51:12 +00:00
|
|
|
for (i = 0; i < cfe->num_iospace; i++) {
|
2000-10-17 06:29:21 +00:00
|
|
|
resource_list_delete(rl, SYS_RES_IOPORT, i);
|
2000-10-16 07:51:12 +00:00
|
|
|
if (cfe->iores[i])
|
2000-10-17 06:29:21 +00:00
|
|
|
bus_release_resource(bus, SYS_RES_IOPORT,
|
2000-10-16 07:51:12 +00:00
|
|
|
cfe->iorid[i], cfe->iores[i]);
|
|
|
|
cfe->iores[i] = NULL;
|
|
|
|
}
|
|
|
|
if (cfe->irqmask && cfe->irqres) {
|
2000-10-17 06:29:21 +00:00
|
|
|
resource_list_delete(rl, SYS_RES_IRQ, cfe->irqrid);
|
|
|
|
bus_release_resource(bus, SYS_RES_IRQ,
|
2000-10-16 07:51:12 +00:00
|
|
|
cfe->irqrid, cfe->irqres);
|
|
|
|
cfe->irqres = NULL;
|
|
|
|
}
|
2000-10-15 17:21:50 +00:00
|
|
|
}
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable a PCCARD function */
|
|
|
|
int
|
2000-01-10 06:58:17 +00:00
|
|
|
pccard_function_enable(struct pccard_function *pf)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
|
|
|
struct pccard_function *tmp;
|
|
|
|
int reg;
|
2000-01-10 06:58:17 +00:00
|
|
|
device_t dev = pf->sc->dev;
|
1999-10-26 06:52:31 +00:00
|
|
|
|
2000-10-19 22:36:41 +00:00
|
|
|
if (pf->cfe == NULL) {
|
|
|
|
DEVPRVERBOSE((dev, "No config entry could be allocated.\n"));
|
|
|
|
return ENOMEM;
|
|
|
|
}
|
1999-10-26 06:52:31 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Increase the reference count on the socket, enabling power, if
|
|
|
|
* necessary.
|
|
|
|
*/
|
|
|
|
if (pf->sc->sc_enabled_count++ == 0)
|
2000-01-10 06:58:17 +00:00
|
|
|
POWER_ENABLE_SOCKET(device_get_parent(dev), dev);
|
|
|
|
DEVPRINTF((dev, "++enabled_count = %d\n", pf->sc->sc_enabled_count));
|
1999-10-26 06:52:31 +00:00
|
|
|
|
|
|
|
if (pf->pf_flags & PFF_ENABLED) {
|
|
|
|
/*
|
|
|
|
* Don't do anything if we're already enabled.
|
|
|
|
*/
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* it's possible for different functions' CCRs to be in the same
|
|
|
|
* underlying page. Check for that.
|
|
|
|
*/
|
|
|
|
STAILQ_FOREACH(tmp, &pf->sc->card.pf_head, pf_list) {
|
|
|
|
if ((tmp->pf_flags & PFF_ENABLED) &&
|
|
|
|
(pf->ccr_base >= (tmp->ccr_base - tmp->pf_ccr_offset)) &&
|
|
|
|
((pf->ccr_base + PCCARD_CCR_SIZE) <=
|
|
|
|
(tmp->ccr_base - tmp->pf_ccr_offset +
|
|
|
|
tmp->pf_ccr_realsize))) {
|
|
|
|
pf->pf_ccrt = tmp->pf_ccrt;
|
|
|
|
pf->pf_ccrh = tmp->pf_ccrh;
|
|
|
|
pf->pf_ccr_realsize = tmp->pf_ccr_realsize;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* pf->pf_ccr_offset = (tmp->pf_ccr_offset -
|
|
|
|
* tmp->ccr_base) + pf->ccr_base;
|
|
|
|
*/
|
|
|
|
pf->pf_ccr_offset =
|
|
|
|
(tmp->pf_ccr_offset + pf->ccr_base) -
|
|
|
|
tmp->ccr_base;
|
|
|
|
pf->pf_ccr_window = tmp->pf_ccr_window;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tmp == NULL) {
|
2000-01-10 06:58:17 +00:00
|
|
|
pf->ccr_rid = 0;
|
|
|
|
pf->ccr_res = bus_alloc_resource(dev, SYS_RES_MEMORY,
|
2000-08-26 00:07:40 +00:00
|
|
|
&pf->ccr_rid, 0xa0000, 0xdffff, 1 << 10, RF_ACTIVE);
|
|
|
|
if (!pf->ccr_res) {
|
|
|
|
DEVPRINTF((dev, "ccr_res == 0\n"));
|
1999-10-26 06:52:31 +00:00
|
|
|
goto bad;
|
2000-08-26 00:07:40 +00:00
|
|
|
}
|
2000-06-18 05:28:59 +00:00
|
|
|
CARD_SET_RES_FLAGS(device_get_parent(dev), dev, SYS_RES_MEMORY,
|
|
|
|
pf->ccr_rid, PCCARD_A_MEM_ATTR);
|
2000-08-26 00:07:40 +00:00
|
|
|
CARD_SET_MEMORY_OFFSET(device_get_parent(dev), dev,
|
|
|
|
pf->ccr_rid, (pf->ccr_rid >> 10) << 10);
|
2000-01-10 06:58:17 +00:00
|
|
|
pf->pf_ccrt = rman_get_bustag(pf->ccr_res);
|
|
|
|
pf->pf_ccrh = rman_get_bushandle(pf->ccr_res);
|
|
|
|
pf->pf_ccr_offset = rman_get_start(pf->ccr_res);
|
|
|
|
pf->pf_ccr_realsize = 1;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
reg = (pf->cfe->number & PCCARD_CCR_OPTION_CFINDEX);
|
|
|
|
reg |= PCCARD_CCR_OPTION_LEVIREQ;
|
|
|
|
if (pccard_mfc(pf->sc)) {
|
|
|
|
reg |= (PCCARD_CCR_OPTION_FUNC_ENABLE |
|
|
|
|
PCCARD_CCR_OPTION_ADDR_DECODE);
|
2000-10-15 17:21:50 +00:00
|
|
|
/*
|
|
|
|
* XXX Need to enable PCCARD_CCR_OPTION_IRQ_ENABLE if
|
|
|
|
* XXX we have an interrupt handler, but we don't know that
|
|
|
|
* XXX at this point.
|
|
|
|
*/
|
2000-10-18 02:21:00 +00:00
|
|
|
reg |= PCCARD_CCR_OPTION_IREQ_ENABLE;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_OPTION, reg);
|
|
|
|
|
|
|
|
reg = 0;
|
|
|
|
if ((pf->cfe->flags & PCCARD_CFE_IO16) == 0)
|
|
|
|
reg |= PCCARD_CCR_STATUS_IOIS8;
|
|
|
|
if (pf->cfe->flags & PCCARD_CFE_AUDIO)
|
|
|
|
reg |= PCCARD_CCR_STATUS_AUDIO;
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_STATUS, reg);
|
|
|
|
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_SOCKETCOPY, 0);
|
|
|
|
|
|
|
|
if (pccard_mfc(pf->sc)) {
|
|
|
|
long tmp, iosize;
|
|
|
|
|
|
|
|
tmp = pf->pf_mfc_iomax - pf->pf_mfc_iobase;
|
|
|
|
/* round up to nearest (2^n)-1 */
|
|
|
|
for (iosize = 1; iosize < tmp; iosize <<= 1)
|
|
|
|
;
|
|
|
|
iosize--;
|
|
|
|
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE0,
|
|
|
|
pf->pf_mfc_iobase & 0xff);
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE1,
|
|
|
|
(pf->pf_mfc_iobase >> 8) & 0xff);
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE2, 0);
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE3, 0);
|
|
|
|
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOSIZE, iosize);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef PCCARDDEBUG
|
|
|
|
if (pccard_debug) {
|
|
|
|
STAILQ_FOREACH(tmp, &pf->sc->card.pf_head, pf_list) {
|
2000-01-06 07:30:28 +00:00
|
|
|
device_printf(tmp->sc->dev,
|
|
|
|
"function %d CCR at %d offset %x: "
|
|
|
|
"%x %x %x %x, %x %x %x %x, %x\n",
|
|
|
|
tmp->number, tmp->pf_ccr_window,
|
|
|
|
tmp->pf_ccr_offset,
|
|
|
|
pccard_ccr_read(tmp, 0x00),
|
|
|
|
pccard_ccr_read(tmp, 0x02),
|
|
|
|
pccard_ccr_read(tmp, 0x04),
|
|
|
|
pccard_ccr_read(tmp, 0x06),
|
|
|
|
pccard_ccr_read(tmp, 0x0A),
|
|
|
|
pccard_ccr_read(tmp, 0x0C),
|
|
|
|
pccard_ccr_read(tmp, 0x0E),
|
|
|
|
pccard_ccr_read(tmp, 0x10),
|
|
|
|
pccard_ccr_read(tmp, 0x12));
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
pf->pf_flags |= PFF_ENABLED;
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
bad:
|
|
|
|
/*
|
|
|
|
* Decrement the reference count, and power down the socket, if
|
|
|
|
* necessary.
|
|
|
|
*/
|
|
|
|
if (--pf->sc->sc_enabled_count == 0)
|
2000-01-10 06:58:17 +00:00
|
|
|
POWER_DISABLE_SOCKET(device_get_parent(dev), dev);
|
2000-08-26 00:07:40 +00:00
|
|
|
DEVPRINTF((dev, "bad --enabled_count = %d\n", pf->sc->sc_enabled_count));
|
1999-10-26 06:52:31 +00:00
|
|
|
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable PCCARD function. */
|
|
|
|
void
|
2000-01-10 06:58:17 +00:00
|
|
|
pccard_function_disable(struct pccard_function *pf)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
|
|
|
struct pccard_function *tmp;
|
2000-01-10 06:58:17 +00:00
|
|
|
device_t dev = pf->sc->dev;
|
1999-10-26 06:52:31 +00:00
|
|
|
|
|
|
|
if (pf->cfe == NULL)
|
2000-06-18 05:28:59 +00:00
|
|
|
panic("pccard_function_disable: function not initialized");
|
1999-10-26 06:52:31 +00:00
|
|
|
|
|
|
|
if ((pf->pf_flags & PFF_ENABLED) == 0) {
|
|
|
|
/*
|
|
|
|
* Don't do anything if we're already disabled.
|
|
|
|
*/
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* it's possible for different functions' CCRs to be in the same
|
|
|
|
* underlying page. Check for that. Note we mark us as disabled
|
|
|
|
* first to avoid matching ourself.
|
|
|
|
*/
|
|
|
|
|
|
|
|
pf->pf_flags &= ~PFF_ENABLED;
|
|
|
|
STAILQ_FOREACH(tmp, &pf->sc->card.pf_head, pf_list) {
|
|
|
|
if ((tmp->pf_flags & PFF_ENABLED) &&
|
|
|
|
(pf->ccr_base >= (tmp->ccr_base - tmp->pf_ccr_offset)) &&
|
|
|
|
((pf->ccr_base + PCCARD_CCR_SIZE) <=
|
|
|
|
(tmp->ccr_base - tmp->pf_ccr_offset + tmp->pf_ccr_realsize)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Not used by anyone else; unmap the CCR. */
|
|
|
|
if (tmp == NULL) {
|
2000-01-10 06:58:17 +00:00
|
|
|
bus_release_resource(dev, SYS_RES_MEMORY, pf->ccr_rid,
|
|
|
|
pf->ccr_res);
|
|
|
|
pf->ccr_res = NULL;
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Decrement the reference count, and power down the socket, if
|
|
|
|
* necessary.
|
|
|
|
*/
|
|
|
|
if (--pf->sc->sc_enabled_count == 0)
|
2000-01-10 06:58:17 +00:00
|
|
|
POWER_DISABLE_SOCKET(device_get_parent(dev), dev);
|
|
|
|
DEVPRINTF((dev, "--enabled_count = %d\n", pf->sc->sc_enabled_count));
|
1999-10-26 06:52:31 +00:00
|
|
|
}
|
|
|
|
|
2000-01-10 06:58:17 +00:00
|
|
|
#if 0
|
|
|
|
/* XXX These functions are needed, but not like this XXX */
|
1999-10-26 06:52:31 +00:00
|
|
|
int
|
2000-01-10 06:58:17 +00:00
|
|
|
pccard_io_map(struct pccard_function *pf, int width, bus_addr_t offset,
|
|
|
|
bus_size_t size, struct pccard_io_handle *pcihp, int *windowp)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
|
|
|
int reg;
|
|
|
|
|
2000-10-15 17:21:50 +00:00
|
|
|
if (pccard_chip_io_map(pf->sc->pct, pf->sc->pch, width, offset, size,
|
|
|
|
pcihp, windowp))
|
1999-10-26 06:52:31 +00:00
|
|
|
return (1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX in the multifunction multi-iospace-per-function case, this
|
|
|
|
* needs to cooperate with io_alloc to make sure that the spaces
|
|
|
|
* don't overlap, and that the ccr's are set correctly
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (pccard_mfc(pf->sc)) {
|
|
|
|
long tmp, iosize;
|
|
|
|
|
|
|
|
if (pf->pf_mfc_iomax == 0) {
|
|
|
|
pf->pf_mfc_iobase = pcihp->addr + offset;
|
|
|
|
pf->pf_mfc_iomax = pf->pf_mfc_iobase + size;
|
|
|
|
} else {
|
|
|
|
/* this makes the assumption that nothing overlaps */
|
|
|
|
if (pf->pf_mfc_iobase > pcihp->addr + offset)
|
|
|
|
pf->pf_mfc_iobase = pcihp->addr + offset;
|
|
|
|
if (pf->pf_mfc_iomax < pcihp->addr + offset + size)
|
|
|
|
pf->pf_mfc_iomax = pcihp->addr + offset + size;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = pf->pf_mfc_iomax - pf->pf_mfc_iobase;
|
|
|
|
/* round up to nearest (2^n)-1 */
|
|
|
|
for (iosize = 1; iosize >= tmp; iosize <<= 1)
|
|
|
|
;
|
|
|
|
iosize--;
|
|
|
|
|
2000-01-10 06:58:17 +00:00
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE0,
|
|
|
|
pf->pf_mfc_iobase & 0xff);
|
1999-10-26 06:52:31 +00:00
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE1,
|
2000-01-10 06:58:17 +00:00
|
|
|
(pf->pf_mfc_iobase >> 8) & 0xff);
|
1999-10-26 06:52:31 +00:00
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE2, 0);
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOBASE3, 0);
|
|
|
|
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_IOSIZE, iosize);
|
|
|
|
|
|
|
|
reg = pccard_ccr_read(pf, PCCARD_CCR_OPTION);
|
|
|
|
reg |= PCCARD_CCR_OPTION_ADDR_DECODE;
|
|
|
|
pccard_ccr_write(pf, PCCARD_CCR_OPTION, reg);
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2000-01-10 06:58:17 +00:00
|
|
|
pccard_io_unmap(struct pccard_function *pf, int window)
|
1999-10-26 06:52:31 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
pccard_chip_io_unmap(pf->sc->pct, pf->sc->pch, window);
|
|
|
|
|
|
|
|
/* XXX Anything for multi-function cards? */
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
Implement indirection in the pccard probe/attach. This should make it
possible to have different probe/attach semantics between the two
systems and yet still use the same driver for both.
Compatibility methods for OLDCARD drivers. We use these routines to make
it possible to call the OLDCARD driver's probe routine in the context that
it expects. For OLDCARD these are implemented as pass throughs to the
device_{probe,attach} routines. For NEWCARD they are implemented such
such that probe becomes strictly a matching routine and attach does both
the old probe and old attach.
compat devices should use the following:
/* Device interface */
DEVMETHOD(device_probe), pccard_compat_probe),
DEVMETHOD(device_attach), pccard_compat_attach),
/* Card interface */
DEVMETHOD(card_compat_match, foo_match), /* newly written */
DEVMETHOD(card_compat_probe, foo_probe), /* old probe */
DEVMETHOD(card_compat_attach, foo_attach), /* old attach */
This will allow a single driver binary image to be used for both
OLDCARD and NEWCARD.
Drivers wishing to not retain OLDCARD compatibility needn't do this.
ep driver minorly updated.
sn driver updated more than minorly. Add module dependencies to allow
module to load. Also change name to if_sn. Add some debugging code.
attempt to fix the cannot allocate memory problem I'd been seeing.
Minor formatting nits.
2000-09-19 04:39:20 +00:00
|
|
|
/*
|
|
|
|
* simulate the old "probe" routine. In the new world order, the driver
|
|
|
|
* needs to grab devices while in the old they were assigned to the device by
|
|
|
|
* the pccardd process. These symbols are exported to the upper layers.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
pccard_compat_probe(device_t dev)
|
|
|
|
{
|
|
|
|
return (CARD_COMPAT_MATCH(dev));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
pccard_compat_attach(device_t dev)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = CARD_COMPAT_PROBE(dev);
|
|
|
|
if (err == 0)
|
|
|
|
err = CARD_COMPAT_ATTACH(dev);
|
|
|
|
return (err);
|
|
|
|
}
|
|
|
|
|
1999-11-29 06:42:55 +00:00
|
|
|
#define PCCARD_NPORT 2
|
|
|
|
#define PCCARD_NMEM 5
|
|
|
|
#define PCCARD_NIRQ 1
|
|
|
|
#define PCCARD_NDRQ 0
|
|
|
|
|
1999-10-26 06:52:31 +00:00
|
|
|
static int
|
|
|
|
pccard_add_children(device_t dev, int busno)
|
|
|
|
{
|
2000-04-13 06:42:58 +00:00
|
|
|
/* Call parent to scan for any current children */
|
1999-10-26 06:52:31 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_probe(device_t dev)
|
|
|
|
{
|
2000-10-19 22:36:41 +00:00
|
|
|
device_set_desc(dev, "16-bit PCCard bus");
|
1999-10-26 06:52:31 +00:00
|
|
|
return pccard_add_children(dev, device_get_unit(dev));
|
|
|
|
}
|
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
static int
|
|
|
|
pccard_attach(device_t dev)
|
|
|
|
{
|
2000-08-19 19:22:04 +00:00
|
|
|
struct pccard_softc *sc = PCCARD_SOFTC(dev);
|
2000-06-18 05:28:59 +00:00
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
sc->dev = dev;
|
2000-06-18 05:28:59 +00:00
|
|
|
sc->sc_enabled_count = 0;
|
2000-04-13 06:42:58 +00:00
|
|
|
return bus_generic_attach(dev);
|
|
|
|
}
|
|
|
|
|
1999-11-29 06:42:55 +00:00
|
|
|
static void
|
|
|
|
pccard_print_resources(struct resource_list *rl, const char *name, int type,
|
|
|
|
int count, const char *format)
|
|
|
|
{
|
|
|
|
struct resource_list_entry *rle;
|
|
|
|
int printed;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
printed = 0;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
rle = resource_list_find(rl, type, i);
|
|
|
|
if (rle) {
|
|
|
|
if (printed == 0)
|
|
|
|
printf(" %s ", name);
|
|
|
|
else if (printed > 0)
|
|
|
|
printf(",");
|
|
|
|
printed++;
|
|
|
|
printf(format, rle->start);
|
|
|
|
if (rle->count > 1) {
|
|
|
|
printf("-");
|
|
|
|
printf(format, rle->start + rle->count - 1);
|
|
|
|
}
|
|
|
|
} else if (i > 3) {
|
|
|
|
/* check the first few regardless */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_print_child(device_t dev, device_t child)
|
|
|
|
{
|
2000-10-09 03:40:15 +00:00
|
|
|
struct pccard_ivar *devi = PCCARD_IVAR(child);
|
1999-11-29 06:42:55 +00:00
|
|
|
struct resource_list *rl = &devi->resources;
|
|
|
|
int retval = 0;
|
|
|
|
|
|
|
|
retval += bus_print_child_header(dev, child);
|
|
|
|
retval += printf(" at");
|
|
|
|
|
|
|
|
if (devi) {
|
|
|
|
pccard_print_resources(rl, "port", SYS_RES_IOPORT,
|
|
|
|
PCCARD_NPORT, "%#lx");
|
|
|
|
pccard_print_resources(rl, "iomem", SYS_RES_MEMORY,
|
|
|
|
PCCARD_NMEM, "%#lx");
|
|
|
|
pccard_print_resources(rl, "irq", SYS_RES_IRQ, PCCARD_NIRQ,
|
|
|
|
"%ld");
|
|
|
|
pccard_print_resources(rl, "drq", SYS_RES_DRQ, PCCARD_NDRQ,
|
|
|
|
"%ld");
|
2000-10-18 02:21:00 +00:00
|
|
|
retval += printf(" function %d config %d", devi->fcn->number,
|
|
|
|
devi->fcn->cfe->number);
|
1999-11-29 06:42:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
retval += bus_print_child_footer(dev, child);
|
|
|
|
|
|
|
|
return (retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_set_resource(device_t dev, device_t child, int type, int rid,
|
|
|
|
u_long start, u_long count)
|
|
|
|
{
|
2000-10-09 03:40:15 +00:00
|
|
|
struct pccard_ivar *devi = PCCARD_IVAR(child);
|
1999-11-29 06:42:55 +00:00
|
|
|
struct resource_list *rl = &devi->resources;
|
|
|
|
|
|
|
|
if (type != SYS_RES_IOPORT && type != SYS_RES_MEMORY
|
|
|
|
&& type != SYS_RES_IRQ && type != SYS_RES_DRQ)
|
|
|
|
return EINVAL;
|
|
|
|
if (rid < 0)
|
|
|
|
return EINVAL;
|
|
|
|
if (type == SYS_RES_IOPORT && rid >= PCCARD_NPORT)
|
|
|
|
return EINVAL;
|
|
|
|
if (type == SYS_RES_MEMORY && rid >= PCCARD_NMEM)
|
|
|
|
return EINVAL;
|
|
|
|
if (type == SYS_RES_IRQ && rid >= PCCARD_NIRQ)
|
|
|
|
return EINVAL;
|
|
|
|
if (type == SYS_RES_DRQ && rid >= PCCARD_NDRQ)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
resource_list_add(rl, type, rid, start, start + count - 1, count);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_get_resource(device_t dev, device_t child, int type, int rid,
|
|
|
|
u_long *startp, u_long *countp)
|
|
|
|
{
|
2000-10-09 03:40:15 +00:00
|
|
|
struct pccard_ivar *devi = PCCARD_IVAR(child);
|
1999-11-29 06:42:55 +00:00
|
|
|
struct resource_list *rl = &devi->resources;
|
|
|
|
struct resource_list_entry *rle;
|
|
|
|
|
|
|
|
rle = resource_list_find(rl, type, rid);
|
|
|
|
if (!rle)
|
|
|
|
return ENOENT;
|
|
|
|
|
|
|
|
if (startp)
|
|
|
|
*startp = rle->start;
|
|
|
|
if (countp)
|
|
|
|
*countp = rle->count;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
pccard_delete_resource(device_t dev, device_t child, int type, int rid)
|
|
|
|
{
|
2000-10-09 03:40:15 +00:00
|
|
|
struct pccard_ivar *devi = PCCARD_IVAR(child);
|
1999-11-29 06:42:55 +00:00
|
|
|
struct resource_list *rl = &devi->resources;
|
|
|
|
resource_list_delete(rl, type, rid);
|
|
|
|
}
|
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
static int
|
|
|
|
pccard_set_res_flags(device_t dev, device_t child, int type, int rid,
|
|
|
|
u_int32_t flags)
|
|
|
|
{
|
|
|
|
return CARD_SET_RES_FLAGS(device_get_parent(dev), child, type,
|
|
|
|
rid, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_set_memory_offset(device_t dev, device_t child, int rid,
|
|
|
|
u_int32_t offset)
|
|
|
|
{
|
|
|
|
return CARD_SET_MEMORY_OFFSET(device_get_parent(dev), child, rid,
|
|
|
|
offset);
|
|
|
|
}
|
|
|
|
|
Implement indirection in the pccard probe/attach. This should make it
possible to have different probe/attach semantics between the two
systems and yet still use the same driver for both.
Compatibility methods for OLDCARD drivers. We use these routines to make
it possible to call the OLDCARD driver's probe routine in the context that
it expects. For OLDCARD these are implemented as pass throughs to the
device_{probe,attach} routines. For NEWCARD they are implemented such
such that probe becomes strictly a matching routine and attach does both
the old probe and old attach.
compat devices should use the following:
/* Device interface */
DEVMETHOD(device_probe), pccard_compat_probe),
DEVMETHOD(device_attach), pccard_compat_attach),
/* Card interface */
DEVMETHOD(card_compat_match, foo_match), /* newly written */
DEVMETHOD(card_compat_probe, foo_probe), /* old probe */
DEVMETHOD(card_compat_attach, foo_attach), /* old attach */
This will allow a single driver binary image to be used for both
OLDCARD and NEWCARD.
Drivers wishing to not retain OLDCARD compatibility needn't do this.
ep driver minorly updated.
sn driver updated more than minorly. Add module dependencies to allow
module to load. Also change name to if_sn. Add some debugging code.
attempt to fix the cannot allocate memory problem I'd been seeing.
Minor formatting nits.
2000-09-19 04:39:20 +00:00
|
|
|
static int
|
|
|
|
pccard_read_ivar(device_t bus, device_t child, int which, u_char *result)
|
|
|
|
{
|
2000-10-09 03:40:15 +00:00
|
|
|
struct pccard_ivar *devi = PCCARD_IVAR(child);
|
2000-10-07 05:48:51 +00:00
|
|
|
struct pccard_function *func = devi->fcn;
|
|
|
|
struct pccard_softc *sc = PCCARD_SOFTC(bus);
|
|
|
|
|
Implement indirection in the pccard probe/attach. This should make it
possible to have different probe/attach semantics between the two
systems and yet still use the same driver for both.
Compatibility methods for OLDCARD drivers. We use these routines to make
it possible to call the OLDCARD driver's probe routine in the context that
it expects. For OLDCARD these are implemented as pass throughs to the
device_{probe,attach} routines. For NEWCARD they are implemented such
such that probe becomes strictly a matching routine and attach does both
the old probe and old attach.
compat devices should use the following:
/* Device interface */
DEVMETHOD(device_probe), pccard_compat_probe),
DEVMETHOD(device_attach), pccard_compat_attach),
/* Card interface */
DEVMETHOD(card_compat_match, foo_match), /* newly written */
DEVMETHOD(card_compat_probe, foo_probe), /* old probe */
DEVMETHOD(card_compat_attach, foo_attach), /* old attach */
This will allow a single driver binary image to be used for both
OLDCARD and NEWCARD.
Drivers wishing to not retain OLDCARD compatibility needn't do this.
ep driver minorly updated.
sn driver updated more than minorly. Add module dependencies to allow
module to load. Also change name to if_sn. Add some debugging code.
attempt to fix the cannot allocate memory problem I'd been seeing.
Minor formatting nits.
2000-09-19 04:39:20 +00:00
|
|
|
/* PCCARD_IVAR_ETHADDR unhandled from oldcard */
|
2000-10-07 05:48:51 +00:00
|
|
|
switch (which) {
|
|
|
|
default:
|
|
|
|
case PCCARD_IVAR_ETHADDR:
|
|
|
|
return (ENOENT);
|
|
|
|
break;
|
|
|
|
case PCCARD_IVAR_VENDOR:
|
|
|
|
*(u_int32_t *) result = sc->card.manufacturer;
|
|
|
|
break;
|
|
|
|
case PCCARD_IVAR_PRODUCT:
|
|
|
|
*(u_int32_t *) result = sc->card.product;
|
|
|
|
break;
|
|
|
|
case PCCARD_IVAR_FUNCTION_NUMBER:
|
2000-10-09 03:40:15 +00:00
|
|
|
if (!func) {
|
|
|
|
device_printf(bus, "No function number, bug!\n");
|
|
|
|
return (ENOENT);
|
|
|
|
}
|
|
|
|
*(u_int32_t *) result = func->number;
|
2000-10-07 05:48:51 +00:00
|
|
|
break;
|
|
|
|
case PCCARD_IVAR_VENDOR_STR:
|
|
|
|
*(char **) result = sc->card.cis1_info[0];
|
|
|
|
break;
|
|
|
|
case PCCARD_IVAR_PRODUCT_STR:
|
|
|
|
*(char **) result = sc->card.cis1_info[1];
|
|
|
|
break;
|
|
|
|
case PCCARD_IVAR_CIS3_STR:
|
|
|
|
*(char **) result = sc->card.cis1_info[2];
|
|
|
|
break;
|
2000-10-15 17:21:50 +00:00
|
|
|
case PCCARD_IVAR_CIS4_STR:
|
|
|
|
*(char **) result = sc->card.cis1_info[2];
|
|
|
|
break;
|
2000-10-07 05:48:51 +00:00
|
|
|
}
|
|
|
|
return (0);
|
Implement indirection in the pccard probe/attach. This should make it
possible to have different probe/attach semantics between the two
systems and yet still use the same driver for both.
Compatibility methods for OLDCARD drivers. We use these routines to make
it possible to call the OLDCARD driver's probe routine in the context that
it expects. For OLDCARD these are implemented as pass throughs to the
device_{probe,attach} routines. For NEWCARD they are implemented such
such that probe becomes strictly a matching routine and attach does both
the old probe and old attach.
compat devices should use the following:
/* Device interface */
DEVMETHOD(device_probe), pccard_compat_probe),
DEVMETHOD(device_attach), pccard_compat_attach),
/* Card interface */
DEVMETHOD(card_compat_match, foo_match), /* newly written */
DEVMETHOD(card_compat_probe, foo_probe), /* old probe */
DEVMETHOD(card_compat_attach, foo_attach), /* old attach */
This will allow a single driver binary image to be used for both
OLDCARD and NEWCARD.
Drivers wishing to not retain OLDCARD compatibility needn't do this.
ep driver minorly updated.
sn driver updated more than minorly. Add module dependencies to allow
module to load. Also change name to if_sn. Add some debugging code.
attempt to fix the cannot allocate memory problem I'd been seeing.
Minor formatting nits.
2000-09-19 04:39:20 +00:00
|
|
|
}
|
|
|
|
|
2000-10-07 05:48:51 +00:00
|
|
|
static void
|
|
|
|
pccard_driver_added(device_t dev, driver_t *driver)
|
|
|
|
{
|
2000-10-15 17:21:50 +00:00
|
|
|
/*
|
|
|
|
* XXX eventually we need to attach stuff when we know we
|
|
|
|
* XXX have kids. For now we do nothing because we normally
|
|
|
|
* XXX add children ourselves. We don't want to necessarily
|
|
|
|
* XXX force a reprobe.
|
|
|
|
*/
|
2000-10-07 05:48:51 +00:00
|
|
|
}
|
Implement indirection in the pccard probe/attach. This should make it
possible to have different probe/attach semantics between the two
systems and yet still use the same driver for both.
Compatibility methods for OLDCARD drivers. We use these routines to make
it possible to call the OLDCARD driver's probe routine in the context that
it expects. For OLDCARD these are implemented as pass throughs to the
device_{probe,attach} routines. For NEWCARD they are implemented such
such that probe becomes strictly a matching routine and attach does both
the old probe and old attach.
compat devices should use the following:
/* Device interface */
DEVMETHOD(device_probe), pccard_compat_probe),
DEVMETHOD(device_attach), pccard_compat_attach),
/* Card interface */
DEVMETHOD(card_compat_match, foo_match), /* newly written */
DEVMETHOD(card_compat_probe, foo_probe), /* old probe */
DEVMETHOD(card_compat_attach, foo_attach), /* old attach */
This will allow a single driver binary image to be used for both
OLDCARD and NEWCARD.
Drivers wishing to not retain OLDCARD compatibility needn't do this.
ep driver minorly updated.
sn driver updated more than minorly. Add module dependencies to allow
module to load. Also change name to if_sn. Add some debugging code.
attempt to fix the cannot allocate memory problem I'd been seeing.
Minor formatting nits.
2000-09-19 04:39:20 +00:00
|
|
|
|
2000-10-17 06:29:21 +00:00
|
|
|
static struct resource *
|
|
|
|
pccard_alloc_resource(device_t dev, device_t child, int type, int *rid,
|
|
|
|
u_long start, u_long end, u_long count, u_int flags)
|
|
|
|
{
|
|
|
|
struct pccard_ivar *ivar;
|
|
|
|
struct pccard_function *pf;
|
2000-10-18 02:21:00 +00:00
|
|
|
struct resource *r = 0;
|
2000-10-17 06:29:21 +00:00
|
|
|
|
|
|
|
if (device_get_parent(child) == dev) {
|
|
|
|
ivar = PCCARD_IVAR(child);
|
|
|
|
pf = ivar->fcn;
|
|
|
|
switch (type) {
|
|
|
|
case SYS_RES_IRQ:
|
|
|
|
if (*rid > 0)
|
|
|
|
return NULL;
|
2000-10-18 02:21:00 +00:00
|
|
|
r = pf->cfe->irqres;
|
|
|
|
break;
|
2000-10-17 06:29:21 +00:00
|
|
|
case SYS_RES_IOPORT:
|
|
|
|
if (*rid > 3) /* XXX */
|
|
|
|
return NULL;
|
2000-10-18 02:21:00 +00:00
|
|
|
r = pf->cfe->iores[*rid];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2000-10-17 06:29:21 +00:00
|
|
|
}
|
|
|
|
}
|
2000-10-18 02:21:00 +00:00
|
|
|
if (r != NULL) {
|
|
|
|
if (flags & RF_ACTIVE)
|
|
|
|
bus_generic_activate_resource(dev, child, type,
|
|
|
|
*rid, r);
|
|
|
|
return (r);
|
|
|
|
}
|
2000-10-17 06:29:21 +00:00
|
|
|
return (bus_generic_alloc_resource(dev, child, type, rid, start,
|
|
|
|
end, count, flags));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_release_resource(device_t dev, device_t child, int type, int rid,
|
|
|
|
struct resource *r)
|
|
|
|
{
|
|
|
|
return bus_generic_release_resource(dev, child, type, rid, r);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_activate_resource(device_t dev, device_t child, int type, int rid,
|
|
|
|
struct resource *r)
|
|
|
|
{
|
2000-10-18 02:21:00 +00:00
|
|
|
/* XXX need to write to the COR to activate this for mf cards */
|
2000-10-17 06:29:21 +00:00
|
|
|
return (bus_generic_activate_resource(dev, child, type, rid, r));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
pccard_deactivate_resource(device_t dev, device_t child, int type, int rid,
|
|
|
|
struct resource *r)
|
|
|
|
{
|
2000-10-18 02:21:00 +00:00
|
|
|
/* XXX need to write to the COR to deactivate this for mf cards */
|
2000-10-17 06:29:21 +00:00
|
|
|
return (bus_generic_deactivate_resource(dev, child, type, rid, r));
|
|
|
|
}
|
|
|
|
|
2000-10-19 22:36:41 +00:00
|
|
|
static void
|
|
|
|
pccard_child_detached(device_t parent, device_t dev)
|
|
|
|
{
|
|
|
|
struct pccard_ivar *ivar = PCCARD_IVAR(dev);
|
|
|
|
|
|
|
|
if (parent == device_get_parent(dev))
|
|
|
|
free(ivar, M_DEVBUF);
|
|
|
|
}
|
|
|
|
|
1999-10-26 06:52:31 +00:00
|
|
|
static device_method_t pccard_methods[] = {
|
|
|
|
/* Device interface */
|
|
|
|
DEVMETHOD(device_probe, pccard_probe),
|
2000-04-13 06:42:58 +00:00
|
|
|
DEVMETHOD(device_attach, pccard_attach),
|
2000-06-18 05:28:59 +00:00
|
|
|
DEVMETHOD(device_detach, bus_generic_detach),
|
1999-10-26 06:52:31 +00:00
|
|
|
DEVMETHOD(device_shutdown, bus_generic_shutdown),
|
|
|
|
DEVMETHOD(device_suspend, bus_generic_suspend),
|
|
|
|
DEVMETHOD(device_resume, bus_generic_resume),
|
|
|
|
|
|
|
|
/* Bus interface */
|
|
|
|
DEVMETHOD(bus_print_child, pccard_print_child),
|
2000-10-07 05:48:51 +00:00
|
|
|
DEVMETHOD(bus_driver_added, pccard_driver_added),
|
2000-10-19 22:36:41 +00:00
|
|
|
DEVMETHOD(bus_child_detached, pccard_child_detached),
|
2000-10-17 06:29:21 +00:00
|
|
|
DEVMETHOD(bus_alloc_resource, pccard_alloc_resource),
|
|
|
|
DEVMETHOD(bus_release_resource, pccard_release_resource),
|
|
|
|
DEVMETHOD(bus_activate_resource, pccard_activate_resource),
|
|
|
|
DEVMETHOD(bus_deactivate_resource, pccard_deactivate_resource),
|
1999-10-26 06:52:31 +00:00
|
|
|
DEVMETHOD(bus_setup_intr, bus_generic_setup_intr),
|
|
|
|
DEVMETHOD(bus_teardown_intr, bus_generic_teardown_intr),
|
|
|
|
DEVMETHOD(bus_set_resource, pccard_set_resource),
|
|
|
|
DEVMETHOD(bus_get_resource, pccard_get_resource),
|
|
|
|
DEVMETHOD(bus_delete_resource, pccard_delete_resource),
|
Implement indirection in the pccard probe/attach. This should make it
possible to have different probe/attach semantics between the two
systems and yet still use the same driver for both.
Compatibility methods for OLDCARD drivers. We use these routines to make
it possible to call the OLDCARD driver's probe routine in the context that
it expects. For OLDCARD these are implemented as pass throughs to the
device_{probe,attach} routines. For NEWCARD they are implemented such
such that probe becomes strictly a matching routine and attach does both
the old probe and old attach.
compat devices should use the following:
/* Device interface */
DEVMETHOD(device_probe), pccard_compat_probe),
DEVMETHOD(device_attach), pccard_compat_attach),
/* Card interface */
DEVMETHOD(card_compat_match, foo_match), /* newly written */
DEVMETHOD(card_compat_probe, foo_probe), /* old probe */
DEVMETHOD(card_compat_attach, foo_attach), /* old attach */
This will allow a single driver binary image to be used for both
OLDCARD and NEWCARD.
Drivers wishing to not retain OLDCARD compatibility needn't do this.
ep driver minorly updated.
sn driver updated more than minorly. Add module dependencies to allow
module to load. Also change name to if_sn. Add some debugging code.
attempt to fix the cannot allocate memory problem I'd been seeing.
Minor formatting nits.
2000-09-19 04:39:20 +00:00
|
|
|
DEVMETHOD(bus_read_ivar, pccard_read_ivar),
|
1999-10-26 06:52:31 +00:00
|
|
|
|
2000-04-13 06:42:58 +00:00
|
|
|
/* Card Interface */
|
|
|
|
DEVMETHOD(card_set_res_flags, pccard_set_res_flags),
|
|
|
|
DEVMETHOD(card_set_memory_offset, pccard_set_memory_offset),
|
|
|
|
DEVMETHOD(card_get_type, pccard_card_gettype),
|
|
|
|
DEVMETHOD(card_attach_card, pccard_attach_card),
|
|
|
|
DEVMETHOD(card_detach_card, pccard_detach_card),
|
|
|
|
|
1999-10-26 06:52:31 +00:00
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
static driver_t pccard_driver = {
|
|
|
|
"pccard",
|
|
|
|
pccard_methods,
|
2000-08-19 19:22:04 +00:00
|
|
|
sizeof(struct pccard_softc)
|
1999-10-26 06:52:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
devclass_t pccard_devclass;
|
|
|
|
|
1999-11-29 06:42:55 +00:00
|
|
|
DRIVER_MODULE(pccard, pcic, pccard_driver, pccard_devclass, 0, 0);
|
1999-10-26 06:52:31 +00:00
|
|
|
DRIVER_MODULE(pccard, pc98pcic, pccard_driver, pccard_devclass, 0, 0);
|
|
|
|
DRIVER_MODULE(pccard, pccbb, pccard_driver, pccard_devclass, 0, 0);
|
1999-11-29 06:42:55 +00:00
|
|
|
DRIVER_MODULE(pccard, tcic, pccard_driver, pccard_devclass, 0, 0);
|
2000-08-22 04:25:57 +00:00
|
|
|
MODULE_VERSION(pccard, 1);
|
|
|
|
MODULE_DEPEND(pccard, pcic, 1, 1, 1);
|