Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / arch / mips / kvm / loongson_ipi.c
blob6ac83a31148cb30cd8ce2ea0f507bb41108d3999
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /*
3 * Loongson-3 Virtual IPI interrupt support.
5 * Copyright (C) 2019 Loongson Technologies, Inc. All rights reserved.
7 * Authors: Chen Zhu <zhuchen@loongson.cn>
8 * Authors: Huacai Chen <chenhc@lemote.com>
9 */
11 #include <linux/kvm_host.h>
13 #include "interrupt.h"
15 #define IPI_BASE 0x3ff01000ULL
17 #define CORE0_STATUS_OFF 0x000
18 #define CORE0_EN_OFF 0x004
19 #define CORE0_SET_OFF 0x008
20 #define CORE0_CLEAR_OFF 0x00c
21 #define CORE0_BUF_20 0x020
22 #define CORE0_BUF_28 0x028
23 #define CORE0_BUF_30 0x030
24 #define CORE0_BUF_38 0x038
26 #define CORE1_STATUS_OFF 0x100
27 #define CORE1_EN_OFF 0x104
28 #define CORE1_SET_OFF 0x108
29 #define CORE1_CLEAR_OFF 0x10c
30 #define CORE1_BUF_20 0x120
31 #define CORE1_BUF_28 0x128
32 #define CORE1_BUF_30 0x130
33 #define CORE1_BUF_38 0x138
35 #define CORE2_STATUS_OFF 0x200
36 #define CORE2_EN_OFF 0x204
37 #define CORE2_SET_OFF 0x208
38 #define CORE2_CLEAR_OFF 0x20c
39 #define CORE2_BUF_20 0x220
40 #define CORE2_BUF_28 0x228
41 #define CORE2_BUF_30 0x230
42 #define CORE2_BUF_38 0x238
44 #define CORE3_STATUS_OFF 0x300
45 #define CORE3_EN_OFF 0x304
46 #define CORE3_SET_OFF 0x308
47 #define CORE3_CLEAR_OFF 0x30c
48 #define CORE3_BUF_20 0x320
49 #define CORE3_BUF_28 0x328
50 #define CORE3_BUF_30 0x330
51 #define CORE3_BUF_38 0x338
53 static int loongson_vipi_read(struct loongson_kvm_ipi *ipi,
54 gpa_t addr, int len, void *val)
56 uint32_t core = (addr >> 8) & 3;
57 uint32_t node = (addr >> 44) & 3;
58 uint32_t id = core + node * 4;
59 uint64_t offset = addr & 0xff;
60 void *pbuf;
61 struct ipi_state *s = &(ipi->ipistate[id]);
63 BUG_ON(offset & (len - 1));
65 switch (offset) {
66 case CORE0_STATUS_OFF:
67 *(uint64_t *)val = s->status;
68 break;
70 case CORE0_EN_OFF:
71 *(uint64_t *)val = s->en;
72 break;
74 case CORE0_SET_OFF:
75 *(uint64_t *)val = 0;
76 break;
78 case CORE0_CLEAR_OFF:
79 *(uint64_t *)val = 0;
80 break;
82 case CORE0_BUF_20 ... CORE0_BUF_38:
83 pbuf = (void *)s->buf + (offset - 0x20);
84 if (len == 8)
85 *(uint64_t *)val = *(uint64_t *)pbuf;
86 else /* Assume len == 4 */
87 *(uint32_t *)val = *(uint32_t *)pbuf;
88 break;
90 default:
91 pr_notice("%s with unknown addr %llx\n", __func__, addr);
92 break;
95 return 0;
98 static int loongson_vipi_write(struct loongson_kvm_ipi *ipi,
99 gpa_t addr, int len, const void *val)
101 uint32_t core = (addr >> 8) & 3;
102 uint32_t node = (addr >> 44) & 3;
103 uint32_t id = core + node * 4;
104 uint64_t data, offset = addr & 0xff;
105 void *pbuf;
106 struct kvm *kvm = ipi->kvm;
107 struct kvm_mips_interrupt irq;
108 struct ipi_state *s = &(ipi->ipistate[id]);
110 data = *(uint64_t *)val;
111 BUG_ON(offset & (len - 1));
113 switch (offset) {
114 case CORE0_STATUS_OFF:
115 break;
117 case CORE0_EN_OFF:
118 s->en = data;
119 break;
121 case CORE0_SET_OFF:
122 s->status |= data;
123 irq.cpu = id;
124 irq.irq = 6;
125 kvm_vcpu_ioctl_interrupt(kvm_get_vcpu(kvm, id), &irq);
126 break;
128 case CORE0_CLEAR_OFF:
129 s->status &= ~data;
130 if (!s->status) {
131 irq.cpu = id;
132 irq.irq = -6;
133 kvm_vcpu_ioctl_interrupt(kvm_get_vcpu(kvm, id), &irq);
135 break;
137 case CORE0_BUF_20 ... CORE0_BUF_38:
138 pbuf = (void *)s->buf + (offset - 0x20);
139 if (len == 8)
140 *(uint64_t *)pbuf = (uint64_t)data;
141 else /* Assume len == 4 */
142 *(uint32_t *)pbuf = (uint32_t)data;
143 break;
145 default:
146 pr_notice("%s with unknown addr %llx\n", __func__, addr);
147 break;
150 return 0;
153 static int kvm_ipi_read(struct kvm_vcpu *vcpu, struct kvm_io_device *dev,
154 gpa_t addr, int len, void *val)
156 unsigned long flags;
157 struct loongson_kvm_ipi *ipi;
158 struct ipi_io_device *ipi_device;
160 ipi_device = container_of(dev, struct ipi_io_device, device);
161 ipi = ipi_device->ipi;
163 spin_lock_irqsave(&ipi->lock, flags);
164 loongson_vipi_read(ipi, addr, len, val);
165 spin_unlock_irqrestore(&ipi->lock, flags);
167 return 0;
170 static int kvm_ipi_write(struct kvm_vcpu *vcpu, struct kvm_io_device *dev,
171 gpa_t addr, int len, const void *val)
173 unsigned long flags;
174 struct loongson_kvm_ipi *ipi;
175 struct ipi_io_device *ipi_device;
177 ipi_device = container_of(dev, struct ipi_io_device, device);
178 ipi = ipi_device->ipi;
180 spin_lock_irqsave(&ipi->lock, flags);
181 loongson_vipi_write(ipi, addr, len, val);
182 spin_unlock_irqrestore(&ipi->lock, flags);
184 return 0;
187 static const struct kvm_io_device_ops kvm_ipi_ops = {
188 .read = kvm_ipi_read,
189 .write = kvm_ipi_write,
192 void kvm_init_loongson_ipi(struct kvm *kvm)
194 int i;
195 unsigned long addr;
196 struct loongson_kvm_ipi *s;
197 struct kvm_io_device *device;
199 s = &kvm->arch.ipi;
200 s->kvm = kvm;
201 spin_lock_init(&s->lock);
204 * Initialize IPI device
206 for (i = 0; i < 4; i++) {
207 device = &s->dev_ipi[i].device;
208 kvm_iodevice_init(device, &kvm_ipi_ops);
209 addr = (((unsigned long)i) << 44) + IPI_BASE;
210 mutex_lock(&kvm->slots_lock);
211 kvm_io_bus_register_dev(kvm, KVM_MMIO_BUS, addr, 0x400, device);
212 mutex_unlock(&kvm->slots_lock);
213 s->dev_ipi[i].ipi = s;
214 s->dev_ipi[i].node_id = i;