xref: /openbmc/qemu/hw/hyperv/hyperv.c (revision 606c34bfd57a0ecda67b395bea022bb307a5384e)
1701189e3SRoman Kagan /*
2701189e3SRoman Kagan  * Hyper-V guest/hypervisor interaction
3701189e3SRoman Kagan  *
4701189e3SRoman Kagan  * Copyright (c) 2015-2018 Virtuozzo International GmbH.
5701189e3SRoman Kagan  *
6701189e3SRoman Kagan  * This work is licensed under the terms of the GNU GPL, version 2 or later.
7701189e3SRoman Kagan  * See the COPYING file in the top-level directory.
8701189e3SRoman Kagan  */
9701189e3SRoman Kagan 
10701189e3SRoman Kagan #include "qemu/osdep.h"
11701189e3SRoman Kagan #include "qemu/main-loop.h"
12*606c34bfSRoman Kagan #include "qapi/error.h"
13701189e3SRoman Kagan #include "sysemu/kvm.h"
14701189e3SRoman Kagan #include "hw/hyperv/hyperv.h"
15701189e3SRoman Kagan 
16*606c34bfSRoman Kagan typedef struct SynICState {
17*606c34bfSRoman Kagan     DeviceState parent_obj;
18*606c34bfSRoman Kagan 
19*606c34bfSRoman Kagan     CPUState *cs;
20*606c34bfSRoman Kagan 
21*606c34bfSRoman Kagan     bool enabled;
22*606c34bfSRoman Kagan     hwaddr msg_page_addr;
23*606c34bfSRoman Kagan     hwaddr event_page_addr;
24*606c34bfSRoman Kagan } SynICState;
25*606c34bfSRoman Kagan 
26*606c34bfSRoman Kagan #define TYPE_SYNIC "hyperv-synic"
27*606c34bfSRoman Kagan #define SYNIC(obj) OBJECT_CHECK(SynICState, (obj), TYPE_SYNIC)
28*606c34bfSRoman Kagan 
29*606c34bfSRoman Kagan static SynICState *get_synic(CPUState *cs)
30*606c34bfSRoman Kagan {
31*606c34bfSRoman Kagan     return SYNIC(object_resolve_path_component(OBJECT(cs), "synic"));
32*606c34bfSRoman Kagan }
33*606c34bfSRoman Kagan 
34*606c34bfSRoman Kagan static void synic_update(SynICState *synic, bool enable,
35*606c34bfSRoman Kagan                          hwaddr msg_page_addr, hwaddr event_page_addr)
36*606c34bfSRoman Kagan {
37*606c34bfSRoman Kagan 
38*606c34bfSRoman Kagan     synic->enabled = enable;
39*606c34bfSRoman Kagan     synic->msg_page_addr = msg_page_addr;
40*606c34bfSRoman Kagan     synic->event_page_addr = event_page_addr;
41*606c34bfSRoman Kagan }
42*606c34bfSRoman Kagan 
43*606c34bfSRoman Kagan void hyperv_synic_update(CPUState *cs, bool enable,
44*606c34bfSRoman Kagan                          hwaddr msg_page_addr, hwaddr event_page_addr)
45*606c34bfSRoman Kagan {
46*606c34bfSRoman Kagan     SynICState *synic = get_synic(cs);
47*606c34bfSRoman Kagan 
48*606c34bfSRoman Kagan     if (!synic) {
49*606c34bfSRoman Kagan         return;
50*606c34bfSRoman Kagan     }
51*606c34bfSRoman Kagan 
52*606c34bfSRoman Kagan     synic_update(synic, enable, msg_page_addr, event_page_addr);
53*606c34bfSRoman Kagan }
54*606c34bfSRoman Kagan 
55*606c34bfSRoman Kagan static void synic_realize(DeviceState *dev, Error **errp)
56*606c34bfSRoman Kagan {
57*606c34bfSRoman Kagan }
58*606c34bfSRoman Kagan 
59*606c34bfSRoman Kagan static void synic_reset(DeviceState *dev)
60*606c34bfSRoman Kagan {
61*606c34bfSRoman Kagan     SynICState *synic = SYNIC(dev);
62*606c34bfSRoman Kagan     synic_update(synic, false, 0, 0);
63*606c34bfSRoman Kagan }
64*606c34bfSRoman Kagan 
65*606c34bfSRoman Kagan static void synic_class_init(ObjectClass *klass, void *data)
66*606c34bfSRoman Kagan {
67*606c34bfSRoman Kagan     DeviceClass *dc = DEVICE_CLASS(klass);
68*606c34bfSRoman Kagan 
69*606c34bfSRoman Kagan     dc->realize = synic_realize;
70*606c34bfSRoman Kagan     dc->reset = synic_reset;
71*606c34bfSRoman Kagan     dc->user_creatable = false;
72*606c34bfSRoman Kagan }
73*606c34bfSRoman Kagan 
74*606c34bfSRoman Kagan void hyperv_synic_add(CPUState *cs)
75*606c34bfSRoman Kagan {
76*606c34bfSRoman Kagan     Object *obj;
77*606c34bfSRoman Kagan     SynICState *synic;
78*606c34bfSRoman Kagan 
79*606c34bfSRoman Kagan     obj = object_new(TYPE_SYNIC);
80*606c34bfSRoman Kagan     synic = SYNIC(obj);
81*606c34bfSRoman Kagan     synic->cs = cs;
82*606c34bfSRoman Kagan     object_property_add_child(OBJECT(cs), "synic", obj, &error_abort);
83*606c34bfSRoman Kagan     object_unref(obj);
84*606c34bfSRoman Kagan     object_property_set_bool(obj, true, "realized", &error_abort);
85*606c34bfSRoman Kagan }
86*606c34bfSRoman Kagan 
87*606c34bfSRoman Kagan void hyperv_synic_reset(CPUState *cs)
88*606c34bfSRoman Kagan {
89*606c34bfSRoman Kagan     device_reset(DEVICE(get_synic(cs)));
90*606c34bfSRoman Kagan }
91*606c34bfSRoman Kagan 
92*606c34bfSRoman Kagan static const TypeInfo synic_type_info = {
93*606c34bfSRoman Kagan     .name = TYPE_SYNIC,
94*606c34bfSRoman Kagan     .parent = TYPE_DEVICE,
95*606c34bfSRoman Kagan     .instance_size = sizeof(SynICState),
96*606c34bfSRoman Kagan     .class_init = synic_class_init,
97*606c34bfSRoman Kagan };
98*606c34bfSRoman Kagan 
99*606c34bfSRoman Kagan static void synic_register_types(void)
100*606c34bfSRoman Kagan {
101*606c34bfSRoman Kagan     type_register_static(&synic_type_info);
102*606c34bfSRoman Kagan }
103*606c34bfSRoman Kagan 
104*606c34bfSRoman Kagan type_init(synic_register_types)
105*606c34bfSRoman Kagan 
106701189e3SRoman Kagan struct HvSintRoute {
107701189e3SRoman Kagan     uint32_t sint;
108*606c34bfSRoman Kagan     SynICState *synic;
109701189e3SRoman Kagan     int gsi;
110701189e3SRoman Kagan     EventNotifier sint_set_notifier;
111701189e3SRoman Kagan     EventNotifier sint_ack_notifier;
112701189e3SRoman Kagan     HvSintAckClb sint_ack_clb;
113701189e3SRoman Kagan     void *sint_ack_clb_data;
114701189e3SRoman Kagan     unsigned refcount;
115701189e3SRoman Kagan };
116701189e3SRoman Kagan 
117701189e3SRoman Kagan static CPUState *hyperv_find_vcpu(uint32_t vp_index)
118701189e3SRoman Kagan {
119701189e3SRoman Kagan     CPUState *cs = qemu_get_cpu(vp_index);
120701189e3SRoman Kagan     assert(hyperv_vp_index(cs) == vp_index);
121701189e3SRoman Kagan     return cs;
122701189e3SRoman Kagan }
123701189e3SRoman Kagan 
124701189e3SRoman Kagan static void kvm_hv_sint_ack_handler(EventNotifier *notifier)
125701189e3SRoman Kagan {
126701189e3SRoman Kagan     HvSintRoute *sint_route = container_of(notifier, HvSintRoute,
127701189e3SRoman Kagan                                            sint_ack_notifier);
128701189e3SRoman Kagan     event_notifier_test_and_clear(notifier);
129701189e3SRoman Kagan     sint_route->sint_ack_clb(sint_route->sint_ack_clb_data);
130701189e3SRoman Kagan }
131701189e3SRoman Kagan 
132701189e3SRoman Kagan HvSintRoute *hyperv_sint_route_new(uint32_t vp_index, uint32_t sint,
133701189e3SRoman Kagan                                    HvSintAckClb sint_ack_clb,
134701189e3SRoman Kagan                                    void *sint_ack_clb_data)
135701189e3SRoman Kagan {
136701189e3SRoman Kagan     HvSintRoute *sint_route;
137701189e3SRoman Kagan     EventNotifier *ack_notifier;
138701189e3SRoman Kagan     int r, gsi;
139701189e3SRoman Kagan     CPUState *cs;
140*606c34bfSRoman Kagan     SynICState *synic;
141701189e3SRoman Kagan 
142701189e3SRoman Kagan     cs = hyperv_find_vcpu(vp_index);
143701189e3SRoman Kagan     if (!cs) {
144701189e3SRoman Kagan         return NULL;
145701189e3SRoman Kagan     }
146701189e3SRoman Kagan 
147*606c34bfSRoman Kagan     synic = get_synic(cs);
148*606c34bfSRoman Kagan     if (!synic) {
149*606c34bfSRoman Kagan         return NULL;
150*606c34bfSRoman Kagan     }
151*606c34bfSRoman Kagan 
152701189e3SRoman Kagan     sint_route = g_new0(HvSintRoute, 1);
153701189e3SRoman Kagan     r = event_notifier_init(&sint_route->sint_set_notifier, false);
154701189e3SRoman Kagan     if (r) {
155701189e3SRoman Kagan         goto err;
156701189e3SRoman Kagan     }
157701189e3SRoman Kagan 
158701189e3SRoman Kagan     ack_notifier = sint_ack_clb ? &sint_route->sint_ack_notifier : NULL;
159701189e3SRoman Kagan     if (ack_notifier) {
160701189e3SRoman Kagan         r = event_notifier_init(ack_notifier, false);
161701189e3SRoman Kagan         if (r) {
162701189e3SRoman Kagan             goto err_sint_set_notifier;
163701189e3SRoman Kagan         }
164701189e3SRoman Kagan 
165701189e3SRoman Kagan         event_notifier_set_handler(ack_notifier, kvm_hv_sint_ack_handler);
166701189e3SRoman Kagan     }
167701189e3SRoman Kagan 
168701189e3SRoman Kagan     gsi = kvm_irqchip_add_hv_sint_route(kvm_state, vp_index, sint);
169701189e3SRoman Kagan     if (gsi < 0) {
170701189e3SRoman Kagan         goto err_gsi;
171701189e3SRoman Kagan     }
172701189e3SRoman Kagan 
173701189e3SRoman Kagan     r = kvm_irqchip_add_irqfd_notifier_gsi(kvm_state,
174701189e3SRoman Kagan                                            &sint_route->sint_set_notifier,
175701189e3SRoman Kagan                                            ack_notifier, gsi);
176701189e3SRoman Kagan     if (r) {
177701189e3SRoman Kagan         goto err_irqfd;
178701189e3SRoman Kagan     }
179701189e3SRoman Kagan     sint_route->gsi = gsi;
180701189e3SRoman Kagan     sint_route->sint_ack_clb = sint_ack_clb;
181701189e3SRoman Kagan     sint_route->sint_ack_clb_data = sint_ack_clb_data;
182*606c34bfSRoman Kagan     sint_route->synic = synic;
183701189e3SRoman Kagan     sint_route->sint = sint;
184701189e3SRoman Kagan     sint_route->refcount = 1;
185701189e3SRoman Kagan 
186701189e3SRoman Kagan     return sint_route;
187701189e3SRoman Kagan 
188701189e3SRoman Kagan err_irqfd:
189701189e3SRoman Kagan     kvm_irqchip_release_virq(kvm_state, gsi);
190701189e3SRoman Kagan err_gsi:
191701189e3SRoman Kagan     if (ack_notifier) {
192701189e3SRoman Kagan         event_notifier_set_handler(ack_notifier, NULL);
193701189e3SRoman Kagan         event_notifier_cleanup(ack_notifier);
194701189e3SRoman Kagan     }
195701189e3SRoman Kagan err_sint_set_notifier:
196701189e3SRoman Kagan     event_notifier_cleanup(&sint_route->sint_set_notifier);
197701189e3SRoman Kagan err:
198701189e3SRoman Kagan     g_free(sint_route);
199701189e3SRoman Kagan 
200701189e3SRoman Kagan     return NULL;
201701189e3SRoman Kagan }
202701189e3SRoman Kagan 
203701189e3SRoman Kagan void hyperv_sint_route_ref(HvSintRoute *sint_route)
204701189e3SRoman Kagan {
205701189e3SRoman Kagan     sint_route->refcount++;
206701189e3SRoman Kagan }
207701189e3SRoman Kagan 
208701189e3SRoman Kagan void hyperv_sint_route_unref(HvSintRoute *sint_route)
209701189e3SRoman Kagan {
210701189e3SRoman Kagan     if (!sint_route) {
211701189e3SRoman Kagan         return;
212701189e3SRoman Kagan     }
213701189e3SRoman Kagan 
214701189e3SRoman Kagan     assert(sint_route->refcount > 0);
215701189e3SRoman Kagan 
216701189e3SRoman Kagan     if (--sint_route->refcount) {
217701189e3SRoman Kagan         return;
218701189e3SRoman Kagan     }
219701189e3SRoman Kagan 
220701189e3SRoman Kagan     kvm_irqchip_remove_irqfd_notifier_gsi(kvm_state,
221701189e3SRoman Kagan                                           &sint_route->sint_set_notifier,
222701189e3SRoman Kagan                                           sint_route->gsi);
223701189e3SRoman Kagan     kvm_irqchip_release_virq(kvm_state, sint_route->gsi);
224701189e3SRoman Kagan     if (sint_route->sint_ack_clb) {
225701189e3SRoman Kagan         event_notifier_set_handler(&sint_route->sint_ack_notifier, NULL);
226701189e3SRoman Kagan         event_notifier_cleanup(&sint_route->sint_ack_notifier);
227701189e3SRoman Kagan     }
228701189e3SRoman Kagan     event_notifier_cleanup(&sint_route->sint_set_notifier);
229701189e3SRoman Kagan     g_free(sint_route);
230701189e3SRoman Kagan }
231701189e3SRoman Kagan 
232701189e3SRoman Kagan int hyperv_sint_route_set_sint(HvSintRoute *sint_route)
233701189e3SRoman Kagan {
234701189e3SRoman Kagan     return event_notifier_set(&sint_route->sint_set_notifier);
235701189e3SRoman Kagan }
236