Statistics
| Branch: | Revision:

root / hw / arm_sysctl.c @ 291155cb

History | View | Annotate | Download (16.7 kB)

1 5fafdf24 ths
/*
2 e69954b9 pbrook
 * Status and system control registers for ARM RealView/Versatile boards.
3 e69954b9 pbrook
 *
4 9ee6e8bb pbrook
 * Copyright (c) 2006-2007 CodeSourcery.
5 e69954b9 pbrook
 * Written by Paul Brook
6 e69954b9 pbrook
 *
7 8e31bf38 Matthew Fernandez
 * This code is licensed under the GPL.
8 e69954b9 pbrook
 */
9 e69954b9 pbrook
10 83c9f4ca Paolo Bonzini
#include "hw/hw.h"
11 1de7afc9 Paolo Bonzini
#include "qemu/timer.h"
12 71538323 Peter Maydell
#include "qemu/bitops.h"
13 83c9f4ca Paolo Bonzini
#include "hw/sysbus.h"
14 83c9f4ca Paolo Bonzini
#include "hw/primecell.h"
15 9c17d615 Paolo Bonzini
#include "sysemu/sysemu.h"
16 e69954b9 pbrook
17 e69954b9 pbrook
#define LOCK_VALUE 0xa05f
18 e69954b9 pbrook
19 e69954b9 pbrook
typedef struct {
20 82634c2d Paul Brook
    SysBusDevice busdev;
21 460d7c53 Avi Kivity
    MemoryRegion iomem;
22 242ea2c6 Peter Maydell
    qemu_irq pl110_mux_ctrl;
23 242ea2c6 Peter Maydell
24 e69954b9 pbrook
    uint32_t sys_id;
25 e69954b9 pbrook
    uint32_t leds;
26 e69954b9 pbrook
    uint16_t lockval;
27 e69954b9 pbrook
    uint32_t cfgdata1;
28 e69954b9 pbrook
    uint32_t cfgdata2;
29 e69954b9 pbrook
    uint32_t flags;
30 e69954b9 pbrook
    uint32_t nvflags;
31 e69954b9 pbrook
    uint32_t resetlevel;
32 26e92f65 Paul Brook
    uint32_t proc_id;
33 b50ff6f5 Peter Maydell
    uint32_t sys_mci;
34 34933c8c Peter Maydell
    uint32_t sys_cfgdata;
35 34933c8c Peter Maydell
    uint32_t sys_cfgctrl;
36 34933c8c Peter Maydell
    uint32_t sys_cfgstat;
37 242ea2c6 Peter Maydell
    uint32_t sys_clcd;
38 e69954b9 pbrook
} arm_sysctl_state;
39 e69954b9 pbrook
40 b5ad0ae7 Peter Maydell
static const VMStateDescription vmstate_arm_sysctl = {
41 b5ad0ae7 Peter Maydell
    .name = "realview_sysctl",
42 242ea2c6 Peter Maydell
    .version_id = 3,
43 b5ad0ae7 Peter Maydell
    .minimum_version_id = 1,
44 b5ad0ae7 Peter Maydell
    .fields = (VMStateField[]) {
45 b5ad0ae7 Peter Maydell
        VMSTATE_UINT32(leds, arm_sysctl_state),
46 b5ad0ae7 Peter Maydell
        VMSTATE_UINT16(lockval, arm_sysctl_state),
47 b5ad0ae7 Peter Maydell
        VMSTATE_UINT32(cfgdata1, arm_sysctl_state),
48 b5ad0ae7 Peter Maydell
        VMSTATE_UINT32(cfgdata2, arm_sysctl_state),
49 b5ad0ae7 Peter Maydell
        VMSTATE_UINT32(flags, arm_sysctl_state),
50 b5ad0ae7 Peter Maydell
        VMSTATE_UINT32(nvflags, arm_sysctl_state),
51 b5ad0ae7 Peter Maydell
        VMSTATE_UINT32(resetlevel, arm_sysctl_state),
52 34933c8c Peter Maydell
        VMSTATE_UINT32_V(sys_mci, arm_sysctl_state, 2),
53 34933c8c Peter Maydell
        VMSTATE_UINT32_V(sys_cfgdata, arm_sysctl_state, 2),
54 34933c8c Peter Maydell
        VMSTATE_UINT32_V(sys_cfgctrl, arm_sysctl_state, 2),
55 34933c8c Peter Maydell
        VMSTATE_UINT32_V(sys_cfgstat, arm_sysctl_state, 2),
56 242ea2c6 Peter Maydell
        VMSTATE_UINT32_V(sys_clcd, arm_sysctl_state, 3),
57 b5ad0ae7 Peter Maydell
        VMSTATE_END_OF_LIST()
58 b5ad0ae7 Peter Maydell
    }
59 b5ad0ae7 Peter Maydell
};
60 b5ad0ae7 Peter Maydell
61 b50ff6f5 Peter Maydell
/* The PB926 actually uses a different format for
62 b50ff6f5 Peter Maydell
 * its SYS_ID register. Fortunately the bits which are
63 b50ff6f5 Peter Maydell
 * board type on later boards are distinct.
64 b50ff6f5 Peter Maydell
 */
65 b50ff6f5 Peter Maydell
#define BOARD_ID_PB926 0x100
66 b50ff6f5 Peter Maydell
#define BOARD_ID_EB 0x140
67 b50ff6f5 Peter Maydell
#define BOARD_ID_PBA8 0x178
68 b50ff6f5 Peter Maydell
#define BOARD_ID_PBX 0x182
69 34933c8c Peter Maydell
#define BOARD_ID_VEXPRESS 0x190
70 b50ff6f5 Peter Maydell
71 b50ff6f5 Peter Maydell
static int board_id(arm_sysctl_state *s)
72 b50ff6f5 Peter Maydell
{
73 b50ff6f5 Peter Maydell
    /* Extract the board ID field from the SYS_ID register value */
74 b50ff6f5 Peter Maydell
    return (s->sys_id >> 16) & 0xfff;
75 b50ff6f5 Peter Maydell
}
76 b50ff6f5 Peter Maydell
77 be0f204a Paul Brook
static void arm_sysctl_reset(DeviceState *d)
78 be0f204a Paul Brook
{
79 1356b98d Andreas Färber
    arm_sysctl_state *s = FROM_SYSBUS(arm_sysctl_state, SYS_BUS_DEVICE(d));
80 be0f204a Paul Brook
81 be0f204a Paul Brook
    s->leds = 0;
82 be0f204a Paul Brook
    s->lockval = 0;
83 be0f204a Paul Brook
    s->cfgdata1 = 0;
84 be0f204a Paul Brook
    s->cfgdata2 = 0;
85 be0f204a Paul Brook
    s->flags = 0;
86 be0f204a Paul Brook
    s->resetlevel = 0;
87 242ea2c6 Peter Maydell
    if (board_id(s) == BOARD_ID_VEXPRESS) {
88 242ea2c6 Peter Maydell
        /* On VExpress this register will RAZ/WI */
89 242ea2c6 Peter Maydell
        s->sys_clcd = 0;
90 242ea2c6 Peter Maydell
    } else {
91 242ea2c6 Peter Maydell
        /* All others: CLCDID 0x1f, indicating VGA */
92 242ea2c6 Peter Maydell
        s->sys_clcd = 0x1f00;
93 242ea2c6 Peter Maydell
    }
94 be0f204a Paul Brook
}
95 be0f204a Paul Brook
96 a8170e5e Avi Kivity
static uint64_t arm_sysctl_read(void *opaque, hwaddr offset,
97 460d7c53 Avi Kivity
                                unsigned size)
98 e69954b9 pbrook
{
99 e69954b9 pbrook
    arm_sysctl_state *s = (arm_sysctl_state *)opaque;
100 e69954b9 pbrook
101 e69954b9 pbrook
    switch (offset) {
102 e69954b9 pbrook
    case 0x00: /* ID */
103 e69954b9 pbrook
        return s->sys_id;
104 e69954b9 pbrook
    case 0x04: /* SW */
105 e69954b9 pbrook
        /* General purpose hardware switches.
106 e69954b9 pbrook
           We don't have a useful way of exposing these to the user.  */
107 e69954b9 pbrook
        return 0;
108 e69954b9 pbrook
    case 0x08: /* LED */
109 e69954b9 pbrook
        return s->leds;
110 e69954b9 pbrook
    case 0x20: /* LOCK */
111 e69954b9 pbrook
        return s->lockval;
112 e69954b9 pbrook
    case 0x0c: /* OSC0 */
113 e69954b9 pbrook
    case 0x10: /* OSC1 */
114 e69954b9 pbrook
    case 0x14: /* OSC2 */
115 e69954b9 pbrook
    case 0x18: /* OSC3 */
116 e69954b9 pbrook
    case 0x1c: /* OSC4 */
117 e69954b9 pbrook
    case 0x24: /* 100HZ */
118 e69954b9 pbrook
        /* ??? Implement these.  */
119 e69954b9 pbrook
        return 0;
120 e69954b9 pbrook
    case 0x28: /* CFGDATA1 */
121 e69954b9 pbrook
        return s->cfgdata1;
122 e69954b9 pbrook
    case 0x2c: /* CFGDATA2 */
123 e69954b9 pbrook
        return s->cfgdata2;
124 e69954b9 pbrook
    case 0x30: /* FLAGS */
125 e69954b9 pbrook
        return s->flags;
126 e69954b9 pbrook
    case 0x38: /* NVFLAGS */
127 e69954b9 pbrook
        return s->nvflags;
128 e69954b9 pbrook
    case 0x40: /* RESETCTL */
129 34933c8c Peter Maydell
        if (board_id(s) == BOARD_ID_VEXPRESS) {
130 34933c8c Peter Maydell
            /* reserved: RAZ/WI */
131 34933c8c Peter Maydell
            return 0;
132 34933c8c Peter Maydell
        }
133 e69954b9 pbrook
        return s->resetlevel;
134 e69954b9 pbrook
    case 0x44: /* PCICTL */
135 e69954b9 pbrook
        return 1;
136 e69954b9 pbrook
    case 0x48: /* MCI */
137 b50ff6f5 Peter Maydell
        return s->sys_mci;
138 e69954b9 pbrook
    case 0x4c: /* FLASH */
139 e69954b9 pbrook
        return 0;
140 e69954b9 pbrook
    case 0x50: /* CLCD */
141 242ea2c6 Peter Maydell
        return s->sys_clcd;
142 e69954b9 pbrook
    case 0x54: /* CLCDSER */
143 e69954b9 pbrook
        return 0;
144 e69954b9 pbrook
    case 0x58: /* BOOTCS */
145 e69954b9 pbrook
        return 0;
146 e69954b9 pbrook
    case 0x5c: /* 24MHz */
147 74475455 Paolo Bonzini
        return muldiv64(qemu_get_clock_ns(vm_clock), 24000000, get_ticks_per_sec());
148 e69954b9 pbrook
    case 0x60: /* MISC */
149 e69954b9 pbrook
        return 0;
150 e69954b9 pbrook
    case 0x84: /* PROCID0 */
151 26e92f65 Paul Brook
        return s->proc_id;
152 e69954b9 pbrook
    case 0x88: /* PROCID1 */
153 e69954b9 pbrook
        return 0xff000000;
154 e69954b9 pbrook
    case 0x64: /* DMAPSR0 */
155 e69954b9 pbrook
    case 0x68: /* DMAPSR1 */
156 e69954b9 pbrook
    case 0x6c: /* DMAPSR2 */
157 e69954b9 pbrook
    case 0x70: /* IOSEL */
158 e69954b9 pbrook
    case 0x74: /* PLDCTL */
159 e69954b9 pbrook
    case 0x80: /* BUSID */
160 e69954b9 pbrook
    case 0x8c: /* OSCRESET0 */
161 e69954b9 pbrook
    case 0x90: /* OSCRESET1 */
162 e69954b9 pbrook
    case 0x94: /* OSCRESET2 */
163 e69954b9 pbrook
    case 0x98: /* OSCRESET3 */
164 e69954b9 pbrook
    case 0x9c: /* OSCRESET4 */
165 e69954b9 pbrook
    case 0xc0: /* SYS_TEST_OSC0 */
166 e69954b9 pbrook
    case 0xc4: /* SYS_TEST_OSC1 */
167 e69954b9 pbrook
    case 0xc8: /* SYS_TEST_OSC2 */
168 e69954b9 pbrook
    case 0xcc: /* SYS_TEST_OSC3 */
169 e69954b9 pbrook
    case 0xd0: /* SYS_TEST_OSC4 */
170 e69954b9 pbrook
        return 0;
171 34933c8c Peter Maydell
    case 0xa0: /* SYS_CFGDATA */
172 34933c8c Peter Maydell
        if (board_id(s) != BOARD_ID_VEXPRESS) {
173 34933c8c Peter Maydell
            goto bad_reg;
174 34933c8c Peter Maydell
        }
175 34933c8c Peter Maydell
        return s->sys_cfgdata;
176 34933c8c Peter Maydell
    case 0xa4: /* SYS_CFGCTRL */
177 34933c8c Peter Maydell
        if (board_id(s) != BOARD_ID_VEXPRESS) {
178 34933c8c Peter Maydell
            goto bad_reg;
179 34933c8c Peter Maydell
        }
180 34933c8c Peter Maydell
        return s->sys_cfgctrl;
181 34933c8c Peter Maydell
    case 0xa8: /* SYS_CFGSTAT */
182 34933c8c Peter Maydell
        if (board_id(s) != BOARD_ID_VEXPRESS) {
183 34933c8c Peter Maydell
            goto bad_reg;
184 34933c8c Peter Maydell
        }
185 34933c8c Peter Maydell
        return s->sys_cfgstat;
186 e69954b9 pbrook
    default:
187 34933c8c Peter Maydell
    bad_reg:
188 0c896f06 Peter Maydell
        qemu_log_mask(LOG_GUEST_ERROR,
189 0c896f06 Peter Maydell
                      "arm_sysctl_read: Bad register offset 0x%x\n",
190 0c896f06 Peter Maydell
                      (int)offset);
191 e69954b9 pbrook
        return 0;
192 e69954b9 pbrook
    }
193 e69954b9 pbrook
}
194 e69954b9 pbrook
195 71538323 Peter Maydell
/* SYS_CFGCTRL functions */
196 71538323 Peter Maydell
#define SYS_CFG_OSC 1
197 71538323 Peter Maydell
#define SYS_CFG_VOLT 2
198 71538323 Peter Maydell
#define SYS_CFG_AMP 3
199 71538323 Peter Maydell
#define SYS_CFG_TEMP 4
200 71538323 Peter Maydell
#define SYS_CFG_RESET 5
201 71538323 Peter Maydell
#define SYS_CFG_SCC 6
202 71538323 Peter Maydell
#define SYS_CFG_MUXFPGA 7
203 71538323 Peter Maydell
#define SYS_CFG_SHUTDOWN 8
204 71538323 Peter Maydell
#define SYS_CFG_REBOOT 9
205 71538323 Peter Maydell
#define SYS_CFG_DVIMODE 11
206 71538323 Peter Maydell
#define SYS_CFG_POWER 12
207 71538323 Peter Maydell
#define SYS_CFG_ENERGY 13
208 71538323 Peter Maydell
209 71538323 Peter Maydell
/* SYS_CFGCTRL site field values */
210 71538323 Peter Maydell
#define SYS_CFG_SITE_MB 0
211 71538323 Peter Maydell
#define SYS_CFG_SITE_DB1 1
212 71538323 Peter Maydell
#define SYS_CFG_SITE_DB2 2
213 71538323 Peter Maydell
214 71538323 Peter Maydell
/**
215 71538323 Peter Maydell
 * vexpress_cfgctrl_read:
216 71538323 Peter Maydell
 * @s: arm_sysctl_state pointer
217 71538323 Peter Maydell
 * @dcc, @function, @site, @position, @device: split out values from
218 71538323 Peter Maydell
 * SYS_CFGCTRL register
219 71538323 Peter Maydell
 * @val: pointer to where to put the read data on success
220 71538323 Peter Maydell
 *
221 71538323 Peter Maydell
 * Handle a VExpress SYS_CFGCTRL register read. On success, return true and
222 71538323 Peter Maydell
 * write the read value to *val. On failure, return false (and val may
223 71538323 Peter Maydell
 * or may not be written to).
224 71538323 Peter Maydell
 */
225 71538323 Peter Maydell
static bool vexpress_cfgctrl_read(arm_sysctl_state *s, unsigned int dcc,
226 71538323 Peter Maydell
                                  unsigned int function, unsigned int site,
227 71538323 Peter Maydell
                                  unsigned int position, unsigned int device,
228 71538323 Peter Maydell
                                  uint32_t *val)
229 71538323 Peter Maydell
{
230 71538323 Peter Maydell
    /* We don't support anything other than DCC 0, board stack position 0
231 71538323 Peter Maydell
     * or sites other than motherboard/daughterboard:
232 71538323 Peter Maydell
     */
233 71538323 Peter Maydell
    if (dcc != 0 || position != 0 ||
234 71538323 Peter Maydell
        (site != SYS_CFG_SITE_MB && site != SYS_CFG_SITE_DB1)) {
235 71538323 Peter Maydell
        goto cfgctrl_unimp;
236 71538323 Peter Maydell
    }
237 71538323 Peter Maydell
238 71538323 Peter Maydell
    switch (function) {
239 71538323 Peter Maydell
    default:
240 71538323 Peter Maydell
        break;
241 71538323 Peter Maydell
    }
242 71538323 Peter Maydell
243 71538323 Peter Maydell
cfgctrl_unimp:
244 71538323 Peter Maydell
    qemu_log_mask(LOG_UNIMP,
245 71538323 Peter Maydell
                  "arm_sysctl: Unimplemented SYS_CFGCTRL read of function "
246 71538323 Peter Maydell
                  "0x%x DCC 0x%x site 0x%x position 0x%x device 0x%x\n",
247 71538323 Peter Maydell
                  function, dcc, site, position, device);
248 71538323 Peter Maydell
    return false;
249 71538323 Peter Maydell
}
250 71538323 Peter Maydell
251 71538323 Peter Maydell
/**
252 71538323 Peter Maydell
 * vexpress_cfgctrl_write:
253 71538323 Peter Maydell
 * @s: arm_sysctl_state pointer
254 71538323 Peter Maydell
 * @dcc, @function, @site, @position, @device: split out values from
255 71538323 Peter Maydell
 * SYS_CFGCTRL register
256 71538323 Peter Maydell
 * @val: data to write
257 71538323 Peter Maydell
 *
258 71538323 Peter Maydell
 * Handle a VExpress SYS_CFGCTRL register write. On success, return true.
259 71538323 Peter Maydell
 * On failure, return false.
260 71538323 Peter Maydell
 */
261 71538323 Peter Maydell
static bool vexpress_cfgctrl_write(arm_sysctl_state *s, unsigned int dcc,
262 71538323 Peter Maydell
                                   unsigned int function, unsigned int site,
263 71538323 Peter Maydell
                                   unsigned int position, unsigned int device,
264 71538323 Peter Maydell
                                   uint32_t val)
265 71538323 Peter Maydell
{
266 71538323 Peter Maydell
    /* We don't support anything other than DCC 0, board stack position 0
267 71538323 Peter Maydell
     * or sites other than motherboard/daughterboard:
268 71538323 Peter Maydell
     */
269 71538323 Peter Maydell
    if (dcc != 0 || position != 0 ||
270 71538323 Peter Maydell
        (site != SYS_CFG_SITE_MB && site != SYS_CFG_SITE_DB1)) {
271 71538323 Peter Maydell
        goto cfgctrl_unimp;
272 71538323 Peter Maydell
    }
273 71538323 Peter Maydell
274 71538323 Peter Maydell
    switch (function) {
275 8ff05c98 Peter Maydell
    case SYS_CFG_MUXFPGA:
276 8ff05c98 Peter Maydell
        if (site == SYS_CFG_SITE_MB && device == 0) {
277 8ff05c98 Peter Maydell
            /* Select whether video output comes from motherboard
278 8ff05c98 Peter Maydell
             * or daughterboard: log and ignore as QEMU doesn't
279 8ff05c98 Peter Maydell
             * support this.
280 8ff05c98 Peter Maydell
             */
281 8ff05c98 Peter Maydell
            qemu_log_mask(LOG_UNIMP, "arm_sysctl: selection of video output "
282 8ff05c98 Peter Maydell
                          "not supported, ignoring\n");
283 8ff05c98 Peter Maydell
            return true;
284 8ff05c98 Peter Maydell
        }
285 8ff05c98 Peter Maydell
        break;
286 71538323 Peter Maydell
    case SYS_CFG_SHUTDOWN:
287 71538323 Peter Maydell
        if (site == SYS_CFG_SITE_MB && device == 0) {
288 71538323 Peter Maydell
            qemu_system_shutdown_request();
289 71538323 Peter Maydell
            return true;
290 71538323 Peter Maydell
        }
291 71538323 Peter Maydell
        break;
292 71538323 Peter Maydell
    case SYS_CFG_REBOOT:
293 71538323 Peter Maydell
        if (site == SYS_CFG_SITE_MB && device == 0) {
294 71538323 Peter Maydell
            qemu_system_reset_request();
295 71538323 Peter Maydell
            return true;
296 71538323 Peter Maydell
        }
297 71538323 Peter Maydell
        break;
298 291155cb Peter Maydell
    case SYS_CFG_DVIMODE:
299 291155cb Peter Maydell
        if (site == SYS_CFG_SITE_MB && device == 0) {
300 291155cb Peter Maydell
            /* Selecting DVI mode is meaningless for QEMU: we will
301 291155cb Peter Maydell
             * always display the output correctly according to the
302 291155cb Peter Maydell
             * pixel height/width programmed into the CLCD controller.
303 291155cb Peter Maydell
             */
304 291155cb Peter Maydell
            return true;
305 291155cb Peter Maydell
        }
306 71538323 Peter Maydell
    default:
307 71538323 Peter Maydell
        break;
308 71538323 Peter Maydell
    }
309 71538323 Peter Maydell
310 71538323 Peter Maydell
cfgctrl_unimp:
311 71538323 Peter Maydell
    qemu_log_mask(LOG_UNIMP,
312 71538323 Peter Maydell
                  "arm_sysctl: Unimplemented SYS_CFGCTRL write of function "
313 71538323 Peter Maydell
                  "0x%x DCC 0x%x site 0x%x position 0x%x device 0x%x\n",
314 71538323 Peter Maydell
                  function, dcc, site, position, device);
315 71538323 Peter Maydell
    return false;
316 71538323 Peter Maydell
}
317 71538323 Peter Maydell
318 a8170e5e Avi Kivity
static void arm_sysctl_write(void *opaque, hwaddr offset,
319 460d7c53 Avi Kivity
                             uint64_t val, unsigned size)
320 e69954b9 pbrook
{
321 e69954b9 pbrook
    arm_sysctl_state *s = (arm_sysctl_state *)opaque;
322 e69954b9 pbrook
323 e69954b9 pbrook
    switch (offset) {
324 e69954b9 pbrook
    case 0x08: /* LED */
325 e69954b9 pbrook
        s->leds = val;
326 bf4229d3 Peter Maydell
        break;
327 e69954b9 pbrook
    case 0x0c: /* OSC0 */
328 e69954b9 pbrook
    case 0x10: /* OSC1 */
329 e69954b9 pbrook
    case 0x14: /* OSC2 */
330 e69954b9 pbrook
    case 0x18: /* OSC3 */
331 e69954b9 pbrook
    case 0x1c: /* OSC4 */
332 e69954b9 pbrook
        /* ??? */
333 e69954b9 pbrook
        break;
334 e69954b9 pbrook
    case 0x20: /* LOCK */
335 e69954b9 pbrook
        if (val == LOCK_VALUE)
336 e69954b9 pbrook
            s->lockval = val;
337 e69954b9 pbrook
        else
338 e69954b9 pbrook
            s->lockval = val & 0x7fff;
339 e69954b9 pbrook
        break;
340 e69954b9 pbrook
    case 0x28: /* CFGDATA1 */
341 e69954b9 pbrook
        /* ??? Need to implement this.  */
342 e69954b9 pbrook
        s->cfgdata1 = val;
343 e69954b9 pbrook
        break;
344 e69954b9 pbrook
    case 0x2c: /* CFGDATA2 */
345 e69954b9 pbrook
        /* ??? Need to implement this.  */
346 e69954b9 pbrook
        s->cfgdata2 = val;
347 e69954b9 pbrook
        break;
348 e69954b9 pbrook
    case 0x30: /* FLAGSSET */
349 e69954b9 pbrook
        s->flags |= val;
350 e69954b9 pbrook
        break;
351 e69954b9 pbrook
    case 0x34: /* FLAGSCLR */
352 e69954b9 pbrook
        s->flags &= ~val;
353 e69954b9 pbrook
        break;
354 e69954b9 pbrook
    case 0x38: /* NVFLAGSSET */
355 e69954b9 pbrook
        s->nvflags |= val;
356 e69954b9 pbrook
        break;
357 e69954b9 pbrook
    case 0x3c: /* NVFLAGSCLR */
358 e69954b9 pbrook
        s->nvflags &= ~val;
359 e69954b9 pbrook
        break;
360 e69954b9 pbrook
    case 0x40: /* RESETCTL */
361 b2887c43 Jean-Christophe DUBOIS
        switch (board_id(s)) {
362 b2887c43 Jean-Christophe DUBOIS
        case BOARD_ID_PB926:
363 b2887c43 Jean-Christophe DUBOIS
            if (s->lockval == LOCK_VALUE) {
364 b2887c43 Jean-Christophe DUBOIS
                s->resetlevel = val;
365 b2887c43 Jean-Christophe DUBOIS
                if (val & 0x100) {
366 b2887c43 Jean-Christophe DUBOIS
                    qemu_system_reset_request();
367 b2887c43 Jean-Christophe DUBOIS
                }
368 b2887c43 Jean-Christophe DUBOIS
            }
369 b2887c43 Jean-Christophe DUBOIS
            break;
370 b2887c43 Jean-Christophe DUBOIS
        case BOARD_ID_PBX:
371 b2887c43 Jean-Christophe DUBOIS
        case BOARD_ID_PBA8:
372 b2887c43 Jean-Christophe DUBOIS
            if (s->lockval == LOCK_VALUE) {
373 b2887c43 Jean-Christophe DUBOIS
                s->resetlevel = val;
374 b2887c43 Jean-Christophe DUBOIS
                if (val & 0x04) {
375 b2887c43 Jean-Christophe DUBOIS
                    qemu_system_reset_request();
376 b2887c43 Jean-Christophe DUBOIS
                }
377 b2887c43 Jean-Christophe DUBOIS
            }
378 b2887c43 Jean-Christophe DUBOIS
            break;
379 b2887c43 Jean-Christophe DUBOIS
        case BOARD_ID_VEXPRESS:
380 b2887c43 Jean-Christophe DUBOIS
        case BOARD_ID_EB:
381 b2887c43 Jean-Christophe DUBOIS
        default:
382 34933c8c Peter Maydell
            /* reserved: RAZ/WI */
383 34933c8c Peter Maydell
            break;
384 34933c8c Peter Maydell
        }
385 e69954b9 pbrook
        break;
386 e69954b9 pbrook
    case 0x44: /* PCICTL */
387 e69954b9 pbrook
        /* nothing to do.  */
388 e69954b9 pbrook
        break;
389 e69954b9 pbrook
    case 0x4c: /* FLASH */
390 242ea2c6 Peter Maydell
        break;
391 e69954b9 pbrook
    case 0x50: /* CLCD */
392 242ea2c6 Peter Maydell
        switch (board_id(s)) {
393 242ea2c6 Peter Maydell
        case BOARD_ID_PB926:
394 242ea2c6 Peter Maydell
            /* On 926 bits 13:8 are R/O, bits 1:0 control
395 242ea2c6 Peter Maydell
             * the mux that defines how to interpret the PL110
396 242ea2c6 Peter Maydell
             * graphics format, and other bits are r/w but we
397 242ea2c6 Peter Maydell
             * don't implement them to do anything.
398 242ea2c6 Peter Maydell
             */
399 242ea2c6 Peter Maydell
            s->sys_clcd &= 0x3f00;
400 242ea2c6 Peter Maydell
            s->sys_clcd |= val & ~0x3f00;
401 242ea2c6 Peter Maydell
            qemu_set_irq(s->pl110_mux_ctrl, val & 3);
402 242ea2c6 Peter Maydell
            break;
403 242ea2c6 Peter Maydell
        case BOARD_ID_EB:
404 242ea2c6 Peter Maydell
            /* The EB is the same except that there is no mux since
405 242ea2c6 Peter Maydell
             * the EB has a PL111.
406 242ea2c6 Peter Maydell
             */
407 242ea2c6 Peter Maydell
            s->sys_clcd &= 0x3f00;
408 242ea2c6 Peter Maydell
            s->sys_clcd |= val & ~0x3f00;
409 242ea2c6 Peter Maydell
            break;
410 242ea2c6 Peter Maydell
        case BOARD_ID_PBA8:
411 242ea2c6 Peter Maydell
        case BOARD_ID_PBX:
412 242ea2c6 Peter Maydell
            /* On PBA8 and PBX bit 7 is r/w and all other bits
413 242ea2c6 Peter Maydell
             * are either r/o or RAZ/WI.
414 242ea2c6 Peter Maydell
             */
415 242ea2c6 Peter Maydell
            s->sys_clcd &= (1 << 7);
416 242ea2c6 Peter Maydell
            s->sys_clcd |= val & ~(1 << 7);
417 242ea2c6 Peter Maydell
            break;
418 242ea2c6 Peter Maydell
        case BOARD_ID_VEXPRESS:
419 242ea2c6 Peter Maydell
        default:
420 242ea2c6 Peter Maydell
            /* On VExpress this register is unimplemented and will RAZ/WI */
421 242ea2c6 Peter Maydell
            break;
422 242ea2c6 Peter Maydell
        }
423 bf4229d3 Peter Maydell
        break;
424 e69954b9 pbrook
    case 0x54: /* CLCDSER */
425 e69954b9 pbrook
    case 0x64: /* DMAPSR0 */
426 e69954b9 pbrook
    case 0x68: /* DMAPSR1 */
427 e69954b9 pbrook
    case 0x6c: /* DMAPSR2 */
428 e69954b9 pbrook
    case 0x70: /* IOSEL */
429 e69954b9 pbrook
    case 0x74: /* PLDCTL */
430 e69954b9 pbrook
    case 0x80: /* BUSID */
431 e69954b9 pbrook
    case 0x84: /* PROCID0 */
432 e69954b9 pbrook
    case 0x88: /* PROCID1 */
433 e69954b9 pbrook
    case 0x8c: /* OSCRESET0 */
434 e69954b9 pbrook
    case 0x90: /* OSCRESET1 */
435 e69954b9 pbrook
    case 0x94: /* OSCRESET2 */
436 e69954b9 pbrook
    case 0x98: /* OSCRESET3 */
437 e69954b9 pbrook
    case 0x9c: /* OSCRESET4 */
438 e69954b9 pbrook
        break;
439 34933c8c Peter Maydell
    case 0xa0: /* SYS_CFGDATA */
440 34933c8c Peter Maydell
        if (board_id(s) != BOARD_ID_VEXPRESS) {
441 34933c8c Peter Maydell
            goto bad_reg;
442 34933c8c Peter Maydell
        }
443 34933c8c Peter Maydell
        s->sys_cfgdata = val;
444 34933c8c Peter Maydell
        return;
445 34933c8c Peter Maydell
    case 0xa4: /* SYS_CFGCTRL */
446 34933c8c Peter Maydell
        if (board_id(s) != BOARD_ID_VEXPRESS) {
447 34933c8c Peter Maydell
            goto bad_reg;
448 34933c8c Peter Maydell
        }
449 71538323 Peter Maydell
        /* Undefined bits [19:18] are RAZ/WI, and writing to
450 71538323 Peter Maydell
         * the start bit just triggers the action; it always reads
451 71538323 Peter Maydell
         * as zero.
452 71538323 Peter Maydell
         */
453 71538323 Peter Maydell
        s->sys_cfgctrl = val & ~((3 << 18) | (1 << 31));
454 71538323 Peter Maydell
        if (val & (1 << 31)) {
455 71538323 Peter Maydell
            /* Start bit set -- actually do something */
456 71538323 Peter Maydell
            unsigned int dcc = extract32(s->sys_cfgctrl, 26, 4);
457 71538323 Peter Maydell
            unsigned int function = extract32(s->sys_cfgctrl, 20, 6);
458 71538323 Peter Maydell
            unsigned int site = extract32(s->sys_cfgctrl, 16, 2);
459 71538323 Peter Maydell
            unsigned int position = extract32(s->sys_cfgctrl, 12, 4);
460 71538323 Peter Maydell
            unsigned int device = extract32(s->sys_cfgctrl, 0, 12);
461 71538323 Peter Maydell
            s->sys_cfgstat = 1;            /* complete */
462 71538323 Peter Maydell
            if (s->sys_cfgctrl & (1 << 30)) {
463 71538323 Peter Maydell
                if (!vexpress_cfgctrl_write(s, dcc, function, site, position,
464 71538323 Peter Maydell
                                            device, s->sys_cfgdata)) {
465 71538323 Peter Maydell
                    s->sys_cfgstat |= 2;        /* error */
466 71538323 Peter Maydell
                }
467 71538323 Peter Maydell
            } else {
468 71538323 Peter Maydell
                uint32_t val;
469 71538323 Peter Maydell
                if (!vexpress_cfgctrl_read(s, dcc, function, site, position,
470 71538323 Peter Maydell
                                           device, &val)) {
471 71538323 Peter Maydell
                    s->sys_cfgstat |= 2;        /* error */
472 71538323 Peter Maydell
                } else {
473 71538323 Peter Maydell
                    s->sys_cfgdata = val;
474 71538323 Peter Maydell
                }
475 71538323 Peter Maydell
            }
476 34933c8c Peter Maydell
        }
477 706872a5 Christoffer Dall
        s->sys_cfgctrl &= ~(1 << 31);
478 34933c8c Peter Maydell
        return;
479 34933c8c Peter Maydell
    case 0xa8: /* SYS_CFGSTAT */
480 34933c8c Peter Maydell
        if (board_id(s) != BOARD_ID_VEXPRESS) {
481 34933c8c Peter Maydell
            goto bad_reg;
482 34933c8c Peter Maydell
        }
483 34933c8c Peter Maydell
        s->sys_cfgstat = val & 3;
484 34933c8c Peter Maydell
        return;
485 e69954b9 pbrook
    default:
486 34933c8c Peter Maydell
    bad_reg:
487 0c896f06 Peter Maydell
        qemu_log_mask(LOG_GUEST_ERROR,
488 0c896f06 Peter Maydell
                      "arm_sysctl_write: Bad register offset 0x%x\n",
489 0c896f06 Peter Maydell
                      (int)offset);
490 e69954b9 pbrook
        return;
491 e69954b9 pbrook
    }
492 e69954b9 pbrook
}
493 e69954b9 pbrook
494 460d7c53 Avi Kivity
static const MemoryRegionOps arm_sysctl_ops = {
495 460d7c53 Avi Kivity
    .read = arm_sysctl_read,
496 460d7c53 Avi Kivity
    .write = arm_sysctl_write,
497 460d7c53 Avi Kivity
    .endianness = DEVICE_NATIVE_ENDIAN,
498 e69954b9 pbrook
};
499 e69954b9 pbrook
500 b50ff6f5 Peter Maydell
static void arm_sysctl_gpio_set(void *opaque, int line, int level)
501 b50ff6f5 Peter Maydell
{
502 b50ff6f5 Peter Maydell
    arm_sysctl_state *s = (arm_sysctl_state *)opaque;
503 b50ff6f5 Peter Maydell
    switch (line) {
504 b50ff6f5 Peter Maydell
    case ARM_SYSCTL_GPIO_MMC_WPROT:
505 b50ff6f5 Peter Maydell
    {
506 b50ff6f5 Peter Maydell
        /* For PB926 and EB write-protect is bit 2 of SYS_MCI;
507 b50ff6f5 Peter Maydell
         * for all later boards it is bit 1.
508 b50ff6f5 Peter Maydell
         */
509 b50ff6f5 Peter Maydell
        int bit = 2;
510 b50ff6f5 Peter Maydell
        if ((board_id(s) == BOARD_ID_PB926) || (board_id(s) == BOARD_ID_EB)) {
511 b50ff6f5 Peter Maydell
            bit = 4;
512 b50ff6f5 Peter Maydell
        }
513 b50ff6f5 Peter Maydell
        s->sys_mci &= ~bit;
514 b50ff6f5 Peter Maydell
        if (level) {
515 b50ff6f5 Peter Maydell
            s->sys_mci |= bit;
516 b50ff6f5 Peter Maydell
        }
517 b50ff6f5 Peter Maydell
        break;
518 b50ff6f5 Peter Maydell
    }
519 b50ff6f5 Peter Maydell
    case ARM_SYSCTL_GPIO_MMC_CARDIN:
520 b50ff6f5 Peter Maydell
        s->sys_mci &= ~1;
521 b50ff6f5 Peter Maydell
        if (level) {
522 b50ff6f5 Peter Maydell
            s->sys_mci |= 1;
523 b50ff6f5 Peter Maydell
        }
524 b50ff6f5 Peter Maydell
        break;
525 b50ff6f5 Peter Maydell
    }
526 b50ff6f5 Peter Maydell
}
527 b50ff6f5 Peter Maydell
528 54de1e5b Peter Maydell
static int arm_sysctl_init(SysBusDevice *dev)
529 e69954b9 pbrook
{
530 82634c2d Paul Brook
    arm_sysctl_state *s = FROM_SYSBUS(arm_sysctl_state, dev);
531 e69954b9 pbrook
532 460d7c53 Avi Kivity
    memory_region_init_io(&s->iomem, &arm_sysctl_ops, s, "arm-sysctl", 0x1000);
533 750ecd44 Avi Kivity
    sysbus_init_mmio(dev, &s->iomem);
534 b50ff6f5 Peter Maydell
    qdev_init_gpio_in(&s->busdev.qdev, arm_sysctl_gpio_set, 2);
535 242ea2c6 Peter Maydell
    qdev_init_gpio_out(&s->busdev.qdev, &s->pl110_mux_ctrl, 1);
536 81a322d4 Gerd Hoffmann
    return 0;
537 e69954b9 pbrook
}
538 82634c2d Paul Brook
539 999e12bb Anthony Liguori
static Property arm_sysctl_properties[] = {
540 999e12bb Anthony Liguori
    DEFINE_PROP_UINT32("sys_id", arm_sysctl_state, sys_id, 0),
541 999e12bb Anthony Liguori
    DEFINE_PROP_UINT32("proc_id", arm_sysctl_state, proc_id, 0),
542 999e12bb Anthony Liguori
    DEFINE_PROP_END_OF_LIST(),
543 999e12bb Anthony Liguori
};
544 999e12bb Anthony Liguori
545 999e12bb Anthony Liguori
static void arm_sysctl_class_init(ObjectClass *klass, void *data)
546 999e12bb Anthony Liguori
{
547 39bffca2 Anthony Liguori
    DeviceClass *dc = DEVICE_CLASS(klass);
548 999e12bb Anthony Liguori
    SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass);
549 999e12bb Anthony Liguori
550 54de1e5b Peter Maydell
    k->init = arm_sysctl_init;
551 39bffca2 Anthony Liguori
    dc->reset = arm_sysctl_reset;
552 39bffca2 Anthony Liguori
    dc->vmsd = &vmstate_arm_sysctl;
553 39bffca2 Anthony Liguori
    dc->props = arm_sysctl_properties;
554 999e12bb Anthony Liguori
}
555 999e12bb Anthony Liguori
556 8c43a6f0 Andreas Färber
static const TypeInfo arm_sysctl_info = {
557 39bffca2 Anthony Liguori
    .name          = "realview_sysctl",
558 39bffca2 Anthony Liguori
    .parent        = TYPE_SYS_BUS_DEVICE,
559 39bffca2 Anthony Liguori
    .instance_size = sizeof(arm_sysctl_state),
560 39bffca2 Anthony Liguori
    .class_init    = arm_sysctl_class_init,
561 ee6847d1 Gerd Hoffmann
};
562 ee6847d1 Gerd Hoffmann
563 83f7d43a Andreas Färber
static void arm_sysctl_register_types(void)
564 82634c2d Paul Brook
{
565 39bffca2 Anthony Liguori
    type_register_static(&arm_sysctl_info);
566 82634c2d Paul Brook
}
567 82634c2d Paul Brook
568 83f7d43a Andreas Färber
type_init(arm_sysctl_register_types)