666 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			666 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | /*
 | ||
|  |  * Copyright (c) 2000, 2001, 2002, 2003, 2004, 2005, 2008, 2009 | ||
|  |  *	The President and Fellows of Harvard College. | ||
|  |  * | ||
|  |  * 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. Neither the name of the University nor the names of its contributors | ||
|  |  *    may be used to endorse or promote products derived from this software | ||
|  |  *    without specific prior written permission. | ||
|  |  * | ||
|  |  * THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY AND CONTRIBUTORS ``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 UNIVERSITY OR CONTRIBUTORS 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. | ||
|  |  */ | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Machine-independent LAMEbus code. | ||
|  |  */ | ||
|  | 
 | ||
|  | #include <types.h>
 | ||
|  | #include <lib.h>
 | ||
|  | #include <cpu.h>
 | ||
|  | #include <membar.h>
 | ||
|  | #include <spinlock.h>
 | ||
|  | #include <current.h>
 | ||
|  | #include <lamebus/lamebus.h>
 | ||
|  | 
 | ||
|  | /* Register offsets within each config region */ | ||
|  | #define CFGREG_VID   0    /* Vendor ID */
 | ||
|  | #define CFGREG_DID   4    /* Device ID */
 | ||
|  | #define CFGREG_DRL   8    /* Device Revision Level */
 | ||
|  | 
 | ||
|  | /* LAMEbus controller private registers (offsets within its config region) */ | ||
|  | #define CTLREG_RAMSZ    0x200
 | ||
|  | #define CTLREG_IRQS     0x204
 | ||
|  | #define CTLREG_PWR      0x208
 | ||
|  | #define CTLREG_IRQE     0x20c
 | ||
|  | #define CTLREG_CPUS     0x210
 | ||
|  | #define CTLREG_CPUE     0x214
 | ||
|  | #define CTLREG_SELF     0x218
 | ||
|  | 
 | ||
|  | /* LAMEbus CPU control registers (offsets within each per-cpu region) */ | ||
|  | #define CTLCPU_CIRQE	0x000
 | ||
|  | #define CTLCPU_CIPI	0x004
 | ||
|  | #define CTLCPU_CRAM	0x300
 | ||
|  | 
 | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Read a config register for the given slot. | ||
|  |  */ | ||
|  | static | ||
|  | inline | ||
|  | uint32_t | ||
|  | read_cfg_register(struct lamebus_softc *lb, int slot, uint32_t offset) | ||
|  | { | ||
|  | 	/* Note that lb might be NULL on some platforms in some contexts. */ | ||
|  | 	offset += LB_CONFIG_SIZE*slot; | ||
|  | 	return lamebus_read_register(lb, LB_CONTROLLER_SLOT, offset); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Write a config register for a given slot. | ||
|  |  */ | ||
|  | static | ||
|  | inline | ||
|  | void | ||
|  | write_cfg_register(struct lamebus_softc *lb, int slot, uint32_t offset, | ||
|  | 		   uint32_t val) | ||
|  | { | ||
|  | 	offset += LB_CONFIG_SIZE*slot; | ||
|  | 	lamebus_write_register(lb, LB_CONTROLLER_SLOT, offset, val); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Read one of the bus controller's registers. | ||
|  |  */ | ||
|  | static | ||
|  | inline | ||
|  | uint32_t | ||
|  | read_ctl_register(struct lamebus_softc *lb, uint32_t offset) | ||
|  | { | ||
|  | 	/* Note that lb might be NULL on some platforms in some contexts. */ | ||
|  | 	return read_cfg_register(lb, LB_CONTROLLER_SLOT, offset); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Write one of the bus controller's registers. | ||
|  |  */ | ||
|  | static | ||
|  | inline | ||
|  | void | ||
|  | write_ctl_register(struct lamebus_softc *lb, uint32_t offset, uint32_t val) | ||
|  | { | ||
|  | 	write_cfg_register(lb, LB_CONTROLLER_SLOT, offset, val); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Write one of the bus controller's CPU control registers. | ||
|  |  */ | ||
|  | static | ||
|  | inline | ||
|  | void | ||
|  | write_ctlcpu_register(struct lamebus_softc *lb, unsigned hw_cpunum, | ||
|  | 		      uint32_t offset, uint32_t val) | ||
|  | { | ||
|  | 	offset += LB_CTLCPU_OFFSET + hw_cpunum * LB_CTLCPU_SIZE; | ||
|  | 	lamebus_write_register(lb, LB_CONTROLLER_SLOT, offset, val); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Find and create secondary CPUs. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_find_cpus(struct lamebus_softc *lamebus) | ||
|  | { | ||
|  | 	uint32_t mainboard_vid, mainboard_did; | ||
|  | 	uint32_t cpumask, self, bit, val; | ||
|  | 	unsigned i, numcpus, bootcpu; | ||
|  | 	unsigned hwnum[32]; | ||
|  | 
 | ||
|  | 	mainboard_vid = read_cfg_register(lamebus, LB_CONTROLLER_SLOT, | ||
|  | 					  CFGREG_VID); | ||
|  | 	mainboard_did = read_cfg_register(lamebus, LB_CONTROLLER_SLOT, | ||
|  | 					  CFGREG_DID); | ||
|  | 	if (mainboard_vid == LB_VENDOR_CS161 && | ||
|  | 	    mainboard_did == LBCS161_UPBUSCTL) { | ||
|  | 		/* Old uniprocessor mainboard; no cpu registers. */ | ||
|  | 		lamebus->ls_uniprocessor = 1; | ||
|  | 		return; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	cpumask = read_ctl_register(lamebus, CTLREG_CPUS); | ||
|  | 	self = read_ctl_register(lamebus, CTLREG_SELF); | ||
|  | 
 | ||
|  | 	numcpus = 0; | ||
|  | 	bootcpu = 0; | ||
|  | 	for (i=0; i<32; i++) { | ||
|  | 		bit = (uint32_t)1 << i; | ||
|  | 		if ((cpumask & bit) != 0) { | ||
|  | 			if (self & bit) { | ||
|  | 				bootcpu = numcpus; | ||
|  | 				curcpu->c_hardware_number = i; | ||
|  | 			} | ||
|  | 			hwnum[numcpus] = i; | ||
|  | 			numcpus++; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	for (i=0; i<numcpus; i++) { | ||
|  | 		if (i != bootcpu) { | ||
|  | 			cpu_create(hwnum[i]); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/*
 | ||
|  | 	 * By default, route all interrupts only to the boot cpu. We | ||
|  | 	 * could be arbitrarily more elaborate, up to things like | ||
|  | 	 * dynamic load balancing. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	for (i=0; i<numcpus; i++) { | ||
|  | 		if (i != bootcpu) { | ||
|  | 			val = 0; | ||
|  | 		} | ||
|  | 		else { | ||
|  | 			val = 0xffffffff; | ||
|  | 		} | ||
|  | 		write_ctlcpu_register(lamebus, hwnum[i], CTLCPU_CIRQE, val); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Start up secondary CPUs. | ||
|  |  * | ||
|  |  * The first word of the CRAM area is set to the entry point for new | ||
|  |  * CPUs; the second to the (software) CPU number. Note that the logic | ||
|  |  * here assumes the boot CPU is CPU 0 and the others are 1-N as | ||
|  |  * created in the function above. This is fine if all CPUs are on | ||
|  |  * LAMEbus; if in some environment there are other CPUs about as well | ||
|  |  * this logic will have to be made more complex. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_start_cpus(struct lamebus_softc *lamebus) | ||
|  | { | ||
|  | 	uint32_t cpumask, self, bit; | ||
|  | 	uint32_t ctlcpuoffset; | ||
|  | 	uint32_t *cram; | ||
|  | 	unsigned i; | ||
|  | 	unsigned cpunum; | ||
|  | 
 | ||
|  | 	if (lamebus->ls_uniprocessor) { | ||
|  | 		return; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	cpumask = read_ctl_register(lamebus, CTLREG_CPUS); | ||
|  | 	self = read_ctl_register(lamebus, CTLREG_SELF); | ||
|  | 
 | ||
|  | 	/* Poke in the startup address. */ | ||
|  | 	cpunum = 1; | ||
|  | 	for (i=0; i<32; i++) { | ||
|  | 		bit = (uint32_t)1 << i; | ||
|  | 		if ((cpumask & bit) != 0) { | ||
|  | 			if (self & bit) { | ||
|  | 				continue; | ||
|  | 			} | ||
|  | 			ctlcpuoffset = LB_CTLCPU_OFFSET + i * LB_CTLCPU_SIZE; | ||
|  | 			cram = lamebus_map_area(lamebus, | ||
|  | 						LB_CONTROLLER_SLOT, | ||
|  | 						ctlcpuoffset + CTLCPU_CRAM); | ||
|  | 			cram[0] = (uint32_t)cpu_start_secondary; | ||
|  | 			cram[1] = cpunum++; | ||
|  | 		} | ||
|  | 	} | ||
|  | 	/* Ensure all the above writes get flushed. */ | ||
|  | 	membar_store_store(); | ||
|  | 
 | ||
|  | 	/* Now, enable them all. */ | ||
|  | 	write_ctl_register(lamebus, CTLREG_CPUE, cpumask); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Probe function. | ||
|  |  * | ||
|  |  * Given a LAMEbus, look for a device that's not already been marked | ||
|  |  * in use, has the specified IDs, and has a device revision level at | ||
|  |  * least as high as the minimum specified. | ||
|  |  * | ||
|  |  * Returns the slot number found (0-31) or -1 if nothing suitable was | ||
|  |  * found. | ||
|  |  * | ||
|  |  * If VERSION_RET is not null, return the device version found. This | ||
|  |  * allows drivers to blacklist specific versions or otherwise conduct | ||
|  |  * more specific checks. | ||
|  |  */ | ||
|  | 
 | ||
|  | int | ||
|  | lamebus_probe(struct lamebus_softc *sc, | ||
|  | 	      uint32_t vendorid, uint32_t deviceid, | ||
|  | 	      uint32_t lowver, uint32_t *version_ret) | ||
|  | { | ||
|  | 	int slot; | ||
|  | 	uint32_t val; | ||
|  | 
 | ||
|  | 	/*
 | ||
|  | 	 * Because the slot information in sc is used when dispatching | ||
|  | 	 * interrupts, disable interrupts while working with it. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	spinlock_acquire(&sc->ls_lock); | ||
|  | 
 | ||
|  | 	for (slot=0; slot<LB_NSLOTS; slot++) { | ||
|  | 		if (sc->ls_slotsinuse & (1<<slot)) { | ||
|  | 			/* Slot already in use; skip */ | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		val = read_cfg_register(sc, slot, CFGREG_VID); | ||
|  | 		if (val!=vendorid) { | ||
|  | 			/* Wrong vendor id */ | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		val = read_cfg_register(sc, slot, CFGREG_DID); | ||
|  | 		if (val != deviceid) { | ||
|  | 			/* Wrong device id */ | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		val = read_cfg_register(sc, slot, CFGREG_DRL); | ||
|  | 		if (val < lowver) { | ||
|  | 			/* Unsupported device revision */ | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 		if (version_ret != NULL) { | ||
|  | 			*version_ret = val; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/* Found something */ | ||
|  | 
 | ||
|  | 		spinlock_release(&sc->ls_lock); | ||
|  | 		return slot; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Found nothing */ | ||
|  | 
 | ||
|  | 	spinlock_release(&sc->ls_lock); | ||
|  | 	return -1; | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Mark that a slot is in use. | ||
|  |  * This prevents the probe routine from returning the same device over | ||
|  |  * and over again. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_mark(struct lamebus_softc *sc, int slot) | ||
|  | { | ||
|  | 	uint32_t mask = ((uint32_t)1) << slot; | ||
|  | 	KASSERT(slot>=0 && slot < LB_NSLOTS); | ||
|  | 
 | ||
|  | 	spinlock_acquire(&sc->ls_lock); | ||
|  | 
 | ||
|  | 	if ((sc->ls_slotsinuse & mask)!=0) { | ||
|  | 		panic("lamebus_mark: slot %d already in use\n", slot); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	sc->ls_slotsinuse |= mask; | ||
|  | 
 | ||
|  | 	spinlock_release(&sc->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Mark that a slot is no longer in use. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_unmark(struct lamebus_softc *sc, int slot) | ||
|  | { | ||
|  | 	uint32_t mask = ((uint32_t)1) << slot; | ||
|  | 	KASSERT(slot>=0 && slot < LB_NSLOTS); | ||
|  | 
 | ||
|  | 	spinlock_acquire(&sc->ls_lock); | ||
|  | 
 | ||
|  | 	if ((sc->ls_slotsinuse & mask)==0) { | ||
|  | 		panic("lamebus_mark: slot %d not marked in use\n", slot); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	sc->ls_slotsinuse &= ~mask; | ||
|  | 
 | ||
|  | 	spinlock_release(&sc->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Register a function (and a device context pointer) to be called | ||
|  |  * when a particular slot signals an interrupt. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_attach_interrupt(struct lamebus_softc *sc, int slot, | ||
|  | 			 void *devdata, | ||
|  | 			 void (*irqfunc)(void *devdata)) | ||
|  | { | ||
|  | 	uint32_t mask = ((uint32_t)1) << slot; | ||
|  | 	KASSERT(slot>=0 && slot < LB_NSLOTS); | ||
|  | 
 | ||
|  | 	spinlock_acquire(&sc->ls_lock); | ||
|  | 
 | ||
|  | 	if ((sc->ls_slotsinuse & mask)==0) { | ||
|  | 		panic("lamebus_attach_interrupt: slot %d not marked in use\n", | ||
|  | 		      slot); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	KASSERT(sc->ls_devdata[slot]==NULL); | ||
|  | 	KASSERT(sc->ls_irqfuncs[slot]==NULL); | ||
|  | 
 | ||
|  | 	sc->ls_devdata[slot] = devdata; | ||
|  | 	sc->ls_irqfuncs[slot] = irqfunc; | ||
|  | 
 | ||
|  | 	spinlock_release(&sc->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Unregister a function that was being called when a particular slot | ||
|  |  * signaled an interrupt. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_detach_interrupt(struct lamebus_softc *sc, int slot) | ||
|  | { | ||
|  | 	uint32_t mask = ((uint32_t)1) << slot; | ||
|  | 	KASSERT(slot>=0 && slot < LB_NSLOTS); | ||
|  | 
 | ||
|  | 	spinlock_acquire(&sc->ls_lock); | ||
|  | 
 | ||
|  | 	if ((sc->ls_slotsinuse & mask)==0) { | ||
|  | 		panic("lamebus_detach_interrupt: slot %d not marked in use\n", | ||
|  | 		      slot); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	KASSERT(sc->ls_irqfuncs[slot]!=NULL); | ||
|  | 
 | ||
|  | 	sc->ls_devdata[slot] = NULL; | ||
|  | 	sc->ls_irqfuncs[slot] = NULL; | ||
|  | 
 | ||
|  | 	spinlock_release(&sc->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Mask/unmask an interrupt using the global IRQE register. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_mask_interrupt(struct lamebus_softc *lamebus, int slot) | ||
|  | { | ||
|  | 	uint32_t bits, mask = ((uint32_t)1) << slot; | ||
|  | 	KASSERT(slot >= 0 && slot < LB_NSLOTS); | ||
|  | 
 | ||
|  | 	spinlock_acquire(&lamebus->ls_lock); | ||
|  | 	bits = read_ctl_register(lamebus, CTLREG_IRQE); | ||
|  | 	bits &= ~mask; | ||
|  | 	write_ctl_register(lamebus, CTLREG_IRQE, bits); | ||
|  | 	spinlock_release(&lamebus->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | void | ||
|  | lamebus_unmask_interrupt(struct lamebus_softc *lamebus, int slot) | ||
|  | { | ||
|  | 	uint32_t bits, mask = ((uint32_t)1) << slot; | ||
|  | 	KASSERT(slot >= 0 && slot < LB_NSLOTS); | ||
|  | 
 | ||
|  | 	spinlock_acquire(&lamebus->ls_lock); | ||
|  | 	bits = read_ctl_register(lamebus, CTLREG_IRQE); | ||
|  | 	bits |= mask; | ||
|  | 	write_ctl_register(lamebus, CTLREG_IRQE, bits); | ||
|  | 	spinlock_release(&lamebus->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * LAMEbus interrupt handling function. (Machine-independent!) | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_interrupt(struct lamebus_softc *lamebus) | ||
|  | { | ||
|  | 	/*
 | ||
|  | 	 * Note that despite the fact that "spl" stands for "set | ||
|  | 	 * priority level", we don't actually support interrupt | ||
|  | 	 * priorities. When an interrupt happens, we look through the | ||
|  | 	 * slots to find the first interrupting device and call its | ||
|  | 	 * interrupt routine, no matter what that device is. | ||
|  | 	 * | ||
|  | 	 * Note that the entire LAMEbus uses only one on-cpu interrupt line. | ||
|  | 	 * Thus, we do not use any on-cpu interrupt priority system either. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	int slot; | ||
|  | 	uint32_t mask; | ||
|  | 	uint32_t irqs; | ||
|  | 	void (*handler)(void *); | ||
|  | 	void *data; | ||
|  | 
 | ||
|  | 	/* For keeping track of how many bogus things happen in a row. */ | ||
|  | 	static int duds = 0; | ||
|  | 	int duds_this_time = 0; | ||
|  | 
 | ||
|  | 	/* and we better have a valid bus instance. */ | ||
|  | 	KASSERT(lamebus != NULL); | ||
|  | 
 | ||
|  | 	/* Lock the softc */ | ||
|  | 	spinlock_acquire(&lamebus->ls_lock); | ||
|  | 
 | ||
|  | 	/*
 | ||
|  | 	 * Read the LAMEbus controller register that tells us which | ||
|  | 	 * slots are asserting an interrupt condition. | ||
|  | 	 */ | ||
|  | 	irqs = read_ctl_register(lamebus, CTLREG_IRQS); | ||
|  | 
 | ||
|  | 	if (irqs == 0) { | ||
|  | 		/*
 | ||
|  | 		 * Huh? None of them? Must be a glitch. | ||
|  | 		 */ | ||
|  | 		kprintf("lamebus: stray interrupt on cpu %u\n", | ||
|  | 			curcpu->c_number); | ||
|  | 		duds++; | ||
|  | 		duds_this_time++; | ||
|  | 
 | ||
|  | 		/*
 | ||
|  | 		 * We could just return now, but instead we'll | ||
|  | 		 * continue ahead. Because irqs == 0, nothing in the | ||
|  | 		 * loop will execute, and passing through it gets us | ||
|  | 		 * to the code that checks how many duds we've | ||
|  | 		 * seen. This is important, because we just might get | ||
|  | 		 * a stray interrupt that latches itself on. If that | ||
|  | 		 * happens, we're pretty much toast, but it's better | ||
|  | 		 * to panic and hopefully reset the system than to | ||
|  | 		 * loop forever printing "stray interrupt". | ||
|  | 		 */ | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/*
 | ||
|  | 	 * Go through the bits in the value we got back to see which | ||
|  | 	 * ones are set. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	for (mask=1, slot=0; slot<LB_NSLOTS; mask<<=1, slot++) { | ||
|  | 		if ((irqs & mask) == 0) { | ||
|  | 			/* Nope. */ | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/*
 | ||
|  | 		 * This slot is signalling an interrupt. | ||
|  | 		 */ | ||
|  | 
 | ||
|  | 		if ((lamebus->ls_slotsinuse & mask)==0) { | ||
|  | 			/*
 | ||
|  | 			 * No device driver is using this slot. | ||
|  | 			 */ | ||
|  | 			duds++; | ||
|  | 			duds_this_time++; | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		if (lamebus->ls_irqfuncs[slot]==NULL) { | ||
|  | 			/*
 | ||
|  | 			 * The device driver hasn't installed an interrupt | ||
|  | 			 * handler. | ||
|  | 			 */ | ||
|  | 			duds++; | ||
|  | 			duds_this_time++; | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/*
 | ||
|  | 		 * Call the interrupt handler. Release the spinlock | ||
|  | 		 * while we do so, in case other CPUs are handling | ||
|  | 		 * interrupts on other devices. | ||
|  | 		 */ | ||
|  | 		handler = lamebus->ls_irqfuncs[slot]; | ||
|  | 		data = lamebus->ls_devdata[slot]; | ||
|  | 		spinlock_release(&lamebus->ls_lock); | ||
|  | 
 | ||
|  | 		handler(data); | ||
|  | 
 | ||
|  | 		spinlock_acquire(&lamebus->ls_lock); | ||
|  | 
 | ||
|  | 		/*
 | ||
|  | 		 * Reload the mask of pending IRQs - if we just called | ||
|  | 		 * hardclock, we might not have come back to this | ||
|  | 		 * context for some time, and it might have changed. | ||
|  | 		 */ | ||
|  | 
 | ||
|  | 		irqs = read_ctl_register(lamebus, CTLREG_IRQS); | ||
|  | 	} | ||
|  | 
 | ||
|  | 
 | ||
|  | 	/*
 | ||
|  | 	 * If we get interrupts for a slot with no driver or no | ||
|  | 	 * interrupt handler, it's fairly serious. Because LAMEbus | ||
|  | 	 * uses level-triggered interrupts, if we don't shut off the | ||
|  | 	 * condition, we'll keep getting interrupted continuously and | ||
|  | 	 * the system will make no progress. But we don't know how to | ||
|  | 	 * do that if there's no driver or no interrupt handler. | ||
|  | 	 * | ||
|  | 	 * So, if we get too many dud interrupts, panic, since it's | ||
|  | 	 * better to panic and reset than to hang. | ||
|  | 	 * | ||
|  | 	 * If we get through here without seeing any duds this time, | ||
|  | 	 * the condition, whatever it was, has gone away. It might be | ||
|  | 	 * some stupid device we don't have a driver for, or it might | ||
|  | 	 * have been an electrical transient. In any case, warn and | ||
|  | 	 * clear the dud count. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	if (duds_this_time == 0 && duds > 0) { | ||
|  | 		kprintf("lamebus: %d dud interrupts\n", duds); | ||
|  | 		duds = 0; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	if (duds > 10000) { | ||
|  | 		panic("lamebus: too many (%d) dud interrupts\n", duds); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Unlock the softc */ | ||
|  | 	spinlock_release(&lamebus->ls_lock); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Have the bus controller power the system off. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_poweroff(struct lamebus_softc *lamebus) | ||
|  | { | ||
|  | 	/*
 | ||
|  | 	 * Write 0 to the power register to shut the system off. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	cpu_irqoff(); | ||
|  | 	write_ctl_register(lamebus, CTLREG_PWR, 0); | ||
|  | 
 | ||
|  | 	/* The power doesn't go off instantly... so halt the cpu. */ | ||
|  | 	cpu_halt(); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Ask the bus controller how much memory we have. | ||
|  |  */ | ||
|  | uint32_t | ||
|  | lamebus_ramsize(void) | ||
|  | { | ||
|  | 	/*
 | ||
|  | 	 * Note that this has to work before bus initialization. | ||
|  | 	 * On machines where lamebus_read_register doesn't work | ||
|  | 	 * before bus initialization, this function can't be used | ||
|  | 	 * for initial RAM size lookup. | ||
|  | 	 */ | ||
|  | 
 | ||
|  | 	return read_ctl_register(NULL, CTLREG_RAMSZ); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Turn on or off the interprocessor interrupt line for a given CPU. | ||
|  |  */ | ||
|  | void | ||
|  | lamebus_assert_ipi(struct lamebus_softc *lamebus, struct cpu *target) | ||
|  | { | ||
|  | 	if (lamebus->ls_uniprocessor) { | ||
|  | 		return; | ||
|  | 	} | ||
|  | 	write_ctlcpu_register(lamebus, target->c_hardware_number, | ||
|  | 			      CTLCPU_CIPI, 1); | ||
|  | } | ||
|  | 
 | ||
|  | void | ||
|  | lamebus_clear_ipi(struct lamebus_softc *lamebus, struct cpu *target) | ||
|  | { | ||
|  | 	if (lamebus->ls_uniprocessor) { | ||
|  | 		return; | ||
|  | 	} | ||
|  | 	write_ctlcpu_register(lamebus, target->c_hardware_number, | ||
|  | 			      CTLCPU_CIPI, 0); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Initial setup. | ||
|  |  * Should be called from mainbus_bootstrap(). | ||
|  |  */ | ||
|  | struct lamebus_softc * | ||
|  | lamebus_init(void) | ||
|  | { | ||
|  | 	struct lamebus_softc *lamebus; | ||
|  | 	int i; | ||
|  | 
 | ||
|  | 	/* Allocate space for lamebus data */ | ||
|  | 	lamebus = kmalloc(sizeof(struct lamebus_softc)); | ||
|  | 	if (lamebus==NULL) { | ||
|  | 		panic("lamebus_init: Out of memory\n"); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	spinlock_init(&lamebus->ls_lock); | ||
|  | 
 | ||
|  | 	/*
 | ||
|  | 	 * Initialize the LAMEbus data structure. | ||
|  | 	 */ | ||
|  | 	lamebus->ls_slotsinuse = 1 << LB_CONTROLLER_SLOT; | ||
|  | 
 | ||
|  | 	for (i=0; i<LB_NSLOTS; i++) { | ||
|  | 		lamebus->ls_devdata[i] = NULL; | ||
|  | 		lamebus->ls_irqfuncs[i] = NULL; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	lamebus->ls_uniprocessor = 0; | ||
|  | 
 | ||
|  | 	return lamebus; | ||
|  | } |