| #include "kvm/kvm.h" |
| #include "kvm/kvm-cpu.h" |
| |
| static int debug_fd; |
| |
| void kvm_cpu__set_debug_fd(int fd) |
| { |
| debug_fd = fd; |
| } |
| |
| int kvm_cpu__get_debug_fd(void) |
| { |
| return debug_fd; |
| } |
| |
| static struct kvm_arm_target *kvm_arm_generic_target; |
| static struct kvm_arm_target *kvm_arm_targets[KVM_ARM_NUM_TARGETS]; |
| |
| void kvm_cpu__set_kvm_arm_generic_target(struct kvm_arm_target *target) |
| { |
| kvm_arm_generic_target = target; |
| } |
| |
| int kvm_cpu__register_kvm_arm_target(struct kvm_arm_target *target) |
| { |
| unsigned int i = 0; |
| |
| for (i = 0; i < ARRAY_SIZE(kvm_arm_targets); ++i) { |
| if (!kvm_arm_targets[i]) { |
| kvm_arm_targets[i] = target; |
| return 0; |
| } |
| } |
| |
| return -ENOSPC; |
| } |
| |
| struct kvm_cpu *kvm_cpu__arch_init(struct kvm *kvm, unsigned long cpu_id) |
| { |
| struct kvm_arm_target *target; |
| struct kvm_cpu *vcpu; |
| int coalesced_offset, mmap_size, err = -1; |
| unsigned int i; |
| struct kvm_vcpu_init preferred_init; |
| struct kvm_vcpu_init vcpu_init = { |
| .features = {}, |
| }; |
| |
| vcpu = calloc(1, sizeof(struct kvm_cpu)); |
| if (!vcpu) |
| return NULL; |
| |
| vcpu->vcpu_fd = ioctl(kvm->vm_fd, KVM_CREATE_VCPU, cpu_id); |
| if (vcpu->vcpu_fd < 0) |
| die_perror("KVM_CREATE_VCPU ioctl"); |
| |
| mmap_size = ioctl(kvm->sys_fd, KVM_GET_VCPU_MMAP_SIZE, 0); |
| if (mmap_size < 0) |
| die_perror("KVM_GET_VCPU_MMAP_SIZE ioctl"); |
| |
| vcpu->kvm_run = mmap(NULL, mmap_size, PROT_RW, MAP_SHARED, |
| vcpu->vcpu_fd, 0); |
| if (vcpu->kvm_run == MAP_FAILED) |
| die("unable to mmap vcpu fd"); |
| |
| /* VCPU 0 is the boot CPU, the others start in a poweroff state. */ |
| if (cpu_id > 0) |
| vcpu_init.features[0] |= (1UL << KVM_ARM_VCPU_POWER_OFF); |
| |
| /* Set KVM_ARM_VCPU_PSCI_0_2 if available */ |
| if (kvm__supports_extension(kvm, KVM_CAP_ARM_PSCI_0_2)) { |
| vcpu_init.features[0] |= (1UL << KVM_ARM_VCPU_PSCI_0_2); |
| } |
| |
| kvm_cpu__select_features(kvm, &vcpu_init); |
| |
| /* |
| * If the preferred target ioctl is successful then |
| * use preferred target else try each and every target type |
| */ |
| err = ioctl(kvm->vm_fd, KVM_ARM_PREFERRED_TARGET, &preferred_init); |
| if (!err) { |
| /* Match preferred target CPU type. */ |
| target = NULL; |
| for (i = 0; i < ARRAY_SIZE(kvm_arm_targets); ++i) { |
| if (!kvm_arm_targets[i]) |
| continue; |
| if (kvm_arm_targets[i]->id == preferred_init.target) { |
| target = kvm_arm_targets[i]; |
| break; |
| } |
| } |
| if (!target) { |
| target = kvm_arm_generic_target; |
| vcpu_init.target = preferred_init.target; |
| } else { |
| vcpu_init.target = target->id; |
| } |
| err = ioctl(vcpu->vcpu_fd, KVM_ARM_VCPU_INIT, &vcpu_init); |
| } else { |
| /* Find an appropriate target CPU type. */ |
| for (i = 0; i < ARRAY_SIZE(kvm_arm_targets); ++i) { |
| if (!kvm_arm_targets[i]) |
| continue; |
| target = kvm_arm_targets[i]; |
| vcpu_init.target = target->id; |
| err = ioctl(vcpu->vcpu_fd, KVM_ARM_VCPU_INIT, &vcpu_init); |
| if (!err) |
| break; |
| } |
| if (err) |
| die("Unable to find matching target"); |
| } |
| |
| /* Populate the vcpu structure. */ |
| vcpu->kvm = kvm; |
| vcpu->cpu_id = cpu_id; |
| vcpu->cpu_type = vcpu_init.target; |
| vcpu->cpu_compatible = target->compatible; |
| vcpu->is_running = true; |
| |
| if (err || target->init(vcpu)) |
| die("Unable to initialise vcpu"); |
| |
| coalesced_offset = ioctl(kvm->sys_fd, KVM_CHECK_EXTENSION, |
| KVM_CAP_COALESCED_MMIO); |
| if (coalesced_offset) |
| vcpu->ring = (void *)vcpu->kvm_run + |
| (coalesced_offset * PAGE_SIZE); |
| |
| if (kvm_cpu__configure_features(vcpu)) |
| die("Unable to configure requested vcpu features"); |
| |
| return vcpu; |
| } |
| |
| void kvm_cpu__arch_nmi(struct kvm_cpu *cpu) |
| { |
| } |
| |
| void kvm_cpu__delete(struct kvm_cpu *vcpu) |
| { |
| kvm_cpu__teardown_pvtime(vcpu->kvm); |
| free(vcpu); |
| } |
| |
| bool kvm_cpu__handle_exit(struct kvm_cpu *vcpu) |
| { |
| return false; |
| } |
| |
| void kvm_cpu__show_page_tables(struct kvm_cpu *vcpu) |
| { |
| } |