xref: /kvmtool/include/kvm/kvm.h (revision db927775acdbc647b5beeb8e29522d1e3fe39987)
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>
11*db927775SAlexandru 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 
131495fbd4eSSasha Levin int 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