2019-05-14 04:43:52 +00:00
|
|
|
// +build linux
|
|
|
|
|
2017-08-16 18:17:25 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 The Kubernetes Authors All rights reserved.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package kvm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2017-10-17 21:35:47 +00:00
|
|
|
"crypto/rand"
|
2017-08-16 18:17:25 +00:00
|
|
|
"fmt"
|
2017-10-17 21:35:47 +00:00
|
|
|
"net"
|
2017-08-16 18:17:25 +00:00
|
|
|
"text/template"
|
|
|
|
|
|
|
|
libvirt "github.com/libvirt/libvirt-go"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
const domainTmpl = `
|
|
|
|
<domain type='kvm'>
|
|
|
|
<name>{{.MachineName}}</name>
|
|
|
|
<memory unit='MB'>{{.Memory}}</memory>
|
|
|
|
<vcpu>{{.CPU}}</vcpu>
|
|
|
|
<features>
|
|
|
|
<acpi/>
|
|
|
|
<apic/>
|
|
|
|
<pae/>
|
2019-03-24 08:45:16 +00:00
|
|
|
{{if .Hidden}}
|
2018-11-20 06:18:23 +00:00
|
|
|
<kvm>
|
|
|
|
<hidden state='on'/>
|
|
|
|
</kvm>
|
2019-03-24 08:45:16 +00:00
|
|
|
{{end}}
|
2017-08-16 18:17:25 +00:00
|
|
|
</features>
|
2018-02-21 09:40:33 +00:00
|
|
|
<cpu mode='host-passthrough'/>
|
2017-08-16 18:17:25 +00:00
|
|
|
<os>
|
|
|
|
<type>hvm</type>
|
|
|
|
<boot dev='cdrom'/>
|
|
|
|
<boot dev='hd'/>
|
|
|
|
<bootmenu enable='no'/>
|
|
|
|
</os>
|
|
|
|
<devices>
|
|
|
|
<disk type='file' device='cdrom'>
|
|
|
|
<source file='{{.ISO}}'/>
|
|
|
|
<target dev='hdc' bus='scsi'/>
|
|
|
|
<readonly/>
|
|
|
|
</disk>
|
|
|
|
<disk type='file' device='disk'>
|
|
|
|
<driver name='qemu' type='raw' cache='default' io='threads' />
|
|
|
|
<source file='{{.DiskPath}}'/>
|
|
|
|
<target dev='hda' bus='virtio'/>
|
|
|
|
</disk>
|
|
|
|
<interface type='network'>
|
|
|
|
<source network='{{.Network}}'/>
|
2017-10-17 21:35:47 +00:00
|
|
|
<mac address='{{.MAC}}'/>
|
2017-08-16 18:17:25 +00:00
|
|
|
<model type='virtio'/>
|
|
|
|
</interface>
|
|
|
|
<interface type='network'>
|
|
|
|
<source network='{{.PrivateNetwork}}'/>
|
2018-08-10 20:07:01 +00:00
|
|
|
<mac address='{{.PrivateMAC}}'/>
|
2017-08-16 18:17:25 +00:00
|
|
|
<model type='virtio'/>
|
|
|
|
</interface>
|
|
|
|
<serial type='pty'>
|
|
|
|
<target port='0'/>
|
|
|
|
</serial>
|
2018-06-08 19:52:33 +00:00
|
|
|
<console type='pty'>
|
|
|
|
<target type='serial' port='0'/>
|
2017-08-16 18:17:25 +00:00
|
|
|
</console>
|
|
|
|
<rng model='virtio'>
|
|
|
|
<backend model='random'>/dev/random</backend>
|
|
|
|
</rng>
|
2018-06-21 22:53:13 +00:00
|
|
|
{{if .GPU}}
|
|
|
|
{{.DevicesXML}}
|
|
|
|
{{end}}
|
2017-08-16 18:17:25 +00:00
|
|
|
</devices>
|
|
|
|
</domain>
|
|
|
|
`
|
|
|
|
|
2017-10-17 21:35:47 +00:00
|
|
|
func randomMAC() (net.HardwareAddr, error) {
|
|
|
|
buf := make([]byte, 6)
|
|
|
|
_, err := rand.Read(buf)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// We unset the first and second least significant bits (LSB) of the MAC
|
|
|
|
//
|
|
|
|
// The LSB of the first octet
|
|
|
|
// 0 for unicast
|
|
|
|
// 1 for multicast
|
|
|
|
//
|
|
|
|
// The second LSB of the first octet
|
|
|
|
// 0 for universally administered addresses
|
|
|
|
// 1 for locally administered addresses
|
2019-05-14 01:31:55 +00:00
|
|
|
buf[0] &= 0xfc
|
2017-10-17 21:35:47 +00:00
|
|
|
return buf, nil
|
|
|
|
}
|
|
|
|
|
2017-08-16 18:17:25 +00:00
|
|
|
func (d *Driver) getDomain() (*libvirt.Domain, *libvirt.Connect, error) {
|
2019-06-28 21:12:58 +00:00
|
|
|
conn, err := getConnection(d.ConnectionURI)
|
2017-08-16 18:17:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "getting domain")
|
|
|
|
}
|
|
|
|
|
|
|
|
dom, err := conn.LookupDomainByName(d.MachineName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "looking up domain")
|
|
|
|
}
|
|
|
|
|
|
|
|
return dom, conn, nil
|
|
|
|
}
|
|
|
|
|
2019-06-28 21:12:58 +00:00
|
|
|
func getConnection(connectionURI string) (*libvirt.Connect, error) {
|
|
|
|
conn, err := libvirt.NewConnect(connectionURI)
|
2017-08-16 18:17:25 +00:00
|
|
|
if err != nil {
|
2019-06-29 13:53:13 +00:00
|
|
|
return nil, errors.Wrap(err, "error connecting to libvirt socket.")
|
2017-08-16 18:17:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return conn, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func closeDomain(dom *libvirt.Domain, conn *libvirt.Connect) error {
|
2019-05-14 01:31:55 +00:00
|
|
|
if err := dom.Free(); err != nil {
|
|
|
|
return err
|
2017-08-16 18:17:25 +00:00
|
|
|
}
|
2019-05-14 01:31:55 +00:00
|
|
|
res, err := conn.Close()
|
|
|
|
if res != 0 {
|
2019-06-29 14:28:54 +00:00
|
|
|
return fmt.Errorf("conn.Close() == %d, expected 0", res)
|
2019-05-14 01:31:55 +00:00
|
|
|
}
|
|
|
|
return err
|
2017-08-16 18:17:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Driver) createDomain() (*libvirt.Domain, error) {
|
2018-08-10 19:30:35 +00:00
|
|
|
// create random MAC addresses first for our NICs
|
|
|
|
if d.MAC == "" {
|
|
|
|
mac, err := randomMAC()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "generating mac address")
|
|
|
|
}
|
|
|
|
d.MAC = mac.String()
|
|
|
|
}
|
|
|
|
|
2018-08-10 20:07:01 +00:00
|
|
|
if d.PrivateMAC == "" {
|
|
|
|
mac, err := randomMAC()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "generating mac address")
|
|
|
|
}
|
|
|
|
d.PrivateMAC = mac.String()
|
|
|
|
}
|
|
|
|
|
2018-08-10 19:30:35 +00:00
|
|
|
// create the XML for the domain using our domainTmpl template
|
2017-08-16 18:17:25 +00:00
|
|
|
tmpl := template.Must(template.New("domain").Parse(domainTmpl))
|
2018-08-10 19:30:35 +00:00
|
|
|
var domainXML bytes.Buffer
|
|
|
|
if err := tmpl.Execute(&domainXML, d); err != nil {
|
2017-08-16 18:17:25 +00:00
|
|
|
return nil, errors.Wrap(err, "executing domain xml")
|
|
|
|
}
|
|
|
|
|
2019-06-28 21:12:58 +00:00
|
|
|
conn, err := getConnection(d.ConnectionURI)
|
2017-08-16 18:17:25 +00:00
|
|
|
if err != nil {
|
2019-06-29 14:28:54 +00:00
|
|
|
return nil, errors.Wrap(err, "error getting libvirt connection")
|
2017-08-16 18:17:25 +00:00
|
|
|
}
|
|
|
|
defer conn.Close()
|
|
|
|
|
2018-08-10 19:30:35 +00:00
|
|
|
// define the domain in libvirt using the generated XML
|
|
|
|
dom, err := conn.DomainDefineXML(domainXML.String())
|
2017-08-16 18:17:25 +00:00
|
|
|
if err != nil {
|
2019-06-29 14:28:54 +00:00
|
|
|
return nil, errors.Wrapf(err, "error defining domain xml: %s", domainXML.String())
|
2017-08-16 18:17:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return dom, nil
|
|
|
|
}
|