NetBSD/sys/arch/x86/pci/rdcpcib.c

302 lines
7.8 KiB
C

/* $NetBSD: rdcpcib.c,v 1.2 2011/07/01 18:22:08 dyoung Exp $ */
/*
* Copyright (c) 2011 Manuel Bouyer.
*
* 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.
*
* 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.
*/
/*
* driver for the RDC vortex86/PMX-1000 SoC PCI-ISA bridge, which also drives
* the watchdog timer
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: rdcpcib.c,v 1.2 2011/07/01 18:22:08 dyoung Exp $");
#include <sys/types.h>
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/sysctl.h>
#include <sys/timetc.h>
#include <sys/gpio.h>
#include <sys/bus.h>
#include <dev/pci/pcivar.h>
#include <dev/pci/pcireg.h>
#include <dev/pci/pcidevs.h>
#include <dev/gpio/gpiovar.h>
#include <dev/sysmon/sysmonvar.h>
#include "pcibvar.h"
/*
* special registers: iospace-registers for indirect access to timer and GPIO
*/
#define RDC_IND_BASE 0x22
#define RDC_IND_SIZE 0x2
#define RDC_IND_ADDR 0
#define RDC_IND_DATA 1
struct rdcpcib_softc {
struct pcib_softc rdc_pcib;
/* indirect registers mapping */
bus_space_tag_t rdc_iot;
bus_space_handle_t rdc_ioh;
/* Watchdog suppoprt */
struct sysmon_wdog rdc_smw;
};
static int rdcpcibmatch(device_t, cfdata_t, void *);
static void rdcpcibattach(device_t, device_t, void *);
static int rdcpcibdetach(device_t, int);
static uint8_t rdc_ind_read(struct rdcpcib_softc *, uint8_t);
static void rdc_ind_write(struct rdcpcib_softc *, uint8_t, uint8_t);
static void rdc_wdtimer_configure(device_t);
static int rdc_wdtimer_unconfigure(device_t, int);
static int rdc_wdtimer_setmode(struct sysmon_wdog *);
static int rdc_wdtimer_tickle(struct sysmon_wdog *);
static void rdc_wdtimer_stop(struct rdcpcib_softc *);
static void rdc_wdtimer_start(struct rdcpcib_softc *);
CFATTACH_DECL2_NEW(rdcpcib, sizeof(struct rdcpcib_softc),
rdcpcibmatch, rdcpcibattach, rdcpcibdetach, NULL,
pcibrescan, pcibchilddet);
static int
rdcpcibmatch(device_t parent, cfdata_t match, void *aux)
{
struct pci_attach_args *pa = aux;
if (PCI_CLASS(pa->pa_class) != PCI_CLASS_BRIDGE ||
PCI_SUBCLASS(pa->pa_class) != PCI_SUBCLASS_BRIDGE_ISA)
return 0;
if (PCI_VENDOR(pa->pa_id) == PCI_VENDOR_RDC &&
PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_RDC_PCIB)
return 10;
return 0;
}
static void
rdcpcibattach(device_t parent, device_t self, void *aux)
{
struct rdcpcib_softc *sc = device_private(self);
/* generic PCI/ISA bridge */
pcibattach(parent, self, aux);
/* map indirect registers */
sc->rdc_iot = x86_bus_space_io;
if (bus_space_map(sc->rdc_iot, RDC_IND_BASE, RDC_IND_SIZE, 0,
&sc->rdc_ioh) != 0) {
aprint_error_dev(self, "couldn't map indirect registers\n");
return;
}
/* Set up the watchdog. */
rdc_wdtimer_configure(self);
/* Install power handler XXX */
if (!pmf_device_register(self, NULL, NULL))
aprint_error_dev(self, "couldn't establish power handler\n");
}
static int
rdcpcibdetach(device_t self, int flags)
{
struct rdcpcib_softc *sc = device_private(self);
int rc;
pmf_device_deregister(self);
if ((rc = rdc_wdtimer_unconfigure(self, flags)) != 0)
return rc;
bus_space_unmap(sc->rdc_iot, sc->rdc_ioh, RDC_IND_SIZE);
return pcibdetach(self, flags);
}
/* indirect registers read/write */
static uint8_t
rdc_ind_read(struct rdcpcib_softc *sc, uint8_t addr)
{
bus_space_write_1(sc->rdc_iot, sc->rdc_ioh, RDC_IND_ADDR, addr);
return bus_space_read_1(sc->rdc_iot, sc->rdc_ioh, RDC_IND_DATA);
}
static void
rdc_ind_write(struct rdcpcib_softc *sc, uint8_t addr, uint8_t data)
{
bus_space_write_1(sc->rdc_iot, sc->rdc_ioh, RDC_IND_ADDR, addr);
bus_space_write_1(sc->rdc_iot, sc->rdc_ioh, RDC_IND_DATA, data);
}
/*
* watchdog timer registers
*/
/* control */
#define RDC_WDT0_CTRL 0x37
#define RDC_WDT0_CTRL_EN 0x40
/* signal select */
#define RDC_WDT0_SSEL 0x38
#define RDC_WDT0_SSEL_MSK 0xf0
#define RDC_WDT0_SSEL_NMI 0xc0
#define RDC_WDT0_SSEL_RST 0xd0
/* counter */
#define RDC_WDT0_CNTL 0x39
#define RDC_WDT0_CNTH 0x3A
#define RDC_WDT0_CNTU 0x3B
#define RDC_WDT0_FREQ 32768 /* Hz */
#define RDC_WDT0_PERIOD_MAX (1 << 24)
/* clear counter */
#define RDC_WDT0_CTRL1 0x3c
#define RDC_WDT0_CTRL1_RELOAD 0x40
#define RDC_WDT0_CRTL1_FIRE 0x80
/*
* Initialize the watchdog timer.
*/
static void
rdc_wdtimer_configure(device_t self)
{
struct rdcpcib_softc *sc = device_private(self);
uint8_t reg;
/* Explicitly stop the timer. */
rdc_wdtimer_stop(sc);
/*
* Register the driver with the sysmon watchdog framework.
*/
sc->rdc_smw.smw_name = device_xname(self);
sc->rdc_smw.smw_cookie = sc;
sc->rdc_smw.smw_setmode = rdc_wdtimer_setmode;
sc->rdc_smw.smw_tickle = rdc_wdtimer_tickle;
sc->rdc_smw.smw_period = RDC_WDT0_PERIOD_MAX / RDC_WDT0_FREQ;
if (sysmon_wdog_register(&sc->rdc_smw)) {
aprint_error_dev(self, "unable to register wdt"
"as a sysmon watchdog device.\n");
return;
}
aprint_verbose_dev(self, "watchdog timer configured.\n");
reg = rdc_ind_read(sc, RDC_WDT0_CTRL1);
if (reg & RDC_WDT0_CRTL1_FIRE) {
aprint_error_dev(self, "watchdog fired bit set, clearing\n");
rdc_ind_write(sc, RDC_WDT0_CTRL1, reg & ~RDC_WDT0_CRTL1_FIRE);
}
}
static int
rdc_wdtimer_unconfigure(device_t self, int flags)
{
struct rdcpcib_softc *sc = device_private(self);
int rc;
if ((rc = sysmon_wdog_unregister(&sc->rdc_smw)) != 0) {
if (rc == ERESTART)
rc = EINTR;
return rc;
}
/* Explicitly stop the timer. */
rdc_wdtimer_stop(sc);
return 0;
}
/*
* Sysmon watchdog callbacks.
*/
static int
rdc_wdtimer_setmode(struct sysmon_wdog *smw)
{
struct rdcpcib_softc *sc = smw->smw_cookie;
unsigned int period;
if ((smw->smw_mode & WDOG_MODE_MASK) == WDOG_MODE_DISARMED) {
/* Stop the timer. */
rdc_wdtimer_stop(sc);
} else {
period = smw->smw_period * RDC_WDT0_FREQ;
if (period < 1 ||
period > RDC_WDT0_PERIOD_MAX)
return EINVAL;
period = period - 1;
/* Stop the timer, */
rdc_wdtimer_stop(sc);
/* set the timeout, */
rdc_ind_write(sc, RDC_WDT0_CNTL, (period >> 0) & 0xff);
rdc_ind_write(sc, RDC_WDT0_CNTH, (period >> 8) & 0xff);
rdc_ind_write(sc, RDC_WDT0_CNTU, (period >> 16) & 0xff);
/* and start the timer again */
rdc_wdtimer_start(sc);
}
return 0;
}
static int
rdc_wdtimer_tickle(struct sysmon_wdog *smw)
{
struct rdcpcib_softc *sc = smw->smw_cookie;
uint8_t reg;
reg = rdc_ind_read(sc, RDC_WDT0_CTRL1);
rdc_ind_write(sc, RDC_WDT0_CTRL1, reg | RDC_WDT0_CTRL1_RELOAD);
return 0;
}
static void
rdc_wdtimer_stop(struct rdcpcib_softc *sc)
{
uint8_t reg;
reg = rdc_ind_read(sc, RDC_WDT0_CTRL);
rdc_ind_write(sc, RDC_WDT0_CTRL, reg & ~RDC_WDT0_CTRL_EN);
}
static void
rdc_wdtimer_start(struct rdcpcib_softc *sc)
{
uint8_t reg;
rdc_ind_write(sc, RDC_WDT0_SSEL, RDC_WDT0_SSEL_RST);
reg = rdc_ind_read(sc, RDC_WDT0_CTRL);
rdc_ind_write(sc, RDC_WDT0_CTRL, reg | RDC_WDT0_CTRL_EN);
}