1*bb4ee6a4SAndroid Build Coastguard Worker // Copyright 2017 The ChromiumOS Authors
2*bb4ee6a4SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*bb4ee6a4SAndroid Build Coastguard Worker // found in the LICENSE file.
4*bb4ee6a4SAndroid Build Coastguard Worker
5*bb4ee6a4SAndroid Build Coastguard Worker #![cfg(any(target_os = "android", target_os = "linux"))]
6*bb4ee6a4SAndroid Build Coastguard Worker #![cfg(target_arch = "x86_64")]
7*bb4ee6a4SAndroid Build Coastguard Worker
8*bb4ee6a4SAndroid Build Coastguard Worker use base::MemoryMappingBuilder;
9*bb4ee6a4SAndroid Build Coastguard Worker use base::SharedMemory;
10*bb4ee6a4SAndroid Build Coastguard Worker use kvm::*;
11*bb4ee6a4SAndroid Build Coastguard Worker use kvm_sys::kvm_regs;
12*bb4ee6a4SAndroid Build Coastguard Worker use vm_memory::GuestAddress;
13*bb4ee6a4SAndroid Build Coastguard Worker use vm_memory::GuestMemory;
14*bb4ee6a4SAndroid Build Coastguard Worker
15*bb4ee6a4SAndroid Build Coastguard Worker #[test]
test_run()16*bb4ee6a4SAndroid Build Coastguard Worker fn test_run() {
17*bb4ee6a4SAndroid Build Coastguard Worker /*
18*bb4ee6a4SAndroid Build Coastguard Worker 0000 881C mov [si],bl
19*bb4ee6a4SAndroid Build Coastguard Worker 0002 F4 hlt
20*bb4ee6a4SAndroid Build Coastguard Worker */
21*bb4ee6a4SAndroid Build Coastguard Worker let code = [0x88, 0x1c, 0xf4];
22*bb4ee6a4SAndroid Build Coastguard Worker let mem_size = 0x10000;
23*bb4ee6a4SAndroid Build Coastguard Worker let load_addr = GuestAddress(0x1000);
24*bb4ee6a4SAndroid Build Coastguard Worker let guest_mem = GuestMemory::new(&[]).unwrap();
25*bb4ee6a4SAndroid Build Coastguard Worker let mem = SharedMemory::new("test", mem_size).expect("failed to create shared memory");
26*bb4ee6a4SAndroid Build Coastguard Worker let mmap = MemoryMappingBuilder::new(mem_size as usize)
27*bb4ee6a4SAndroid Build Coastguard Worker .from_shared_memory(&mem)
28*bb4ee6a4SAndroid Build Coastguard Worker .build()
29*bb4ee6a4SAndroid Build Coastguard Worker .expect("failed to create memory mapping");
30*bb4ee6a4SAndroid Build Coastguard Worker
31*bb4ee6a4SAndroid Build Coastguard Worker mmap.write_slice(&code[..], load_addr.offset() as usize)
32*bb4ee6a4SAndroid Build Coastguard Worker .expect("Writing code to memory failed.");
33*bb4ee6a4SAndroid Build Coastguard Worker
34*bb4ee6a4SAndroid Build Coastguard Worker let kvm = Kvm::new().expect("new kvm failed");
35*bb4ee6a4SAndroid Build Coastguard Worker let mut vm = Vm::new(&kvm, guest_mem).expect("new vm failed");
36*bb4ee6a4SAndroid Build Coastguard Worker let vcpu = Vcpu::new(0, &kvm, &vm).expect("new vcpu failed");
37*bb4ee6a4SAndroid Build Coastguard Worker let mut vcpu_sregs = vcpu.get_sregs().expect("get sregs failed");
38*bb4ee6a4SAndroid Build Coastguard Worker vcpu_sregs.cs.base = 0;
39*bb4ee6a4SAndroid Build Coastguard Worker vcpu_sregs.cs.selector = 0;
40*bb4ee6a4SAndroid Build Coastguard Worker vcpu.set_sregs(&vcpu_sregs).expect("set sregs failed");
41*bb4ee6a4SAndroid Build Coastguard Worker
42*bb4ee6a4SAndroid Build Coastguard Worker // SAFETY: trivially safe
43*bb4ee6a4SAndroid Build Coastguard Worker let mut vcpu_regs: kvm_regs = unsafe { std::mem::zeroed() };
44*bb4ee6a4SAndroid Build Coastguard Worker vcpu_regs.rip = load_addr.offset();
45*bb4ee6a4SAndroid Build Coastguard Worker vcpu_regs.rflags = 2;
46*bb4ee6a4SAndroid Build Coastguard Worker // Write 0x12 to the beginning of the 9th page.
47*bb4ee6a4SAndroid Build Coastguard Worker vcpu_regs.rsi = 0x8000;
48*bb4ee6a4SAndroid Build Coastguard Worker vcpu_regs.rbx = 0x12;
49*bb4ee6a4SAndroid Build Coastguard Worker vcpu.set_regs(&vcpu_regs).expect("set regs failed");
50*bb4ee6a4SAndroid Build Coastguard Worker let slot = vm
51*bb4ee6a4SAndroid Build Coastguard Worker .add_memory_region(
52*bb4ee6a4SAndroid Build Coastguard Worker GuestAddress(0),
53*bb4ee6a4SAndroid Build Coastguard Worker Box::new(
54*bb4ee6a4SAndroid Build Coastguard Worker MemoryMappingBuilder::new(mem_size as usize)
55*bb4ee6a4SAndroid Build Coastguard Worker .from_shared_memory(&mem)
56*bb4ee6a4SAndroid Build Coastguard Worker .build()
57*bb4ee6a4SAndroid Build Coastguard Worker .expect("failed to create memory mapping"),
58*bb4ee6a4SAndroid Build Coastguard Worker ),
59*bb4ee6a4SAndroid Build Coastguard Worker false,
60*bb4ee6a4SAndroid Build Coastguard Worker true,
61*bb4ee6a4SAndroid Build Coastguard Worker )
62*bb4ee6a4SAndroid Build Coastguard Worker .expect("failed to register memory");
63*bb4ee6a4SAndroid Build Coastguard Worker
64*bb4ee6a4SAndroid Build Coastguard Worker let runnable_vcpu = vcpu.to_runnable(None).unwrap();
65*bb4ee6a4SAndroid Build Coastguard Worker let run_result = runnable_vcpu.run().expect("run failed");
66*bb4ee6a4SAndroid Build Coastguard Worker if !matches!(run_result, VcpuExit::Hlt) {
67*bb4ee6a4SAndroid Build Coastguard Worker panic!("unexpected exit reason: {:?}", run_result);
68*bb4ee6a4SAndroid Build Coastguard Worker }
69*bb4ee6a4SAndroid Build Coastguard Worker
70*bb4ee6a4SAndroid Build Coastguard Worker let mut dirty_log = [0x0, 0x0];
71*bb4ee6a4SAndroid Build Coastguard Worker vm.get_dirty_log(slot, &mut dirty_log[..])
72*bb4ee6a4SAndroid Build Coastguard Worker .expect("failed to get dirty log");
73*bb4ee6a4SAndroid Build Coastguard Worker // Tests the 9th page was written to.
74*bb4ee6a4SAndroid Build Coastguard Worker assert_eq!(dirty_log[1], 0x1);
75*bb4ee6a4SAndroid Build Coastguard Worker assert_eq!(
76*bb4ee6a4SAndroid Build Coastguard Worker mmap.read_obj::<u64>(vcpu_regs.rsi as usize).unwrap(),
77*bb4ee6a4SAndroid Build Coastguard Worker vcpu_regs.rbx
78*bb4ee6a4SAndroid Build Coastguard Worker );
79*bb4ee6a4SAndroid Build Coastguard Worker }
80