NetBSD/sys/arch/sun3/sun3/leds.c

156 lines
4.2 KiB
C

/* $NetBSD: leds.c,v 1.5 1998/02/05 04:57:38 gwr Exp $ */
/*-
* Copyright (c) 1997 The NetBSD Foundation, Inc.
* All rights reserved.
*
* This code is derived from software contributed to The NetBSD Foundation
* by Gordon W. Ross and der Mouse.
*
* 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 the NetBSD
* Foundation, Inc. and its contributors.
* 4. Neither the name of The NetBSD Foundation 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 NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
*/
/*
* Functions to flash the LEDs with some pattern.
* The 3/80 has just one LED on the front panel,
* which we turn on during boot and leave alone.
* All other Sun3 machines have an 8-position LED
* array in which some pattern is animated.
*/
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/conf.h>
#include <sys/buf.h>
#include <sys/malloc.h>
#include <sys/proc.h>
#include <machine/autoconf.h>
#include <machine/idprom.h>
#include <machine/leds.h>
#include <sun3/sun3/machdep.h>
#ifdef _SUN3_
#include <sun3/sun3/control.h>
#endif
#ifdef _SUN3X_
#include <sun3/sun3x/obio.h>
static volatile u_int8_t *diagreg;
#endif
static u_char led_countdown = 0;
static u_char led_px = 0;
/*
* Initial value is the default pattern set. Note, only bit zero
* shows on the 3/80, and it is active-high (3/470 is active-low).
* We normally want the 3/80's LED to just stay ON, so the 3/80
* will change the default patlen to one, causing the LED to keep
* using the first byte of the pattern where bit zero is ON.
*/
static struct led_patterns ledpat = {
8, /* divisor */
8, /* patlen */
{ /* patterns */
0x0F, 0x1E, 0x3C, 0x78,
0xF0, 0xE1, 0xC3, 0x87,
}
};
/*
* This is called early during startup to find the
* diag register (LEDs) and turn on the light(s).
*/
void
leds_init()
{
#ifdef _SUN3X_
diagreg = obio_find_mapping(OBIO_DIAGREG, 1);
if (cpu_machine_id == SUN3X_MACH_80)
ledpat.patlen = 1;
#endif /* SUN3X */
/* Turn on some lights. */
leds_intr();
}
/*
* This is called by the clock interrupt.
*/
void
leds_intr()
{
register u_char i;
if (led_countdown) {
led_countdown--;
return;
}
led_countdown = ledpat.divisor - 1;
i = led_px;
#ifdef _SUN3X_
*diagreg = (char) ledpat.pat[i];
#else
set_control_byte(DIAG_REG, ledpat.pat[i]);
#endif
i = i+1;
if (i == ledpat.patlen)
i = 0;
led_px = i;
}
/*
* This is called by mem.c to handle /dev/leds
*/
int
leds_uio(struct uio *uio)
{
int cnt, error;
int off; /* NOT off_t */
caddr_t va;
off = uio->uio_offset;
if ((off < 0) || (off > sizeof(ledpat)))
return (EIO);
cnt = min(uio->uio_resid, (sizeof(ledpat) - off));
if (cnt == 0)
return (0); /* EOF */
va = ((char*)(&ledpat)) + off;
error = uiomove(va, cnt, uio);
return (error);
}