Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2024 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
| 17 | //! Responsible for starting an instance of the Microfuchsia VM. |
| 18 | |
| 19 | use android_system_virtualizationservice::aidl::android::system::virtualizationservice::{ |
| 20 | CpuTopology::CpuTopology, IVirtualizationService::IVirtualizationService, |
| 21 | VirtualMachineConfig::VirtualMachineConfig, VirtualMachineRawConfig::VirtualMachineRawConfig, |
| 22 | }; |
| 23 | use anyhow::{ensure, Context, Result}; |
| 24 | use binder::{LazyServiceGuard, ParcelFileDescriptor}; |
| 25 | use log::info; |
| 26 | use std::ffi::CStr; |
| 27 | use std::fs::File; |
Jiyong Park | af63d1c | 2024-09-04 16:15:42 +0900 | [diff] [blame] | 28 | use std::os::fd::FromRawFd; |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 29 | use vmclient::VmInstance; |
| 30 | |
| 31 | pub struct MicrofuchsiaInstance { |
| 32 | _vm_instance: VmInstance, |
| 33 | _lazy_service_guard: LazyServiceGuard, |
James Robinson | 6eea2b8 | 2024-11-19 17:28:00 +0000 | [diff] [blame] | 34 | _pty: Option<Pty>, |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 35 | } |
| 36 | |
| 37 | pub struct InstanceStarter { |
| 38 | instance_name: String, |
| 39 | instance_id: u8, |
| 40 | } |
| 41 | |
| 42 | impl InstanceStarter { |
| 43 | pub fn new(instance_name: &str, instance_id: u8) -> Self { |
| 44 | Self { instance_name: instance_name.to_owned(), instance_id } |
| 45 | } |
| 46 | |
| 47 | pub fn start_new_instance( |
| 48 | &self, |
| 49 | virtualization_service: &dyn IVirtualizationService, |
| 50 | ) -> Result<MicrofuchsiaInstance> { |
| 51 | info!("Creating {} instance", self.instance_name); |
| 52 | |
| 53 | // Always use instance id 0, because we will only ever have one instance. |
| 54 | let mut instance_id = [0u8; 64]; |
| 55 | instance_id[0] = self.instance_id; |
| 56 | |
| 57 | // Open the kernel and initrd files from the microfuchsia.images apex. |
| 58 | let kernel_fd = |
| 59 | File::open("/apex/com.android.microfuchsia.images/etc/linux-arm64-boot-shim.bin") |
| 60 | .context("Failed to open the boot-shim")?; |
| 61 | let initrd_fd = File::open("/apex/com.android.microfuchsia.images/etc/fuchsia.zbi") |
| 62 | .context("Failed to open the fuchsia ZBI")?; |
| 63 | let kernel = Some(ParcelFileDescriptor::new(kernel_fd)); |
| 64 | let initrd = Some(ParcelFileDescriptor::new(initrd_fd)); |
| 65 | |
| 66 | // Prepare a pty for console input/output. |
James Robinson | 6eea2b8 | 2024-11-19 17:28:00 +0000 | [diff] [blame] | 67 | let (pty, console_in, console_out) = if cfg!(enable_console) { |
| 68 | let pty = openpty()?; |
| 69 | let console_in = Some(pty.leader.try_clone().context("cloning pty")?); |
| 70 | let console_out = Some(pty.leader.try_clone().context("cloning pty")?); |
| 71 | (Some(pty), console_in, console_out) |
| 72 | } else { |
| 73 | (None, None, None) |
| 74 | }; |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 75 | let config = VirtualMachineConfig::RawConfig(VirtualMachineRawConfig { |
| 76 | name: "Microfuchsia".into(), |
| 77 | instanceId: instance_id, |
| 78 | kernel, |
| 79 | initrd, |
| 80 | params: None, |
| 81 | bootloader: None, |
| 82 | disks: vec![], |
| 83 | protectedVm: false, |
| 84 | memoryMib: 256, |
| 85 | cpuTopology: CpuTopology::ONE_CPU, |
| 86 | platformVersion: "1.0.0".into(), |
James Robinson | 6eea2b8 | 2024-11-19 17:28:00 +0000 | [diff] [blame] | 87 | #[cfg(enable_console)] |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 88 | consoleInputDevice: Some("ttyS0".into()), |
Frederick Mayle | cc4e2d7 | 2024-12-06 16:54:40 -0800 | [diff] [blame] | 89 | balloon: true, |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 90 | ..Default::default() |
| 91 | }); |
| 92 | let vm_instance = VmInstance::create( |
| 93 | virtualization_service, |
| 94 | &config, |
| 95 | console_out, |
| 96 | console_in, |
| 97 | /* log= */ None, |
Elie Kheirallah | 5c807a2 | 2024-09-23 20:40:42 +0000 | [diff] [blame] | 98 | /* dump_dt= */ None, |
Chaitanya Cheemala (xWF) | 3da8a16 | 2025-01-21 08:57:09 -0800 | [diff] [blame] | 99 | None, |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 100 | ) |
| 101 | .context("Failed to create VM")?; |
James Robinson | 6eea2b8 | 2024-11-19 17:28:00 +0000 | [diff] [blame] | 102 | if let Some(pty) = &pty { |
| 103 | vm_instance |
| 104 | .vm |
| 105 | .setHostConsoleName(&pty.follower_name) |
| 106 | .context("Setting host console name")?; |
| 107 | } |
Chaitanya Cheemala (xWF) | 3da8a16 | 2025-01-21 08:57:09 -0800 | [diff] [blame] | 108 | vm_instance.start().context("Starting VM")?; |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 109 | |
| 110 | Ok(MicrofuchsiaInstance { |
| 111 | _vm_instance: vm_instance, |
| 112 | _lazy_service_guard: Default::default(), |
| 113 | _pty: pty, |
| 114 | }) |
| 115 | } |
| 116 | } |
| 117 | |
| 118 | struct Pty { |
| 119 | leader: File, |
| 120 | follower_name: String, |
| 121 | } |
| 122 | |
| 123 | /// Opens a pseudoterminal (pty), configures it to be a raw terminal, and returns the file pair. |
| 124 | fn openpty() -> Result<Pty> { |
| 125 | // Create a pty pair. |
| 126 | let mut leader: libc::c_int = -1; |
| 127 | let mut _follower: libc::c_int = -1; |
| 128 | let mut follower_name: Vec<libc::c_char> = vec![0; 32]; |
| 129 | |
| 130 | // SAFETY: calling openpty with valid+initialized variables is safe. |
| 131 | // The two null pointers are valid inputs for openpty. |
| 132 | unsafe { |
| 133 | ensure!( |
| 134 | libc::openpty( |
| 135 | &mut leader, |
| 136 | &mut _follower, |
| 137 | follower_name.as_mut_ptr(), |
| 138 | std::ptr::null_mut(), |
| 139 | std::ptr::null_mut(), |
| 140 | ) == 0, |
| 141 | "failed to openpty" |
| 142 | ); |
| 143 | } |
| 144 | |
| 145 | // SAFETY: calling these libc functions with valid+initialized variables is safe. |
| 146 | unsafe { |
| 147 | // Fetch the termios attributes. |
| 148 | let mut attr = libc::termios { |
| 149 | c_iflag: 0, |
| 150 | c_oflag: 0, |
| 151 | c_cflag: 0, |
| 152 | c_lflag: 0, |
| 153 | c_line: 0, |
| 154 | c_cc: [0u8; 19], |
| 155 | }; |
Jiyong Park | af63d1c | 2024-09-04 16:15:42 +0900 | [diff] [blame] | 156 | ensure!(libc::tcgetattr(leader, &mut attr) == 0, "failed to get termios attributes"); |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 157 | |
| 158 | // Force it to be a raw pty and re-set it. |
| 159 | libc::cfmakeraw(&mut attr); |
| 160 | ensure!( |
Jiyong Park | af63d1c | 2024-09-04 16:15:42 +0900 | [diff] [blame] | 161 | libc::tcsetattr(leader, libc::TCSANOW, &attr) == 0, |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 162 | "failed to set termios attributes" |
| 163 | ); |
| 164 | } |
| 165 | |
| 166 | // Construct the return value. |
Jiyong Park | af63d1c | 2024-09-04 16:15:42 +0900 | [diff] [blame] | 167 | // SAFETY: The file descriptors are valid because openpty returned without error (above). |
| 168 | let leader = unsafe { File::from_raw_fd(leader) }; |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 169 | let follower_name: Vec<u8> = follower_name.iter_mut().map(|x| *x as _).collect(); |
| 170 | let follower_name = CStr::from_bytes_until_nul(&follower_name) |
| 171 | .context("pty filename missing NUL")? |
| 172 | .to_str() |
| 173 | .context("pty filename invalid utf8")? |
| 174 | .to_string(); |
Jiyong Park | af63d1c | 2024-09-04 16:15:42 +0900 | [diff] [blame] | 175 | Ok(Pty { leader, follower_name }) |
Aidan Wolter | bb91a6b | 2024-07-01 15:42:20 +0000 | [diff] [blame] | 176 | } |