sh_timer.c 8.8 KB
Newer Older
1 2 3 4 5 6 7
/*
 * SuperH Timer modules.
 *
 * Copyright (c) 2007 Magnus Damm
 * Based on arm_timer.c by Paul Brook
 * Copyright (c) 2005-2006 CodeSourcery.
 *
M
Matthew Fernandez 已提交
8
 * This code is licensed under the GPL.
9 10
 */

11
#include "hw/hw.h"
P
Paolo Bonzini 已提交
12
#include "hw/sh4/sh.h"
13
#include "qemu/timer.h"
14
#include "exec/address-spaces.h"
15
#include "hw/ptimer.h"
16 17 18 19 20 21 22 23 24 25 26 27 28 29

//#define DEBUG_TIMER

#define TIMER_TCR_TPSC          (7 << 0)
#define TIMER_TCR_CKEG          (3 << 3)
#define TIMER_TCR_UNIE          (1 << 5)
#define TIMER_TCR_ICPE          (3 << 6)
#define TIMER_TCR_UNF           (1 << 8)
#define TIMER_TCR_ICPF          (1 << 9)
#define TIMER_TCR_RESERVED      (0x3f << 10)

#define TIMER_FEAT_CAPT   (1 << 0)
#define TIMER_FEAT_EXTCLK (1 << 1)

A
aurel32 已提交
30 31 32 33 34
#define OFFSET_TCOR   0
#define OFFSET_TCNT   1
#define OFFSET_TCR    2
#define OFFSET_TCPR   3

35 36 37 38 39 40 41 42
typedef struct {
    ptimer_state *timer;
    uint32_t tcnt;
    uint32_t tcor;
    uint32_t tcr;
    uint32_t tcpr;
    int freq;
    int int_level;
43
    int old_level;
44 45
    int feat;
    int enabled;
A
aurel32 已提交
46
    qemu_irq irq;
47 48 49 50 51 52
} sh_timer_state;

/* Check all active timers, and schedule the next timer interrupt. */

static void sh_timer_update(sh_timer_state *s)
{
53 54 55
    int new_level = s->int_level && (s->tcr & TIMER_TCR_UNIE);

    if (new_level != s->old_level)
A
aurel32 已提交
56
      qemu_set_irq (s->irq, new_level);
57 58 59

    s->old_level = s->int_level;
    s->int_level = new_level;
60 61
}

A
Avi Kivity 已提交
62
static uint32_t sh_timer_read(void *opaque, hwaddr offset)
63 64 65 66
{
    sh_timer_state *s = (sh_timer_state *)opaque;

    switch (offset >> 2) {
A
aurel32 已提交
67
    case OFFSET_TCOR:
68
        return s->tcor;
A
aurel32 已提交
69
    case OFFSET_TCNT:
70
        return ptimer_get_count(s->timer);
A
aurel32 已提交
71
    case OFFSET_TCR:
72
        return s->tcr | (s->int_level ? TIMER_TCR_UNF : 0);
A
aurel32 已提交
73
    case OFFSET_TCPR:
74 75 76
        if (s->feat & TIMER_FEAT_CAPT)
            return s->tcpr;
    default:
P
Paul Brook 已提交
77
        hw_error("sh_timer_read: Bad offset %x\n", (int)offset);
78 79 80 81
        return 0;
    }
}

A
Avi Kivity 已提交
82
static void sh_timer_write(void *opaque, hwaddr offset,
83 84 85 86 87 88
                            uint32_t value)
{
    sh_timer_state *s = (sh_timer_state *)opaque;
    int freq;

    switch (offset >> 2) {
A
aurel32 已提交
89
    case OFFSET_TCOR:
90 91 92
        s->tcor = value;
        ptimer_set_limit(s->timer, s->tcor, 0);
        break;
A
aurel32 已提交
93
    case OFFSET_TCNT:
94 95 96
        s->tcnt = value;
        ptimer_set_count(s->timer, s->tcnt);
        break;
A
aurel32 已提交
97
    case OFFSET_TCR:
98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
        if (s->enabled) {
            /* Pause the timer if it is running.  This may cause some
               inaccuracy dure to rounding, but avoids a whole lot of other
               messyness.  */
            ptimer_stop(s->timer);
        }
        freq = s->freq;
        /* ??? Need to recalculate expiry time after changing divisor.  */
        switch (value & TIMER_TCR_TPSC) {
        case 0: freq >>= 2; break;
        case 1: freq >>= 4; break;
        case 2: freq >>= 6; break;
        case 3: freq >>= 8; break;
        case 4: freq >>= 10; break;
	case 6:
	case 7: if (s->feat & TIMER_FEAT_EXTCLK) break;
P
Paul Brook 已提交
114
	default: hw_error("sh_timer_write: Reserved TPSC value\n"); break;
115 116 117 118 119 120
        }
        switch ((value & TIMER_TCR_CKEG) >> 3) {
	case 0: break;
        case 1:
        case 2:
        case 3: if (s->feat & TIMER_FEAT_EXTCLK) break;
P
Paul Brook 已提交
121
	default: hw_error("sh_timer_write: Reserved CKEG value\n"); break;
122 123 124 125 126
        }
        switch ((value & TIMER_TCR_ICPE) >> 6) {
	case 0: break;
        case 2:
        case 3: if (s->feat & TIMER_FEAT_CAPT) break;
P
Paul Brook 已提交
127
	default: hw_error("sh_timer_write: Reserved ICPE value\n"); break;
128 129 130 131 132 133 134
        }
	if ((value & TIMER_TCR_UNF) == 0)
            s->int_level = 0;

	value &= ~TIMER_TCR_UNF;

	if ((value & TIMER_TCR_ICPF) && (!(s->feat & TIMER_FEAT_CAPT)))
P
Paul Brook 已提交
135
            hw_error("sh_timer_write: Reserved ICPF value\n");
136 137 138 139

	value &= ~TIMER_TCR_ICPF; /* capture not supported */

	if (value & TIMER_TCR_RESERVED)
P
Paul Brook 已提交
140
            hw_error("sh_timer_write: Reserved TCR bits set\n");
141 142 143 144 145 146 147 148
        s->tcr = value;
        ptimer_set_limit(s->timer, s->tcor, 0);
        ptimer_set_freq(s->timer, freq);
        if (s->enabled) {
            /* Restart the timer if still enabled.  */
            ptimer_run(s->timer, 0);
        }
        break;
A
aurel32 已提交
149
    case OFFSET_TCPR:
150 151 152 153 154
        if (s->feat & TIMER_FEAT_CAPT) {
            s->tcpr = value;
	    break;
	}
    default:
P
Paul Brook 已提交
155
        hw_error("sh_timer_write: Bad offset %x\n", (int)offset);
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187
    }
    sh_timer_update(s);
}

static void sh_timer_start_stop(void *opaque, int enable)
{
    sh_timer_state *s = (sh_timer_state *)opaque;

#ifdef DEBUG_TIMER
    printf("sh_timer_start_stop %d (%d)\n", enable, s->enabled);
#endif

    if (s->enabled && !enable) {
        ptimer_stop(s->timer);
    }
    if (!s->enabled && enable) {
        ptimer_run(s->timer, 0);
    }
    s->enabled = !!enable;

#ifdef DEBUG_TIMER
    printf("sh_timer_start_stop done %d\n", s->enabled);
#endif
}

static void sh_timer_tick(void *opaque)
{
    sh_timer_state *s = (sh_timer_state *)opaque;
    s->int_level = s->enabled;
    sh_timer_update(s);
}

A
aurel32 已提交
188
static void *sh_timer_init(uint32_t freq, int feat, qemu_irq irq)
189 190 191 192
{
    sh_timer_state *s;
    QEMUBH *bh;

193
    s = (sh_timer_state *)g_malloc0(sizeof(sh_timer_state));
194 195 196 197 198
    s->freq = freq;
    s->feat = feat;
    s->tcor = 0xffffffff;
    s->tcnt = 0xffffffff;
    s->tcpr = 0xdeadbeef;
A
aurel32 已提交
199
    s->tcr = 0;
200
    s->enabled = 0;
201
    s->irq = irq;
202 203 204

    bh = qemu_bh_new(sh_timer_tick, s);
    s->timer = ptimer_init(bh);
A
aurel32 已提交
205 206 207 208 209

    sh_timer_write(s, OFFSET_TCOR >> 2, s->tcor);
    sh_timer_write(s, OFFSET_TCNT >> 2, s->tcnt);
    sh_timer_write(s, OFFSET_TCPR >> 2, s->tcpr);
    sh_timer_write(s, OFFSET_TCR  >> 2, s->tcpr);
210 211 212 213 214
    /* ??? Save/restore.  */
    return s;
}

typedef struct {
B
Benoît Canet 已提交
215 216 217
    MemoryRegion iomem;
    MemoryRegion iomem_p4;
    MemoryRegion iomem_a7;
218 219 220 221 222 223 224
    void *timer[3];
    int level[3];
    uint32_t tocr;
    uint32_t tstr;
    int feat;
} tmu012_state;

A
Avi Kivity 已提交
225
static uint64_t tmu012_read(void *opaque, hwaddr offset,
B
Benoît Canet 已提交
226
                            unsigned size)
227 228 229 230 231 232 233 234 235
{
    tmu012_state *s = (tmu012_state *)opaque;

#ifdef DEBUG_TIMER
    printf("tmu012_read 0x%lx\n", (unsigned long) offset);
#endif

    if (offset >= 0x20) {
        if (!(s->feat & TMU012_FEAT_3CHAN))
P
Paul Brook 已提交
236
	    hw_error("tmu012_write: Bad channel offset %x\n", (int)offset);
237 238 239 240 241 242 243 244 245 246 247 248 249 250 251
        return sh_timer_read(s->timer[2], offset - 0x20);
    }

    if (offset >= 0x14)
        return sh_timer_read(s->timer[1], offset - 0x14);

    if (offset >= 0x08)
        return sh_timer_read(s->timer[0], offset - 0x08);

    if (offset == 4)
        return s->tstr;

    if ((s->feat & TMU012_FEAT_TOCR) && offset == 0)
        return s->tocr;

P
Paul Brook 已提交
252
    hw_error("tmu012_write: Bad offset %x\n", (int)offset);
253 254 255
    return 0;
}

A
Avi Kivity 已提交
256
static void tmu012_write(void *opaque, hwaddr offset,
B
Benoît Canet 已提交
257
                        uint64_t value, unsigned size)
258 259 260 261 262 263 264 265 266
{
    tmu012_state *s = (tmu012_state *)opaque;

#ifdef DEBUG_TIMER
    printf("tmu012_write 0x%lx 0x%08x\n", (unsigned long) offset, value);
#endif

    if (offset >= 0x20) {
        if (!(s->feat & TMU012_FEAT_3CHAN))
P
Paul Brook 已提交
267
	    hw_error("tmu012_write: Bad channel offset %x\n", (int)offset);
268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288
        sh_timer_write(s->timer[2], offset - 0x20, value);
	return;
    }

    if (offset >= 0x14) {
        sh_timer_write(s->timer[1], offset - 0x14, value);
	return;
    }

    if (offset >= 0x08) {
        sh_timer_write(s->timer[0], offset - 0x08, value);
	return;
    }

    if (offset == 4) {
        sh_timer_start_stop(s->timer[0], value & (1 << 0));
        sh_timer_start_stop(s->timer[1], value & (1 << 1));
        if (s->feat & TMU012_FEAT_3CHAN)
            sh_timer_start_stop(s->timer[2], value & (1 << 2));
	else
            if (value & (1 << 2))
P
Paul Brook 已提交
289
                hw_error("tmu012_write: Bad channel\n");
290 291 292 293 294 295 296 297 298 299

	s->tstr = value;
	return;
    }

    if ((s->feat & TMU012_FEAT_TOCR) && offset == 0) {
        s->tocr = value & (1 << 0);
    }
}

B
Benoît Canet 已提交
300 301 302 303
static const MemoryRegionOps tmu012_ops = {
    .read = tmu012_read,
    .write = tmu012_write,
    .endianness = DEVICE_NATIVE_ENDIAN,
304 305
};

A
Avi Kivity 已提交
306
void tmu012_init(MemoryRegion *sysmem, hwaddr base,
B
Benoît Canet 已提交
307
                 int feat, uint32_t freq,
A
aurel32 已提交
308 309
		 qemu_irq ch0_irq, qemu_irq ch1_irq,
		 qemu_irq ch2_irq0, qemu_irq ch2_irq1)
310 311 312 313
{
    tmu012_state *s;
    int timer_feat = (feat & TMU012_FEAT_EXTCLK) ? TIMER_FEAT_EXTCLK : 0;

314
    s = (tmu012_state *)g_malloc0(sizeof(tmu012_state));
315
    s->feat = feat;
316 317
    s->timer[0] = sh_timer_init(freq, timer_feat, ch0_irq);
    s->timer[1] = sh_timer_init(freq, timer_feat, ch1_irq);
318
    if (feat & TMU012_FEAT_3CHAN)
319 320
        s->timer[2] = sh_timer_init(freq, timer_feat | TIMER_FEAT_CAPT,
				    ch2_irq0); /* ch2_irq1 not supported */
B
Benoît Canet 已提交
321 322 323 324 325 326 327 328 329 330 331

    memory_region_init_io(&s->iomem, &tmu012_ops, s,
                          "timer", 0x100000000ULL);

    memory_region_init_alias(&s->iomem_p4, "timer-p4",
                             &s->iomem, 0, 0x1000);
    memory_region_add_subregion(sysmem, P4ADDR(base), &s->iomem_p4);

    memory_region_init_alias(&s->iomem_a7, "timer-a7",
                             &s->iomem, 0, 0x1000);
    memory_region_add_subregion(sysmem, A7ADDR(base), &s->iomem_a7);
332 333
    /* ??? Save/restore.  */
}