Statistics
| Branch: | Revision:

root / hw / etraxfs_timer.c @ 71cf9e62

History | View | Annotate | Download (8.4 kB)

1 83fa1010 ths
/*
2 e62b5b13 edgar_igl
 * QEMU ETRAX Timers
3 83fa1010 ths
 *
4 83fa1010 ths
 * Copyright (c) 2007 Edgar E. Iglesias, Axis Communications AB.
5 83fa1010 ths
 *
6 83fa1010 ths
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 83fa1010 ths
 * of this software and associated documentation files (the "Software"), to deal
8 83fa1010 ths
 * in the Software without restriction, including without limitation the rights
9 83fa1010 ths
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 83fa1010 ths
 * copies of the Software, and to permit persons to whom the Software is
11 83fa1010 ths
 * furnished to do so, subject to the following conditions:
12 83fa1010 ths
 *
13 83fa1010 ths
 * The above copyright notice and this permission notice shall be included in
14 83fa1010 ths
 * all copies or substantial portions of the Software.
15 83fa1010 ths
 *
16 83fa1010 ths
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 83fa1010 ths
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 83fa1010 ths
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 83fa1010 ths
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 83fa1010 ths
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 83fa1010 ths
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 83fa1010 ths
 * THE SOFTWARE.
23 83fa1010 ths
 */
24 3b1fd90e Edgar E. Iglesias
#include "sysbus.h"
25 5439779e edgar_igl
#include "sysemu.h"
26 87ecb68b pbrook
#include "qemu-timer.h"
27 83fa1010 ths
28 bbaf29c7 edgar_igl
#define D(x)
29 bbaf29c7 edgar_igl
30 ca87d03b edgar_igl
#define RW_TMR0_DIV   0x00
31 ca87d03b edgar_igl
#define R_TMR0_DATA   0x04
32 ca87d03b edgar_igl
#define RW_TMR0_CTRL  0x08
33 ca87d03b edgar_igl
#define RW_TMR1_DIV   0x10
34 ca87d03b edgar_igl
#define R_TMR1_DATA   0x14
35 ca87d03b edgar_igl
#define RW_TMR1_CTRL  0x18
36 ca87d03b edgar_igl
#define R_TIME        0x38
37 ca87d03b edgar_igl
#define RW_WD_CTRL    0x40
38 5439779e edgar_igl
#define R_WD_STAT     0x44
39 ca87d03b edgar_igl
#define RW_INTR_MASK  0x48
40 ca87d03b edgar_igl
#define RW_ACK_INTR   0x4c
41 ca87d03b edgar_igl
#define R_INTR        0x50
42 ca87d03b edgar_igl
#define R_MASKED_INTR 0x54
43 83fa1010 ths
44 3b1fd90e Edgar E. Iglesias
struct etrax_timer {
45 3b1fd90e Edgar E. Iglesias
    SysBusDevice busdev;
46 b8e5da2c Edgar E. Iglesias
    MemoryRegion mmio;
47 3b1fd90e Edgar E. Iglesias
    qemu_irq irq;
48 3b1fd90e Edgar E. Iglesias
    qemu_irq nmi;
49 84ceea57 Edgar E. Iglesias
50 84ceea57 Edgar E. Iglesias
    QEMUBH *bh_t0;
51 84ceea57 Edgar E. Iglesias
    QEMUBH *bh_t1;
52 84ceea57 Edgar E. Iglesias
    QEMUBH *bh_wd;
53 84ceea57 Edgar E. Iglesias
    ptimer_state *ptimer_t0;
54 84ceea57 Edgar E. Iglesias
    ptimer_state *ptimer_t1;
55 84ceea57 Edgar E. Iglesias
    ptimer_state *ptimer_wd;
56 84ceea57 Edgar E. Iglesias
57 84ceea57 Edgar E. Iglesias
    int wd_hits;
58 84ceea57 Edgar E. Iglesias
59 84ceea57 Edgar E. Iglesias
    /* Control registers.  */
60 84ceea57 Edgar E. Iglesias
    uint32_t rw_tmr0_div;
61 84ceea57 Edgar E. Iglesias
    uint32_t r_tmr0_data;
62 84ceea57 Edgar E. Iglesias
    uint32_t rw_tmr0_ctrl;
63 84ceea57 Edgar E. Iglesias
64 84ceea57 Edgar E. Iglesias
    uint32_t rw_tmr1_div;
65 84ceea57 Edgar E. Iglesias
    uint32_t r_tmr1_data;
66 84ceea57 Edgar E. Iglesias
    uint32_t rw_tmr1_ctrl;
67 84ceea57 Edgar E. Iglesias
68 84ceea57 Edgar E. Iglesias
    uint32_t rw_wd_ctrl;
69 84ceea57 Edgar E. Iglesias
70 84ceea57 Edgar E. Iglesias
    uint32_t rw_intr_mask;
71 84ceea57 Edgar E. Iglesias
    uint32_t rw_ack_intr;
72 84ceea57 Edgar E. Iglesias
    uint32_t r_intr;
73 84ceea57 Edgar E. Iglesias
    uint32_t r_masked_intr;
74 83fa1010 ths
};
75 83fa1010 ths
76 b8e5da2c Edgar E. Iglesias
static uint64_t
77 b8e5da2c Edgar E. Iglesias
timer_read(void *opaque, target_phys_addr_t addr, unsigned int size)
78 83fa1010 ths
{
79 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = opaque;
80 84ceea57 Edgar E. Iglesias
    uint32_t r = 0;
81 84ceea57 Edgar E. Iglesias
82 84ceea57 Edgar E. Iglesias
    switch (addr) {
83 84ceea57 Edgar E. Iglesias
    case R_TMR0_DATA:
84 84ceea57 Edgar E. Iglesias
        r = ptimer_get_count(t->ptimer_t0);
85 84ceea57 Edgar E. Iglesias
        break;
86 84ceea57 Edgar E. Iglesias
    case R_TMR1_DATA:
87 84ceea57 Edgar E. Iglesias
        r = ptimer_get_count(t->ptimer_t1);
88 84ceea57 Edgar E. Iglesias
        break;
89 84ceea57 Edgar E. Iglesias
    case R_TIME:
90 74475455 Paolo Bonzini
        r = qemu_get_clock_ns(vm_clock) / 10;
91 84ceea57 Edgar E. Iglesias
        break;
92 84ceea57 Edgar E. Iglesias
    case RW_INTR_MASK:
93 84ceea57 Edgar E. Iglesias
        r = t->rw_intr_mask;
94 84ceea57 Edgar E. Iglesias
        break;
95 84ceea57 Edgar E. Iglesias
    case R_MASKED_INTR:
96 84ceea57 Edgar E. Iglesias
        r = t->r_intr & t->rw_intr_mask;
97 84ceea57 Edgar E. Iglesias
        break;
98 84ceea57 Edgar E. Iglesias
    default:
99 84ceea57 Edgar E. Iglesias
        D(printf ("%s %x\n", __func__, addr));
100 84ceea57 Edgar E. Iglesias
        break;
101 84ceea57 Edgar E. Iglesias
    }
102 84ceea57 Edgar E. Iglesias
    return r;
103 83fa1010 ths
}
104 83fa1010 ths
105 3b1fd90e Edgar E. Iglesias
static void update_ctrl(struct etrax_timer *t, int tnum)
106 83fa1010 ths
{
107 84ceea57 Edgar E. Iglesias
    unsigned int op;
108 84ceea57 Edgar E. Iglesias
    unsigned int freq;
109 84ceea57 Edgar E. Iglesias
    unsigned int freq_hz;
110 84ceea57 Edgar E. Iglesias
    unsigned int div;
111 84ceea57 Edgar E. Iglesias
    uint32_t ctrl;
112 84ceea57 Edgar E. Iglesias
113 84ceea57 Edgar E. Iglesias
    ptimer_state *timer;
114 84ceea57 Edgar E. Iglesias
115 84ceea57 Edgar E. Iglesias
    if (tnum == 0) {
116 84ceea57 Edgar E. Iglesias
        ctrl = t->rw_tmr0_ctrl;
117 84ceea57 Edgar E. Iglesias
        div = t->rw_tmr0_div;
118 84ceea57 Edgar E. Iglesias
        timer = t->ptimer_t0;
119 84ceea57 Edgar E. Iglesias
    } else {
120 84ceea57 Edgar E. Iglesias
        ctrl = t->rw_tmr1_ctrl;
121 84ceea57 Edgar E. Iglesias
        div = t->rw_tmr1_div;
122 84ceea57 Edgar E. Iglesias
        timer = t->ptimer_t1;
123 84ceea57 Edgar E. Iglesias
    }
124 84ceea57 Edgar E. Iglesias
125 84ceea57 Edgar E. Iglesias
126 84ceea57 Edgar E. Iglesias
    op = ctrl & 3;
127 84ceea57 Edgar E. Iglesias
    freq = ctrl >> 2;
128 84ceea57 Edgar E. Iglesias
    freq_hz = 32000000;
129 84ceea57 Edgar E. Iglesias
130 84ceea57 Edgar E. Iglesias
    switch (freq)
131 84ceea57 Edgar E. Iglesias
    {
132 84ceea57 Edgar E. Iglesias
    case 0:
133 84ceea57 Edgar E. Iglesias
    case 1:
134 84ceea57 Edgar E. Iglesias
        D(printf ("extern or disabled timer clock?\n"));
135 84ceea57 Edgar E. Iglesias
        break;
136 84ceea57 Edgar E. Iglesias
    case 4: freq_hz =  29493000; break;
137 84ceea57 Edgar E. Iglesias
    case 5: freq_hz =  32000000; break;
138 84ceea57 Edgar E. Iglesias
    case 6: freq_hz =  32768000; break;
139 84ceea57 Edgar E. Iglesias
    case 7: freq_hz = 100000000; break;
140 84ceea57 Edgar E. Iglesias
    default:
141 84ceea57 Edgar E. Iglesias
        abort();
142 84ceea57 Edgar E. Iglesias
        break;
143 84ceea57 Edgar E. Iglesias
    }
144 84ceea57 Edgar E. Iglesias
145 84ceea57 Edgar E. Iglesias
    D(printf ("freq_hz=%d div=%d\n", freq_hz, div));
146 84ceea57 Edgar E. Iglesias
    ptimer_set_freq(timer, freq_hz);
147 84ceea57 Edgar E. Iglesias
    ptimer_set_limit(timer, div, 0);
148 84ceea57 Edgar E. Iglesias
149 84ceea57 Edgar E. Iglesias
    switch (op)
150 84ceea57 Edgar E. Iglesias
    {
151 84ceea57 Edgar E. Iglesias
        case 0:
152 84ceea57 Edgar E. Iglesias
            /* Load.  */
153 84ceea57 Edgar E. Iglesias
            ptimer_set_limit(timer, div, 1);
154 84ceea57 Edgar E. Iglesias
            break;
155 84ceea57 Edgar E. Iglesias
        case 1:
156 84ceea57 Edgar E. Iglesias
            /* Hold.  */
157 84ceea57 Edgar E. Iglesias
            ptimer_stop(timer);
158 84ceea57 Edgar E. Iglesias
            break;
159 84ceea57 Edgar E. Iglesias
        case 2:
160 84ceea57 Edgar E. Iglesias
            /* Run.  */
161 84ceea57 Edgar E. Iglesias
            ptimer_run(timer, 0);
162 84ceea57 Edgar E. Iglesias
            break;
163 84ceea57 Edgar E. Iglesias
        default:
164 84ceea57 Edgar E. Iglesias
            abort();
165 84ceea57 Edgar E. Iglesias
            break;
166 84ceea57 Edgar E. Iglesias
    }
167 83fa1010 ths
}
168 83fa1010 ths
169 3b1fd90e Edgar E. Iglesias
static void timer_update_irq(struct etrax_timer *t)
170 83fa1010 ths
{
171 84ceea57 Edgar E. Iglesias
    t->r_intr &= ~(t->rw_ack_intr);
172 84ceea57 Edgar E. Iglesias
    t->r_masked_intr = t->r_intr & t->rw_intr_mask;
173 60237223 edgar_igl
174 84ceea57 Edgar E. Iglesias
    D(printf("%s: masked_intr=%x\n", __func__, t->r_masked_intr));
175 3b1fd90e Edgar E. Iglesias
    qemu_set_irq(t->irq, !!t->r_masked_intr);
176 83fa1010 ths
}
177 83fa1010 ths
178 5439779e edgar_igl
static void timer0_hit(void *opaque)
179 60237223 edgar_igl
{
180 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = opaque;
181 84ceea57 Edgar E. Iglesias
    t->r_intr |= 1;
182 84ceea57 Edgar E. Iglesias
    timer_update_irq(t);
183 60237223 edgar_igl
}
184 60237223 edgar_igl
185 5439779e edgar_igl
static void timer1_hit(void *opaque)
186 5439779e edgar_igl
{
187 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = opaque;
188 84ceea57 Edgar E. Iglesias
    t->r_intr |= 2;
189 84ceea57 Edgar E. Iglesias
    timer_update_irq(t);
190 5439779e edgar_igl
}
191 5439779e edgar_igl
192 5439779e edgar_igl
static void watchdog_hit(void *opaque)
193 5439779e edgar_igl
{
194 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = opaque;
195 84ceea57 Edgar E. Iglesias
    if (t->wd_hits == 0) {
196 84ceea57 Edgar E. Iglesias
        /* real hw gives a single tick before reseting but we are
197 84ceea57 Edgar E. Iglesias
           a bit friendlier to compensate for our slower execution.  */
198 84ceea57 Edgar E. Iglesias
        ptimer_set_count(t->ptimer_wd, 10);
199 84ceea57 Edgar E. Iglesias
        ptimer_run(t->ptimer_wd, 1);
200 3b1fd90e Edgar E. Iglesias
        qemu_irq_raise(t->nmi);
201 84ceea57 Edgar E. Iglesias
    }
202 84ceea57 Edgar E. Iglesias
    else
203 84ceea57 Edgar E. Iglesias
        qemu_system_reset_request();
204 84ceea57 Edgar E. Iglesias
205 84ceea57 Edgar E. Iglesias
    t->wd_hits++;
206 5439779e edgar_igl
}
207 5439779e edgar_igl
208 3b1fd90e Edgar E. Iglesias
static inline void timer_watchdog_update(struct etrax_timer *t, uint32_t value)
209 5439779e edgar_igl
{
210 84ceea57 Edgar E. Iglesias
    unsigned int wd_en = t->rw_wd_ctrl & (1 << 8);
211 84ceea57 Edgar E. Iglesias
    unsigned int wd_key = t->rw_wd_ctrl >> 9;
212 84ceea57 Edgar E. Iglesias
    unsigned int wd_cnt = t->rw_wd_ctrl & 511;
213 84ceea57 Edgar E. Iglesias
    unsigned int new_key = value >> 9 & ((1 << 7) - 1);
214 84ceea57 Edgar E. Iglesias
    unsigned int new_cmd = (value >> 8) & 1;
215 5439779e edgar_igl
216 84ceea57 Edgar E. Iglesias
    /* If the watchdog is enabled, they written key must match the
217 84ceea57 Edgar E. Iglesias
       complement of the previous.  */
218 84ceea57 Edgar E. Iglesias
    wd_key = ~wd_key & ((1 << 7) - 1);
219 5439779e edgar_igl
220 84ceea57 Edgar E. Iglesias
    if (wd_en && wd_key != new_key)
221 84ceea57 Edgar E. Iglesias
        return;
222 5439779e edgar_igl
223 84ceea57 Edgar E. Iglesias
    D(printf("en=%d new_key=%x oldkey=%x cmd=%d cnt=%d\n", 
224 84ceea57 Edgar E. Iglesias
         wd_en, new_key, wd_key, new_cmd, wd_cnt));
225 5439779e edgar_igl
226 84ceea57 Edgar E. Iglesias
    if (t->wd_hits)
227 3b1fd90e Edgar E. Iglesias
        qemu_irq_lower(t->nmi);
228 5ef98b47 edgar_igl
229 84ceea57 Edgar E. Iglesias
    t->wd_hits = 0;
230 5ef98b47 edgar_igl
231 84ceea57 Edgar E. Iglesias
    ptimer_set_freq(t->ptimer_wd, 760);
232 84ceea57 Edgar E. Iglesias
    if (wd_cnt == 0)
233 84ceea57 Edgar E. Iglesias
        wd_cnt = 256;
234 84ceea57 Edgar E. Iglesias
    ptimer_set_count(t->ptimer_wd, wd_cnt);
235 84ceea57 Edgar E. Iglesias
    if (new_cmd)
236 84ceea57 Edgar E. Iglesias
        ptimer_run(t->ptimer_wd, 1);
237 84ceea57 Edgar E. Iglesias
    else
238 84ceea57 Edgar E. Iglesias
        ptimer_stop(t->ptimer_wd);
239 5439779e edgar_igl
240 84ceea57 Edgar E. Iglesias
    t->rw_wd_ctrl = value;
241 5439779e edgar_igl
}
242 5439779e edgar_igl
243 83fa1010 ths
static void
244 b8e5da2c Edgar E. Iglesias
timer_write(void *opaque, target_phys_addr_t addr,
245 b8e5da2c Edgar E. Iglesias
            uint64_t val64, unsigned int size)
246 83fa1010 ths
{
247 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = opaque;
248 b8e5da2c Edgar E. Iglesias
    uint32_t value = val64;
249 84ceea57 Edgar E. Iglesias
250 84ceea57 Edgar E. Iglesias
    switch (addr)
251 84ceea57 Edgar E. Iglesias
    {
252 84ceea57 Edgar E. Iglesias
        case RW_TMR0_DIV:
253 84ceea57 Edgar E. Iglesias
            t->rw_tmr0_div = value;
254 84ceea57 Edgar E. Iglesias
            break;
255 84ceea57 Edgar E. Iglesias
        case RW_TMR0_CTRL:
256 84ceea57 Edgar E. Iglesias
            D(printf ("RW_TMR0_CTRL=%x\n", value));
257 84ceea57 Edgar E. Iglesias
            t->rw_tmr0_ctrl = value;
258 84ceea57 Edgar E. Iglesias
            update_ctrl(t, 0);
259 84ceea57 Edgar E. Iglesias
            break;
260 84ceea57 Edgar E. Iglesias
        case RW_TMR1_DIV:
261 84ceea57 Edgar E. Iglesias
            t->rw_tmr1_div = value;
262 84ceea57 Edgar E. Iglesias
            break;
263 84ceea57 Edgar E. Iglesias
        case RW_TMR1_CTRL:
264 84ceea57 Edgar E. Iglesias
            D(printf ("RW_TMR1_CTRL=%x\n", value));
265 84ceea57 Edgar E. Iglesias
            t->rw_tmr1_ctrl = value;
266 84ceea57 Edgar E. Iglesias
            update_ctrl(t, 1);
267 84ceea57 Edgar E. Iglesias
            break;
268 84ceea57 Edgar E. Iglesias
        case RW_INTR_MASK:
269 84ceea57 Edgar E. Iglesias
            D(printf ("RW_INTR_MASK=%x\n", value));
270 84ceea57 Edgar E. Iglesias
            t->rw_intr_mask = value;
271 84ceea57 Edgar E. Iglesias
            timer_update_irq(t);
272 84ceea57 Edgar E. Iglesias
            break;
273 84ceea57 Edgar E. Iglesias
        case RW_WD_CTRL:
274 84ceea57 Edgar E. Iglesias
            timer_watchdog_update(t, value);
275 84ceea57 Edgar E. Iglesias
            break;
276 84ceea57 Edgar E. Iglesias
        case RW_ACK_INTR:
277 84ceea57 Edgar E. Iglesias
            t->rw_ack_intr = value;
278 84ceea57 Edgar E. Iglesias
            timer_update_irq(t);
279 84ceea57 Edgar E. Iglesias
            t->rw_ack_intr = 0;
280 84ceea57 Edgar E. Iglesias
            break;
281 84ceea57 Edgar E. Iglesias
        default:
282 84ceea57 Edgar E. Iglesias
            printf ("%s " TARGET_FMT_plx " %x\n",
283 84ceea57 Edgar E. Iglesias
                __func__, addr, value);
284 84ceea57 Edgar E. Iglesias
            break;
285 84ceea57 Edgar E. Iglesias
    }
286 83fa1010 ths
}
287 83fa1010 ths
288 b8e5da2c Edgar E. Iglesias
static const MemoryRegionOps timer_ops = {
289 b8e5da2c Edgar E. Iglesias
    .read = timer_read,
290 b8e5da2c Edgar E. Iglesias
    .write = timer_write,
291 b8e5da2c Edgar E. Iglesias
    .endianness = DEVICE_LITTLE_ENDIAN,
292 b8e5da2c Edgar E. Iglesias
    .valid = {
293 b8e5da2c Edgar E. Iglesias
        .min_access_size = 4,
294 b8e5da2c Edgar E. Iglesias
        .max_access_size = 4
295 b8e5da2c Edgar E. Iglesias
    }
296 83fa1010 ths
};
297 83fa1010 ths
298 5439779e edgar_igl
static void etraxfs_timer_reset(void *opaque)
299 5439779e edgar_igl
{
300 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = opaque;
301 84ceea57 Edgar E. Iglesias
302 84ceea57 Edgar E. Iglesias
    ptimer_stop(t->ptimer_t0);
303 84ceea57 Edgar E. Iglesias
    ptimer_stop(t->ptimer_t1);
304 84ceea57 Edgar E. Iglesias
    ptimer_stop(t->ptimer_wd);
305 84ceea57 Edgar E. Iglesias
    t->rw_wd_ctrl = 0;
306 84ceea57 Edgar E. Iglesias
    t->r_intr = 0;
307 84ceea57 Edgar E. Iglesias
    t->rw_intr_mask = 0;
308 3b1fd90e Edgar E. Iglesias
    qemu_irq_lower(t->irq);
309 5439779e edgar_igl
}
310 5439779e edgar_igl
311 81a322d4 Gerd Hoffmann
static int etraxfs_timer_init(SysBusDevice *dev)
312 83fa1010 ths
{
313 3b1fd90e Edgar E. Iglesias
    struct etrax_timer *t = FROM_SYSBUS(typeof (*t), dev);
314 83fa1010 ths
315 84ceea57 Edgar E. Iglesias
    t->bh_t0 = qemu_bh_new(timer0_hit, t);
316 84ceea57 Edgar E. Iglesias
    t->bh_t1 = qemu_bh_new(timer1_hit, t);
317 84ceea57 Edgar E. Iglesias
    t->bh_wd = qemu_bh_new(watchdog_hit, t);
318 84ceea57 Edgar E. Iglesias
    t->ptimer_t0 = ptimer_init(t->bh_t0);
319 84ceea57 Edgar E. Iglesias
    t->ptimer_t1 = ptimer_init(t->bh_t1);
320 84ceea57 Edgar E. Iglesias
    t->ptimer_wd = ptimer_init(t->bh_wd);
321 3b1fd90e Edgar E. Iglesias
322 3b1fd90e Edgar E. Iglesias
    sysbus_init_irq(dev, &t->irq);
323 3b1fd90e Edgar E. Iglesias
    sysbus_init_irq(dev, &t->nmi);
324 83fa1010 ths
325 b8e5da2c Edgar E. Iglesias
    memory_region_init_io(&t->mmio, &timer_ops, t, "etraxfs-timer", 0x5c);
326 b8e5da2c Edgar E. Iglesias
    sysbus_init_mmio_region(dev, &t->mmio);
327 a08d4367 Jan Kiszka
    qemu_register_reset(etraxfs_timer_reset, t);
328 81a322d4 Gerd Hoffmann
    return 0;
329 83fa1010 ths
}
330 3b1fd90e Edgar E. Iglesias
331 3b1fd90e Edgar E. Iglesias
static void etraxfs_timer_register(void)
332 3b1fd90e Edgar E. Iglesias
{
333 3b1fd90e Edgar E. Iglesias
    sysbus_register_dev("etraxfs,timer", sizeof (struct etrax_timer),
334 3b1fd90e Edgar E. Iglesias
                        etraxfs_timer_init);
335 3b1fd90e Edgar E. Iglesias
}
336 3b1fd90e Edgar E. Iglesias
337 3b1fd90e Edgar E. Iglesias
device_init(etraxfs_timer_register)