summaryrefslogtreecommitdiff
path: root/sys/arch/sparc64/dev/sbbc.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/sparc64/dev/sbbc.c')
-rw-r--r--sys/arch/sparc64/dev/sbbc.c201
1 files changed, 201 insertions, 0 deletions
diff --git a/sys/arch/sparc64/dev/sbbc.c b/sys/arch/sparc64/dev/sbbc.c
new file mode 100644
index 00000000000..9bc9e033def
--- /dev/null
+++ b/sys/arch/sparc64/dev/sbbc.c
@@ -0,0 +1,201 @@
+/* $OpenBSD: sbbc.c,v 1.1 2008/07/06 07:27:43 kettenis Exp $ */
+/*
+ * Copyright (c) 2008 Mark Kettenis
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+#include <sys/param.h>
+#include <sys/device.h>
+#include <sys/malloc.h>
+#include <sys/systm.h>
+
+#include <machine/autoconf.h>
+#include <machine/openfirm.h>
+
+#include <dev/pci/pcidevs.h>
+#include <dev/pci/pcireg.h>
+#include <dev/pci/pcivar.h>
+
+#include <dev/clock_subr.h>
+
+extern todr_chip_handle_t todr_handle;
+
+#define SBBC_PCI_BAR PCI_MAPREG_START
+
+#define SBBC_REGS_OFFSET 0x800000
+#define SBBC_REGS_SIZE 0x6230
+#define SBBC_SRAM_OFFSET 0x900000
+#define SBBC_SRAM_SIZE 0x20000 /* 128KB SRAM */
+
+#define SBBC_MAX_TAGS 32
+
+struct sbbc_sram_tag {
+ char tag_key[8];
+ uint32_t tag_size;
+ uint32_t tag_offset;
+};
+
+struct sbbc_sram_toc {
+ char toc_magic[8];
+ uint8_t toc_reserved;
+ uint8_t toc_type;
+ uint16_t toc_version;
+ uint32_t toc_ntags;
+ struct sbbc_sram_tag toc_tag[SBBC_MAX_TAGS];
+};
+
+/* Time of day service. */
+struct sbbc_sram_tod {
+ uint32_t tod_magic;
+ uint32_t tod_version;
+ uint64_t tod_time;
+ uint64_t tod_skew;
+ uint32_t tod_reserved;
+ uint32_t tod_heartbeat;
+ uint32_t tod_timeout;
+};
+
+#define SBBC_TOD_MAGIC 0x54443100
+#define SBBC_TOD_VERSION 1
+
+struct sbbc_softc {
+ struct device sc_dv;
+ bus_space_tag_t sc_iot;
+ bus_space_handle_t sc_reg_ioh;
+ bus_space_handle_t sc_sram_ioh;
+ caddr_t sc_sram;
+ uint32_t sc_sram_toc;
+
+ struct sparc_bus_space_tag sc_bbt;
+};
+
+int sbbc_match(struct device *, void *, void *);
+void sbbc_attach(struct device *, struct device *, void *);
+
+struct cfattach sbbc_ca = {
+ sizeof(struct sbbc_softc), sbbc_match, sbbc_attach
+};
+
+struct cfdriver sbbc_cd = {
+ NULL, "sbbc", DV_DULL
+};
+
+void sbbc_attach_tod(struct sbbc_softc *, uint32_t);
+int sbbc_tod_gettime(todr_chip_handle_t, struct timeval *);
+int sbbc_tod_settime(todr_chip_handle_t, struct timeval *);
+
+int
+sbbc_match(struct device *parent, void *match, void *aux)
+{
+ struct pci_attach_args *pa = aux;
+
+ if (PCI_VENDOR(pa->pa_id) == PCI_VENDOR_SUN &&
+ (PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_SUN_SBBC))
+ return (1);
+
+ return (0);
+}
+
+void
+sbbc_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct sbbc_softc *sc = (void *)self;
+ struct pci_attach_args *pa = aux;
+ struct sbbc_sram_toc *toc;
+ bus_addr_t base;
+ bus_size_t size;
+ int chosen, iosram;
+ int i;
+
+ /* XXX Don't byteswap. */
+ sc->sc_bbt = *pa->pa_memt;
+ sc->sc_bbt.sasi = ASI_PRIMARY;
+ sc->sc_iot = &sc->sc_bbt;
+
+ if (pci_mapreg_info(pa->pa_pc, pa->pa_tag, SBBC_PCI_BAR,
+ PCI_MAPREG_TYPE_MEM, &base, &size, NULL)) {
+ printf(": can't find register space\n");
+ return;
+ }
+
+ if (bus_space_map(sc->sc_iot, base + SBBC_SRAM_OFFSET,
+ SBBC_SRAM_SIZE, 0, &sc->sc_sram_ioh)) {
+ printf(": can't map SRAM\n");
+ return;
+ }
+
+ printf("\n");
+
+ /* Check if we are the chosen one. */
+ chosen = OF_finddevice("/chosen");
+ if (OF_getprop(chosen, "iosram", &iosram, sizeof(iosram)) <= 0 ||
+ PCITAG_NODE(pa->pa_tag) != iosram)
+ return;
+
+ /* SRAM TOC offset defaults to 0. */
+ if (OF_getprop(chosen, "iosram-toc", &sc->sc_sram_toc,
+ sizeof(sc->sc_sram_toc)) <= 0)
+ sc->sc_sram_toc = 0;
+
+ sc->sc_sram = bus_space_vaddr(sc->sc_iot, sc->sc_sram_ioh);
+ toc = (struct sbbc_sram_toc *)(sc->sc_sram + sc->sc_sram_toc);
+
+ for (i = 0; i < toc->toc_ntags; i++) {
+ if (strcmp(toc->toc_tag[i].tag_key, "TODDATA") == 0)
+ sbbc_attach_tod(sc, toc->toc_tag[i].tag_offset);
+ }
+}
+
+void
+sbbc_attach_tod(struct sbbc_softc *sc, uint32_t offset)
+{
+ struct sbbc_sram_tod *tod;
+ todr_chip_handle_t handle;
+
+ tod = (struct sbbc_sram_tod *)(sc->sc_sram + offset);
+ if (tod->tod_magic != SBBC_TOD_MAGIC ||
+ tod->tod_version < SBBC_TOD_VERSION)
+ return;
+
+ handle = malloc(sizeof(struct todr_chip_handle), M_DEVBUF, M_NOWAIT);
+ if (handle == NULL)
+ panic("couldn't allocate todr_handle");
+
+ handle->cookie = tod;
+ handle->todr_gettime = sbbc_tod_gettime;
+ handle->todr_settime = sbbc_tod_settime;
+
+ handle->bus_cookie = NULL;
+ handle->todr_setwen = NULL;
+ todr_handle = handle;
+}
+
+int
+sbbc_tod_gettime(todr_chip_handle_t handle, struct timeval *tv)
+{
+ struct sbbc_sram_tod *tod = handle->cookie;
+
+ tv->tv_sec = tod->tod_time + tod->tod_skew;
+ tv->tv_usec = 0;
+ return (0);
+}
+
+int
+sbbc_tod_settime(todr_chip_handle_t handle, struct timeval *tv)
+{
+ struct sbbc_sram_tod *tod = handle->cookie;
+
+ tod->tod_skew = tv->tv_sec - tod->tod_time;
+ return (0);
+}