#include #include #include #include #include #include #include #include #include #include #include /* * Compatibility code. Remove this when we move to tools/kvm. */ #ifndef KVM_EXIT_INTERNAL_ERROR # define KVM_EXIT_INTERNAL_ERROR 17 #endif struct kvm { int sys_fd; /* For system ioctls(), i.e. /dev/kvm */ int vm_fd; /* For VM ioctls() */ int vcpu_fd; /* For VCPU ioctls() */ struct kvm_run *kvm_run; uint64_t ram_size; void *ram_start; struct kvm_regs regs; }; static void die_perror(const char *s) { perror(s); exit(1); } static void die(const char *format, ...) { va_list ap; va_start(ap, format); vprintf(format, ap); va_end(ap); printf("\n"); exit(1); } static inline bool kvm__supports_extension(struct kvm *self, unsigned int extension) { int ret; ret = ioctl(self->sys_fd, KVM_CHECK_EXTENSION, extension); if (ret < 0) return false; return ret; } static struct kvm *kvm__new(void) { struct kvm *self = calloc(1, sizeof *self); if (!self) die("out of memory"); return self; } static struct kvm *kvm__init(void) { struct kvm_userspace_memory_region mem; struct kvm *self; long page_size; int mmap_size; int ret; self = kvm__new(); self->sys_fd = open("/dev/kvm", O_RDWR); if (self->sys_fd < 0) die_perror("open"); ret = ioctl(self->sys_fd, KVM_GET_API_VERSION, 0); if (ret != KVM_API_VERSION) die_perror("KVM_API_VERSION ioctl"); self->vm_fd = ioctl(self->sys_fd, KVM_CREATE_VM, 0); if (self->vm_fd < 0) die_perror("KVM_CREATE_VM ioctl"); if (!kvm__supports_extension(self, KVM_CAP_USER_MEMORY)) die("KVM_CAP_USER_MEMORY is not supported"); self->ram_size = 64UL * 1024UL * 1024UL; page_size = sysconf(_SC_PAGESIZE); if (posix_memalign(&self->ram_start, page_size, self->ram_size) != 0) die("out of memory"); mem = (struct kvm_userspace_memory_region) { .slot = 0, .guest_phys_addr = 0x0UL, .memory_size = self->ram_size, .userspace_addr = (unsigned long) self->ram_start, }; ret = ioctl(self->vm_fd, KVM_SET_USER_MEMORY_REGION, &mem, 1); if (ret < 0) die_perror("KVM_SET_USER_MEMORY_REGION ioctl"); if (!kvm__supports_extension(self, KVM_CAP_SET_TSS_ADDR)) die("KVM_CAP_SET_TSS_ADDR is not supported"); ret = ioctl(self->vm_fd, KVM_SET_TSS_ADDR, 0xfffbd000); if (ret < 0) die_perror("KVM_SET_TSS_ADDR ioctl"); self->vcpu_fd = ioctl(self->vm_fd, KVM_CREATE_VCPU, 0); if (self->vcpu_fd < 0) die_perror("KVM_CREATE_VCPU ioctl"); mmap_size = ioctl(self->sys_fd, KVM_GET_VCPU_MMAP_SIZE, 0); if (mmap_size < 0) die_perror("KVM_GET_VCPU_MMAP_SIZE ioctl"); self->kvm_run = mmap(NULL, mmap_size, PROT_READ|PROT_WRITE, MAP_SHARED, self->vcpu_fd, 0); if (self->kvm_run == MAP_FAILED) die("unable to mmap vcpu fd"); return self; } static void kvm__run(struct kvm *self) { if (ioctl(self->vcpu_fd, KVM_SET_REGS, &self->regs) < 0) die_perror("KVM_SET_REGS failed"); if (ioctl(self->vcpu_fd, KVM_RUN, 0) < 0) die_perror("KVM_RUN failed"); } static void kvm__show_registers(struct kvm *self) { unsigned long rax, rbx, rcx; unsigned long rdx, rsi, rdi; unsigned long rbp, r8, r9; unsigned long r10, r11, r12; unsigned long r13, r14, r15; unsigned long rip, rsp; struct kvm_regs regs; if (ioctl(self->vcpu_fd, KVM_GET_REGS, ®s) < 0) die("KVM_GET_REGS failed"); rip = regs.rip; rsp = regs.rsp; rax = regs.rax; rbx = regs.rbx; rcx = regs.rcx; rdx = regs.rdx; rsi = regs.rsi; rdi = regs.rdi; rbp = regs.rbp; r8 = regs.r8; r9 = regs.r9; r10 = regs.r10; r11 = regs.r11; r12 = regs.r12; r13 = regs.r13; r14 = regs.r14; r15 = regs.r15; printf("Registers:\n"); printf(" rip: %016lx rsp: %016lx\n", rip, rsp); printf(" rax: %016lx ebx: %016lx ecx: %016lx\n", rax, rbx, rcx); printf(" rdx: %016lx rsi: %016lx rdi: %016lx\n", rdx, rsi, rdi); printf(" rbp: %016lx r8: %016lx r9: %016lx\n", rbp, r8, r9); printf(" r10: %016lx r11: %016lx r12: %016lx\n", r10, r11, r12); printf(" r13: %016lx r14: %016lx r15: %016lx\n", r13, r14, r15); } static inline void *guest_addr_to_host(struct kvm *self, unsigned long offset) { return self->ram_start + offset; } static void kvm__show_code(struct kvm *self) { unsigned int code_bytes = 64; unsigned int code_prologue = code_bytes * 43 / 64; unsigned int code_len = code_bytes; unsigned char c; uint8_t *ip; int i; ip = guest_addr_to_host(self, self->regs.rip - code_prologue); printf("Code: "); for (i = 0; i < code_len; i++, ip++) { c = *ip; if (ip == guest_addr_to_host(self, self->regs.rip)) printf("<%02x> ", c); else printf("%02x ", c); } printf("\n"); } /* bzImages are loaded at 1 MB by default. */ #define KERNEL_START_ADDR (1024ULL * 1024ULL) static const char *BZIMAGE_MAGIC = "HdrS"; static uint32_t load_bzimage(struct kvm *kvm, int fd) { struct boot_params boot; void *p; int nr; read(fd, &boot, sizeof(boot)); if (memcmp(&boot.hdr.header, BZIMAGE_MAGIC, strlen(BZIMAGE_MAGIC)) != 0) return 0; lseek(fd, (boot.hdr.setup_sects+1) * 512, SEEK_SET); p = guest_addr_to_host(kvm, KERNEL_START_ADDR); while ((nr = read(fd, p, 65536)) > 0) p += nr; return boot.hdr.code32_start; } static uint32_t kvm__load_kernel(struct kvm *kvm, const char *kernel_filename) { uint32_t ret; int fd; fd = open(kernel_filename, O_RDONLY); if (fd < 0) die("unable to open kernel"); ret = load_bzimage(kvm, fd); if (!ret) die("%s is not a valid bzImage", kernel_filename); return ret; } #define DEFINE_KVM_EXIT_REASON(reason) [reason] = #reason static const char *exit_reasons[] = { DEFINE_KVM_EXIT_REASON(KVM_EXIT_UNKNOWN), DEFINE_KVM_EXIT_REASON(KVM_EXIT_EXCEPTION), DEFINE_KVM_EXIT_REASON(KVM_EXIT_IO), DEFINE_KVM_EXIT_REASON(KVM_EXIT_HYPERCALL), DEFINE_KVM_EXIT_REASON(KVM_EXIT_DEBUG), DEFINE_KVM_EXIT_REASON(KVM_EXIT_HLT), DEFINE_KVM_EXIT_REASON(KVM_EXIT_MMIO), DEFINE_KVM_EXIT_REASON(KVM_EXIT_IRQ_WINDOW_OPEN), DEFINE_KVM_EXIT_REASON(KVM_EXIT_SHUTDOWN), DEFINE_KVM_EXIT_REASON(KVM_EXIT_FAIL_ENTRY), DEFINE_KVM_EXIT_REASON(KVM_EXIT_INTR), DEFINE_KVM_EXIT_REASON(KVM_EXIT_SET_TPR), DEFINE_KVM_EXIT_REASON(KVM_EXIT_TPR_ACCESS), DEFINE_KVM_EXIT_REASON(KVM_EXIT_S390_SIEIC), DEFINE_KVM_EXIT_REASON(KVM_EXIT_S390_RESET), DEFINE_KVM_EXIT_REASON(KVM_EXIT_DCR), DEFINE_KVM_EXIT_REASON(KVM_EXIT_NMI), DEFINE_KVM_EXIT_REASON(KVM_EXIT_INTERNAL_ERROR), }; static void usage(char *argv[]) { fprintf(stderr, " usage: %s \n", argv[0]); exit(1); } int main(int argc, char *argv[]) { const char *kernel_filename; struct kvm *kvm; int ret; if (argc < 2) usage(argv); kernel_filename = argv[1]; kvm = kvm__init(); kvm->regs.rip = kvm__load_kernel(kvm, kernel_filename); kvm__run(kvm); fprintf(stderr, "KVM exit reason: %" PRIu32 " (\"%s\")\n", kvm->kvm_run->exit_reason, exit_reasons[kvm->kvm_run->exit_reason]); kvm__show_registers(kvm); kvm__show_code(kvm); return 0; }