LXC: Wire up the virDomainCreate{XML}WithFiles methods

Wire up the new virDomainCreate{XML}WithFiles methods in the
LXC driver, so that FDs get passed down to the init process.

The lxc_container code needs to do a little dance in order
to renumber the file descriptors it receives into linear
order, starting from STDERR_FILENO + 1.

Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
This commit is contained in:
Daniel P. Berrange 2013-07-09 18:15:45 +01:00
parent ddaf15d7a3
commit 11693bc6f0
6 changed files with 203 additions and 49 deletions

View File

@ -103,8 +103,10 @@ struct __lxc_child_argv {
size_t nveths; size_t nveths;
char **veths; char **veths;
int monitor; int monitor;
char **ttyPaths; size_t npassFDs;
int *passFDs;
size_t nttyPaths; size_t nttyPaths;
char **ttyPaths;
int handshakefd; int handshakefd;
}; };
@ -217,20 +219,28 @@ static virCommandPtr lxcContainerBuildInitCmd(virDomainDefPtr vmDef)
} }
/** /**
* lxcContainerSetStdio: * lxcContainerSetupFDs:
* @control: control FD from parent * @control: control FD from parent
* @ttyfd: FD of tty to set as the container console * @ttyfd: FD of tty to set as the container console
* @npassFDs: number of extra FDs
* @passFDs: list of extra FDs
* *
* Sets the given tty as the primary conosole for the container as well as * Setup file descriptors in the container. @ttyfd is set to be
* stdout, stdin and stderr. * the container's stdin, stdout & stderr. Any FDs included in
* @passFDs, will be dup()'d such that they start from stderr+1
* with no gaps.
* *
* Returns 0 on success or -1 in case of error * Returns 0 on success or -1 in case of error
*/ */
static int lxcContainerSetStdio(int control, int ttyfd, int handshakefd) static int lxcContainerSetupFDs(int *ttyfd,
size_t npassFDs, int *passFDs)
{ {
int rc = -1; int rc = -1;
int open_max; int open_max;
int fd; int fd;
int last_fd;
size_t i;
size_t j;
if (setsid() < 0) { if (setsid() < 0) {
virReportSystemError(errno, "%s", virReportSystemError(errno, "%s",
@ -238,42 +248,97 @@ static int lxcContainerSetStdio(int control, int ttyfd, int handshakefd)
goto cleanup; goto cleanup;
} }
if (ioctl(ttyfd, TIOCSCTTY, NULL) < 0) { if (ioctl(*ttyfd, TIOCSCTTY, NULL) < 0) {
virReportSystemError(errno, "%s", virReportSystemError(errno, "%s",
_("ioctl(TIOCSTTY) failed")); _("ioctl(TIOCSTTY) failed"));
goto cleanup; goto cleanup;
} }
if (dup2(*ttyfd, STDIN_FILENO) < 0) {
virReportSystemError(errno, "%s",
_("dup2(stdin) failed"));
goto cleanup;
}
if (dup2(*ttyfd, STDOUT_FILENO) < 0) {
virReportSystemError(errno, "%s",
_("dup2(stdout) failed"));
goto cleanup;
}
if (dup2(*ttyfd, STDERR_FILENO) < 0) {
virReportSystemError(errno, "%s",
_("dup2(stderr) failed"));
goto cleanup;
}
VIR_FORCE_CLOSE(*ttyfd);
/* Any FDs in @passFDs need to be moved around so that
* they are numbered, without gaps, starting from
* STDERR_FILENO + 1
*/
for (i = 0; i < npassFDs; i++) {
int wantfd;
wantfd = STDERR_FILENO + i + 1;
VIR_DEBUG("Pass %d onto %d", passFDs[i], wantfd);
/* If we already have desired FD number, life
* is easy. Nothing needs renumbering */
if (passFDs[i] == wantfd)
continue;
/*
* Lets check to see if any later FDs are occupying
* our desired FD number. If so, we must move them
* out of the way
*/
for (j = i + 1; j < npassFDs; j++) {
if (passFDs[j] == wantfd) {
VIR_DEBUG("Clash %zu", j);
int newfd = dup(passFDs[j]);
if (newfd < 0) {
virReportSystemError(errno,
_("Cannot move fd %d out of the way"),
passFDs[j]);
goto cleanup;
}
/* We're intentionally not closing the
* old value of passFDs[j], because we
* don't want later iterations of the
* loop to take it back. dup2() will
* cause it to be closed shortly anyway
*/
VIR_DEBUG("Moved clash onto %d", newfd);
passFDs[j] = newfd;
}
}
/* Finally we can move into our desired FD number */
if (dup2(passFDs[i], wantfd) < 0) {
virReportSystemError(errno,
_("Cannot duplicate fd %d onto fd %d"),
passFDs[i], wantfd);
goto cleanup;
}
VIR_FORCE_CLOSE(passFDs[i]);
}
last_fd = STDERR_FILENO + npassFDs;
/* Just in case someone forget to set FD_CLOEXEC, explicitly /* Just in case someone forget to set FD_CLOEXEC, explicitly
* close all FDs before executing the container */ * close all remaining FDs before executing the container */
open_max = sysconf(_SC_OPEN_MAX); open_max = sysconf(_SC_OPEN_MAX);
if (open_max < 0) { if (open_max < 0) {
virReportSystemError(errno, "%s", virReportSystemError(errno, "%s",
_("sysconf(_SC_OPEN_MAX) failed")); _("sysconf(_SC_OPEN_MAX) failed"));
goto cleanup; goto cleanup;
} }
for (fd = 0; fd < open_max; fd++)
if (fd != ttyfd && fd != control && fd != handshakefd) {
int tmpfd = fd;
VIR_MASS_CLOSE(tmpfd);
}
if (dup2(ttyfd, 0) < 0) { for (fd = last_fd + 1; fd < open_max; fd++) {
virReportSystemError(errno, "%s", int tmpfd = fd;
_("dup2(stdin) failed")); VIR_MASS_CLOSE(tmpfd);
goto cleanup;
}
if (dup2(ttyfd, 1) < 0) {
virReportSystemError(errno, "%s",
_("dup2(stdout) failed"));
goto cleanup;
}
if (dup2(ttyfd, 2) < 0) {
virReportSystemError(errno, "%s",
_("dup2(stderr) failed"));
goto cleanup;
} }
rc = 0; rc = 0;
@ -1677,9 +1742,11 @@ static int lxcContainerChild(void *data)
if (virSecurityManagerSetProcessLabel(argv->securityDriver, vmDef) < 0) if (virSecurityManagerSetProcessLabel(argv->securityDriver, vmDef) < 0)
goto cleanup; goto cleanup;
if (lxcContainerSetStdio(argv->monitor, ttyfd, argv->handshakefd) < 0) { VIR_FORCE_CLOSE(argv->handshakefd);
VIR_FORCE_CLOSE(argv->monitor);
if (lxcContainerSetupFDs(&ttyfd,
argv->npassFDs, argv->passFDs) < 0)
goto cleanup; goto cleanup;
}
ret = 0; ret = 0;
cleanup: cleanup:
@ -1762,18 +1829,29 @@ int lxcContainerStart(virDomainDefPtr def,
virSecurityManagerPtr securityDriver, virSecurityManagerPtr securityDriver,
size_t nveths, size_t nveths,
char **veths, char **veths,
size_t npassFDs,
int *passFDs,
int control, int control,
int handshakefd, int handshakefd,
char **ttyPaths, size_t nttyPaths,
size_t nttyPaths) char **ttyPaths)
{ {
pid_t pid; pid_t pid;
int cflags; int cflags;
int stacksize = getpagesize() * 4; int stacksize = getpagesize() * 4;
char *stack, *stacktop; char *stack, *stacktop;
lxc_child_argv_t args = { def, securityDriver, lxc_child_argv_t args = {
nveths, veths, control, .config = def,
ttyPaths, nttyPaths, handshakefd}; .securityDriver = securityDriver,
.nveths = nveths,
.veths = veths,
.npassFDs = npassFDs,
.passFDs = passFDs,
.monitor = control,
.nttyPaths = nttyPaths,
.ttyPaths = ttyPaths,
.handshakefd = handshakefd
};
/* allocate a stack for the container */ /* allocate a stack for the container */
if (VIR_ALLOC_N(stack, stacksize) < 0) if (VIR_ALLOC_N(stack, stacksize) < 0)

View File

@ -56,10 +56,12 @@ int lxcContainerStart(virDomainDefPtr def,
virSecurityManagerPtr securityDriver, virSecurityManagerPtr securityDriver,
size_t nveths, size_t nveths,
char **veths, char **veths,
size_t npassFDs,
int *passFDs,
int control, int control,
int handshakefd, int handshakefd,
char **ttyPaths, size_t nttyPaths,
size_t nttyPaths); char **ttyPaths);
int lxcContainerAvailable(int features); int lxcContainerAvailable(int features);

View File

@ -108,6 +108,9 @@ struct _virLXCController {
size_t nveths; size_t nveths;
char **veths; char **veths;
size_t npassFDs;
int *passFDs;
size_t nconsoles; size_t nconsoles;
virLXCControllerConsolePtr consoles; virLXCControllerConsolePtr consoles;
char *devptmx; char *devptmx;
@ -253,6 +256,10 @@ static void virLXCControllerFree(virLXCControllerPtr ctrl)
VIR_FREE(ctrl->veths[i]); VIR_FREE(ctrl->veths[i]);
VIR_FREE(ctrl->veths); VIR_FREE(ctrl->veths);
for (i = 0; i < ctrl->npassFDs; i++)
VIR_FORCE_CLOSE(ctrl->passFDs[i]);
VIR_FREE(ctrl->passFDs);
for (i = 0; i < ctrl->nconsoles; i++) for (i = 0; i < ctrl->nconsoles; i++)
virLXCControllerConsoleClose(&(ctrl->consoles[i])); virLXCControllerConsoleClose(&(ctrl->consoles[i]));
VIR_FREE(ctrl->consoles); VIR_FREE(ctrl->consoles);
@ -2135,14 +2142,19 @@ virLXCControllerRun(virLXCControllerPtr ctrl)
ctrl->securityManager, ctrl->securityManager,
ctrl->nveths, ctrl->nveths,
ctrl->veths, ctrl->veths,
ctrl->npassFDs,
ctrl->passFDs,
control[1], control[1],
containerhandshake[1], containerhandshake[1],
containerTTYPaths, ctrl->nconsoles,
ctrl->nconsoles)) < 0) containerTTYPaths)) < 0)
goto cleanup; goto cleanup;
VIR_FORCE_CLOSE(control[1]); VIR_FORCE_CLOSE(control[1]);
VIR_FORCE_CLOSE(containerhandshake[1]); VIR_FORCE_CLOSE(containerhandshake[1]);
for (i = 0; i < ctrl->npassFDs; i++)
VIR_FORCE_CLOSE(ctrl->passFDs[i]);
if (virLXCControllerSetupUserns(ctrl) < 0) if (virLXCControllerSetupUserns(ctrl) < 0)
goto cleanup; goto cleanup;
@ -2209,6 +2221,7 @@ int main(int argc, char *argv[])
{ "name", 1, NULL, 'n' }, { "name", 1, NULL, 'n' },
{ "veth", 1, NULL, 'v' }, { "veth", 1, NULL, 'v' },
{ "console", 1, NULL, 'c' }, { "console", 1, NULL, 'c' },
{ "passfd", 1, NULL, 'p' },
{ "handshakefd", 1, NULL, 's' }, { "handshakefd", 1, NULL, 's' },
{ "security", 1, NULL, 'S' }, { "security", 1, NULL, 'S' },
{ "help", 0, NULL, 'h' }, { "help", 0, NULL, 'h' },
@ -2216,6 +2229,8 @@ int main(int argc, char *argv[])
}; };
int *ttyFDs = NULL; int *ttyFDs = NULL;
size_t nttyFDs = 0; size_t nttyFDs = 0;
int *passFDs = NULL;
size_t npassFDs = 0;
virLXCControllerPtr ctrl = NULL; virLXCControllerPtr ctrl = NULL;
size_t i; size_t i;
const char *securityDriver = "none"; const char *securityDriver = "none";
@ -2233,7 +2248,7 @@ int main(int argc, char *argv[])
while (1) { while (1) {
int c; int c;
c = getopt_long(argc, argv, "dn:v:m:c:s:h:S:", c = getopt_long(argc, argv, "dn:v:p:m:c:s:h:S:",
options, NULL); options, NULL);
if (c == -1) if (c == -1)
@ -2265,6 +2280,15 @@ int main(int argc, char *argv[])
} }
break; break;
case 'p':
if (VIR_REALLOC_N(passFDs, npassFDs + 1) < 0)
goto cleanup;
if (virStrToLong_i(optarg, NULL, 10, &passFDs[npassFDs++]) < 0) {
fprintf(stderr, "malformed --passfd argument '%s'", optarg);
goto cleanup;
}
break;
case 's': case 's':
if (virStrToLong_i(optarg, NULL, 10, &handshakeFd) < 0) { if (virStrToLong_i(optarg, NULL, 10, &handshakeFd) < 0) {
fprintf(stderr, "malformed --handshakefd argument '%s'", fprintf(stderr, "malformed --handshakefd argument '%s'",
@ -2337,6 +2361,9 @@ int main(int argc, char *argv[])
ctrl->veths = veths; ctrl->veths = veths;
ctrl->nveths = nveths; ctrl->nveths = nveths;
ctrl->passFDs = passFDs;
ctrl->npassFDs = npassFDs;
for (i = 0; i < nttyFDs; i++) { for (i = 0; i < nttyFDs; i++) {
if (virLXCControllerAddConsole(ctrl, ttyFDs[i]) < 0) if (virLXCControllerAddConsole(ctrl, ttyFDs[i]) < 0)
goto cleanup; goto cleanup;
@ -2402,6 +2429,9 @@ cleanup:
for (i = 0; i < nttyFDs; i++) for (i = 0; i < nttyFDs; i++)
VIR_FORCE_CLOSE(ttyFDs[i]); VIR_FORCE_CLOSE(ttyFDs[i]);
VIR_FREE(ttyFDs); VIR_FREE(ttyFDs);
for (i = 0; i < npassFDs; i++)
VIR_FORCE_CLOSE(passFDs[i]);
VIR_FREE(passFDs);
virLXCControllerFree(ctrl); virLXCControllerFree(ctrl);

View File

@ -1021,7 +1021,10 @@ cleanup:
* *
* Returns 0 on success or -1 in case of error * Returns 0 on success or -1 in case of error
*/ */
static int lxcDomainCreateWithFlags(virDomainPtr dom, unsigned int flags) static int lxcDomainCreateWithFiles(virDomainPtr dom,
unsigned int nfiles,
int *files,
unsigned int flags)
{ {
virLXCDriverPtr driver = dom->conn->privateData; virLXCDriverPtr driver = dom->conn->privateData;
virDomainObjPtr vm; virDomainObjPtr vm;
@ -1040,7 +1043,7 @@ static int lxcDomainCreateWithFlags(virDomainPtr dom, unsigned int flags)
goto cleanup; goto cleanup;
} }
if (virDomainCreateWithFlagsEnsureACL(dom->conn, vm->def) < 0) if (virDomainCreateWithFilesEnsureACL(dom->conn, vm->def) < 0)
goto cleanup; goto cleanup;
if ((vm->def->nets != NULL) && !(driver->have_netns)) { if ((vm->def->nets != NULL) && !(driver->have_netns)) {
@ -1056,6 +1059,7 @@ static int lxcDomainCreateWithFlags(virDomainPtr dom, unsigned int flags)
} }
ret = virLXCProcessStart(dom->conn, driver, vm, ret = virLXCProcessStart(dom->conn, driver, vm,
nfiles, files,
(flags & VIR_DOMAIN_START_AUTODESTROY), (flags & VIR_DOMAIN_START_AUTODESTROY),
VIR_DOMAIN_RUNNING_BOOTED); VIR_DOMAIN_RUNNING_BOOTED);
@ -1087,7 +1091,21 @@ cleanup:
*/ */
static int lxcDomainCreate(virDomainPtr dom) static int lxcDomainCreate(virDomainPtr dom)
{ {
return lxcDomainCreateWithFlags(dom, 0); return lxcDomainCreateWithFiles(dom, 0, NULL, 0);
}
/**
* lxcDomainCreateWithFlags:
* @dom: domain to start
*
* Looks up domain and starts it.
*
* Returns 0 on success or -1 in case of error
*/
static int lxcDomainCreateWithFlags(virDomainPtr dom,
unsigned int flags)
{
return lxcDomainCreateWithFiles(dom, 0, NULL, flags);
} }
/** /**
@ -1101,9 +1119,11 @@ static int lxcDomainCreate(virDomainPtr dom)
* Returns 0 on success or -1 in case of error * Returns 0 on success or -1 in case of error
*/ */
static virDomainPtr static virDomainPtr
lxcDomainCreateXML(virConnectPtr conn, lxcDomainCreateXMLWithFiles(virConnectPtr conn,
const char *xml, const char *xml,
unsigned int flags) { unsigned int nfiles,
int *files,
unsigned int flags) {
virLXCDriverPtr driver = conn->privateData; virLXCDriverPtr driver = conn->privateData;
virDomainObjPtr vm = NULL; virDomainObjPtr vm = NULL;
virDomainDefPtr def; virDomainDefPtr def;
@ -1118,7 +1138,7 @@ lxcDomainCreateXML(virConnectPtr conn,
VIR_DOMAIN_XML_INACTIVE))) VIR_DOMAIN_XML_INACTIVE)))
goto cleanup; goto cleanup;
if (virDomainCreateXMLEnsureACL(conn, def) < 0) if (virDomainCreateXMLWithFilesEnsureACL(conn, def) < 0)
goto cleanup; goto cleanup;
if (virSecurityManagerVerify(driver->securityManager, def) < 0) if (virSecurityManagerVerify(driver->securityManager, def) < 0)
@ -1139,6 +1159,7 @@ lxcDomainCreateXML(virConnectPtr conn,
def = NULL; def = NULL;
if (virLXCProcessStart(conn, driver, vm, if (virLXCProcessStart(conn, driver, vm,
nfiles, files,
(flags & VIR_DOMAIN_START_AUTODESTROY), (flags & VIR_DOMAIN_START_AUTODESTROY),
VIR_DOMAIN_RUNNING_BOOTED) < 0) { VIR_DOMAIN_RUNNING_BOOTED) < 0) {
virDomainAuditStart(vm, "booted", false); virDomainAuditStart(vm, "booted", false);
@ -1167,6 +1188,14 @@ cleanup:
} }
static virDomainPtr
lxcDomainCreateXML(virConnectPtr conn,
const char *xml,
unsigned int flags) {
return lxcDomainCreateXMLWithFiles(conn, xml, 0, NULL, flags);
}
static int lxcDomainGetSecurityLabel(virDomainPtr dom, virSecurityLabelPtr seclabel) static int lxcDomainGetSecurityLabel(virDomainPtr dom, virSecurityLabelPtr seclabel)
{ {
virLXCDriverPtr driver = dom->conn->privateData; virLXCDriverPtr driver = dom->conn->privateData;
@ -4849,6 +4878,7 @@ static virDriver lxcDriver = {
.connectNumOfDomains = lxcConnectNumOfDomains, /* 0.4.2 */ .connectNumOfDomains = lxcConnectNumOfDomains, /* 0.4.2 */
.connectListAllDomains = lxcConnectListAllDomains, /* 0.9.13 */ .connectListAllDomains = lxcConnectListAllDomains, /* 0.9.13 */
.domainCreateXML = lxcDomainCreateXML, /* 0.4.4 */ .domainCreateXML = lxcDomainCreateXML, /* 0.4.4 */
.domainCreateXMLWithFiles = lxcDomainCreateXMLWithFiles, /* 1.1.1 */
.domainLookupByID = lxcDomainLookupByID, /* 0.4.2 */ .domainLookupByID = lxcDomainLookupByID, /* 0.4.2 */
.domainLookupByUUID = lxcDomainLookupByUUID, /* 0.4.2 */ .domainLookupByUUID = lxcDomainLookupByUUID, /* 0.4.2 */
.domainLookupByName = lxcDomainLookupByName, /* 0.4.2 */ .domainLookupByName = lxcDomainLookupByName, /* 0.4.2 */
@ -4873,6 +4903,7 @@ static virDriver lxcDriver = {
.connectNumOfDefinedDomains = lxcConnectNumOfDefinedDomains, /* 0.4.2 */ .connectNumOfDefinedDomains = lxcConnectNumOfDefinedDomains, /* 0.4.2 */
.domainCreate = lxcDomainCreate, /* 0.4.4 */ .domainCreate = lxcDomainCreate, /* 0.4.4 */
.domainCreateWithFlags = lxcDomainCreateWithFlags, /* 0.8.2 */ .domainCreateWithFlags = lxcDomainCreateWithFlags, /* 0.8.2 */
.domainCreateWithFiles = lxcDomainCreateWithFiles, /* 1.1.1 */
.domainDefineXML = lxcDomainDefineXML, /* 0.4.2 */ .domainDefineXML = lxcDomainDefineXML, /* 0.4.2 */
.domainUndefine = lxcDomainUndefine, /* 0.4.2 */ .domainUndefine = lxcDomainUndefine, /* 0.4.2 */
.domainUndefineFlags = lxcDomainUndefineFlags, /* 0.9.4 */ .domainUndefineFlags = lxcDomainUndefineFlags, /* 0.9.4 */

View File

@ -192,7 +192,8 @@ virLXCProcessReboot(virLXCDriverPtr driver,
vm->newDef = NULL; vm->newDef = NULL;
virLXCProcessStop(driver, vm, VIR_DOMAIN_SHUTOFF_SHUTDOWN); virLXCProcessStop(driver, vm, VIR_DOMAIN_SHUTOFF_SHUTDOWN);
vm->newDef = savedDef; vm->newDef = savedDef;
if (virLXCProcessStart(conn, driver, vm, autodestroy, reason) < 0) { if (virLXCProcessStart(conn, driver, vm,
0, NULL, autodestroy, reason) < 0) {
VIR_WARN("Unable to handle reboot of vm %s", VIR_WARN("Unable to handle reboot of vm %s",
vm->def->name); vm->def->name);
goto cleanup; goto cleanup;
@ -803,6 +804,8 @@ virLXCProcessBuildControllerCmd(virLXCDriverPtr driver,
char **veths, char **veths,
int *ttyFDs, int *ttyFDs,
size_t nttyFDs, size_t nttyFDs,
int *files,
size_t nfiles,
int handshakefd) int handshakefd)
{ {
size_t i; size_t i;
@ -853,6 +856,12 @@ virLXCProcessBuildControllerCmd(virLXCDriverPtr driver,
virCommandPreserveFD(cmd, ttyFDs[i]); virCommandPreserveFD(cmd, ttyFDs[i]);
} }
for (i = 0; i < nfiles; i++) {
virCommandAddArg(cmd, "--passfd");
virCommandAddArgFormat(cmd, "%d", files[i]);
virCommandPreserveFD(cmd, files[i]);
}
virCommandAddArgPair(cmd, "--security", virCommandAddArgPair(cmd, "--security",
virSecurityManagerGetModel(driver->securityManager)); virSecurityManagerGetModel(driver->securityManager));
@ -1024,6 +1033,7 @@ error:
int virLXCProcessStart(virConnectPtr conn, int virLXCProcessStart(virConnectPtr conn,
virLXCDriverPtr driver, virLXCDriverPtr driver,
virDomainObjPtr vm, virDomainObjPtr vm,
unsigned int nfiles, int *files,
bool autoDestroy, bool autoDestroy,
virDomainRunningReason reason) virDomainRunningReason reason)
{ {
@ -1189,6 +1199,7 @@ int virLXCProcessStart(virConnectPtr conn,
vm, vm,
nveths, veths, nveths, veths,
ttyFDs, nttyFDs, ttyFDs, nttyFDs,
files, nfiles,
handshakefds[1]))) handshakefds[1])))
goto cleanup; goto cleanup;
virCommandSetOutputFD(cmd, &logfd); virCommandSetOutputFD(cmd, &logfd);
@ -1382,7 +1393,8 @@ virLXCProcessAutostartDomain(virDomainObjPtr vm,
virObjectLock(vm); virObjectLock(vm);
if (vm->autostart && if (vm->autostart &&
!virDomainObjIsActive(vm)) { !virDomainObjIsActive(vm)) {
ret = virLXCProcessStart(data->conn, data->driver, vm, false, ret = virLXCProcessStart(data->conn, data->driver, vm,
0, NULL, false,
VIR_DOMAIN_RUNNING_BOOTED); VIR_DOMAIN_RUNNING_BOOTED);
virDomainAuditStart(vm, "booted", ret >= 0); virDomainAuditStart(vm, "booted", ret >= 0);
if (ret < 0) { if (ret < 0) {

View File

@ -27,6 +27,7 @@
int virLXCProcessStart(virConnectPtr conn, int virLXCProcessStart(virConnectPtr conn,
virLXCDriverPtr driver, virLXCDriverPtr driver,
virDomainObjPtr vm, virDomainObjPtr vm,
unsigned int nfiles, int *files,
bool autoDestroy, bool autoDestroy,
virDomainRunningReason reason); virDomainRunningReason reason);
int virLXCProcessStop(virLXCDriverPtr driver, int virLXCProcessStop(virLXCDriverPtr driver,