2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
/*
|
2010-03-02 21:19:24 +00:00
|
|
|
* esx_vmx.c: VMX related functions for the VMware ESX driver
|
2009-07-23 20:21:08 +00:00
|
|
|
*
|
2010-03-01 23:38:28 +00:00
|
|
|
* Copyright (C) 2010 Red Hat, Inc.
|
2009-07-23 20:21:08 +00:00
|
|
|
* Copyright (C) 2009 Matthias Bolte <matthias.bolte@googlemail.com>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
#include <c-ctype.h>
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
#include "internal.h"
|
|
|
|
#include "virterror_internal.h"
|
|
|
|
#include "memory.h"
|
|
|
|
#include "logging.h"
|
2010-07-06 17:02:48 +00:00
|
|
|
#include "esx_vi_methods.h"
|
2010-04-02 19:34:31 +00:00
|
|
|
#include "esx_private.h"
|
2009-07-23 20:21:08 +00:00
|
|
|
#include "esx_util.h"
|
|
|
|
#include "esx_vmx.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
mapping:
|
|
|
|
|
|
|
|
domain-xml <=> vmx
|
|
|
|
|
|
|
|
|
|
|
|
config.version = "8" # essential
|
2009-07-27 12:27:09 +00:00
|
|
|
virtualHW.version = "4" # essential for ESX 3.5
|
|
|
|
virtualHW.version = "7" # essential for ESX 4.0
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
??? <=> guestOS = "<value>" # essential, FIXME: not representable
|
|
|
|
def->id = <value> <=> ??? # not representable
|
|
|
|
def->uuid = <value> <=> uuid.bios = "<value>"
|
|
|
|
def->name = <value> <=> displayName = "<value>"
|
|
|
|
def->maxmem = <value kilobyte> <=> memsize = "<value megabyte>" # must be a multiple of 4, defaults to 32
|
|
|
|
def->memory = <value kilobyte> <=> sched.mem.max = "<value megabyte>" # defaults to "unlimited" -> def->memory = def->maxmem
|
|
|
|
def->vcpus = <value> <=> numvcpus = "<value>" # must be 1 or a multiple of 2, defaults to 1
|
|
|
|
def->cpumask = <uint list> <=> sched.cpu.affinity = "<uint list>"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
## os ##########################################################################
|
|
|
|
|
|
|
|
def->os
|
|
|
|
|
|
|
|
->type = "hvm"
|
2009-09-23 12:16:41 +00:00
|
|
|
->arch = <arch> <=> guestOS = "<value>" # if <value> ends with -64 than <arch> is x86_64, otherwise <arch> is i686
|
2009-07-23 20:21:08 +00:00
|
|
|
->machine
|
|
|
|
->nBootDevs
|
|
|
|
->bootDevs
|
|
|
|
->init
|
|
|
|
->kernel
|
|
|
|
->initrd
|
|
|
|
->cmdline
|
|
|
|
->root
|
|
|
|
->loader
|
|
|
|
->bootloader
|
|
|
|
->bootloaderArgs
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
## disks #######################################################################
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
scsi[0..3]:[0..6,8..15] -> <controller>:<unit> with 1 bus per controller
|
|
|
|
ide[0..1]:[0..1] -> <bus>:<unit> with 1 controller
|
|
|
|
floppy[0..1] -> <unit> with 1 controller and 1 bus per controller
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
def->disks[0]...
|
|
|
|
|
|
|
|
## disks: scsi hard drive from .vmdk image #####################################
|
|
|
|
|
|
|
|
scsi0.present = "true" # defaults to "false"
|
|
|
|
scsi0:0.present = "true" # defaults to "false"
|
|
|
|
scsi0:0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
??? <=> scsi0:0.mode = "persistent" # defaults to "persistent"
|
|
|
|
scsi0:0.mode = "undoable"
|
|
|
|
scsi0:0.mode = "independent-persistent"
|
|
|
|
scsi0:0.mode = "independent-nonpersistent"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_FILE <=> scsi0:0.deviceType = "scsi-hardDisk" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_DISK <=> scsi0:0.deviceType = "scsi-hardDisk" # defaults to ?
|
|
|
|
->bus = _DISK_BUS_SCSI
|
|
|
|
->src = <value>.vmdk <=> scsi0:0.fileName = "<value>.vmdk"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = sd[<controller> * 15 + <unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName = <driver> <=> scsi0.virtualDev = "<driver>" # default depends on guestOS value
|
|
|
|
->driverType
|
|
|
|
->cachemode <=> scsi0:0.writeThrough = "<value>" # defaults to false, true -> _DISK_CACHE_WRITETHRU, false _DISK_CACHE_DEFAULT
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: ide hard drive from .vmdk image ######################################
|
|
|
|
|
|
|
|
ide0:0.present = "true" # defaults to "false"
|
|
|
|
ide0:0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
??? <=> ide0:0.mode = "persistent" # defaults to "persistent"
|
|
|
|
ide0:0.mode = "undoable"
|
|
|
|
ide0:0.mode = "independent-persistent"
|
|
|
|
ide0:0.mode = "independent-nonpersistent"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_FILE <=> ide0:0.deviceType = "ata-hardDisk" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_DISK <=> ide0:0.deviceType = "ata-hardDisk" # defaults to ?
|
|
|
|
->bus = _DISK_BUS_IDE
|
|
|
|
->src = <value>.vmdk <=> ide0:0.fileName = "<value>.vmdk"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = hd[<bus> * 2 + <unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName
|
|
|
|
->driverType
|
|
|
|
->cachemode <=> ide0:0.writeThrough = "<value>" # defaults to false, true -> _DISK_CACHE_WRITETHRU, false _DISK_CACHE_DEFAULT
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: scsi cdrom from .iso image ###########################################
|
|
|
|
|
|
|
|
scsi0.present = "true" # defaults to "false"
|
|
|
|
scsi0:0.present = "true" # defaults to "false"
|
|
|
|
scsi0:0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_FILE <=> scsi0:0.deviceType = "cdrom-image" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_CDROM <=> scsi0:0.deviceType = "cdrom-image" # defaults to ?
|
|
|
|
->bus = _DISK_BUS_SCSI
|
|
|
|
->src = <value>.iso <=> scsi0:0.fileName = "<value>.iso"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = sd[<controller> * 15 + <unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName = <driver> <=> scsi0.virtualDev = "<driver>" # default depends on guestOS value
|
|
|
|
->driverType
|
|
|
|
->cachemode
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: ide cdrom from .iso image ############################################
|
|
|
|
|
|
|
|
ide0:0.present = "true" # defaults to "false"
|
|
|
|
ide0:0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_FILE <=> ide0:0.deviceType = "cdrom-image" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_CDROM <=> ide0:0.deviceType = "cdrom-image" # defaults to ?
|
|
|
|
->bus = _DISK_BUS_IDE
|
|
|
|
->src = <value>.iso <=> ide0:0.fileName = "<value>.iso"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = hd[<bus> * 2 + <unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName
|
|
|
|
->driverType
|
|
|
|
->cachemode
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: scsi cdrom from host device ##########################################
|
|
|
|
|
|
|
|
scsi0.present = "true" # defaults to "false"
|
|
|
|
scsi0:0.present = "true" # defaults to "false"
|
|
|
|
scsi0:0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_BLOCK <=> scsi0:0.deviceType = "atapi-cdrom" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_CDROM <=> scsi0:0.deviceType = "atapi-cdrom" # defaults to ?
|
|
|
|
->bus = _DISK_BUS_SCSI
|
|
|
|
->src = <value> <=> scsi0:0.fileName = "<value>" # e.g. "/dev/scd0" ?
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = sd[<controller> * 15 + <unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName = <driver> <=> scsi0.virtualDev = "<driver>" # default depends on guestOS value
|
|
|
|
->driverType
|
|
|
|
->cachemode
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: ide cdrom from host device ###########################################
|
|
|
|
|
|
|
|
ide0:0.present = "true" # defaults to "false"
|
|
|
|
ide0:0.startConnected = "true" # defaults to "true"
|
|
|
|
ide0:0.clientDevice = "false" # defaults to "false"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_BLOCK <=> ide0:0.deviceType = "atapi-cdrom" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_CDROM <=> ide0:0.deviceType = "atapi-cdrom" # defaults to ?
|
|
|
|
->bus = _DISK_BUS_IDE
|
|
|
|
->src = <value> <=> ide0:0.fileName = "<value>" # e.g. "/dev/scd0"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = hd[<bus> * 2 + <unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName
|
|
|
|
->driverType
|
|
|
|
->cachemode
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: floppy from .flp image ###############################################
|
|
|
|
|
|
|
|
floppy0.present = "true" # defaults to "false"
|
|
|
|
floppy0.startConnected = "true" # defaults to "true"
|
|
|
|
floppy0.clientDevice = "false" # defaults to "false"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_FILE <=> floppy0.fileType = "file" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_FLOPPY
|
|
|
|
->bus = _DISK_BUS_FDC
|
|
|
|
->src = <value>.flp <=> floppy0.fileName = "<value>.flp"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = fd[<unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName
|
|
|
|
->driverType
|
|
|
|
->cachemode
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
## disks: floppy from host device ##############################################
|
|
|
|
|
|
|
|
floppy0.present = "true" # defaults to "false"
|
|
|
|
floppy0.startConnected = "true" # defaults to "true"
|
|
|
|
floppy0.clientDevice = "false" # defaults to "false"
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _DISK_TYPE_BLOCK <=> floppy0.fileType = "device" # defaults to ?
|
|
|
|
->device = _DISK_DEVICE_FLOPPY
|
|
|
|
->bus = _DISK_BUS_FDC
|
|
|
|
->src = <value> <=> floppy0.fileName = "<value>" # e.g. "/dev/fd0"
|
2010-06-17 17:57:12 +00:00
|
|
|
->dst = fd[<unit> mapped to [a-z]+]
|
2009-07-23 20:21:08 +00:00
|
|
|
->driverName
|
|
|
|
->driverType
|
|
|
|
->cachemode
|
|
|
|
->readonly
|
|
|
|
->shared
|
|
|
|
->slotnum
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
## nets ########################################################################
|
|
|
|
|
|
|
|
ethernet[0..3] -> <controller>
|
|
|
|
|
|
|
|
ethernet0.present = "true" # defaults to "false"
|
|
|
|
ethernet0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
ethernet0.networkName = "VM Network" # FIXME
|
|
|
|
|
|
|
|
def->nets[0]...
|
|
|
|
->model = <model> <=> ethernet0.virtualDev = "<model>" # default depends on guestOS value
|
2010-04-17 09:43:01 +00:00
|
|
|
ethernet0.features = "15" # if present and virtualDev is "vmxnet" => vmxnet2 (enhanced)
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
ethernet0.addressType = "generated" # default to "generated"
|
|
|
|
->mac = <value> <=> ethernet0.generatedAddress = "<value>"
|
2010-01-02 22:58:24 +00:00
|
|
|
ethernet0.generatedAddressOffset = "0" # ?
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
ethernet0.addressType = "static" # default to "generated"
|
|
|
|
->mac = <value> <=> ethernet0.address = "<value>"
|
|
|
|
|
2009-09-02 14:22:17 +00:00
|
|
|
|
|
|
|
ethernet0.addressType = "vpx" # default to "generated"
|
|
|
|
->mac = <value> <=> ethernet0.generatedAddress = "<value>"
|
|
|
|
|
2010-01-02 22:58:24 +00:00
|
|
|
|
|
|
|
ethernet0.addressType = "static" # default to "generated"
|
|
|
|
->mac = <value> <=> ethernet0.address = "<value>"
|
|
|
|
ethernet0.checkMACAddress = "false" # mac address outside the VMware prefixes
|
|
|
|
|
2009-11-14 18:51:05 +00:00
|
|
|
# 00:0c:29 prefix for autogenerated mac's -> ethernet0.addressType = "generated"
|
2009-07-23 20:21:08 +00:00
|
|
|
# 00:50:56 prefix for manual configured mac's
|
2009-11-14 18:51:05 +00:00
|
|
|
# 00:50:56:00:00:00 - 00:50:56:3f:ff:ff -> ethernet0.addressType = "static"
|
2010-01-02 22:58:24 +00:00
|
|
|
# 00:50:56:80:00:00 - 00:50:56:bf:ff:ff -> ethernet0.addressType = "vpx"
|
2009-07-23 20:21:08 +00:00
|
|
|
# 00:05:69 old prefix from esx 1.5
|
|
|
|
|
|
|
|
|
|
|
|
## nets: bridged ###############################################################
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _NET_TYPE_BRIDGE <=> ethernet0.connectionType = "bridged" # defaults to "bridged"
|
2009-09-23 12:16:41 +00:00
|
|
|
->data.bridge.brname = <value> <=> ethernet0.networkName = "<value>"
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
## nets: hostonly ##############################################################
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
... # FIXME: Investigate if ESX supports this
|
2009-07-23 20:21:08 +00:00
|
|
|
->type = _NET_TYPE_NETWORK <=> ethernet0.connectionType = "hostonly" # defaults to "bridged"
|
|
|
|
|
|
|
|
|
|
|
|
## nets: nat ###################################################################
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
... # FIXME: Investigate if ESX supports this
|
2009-07-23 20:21:08 +00:00
|
|
|
->type = _NET_TYPE_USER <=> ethernet0.connectionType = "nat" # defaults to "bridged"
|
|
|
|
|
|
|
|
|
|
|
|
## nets: custom ################################################################
|
|
|
|
|
|
|
|
...
|
|
|
|
->type = _NET_TYPE_BRIDGE <=> ethernet0.connectionType = "custom" # defaults to "bridged"
|
2009-09-23 12:16:41 +00:00
|
|
|
->data.bridge.brname = <value> <=> ethernet0.networkName = "<value>"
|
|
|
|
->ifname = <value> <=> ethernet0.vnet = "<value>"
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
## serials #####################################################################
|
|
|
|
|
|
|
|
serial[0..3] -> <port>
|
|
|
|
|
|
|
|
serial0.present = "true" # defaults to "false"
|
|
|
|
serial0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
def->serials[0]...
|
2009-11-05 13:19:14 +00:00
|
|
|
->target.port = <port>
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
## serials: device #############################################################
|
|
|
|
|
|
|
|
->type = _CHR_TYPE_DEV <=> serial0.fileType = "device"
|
|
|
|
->data.file.path = <value> <=> serial0.fileName = "<value>" # e.g. "/dev/ttyS0"
|
|
|
|
??? <=> serial0.tryNoRxLoss = "false" # defaults to "false", FIXME: not representable
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.yieldOnMsrRead = "true" # defaults to "false", FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
## serials: file ###############################################################
|
|
|
|
|
|
|
|
->type = _CHR_TYPE_FILE <=> serial0.fileType = "file"
|
|
|
|
->data.file.path = <value> <=> serial0.fileName = "<value>" # e.g. "serial0.file"
|
|
|
|
??? <=> serial0.tryNoRxLoss = "false" # defaults to "false", FIXME: not representable
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.yieldOnMsrRead = "true" # defaults to "false", FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
## serials: pipe, far end -> app ###############################################
|
|
|
|
|
|
|
|
->type = _CHR_TYPE_PIPE <=> serial0.fileType = "pipe"
|
|
|
|
->data.file.path = <value> <=> serial0.fileName = "<value>" # e.g. "serial0.pipe"
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.pipe.endPoint = "client" # defaults to ?, FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
??? <=> serial0.tryNoRxLoss = "true" # defaults to "false", FIXME: not representable
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.yieldOnMsrRead = "true" # defaults to "false", FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
->type = _CHR_TYPE_PIPE <=> serial0.fileType = "pipe"
|
|
|
|
->data.file.path = <value> <=> serial0.fileName = "<value>" # e.g. "serial0.pipe"
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.pipe.endPoint = "server" # defaults to ?, FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
??? <=> serial0.tryNoRxLoss = "true" # defaults to "false", FIXME: not representable
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.yieldOnMsrRead = "true" # defaults to "false", FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
## serials: pipe, far end -> vm ################################################
|
|
|
|
|
|
|
|
->type = _CHR_TYPE_PIPE <=> serial0.fileType = "pipe"
|
|
|
|
->data.file.path = <value> <=> serial0.fileName = "<value>" # e.g. "serial0.pipe"
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.pipe.endPoint = "client" # defaults to ?, FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
??? <=> serial0.tryNoRxLoss = "false" # defaults to "false", FIXME: not representable
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.yieldOnMsrRead = "true" # defaults to "false", FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
->type = _CHR_TYPE_PIPE <=> serial0.fileType = "pipe"
|
|
|
|
->data.file.path = <value> <=> serial0.fileName = "<value>" # e.g. "serial0.pipe"
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.pipe.endPoint = "server" # defaults to ?, FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
??? <=> serial0.tryNoRxLoss = "false" # defaults to "false", FIXME: not representable
|
2009-09-23 12:16:41 +00:00
|
|
|
??? <=> serial0.yieldOnMsrRead = "true" # defaults to "false", FIXME: not representable
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
## parallels ###################################################################
|
|
|
|
|
|
|
|
parallel[0..2] -> <port>
|
|
|
|
|
|
|
|
parallel0.present = "true" # defaults to "false"
|
|
|
|
parallel0.startConnected = "true" # defaults to "true"
|
|
|
|
|
|
|
|
def->parallels[0]...
|
2009-11-05 13:19:14 +00:00
|
|
|
->target.port = <port>
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
## parallels: device #############################################################
|
|
|
|
|
|
|
|
->type = _CHR_TYPE_DEV <=> parallel0.fileType = "device"
|
|
|
|
->data.file.path = <value> <=> parallel0.fileName = "<value>" # e.g. "/dev/parport0"
|
|
|
|
??? <=> parallel0.bidirectional = "<value>" # defaults to ?, FIXME: not representable
|
|
|
|
|
|
|
|
|
|
|
|
## parallels: file #############################################################
|
|
|
|
|
|
|
|
->type = _CHR_TYPE_FILE <=> parallel0.fileType = "file"
|
|
|
|
->data.file.path = <value> <=> parallel0.fileName = "<value>" # e.g. "parallel0.file"
|
|
|
|
??? <=> parallel0.bidirectional = "<value>" # must be "false" for fileType = "file", FIXME: not representable
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
## sound #######################################################################
|
|
|
|
|
|
|
|
sound.present = "true" # defaults to "false"
|
|
|
|
sound.startConnected = "true" # defaults to "true"
|
|
|
|
sound.autodetect = "true"
|
|
|
|
sound.fileName = "-1"
|
|
|
|
|
|
|
|
FIXME: Investigate if ESX supports this,
|
|
|
|
at least the VI Client GUI has no
|
|
|
|
options to add a sound device, but
|
|
|
|
the VI API contains a VirtualSoundCard
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_ESX
|
|
|
|
|
|
|
|
#define ESX_BUILD_VMX_NAME(_suffix) \
|
2009-08-03 11:39:07 +00:00
|
|
|
snprintf(_suffix##_name, sizeof(_suffix##_name), "%s."#_suffix, prefix);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2010-06-24 14:58:54 +00:00
|
|
|
/* directly map the virDomainControllerModel to esxVMX_SCSIControllerModel,
|
|
|
|
* this is good enough for now because all virDomainControllerModel values
|
|
|
|
* are actually SCSI controller models in the ESX case */
|
|
|
|
VIR_ENUM_DECL(esxVMX_SCSIControllerModel)
|
|
|
|
VIR_ENUM_IMPL(esxVMX_SCSIControllerModel, VIR_DOMAIN_CONTROLLER_MODEL_LAST,
|
2010-07-06 17:02:48 +00:00
|
|
|
"auto", /* just to match virDomainControllerModel, will never be used */
|
2010-06-24 14:58:54 +00:00
|
|
|
"buslogic",
|
|
|
|
"lsilogic",
|
|
|
|
"lsisas1068",
|
|
|
|
"pvscsi");
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
int
|
2010-06-17 17:57:12 +00:00
|
|
|
esxVMX_SCSIDiskNameToControllerAndUnit(const char *name, int *controller, int *unit)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if (! STRPREFIX(name, "sd")) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Expecting domain XML attribute 'dev' of entry "
|
|
|
|
"'devices/disk/target' to start with 'sd'"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
idx = virDiskNameToIndex(name);
|
|
|
|
|
|
|
|
if (idx < 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Could not parse valid disk index from '%s'"), name);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* Each of the 4 SCSI controllers has 1 bus with 15 units each for devices */
|
2009-09-23 12:16:41 +00:00
|
|
|
if (idx >= (4 * 15)) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("SCSI disk index (parsed from '%s') is too large"), name);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*controller = idx / 15;
|
2010-06-17 17:57:12 +00:00
|
|
|
*unit = idx % 15;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* Skip the controller ifself at unit 7 */
|
|
|
|
if (*unit >= 7) {
|
|
|
|
++(*unit);
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-06-17 17:57:12 +00:00
|
|
|
esxVMX_IDEDiskNameToBusAndUnit(const char *name, int *bus, int *unit)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if (! STRPREFIX(name, "hd")) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Expecting domain XML attribute 'dev' of entry "
|
|
|
|
"'devices/disk/target' to start with 'hd'"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
idx = virDiskNameToIndex(name);
|
|
|
|
|
|
|
|
if (idx < 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Could not parse valid disk index from '%s'"), name);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* The IDE controller has 2 buses with 2 units each for devices */
|
2009-09-23 12:16:41 +00:00
|
|
|
if (idx >= (2 * 2)) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("IDE disk index (parsed from '%s') is too large"), name);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
*bus = idx / 2;
|
|
|
|
*unit = idx % 2;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-06-17 17:57:12 +00:00
|
|
|
esxVMX_FloppyDiskNameToUnit(const char *name, int *unit)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if (! STRPREFIX(name, "fd")) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Expecting domain XML attribute 'dev' of entry "
|
|
|
|
"'devices/disk/target' to start with 'fd'"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
idx = virDiskNameToIndex(name);
|
|
|
|
|
|
|
|
if (idx < 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Could not parse valid disk index from '%s'"), name);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* The FDC controller has 1 bus with 2 units for devices */
|
2009-09-23 12:16:41 +00:00
|
|
|
if (idx >= 2) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Floppy disk index (parsed from '%s') is too large"), name);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
*unit = idx;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
esxVMX_VerifyDiskAddress(virCapsPtr caps, virDomainDiskDefPtr disk)
|
|
|
|
{
|
|
|
|
virDomainDiskDef def;
|
|
|
|
virDomainDeviceDriveAddressPtr drive;
|
|
|
|
|
|
|
|
memset(&def, 0, sizeof(def));
|
|
|
|
|
|
|
|
if (disk->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DRIVE) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unsupported disk address type '%s'"),
|
|
|
|
virDomainDeviceAddressTypeToString(disk->info.type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
drive = &disk->info.addr.drive;
|
|
|
|
|
|
|
|
def.dst = disk->dst;
|
|
|
|
def.bus = disk->bus;
|
|
|
|
|
|
|
|
if (virDomainDiskDefAssignAddress(caps, &def) < 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Could not verify disk address"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def.info.addr.drive.controller != drive->controller ||
|
|
|
|
def.info.addr.drive.bus != drive->bus ||
|
|
|
|
def.info.addr.drive.unit != drive->unit) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Disk address %d:%d:%d doesn't match target device '%s'"),
|
|
|
|
drive->controller, drive->bus, drive->unit, disk->dst);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* drive->{controller|bus|unit} is unsigned, no >= 0 checks are necessary */
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
if (drive->controller > 3) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("SCSI controller index %d out of [0..3] range"),
|
|
|
|
drive->controller);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive->bus != 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("SCSI bus index %d out of [0] range"),
|
|
|
|
drive->bus);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive->unit > 15 || drive->unit == 7) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("SCSI unit index %d out of [0..6,8..15] range"),
|
|
|
|
drive->unit);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (disk->bus == VIR_DOMAIN_DISK_BUS_IDE) {
|
|
|
|
if (drive->controller != 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("IDE controller index %d out of [0] range"),
|
|
|
|
drive->controller);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive->bus > 1) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("IDE bus index %d out of [0..1] range"),
|
|
|
|
drive->bus);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive->unit > 1) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("IDE unit index %d out of [0..1] range"),
|
|
|
|
drive->unit);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (disk->bus == VIR_DOMAIN_DISK_BUS_FDC) {
|
|
|
|
if (drive->controller != 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("FDC controller index %d out of [0] range"),
|
|
|
|
drive->controller);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive->bus != 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("FDC bus index %d out of [0] range"),
|
|
|
|
drive->bus);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive->unit > 1) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("FDC unit index %d out of [0..1] range"),
|
|
|
|
drive->unit);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unsupported bus type '%s'"),
|
|
|
|
virDomainDiskBusTypeToString(disk->bus));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
esxVMX_HandleLegacySCSIDiskDriverName(virDomainDefPtr def,
|
|
|
|
virDomainDiskDefPtr disk)
|
|
|
|
{
|
|
|
|
char *tmp;
|
|
|
|
int model, i;
|
|
|
|
virDomainControllerDefPtr controller = NULL;
|
|
|
|
|
|
|
|
if (disk->bus != VIR_DOMAIN_DISK_BUS_SCSI || disk->driverName == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = disk->driverName;
|
|
|
|
|
|
|
|
for (; *tmp != '\0'; ++tmp) {
|
|
|
|
*tmp = c_tolower(*tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
model = virDomainControllerModelTypeFromString(disk->driverName);
|
|
|
|
|
|
|
|
if (model < 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unknown driver name '%s'"), disk->driverName);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < def->ncontrollers; ++i) {
|
|
|
|
if (def->controllers[i]->idx == disk->info.addr.drive.controller) {
|
|
|
|
controller = def->controllers[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (controller == NULL) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Missing SCSI controller for index %d"),
|
|
|
|
disk->info.addr.drive.controller);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (controller->model == -1) {
|
|
|
|
controller->model = model;
|
|
|
|
} else if (controller->model != model) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Inconsistent SCSI controller model ('%s' is not '%s') "
|
|
|
|
"for SCSI controller index %d"), disk->driverName,
|
|
|
|
virDomainControllerModelTypeToString(controller->model),
|
|
|
|
controller->idx);
|
|
|
|
return -1;
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-07-06 17:02:48 +00:00
|
|
|
esxVMX_AutodetectSCSIControllerModel(esxVI_Context *ctx,
|
|
|
|
virDomainDiskDefPtr def, int *model)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
2010-07-06 17:02:48 +00:00
|
|
|
int result = -1;
|
|
|
|
char *datastoreName = NULL;
|
|
|
|
char *directoryName = NULL;
|
|
|
|
char *fileName = NULL;
|
|
|
|
char *datastorePath = NULL;
|
|
|
|
esxVI_String *propertyNameList = NULL;
|
|
|
|
esxVI_ObjectContent *datastore = NULL;
|
|
|
|
esxVI_ManagedObjectReference *hostDatastoreBrowser = NULL;
|
|
|
|
esxVI_HostDatastoreBrowserSearchSpec *searchSpec = NULL;
|
|
|
|
esxVI_VmDiskFileQuery *vmDiskFileQuery = NULL;
|
|
|
|
esxVI_ManagedObjectReference *task = NULL;
|
|
|
|
esxVI_TaskInfoState taskInfoState;
|
|
|
|
esxVI_TaskInfo *taskInfo = NULL;
|
|
|
|
esxVI_HostDatastoreBrowserSearchResults *searchResults = NULL;
|
|
|
|
esxVI_VmDiskFileInfo *vmDiskFileInfo = NULL;
|
|
|
|
|
|
|
|
if (def->device != VIR_DOMAIN_DISK_DEVICE_DISK ||
|
|
|
|
def->bus != VIR_DOMAIN_DISK_BUS_SCSI ||
|
|
|
|
def->type != VIR_DOMAIN_DISK_TYPE_FILE ||
|
|
|
|
def->src == NULL ||
|
|
|
|
! STRPREFIX(def->src, "[")) {
|
|
|
|
/*
|
|
|
|
* This isn't a file-based SCSI disk device with a datastore related
|
|
|
|
* source path => do nothing.
|
|
|
|
*/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (esxUtil_ParseDatastoreRelatedPath(def->src, &datastoreName,
|
|
|
|
&directoryName, &fileName) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (directoryName == NULL) {
|
|
|
|
if (virAsprintf(&datastorePath, "[%s]", datastoreName) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (virAsprintf(&datastorePath, "[%s] %s", datastoreName,
|
|
|
|
directoryName) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Lookup HostDatastoreBrowser */
|
|
|
|
if (esxVI_String_AppendValueToList(&propertyNameList, "browser") < 0 ||
|
|
|
|
esxVI_LookupDatastoreByName(ctx, datastoreName, propertyNameList,
|
|
|
|
&datastore,
|
|
|
|
esxVI_Occurrence_RequiredItem) < 0 ||
|
|
|
|
esxVI_GetManagedObjectReference(datastore, "browser",
|
|
|
|
&hostDatastoreBrowser,
|
|
|
|
esxVI_Occurrence_RequiredItem) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Build HostDatastoreBrowserSearchSpec */
|
|
|
|
if (esxVI_HostDatastoreBrowserSearchSpec_Alloc(&searchSpec) < 0 ||
|
|
|
|
esxVI_FileQueryFlags_Alloc(&searchSpec->details) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
searchSpec->details->fileType = esxVI_Boolean_True;
|
|
|
|
searchSpec->details->fileSize = esxVI_Boolean_False;
|
|
|
|
searchSpec->details->modification = esxVI_Boolean_False;
|
|
|
|
|
|
|
|
if (esxVI_VmDiskFileQuery_Alloc(&vmDiskFileQuery) < 0 ||
|
|
|
|
esxVI_VmDiskFileQueryFlags_Alloc(&vmDiskFileQuery->details) < 0 ||
|
|
|
|
esxVI_FileQuery_AppendToList
|
|
|
|
(&searchSpec->query,
|
|
|
|
esxVI_FileQuery_DynamicCast(vmDiskFileQuery)) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
vmDiskFileQuery->details->diskType = esxVI_Boolean_False;
|
|
|
|
vmDiskFileQuery->details->capacityKb = esxVI_Boolean_False;
|
|
|
|
vmDiskFileQuery->details->hardwareVersion = esxVI_Boolean_False;
|
|
|
|
vmDiskFileQuery->details->controllerType = esxVI_Boolean_True;
|
|
|
|
vmDiskFileQuery->details->diskExtents = esxVI_Boolean_False;
|
|
|
|
|
|
|
|
if (esxVI_String_Alloc(&searchSpec->matchPattern) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
searchSpec->matchPattern->value = fileName;
|
|
|
|
|
|
|
|
/* Search datastore for file */
|
|
|
|
if (esxVI_SearchDatastore_Task(ctx, hostDatastoreBrowser, datastorePath,
|
|
|
|
searchSpec, &task) < 0 ||
|
|
|
|
esxVI_WaitForTaskCompletion(ctx, task, NULL, esxVI_Boolean_False,
|
|
|
|
&taskInfoState) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (taskInfoState != esxVI_TaskInfoState_Success) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not serach in datastore '%s'"), datastoreName);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (esxVI_LookupTaskInfoByTask(ctx, task, &taskInfo) < 0 ||
|
|
|
|
esxVI_HostDatastoreBrowserSearchResults_CastFromAnyType
|
|
|
|
(taskInfo->result, &searchResults) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Interpret search result */
|
|
|
|
vmDiskFileInfo = esxVI_VmDiskFileInfo_DynamicCast(searchResults->file);
|
|
|
|
|
|
|
|
if (vmDiskFileInfo == NULL || vmDiskFileInfo->controllerType == NULL) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not lookup controller model for '%s'"), def->src);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STRCASEEQ(vmDiskFileInfo->controllerType,
|
|
|
|
"VirtualBusLogicController")) {
|
|
|
|
*model = VIR_DOMAIN_CONTROLLER_MODEL_BUSLOGIC;
|
|
|
|
} else if (STRCASEEQ(vmDiskFileInfo->controllerType,
|
|
|
|
"VirtualLsiLogicController")) {
|
|
|
|
*model = VIR_DOMAIN_CONTROLLER_MODEL_LSILOGIC;
|
|
|
|
} else if (STRCASEEQ(vmDiskFileInfo->controllerType,
|
|
|
|
"VirtualLsiLogicSASController")) {
|
|
|
|
*model = VIR_DOMAIN_CONTROLLER_MODEL_LSISAS1068;
|
|
|
|
} else if (STRCASEEQ(vmDiskFileInfo->controllerType,
|
|
|
|
"ParaVirtualSCSIController")) {
|
|
|
|
*model = VIR_DOMAIN_CONTROLLER_MODEL_VMPVSCSI;
|
|
|
|
} else {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Found unexpected controller model '%s' for disk '%s'"),
|
|
|
|
vmDiskFileInfo->controllerType, def->src);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
/* Don't double free fileName */
|
|
|
|
if (searchSpec != NULL && searchSpec->matchPattern != NULL) {
|
|
|
|
searchSpec->matchPattern->value = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(datastoreName);
|
|
|
|
VIR_FREE(directoryName);
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
VIR_FREE(datastorePath);
|
|
|
|
esxVI_String_Free(&propertyNameList);
|
|
|
|
esxVI_ObjectContent_Free(&datastore);
|
|
|
|
esxVI_ManagedObjectReference_Free(&hostDatastoreBrowser);
|
|
|
|
esxVI_HostDatastoreBrowserSearchSpec_Free(&searchSpec);
|
|
|
|
esxVI_ManagedObjectReference_Free(&task);
|
|
|
|
esxVI_TaskInfo_Free(&taskInfo);
|
|
|
|
esxVI_HostDatastoreBrowserSearchResults_Free(&searchResults);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
esxVMX_GatherSCSIControllers(esxVI_Context *ctx, virDomainDefPtr def,
|
|
|
|
int virtualDev[4], bool present[4])
|
|
|
|
{
|
|
|
|
int result = -1;
|
|
|
|
int i, k;
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskDefPtr disk;
|
2010-07-06 17:02:48 +00:00
|
|
|
virDomainControllerDefPtr controller;
|
|
|
|
bool controllerHasDisksAttached;
|
|
|
|
int count = 0;
|
|
|
|
int *autodetectedModels;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
if (VIR_ALLOC_N(autodetectedModels, def->ndisks) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < def->ncontrollers; ++i) {
|
|
|
|
controller = def->controllers[i];
|
2009-09-23 12:16:41 +00:00
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
if (controller->type != VIR_DOMAIN_CONTROLLER_TYPE_SCSI) {
|
|
|
|
// skip non-SCSI controllers
|
2009-09-23 12:16:41 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
controllerHasDisksAttached = false;
|
2010-06-17 17:57:12 +00:00
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
for (k = 0; k < def->ndisks; ++k) {
|
|
|
|
disk = def->disks[k];
|
|
|
|
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_SCSI &&
|
|
|
|
disk->info.addr.drive.controller == controller->idx) {
|
|
|
|
controllerHasDisksAttached = true;
|
2010-06-17 17:57:12 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
if (! controllerHasDisksAttached) {
|
|
|
|
// skip SCSI controllers without attached disks
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx != NULL &&
|
|
|
|
controller->model == VIR_DOMAIN_CONTROLLER_MODEL_AUTO) {
|
|
|
|
count = 0;
|
|
|
|
|
|
|
|
// try to autodetect the SCSI controller model by collecting
|
|
|
|
// SCSI controller model of all disks attached to this controller
|
|
|
|
for (k = 0; k < def->ndisks; ++k) {
|
|
|
|
disk = def->disks[k];
|
|
|
|
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_SCSI &&
|
|
|
|
disk->info.addr.drive.controller == controller->idx) {
|
|
|
|
if (esxVMX_AutodetectSCSIControllerModel
|
|
|
|
(ctx, disk, &autodetectedModels[count]) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// autodetection fails when the disks attached to one controller
|
|
|
|
// have inconsistent SCSI controller models
|
|
|
|
for (k = 0; k < count; ++k) {
|
|
|
|
if (autodetectedModels[k] != autodetectedModels[0]) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Disks on SCSI controller %d have inconsistent "
|
|
|
|
"controller models, cannot autodetect model"),
|
|
|
|
controller->idx);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
controller->model = autodetectedModels[0];
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (controller->model != -1 &&
|
|
|
|
controller->model != VIR_DOMAIN_CONTROLLER_MODEL_BUSLOGIC &&
|
|
|
|
controller->model != VIR_DOMAIN_CONTROLLER_MODEL_LSILOGIC &&
|
2010-06-24 14:58:54 +00:00
|
|
|
controller->model != VIR_DOMAIN_CONTROLLER_MODEL_LSISAS1068 &&
|
|
|
|
controller->model != VIR_DOMAIN_CONTROLLER_MODEL_VMPVSCSI) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-06-17 17:57:12 +00:00
|
|
|
_("Expecting domain XML attribute 'model' of entry "
|
|
|
|
"'controller' to be 'buslogic' or 'lsilogic' or "
|
2010-06-24 14:58:54 +00:00
|
|
|
"'lsisas1068' or 'vmpvscsi' but found '%s'"),
|
2010-06-17 17:57:12 +00:00
|
|
|
virDomainControllerModelTypeToString(controller->model));
|
2010-07-06 17:02:48 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
2010-06-17 17:57:12 +00:00
|
|
|
|
|
|
|
present[controller->idx] = true;
|
|
|
|
virtualDev[controller->idx] = controller->model;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
result = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(autodetectedModels);
|
|
|
|
|
|
|
|
return result;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
char *
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_AbsolutePathToDatastoreRelatedPath(esxVI_Context *ctx,
|
2009-09-23 12:52:36 +00:00
|
|
|
const char *absolutePath)
|
|
|
|
{
|
2010-05-19 20:59:32 +00:00
|
|
|
bool success = false;
|
2010-04-13 22:03:12 +00:00
|
|
|
char *copyOfAbsolutePath = NULL;
|
|
|
|
char *tmp = NULL;
|
|
|
|
char *saveptr = NULL;
|
2009-09-23 12:52:36 +00:00
|
|
|
char *datastoreRelatedPath = NULL;
|
|
|
|
char *preliminaryDatastoreName = NULL;
|
|
|
|
char *directoryAndFileName = NULL;
|
|
|
|
esxVI_DynamicProperty *dynamicProperty = NULL;
|
|
|
|
esxVI_ObjectContent *datastore = NULL;
|
|
|
|
const char *datastoreName = NULL;
|
|
|
|
|
2010-04-13 22:03:12 +00:00
|
|
|
if (esxVI_String_DeepCopyValue(©OfAbsolutePath, absolutePath) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
return NULL;
|
2010-04-13 22:03:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Expected format: '/vmfs/volumes/<datastore>/<path>' */
|
|
|
|
if ((tmp = STRSKIP(copyOfAbsolutePath, "/vmfs/volumes/")) == NULL ||
|
|
|
|
(preliminaryDatastoreName = strtok_r(tmp, "/", &saveptr)) == NULL ||
|
|
|
|
(directoryAndFileName = strtok_r(NULL, "", &saveptr)) == NULL) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Absolute path '%s' doesn't have expected format "
|
|
|
|
"'/vmfs/volumes/<datastore>/<path>'"), absolutePath);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx != NULL) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVI_LookupDatastoreByName(ctx, preliminaryDatastoreName,
|
2009-09-23 12:52:36 +00:00
|
|
|
NULL, &datastore,
|
2009-11-22 20:18:45 +00:00
|
|
|
esxVI_Occurrence_OptionalItem) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
|
2009-10-16 23:42:05 +00:00
|
|
|
if (datastore != NULL) {
|
|
|
|
for (dynamicProperty = datastore->propSet; dynamicProperty != NULL;
|
|
|
|
dynamicProperty = dynamicProperty->_next) {
|
|
|
|
if (STREQ(dynamicProperty->name, "summary.accessible")) {
|
|
|
|
/* Ignore it */
|
|
|
|
} else if (STREQ(dynamicProperty->name, "summary.name")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVI_AnyType_ExpectType(dynamicProperty->val,
|
2009-10-16 23:42:05 +00:00
|
|
|
esxVI_Type_String) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-10-16 23:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
datastoreName = dynamicProperty->val->string;
|
|
|
|
break;
|
|
|
|
} else if (STREQ(dynamicProperty->name, "summary.url")) {
|
|
|
|
/* Ignore it */
|
|
|
|
} else {
|
|
|
|
VIR_WARN("Unexpected '%s' property", dynamicProperty->name);
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (datastoreName == NULL) {
|
2009-10-16 23:42:05 +00:00
|
|
|
VIR_WARN("Could not retrieve datastore name for absolute "
|
|
|
|
"path '%s', falling back to preliminary name '%s'",
|
|
|
|
absolutePath, preliminaryDatastoreName);
|
|
|
|
|
|
|
|
datastoreName = preliminaryDatastoreName;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
datastoreName = preliminaryDatastoreName;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virAsprintf(&datastoreRelatedPath, "[%s] %s", datastoreName,
|
|
|
|
directoryAndFileName) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Check if referenced path/file really exists */
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
success = true;
|
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (! success) {
|
|
|
|
VIR_FREE(datastoreRelatedPath);
|
|
|
|
}
|
|
|
|
|
2010-04-13 22:03:12 +00:00
|
|
|
VIR_FREE(copyOfAbsolutePath);
|
2009-09-23 12:52:36 +00:00
|
|
|
esxVI_ObjectContent_Free(&datastore);
|
|
|
|
|
|
|
|
return datastoreRelatedPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
* VMX -> Domain XML
|
|
|
|
*/
|
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
char *
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_ParseFileName(esxVI_Context *ctx, const char *fileName,
|
|
|
|
const char *datastoreName, const char *directoryName)
|
2009-09-23 12:52:36 +00:00
|
|
|
{
|
|
|
|
char *src = NULL;
|
|
|
|
|
|
|
|
if (STRPREFIX(fileName, "/vmfs/volumes/")) {
|
|
|
|
/* Found absolute path referencing a file inside a datastore */
|
2010-01-15 22:05:26 +00:00
|
|
|
return esxVMX_AbsolutePathToDatastoreRelatedPath(ctx, fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
} else if (STRPREFIX(fileName, "/")) {
|
|
|
|
/* Found absolute path referencing a file outside a datastore */
|
|
|
|
src = strdup(fileName);
|
|
|
|
|
|
|
|
if (src == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-09-23 12:52:36 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Check if referenced path/file really exists */
|
|
|
|
|
|
|
|
return src;
|
|
|
|
} else if (strchr(fileName, '/') != NULL) {
|
|
|
|
/* Found relative path, this is not supported */
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Found relative path '%s' in VMX file, this is not "
|
|
|
|
"supported"), fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
return NULL;
|
|
|
|
} else {
|
|
|
|
/* Found single file name referencing a file inside a datastore */
|
|
|
|
if (virAsprintf(&src, "[%s] %s/%s", datastoreName, directoryName,
|
|
|
|
fileName) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-09-23 12:52:36 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Check if referenced path/file really exists */
|
|
|
|
|
|
|
|
return src;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
virDomainDefPtr
|
2010-06-17 17:57:12 +00:00
|
|
|
esxVMX_ParseConfig(esxVI_Context *ctx, virCapsPtr caps, const char *vmx,
|
2009-09-23 12:52:36 +00:00
|
|
|
const char *datastoreName, const char *directoryName,
|
2010-04-18 16:04:24 +00:00
|
|
|
esxVI_ProductVersion productVersion)
|
2009-07-23 20:21:08 +00:00
|
|
|
{
|
2010-05-19 20:59:32 +00:00
|
|
|
bool success = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
virConfPtr conf = NULL;
|
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
long long config_version = 0;
|
|
|
|
long long virtualHW_version = 0;
|
|
|
|
long long memsize = 0;
|
|
|
|
long long memory = 0;
|
|
|
|
long long numvcpus = 0;
|
|
|
|
char *sched_cpu_affinity = NULL;
|
2009-09-23 12:16:41 +00:00
|
|
|
char *guestOS = NULL;
|
2009-07-23 20:21:08 +00:00
|
|
|
int controller;
|
2010-06-17 17:57:12 +00:00
|
|
|
int bus;
|
2009-07-23 20:21:08 +00:00
|
|
|
int port;
|
2010-06-18 16:16:22 +00:00
|
|
|
bool present;
|
2010-06-17 17:57:12 +00:00
|
|
|
int scsi_virtualDev[4] = { -1, -1, -1, -1 };
|
|
|
|
int unit;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
conf = virConfReadMem(vmx, strlen(vmx), VIR_CONF_FLAG_VMX_FORMAT);
|
|
|
|
|
|
|
|
if (conf == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
return NULL;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->virtType = VIR_DOMAIN_VIRT_VMWARE; /* FIXME: maybe add VIR_DOMAIN_VIRT_ESX ? */
|
|
|
|
def->id = -1;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:config.version */
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigLong(conf, "config.version", &config_version, 0,
|
2010-06-18 16:16:22 +00:00
|
|
|
false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (config_version != 8) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'config.version' to be 8 but found "
|
|
|
|
"%lld"), config_version);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:virtualHW.version */
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigLong(conf, "virtualHW.version", &virtualHW_version, 0,
|
2010-06-18 16:16:22 +00:00
|
|
|
false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-04-18 16:04:24 +00:00
|
|
|
/*
|
|
|
|
* virtualHW.version compatibility matrix:
|
|
|
|
*
|
|
|
|
* 4 7 API
|
|
|
|
* ESX 3.5 + 2.5
|
|
|
|
* ESX 4.0 + + 4.0
|
|
|
|
* GSX 2.0 + + 2.5
|
|
|
|
*/
|
|
|
|
switch (productVersion) {
|
|
|
|
case esxVI_ProductVersion_ESX35:
|
2009-07-27 12:27:09 +00:00
|
|
|
if (virtualHW_version != 4) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-18 16:04:24 +00:00
|
|
|
_("Expecting VMX entry 'virtualHW.version' to be 4 "
|
|
|
|
"but found %lld"),
|
2010-04-02 19:34:31 +00:00
|
|
|
virtualHW_version);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-27 12:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2010-04-18 16:04:24 +00:00
|
|
|
case esxVI_ProductVersion_GSX20:
|
|
|
|
case esxVI_ProductVersion_ESX40:
|
2009-12-23 22:08:21 +00:00
|
|
|
if (virtualHW_version != 4 && virtualHW_version != 7) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'virtualHW.version' to be 4 or 7 "
|
2010-04-18 16:04:24 +00:00
|
|
|
"but found %lld"),
|
2010-04-02 19:34:31 +00:00
|
|
|
virtualHW_version);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-27 12:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-04-18 16:04:24 +00:00
|
|
|
_("Unexpected product version"));
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:uuid.bios -> def:uuid */
|
2009-07-23 20:21:08 +00:00
|
|
|
/* FIXME: Need to handle 'uuid.action = "create"' */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigUUID(conf, "uuid.bios", def->uuid, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:displayName -> def:name */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, "displayName", &def->name, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:memsize -> def:maxmem */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigLong(conf, "memsize", &memsize, 32, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (memsize <= 0 || memsize % 4 != 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'memsize' to be an unsigned "
|
|
|
|
"integer (multiple of 4) but found %lld"), memsize);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->maxmem = memsize * 1024; /* Scale from megabytes to kilobytes */
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:sched.mem.max -> def:memory */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigLong(conf, "sched.mem.max", &memory, memsize,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (memory < 0) {
|
|
|
|
memory = memsize;
|
|
|
|
}
|
|
|
|
|
|
|
|
def->memory = memory * 1024; /* Scale from megabytes to kilobytes */
|
|
|
|
|
|
|
|
if (def->memory > def->maxmem) {
|
|
|
|
def->memory = def->maxmem;
|
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:numvcpus -> def:vcpus */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigLong(conf, "numvcpus", &numvcpus, 1, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (numvcpus <= 0 || (numvcpus % 2 != 0 && numvcpus != 1)) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'numvcpus' to be an unsigned "
|
|
|
|
"integer (1 or a multiple of 2) but found %lld"), numvcpus);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->vcpus = numvcpus;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:sched.cpu.affinity -> def:cpumask */
|
2009-07-23 20:21:08 +00:00
|
|
|
// VirtualMachine:config.cpuAffinity.affinitySet
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, "sched.cpu.affinity", &sched_cpu_affinity,
|
2010-06-18 16:16:22 +00:00
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (sched_cpu_affinity != NULL && STRNEQ(sched_cpu_affinity, "all")) {
|
|
|
|
const char *current = sched_cpu_affinity;
|
|
|
|
int number, count = 0;
|
|
|
|
|
|
|
|
def->cpumasklen = 0;
|
|
|
|
|
|
|
|
if (VIR_ALLOC_N(def->cpumask, VIR_DOMAIN_CPUMASK_LEN) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
while (*current != '\0') {
|
|
|
|
virSkipSpaces(¤t);
|
|
|
|
|
|
|
|
number = virParseNumber(¤t);
|
|
|
|
|
|
|
|
if (number < 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'sched.cpu.affinity' to be "
|
|
|
|
"a comma separated list of unsigned integers but "
|
|
|
|
"found '%s'"), sched_cpu_affinity);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (number >= VIR_DOMAIN_CPUMASK_LEN) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("VMX entry 'sched.cpu.affinity' contains a %d, "
|
|
|
|
"this value is too large"), number);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (number + 1 > def->cpumasklen) {
|
|
|
|
def->cpumasklen = number + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
def->cpumask[number] = 1;
|
|
|
|
++count;
|
|
|
|
|
|
|
|
virSkipSpaces(¤t);
|
|
|
|
|
|
|
|
if (*current == ',') {
|
|
|
|
++current;
|
|
|
|
} else if (*current == '\0') {
|
|
|
|
break;
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'sched.cpu.affinity' to be "
|
|
|
|
"a comma separated list of unsigned integers but "
|
|
|
|
"found '%s'"), sched_cpu_affinity);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virSkipSpaces(¤t);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count < numvcpus) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry 'sched.cpu.affinity' to contain "
|
|
|
|
"at least as many values as 'numvcpus' (%lld) but "
|
|
|
|
"found only %d value(s)"), numvcpus, count);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:lifecycle */
|
|
|
|
def->onReboot = VIR_DOMAIN_LIFECYCLE_RESTART;
|
|
|
|
def->onPoweroff = VIR_DOMAIN_LIFECYCLE_DESTROY;
|
|
|
|
def->onCrash = VIR_DOMAIN_LIFECYCLE_DESTROY;
|
|
|
|
|
|
|
|
/* def:os */
|
|
|
|
def->os.type = strdup("hvm");
|
|
|
|
|
|
|
|
if (def->os.type == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:guestOS -> def:os.arch */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, "guestOS", &guestOS, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:56:05 +00:00
|
|
|
if (guestOS != NULL && virFileHasSuffix(guestOS, "-64")) {
|
2009-09-23 12:16:41 +00:00
|
|
|
def->os.arch = strdup("x86_64");
|
|
|
|
} else {
|
|
|
|
def->os.arch = strdup("i686");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->os.arch == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* def:features */
|
|
|
|
/* FIXME */
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* def:clock */
|
|
|
|
/* FIXME */
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
/* def:graphics */
|
2010-01-16 12:52:34 +00:00
|
|
|
if (VIR_ALLOC_N(def->graphics, 1) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2010-01-16 12:52:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->ngraphics = 0;
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_ParseVNC(conf, &def->graphics[def->ngraphics]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2010-01-16 12:52:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->graphics[def->ngraphics] != NULL) {
|
|
|
|
++def->ngraphics;
|
|
|
|
}
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* def:disks: 4 * 15 scsi + 2 * 2 ide + 2 floppy = 66 */
|
|
|
|
if (VIR_ALLOC_N(def->disks, 66) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->ndisks = 0;
|
|
|
|
|
|
|
|
/* def:disks (scsi) */
|
|
|
|
for (controller = 0; controller < 4; ++controller) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_ParseSCSIController(conf, controller, &present,
|
2010-06-17 17:57:12 +00:00
|
|
|
&scsi_virtualDev[controller]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (! present) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
for (unit = 0; unit < 16; ++unit) {
|
|
|
|
if (unit == 7) {
|
2009-07-23 20:21:08 +00:00
|
|
|
/*
|
2010-06-17 17:57:12 +00:00
|
|
|
* SCSI unit 7 is assigned to the SCSI controller and cannot be
|
2009-07-23 20:21:08 +00:00
|
|
|
* used for disk devices.
|
|
|
|
*/
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_ParseDisk(ctx, caps, conf, VIR_DOMAIN_DISK_DEVICE_DISK,
|
|
|
|
VIR_DOMAIN_DISK_BUS_SCSI, controller, unit,
|
|
|
|
datastoreName, directoryName,
|
2009-07-23 20:21:08 +00:00
|
|
|
&def->disks[def->ndisks]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->disks[def->ndisks] != NULL) {
|
|
|
|
++def->ndisks;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_ParseDisk(ctx, caps, conf, VIR_DOMAIN_DISK_DEVICE_CDROM,
|
|
|
|
VIR_DOMAIN_DISK_BUS_SCSI, controller, unit,
|
|
|
|
datastoreName, directoryName,
|
2009-07-23 20:21:08 +00:00
|
|
|
&def->disks[def->ndisks]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->disks[def->ndisks] != NULL) {
|
|
|
|
++def->ndisks;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:disks (ide) */
|
2010-06-17 17:57:12 +00:00
|
|
|
for (bus = 0; bus < 2; ++bus) {
|
|
|
|
for (unit = 0; unit < 2; ++unit) {
|
|
|
|
if (esxVMX_ParseDisk(ctx, caps, conf, VIR_DOMAIN_DISK_DEVICE_DISK,
|
|
|
|
VIR_DOMAIN_DISK_BUS_IDE, bus, unit,
|
|
|
|
datastoreName, directoryName,
|
2009-09-23 12:52:36 +00:00
|
|
|
&def->disks[def->ndisks]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->disks[def->ndisks] != NULL) {
|
|
|
|
++def->ndisks;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_ParseDisk(ctx, caps, conf, VIR_DOMAIN_DISK_DEVICE_CDROM,
|
|
|
|
VIR_DOMAIN_DISK_BUS_IDE, bus, unit,
|
|
|
|
datastoreName, directoryName,
|
2009-09-23 12:52:36 +00:00
|
|
|
&def->disks[def->ndisks]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->disks[def->ndisks] != NULL) {
|
|
|
|
++def->ndisks;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:disks (floppy) */
|
2010-06-17 17:57:12 +00:00
|
|
|
for (unit = 0; unit < 2; ++unit) {
|
|
|
|
if (esxVMX_ParseDisk(ctx, caps, conf, VIR_DOMAIN_DISK_DEVICE_FLOPPY,
|
|
|
|
VIR_DOMAIN_DISK_BUS_FDC, 0, unit,
|
2009-09-23 12:52:36 +00:00
|
|
|
datastoreName, directoryName,
|
2009-07-23 20:21:08 +00:00
|
|
|
&def->disks[def->ndisks]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->disks[def->ndisks] != NULL) {
|
|
|
|
++def->ndisks;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
/* def:controllers */
|
|
|
|
if (virDomainDefAddImplicitControllers(def) < 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Could not add controllers"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (controller = 0; controller < def->ncontrollers; ++controller) {
|
|
|
|
if (def->controllers[controller]->type == VIR_DOMAIN_CONTROLLER_TYPE_SCSI) {
|
|
|
|
if (def->controllers[controller]->idx < 0 ||
|
|
|
|
def->controllers[controller]->idx > 3) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("SCSI controller index %d out of [0..3] range"),
|
|
|
|
def->controllers[controller]->idx);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
def->controllers[controller]->model =
|
|
|
|
scsi_virtualDev[def->controllers[controller]->idx];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
/* def:fss */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:nets */
|
|
|
|
if (VIR_ALLOC_N(def->nets, 4) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->nnets = 0;
|
|
|
|
|
|
|
|
for (controller = 0; controller < 4; ++controller) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_ParseEthernet(conf, controller,
|
2009-07-23 20:21:08 +00:00
|
|
|
&def->nets[def->nnets]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->nets[def->nnets] != NULL) {
|
|
|
|
++def->nnets;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:inputs */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:sounds */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:hostdevs */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:serials */
|
|
|
|
if (VIR_ALLOC_N(def->serials, 4) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->nserials = 0;
|
|
|
|
|
|
|
|
for (port = 0; port < 4; ++port) {
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_ParseSerial(ctx, conf, port, datastoreName, directoryName,
|
2009-07-23 20:21:08 +00:00
|
|
|
&def->serials[def->nserials]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->serials[def->nserials] != NULL) {
|
|
|
|
++def->nserials;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:parallels */
|
|
|
|
if (VIR_ALLOC_N(def->parallels, 3) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def->nparallels = 0;
|
|
|
|
|
|
|
|
for (port = 0; port < 3; ++port) {
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_ParseParallel(ctx, conf, port, datastoreName, directoryName,
|
2009-07-23 20:21:08 +00:00
|
|
|
&def->parallels[def->nparallels]) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->parallels[def->nparallels] != NULL) {
|
|
|
|
++def->nparallels;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
success = true;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (! success) {
|
|
|
|
virDomainDefFree(def);
|
|
|
|
def = NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
virConfFree(conf);
|
|
|
|
VIR_FREE(sched_cpu_affinity);
|
2009-09-23 12:16:41 +00:00
|
|
|
VIR_FREE(guestOS);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
return def;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-01-16 12:52:34 +00:00
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_ParseVNC(virConfPtr conf, virDomainGraphicsDefPtr *def)
|
2010-01-16 12:52:34 +00:00
|
|
|
{
|
2010-06-18 16:16:22 +00:00
|
|
|
bool enabled = false;
|
2010-01-16 12:52:34 +00:00
|
|
|
long long port = 0;
|
|
|
|
|
|
|
|
if (def == NULL || *def != NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2010-01-16 12:52:34 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, "RemoteDisplay.vnc.enabled", &enabled,
|
2010-06-18 16:16:22 +00:00
|
|
|
false, true) < 0) {
|
2010-01-16 12:52:34 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! enabled) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(*def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-01-16 12:52:34 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
(*def)->type = VIR_DOMAIN_GRAPHICS_TYPE_VNC;
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigLong(conf, "RemoteDisplay.vnc.port", &port, -1,
|
2010-06-18 16:16:22 +00:00
|
|
|
true) < 0 ||
|
2010-01-15 22:05:26 +00:00
|
|
|
esxUtil_GetConfigString(conf, "RemoteDisplay.vnc.ip",
|
2010-06-18 16:16:22 +00:00
|
|
|
&(*def)->data.vnc.listenAddr, true) < 0 ||
|
2010-01-15 22:05:26 +00:00
|
|
|
esxUtil_GetConfigString(conf, "RemoteDisplay.vnc.keymap",
|
2010-06-18 16:16:22 +00:00
|
|
|
&(*def)->data.vnc.keymap, true) < 0 ||
|
2010-01-15 22:05:26 +00:00
|
|
|
esxUtil_GetConfigString(conf, "RemoteDisplay.vnc.password",
|
2010-06-18 16:16:22 +00:00
|
|
|
&(*def)->data.vnc.passwd, true) < 0) {
|
2010-01-16 12:52:34 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (port < 0) {
|
|
|
|
VIR_WARN0("VNC is enabled but VMX entry 'RemoteDisplay.vnc.port' "
|
|
|
|
"is missing, the VNC port is unknown");
|
|
|
|
|
|
|
|
(*def)->data.vnc.port = 0;
|
|
|
|
(*def)->data.vnc.autoport = 1;
|
|
|
|
} else {
|
|
|
|
if (port < 5900 || port > 5964) {
|
|
|
|
VIR_WARN("VNC port %lld it out of [5900..5964] range", port);
|
|
|
|
}
|
|
|
|
|
|
|
|
(*def)->data.vnc.port = port;
|
|
|
|
(*def)->data.vnc.autoport = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
failure:
|
|
|
|
virDomainGraphicsDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
int
|
2010-06-18 16:16:22 +00:00
|
|
|
esxVMX_ParseSCSIController(virConfPtr conf, int controller, bool *present,
|
2010-06-17 17:57:12 +00:00
|
|
|
int *virtualDev)
|
2009-07-23 20:21:08 +00:00
|
|
|
{
|
|
|
|
char present_name[32];
|
|
|
|
char virtualDev_name[32];
|
2010-06-17 17:57:12 +00:00
|
|
|
char *virtualDev_string = NULL;
|
|
|
|
char *tmp;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (virtualDev == NULL || *virtualDev != -1) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-12-15 22:49:56 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (controller < 0 || controller > 3) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("SCSI controller index %d out of [0..3] range"),
|
2009-07-23 20:21:08 +00:00
|
|
|
controller);
|
2009-12-15 22:49:56 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-08-03 11:39:07 +00:00
|
|
|
snprintf(present_name, sizeof(present_name), "scsi%d.present", controller);
|
|
|
|
snprintf(virtualDev_name, sizeof(virtualDev_name), "scsi%d.virtualDev",
|
|
|
|
controller);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, present_name, present, false,
|
|
|
|
true) < 0) {
|
2009-07-23 20:21:08 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! *present) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, virtualDev_name, &virtualDev_string,
|
2010-06-18 16:16:22 +00:00
|
|
|
true) < 0) {
|
2009-07-23 20:21:08 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (virtualDev_string != NULL) {
|
|
|
|
tmp = virtualDev_string;
|
|
|
|
|
|
|
|
for (; *tmp != '\0'; ++tmp) {
|
|
|
|
*tmp = c_tolower(*tmp);
|
|
|
|
}
|
|
|
|
|
2010-06-24 14:58:54 +00:00
|
|
|
*virtualDev = esxVMX_SCSIControllerModelTypeFromString(virtualDev_string);
|
2010-06-17 17:57:12 +00:00
|
|
|
|
|
|
|
if (*virtualDev == -1 ||
|
|
|
|
(*virtualDev != VIR_DOMAIN_CONTROLLER_MODEL_BUSLOGIC &&
|
|
|
|
*virtualDev != VIR_DOMAIN_CONTROLLER_MODEL_LSILOGIC &&
|
2010-06-24 14:58:54 +00:00
|
|
|
*virtualDev != VIR_DOMAIN_CONTROLLER_MODEL_LSISAS1068 &&
|
|
|
|
*virtualDev != VIR_DOMAIN_CONTROLLER_MODEL_VMPVSCSI)) {
|
2010-06-17 17:57:12 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Expecting VMX entry '%s' to be 'buslogic' or 'lsilogic' "
|
2010-06-24 14:58:54 +00:00
|
|
|
"or 'lsisas1068' or 'pvscsi' but found '%s'"),
|
|
|
|
virtualDev_name, virtualDev_string);
|
2010-06-17 17:57:12 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
failure:
|
2010-06-17 17:57:12 +00:00
|
|
|
VIR_FREE(virtualDev_string);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
struct _virDomainDiskDef {
|
|
|
|
int type; // partly done
|
|
|
|
int device; // done
|
|
|
|
int bus; // done
|
|
|
|
char *src; // done
|
|
|
|
char *dst; // done
|
|
|
|
char *driverName; // done
|
|
|
|
char *driverType;
|
|
|
|
int cachemode; // done
|
|
|
|
unsigned int readonly : 1;
|
|
|
|
unsigned int shared : 1;
|
|
|
|
int slotnum;
|
|
|
|
};*/
|
|
|
|
|
|
|
|
int
|
2010-06-17 17:57:12 +00:00
|
|
|
esxVMX_ParseDisk(esxVI_Context *ctx, virCapsPtr caps, virConfPtr conf,
|
|
|
|
int device, int busType, int controllerOrBus, int unit,
|
2010-01-15 22:05:26 +00:00
|
|
|
const char *datastoreName, const char *directoryName,
|
|
|
|
virDomainDiskDefPtr *def)
|
2009-07-23 20:21:08 +00:00
|
|
|
{
|
|
|
|
/*
|
2010-06-17 17:57:12 +00:00
|
|
|
* device = {VIR_DOMAIN_DISK_DEVICE_DISK, VIR_DOMAIN_DISK_DEVICE_CDROM}
|
|
|
|
* busType = VIR_DOMAIN_DISK_BUS_SCSI
|
|
|
|
* controllerOrBus = [0..3] -> controller
|
|
|
|
* unit = [0..6,8..15]
|
2009-07-23 20:21:08 +00:00
|
|
|
*
|
2010-06-17 17:57:12 +00:00
|
|
|
* device = {VIR_DOMAIN_DISK_DEVICE_DISK, VIR_DOMAIN_DISK_DEVICE_CDROM}
|
|
|
|
* busType = VIR_DOMAIN_DISK_BUS_IDE
|
|
|
|
* controllerOrBus = [0..1] -> bus
|
|
|
|
* unit = [0..1]
|
2009-07-23 20:21:08 +00:00
|
|
|
*
|
2010-06-17 17:57:12 +00:00
|
|
|
* device = VIR_DOMAIN_DISK_DEVICE_FLOPPY
|
|
|
|
* busType = VIR_DOMAIN_DISK_BUS_FDC
|
|
|
|
* controllerOrBus = [0]
|
|
|
|
* unit = [0..1]
|
2009-07-23 20:21:08 +00:00
|
|
|
*/
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
int result = -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
char *prefix = NULL;
|
|
|
|
|
|
|
|
char present_name[32] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool present = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char startConnected_name[32] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool startConnected = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char deviceType_name[32] = "";
|
|
|
|
char *deviceType = NULL;
|
|
|
|
|
|
|
|
char clientDevice_name[32] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool clientDevice = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char fileType_name[32] = "";
|
|
|
|
char *fileType = NULL;
|
|
|
|
|
|
|
|
char fileName_name[32] = "";
|
|
|
|
char *fileName = NULL;
|
|
|
|
|
|
|
|
char writeThrough_name[32] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool writeThrough = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
if (def == NULL || *def != NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-07-23 20:21:08 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(*def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
(*def)->device = device;
|
2010-06-17 17:57:12 +00:00
|
|
|
(*def)->bus = busType;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
/* def:dst, def:driverName */
|
|
|
|
if (device == VIR_DOMAIN_DISK_DEVICE_DISK ||
|
|
|
|
device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2010-06-17 17:57:12 +00:00
|
|
|
if (busType == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
if (controllerOrBus < 0 || controllerOrBus > 3) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("SCSI controller index %d out of [0..3] range"),
|
2010-06-17 17:57:12 +00:00
|
|
|
controllerOrBus);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (unit < 0 || unit > 15 || unit == 7) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-06-17 17:57:12 +00:00
|
|
|
_("SCSI unit index %d out of [0..6,8..15] range"),
|
|
|
|
unit);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (virAsprintf(&prefix, "scsi%d:%d", controllerOrBus, unit) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
(*def)->dst =
|
2009-12-03 16:17:40 +00:00
|
|
|
virIndexToDiskName
|
2010-06-17 17:57:12 +00:00
|
|
|
(controllerOrBus * 15 + (unit < 7 ? unit : unit - 1), "sd");
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
if ((*def)->dst == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
2010-06-17 17:57:12 +00:00
|
|
|
} else if (busType == VIR_DOMAIN_DISK_BUS_IDE) {
|
|
|
|
if (controllerOrBus < 0 || controllerOrBus > 1) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-06-17 17:57:12 +00:00
|
|
|
_("IDE bus index %d out of [0..1] range"),
|
|
|
|
controllerOrBus);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (unit < 0 || unit > 1) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-06-17 17:57:12 +00:00
|
|
|
_("IDE unit index %d out of [0..1] range"), unit);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (virAsprintf(&prefix, "ide%d:%d", controllerOrBus, unit) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
(*def)->dst = virIndexToDiskName(controllerOrBus * 2 + unit, "hd");
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
if ((*def)->dst == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported bus type '%s' for device type '%s'"),
|
2010-06-17 17:57:12 +00:00
|
|
|
virDomainDiskBusTypeToString(busType),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskDeviceTypeToString(device));
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else if (device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
2010-06-17 17:57:12 +00:00
|
|
|
if (busType == VIR_DOMAIN_DISK_BUS_FDC) {
|
|
|
|
if (controllerOrBus != 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("FDC controller index %d out of [0] range"),
|
|
|
|
controllerOrBus);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unit < 0 || unit > 1) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-06-17 17:57:12 +00:00
|
|
|
_("FDC unit index %d out of [0..1] range"),
|
|
|
|
unit);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (virAsprintf(&prefix, "floppy%d", unit) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
(*def)->dst = virIndexToDiskName(unit, "fd");
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
if ((*def)->dst == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported bus type '%s' for device type '%s'"),
|
2010-06-17 17:57:12 +00:00
|
|
|
virDomainDiskBusTypeToString(busType),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskDeviceTypeToString(device));
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported device type '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskDeviceTypeToString(device));
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ESX_BUILD_VMX_NAME(present);
|
|
|
|
ESX_BUILD_VMX_NAME(startConnected);
|
|
|
|
ESX_BUILD_VMX_NAME(deviceType);
|
|
|
|
ESX_BUILD_VMX_NAME(clientDevice);
|
|
|
|
ESX_BUILD_VMX_NAME(fileType);
|
|
|
|
ESX_BUILD_VMX_NAME(fileName);
|
|
|
|
ESX_BUILD_VMX_NAME(writeThrough);
|
|
|
|
|
|
|
|
/* vmx:present */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, present_name, &present, false,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:startConnected */
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, startConnected_name, &startConnected,
|
2010-06-18 16:16:22 +00:00
|
|
|
true, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Need to distiguish between active and inactive domains here */
|
|
|
|
if (! present/* && ! startConnected*/) {
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:deviceType -> def:type */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, deviceType_name, &deviceType, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:clientDevice */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, clientDevice_name, &clientDevice, false,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (clientDevice) {
|
|
|
|
/*
|
|
|
|
* Just ignore devices in client mode, because I have no clue how to
|
2009-09-23 12:16:41 +00:00
|
|
|
* handle them (e.g. assign an image) without the VI Client GUI.
|
2009-07-23 20:21:08 +00:00
|
|
|
*/
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:fileType -> def:type */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, fileType_name, &fileType, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:fileName -> def:src, def:type */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, fileName_name, &fileName, false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:writeThrough -> def:cachemode */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, writeThrough_name, &writeThrough, false,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup virDomainDiskDef */
|
|
|
|
if (device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
2009-09-23 12:56:05 +00:00
|
|
|
if (virFileHasSuffix(fileName, ".vmdk")) {
|
2009-07-23 20:21:08 +00:00
|
|
|
if (deviceType != NULL) {
|
2010-06-17 17:57:12 +00:00
|
|
|
if (busType == VIR_DOMAIN_DISK_BUS_SCSI &&
|
2010-06-11 21:18:17 +00:00
|
|
|
STRCASENEQ(deviceType, "scsi-hardDisk") &&
|
|
|
|
STRCASENEQ(deviceType, "disk")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'scsi-hardDisk' "
|
2010-06-11 21:18:17 +00:00
|
|
|
"or 'disk' but found '%s'"), deviceType_name,
|
|
|
|
deviceType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2010-06-17 17:57:12 +00:00
|
|
|
} else if (busType == VIR_DOMAIN_DISK_BUS_IDE &&
|
2010-06-11 21:18:17 +00:00
|
|
|
STRCASENEQ(deviceType, "ata-hardDisk") &&
|
|
|
|
STRCASENEQ(deviceType, "disk")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'ata-hardDisk' "
|
2010-06-11 21:18:17 +00:00
|
|
|
"or 'disk' but found '%s'"), deviceType_name,
|
|
|
|
deviceType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
(*def)->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
2010-01-15 22:05:26 +00:00
|
|
|
(*def)->src = esxVMX_ParseFileName(ctx, fileName, datastoreName,
|
|
|
|
directoryName);
|
2009-07-23 20:21:08 +00:00
|
|
|
(*def)->cachemode = writeThrough ? VIR_DOMAIN_DISK_CACHE_WRITETHRU
|
|
|
|
: VIR_DOMAIN_DISK_CACHE_DEFAULT;
|
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
if ((*def)->src == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-09-23 12:56:05 +00:00
|
|
|
} else if (virFileHasSuffix(fileName, ".iso") ||
|
2009-07-23 20:21:08 +00:00
|
|
|
STREQ(deviceType, "atapi-cdrom")) {
|
|
|
|
/*
|
|
|
|
* This function was called in order to parse a harddisk device,
|
|
|
|
* but .iso files and 'atapi-cdrom' devices are for CDROM devices
|
|
|
|
* only. Just ignore it, another call to this function to parse a
|
|
|
|
* CDROM device may handle it.
|
|
|
|
*/
|
|
|
|
goto ignore;
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Invalid or not yet handled value '%s' for VMX entry "
|
|
|
|
"'%s'"), fileName, fileName_name);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else if (device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2009-09-23 12:56:05 +00:00
|
|
|
if (virFileHasSuffix(fileName, ".iso")) {
|
2009-07-23 20:21:08 +00:00
|
|
|
if (deviceType != NULL) {
|
|
|
|
if (STRCASENEQ(deviceType, "cdrom-image")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'cdrom-image' "
|
|
|
|
"but found '%s'"), deviceType_name, deviceType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
(*def)->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
2010-01-15 22:05:26 +00:00
|
|
|
(*def)->src = esxVMX_ParseFileName(ctx, fileName, datastoreName,
|
|
|
|
directoryName);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
if ((*def)->src == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-09-23 12:56:05 +00:00
|
|
|
} else if (virFileHasSuffix(fileName, ".vmdk")) {
|
2009-07-23 20:21:08 +00:00
|
|
|
/*
|
|
|
|
* This function was called in order to parse a CDROM device, but
|
|
|
|
* .vmdk files are for harddisk devices only. Just ignore it,
|
|
|
|
* another call to this function to parse a harddisk device may
|
|
|
|
* handle it.
|
|
|
|
*/
|
|
|
|
goto ignore;
|
|
|
|
} else if (STREQ(deviceType, "atapi-cdrom")) {
|
|
|
|
(*def)->type = VIR_DOMAIN_DISK_TYPE_BLOCK;
|
|
|
|
(*def)->src = fileName;
|
|
|
|
|
|
|
|
fileName = NULL;
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Invalid or not yet handled value '%s' for VMX entry "
|
|
|
|
"'%s'"), fileName, fileName_name);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else if (device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
2009-09-23 12:56:05 +00:00
|
|
|
if (virFileHasSuffix(fileName, ".flp")) {
|
2009-07-23 20:21:08 +00:00
|
|
|
if (fileType != NULL) {
|
|
|
|
if (STRCASENEQ(fileType, "file")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'file' but "
|
|
|
|
"found '%s'"), fileType_name, fileType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
(*def)->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
2010-01-15 22:05:26 +00:00
|
|
|
(*def)->src = esxVMX_ParseFileName(ctx, fileName, datastoreName,
|
|
|
|
directoryName);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
if ((*def)->src == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-07-23 20:21:08 +00:00
|
|
|
} else if (fileType != NULL && STREQ(fileType, "device")) {
|
|
|
|
(*def)->type = VIR_DOMAIN_DISK_TYPE_BLOCK;
|
|
|
|
(*def)->src = fileName;
|
|
|
|
|
|
|
|
fileName = NULL;
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Invalid or not yet handled value '%s' for VMX entry "
|
|
|
|
"'%s'"), fileName, fileName_name);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
} else {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, _("Unsupported device type '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskDeviceTypeToString(device));
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (virDomainDiskDefAssignAddress(caps, *def) < 0) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not assign address to disk '%s'"), (*def)->src);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (result < 0) {
|
|
|
|
virDomainDiskDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
VIR_FREE(prefix);
|
|
|
|
VIR_FREE(deviceType);
|
|
|
|
VIR_FREE(fileType);
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
ignore:
|
2009-07-23 20:21:08 +00:00
|
|
|
virDomainDiskDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_ParseEthernet(virConfPtr conf, int controller, virDomainNetDefPtr *def)
|
2009-07-23 20:21:08 +00:00
|
|
|
{
|
2010-05-19 20:59:32 +00:00
|
|
|
int result = -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
char prefix[48] = "";
|
|
|
|
|
|
|
|
char present_name[48] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool present = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char startConnected_name[48] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool startConnected = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char connectionType_name[48] = "";
|
|
|
|
char *connectionType = NULL;
|
|
|
|
|
|
|
|
char addressType_name[48] = "";
|
|
|
|
char *addressType = NULL;
|
|
|
|
|
|
|
|
char generatedAddress_name[48] = "";
|
|
|
|
char *generatedAddress = NULL;
|
|
|
|
|
|
|
|
char address_name[48] = "";
|
|
|
|
char *address = NULL;
|
|
|
|
|
|
|
|
char virtualDev_name[48] = "";
|
|
|
|
char *virtualDev = NULL;
|
|
|
|
|
2010-04-17 09:43:01 +00:00
|
|
|
char features_name[48] = "";
|
|
|
|
long long features = 0;
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
char vnet_name[48] = "";
|
|
|
|
char *vnet = NULL;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
char networkName_name[48] = "";
|
|
|
|
char *networkName = NULL;
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
if (def == NULL || *def != NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-07-23 20:21:08 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (controller < 0 || controller > 3) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Ethernet controller index %d out of [0..3] range"),
|
2009-07-23 20:21:08 +00:00
|
|
|
controller);
|
2009-12-15 22:49:56 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(*def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-08-03 11:39:07 +00:00
|
|
|
snprintf(prefix, sizeof(prefix), "ethernet%d", controller);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
ESX_BUILD_VMX_NAME(present);
|
|
|
|
ESX_BUILD_VMX_NAME(startConnected);
|
|
|
|
ESX_BUILD_VMX_NAME(connectionType);
|
|
|
|
ESX_BUILD_VMX_NAME(addressType);
|
|
|
|
ESX_BUILD_VMX_NAME(generatedAddress);
|
|
|
|
ESX_BUILD_VMX_NAME(address);
|
|
|
|
ESX_BUILD_VMX_NAME(virtualDev);
|
2010-04-17 09:43:01 +00:00
|
|
|
ESX_BUILD_VMX_NAME(features);
|
2009-09-23 12:16:41 +00:00
|
|
|
ESX_BUILD_VMX_NAME(networkName);
|
2009-08-05 08:23:59 +00:00
|
|
|
ESX_BUILD_VMX_NAME(vnet);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
/* vmx:present */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, present_name, &present, false,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:startConnected */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, startConnected_name, &startConnected,
|
|
|
|
true, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Need to distiguish between active and inactive domains here */
|
|
|
|
if (! present/* && ! startConnected*/) {
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:connectionType -> def:type */
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, connectionType_name, &connectionType,
|
2010-06-18 16:16:22 +00:00
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:addressType, vmx:generatedAddress, vmx:address -> def:mac */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, addressType_name, &addressType,
|
|
|
|
true) < 0 ||
|
2010-01-15 22:05:26 +00:00
|
|
|
esxUtil_GetConfigString(conf, generatedAddress_name, &generatedAddress,
|
2010-06-18 16:16:22 +00:00
|
|
|
true) < 0 ||
|
|
|
|
esxUtil_GetConfigString(conf, address_name, &address, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-02 14:22:17 +00:00
|
|
|
if (addressType == NULL || STRCASEEQ(addressType, "generated") ||
|
|
|
|
STRCASEEQ(addressType, "vpx")) {
|
2009-07-23 20:21:08 +00:00
|
|
|
if (generatedAddress != NULL) {
|
|
|
|
if (virParseMacAddr(generatedAddress, (*def)->mac) < 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be MAC address but "
|
|
|
|
"found '%s'"), generatedAddress_name,
|
2009-07-23 20:21:08 +00:00
|
|
|
generatedAddress);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (STRCASEEQ(addressType, "static")) {
|
|
|
|
if (address != NULL) {
|
|
|
|
if (virParseMacAddr(address, (*def)->mac) < 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be MAC address but "
|
|
|
|
"found '%s'"), address_name, address);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'generated' or 'static' or "
|
|
|
|
"'vpx' but found '%s'"), addressType_name, addressType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-04-17 09:43:01 +00:00
|
|
|
/* vmx:virtualDev, vmx:features -> def:model */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, virtualDev_name, &virtualDev, true) < 0 ||
|
|
|
|
esxUtil_GetConfigLong(conf, features_name, &features, 0, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-04-17 09:43:01 +00:00
|
|
|
if (virtualDev != NULL) {
|
|
|
|
if (STRCASENEQ(virtualDev, "vlance") &&
|
|
|
|
STRCASENEQ(virtualDev, "vmxnet") &&
|
|
|
|
STRCASENEQ(virtualDev, "vmxnet3") &&
|
|
|
|
STRCASENEQ(virtualDev, "e1000")) {
|
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Expecting VMX entry '%s' to be 'vlance' or 'vmxnet' or "
|
|
|
|
"'vmxnet3' or 'e1000' but found '%s'"), virtualDev_name,
|
|
|
|
virtualDev);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2010-04-17 09:43:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (STRCASEEQ(virtualDev, "vmxnet") && features == 15) {
|
|
|
|
VIR_FREE(virtualDev);
|
|
|
|
|
|
|
|
virtualDev = strdup("vmxnet2");
|
|
|
|
|
|
|
|
if (virtualDev == NULL) {
|
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2010-04-17 09:43:01 +00:00
|
|
|
}
|
|
|
|
}
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* vmx:networkName -> def:data.bridge.brname */
|
|
|
|
if ((connectionType == NULL ||
|
|
|
|
STRCASEEQ(connectionType, "bridged") ||
|
|
|
|
STRCASEEQ(connectionType, "custom")) &&
|
2010-06-18 16:16:22 +00:00
|
|
|
esxUtil_GetConfigString(conf, networkName_name, &networkName,
|
|
|
|
false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:vnet -> def:data.ifname */
|
2009-07-23 20:21:08 +00:00
|
|
|
if (connectionType != NULL && STRCASEEQ(connectionType, "custom") &&
|
2010-06-18 16:16:22 +00:00
|
|
|
esxUtil_GetConfigString(conf, vnet_name, &vnet, false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup virDomainNetDef */
|
|
|
|
if (connectionType == NULL || STRCASEEQ(connectionType, "bridged")) {
|
|
|
|
(*def)->type = VIR_DOMAIN_NET_TYPE_BRIDGE;
|
|
|
|
(*def)->model = virtualDev;
|
2009-09-23 12:16:41 +00:00
|
|
|
(*def)->data.bridge.brname = networkName;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
virtualDev = NULL;
|
2009-09-23 12:16:41 +00:00
|
|
|
networkName = NULL;
|
2009-07-23 20:21:08 +00:00
|
|
|
} else if (STRCASEEQ(connectionType, "hostonly")) {
|
|
|
|
/* FIXME */
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("No yet handled value '%s' for VMX entry '%s'"),
|
2009-07-23 20:21:08 +00:00
|
|
|
connectionType, connectionType_name);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
} else if (STRCASEEQ(connectionType, "nat")) {
|
|
|
|
/* FIXME */
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("No yet handled value '%s' for VMX entry '%s'"),
|
2009-07-23 20:21:08 +00:00
|
|
|
connectionType, connectionType_name);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
} else if (STRCASEEQ(connectionType, "custom")) {
|
|
|
|
(*def)->type = VIR_DOMAIN_NET_TYPE_BRIDGE;
|
|
|
|
(*def)->model = virtualDev;
|
2009-09-23 12:16:41 +00:00
|
|
|
(*def)->data.bridge.brname = networkName;
|
|
|
|
(*def)->ifname = vnet;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
virtualDev = NULL;
|
2009-09-23 12:16:41 +00:00
|
|
|
networkName = NULL;
|
2009-07-23 20:21:08 +00:00
|
|
|
vnet = NULL;
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Invalid value '%s' for VMX entry '%s'"), connectionType,
|
2009-07-23 20:21:08 +00:00
|
|
|
connectionType_name);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (result < 0) {
|
|
|
|
virDomainNetDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
VIR_FREE(connectionType);
|
|
|
|
VIR_FREE(addressType);
|
|
|
|
VIR_FREE(generatedAddress);
|
|
|
|
VIR_FREE(address);
|
|
|
|
VIR_FREE(virtualDev);
|
|
|
|
VIR_FREE(vnet);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
ignore:
|
2009-07-23 20:21:08 +00:00
|
|
|
virDomainNetDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_ParseSerial(esxVI_Context *ctx, virConfPtr conf, int port,
|
|
|
|
const char *datastoreName, const char *directoryName,
|
|
|
|
virDomainChrDefPtr *def)
|
2009-07-23 20:21:08 +00:00
|
|
|
{
|
2010-05-19 20:59:32 +00:00
|
|
|
int result = -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
char prefix[48] = "";
|
|
|
|
|
|
|
|
char present_name[48] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool present = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char startConnected_name[48] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool startConnected = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char fileType_name[48] = "";
|
|
|
|
char *fileType = NULL;
|
|
|
|
|
|
|
|
char fileName_name[48] = "";
|
|
|
|
char *fileName = NULL;
|
|
|
|
|
|
|
|
if (def == NULL || *def != NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-07-23 20:21:08 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (port < 0 || port > 3) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Serial port index %d out of [0..3] range"), port);
|
2009-12-15 22:49:56 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(*def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->targetType = VIR_DOMAIN_CHR_TARGET_TYPE_SERIAL;
|
|
|
|
|
2009-08-03 11:39:07 +00:00
|
|
|
snprintf(prefix, sizeof(prefix), "serial%d", port);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
ESX_BUILD_VMX_NAME(present);
|
|
|
|
ESX_BUILD_VMX_NAME(startConnected);
|
|
|
|
ESX_BUILD_VMX_NAME(fileType);
|
|
|
|
ESX_BUILD_VMX_NAME(fileName);
|
|
|
|
|
|
|
|
/* vmx:present */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, present_name, &present, false,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:startConnected */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, startConnected_name, &startConnected,
|
|
|
|
true, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Need to distiguish between active and inactive domains here */
|
|
|
|
if (! present/* && ! startConnected*/) {
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:fileType -> def:type */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, fileType_name, &fileType, false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:fileName -> def:data.file.path */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, fileName_name, &fileName, false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup virDomainChrDef */
|
|
|
|
if (STRCASEEQ(fileType, "device")) {
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->target.port = port;
|
2009-07-23 20:21:08 +00:00
|
|
|
(*def)->type = VIR_DOMAIN_CHR_TYPE_DEV;
|
|
|
|
(*def)->data.file.path = fileName;
|
|
|
|
|
|
|
|
fileName = NULL;
|
|
|
|
} else if (STRCASEEQ(fileType, "file")) {
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->target.port = port;
|
2009-07-23 20:21:08 +00:00
|
|
|
(*def)->type = VIR_DOMAIN_CHR_TYPE_FILE;
|
2010-01-15 22:05:26 +00:00
|
|
|
(*def)->data.file.path = esxVMX_ParseFileName(ctx, fileName,
|
2009-09-23 12:52:36 +00:00
|
|
|
datastoreName,
|
|
|
|
directoryName);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
if ((*def)->data.file.path == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-07-23 20:21:08 +00:00
|
|
|
} else if (STRCASEEQ(fileType, "pipe")) {
|
2009-09-23 12:16:41 +00:00
|
|
|
/*
|
|
|
|
* FIXME: Differences between client/server and VM/application pipes
|
|
|
|
* not representable in domain XML form
|
|
|
|
*/
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->target.port = port;
|
2009-09-23 12:16:41 +00:00
|
|
|
(*def)->type = VIR_DOMAIN_CHR_TYPE_PIPE;
|
|
|
|
(*def)->data.file.path = fileName;
|
|
|
|
|
|
|
|
fileName = NULL;
|
2009-07-23 20:21:08 +00:00
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'device', 'file' or 'pipe' "
|
|
|
|
"but found '%s'"), fileType_name, fileType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (result < 0) {
|
|
|
|
virDomainChrDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
VIR_FREE(fileType);
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
ignore:
|
2009-07-23 20:21:08 +00:00
|
|
|
virDomainChrDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_ParseParallel(esxVI_Context *ctx, virConfPtr conf, int port,
|
|
|
|
const char *datastoreName, const char *directoryName,
|
|
|
|
virDomainChrDefPtr *def)
|
2009-07-23 20:21:08 +00:00
|
|
|
{
|
2010-05-19 20:59:32 +00:00
|
|
|
int result = -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
char prefix[48] = "";
|
|
|
|
|
|
|
|
char present_name[48] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool present = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char startConnected_name[48] = "";
|
2010-06-18 16:16:22 +00:00
|
|
|
bool startConnected = false;
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
char fileType_name[48] = "";
|
|
|
|
char *fileType = NULL;
|
|
|
|
|
|
|
|
char fileName_name[48] = "";
|
|
|
|
char *fileName = NULL;
|
|
|
|
|
|
|
|
if (def == NULL || *def != NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-07-23 20:21:08 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (port < 0 || port > 2) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Parallel port index %d out of [0..2] range"), port);
|
2009-12-15 22:49:56 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(*def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
return -1;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->targetType = VIR_DOMAIN_CHR_TARGET_TYPE_PARALLEL;
|
|
|
|
|
2009-08-03 11:39:07 +00:00
|
|
|
snprintf(prefix, sizeof(prefix), "parallel%d", port);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
|
|
|
ESX_BUILD_VMX_NAME(present);
|
|
|
|
ESX_BUILD_VMX_NAME(startConnected);
|
|
|
|
ESX_BUILD_VMX_NAME(fileType);
|
|
|
|
ESX_BUILD_VMX_NAME(fileName);
|
|
|
|
|
|
|
|
/* vmx:present */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, present_name, &present, false,
|
|
|
|
true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:startConnected */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigBoolean(conf, startConnected_name, &startConnected,
|
|
|
|
true, true) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Need to distiguish between active and inactive domains here */
|
|
|
|
if (! present/* && ! startConnected*/) {
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:fileType -> def:type */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, fileType_name, &fileType, false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:fileName -> def:data.file.path */
|
2010-06-18 16:16:22 +00:00
|
|
|
if (esxUtil_GetConfigString(conf, fileName_name, &fileName, false) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup virDomainChrDef */
|
|
|
|
if (STRCASEEQ(fileType, "device")) {
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->target.port = port;
|
2009-07-23 20:21:08 +00:00
|
|
|
(*def)->type = VIR_DOMAIN_CHR_TYPE_DEV;
|
|
|
|
(*def)->data.file.path = fileName;
|
|
|
|
|
|
|
|
fileName = NULL;
|
|
|
|
} else if (STRCASEEQ(fileType, "file")) {
|
2009-11-05 13:19:14 +00:00
|
|
|
(*def)->target.port = port;
|
2009-07-23 20:21:08 +00:00
|
|
|
(*def)->type = VIR_DOMAIN_CHR_TYPE_FILE;
|
2010-01-15 22:05:26 +00:00
|
|
|
(*def)->data.file.path = esxVMX_ParseFileName(ctx, fileName,
|
2009-09-23 12:52:36 +00:00
|
|
|
datastoreName,
|
|
|
|
directoryName);
|
2009-07-23 20:21:08 +00:00
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
if ((*def)->data.file.path == NULL) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-07-23 20:21:08 +00:00
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting VMX entry '%s' to be 'device' or 'file' but "
|
|
|
|
"found '%s'"), fileType_name, fileType);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-07-23 20:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (result < 0) {
|
|
|
|
virDomainChrDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
VIR_FREE(fileType);
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
ignore:
|
2009-07-23 20:21:08 +00:00
|
|
|
virDomainChrDefFree(*def);
|
|
|
|
*def = NULL;
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
result = 0;
|
|
|
|
|
2009-07-23 20:21:08 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
* Domain XML -> VMX
|
|
|
|
*/
|
|
|
|
|
|
|
|
char *
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatFileName(esxVI_Context *ctx ATTRIBUTE_UNUSED, const char *src)
|
2009-09-23 12:52:36 +00:00
|
|
|
{
|
2010-05-19 20:59:32 +00:00
|
|
|
bool success = false;
|
2009-09-23 12:52:36 +00:00
|
|
|
char *datastoreName = NULL;
|
|
|
|
char *directoryName = NULL;
|
|
|
|
char *fileName = NULL;
|
|
|
|
char *absolutePath = NULL;
|
|
|
|
|
|
|
|
if (STRPREFIX(src, "[")) {
|
|
|
|
/* Found potential datastore related path */
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxUtil_ParseDatastoreRelatedPath(src, &datastoreName,
|
2009-09-23 12:52:36 +00:00
|
|
|
&directoryName, &fileName) < 0) {
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (directoryName == NULL) {
|
|
|
|
if (virAsprintf(&absolutePath, "/vmfs/volumes/%s/%s",
|
|
|
|
datastoreName, fileName) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (virAsprintf(&absolutePath, "/vmfs/volumes/%s/%s/%s",
|
|
|
|
datastoreName, directoryName, fileName) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (STRPREFIX(src, "/")) {
|
|
|
|
/* Found absolute path */
|
|
|
|
absolutePath = strdup(src);
|
|
|
|
|
|
|
|
if (absolutePath == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Found relative path, this is not supported */
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Found relative path '%s' in domain XML, this is not "
|
|
|
|
"supported"), src);
|
2010-05-19 20:59:32 +00:00
|
|
|
goto cleanup;
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Check if referenced path/file really exists */
|
|
|
|
|
2010-05-19 20:59:32 +00:00
|
|
|
success = true;
|
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
cleanup:
|
2010-05-19 20:59:32 +00:00
|
|
|
if (! success) {
|
|
|
|
VIR_FREE(absolutePath);
|
|
|
|
}
|
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
VIR_FREE(datastoreName);
|
|
|
|
VIR_FREE(directoryName);
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
|
|
|
|
return absolutePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
2010-06-17 17:57:12 +00:00
|
|
|
esxVMX_FormatConfig(esxVI_Context *ctx, virCapsPtr caps, virDomainDefPtr def,
|
2010-04-18 16:04:24 +00:00
|
|
|
esxVI_ProductVersion productVersion)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int sched_cpu_affinity_length;
|
|
|
|
unsigned char zero[VIR_UUID_BUFLEN];
|
|
|
|
virBuffer buffer = VIR_BUFFER_INITIALIZER;
|
2010-06-17 17:57:12 +00:00
|
|
|
bool scsi_present[4] = { false, false, false, false };
|
|
|
|
int scsi_virtualDev[4] = { -1, -1, -1, -1 };
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
memset(zero, 0, VIR_UUID_BUFLEN);
|
|
|
|
|
|
|
|
if (def->virtType != VIR_DOMAIN_VIRT_VMWARE) { /* FIXME: maybe add VIR_DOMAIN_VIRT_ESX ? */
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting virt type to be '%s' but found '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainVirtTypeToString(VIR_DOMAIN_VIRT_VMWARE),
|
|
|
|
virDomainVirtTypeToString(def->virtType));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:config.version */
|
|
|
|
virBufferAddLit(&buffer, "config.version = \"8\"\n");
|
|
|
|
|
|
|
|
/* vmx:virtualHW.version */
|
2010-04-18 16:04:24 +00:00
|
|
|
switch (productVersion) {
|
|
|
|
case esxVI_ProductVersion_ESX35:
|
2009-09-23 12:16:41 +00:00
|
|
|
virBufferAddLit(&buffer, "virtualHW.version = \"4\"\n");
|
|
|
|
break;
|
|
|
|
|
2010-04-18 16:04:24 +00:00
|
|
|
case esxVI_ProductVersion_GSX20:
|
|
|
|
case esxVI_ProductVersion_ESX40:
|
2009-09-23 12:16:41 +00:00
|
|
|
virBufferAddLit(&buffer, "virtualHW.version = \"7\"\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-04-18 16:04:24 +00:00
|
|
|
_("Unexpected product version"));
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:arch -> vmx:guestOS */
|
|
|
|
if (def->os.arch == NULL || STRCASEEQ(def->os.arch, "i686")) {
|
|
|
|
virBufferAddLit(&buffer, "guestOS = \"other\"\n");
|
|
|
|
} else if (STRCASEEQ(def->os.arch, "x86_64")) {
|
|
|
|
virBufferAddLit(&buffer, "guestOS = \"other-64\"\n");
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting domain XML attribute 'arch' of entry 'os/type' "
|
|
|
|
"to be 'i686' or 'x86_64' but found '%s'"), def->os.arch);
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:uuid -> vmx:uuid.action, vmx:uuid.bios */
|
|
|
|
if (memcmp(def->uuid, zero, VIR_UUID_BUFLEN) == 0) {
|
|
|
|
virBufferAddLit(&buffer, "uuid.action = \"create\"\n");
|
|
|
|
} else {
|
|
|
|
virBufferVSprintf(&buffer, "uuid.bios = \"%02x %02x %02x %02x %02x %02x "
|
|
|
|
"%02x %02x-%02x %02x %02x %02x %02x %02x %02x %02x\"\n",
|
|
|
|
def->uuid[0], def->uuid[1], def->uuid[2], def->uuid[3],
|
|
|
|
def->uuid[4], def->uuid[5], def->uuid[6], def->uuid[7],
|
|
|
|
def->uuid[8], def->uuid[9], def->uuid[10], def->uuid[11],
|
|
|
|
def->uuid[12], def->uuid[13], def->uuid[14],
|
|
|
|
def->uuid[15]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:name -> vmx:displayName */
|
|
|
|
virBufferVSprintf(&buffer, "displayName = \"%s\"\n", def->name);
|
|
|
|
|
|
|
|
/* def:maxmem -> vmx:memsize */
|
|
|
|
if (def->maxmem <= 0 || def->maxmem % 4096 != 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting domain XML entry 'memory' to be an unsigned "
|
|
|
|
"integer (multiple of 4096) but found %lld"),
|
2009-09-23 12:16:41 +00:00
|
|
|
(unsigned long long)def->maxmem);
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Scale from kilobytes to megabytes */
|
|
|
|
virBufferVSprintf(&buffer, "memsize = \"%d\"\n",
|
|
|
|
(int)(def->maxmem / 1024));
|
|
|
|
|
|
|
|
/* def:memory -> vmx:sched.mem.max */
|
|
|
|
if (def->memory < def->maxmem) {
|
|
|
|
if (def->memory <= 0 || def->memory % 1024 != 0) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting domain XML entry 'currentMemory' to be an "
|
|
|
|
"unsigned integer (multiple of 1024) but found %lld"),
|
2009-09-23 12:16:41 +00:00
|
|
|
(unsigned long long)def->memory);
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Scale from kilobytes to megabytes */
|
|
|
|
virBufferVSprintf(&buffer, "sched.mem.max = \"%d\"\n",
|
|
|
|
(int)(def->memory / 1024));
|
|
|
|
}
|
|
|
|
|
2010-01-16 12:52:34 +00:00
|
|
|
/* def:vcpus -> vmx:numvcpus */
|
2009-09-23 12:16:41 +00:00
|
|
|
if (def->vcpus <= 0 || (def->vcpus % 2 != 0 && def->vcpus != 1)) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting domain XML entry 'vcpu' to be an unsigned "
|
|
|
|
"integer (1 or a multiple of 2) but found %d"),
|
2009-09-23 12:16:41 +00:00
|
|
|
(int)def->vcpus);
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(&buffer, "numvcpus = \"%d\"\n", (int)def->vcpus);
|
|
|
|
|
|
|
|
/* def:cpumask -> vmx:sched.cpu.affinity */
|
|
|
|
if (def->cpumasklen > 0) {
|
|
|
|
virBufferAddLit(&buffer, "sched.cpu.affinity = \"");
|
|
|
|
|
|
|
|
sched_cpu_affinity_length = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < def->cpumasklen; ++i) {
|
|
|
|
if (def->cpumask[i]) {
|
|
|
|
++sched_cpu_affinity_length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sched_cpu_affinity_length < def->vcpus) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting domain XML attribute 'cpuset' of entry "
|
|
|
|
"'vcpu' to contains at least %d CPU(s)"),
|
2009-09-23 12:16:41 +00:00
|
|
|
(int)def->vcpus);
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < def->cpumasklen; ++i) {
|
|
|
|
if (def->cpumask[i]) {
|
|
|
|
virBufferVSprintf(&buffer, "%d", i);
|
|
|
|
|
|
|
|
if (sched_cpu_affinity_length > 1) {
|
|
|
|
virBufferAddChar(&buffer, ',');
|
|
|
|
}
|
|
|
|
|
|
|
|
--sched_cpu_affinity_length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAddLit(&buffer, "\"\n");
|
|
|
|
}
|
|
|
|
|
2010-01-16 12:52:34 +00:00
|
|
|
/* def:graphics */
|
|
|
|
for (i = 0; i < def->ngraphics; ++i) {
|
|
|
|
switch (def->graphics[i]->type) {
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_VNC:
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatVNC(def->graphics[i], &buffer) < 0) {
|
2010-01-16 12:52:34 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported graphics type '%s'"),
|
2010-01-16 12:52:34 +00:00
|
|
|
virDomainGraphicsTypeToString(def->graphics[i]->type));
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
/* def:disks */
|
2010-06-17 17:57:12 +00:00
|
|
|
for (i = 0; i < def->ndisks; ++i) {
|
|
|
|
if (esxVMX_VerifyDiskAddress(caps, def->disks[i]) < 0 ||
|
|
|
|
esxVMX_HandleLegacySCSIDiskDriverName(def, def->disks[i]) < 0) {
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
|
2010-07-06 17:02:48 +00:00
|
|
|
if (esxVMX_GatherSCSIControllers(ctx, def, scsi_virtualDev,
|
|
|
|
scsi_present) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 4; ++i) {
|
|
|
|
if (scsi_present[i]) {
|
|
|
|
virBufferVSprintf(&buffer, "scsi%d.present = \"true\"\n", i);
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (scsi_virtualDev[i] != -1) {
|
2009-09-23 12:16:41 +00:00
|
|
|
virBufferVSprintf(&buffer, "scsi%d.virtualDev = \"%s\"\n", i,
|
2010-06-24 14:58:54 +00:00
|
|
|
esxVMX_SCSIControllerModelTypeToString
|
2010-06-17 17:57:12 +00:00
|
|
|
(scsi_virtualDev[i]));
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; ++i) {
|
|
|
|
switch (def->disks[i]->device) {
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_DISK:
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatHardDisk(ctx, def->disks[i], &buffer) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_CDROM:
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatCDROM(ctx, def->disks[i], &buffer) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_FLOPPY:
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatFloppy(ctx, def->disks[i], &buffer) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported disk device type '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskDeviceTypeToString(def->disks[i]->device));
|
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:fss */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:nets */
|
|
|
|
for (i = 0; i < def->nnets; ++i) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatEthernet(def->nets[i], i, &buffer) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:inputs */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:sounds */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:hostdevs */
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
/* def:serials */
|
|
|
|
for (i = 0; i < def->nserials; ++i) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatSerial(ctx, def->serials[i], &buffer) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* def:parallels */
|
|
|
|
for (i = 0; i < def->nparallels; ++i) {
|
2010-01-15 22:05:26 +00:00
|
|
|
if (esxVMX_FormatParallel(ctx, def->parallels[i], &buffer) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get final VMX output */
|
|
|
|
if (virBufferError(&buffer)) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-09-23 12:16:41 +00:00
|
|
|
goto failure;
|
|
|
|
}
|
|
|
|
|
2009-12-09 23:00:50 +00:00
|
|
|
return virBufferContentAndReset(&buffer);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
failure:
|
2009-12-09 23:00:50 +00:00
|
|
|
virBufferFreeAndReset(&buffer);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-01-16 12:52:34 +00:00
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatVNC(virDomainGraphicsDefPtr def, virBufferPtr buffer)
|
2010-01-16 12:52:34 +00:00
|
|
|
{
|
|
|
|
if (def->type != VIR_DOMAIN_GRAPHICS_TYPE_VNC) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2010-01-16 12:52:34 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "RemoteDisplay.vnc.enabled = \"true\"\n");
|
|
|
|
|
|
|
|
if (def->data.vnc.autoport) {
|
|
|
|
VIR_WARN0("VNC autoport is enabled, but the automatically assigned "
|
|
|
|
"VNC port cannot be read back");
|
|
|
|
} else {
|
|
|
|
if (def->data.vnc.port < 5900 || def->data.vnc.port > 5964) {
|
|
|
|
VIR_WARN("VNC port %d it out of [5900..5964] range",
|
|
|
|
def->data.vnc.port);
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "RemoteDisplay.vnc.port = \"%d\"\n",
|
|
|
|
def->data.vnc.port);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->data.vnc.listenAddr != NULL) {
|
|
|
|
virBufferVSprintf(buffer, "RemoteDisplay.vnc.ip = \"%s\"\n",
|
|
|
|
def->data.vnc.listenAddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->data.vnc.keymap != NULL) {
|
|
|
|
virBufferVSprintf(buffer, "RemoteDisplay.vnc.keymap = \"%s\"\n",
|
|
|
|
def->data.vnc.keymap);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->data.vnc.passwd != NULL) {
|
|
|
|
virBufferVSprintf(buffer, "RemoteDisplay.vnc.password = \"%s\"\n",
|
|
|
|
def->data.vnc.passwd);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatHardDisk(esxVI_Context *ctx, virDomainDiskDefPtr def,
|
|
|
|
virBufferPtr buffer)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
2010-06-17 17:57:12 +00:00
|
|
|
int controllerOrBus, unit;
|
2009-09-23 12:16:41 +00:00
|
|
|
const char *busName = NULL;
|
|
|
|
const char *entryPrefix = NULL;
|
|
|
|
const char *deviceTypePrefix = NULL;
|
2009-09-23 12:52:36 +00:00
|
|
|
char *fileName = NULL;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (def->device != VIR_DOMAIN_DISK_DEVICE_DISK) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->bus == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
busName = "SCSI";
|
|
|
|
entryPrefix = "scsi";
|
|
|
|
deviceTypePrefix = "scsi";
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_SCSIDiskNameToControllerAndUnit(def->dst, &controllerOrBus,
|
|
|
|
&unit) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (def->bus == VIR_DOMAIN_DISK_BUS_IDE) {
|
|
|
|
busName = "IDE";
|
|
|
|
entryPrefix = "ide";
|
|
|
|
deviceTypePrefix = "ata";
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_IDEDiskNameToBusAndUnit(def->dst, &controllerOrBus,
|
|
|
|
&unit) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported bus type '%s' for harddisk"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskBusTypeToString(def->bus));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->type != VIR_DOMAIN_DISK_TYPE_FILE) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("%s harddisk '%s' has unsupported type '%s', expecting '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
busName, def->dst, virDomainDiskTypeToString(def->type),
|
|
|
|
virDomainDiskTypeToString(VIR_DOMAIN_DISK_TYPE_FILE));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.present = \"true\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit);
|
2009-09-23 12:16:41 +00:00
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.deviceType = \"%s-hardDisk\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit, deviceTypePrefix);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (def->src != NULL) {
|
2009-09-23 12:56:05 +00:00
|
|
|
if (! virFileHasSuffix(def->src, ".vmdk")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Image file for %s harddisk '%s' has unsupported suffix, "
|
|
|
|
"expecting '.vmdk'"), busName, def->dst);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
fileName = esxVMX_FormatFileName(ctx, def->src);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (fileName == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-09-23 12:16:41 +00:00
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.fileName = \"%s\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit, fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
VIR_FREE(fileName);
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->bus == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
if (def->cachemode == VIR_DOMAIN_DISK_CACHE_WRITETHRU) {
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.writeThrough = \"true\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit);
|
2009-09-23 12:16:41 +00:00
|
|
|
} else if (def->cachemode != VIR_DOMAIN_DISK_CACHE_DEFAULT) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("%s harddisk '%s' has unsupported cache mode '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
busName, def->dst,
|
|
|
|
virDomainDiskCacheTypeToString(def->cachemode));
|
2009-09-23 12:52:36 +00:00
|
|
|
return -1;
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatCDROM(esxVI_Context *ctx, virDomainDiskDefPtr def,
|
|
|
|
virBufferPtr buffer)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
2010-06-17 17:57:12 +00:00
|
|
|
int controllerOrBus, unit;
|
2009-09-23 12:16:41 +00:00
|
|
|
const char *busName = NULL;
|
|
|
|
const char *entryPrefix = NULL;
|
2009-09-23 12:52:36 +00:00
|
|
|
char *fileName = NULL;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (def->device != VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->bus == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
busName = "SCSI";
|
|
|
|
entryPrefix = "scsi";
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_SCSIDiskNameToControllerAndUnit(def->dst, &controllerOrBus,
|
|
|
|
&unit) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (def->bus == VIR_DOMAIN_DISK_BUS_IDE) {
|
|
|
|
busName = "IDE";
|
|
|
|
entryPrefix = "ide";
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_IDEDiskNameToBusAndUnit(def->dst, &controllerOrBus,
|
|
|
|
&unit) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unsupported bus type '%s' for cdrom"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskBusTypeToString(def->bus));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.present = \"true\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (def->type == VIR_DOMAIN_DISK_TYPE_FILE) {
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.deviceType = \"cdrom-image\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (def->src != NULL) {
|
2009-09-23 12:56:05 +00:00
|
|
|
if (! virFileHasSuffix(def->src, ".iso")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Image file for %s cdrom '%s' has unsupported "
|
|
|
|
"suffix, expecting '.iso'"), busName, def->dst);
|
2009-09-23 12:52:36 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
fileName = esxVMX_FormatFileName(ctx, def->src);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (fileName == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.fileName = \"%s\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit, fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
} else if (def->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.deviceType = \"atapi-cdrom\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (def->src != NULL) {
|
|
|
|
virBufferVSprintf(buffer, "%s%d:%d.fileName = \"%s\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
entryPrefix, controllerOrBus, unit, def->src);
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("%s cdrom '%s' has unsupported type '%s', expecting '%s' "
|
|
|
|
"or '%s'"), busName, def->dst,
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskTypeToString(def->type),
|
|
|
|
virDomainDiskTypeToString(VIR_DOMAIN_DISK_TYPE_FILE),
|
|
|
|
virDomainDiskTypeToString(VIR_DOMAIN_DISK_TYPE_BLOCK));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatFloppy(esxVI_Context *ctx, virDomainDiskDefPtr def,
|
|
|
|
virBufferPtr buffer)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
2010-06-17 17:57:12 +00:00
|
|
|
int unit;
|
2009-09-23 12:52:36 +00:00
|
|
|
char *fileName = NULL;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (def->device != VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
if (esxVMX_FloppyDiskNameToUnit(def->dst, &unit) < 0) {
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-06-17 17:57:12 +00:00
|
|
|
virBufferVSprintf(buffer, "floppy%d.present = \"true\"\n", unit);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (def->type == VIR_DOMAIN_DISK_TYPE_FILE) {
|
2010-06-17 17:57:12 +00:00
|
|
|
virBufferVSprintf(buffer, "floppy%d.fileType = \"file\"\n", unit);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (def->src != NULL) {
|
2009-09-23 12:56:05 +00:00
|
|
|
if (! virFileHasSuffix(def->src, ".flp")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Image file for floppy '%s' has unsupported "
|
|
|
|
"suffix, expecting '.flp'"), def->dst);
|
2009-09-23 12:52:36 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
fileName = esxVMX_FormatFileName(ctx, def->src);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (fileName == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "floppy%d.fileName = \"%s\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
unit, fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
VIR_FREE(fileName);
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
} else if (def->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
2010-06-17 17:57:12 +00:00
|
|
|
virBufferVSprintf(buffer, "floppy%d.fileType = \"device\"\n", unit);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (def->src != NULL) {
|
|
|
|
virBufferVSprintf(buffer, "floppy%d.fileName = \"%s\"\n",
|
2010-06-17 17:57:12 +00:00
|
|
|
unit, def->src);
|
2009-09-23 12:52:36 +00:00
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
} else {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Floppy '%s' has unsupported type '%s', expecting '%s' "
|
|
|
|
"or '%s'"), def->dst,
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainDiskTypeToString(def->type),
|
|
|
|
virDomainDiskTypeToString(VIR_DOMAIN_DISK_TYPE_FILE),
|
|
|
|
virDomainDiskTypeToString(VIR_DOMAIN_DISK_TYPE_BLOCK));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatEthernet(virDomainNetDefPtr def, int controller,
|
|
|
|
virBufferPtr buffer)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
|
|
|
char mac_string[VIR_MAC_STRING_BUFLEN];
|
2010-01-02 22:58:24 +00:00
|
|
|
unsigned int prefix, suffix;
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
if (controller < 0 || controller > 3) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Ethernet controller index %d out of [0..3] range"),
|
2009-09-23 12:16:41 +00:00
|
|
|
controller);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.present = \"true\"\n", controller);
|
|
|
|
|
2010-04-17 09:43:01 +00:00
|
|
|
/* def:model -> vmx:virtualDev, vmx:features */
|
2009-09-23 12:16:41 +00:00
|
|
|
if (def->model != NULL) {
|
|
|
|
if (STRCASENEQ(def->model, "vlance") &&
|
|
|
|
STRCASENEQ(def->model, "vmxnet") &&
|
2010-04-17 09:43:01 +00:00
|
|
|
STRCASENEQ(def->model, "vmxnet2") &&
|
2009-11-13 21:55:25 +00:00
|
|
|
STRCASENEQ(def->model, "vmxnet3") &&
|
2009-09-23 12:16:41 +00:00
|
|
|
STRCASENEQ(def->model, "e1000")) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Expecting domain XML entry 'devices/interfase/model' "
|
2010-04-17 09:43:01 +00:00
|
|
|
"to be 'vlance' or 'vmxnet' or 'vmxnet2' or 'vmxnet3' "
|
|
|
|
"or 'e1000' but found '%s'"), def->model);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-04-17 09:43:01 +00:00
|
|
|
if (STRCASEEQ(def->model, "vmxnet2")) {
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.virtualDev = \"vmxnet\"\n",
|
|
|
|
controller);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.features = \"15\"\n",
|
|
|
|
controller);
|
|
|
|
} else {
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.virtualDev = \"%s\"\n",
|
|
|
|
controller, def->model);
|
|
|
|
}
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* def:type, def:ifname -> vmx:connectionType */
|
|
|
|
switch (def->type) {
|
|
|
|
case VIR_DOMAIN_NET_TYPE_BRIDGE:
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.networkName = \"%s\"\n",
|
|
|
|
controller, def->data.bridge.brname);
|
|
|
|
|
|
|
|
if (def->ifname != NULL) {
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.connectionType = \"custom\"\n",
|
|
|
|
controller);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.vnet = \"%s\"\n",
|
|
|
|
controller, def->ifname);
|
|
|
|
} else {
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.connectionType = \"bridged\"\n",
|
|
|
|
controller);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, _("Unsupported net type '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainNetTypeToString(def->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-01-02 22:58:24 +00:00
|
|
|
/* def:mac -> vmx:addressType, vmx:(generated)Address, vmx:checkMACAddress */
|
2009-09-23 12:16:41 +00:00
|
|
|
virFormatMacAddr(def->mac, mac_string);
|
|
|
|
|
2010-01-02 22:58:24 +00:00
|
|
|
prefix = (def->mac[0] << 16) | (def->mac[1] << 8) | def->mac[2];
|
|
|
|
suffix = (def->mac[3] << 16) | (def->mac[4] << 8) | def->mac[5];
|
|
|
|
|
|
|
|
if (prefix == 0x000c29) {
|
2009-09-23 12:16:41 +00:00
|
|
|
virBufferVSprintf(buffer, "ethernet%d.addressType = \"generated\"\n",
|
|
|
|
controller);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.generatedAddress = \"%s\"\n",
|
|
|
|
controller, mac_string);
|
2009-11-14 18:51:05 +00:00
|
|
|
virBufferVSprintf(buffer, "ethernet%d.generatedAddressOffset = \"0\"\n",
|
|
|
|
controller);
|
2010-01-02 22:58:24 +00:00
|
|
|
} else if (prefix == 0x005056 && suffix <= 0x3fffff) {
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.addressType = \"static\"\n",
|
|
|
|
controller);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.address = \"%s\"\n",
|
|
|
|
controller, mac_string);
|
|
|
|
} else if (prefix == 0x005056 && suffix >= 0x800000 && suffix <= 0xbfffff) {
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.addressType = \"vpx\"\n",
|
|
|
|
controller);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.generatedAddress = \"%s\"\n",
|
|
|
|
controller, mac_string);
|
2009-09-23 12:16:41 +00:00
|
|
|
} else {
|
2010-01-02 22:58:24 +00:00
|
|
|
virBufferVSprintf(buffer, "ethernet%d.addressType = \"static\"\n",
|
|
|
|
controller);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.address = \"%s\"\n",
|
|
|
|
controller, mac_string);
|
|
|
|
virBufferVSprintf(buffer, "ethernet%d.checkMACAddress = \"false\"\n",
|
|
|
|
controller);
|
2009-09-23 12:16:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatSerial(esxVI_Context *ctx, virDomainChrDefPtr def,
|
|
|
|
virBufferPtr buffer)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
2009-09-23 12:52:36 +00:00
|
|
|
char *fileName = NULL;
|
|
|
|
|
2009-11-05 13:19:14 +00:00
|
|
|
if (def->target.port < 0 || def->target.port > 3) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Serial port index %d out of [0..3] range"),
|
|
|
|
def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->data.file.path == NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Expecting domain XML attribute 'path' of entry "
|
|
|
|
"'devices/serial/source' to be present"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-11-05 13:19:14 +00:00
|
|
|
virBufferVSprintf(buffer, "serial%d.present = \"true\"\n", def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
/* def:type -> vmx:fileType and def:data.file.path -> vmx:fileName */
|
2009-09-23 12:16:41 +00:00
|
|
|
switch (def->type) {
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_DEV:
|
|
|
|
virBufferVSprintf(buffer, "serial%d.fileType = \"device\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:52:36 +00:00
|
|
|
virBufferVSprintf(buffer, "serial%d.fileName = \"%s\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port, def->data.file.path);
|
2009-09-23 12:16:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_FILE:
|
|
|
|
virBufferVSprintf(buffer, "serial%d.fileType = \"file\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
fileName = esxVMX_FormatFileName(ctx, def->data.file.path);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (fileName == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "serial%d.fileName = \"%s\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port, fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
VIR_FREE(fileName);
|
2009-09-23 12:16:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_PIPE:
|
|
|
|
virBufferVSprintf(buffer, "serial%d.fileType = \"pipe\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
/* FIXME: Based on VI Client GUI default */
|
|
|
|
virBufferVSprintf(buffer, "serial%d.pipe.endPoint = \"client\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
/* FIXME: Based on VI Client GUI default */
|
|
|
|
virBufferVSprintf(buffer, "serial%d.tryNoRxLoss = \"false\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:52:36 +00:00
|
|
|
virBufferVSprintf(buffer, "serial%d.fileName = \"%s\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port, def->data.file.path);
|
2009-09-23 12:16:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported character device type '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainChrTypeToString(def->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vmx:yieldOnMsrRead */
|
|
|
|
/* FIXME: Based on VI Client GUI default */
|
|
|
|
virBufferVSprintf(buffer, "serial%d.yieldOnMsrRead = \"true\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2010-01-15 22:05:26 +00:00
|
|
|
esxVMX_FormatParallel(esxVI_Context *ctx, virDomainChrDefPtr def,
|
|
|
|
virBufferPtr buffer)
|
2009-09-23 12:16:41 +00:00
|
|
|
{
|
2009-09-23 12:52:36 +00:00
|
|
|
char *fileName = NULL;
|
|
|
|
|
2009-11-05 13:19:14 +00:00
|
|
|
if (def->target.port < 0 || def->target.port > 2) {
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Parallel port index %d out of [0..2] range"),
|
2010-01-15 22:05:26 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->data.file.path == NULL) {
|
2010-04-02 19:34:31 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Expecting domain XML attribute 'path' of entry "
|
|
|
|
"'devices/parallel/source' to be present"));
|
2009-09-23 12:16:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
virBufferVSprintf(buffer, "parallel%d.present = \"true\"\n",
|
|
|
|
def->target.port);
|
2009-09-23 12:16:41 +00:00
|
|
|
|
2009-09-23 12:52:36 +00:00
|
|
|
/* def:type -> vmx:fileType and def:data.file.path -> vmx:fileName */
|
2009-09-23 12:16:41 +00:00
|
|
|
switch (def->type) {
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_DEV:
|
|
|
|
virBufferVSprintf(buffer, "parallel%d.fileType = \"device\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:52:36 +00:00
|
|
|
virBufferVSprintf(buffer, "parallel%d.fileName = \"%s\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port, def->data.file.path);
|
2009-09-23 12:16:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_FILE:
|
|
|
|
virBufferVSprintf(buffer, "parallel%d.fileType = \"file\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
2010-01-15 22:05:26 +00:00
|
|
|
fileName = esxVMX_FormatFileName(ctx, def->data.file.path);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
if (fileName == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferVSprintf(buffer, "parallel%d.fileName = \"%s\"\n",
|
2009-11-05 13:19:14 +00:00
|
|
|
def->target.port, fileName);
|
2009-09-23 12:52:36 +00:00
|
|
|
|
|
|
|
VIR_FREE(fileName);
|
2009-09-23 12:16:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-01-15 22:05:26 +00:00
|
|
|
ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
|
2010-04-02 19:34:31 +00:00
|
|
|
_("Unsupported character device type '%s'"),
|
2009-09-23 12:16:41 +00:00
|
|
|
virDomainChrTypeToString(def->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|