Initial Linux containers work

* configure.in include/libvirt/virterror.h src/Makefile.am
  src/driver.h src/lxc_conf.[ch] src/lxc_driver.[ch] src/virterror.c:
  Applied 3 patches from Dave Leskovec for intial support of
  Linux containers, configured off by default, work in progress.
* src/libvirt.c: improve virDomainCreateLinux xmlDesc description
Daniel
This commit is contained in:
Daniel Veillard 2008-03-21 15:03:37 +00:00
parent ccb19376f7
commit f163895204
11 changed files with 1641 additions and 1 deletions

View File

@ -1,3 +1,11 @@
Fri Mar 21 15:59:53 CET 2008 Daniel Veillard <veillard@redhat.com>
* configure.in include/libvirt/virterror.h src/Makefile.am
src/driver.h src/lxc_conf.[ch] src/lxc_driver.[ch] src/virterror.c:
Applied 3 patches from Dave Leskovec for intial support of
Linux containers, configured off by default, work in progress.
* src/libvirt.c: improve virDomainCreateLinux xmlDesc description
Thu Mar 20 12:23:03 CET 2008 Daniel Veillard <veillard@redhat.com>
* src/util.c src/util.h src/xml.c: applied patch from Hiroyuki Kaguchi

View File

@ -132,6 +132,8 @@ AC_ARG_WITH(qemu,
[ --with-qemu add QEMU/KVM support (on)],[],[with_qemu=yes])
AC_ARG_WITH(openvz,
[ --with-openvz add OpenVZ support (off)],[],[with_openvz=no])
AC_ARG_WITH(lxc,
[ --with-lxc add Linux Container support (off)],[],[with_lxc=no])
AC_ARG_WITH(test,
[ --with-test add test driver support (on)],[],[with_test=yes])
AC_ARG_WITH(remote,
@ -229,6 +231,9 @@ WITH_XEN=0
if test "$with_openvz" = "yes" ; then
LIBVIRT_FEATURES="$LIBVIRT_FEATURES -DWITH_OPENVZ"
fi
if test "$with_lxc" = "yes" ; then
LIBVIRT_FEATURES="$LIBVIRT_FEATURES -DWITH_LXC"
fi
if test "$with_qemu" = "yes" ; then
LIBVIRT_FEATURES="$LIBVIRT_FEATURES -DWITH_QEMU"
fi
@ -950,6 +955,7 @@ AC_MSG_NOTICE([ Xen: $with_xen])
AC_MSG_NOTICE([ Proxy: $with_xen_proxy])
AC_MSG_NOTICE([ QEMU: $with_qemu])
AC_MSG_NOTICE([ OpenVZ: $with_openvz])
AC_MSG_NOTICE([ LXC: $with_lxc])
AC_MSG_NOTICE([ Test: $with_test])
AC_MSG_NOTICE([ Remote: $with_remote])
AC_MSG_NOTICE([Libvirtd: $with_libvirtd])

View File

@ -54,6 +54,7 @@ typedef enum {
VIR_FROM_OPENVZ, /* Error from OpenVZ driver */
VIR_FROM_XENXM, /* Error at Xen XM layer */
VIR_FROM_STATS_LINUX, /* Error in the Linux Stats code */
VIR_FROM_LXC, /* Error from Linux Container driver */
VIR_FROM_STORAGE, /* Error from storage driver */
} virErrorDomain;

View File

@ -59,6 +59,8 @@ CLIENT_SOURCES = \
qemu_conf.c qemu_conf.h \
openvz_conf.c openvz_conf.h \
openvz_driver.c openvz_driver.h \
lxc_driver.c lxc_driver.h \
lxc_conf.c lxc_conf.h \
nodeinfo.h nodeinfo.c \
storage_conf.h storage_conf.c \
storage_driver.h storage_driver.c \

View File

@ -24,6 +24,7 @@ typedef enum {
VIR_DRV_QEMU = 3,
VIR_DRV_REMOTE = 4,
VIR_DRV_OPENVZ = 5,
VIR_DRV_LXC = 6
} virDrvNo;

View File

@ -45,6 +45,9 @@
#ifdef WITH_OPENVZ
#include "openvz_driver.h"
#endif
#ifdef WITH_LXC
#include "lxc_driver.h"
#endif
/*
* TODO:
@ -271,6 +274,9 @@ virInitialize(void)
#endif
#ifdef WITH_OPENVZ
if (openvzRegister() == -1) return -1;
#endif
#ifdef WITH_LXC
if (lxcRegister() == -1) return -1;
#endif
if (storageRegister() == -1) return -1;
#ifdef WITH_REMOTE
@ -1183,7 +1189,7 @@ virDomainGetConnect (virDomainPtr dom)
/**
* virDomainCreateLinux:
* @conn: pointer to the hypervisor connection
* @xmlDesc: an XML description of the domain
* @xmlDesc: string containing an XML description of the domain
* @flags: an optional set of virDomainFlags
*
* Launch a new Linux guest domain, based on an XML description similar

942
src/lxc_conf.c Normal file
View File

@ -0,0 +1,942 @@
/*
* Copyright IBM Corp. 2008
*
* lxc_conf.c: config functions for managing linux containers
*
* Authors:
* David L. Leskovec <dlesko at linux.vnet.ibm.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
*
*/
/* includes */
#include <config.h>
#ifdef WITH_LXC
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <libxml/parser.h>
#include <libxml/tree.h>
#include <libxml/uri.h>
#include <libxml/xpath.h>
#include "buf.h"
#include "util.h"
#include "uuid.h"
#include "lxc_conf.h"
/* debug macros */
#define DEBUG(fmt,...) VIR_DEBUG(__FILE__, fmt, __VA_ARGS__)
#define DEBUG0(msg) VIR_DEBUG(__FILE__, "%s", msg)
/* Functions */
void lxcError(virConnectPtr conn, virDomainPtr dom, int code,
const char *fmt, ...)
{
va_list args;
char errorMessage[LXC_MAX_ERROR_LEN];
const char *codeErrorMessage;
if (fmt) {
va_start(args, fmt);
vsnprintf(errorMessage, LXC_MAX_ERROR_LEN-1, fmt, args);
va_end(args);
} else {
errorMessage[0] = '\0';
}
codeErrorMessage = __virErrorMsg(code, fmt);
__virRaiseError(conn, dom, NULL, VIR_FROM_LXC, code, VIR_ERR_ERROR,
codeErrorMessage, errorMessage, NULL, 0, 0,
codeErrorMessage, errorMessage);
}
static inline int lxcIsEmptyXPathStringObj(xmlXPathObjectPtr xpathObj)
{
if ((xpathObj == NULL) ||
(xpathObj->type != XPATH_STRING) ||
(xpathObj->stringval == NULL) ||
(xpathObj->stringval[0] == 0)) {
return 1;
} else {
return 0;
}
}
static int lxcParseMountXML(virConnectPtr conn, xmlNodePtr nodePtr,
lxc_mount_t *lxcMount)
{
xmlChar *fsType = NULL;
xmlNodePtr curNode;
xmlChar *mountSource = NULL;
xmlChar *mountTarget = NULL;
int strLen;
int rc = -1;
if (NULL == (fsType = xmlGetProp(nodePtr, BAD_CAST "type"))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("missing filesystem type"));
goto error;
}
if (xmlStrEqual(fsType, BAD_CAST "mount") == 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("invalid filesystem type"));
goto error;
}
for (curNode = nodePtr->children;
NULL != curNode;
curNode = curNode->next) {
if (curNode->type != XML_ELEMENT_NODE) {
continue;
}
if ((mountSource == NULL) &&
(xmlStrEqual(curNode->name, BAD_CAST "source"))) {
mountSource = xmlGetProp(curNode, BAD_CAST "dir");
} else if ((mountTarget == NULL) &&
(xmlStrEqual(curNode->name, BAD_CAST "target"))) {
mountTarget = xmlGetProp(curNode, BAD_CAST "dir");
}
}
if (mountSource == NULL) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("missing mount source"));
goto error;
}
strLen = xmlStrlen(mountSource);
if ((strLen > (PATH_MAX-1)) || (0 == strLen)) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("empty or invalid mount source"));
goto error;
}
strncpy(lxcMount->source, (char *)mountSource, strLen);
lxcMount->source[strLen] = '\0';
if (mountTarget == NULL) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("missing mount target"));
goto error;
}
strLen = xmlStrlen(mountTarget);
if ((strLen > (PATH_MAX-1)) || (0 == strLen)) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("empty or invalid mount target"));
goto error;
}
strncpy(lxcMount->target, (char *)mountTarget, strLen);
lxcMount->target[strLen] = '\0';
rc = 0;
error:
xmlFree(mountSource);
xmlFree(mountTarget);
return rc;
}
static int lxcParseDomainName(virConnectPtr conn, char **name,
xmlXPathContextPtr contextPtr)
{
int rc = -1;
xmlXPathObjectPtr xpathObj = NULL;
xpathObj = xmlXPathEval(BAD_CAST "string(/domain/name[1])", contextPtr);
if (lxcIsEmptyXPathStringObj(xpathObj)) {
lxcError(conn, NULL, VIR_ERR_NO_NAME, NULL);
goto parse_complete;
}
*name = strdup((const char *)xpathObj->stringval);
if (NULL == *name) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, NULL);
goto parse_complete;
}
rc = 0;
parse_complete:
xmlXPathFreeObject(xpathObj);
return rc;
}
static int lxcParseDomainUUID(virConnectPtr conn, unsigned char *uuid,
xmlXPathContextPtr contextPtr)
{
int rc = -1;
xmlXPathObjectPtr xpathObj = NULL;
xpathObj = xmlXPathEval(BAD_CAST "string(/domain/uuid[1])", contextPtr);
if (lxcIsEmptyXPathStringObj(xpathObj)) {
if ((rc = virUUIDGenerate(uuid))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("failed to generate uuid: %s"), strerror(rc));
goto parse_complete;
}
} else {
if (virUUIDParse((const char *)xpathObj->stringval, uuid) < 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("invalid uuid element"));
goto parse_complete;
}
}
rc = 0;
parse_complete:
xmlXPathFreeObject(xpathObj);
return rc;
}
static int lxcParseDomainMounts(virConnectPtr conn,
lxc_mount_t **mounts,
xmlXPathContextPtr contextPtr)
{
int rc = -1;
xmlXPathObjectPtr xpathObj = NULL;
int i;
lxc_mount_t *mountObj;
lxc_mount_t *prevObj = NULL;
int nmounts = 0;
xpathObj = xmlXPathEval(BAD_CAST "/domain/devices/filesystem",
contextPtr);
if ((xpathObj != NULL) &&
(xpathObj->type == XPATH_NODESET) &&
(xpathObj->nodesetval != NULL) &&
(xpathObj->nodesetval->nodeNr >= 0)) {
for (i = 0; i < xpathObj->nodesetval->nodeNr; ++i) {
mountObj = calloc(1, sizeof(lxc_mount_t));
if (NULL == mountObj) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "mount");
goto parse_complete;
}
rc = lxcParseMountXML(conn, xpathObj->nodesetval->nodeTab[i],
mountObj);
if (0 > rc) {
free(mountObj);
goto parse_complete;
}
/* set the linked list pointers */
nmounts++;
mountObj->next = NULL;
if (0 == i) {
*mounts = mountObj;
} else {
prevObj->next = mountObj;
}
prevObj = mountObj;
}
}
rc = nmounts;
parse_complete:
xmlXPathFreeObject(xpathObj);
return rc;
}
static int lxcParseDomainInit(virConnectPtr conn, char* init, xmlXPathContextPtr contextPtr)
{
int rc = -1;
xmlXPathObjectPtr xpathObj = NULL;
xpathObj = xmlXPathEval(BAD_CAST "string(/domain/os/init[1])",
contextPtr);
if (lxcIsEmptyXPathStringObj(xpathObj)) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("invalid or missing init element"));
goto parse_complete;
}
if (strlen((const char*)xpathObj->stringval) >= PATH_MAX - 1) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("init string too long"));
goto parse_complete;
}
strcpy(init, (const char *)xpathObj->stringval);
rc = 0;
parse_complete:
xmlXPathFreeObject(xpathObj);
return rc;
}
static int lxcParseDomainTty(virConnectPtr conn, char *tty, xmlXPathContextPtr contextPtr)
{
int rc = -1;
xmlXPathObjectPtr xpathObj = NULL;
xpathObj = xmlXPathEval(BAD_CAST "string(/domain/devices/console[1]/@tty)",
contextPtr);
if (lxcIsEmptyXPathStringObj(xpathObj)) {
/* make sure the tty string is empty */
tty[0] = 0x00;
} else {
/* check the source string length */
if (strlen((const char*)xpathObj->stringval) >= LXC_MAX_TTY_NAME - 1) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("tty name is too long"));
goto parse_complete;
}
strcpy(tty, (const char *)xpathObj->stringval);
}
rc = 0;
parse_complete:
xmlXPathFreeObject(xpathObj);
return rc;
}
static int lxcParseDomainMemory(virConnectPtr conn, int* memory, xmlXPathContextPtr contextPtr)
{
int rc = -1;
xmlXPathObjectPtr xpathObj = NULL;
char *endChar = NULL;
xpathObj = xmlXPathEval(BAD_CAST "string(/domain/memory[1])", contextPtr);
if (lxcIsEmptyXPathStringObj(xpathObj)) {
/* not an error, default to an invalid value so it's not used */
*memory = -1;
} else {
*memory = strtoll((const char*)xpathObj->stringval,
&endChar, 10);
if ((endChar == (const char*)xpathObj->stringval) ||
(*endChar != '\0')) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("invalid memory value"));
goto parse_complete;
}
}
rc = 0;
parse_complete:
xmlXPathFreeObject(xpathObj);
return rc;
}
static lxc_vm_def_t * lxcParseXML(virConnectPtr conn, xmlDocPtr docPtr)
{
xmlNodePtr rootNodePtr = NULL;
xmlXPathContextPtr contextPtr = NULL;
xmlChar *xmlProp = NULL;
lxc_vm_def_t *containerDef;
if (!(containerDef = calloc(1, sizeof(*containerDef)))) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "containerDef");
return NULL;
}
/* Prepare parser / xpath context */
rootNodePtr = xmlDocGetRootElement(docPtr);
if ((rootNodePtr == NULL) ||
(!xmlStrEqual(rootNodePtr->name, BAD_CAST "domain"))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("invalid root element"));
goto error;
}
contextPtr = xmlXPathNewContext(docPtr);
if (contextPtr == NULL) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "context");
goto error;
}
/* Verify the domain type is linuxcontainer */
if (!(xmlProp = xmlGetProp(rootNodePtr, BAD_CAST "type"))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("missing domain type"));
goto error;
}
if (!(xmlStrEqual(xmlProp, BAD_CAST LXC_DOMAIN_TYPE))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("invalid domain type"));
goto error;
}
free(xmlProp);
xmlProp = NULL;
if ((xmlProp = xmlGetProp(rootNodePtr, BAD_CAST "id"))) {
if (0 > virStrToLong_i((char*)xmlProp, NULL, 10, &(containerDef->id))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
"invalid domain id");
goto error;
}
} else {
containerDef->id = -1;
}
free(xmlProp);
xmlProp = NULL;
if (lxcParseDomainName(conn, &(containerDef->name), contextPtr) < 0) {
goto error;
}
if (lxcParseDomainInit(conn, containerDef->init, contextPtr) < 0) {
goto error;
}
if (lxcParseDomainUUID(conn, containerDef->uuid, contextPtr) < 0) {
goto error;
}
containerDef->nmounts = lxcParseDomainMounts(conn, &(containerDef->mounts),
contextPtr);
if (0 > containerDef->nmounts) {
goto error;
}
if (lxcParseDomainTty(conn, containerDef->tty, contextPtr) < 0) {
goto error;
}
if (lxcParseDomainMemory(conn, &(containerDef->maxMemory), contextPtr) < 0) {
goto error;
}
xmlXPathFreeContext(contextPtr);
return containerDef;
error:
free(xmlProp);
xmlXPathFreeContext(contextPtr);
lxcFreeVMDef(containerDef);
return NULL;
}
lxc_vm_def_t * lxcParseVMDef(virConnectPtr conn,
const char* xmlString,
const char* fileName)
{
xmlDocPtr xml;
lxc_vm_def_t *containerDef;
xml = xmlReadDoc(BAD_CAST xmlString,
fileName ? fileName : "domain.xml",
NULL, XML_PARSE_NOENT |
XML_PARSE_NONET | XML_PARSE_NOERROR |
XML_PARSE_NOWARNING);
if (!xml) {
lxcError(conn, NULL, VIR_ERR_XML_ERROR, NULL);
return NULL;
}
containerDef = lxcParseXML(conn, xml);
xmlFreeDoc(xml);
return containerDef;
}
lxc_vm_t * lxcAssignVMDef(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_def_t *def)
{
lxc_vm_t *vm = NULL;
if ((vm = lxcFindVMByName(driver, def->name))) {
if (!lxcIsActiveVM(vm)) {
lxcFreeVMDef(vm->def);
vm->def = def;
} else {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("Can't redefine active VM with name %s"), def->name);
return NULL;
}
return vm;
}
if (!(vm = calloc(1, sizeof(lxc_vm_t)))) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "vm");
return NULL;
}
vm->pid = -1;
vm->def = def;
vm->next = driver->vms;
driver->vms = vm;
if (lxcIsActiveVM(vm)) {
vm->state = VIR_DOMAIN_RUNNING;
driver->nactivevms++;
} else {
vm->state = VIR_DOMAIN_SHUTOFF;
driver->ninactivevms++;
}
return vm;
}
void lxcRemoveInactiveVM(lxc_driver_t *driver,
lxc_vm_t *vm)
{
lxc_vm_t *prevVm = NULL;
lxc_vm_t *curVm;
for (curVm = driver->vms;
(curVm != vm) && (NULL != curVm);
curVm = curVm->next) {
prevVm = curVm;
}
if (curVm) {
if (prevVm) {
prevVm->next = curVm->next;
} else {
driver->vms = curVm->next;
}
driver->ninactivevms--;
}
lxcFreeVM(vm);
}
/* Save a container's config data into a persistent file */
int lxcSaveConfig(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_t *vm,
lxc_vm_def_t *def)
{
int rc = -1;
char *xmlDef;
int fd = -1;
int amtToWrite;
if (!(xmlDef = lxcGenerateXML(conn, driver, vm, def))) {
return -1;
}
if ((fd = open(vm->configFile,
O_WRONLY | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR )) < 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("cannot create config file %s: %s"),
vm->configFile, strerror(errno));
goto cleanup;
}
amtToWrite = strlen(xmlDef);
if (safewrite(fd, xmlDef, amtToWrite) < 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("cannot write config file %s: %s"),
vm->configFile, strerror(errno));
goto cleanup;
}
if (close(fd) < 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("cannot save config file %s: %s"),
vm->configFile, strerror(errno));
goto cleanup;
}
rc = 0;
cleanup:
if (fd != -1) {
close(fd);
}
free(xmlDef);
return rc;
}
int lxcSaveVMDef(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_t *vm,
lxc_vm_def_t *def)
{
int rc = -1;
char uuidstr[VIR_UUID_STRING_BUFLEN];
if (vm->configFile[0] == '\0') {
if ((rc = virFileMakePath(driver->configDir))) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("cannot create config directory %s: %s"),
driver->configDir, strerror(rc));
goto save_complete;
}
virUUIDFormat(def->uuid, uuidstr);
if (virFileBuildPath(driver->configDir, uuidstr, ".xml",
vm->configFile, PATH_MAX) < 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("cannot construct config file path"));
goto save_complete;
}
strncpy(vm->configFileBase, uuidstr, PATH_MAX);
strncat(vm->configFileBase, ".xml", PATH_MAX - strlen(uuidstr));
}
rc = lxcSaveConfig(conn, driver, vm, def);
save_complete:
return rc;
}
static lxc_vm_t * lxcLoadConfig(lxc_driver_t *driver,
const char *file,
const char *fullFilePath,
const char *xmlData)
{
lxc_vm_def_t *containerDef;
lxc_vm_t * vm;
char uuidstr[VIR_UUID_STRING_BUFLEN];
containerDef = lxcParseVMDef(NULL, xmlData, file);
if (NULL == containerDef) {
DEBUG0("Error parsing container config");
return NULL;
}
virUUIDFormat(containerDef->uuid, uuidstr);
if (!virFileMatchesNameSuffix(file, uuidstr, ".xml")) {
DEBUG0("Container uuid does not match config file name");
lxcFreeVMDef(containerDef);
return NULL;
}
vm = lxcAssignVMDef(NULL, driver, containerDef);
if (NULL == vm) {
DEBUG0("Failed to load container config");
lxcFreeVMDef(containerDef);
return NULL;
}
strncpy(vm->configFile, fullFilePath, PATH_MAX);
vm->configFile[PATH_MAX-1] = '\0';
strncpy(vm->configFileBase, file, PATH_MAX);
vm->configFile[PATH_MAX-1] = '\0';
return vm;
}
int lxcLoadDriverConfig(virConnectPtr conn)
{
lxc_driver_t *driverPtr = (lxc_driver_t*)conn->privateData;
/* Set the container configuration directory */
driverPtr->configDir = strdup(SYSCONF_DIR "/libvirt/lxc");
if (NULL == driverPtr->configDir) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "configDir");
return -1;
}
return 0;
}
int lxcLoadContainerConfigFile(lxc_driver_t *driver,
const char *file)
{
int rc = -1;
char tempPath[PATH_MAX];
char* xmlData;
rc = virFileBuildPath(driver->configDir, file, NULL, tempPath,
PATH_MAX);
if (0 > rc) {
DEBUG0("config file name too long");
goto load_complete;
}
if ((rc = virFileReadAll(tempPath, LXC_MAX_XML_LENGTH, &xmlData)) < 0) {
goto load_complete;
}
lxcLoadConfig(driver, file, tempPath, xmlData);
free(xmlData);
load_complete:
return rc;
}
int lxcLoadContainerInfo(virConnectPtr conn)
{
int rc = -1;
lxc_driver_t *driverPtr = (lxc_driver_t*)conn->privateData;
DIR *dir;
struct dirent *dirEntry;
if (!(dir = opendir(driverPtr->configDir))) {
if (ENOENT == errno) {
/* no config dir => no containers */
rc = 0;
} else {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("failed to open config directory: %s"), strerror(errno));
}
goto load_complete;
}
while ((dirEntry = readdir(dir))) {
if (dirEntry->d_name[0] == '.') {
continue;
}
if (!virFileHasSuffix(dirEntry->d_name, ".xml")) {
continue;
}
lxcLoadContainerConfigFile(driverPtr, dirEntry->d_name);
}
closedir(dir);
rc = 0;
load_complete:
return rc;
}
/* Generate an XML document describing the vm's configuration */
char *lxcGenerateXML(virConnectPtr conn,
lxc_driver_t *driver ATTRIBUTE_UNUSED,
lxc_vm_t *vm,
lxc_vm_def_t *def)
{
virBufferPtr buf = 0;
unsigned char *uuid;
char uuidstr[VIR_UUID_STRING_BUFLEN];
lxc_mount_t *mount;
buf = virBufferNew(LXC_MAX_XML_LENGTH);
if (!buf) {
goto no_memory;
}
if (lxcIsActiveVM(vm)) {
if (virBufferVSprintf(buf, "<domain type='linuxcontainer' id='%d'>\n",
vm->def->id) < 0) {
goto no_memory;
}
} else {
if (virBufferAddLit(buf, "<domain type='linuxcontainer'>\n") < 0) {
goto no_memory;
}
}
if (virBufferVSprintf(buf, " <name>%s</name>\n", def->name) < 0) {
goto no_memory;
}
uuid = def->uuid;
virUUIDFormat(uuid, uuidstr);
if (virBufferVSprintf(buf, " <uuid>%s</uuid>\n", uuidstr) < 0) {
goto no_memory;
}
if (virBufferAddLit(buf, " <os>\n") < 0) {
goto no_memory;
}
if (virBufferVSprintf(buf, " <init>%s</init>\n", def->init) < 0) {
goto no_memory;
}
if (virBufferAddLit(buf, " </os>\n") < 0) {
goto no_memory;
}
if (virBufferVSprintf(buf, " <memory>%d</memory>\n", def->maxMemory) < 0) {
goto no_memory;
}
if (virBufferAddLit(buf, " <devices>\n") < 0) {
goto no_memory;
}
/* loop adding mounts */
for (mount = def->mounts; mount; mount = mount->next) {
if (virBufferAddLit(buf, " <filesystem type='mount'>\n") < 0) {
goto no_memory;
}
if (virBufferVSprintf(buf, " <source dir='%s'/>\n",
mount->source) < 0) {
goto no_memory;
}
if (virBufferVSprintf(buf, " <target dir='%s'/>\n",
mount->target) < 0) {
goto no_memory;
}
if (virBufferAddLit(buf, " </filesystem>\n") < 0) {
goto no_memory;
}
}
if (virBufferVSprintf(buf, " <console tty='%s'/>\n", def->tty) < 0) {
goto no_memory;
}
if (virBufferAddLit(buf, " </devices>\n") < 0) {
goto no_memory;
}
if (virBufferAddLit(buf, "</domain>\n") < 0) {
goto no_memory;
}
return virBufferContentAndFree(buf);
no_memory:
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "generateXml");
virBufferFree(buf);
return NULL;
}
void lxcFreeVMDef(lxc_vm_def_t *vmdef)
{
lxc_mount_t *curMount = vmdef->mounts;
lxc_mount_t *nextMount;
while (curMount) {
nextMount = curMount->next;
free(curMount);
curMount = nextMount;
}
free(vmdef->name);
vmdef->name = NULL;
}
void lxcFreeVMs(lxc_vm_t *vms)
{
lxc_vm_t *curVm = vms;
lxc_vm_t *nextVm;
while (curVm) {
lxcFreeVM(curVm);
nextVm = curVm->next;
free(curVm);
curVm = nextVm;
}
}
void lxcFreeVM(lxc_vm_t *vm)
{
lxcFreeVMDef(vm->def);
free(vm->def);
}
lxc_vm_t *lxcFindVMByID(const lxc_driver_t *driver, int id)
{
lxc_vm_t *vm;
for (vm = driver->vms; vm; vm = vm->next) {
if (lxcIsActiveVM(vm) && (vm->def->id == id)) {
return vm;
}
}
return NULL;
}
lxc_vm_t *lxcFindVMByUUID(const lxc_driver_t *driver,
const unsigned char *uuid)
{
lxc_vm_t *vm;
for (vm = driver->vms; vm; vm = vm->next) {
if (!memcmp(vm->def->uuid, uuid, VIR_UUID_BUFLEN)) {
return vm;
}
}
return NULL;
}
lxc_vm_t *lxcFindVMByName(const lxc_driver_t *driver,
const char *name)
{
lxc_vm_t *vm;
for (vm = driver->vms; vm; vm = vm->next) {
if (STREQ(vm->def->name, name)) {
return vm;
}
}
return NULL;
}
int lxcDeleteConfig(virConnectPtr conn,
lxc_driver_t *driver ATTRIBUTE_UNUSED,
const char *configFile,
const char *name)
{
if (!configFile[0]) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("no config file for %s"), name);
return -1;
}
if (unlink(configFile) < 0) {
lxcError(conn, NULL, VIR_ERR_INTERNAL_ERROR,
_("cannot remove config for %s"), name);
return -1;
}
return 0;
}
#endif /* WITH_LXC */

142
src/lxc_conf.h Normal file
View File

@ -0,0 +1,142 @@
/*
* Copyright IBM Corp. 2008
*
* lxc_conf.h: header file for linux container config functions
*
* Authors:
* David L. Leskovec <dlesko at linux.vnet.ibm.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
*/
#ifndef LXC_CONF_H
#define LXC_CONF_H
#include <config.h>
#ifdef WITH_LXC
#include "internal.h"
/* Defines */
#define LXC_MAX_TTY_NAME 32
#define LXC_MAX_XML_LENGTH 16384
#define LXC_MAX_ERROR_LEN 1024
#define LXC_DOMAIN_TYPE "lxc"
typedef struct __lxc_mount lxc_mount_t;
struct __lxc_mount {
char source[PATH_MAX]; /* user's directory */
char target[PATH_MAX];
lxc_mount_t *next;
};
typedef struct __lxc_vm_def lxc_vm_def_t;
struct __lxc_vm_def {
unsigned char uuid[VIR_UUID_BUFLEN];
char* name;
int id;
/* init command string */
char init[PATH_MAX];
int maxMemory;
/* mounts - list of mount structs */
int nmounts;
lxc_mount_t *mounts;
/* tty device */
char tty[LXC_MAX_TTY_NAME];
};
typedef struct __lxc_vm lxc_vm_t;
struct __lxc_vm {
int pid;
int state;
char configFile[PATH_MAX];
char configFileBase[PATH_MAX];
int parentTty;
lxc_vm_def_t *def;
lxc_vm_t *next;
};
typedef struct __lxc_driver lxc_driver_t;
struct __lxc_driver {
lxc_vm_t *vms;
int nactivevms;
int ninactivevms;
char* configDir;
};
/* Types and structs */
/* Inline Functions */
static inline int lxcIsActiveVM(lxc_vm_t *vm)
{
return vm->def->id != -1;
}
/* Function declarations */
lxc_vm_def_t * lxcParseVMDef(virConnectPtr conn,
const char* xmlString,
const char* fileName);
int lxcSaveVMDef(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_t *vm,
lxc_vm_def_t *def);
int lxcLoadDriverConfig(virConnectPtr conn);
int lxcSaveConfig(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_t *vm,
lxc_vm_def_t *def);
int lxcLoadContainerInfo(virConnectPtr conn);
int lxcLoadContainerConfigFile(lxc_driver_t *driver,
const char *file);
lxc_vm_t * lxcAssignVMDef(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_def_t *def);
char *lxcGenerateXML(virConnectPtr conn,
lxc_driver_t *driver,
lxc_vm_t *vm,
lxc_vm_def_t *def);
lxc_vm_t *lxcFindVMByID(const lxc_driver_t *driver, int id);
lxc_vm_t *lxcFindVMByUUID(const lxc_driver_t *driver,
const unsigned char *uuid);
lxc_vm_t *lxcFindVMByName(const lxc_driver_t *driver,
const char *name);
void lxcRemoveInactiveVM(lxc_driver_t *driver,
lxc_vm_t *vm);
void lxcFreeVMs(lxc_vm_t *vms);
void lxcFreeVM(lxc_vm_t *vm);
void lxcFreeVMDef(lxc_vm_def_t *vmdef);
int lxcDeleteConfig(virConnectPtr conn,
lxc_driver_t *driver,
const char *configFile,
const char *name);
void lxcError(virConnectPtr conn,
virDomainPtr dom,
int code, const char *fmt, ...)
ATTRIBUTE_FORMAT(printf,4,5);
#endif /* WITH_LXC */
#endif /* LXC_CONF_H */

484
src/lxc_driver.c Normal file
View File

@ -0,0 +1,484 @@
/*
* Copyright IBM Corp. 2008
*
* lxc_driver.c: linux container driver functions
*
* Authors:
* David L. Leskovec <dlesko at linux.vnet.ibm.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>
#ifdef WITH_LXC
#include <sched.h>
#include <sys/utsname.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <wait.h>
#include "lxc_conf.h"
#include "lxc_driver.h"
#include "driver.h"
#include "internal.h"
/* debug macros */
#define DEBUG(fmt,...) VIR_DEBUG(__FILE__, fmt, __VA_ARGS__)
#define DEBUG0(msg) VIR_DEBUG(__FILE__, "%s", msg)
static int lxcStartup(virConnectPtr conn);
static int lxcShutdown(virConnectPtr conn);
/* Functions */
static int lxcDummyChild( void *argv ATTRIBUTE_UNUSED )
{
exit(0);
}
static int lxcCheckContainerSupport( void )
{
int rc = 0;
int flags = CLONE_NEWPID|CLONE_NEWNS|CLONE_NEWUTS|CLONE_NEWUSER|
CLONE_NEWIPC|SIGCHLD;
int cpid;
char *childStack;
char *stack;
int childStatus;
stack = malloc(getpagesize() * 4);
if(!stack) {
DEBUG0("Unable to allocate stack");
rc = -1;
goto check_complete;
}
childStack = stack + (getpagesize() * 4);
cpid = clone(lxcDummyChild, childStack, flags, NULL);
if ((0 > cpid) && (EINVAL == errno)) {
DEBUG0("clone call returned EINVAL, container support is not enabled");
rc = -1;
} else {
waitpid(cpid, &childStatus, 0);
}
free(stack);
check_complete:
return rc;
}
static const char *lxcProbe(void)
{
#ifdef __linux__
if (0 == lxcCheckContainerSupport()) {
return("lxc:///");
}
#endif
return(NULL);
}
static virDrvOpenStatus lxcOpen(virConnectPtr conn,
xmlURIPtr uri,
virConnectAuthPtr auth ATTRIBUTE_UNUSED,
int flags ATTRIBUTE_UNUSED)
{
uid_t uid = getuid();
/* Check that the user is root */
if (0 != uid) {
goto declineConnection;
}
/* Verify uri was specified */
if ((NULL == uri) || (NULL == uri->scheme)) {
goto declineConnection;
}
/* Check that the uri scheme is lxc */
if (STRNEQ(uri->scheme, "lxc")) {
goto declineConnection;
}
/* Check that this is a container enabled kernel */
if(0 != lxcCheckContainerSupport()) {
goto declineConnection;
}
/* initialize driver data */
if (0 > lxcStartup(conn)) {
goto declineConnection;
}
return VIR_DRV_OPEN_SUCCESS;
declineConnection:
return VIR_DRV_OPEN_DECLINED;
}
static int lxcClose(virConnectPtr conn)
{
return lxcShutdown(conn);
}
static virDomainPtr lxcDomainLookupByID(virConnectPtr conn,
int id)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_t *vm = lxcFindVMByID(driver, id);
virDomainPtr dom;
if (!vm) {
lxcError(conn, NULL, VIR_ERR_NO_DOMAIN, NULL);
return NULL;
}
dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
if (dom) {
dom->id = vm->def->id;
}
return dom;
}
static virDomainPtr lxcDomainLookupByUUID(virConnectPtr conn,
const unsigned char *uuid)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_t *vm = lxcFindVMByUUID(driver, uuid);
virDomainPtr dom;
if (!vm) {
lxcError(conn, NULL, VIR_ERR_NO_DOMAIN, NULL);
return NULL;
}
dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
if (dom) {
dom->id = vm->def->id;
}
return dom;
}
static virDomainPtr lxcDomainLookupByName(virConnectPtr conn,
const char *name)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_t *vm = lxcFindVMByName(driver, name);
virDomainPtr dom;
if (!vm) {
lxcError(conn, NULL, VIR_ERR_NO_DOMAIN, NULL);
return NULL;
}
dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
if (dom) {
dom->id = vm->def->id;
}
return dom;
}
static int lxcListDomains(virConnectPtr conn, int *ids, int nids)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_t *vm;
int numDoms = 0;
for (vm = driver->vms; vm && (numDoms < nids); vm = vm->next) {
if (lxcIsActiveVM(vm)) {
ids[numDoms] = vm->def->id;
numDoms++;
}
}
return numDoms;
}
static int lxcNumDomains(virConnectPtr conn)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
return driver->nactivevms;
}
static int lxcListDefinedDomains(virConnectPtr conn,
char **const names, int nnames)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_t *vm;
int numDoms = 0;
int i;
for (vm = driver->vms; vm && (numDoms < nnames); vm = vm->next) {
if (!lxcIsActiveVM(vm)) {
if (!(names[numDoms] = strdup(vm->def->name))) {
lxcError(conn, NULL, VIR_ERR_NO_MEMORY, "names");
goto cleanup;
}
numDoms++;
}
}
return numDoms;
cleanup:
for (i = 0 ; i < numDoms ; i++) {
free(names[i]);
}
return -1;
}
static int lxcNumDefinedDomains(virConnectPtr conn)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
return driver->ninactivevms;
}
static virDomainPtr lxcDomainDefine(virConnectPtr conn, const char *xml)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_def_t *def;
lxc_vm_t *vm;
virDomainPtr dom;
if (!(def = lxcParseVMDef(conn, xml, NULL))) {
return NULL;
}
if (!(vm = lxcAssignVMDef(conn, driver, def))) {
lxcFreeVMDef(def);
return NULL;
}
if (lxcSaveVMDef(conn, driver, vm, def) < 0) {
lxcRemoveInactiveVM(driver, vm);
return NULL;
}
dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
if (dom) {
dom->id = vm->def->id;
}
return dom;
}
static int lxcDomainUndefine(virDomainPtr dom)
{
lxc_driver_t *driver = (lxc_driver_t *)dom->conn->privateData;
lxc_vm_t *vm = lxcFindVMByUUID(driver, dom->uuid);
if (!vm) {
lxcError(dom->conn, dom, VIR_ERR_INVALID_DOMAIN,
_("no domain with matching uuid"));
return -1;
}
if (lxcIsActiveVM(vm)) {
lxcError(dom->conn, dom, VIR_ERR_INTERNAL_ERROR,
_("cannot delete active domain"));
return -1;
}
if (lxcDeleteConfig(dom->conn, driver, vm->configFile, vm->def->name) < 0) {
return -1;
}
vm->configFile[0] = '\0';
lxcRemoveInactiveVM(driver, vm);
return 0;
}
static int lxcDomainGetInfo(virDomainPtr dom,
virDomainInfoPtr info)
{
lxc_driver_t *driver = (lxc_driver_t *)dom->conn->privateData;
lxc_vm_t *vm = lxcFindVMByUUID(driver, dom->uuid);
if (!vm) {
lxcError(dom->conn, dom, VIR_ERR_INVALID_DOMAIN,
_("no domain with matching uuid"));
return -1;
}
info->state = vm->state;
if (!lxcIsActiveVM(vm)) {
info->cpuTime = 0;
} else {
info->cpuTime = 0;
}
info->maxMem = vm->def->maxMemory;
info->memory = vm->def->maxMemory;
info->nrVirtCpu = 1;
return 0;
}
static char *lxcGetOSType(virDomainPtr dom ATTRIBUTE_UNUSED)
{
/* Linux containers only run on Linux */
return strdup("linux");
}
static char *lxcDomainDumpXML(virDomainPtr dom,
int flags ATTRIBUTE_UNUSED)
{
lxc_driver_t *driver = (lxc_driver_t *)dom->conn->privateData;
lxc_vm_t *vm = lxcFindVMByUUID(driver, dom->uuid);
if (!vm) {
lxcError(dom->conn, dom, VIR_ERR_INVALID_DOMAIN,
_("no domain with matching uuid"));
return NULL;
}
return lxcGenerateXML(dom->conn, driver, vm, vm->def);
}
static int lxcStartup(virConnectPtr conn)
{
lxc_driver_t *driver;
driver = calloc(1, sizeof(lxc_driver_t));
if (NULL == driver) {
return -1;
}
conn->privateData = driver;
/* Call function to load lxc driver configuration information */
if (lxcLoadDriverConfig(conn) < 0) {
lxcShutdown(conn);
return -1;
}
/* Call function to load the container configuration files */
if (lxcLoadContainerInfo(conn) < 0) {
lxcShutdown(conn);
return -1;
}
return 0;
}
static void lxcFreeDriver(lxc_driver_t *driver)
{
free(driver->configDir);
free(driver);
}
static int lxcShutdown(virConnectPtr conn)
{
lxc_driver_t *driver = (lxc_driver_t *)conn->privateData;
lxc_vm_t *vms = driver->vms;
lxcFreeVMs(vms);
driver->vms = NULL;
lxcFreeDriver(driver);
conn->privateData = NULL;
return 0;
}
/* Function Tables */
static virDriver lxcDriver = {
VIR_DRV_LXC, /* the number virDrvNo */
"LXC", /* the name of the driver */
LIBVIR_VERSION_NUMBER, /* the version of the backend */
lxcProbe, /* probe */
lxcOpen, /* open */
lxcClose, /* close */
NULL, /* supports_feature */
NULL, /* type */
NULL, /* version */
NULL, /* getHostname */
NULL, /* getURI */
NULL, /* getMaxVcpus */
NULL, /* nodeGetInfo */
NULL, /* getCapabilities */
lxcListDomains, /* listDomains */
lxcNumDomains, /* numOfDomains */
NULL/*lxcDomainCreateLinux*/, /* domainCreateLinux */
lxcDomainLookupByID, /* domainLookupByID */
lxcDomainLookupByUUID, /* domainLookupByUUID */
lxcDomainLookupByName, /* domainLookupByName */
NULL, /* domainSuspend */
NULL, /* domainResume */
NULL, /* domainShutdown */
NULL, /* domainReboot */
NULL, /* domainDestroy */
lxcGetOSType, /* domainGetOSType */
NULL, /* domainGetMaxMemory */
NULL, /* domainSetMaxMemory */
NULL, /* domainSetMemory */
lxcDomainGetInfo, /* domainGetInfo */
NULL, /* domainSave */
NULL, /* domainRestore */
NULL, /* domainCoreDump */
NULL, /* domainSetVcpus */
NULL, /* domainPinVcpu */
NULL, /* domainGetVcpus */
NULL, /* domainGetMaxVcpus */
lxcDomainDumpXML, /* domainDumpXML */
lxcListDefinedDomains, /* listDefinedDomains */
lxcNumDefinedDomains, /* numOfDefinedDomains */
NULL, /* domainCreate */
lxcDomainDefine, /* domainDefineXML */
lxcDomainUndefine, /* domainUndefine */
NULL, /* domainAttachDevice */
NULL, /* domainDetachDevice */
NULL, /* domainGetAutostart */
NULL, /* domainSetAutostart */
NULL, /* domainGetSchedulerType */
NULL, /* domainGetSchedulerParameters */
NULL, /* domainSetSchedulerParameters */
NULL, /* domainMigratePrepare */
NULL, /* domainMigratePerform */
NULL, /* domainMigrateFinish */
NULL, /* domainBlockStats */
NULL, /* domainInterfaceStats */
NULL, /* nodeGetCellsFreeMemory */
NULL, /* getFreeMemory */
};
int lxcRegister(void)
{
virRegisterDriver(&lxcDriver);
return 0;
}
#endif /* WITH_LXC */
/*
* Local variables:
* indent-tabs-mode: nil
* c-indent-level: 4
* c-basic-offset: 4
* tab-width: 4
* End:
*/

45
src/lxc_driver.h Normal file
View File

@ -0,0 +1,45 @@
/*
* Copyright IBM Corp. 2008
*
* lxc_driver.h: header file for linux container driver functions
*
* Authors:
* David L. Leskovec <dlesko at linux.vnet.ibm.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
*/
#ifndef LXC_DRIVER_H
#define LXC_DRIVER_H
#include <config.h>
#ifdef WITH_LXC
/* Function declarations */
int lxcRegister(void);
#endif /* WITH_LXC */
#endif /* LXC_DRIVER_H */
/*
* Local variables:
* indent-tabs-mode: nil
* c-indent-level: 4
* c-basic-offset: 4
* tab-width: 4
* End:
*/

View File

@ -296,6 +296,9 @@ virDefaultErrorFunc(virErrorPtr err)
case VIR_FROM_STATS_LINUX:
dom = "Linux Stats ";
break;
case VIR_FROM_LXC:
dom = "Linux Container ";
break;
case VIR_FROM_STORAGE:
dom = "Storage ";
break;