1*a2c472e7SAleksandr Rybalko /*- 2*a2c472e7SAleksandr Rybalko * Copyright (c) 2012 The FreeBSD Foundation 3*a2c472e7SAleksandr Rybalko * All rights reserved. 4*a2c472e7SAleksandr Rybalko * 5*a2c472e7SAleksandr Rybalko * This software was developed by Oleksandr Rybalko under sponsorship 6*a2c472e7SAleksandr Rybalko * from the FreeBSD Foundation. 7*a2c472e7SAleksandr Rybalko * 8*a2c472e7SAleksandr Rybalko * Redistribution and use in source and binary forms, with or without 9*a2c472e7SAleksandr Rybalko * modification, are permitted provided that the following conditions 10*a2c472e7SAleksandr Rybalko * are met: 11*a2c472e7SAleksandr Rybalko * 1. Redistributions of source code must retain the above copyright 12*a2c472e7SAleksandr Rybalko * notice, this list of conditions and the following disclaimer. 13*a2c472e7SAleksandr Rybalko * 2. Redistributions in binary form must reproduce the above copyright 14*a2c472e7SAleksandr Rybalko * notice, this list of conditions and the following disclaimer in the 15*a2c472e7SAleksandr Rybalko * documentation and/or other materials provided with the distribution. 16*a2c472e7SAleksandr Rybalko * 17*a2c472e7SAleksandr Rybalko * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 18*a2c472e7SAleksandr Rybalko * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19*a2c472e7SAleksandr Rybalko * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20*a2c472e7SAleksandr Rybalko * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 21*a2c472e7SAleksandr Rybalko * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 22*a2c472e7SAleksandr Rybalko * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 23*a2c472e7SAleksandr Rybalko * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 24*a2c472e7SAleksandr Rybalko * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 25*a2c472e7SAleksandr Rybalko * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 26*a2c472e7SAleksandr Rybalko * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 27*a2c472e7SAleksandr Rybalko * SUCH DAMAGE. 28*a2c472e7SAleksandr Rybalko */ 29*a2c472e7SAleksandr Rybalko 30*a2c472e7SAleksandr Rybalko #include <sys/cdefs.h> 31*a2c472e7SAleksandr Rybalko __FBSDID("$FreeBSD$"); 32*a2c472e7SAleksandr Rybalko 33*a2c472e7SAleksandr Rybalko #include <sys/param.h> 34*a2c472e7SAleksandr Rybalko #include <sys/systm.h> 35*a2c472e7SAleksandr Rybalko #include <sys/bus.h> 36*a2c472e7SAleksandr Rybalko #include <sys/kernel.h> 37*a2c472e7SAleksandr Rybalko #include <sys/module.h> 38*a2c472e7SAleksandr Rybalko #include <sys/malloc.h> 39*a2c472e7SAleksandr Rybalko #include <sys/rman.h> 40*a2c472e7SAleksandr Rybalko #include <sys/timeet.h> 41*a2c472e7SAleksandr Rybalko #include <sys/timetc.h> 42*a2c472e7SAleksandr Rybalko #include <sys/watchdog.h> 43*a2c472e7SAleksandr Rybalko #include <machine/bus.h> 44*a2c472e7SAleksandr Rybalko #include <machine/cpu.h> 45*a2c472e7SAleksandr Rybalko #include <machine/frame.h> 46*a2c472e7SAleksandr Rybalko #include <machine/intr.h> 47*a2c472e7SAleksandr Rybalko 48*a2c472e7SAleksandr Rybalko #include <machine/fdt.h> 49*a2c472e7SAleksandr Rybalko #include <dev/fdt/fdt_common.h> 50*a2c472e7SAleksandr Rybalko #include <dev/ofw/openfirm.h> 51*a2c472e7SAleksandr Rybalko #include <dev/ofw/ofw_bus.h> 52*a2c472e7SAleksandr Rybalko #include <dev/ofw/ofw_bus_subr.h> 53*a2c472e7SAleksandr Rybalko 54*a2c472e7SAleksandr Rybalko #include <arm/freescale/imx/imx_gptvar.h> 55*a2c472e7SAleksandr Rybalko #include <arm/freescale/imx/imx_gptreg.h> 56*a2c472e7SAleksandr Rybalko 57*a2c472e7SAleksandr Rybalko #include <sys/kdb.h> 58*a2c472e7SAleksandr Rybalko #include <arm/freescale/imx/imx51_ccmvar.h> 59*a2c472e7SAleksandr Rybalko 60*a2c472e7SAleksandr Rybalko #define MIN_PERIOD 100LLU 61*a2c472e7SAleksandr Rybalko 62*a2c472e7SAleksandr Rybalko #define WRITE4(_sc, _r, _v) \ 63*a2c472e7SAleksandr Rybalko bus_space_write_4((_sc)->sc_iot, (_sc)->sc_ioh, (_r), (_v)) 64*a2c472e7SAleksandr Rybalko #define READ4(_sc, _r) \ 65*a2c472e7SAleksandr Rybalko bus_space_read_4((_sc)->sc_iot, (_sc)->sc_ioh, (_r)) 66*a2c472e7SAleksandr Rybalko #define SET4(_sc, _r, _m) \ 67*a2c472e7SAleksandr Rybalko WRITE4((_sc), (_r), READ4((_sc), (_r)) | (_m)) 68*a2c472e7SAleksandr Rybalko #define CLEAR4(_sc, _r, _m) \ 69*a2c472e7SAleksandr Rybalko WRITE4((_sc), (_r), READ4((_sc), (_r)) & ~(_m)) 70*a2c472e7SAleksandr Rybalko 71*a2c472e7SAleksandr Rybalko static u_int imx_gpt_get_timecount(struct timecounter *); 72*a2c472e7SAleksandr Rybalko static int imx_gpt_timer_start(struct eventtimer *, sbintime_t, 73*a2c472e7SAleksandr Rybalko sbintime_t); 74*a2c472e7SAleksandr Rybalko static int imx_gpt_timer_stop(struct eventtimer *); 75*a2c472e7SAleksandr Rybalko 76*a2c472e7SAleksandr Rybalko static int imx_gpt_intr(void *); 77*a2c472e7SAleksandr Rybalko static int imx_gpt_probe(device_t); 78*a2c472e7SAleksandr Rybalko static int imx_gpt_attach(device_t); 79*a2c472e7SAleksandr Rybalko 80*a2c472e7SAleksandr Rybalko static struct timecounter imx_gpt_timecounter = { 81*a2c472e7SAleksandr Rybalko .tc_name = "i.MX GPT Timecounter", 82*a2c472e7SAleksandr Rybalko .tc_get_timecount = imx_gpt_get_timecount, 83*a2c472e7SAleksandr Rybalko .tc_counter_mask = ~0u, 84*a2c472e7SAleksandr Rybalko .tc_frequency = 0, 85*a2c472e7SAleksandr Rybalko .tc_quality = 500, 86*a2c472e7SAleksandr Rybalko }; 87*a2c472e7SAleksandr Rybalko 88*a2c472e7SAleksandr Rybalko struct imx_gpt_softc *imx_gpt_sc = NULL; 89*a2c472e7SAleksandr Rybalko static volatile int imx_gpt_delay_count = 300; 90*a2c472e7SAleksandr Rybalko 91*a2c472e7SAleksandr Rybalko static struct resource_spec imx_gpt_spec[] = { 92*a2c472e7SAleksandr Rybalko { SYS_RES_MEMORY, 0, RF_ACTIVE }, 93*a2c472e7SAleksandr Rybalko { SYS_RES_IRQ, 0, RF_ACTIVE }, 94*a2c472e7SAleksandr Rybalko { -1, 0 } 95*a2c472e7SAleksandr Rybalko }; 96*a2c472e7SAleksandr Rybalko 97*a2c472e7SAleksandr Rybalko static int 98*a2c472e7SAleksandr Rybalko imx_gpt_probe(device_t dev) 99*a2c472e7SAleksandr Rybalko { 100*a2c472e7SAleksandr Rybalko 101*a2c472e7SAleksandr Rybalko if (!ofw_bus_is_compatible(dev, "fsl,imx51-gpt")) 102*a2c472e7SAleksandr Rybalko return (ENXIO); 103*a2c472e7SAleksandr Rybalko 104*a2c472e7SAleksandr Rybalko device_set_desc(dev, "Freescale i.MXxxx GPT timer"); 105*a2c472e7SAleksandr Rybalko return (BUS_PROBE_DEFAULT); 106*a2c472e7SAleksandr Rybalko } 107*a2c472e7SAleksandr Rybalko 108*a2c472e7SAleksandr Rybalko static int 109*a2c472e7SAleksandr Rybalko imx_gpt_attach(device_t dev) 110*a2c472e7SAleksandr Rybalko { 111*a2c472e7SAleksandr Rybalko struct imx_gpt_softc *sc; 112*a2c472e7SAleksandr Rybalko int err; 113*a2c472e7SAleksandr Rybalko 114*a2c472e7SAleksandr Rybalko sc = device_get_softc(dev); 115*a2c472e7SAleksandr Rybalko 116*a2c472e7SAleksandr Rybalko if (bus_alloc_resources(dev, imx_gpt_spec, sc->res)) { 117*a2c472e7SAleksandr Rybalko device_printf(dev, "could not allocate resources\n"); 118*a2c472e7SAleksandr Rybalko return (ENXIO); 119*a2c472e7SAleksandr Rybalko } 120*a2c472e7SAleksandr Rybalko 121*a2c472e7SAleksandr Rybalko sc->sc_dev = dev; 122*a2c472e7SAleksandr Rybalko sc->sc_clksrc = GPT_CR_CLKSRC_IPG; 123*a2c472e7SAleksandr Rybalko sc->sc_iot = rman_get_bustag(sc->res[0]); 124*a2c472e7SAleksandr Rybalko sc->sc_ioh = rman_get_bushandle(sc->res[0]); 125*a2c472e7SAleksandr Rybalko 126*a2c472e7SAleksandr Rybalko switch (sc->sc_clksrc) { 127*a2c472e7SAleksandr Rybalko case GPT_CR_CLKSRC_NONE: 128*a2c472e7SAleksandr Rybalko device_printf(dev, "can't run timer without clock source\n"); 129*a2c472e7SAleksandr Rybalko return (EINVAL); 130*a2c472e7SAleksandr Rybalko case GPT_CR_CLKSRC_EXT: 131*a2c472e7SAleksandr Rybalko device_printf(dev, "Not implemented. Geve me the way to get " 132*a2c472e7SAleksandr Rybalko "external clock source frequency\n"); 133*a2c472e7SAleksandr Rybalko return (EINVAL); 134*a2c472e7SAleksandr Rybalko case GPT_CR_CLKSRC_32K: 135*a2c472e7SAleksandr Rybalko sc->clkfreq = 32768; 136*a2c472e7SAleksandr Rybalko break; 137*a2c472e7SAleksandr Rybalko case GPT_CR_CLKSRC_IPG_HIGH: 138*a2c472e7SAleksandr Rybalko sc->clkfreq = imx51_get_clock(IMX51CLK_IPG_CLK_ROOT) * 2; 139*a2c472e7SAleksandr Rybalko break; 140*a2c472e7SAleksandr Rybalko default: 141*a2c472e7SAleksandr Rybalko sc->clkfreq = imx51_get_clock(IMX51CLK_IPG_CLK_ROOT); 142*a2c472e7SAleksandr Rybalko } 143*a2c472e7SAleksandr Rybalko device_printf(dev, "Run on %dKHz clock.\n", sc->clkfreq / 1000); 144*a2c472e7SAleksandr Rybalko 145*a2c472e7SAleksandr Rybalko /* Reset */ 146*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_CR, GPT_CR_SWR); 147*a2c472e7SAleksandr Rybalko /* Enable and setup counters */ 148*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_CR, 149*a2c472e7SAleksandr Rybalko GPT_CR_CLKSRC_IPG | /* Use IPG clock */ 150*a2c472e7SAleksandr Rybalko GPT_CR_FRR | /* Just count (FreeRunner mode) */ 151*a2c472e7SAleksandr Rybalko GPT_CR_STOPEN | /* Run in STOP mode */ 152*a2c472e7SAleksandr Rybalko GPT_CR_WAITEN | /* Run in WAIT mode */ 153*a2c472e7SAleksandr Rybalko GPT_CR_DBGEN); /* Run in DEBUG mode */ 154*a2c472e7SAleksandr Rybalko 155*a2c472e7SAleksandr Rybalko /* Disable interrupts */ 156*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_IR, 0); 157*a2c472e7SAleksandr Rybalko 158*a2c472e7SAleksandr Rybalko /* Tick every 10us */ 159*a2c472e7SAleksandr Rybalko /* XXX: must be calculated from clock source frequency */ 160*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_PR, 665); 161*a2c472e7SAleksandr Rybalko /* Use 100 KHz */ 162*a2c472e7SAleksandr Rybalko sc->clkfreq = 100000; 163*a2c472e7SAleksandr Rybalko 164*a2c472e7SAleksandr Rybalko /* Setup and enable the timer interrupt */ 165*a2c472e7SAleksandr Rybalko err = bus_setup_intr(dev, sc->res[1], INTR_TYPE_CLK, imx_gpt_intr, 166*a2c472e7SAleksandr Rybalko NULL, sc, &sc->sc_ih); 167*a2c472e7SAleksandr Rybalko if (err != 0) { 168*a2c472e7SAleksandr Rybalko bus_release_resources(dev, imx_gpt_spec, sc->res); 169*a2c472e7SAleksandr Rybalko device_printf(dev, "Unable to setup the clock irq handler, " 170*a2c472e7SAleksandr Rybalko "err = %d\n", err); 171*a2c472e7SAleksandr Rybalko return (ENXIO); 172*a2c472e7SAleksandr Rybalko } 173*a2c472e7SAleksandr Rybalko 174*a2c472e7SAleksandr Rybalko sc->et.et_name = "i.MXxxx GPT Eventtimer"; 175*a2c472e7SAleksandr Rybalko sc->et.et_flags = ET_FLAGS_ONESHOT | ET_FLAGS_PERIODIC; 176*a2c472e7SAleksandr Rybalko sc->et.et_quality = 1000; 177*a2c472e7SAleksandr Rybalko sc->et.et_frequency = sc->clkfreq; 178*a2c472e7SAleksandr Rybalko sc->et.et_min_period = (MIN_PERIOD << 32) / sc->et.et_frequency; 179*a2c472e7SAleksandr Rybalko sc->et.et_max_period = (0xfffffffeLLU << 32) / sc->et.et_frequency; 180*a2c472e7SAleksandr Rybalko sc->et.et_start = imx_gpt_timer_start; 181*a2c472e7SAleksandr Rybalko sc->et.et_stop = imx_gpt_timer_stop; 182*a2c472e7SAleksandr Rybalko sc->et.et_priv = sc; 183*a2c472e7SAleksandr Rybalko et_register(&sc->et); 184*a2c472e7SAleksandr Rybalko 185*a2c472e7SAleksandr Rybalko /* Disable interrupts */ 186*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_IR, 0); 187*a2c472e7SAleksandr Rybalko /* ACK any panding interrupts */ 188*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_SR, (GPT_IR_ROV << 1) - 1); 189*a2c472e7SAleksandr Rybalko 190*a2c472e7SAleksandr Rybalko if (device_get_unit(dev) == 0) 191*a2c472e7SAleksandr Rybalko imx_gpt_sc = sc; 192*a2c472e7SAleksandr Rybalko 193*a2c472e7SAleksandr Rybalko imx_gpt_timecounter.tc_frequency = sc->clkfreq; 194*a2c472e7SAleksandr Rybalko tc_init(&imx_gpt_timecounter); 195*a2c472e7SAleksandr Rybalko 196*a2c472e7SAleksandr Rybalko printf("clock: hz=%d stathz = %d\n", hz, stathz); 197*a2c472e7SAleksandr Rybalko 198*a2c472e7SAleksandr Rybalko device_printf(sc->sc_dev, "timer clock frequency %d\n", sc->clkfreq); 199*a2c472e7SAleksandr Rybalko 200*a2c472e7SAleksandr Rybalko imx_gpt_delay_count = imx51_get_clock(IMX51CLK_ARM_ROOT) / 4000000; 201*a2c472e7SAleksandr Rybalko 202*a2c472e7SAleksandr Rybalko SET4(sc, IMX_GPT_CR, GPT_CR_EN); 203*a2c472e7SAleksandr Rybalko 204*a2c472e7SAleksandr Rybalko return (0); 205*a2c472e7SAleksandr Rybalko } 206*a2c472e7SAleksandr Rybalko 207*a2c472e7SAleksandr Rybalko static int 208*a2c472e7SAleksandr Rybalko imx_gpt_timer_start(struct eventtimer *et, sbintime_t first, sbintime_t period) 209*a2c472e7SAleksandr Rybalko { 210*a2c472e7SAleksandr Rybalko struct imx_gpt_softc *sc; 211*a2c472e7SAleksandr Rybalko uint32_t ticks; 212*a2c472e7SAleksandr Rybalko 213*a2c472e7SAleksandr Rybalko sc = (struct imx_gpt_softc *)et->et_priv; 214*a2c472e7SAleksandr Rybalko 215*a2c472e7SAleksandr Rybalko if (period != 0) { 216*a2c472e7SAleksandr Rybalko sc->sc_period = ((uint32_t)et->et_frequency * period) >> 32; 217*a2c472e7SAleksandr Rybalko /* Set expected value */ 218*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_OCR2, READ4(sc, IMX_GPT_CNT) + sc->sc_period); 219*a2c472e7SAleksandr Rybalko /* Enable compare register 2 Interrupt */ 220*a2c472e7SAleksandr Rybalko SET4(sc, IMX_GPT_IR, GPT_IR_OF2); 221*a2c472e7SAleksandr Rybalko } else if (first != 0) { 222*a2c472e7SAleksandr Rybalko ticks = ((uint32_t)et->et_frequency * first) >> 32; 223*a2c472e7SAleksandr Rybalko 224*a2c472e7SAleksandr Rybalko /* 225*a2c472e7SAleksandr Rybalko * TODO: setupt second compare reg with time which will save 226*a2c472e7SAleksandr Rybalko * us in case correct one lost, f.e. if period to short and 227*a2c472e7SAleksandr Rybalko * setup done later than counter reach target value. 228*a2c472e7SAleksandr Rybalko */ 229*a2c472e7SAleksandr Rybalko /* Do not disturb, otherwise event will be lost */ 230*a2c472e7SAleksandr Rybalko spinlock_enter(); 231*a2c472e7SAleksandr Rybalko /* Set expected value */ 232*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_OCR1, READ4(sc, IMX_GPT_CNT) + ticks); 233*a2c472e7SAleksandr Rybalko /* Enable compare register 1 Interrupt */ 234*a2c472e7SAleksandr Rybalko SET4(sc, IMX_GPT_IR, GPT_IR_OF1); 235*a2c472e7SAleksandr Rybalko /* Now everybody can relax */ 236*a2c472e7SAleksandr Rybalko spinlock_exit(); 237*a2c472e7SAleksandr Rybalko 238*a2c472e7SAleksandr Rybalko return (0); 239*a2c472e7SAleksandr Rybalko } 240*a2c472e7SAleksandr Rybalko 241*a2c472e7SAleksandr Rybalko return (EINVAL); 242*a2c472e7SAleksandr Rybalko } 243*a2c472e7SAleksandr Rybalko 244*a2c472e7SAleksandr Rybalko static int 245*a2c472e7SAleksandr Rybalko imx_gpt_timer_stop(struct eventtimer *et) 246*a2c472e7SAleksandr Rybalko { 247*a2c472e7SAleksandr Rybalko struct imx_gpt_softc *sc; 248*a2c472e7SAleksandr Rybalko 249*a2c472e7SAleksandr Rybalko sc = (struct imx_gpt_softc *)et->et_priv; 250*a2c472e7SAleksandr Rybalko 251*a2c472e7SAleksandr Rybalko /* Disable OF2 Interrupt */ 252*a2c472e7SAleksandr Rybalko CLEAR4(sc, IMX_GPT_IR, GPT_IR_OF2); 253*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_SR, GPT_IR_OF2); 254*a2c472e7SAleksandr Rybalko sc->sc_period = 0; 255*a2c472e7SAleksandr Rybalko 256*a2c472e7SAleksandr Rybalko return (0); 257*a2c472e7SAleksandr Rybalko } 258*a2c472e7SAleksandr Rybalko 259*a2c472e7SAleksandr Rybalko int 260*a2c472e7SAleksandr Rybalko imx_gpt_get_timerfreq(struct imx_gpt_softc *sc) 261*a2c472e7SAleksandr Rybalko { 262*a2c472e7SAleksandr Rybalko 263*a2c472e7SAleksandr Rybalko return (sc->clkfreq); 264*a2c472e7SAleksandr Rybalko } 265*a2c472e7SAleksandr Rybalko 266*a2c472e7SAleksandr Rybalko void 267*a2c472e7SAleksandr Rybalko cpu_initclocks(void) 268*a2c472e7SAleksandr Rybalko { 269*a2c472e7SAleksandr Rybalko 270*a2c472e7SAleksandr Rybalko if (!imx_gpt_sc) { 271*a2c472e7SAleksandr Rybalko panic("%s: driver has not been initialized!", __func__); 272*a2c472e7SAleksandr Rybalko } 273*a2c472e7SAleksandr Rybalko 274*a2c472e7SAleksandr Rybalko cpu_initclocks_bsp(); 275*a2c472e7SAleksandr Rybalko 276*a2c472e7SAleksandr Rybalko /* Switch to DELAY using counter */ 277*a2c472e7SAleksandr Rybalko imx_gpt_delay_count = 0; 278*a2c472e7SAleksandr Rybalko device_printf(imx_gpt_sc->sc_dev, 279*a2c472e7SAleksandr Rybalko "switch DELAY to use H/W counter\n"); 280*a2c472e7SAleksandr Rybalko } 281*a2c472e7SAleksandr Rybalko 282*a2c472e7SAleksandr Rybalko static int 283*a2c472e7SAleksandr Rybalko imx_gpt_intr(void *arg) 284*a2c472e7SAleksandr Rybalko { 285*a2c472e7SAleksandr Rybalko struct imx_gpt_softc *sc; 286*a2c472e7SAleksandr Rybalko uint32_t status; 287*a2c472e7SAleksandr Rybalko 288*a2c472e7SAleksandr Rybalko sc = (struct imx_gpt_softc *)arg; 289*a2c472e7SAleksandr Rybalko 290*a2c472e7SAleksandr Rybalko /* Sometime we not get staus bit when interrupt arrive. Cache? */ 291*a2c472e7SAleksandr Rybalko while (!(status = READ4(sc, IMX_GPT_SR))) 292*a2c472e7SAleksandr Rybalko ; 293*a2c472e7SAleksandr Rybalko 294*a2c472e7SAleksandr Rybalko if (status & GPT_IR_OF1) { 295*a2c472e7SAleksandr Rybalko if (sc->et.et_active) { 296*a2c472e7SAleksandr Rybalko sc->et.et_event_cb(&sc->et, sc->et.et_arg); 297*a2c472e7SAleksandr Rybalko } 298*a2c472e7SAleksandr Rybalko } 299*a2c472e7SAleksandr Rybalko if (status & GPT_IR_OF2) { 300*a2c472e7SAleksandr Rybalko if (sc->et.et_active) { 301*a2c472e7SAleksandr Rybalko sc->et.et_event_cb(&sc->et, sc->et.et_arg); 302*a2c472e7SAleksandr Rybalko /* Set expected value */ 303*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_OCR2, READ4(sc, IMX_GPT_CNT) + 304*a2c472e7SAleksandr Rybalko sc->sc_period); 305*a2c472e7SAleksandr Rybalko } 306*a2c472e7SAleksandr Rybalko } 307*a2c472e7SAleksandr Rybalko 308*a2c472e7SAleksandr Rybalko /* ACK */ 309*a2c472e7SAleksandr Rybalko WRITE4(sc, IMX_GPT_SR, status); 310*a2c472e7SAleksandr Rybalko 311*a2c472e7SAleksandr Rybalko return (FILTER_HANDLED); 312*a2c472e7SAleksandr Rybalko } 313*a2c472e7SAleksandr Rybalko 314*a2c472e7SAleksandr Rybalko u_int 315*a2c472e7SAleksandr Rybalko imx_gpt_get_timecount(struct timecounter *tc) 316*a2c472e7SAleksandr Rybalko { 317*a2c472e7SAleksandr Rybalko 318*a2c472e7SAleksandr Rybalko if (imx_gpt_sc == NULL) 319*a2c472e7SAleksandr Rybalko return (0); 320*a2c472e7SAleksandr Rybalko 321*a2c472e7SAleksandr Rybalko return (READ4(imx_gpt_sc, IMX_GPT_CNT)); 322*a2c472e7SAleksandr Rybalko } 323*a2c472e7SAleksandr Rybalko 324*a2c472e7SAleksandr Rybalko static device_method_t imx_gpt_methods[] = { 325*a2c472e7SAleksandr Rybalko DEVMETHOD(device_probe, imx_gpt_probe), 326*a2c472e7SAleksandr Rybalko DEVMETHOD(device_attach, imx_gpt_attach), 327*a2c472e7SAleksandr Rybalko 328*a2c472e7SAleksandr Rybalko DEVMETHOD_END 329*a2c472e7SAleksandr Rybalko }; 330*a2c472e7SAleksandr Rybalko 331*a2c472e7SAleksandr Rybalko static driver_t imx_gpt_driver = { 332*a2c472e7SAleksandr Rybalko "imx_gpt", 333*a2c472e7SAleksandr Rybalko imx_gpt_methods, 334*a2c472e7SAleksandr Rybalko sizeof(struct imx_gpt_softc), 335*a2c472e7SAleksandr Rybalko }; 336*a2c472e7SAleksandr Rybalko 337*a2c472e7SAleksandr Rybalko static devclass_t imx_gpt_devclass; 338*a2c472e7SAleksandr Rybalko 339*a2c472e7SAleksandr Rybalko EARLY_DRIVER_MODULE(imx_gpt, simplebus, imx_gpt_driver, imx_gpt_devclass, 0, 340*a2c472e7SAleksandr Rybalko 0, BUS_PASS_TIMER); 341*a2c472e7SAleksandr Rybalko 342*a2c472e7SAleksandr Rybalko void 343*a2c472e7SAleksandr Rybalko DELAY(int usec) 344*a2c472e7SAleksandr Rybalko { 345*a2c472e7SAleksandr Rybalko int32_t counts; 346*a2c472e7SAleksandr Rybalko uint32_t last; 347*a2c472e7SAleksandr Rybalko 348*a2c472e7SAleksandr Rybalko /* 349*a2c472e7SAleksandr Rybalko * Check the timers are setup, if not just use a for loop for the 350*a2c472e7SAleksandr Rybalko * meantime. 351*a2c472e7SAleksandr Rybalko */ 352*a2c472e7SAleksandr Rybalko if (imx_gpt_delay_count) { 353*a2c472e7SAleksandr Rybalko for (; usec > 0; usec--) 354*a2c472e7SAleksandr Rybalko for (counts = imx_gpt_delay_count; counts > 0; 355*a2c472e7SAleksandr Rybalko counts--) 356*a2c472e7SAleksandr Rybalko /* Prevent optimizing out the loop */ 357*a2c472e7SAleksandr Rybalko cpufunc_nullop(); 358*a2c472e7SAleksandr Rybalko return; 359*a2c472e7SAleksandr Rybalko } 360*a2c472e7SAleksandr Rybalko 361*a2c472e7SAleksandr Rybalko /* At least 1 count */ 362*a2c472e7SAleksandr Rybalko usec = MAX(1, usec / 100); 363*a2c472e7SAleksandr Rybalko 364*a2c472e7SAleksandr Rybalko last = READ4(imx_gpt_sc, IMX_GPT_CNT) + usec; 365*a2c472e7SAleksandr Rybalko while (READ4(imx_gpt_sc, IMX_GPT_CNT) < last) { 366*a2c472e7SAleksandr Rybalko /* Prevent optimizing out the loop */ 367*a2c472e7SAleksandr Rybalko cpufunc_nullop(); 368*a2c472e7SAleksandr Rybalko } 369*a2c472e7SAleksandr Rybalko /* TODO: use interrupt on OCR2 */ 370*a2c472e7SAleksandr Rybalko } 371