Commit 24be5ae3a075319a57477fd03b1bdfb74f7f4d05
1 parent
2e719ba3
Add PowerPC 405 input pins (IRQ, resets, ...) model.
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@2654 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
5 changed files
with
133 additions
and
24 deletions
hw/ppc.c
@@ -56,23 +56,23 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | @@ -56,23 +56,23 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | ||
56 | #endif | 56 | #endif |
57 | cur_level = (env->irq_input_state >> pin) & 1; | 57 | cur_level = (env->irq_input_state >> pin) & 1; |
58 | /* Don't generate spurious events */ | 58 | /* Don't generate spurious events */ |
59 | - if ((cur_level == 1 && level == 0) || (cur_level == 0 && level != 0) || 0) { | 59 | + if ((cur_level == 1 && level == 0) || (cur_level == 0 && level != 0)) { |
60 | switch (pin) { | 60 | switch (pin) { |
61 | - case PPC_INPUT_INT: | ||
62 | - /* Level sensitive - asserted high */ | 61 | + case PPC6xx_INPUT_INT: |
62 | + /* Level sensitive - active high */ | ||
63 | #if defined(PPC_DEBUG_IRQ) | 63 | #if defined(PPC_DEBUG_IRQ) |
64 | printf("%s: set the external IRQ state to %d\n", __func__, level); | 64 | printf("%s: set the external IRQ state to %d\n", __func__, level); |
65 | #endif | 65 | #endif |
66 | ppc_set_irq(env, PPC_INTERRUPT_EXT, level); | 66 | ppc_set_irq(env, PPC_INTERRUPT_EXT, level); |
67 | break; | 67 | break; |
68 | - case PPC_INPUT_SMI: | 68 | + case PPC6xx_INPUT_SMI: |
69 | /* Level sensitive - active high */ | 69 | /* Level sensitive - active high */ |
70 | #if defined(PPC_DEBUG_IRQ) | 70 | #if defined(PPC_DEBUG_IRQ) |
71 | printf("%s: set the SMI IRQ state to %d\n", __func__, level); | 71 | printf("%s: set the SMI IRQ state to %d\n", __func__, level); |
72 | #endif | 72 | #endif |
73 | ppc_set_irq(env, PPC_INTERRUPT_SMI, level); | 73 | ppc_set_irq(env, PPC_INTERRUPT_SMI, level); |
74 | break; | 74 | break; |
75 | - case PPC_INPUT_MCP: | 75 | + case PPC6xx_INPUT_MCP: |
76 | /* Negative edge sensitive */ | 76 | /* Negative edge sensitive */ |
77 | /* XXX: TODO: actual reaction may depends on HID0 status | 77 | /* XXX: TODO: actual reaction may depends on HID0 status |
78 | * 603/604/740/750: check HID0[EMCP] | 78 | * 603/604/740/750: check HID0[EMCP] |
@@ -84,7 +84,7 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | @@ -84,7 +84,7 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | ||
84 | ppc_set_irq(env, PPC_INTERRUPT_MCK, 1); | 84 | ppc_set_irq(env, PPC_INTERRUPT_MCK, 1); |
85 | } | 85 | } |
86 | break; | 86 | break; |
87 | - case PPC_INPUT_CKSTP_IN: | 87 | + case PPC6xx_INPUT_CKSTP_IN: |
88 | /* Level sensitive - active low */ | 88 | /* Level sensitive - active low */ |
89 | /* XXX: TODO: relay the signal to CKSTP_OUT pin */ | 89 | /* XXX: TODO: relay the signal to CKSTP_OUT pin */ |
90 | if (level) { | 90 | if (level) { |
@@ -99,7 +99,7 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | @@ -99,7 +99,7 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | ||
99 | env->halted = 0; | 99 | env->halted = 0; |
100 | } | 100 | } |
101 | break; | 101 | break; |
102 | - case PPC_INPUT_HRESET: | 102 | + case PPC6xx_INPUT_HRESET: |
103 | /* Level sensitive - active low */ | 103 | /* Level sensitive - active low */ |
104 | if (level) { | 104 | if (level) { |
105 | #if 0 // XXX: TOFIX | 105 | #if 0 // XXX: TOFIX |
@@ -110,7 +110,7 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | @@ -110,7 +110,7 @@ static void ppc6xx_set_irq (void *opaque, int pin, int level) | ||
110 | #endif | 110 | #endif |
111 | } | 111 | } |
112 | break; | 112 | break; |
113 | - case PPC_INPUT_SRESET: | 113 | + case PPC6xx_INPUT_SRESET: |
114 | #if defined(PPC_DEBUG_IRQ) | 114 | #if defined(PPC_DEBUG_IRQ) |
115 | printf("%s: set the RESET IRQ state to %d\n", __func__, level); | 115 | printf("%s: set the RESET IRQ state to %d\n", __func__, level); |
116 | #endif | 116 | #endif |
@@ -135,6 +135,92 @@ void ppc6xx_irq_init (CPUState *env) | @@ -135,6 +135,92 @@ void ppc6xx_irq_init (CPUState *env) | ||
135 | env->irq_inputs = (void **)qemu_allocate_irqs(&ppc6xx_set_irq, env, 6); | 135 | env->irq_inputs = (void **)qemu_allocate_irqs(&ppc6xx_set_irq, env, 6); |
136 | } | 136 | } |
137 | 137 | ||
138 | +/* PowerPC 405 internal IRQ controller */ | ||
139 | +static void ppc405_set_irq (void *opaque, int pin, int level) | ||
140 | +{ | ||
141 | + CPUState *env = opaque; | ||
142 | + int cur_level; | ||
143 | + | ||
144 | +#if defined(PPC_DEBUG_IRQ) | ||
145 | + printf("%s: env %p pin %d level %d\n", __func__, env, pin, level); | ||
146 | +#endif | ||
147 | + cur_level = (env->irq_input_state >> pin) & 1; | ||
148 | + /* Don't generate spurious events */ | ||
149 | + if ((cur_level == 1 && level == 0) || (cur_level == 0 && level != 0)) { | ||
150 | + switch (pin) { | ||
151 | + case PPC405_INPUT_RESET_SYS: | ||
152 | + /* XXX: TODO: reset all peripherals */ | ||
153 | + /* No break here */ | ||
154 | + case PPC405_INPUT_RESET_CHIP: | ||
155 | + /* XXX: TODO: reset on-chip peripherals */ | ||
156 | + /* No break here */ | ||
157 | + case PPC405_INPUT_RESET_CORE: | ||
158 | + /* XXX: TODO: update DBSR[MRR] */ | ||
159 | + if (level) { | ||
160 | +#if 0 // XXX: TOFIX | ||
161 | +#if defined(PPC_DEBUG_IRQ) | ||
162 | + printf("%s: reset the CPU\n", __func__); | ||
163 | +#endif | ||
164 | + cpu_reset(env); | ||
165 | +#endif | ||
166 | + } | ||
167 | + break; | ||
168 | + case PPC405_INPUT_CINT: | ||
169 | + /* Level sensitive - active high */ | ||
170 | +#if defined(PPC_DEBUG_IRQ) | ||
171 | + printf("%s: set the critical IRQ state to %d\n", __func__, level); | ||
172 | +#endif | ||
173 | + /* XXX: TOFIX */ | ||
174 | + ppc_set_irq(env, PPC_INTERRUPT_RESET, level); | ||
175 | + break; | ||
176 | + case PPC405_INPUT_INT: | ||
177 | + /* Level sensitive - active high */ | ||
178 | +#if defined(PPC_DEBUG_IRQ) | ||
179 | + printf("%s: set the external IRQ state to %d\n", __func__, level); | ||
180 | +#endif | ||
181 | + ppc_set_irq(env, PPC_INTERRUPT_EXT, level); | ||
182 | + break; | ||
183 | + case PPC405_INPUT_HALT: | ||
184 | + /* Level sensitive - active low */ | ||
185 | + if (level) { | ||
186 | +#if defined(PPC_DEBUG_IRQ) | ||
187 | + printf("%s: stop the CPU\n", __func__); | ||
188 | +#endif | ||
189 | + env->halted = 1; | ||
190 | + } else { | ||
191 | +#if defined(PPC_DEBUG_IRQ) | ||
192 | + printf("%s: restart the CPU\n", __func__); | ||
193 | +#endif | ||
194 | + env->halted = 0; | ||
195 | + } | ||
196 | + break; | ||
197 | + case PPC405_INPUT_DEBUG: | ||
198 | + /* Level sensitive - active high */ | ||
199 | +#if defined(PPC_DEBUG_IRQ) | ||
200 | + printf("%s: set the external IRQ state to %d\n", __func__, level); | ||
201 | +#endif | ||
202 | + ppc_set_irq(env, EXCP_40x_DEBUG, level); | ||
203 | + break; | ||
204 | + default: | ||
205 | + /* Unknown pin - do nothing */ | ||
206 | +#if defined(PPC_DEBUG_IRQ) | ||
207 | + printf("%s: unknown IRQ pin %d\n", __func__, pin); | ||
208 | +#endif | ||
209 | + return; | ||
210 | + } | ||
211 | + if (level) | ||
212 | + env->irq_input_state |= 1 << pin; | ||
213 | + else | ||
214 | + env->irq_input_state &= ~(1 << pin); | ||
215 | + } | ||
216 | +} | ||
217 | + | ||
218 | +void ppc405_irq_init (CPUState *env) | ||
219 | +{ | ||
220 | + printf("%s\n", __func__); | ||
221 | + env->irq_inputs = (void **)qemu_allocate_irqs(&ppc405_set_irq, env, 7); | ||
222 | +} | ||
223 | + | ||
138 | /*****************************************************************************/ | 224 | /*****************************************************************************/ |
139 | /* PowerPC time base and decrementer emulation */ | 225 | /* PowerPC time base and decrementer emulation */ |
140 | //#define DEBUG_TB | 226 | //#define DEBUG_TB |
hw/ppc_chrp.c
@@ -470,14 +470,14 @@ static void ppc_chrp_init (int ram_size, int vga_ram_size, int boot_device, | @@ -470,14 +470,14 @@ static void ppc_chrp_init (int ram_size, int vga_ram_size, int boot_device, | ||
470 | */ | 470 | */ |
471 | openpic_irqs[i] = openpic_irqs[0] + (i * OPENPIC_OUTPUT_NB); | 471 | openpic_irqs[i] = openpic_irqs[0] + (i * OPENPIC_OUTPUT_NB); |
472 | openpic_irqs[i][OPENPIC_OUTPUT_INT] = | 472 | openpic_irqs[i][OPENPIC_OUTPUT_INT] = |
473 | - ((qemu_irq *)env->irq_inputs)[PPC_INPUT_INT]; | 473 | + ((qemu_irq *)env->irq_inputs)[PPC6xx_INPUT_INT]; |
474 | openpic_irqs[i][OPENPIC_OUTPUT_CINT] = | 474 | openpic_irqs[i][OPENPIC_OUTPUT_CINT] = |
475 | - ((qemu_irq *)env->irq_inputs)[PPC_INPUT_INT]; | 475 | + ((qemu_irq *)env->irq_inputs)[PPC6xx_INPUT_INT]; |
476 | openpic_irqs[i][OPENPIC_OUTPUT_MCK] = | 476 | openpic_irqs[i][OPENPIC_OUTPUT_MCK] = |
477 | - ((qemu_irq *)env->irq_inputs)[PPC_INPUT_MCP]; | 477 | + ((qemu_irq *)env->irq_inputs)[PPC6xx_INPUT_MCP]; |
478 | openpic_irqs[i][OPENPIC_OUTPUT_DEBUG] = NULL; /* Not connected ? */ | 478 | openpic_irqs[i][OPENPIC_OUTPUT_DEBUG] = NULL; /* Not connected ? */ |
479 | openpic_irqs[i][OPENPIC_OUTPUT_RESET] = | 479 | openpic_irqs[i][OPENPIC_OUTPUT_RESET] = |
480 | - ((qemu_irq *)env->irq_inputs)[PPC_INPUT_HRESET]; /* Check this */ | 480 | + ((qemu_irq *)env->irq_inputs)[PPC6xx_INPUT_HRESET]; /* Check this */ |
481 | } | 481 | } |
482 | pic = openpic_init(NULL, &openpic_mem_index, smp_cpus, | 482 | pic = openpic_init(NULL, &openpic_mem_index, smp_cpus, |
483 | openpic_irqs, NULL); | 483 | openpic_irqs, NULL); |
hw/ppc_prep.c
@@ -598,7 +598,7 @@ static void ppc_prep_init (int ram_size, int vga_ram_size, int boot_device, | @@ -598,7 +598,7 @@ static void ppc_prep_init (int ram_size, int vga_ram_size, int boot_device, | ||
598 | } | 598 | } |
599 | 599 | ||
600 | isa_mem_base = 0xc0000000; | 600 | isa_mem_base = 0xc0000000; |
601 | - i8259 = i8259_init(first_cpu->irq_inputs[PPC_INPUT_INT]); | 601 | + i8259 = i8259_init(first_cpu->irq_inputs[PPC6xx_INPUT_INT]); |
602 | pci_bus = pci_prep_init(i8259); | 602 | pci_bus = pci_prep_init(i8259); |
603 | // pci_bus = i440fx_init(); | 603 | // pci_bus = i440fx_init(); |
604 | /* Register 8 MB of ISA IO space (needed for non-contiguous map) */ | 604 | /* Register 8 MB of ISA IO space (needed for non-contiguous map) */ |
target-ppc/cpu.h
@@ -1315,15 +1315,34 @@ enum { | @@ -1315,15 +1315,34 @@ enum { | ||
1315 | /* Input pins definitions */ | 1315 | /* Input pins definitions */ |
1316 | enum { | 1316 | enum { |
1317 | /* 6xx bus input pins */ | 1317 | /* 6xx bus input pins */ |
1318 | - PPC_INPUT_HRESET = 0, | ||
1319 | - PPC_INPUT_SRESET = 1, | ||
1320 | - PPC_INPUT_CKSTP_IN = 2, | ||
1321 | - PPC_INPUT_MCP = 3, | ||
1322 | - PPC_INPUT_SMI = 4, | ||
1323 | - PPC_INPUT_INT = 5, | 1318 | + PPC6xx_INPUT_HRESET = 0, |
1319 | + PPC6xx_INPUT_SRESET = 1, | ||
1320 | + PPC6xx_INPUT_CKSTP_IN = 2, | ||
1321 | + PPC6xx_INPUT_MCP = 3, | ||
1322 | + PPC6xx_INPUT_SMI = 4, | ||
1323 | + PPC6xx_INPUT_INT = 5, | ||
1324 | +}; | ||
1325 | + | ||
1326 | +enum { | ||
1324 | /* Embedded PowerPC input pins */ | 1327 | /* Embedded PowerPC input pins */ |
1325 | - PPC_INPUT_CINT = 6, | ||
1326 | - PPC_INPUT_NB, | 1328 | + PPCBookE_INPUT_HRESET = 0, |
1329 | + PPCBookE_INPUT_SRESET = 1, | ||
1330 | + PPCBookE_INPUT_CKSTP_IN = 2, | ||
1331 | + PPCBookE_INPUT_MCP = 3, | ||
1332 | + PPCBookE_INPUT_SMI = 4, | ||
1333 | + PPCBookE_INPUT_INT = 5, | ||
1334 | + PPCBookE_INPUT_CINT = 6, | ||
1335 | +}; | ||
1336 | + | ||
1337 | +enum { | ||
1338 | + /* PowerPC 405 input pins */ | ||
1339 | + PPC405_INPUT_RESET_CORE = 0, | ||
1340 | + PPC405_INPUT_RESET_CHIP = 1, | ||
1341 | + PPC405_INPUT_RESET_SYS = 2, | ||
1342 | + PPC405_INPUT_CINT = 3, | ||
1343 | + PPC405_INPUT_INT = 4, | ||
1344 | + PPC405_INPUT_HALT = 5, | ||
1345 | + PPC405_INPUT_DEBUG = 6, | ||
1327 | }; | 1346 | }; |
1328 | 1347 | ||
1329 | /* Hardware exceptions definitions */ | 1348 | /* Hardware exceptions definitions */ |
target-ppc/translate_init.c
@@ -45,6 +45,7 @@ static inline void glue(glue(ppc, name),_irq_init) (CPUPPCState *env) \ | @@ -45,6 +45,7 @@ static inline void glue(glue(ppc, name),_irq_init) (CPUPPCState *env) \ | ||
45 | #define PPC_IRQ_INIT_FN(name) \ | 45 | #define PPC_IRQ_INIT_FN(name) \ |
46 | void glue(glue(ppc, name),_irq_init) (CPUPPCState *env); | 46 | void glue(glue(ppc, name),_irq_init) (CPUPPCState *env); |
47 | #endif | 47 | #endif |
48 | +PPC_IRQ_INIT_FN(405); | ||
48 | PPC_IRQ_INIT_FN(6xx); | 49 | PPC_IRQ_INIT_FN(6xx); |
49 | 50 | ||
50 | /* Generic callbacks: | 51 | /* Generic callbacks: |
@@ -1909,7 +1910,8 @@ static void init_ppc_proc (CPUPPCState *env, ppc_def_t *def) | @@ -1909,7 +1910,8 @@ static void init_ppc_proc (CPUPPCState *env, ppc_def_t *def) | ||
1909 | env->nb_tlb = 64; | 1910 | env->nb_tlb = 64; |
1910 | env->nb_ways = 1; | 1911 | env->nb_ways = 1; |
1911 | env->id_tlbs = 0; | 1912 | env->id_tlbs = 0; |
1912 | - /* XXX: TODO: allocate internal IRQ controller */ | 1913 | + /* Allocate hardware IRQ controller */ |
1914 | + ppc405_irq_init(env); | ||
1913 | break; | 1915 | break; |
1914 | 1916 | ||
1915 | case CPU_PPC_NPE405H: /* NPe405 H family */ | 1917 | case CPU_PPC_NPE405H: /* NPe405 H family */ |
@@ -1924,7 +1926,8 @@ static void init_ppc_proc (CPUPPCState *env, ppc_def_t *def) | @@ -1924,7 +1926,8 @@ static void init_ppc_proc (CPUPPCState *env, ppc_def_t *def) | ||
1924 | env->nb_tlb = 64; | 1926 | env->nb_tlb = 64; |
1925 | env->nb_ways = 1; | 1927 | env->nb_ways = 1; |
1926 | env->id_tlbs = 0; | 1928 | env->id_tlbs = 0; |
1927 | - /* XXX: TODO: allocate internal IRQ controller */ | 1929 | + /* Allocate hardware IRQ controller */ |
1930 | + ppc405_irq_init(env); | ||
1928 | break; | 1931 | break; |
1929 | 1932 | ||
1930 | #if defined (TODO) | 1933 | #if defined (TODO) |
@@ -1956,7 +1959,8 @@ static void init_ppc_proc (CPUPPCState *env, ppc_def_t *def) | @@ -1956,7 +1959,8 @@ static void init_ppc_proc (CPUPPCState *env, ppc_def_t *def) | ||
1956 | env->nb_tlb = 64; | 1959 | env->nb_tlb = 64; |
1957 | env->nb_ways = 1; | 1960 | env->nb_ways = 1; |
1958 | env->id_tlbs = 0; | 1961 | env->id_tlbs = 0; |
1959 | - /* XXX: TODO: allocate internal IRQ controller */ | 1962 | + /* Allocate hardware IRQ controller */ |
1963 | + ppc405_irq_init(env); | ||
1960 | break; | 1964 | break; |
1961 | 1965 | ||
1962 | case CPU_PPC_440EP: /* 440 EP family */ | 1966 | case CPU_PPC_440EP: /* 440 EP family */ |