132d1354aSWojciech Macek /*- 232d1354aSWojciech Macek * Copyright (c) 2015 Nathan Whitehorn 332d1354aSWojciech Macek * All rights reserved. 432d1354aSWojciech Macek * 532d1354aSWojciech Macek * Redistribution and use in source and binary forms, with or without 632d1354aSWojciech Macek * modification, are permitted provided that the following conditions 732d1354aSWojciech Macek * are met: 832d1354aSWojciech Macek * 1. Redistributions of source code must retain the above copyright 932d1354aSWojciech Macek * notice, this list of conditions and the following disclaimer. 1032d1354aSWojciech Macek * 2. Redistributions in binary form must reproduce the above copyright 1132d1354aSWojciech Macek * notice, this list of conditions and the following disclaimer in the 1232d1354aSWojciech Macek * documentation and/or other materials provided with the distribution. 1332d1354aSWojciech Macek * 1432d1354aSWojciech Macek * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 1532d1354aSWojciech Macek * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 1632d1354aSWojciech Macek * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 1732d1354aSWojciech Macek * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 1832d1354aSWojciech Macek * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 1932d1354aSWojciech Macek * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 2032d1354aSWojciech Macek * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 2132d1354aSWojciech Macek * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2232d1354aSWojciech Macek * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2332d1354aSWojciech Macek * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 2432d1354aSWojciech Macek * SUCH DAMAGE. 2532d1354aSWojciech Macek */ 2632d1354aSWojciech Macek 2732d1354aSWojciech Macek #include <sys/cdefs.h> 2832d1354aSWojciech Macek __FBSDID("$FreeBSD$"); 2932d1354aSWojciech Macek 3032d1354aSWojciech Macek #include <sys/param.h> 3132d1354aSWojciech Macek #include <sys/systm.h> 3232d1354aSWojciech Macek #include <sys/module.h> 3332d1354aSWojciech Macek #include <sys/bus.h> 3432d1354aSWojciech Macek #include <sys/conf.h> 3532d1354aSWojciech Macek #include <sys/clock.h> 3632d1354aSWojciech Macek #include <sys/cpu.h> 3732d1354aSWojciech Macek #include <sys/kernel.h> 3832d1354aSWojciech Macek #include <sys/reboot.h> 3932d1354aSWojciech Macek #include <sys/sysctl.h> 40504d9b60SWojciech Macek #include <sys/endian.h> 41504d9b60SWojciech Macek 42504d9b60SWojciech Macek #include <vm/vm.h> 43504d9b60SWojciech Macek #include <vm/pmap.h> 4432d1354aSWojciech Macek 4532d1354aSWojciech Macek #include <dev/ofw/ofw_bus.h> 4632d1354aSWojciech Macek #include <dev/ofw/ofw_bus_subr.h> 4732d1354aSWojciech Macek #include <dev/ofw/openfirm.h> 4832d1354aSWojciech Macek 4932d1354aSWojciech Macek #include "clock_if.h" 5032d1354aSWojciech Macek #include "opal.h" 5132d1354aSWojciech Macek 5232d1354aSWojciech Macek static int opaldev_probe(device_t); 5332d1354aSWojciech Macek static int opaldev_attach(device_t); 5432d1354aSWojciech Macek /* clock interface */ 5532d1354aSWojciech Macek static int opal_gettime(device_t dev, struct timespec *ts); 5632d1354aSWojciech Macek static int opal_settime(device_t dev, struct timespec *ts); 5732d1354aSWojciech Macek /* ofw bus interface */ 5832d1354aSWojciech Macek static const struct ofw_bus_devinfo *opaldev_get_devinfo(device_t dev, 5932d1354aSWojciech Macek device_t child); 6032d1354aSWojciech Macek 6132d1354aSWojciech Macek static void opal_shutdown(void *arg, int howto); 62504d9b60SWojciech Macek static void opal_intr(void *); 6332d1354aSWojciech Macek 6432d1354aSWojciech Macek static device_method_t opaldev_methods[] = { 6532d1354aSWojciech Macek /* Device interface */ 6632d1354aSWojciech Macek DEVMETHOD(device_probe, opaldev_probe), 6732d1354aSWojciech Macek DEVMETHOD(device_attach, opaldev_attach), 6832d1354aSWojciech Macek 6932d1354aSWojciech Macek /* clock interface */ 7032d1354aSWojciech Macek DEVMETHOD(clock_gettime, opal_gettime), 7132d1354aSWojciech Macek DEVMETHOD(clock_settime, opal_settime), 7232d1354aSWojciech Macek 7332d1354aSWojciech Macek /* ofw_bus interface */ 7432d1354aSWojciech Macek DEVMETHOD(ofw_bus_get_devinfo, opaldev_get_devinfo), 7532d1354aSWojciech Macek DEVMETHOD(ofw_bus_get_compat, ofw_bus_gen_get_compat), 7632d1354aSWojciech Macek DEVMETHOD(ofw_bus_get_model, ofw_bus_gen_get_model), 7732d1354aSWojciech Macek DEVMETHOD(ofw_bus_get_name, ofw_bus_gen_get_name), 7832d1354aSWojciech Macek DEVMETHOD(ofw_bus_get_node, ofw_bus_gen_get_node), 7932d1354aSWojciech Macek DEVMETHOD(ofw_bus_get_type, ofw_bus_gen_get_type), 8032d1354aSWojciech Macek 8132d1354aSWojciech Macek DEVMETHOD_END 8232d1354aSWojciech Macek }; 8332d1354aSWojciech Macek 8432d1354aSWojciech Macek static driver_t opaldev_driver = { 8532d1354aSWojciech Macek "opal", 8632d1354aSWojciech Macek opaldev_methods, 8732d1354aSWojciech Macek 0 8832d1354aSWojciech Macek }; 8932d1354aSWojciech Macek 9032d1354aSWojciech Macek static devclass_t opaldev_devclass; 9132d1354aSWojciech Macek 9232d1354aSWojciech Macek DRIVER_MODULE(opaldev, ofwbus, opaldev_driver, opaldev_devclass, 0, 0); 9332d1354aSWojciech Macek 9432d1354aSWojciech Macek static int 9532d1354aSWojciech Macek opaldev_probe(device_t dev) 9632d1354aSWojciech Macek { 97504d9b60SWojciech Macek phandle_t iparent; 98504d9b60SWojciech Macek pcell_t *irqs; 99504d9b60SWojciech Macek int i, n_irqs; 10032d1354aSWojciech Macek 10132d1354aSWojciech Macek if (!ofw_bus_is_compatible(dev, "ibm,opal-v3")) 10232d1354aSWojciech Macek return (ENXIO); 10332d1354aSWojciech Macek if (opal_check() != 0) 10432d1354aSWojciech Macek return (ENXIO); 10532d1354aSWojciech Macek 10632d1354aSWojciech Macek device_set_desc(dev, "OPAL Abstraction Firmware"); 107504d9b60SWojciech Macek 108504d9b60SWojciech Macek /* Manually add IRQs before attaching */ 109504d9b60SWojciech Macek if (OF_hasprop(ofw_bus_get_node(dev), "opal-interrupts")) { 110504d9b60SWojciech Macek iparent = OF_finddevice("/interrupt-controller@0"); 111504d9b60SWojciech Macek iparent = OF_xref_from_node(iparent); 112504d9b60SWojciech Macek 113504d9b60SWojciech Macek n_irqs = OF_getproplen(ofw_bus_get_node(dev), 114504d9b60SWojciech Macek "opal-interrupts") / sizeof(*irqs); 115504d9b60SWojciech Macek irqs = malloc(n_irqs * sizeof(*irqs), M_DEVBUF, M_WAITOK); 116504d9b60SWojciech Macek OF_getencprop(ofw_bus_get_node(dev), "opal-interrupts", irqs, 117504d9b60SWojciech Macek n_irqs * sizeof(*irqs)); 118504d9b60SWojciech Macek for (i = 0; i < n_irqs; i++) 119504d9b60SWojciech Macek bus_set_resource(dev, SYS_RES_IRQ, i, 120504d9b60SWojciech Macek ofw_bus_map_intr(dev, iparent, 1, &irqs[i]), 1); 121504d9b60SWojciech Macek free(irqs, M_DEVBUF); 122504d9b60SWojciech Macek } 123504d9b60SWojciech Macek 124504d9b60SWojciech Macek 12532d1354aSWojciech Macek return (BUS_PROBE_SPECIFIC); 12632d1354aSWojciech Macek } 12732d1354aSWojciech Macek 12832d1354aSWojciech Macek static int 12932d1354aSWojciech Macek opaldev_attach(device_t dev) 13032d1354aSWojciech Macek { 13132d1354aSWojciech Macek phandle_t child; 13232d1354aSWojciech Macek device_t cdev; 133504d9b60SWojciech Macek uint64_t junk; 134504d9b60SWojciech Macek int i, rv; 13532d1354aSWojciech Macek struct ofw_bus_devinfo *dinfo; 136504d9b60SWojciech Macek struct resource *irq; 13732d1354aSWojciech Macek 138504d9b60SWojciech Macek /* Test for RTC support and register clock if it works */ 139504d9b60SWojciech Macek rv = opal_call(OPAL_RTC_READ, vtophys(&junk), vtophys(&junk)); 140504d9b60SWojciech Macek do { 141504d9b60SWojciech Macek rv = opal_call(OPAL_RTC_READ, vtophys(&junk), vtophys(&junk)); 142504d9b60SWojciech Macek if (rv == OPAL_BUSY_EVENT) 143504d9b60SWojciech Macek rv = opal_call(OPAL_POLL_EVENTS, 0); 144504d9b60SWojciech Macek } while (rv == OPAL_BUSY_EVENT); 145504d9b60SWojciech Macek 146504d9b60SWojciech Macek if (rv == OPAL_SUCCESS) 14732d1354aSWojciech Macek clock_register(dev, 2000); 14832d1354aSWojciech Macek 14932d1354aSWojciech Macek EVENTHANDLER_REGISTER(shutdown_final, opal_shutdown, NULL, 15032d1354aSWojciech Macek SHUTDOWN_PRI_LAST); 15132d1354aSWojciech Macek 152504d9b60SWojciech Macek /* Bind to interrupts */ 153504d9b60SWojciech Macek for (i = 0; (irq = bus_alloc_resource_any(dev, SYS_RES_IRQ, &i, 154504d9b60SWojciech Macek RF_ACTIVE)) != NULL; i++) 155504d9b60SWojciech Macek bus_setup_intr(dev, irq, INTR_TYPE_TTY | INTR_MPSAFE | 156504d9b60SWojciech Macek INTR_ENTROPY, NULL, opal_intr, (void *)rman_get_start(irq), 157504d9b60SWojciech Macek NULL); 158504d9b60SWojciech Macek 15932d1354aSWojciech Macek for (child = OF_child(ofw_bus_get_node(dev)); child != 0; 16032d1354aSWojciech Macek child = OF_peer(child)) { 16132d1354aSWojciech Macek dinfo = malloc(sizeof(*dinfo), M_DEVBUF, M_WAITOK | M_ZERO); 16232d1354aSWojciech Macek if (ofw_bus_gen_setup_devinfo(dinfo, child) != 0) { 16332d1354aSWojciech Macek free(dinfo, M_DEVBUF); 16432d1354aSWojciech Macek continue; 16532d1354aSWojciech Macek } 16632d1354aSWojciech Macek cdev = device_add_child(dev, NULL, -1); 16732d1354aSWojciech Macek if (cdev == NULL) { 16832d1354aSWojciech Macek device_printf(dev, "<%s>: device_add_child failed\n", 16932d1354aSWojciech Macek dinfo->obd_name); 17032d1354aSWojciech Macek ofw_bus_gen_destroy_devinfo(dinfo); 17132d1354aSWojciech Macek free(dinfo, M_DEVBUF); 17232d1354aSWojciech Macek continue; 17332d1354aSWojciech Macek } 17432d1354aSWojciech Macek device_set_ivars(cdev, dinfo); 17532d1354aSWojciech Macek } 17632d1354aSWojciech Macek 17732d1354aSWojciech Macek return (bus_generic_attach(dev)); 17832d1354aSWojciech Macek } 17932d1354aSWojciech Macek 18032d1354aSWojciech Macek static int 181504d9b60SWojciech Macek bcd2bin32(int bcd) 182504d9b60SWojciech Macek { 183504d9b60SWojciech Macek int out = 0; 184504d9b60SWojciech Macek 185504d9b60SWojciech Macek out += bcd2bin(bcd & 0xff); 186504d9b60SWojciech Macek out += 100*bcd2bin((bcd & 0x0000ff00) >> 8); 187504d9b60SWojciech Macek out += 10000*bcd2bin((bcd & 0x00ff0000) >> 16); 188504d9b60SWojciech Macek out += 1000000*bcd2bin((bcd & 0xffff0000) >> 24); 189504d9b60SWojciech Macek 190504d9b60SWojciech Macek return (out); 191504d9b60SWojciech Macek } 192504d9b60SWojciech Macek 193504d9b60SWojciech Macek static int 194*eb96cc13SWojciech Macek bin2bcd32(int bin) 195*eb96cc13SWojciech Macek { 196*eb96cc13SWojciech Macek int out = 0; 197*eb96cc13SWojciech Macek int tmp; 198*eb96cc13SWojciech Macek 199*eb96cc13SWojciech Macek tmp = bin % 100; 200*eb96cc13SWojciech Macek out += bin2bcd(tmp) * 1; 201*eb96cc13SWojciech Macek bin = bin / 100; 202*eb96cc13SWojciech Macek 203*eb96cc13SWojciech Macek tmp = bin % 100; 204*eb96cc13SWojciech Macek out += bin2bcd(tmp) * 100; 205*eb96cc13SWojciech Macek bin = bin / 100; 206*eb96cc13SWojciech Macek 207*eb96cc13SWojciech Macek tmp = bin % 100; 208*eb96cc13SWojciech Macek out += bin2bcd(tmp) * 10000; 209*eb96cc13SWojciech Macek 210*eb96cc13SWojciech Macek return (out); 211*eb96cc13SWojciech Macek } 212*eb96cc13SWojciech Macek 213*eb96cc13SWojciech Macek static int 214504d9b60SWojciech Macek opal_gettime(device_t dev, struct timespec *ts) 215504d9b60SWojciech Macek { 216504d9b60SWojciech Macek int rv; 217504d9b60SWojciech Macek struct clocktime ct; 218504d9b60SWojciech Macek uint32_t ymd; 219504d9b60SWojciech Macek uint64_t hmsm; 220504d9b60SWojciech Macek 221504d9b60SWojciech Macek do { 222504d9b60SWojciech Macek rv = opal_call(OPAL_RTC_READ, vtophys(&ymd), vtophys(&hmsm)); 223504d9b60SWojciech Macek if (rv == OPAL_BUSY_EVENT) { 224504d9b60SWojciech Macek rv = opal_call(OPAL_POLL_EVENTS, 0); 225504d9b60SWojciech Macek pause("opalrtc", 1); 226504d9b60SWojciech Macek } 227504d9b60SWojciech Macek } while (rv == OPAL_BUSY_EVENT); 228504d9b60SWojciech Macek 229504d9b60SWojciech Macek if (rv != OPAL_SUCCESS) 23032d1354aSWojciech Macek return (ENXIO); 231504d9b60SWojciech Macek 232504d9b60SWojciech Macek hmsm = be64toh(hmsm); 233504d9b60SWojciech Macek ymd = be32toh(ymd); 234504d9b60SWojciech Macek 235504d9b60SWojciech Macek ct.nsec = bcd2bin32((hmsm & 0x000000ffffff0000) >> 16) * 1000; 236504d9b60SWojciech Macek ct.sec = bcd2bin((hmsm & 0x0000ff0000000000) >> 40); 237504d9b60SWojciech Macek ct.min = bcd2bin((hmsm & 0x00ff000000000000) >> 48); 238504d9b60SWojciech Macek ct.hour = bcd2bin((hmsm & 0xff00000000000000) >> 56); 239504d9b60SWojciech Macek 240504d9b60SWojciech Macek ct.day = bcd2bin((ymd & 0x000000ff) >> 0); 241504d9b60SWojciech Macek ct.mon = bcd2bin((ymd & 0x0000ff00) >> 8); 242504d9b60SWojciech Macek ct.year = bcd2bin32((ymd & 0xffff0000) >> 16); 243504d9b60SWojciech Macek 244504d9b60SWojciech Macek return (clock_ct_to_ts(&ct, ts)); 24532d1354aSWojciech Macek } 24632d1354aSWojciech Macek 24732d1354aSWojciech Macek static int 24832d1354aSWojciech Macek opal_settime(device_t dev, struct timespec *ts) 24932d1354aSWojciech Macek { 250*eb96cc13SWojciech Macek int rv; 251*eb96cc13SWojciech Macek struct clocktime ct; 252*eb96cc13SWojciech Macek uint32_t ymd = 0; 253*eb96cc13SWojciech Macek uint64_t hmsm = 0; 254*eb96cc13SWojciech Macek 255*eb96cc13SWojciech Macek clock_ts_to_ct(ts, &ct); 256*eb96cc13SWojciech Macek 257*eb96cc13SWojciech Macek ymd |= (uint32_t)bin2bcd(ct.day); 258*eb96cc13SWojciech Macek ymd |= ((uint32_t)bin2bcd(ct.mon) << 8); 259*eb96cc13SWojciech Macek ymd |= ((uint32_t)bin2bcd32(ct.year) << 16); 260*eb96cc13SWojciech Macek 261*eb96cc13SWojciech Macek hmsm |= ((uint64_t)bin2bcd32(ct.nsec/1000) << 16); 262*eb96cc13SWojciech Macek hmsm |= ((uint64_t)bin2bcd(ct.sec) << 40); 263*eb96cc13SWojciech Macek hmsm |= ((uint64_t)bin2bcd(ct.min) << 48); 264*eb96cc13SWojciech Macek hmsm |= ((uint64_t)bin2bcd(ct.hour) << 56); 265*eb96cc13SWojciech Macek 266*eb96cc13SWojciech Macek hmsm = htobe64(hmsm); 267*eb96cc13SWojciech Macek ymd = htobe32(ymd); 268*eb96cc13SWojciech Macek 269*eb96cc13SWojciech Macek do { 270*eb96cc13SWojciech Macek rv = opal_call(OPAL_RTC_WRITE, vtophys(&ymd), vtophys(&hmsm)); 271*eb96cc13SWojciech Macek if (rv == OPAL_BUSY_EVENT) { 272*eb96cc13SWojciech Macek rv = opal_call(OPAL_POLL_EVENTS, 0); 273*eb96cc13SWojciech Macek pause("opalrtc", 1); 274*eb96cc13SWojciech Macek } 275*eb96cc13SWojciech Macek } while (rv == OPAL_BUSY_EVENT); 276*eb96cc13SWojciech Macek 277*eb96cc13SWojciech Macek if (rv != OPAL_SUCCESS) 278*eb96cc13SWojciech Macek return (ENXIO); 279504d9b60SWojciech Macek 28032d1354aSWojciech Macek return (0); 28132d1354aSWojciech Macek } 28232d1354aSWojciech Macek 28332d1354aSWojciech Macek static const struct ofw_bus_devinfo * 28432d1354aSWojciech Macek opaldev_get_devinfo(device_t dev, device_t child) 28532d1354aSWojciech Macek { 28632d1354aSWojciech Macek return (device_get_ivars(child)); 28732d1354aSWojciech Macek } 28832d1354aSWojciech Macek 28932d1354aSWojciech Macek static void 29032d1354aSWojciech Macek opal_shutdown(void *arg, int howto) 29132d1354aSWojciech Macek { 29232d1354aSWojciech Macek 29332d1354aSWojciech Macek if (howto & RB_HALT) 29432d1354aSWojciech Macek opal_call(OPAL_CEC_POWER_DOWN, 0 /* Normal power off */); 29532d1354aSWojciech Macek else 29632d1354aSWojciech Macek opal_call(OPAL_CEC_REBOOT); 297504d9b60SWojciech Macek 298504d9b60SWojciech Macek opal_call(OPAL_RETURN_CPU); 299504d9b60SWojciech Macek } 300504d9b60SWojciech Macek 301504d9b60SWojciech Macek static void 302504d9b60SWojciech Macek opal_intr(void *xintr) 303504d9b60SWojciech Macek { 304504d9b60SWojciech Macek uint64_t events = 0; 305504d9b60SWojciech Macek 306504d9b60SWojciech Macek opal_call(OPAL_HANDLE_INTERRUPT, (uint32_t)(uint64_t)xintr, 307504d9b60SWojciech Macek vtophys(&events)); 308504d9b60SWojciech Macek /* XXX: do something useful with this information */ 309504d9b60SWojciech Macek 31032d1354aSWojciech Macek } 31132d1354aSWojciech Macek 312