]> git.proxmox.com Git - mirror_qemu.git/blame - hw/slavio_misc.c
Revert "Get rid of _t suffix"
[mirror_qemu.git] / hw / slavio_misc.c
CommitLineData
3475187d
FB
1/*
2 * QEMU Sparc SLAVIO aux io port emulation
5fafdf24 3 *
3475187d 4 * Copyright (c) 2005 Fabrice Bellard
5fafdf24 5 *
3475187d
FB
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 * THE SOFTWARE.
23 */
2582cfa0 24
87ecb68b
PB
25#include "sun4m.h"
26#include "sysemu.h"
2582cfa0 27#include "sysbus.h"
87ecb68b 28
3475187d
FB
29/* debug misc */
30//#define DEBUG_MISC
31
32/*
33 * This is the auxio port, chip control and system control part of
34 * chip STP2001 (Slave I/O), also produced as NCR89C105. See
35 * http://www.ibiblio.org/pub/historic-linux/early-ports/Sparc/NCR/NCR89C105.txt
36 *
37 * This also includes the PMC CPU idle controller.
38 */
39
40#ifdef DEBUG_MISC
001faf32
BS
41#define MISC_DPRINTF(fmt, ...) \
42 do { printf("MISC: " fmt , ## __VA_ARGS__); } while (0)
3475187d 43#else
001faf32 44#define MISC_DPRINTF(fmt, ...)
3475187d
FB
45#endif
46
47typedef struct MiscState {
2582cfa0 48 SysBusDevice busdev;
d537cf6c 49 qemu_irq irq;
d37adb09 50 uint32_t dummy;
3475187d
FB
51 uint8_t config;
52 uint8_t aux1, aux2;
bfa30a38 53 uint8_t diag, mctrl;
d37adb09 54 uint8_t sysctrl;
6a3b9cc9 55 uint16_t leds;
2be17ebd 56 qemu_irq fdc_tc;
3475187d
FB
57} MiscState;
58
2582cfa0
BS
59typedef struct APCState {
60 SysBusDevice busdev;
61 qemu_irq cpu_halt;
62} APCState;
63
5aca8c3b 64#define MISC_SIZE 1
a8f48dcc 65#define SYSCTRL_SIZE 4
3475187d 66
2be17ebd
BS
67#define AUX1_TC 0x02
68
7debeb82
BS
69#define AUX2_PWROFF 0x01
70#define AUX2_PWRINTCLR 0x02
71#define AUX2_PWRFAIL 0x20
72
73#define CFG_PWRINTEN 0x08
74
75#define SYS_RESET 0x01
76#define SYS_RESETSTAT 0x02
77
3475187d
FB
78static void slavio_misc_update_irq(void *opaque)
79{
80 MiscState *s = opaque;
81
7debeb82 82 if ((s->aux2 & AUX2_PWRFAIL) && (s->config & CFG_PWRINTEN)) {
d537cf6c
PB
83 MISC_DPRINTF("Raise IRQ\n");
84 qemu_irq_raise(s->irq);
3475187d 85 } else {
d537cf6c
PB
86 MISC_DPRINTF("Lower IRQ\n");
87 qemu_irq_lower(s->irq);
3475187d
FB
88 }
89}
90
91static void slavio_misc_reset(void *opaque)
92{
93 MiscState *s = opaque;
94
4e3b1ea1 95 // Diagnostic and system control registers not cleared in reset
3475187d
FB
96 s->config = s->aux1 = s->aux2 = s->mctrl = 0;
97}
98
b2b6f6ec 99static void slavio_set_power_fail(void *opaque, int irq, int power_failing)
3475187d
FB
100{
101 MiscState *s = opaque;
102
103 MISC_DPRINTF("Power fail: %d, config: %d\n", power_failing, s->config);
7debeb82
BS
104 if (power_failing && (s->config & CFG_PWRINTEN)) {
105 s->aux2 |= AUX2_PWRFAIL;
3475187d 106 } else {
7debeb82 107 s->aux2 &= ~AUX2_PWRFAIL;
3475187d
FB
108 }
109 slavio_misc_update_irq(s);
110}
111
c227f099 112static void slavio_cfg_mem_writeb(void *opaque, target_phys_addr_t addr,
a8f48dcc
BS
113 uint32_t val)
114{
115 MiscState *s = opaque;
116
117 MISC_DPRINTF("Write config %2.2x\n", val & 0xff);
118 s->config = val & 0xff;
119 slavio_misc_update_irq(s);
120}
121
c227f099 122static uint32_t slavio_cfg_mem_readb(void *opaque, target_phys_addr_t addr)
a8f48dcc
BS
123{
124 MiscState *s = opaque;
125 uint32_t ret = 0;
126
127 ret = s->config;
128 MISC_DPRINTF("Read config %2.2x\n", ret);
129 return ret;
130}
131
d60efc6b 132static CPUReadMemoryFunc * const slavio_cfg_mem_read[3] = {
a8f48dcc
BS
133 slavio_cfg_mem_readb,
134 NULL,
135 NULL,
136};
137
d60efc6b 138static CPUWriteMemoryFunc * const slavio_cfg_mem_write[3] = {
a8f48dcc
BS
139 slavio_cfg_mem_writeb,
140 NULL,
141 NULL,
142};
143
c227f099 144static void slavio_diag_mem_writeb(void *opaque, target_phys_addr_t addr,
bfa30a38 145 uint32_t val)
3475187d
FB
146{
147 MiscState *s = opaque;
148
a8f48dcc
BS
149 MISC_DPRINTF("Write diag %2.2x\n", val & 0xff);
150 s->diag = val & 0xff;
3475187d
FB
151}
152
c227f099 153static uint32_t slavio_diag_mem_readb(void *opaque, target_phys_addr_t addr)
3475187d
FB
154{
155 MiscState *s = opaque;
156 uint32_t ret = 0;
157
a8f48dcc
BS
158 ret = s->diag;
159 MISC_DPRINTF("Read diag %2.2x\n", ret);
160 return ret;
161}
162
d60efc6b 163static CPUReadMemoryFunc * const slavio_diag_mem_read[3] = {
a8f48dcc
BS
164 slavio_diag_mem_readb,
165 NULL,
166 NULL,
167};
168
d60efc6b 169static CPUWriteMemoryFunc * const slavio_diag_mem_write[3] = {
a8f48dcc
BS
170 slavio_diag_mem_writeb,
171 NULL,
172 NULL,
173};
174
c227f099 175static void slavio_mdm_mem_writeb(void *opaque, target_phys_addr_t addr,
a8f48dcc
BS
176 uint32_t val)
177{
178 MiscState *s = opaque;
179
180 MISC_DPRINTF("Write modem control %2.2x\n", val & 0xff);
181 s->mctrl = val & 0xff;
182}
183
c227f099 184static uint32_t slavio_mdm_mem_readb(void *opaque, target_phys_addr_t addr)
a8f48dcc
BS
185{
186 MiscState *s = opaque;
187 uint32_t ret = 0;
188
189 ret = s->mctrl;
190 MISC_DPRINTF("Read modem control %2.2x\n", ret);
3475187d
FB
191 return ret;
192}
193
d60efc6b 194static CPUReadMemoryFunc * const slavio_mdm_mem_read[3] = {
a8f48dcc 195 slavio_mdm_mem_readb,
7c560456
BS
196 NULL,
197 NULL,
3475187d
FB
198};
199
d60efc6b 200static CPUWriteMemoryFunc * const slavio_mdm_mem_write[3] = {
a8f48dcc 201 slavio_mdm_mem_writeb,
7c560456
BS
202 NULL,
203 NULL,
3475187d
FB
204};
205
c227f099 206static void slavio_aux1_mem_writeb(void *opaque, target_phys_addr_t addr,
0019ad53
BS
207 uint32_t val)
208{
209 MiscState *s = opaque;
210
211 MISC_DPRINTF("Write aux1 %2.2x\n", val & 0xff);
2be17ebd
BS
212 if (val & AUX1_TC) {
213 // Send a pulse to floppy terminal count line
214 if (s->fdc_tc) {
215 qemu_irq_raise(s->fdc_tc);
216 qemu_irq_lower(s->fdc_tc);
217 }
218 val &= ~AUX1_TC;
219 }
0019ad53
BS
220 s->aux1 = val & 0xff;
221}
222
c227f099 223static uint32_t slavio_aux1_mem_readb(void *opaque, target_phys_addr_t addr)
0019ad53
BS
224{
225 MiscState *s = opaque;
226 uint32_t ret = 0;
227
228 ret = s->aux1;
229 MISC_DPRINTF("Read aux1 %2.2x\n", ret);
230
231 return ret;
232}
233
d60efc6b 234static CPUReadMemoryFunc * const slavio_aux1_mem_read[3] = {
0019ad53
BS
235 slavio_aux1_mem_readb,
236 NULL,
237 NULL,
238};
239
d60efc6b 240static CPUWriteMemoryFunc * const slavio_aux1_mem_write[3] = {
0019ad53
BS
241 slavio_aux1_mem_writeb,
242 NULL,
243 NULL,
244};
245
c227f099 246static void slavio_aux2_mem_writeb(void *opaque, target_phys_addr_t addr,
0019ad53
BS
247 uint32_t val)
248{
249 MiscState *s = opaque;
250
251 val &= AUX2_PWRINTCLR | AUX2_PWROFF;
252 MISC_DPRINTF("Write aux2 %2.2x\n", val);
253 val |= s->aux2 & AUX2_PWRFAIL;
254 if (val & AUX2_PWRINTCLR) // Clear Power Fail int
255 val &= AUX2_PWROFF;
256 s->aux2 = val;
257 if (val & AUX2_PWROFF)
258 qemu_system_shutdown_request();
259 slavio_misc_update_irq(s);
260}
261
c227f099 262static uint32_t slavio_aux2_mem_readb(void *opaque, target_phys_addr_t addr)
0019ad53
BS
263{
264 MiscState *s = opaque;
265 uint32_t ret = 0;
266
267 ret = s->aux2;
268 MISC_DPRINTF("Read aux2 %2.2x\n", ret);
269
270 return ret;
271}
272
d60efc6b 273static CPUReadMemoryFunc * const slavio_aux2_mem_read[3] = {
0019ad53
BS
274 slavio_aux2_mem_readb,
275 NULL,
276 NULL,
277};
278
d60efc6b 279static CPUWriteMemoryFunc * const slavio_aux2_mem_write[3] = {
0019ad53
BS
280 slavio_aux2_mem_writeb,
281 NULL,
282 NULL,
283};
284
c227f099 285static void apc_mem_writeb(void *opaque, target_phys_addr_t addr, uint32_t val)
0019ad53 286{
2582cfa0 287 APCState *s = opaque;
0019ad53
BS
288
289 MISC_DPRINTF("Write power management %2.2x\n", val & 0xff);
6d0c293d 290 qemu_irq_raise(s->cpu_halt);
0019ad53
BS
291}
292
c227f099 293static uint32_t apc_mem_readb(void *opaque, target_phys_addr_t addr)
0019ad53
BS
294{
295 uint32_t ret = 0;
296
297 MISC_DPRINTF("Read power management %2.2x\n", ret);
298 return ret;
299}
300
d60efc6b 301static CPUReadMemoryFunc * const apc_mem_read[3] = {
0019ad53
BS
302 apc_mem_readb,
303 NULL,
304 NULL,
305};
306
d60efc6b 307static CPUWriteMemoryFunc * const apc_mem_write[3] = {
0019ad53
BS
308 apc_mem_writeb,
309 NULL,
310 NULL,
311};
312
c227f099 313static uint32_t slavio_sysctrl_mem_readl(void *opaque, target_phys_addr_t addr)
bfa30a38
BS
314{
315 MiscState *s = opaque;
a8f48dcc 316 uint32_t ret = 0;
bfa30a38 317
a8f48dcc 318 switch (addr) {
bfa30a38
BS
319 case 0:
320 ret = s->sysctrl;
321 break;
322 default:
323 break;
324 }
5626b017 325 MISC_DPRINTF("Read system control %08x\n", ret);
bfa30a38
BS
326 return ret;
327}
328
c227f099 329static void slavio_sysctrl_mem_writel(void *opaque, target_phys_addr_t addr,
bfa30a38
BS
330 uint32_t val)
331{
332 MiscState *s = opaque;
bfa30a38 333
5626b017 334 MISC_DPRINTF("Write system control %08x\n", val);
a8f48dcc 335 switch (addr) {
bfa30a38 336 case 0:
7debeb82
BS
337 if (val & SYS_RESET) {
338 s->sysctrl = SYS_RESETSTAT;
bfa30a38
BS
339 qemu_system_reset_request();
340 }
341 break;
342 default:
343 break;
344 }
345}
346
d60efc6b 347static CPUReadMemoryFunc * const slavio_sysctrl_mem_read[3] = {
7c560456
BS
348 NULL,
349 NULL,
bfa30a38
BS
350 slavio_sysctrl_mem_readl,
351};
352
d60efc6b 353static CPUWriteMemoryFunc * const slavio_sysctrl_mem_write[3] = {
7c560456
BS
354 NULL,
355 NULL,
bfa30a38
BS
356 slavio_sysctrl_mem_writel,
357};
358
c227f099 359static uint32_t slavio_led_mem_readw(void *opaque, target_phys_addr_t addr)
6a3b9cc9
BS
360{
361 MiscState *s = opaque;
a8f48dcc 362 uint32_t ret = 0;
6a3b9cc9 363
a8f48dcc 364 switch (addr) {
6a3b9cc9
BS
365 case 0:
366 ret = s->leds;
367 break;
368 default:
369 break;
370 }
5626b017 371 MISC_DPRINTF("Read diagnostic LED %04x\n", ret);
6a3b9cc9
BS
372 return ret;
373}
374
c227f099 375static void slavio_led_mem_writew(void *opaque, target_phys_addr_t addr,
6a3b9cc9
BS
376 uint32_t val)
377{
378 MiscState *s = opaque;
6a3b9cc9 379
5626b017 380 MISC_DPRINTF("Write diagnostic LED %04x\n", val & 0xffff);
a8f48dcc 381 switch (addr) {
6a3b9cc9 382 case 0:
d5296cb5 383 s->leds = val;
6a3b9cc9
BS
384 break;
385 default:
386 break;
387 }
388}
389
d60efc6b 390static CPUReadMemoryFunc * const slavio_led_mem_read[3] = {
7c560456
BS
391 NULL,
392 slavio_led_mem_readw,
393 NULL,
6a3b9cc9
BS
394};
395
d60efc6b 396static CPUWriteMemoryFunc * const slavio_led_mem_write[3] = {
7c560456
BS
397 NULL,
398 slavio_led_mem_writew,
399 NULL,
6a3b9cc9
BS
400};
401
d37adb09
BS
402static const VMStateDescription vmstate_misc = {
403 .name ="slavio_misc",
404 .version_id = 1,
405 .minimum_version_id = 1,
406 .minimum_version_id_old = 1,
407 .fields = (VMStateField []) {
408 VMSTATE_UINT32(dummy, MiscState),
409 VMSTATE_UINT8(config, MiscState),
410 VMSTATE_UINT8(aux1, MiscState),
411 VMSTATE_UINT8(aux2, MiscState),
412 VMSTATE_UINT8(diag, MiscState),
413 VMSTATE_UINT8(mctrl, MiscState),
414 VMSTATE_UINT8(sysctrl, MiscState),
415 VMSTATE_END_OF_LIST()
416 }
417};
3475187d 418
81a322d4 419static int apc_init1(SysBusDevice *dev)
2582cfa0
BS
420{
421 APCState *s = FROM_SYSBUS(APCState, dev);
422 int io;
3475187d 423
2582cfa0
BS
424 sysbus_init_irq(dev, &s->cpu_halt);
425
426 /* Power management (APC) XXX: not a Slavio device */
427 io = cpu_register_io_memory(apc_mem_read, apc_mem_write, s);
428 sysbus_init_mmio(dev, MISC_SIZE, io);
81a322d4 429 return 0;
2582cfa0
BS
430}
431
81a322d4 432static int slavio_misc_init1(SysBusDevice *dev)
2582cfa0
BS
433{
434 MiscState *s = FROM_SYSBUS(MiscState, dev);
435 int io;
436
437 sysbus_init_irq(dev, &s->irq);
438 sysbus_init_irq(dev, &s->fdc_tc);
439
440 /* 8 bit registers */
441 /* Slavio control */
442 io = cpu_register_io_memory(slavio_cfg_mem_read,
443 slavio_cfg_mem_write, s);
444 sysbus_init_mmio(dev, MISC_SIZE, io);
445
446 /* Diagnostics */
447 io = cpu_register_io_memory(slavio_diag_mem_read,
448 slavio_diag_mem_write, s);
449 sysbus_init_mmio(dev, MISC_SIZE, io);
450
451 /* Modem control */
452 io = cpu_register_io_memory(slavio_mdm_mem_read,
453 slavio_mdm_mem_write, s);
454 sysbus_init_mmio(dev, MISC_SIZE, io);
455
456 /* 16 bit registers */
457 /* ss600mp diag LEDs */
458 io = cpu_register_io_memory(slavio_led_mem_read,
459 slavio_led_mem_write, s);
460 sysbus_init_mmio(dev, MISC_SIZE, io);
461
462 /* 32 bit registers */
463 /* System control */
464 io = cpu_register_io_memory(slavio_sysctrl_mem_read,
465 slavio_sysctrl_mem_write, s);
466 sysbus_init_mmio(dev, SYSCTRL_SIZE, io);
467
468 /* AUX 1 (Misc System Functions) */
469 io = cpu_register_io_memory(slavio_aux1_mem_read,
470 slavio_aux1_mem_write, s);
471 sysbus_init_mmio(dev, MISC_SIZE, io);
472
473 /* AUX 2 (Software Powerdown Control) */
474 io = cpu_register_io_memory(slavio_aux2_mem_read,
475 slavio_aux2_mem_write, s);
476 sysbus_init_mmio(dev, MISC_SIZE, io);
477
b2b6f6ec
BS
478 qdev_init_gpio_in(&dev->qdev, slavio_set_power_fail, 1);
479
d37adb09 480 vmstate_register(-1, &vmstate_misc, s);
a08d4367 481 qemu_register_reset(slavio_misc_reset, s);
3475187d 482 slavio_misc_reset(s);
81a322d4 483 return 0;
2582cfa0 484}
0019ad53 485
2582cfa0
BS
486static SysBusDeviceInfo slavio_misc_info = {
487 .init = slavio_misc_init1,
488 .qdev.name = "slavio_misc",
489 .qdev.size = sizeof(MiscState),
2582cfa0
BS
490};
491
492static SysBusDeviceInfo apc_info = {
493 .init = apc_init1,
494 .qdev.name = "apc",
495 .qdev.size = sizeof(MiscState),
2582cfa0
BS
496};
497
498static void slavio_misc_register_devices(void)
499{
500 sysbus_register_withprop(&slavio_misc_info);
501 sysbus_register_withprop(&apc_info);
3475187d 502}
2582cfa0
BS
503
504device_init(slavio_misc_register_devices)