NetBSD/sys/arch/x68k/dev/pow.c
2007-03-11 08:09:23 +00:00

312 lines
6.8 KiB
C

/* $NetBSD: pow.c,v 1.17 2007/03/11 08:09:25 isaki Exp $ */
/*
* Copyright (c) 1995 MINOURA Makoto.
* 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 Minoura Makoto.
* 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.
*/
/*
* Power switch device driver.
* Useful for
* 1. accessing boot information.
* 2. looking at the front or external power switch.
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: pow.c,v 1.17 2007/03/11 08:09:25 isaki Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/proc.h>
#include <sys/ioctl.h>
#include <sys/time.h>
#include <sys/kernel.h>
#include <sys/fcntl.h>
#include <sys/signalvar.h>
#include <sys/conf.h>
#include <machine/powioctl.h>
#include <x68k/dev/powvar.h>
#include <x68k/x68k/iodevice.h>
#include "pow.h"
#define sramtop (IODEVbase->io_sram)
#define rtc (IODEVbase->io_rtc)
struct pow_softc pows[NPOW];
void powattach(int);
void powintr(void);
static int setalarm(struct x68k_alarminfo *);
static void pow_check_switch(void *);
dev_type_open(powopen);
dev_type_close(powclose);
dev_type_ioctl(powioctl);
const struct cdevsw pow_cdevsw = {
powopen, powclose, noread, nowrite, powioctl,
nostop, notty, nopoll, nommap, nokqfilter,
};
/* ARGSUSED */
void
powattach(int num)
{
int minor;
int sw;
sw = ~mfp.gpip & 7;
mfp.ierb &= ~7; /* disable mfp power switch interrupt */
mfp.aer &= ~7;
for (minor = 0; minor < num; minor++) {
if (minor == 0)
pows[minor].status = POW_FREE;
else
pows[minor].status = POW_ANY;
pows[minor].sw = sw;
if (sw) {
mfp.aer |= sw;
mfp.ierb |= sw;
}
printf("pow%d: started by ", minor);
if (sw & POW_EXTERNALSW)
printf("external power switch.\n");
else if (sw & POW_FRONTSW)
printf("front power switch.\n");
/* XXX: I don't know why POW_ALARMSW should not be checked */
#if 0
else if ((sw & POW_ALARMSW) && sramtop[0x26] == 0)
printf("RTC alarm.\n");
else
printf("???.\n");
#else
else
printf("RTC alarm.\n");
#endif
}
shutdownhook_establish(pow_check_switch, 0);
}
/*ARGSUSED*/
int
powopen(dev_t dev, int flags, int mode, struct lwp *l)
{
struct pow_softc *sc = &pows[minor(dev)];
if (minor(dev) >= NPOW)
return EXDEV;
if (sc->status == POW_BUSY)
return EBUSY;
sc->pid = 0;
if (sc->status == POW_FREE)
sc->status = POW_BUSY;
sc->rw = (flags & (FREAD|FWRITE));
return 0;
}
/*ARGSUSED*/
int
powclose(dev_t dev, int flags, int mode, struct lwp *l)
{
struct pow_softc *sc = &pows[minor(dev)];
if (sc->status == POW_BUSY)
sc->status = POW_FREE;
sc->pid = 0;
return 0;
}
#define SRAMINT(offset) (*((volatile int *) (&sramtop[offset])))
#define RTCWAIT DELAY(100)
static int
setalarm(struct x68k_alarminfo *bp)
{
int s, ontime;
s = splclock();
sysport.sramwp = 0x31;
if (bp->al_enable) {
SRAMINT(0x1e) = bp->al_dowhat;
SRAMINT(0x22) = bp->al_ontime;
SRAMINT(0x14) = (bp->al_offtime / 60) - 1;
sramtop[0x26] = 0;
} else {
sramtop[0x26] = 7;
}
sysport.sramwp = 0;
rtc.bank0.mode = 9;
RTCWAIT;
rtc.bank1.reset = 5;
RTCWAIT;
if (bp->al_enable) {
ontime = bp->al_ontime;
if ((ontime & 0x0f) <= 9)
rtc.bank1.al_min = ontime & 0x0f;
RTCWAIT;
ontime >>= 4;
if ((ontime & 0x0f) <= 6)
rtc.bank1.al_min10 = ontime & 0x0f;
RTCWAIT;
ontime >>= 4;
if ((ontime & 0x0f) <= 9)
rtc.bank1.al_hour = ontime & 0x0f;
RTCWAIT;
ontime >>= 4;
if ((ontime & 0x0f) <= 2)
rtc.bank1.al_hour10 = ontime & 0x0f;
RTCWAIT;
ontime >>= 4;
if ((ontime & 0x0f) <= 9)
rtc.bank1.al_day = ontime & 0x0f;
RTCWAIT;
ontime >>= 4;
if ((ontime & 0x0f) <= 3)
rtc.bank1.al_day10 = ontime & 0x0f;
RTCWAIT;
ontime >>= 4;
if ((ontime & 0x0f) <= 6)
rtc.bank1.al_week = ontime & 0x0f;
RTCWAIT;
rtc.bank1.clkout = 0;
RTCWAIT;
rtc.bank1.mode = 0x0c;
} else {
rtc.bank1.clkout = 7;
RTCWAIT;
rtc.bank1.mode = 0x08;
}
splx(s);
return 0;
}
/*ARGSUSED*/
int
powioctl(dev_t dev, u_long cmd, void *addr, int flag, struct lwp *l)
{
struct pow_softc *sc = &pows[minor(dev)];
switch (cmd) {
case POWIOCGPOWERINFO:
{
struct x68k_powerinfo *bp = (void *)addr;
if (!(sc->rw & FREAD))
return EBADF;
bp->pow_switch_boottime = sc->sw;
bp->pow_switch_current = ~mfp.gpip & 7;
bp->pow_boottime = boottime.tv_sec;
bp->pow_bootcount = SRAMINT(0x44);
bp->pow_usedtotal = SRAMINT(0x40) * 60;
}
break;
case POWIOCGALARMINFO:
{
struct x68k_alarminfo *bp = (void *) addr;
if (!(sc->rw & FREAD))
return EBADF;
bp->al_enable = (sramtop[0x26] == 0);
bp->al_ontime = SRAMINT(0x22);
bp->al_dowhat = SRAMINT(0x1e);
bp->al_offtime = (SRAMINT(0x14) + 1) * 60;
}
break;
case POWIOCSALARMINFO:
if (!(sc->rw & FWRITE))
return EBADF;
return setalarm ((void *) addr);
case POWIOCSSIGNAL:
if (minor(dev) != 0)
return EOPNOTSUPP;
if (!(sc->rw & FWRITE))
return EBADF;
{
int signum = *(int *) addr;
if (signum <= 0 || signum > 31)
return EINVAL;
sc->signum = signum;
sc->proc = l->l_proc;
sc->pid = l->l_proc->p_pid;
}
break;
default:
return EINVAL;
}
return 0;
}
void
powintr(void)
{
int sw;
int s;
s = spl6();
sw = ~mfp.gpip & 6;
mfp.aer &= ~sw;
mfp.ierb |= sw;
if (pows[0].status == POW_BUSY && pows[0].pid != 0)
psignal(pows[0].proc, pows[0].signum);
splx(s);
}
static void
pow_check_switch(void *dummy)
{
extern int power_switch_is_off;
if ((~mfp.gpip & (POW_FRONTSW | POW_EXTERNALSW)) == 0)
power_switch_is_off = 1;
}