1ae1fae34SPekka Enberg #ifndef KVM__KVM_H 2ae1fae34SPekka Enberg #define KVM__KVM_H 3ae1fae34SPekka Enberg 4af7b0868SMatt Evans #include "kvm/kvm-arch.h" 547621338SSasha Levin #include "kvm/kvm-config.h" 649a8afd1SSasha Levin #include "kvm/util-init.h" 749a8afd1SSasha Levin #include "kvm/kvm.h" 8da8883c1SPekka Enberg 9ae1fae34SPekka Enberg #include <stdbool.h> 103fdf659dSSasha Levin #include <linux/types.h> 11db927775SAlexandru Elisei #include <linux/compiler.h> 12ce79f1caSPekka Enberg #include <time.h> 1382d2f21eSSasha Levin #include <signal.h> 14a4d8c55eSSasha Levin #include <sys/prctl.h> 154095fac8SAndre Przywara #include <limits.h> 16ae1fae34SPekka Enberg 17b1c2516bSSasha Levin #define SIGKVMEXIT (SIGRTMIN + 0) 184298ddadSSasha Levin #define SIGKVMPAUSE (SIGRTMIN + 1) 19e300a5eeSMichael Ellerman #define SIGKVMTASK (SIGRTMIN + 2) 2049e5227dSSasha Levin 21354b198bSAsias He #define KVM_PID_FILE_PATH "/.lkvm/" 2282d65b5eSSasha Levin #define HOME_DIR getenv("HOME") 23ee8b1456SWanlong Gao #define KVM_BINARY_NAME "lkvm" 2482d65b5eSSasha Levin 254095fac8SAndre Przywara #ifndef PAGE_SIZE 26af7b0868SMatt Evans #define PAGE_SIZE (sysconf(_SC_PAGE_SIZE)) 274095fac8SAndre Przywara #endif 28ae1fae34SPekka Enberg 29af7b0868SMatt Evans #define DEFINE_KVM_EXT(ext) \ 30af7b0868SMatt Evans .name = #ext, \ 31af7b0868SMatt Evans .code = ext 320c7c14a7SCyrill Gorcunov 337021c50bSAsias He enum { 347021c50bSAsias He KVM_VMSTATE_RUNNING, 357021c50bSAsias He KVM_VMSTATE_PAUSED, 367021c50bSAsias He }; 377021c50bSAsias He 388f46c736SJean-Philippe Brucker enum kvm_mem_type { 398f46c736SJean-Philippe Brucker KVM_MEM_TYPE_RAM = 1 << 0, 408f46c736SJean-Philippe Brucker KVM_MEM_TYPE_DEVICE = 1 << 1, 41fa1076abSJean-Philippe Brucker KVM_MEM_TYPE_RESERVED = 1 << 2, 428f46c736SJean-Philippe Brucker 438f46c736SJean-Philippe Brucker KVM_MEM_TYPE_ALL = KVM_MEM_TYPE_RAM 448f46c736SJean-Philippe Brucker | KVM_MEM_TYPE_DEVICE 45fa1076abSJean-Philippe Brucker | KVM_MEM_TYPE_RESERVED 468f46c736SJean-Philippe Brucker }; 478f46c736SJean-Philippe Brucker 48af7b0868SMatt Evans struct kvm_ext { 495358b0e6SSasha Levin const char *name; 50af7b0868SMatt Evans int code; 51ae1fae34SPekka Enberg }; 52ae1fae34SPekka Enberg 53d82350d3SWill Deacon struct kvm_mem_bank { 54d82350d3SWill Deacon struct list_head list; 55d82350d3SWill Deacon u64 guest_phys_addr; 56d82350d3SWill Deacon void *host_addr; 57d82350d3SWill Deacon u64 size; 588f46c736SJean-Philippe Brucker enum kvm_mem_type type; 59d82350d3SWill Deacon }; 60d82350d3SWill Deacon 6142ac24f9SSasha Levin struct kvm { 6242ac24f9SSasha Levin struct kvm_arch arch; 6347621338SSasha Levin struct kvm_config cfg; 6442ac24f9SSasha Levin int sys_fd; /* For system ioctls(), i.e. /dev/kvm */ 6542ac24f9SSasha Levin int vm_fd; /* For VM ioctls() */ 6642ac24f9SSasha Levin timer_t timerid; /* Posix timer for interrupts */ 6742ac24f9SSasha Levin 6842ac24f9SSasha Levin int nrcpus; /* Number of cpus to run */ 69df4239fbSSasha Levin struct kvm_cpu **cpus; 7042ac24f9SSasha Levin 7142ac24f9SSasha Levin u32 mem_slots; /* for KVM_SET_USER_MEMORY_REGION */ 7242ac24f9SSasha Levin u64 ram_size; 7342ac24f9SSasha Levin void *ram_start; 7442ac24f9SSasha Levin u64 ram_pagesize; 75d82350d3SWill Deacon struct list_head mem_banks; 7642ac24f9SSasha Levin 7742ac24f9SSasha Levin bool nmi_disabled; 78714ab9e6SAndre Przywara bool msix_needs_devid; 7942ac24f9SSasha Levin 8042ac24f9SSasha Levin const char *vmlinux; 8142ac24f9SSasha Levin struct disk_image **disks; 8242ac24f9SSasha Levin int nr_disks; 8342ac24f9SSasha Levin 8442ac24f9SSasha Levin int vm_state; 8520b65266SJulien Thierry 8620b65266SJulien Thierry #ifdef KVM_BRLOCK_DEBUG 8720b65266SJulien Thierry pthread_rwlock_t brlock_sem; 8820b65266SJulien Thierry #endif 8942ac24f9SSasha Levin }; 9042ac24f9SSasha Levin 919667701cSPekka Enberg void kvm__set_dir(const char *fmt, ...); 929667701cSPekka Enberg const char *kvm__get_dir(void); 939667701cSPekka Enberg 9447621338SSasha Levin int kvm__init(struct kvm *kvm); 9547621338SSasha Levin struct kvm *kvm__new(void); 968259b8ccSSasha Levin int kvm__recommended_cpus(struct kvm *kvm); 9743835ac9SSasha Levin int kvm__max_cpus(struct kvm *kvm); 9843835ac9SSasha Levin void kvm__init_ram(struct kvm *kvm); 99495fbd4eSSasha Levin int kvm__exit(struct kvm *kvm); 1005ad8db5eSPekka Enberg bool kvm__load_firmware(struct kvm *kvm, const char *firmware_filename); 1012065a6f7SCyrill Gorcunov bool kvm__load_kernel(struct kvm *kvm, const char *kernel_filename, 102ff7ba6faSWill Deacon const char *initrd_filename, const char *kernel_cmdline); 103b4532ca9SSasha Levin int kvm_timer__init(struct kvm *kvm); 104b4532ca9SSasha Levin int kvm_timer__exit(struct kvm *kvm); 10543835ac9SSasha Levin void kvm__irq_line(struct kvm *kvm, int irq, int level); 106bfaed61cSSasha Levin void kvm__irq_trigger(struct kvm *kvm, int irq); 1074123ca55SMarc Zyngier bool kvm__emulate_io(struct kvm_cpu *vcpu, u16 port, void *data, int direction, int size, u32 count); 1089b735910SMarc Zyngier bool kvm__emulate_mmio(struct kvm_cpu *vcpu, u64 phys_addr, u8 *data, u32 len, u8 is_write); 1098f46c736SJean-Philippe Brucker int kvm__register_mem(struct kvm *kvm, u64 guest_phys, u64 size, void *userspace_addr, 1108f46c736SJean-Philippe Brucker enum kvm_mem_type type); 1118f46c736SJean-Philippe Brucker static inline int kvm__register_ram(struct kvm *kvm, u64 guest_phys, u64 size, 1128f46c736SJean-Philippe Brucker void *userspace_addr) 1138f46c736SJean-Philippe Brucker { 1148f46c736SJean-Philippe Brucker return kvm__register_mem(kvm, guest_phys, size, userspace_addr, 1158f46c736SJean-Philippe Brucker KVM_MEM_TYPE_RAM); 1168f46c736SJean-Philippe Brucker } 1178f46c736SJean-Philippe Brucker 1188f46c736SJean-Philippe Brucker static inline int kvm__register_dev_mem(struct kvm *kvm, u64 guest_phys, 1198f46c736SJean-Philippe Brucker u64 size, void *userspace_addr) 1208f46c736SJean-Philippe Brucker { 1218f46c736SJean-Philippe Brucker return kvm__register_mem(kvm, guest_phys, size, userspace_addr, 1228f46c736SJean-Philippe Brucker KVM_MEM_TYPE_DEVICE); 1238f46c736SJean-Philippe Brucker } 1248f46c736SJean-Philippe Brucker 125fa1076abSJean-Philippe Brucker static inline int kvm__reserve_mem(struct kvm *kvm, u64 guest_phys, u64 size) 126fa1076abSJean-Philippe Brucker { 127fa1076abSJean-Philippe Brucker return kvm__register_mem(kvm, guest_phys, size, NULL, 128fa1076abSJean-Philippe Brucker KVM_MEM_TYPE_RESERVED); 129fa1076abSJean-Philippe Brucker } 130fa1076abSJean-Philippe Brucker 131*8f160708SAlexandru Elisei int __must_check kvm__register_mmio(struct kvm *kvm, u64 phys_addr, u64 phys_addr_len, bool coalesce, 1329b735910SMarc Zyngier void (*mmio_fn)(struct kvm_cpu *vcpu, u64 addr, u8 *data, u32 len, u8 is_write, void *ptr), 1339aa9d62aSSasha Levin void *ptr); 13473f7e5b3SSasha Levin bool kvm__deregister_mmio(struct kvm *kvm, u64 phys_addr); 1352aa76b26SWill Deacon void kvm__reboot(struct kvm *kvm); 1364346fd8fSSasha Levin void kvm__pause(struct kvm *kvm); 1374346fd8fSSasha Levin void kvm__continue(struct kvm *kvm); 1384298ddadSSasha Levin void kvm__notify_paused(void); 1394b1addaeSSasha Levin int kvm__get_sock_by_instance(const char *name); 140886af5f2SLiming Wang int kvm__enumerate_instances(int (*callback)(const char *name, int pid)); 1414b1addaeSSasha Levin void kvm__remove_socket(const char *name); 142ae1fae34SPekka Enberg 1438e704a7aSMatt Evans void kvm__arch_set_cmdline(char *cmdline, bool video); 1447eff9f49SWanlong Gao void kvm__arch_init(struct kvm *kvm, const char *hugetlbfs_path, u64 ram_size); 145e56e2de7SLai Jiangshan void kvm__arch_delete_ram(struct kvm *kvm); 146f7f9d02bSCyrill Gorcunov int kvm__arch_setup_firmware(struct kvm *kvm); 1471add9f73SSasha Levin int kvm__arch_free_firmware(struct kvm *kvm); 148af7b0868SMatt Evans bool kvm__arch_cpu_supports_vm(void); 14912c406a8SJonathan Austin void kvm__arch_read_term(struct kvm *kvm); 150af7b0868SMatt Evans 151f412251fSWill Deacon void *guest_flat_to_host(struct kvm *kvm, u64 offset); 1520cb41990SWill Deacon u64 host_to_guest_flat(struct kvm *kvm, void *ptr); 153f412251fSWill Deacon 154004f7684SAndre Przywara bool kvm__arch_load_kernel_image(struct kvm *kvm, int fd_kernel, int fd_initrd, 155004f7684SAndre Przywara const char *kernel_cmdline); 156af7b0868SMatt Evans 1578f46c736SJean-Philippe Brucker static inline const char *kvm_mem_type_to_string(enum kvm_mem_type type) 1588f46c736SJean-Philippe Brucker { 1598f46c736SJean-Philippe Brucker switch (type) { 1608f46c736SJean-Philippe Brucker case KVM_MEM_TYPE_ALL: 1618f46c736SJean-Philippe Brucker return "(all)"; 1628f46c736SJean-Philippe Brucker case KVM_MEM_TYPE_RAM: 1638f46c736SJean-Philippe Brucker return "RAM"; 1648f46c736SJean-Philippe Brucker case KVM_MEM_TYPE_DEVICE: 1658f46c736SJean-Philippe Brucker return "device"; 166fa1076abSJean-Philippe Brucker case KVM_MEM_TYPE_RESERVED: 167fa1076abSJean-Philippe Brucker return "reserved"; 1688f46c736SJean-Philippe Brucker } 1698f46c736SJean-Philippe Brucker 1708f46c736SJean-Philippe Brucker return "???"; 1718f46c736SJean-Philippe Brucker } 1728f46c736SJean-Philippe Brucker 1738f46c736SJean-Philippe Brucker int kvm__for_each_mem_bank(struct kvm *kvm, enum kvm_mem_type type, 1748f46c736SJean-Philippe Brucker int (*fun)(struct kvm *kvm, struct kvm_mem_bank *bank, void *data), 1758f46c736SJean-Philippe Brucker void *data); 1768f46c736SJean-Philippe Brucker 177ae1fae34SPekka Enberg /* 178ae1fae34SPekka Enberg * Debugging 179ae1fae34SPekka Enberg */ 180b2cf1e9fSAsias He void kvm__dump_mem(struct kvm *kvm, unsigned long addr, unsigned long size, int debug_fd); 181ae1fae34SPekka Enberg 182ae1fae34SPekka Enberg extern const char *kvm_exit_reasons[]; 183ae1fae34SPekka Enberg 18443835ac9SSasha Levin static inline bool host_ptr_in_ram(struct kvm *kvm, void *p) 1859292f776SCyrill Gorcunov { 18643835ac9SSasha Levin return kvm->ram_start <= p && p < (kvm->ram_start + kvm->ram_size); 1879292f776SCyrill Gorcunov } 1889292f776SCyrill Gorcunov 1891d6fb3f2SSasha Levin bool kvm__supports_extension(struct kvm *kvm, unsigned int extension); 190663165a2SAndre Przywara bool kvm__supports_vm_extension(struct kvm *kvm, unsigned int extension); 1916930e42fSSasha Levin 192a4d8c55eSSasha Levin static inline void kvm__set_thread_name(const char *name) 193a4d8c55eSSasha Levin { 194a4d8c55eSSasha Levin prctl(PR_SET_NAME, name); 195a4d8c55eSSasha Levin } 196a4d8c55eSSasha Levin 197ae1fae34SPekka Enberg #endif /* KVM__KVM_H */ 198