mirror of https://github.com/tailscale/tailscale/
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
187 lines
7.3 KiB
Swift
187 lines
7.3 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()]
|
|
|
|
if let dir = config.sharedDir, let shareConfig = helper.createDirectoryShareConfiguration(tag: "vmshare") {
|
|
print("Sharing \(dir) as vmshare. Use: mount_virtiofs vmshare <path> in the guest to mount.")
|
|
virtualMachineConfiguration.directorySharingDevices = [shareConfig]
|
|
} else {
|
|
print("No shared directory created. \(config.sharedDir ?? "none") was requested.")
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|