podman/pkg/machine/provider/platform_darwin.go

129 lines
3.4 KiB
Go

package provider
import (
"bytes"
"fmt"
"os"
"os/exec"
"runtime"
"strings"
"github.com/blang/semver/v4"
"github.com/containers/common/pkg/config"
"github.com/containers/podman/v5/pkg/machine/applehv"
"github.com/containers/podman/v5/pkg/machine/define"
"github.com/containers/podman/v5/pkg/machine/libkrun"
"github.com/containers/podman/v5/pkg/machine/vmconfigs"
"github.com/sirupsen/logrus"
)
func Get() (vmconfigs.VMProvider, error) {
cfg, err := config.Default()
if err != nil {
return nil, err
}
provider := cfg.Machine.Provider
if providerOverride, found := os.LookupEnv("CONTAINERS_MACHINE_PROVIDER"); found {
provider = providerOverride
}
resolvedVMType, err := define.ParseVMType(provider, define.AppleHvVirt)
if err != nil {
return nil, err
}
logrus.Debugf("Using Podman machine with `%s` virtualization provider", resolvedVMType.String())
switch resolvedVMType {
case define.AppleHvVirt:
return new(applehv.AppleHVStubber), nil
case define.LibKrun:
return new(libkrun.LibKrunStubber), nil
default:
return nil, fmt.Errorf("unsupported virtualization provider: `%s`", resolvedVMType.String())
}
}
func GetAll(_ bool) ([]vmconfigs.VMProvider, error) {
return []vmconfigs.VMProvider{
new(applehv.AppleHVStubber),
new(libkrun.LibKrunStubber),
}, nil
}
// SupportedProviders returns the providers that are supported on the host operating system
func SupportedProviders() []define.VMType {
supported := []define.VMType{define.AppleHvVirt}
if runtime.GOARCH == "arm64" {
return append(supported, define.LibKrun)
}
return supported
}
// InstalledProviders returns the supported providers that are installed on the host
func InstalledProviders() ([]define.VMType, error) {
installed := []define.VMType{}
appleHvInstalled, err := appleHvInstalled()
if err != nil {
return nil, err
}
if appleHvInstalled {
installed = append(installed, define.AppleHvVirt)
}
libKrunInstalled, err := libKrunInstalled()
if err != nil {
return nil, err
}
if libKrunInstalled {
installed = append(installed, define.LibKrun)
}
return installed, nil
}
func appleHvInstalled() (bool, error) {
var outBuf bytes.Buffer
// Apple's Virtualization.Framework is only supported on MacOS 11.0+,
// but to use EFI MacOS 13.0+ is required
expectedVer, err := semver.Make("13.0.0")
if err != nil {
return false, err
}
cmd := exec.Command("sw_vers", "--productVersion")
cmd.Stdout = &outBuf
if err := cmd.Run(); err != nil {
return false, fmt.Errorf("unable to check current macOS version using `sw_vers --productVersion`: %s", err)
}
// the output will be in the format of MAJOR.MINOR.PATCH
output := strings.TrimSuffix(outBuf.String(), "\n")
currentVer, err := semver.Make(output)
if err != nil {
return false, err
}
return currentVer.GTE(expectedVer), nil
}
func libKrunInstalled() (bool, error) {
if runtime.GOARCH != "arm64" {
return false, nil
}
// need to verify that krunkit, virglrenderer, and libkrun-efi are installed
cfg, err := config.Default()
if err != nil {
return false, err
}
_, err = cfg.FindHelperBinary("krunkit", false)
return err == nil, nil
}
// HasPermsForProvider returns whether the host operating system has the proper permissions to use the given provider
func HasPermsForProvider(provider define.VMType) bool {
// there are no permissions required for AppleHV or LibKrun
return provider == define.AppleHvVirt || provider == define.LibKrun
}