You need to sign in or sign up before continuing.
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 "qemu/main-loop.h"
15
#include "exec/address-spaces.h"
16
#include "hw/ptimer.h"
17 18 19 20 21 22 23 24 25 26 27 28 29 30

//#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 已提交
31 32 33 34 35
#define OFFSET_TCOR   0
#define OFFSET_TCNT   1
#define OFFSET_TCR    2
#define OFFSET_TCPR   3

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

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

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

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

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

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

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

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

    switch (offset >> 2) {
A
aurel32 已提交
90
    case OFFSET_TCOR:
91 92 93
        s->tcor = value;
        ptimer_set_limit(s->timer, s->tcor, 0);
        break;
A
aurel32 已提交
94
    case OFFSET_TCNT:
95 96 97
        s->tcnt = value;
        ptimer_set_count(s->timer, s->tcnt);
        break;
A
aurel32 已提交
98
    case OFFSET_TCR:
99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114
        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 已提交
115
	default: hw_error("sh_timer_write: Reserved TPSC value\n"); break;
116 117 118 119 120 121
        }
        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 已提交
122
	default: hw_error("sh_timer_write: Reserved CKEG value\n"); break;
123 124 125 126 127
        }
        switch ((value & TIMER_TCR_ICPE) >> 6) {
	case 0: break;
        case 2:
        case 3: if (s->feat & TIMER_FEAT_CAPT) break;
P
Paul Brook 已提交
128
	default: hw_error("sh_timer_write: Reserved ICPE value\n"); break;
129 130 131 132 133 134 135
        }
	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 已提交
136
            hw_error("sh_timer_write: Reserved ICPF value\n");
137 138 139 140

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

	if (value & TIMER_TCR_RESERVED)
P
Paul Brook 已提交
141
            hw_error("sh_timer_write: Reserved TCR bits set\n");
142 143 144 145 146 147 148 149
        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 已提交
150
    case OFFSET_TCPR:
151 152 153 154 155
        if (s->feat & TIMER_FEAT_CAPT) {
            s->tcpr = value;
	    break;
	}
    default:
P
Paul Brook 已提交
156
        hw_error("sh_timer_write: Bad offset %x\n", (int)offset);
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 188
    }
    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 已提交
189
static void *sh_timer_init(uint32_t freq, int feat, qemu_irq irq)
190 191 192 193
{
    sh_timer_state *s;
    QEMUBH *bh;

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

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

    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);
211 212 213 214 215
    /* ??? Save/restore.  */
    return s;
}

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

A
Avi Kivity 已提交
226
static uint64_t tmu012_read(void *opaque, hwaddr offset,
B
Benoît Canet 已提交
227
                            unsigned size)
228 229 230 231 232 233 234 235 236
{
    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 已提交
237
	    hw_error("tmu012_write: Bad channel offset %x\n", (int)offset);
238 239 240 241 242 243 244 245 246 247 248 249 250 251 252
        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 已提交
253
    hw_error("tmu012_write: Bad offset %x\n", (int)offset);
254 255 256
    return 0;
}

A
Avi Kivity 已提交
257
static void tmu012_write(void *opaque, hwaddr offset,
B
Benoît Canet 已提交
258
                        uint64_t value, unsigned size)
259 260 261 262 263 264 265 266 267
{
    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 已提交
268
	    hw_error("tmu012_write: Bad channel offset %x\n", (int)offset);
269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289
        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 已提交
290
                hw_error("tmu012_write: Bad channel\n");
291 292 293 294 295 296 297 298 299 300

	s->tstr = value;
	return;
    }

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

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

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

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

323
    memory_region_init_io(&s->iomem, NULL, &tmu012_ops, s,
B
Benoît Canet 已提交
324 325
                          "timer", 0x100000000ULL);

326
    memory_region_init_alias(&s->iomem_p4, NULL, "timer-p4",
B
Benoît Canet 已提交
327 328 329
                             &s->iomem, 0, 0x1000);
    memory_region_add_subregion(sysmem, P4ADDR(base), &s->iomem_p4);

330
    memory_region_init_alias(&s->iomem_a7, NULL, "timer-a7",
B
Benoît Canet 已提交
331 332
                             &s->iomem, 0, 0x1000);
    memory_region_add_subregion(sysmem, A7ADDR(base), &s->iomem_a7);
333 334
    /* ??? Save/restore.  */
}