123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540 |
- #include <stdint.h>
- #include <stdio.h>
- #include <string.h>
- #include <sys/ioctl.h>
- #include <sys/mman.h>
- #include <fcntl.h>
- typedef __signed__ char __s8;
- typedef unsigned char __u8;
- typedef __signed__ short __s16;
- typedef unsigned short __u16;
- typedef __signed__ int __s32;
- typedef unsigned int __u32;
- #ifdef __GNUC__
- __extension__ typedef __signed__ long long __s64;
- __extension__ typedef unsigned long long __u64;
- #else
- typedef __signed__ long long __s64;
- typedef unsigned long long __u64;
- #endif
- #define KVM_CREATE_VM _IO(KVMIO, 0x01)
- #define KVM_CREATE_VCPU _IO(KVMIO, 0x41)
- #define KVM_GET_VCPU_MMAP_SIZE _IO(KVMIO, 0x04)
- #define KVM_RUN _IO(KVMIO, 0x80)
- #define KVM_GET_REGS _IOR(KVMIO, 0x81, struct kvm_regs)
- #define KVM_SET_REGS _IOW(KVMIO, 0x82, struct kvm_regs)
- #define KVM_GET_SREGS _IOR(KVMIO, 0x83, struct kvm_sregs)
- #define KVM_SET_SREGS _IOW(KVMIO, 0x84, struct kvm_sregs)
- #define KVMIO 0xAE
- #define KVM_SET_USER_MEMORY_REGION _IOW(KVMIO, 0x46, \
- struct kvm_userspace_memory_region)
- #define KVM_NR_INTERRUPTS 256
- struct kvm_hyperv_exit {
- #define KVM_EXIT_HYPERV_SYNIC 1
- #define KVM_EXIT_HYPERV_HCALL 2
- #define KVM_EXIT_HYPERV_SYNDBG 3
- __u32 type;
- __u32 pad1;
- union {
- struct {
- __u32 msr;
- __u32 pad2;
- __u64 control;
- __u64 evt_page;
- __u64 msg_page;
- } synic;
- struct {
- __u64 input;
- __u64 result;
- __u64 params[2];
- } hcall;
- struct {
- __u32 msr;
- __u32 pad2;
- __u64 control;
- __u64 status;
- __u64 send_page;
- __u64 recv_page;
- __u64 pending_page;
- } syndbg;
- } u;
- };
- struct kvm_debug_exit_arch {
- __u32 exception;
- __u32 pad;
- __u64 pc;
- __u64 dr6;
- __u64 dr7;
- };
- struct kvm_userspace_memory_region {
- __u32 slot;
- __u32 flags;
- __u64 guest_phys_addr;
- __u64 memory_size;
- __u64 userspace_addr;
- };
- struct kvm_xen_exit {
- #define KVM_EXIT_XEN_HCALL 1
- __u32 type;
- union {
- struct {
- __u32 longmode;
- __u32 cpl;
- __u64 input;
- __u64 result;
- __u64 params[6];
- } hcall;
- } u;
- };
- struct kvm_regs {
-
- __u64 rax, rbx, rcx, rdx;
- __u64 rsi, rdi, rsp, rbp;
- __u64 r8, r9, r10, r11;
- __u64 r12, r13, r14, r15;
- __u64 rip, rflags;
- };
- struct my_kvm_segment {
- __u64 base;
- __u32 limit;
- __u16 selector;
- __u8 type;
- __u8 present, dpl, db, s, l, g, avl;
- __u8 unusable;
- __u8 padding;
- };
- struct kvm_dtable {
- __u64 base;
- __u16 limit;
- __u16 padding[3];
- };
- struct kvm_sregs {
-
- struct my_kvm_segment cs, ds, es, fs, gs, ss;
- struct my_kvm_segment tr, ldt;
- struct kvm_dtable gdt, idt;
- __u64 cr0, cr2, cr3, cr4, cr8;
- __u64 efer;
- __u64 apic_base;
- __u64 interrupt_bitmap[(KVM_NR_INTERRUPTS + 63) / 64];
- };
- struct kvm_vcpu_events {
- struct {
- __u8 injected;
- __u8 nr;
- __u8 has_error_code;
- __u8 pending;
- __u32 error_code;
- } exception;
- struct {
- __u8 injected;
- __u8 nr;
- __u8 soft;
- __u8 shadow;
- } interrupt;
- struct {
- __u8 injected;
- __u8 pending;
- __u8 masked;
- __u8 pad;
- } nmi;
- __u32 sipi_vector;
- __u32 flags;
- struct {
- __u8 smm;
- __u8 pending;
- __u8 smm_inside_nmi;
- __u8 latched_init;
- } smi;
- __u8 reserved[27];
- __u8 exception_has_payload;
- __u64 exception_payload;
- };
- struct kvm_sync_regs {
-
- struct kvm_regs regs;
- struct kvm_sregs sregs;
- struct kvm_vcpu_events events;
- };
- struct kvm_run {
-
- __u8 request_interrupt_window;
- __u8 immediate_exit;
- __u8 padding1[6];
-
- __u32 exit_reason;
- __u8 ready_for_interrupt_injection;
- __u8 if_flag;
- __u16 flags;
-
- __u64 cr8;
- __u64 apic_base;
- #ifdef __KVM_S390
-
- __u64 psw_mask;
- __u64 psw_addr;
- #endif
- union {
-
- struct {
- __u64 hardware_exit_reason;
- } hw;
-
- struct {
- __u64 hardware_entry_failure_reason;
- __u32 cpu;
- } fail_entry;
-
- struct {
- __u32 exception;
- __u32 error_code;
- } ex;
-
- struct {
- #define KVM_EXIT_IO_IN 0
- #define KVM_EXIT_IO_OUT 1
- __u8 direction;
- __u8 size;
- __u16 port;
- __u32 count;
- __u64 data_offset;
- } io;
-
- struct {
- struct kvm_debug_exit_arch arch;
- } debug;
-
- struct {
- __u64 phys_addr;
- __u8 data[8];
- __u32 len;
- __u8 is_write;
- } mmio;
-
- struct {
- __u64 nr;
- __u64 args[6];
- __u64 ret;
- __u32 longmode;
- __u32 pad;
- } hypercall;
-
- struct {
- __u64 rip;
- __u32 is_write;
- __u32 pad;
- } tpr_access;
-
- struct {
- __u8 icptcode;
- __u16 ipa;
- __u32 ipb;
- } s390_sieic;
-
- #define KVM_S390_RESET_POR 1
- #define KVM_S390_RESET_CLEAR 2
- #define KVM_S390_RESET_SUBSYSTEM 4
- #define KVM_S390_RESET_CPU_INIT 8
- #define KVM_S390_RESET_IPL 16
- __u64 s390_reset_flags;
-
- struct {
- __u64 trans_exc_code;
- __u32 pgm_code;
- } s390_ucontrol;
-
- struct {
- __u32 dcrn;
- __u32 data;
- __u8 is_write;
- } dcr;
-
- struct {
- __u32 suberror;
-
- __u32 ndata;
- __u64 data[16];
- } internal;
-
- struct {
- __u32 suberror;
- __u32 ndata;
- __u64 flags;
- __u8 insn_size;
- __u8 insn_bytes[15];
- } emulation_failure;
-
- struct {
- __u64 gprs[32];
- } osi;
-
- struct {
- __u64 nr;
- __u64 ret;
- __u64 args[9];
- } papr_hcall;
-
- struct {
- __u16 subchannel_id;
- __u16 subchannel_nr;
- __u32 io_int_parm;
- __u32 io_int_word;
- __u32 ipb;
- __u8 dequeued;
- } s390_tsch;
-
- struct {
- __u32 epr;
- } epr;
-
- struct {
- #define KVM_SYSTEM_EVENT_SHUTDOWN 1
- #define KVM_SYSTEM_EVENT_RESET 2
- #define KVM_SYSTEM_EVENT_CRASH 3
- __u32 type;
- __u64 flags;
- } system_event;
-
- struct {
- __u64 addr;
- __u8 ar;
- __u8 reserved;
- __u8 fc;
- __u8 sel1;
- __u16 sel2;
- } s390_stsi;
-
- struct {
- __u8 vector;
- } eoi;
-
- struct kvm_hyperv_exit hyperv;
-
- struct {
- __u64 esr_iss;
- __u64 fault_ipa;
- } arm_nisv;
-
- struct {
- __u8 error;
- __u8 pad[7];
- #define KVM_MSR_EXIT_REASON_INVAL (1 << 0)
- #define KVM_MSR_EXIT_REASON_UNKNOWN (1 << 1)
- #define KVM_MSR_EXIT_REASON_FILTER (1 << 2)
- __u32 reason;
- __u32 index;
- __u64 data;
- } msr;
-
- struct kvm_xen_exit xen;
-
- char padding[256];
- };
-
- #define SYNC_REGS_SIZE_BYTES 2048
-
- __u64 kvm_valid_regs;
- __u64 kvm_dirty_regs;
- union {
- struct kvm_sync_regs regs;
- char padding[SYNC_REGS_SIZE_BYTES];
- } s;
- };
- int kvm(uint8_t code[], size_t code_len)
- {
-
- int kvmfd = open("/dev/kvm", O_RDWR | O_CLOEXEC);
- if (kvmfd == -1)
- {
- printf("failed to open /dev/kvm\n");
- return 0;
- }
-
- int vmfd = ioctl(kvmfd, KVM_CREATE_VM, 0);
- printf("vmfd %d\n", vmfd);
-
- size_t mem_size = 0x100000;
- void *mem = mmap(0, mem_size, PROT_READ | PROT_WRITE,
- MAP_SHARED | MAP_ANONYMOUS, -1, 0);
- printf("map mem %p\n", mem);
- int user_entry = 0x0;
- memcpy((void *)((size_t)mem + user_entry), code, code_len);
- struct kvm_userspace_memory_region region = {
- .slot = 0,
- .flags = 0,
- .guest_phys_addr = 0,
- .memory_size = mem_size,
- .userspace_addr = (size_t)mem};
- ioctl(vmfd, KVM_SET_USER_MEMORY_REGION, ®ion);
-
-
- int vcpufd = ioctl(vmfd, KVM_CREATE_VCPU, 0);
- printf("create vcpu,fd: %p\n", vcpufd);
-
- size_t vcpu_mmap_size = ioctl(kvmfd, KVM_GET_VCPU_MMAP_SIZE, NULL);
- struct kvm_run *run = (struct kvm_run *)mmap(0, vcpu_mmap_size, PROT_READ | PROT_WRITE, MAP_SHARED, vcpufd, 0);
-
-
- struct kvm_regs regs;
- ioctl(vcpufd, KVM_GET_REGS, ®s);
- regs.rip = user_entry;
- regs.rsp = 0x200000;
- regs.rflags = 0x2;
- ioctl(vcpufd, KVM_SET_REGS, ®s);
-
- struct kvm_sregs sregs;
- ioctl(vcpufd, KVM_GET_SREGS, &sregs);
- sregs.cs.base = sregs.cs.selector = 0;
- ioctl(vcpufd, KVM_SET_SREGS, &sregs);
- ioctl(vcpufd, KVM_GET_SREGS, &sregs);
-
- #define KVM_EXIT_UNKNOWN 0
- #define KVM_EXIT_EXCEPTION 1
- #define KVM_EXIT_IO 2
- #define KVM_EXIT_HYPERCALL 3
- #define KVM_EXIT_DEBUG 4
- #define KVM_EXIT_HLT 5
- #define KVM_EXIT_MMIO 6
- #define KVM_EXIT_IRQ_WINDOW_OPEN 7
- #define KVM_EXIT_SHUTDOWN 8
- #define KVM_EXIT_FAIL_ENTRY 9
- #define KVM_EXIT_INTR 10
- #define KVM_EXIT_SET_TPR 11
- #define KVM_EXIT_TPR_ACCESS 12
- #define KVM_EXIT_S390_SIEIC 13
- #define KVM_EXIT_S390_RESET 14
- #define KVM_EXIT_DCR 15
- #define KVM_EXIT_NMI 16
- #define KVM_EXIT_INTERNAL_ERROR 17
- #define KVM_EXIT_OSI 18
- #define KVM_EXIT_PAPR_HCALL 19
- #define KVM_EXIT_S390_UCONTROL 20
- #define KVM_EXIT_WATCHDOG 21
- #define KVM_EXIT_S390_TSCH 22
- #define KVM_EXIT_EPR 23
- #define KVM_EXIT_SYSTEM_EVENT 24
- #define KVM_EXIT_S390_STSI 25
- #define KVM_EXIT_IOAPIC_EOI 26
- #define KVM_EXIT_HYPERV 27
- #define KVM_EXIT_ARM_NISV 28
- #define KVM_EXIT_X86_RDMSR 29
- #define KVM_EXIT_X86_WRMSR 30
- #define KVM_EXIT_DIRTY_RING_FULL 31
- #define KVM_EXIT_AP_RESET_HOLD 32
- #define KVM_EXIT_X86_BUS_LOCK 33
- #define KVM_EXIT_XEN 34
- while (1)
- {
- ioctl(vcpufd, KVM_RUN, NULL);
- ioctl(vcpufd, KVM_GET_SREGS, &sregs);
- printf("Guest CR3: 0x%llx\n", sregs.cr3);
- switch (run->exit_reason)
- {
- case KVM_EXIT_HLT:
- fputs("KVM_EXIT_HLT \n", stderr);
- return 0;
- case KVM_EXIT_IO:
-
- putchar(*(((char *)run) + run->io.data_offset));
- printf("KVM_EXIT_IO: run->io.port = %lx \n",
- run->io.port);
- break;
- case KVM_EXIT_FAIL_ENTRY:
- printf("KVM_EXIT_FAIL_ENTRY: hardware_entry_failure_reason = 0x%lx",
- run->fail_entry.hardware_entry_failure_reason);
- return 0;
- case KVM_EXIT_INTERNAL_ERROR:
- printf("KVM_EXIT_INTERNAL_ERROR: suberror = 0x%x",
- run->internal.suberror);
- return 0;
- case KVM_EXIT_SHUTDOWN:
- printf("KVM_EXIT_SHUTDOWN");
- return 0;
- default:
- printf("Unhandled reason: %d", run->exit_reason);
- return 0;
- }
- }
- }
-
- int main()
- {
-
-
- uint8_t code[] = "\xB0\x61\xF4";
- kvm(code, sizeof(code));
- return 0;
- }
|