svn commit: r259744 - head/sys/arm/ti/am335x
Ian Lepore
ian at FreeBSD.org
Sun Dec 22 21:44:33 UTC 2013
Author: ian
Date: Sun Dec 22 21:44:32 2013
New Revision: 259744
URL: http://svnweb.freebsd.org/changeset/base/259744
Log:
A variety of cleanups...
- Use named constants for register bits, instead of mystery numebrs
scattered around in the code.
- Use inline functions for bus space read/write, instead of macros
that rely on global variables.
- Move the timecounter struct into the softc instead of treating it
as a global variable. Backlink from it to the softc.
- This leaves a pointer to the softc as the only static/global variable
and it's now used only by DELAY().
Modified:
head/sys/arm/ti/am335x/am335x_dmtimer.c
Modified: head/sys/arm/ti/am335x/am335x_dmtimer.c
==============================================================================
--- head/sys/arm/ti/am335x/am335x_dmtimer.c Sun Dec 22 21:35:18 2013 (r259743)
+++ head/sys/arm/ti/am335x/am335x_dmtimer.c Sun Dec 22 21:44:32 2013 (r259744)
@@ -99,13 +99,16 @@ struct am335x_dmtimer_softc {
struct resource * tmr_mem_res[AM335X_NUM_TIMERS];
struct resource * tmr_irq_res[AM335X_NUM_TIMERS];
uint32_t sysclk_freq;
- struct am335x_dmtimer {
- bus_space_tag_t bst;
- bus_space_handle_t bsh;
- struct eventtimer et;
- } t[AM335X_NUM_TIMERS];
+ uint32_t tc_num; /* Which timer number is tc. */
+ uint32_t et_num; /* Which timer number is et. */
+ struct resource * tc_memres; /* Resources for tc timer. */
+ struct resource * et_memres; /* Resources for et timer. */
+ struct timecounter tc;
+ struct eventtimer et;
};
+static struct am335x_dmtimer_softc *am335x_dmtimer_sc;
+
static struct resource_spec am335x_dmtimer_mem_spec[] = {
{ SYS_RES_MEMORY, 0, RF_ACTIVE },
{ SYS_RES_MEMORY, 1, RF_ACTIVE },
@@ -129,51 +132,58 @@ static struct resource_spec am335x_dmtim
{ -1, 0, 0 }
};
-static struct am335x_dmtimer *am335x_dmtimer_tc_tmr = NULL;
+static inline uint32_t
+am335x_dmtimer_tc_read_4(struct am335x_dmtimer_softc *sc, uint32_t reg)
+{
-/* Read/Write macros for Timer used as timecounter */
-#define am335x_dmtimer_tc_read_4(reg) \
- bus_space_read_4(am335x_dmtimer_tc_tmr->bst, \
- am335x_dmtimer_tc_tmr->bsh, reg)
-
-#define am335x_dmtimer_tc_write_4(reg, val) \
- bus_space_write_4(am335x_dmtimer_tc_tmr->bst, \
- am335x_dmtimer_tc_tmr->bsh, reg, val)
-
-/* Read/Write macros for Timer used as eventtimer */
-#define am335x_dmtimer_et_read_4(reg) \
- bus_space_read_4(tmr->bst, tmr->bsh, reg)
-
-#define am335x_dmtimer_et_write_4(reg, val) \
- bus_space_write_4(tmr->bst, tmr->bsh, reg, val)
-
-static unsigned am335x_dmtimer_tc_get_timecount(struct timecounter *);
-
-static struct timecounter am335x_dmtimer_tc = {
- .tc_name = "AM335x Timecounter",
- .tc_get_timecount = am335x_dmtimer_tc_get_timecount,
- .tc_poll_pps = NULL,
- .tc_counter_mask = ~0u,
- .tc_frequency = 0,
- .tc_quality = 1000,
-};
+ return (bus_read_4(sc->tc_memres, reg));
+}
+
+static inline void
+am335x_dmtimer_tc_write_4(struct am335x_dmtimer_softc *sc, uint32_t reg,
+ uint32_t val)
+{
+
+ bus_write_4(sc->tc_memres, reg, val);
+}
+
+static inline uint32_t
+am335x_dmtimer_et_read_4(struct am335x_dmtimer_softc *sc, uint32_t reg)
+{
+
+ return (bus_read_4(sc->et_memres, reg));
+}
+
+static inline void
+am335x_dmtimer_et_write_4(struct am335x_dmtimer_softc *sc, uint32_t reg,
+ uint32_t val)
+{
+
+ bus_write_4(sc->et_memres, reg, val);
+}
static unsigned
am335x_dmtimer_tc_get_timecount(struct timecounter *tc)
{
- return am335x_dmtimer_tc_read_4(DMT_TCRR);
+ struct am335x_dmtimer_softc *sc;
+
+ sc = tc->tc_priv;
+
+ return (am335x_dmtimer_tc_read_4(sc, DMT_TCRR));
}
static int
am335x_dmtimer_start(struct eventtimer *et, sbintime_t first, sbintime_t period)
{
- struct am335x_dmtimer *tmr = (struct am335x_dmtimer *)et->et_priv;
- uint32_t load, count;
- uint32_t tclr = 0;
+ struct am335x_dmtimer_softc *sc;
+ uint32_t count, load, tclr;
+
+ sc = et->et_priv;
+ tclr = 0;
if (period != 0) {
load = ((uint32_t)et->et_frequency * period) >> 32;
- tclr |= 2; /* autoreload bit */
+ tclr |= DMT_TCLR_AUTOLOAD;
panic("periodic timer not implemented\n");
} else {
load = 0;
@@ -185,23 +195,24 @@ am335x_dmtimer_start(struct eventtimer *
count = load;
/* Reset Timer */
- am335x_dmtimer_et_write_4(DMT_TSICR, 2);
+ am335x_dmtimer_et_write_4(sc, DMT_TSICR, DMT_TSICR_RESET);
/* Wait for reset to complete */
- while (am335x_dmtimer_et_read_4(DMT_TIOCP_CFG) & 1);
+ while (am335x_dmtimer_et_read_4(sc, DMT_TIOCP_CFG) & DMT_TIOCP_RESET)
+ continue;
/* set load value */
- am335x_dmtimer_et_write_4(DMT_TLDR, 0xFFFFFFFE - load);
+ am335x_dmtimer_et_write_4(sc, DMT_TLDR, 0xFFFFFFFE - load);
/* set counter value */
- am335x_dmtimer_et_write_4(DMT_TCRR, 0xFFFFFFFE - count);
+ am335x_dmtimer_et_write_4(sc, DMT_TCRR, 0xFFFFFFFE - count);
/* enable overflow interrupt */
- am335x_dmtimer_et_write_4(DMT_IRQENABLE_SET, 2);
+ am335x_dmtimer_et_write_4(sc, DMT_IRQENABLE_SET, DMT_IRQ_OVF);
/* start timer(ST) */
- tclr |= 1;
- am335x_dmtimer_et_write_4(DMT_TCLR, tclr);
+ tclr |= DMT_TCLR_START;
+ am335x_dmtimer_et_write_4(sc, DMT_TCLR, tclr);
return (0);
}
@@ -209,13 +220,15 @@ am335x_dmtimer_start(struct eventtimer *
static int
am335x_dmtimer_stop(struct eventtimer *et)
{
- struct am335x_dmtimer *tmr = (struct am335x_dmtimer *)et->et_priv;
+ struct am335x_dmtimer_softc *sc;
+
+ sc = et->et_priv;
/* Disable all interrupts */
- am335x_dmtimer_et_write_4(DMT_IRQENABLE_CLR, 7);
+ am335x_dmtimer_et_write_4(sc, DMT_IRQENABLE_CLR, DMT_IRQ_MASK);
/* Stop Timer */
- am335x_dmtimer_et_write_4(DMT_TCLR, 0);
+ am335x_dmtimer_et_write_4(sc, DMT_TCLR, 0);
return (0);
}
@@ -223,12 +236,13 @@ am335x_dmtimer_stop(struct eventtimer *e
static int
am335x_dmtimer_intr(void *arg)
{
- struct am335x_dmtimer *tmr = (struct am335x_dmtimer *)arg;
+ struct am335x_dmtimer_softc *sc;
+ sc = arg;
/* Ack interrupt */
- am335x_dmtimer_et_write_4(DMT_IRQSTATUS, 7);
- if (tmr->et.et_active)
- tmr->et.et_event_cb(&tmr->et, tmr->et.et_arg);
+ am335x_dmtimer_et_write_4(sc, DMT_IRQSTATUS, DMT_IRQ_OVF);
+ if (sc->et.et_active)
+ sc->et.et_event_cb(&sc->et, sc->et.et_arg);
return (FILTER_HANDLED);
}
@@ -236,8 +250,6 @@ am335x_dmtimer_intr(void *arg)
static int
am335x_dmtimer_probe(device_t dev)
{
- struct am335x_dmtimer_softc *sc;
- sc = (struct am335x_dmtimer_softc *)device_get_softc(dev);
if (ofw_bus_is_compatible(dev, "ti,am335x-dmtimer")) {
device_set_desc(dev, "AM335x DMTimer");
@@ -250,22 +262,29 @@ am335x_dmtimer_probe(device_t dev)
static int
am335x_dmtimer_attach(device_t dev)
{
- struct am335x_dmtimer_softc *sc = device_get_softc(dev);
+ struct am335x_dmtimer_softc *sc;
void *ihl;
int err;
- int i;
- if (am335x_dmtimer_tc_tmr != NULL)
+ /*
+ * Note that if this routine returns an error status rather than running
+ * to completion it makes no attempt to clean up allocated resources;
+ * the system is essentially dead anyway without functional timers.
+ */
+
+ sc = device_get_softc(dev);
+
+ if (am335x_dmtimer_sc != NULL)
return (EINVAL);
- /* Get the base clock frequency */
+ /* Get the base clock frequency. */
err = ti_prcm_clk_get_source_freq(SYS_CLK, &sc->sysclk_freq);
if (err) {
device_printf(dev, "Error: could not get sysclk frequency\n");
return (ENXIO);
}
- /* Request the memory resources */
+ /* Request the memory resources. */
err = bus_alloc_resources(dev, am335x_dmtimer_mem_spec,
sc->tmr_mem_res);
if (err) {
@@ -273,7 +292,7 @@ am335x_dmtimer_attach(device_t dev)
return (ENXIO);
}
- /* Request the IRQ resources */
+ /* Request the IRQ resources. */
err = bus_alloc_resources(dev, am335x_dmtimer_irq_spec,
sc->tmr_irq_res);
if (err) {
@@ -281,65 +300,64 @@ am335x_dmtimer_attach(device_t dev)
return (ENXIO);
}
- for(i=0;i<AM335X_NUM_TIMERS;i++) {
- sc->t[i].bst = rman_get_bustag(sc->tmr_mem_res[i]);
- sc->t[i].bsh = rman_get_bushandle(sc->tmr_mem_res[i]);
- }
-
- /* Configure DMTimer2 and DMTimer3 source and enable them */
- err = ti_prcm_clk_set_source(DMTIMER2_CLK, SYSCLK_CLK);
- err |= ti_prcm_clk_enable(DMTIMER2_CLK);
- err |= ti_prcm_clk_set_source(DMTIMER3_CLK, SYSCLK_CLK);
- err |= ti_prcm_clk_enable(DMTIMER3_CLK);
+ /* Configure DMTimer3 as eventtimer and DMTimer4 as timecounter. */
+ sc->et_num = 3;
+ sc->tc_num = 2;
+ sc->et_memres = sc->tmr_mem_res[sc->et_num];
+ sc->tc_memres = sc->tmr_mem_res[sc->tc_num];
+
+ err = ti_prcm_clk_set_source(DMTIMER0_CLK + sc->et_num, SYSCLK_CLK);
+ err |= ti_prcm_clk_enable(DMTIMER0_CLK + sc->et_num);
+ err |= ti_prcm_clk_set_source(DMTIMER0_CLK + sc->tc_num, SYSCLK_CLK);
+ err |= ti_prcm_clk_enable(DMTIMER0_CLK + sc->tc_num);
if (err) {
device_printf(dev, "Error: could not setup timer clock\n");
return (ENXIO);
}
- /* Take DMTimer2 for TC */
- am335x_dmtimer_tc_tmr = &sc->t[2];
-
- /* Reset Timer */
- am335x_dmtimer_tc_write_4(DMT_TSICR, 2);
-
- /* Wait for reset to complete */
- while (am335x_dmtimer_tc_read_4(DMT_TIOCP_CFG) & 1);
-
- /* set load value */
- am335x_dmtimer_tc_write_4(DMT_TLDR, 0);
-
- /* set counter value */
- am335x_dmtimer_tc_write_4(DMT_TCRR, 0);
-
- /* Set Timer autoreload(AR) and start timer(ST) */
- am335x_dmtimer_tc_write_4(DMT_TCLR, 3);
-
- am335x_dmtimer_tc.tc_frequency = sc->sysclk_freq;
- tc_init(&am335x_dmtimer_tc);
-
- /* Register DMTimer3 as ET */
-
- /* Setup and enable the timer */
- if (bus_setup_intr(dev, sc->tmr_irq_res[3], INTR_TYPE_CLK,
- am335x_dmtimer_intr, NULL, &sc->t[3], &ihl) != 0) {
+ /* Set up timecounter; register tc. */
+ am335x_dmtimer_tc_write_4(sc, DMT_TSICR, DMT_TSICR_RESET);
+ while (am335x_dmtimer_tc_read_4(sc, DMT_TIOCP_CFG) & DMT_TIOCP_RESET)
+ continue;
+
+ am335x_dmtimer_tc_write_4(sc, DMT_TLDR, 0);
+ am335x_dmtimer_tc_write_4(sc, DMT_TCRR, 0);
+ am335x_dmtimer_tc_write_4(sc, DMT_TCLR,
+ DMT_TCLR_START | DMT_TCLR_AUTOLOAD);
+
+ sc->tc.tc_name = "AM335x Timecounter";
+ sc->tc.tc_get_timecount = am335x_dmtimer_tc_get_timecount;
+ sc->tc.tc_poll_pps = NULL;
+ sc->tc.tc_counter_mask = ~0u;
+ sc->tc.tc_frequency = sc->sysclk_freq;
+ sc->tc.tc_quality = 1000;
+ sc->tc.tc_priv = sc;
+ tc_init(&sc->tc);
+
+ /* Setup eventtimer; register et. */
+ if (bus_setup_intr(dev, sc->tmr_irq_res[sc->et_num], INTR_TYPE_CLK,
+ am335x_dmtimer_intr, NULL, sc, &ihl) != 0) {
bus_release_resources(dev, am335x_dmtimer_irq_spec,
sc->tmr_irq_res);
device_printf(dev, "Unable to setup the clock irq handler.\n");
return (ENXIO);
}
- sc->t[3].et.et_name = "AM335x Eventtimer0";
- sc->t[3].et.et_flags = ET_FLAGS_PERIODIC | ET_FLAGS_ONESHOT;
- sc->t[3].et.et_quality = 1000;
- sc->t[3].et.et_frequency = sc->sysclk_freq;
- sc->t[3].et.et_min_period =
- (0x00000002LLU << 32) / sc->t[3].et.et_frequency;
- sc->t[3].et.et_max_period =
- (0xfffffffeLLU << 32) / sc->t[3].et.et_frequency;
- sc->t[3].et.et_start = am335x_dmtimer_start;
- sc->t[3].et.et_stop = am335x_dmtimer_stop;
- sc->t[3].et.et_priv = &sc->t[3];
- et_register(&sc->t[3].et);
+ sc->et.et_name = "AM335x Eventtimer";
+ sc->et.et_flags = ET_FLAGS_ONESHOT;
+ sc->et.et_quality = 1000;
+ sc->et.et_frequency = sc->sysclk_freq;
+ sc->et.et_min_period =
+ ((0x00000005LLU << 32) / sc->et.et_frequency);
+ sc->et.et_max_period =
+ (0xfffffffeLLU << 32) / sc->et.et_frequency;
+ sc->et.et_start = am335x_dmtimer_start;
+ sc->et.et_stop = am335x_dmtimer_stop;
+ sc->et.et_priv = sc;
+ et_register(&sc->et);
+
+ /* Store a pointer to the softc for use in DELAY(). */
+ am335x_dmtimer_sc = sc;
return (0);
}
@@ -370,10 +388,13 @@ cpu_initclocks(void)
void
DELAY(int usec)
{
- int32_t counts;
+ struct am335x_dmtimer_softc *sc;
+ int32_t counts;
uint32_t first, last;
- if (am335x_dmtimer_tc_tmr == NULL) {
+ sc = am335x_dmtimer_sc;
+
+ if (sc == NULL) {
for (; usec > 0; usec--)
for (counts = 200; counts > 0; counts--)
/* Prevent gcc from optimizing out the loop */
@@ -382,13 +403,13 @@ DELAY(int usec)
}
/* Get the number of times to count */
- counts = usec * (am335x_dmtimer_tc.tc_frequency / 1000000) + 1;
+ counts = (usec + 1) * (sc->sysclk_freq / 1000000);
- first = am335x_dmtimer_tc_read_4(DMT_TCRR);
+ first = am335x_dmtimer_tc_read_4(sc, DMT_TCRR);
while (counts > 0) {
- last = am335x_dmtimer_tc_read_4(DMT_TCRR);
- if (last>first) {
+ last = am335x_dmtimer_tc_read_4(sc, DMT_TCRR);
+ if (last > first) {
counts -= (int32_t)(last - first);
} else {
counts -= (int32_t)((0xFFFFFFFF - first) + last);
More information about the svn-src-head
mailing list