142c2a33ba
by Slava Semushin <slava.semushin@gmail.com>. To verify that no nasty side effects of duplicate includes (or their removal) have an effect here, I've compiled an i386/ALL kernel with and without the patch, and the only difference in the resulting .o files was in shifted line numbers in some assert() calls. The comparison of the .o files was based on the output of "objdump -D". Thanks to martin@ for the input on testing.
273 lines
6.8 KiB
C
273 lines
6.8 KiB
C
/* $NetBSD: panel.c,v 1.11 2007/01/24 13:08:14 hubertf Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2002 Dennis I. Chernoivanov
|
|
* 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. 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/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: panel.c,v 1.11 2007/01/24 13:08:14 hubertf Exp $");
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/poll.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/uio.h>
|
|
#include <sys/types.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/device.h>
|
|
#include <sys/callout.h>
|
|
#include <sys/select.h>
|
|
|
|
#include <machine/bus.h>
|
|
#include <machine/autoconf.h>
|
|
|
|
#include <dev/ic/hd44780reg.h>
|
|
#include <dev/ic/hd44780var.h>
|
|
#include <dev/ic/lcdkp_subr.h>
|
|
|
|
#include "ioconf.h"
|
|
|
|
#define PANEL_POLLRATE (hz / 10)
|
|
#define PANEL_REGION 0x20
|
|
#define DATA_OFFSET 0x10
|
|
|
|
struct panel_softc {
|
|
struct device sc_dev;
|
|
|
|
struct hd44780_chip sc_lcd;
|
|
struct lcdkp_chip sc_kp;
|
|
|
|
struct selinfo sc_selq;
|
|
struct callout sc_callout;
|
|
};
|
|
|
|
static int panel_match(struct device *, struct cfdata *, void *);
|
|
static void panel_attach(struct device *, struct device *, void *);
|
|
|
|
static void panel_soft(void *);
|
|
|
|
static uint8_t panel_cbt_kprread(bus_space_tag_t, bus_space_handle_t);
|
|
static uint8_t panel_cbt_hdreadreg(struct hd44780_chip *, uint32_t, uint32_t);
|
|
static void panel_cbt_hdwritereg(struct hd44780_chip *, uint32_t, uint32_t,
|
|
uint8_t);
|
|
|
|
dev_type_open(panelopen);
|
|
dev_type_close(panelclose);
|
|
dev_type_read(panelread);
|
|
dev_type_write(panelwrite);
|
|
dev_type_ioctl(panelioctl);
|
|
dev_type_poll(panelpoll);
|
|
|
|
const struct cdevsw panel_cdevsw = {
|
|
panelopen, panelclose, panelread, panelwrite, panelioctl,
|
|
nostop, notty, panelpoll, nommap,
|
|
};
|
|
|
|
CFATTACH_DECL(panel, sizeof(struct panel_softc),
|
|
panel_match, panel_attach, NULL, NULL);
|
|
|
|
static int
|
|
panel_match(struct device *parent, struct cfdata *match, void *aux)
|
|
{
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
panel_attach(struct device *parent, struct device *self, void *aux)
|
|
{
|
|
struct panel_softc *sc = (void *)self;
|
|
struct mainbus_attach_args *maa = aux;
|
|
|
|
static struct lcdkp_xlate keys[] = {
|
|
{ 0xfa, 'h' },
|
|
{ 0xf6, 'k' },
|
|
{ 0xde, 'l' },
|
|
{ 0xee, 'j' },
|
|
{ 0x7e, 's' },
|
|
{ 0xbe, 'e' }
|
|
};
|
|
|
|
sc->sc_lcd.sc_iot = maa->ma_iot;
|
|
if (bus_space_map(sc->sc_lcd.sc_iot, maa->ma_addr, PANEL_REGION,
|
|
0, &sc->sc_lcd.sc_ioir)) {
|
|
printf(": unable to map registers\n");
|
|
return;
|
|
}
|
|
bus_space_subregion(sc->sc_lcd.sc_iot, sc->sc_lcd.sc_ioir, DATA_OFFSET,
|
|
1, &sc->sc_lcd.sc_iodr);
|
|
|
|
sc->sc_lcd.sc_dev_ok = 1;
|
|
sc->sc_lcd.sc_cols = 16;
|
|
sc->sc_lcd.sc_vcols = 40;
|
|
sc->sc_lcd.sc_flags = HD_8BIT | HD_MULTILINE | HD_KEYPAD;
|
|
|
|
sc->sc_lcd.sc_writereg = panel_cbt_hdwritereg;
|
|
sc->sc_lcd.sc_readreg = panel_cbt_hdreadreg;
|
|
sc->sc_lcd.sc_dev = self;
|
|
|
|
hd44780_attach_subr(&sc->sc_lcd);
|
|
|
|
sc->sc_kp.sc_iot = maa->ma_iot;
|
|
sc->sc_kp.sc_ioh = MIPS_PHYS_TO_KSEG1(0x1d000000); /* XXX */
|
|
|
|
sc->sc_kp.sc_knum = sizeof(keys) / sizeof(struct lcdkp_xlate);
|
|
sc->sc_kp.sc_kpad = keys;
|
|
sc->sc_kp.sc_rread = panel_cbt_kprread;
|
|
|
|
lcdkp_attach_subr(&sc->sc_kp);
|
|
|
|
callout_init(&sc->sc_callout);
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
static uint8_t
|
|
panel_cbt_kprread(bus_space_tag_t iot, bus_space_handle_t ioh)
|
|
{
|
|
|
|
delay(HD_TIMEOUT_NORMAL);
|
|
return (bus_space_read_4(iot, ioh, 0x00) >> 24) & 0xff;
|
|
}
|
|
|
|
|
|
static void
|
|
panel_cbt_hdwritereg(struct hd44780_chip *hd, uint32_t en, uint32_t rs,
|
|
uint8_t dat)
|
|
{
|
|
|
|
if (rs)
|
|
bus_space_write_4(hd->sc_iot, hd->sc_iodr, 0x00, dat << 24);
|
|
else
|
|
bus_space_write_4(hd->sc_iot, hd->sc_ioir, 0x00, dat << 24);
|
|
delay(HD_TIMEOUT_NORMAL);
|
|
}
|
|
|
|
static uint8_t
|
|
panel_cbt_hdreadreg(struct hd44780_chip *hd, uint32_t en, uint32_t rs)
|
|
{
|
|
|
|
delay(HD_TIMEOUT_NORMAL);
|
|
if (rs)
|
|
return (bus_space_read_4(hd->sc_iot, hd->sc_iodr, 0x00) >> 24)
|
|
& 0xff;
|
|
else
|
|
return (bus_space_read_4(hd->sc_iot, hd->sc_ioir, 0x00) >> 24)
|
|
& 0xff;
|
|
}
|
|
|
|
int
|
|
panelopen(dev_t dev, int flag, int mode, struct lwp *l)
|
|
{
|
|
struct panel_softc *sc = device_lookup(&panel_cd, minor(dev));
|
|
|
|
return (sc->sc_lcd.sc_dev_ok == 0) ? ENXIO : 0;
|
|
}
|
|
|
|
int
|
|
panelclose(dev_t dev, int flag, int mode, struct lwp *l)
|
|
{
|
|
struct panel_softc *sc = device_lookup(&panel_cd, minor(dev));
|
|
|
|
selwakeup(&sc->sc_selq);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
panelread(dev_t dev, struct uio *uio, int flag)
|
|
{
|
|
int error;
|
|
uint8_t b;
|
|
struct panel_softc *sc = device_lookup(&panel_cd, minor(dev));
|
|
|
|
if (uio->uio_resid < sizeof(b))
|
|
return EIO;
|
|
|
|
if ((error = lcdkp_readkey(&sc->sc_kp, &b)) != 0)
|
|
return error;
|
|
|
|
return uiomove((void*)&b, sizeof(b), uio);
|
|
}
|
|
|
|
int
|
|
panelwrite(dev_t dev, struct uio *uio, int flag)
|
|
{
|
|
int error;
|
|
struct hd44780_io io;
|
|
struct panel_softc *sc = device_lookup(&panel_cd, minor(dev));
|
|
|
|
io.dat = 0;
|
|
io.len = uio->uio_resid;
|
|
if (io.len > HD_MAX_CHARS)
|
|
io.len = HD_MAX_CHARS;
|
|
|
|
if ((error = uiomove((void*)io.buf, io.len, uio)) != 0)
|
|
return error;
|
|
|
|
hd44780_ddram_redraw(&sc->sc_lcd, 0, &io);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
panelioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct lwp *l)
|
|
{
|
|
struct panel_softc *sc = device_lookup(&panel_cd, minor(dev));
|
|
|
|
return hd44780_ioctl_subr(&sc->sc_lcd, cmd, data);
|
|
}
|
|
|
|
int
|
|
panelpoll(dev_t dev, int events, struct lwp *l)
|
|
{
|
|
int revents = 0;
|
|
|
|
if ((events & (POLLIN | POLLRDNORM)) != 0) {
|
|
struct panel_softc *sc = device_lookup(&panel_cd, minor(dev));
|
|
|
|
if (lcdkp_scankey(&sc->sc_kp) != 0) {
|
|
revents = events & (POLLIN | POLLRDNORM);
|
|
} else {
|
|
selrecord(l, &sc->sc_selq);
|
|
callout_reset(&sc->sc_callout, PANEL_POLLRATE,
|
|
panel_soft, sc);
|
|
}
|
|
}
|
|
|
|
return revents;
|
|
}
|
|
|
|
static void
|
|
panel_soft(void *arg)
|
|
{
|
|
struct panel_softc *sc = arg;
|
|
|
|
if (lcdkp_scankey(&sc->sc_kp) != 0)
|
|
selwakeup(&sc->sc_selq);
|
|
else
|
|
callout_reset(&sc->sc_callout, PANEL_POLLRATE, panel_soft, sc);
|
|
}
|