summaryrefslogtreecommitdiff
path: root/sys/arch/landisk/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/landisk/dev')
-rw-r--r--sys/arch/landisk/dev/obio.c955
-rw-r--r--sys/arch/landisk/dev/obiovar.h5
-rw-r--r--sys/arch/landisk/dev/rs5c313.c437
-rw-r--r--sys/arch/landisk/dev/wdc_obio.c142
4 files changed, 1537 insertions, 2 deletions
diff --git a/sys/arch/landisk/dev/obio.c b/sys/arch/landisk/dev/obio.c
new file mode 100644
index 00000000000..13aa0156a71
--- /dev/null
+++ b/sys/arch/landisk/dev/obio.c
@@ -0,0 +1,955 @@
+/* $OpenBSD: obio.c,v 1.1 2006/10/07 20:52:40 miod Exp $ */
+/* $NetBSD: obio.c,v 1.1 2006/09/01 21:26:18 uwe Exp $ */
+
+/*-
+ * Copyright (c) 1998 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Charles M. Hannum.
+ *
+ * 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.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+
+#include <uvm/uvm_extern.h>
+
+#include <sh/devreg.h>
+#include <sh/mmu.h>
+#include <sh/pmap.h>
+#include <sh/pte.h>
+
+#include <machine/bus.h>
+#include <machine/cpu.h>
+#include <machine/intr.h>
+
+#include <landisk/dev/obiovar.h>
+
+int obio_match(struct device *, void *, void *);
+void obio_attach(struct device *, struct device *, void *);
+int obio_print(void *, const char *);
+int obio_search(struct device *, void *, void *);
+
+struct cfattach obio_ca = {
+ sizeof(struct device), obio_match, obio_attach
+};
+
+struct cfdriver obio_cd = {
+ 0, "obio", DV_DULL
+};
+
+int
+obio_match(struct device *parent, void *vcf, void *aux)
+{
+ struct obiobus_attach_args *oba = aux;
+
+ if (strcmp(oba->oba_busname, obio_cd.cd_name) != 0)
+ return (0);
+
+ return (1);
+}
+
+void
+obio_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct obio_softc *sc = (struct obio_softc *)self;
+ struct obiobus_attach_args *oba = aux;
+
+ printf("\n");
+
+ sc->sc_iot = oba->oba_iot;
+ sc->sc_memt = oba->oba_memt;
+
+ config_search(obio_search, self, NULL);
+}
+
+int
+obio_search(struct device *parent, void *vcf, void *aux)
+{
+ struct obio_softc *sc = (struct obio_softc *)parent;
+ struct cfdata *cf = vcf;
+ struct obio_attach_args oa;
+ struct obio_io res_io[1];
+ struct obio_iomem res_mem[1];
+ struct obio_irq res_irq[1];
+
+ oa.oa_iot = sc->sc_iot;
+ oa.oa_memt = sc->sc_memt;
+
+ res_io[0].or_addr = cf->cf_iobase;
+ res_io[0].or_size = cf->cf_iosize;
+
+ res_mem[0].or_addr = cf->cf_maddr;
+ res_mem[0].or_size = cf->cf_msize;
+
+ res_irq[0].or_irq = cf->cf_irq;
+
+ oa.oa_io = res_io;
+ oa.oa_nio = 1;
+
+ oa.oa_iomem = res_mem;
+ oa.oa_niomem = 1;
+
+ oa.oa_irq = res_irq;
+ oa.oa_nirq = 1;
+
+ if ((*cf->cf_attach->ca_match)(parent, cf, &oa) == 0)
+ return (0);
+
+ config_attach(parent, cf, &oa, obio_print);
+ return (1);
+}
+
+int
+obio_print(void *args, const char *name)
+{
+ struct obio_attach_args *oa = args;
+ const char *sep;
+ int i;
+
+ if (oa->oa_nio) {
+ sep = "";
+ printf(" port ");
+ for (i = 0; i < oa->oa_nio; i++) {
+ if (oa->oa_io[i].or_size == 0)
+ continue;
+ printf("%s0x%x", sep, oa->oa_io[i].or_addr);
+ if (oa->oa_io[i].or_size > 1)
+ printf("-0x%x", oa->oa_io[i].or_addr +
+ oa->oa_io[i].or_size - 1);
+ sep = ",";
+ }
+ }
+
+ if (oa->oa_niomem) {
+ sep = "";
+ printf(" iomem ");
+ for (i = 0; i < oa->oa_niomem; i++) {
+ if (oa->oa_iomem[i].or_size == 0)
+ continue;
+ printf("%s0x%x", sep, oa->oa_iomem[i].or_addr);
+ if (oa->oa_iomem[i].or_size > 1)
+ printf("-0x%x", oa->oa_iomem[i].or_addr +
+ oa->oa_iomem[i].or_size - 1);
+ sep = ",";
+ }
+ }
+
+ if (oa->oa_nirq) {
+ sep = "";
+ printf(" irq ");
+ for (i = 0; i < oa->oa_nirq; i++) {
+ if (oa->oa_irq[i].or_irq == IRQUNK)
+ continue;
+ printf("%s%d", sep, oa->oa_irq[i].or_irq);
+ sep = ",";
+ }
+ }
+
+ return (UNCONF);
+}
+
+/*
+ * Set up an interrupt handler to start being called.
+ */
+void *
+obio_intr_establish(int irq, int level, int (*ih_fun)(void *), void *ih_arg,
+ const char *ih_name)
+{
+ return extintr_establish(irq, level, ih_fun, ih_arg, ih_name);
+}
+
+/*
+ * Deregister an interrupt handler.
+ */
+void
+obio_intr_disestablish(void *arg)
+{
+ extintr_disestablish(arg);
+}
+
+/*
+ * on-board I/O bus space
+ */
+#define OBIO_IOMEM_IO 0 /* space is i/o space */
+#define OBIO_IOMEM_MEM 1 /* space is mem space */
+#define OBIO_IOMEM_PCMCIA_IO 2 /* PCMCIA IO space */
+#define OBIO_IOMEM_PCMCIA_MEM 3 /* PCMCIA Mem space */
+#define OBIO_IOMEM_PCMCIA_ATT 4 /* PCMCIA Attr space */
+#define OBIO_IOMEM_PCMCIA_8BIT 0x8000 /* PCMCIA BUS 8 BIT WIDTH */
+#define OBIO_IOMEM_PCMCIA_IO8 \
+ (OBIO_IOMEM_PCMCIA_IO|OBIO_IOMEM_PCMCIA_8BIT)
+#define OBIO_IOMEM_PCMCIA_MEM8 \
+ (OBIO_IOMEM_PCMCIA_MEM|OBIO_IOMEM_PCMCIA_8BIT)
+#define OBIO_IOMEM_PCMCIA_ATT8 \
+ (OBIO_IOMEM_PCMCIA_ATT|OBIO_IOMEM_PCMCIA_8BIT)
+
+int obio_iomem_map(void *v, bus_addr_t bpa, bus_size_t size, int flags,
+ bus_space_handle_t *bshp);
+void obio_iomem_unmap(void *v, bus_space_handle_t bsh, bus_size_t size);
+int obio_iomem_subregion(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, bus_size_t size, bus_space_handle_t *nbshp);
+int obio_iomem_alloc(void *v, bus_addr_t rstart, bus_addr_t rend,
+ bus_size_t size, bus_size_t alignment, bus_size_t boundary, int flags,
+ bus_addr_t *bpap, bus_space_handle_t *bshp);
+void obio_iomem_free(void *v, bus_space_handle_t bsh, bus_size_t size);
+
+int obio_iomem_add_mapping(bus_addr_t, bus_size_t, int,
+ bus_space_handle_t *);
+
+int
+obio_iomem_add_mapping(bus_addr_t bpa, bus_size_t size, int type,
+ bus_space_handle_t *bshp)
+{
+ u_long pa, endpa;
+ vaddr_t va;
+ pt_entry_t *pte;
+ unsigned int m = 0;
+ int io_type = type & ~OBIO_IOMEM_PCMCIA_8BIT;
+
+ pa = trunc_page(bpa);
+ endpa = round_page(bpa + size);
+
+#ifdef DIAGNOSTIC
+ if (endpa <= pa)
+ panic("obio_iomem_add_mapping: overflow");
+#endif
+
+ va = uvm_km_alloc(kernel_map, endpa - pa);
+ if (va == 0){
+ printf("obio_iomem_add_mapping: nomem\n");
+ return (ENOMEM);
+ }
+
+ *bshp = (bus_space_handle_t)(va + (bpa & PGOFSET));
+
+#define MODE(t, s) \
+ ((t) & OBIO_IOMEM_PCMCIA_8BIT) ? \
+ _PG_PCMCIA_ ## s ## 8 : \
+ _PG_PCMCIA_ ## s ## 16
+ switch (io_type) {
+ default:
+ panic("unknown pcmcia space.");
+ /* NOTREACHED */
+ case OBIO_IOMEM_PCMCIA_IO:
+ m = MODE(type, IO);
+ break;
+ case OBIO_IOMEM_PCMCIA_MEM:
+ m = MODE(type, MEM);
+ break;
+ case OBIO_IOMEM_PCMCIA_ATT:
+ m = MODE(type, ATTR);
+ break;
+ }
+#undef MODE
+
+ for (; pa < endpa; pa += PAGE_SIZE, va += PAGE_SIZE) {
+ pmap_kenter_pa(va, pa, VM_PROT_READ | VM_PROT_WRITE);
+ pte = __pmap_kpte_lookup(va);
+ KDASSERT(pte);
+ *pte |= m; /* PTEA PCMCIA assistant bit */
+ sh_tlb_update(0, va, *pte);
+ }
+
+ return (0);
+}
+
+int
+obio_iomem_map(void *v, bus_addr_t bpa, bus_size_t size,
+ int flags, bus_space_handle_t *bshp)
+{
+ bus_addr_t addr = SH3_PHYS_TO_P2SEG(bpa);
+ int error;
+
+ KASSERT((bpa & SH3_PHYS_MASK) == bpa);
+
+ if (bpa < 0x14000000 || bpa >= 0x1c000000) {
+ /* CS0,1,2,3,4,7 */
+ *bshp = (bus_space_handle_t)addr;
+ return (0);
+ }
+
+ /* CS5,6 */
+ error = obio_iomem_add_mapping(addr, size, (int)(u_long)v, bshp);
+
+ return (error);
+}
+
+void
+obio_iomem_unmap(void *v, bus_space_handle_t bsh, bus_size_t size)
+{
+ u_long va, endva;
+ bus_addr_t bpa;
+
+ if (bsh >= SH3_P2SEG_BASE && bsh <= SH3_P2SEG_END) {
+ /* maybe CS0,1,2,3,4,7 */
+ return;
+ }
+
+ /* CS5,6 */
+ va = trunc_page(bsh);
+ endva = round_page(bsh + size);
+
+#ifdef DIAGNOSTIC
+ if (endva <= va)
+ panic("obio_io_unmap: overflow");
+#endif
+
+ pmap_extract(pmap_kernel(), va, &bpa);
+ bpa += bsh & PGOFSET;
+
+ pmap_kremove(va, endva - va);
+
+ /*
+ * Free the kernel virtual mapping.
+ */
+ uvm_km_free(kernel_map, va, endva - va);
+}
+
+int
+obio_iomem_subregion(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, bus_size_t size, bus_space_handle_t *nbshp)
+{
+ *nbshp = bsh + offset;
+
+ return (0);
+}
+
+int
+obio_iomem_alloc(void *v, bus_addr_t rstart, bus_addr_t rend,
+ bus_size_t size, bus_size_t alignment, bus_size_t boundary, int flags,
+ bus_addr_t *bpap, bus_space_handle_t *bshp)
+{
+ *bshp = *bpap = rstart;
+
+ return (0);
+}
+
+void
+obio_iomem_free(void *v, bus_space_handle_t bsh, bus_size_t size)
+{
+ obio_iomem_unmap(v, bsh, size);
+}
+
+/*
+ * on-board I/O bus space read/write
+ */
+uint8_t obio_iomem_read_1(void *v, bus_space_handle_t bsh, bus_size_t offset);
+uint16_t obio_iomem_read_2(void *v, bus_space_handle_t bsh, bus_size_t offset);
+uint32_t obio_iomem_read_4(void *v, bus_space_handle_t bsh, bus_size_t offset);
+void obio_iomem_read_multi_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count);
+void obio_iomem_read_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t *addr, bus_size_t count);
+void obio_iomem_read_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t *addr, bus_size_t count);
+void obio_iomem_read_raw_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count);
+void obio_iomem_read_raw_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count);
+void obio_iomem_read_region_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count);
+void obio_iomem_read_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t *addr, bus_size_t count);
+void obio_iomem_read_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t *addr, bus_size_t count);
+void obio_iomem_read_raw_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count);
+void obio_iomem_read_raw_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count);
+void obio_iomem_write_1(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint8_t value);
+void obio_iomem_write_2(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint16_t value);
+void obio_iomem_write_4(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint32_t value);
+void obio_iomem_write_multi_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count);
+void obio_iomem_write_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint16_t *addr, bus_size_t count);
+void obio_iomem_write_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint32_t *addr, bus_size_t count);
+void obio_iomem_write_raw_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count);
+void obio_iomem_write_raw_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count);
+void obio_iomem_write_region_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count);
+void obio_iomem_write_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint16_t *addr, bus_size_t count);
+void obio_iomem_write_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint32_t *addr, bus_size_t count);
+void obio_iomem_write_raw_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count);
+void obio_iomem_write_raw_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count);
+void obio_iomem_set_multi_1(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint8_t val, bus_size_t count);
+void obio_iomem_set_multi_2(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint16_t val, bus_size_t count);
+void obio_iomem_set_multi_4(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint32_t val, bus_size_t count);
+void obio_iomem_set_region_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t val, bus_size_t count);
+void obio_iomem_set_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t val, bus_size_t count);
+void obio_iomem_set_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t val, bus_size_t count);
+void obio_iomem_copy_region_1(void *v, bus_space_handle_t h1, bus_size_t o1,
+ bus_space_handle_t h2, bus_size_t o2, bus_size_t count);
+void obio_iomem_copy_region_2(void *v, bus_space_handle_t h1, bus_size_t o1,
+ bus_space_handle_t h2, bus_size_t o2, bus_size_t count);
+void obio_iomem_copy_region_4(void *v, bus_space_handle_t h1, bus_size_t o1,
+ bus_space_handle_t h2, bus_size_t o2, bus_size_t count);
+
+struct _bus_space obio_bus_io =
+{
+ .bs_cookie = (void *)OBIO_IOMEM_PCMCIA_IO,
+
+ .bs_map = obio_iomem_map,
+ .bs_unmap = obio_iomem_unmap,
+ .bs_subregion = obio_iomem_subregion,
+
+ .bs_alloc = obio_iomem_alloc,
+ .bs_free = obio_iomem_free,
+
+ .bs_r_1 = obio_iomem_read_1,
+ .bs_r_2 = obio_iomem_read_2,
+ .bs_r_4 = obio_iomem_read_4,
+
+ .bs_rm_1 = obio_iomem_read_multi_1,
+ .bs_rm_2 = obio_iomem_read_multi_2,
+ .bs_rm_4 = obio_iomem_read_multi_4,
+
+ .bs_rrm_2 = obio_iomem_read_raw_multi_2,
+ .bs_rrm_4 = obio_iomem_read_raw_multi_4,
+
+ .bs_rr_1 = obio_iomem_read_region_1,
+ .bs_rr_2 = obio_iomem_read_region_2,
+ .bs_rr_4 = obio_iomem_read_region_4,
+
+ .bs_rrr_2 = obio_iomem_read_raw_region_2,
+ .bs_rrr_4 = obio_iomem_read_raw_region_4,
+
+ .bs_w_1 = obio_iomem_write_1,
+ .bs_w_2 = obio_iomem_write_2,
+ .bs_w_4 = obio_iomem_write_4,
+
+ .bs_wm_1 = obio_iomem_write_multi_1,
+ .bs_wm_2 = obio_iomem_write_multi_2,
+ .bs_wm_4 = obio_iomem_write_multi_4,
+
+ .bs_wrm_2 = obio_iomem_write_raw_multi_2,
+ .bs_wrm_4 = obio_iomem_write_raw_multi_4,
+
+ .bs_wr_1 = obio_iomem_write_region_1,
+ .bs_wr_2 = obio_iomem_write_region_2,
+ .bs_wr_4 = obio_iomem_write_region_4,
+
+ .bs_wrr_2 = obio_iomem_write_raw_region_2,
+ .bs_wrr_4 = obio_iomem_write_raw_region_4,
+
+ .bs_sm_1 = obio_iomem_set_multi_1,
+ .bs_sm_2 = obio_iomem_set_multi_2,
+ .bs_sm_4 = obio_iomem_set_multi_4,
+
+ .bs_sr_1 = obio_iomem_set_region_1,
+ .bs_sr_2 = obio_iomem_set_region_2,
+ .bs_sr_4 = obio_iomem_set_region_4,
+
+ .bs_c_1 = obio_iomem_copy_region_1,
+ .bs_c_2 = obio_iomem_copy_region_2,
+ .bs_c_4 = obio_iomem_copy_region_4,
+};
+
+struct _bus_space obio_bus_mem =
+{
+ .bs_cookie = (void *)OBIO_IOMEM_PCMCIA_MEM,
+
+ .bs_map = obio_iomem_map,
+ .bs_unmap = obio_iomem_unmap,
+ .bs_subregion = obio_iomem_subregion,
+
+ .bs_alloc = obio_iomem_alloc,
+ .bs_free = obio_iomem_free,
+
+ .bs_r_1 = obio_iomem_read_1,
+ .bs_r_2 = obio_iomem_read_2,
+ .bs_r_4 = obio_iomem_read_4,
+
+ .bs_rm_1 = obio_iomem_read_multi_1,
+ .bs_rm_2 = obio_iomem_read_multi_2,
+ .bs_rm_4 = obio_iomem_read_multi_4,
+
+ .bs_rrm_2 = obio_iomem_read_raw_multi_2,
+ .bs_rrm_4 = obio_iomem_read_raw_multi_4,
+
+ .bs_rr_1 = obio_iomem_read_region_1,
+ .bs_rr_2 = obio_iomem_read_region_2,
+ .bs_rr_4 = obio_iomem_read_region_4,
+
+ .bs_rrr_2 = obio_iomem_read_raw_region_2,
+ .bs_rrr_4 = obio_iomem_read_raw_region_4,
+
+ .bs_w_1 = obio_iomem_write_1,
+ .bs_w_2 = obio_iomem_write_2,
+ .bs_w_4 = obio_iomem_write_4,
+
+ .bs_wm_1 = obio_iomem_write_multi_1,
+ .bs_wm_2 = obio_iomem_write_multi_2,
+ .bs_wm_4 = obio_iomem_write_multi_4,
+
+ .bs_wrm_2 = obio_iomem_write_raw_multi_2,
+ .bs_wrm_4 = obio_iomem_write_raw_multi_4,
+
+ .bs_wr_1 = obio_iomem_write_region_1,
+ .bs_wr_2 = obio_iomem_write_region_2,
+ .bs_wr_4 = obio_iomem_write_region_4,
+
+ .bs_wrr_2 = obio_iomem_write_raw_region_2,
+ .bs_wrr_4 = obio_iomem_write_raw_region_4,
+
+ .bs_sm_1 = obio_iomem_set_multi_1,
+ .bs_sm_2 = obio_iomem_set_multi_2,
+ .bs_sm_4 = obio_iomem_set_multi_4,
+
+ .bs_sr_1 = obio_iomem_set_region_1,
+ .bs_sr_2 = obio_iomem_set_region_2,
+ .bs_sr_4 = obio_iomem_set_region_4,
+
+ .bs_c_1 = obio_iomem_copy_region_1,
+ .bs_c_2 = obio_iomem_copy_region_2,
+ .bs_c_4 = obio_iomem_copy_region_4,
+};
+
+/* read */
+uint8_t
+obio_iomem_read_1(void *v, bus_space_handle_t bsh, bus_size_t offset)
+{
+ return *(volatile uint8_t *)(bsh + offset);
+}
+
+uint16_t
+obio_iomem_read_2(void *v, bus_space_handle_t bsh, bus_size_t offset)
+{
+ return *(volatile uint16_t *)(bsh + offset);
+}
+
+uint32_t
+obio_iomem_read_4(void *v, bus_space_handle_t bsh, bus_size_t offset)
+{
+ return *(volatile uint32_t *)(bsh + offset);
+}
+
+void
+obio_iomem_read_multi_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count)
+{
+ volatile uint8_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = *p;
+ }
+}
+
+void
+obio_iomem_read_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = *p;
+ }
+}
+
+void
+obio_iomem_read_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = *p;
+ }
+}
+
+void
+obio_iomem_read_raw_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ count >>= 1;
+ while (count--) {
+ *(uint16_t *)addr = *p;
+ addr += 2;
+ }
+}
+
+void
+obio_iomem_read_raw_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ count >>= 2;
+ while (count--) {
+ *(uint32_t *)addr = *p;
+ addr += 4;
+ }
+}
+
+void
+obio_iomem_read_region_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count)
+{
+ volatile uint8_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = *p++;
+ }
+}
+
+void
+obio_iomem_read_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = *p++;
+ }
+}
+
+void
+obio_iomem_read_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = *p++;
+ }
+}
+
+void
+obio_iomem_read_raw_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ count >>= 1;
+ while (count--) {
+ *(uint16_t *)addr = *p++;
+ addr += 2;
+ }
+}
+
+void
+obio_iomem_read_raw_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ count >>= 2;
+ while (count--) {
+ *(uint32_t *)addr = *p++;
+ addr += 4;
+ }
+}
+
+/* write */
+void
+obio_iomem_write_1(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint8_t value)
+{
+ *(volatile uint8_t *)(bsh + offset) = value;
+}
+
+void
+obio_iomem_write_2(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint16_t value)
+{
+ *(volatile uint16_t *)(bsh + offset) = value;
+}
+
+void
+obio_iomem_write_4(void *v, bus_space_handle_t bsh, bus_size_t offset,
+ uint32_t value)
+{
+ *(volatile uint32_t *)(bsh + offset) = value;
+}
+
+void
+obio_iomem_write_multi_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count)
+{
+ volatile uint8_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p = *addr++;
+ }
+}
+
+void
+obio_iomem_write_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint16_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p = *addr++;
+ }
+}
+
+void
+obio_iomem_write_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint32_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p = *addr++;
+ }
+}
+
+void
+obio_iomem_write_raw_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ count >>= 1;
+ while (count--) {
+ *p = *(uint16_t *)addr;
+ addr += 2;
+ }
+}
+
+void
+obio_iomem_write_raw_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ count >>= 2;
+ while (count--) {
+ *p = *(uint32_t *)addr;
+ addr += 4;
+ }
+}
+
+void
+obio_iomem_write_region_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count)
+{
+ volatile uint8_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p++ = *addr++;
+ }
+}
+
+void
+obio_iomem_write_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint16_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p++ = *addr++;
+ }
+}
+
+void
+obio_iomem_write_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint32_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p++ = *addr++;
+ }
+}
+
+void
+obio_iomem_write_raw_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ count >>= 1;
+ while (count--) {
+ *p++ = *(uint16_t *)addr;
+ addr += 2;
+ }
+}
+
+void
+obio_iomem_write_raw_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, const uint8_t *addr, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ count >>= 2;
+ while (count--) {
+ *p++ = *(uint32_t *)addr;
+ addr += 4;
+ }
+}
+
+void
+obio_iomem_set_multi_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t val, bus_size_t count)
+{
+ volatile uint8_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p = val;
+ }
+}
+
+void
+obio_iomem_set_multi_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t val, bus_size_t count)
+{
+ volatile uint16_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p = val;
+ }
+}
+
+void
+obio_iomem_set_multi_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t val, bus_size_t count)
+{
+ volatile uint32_t *p = (void *)(bsh + offset);
+
+ while (count--) {
+ *p = val;
+ }
+}
+
+void
+obio_iomem_set_region_1(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint8_t val, bus_size_t count)
+{
+ volatile uint8_t *addr = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = val;
+ }
+}
+
+void
+obio_iomem_set_region_2(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint16_t val, bus_size_t count)
+{
+ volatile uint16_t *addr = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = val;
+ }
+}
+
+void
+obio_iomem_set_region_4(void *v, bus_space_handle_t bsh,
+ bus_size_t offset, uint32_t val, bus_size_t count)
+{
+ volatile uint32_t *addr = (void *)(bsh + offset);
+
+ while (count--) {
+ *addr++ = val;
+ }
+}
+
+void
+obio_iomem_copy_region_1(void *v, bus_space_handle_t h1, bus_size_t o1,
+ bus_space_handle_t h2, bus_size_t o2, bus_size_t count)
+{
+ volatile uint8_t *addr1 = (void *)(h1 + o1);
+ volatile uint8_t *addr2 = (void *)(h2 + o2);
+
+ if (addr1 >= addr2) { /* src after dest: copy forward */
+ while (count--) {
+ *addr2++ = *addr1++;
+ }
+ } else { /* dest after src: copy backwards */
+ addr1 += count - 1;
+ addr2 += count - 1;
+ while (count--) {
+ *addr2-- = *addr1--;
+ }
+ }
+}
+
+void
+obio_iomem_copy_region_2(void *v, bus_space_handle_t h1, bus_size_t o1,
+ bus_space_handle_t h2, bus_size_t o2, bus_size_t count)
+{
+ volatile uint16_t *addr1 = (void *)(h1 + o1);
+ volatile uint16_t *addr2 = (void *)(h2 + o2);
+
+ if (addr1 >= addr2) { /* src after dest: copy forward */
+ while (count--) {
+ *addr2++ = *addr1++;
+ }
+ } else { /* dest after src: copy backwards */
+ addr1 += count - 1;
+ addr2 += count - 1;
+ while (count--) {
+ *addr2-- = *addr1--;
+ }
+ }
+}
+
+void
+obio_iomem_copy_region_4(void *v, bus_space_handle_t h1, bus_size_t o1,
+ bus_space_handle_t h2, bus_size_t o2, bus_size_t count)
+{
+ volatile uint32_t *addr1 = (void *)(h1 + o1);
+ volatile uint32_t *addr2 = (void *)(h2 + o2);
+
+ if (addr1 >= addr2) { /* src after dest: copy forward */
+ while (count--) {
+ *addr2++ = *addr1++;
+ }
+ } else { /* dest after src: copy backwards */
+ addr1 += count - 1;
+ addr2 += count - 1;
+ while (count--) {
+ *addr2-- = *addr1--;
+ }
+ }
+}
diff --git a/sys/arch/landisk/dev/obiovar.h b/sys/arch/landisk/dev/obiovar.h
index 8b263f49a8e..72c44a01865 100644
--- a/sys/arch/landisk/dev/obiovar.h
+++ b/sys/arch/landisk/dev/obiovar.h
@@ -1,3 +1,4 @@
+/* $OpenBSD: obiovar.h,v 1.2 2006/10/07 20:52:40 miod Exp $ */
/* $NetBSD: obiovar.h,v 1.1 2006/09/01 21:26:18 uwe Exp $ */
/*-
@@ -150,7 +151,7 @@ struct obio_softc {
#define cf_msize cf_loc[3]
#define cf_irq cf_loc[4]
-void *obio_intr_establish(int irq, int level, int (*func)(void *), void *arg);
-void obio_intr_disestablish(void *ih);
+void *obio_intr_establish(int, int, int (*)(void *), void *, const char *);
+void obio_intr_disestablish(void *);
#endif /* _LANDISK_OBIOVAR_H_ */
diff --git a/sys/arch/landisk/dev/rs5c313.c b/sys/arch/landisk/dev/rs5c313.c
new file mode 100644
index 00000000000..e606c21872c
--- /dev/null
+++ b/sys/arch/landisk/dev/rs5c313.c
@@ -0,0 +1,437 @@
+/* $OpenBSD: rs5c313.c,v 1.1 2006/10/07 20:52:40 miod Exp $ */
+/* $NetBSD: rs5c313.c,v 1.1 2006/09/07 01:12:00 uwe Exp $ */
+/* $NetBSD: rs5c313_landisk.c,v 1.1 2006/09/07 01:55:03 uwe Exp $ */
+
+/*-
+ * Copyright (c) 2002 The NetBSD Foundation, Inc.
+ * 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 the NetBSD
+ * Foundation, Inc. and its contributors.
+ *
+ * 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.
+ */
+
+/*
+ * RICOH RS5C313 Real Time Clock
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <sys/kernel.h>
+
+#include <dev/clock_subr.h>
+#include <sh/clock.h>
+
+#include <sh/devreg.h>
+#include <sh/dev/scireg.h>
+
+#include <landisk/dev/rs5c313reg.h>
+#include <landisk/landisk/landiskreg.h>
+
+struct rs5c313_softc {
+ struct device sc_dev;
+
+ int sc_valid; /* oscillation halt sensing on init */
+};
+
+/* chip access methods */
+void rtc_begin(struct rs5c313_softc *);
+void rtc_ce(struct rs5c313_softc *, int);
+void rtc_dir(struct rs5c313_softc *, int);
+void rtc_clk(struct rs5c313_softc *, int);
+int rtc_read(struct rs5c313_softc *);
+void rtc_write(struct rs5c313_softc *, int);
+
+int rs5c313_init(struct rs5c313_softc *);
+int rs5c313_read_reg(struct rs5c313_softc *, int);
+void rs5c313_write_reg(struct rs5c313_softc *, int, int);
+void rs5c313_gettime(void *, time_t, struct clock_ymdhms *);
+void rs5c313_settime(void *, struct clock_ymdhms *);
+
+int
+rs5c313_init(struct rs5c313_softc *sc)
+{
+ int status = 0;
+ int retry;
+
+ rtc_ce(sc, 0);
+
+ rtc_begin(sc);
+ rtc_ce(sc, 1);
+
+ if ((rs5c313_read_reg(sc, RS5C313_CTRL) & CTRL_XSTP) == 0) {
+ sc->sc_valid = 1;
+ goto done;
+ }
+
+ sc->sc_valid = 0;
+ printf("%s: time not valid\n", sc->sc_dev.dv_xname);
+
+ rs5c313_write_reg(sc, RS5C313_TINT, 0);
+ rs5c313_write_reg(sc, RS5C313_CTRL, (CTRL_BASE | CTRL_ADJ));
+
+ for (retry = 1000; retry > 0; --retry) {
+ if (rs5c313_read_reg(sc, RS5C313_CTRL) & CTRL_BSY)
+ delay(1);
+ else
+ break;
+ }
+
+ if (retry == 0) {
+ status = EIO;
+ goto done;
+ }
+
+ rs5c313_write_reg(sc, RS5C313_CTRL, CTRL_BASE);
+
+done:
+ rtc_ce(sc, 0);
+ return status;
+}
+
+int
+rs5c313_read_reg(struct rs5c313_softc *sc, int addr)
+{
+ int data;
+
+ /* output */
+ rtc_dir(sc, 1);
+
+ /* control */
+ rtc_write(sc, 1); /* ignored */
+ rtc_write(sc, 1); /* R/#W = 1(READ) */
+ rtc_write(sc, 1); /* AD = 1 */
+ rtc_write(sc, 0); /* DT = 0 */
+
+ /* address */
+ rtc_write(sc, addr & 0x8); /* A3 */
+ rtc_write(sc, addr & 0x4); /* A2 */
+ rtc_write(sc, addr & 0x2); /* A1 */
+ rtc_write(sc, addr & 0x1); /* A0 */
+
+ /* input */
+ rtc_dir(sc, 0);
+
+ /* ignore */
+ (void)rtc_read(sc);
+ (void)rtc_read(sc);
+ (void)rtc_read(sc);
+ (void)rtc_read(sc);
+
+ /* data */
+ data = rtc_read(sc); /* D3 */
+ data <<= 1;
+ data |= rtc_read(sc); /* D2 */
+ data <<= 1;
+ data |= rtc_read(sc); /* D1 */
+ data <<= 1;
+ data |= rtc_read(sc); /* D0 */
+
+ return data;
+}
+
+void
+rs5c313_write_reg(struct rs5c313_softc *sc, int addr, int data)
+{
+ /* output */
+ rtc_dir(sc, 1);
+
+ /* control */
+ rtc_write(sc, 1); /* ignored */
+ rtc_write(sc, 0); /* R/#W = 0 (WRITE) */
+ rtc_write(sc, 1); /* AD = 1 */
+ rtc_write(sc, 0); /* DT = 0 */
+
+ /* address */
+ rtc_write(sc, addr & 0x8); /* A3 */
+ rtc_write(sc, addr & 0x4); /* A2 */
+ rtc_write(sc, addr & 0x2); /* A1 */
+ rtc_write(sc, addr & 0x1); /* A0 */
+
+ /* control */
+ rtc_write(sc, 1); /* ignored */
+ rtc_write(sc, 0); /* R/#W = 0(WRITE) */
+ rtc_write(sc, 0); /* AD = 0 */
+ rtc_write(sc, 1); /* DT = 1 */
+
+ /* data */
+ rtc_write(sc, data & 0x8); /* D3 */
+ rtc_write(sc, data & 0x4); /* D2 */
+ rtc_write(sc, data & 0x2); /* D1 */
+ rtc_write(sc, data & 0x1); /* D0 */
+}
+
+void
+rs5c313_gettime(void *cookie, time_t base, struct clock_ymdhms *dt)
+{
+ struct rs5c313_softc *sc = cookie;
+ int retry;
+ int s;
+
+ /*
+ * If chip had invalid data on init, don't bother reading
+ * bogus values.
+ */
+ if (sc->sc_valid == 0)
+ return;
+
+ s = splhigh();
+
+ rtc_begin(sc);
+ for (retry = 10; retry > 0; --retry) {
+ rtc_ce(sc, 1);
+
+ rs5c313_write_reg(sc, RS5C313_CTRL, CTRL_BASE);
+ if ((rs5c313_read_reg(sc, RS5C313_CTRL) & CTRL_BSY) == 0)
+ break;
+
+ rtc_ce(sc, 0);
+ delay(1);
+ }
+
+ if (retry == 0) {
+ splx(s);
+ return;
+ }
+
+#define RTCGET(x, y) \
+ do { \
+ int ones = rs5c313_read_reg(sc, RS5C313_ ## y ## 1); \
+ int tens = rs5c313_read_reg(sc, RS5C313_ ## y ## 10); \
+ dt->dt_ ## x = tens * 10 + ones; \
+ } while (/* CONSTCOND */0)
+
+ RTCGET(sec, SEC);
+ RTCGET(min, MIN);
+ RTCGET(hour, HOUR);
+ RTCGET(day, DAY);
+ RTCGET(mon, MON);
+ RTCGET(year, YEAR);
+#undef RTCGET
+ dt->dt_wday = rs5c313_read_reg(sc, RS5C313_WDAY);
+
+ rtc_ce(sc, 0);
+ splx(s);
+
+ dt->dt_year = (dt->dt_year % 100) + 1900;
+ if (dt->dt_year < 1970) {
+ dt->dt_year += 100;
+ }
+}
+
+void
+rs5c313_settime(void *cookie, struct clock_ymdhms *dt)
+{
+ struct rs5c313_softc *sc = cookie;
+ int retry;
+ int t;
+ int s;
+
+ s = splhigh();
+
+ rtc_begin(sc);
+ for (retry = 10; retry > 0; --retry) {
+ rtc_ce(sc, 1);
+
+ rs5c313_write_reg(sc, RS5C313_CTRL, CTRL_BASE);
+ if ((rs5c313_read_reg(sc, RS5C313_CTRL) & CTRL_BSY) == 0)
+ break;
+
+ rtc_ce(sc, 0);
+ delay(1);
+ }
+
+ if (retry == 0) {
+ splx(s);
+ return;
+ }
+
+#define RTCSET(x, y) \
+ do { \
+ t = TOBCD(dt->dt_ ## y) & 0xff; \
+ rs5c313_write_reg(sc, RS5C313_ ## x ## 1, t & 0x0f); \
+ rs5c313_write_reg(sc, RS5C313_ ## x ## 10, (t >> 4) & 0x0f); \
+ } while (/* CONSTCOND */0)
+
+ RTCSET(SEC, sec);
+ RTCSET(MIN, min);
+ RTCSET(HOUR, hour);
+ RTCSET(DAY, day);
+ RTCSET(MON, mon);
+
+#undef RTCSET
+
+ t = dt->dt_year % 100;
+ t = TOBCD(t);
+ rs5c313_write_reg(sc, RS5C313_YEAR1, t & 0x0f);
+ rs5c313_write_reg(sc, RS5C313_YEAR10, (t >> 4) & 0x0f);
+
+ rs5c313_write_reg(sc, RS5C313_WDAY, dt->dt_wday);
+
+ rtc_ce(sc, 0);
+ splx(s);
+
+ sc->sc_valid = 1;
+}
+
+struct rtc_ops rs5c313_ops = {
+ NULL,
+ NULL, /* not used */
+ rs5c313_gettime,
+ rs5c313_settime
+};
+
+void
+rtc_begin(struct rs5c313_softc *sc)
+{
+ SHREG_SCSPTR = SCSPTR_SPB1IO | SCSPTR_SPB1DT
+ | SCSPTR_SPB0IO | SCSPTR_SPB0DT;
+ delay(100);
+}
+
+/*
+ * CE pin
+ */
+void
+rtc_ce(struct rs5c313_softc *sc, int onoff)
+{
+ if (onoff)
+ _reg_write_1(LANDISK_PWRMNG, PWRMNG_RTC_CE);
+ else
+ _reg_write_1(LANDISK_PWRMNG, 0);
+ delay(600);
+}
+
+/*
+ * SCLK pin is connnected to SPB0DT.
+ * SPB0DT is always in output mode, we set SPB0IO in rtc_begin.
+ */
+void
+rtc_clk(struct rs5c313_softc *sc, int onoff)
+{
+ uint8_t r = SHREG_SCSPTR;
+
+ if (onoff)
+ r |= SCSPTR_SPB0DT;
+ else
+ r &= ~SCSPTR_SPB0DT;
+ SHREG_SCSPTR = r;
+}
+
+/*
+ * SIO pin is connected to SPB1DT.
+ * SPB1DT is output when SPB1IO is set.
+ */
+void
+rtc_dir(struct rs5c313_softc *sc, int output)
+{
+ uint8_t r = SHREG_SCSPTR;
+
+ if (output)
+ r |= SCSPTR_SPB1IO;
+ else
+ r &= ~SCSPTR_SPB1IO;
+ SHREG_SCSPTR = r;
+}
+
+/*
+ * Read bit from SPB1DT pin.
+ */
+int
+rtc_read(struct rs5c313_softc *sc)
+{
+ int bit;
+
+ delay(300);
+
+ bit = (SHREG_SCSPTR & SCSPTR_SPB1DT) ? 1 : 0;
+
+ rtc_clk(sc, 0);
+ delay(300);
+ rtc_clk(sc, 1);
+
+ return bit;
+}
+
+/*
+ * Write bit via SPB1DT pin.
+ */
+void
+rtc_write(struct rs5c313_softc *sc, int bit)
+{
+ uint8_t r = SHREG_SCSPTR;
+
+ if (bit)
+ r |= SCSPTR_SPB1DT;
+ else
+ r &= ~SCSPTR_SPB1DT;
+ SHREG_SCSPTR = r;
+
+ delay(300);
+
+ rtc_clk(sc, 0);
+ delay(300);
+ rtc_clk(sc, 1);
+}
+
+/* autoconf glue */
+int rs5c313_landisk_match(struct device *, void *, void *);
+void rs5c313_landisk_attach(struct device *, struct device *, void *);
+
+const struct cfattach rsclock_ca = {
+ sizeof (struct rs5c313_softc),
+ rs5c313_landisk_match, rs5c313_landisk_attach
+};
+
+struct cfdriver rsclock_cd = {
+ 0, "rsclock", DV_DULL
+};
+
+int
+rs5c313_landisk_match(struct device *parent, void *vcf, void *aux)
+{
+ static int matched = 0;
+
+ if (matched)
+ return (0);
+
+ return (matched = 1);
+}
+
+void
+rs5c313_landisk_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct rs5c313_softc *sc = (void *)self;
+
+ printf(": RS5C313 real time clock\n");
+
+ if (rs5c313_init(sc) != 0) {
+ printf("%s: init failed\n", self->dv_xname);
+ return;
+ }
+
+ rs5c313_ops._cookie = sc;
+ sh_clock_init(0, &rs5c313_ops);
+}
diff --git a/sys/arch/landisk/dev/wdc_obio.c b/sys/arch/landisk/dev/wdc_obio.c
new file mode 100644
index 00000000000..b2d7f40d8ce
--- /dev/null
+++ b/sys/arch/landisk/dev/wdc_obio.c
@@ -0,0 +1,142 @@
+/* $OpenBSD: wdc_obio.c,v 1.1 2006/10/07 20:52:40 miod Exp $ */
+/* $NetBSD: wdc_obio.c,v 1.1 2006/09/01 21:26:18 uwe Exp $ */
+
+/*-
+ * Copyright (c) 1998, 2003 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Charles M. Hannum and by Onno van der Linden.
+ *
+ * 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.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <sys/malloc.h>
+
+#include <machine/bus.h>
+#include <machine/intr.h>
+
+#include <dev/ata/atavar.h>
+#include <dev/ic/wdcvar.h>
+
+#include <landisk/dev/obiovar.h>
+
+struct wdc_obio_softc {
+ struct wdc_softc sc_wdcdev;
+ struct channel_softc *sc_chanptr;
+ struct channel_softc sc_channel;
+
+ void *sc_ih;
+};
+
+int wdc_obio_match(struct device *, void *, void *);
+void wdc_obio_attach(struct device *, struct device *, void *);
+
+struct cfattach wdc_obio_ca = {
+ sizeof(struct wdc_obio_softc), wdc_obio_match, wdc_obio_attach
+};
+
+#define WDC_OBIO_REG_NPORTS WDC_NREG
+#define WDC_OBIO_REG_SIZE (WDC_OBIO_REG_NPORTS * 2)
+#define WDC_OBIO_AUXREG_NPORTS 1
+#define WDC_OBIO_AUXREG_SIZE (WDC_OBIO_AUXREG_NPORTS * 2)
+#define WDC_OBIO_AUXREG_OFFSET 0x2c
+
+int
+wdc_obio_match(struct device *parent, void *vcf, void *aux)
+{
+ struct obio_attach_args *oa = aux;
+
+ if (oa->oa_nio != 1)
+ return (0);
+ if (oa->oa_nirq != 1)
+ return (0);
+ if (oa->oa_niomem != 0)
+ return (0);
+
+ if (oa->oa_io[0].or_addr == IOBASEUNK)
+ return (0);
+ if (oa->oa_irq[0].or_irq == IRQUNK)
+ return (0);
+
+ /* XXX should probe for hardware */
+
+ oa->oa_io[0].or_size = WDC_OBIO_REG_SIZE;
+
+ return (1);
+}
+
+void
+wdc_obio_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct wdc_obio_softc *sc = (void *)self;
+ struct obio_attach_args *oa = aux;
+ struct channel_softc *chp = &sc->sc_channel;
+
+ printf("\n");
+
+ chp->cmd_iot = chp->ctl_iot = oa->oa_iot;
+
+ if (bus_space_map(chp->cmd_iot, oa->oa_io[0].or_addr,
+ WDC_OBIO_REG_SIZE, 0, &chp->cmd_ioh)
+ || bus_space_map(chp->ctl_iot,
+ oa->oa_io[0].or_addr + WDC_OBIO_AUXREG_OFFSET,
+ WDC_OBIO_AUXREG_SIZE, 0, &chp->ctl_ioh)) {
+ printf(": couldn't map registers\n");
+ return;
+ }
+ chp->data32iot = chp->cmd_iot;
+ chp->data32ioh = chp->cmd_ioh;
+
+ sc->sc_ih = obio_intr_establish(oa->oa_irq[0].or_irq, IPL_BIO, wdcintr,
+ chp, self->dv_xname);
+
+ sc->sc_wdcdev.cap |= WDC_CAPABILITY_DATA16 | WDC_CAPABILITY_PREATA;
+ sc->sc_wdcdev.PIO_cap = 0;
+ sc->sc_chanptr = chp;
+ sc->sc_wdcdev.channels = &sc->sc_chanptr;
+ sc->sc_wdcdev.nchannels = 1;
+ chp->channel = 0;
+ chp->wdc = &sc->sc_wdcdev;
+
+ chp->ch_queue = malloc(sizeof(struct channel_queue), M_DEVBUF,
+ M_NOWAIT);
+ if (chp->ch_queue == NULL) {
+ printf("%s: can't allocate memory for command queue\n",
+ self->dv_xname);
+ obio_intr_disestablish(sc->sc_ih);
+ return;
+ }
+
+ wdcattach(chp);
+ wdc_print_current_modes(chp);
+}