13f0e7e57SJagannathan Raman /* 23f0e7e57SJagannathan Raman * Machine for remote device 33f0e7e57SJagannathan Raman * 43f0e7e57SJagannathan Raman * This machine type is used by the remote device process in multi-process 53f0e7e57SJagannathan Raman * QEMU. QEMU device models depend on parent busses, interrupt controllers, 63f0e7e57SJagannathan Raman * memory regions, etc. The remote machine type offers this environment so 73f0e7e57SJagannathan Raman * that QEMU device models can be used as remote devices. 83f0e7e57SJagannathan Raman * 93f0e7e57SJagannathan Raman * Copyright © 2018, 2021 Oracle and/or its affiliates. 103f0e7e57SJagannathan Raman * 113f0e7e57SJagannathan Raman * This work is licensed under the terms of the GNU GPL, version 2 or later. 123f0e7e57SJagannathan Raman * See the COPYING file in the top-level directory. 133f0e7e57SJagannathan Raman * 143f0e7e57SJagannathan Raman */ 153f0e7e57SJagannathan Raman 163f0e7e57SJagannathan Raman #include "qemu/osdep.h" 173f0e7e57SJagannathan Raman 183f0e7e57SJagannathan Raman #include "hw/remote/machine.h" 193f0e7e57SJagannathan Raman #include "exec/memory.h" 203f0e7e57SJagannathan Raman #include "qapi/error.h" 21bd36adb8SJagannathan Raman #include "hw/pci/pci_host.h" 22bd36adb8SJagannathan Raman #include "hw/remote/iohub.h" 23661e21c4SJagannathan Raman #include "hw/qdev-core.h" 243f0e7e57SJagannathan Raman 253f0e7e57SJagannathan Raman static void remote_machine_init(MachineState *machine) 263f0e7e57SJagannathan Raman { 273f0e7e57SJagannathan Raman MemoryRegion *system_memory, *system_io, *pci_memory; 283f0e7e57SJagannathan Raman RemoteMachineState *s = REMOTE_MACHINE(machine); 293f0e7e57SJagannathan Raman RemotePCIHost *rem_host; 30bd36adb8SJagannathan Raman PCIHostState *pci_host; 313f0e7e57SJagannathan Raman 323f0e7e57SJagannathan Raman system_memory = get_system_memory(); 333f0e7e57SJagannathan Raman system_io = get_system_io(); 343f0e7e57SJagannathan Raman 353f0e7e57SJagannathan Raman pci_memory = g_new(MemoryRegion, 1); 363f0e7e57SJagannathan Raman memory_region_init(pci_memory, NULL, "pci", UINT64_MAX); 373f0e7e57SJagannathan Raman 383f0e7e57SJagannathan Raman rem_host = REMOTE_PCIHOST(qdev_new(TYPE_REMOTE_PCIHOST)); 393f0e7e57SJagannathan Raman 403f0e7e57SJagannathan Raman rem_host->mr_pci_mem = pci_memory; 413f0e7e57SJagannathan Raman rem_host->mr_sys_mem = system_memory; 423f0e7e57SJagannathan Raman rem_host->mr_sys_io = system_io; 433f0e7e57SJagannathan Raman 443f0e7e57SJagannathan Raman s->host = rem_host; 453f0e7e57SJagannathan Raman 463f0e7e57SJagannathan Raman object_property_add_child(OBJECT(s), "remote-pcihost", OBJECT(rem_host)); 473f0e7e57SJagannathan Raman memory_region_add_subregion_overlap(system_memory, 0x0, pci_memory, -1); 483f0e7e57SJagannathan Raman 493f0e7e57SJagannathan Raman qdev_realize(DEVICE(rem_host), sysbus_get_default(), &error_fatal); 50bd36adb8SJagannathan Raman 51bd36adb8SJagannathan Raman pci_host = PCI_HOST_BRIDGE(rem_host); 52bd36adb8SJagannathan Raman 53bd36adb8SJagannathan Raman remote_iohub_init(&s->iohub); 54bd36adb8SJagannathan Raman 55bd36adb8SJagannathan Raman pci_bus_irqs(pci_host->bus, remote_iohub_set_irq, remote_iohub_map_irq, 56bd36adb8SJagannathan Raman &s->iohub, REMOTE_IOHUB_NB_PIRQS); 57661e21c4SJagannathan Raman 58661e21c4SJagannathan Raman qbus_set_hotplug_handler(BUS(pci_host->bus), OBJECT(s)); 593f0e7e57SJagannathan Raman } 603f0e7e57SJagannathan Raman 61*9b5b473eSJagannathan Raman static bool remote_machine_get_vfio_user(Object *obj, Error **errp) 62*9b5b473eSJagannathan Raman { 63*9b5b473eSJagannathan Raman RemoteMachineState *s = REMOTE_MACHINE(obj); 64*9b5b473eSJagannathan Raman 65*9b5b473eSJagannathan Raman return s->vfio_user; 66*9b5b473eSJagannathan Raman } 67*9b5b473eSJagannathan Raman 68*9b5b473eSJagannathan Raman static void remote_machine_set_vfio_user(Object *obj, bool value, Error **errp) 69*9b5b473eSJagannathan Raman { 70*9b5b473eSJagannathan Raman RemoteMachineState *s = REMOTE_MACHINE(obj); 71*9b5b473eSJagannathan Raman 72*9b5b473eSJagannathan Raman if (phase_check(PHASE_MACHINE_CREATED)) { 73*9b5b473eSJagannathan Raman error_setg(errp, "Error enabling vfio-user - machine already created"); 74*9b5b473eSJagannathan Raman return; 75*9b5b473eSJagannathan Raman } 76*9b5b473eSJagannathan Raman 77*9b5b473eSJagannathan Raman s->vfio_user = value; 78*9b5b473eSJagannathan Raman } 79*9b5b473eSJagannathan Raman 803f0e7e57SJagannathan Raman static void remote_machine_class_init(ObjectClass *oc, void *data) 813f0e7e57SJagannathan Raman { 823f0e7e57SJagannathan Raman MachineClass *mc = MACHINE_CLASS(oc); 83661e21c4SJagannathan Raman HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(oc); 843f0e7e57SJagannathan Raman 853f0e7e57SJagannathan Raman mc->init = remote_machine_init; 863f0e7e57SJagannathan Raman mc->desc = "Experimental remote machine"; 87661e21c4SJagannathan Raman 88661e21c4SJagannathan Raman hc->unplug = qdev_simple_device_unplug_cb; 89*9b5b473eSJagannathan Raman 90*9b5b473eSJagannathan Raman object_class_property_add_bool(oc, "vfio-user", 91*9b5b473eSJagannathan Raman remote_machine_get_vfio_user, 92*9b5b473eSJagannathan Raman remote_machine_set_vfio_user); 933f0e7e57SJagannathan Raman } 943f0e7e57SJagannathan Raman 953f0e7e57SJagannathan Raman static const TypeInfo remote_machine = { 963f0e7e57SJagannathan Raman .name = TYPE_REMOTE_MACHINE, 973f0e7e57SJagannathan Raman .parent = TYPE_MACHINE, 983f0e7e57SJagannathan Raman .instance_size = sizeof(RemoteMachineState), 993f0e7e57SJagannathan Raman .class_init = remote_machine_class_init, 100661e21c4SJagannathan Raman .interfaces = (InterfaceInfo[]) { 101661e21c4SJagannathan Raman { TYPE_HOTPLUG_HANDLER }, 102661e21c4SJagannathan Raman { } 103661e21c4SJagannathan Raman } 1043f0e7e57SJagannathan Raman }; 1053f0e7e57SJagannathan Raman 1063f0e7e57SJagannathan Raman static void remote_machine_register_types(void) 1073f0e7e57SJagannathan Raman { 1083f0e7e57SJagannathan Raman type_register_static(&remote_machine); 1093f0e7e57SJagannathan Raman } 1103f0e7e57SJagannathan Raman 1113f0e7e57SJagannathan Raman type_init(remote_machine_register_types); 112