xen_inotify.c 15.3 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
/*
 * xen_inofify.c: Xen notification of xml file activity in the
 *                following dirs:
 *                /etc/xen
 *                /var/lib/xend/domains
 *
 * Copyright (C) 2008 VirtualIron
 *
 * 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
 *
 * Author: Ben Guthro
 */
#include <config.h>
#include <dirent.h>
#include <sys/inotify.h>

#include "virterror_internal.h"
#include "datatypes.h"
#include "driver.h"
#include "memory.h"
#include "event.h"
34
#include "xen_driver.h"
35 36 37 38 39 40 41 42 43
#include "conf.h"
#include "domain_conf.h"
#include "xen_inotify.h"
#include "xend_internal.h"
#include "logging.h"
#include "uuid.h"

#include "xm_internal.h" /* for xenXMDomainConfigParse */

44 45
#define VIR_FROM_THIS VIR_FROM_XEN_INOTIFY

46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
#define virXenInotifyError(conn, code, fmt...)                                 \
        virReportErrorHelper(NULL, VIR_FROM_XEN_INOTIFY, code, __FILE__,      \
                               __FUNCTION__, __LINE__, fmt)

struct xenUnifiedDriver xenInotifyDriver = {
    xenInotifyOpen, /* open */
    xenInotifyClose, /* close */
    NULL, /* version */
    NULL, /* hostname */
    NULL, /* nodeGetInfo */
    NULL, /* getCapabilities */
    NULL, /* listDomains */
    NULL, /* numOfDomains */
    NULL, /* domainCreateLinux */
    NULL, /* domainSuspend */
    NULL, /* domainResume */
    NULL, /* domainShutdown */
    NULL, /* domainReboot */
    NULL, /* domainDestroy */
    NULL, /* domainGetOSType */
    NULL, /* domainGetMaxMemory */
    NULL, /* domainSetMaxMemory */
    NULL, /* domainSetMemory */
    NULL, /* domainGetInfo */
    NULL, /* domainSave */
    NULL, /* domainRestore */
    NULL, /* domainCoreDump */
    NULL, /* domainSetVcpus */
    NULL, /* domainPinVcpu */
    NULL, /* domainGetVcpus */
    NULL, /* domainGetMaxVcpus */
    NULL, /* listDefinedDomains */
    NULL, /* numOfDefinedDomains */
    NULL, /* domainCreate */
    NULL, /* domainDefineXML */
    NULL, /* domainUndefine */
82 83
    NULL, /* domainAttachDeviceFlags */
    NULL, /* domainDetachDeviceFlags */
84 85 86 87 88 89 90
    NULL, /* domainGetAutostart */
    NULL, /* domainSetAutostart */
    NULL, /* domainGetSchedulerType */
    NULL, /* domainGetSchedulerParameters */
    NULL, /* domainSetSchedulerParameters */
};

91
static int
92 93
xenInotifyXenCacheLookup(virConnectPtr conn,
                         const char *filename,
94
                         char **name, unsigned char *uuid) {
95
    xenUnifiedPrivatePtr priv = conn->privateData;
96 97
    xenXMConfCachePtr entry;

98
    if (!(entry = virHashLookup(priv->configCache, filename))) {
99
        DEBUG("No config found for %s", filename);
100
        return -1;
101 102
    }

103 104 105 106
    *name = strdup(entry->def->name);
    memcpy(uuid, entry->def->uuid, VIR_UUID_BUFLEN);

    if (!*name) {
107
        DEBUG0("Error getting dom from def");
108
        virReportOOMError();
109
        return -1;
110
    }
111
    return 0;
112 113
}

114 115 116
static int
xenInotifyXendDomainsDirLookup(virConnectPtr conn, const char *filename,
                               char **name, unsigned char *uuid) {
117 118 119
    int i;
    virDomainPtr dom;
    const char *uuid_str;
120
    unsigned char rawuuid[VIR_UUID_BUFLEN];
121
    xenUnifiedPrivatePtr priv = conn->privateData;
122 123 124 125 126

    /* xend is managing domains. we will get
    * a filename in the manner:
    * /var/lib/xend/domains/<uuid>/
    */
127
    uuid_str = filename + strlen(XEND_DOMAINS_DIR) + 1;
128

129
    if (virUUIDParse(uuid_str, rawuuid) < 0) {
130
        virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
131
                           _("parsing uuid %s"), uuid_str);
132
        return -1;
133 134 135 136 137
    }
    /* call directly into xend here, as driver may not yet
       be set during open while we are building our
       initial list of domains */
    DEBUG("Looking for dom with uuid: %s", uuid_str);
138 139
    /* XXX Should not have to go via a virDomainPtr obj instance */
    if(!(dom = xenDaemonLookupByUUID(conn, rawuuid))) {
140 141 142
        /* If we are here, the domain has gone away.
           search for, and create a domain from the stored
           list info */
143
        for (i = 0 ; i < priv->configInfoList->count ; i++) {
144
            if (!memcmp(rawuuid, priv->configInfoList->doms[i]->uuid, VIR_UUID_BUFLEN)) {
145
                *name = strdup(priv->configInfoList->doms[i]->name);
146
                if (!*name) {
147
                    virReportOOMError();
148
                    return -1;
149
                }
150
                memcpy(uuid, priv->configInfoList->doms[i]->uuid, VIR_UUID_BUFLEN);
151
                DEBUG0("Found dom on list");
152
                return 0;
153 154 155
            }
        }
        virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
156 157
                           "%s", _("finding dom on config list"));
        return -1;
158 159
    }

160
    if (!(*name = strdup(dom->name))) {
161
        virReportOOMError();
162
        return -1;
163
    }
164 165
    memcpy(uuid, dom->uuid, VIR_UUID_BUFLEN);
    virDomainFree(dom);
166
    /* succeeded too find domain by uuid */
167
    return 0;
168 169
}

170 171 172 173
static int
xenInotifyDomainLookup(virConnectPtr conn,
                       const char *filename,
                       char **name, unsigned char *uuid) {
174 175
    xenUnifiedPrivatePtr priv = conn->privateData;
    if (priv->useXenConfigCache)
176
        return xenInotifyXenCacheLookup(conn, filename, name, uuid);
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194
    else
        return xenInotifyXendDomainsDirLookup(conn, filename, name, uuid);
}

static virDomainEventPtr
xenInotifyDomainEventFromFile(virConnectPtr conn,
                              const char *filename,
                              int type, int detail) {
    virDomainEventPtr event;
    char *name = NULL;
    unsigned char uuid[VIR_UUID_BUFLEN];

    if (xenInotifyDomainLookup(conn, filename, &name, uuid) < 0)
        return NULL;

    event = virDomainEventNew(-1, name, uuid, type, detail);
    VIR_FREE(name);
    return event;
195 196 197
}

static int
198
xenInotifyXendDomainsDirRemoveEntry(virConnectPtr conn,
199
                                    const char *fname) {
200
    xenUnifiedPrivatePtr priv = conn->privateData;
201
    const char *uuidstr = fname + strlen(XEND_DOMAINS_DIR) + 1;
202 203 204 205 206
    unsigned char uuid[VIR_UUID_BUFLEN];
    int i;

    if (virUUIDParse(uuidstr, uuid) < 0) {
        virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
207
                           _("parsing uuid %s"), uuidstr);
208 209 210 211
        return -1;
    }

    /* match and remove on uuid */
212 213 214 215 216 217 218 219 220 221 222 223 224
    for (i = 0 ; i < priv->configInfoList->count ; i++) {
        if (!memcmp(uuid, priv->configInfoList->doms[i]->uuid, VIR_UUID_BUFLEN)) {
            VIR_FREE(priv->configInfoList->doms[i]->name);
            VIR_FREE(priv->configInfoList->doms[i]);

            if (i < (priv->configInfoList->count - 1))
                memmove(priv->configInfoList->doms + i,
                        priv->configInfoList->doms + i + 1,
                        sizeof(*(priv->configInfoList->doms)) *
                                (priv->configInfoList->count - (i + 1)));

            if (VIR_REALLOC_N(priv->configInfoList->doms,
                              priv->configInfoList->count - 1) < 0) {
225 226
                ; /* Failure to reduce memory allocation isn't fatal */
            }
227
            priv->configInfoList->count--;
228 229 230 231 232 233 234 235 236
            return 0;
        }
    }
    return -1;
}

static int
xenInotifyXendDomainsDirAddEntry(virConnectPtr conn,
                                 const char *fname) {
237 238
    char *name = NULL;
    unsigned char uuid[VIR_UUID_BUFLEN];
239 240
    xenUnifiedPrivatePtr priv = conn->privateData;

241
    if (xenInotifyDomainLookup(conn, fname, &name, uuid) < 0) {
242
        virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
243
                           "%s", _("Error looking up domain"));
244 245 246
        return -1;
    }

247
    if (xenUnifiedAddDomainInfo(priv->configInfoList,
248
                                -1, name, uuid) < 0) {
249 250
        virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
                        "%s", _("Error adding file to config cache"));
251
        VIR_FREE(name);
252 253
        return -1;
    }
254
    VIR_FREE(name);
255 256 257 258 259 260
    return 0;
}

static int
xenInotifyRemoveDomainConfigInfo(virConnectPtr conn,
                                 const char *fname) {
261 262 263 264
    xenUnifiedPrivatePtr priv = conn->privateData;
    return priv->useXenConfigCache ?
        xenXMConfigCacheRemoveFile(conn, fname) :
        xenInotifyXendDomainsDirRemoveEntry(conn, fname);
265 266 267 268 269
}

static int
xenInotifyAddDomainConfigInfo(virConnectPtr conn,
                              const char *fname) {
270 271 272 273
    xenUnifiedPrivatePtr priv = conn->privateData;
    return priv->useXenConfigCache ?
        xenXMConfigCacheAddFile(conn, fname) :
        xenInotifyXendDomainsDirAddEntry(conn, fname);
274 275 276 277 278 279 280 281 282 283 284 285 286
}

static void
xenInotifyEvent(int watch ATTRIBUTE_UNUSED,
                int fd,
                int events ATTRIBUTE_UNUSED,
                void *data)
{
    char buf[1024];
    char fname[1024];
    struct inotify_event *e;
    int got;
    char *tmp, *name;
287
    virConnectPtr conn = data;
288 289 290 291 292 293 294 295 296 297 298 299
    xenUnifiedPrivatePtr priv = NULL;

    DEBUG0("got inotify event");

    if( conn && conn->privateData ) {
        priv = conn->privateData;
    } else {
        virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
                           "%s", _("conn, or private data is NULL"));
        return;
    }

D
Daniel P. Berrange 已提交
300 301
    xenUnifiedLock(priv);

302 303 304 305 306
reread:
    got = read(fd, buf, sizeof(buf));
    if (got == -1) {
        if (errno == EINTR)
            goto reread;
D
Daniel P. Berrange 已提交
307
        goto cleanup;
308 309 310 311 312
    }

    tmp = buf;
    while (got) {
        if (got < sizeof(struct inotify_event))
D
Daniel P. Berrange 已提交
313
            goto cleanup; /* bad */
314 315 316 317 318 319

        e = (struct inotify_event *)tmp;
        tmp += sizeof(struct inotify_event);
        got -= sizeof(struct inotify_event);

        if (got < e->len)
D
Daniel P. Berrange 已提交
320
            goto cleanup;
321 322 323 324 325 326

        tmp += e->len;
        got -= e->len;

        name = (char *)&(e->name);

327 328
        snprintf(fname, 1024, "%s/%s",
                 priv->configDir, name);
329 330

        if (e->mask & (IN_DELETE | IN_MOVED_FROM)) {
331 332 333 334
            virDomainEventPtr event =
                xenInotifyDomainEventFromFile(conn, fname,
                                              VIR_DOMAIN_EVENT_UNDEFINED,
                                              VIR_DOMAIN_EVENT_UNDEFINED_REMOVED);
335
            if (event)
336 337
                xenUnifiedDomainEventDispatch(conn->privateData, event);
            else
338
                virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
339
                                   "%s", _("looking up dom"));
340 341 342 343

            if (xenInotifyRemoveDomainConfigInfo(conn, fname) < 0 ) {
                virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
                                   "%s", _("Error adding file to config cache"));
D
Daniel P. Berrange 已提交
344
                goto cleanup;
345 346
            }
        } else if (e->mask & ( IN_CREATE | IN_CLOSE_WRITE | IN_MOVED_TO) ) {
347
            virDomainEventPtr event;
348 349 350
            if (xenInotifyAddDomainConfigInfo(conn, fname) < 0 ) {
                virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
                                   "%s", _("Error adding file to config cache"));
D
Daniel P. Berrange 已提交
351
                goto cleanup;
352 353
            }

354 355 356 357 358 359 360
            event = xenInotifyDomainEventFromFile(conn, fname,
                                                  VIR_DOMAIN_EVENT_DEFINED,
                                                  VIR_DOMAIN_EVENT_DEFINED_ADDED);

            if (event)
                xenUnifiedDomainEventDispatch(conn->privateData, event);
            else
361
                virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
362
                                   "%s", _("looking up dom"));
363 364 365 366

        }

    }
D
Daniel P. Berrange 已提交
367 368 369

cleanup:
    xenUnifiedUnlock(priv);
370 371 372 373 374 375 376 377 378 379 380 381
}

/**
 * xenInotifyOpen:
 * @conn: pointer to the connection block
 * @name: URL for the target, NULL for local
 * @flags: combination of virDrvOpenFlag(s)
 *
 * Connects and starts listening for inotify events
 *
 * Returns 0 or -1 in case of error.
 */
382
virDrvOpenStatus
383
xenInotifyOpen(virConnectPtr conn,
384 385 386 387 388 389 390 391
             virConnectAuthPtr auth ATTRIBUTE_UNUSED,
             int flags ATTRIBUTE_UNUSED)
{
    DIR *dh;
    struct dirent *ent;
    char path[PATH_MAX];
    xenUnifiedPrivatePtr priv = (xenUnifiedPrivatePtr) conn->privateData;

392
    if (priv->configDir) {
393
        priv->useXenConfigCache = 1;
394 395
    } else {
        /* /var/lib/xend/domains/<uuid>/config.sxp */
396
        priv->configDir = XEND_DOMAINS_DIR;
397
        priv->useXenConfigCache = 0;
398

399
        if (VIR_ALLOC(priv->configInfoList) < 0) {
400
            virReportOOMError();
401 402 403 404
            return -1;
        }

        /* populate initial list */
405
        if (!(dh = opendir(priv->configDir))) {
406
            virReportSystemError(errno,
407
                                 _("cannot open directory: %s"),
408
                                 priv->configDir);
409 410 411 412 413 414 415
            return -1;
        }
        while ((ent = readdir(dh))) {
            if (STRPREFIX(ent->d_name, "."))
                continue;

            /* Build the full file path */
416 417
            if ((strlen(priv->configDir) + 1 +
                 strlen(ent->d_name) + 1) > PATH_MAX)
418
                continue;
419
            strcpy(path, priv->configDir);
420 421 422 423 424 425
            strcat(path, "/");
            strcat(path, ent->d_name);

            if (xenInotifyAddDomainConfigInfo(conn, path) < 0 ) {
                virXenInotifyError(NULL, VIR_ERR_INTERNAL_ERROR,
                                   "%s", _("Error adding file to config list"));
426
                closedir(dh);
427 428 429
                return -1;
            }
        }
430
        closedir(dh);
431 432 433
    }

    if ((priv->inotifyFD = inotify_init()) < 0) {
434
        virReportSystemError(errno,
435
                             "%s", _("initializing inotify"));
436 437 438
        return -1;
    }

439
    DEBUG("Adding a watch on %s", priv->configDir);
440
    if (inotify_add_watch(priv->inotifyFD,
441
                          priv->configDir,
442 443 444
                          IN_CREATE |
                          IN_CLOSE_WRITE | IN_DELETE |
                          IN_MOVED_TO | IN_MOVED_FROM) < 0) {
445
        virReportSystemError(errno,
446 447
                             _("adding watch on %s"),
                             priv->configDir);
448 449 450 451
        return -1;
    }

    DEBUG0("Building initial config cache");
452
    if (priv->useXenConfigCache &&
453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478
        xenXMConfigCacheRefresh (conn) < 0) {
        DEBUG("Failed to enable XM config cache %s", conn->err.message);
        return -1;
    }

    DEBUG0("Registering with event loop");
    /* Add the handle for monitoring */
    if ((priv->inotifyWatch = virEventAddHandle(priv->inotifyFD, VIR_EVENT_HANDLE_READABLE,
                                                xenInotifyEvent, conn, NULL)) < 0) {
        DEBUG0("Failed to add inotify handle, disabling events");
    }

    return 0;
}

/**
 * xenInotifyClose:
 * @conn: pointer to the connection block
 *
 * Close and stop listening for inotify events
 *
 * Returns 0 in case of success or -1 in case of error.
 */
int
xenInotifyClose(virConnectPtr conn)
{
479
    xenUnifiedPrivatePtr priv = conn->privateData;
480

481 482
    if (priv->configInfoList)
        xenUnifiedDomainInfoListFree(priv->configInfoList);
483 484 485 486 487 488 489

    if (priv->inotifyWatch != -1)
        virEventRemoveHandle(priv->inotifyWatch);
    close(priv->inotifyFD);

    return 0;
}