Blame view

arch/powerpc/kvm/powerpc.c 16.1 KB
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
  /*
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License, version 2, as
   * published by the Free Software Foundation.
   *
   * This program is distributed in the hope that it will be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   * GNU General Public License for more details.
   *
   * You should have received a copy of the GNU General Public License
   * along with this program; if not, write to the Free Software
   * Foundation, 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
   *
   * Copyright IBM Corp. 2007
   *
   * Authors: Hollis Blanchard <hollisb@us.ibm.com>
   *          Christian Ehrhardt <ehrhardt@linux.vnet.ibm.com>
   */
  
  #include <linux/errno.h>
  #include <linux/err.h>
  #include <linux/kvm_host.h>
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
24
  #include <linux/vmalloc.h>
544c6761b   Alexander Graf   Use hrtimers for ...
25
  #include <linux/hrtimer.h>
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
26
  #include <linux/fs.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
27
  #include <linux/slab.h>
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
28
29
30
  #include <asm/cputable.h>
  #include <asm/uaccess.h>
  #include <asm/kvm_ppc.h>
83aae4a80   Hollis Blanchard   KVM: ppc: Write o...
31
  #include <asm/tlbflush.h>
371fefd6f   Paul Mackerras   KVM: PPC: Allow b...
32
  #include <asm/cputhreads.h>
73e75b416   Hollis Blanchard   KVM: ppc: Impleme...
33
  #include "timing.h"
fad7b9b51   Paul Mackerras   powerpc: Fix KVM ...
34
  #include "../mm/mmu_decl.h"
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
35

46f43c6ee   Marcelo Tosatti   KVM: powerpc: con...
36
37
  #define CREATE_TRACE_POINTS
  #include "trace.h"
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
38
39
  int kvm_arch_vcpu_runnable(struct kvm_vcpu *v)
  {
666e7252a   Alexander Graf   KVM: PPC: Convert...
40
41
  	return !(v->arch.shared->msr & MSR_WE) ||
  	       !!(v->arch.pending_exceptions);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
42
  }
2a342ed57   Alexander Graf   KVM: PPC: Impleme...
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
  int kvmppc_kvm_pv(struct kvm_vcpu *vcpu)
  {
  	int nr = kvmppc_get_gpr(vcpu, 11);
  	int r;
  	unsigned long __maybe_unused param1 = kvmppc_get_gpr(vcpu, 3);
  	unsigned long __maybe_unused param2 = kvmppc_get_gpr(vcpu, 4);
  	unsigned long __maybe_unused param3 = kvmppc_get_gpr(vcpu, 5);
  	unsigned long __maybe_unused param4 = kvmppc_get_gpr(vcpu, 6);
  	unsigned long r2 = 0;
  
  	if (!(vcpu->arch.shared->msr & MSR_SF)) {
  		/* 32 bit mode */
  		param1 &= 0xffffffff;
  		param2 &= 0xffffffff;
  		param3 &= 0xffffffff;
  		param4 &= 0xffffffff;
  	}
  
  	switch (nr) {
5fc87407b   Alexander Graf   KVM: PPC: Expose ...
62
63
64
65
  	case HC_VENDOR_KVM | KVM_HC_PPC_MAP_MAGIC_PAGE:
  	{
  		vcpu->arch.magic_page_pa = param1;
  		vcpu->arch.magic_page_ea = param2;
df1bfa25d   Alexander Graf   KVM: PPC: Put seg...
66
  		r2 = KVM_MAGIC_FEAT_SR;
7508e16c9   Alexander Graf   KVM: PPC: Add fea...
67

5fc87407b   Alexander Graf   KVM: PPC: Expose ...
68
69
70
  		r = HC_EV_SUCCESS;
  		break;
  	}
2a342ed57   Alexander Graf   KVM: PPC: Impleme...
71
72
  	case HC_VENDOR_KVM | KVM_HC_FEATURES:
  		r = HC_EV_SUCCESS;
a4cd8b23a   Scott Wood   KVM: PPC: e500: e...
73
74
  #if defined(CONFIG_PPC_BOOK3S) || defined(CONFIG_KVM_E500)
  		/* XXX Missing magic page on 44x */
5fc87407b   Alexander Graf   KVM: PPC: Expose ...
75
76
  		r2 |= (1 << KVM_FEATURE_MAGIC_PAGE);
  #endif
2a342ed57   Alexander Graf   KVM: PPC: Impleme...
77
78
  
  		/* Second return value is in r4 */
2a342ed57   Alexander Graf   KVM: PPC: Impleme...
79
80
81
82
83
  		break;
  	default:
  		r = HC_EV_UNIMPLEMENTED;
  		break;
  	}
7508e16c9   Alexander Graf   KVM: PPC: Add fea...
84
  	kvmppc_set_gpr(vcpu, 4, r2);
2a342ed57   Alexander Graf   KVM: PPC: Impleme...
85
86
  	return r;
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
87

af8f38b34   Alexander Graf   KVM: PPC: Add san...
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
  int kvmppc_sanity_check(struct kvm_vcpu *vcpu)
  {
  	int r = false;
  
  	/* We have to know what CPU to virtualize */
  	if (!vcpu->arch.pvr)
  		goto out;
  
  	/* PAPR only works with book3s_64 */
  	if ((vcpu->arch.cpu_type != KVM_CPU_3S_64) && vcpu->arch.papr_enabled)
  		goto out;
  
  #ifdef CONFIG_KVM_BOOK3S_64_HV
  	/* HV KVM can only do PAPR mode for now */
  	if (!vcpu->arch.papr_enabled)
  		goto out;
  #endif
  
  	r = true;
  
  out:
  	vcpu->arch.sane = r;
  	return r ? 0 : -EINVAL;
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
  int kvmppc_emulate_mmio(struct kvm_run *run, struct kvm_vcpu *vcpu)
  {
  	enum emulation_result er;
  	int r;
  
  	er = kvmppc_emulate_instruction(run, vcpu);
  	switch (er) {
  	case EMULATE_DONE:
  		/* Future optimization: only reload non-volatiles if they were
  		 * actually modified. */
  		r = RESUME_GUEST_NV;
  		break;
  	case EMULATE_DO_MMIO:
  		run->exit_reason = KVM_EXIT_MMIO;
  		/* We must reload nonvolatiles because "update" load/store
  		 * instructions modify register state. */
  		/* Future optimization: only reload non-volatiles if they were
  		 * actually modified. */
  		r = RESUME_HOST_NV;
  		break;
  	case EMULATE_FAIL:
  		/* XXX Deliver Program interrupt to guest. */
  		printk(KERN_EMERG "%s: emulation failed (%08x)
  ", __func__,
c7f38f46f   Alexander Graf   KVM: PPC: Improve...
136
  		       kvmppc_get_last_inst(vcpu));
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
137
138
139
140
141
142
143
144
  		r = RESUME_HOST;
  		break;
  	default:
  		BUG();
  	}
  
  	return r;
  }
10474ae89   Alexander Graf   KVM: Activate Vir...
145
  int kvm_arch_hardware_enable(void *garbage)
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
146
  {
10474ae89   Alexander Graf   KVM: Activate Vir...
147
  	return 0;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
  }
  
  void kvm_arch_hardware_disable(void *garbage)
  {
  }
  
  int kvm_arch_hardware_setup(void)
  {
  	return 0;
  }
  
  void kvm_arch_hardware_unsetup(void)
  {
  }
  
  void kvm_arch_check_processor_compat(void *rtn)
  {
9dd921cfe   Hollis Blanchard   KVM: ppc: Refacto...
165
  	*(int *)rtn = kvmppc_core_check_processor_compat();
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
166
  }
d89f5eff7   Jan Kiszka   KVM: Clean up vm ...
167
  int kvm_arch_init_vm(struct kvm *kvm)
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
168
  {
f9e0554de   Paul Mackerras   KVM: PPC: Pass in...
169
  	return kvmppc_core_init_vm(kvm);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
170
  }
d89f5eff7   Jan Kiszka   KVM: Clean up vm ...
171
  void kvm_arch_destroy_vm(struct kvm *kvm)
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
172
173
  {
  	unsigned int i;
988a2cae6   Gleb Natapov   KVM: Use macro to...
174
  	struct kvm_vcpu *vcpu;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
175

988a2cae6   Gleb Natapov   KVM: Use macro to...
176
177
178
179
180
181
182
183
  	kvm_for_each_vcpu(i, vcpu, kvm)
  		kvm_arch_vcpu_free(vcpu);
  
  	mutex_lock(&kvm->lock);
  	for (i = 0; i < atomic_read(&kvm->online_vcpus); i++)
  		kvm->vcpus[i] = NULL;
  
  	atomic_set(&kvm->online_vcpus, 0);
f9e0554de   Paul Mackerras   KVM: PPC: Pass in...
184
185
  
  	kvmppc_core_destroy_vm(kvm);
988a2cae6   Gleb Natapov   KVM: Use macro to...
186
  	mutex_unlock(&kvm->lock);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
187
  }
ad8ba2cd4   Sheng Yang   KVM: Add kvm_arch...
188
189
190
  void kvm_arch_sync_events(struct kvm *kvm)
  {
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
191
192
193
194
195
  int kvm_dev_ioctl_check_extension(long ext)
  {
  	int r;
  
  	switch (ext) {
5ce941ee4   Scott Wood   KVM: PPC: booke: ...
196
197
198
  #ifdef CONFIG_BOOKE
  	case KVM_CAP_PPC_BOOKE_SREGS:
  #else
e15a11370   Alexander Graf   powerpc/kvm: Sync...
199
  	case KVM_CAP_PPC_SEGSTATE:
930b412a0   Alexander Graf   KVM: PPC: Enable ...
200
  	case KVM_CAP_PPC_PAPR:
5ce941ee4   Scott Wood   KVM: PPC: booke: ...
201
  #endif
18978768d   Alexander Graf   KVM: PPC: Allow u...
202
  	case KVM_CAP_PPC_UNSET_IRQ:
7b4203e8c   Alexander Graf   KVM: PPC: Expose ...
203
  	case KVM_CAP_PPC_IRQ_LEVEL:
71fbfd5f3   Alexander Graf   KVM: Add support ...
204
  	case KVM_CAP_ENABLE_CAP:
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
205
206
207
208
  		r = 1;
  		break;
  #ifndef CONFIG_KVM_BOOK3S_64_HV
  	case KVM_CAP_PPC_PAIRED_SINGLES:
ad0a048b0   Alexander Graf   KVM: PPC: Add OSI...
209
  	case KVM_CAP_PPC_OSI:
15711e9c9   Alexander Graf   KVM: PPC: Add get...
210
  	case KVM_CAP_PPC_GET_PVINFO:
e15a11370   Alexander Graf   powerpc/kvm: Sync...
211
212
  		r = 1;
  		break;
588968b6b   Laurent Vivier   KVM: Add coalesce...
213
214
215
  	case KVM_CAP_COALESCED_MMIO:
  		r = KVM_COALESCED_MMIO_PAGE_OFFSET;
  		break;
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
216
  #endif
54738c097   David Gibson   KVM: PPC: Acceler...
217
218
219
220
  #ifdef CONFIG_KVM_BOOK3S_64_HV
  	case KVM_CAP_SPAPR_TCE:
  		r = 1;
  		break;
371fefd6f   Paul Mackerras   KVM: PPC: Allow b...
221
222
223
  	case KVM_CAP_PPC_SMT:
  		r = threads_per_core;
  		break;
aa04b4cc5   Paul Mackerras   KVM: PPC: Allocat...
224
225
  	case KVM_CAP_PPC_RMA:
  		r = 1;
9e368f291   Paul Mackerras   KVM: PPC: book3s_...
226
227
228
  		/* PPC970 requires an RMA */
  		if (cpu_has_feature(CPU_FTR_ARCH_201))
  			r = 2;
aa04b4cc5   Paul Mackerras   KVM: PPC: Allocat...
229
  		break;
54738c097   David Gibson   KVM: PPC: Acceler...
230
  #endif
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
231
232
233
234
235
236
237
238
239
240
241
242
243
  	default:
  		r = 0;
  		break;
  	}
  	return r;
  
  }
  
  long kvm_arch_dev_ioctl(struct file *filp,
                          unsigned int ioctl, unsigned long arg)
  {
  	return -EINVAL;
  }
f7784b8ec   Marcelo Tosatti   KVM: split kvm_ar...
244
245
246
247
248
  int kvm_arch_prepare_memory_region(struct kvm *kvm,
                                     struct kvm_memory_slot *memslot,
                                     struct kvm_memory_slot old,
                                     struct kvm_userspace_memory_region *mem,
                                     int user_alloc)
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
249
  {
f9e0554de   Paul Mackerras   KVM: PPC: Pass in...
250
  	return kvmppc_core_prepare_memory_region(kvm, mem);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
251
  }
f7784b8ec   Marcelo Tosatti   KVM: split kvm_ar...
252
253
254
255
256
  void kvm_arch_commit_memory_region(struct kvm *kvm,
                 struct kvm_userspace_memory_region *mem,
                 struct kvm_memory_slot old,
                 int user_alloc)
  {
f9e0554de   Paul Mackerras   KVM: PPC: Pass in...
257
  	kvmppc_core_commit_memory_region(kvm, mem);
f7784b8ec   Marcelo Tosatti   KVM: split kvm_ar...
258
  }
34d4cb8fc   Marcelo Tosatti   KVM: MMU: nuke sh...
259
260
261
  void kvm_arch_flush_shadow(struct kvm *kvm)
  {
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
262
263
  struct kvm_vcpu *kvm_arch_vcpu_create(struct kvm *kvm, unsigned int id)
  {
73e75b416   Hollis Blanchard   KVM: ppc: Impleme...
264
265
  	struct kvm_vcpu *vcpu;
  	vcpu = kvmppc_core_vcpu_create(kvm, id);
19ccb76a1   Paul Mackerras   KVM: PPC: Impleme...
266
  	vcpu->arch.wqp = &vcpu->wq;
06056bfb9   Wei Yongjun   KVM: PPC: Do not ...
267
268
  	if (!IS_ERR(vcpu))
  		kvmppc_create_vcpu_debugfs(vcpu, id);
73e75b416   Hollis Blanchard   KVM: ppc: Impleme...
269
  	return vcpu;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
270
271
272
273
  }
  
  void kvm_arch_vcpu_free(struct kvm_vcpu *vcpu)
  {
a595405df   Alexander Graf   KVM: PPC: Destory...
274
275
276
  	/* Make sure we're not using the vcpu anymore */
  	hrtimer_cancel(&vcpu->arch.dec_timer);
  	tasklet_kill(&vcpu->arch.tasklet);
73e75b416   Hollis Blanchard   KVM: ppc: Impleme...
277
  	kvmppc_remove_vcpu_debugfs(vcpu);
db93f5745   Hollis Blanchard   KVM: ppc: create ...
278
  	kvmppc_core_vcpu_free(vcpu);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
279
280
281
282
283
284
285
286
287
  }
  
  void kvm_arch_vcpu_destroy(struct kvm_vcpu *vcpu)
  {
  	kvm_arch_vcpu_free(vcpu);
  }
  
  int kvm_cpu_has_pending_timer(struct kvm_vcpu *vcpu)
  {
9dd921cfe   Hollis Blanchard   KVM: ppc: Refacto...
288
  	return kvmppc_core_pending_dec(vcpu);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
289
290
291
292
293
  }
  
  static void kvmppc_decrementer_func(unsigned long data)
  {
  	struct kvm_vcpu *vcpu = (struct kvm_vcpu *)data;
9dd921cfe   Hollis Blanchard   KVM: ppc: Refacto...
294
  	kvmppc_core_queue_dec(vcpu);
45c5eb67d   Hollis Blanchard   KVM: ppc: Handle ...
295

19ccb76a1   Paul Mackerras   KVM: PPC: Impleme...
296
297
  	if (waitqueue_active(vcpu->arch.wqp)) {
  		wake_up_interruptible(vcpu->arch.wqp);
45c5eb67d   Hollis Blanchard   KVM: ppc: Handle ...
298
299
  		vcpu->stat.halt_wakeup++;
  	}
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
300
  }
544c6761b   Alexander Graf   Use hrtimers for ...
301
302
303
304
305
306
307
308
309
310
311
312
313
  /*
   * low level hrtimer wake routine. Because this runs in hardirq context
   * we schedule a tasklet to do the real work.
   */
  enum hrtimer_restart kvmppc_decrementer_wakeup(struct hrtimer *timer)
  {
  	struct kvm_vcpu *vcpu;
  
  	vcpu = container_of(timer, struct kvm_vcpu, arch.dec_timer);
  	tasklet_schedule(&vcpu->arch.tasklet);
  
  	return HRTIMER_NORESTART;
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
314
315
  int kvm_arch_vcpu_init(struct kvm_vcpu *vcpu)
  {
544c6761b   Alexander Graf   Use hrtimers for ...
316
317
318
  	hrtimer_init(&vcpu->arch.dec_timer, CLOCK_REALTIME, HRTIMER_MODE_ABS);
  	tasklet_init(&vcpu->arch.tasklet, kvmppc_decrementer_func, (ulong)vcpu);
  	vcpu->arch.dec_timer.function = kvmppc_decrementer_wakeup;
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
319
  	vcpu->arch.dec_expires = ~(u64)0;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
320

09000adb8   Bharat Bhushan   KVM: PPC: Fix iss...
321
322
323
  #ifdef CONFIG_KVM_EXIT_TIMING
  	mutex_init(&vcpu->arch.exit_timing_lock);
  #endif
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
324
325
326
327
328
  	return 0;
  }
  
  void kvm_arch_vcpu_uninit(struct kvm_vcpu *vcpu)
  {
ecc0981ff   Hollis Blanchard   KVM: ppc: cosmeti...
329
  	kvmppc_mmu_destroy(vcpu);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
330
331
332
333
  }
  
  void kvm_arch_vcpu_load(struct kvm_vcpu *vcpu, int cpu)
  {
eab176722   Scott Wood   KVM: PPC: booke: ...
334
335
336
337
338
339
340
341
342
343
  #ifdef CONFIG_BOOKE
  	/*
  	 * vrsave (formerly usprg0) isn't used by Linux, but may
  	 * be used by the guest.
  	 *
  	 * On non-booke this is associated with Altivec and
  	 * is handled by code in book3s.c.
  	 */
  	mtspr(SPRN_VRSAVE, vcpu->arch.vrsave);
  #endif
9dd921cfe   Hollis Blanchard   KVM: ppc: Refacto...
344
  	kvmppc_core_vcpu_load(vcpu, cpu);
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
345
  	vcpu->cpu = smp_processor_id();
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
346
347
348
349
  }
  
  void kvm_arch_vcpu_put(struct kvm_vcpu *vcpu)
  {
9dd921cfe   Hollis Blanchard   KVM: ppc: Refacto...
350
  	kvmppc_core_vcpu_put(vcpu);
eab176722   Scott Wood   KVM: PPC: booke: ...
351
352
353
  #ifdef CONFIG_BOOKE
  	vcpu->arch.vrsave = mfspr(SPRN_VRSAVE);
  #endif
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
354
  	vcpu->cpu = -1;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
355
  }
d0bfb940e   Jan Kiszka   KVM: New guest de...
356
  int kvm_arch_vcpu_ioctl_set_guest_debug(struct kvm_vcpu *vcpu,
f5d0906b5   Hollis Blanchard   KVM: ppc: remove ...
357
                                          struct kvm_guest_debug *dbg)
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
358
  {
f5d0906b5   Hollis Blanchard   KVM: ppc: remove ...
359
  	return -EINVAL;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
360
361
362
363
364
  }
  
  static void kvmppc_complete_dcr_load(struct kvm_vcpu *vcpu,
                                       struct kvm_run *run)
  {
8e5b26b55   Alexander Graf   KVM: PPC: Use acc...
365
  	kvmppc_set_gpr(vcpu, vcpu->arch.io_gpr, run->dcr.data);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
366
367
368
369
370
  }
  
  static void kvmppc_complete_mmio_load(struct kvm_vcpu *vcpu,
                                        struct kvm_run *run)
  {
69b61833f   Denis Kirjanov   KVM: PPC: fix bui...
371
  	u64 uninitialized_var(gpr);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
372

8e5b26b55   Alexander Graf   KVM: PPC: Use acc...
373
  	if (run->mmio.len > sizeof(gpr)) {
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
374
375
376
377
378
379
380
  		printk(KERN_ERR "bad MMIO length: %d
  ", run->mmio.len);
  		return;
  	}
  
  	if (vcpu->arch.mmio_is_bigendian) {
  		switch (run->mmio.len) {
b104d0663   Alexander Graf   KVM: PPC: Enable ...
381
  		case 8: gpr = *(u64 *)run->mmio.data; break;
8e5b26b55   Alexander Graf   KVM: PPC: Use acc...
382
383
384
  		case 4: gpr = *(u32 *)run->mmio.data; break;
  		case 2: gpr = *(u16 *)run->mmio.data; break;
  		case 1: gpr = *(u8 *)run->mmio.data; break;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
385
386
387
388
  		}
  	} else {
  		/* Convert BE data from userland back to LE. */
  		switch (run->mmio.len) {
8e5b26b55   Alexander Graf   KVM: PPC: Use acc...
389
390
391
  		case 4: gpr = ld_le32((u32 *)run->mmio.data); break;
  		case 2: gpr = ld_le16((u16 *)run->mmio.data); break;
  		case 1: gpr = *(u8 *)run->mmio.data; break;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
392
393
  		}
  	}
8e5b26b55   Alexander Graf   KVM: PPC: Use acc...
394

3587d5348   Alexander Graf   KVM: PPC: Teach M...
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
  	if (vcpu->arch.mmio_sign_extend) {
  		switch (run->mmio.len) {
  #ifdef CONFIG_PPC64
  		case 4:
  			gpr = (s64)(s32)gpr;
  			break;
  #endif
  		case 2:
  			gpr = (s64)(s16)gpr;
  			break;
  		case 1:
  			gpr = (s64)(s8)gpr;
  			break;
  		}
  	}
8e5b26b55   Alexander Graf   KVM: PPC: Use acc...
410
  	kvmppc_set_gpr(vcpu, vcpu->arch.io_gpr, gpr);
b104d0663   Alexander Graf   KVM: PPC: Enable ...
411
412
413
414
415
416
417
418
  
  	switch (vcpu->arch.io_gpr & KVM_REG_EXT_MASK) {
  	case KVM_REG_GPR:
  		kvmppc_set_gpr(vcpu, vcpu->arch.io_gpr, gpr);
  		break;
  	case KVM_REG_FPR:
  		vcpu->arch.fpr[vcpu->arch.io_gpr & KVM_REG_MASK] = gpr;
  		break;
287d5611f   Alexander Graf   KVM: PPC: Only us...
419
  #ifdef CONFIG_PPC_BOOK3S
b104d0663   Alexander Graf   KVM: PPC: Enable ...
420
421
422
423
424
425
426
  	case KVM_REG_QPR:
  		vcpu->arch.qpr[vcpu->arch.io_gpr & KVM_REG_MASK] = gpr;
  		break;
  	case KVM_REG_FQPR:
  		vcpu->arch.fpr[vcpu->arch.io_gpr & KVM_REG_MASK] = gpr;
  		vcpu->arch.qpr[vcpu->arch.io_gpr & KVM_REG_MASK] = gpr;
  		break;
287d5611f   Alexander Graf   KVM: PPC: Only us...
427
  #endif
b104d0663   Alexander Graf   KVM: PPC: Enable ...
428
429
430
  	default:
  		BUG();
  	}
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
  }
  
  int kvmppc_handle_load(struct kvm_run *run, struct kvm_vcpu *vcpu,
                         unsigned int rt, unsigned int bytes, int is_bigendian)
  {
  	if (bytes > sizeof(run->mmio.data)) {
  		printk(KERN_ERR "%s: bad MMIO length: %d
  ", __func__,
  		       run->mmio.len);
  	}
  
  	run->mmio.phys_addr = vcpu->arch.paddr_accessed;
  	run->mmio.len = bytes;
  	run->mmio.is_write = 0;
  
  	vcpu->arch.io_gpr = rt;
  	vcpu->arch.mmio_is_bigendian = is_bigendian;
  	vcpu->mmio_needed = 1;
  	vcpu->mmio_is_write = 0;
3587d5348   Alexander Graf   KVM: PPC: Teach M...
450
  	vcpu->arch.mmio_sign_extend = 0;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
451
452
453
  
  	return EMULATE_DO_MMIO;
  }
3587d5348   Alexander Graf   KVM: PPC: Teach M...
454
455
456
457
458
459
460
461
462
463
464
  /* Same as above, but sign extends */
  int kvmppc_handle_loads(struct kvm_run *run, struct kvm_vcpu *vcpu,
                          unsigned int rt, unsigned int bytes, int is_bigendian)
  {
  	int r;
  
  	r = kvmppc_handle_load(run, vcpu, rt, bytes, is_bigendian);
  	vcpu->arch.mmio_sign_extend = 1;
  
  	return r;
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
465
  int kvmppc_handle_store(struct kvm_run *run, struct kvm_vcpu *vcpu,
b104d0663   Alexander Graf   KVM: PPC: Enable ...
466
                          u64 val, unsigned int bytes, int is_bigendian)
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
  {
  	void *data = run->mmio.data;
  
  	if (bytes > sizeof(run->mmio.data)) {
  		printk(KERN_ERR "%s: bad MMIO length: %d
  ", __func__,
  		       run->mmio.len);
  	}
  
  	run->mmio.phys_addr = vcpu->arch.paddr_accessed;
  	run->mmio.len = bytes;
  	run->mmio.is_write = 1;
  	vcpu->mmio_needed = 1;
  	vcpu->mmio_is_write = 1;
  
  	/* Store the value at the lowest bytes in 'data'. */
  	if (is_bigendian) {
  		switch (bytes) {
b104d0663   Alexander Graf   KVM: PPC: Enable ...
485
  		case 8: *(u64 *)data = val; break;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
  		case 4: *(u32 *)data = val; break;
  		case 2: *(u16 *)data = val; break;
  		case 1: *(u8  *)data = val; break;
  		}
  	} else {
  		/* Store LE value into 'data'. */
  		switch (bytes) {
  		case 4: st_le32(data, val); break;
  		case 2: st_le16(data, val); break;
  		case 1: *(u8 *)data = val; break;
  		}
  	}
  
  	return EMULATE_DO_MMIO;
  }
  
  int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *run)
  {
  	int r;
  	sigset_t sigsaved;
  
  	if (vcpu->sigset_active)
  		sigprocmask(SIG_SETMASK, &vcpu->sigset, &sigsaved);
  
  	if (vcpu->mmio_needed) {
  		if (!vcpu->mmio_is_write)
  			kvmppc_complete_mmio_load(vcpu, run);
  		vcpu->mmio_needed = 0;
  	} else if (vcpu->arch.dcr_needed) {
  		if (!vcpu->arch.dcr_is_write)
  			kvmppc_complete_dcr_load(vcpu, run);
  		vcpu->arch.dcr_needed = 0;
ad0a048b0   Alexander Graf   KVM: PPC: Add OSI...
518
519
520
521
522
523
524
  	} else if (vcpu->arch.osi_needed) {
  		u64 *gprs = run->osi.gprs;
  		int i;
  
  		for (i = 0; i < 32; i++)
  			kvmppc_set_gpr(vcpu, i, gprs[i]);
  		vcpu->arch.osi_needed = 0;
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
525
526
527
528
529
530
531
  	} else if (vcpu->arch.hcall_needed) {
  		int i;
  
  		kvmppc_set_gpr(vcpu, 3, run->papr_hcall.ret);
  		for (i = 0; i < 9; ++i)
  			kvmppc_set_gpr(vcpu, 4 + i, run->papr_hcall.args[i]);
  		vcpu->arch.hcall_needed = 0;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
532
  	}
9dd921cfe   Hollis Blanchard   KVM: ppc: Refacto...
533
  	kvmppc_core_deliver_interrupts(vcpu);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
534

df6909e5d   Paul Mackerras   KVM: PPC: Move gu...
535
  	r = kvmppc_vcpu_run(run, vcpu);
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
536
537
538
539
540
541
542
543
544
  
  	if (vcpu->sigset_active)
  		sigprocmask(SIG_SETMASK, &sigsaved, NULL);
  
  	return r;
  }
  
  int kvm_vcpu_ioctl_interrupt(struct kvm_vcpu *vcpu, struct kvm_interrupt *irq)
  {
19ccb76a1   Paul Mackerras   KVM: PPC: Impleme...
545
  	if (irq->irq == KVM_INTERRUPT_UNSET) {
18978768d   Alexander Graf   KVM: PPC: Allow u...
546
  		kvmppc_core_dequeue_external(vcpu, irq);
19ccb76a1   Paul Mackerras   KVM: PPC: Impleme...
547
548
549
550
  		return 0;
  	}
  
  	kvmppc_core_queue_external(vcpu, irq);
45c5eb67d   Hollis Blanchard   KVM: ppc: Handle ...
551

19ccb76a1   Paul Mackerras   KVM: PPC: Impleme...
552
553
  	if (waitqueue_active(vcpu->arch.wqp)) {
  		wake_up_interruptible(vcpu->arch.wqp);
45c5eb67d   Hollis Blanchard   KVM: ppc: Handle ...
554
  		vcpu->stat.halt_wakeup++;
de56a948b   Paul Mackerras   KVM: PPC: Add sup...
555
556
  	} else if (vcpu->cpu != -1) {
  		smp_send_reschedule(vcpu->cpu);
45c5eb67d   Hollis Blanchard   KVM: ppc: Handle ...
557
  	}
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
558
559
  	return 0;
  }
71fbfd5f3   Alexander Graf   KVM: Add support ...
560
561
562
563
564
565
566
567
568
  static int kvm_vcpu_ioctl_enable_cap(struct kvm_vcpu *vcpu,
  				     struct kvm_enable_cap *cap)
  {
  	int r;
  
  	if (cap->flags)
  		return -EINVAL;
  
  	switch (cap->cap) {
ad0a048b0   Alexander Graf   KVM: PPC: Add OSI...
569
570
571
572
  	case KVM_CAP_PPC_OSI:
  		r = 0;
  		vcpu->arch.osi_enabled = true;
  		break;
930b412a0   Alexander Graf   KVM: PPC: Enable ...
573
574
575
576
  	case KVM_CAP_PPC_PAPR:
  		r = 0;
  		vcpu->arch.papr_enabled = true;
  		break;
71fbfd5f3   Alexander Graf   KVM: Add support ...
577
578
579
580
  	default:
  		r = -EINVAL;
  		break;
  	}
af8f38b34   Alexander Graf   KVM: PPC: Add san...
581
582
  	if (!r)
  		r = kvmppc_sanity_check(vcpu);
71fbfd5f3   Alexander Graf   KVM: Add support ...
583
584
  	return r;
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
  int kvm_arch_vcpu_ioctl_get_mpstate(struct kvm_vcpu *vcpu,
                                      struct kvm_mp_state *mp_state)
  {
  	return -EINVAL;
  }
  
  int kvm_arch_vcpu_ioctl_set_mpstate(struct kvm_vcpu *vcpu,
                                      struct kvm_mp_state *mp_state)
  {
  	return -EINVAL;
  }
  
  long kvm_arch_vcpu_ioctl(struct file *filp,
                           unsigned int ioctl, unsigned long arg)
  {
  	struct kvm_vcpu *vcpu = filp->private_data;
  	void __user *argp = (void __user *)arg;
  	long r;
937366246   Avi Kivity   KVM: Consolidate ...
603
604
  	switch (ioctl) {
  	case KVM_INTERRUPT: {
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
605
606
607
  		struct kvm_interrupt irq;
  		r = -EFAULT;
  		if (copy_from_user(&irq, argp, sizeof(irq)))
937366246   Avi Kivity   KVM: Consolidate ...
608
  			goto out;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
609
  		r = kvm_vcpu_ioctl_interrupt(vcpu, &irq);
937366246   Avi Kivity   KVM: Consolidate ...
610
  		goto out;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
611
  	}
19483d144   Avi Kivity   KVM: PPC: Central...
612

71fbfd5f3   Alexander Graf   KVM: Add support ...
613
614
615
616
617
618
619
620
621
  	case KVM_ENABLE_CAP:
  	{
  		struct kvm_enable_cap cap;
  		r = -EFAULT;
  		if (copy_from_user(&cap, argp, sizeof(cap)))
  			goto out;
  		r = kvm_vcpu_ioctl_enable_cap(vcpu, &cap);
  		break;
  	}
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
622
623
624
625
626
627
628
  	default:
  		r = -EINVAL;
  	}
  
  out:
  	return r;
  }
15711e9c9   Alexander Graf   KVM: PPC: Add get...
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
  static int kvm_vm_ioctl_get_pvinfo(struct kvm_ppc_pvinfo *pvinfo)
  {
  	u32 inst_lis = 0x3c000000;
  	u32 inst_ori = 0x60000000;
  	u32 inst_nop = 0x60000000;
  	u32 inst_sc = 0x44000002;
  	u32 inst_imm_mask = 0xffff;
  
  	/*
  	 * The hypercall to get into KVM from within guest context is as
  	 * follows:
  	 *
  	 *    lis r0, r0, KVM_SC_MAGIC_R0@h
  	 *    ori r0, KVM_SC_MAGIC_R0@l
  	 *    sc
  	 *    nop
  	 */
  	pvinfo->hcall[0] = inst_lis | ((KVM_SC_MAGIC_R0 >> 16) & inst_imm_mask);
  	pvinfo->hcall[1] = inst_ori | (KVM_SC_MAGIC_R0 & inst_imm_mask);
  	pvinfo->hcall[2] = inst_sc;
  	pvinfo->hcall[3] = inst_nop;
  
  	return 0;
  }
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
653
654
655
  long kvm_arch_vm_ioctl(struct file *filp,
                         unsigned int ioctl, unsigned long arg)
  {
15711e9c9   Alexander Graf   KVM: PPC: Add get...
656
  	void __user *argp = (void __user *)arg;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
657
658
659
  	long r;
  
  	switch (ioctl) {
15711e9c9   Alexander Graf   KVM: PPC: Add get...
660
661
  	case KVM_PPC_GET_PVINFO: {
  		struct kvm_ppc_pvinfo pvinfo;
d8cdddcd6   Vasiliy Kulikov   KVM: PPC: fix inf...
662
  		memset(&pvinfo, 0, sizeof(pvinfo));
15711e9c9   Alexander Graf   KVM: PPC: Add get...
663
664
665
666
667
668
669
670
  		r = kvm_vm_ioctl_get_pvinfo(&pvinfo);
  		if (copy_to_user(argp, &pvinfo, sizeof(pvinfo))) {
  			r = -EFAULT;
  			goto out;
  		}
  
  		break;
  	}
54738c097   David Gibson   KVM: PPC: Acceler...
671
672
673
674
675
676
677
678
679
680
681
  #ifdef CONFIG_KVM_BOOK3S_64_HV
  	case KVM_CREATE_SPAPR_TCE: {
  		struct kvm_create_spapr_tce create_tce;
  		struct kvm *kvm = filp->private_data;
  
  		r = -EFAULT;
  		if (copy_from_user(&create_tce, argp, sizeof(create_tce)))
  			goto out;
  		r = kvm_vm_ioctl_create_spapr_tce(kvm, &create_tce);
  		goto out;
  	}
aa04b4cc5   Paul Mackerras   KVM: PPC: Allocat...
682
683
684
685
686
687
688
689
690
691
  
  	case KVM_ALLOCATE_RMA: {
  		struct kvm *kvm = filp->private_data;
  		struct kvm_allocate_rma rma;
  
  		r = kvm_vm_ioctl_allocate_rma(kvm, &rma);
  		if (r >= 0 && copy_to_user(argp, &rma, sizeof(rma)))
  			r = -EFAULT;
  		break;
  	}
54738c097   David Gibson   KVM: PPC: Acceler...
692
  #endif /* CONFIG_KVM_BOOK3S_64_HV */
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
693
  	default:
367e1319b   Avi Kivity   KVM: Return -ENOT...
694
  		r = -ENOTTY;
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
695
  	}
15711e9c9   Alexander Graf   KVM: PPC: Add get...
696
  out:
bbf45ba57   Hollis Blanchard   KVM: ppc: PowerPC...
697
698
699
700
701
702
703
704
705
706
707
  	return r;
  }
  
  int kvm_arch_init(void *opaque)
  {
  	return 0;
  }
  
  void kvm_arch_exit(void)
  {
  }