/*
* Copyright (C) 2010-2012 Red Hat, Inc.
*
* lxc_monitor.c: client for LXC controller monitor
*
* 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 "lxc_monitor.h"
#include "lxc_conf.h"
#include "memory.h"
#include "virterror_internal.h"
#include "logging.h"
#include "threads.h"
#include "rpc/virnetclient.h"
#define VIR_FROM_THIS VIR_FROM_LXC
struct _virLXCMonitor {
int refs;
virMutex lock; /* also used to protect refs */
virDomainObjPtr vm;
virLXCMonitorCallbacksPtr cb;
virNetClientPtr client;
};
static void virLXCMonitorFree(virLXCMonitorPtr mon);
static void virLXCMonitorEOFNotify(virNetClientPtr client ATTRIBUTE_UNUSED,
int reason ATTRIBUTE_UNUSED,
void *opaque)
{
virLXCMonitorPtr mon = opaque;
virLXCMonitorCallbackEOFNotify eofNotify;
virDomainObjPtr vm;
virLXCMonitorLock(mon);
eofNotify = mon->cb->eofNotify;
vm = mon->vm;
virLXCMonitorUnlock(mon);
eofNotify(mon, vm);
}
static void virLXCMonitorCloseFreeCallback(void *opaque)
{
virLXCMonitorPtr mon = opaque;
virLXCMonitorLock(mon);
if (virLXCMonitorUnref(mon) > 0)
virLXCMonitorUnlock(mon);
}
virLXCMonitorPtr virLXCMonitorNew(virDomainObjPtr vm,
const char *socketdir,
virLXCMonitorCallbacksPtr cb)
{
virLXCMonitorPtr mon;
char *sockpath = NULL;
if (VIR_ALLOC(mon) < 0) {
virReportOOMError();
return NULL;
}
mon->refs = 1;
if (virMutexInit(&mon->lock) < 0) {
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
_("cannot initialize monitor mutex"));
VIR_FREE(mon);
return NULL;
}
if (virAsprintf(&sockpath, "%s/%s.sock",
socketdir, vm->def->name) < 0)
goto no_memory;
if (!(mon->client = virNetClientNewUNIX(sockpath, false, NULL)))
goto error;
mon->vm = vm;
mon->cb = cb;
virLXCMonitorRef(mon);
virNetClientSetCloseCallback(mon->client, virLXCMonitorEOFNotify, mon,
virLXCMonitorCloseFreeCallback);
cleanup:
VIR_FREE(sockpath);
return mon;
no_memory:
virReportOOMError();
error:
virLXCMonitorFree(mon);
mon = NULL;
goto cleanup;
}
static void virLXCMonitorFree(virLXCMonitorPtr mon)
{
VIR_DEBUG("mon=%p", mon);
if (mon->client)
virLXCMonitorClose(mon);
if (mon->cb && mon->cb->destroy)
(mon->cb->destroy)(mon, mon->vm);
virMutexDestroy(&mon->lock);
VIR_FREE(mon);
}
int virLXCMonitorRef(virLXCMonitorPtr mon)
{
mon->refs++;
return mon->refs;
}
int virLXCMonitorUnref(virLXCMonitorPtr mon)
{
mon->refs--;
if (mon->refs == 0) {
virLXCMonitorUnlock(mon);
virLXCMonitorFree(mon);
return 0;
}
return mon->refs;
}
void virLXCMonitorClose(virLXCMonitorPtr mon)
{
if (mon->client) {
virNetClientClose(mon->client);
virNetClientFree(mon->client);
mon->client = NULL;
}
}
void virLXCMonitorLock(virLXCMonitorPtr mon)
{
virMutexLock(&mon->lock);
}
void virLXCMonitorUnlock(virLXCMonitorPtr mon)
{
virMutexUnlock(&mon->lock);
}