/* * esx_interface_driver.c: interface driver functions for managing VMware ESX * host interfaces * * Copyright (C) 2010-2011 Red Hat, Inc. * Copyright (C) 2010-2012 Matthias Bolte * * 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, see * . * */ #include #include "internal.h" #include "interface_conf.h" #include "virsocketaddr.h" #include "esx_private.h" #include "esx_vi.h" #define VIR_FROM_THIS VIR_FROM_ESX static int esxConnectNumOfInterfaces(virConnectPtr conn) { esxPrivate *priv = conn->privateData; esxVI_PhysicalNic *physicalNicList = NULL; esxVI_PhysicalNic *physicalNic = NULL; int count = 0; if (esxVI_EnsureSession(priv->primary) < 0 || esxVI_LookupPhysicalNicList(priv->primary, &physicalNicList) < 0) { return -1; } for (physicalNic = physicalNicList; physicalNic; physicalNic = physicalNic->_next) { ++count; } esxVI_PhysicalNic_Free(&physicalNicList); return count; } static int esxConnectListInterfaces(virConnectPtr conn, char **const names, int maxnames) { esxPrivate *priv = conn->privateData; esxVI_PhysicalNic *physicalNicList = NULL; esxVI_PhysicalNic *physicalNic = NULL; int count = 0; if (maxnames == 0) return 0; if (esxVI_EnsureSession(priv->primary) < 0 || esxVI_LookupPhysicalNicList(priv->primary, &physicalNicList) < 0) { return -1; } for (physicalNic = physicalNicList; physicalNic; physicalNic = physicalNic->_next) { names[count] = g_strdup(physicalNic->device); ++count; } esxVI_PhysicalNic_Free(&physicalNicList); return count; } static int esxConnectNumOfDefinedInterfaces(virConnectPtr conn G_GNUC_UNUSED) { /* ESX interfaces are always active */ return 0; } static int esxConnectListDefinedInterfaces(virConnectPtr conn G_GNUC_UNUSED, char **const names G_GNUC_UNUSED, int maxnames G_GNUC_UNUSED) { /* ESX interfaces are always active */ return 0; } static virInterfacePtr esxInterfaceLookupByName(virConnectPtr conn, const char *name) { virInterfacePtr iface = NULL; esxPrivate *priv = conn->privateData; esxVI_PhysicalNic *physicalNic = NULL; if (esxVI_EnsureSession(priv->primary) < 0 || esxVI_LookupPhysicalNicByMACAddress(priv->primary, name, &physicalNic, esxVI_Occurrence_RequiredItem) < 0) { return NULL; } iface = virGetInterface(conn, physicalNic->device, physicalNic->mac); esxVI_PhysicalNic_Free(&physicalNic); return iface; } static virInterfacePtr esxInterfaceLookupByMACString(virConnectPtr conn, const char *mac) { virInterfacePtr iface = NULL; esxPrivate *priv = conn->privateData; esxVI_PhysicalNic *physicalNic = NULL; if (esxVI_EnsureSession(priv->primary) < 0 || esxVI_LookupPhysicalNicByMACAddress(priv->primary, mac, &physicalNic, esxVI_Occurrence_RequiredItem) < 0) { return NULL; } iface = virGetInterface(conn, physicalNic->device, physicalNic->mac); esxVI_PhysicalNic_Free(&physicalNic); return iface; } static char * esxInterfaceGetXMLDesc(virInterfacePtr iface, unsigned int flags) { char *xml = NULL; esxPrivate *priv = iface->conn->privateData; esxVI_PhysicalNic *physicalNic = NULL; virInterfaceDef def = { 0 }; bool hasAddress = false; virInterfaceProtocolDef *protocols; virInterfaceProtocolDef protocol = { 0 }; virSocketAddr socketAddress = { 0 }; virInterfaceIPDef *ips; virInterfaceIPDef ip = { 0 }; virCheckFlags(VIR_INTERFACE_XML_INACTIVE, NULL); if (esxVI_EnsureSession(priv->primary) < 0 || esxVI_LookupPhysicalNicByMACAddress(priv->primary, iface->mac, &physicalNic, esxVI_Occurrence_RequiredItem) < 0) { return NULL; } def.type = VIR_INTERFACE_TYPE_ETHERNET; def.name = physicalNic->device; def.mac = physicalNic->mac; def.startmode = VIR_INTERFACE_START_ONBOOT; /* FIXME: Add support for IPv6, requires to use vSphere API 4.0 */ if (physicalNic->spec->ip) { protocol.family = (char *)"ipv4"; if (physicalNic->spec->ip->dhcp == esxVI_Boolean_True) protocol.dhcp = 1; if (physicalNic->spec->ip->ipAddress && physicalNic->spec->ip->subnetMask && strlen(physicalNic->spec->ip->ipAddress) > 0 && strlen(physicalNic->spec->ip->subnetMask) > 0) { hasAddress = true; } if (protocol.dhcp || hasAddress) { protocols = &protocol; def.nprotos = 1; def.protos = &protocols; } if (hasAddress && !(protocol.dhcp && (flags & VIR_INTERFACE_XML_INACTIVE))) { ips = &ip; protocol.nips = 1; protocol.ips = &ips; if (virSocketAddrParseIPv4(&socketAddress, physicalNic->spec->ip->subnetMask) < 0) { goto cleanup; } ip.address = physicalNic->spec->ip->ipAddress; ip.prefix = virSocketAddrGetNumNetmaskBits(&socketAddress); } } xml = virInterfaceDefFormat(&def); cleanup: esxVI_PhysicalNic_Free(&physicalNic); return xml; } static int esxInterfaceIsActive(virInterfacePtr iface G_GNUC_UNUSED) { /* ESX interfaces are always active */ return 1; } virInterfaceDriver esxInterfaceDriver = { .connectNumOfInterfaces = esxConnectNumOfInterfaces, /* 0.10.0 */ .connectListInterfaces = esxConnectListInterfaces, /* 0.10.0 */ .connectNumOfDefinedInterfaces = esxConnectNumOfDefinedInterfaces, /* 0.10.0 */ .connectListDefinedInterfaces = esxConnectListDefinedInterfaces, /* 0.10.0 */ .interfaceLookupByName = esxInterfaceLookupByName, /* 0.10.0 */ .interfaceLookupByMACString = esxInterfaceLookupByMACString, /* 0.10.0 */ .interfaceGetXMLDesc = esxInterfaceGetXMLDesc, /* 0.10.0 */ .interfaceIsActive = esxInterfaceIsActive, /* 0.10.0 */ };