aboutsummaryrefslogtreecommitdiffstats
path: root/arch/mips
diff options
context:
space:
mode:
authorDavid Daney <david.daney@cavium.com>2014-05-28 17:52:13 -0400
committerRalf Baechle <ralf@linux-mips.org>2014-06-02 06:34:20 -0400
commit18280edafef1b8ffc920743eddaf6cf6612b1509 (patch)
tree42136302ac56fb113d16de3128becd1fc6aed7a6 /arch/mips
parent90dfdc7ceb577c0d7b7635def3c62039a091e50d (diff)
MIPS: Add code for new system 'paravirt'
For para-virtualized guests running under KVM or other equivalent hypervisor. Signed-off-by: David Daney <david.daney@cavium.com> Signed-off-by: Andreas Herrmann <andreas.herrmann@caviumnetworks.com> Cc: linux-mips@linux-mips.org Cc: James Hogan <james.hogan@imgtec.com> Cc: kvm@vger.kernel.org Patchwork: https://patchwork.linux-mips.org/patch/7004/ Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
Diffstat (limited to 'arch/mips')
-rw-r--r--arch/mips/include/asm/mach-paravirt/cpu-feature-overrides.h36
-rw-r--r--arch/mips/include/asm/mach-paravirt/irq.h19
-rw-r--r--arch/mips/include/asm/mach-paravirt/kernel-entry-init.h50
-rw-r--r--arch/mips/include/asm/mach-paravirt/war.h25
-rw-r--r--arch/mips/mm/tlbex.c8
-rw-r--r--arch/mips/paravirt/Makefile14
-rw-r--r--arch/mips/paravirt/Platform8
-rw-r--r--arch/mips/paravirt/paravirt-irq.c368
-rw-r--r--arch/mips/paravirt/paravirt-smp.c148
-rw-r--r--arch/mips/paravirt/serial.c40
-rw-r--r--arch/mips/paravirt/setup.c67
11 files changed, 777 insertions, 6 deletions
diff --git a/arch/mips/include/asm/mach-paravirt/cpu-feature-overrides.h b/arch/mips/include/asm/mach-paravirt/cpu-feature-overrides.h
new file mode 100644
index 000000000000..725e1ed83f6a
--- /dev/null
+++ b/arch/mips/include/asm/mach-paravirt/cpu-feature-overrides.h
@@ -0,0 +1,36 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc.
7 */
8#ifndef __ASM_MACH_PARAVIRT_CPU_FEATURE_OVERRIDES_H
9#define __ASM_MACH_PARAVIRT_CPU_FEATURE_OVERRIDES_H
10
11#define cpu_has_4kex 1
12#define cpu_has_3k_cache 0
13#define cpu_has_tx39_cache 0
14#define cpu_has_counter 1
15#define cpu_has_llsc 1
16/*
17 * We Disable LL/SC on non SMP systems as it is faster to disable
18 * interrupts for atomic access than a LL/SC.
19 */
20#ifdef CONFIG_SMP
21# define kernel_uses_llsc 1
22#else
23# define kernel_uses_llsc 0
24#endif
25
26#ifdef CONFIG_CPU_CAVIUM_OCTEON
27#define cpu_dcache_line_size() 128
28#define cpu_icache_line_size() 128
29#define cpu_has_octeon_cache 1
30#define cpu_has_4k_cache 0
31#else
32#define cpu_has_octeon_cache 0
33#define cpu_has_4k_cache 1
34#endif
35
36#endif /* __ASM_MACH_PARAVIRT_CPU_FEATURE_OVERRIDES_H */
diff --git a/arch/mips/include/asm/mach-paravirt/irq.h b/arch/mips/include/asm/mach-paravirt/irq.h
new file mode 100644
index 000000000000..9b4d35eca977
--- /dev/null
+++ b/arch/mips/include/asm/mach-paravirt/irq.h
@@ -0,0 +1,19 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc.
7 */
8#ifndef __ASM_MACH_PARAVIRT_IRQ_H__
9#define __ASM_MACH_PARAVIRT_IRQ_H__
10
11#define NR_IRQS 64
12#define MIPS_CPU_IRQ_BASE 1
13
14#define MIPS_IRQ_PCIA (MIPS_CPU_IRQ_BASE + 8)
15
16#define MIPS_IRQ_MBOX0 (MIPS_CPU_IRQ_BASE + 32)
17#define MIPS_IRQ_MBOX1 (MIPS_CPU_IRQ_BASE + 33)
18
19#endif /* __ASM_MACH_PARAVIRT_IRQ_H__ */
diff --git a/arch/mips/include/asm/mach-paravirt/kernel-entry-init.h b/arch/mips/include/asm/mach-paravirt/kernel-entry-init.h
new file mode 100644
index 000000000000..2f82bfa3a773
--- /dev/null
+++ b/arch/mips/include/asm/mach-paravirt/kernel-entry-init.h
@@ -0,0 +1,50 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc
7 */
8#ifndef __ASM_MACH_PARAVIRT_KERNEL_ENTRY_H
9#define __ASM_MACH_PARAVIRT_KERNEL_ENTRY_H
10
11#define CP0_EBASE $15, 1
12
13 .macro kernel_entry_setup
14 mfc0 t0, CP0_EBASE
15 andi t0, t0, 0x3ff # CPUNum
16 beqz t0, 1f
17 # CPUs other than zero goto smp_bootstrap
18 j smp_bootstrap
19
201:
21 .endm
22
23/*
24 * Do SMP slave processor setup necessary before we can safely execute
25 * C code.
26 */
27 .macro smp_slave_setup
28 mfc0 t0, CP0_EBASE
29 andi t0, t0, 0x3ff # CPUNum
30 slti t1, t0, NR_CPUS
31 bnez t1, 1f
322:
33 di
34 wait
35 b 2b # Unknown CPU, loop forever.
361:
37 PTR_LA t1, paravirt_smp_sp
38 PTR_SLL t0, PTR_SCALESHIFT
39 PTR_ADDU t1, t1, t0
403:
41 PTR_L sp, 0(t1)
42 beqz sp, 3b # Spin until told to proceed.
43
44 PTR_LA t1, paravirt_smp_gp
45 PTR_ADDU t1, t1, t0
46 sync
47 PTR_L gp, 0(t1)
48 .endm
49
50#endif /* __ASM_MACH_PARAVIRT_KERNEL_ENTRY_H */
diff --git a/arch/mips/include/asm/mach-paravirt/war.h b/arch/mips/include/asm/mach-paravirt/war.h
new file mode 100644
index 000000000000..36d3afb98451
--- /dev/null
+++ b/arch/mips/include/asm/mach-paravirt/war.h
@@ -0,0 +1,25 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2002, 2004, 2007 by Ralf Baechle <ralf@linux-mips.org>
7 * Copyright (C) 2013 Cavium Networks <support@caviumnetworks.com>
8 */
9#ifndef __ASM_MIPS_MACH_PARAVIRT_WAR_H
10#define __ASM_MIPS_MACH_PARAVIRT_WAR_H
11
12#define R4600_V1_INDEX_ICACHEOP_WAR 0
13#define R4600_V1_HIT_CACHEOP_WAR 0
14#define R4600_V2_HIT_CACHEOP_WAR 0
15#define R5432_CP0_INTERRUPT_WAR 0
16#define BCM1250_M3_WAR 0
17#define SIBYTE_1956_WAR 0
18#define MIPS4K_ICACHE_REFILL_WAR 0
19#define MIPS_CACHE_SYNC_WAR 0
20#define TX49XX_ICACHE_INDEX_INV_WAR 0
21#define ICACHE_REFILLS_WORKAROUND_WAR 0
22#define R10000_LLSC_WAR 0
23#define MIPS34K_MISSED_ITLB_WAR 0
24
25#endif /* __ASM_MIPS_MACH_PARAVIRT_WAR_H */
diff --git a/arch/mips/mm/tlbex.c b/arch/mips/mm/tlbex.c
index af91f3ed1a82..e80e10bafc83 100644
--- a/arch/mips/mm/tlbex.c
+++ b/arch/mips/mm/tlbex.c
@@ -1250,17 +1250,13 @@ static void build_r4000_tlb_refill_handler(void)
1250 unsigned int final_len; 1250 unsigned int final_len;
1251 struct mips_huge_tlb_info htlb_info __maybe_unused; 1251 struct mips_huge_tlb_info htlb_info __maybe_unused;
1252 enum vmalloc64_mode vmalloc_mode __maybe_unused; 1252 enum vmalloc64_mode vmalloc_mode __maybe_unused;
1253#ifdef CONFIG_64BIT 1253
1254 bool is64bit = true;
1255#else
1256 bool is64bit = false;
1257#endif
1258 memset(tlb_handler, 0, sizeof(tlb_handler)); 1254 memset(tlb_handler, 0, sizeof(tlb_handler));
1259 memset(labels, 0, sizeof(labels)); 1255 memset(labels, 0, sizeof(labels));
1260 memset(relocs, 0, sizeof(relocs)); 1256 memset(relocs, 0, sizeof(relocs));
1261 memset(final_handler, 0, sizeof(final_handler)); 1257 memset(final_handler, 0, sizeof(final_handler));
1262 1258
1263 if (is64bit && (scratch_reg >= 0 || scratchpad_available()) && use_bbit_insns()) { 1259 if (IS_ENABLED(CONFIG_64BIT) && (scratch_reg >= 0 || scratchpad_available()) && use_bbit_insns()) {
1264 htlb_info = build_fast_tlb_refill_handler(&p, &l, &r, K0, K1, 1260 htlb_info = build_fast_tlb_refill_handler(&p, &l, &r, K0, K1,
1265 scratch_reg); 1261 scratch_reg);
1266 vmalloc_mode = refill_scratch; 1262 vmalloc_mode = refill_scratch;
diff --git a/arch/mips/paravirt/Makefile b/arch/mips/paravirt/Makefile
new file mode 100644
index 000000000000..5023af733a35
--- /dev/null
+++ b/arch/mips/paravirt/Makefile
@@ -0,0 +1,14 @@
1#
2# Makefile for MIPS para-virtualized specific kernel interface routines
3# under Linux.
4#
5# This file is subject to the terms and conditions of the GNU General Public
6# License. See the file "COPYING" in the main directory of this archive
7# for more details.
8#
9# Copyright (C) 2013 Cavium, Inc.
10#
11
12obj-y := setup.o serial.o paravirt-irq.o
13
14obj-$(CONFIG_SMP) += paravirt-smp.o
diff --git a/arch/mips/paravirt/Platform b/arch/mips/paravirt/Platform
new file mode 100644
index 000000000000..7e76ef25ea17
--- /dev/null
+++ b/arch/mips/paravirt/Platform
@@ -0,0 +1,8 @@
1#
2# Generic para-virtualized guest.
3#
4platform-$(CONFIG_MIPS_PARAVIRT) += paravirt/
5cflags-$(CONFIG_MIPS_PARAVIRT) += \
6 -I$(srctree)/arch/mips/include/asm/mach-paravirt
7
8load-$(CONFIG_MIPS_PARAVIRT) = 0xffffffff80010000
diff --git a/arch/mips/paravirt/paravirt-irq.c b/arch/mips/paravirt/paravirt-irq.c
new file mode 100644
index 000000000000..8987b06c9de9
--- /dev/null
+++ b/arch/mips/paravirt/paravirt-irq.c
@@ -0,0 +1,368 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc.
7 */
8
9#include <linux/interrupt.h>
10#include <linux/cpumask.h>
11#include <linux/kernel.h>
12#include <linux/mutex.h>
13
14#include <asm/io.h>
15
16#define MBOX_BITS_PER_CPU 2
17
18static int cpunum_for_cpu(int cpu)
19{
20#ifdef CONFIG_SMP
21 return cpu_logical_map(cpu);
22#else
23 return get_ebase_cpunum();
24#endif
25}
26
27struct core_chip_data {
28 struct mutex core_irq_mutex;
29 bool current_en;
30 bool desired_en;
31 u8 bit;
32};
33
34static struct core_chip_data irq_core_chip_data[8];
35
36static void irq_core_ack(struct irq_data *data)
37{
38 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
39 unsigned int bit = cd->bit;
40
41 /*
42 * We don't need to disable IRQs to make these atomic since
43 * they are already disabled earlier in the low level
44 * interrupt code.
45 */
46 clear_c0_status(0x100 << bit);
47 /* The two user interrupts must be cleared manually. */
48 if (bit < 2)
49 clear_c0_cause(0x100 << bit);
50}
51
52static void irq_core_eoi(struct irq_data *data)
53{
54 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
55
56 /*
57 * We don't need to disable IRQs to make these atomic since
58 * they are already disabled earlier in the low level
59 * interrupt code.
60 */
61 set_c0_status(0x100 << cd->bit);
62}
63
64static void irq_core_set_enable_local(void *arg)
65{
66 struct irq_data *data = arg;
67 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
68 unsigned int mask = 0x100 << cd->bit;
69
70 /*
71 * Interrupts are already disabled, so these are atomic.
72 */
73 if (cd->desired_en)
74 set_c0_status(mask);
75 else
76 clear_c0_status(mask);
77
78}
79
80static void irq_core_disable(struct irq_data *data)
81{
82 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
83 cd->desired_en = false;
84}
85
86static void irq_core_enable(struct irq_data *data)
87{
88 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
89 cd->desired_en = true;
90}
91
92static void irq_core_bus_lock(struct irq_data *data)
93{
94 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
95
96 mutex_lock(&cd->core_irq_mutex);
97}
98
99static void irq_core_bus_sync_unlock(struct irq_data *data)
100{
101 struct core_chip_data *cd = irq_data_get_irq_chip_data(data);
102
103 if (cd->desired_en != cd->current_en) {
104 on_each_cpu(irq_core_set_enable_local, data, 1);
105 cd->current_en = cd->desired_en;
106 }
107
108 mutex_unlock(&cd->core_irq_mutex);
109}
110
111static struct irq_chip irq_chip_core = {
112 .name = "Core",
113 .irq_enable = irq_core_enable,
114 .irq_disable = irq_core_disable,
115 .irq_ack = irq_core_ack,
116 .irq_eoi = irq_core_eoi,
117 .irq_bus_lock = irq_core_bus_lock,
118 .irq_bus_sync_unlock = irq_core_bus_sync_unlock,
119
120 .irq_cpu_online = irq_core_eoi,
121 .irq_cpu_offline = irq_core_ack,
122 .flags = IRQCHIP_ONOFFLINE_ENABLED,
123};
124
125static void __init irq_init_core(void)
126{
127 int i;
128 int irq;
129 struct core_chip_data *cd;
130
131 /* Start with a clean slate */
132 clear_c0_status(ST0_IM);
133 clear_c0_cause(CAUSEF_IP0 | CAUSEF_IP1);
134
135 for (i = 0; i < ARRAY_SIZE(irq_core_chip_data); i++) {
136 cd = irq_core_chip_data + i;
137 cd->current_en = false;
138 cd->desired_en = false;
139 cd->bit = i;
140 mutex_init(&cd->core_irq_mutex);
141
142 irq = MIPS_CPU_IRQ_BASE + i;
143
144 switch (i) {
145 case 0: /* SW0 */
146 case 1: /* SW1 */
147 case 5: /* IP5 */
148 case 6: /* IP6 */
149 case 7: /* IP7 */
150 irq_set_chip_data(irq, cd);
151 irq_set_chip_and_handler(irq, &irq_chip_core,
152 handle_percpu_irq);
153 break;
154 default:
155 break;
156 }
157 }
158}
159
160static void __iomem *mips_irq_chip;
161#define MIPS_IRQ_CHIP_NUM_BITS 0
162#define MIPS_IRQ_CHIP_REGS 8
163
164static int mips_irq_cpu_stride;
165static int mips_irq_chip_reg_raw;
166static int mips_irq_chip_reg_src;
167static int mips_irq_chip_reg_en;
168static int mips_irq_chip_reg_raw_w1s;
169static int mips_irq_chip_reg_raw_w1c;
170static int mips_irq_chip_reg_en_w1s;
171static int mips_irq_chip_reg_en_w1c;
172
173static void irq_pci_enable(struct irq_data *data)
174{
175 u32 mask = 1u << data->irq;
176
177 __raw_writel(mask, mips_irq_chip + mips_irq_chip_reg_en_w1s);
178}
179
180static void irq_pci_disable(struct irq_data *data)
181{
182 u32 mask = 1u << data->irq;
183
184 __raw_writel(mask, mips_irq_chip + mips_irq_chip_reg_en_w1c);
185}
186
187static void irq_pci_ack(struct irq_data *data)
188{
189}
190
191static void irq_pci_mask(struct irq_data *data)
192{
193 u32 mask = 1u << data->irq;
194
195 __raw_writel(mask, mips_irq_chip + mips_irq_chip_reg_en_w1c);
196}
197
198static void irq_pci_unmask(struct irq_data *data)
199{
200 u32 mask = 1u << data->irq;
201
202 __raw_writel(mask, mips_irq_chip + mips_irq_chip_reg_en_w1s);
203}
204
205static struct irq_chip irq_chip_pci = {
206 .name = "PCI",
207 .irq_enable = irq_pci_enable,
208 .irq_disable = irq_pci_disable,
209 .irq_ack = irq_pci_ack,
210 .irq_mask = irq_pci_mask,
211 .irq_unmask = irq_pci_unmask,
212};
213
214static void irq_mbox_all(struct irq_data *data, void __iomem *base)
215{
216 int cpu;
217 unsigned int mbox = data->irq - MIPS_IRQ_MBOX0;
218 u32 mask;
219
220 WARN_ON(mbox >= MBOX_BITS_PER_CPU);
221
222 for_each_online_cpu(cpu) {
223 unsigned int cpuid = cpunum_for_cpu(cpu);
224 mask = 1 << (cpuid * MBOX_BITS_PER_CPU + mbox);
225 __raw_writel(mask, base + (cpuid * mips_irq_cpu_stride));
226 }
227}
228
229static void irq_mbox_enable(struct irq_data *data)
230{
231 irq_mbox_all(data, mips_irq_chip + mips_irq_chip_reg_en_w1s + sizeof(u32));
232}
233
234static void irq_mbox_disable(struct irq_data *data)
235{
236 irq_mbox_all(data, mips_irq_chip + mips_irq_chip_reg_en_w1c + sizeof(u32));
237}
238
239static void irq_mbox_ack(struct irq_data *data)
240{
241 u32 mask;
242 unsigned int mbox = data->irq - MIPS_IRQ_MBOX0;
243
244 WARN_ON(mbox >= MBOX_BITS_PER_CPU);
245
246 mask = 1 << (get_ebase_cpunum() * MBOX_BITS_PER_CPU + mbox);
247 __raw_writel(mask, mips_irq_chip + mips_irq_chip_reg_raw_w1c + sizeof(u32));
248}
249
250void irq_mbox_ipi(int cpu, unsigned int actions)
251{
252 unsigned int cpuid = cpunum_for_cpu(cpu);
253 u32 mask;
254
255 WARN_ON(actions >= (1 << MBOX_BITS_PER_CPU));
256
257 mask = actions << (cpuid * MBOX_BITS_PER_CPU);
258 __raw_writel(mask, mips_irq_chip + mips_irq_chip_reg_raw_w1s + sizeof(u32));
259}
260
261static void irq_mbox_cpu_onoffline(struct irq_data *data, void __iomem *base)
262{
263 unsigned int mbox = data->irq - MIPS_IRQ_MBOX0;
264 unsigned int cpuid = get_ebase_cpunum();
265 u32 mask;
266
267 WARN_ON(mbox >= MBOX_BITS_PER_CPU);
268
269 mask = 1 << (cpuid * MBOX_BITS_PER_CPU + mbox);
270 __raw_writel(mask, base + (cpuid * mips_irq_cpu_stride));
271
272}
273
274static void irq_mbox_cpu_online(struct irq_data *data)
275{
276 irq_mbox_cpu_onoffline(data, mips_irq_chip + mips_irq_chip_reg_en_w1s + sizeof(u32));
277}
278
279static void irq_mbox_cpu_offline(struct irq_data *data)
280{
281 irq_mbox_cpu_onoffline(data, mips_irq_chip + mips_irq_chip_reg_en_w1c + sizeof(u32));
282}
283
284static struct irq_chip irq_chip_mbox = {
285 .name = "MBOX",
286 .irq_enable = irq_mbox_enable,
287 .irq_disable = irq_mbox_disable,
288 .irq_ack = irq_mbox_ack,
289 .irq_cpu_online = irq_mbox_cpu_online,
290 .irq_cpu_offline = irq_mbox_cpu_offline,
291 .flags = IRQCHIP_ONOFFLINE_ENABLED,
292};
293
294static void __init irq_pci_init(void)
295{
296 int i, stride;
297 u32 num_bits;
298
299 mips_irq_chip = ioremap(0x1e010000, 4096);
300
301 num_bits = __raw_readl(mips_irq_chip + MIPS_IRQ_CHIP_NUM_BITS);
302 stride = 8 * (1 + ((num_bits - 1) / 64));
303
304
305 pr_notice("mips_irq_chip: %u bits, reg stride: %d\n", num_bits, stride);
306 mips_irq_chip_reg_raw = MIPS_IRQ_CHIP_REGS + 0 * stride;
307 mips_irq_chip_reg_raw_w1s = MIPS_IRQ_CHIP_REGS + 1 * stride;
308 mips_irq_chip_reg_raw_w1c = MIPS_IRQ_CHIP_REGS + 2 * stride;
309 mips_irq_chip_reg_src = MIPS_IRQ_CHIP_REGS + 3 * stride;
310 mips_irq_chip_reg_en = MIPS_IRQ_CHIP_REGS + 4 * stride;
311 mips_irq_chip_reg_en_w1s = MIPS_IRQ_CHIP_REGS + 5 * stride;
312 mips_irq_chip_reg_en_w1c = MIPS_IRQ_CHIP_REGS + 6 * stride;
313 mips_irq_cpu_stride = stride * 4;
314
315 for (i = 0; i < 4; i++)
316 irq_set_chip_and_handler(i + MIPS_IRQ_PCIA, &irq_chip_pci, handle_level_irq);
317
318 for (i = 0; i < 2; i++)
319 irq_set_chip_and_handler(i + MIPS_IRQ_MBOX0, &irq_chip_mbox, handle_percpu_irq);
320
321
322 set_c0_status(STATUSF_IP2);
323}
324
325static void irq_pci_dispatch(void)
326{
327 unsigned int cpuid = get_ebase_cpunum();
328 u32 en;
329
330 en = __raw_readl(mips_irq_chip + mips_irq_chip_reg_src +
331 (cpuid * mips_irq_cpu_stride));
332
333 if (!en) {
334 en = __raw_readl(mips_irq_chip + mips_irq_chip_reg_src + (cpuid * mips_irq_cpu_stride) + sizeof(u32));
335 en = (en >> (2 * cpuid)) & 3;
336
337 if (!en)
338 spurious_interrupt();
339 else
340 do_IRQ(__ffs(en) + MIPS_IRQ_MBOX0); /* MBOX type */
341 } else {
342 do_IRQ(__ffs(en));
343 }
344}
345
346
347void __init arch_init_irq(void)
348{
349 irq_init_core();
350 irq_pci_init();
351}
352
353asmlinkage void plat_irq_dispatch(void)
354{
355 unsigned int pending = read_c0_cause() & read_c0_status() & ST0_IM;
356 int ip;
357
358 if (unlikely(!pending)) {
359 spurious_interrupt();
360 return;
361 }
362
363 ip = ffs(pending) - 1 - STATUSB_IP0;
364 if (ip == 2)
365 irq_pci_dispatch();
366 else
367 do_IRQ(MIPS_CPU_IRQ_BASE + ip);
368}
diff --git a/arch/mips/paravirt/paravirt-smp.c b/arch/mips/paravirt/paravirt-smp.c
new file mode 100644
index 000000000000..73a123e08c28
--- /dev/null
+++ b/arch/mips/paravirt/paravirt-smp.c
@@ -0,0 +1,148 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc.
7 */
8
9#include <linux/interrupt.h>
10#include <linux/cpumask.h>
11#include <linux/kernel.h>
12#include <linux/sched.h>
13
14#include <asm/mipsregs.h>
15#include <asm/setup.h>
16#include <asm/time.h>
17#include <asm/smp.h>
18
19/*
20 * Writing the sp releases the CPU, so writes must be ordered, gp
21 * first, then sp.
22 */
23unsigned long paravirt_smp_sp[NR_CPUS];
24unsigned long paravirt_smp_gp[NR_CPUS];
25
26static int numcpus = 1;
27
28static int __init set_numcpus(char *str)
29{
30 int newval;
31
32 if (get_option(&str, &newval)) {
33 if (newval < 1 || newval >= NR_CPUS)
34 goto bad;
35 numcpus = newval;
36 return 0;
37 }
38bad:
39 return -EINVAL;
40}
41early_param("numcpus", set_numcpus);
42
43
44static void paravirt_smp_setup(void)
45{
46 int id;
47 unsigned int cpunum = get_ebase_cpunum();
48
49 if (WARN_ON(cpunum >= NR_CPUS))
50 return;
51
52 /* The present CPUs are initially just the boot cpu (CPU 0). */
53 for (id = 0; id < NR_CPUS; id++) {
54 set_cpu_possible(id, id == 0);
55 set_cpu_present(id, id == 0);
56 }
57 __cpu_number_map[cpunum] = 0;
58 __cpu_logical_map[0] = cpunum;
59
60 for (id = 0; id < numcpus; id++) {
61 set_cpu_possible(id, true);
62 set_cpu_present(id, true);
63 __cpu_number_map[id] = id;
64 __cpu_logical_map[id] = id;
65 }
66}
67
68void irq_mbox_ipi(int cpu, unsigned int actions);
69static void paravirt_send_ipi_single(int cpu, unsigned int action)
70{
71 irq_mbox_ipi(cpu, action);
72}
73
74static void paravirt_send_ipi_mask(const struct cpumask *mask, unsigned int action)
75{
76 unsigned int cpu;
77
78 for_each_cpu_mask(cpu, *mask)
79 paravirt_send_ipi_single(cpu, action);
80}
81
82static void paravirt_init_secondary(void)
83{
84 unsigned int sr;
85
86 sr = set_c0_status(ST0_BEV);
87 write_c0_ebase((u32)ebase);
88
89 sr |= STATUSF_IP2; /* Interrupt controller on IP2 */
90 write_c0_status(sr);
91
92 irq_cpu_online();
93}
94
95static void paravirt_smp_finish(void)
96{
97 /* to generate the first CPU timer interrupt */
98 write_c0_compare(read_c0_count() + mips_hpt_frequency / HZ);
99 local_irq_enable();
100}
101
102static void paravirt_cpus_done(void)
103{
104}
105
106static void paravirt_boot_secondary(int cpu, struct task_struct *idle)
107{
108 paravirt_smp_gp[cpu] = (unsigned long)task_thread_info(idle);
109 smp_wmb();
110 paravirt_smp_sp[cpu] = __KSTK_TOS(idle);
111}
112
113static irqreturn_t paravirt_reched_interrupt(int irq, void *dev_id)
114{
115 scheduler_ipi();
116 return IRQ_HANDLED;
117}
118
119static irqreturn_t paravirt_function_interrupt(int irq, void *dev_id)
120{
121 smp_call_function_interrupt();
122 return IRQ_HANDLED;
123}
124
125static void paravirt_prepare_cpus(unsigned int max_cpus)
126{
127 if (request_irq(MIPS_IRQ_MBOX0, paravirt_reched_interrupt,
128 IRQF_PERCPU | IRQF_NO_THREAD, "Scheduler",
129 paravirt_reched_interrupt)) {
130 panic("Cannot request_irq for SchedulerIPI");
131 }
132 if (request_irq(MIPS_IRQ_MBOX1, paravirt_function_interrupt,
133 IRQF_PERCPU | IRQF_NO_THREAD, "SMP-Call",
134 paravirt_function_interrupt)) {
135 panic("Cannot request_irq for SMP-Call");
136 }
137}
138
139struct plat_smp_ops paravirt_smp_ops = {
140 .send_ipi_single = paravirt_send_ipi_single,
141 .send_ipi_mask = paravirt_send_ipi_mask,
142 .init_secondary = paravirt_init_secondary,
143 .smp_finish = paravirt_smp_finish,
144 .cpus_done = paravirt_cpus_done,
145 .boot_secondary = paravirt_boot_secondary,
146 .smp_setup = paravirt_smp_setup,
147 .prepare_cpus = paravirt_prepare_cpus,
148};
diff --git a/arch/mips/paravirt/serial.c b/arch/mips/paravirt/serial.c
new file mode 100644
index 000000000000..02b665c02272
--- /dev/null
+++ b/arch/mips/paravirt/serial.c
@@ -0,0 +1,40 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc.
7 */
8
9#include <linux/kernel.h>
10#include <linux/virtio_console.h>
11#include <linux/kvm_para.h>
12
13/*
14 * Emit one character to the boot console.
15 */
16int prom_putchar(char c)
17{
18 kvm_hypercall3(KVM_HC_MIPS_CONSOLE_OUTPUT, 0 /* port 0 */,
19 (unsigned long)&c, 1 /* len == 1 */);
20
21 return 1;
22}
23
24#ifdef CONFIG_VIRTIO_CONSOLE
25static int paravirt_put_chars(u32 vtermno, const char *buf, int count)
26{
27 kvm_hypercall3(KVM_HC_MIPS_CONSOLE_OUTPUT, vtermno,
28 (unsigned long)buf, count);
29
30 return count;
31}
32
33static int __init paravirt_cons_init(void)
34{
35 virtio_cons_early_init(paravirt_put_chars);
36 return 0;
37}
38core_initcall(paravirt_cons_init);
39
40#endif
diff --git a/arch/mips/paravirt/setup.c b/arch/mips/paravirt/setup.c
new file mode 100644
index 000000000000..cb8448b373a7
--- /dev/null
+++ b/arch/mips/paravirt/setup.c
@@ -0,0 +1,67 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2013 Cavium, Inc.
7 */
8
9#include <linux/kernel.h>
10#include <linux/kvm_para.h>
11
12#include <asm/reboot.h>
13#include <asm/bootinfo.h>
14#include <asm/smp-ops.h>
15#include <asm/time.h>
16
17extern struct plat_smp_ops paravirt_smp_ops;
18
19const char *get_system_type(void)
20{
21 return "MIPS Para-Virtualized Guest";
22}
23
24void __init plat_time_init(void)
25{
26 mips_hpt_frequency = kvm_hypercall0(KVM_HC_MIPS_GET_CLOCK_FREQ);
27
28 preset_lpj = mips_hpt_frequency / (2 * HZ);
29}
30
31static void pv_machine_halt(void)
32{
33 kvm_hypercall0(KVM_HC_MIPS_EXIT_VM);
34}
35
36/*
37 * Early entry point for arch setup
38 */
39void __init prom_init(void)
40{
41 int i;
42 int argc = fw_arg0;
43 char **argv = (char **)fw_arg1;
44
45#ifdef CONFIG_32BIT
46 set_io_port_base(KSEG1ADDR(0x1e000000));
47#else /* CONFIG_64BIT */
48 set_io_port_base(PHYS_TO_XKSEG_UNCACHED(0x1e000000));
49#endif
50
51 for (i = 0; i < argc; i++) {
52 strlcat(arcs_cmdline, argv[i], COMMAND_LINE_SIZE);
53 if (i < argc - 1)
54 strlcat(arcs_cmdline, " ", COMMAND_LINE_SIZE);
55 }
56 _machine_halt = pv_machine_halt;
57 register_smp_ops(&paravirt_smp_ops);
58}
59
60void __init plat_mem_setup(void)
61{
62 /* Do nothing, the "mem=???" parser handles our memory. */
63}
64
65void __init prom_free_prom_memory(void)
66{
67}