tstest/tailmac: add customized macOS virtualization tooling (#13146)
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>
This commit is contained in:
@@ -0,0 +1,125 @@
|
||||
// Copyright (c) Tailscale Inc & AUTHORS
|
||||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
import Foundation
|
||||
|
||||
let kDefaultDiskSizeGb: Int64 = 72
|
||||
let kDefaultMemSizeGb: UInt64 = 72
|
||||
|
||||
|
||||
/// Represents a configuration for a virtual machine
|
||||
class Config: Codable {
|
||||
var serverSocket = "/tmp/qemu-dgram.sock"
|
||||
var memorySize = (kDefaultMemSizeGb * 1024 * 1024 * 1024) as UInt64
|
||||
var mac = "52:cc:cc:cc:cc:01"
|
||||
var ethermac = "52:cc:cc:cc:ce:01"
|
||||
var port: UInt32 = 51009
|
||||
|
||||
// The virtual machines ID. Also double as the directory name under which
|
||||
// we will store configuration, block device, etc.
|
||||
let vmID: String
|
||||
|
||||
required init(from decoder: Decoder) throws {
|
||||
let container = try decoder.container(keyedBy: CodingKeys.self)
|
||||
if let ethermac = try container.decodeIfPresent(String.self, forKey: .ethermac) {
|
||||
self.ethermac = ethermac
|
||||
}
|
||||
if let serverSocket = try container.decodeIfPresent(String.self, forKey: .serverSocket) {
|
||||
self.serverSocket = serverSocket
|
||||
}
|
||||
if let memorySize = try container.decodeIfPresent(UInt64.self, forKey: .memorySize) {
|
||||
self.memorySize = memorySize
|
||||
}
|
||||
if let port = try container.decodeIfPresent(UInt32.self, forKey: .port) {
|
||||
self.port = port
|
||||
}
|
||||
if let mac = try container.decodeIfPresent(String.self, forKey: .mac) {
|
||||
self.mac = mac
|
||||
}
|
||||
if let vmID = try container.decodeIfPresent(String.self, forKey: .vmID) {
|
||||
self.vmID = vmID
|
||||
} else {
|
||||
self.vmID = "default"
|
||||
}
|
||||
}
|
||||
|
||||
init(_ vmID: String = "default") {
|
||||
self.vmID = vmID
|
||||
let configFile = vmDataURL.appendingPathComponent("config.json")
|
||||
if FileManager.default.fileExists(atPath: configFile.path()) {
|
||||
print("Using config file at path \(configFile)")
|
||||
if let jsonData = try? Data(contentsOf: configFile) {
|
||||
let config = try! JSONDecoder().decode(Config.self, from: jsonData)
|
||||
self.serverSocket = config.serverSocket
|
||||
self.memorySize = config.memorySize
|
||||
self.mac = config.mac
|
||||
self.port = config.port
|
||||
self.ethermac = config.ethermac
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func persist() {
|
||||
let configFile = vmDataURL.appendingPathComponent("config.json")
|
||||
let data = try! JSONEncoder().encode(self)
|
||||
try! data.write(to: configFile)
|
||||
}
|
||||
|
||||
lazy var restoreImageURL: URL = {
|
||||
vmBundleURL.appendingPathComponent("RestoreImage.ipsw")
|
||||
}()
|
||||
|
||||
// The VM Data URL holds the specific files composing a unique VM guest instance
|
||||
// By default, VM's are persisted at ~/VM.bundle/<vmID>
|
||||
lazy var vmDataURL = {
|
||||
let dataURL = vmBundleURL.appendingPathComponent(vmID)
|
||||
return dataURL
|
||||
}()
|
||||
|
||||
lazy var auxiliaryStorageURL = {
|
||||
vmDataURL.appendingPathComponent("AuxiliaryStorage")
|
||||
}()
|
||||
|
||||
lazy var diskImageURL = {
|
||||
vmDataURL.appendingPathComponent("Disk.img")
|
||||
}()
|
||||
|
||||
lazy var diskSize: Int64 = {
|
||||
kDefaultDiskSizeGb * 1024 * 1024 * 1024
|
||||
}()
|
||||
|
||||
lazy var hardwareModelURL = {
|
||||
vmDataURL.appendingPathComponent("HardwareModel")
|
||||
}()
|
||||
|
||||
lazy var machineIdentifierURL = {
|
||||
vmDataURL.appendingPathComponent("MachineIdentifier")
|
||||
}()
|
||||
|
||||
lazy var saveFileURL = {
|
||||
vmDataURL.appendingPathComponent("SaveFile.vzvmsave")
|
||||
}()
|
||||
|
||||
}
|
||||
|
||||
// The VM Bundle URL holds the restore image and a set of VM images
|
||||
// By default, VM's are persisted at ~/VM.bundle
|
||||
var vmBundleURL: URL = {
|
||||
let vmBundlePath = NSHomeDirectory() + "/VM.bundle/"
|
||||
createDir(vmBundlePath)
|
||||
let bundleURL = URL(fileURLWithPath: vmBundlePath)
|
||||
return bundleURL
|
||||
}()
|
||||
|
||||
|
||||
func createDir(_ path: String) {
|
||||
do {
|
||||
try FileManager.default.createDirectory(atPath: path, withIntermediateDirectories: true)
|
||||
} catch {
|
||||
fatalError("Unable to create dir at \(path) \(error)")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
Reference in New Issue
Block a user