bcm53xx_sdhc.c revision 1.1
11.1Smatt/*-
21.1Smatt * Copyright (c) 2012 The NetBSD Foundation, Inc.
31.1Smatt * All rights reserved.
41.1Smatt *
51.1Smatt * This code is derived from software contributed to The NetBSD Foundation
61.1Smatt * by Matt Thomas of 3am Software Foundry.
71.1Smatt *
81.1Smatt * Redistribution and use in source and binary forms, with or without
91.1Smatt * modification, are permitted provided that the following conditions
101.1Smatt * are met:
111.1Smatt * 1. Redistributions of source code must retain the above copyright
121.1Smatt *    notice, this list of conditions and the following disclaimer.
131.1Smatt * 2. Redistributions in binary form must reproduce the above copyright
141.1Smatt *    notice, this list of conditions and the following disclaimer in the
151.1Smatt *    documentation and/or other materials provided with the distribution.
161.1Smatt *
171.1Smatt * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
181.1Smatt * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
191.1Smatt * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
201.1Smatt * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
211.1Smatt * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
221.1Smatt * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
231.1Smatt * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
241.1Smatt * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
251.1Smatt * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
261.1Smatt * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
271.1Smatt * POSSIBILITY OF SUCH DAMAGE.
281.1Smatt */
291.1Smatt
301.1Smatt#include "locators.h"
311.1Smatt
321.1Smatt#include <sys/cdefs.h>
331.1Smatt
341.1Smatt__KERNEL_RCSID(1, "$NetBSD: bcm53xx_sdhc.c,v 1.1 2012/09/01 00:04:44 matt Exp $");
351.1Smatt
361.1Smatt#include <sys/bus.h>
371.1Smatt#include <sys/device.h>
381.1Smatt#include <sys/intr.h>
391.1Smatt#include <sys/systm.h>
401.1Smatt
411.1Smatt#include <arm/broadcom/bcm53xx_reg.h>
421.1Smatt#include <arm/broadcom/bcm53xx_var.h>
431.1Smatt
441.1Smatt#include <dev/sdmmc/sdhcreg.h>
451.1Smatt#include <dev/sdmmc/sdhcvar.h>
461.1Smatt
471.1Smattstatic int sdhc_ccb_match(device_t, cfdata_t, void *);
481.1Smattstatic void sdhc_ccb_attach(device_t, device_t, void *);
491.1Smatt
501.1Smattstruct sdhc_ccb_softc {
511.1Smatt	struct sdhc_softc ccbsc_sc;
521.1Smatt	bus_space_tag_t ccbsc_bst;
531.1Smatt	bus_space_handle_t ccbsc_bsh;
541.1Smatt	struct sdhc_host *ccbsc_hosts[1];
551.1Smatt	void *ccbsc_ih;
561.1Smatt};
571.1Smatt
581.1SmattCFATTACH_DECL_NEW(sdhc_ccb, sizeof(struct sdhc_ccb_softc),
591.1Smatt	sdhc_ccb_match, sdhc_ccb_attach, NULL, NULL);
601.1Smatt
611.1Smattstatic int
621.1Smattsdhc_ccb_match(device_t parent, cfdata_t cf, void *aux)
631.1Smatt{
641.1Smatt	struct bcmccb_attach_args * const ccbaa = aux;
651.1Smatt	const struct bcm_locators * const loc = &ccbaa->ccbaa_loc;
661.1Smatt
671.1Smatt	if (strcmp(cf->cf_name, loc->loc_name))
681.1Smatt		return 0;
691.1Smatt
701.1Smatt	KASSERT(cf->cf_loc[BCMCCBCF_PORT] == BCMCCBCF_PORT_DEFAULT);
711.1Smatt
721.1Smatt	return 1;
731.1Smatt}
741.1Smatt
751.1Smattstatic void
761.1Smattsdhc_ccb_attach(device_t parent, device_t self, void *aux)
771.1Smatt{
781.1Smatt	struct sdhc_ccb_softc * const ccbsc = device_private(self);
791.1Smatt	struct sdhc_softc * const sc = &ccbsc->ccbsc_sc;
801.1Smatt	struct bcmccb_attach_args * const ccbaa = aux;
811.1Smatt	const struct bcm_locators * const loc = &ccbaa->ccbaa_loc;
821.1Smatt
831.1Smatt	ccbsc->ccbsc_bst = ccbaa->ccbaa_ccb_bst;
841.1Smatt	bus_space_subregion(ccbsc->ccbsc_bst, ccbaa->ccbaa_ccb_bsh,
851.1Smatt	    loc->loc_offset, loc->loc_size, &ccbsc->ccbsc_bsh);
861.1Smatt
871.1Smatt	sc->sc_dev = self;
881.1Smatt	sc->sc_dmat = ccbaa->ccbaa_dmat;
891.1Smatt	sc->sc_host = ccbsc->ccbsc_hosts;
901.1Smatt	sc->sc_flags |= SDHC_FLAG_32BIT_ACCESS;
911.1Smatt	sc->sc_flags |= SDHC_FLAG_HAVE_CGM;
921.1Smatt	//sc->sc_flags |= SDHC_FLAG_USE_DMA;
931.1Smatt
941.1Smatt	aprint_naive(": SDHC controller\n");
951.1Smatt	aprint_normal(": SDHC controller%s\n",
961.1Smatt	   (sc->sc_flags & SDHC_FLAG_USE_DMA) ? " (DMA enabled)" : "");
971.1Smatt
981.1Smatt#if 1
991.1Smatt	int error = sdhc_host_found(sc, ccbsc->ccbsc_bst, ccbsc->ccbsc_bsh,
1001.1Smatt	    loc->loc_size);
1011.1Smatt	if (error != 0) {
1021.1Smatt		aprint_error_dev(self, "couldn't initialize host, error=%d\n",
1031.1Smatt		    error);
1041.1Smatt		goto fail;
1051.1Smatt	}
1061.1Smatt#endif
1071.1Smatt
1081.1Smatt	ccbsc->ccbsc_ih = intr_establish(loc->loc_intrs[0], IPL_VM, IST_LEVEL,
1091.1Smatt	    sdhc_intr, sc);
1101.1Smatt	if (ccbsc->ccbsc_ih == NULL) {
1111.1Smatt		aprint_error_dev(self, "failed to establish interrupt %d\n",
1121.1Smatt		     loc->loc_intrs[0]);
1131.1Smatt		goto fail;
1141.1Smatt	}
1151.1Smatt	aprint_normal_dev(self, "interrupting on irq %d\n",
1161.1Smatt	     loc->loc_intrs[0]);
1171.1Smatt
1181.1Smatt	return;
1191.1Smatt
1201.1Smattfail:
1211.1Smatt	if (ccbsc->ccbsc_ih) {
1221.1Smatt		intr_disestablish(ccbsc->ccbsc_ih);
1231.1Smatt		ccbsc->ccbsc_ih = NULL;
1241.1Smatt	}
1251.1Smatt}
126