mirror of
https://github.com/containers/podman.git
synced 2025-07-29 03:12:47 +08:00

podman machine with Microsoft HyperV will use hvsock on the guest and vsock on the guest for its networking. this pr enables the basics for this to happen as well as changes to ignition to automatically set this up with network manager. the vm binary referenced in this pr is in containers/gvisor-tap-vsock and will need to be added to distributions. [NO NEW TESTS NEEDED] Signed-off-by: Brent Baude <bbaude@redhat.com>
271 lines
7.2 KiB
Go
271 lines
7.2 KiB
Go
//go:build windows
|
|
// +build windows
|
|
|
|
package hyperv
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/Microsoft/go-winio"
|
|
"github.com/containers/podman/v4/utils"
|
|
"github.com/sirupsen/logrus"
|
|
"golang.org/x/sys/windows/registry"
|
|
)
|
|
|
|
var ErrVSockRegistryEntryExists = errors.New("registry entry already exists")
|
|
|
|
const (
|
|
// HvsockMachineName is the string identifier for the machine name in a registry entry
|
|
HvsockMachineName = "MachineName"
|
|
// HvsockPurpose is the string identifier for the sock purpose in a registry entry
|
|
HvsockPurpose = "Purpose"
|
|
// VsockRegistryPath describes the registry path to where the hvsock registry entries live
|
|
VsockRegistryPath = `SOFTWARE\Microsoft\Windows NT\CurrentVersion\Virtualization\GuestCommunicationServices`
|
|
// LinuxVm is the default guid for a Linux VM on Windows
|
|
LinuxVm = "FACB-11E6-BD58-64006A7986D3"
|
|
)
|
|
|
|
// HVSockPurpose describes what the hvsock is needed for
|
|
type HVSockPurpose int
|
|
|
|
const (
|
|
// Network implies the sock is used for user-mode networking
|
|
Network HVSockPurpose = iota
|
|
// Events implies the sock is used for notification (like "Ready")
|
|
Events
|
|
)
|
|
|
|
func (hv HVSockPurpose) string() string {
|
|
switch hv {
|
|
case Network:
|
|
return "Network"
|
|
case Events:
|
|
return "Events"
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (hv HVSockPurpose) Equal(purpose string) bool {
|
|
return hv.string() == purpose
|
|
}
|
|
|
|
func toHVSockPurpose(p string) (HVSockPurpose, error) {
|
|
switch p {
|
|
case "Network":
|
|
return Network, nil
|
|
case "Events":
|
|
return Events, nil
|
|
}
|
|
return 0, fmt.Errorf("unknown hvsockpurpose: %s", p)
|
|
}
|
|
|
|
func openVSockRegistryEntry(entry string) (registry.Key, error) {
|
|
return registry.OpenKey(registry.LOCAL_MACHINE, entry, registry.QUERY_VALUE)
|
|
}
|
|
|
|
// HVSockRegistryEntry describes a registry entry used in Windows for HVSOCK implementations
|
|
type HVSockRegistryEntry struct {
|
|
KeyName string `json:"key_name"`
|
|
Purpose HVSockPurpose `json:"purpose"`
|
|
Port uint64 `json:"port"`
|
|
MachineName string `json:"machineName"`
|
|
Key registry.Key `json:"key,omitempty"`
|
|
}
|
|
|
|
// Add creates a new Windows registry entry with string values from the
|
|
// HVSockRegistryEntry.
|
|
func (hv *HVSockRegistryEntry) Add() error {
|
|
if err := hv.validate(); err != nil {
|
|
return err
|
|
}
|
|
exists, err := hv.exists()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if exists {
|
|
return fmt.Errorf("%q: %s", ErrVSockRegistryEntryExists, hv.KeyName)
|
|
}
|
|
parentKey, err := registry.OpenKey(registry.LOCAL_MACHINE, VsockRegistryPath, registry.QUERY_VALUE)
|
|
defer func() {
|
|
if err := parentKey.Close(); err != nil {
|
|
logrus.Error(err)
|
|
}
|
|
}()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
newKey, _, err := registry.CreateKey(parentKey, hv.KeyName, registry.WRITE)
|
|
defer func() {
|
|
if err := newKey.Close(); err != nil {
|
|
logrus.Error(err)
|
|
}
|
|
}()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := newKey.SetStringValue(HvsockPurpose, hv.Purpose.string()); err != nil {
|
|
return err
|
|
}
|
|
return newKey.SetStringValue(HvsockMachineName, hv.MachineName)
|
|
}
|
|
|
|
// Remove deletes the registry key and its string values
|
|
func (hv *HVSockRegistryEntry) Remove() error {
|
|
return registry.DeleteKey(registry.LOCAL_MACHINE, hv.fqPath())
|
|
}
|
|
|
|
func (hv *HVSockRegistryEntry) fqPath() string {
|
|
return fmt.Sprintf("%s\\%s", VsockRegistryPath, hv.KeyName)
|
|
}
|
|
|
|
func (hv *HVSockRegistryEntry) validate() error {
|
|
if hv.Port < 1 {
|
|
return errors.New("port must be larger than 1")
|
|
}
|
|
if len(hv.Purpose.string()) < 1 {
|
|
return errors.New("required field purpose is empty")
|
|
}
|
|
if len(hv.MachineName) < 1 {
|
|
return errors.New("required field machinename is empty")
|
|
}
|
|
if len(hv.KeyName) < 1 {
|
|
return errors.New("required field keypath is empty")
|
|
}
|
|
//decimal_num, err = strconv.ParseInt(hexadecimal_num, 16, 64)
|
|
return nil
|
|
}
|
|
|
|
func (hv *HVSockRegistryEntry) exists() (bool, error) {
|
|
foo := hv.fqPath()
|
|
_ = foo
|
|
_, err := openVSockRegistryEntry(hv.fqPath())
|
|
if err == nil {
|
|
return true, err
|
|
}
|
|
if errors.Is(err, registry.ErrNotExist) {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
|
|
// findOpenHVSockPort looks for an open random port. it verifies the port is not
|
|
// already being used by another hvsock in the Windows registry.
|
|
func findOpenHVSockPort() (uint64, error) {
|
|
// If we cannot find a free port in 10 attempts, something is wrong
|
|
for i := 0; i < 10; i++ {
|
|
port, err := utils.GetRandomPort()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
// Try and load registry entries by port to see if they exist
|
|
_, err = LoadHVSockRegistryEntry(uint64(port))
|
|
if err == nil {
|
|
// the port is no good, it is being used; try again
|
|
logrus.Errorf("port %d is already used for hvsock", port)
|
|
continue
|
|
}
|
|
if errors.Is(err, registry.ErrNotExist) {
|
|
// the port is good to go
|
|
return uint64(port), nil
|
|
}
|
|
if err != nil {
|
|
// something went wrong
|
|
return 0, err
|
|
}
|
|
}
|
|
return 0, errors.New("unable to find a free port for hvsock use")
|
|
}
|
|
|
|
// NewHVSockRegistryEntry is a constructor to make a new registry entry in Windows. After making the new
|
|
// object, you must call the add() method to *actually* add it to the Windows registry.
|
|
func NewHVSockRegistryEntry(machineName string, purpose HVSockPurpose) (*HVSockRegistryEntry, error) {
|
|
// a so-called wildcard entry ... everything from FACB -> 6D3 is MS special sauce
|
|
// for a " linux vm". this first segment is hexi for the hvsock port number
|
|
//00000400-FACB-11E6-BD58-64006A7986D3
|
|
port, err := findOpenHVSockPort()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
r := HVSockRegistryEntry{
|
|
KeyName: portToKeyName(port),
|
|
Purpose: purpose,
|
|
Port: port,
|
|
MachineName: machineName,
|
|
}
|
|
if err := r.Add(); err != nil {
|
|
return nil, err
|
|
}
|
|
return &r, nil
|
|
}
|
|
|
|
func portToKeyName(port uint64) string {
|
|
// this could be flattened but given the complexity, I thought it might
|
|
// be more difficult to read
|
|
hexi := strings.ToUpper(fmt.Sprintf("%08x", port))
|
|
return fmt.Sprintf("%s-%s", hexi, LinuxVm)
|
|
}
|
|
|
|
func LoadHVSockRegistryEntry(port uint64) (*HVSockRegistryEntry, error) {
|
|
keyName := portToKeyName(port)
|
|
fqPath := fmt.Sprintf("%s\\%s", VsockRegistryPath, keyName)
|
|
k, err := openVSockRegistryEntry(fqPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
p, _, err := k.GetStringValue(HvsockPurpose)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
purpose, err := toHVSockPurpose(p)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
machineName, _, err := k.GetStringValue(HvsockMachineName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &HVSockRegistryEntry{
|
|
KeyName: keyName,
|
|
Purpose: purpose,
|
|
Port: port,
|
|
MachineName: machineName,
|
|
Key: k,
|
|
}, nil
|
|
}
|
|
|
|
// Listen is used on the windows side to listen for anything to come
|
|
// over the hvsock as a signal the vm is booted
|
|
func (hv *HVSockRegistryEntry) Listen() error {
|
|
n := winio.HvsockAddr{
|
|
VMID: winio.HvsockGUIDWildcard(), // When listening on the host side, use equiv of 0.0.0.0
|
|
ServiceID: winio.VsockServiceID(uint32(hv.Port)),
|
|
}
|
|
listener, err := winio.ListenHvsock(&n)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if err := listener.Close(); err != nil {
|
|
logrus.Error(err)
|
|
}
|
|
}()
|
|
conn, err := listener.Accept()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if err := conn.Close(); err != nil {
|
|
logrus.Error(err)
|
|
}
|
|
}()
|
|
// Right now we just listen for anything down the pipe (like qemu)
|
|
_, err = bufio.NewReader(conn).ReadString('\n')
|
|
return err
|
|
}
|