mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-09 17:43:40 +00:00
8fad8c4b9b
updates tailcale/corp#22371 Adds custom macOS vm tooling. See the README for the general gist, but this will spin up VMs with unixgram capable network interfaces listening to a named socket, and with a virtio socket device for host-guest communication. We can add other devices like consoles, serial, etc as needed. The whole things is buildable with a single make command, and everything is controllable via the command line using the TailMac utility. This should all be generally functional but takes a few shortcuts with error handling and the like. The virtio socket device support has not been tested and may require some refinement. Signed-off-by: Jonathan Nobels <jonathan@tailscale.com>
180 lines
6.9 KiB
Swift
180 lines
6.9 KiB
Swift
// Copyright (c) Tailscale Inc & AUTHORS
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
import Cocoa
|
|
import Foundation
|
|
import Virtualization
|
|
import Foundation
|
|
|
|
class VMController: NSObject, VZVirtualMachineDelegate {
|
|
var virtualMachine: VZVirtualMachine!
|
|
|
|
lazy var helper = TailMacConfigHelper(config: config)
|
|
|
|
override init() {
|
|
super.init()
|
|
listenForNotifications()
|
|
}
|
|
|
|
func listenForNotifications() {
|
|
let nc = DistributedNotificationCenter()
|
|
nc.addObserver(forName: Notifications.stop, object: nil, queue: nil) { notification in
|
|
if let vmID = notification.userInfo?["id"] as? String {
|
|
if config.vmID == vmID {
|
|
print("We've been asked to stop... Saving state and exiting")
|
|
self.pauseAndSaveVirtualMachine {
|
|
exit(0)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
nc.addObserver(forName: Notifications.halt, object: nil, queue: nil) { notification in
|
|
if let vmID = notification.userInfo?["id"] as? String {
|
|
if config.vmID == vmID {
|
|
print("We've been asked to stop... Saving state and exiting")
|
|
self.virtualMachine.pause { (result) in
|
|
if case let .failure(error) = result {
|
|
fatalError("Virtual machine failed to pause with \(error)")
|
|
}
|
|
exit(0)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func createMacPlaform() -> VZMacPlatformConfiguration {
|
|
let macPlatform = VZMacPlatformConfiguration()
|
|
|
|
let auxiliaryStorage = VZMacAuxiliaryStorage(contentsOf: config.auxiliaryStorageURL)
|
|
macPlatform.auxiliaryStorage = auxiliaryStorage
|
|
|
|
if !FileManager.default.fileExists(atPath: config.vmDataURL.path()) {
|
|
fatalError("Missing Virtual Machine Bundle at \(config.vmDataURL). Run InstallationTool first to create it.")
|
|
}
|
|
|
|
// Retrieve the hardware model and save this value to disk during installation.
|
|
guard let hardwareModelData = try? Data(contentsOf: config.hardwareModelURL) else {
|
|
fatalError("Failed to retrieve hardware model data.")
|
|
}
|
|
|
|
guard let hardwareModel = VZMacHardwareModel(dataRepresentation: hardwareModelData) else {
|
|
fatalError("Failed to create hardware model.")
|
|
}
|
|
|
|
if !hardwareModel.isSupported {
|
|
fatalError("The hardware model isn't supported on the current host")
|
|
}
|
|
macPlatform.hardwareModel = hardwareModel
|
|
|
|
// Retrieve the machine identifier and save this value to disk during installation.
|
|
guard let machineIdentifierData = try? Data(contentsOf: config.machineIdentifierURL) else {
|
|
fatalError("Failed to retrieve machine identifier data.")
|
|
}
|
|
|
|
guard let machineIdentifier = VZMacMachineIdentifier(dataRepresentation: machineIdentifierData) else {
|
|
fatalError("Failed to create machine identifier.")
|
|
}
|
|
macPlatform.machineIdentifier = machineIdentifier
|
|
|
|
return macPlatform
|
|
}
|
|
|
|
func createVirtualMachine() {
|
|
let virtualMachineConfiguration = VZVirtualMachineConfiguration()
|
|
|
|
virtualMachineConfiguration.platform = createMacPlaform()
|
|
virtualMachineConfiguration.bootLoader = helper.createBootLoader()
|
|
virtualMachineConfiguration.cpuCount = helper.computeCPUCount()
|
|
virtualMachineConfiguration.memorySize = helper.computeMemorySize()
|
|
virtualMachineConfiguration.graphicsDevices = [helper.createGraphicsDeviceConfiguration()]
|
|
virtualMachineConfiguration.storageDevices = [helper.createBlockDeviceConfiguration()]
|
|
virtualMachineConfiguration.networkDevices = [helper.createNetworkDeviceConfiguration(), helper.createSocketNetworkDeviceConfiguration()]
|
|
virtualMachineConfiguration.pointingDevices = [helper.createPointingDeviceConfiguration()]
|
|
virtualMachineConfiguration.keyboards = [helper.createKeyboardConfiguration()]
|
|
virtualMachineConfiguration.socketDevices = [helper.createSocketDeviceConfiguration()]
|
|
|
|
try! virtualMachineConfiguration.validate()
|
|
try! virtualMachineConfiguration.validateSaveRestoreSupport()
|
|
|
|
virtualMachine = VZVirtualMachine(configuration: virtualMachineConfiguration)
|
|
virtualMachine.delegate = self
|
|
}
|
|
|
|
|
|
func startVirtualMachine() {
|
|
virtualMachine.start(completionHandler: { (result) in
|
|
if case let .failure(error) = result {
|
|
fatalError("Virtual machine failed to start with \(error)")
|
|
}
|
|
self.startSocketDevice()
|
|
})
|
|
}
|
|
|
|
func startSocketDevice() {
|
|
if let device = virtualMachine.socketDevices.first as? VZVirtioSocketDevice {
|
|
print("Configuring socket device at port \(config.port)")
|
|
device.connect(toPort: config.port) { connection in
|
|
//TODO: Anything? Or is this enough to bootstrap it on both ends?
|
|
}
|
|
} else {
|
|
print("Virtual machine could not start it's socket device")
|
|
}
|
|
}
|
|
|
|
func resumeVirtualMachine() {
|
|
virtualMachine.resume(completionHandler: { (result) in
|
|
if case let .failure(error) = result {
|
|
fatalError("Virtual machine failed to resume with \(error)")
|
|
}
|
|
})
|
|
}
|
|
|
|
func restoreVirtualMachine() {
|
|
virtualMachine.restoreMachineStateFrom(url: config.saveFileURL, completionHandler: { [self] (error) in
|
|
// Remove the saved file. Whether success or failure, the state no longer matches the VM's disk.
|
|
let fileManager = FileManager.default
|
|
try! fileManager.removeItem(at: config.saveFileURL)
|
|
|
|
if error == nil {
|
|
self.resumeVirtualMachine()
|
|
} else {
|
|
self.startVirtualMachine()
|
|
}
|
|
})
|
|
}
|
|
|
|
func saveVirtualMachine(completionHandler: @escaping () -> Void) {
|
|
virtualMachine.saveMachineStateTo(url: config.saveFileURL, completionHandler: { (error) in
|
|
guard error == nil else {
|
|
fatalError("Virtual machine failed to save with \(error!)")
|
|
}
|
|
|
|
completionHandler()
|
|
})
|
|
}
|
|
|
|
func pauseAndSaveVirtualMachine(completionHandler: @escaping () -> Void) {
|
|
virtualMachine.pause { result in
|
|
if case let .failure(error) = result {
|
|
fatalError("Virtual machine failed to pause with \(error)")
|
|
}
|
|
|
|
self.saveVirtualMachine(completionHandler: completionHandler)
|
|
}
|
|
}
|
|
|
|
// MARK: - VZVirtualMachineDeleate
|
|
|
|
func virtualMachine(_ virtualMachine: VZVirtualMachine, didStopWithError error: Error) {
|
|
print("Virtual machine did stop with error: \(error.localizedDescription)")
|
|
exit(-1)
|
|
}
|
|
|
|
func guestDidStop(_ virtualMachine: VZVirtualMachine) {
|
|
print("Guest did stop virtual machine.")
|
|
exit(0)
|
|
}
|
|
}
|