381 lines
7.3 KiB
C
381 lines
7.3 KiB
C
|
/*
|
||
|
Copyright Red Hat, Inc. 2010
|
||
|
|
||
|
This program is free software; you can redistribute it and/or modify it
|
||
|
under the terms of the GNU General Public License as published by the
|
||
|
Free Software Foundation; either version 2, or (at your option) any
|
||
|
later version.
|
||
|
|
||
|
This program 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
|
||
|
General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License
|
||
|
along with this program; see the file COPYING. If not, write to the
|
||
|
Free Software Foundation, Inc., 675 Mass Ave, Cambridge,
|
||
|
MA 02139, USA.
|
||
|
*/
|
||
|
/*
|
||
|
* Author: Lon Hohberger <lhh at redhat.com>
|
||
|
*/
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <unistd.h>
|
||
|
#include <signal.h>
|
||
|
#include <string.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/wait.h>
|
||
|
#include <sys/un.h>
|
||
|
#include <sys/socket.h>
|
||
|
#include <sys/select.h>
|
||
|
#include <sys/ioctl.h>
|
||
|
#include <arpa/inet.h>
|
||
|
#include <net/if.h>
|
||
|
#include <netinet/in.h>
|
||
|
#include <netdb.h>
|
||
|
#include <sys/time.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <errno.h>
|
||
|
#include <pthread.h>
|
||
|
#include <nss.h>
|
||
|
#include <libgen.h>
|
||
|
#include <list.h>
|
||
|
#include <simpleconfig.h>
|
||
|
#include <server_plugin.h>
|
||
|
#include <history.h>
|
||
|
#include <xvm.h>
|
||
|
|
||
|
|
||
|
/* Local includes */
|
||
|
#include "debug.h"
|
||
|
#include "fdops.h"
|
||
|
#include "virt-sockets.h"
|
||
|
|
||
|
#define NAME "serial"
|
||
|
#define VERSION "0.3"
|
||
|
|
||
|
#define SERIAL_PLUG_MAGIC 0x1227a000
|
||
|
|
||
|
#define VALIDATE(info) \
|
||
|
do {\
|
||
|
if (!info || info->magic != SERIAL_PLUG_MAGIC)\
|
||
|
return -EINVAL;\
|
||
|
} while(0)
|
||
|
|
||
|
|
||
|
typedef struct _serial_info {
|
||
|
uint64_t magic;
|
||
|
const fence_callbacks_t *cb;
|
||
|
void *priv;
|
||
|
char *uri;
|
||
|
history_info_t *history;
|
||
|
} serial_info;
|
||
|
|
||
|
|
||
|
struct serial_hostlist_arg {
|
||
|
int fd;
|
||
|
};
|
||
|
|
||
|
|
||
|
/*
|
||
|
* See if we fenced this node recently (successfully)
|
||
|
* If so, ignore the request for a few seconds.
|
||
|
*
|
||
|
* We purge our history when the entries time out.
|
||
|
*/
|
||
|
static int
|
||
|
check_history(void *a, void *b) {
|
||
|
fence_req_t *old = a, *current = b;
|
||
|
|
||
|
if (old->request == current->request &&
|
||
|
old->seqno == current->seqno &&
|
||
|
!strcasecmp((const char *)old->domain,
|
||
|
(const char *)current->domain)) {
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_hostlist(const char *vm_name, const char *vm_uuid,
|
||
|
int state, void *priv)
|
||
|
{
|
||
|
struct serial_hostlist_arg *arg = (struct serial_hostlist_arg *)priv;
|
||
|
host_state_t hinfo;
|
||
|
struct timeval tv;
|
||
|
int ret;
|
||
|
|
||
|
strncpy((char *)hinfo.domain, vm_name, sizeof(hinfo.domain));
|
||
|
strncpy((char *)hinfo.uuid, vm_uuid, sizeof(hinfo.uuid));
|
||
|
hinfo.state = state;
|
||
|
|
||
|
tv.tv_sec = 1;
|
||
|
tv.tv_usec = 0;
|
||
|
|
||
|
printf("%d\n", arg->fd);
|
||
|
ret = _write_retry(arg->fd, &hinfo, sizeof(hinfo), &tv);
|
||
|
if (ret == sizeof(hinfo))
|
||
|
return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_hostlist_begin(int fd)
|
||
|
{
|
||
|
struct timeval tv;
|
||
|
serial_resp_t resp;
|
||
|
|
||
|
resp.magic = SERIAL_MAGIC;
|
||
|
resp.response = 253;
|
||
|
|
||
|
tv.tv_sec = 1;
|
||
|
tv.tv_usec = 0;
|
||
|
return _write_retry(fd, &resp, sizeof(resp), &tv);
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_hostlist_end(int fd)
|
||
|
{
|
||
|
host_state_t hinfo;
|
||
|
struct timeval tv;
|
||
|
int ret;
|
||
|
|
||
|
printf("Sending terminator packet\n");
|
||
|
|
||
|
memset(&hinfo, 0, sizeof(hinfo));
|
||
|
|
||
|
tv.tv_sec = 1;
|
||
|
tv.tv_usec = 0;
|
||
|
ret = _write_retry(fd, &hinfo, sizeof(hinfo), &tv);
|
||
|
if (ret == sizeof(hinfo))
|
||
|
return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
do_fence_request(int fd, serial_req_t *req, serial_info *info)
|
||
|
{
|
||
|
char response = 1;
|
||
|
struct serial_hostlist_arg arg;
|
||
|
serial_resp_t resp;
|
||
|
|
||
|
arg.fd = fd;
|
||
|
|
||
|
switch(req->request) {
|
||
|
case FENCE_NULL:
|
||
|
response = info->cb->null((char *)req->domain, info->priv);
|
||
|
break;
|
||
|
case FENCE_ON:
|
||
|
response = info->cb->on((char *)req->domain, req->seqno,
|
||
|
info->priv);
|
||
|
break;
|
||
|
case FENCE_OFF:
|
||
|
response = info->cb->off((char *)req->domain, req->seqno,
|
||
|
info->priv);
|
||
|
break;
|
||
|
case FENCE_REBOOT:
|
||
|
response = info->cb->reboot((char *)req->domain, req->seqno,
|
||
|
info->priv);
|
||
|
break;
|
||
|
case FENCE_STATUS:
|
||
|
response = info->cb->status((char *)req->domain, info->priv);
|
||
|
break;
|
||
|
case FENCE_DEVSTATUS:
|
||
|
response = info->cb->devstatus(info->priv);
|
||
|
break;
|
||
|
case FENCE_HOSTLIST:
|
||
|
serial_hostlist_begin(arg.fd);
|
||
|
response = info->cb->hostlist(serial_hostlist, &arg,
|
||
|
info->priv);
|
||
|
serial_hostlist_end(arg.fd);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
resp.magic = SERIAL_MAGIC;
|
||
|
resp.response = response;
|
||
|
|
||
|
dbg_printf(3, "Sending response to caller...\n");
|
||
|
if (write(fd, &resp, sizeof(resp)) < 0) {
|
||
|
perror("write");
|
||
|
}
|
||
|
|
||
|
/* XVM shotguns multicast packets, so we want to avoid
|
||
|
* acting on the same request multiple times if the first
|
||
|
* attempt was successful.
|
||
|
*/
|
||
|
history_record(info->history, req);
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_dispatch(listener_context_t c, struct timeval *timeout)
|
||
|
{
|
||
|
serial_info *info;
|
||
|
serial_req_t data;
|
||
|
fd_set rfds;
|
||
|
struct timeval tv;
|
||
|
int max;
|
||
|
int n, x, ret;
|
||
|
|
||
|
info = (serial_info *)c;
|
||
|
VALIDATE(info);
|
||
|
|
||
|
FD_ZERO(&rfds);
|
||
|
domain_sock_fdset(&rfds, &max);
|
||
|
|
||
|
n = select(max+1, &rfds, NULL, NULL, timeout);
|
||
|
printf("n = %d max = %d\n", n, max);
|
||
|
if (n < 0) {
|
||
|
perror("select");
|
||
|
return n;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* If no requests, we're done
|
||
|
*/
|
||
|
if (n == 0)
|
||
|
return 0;
|
||
|
|
||
|
/* find & read request */
|
||
|
for (x = 0; x <= max; x++) {
|
||
|
if (FD_ISSET(x, &rfds)) {
|
||
|
tv.tv_sec = 1;
|
||
|
tv.tv_usec = 0;
|
||
|
ret = _read_retry(x, &data, sizeof(data), &tv);
|
||
|
printf("the read...%d\n",ret);
|
||
|
if (ret != sizeof(data)) {
|
||
|
if (--n)
|
||
|
continue;
|
||
|
else
|
||
|
return 0;
|
||
|
} else {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
printf("Sock %d Request %d seqno %d domain %s\n", x, data.request, data.seqno,
|
||
|
data.domain);
|
||
|
|
||
|
if (history_check(info->history, &data) == 1) {
|
||
|
printf("We just did this request; dropping packet\n");
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
do_fence_request(x, &data, info);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_config(config_object_t *config, serial_info *args)
|
||
|
{
|
||
|
char value[1024];
|
||
|
int errors = 0;
|
||
|
|
||
|
#ifdef _MODULE
|
||
|
if (sc_get(config, "fence_virtd/@debug", value, sizeof(value))==0)
|
||
|
dset(atoi(value));
|
||
|
#endif
|
||
|
|
||
|
if (sc_get(config, "listeners/serial/@uri",
|
||
|
value, sizeof(value)-1) == 0) {
|
||
|
dbg_printf(1, "Got %s for uri\n", value);
|
||
|
args->uri = strdup(value);
|
||
|
}
|
||
|
|
||
|
return errors;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_init(listener_context_t *c, const fence_callbacks_t *cb,
|
||
|
config_object_t *config, void *priv)
|
||
|
{
|
||
|
serial_info *info;
|
||
|
int ret;
|
||
|
|
||
|
info = malloc(sizeof(*info));
|
||
|
if (!info)
|
||
|
return -1;
|
||
|
memset(info, 0, sizeof(*info));
|
||
|
|
||
|
info->priv = priv;
|
||
|
info->cb = cb;
|
||
|
|
||
|
ret = serial_config(config, info);
|
||
|
if (ret < 0) {
|
||
|
perror("serial_config");
|
||
|
return -1;
|
||
|
} else if (ret > 0) {
|
||
|
printf("%d errors found during configuration\n",ret);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
info->magic = SERIAL_PLUG_MAGIC;
|
||
|
info->history = history_init(check_history, 10, sizeof(fence_req_t));
|
||
|
*c = (listener_context_t)info;
|
||
|
start_event_listener(info->uri);
|
||
|
sleep(1);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
serial_shutdown(listener_context_t c)
|
||
|
{
|
||
|
serial_info *info = (serial_info *)c;
|
||
|
|
||
|
dbg_printf(3, "Shutting down serial\n");
|
||
|
|
||
|
VALIDATE(info);
|
||
|
info->magic = 0;
|
||
|
stop_event_listener();
|
||
|
free(info->uri);
|
||
|
free(info);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static listener_plugin_t serial_plugin = {
|
||
|
.name = NAME,
|
||
|
.version = VERSION,
|
||
|
.init = serial_init,
|
||
|
.dispatch = serial_dispatch,
|
||
|
.cleanup = serial_shutdown,
|
||
|
};
|
||
|
|
||
|
|
||
|
#ifdef _MODULE
|
||
|
double
|
||
|
LISTENER_VER_SYM(void)
|
||
|
{
|
||
|
return PLUGIN_VERSION_LISTENER;
|
||
|
}
|
||
|
|
||
|
const listener_plugin_t *
|
||
|
LISTENER_INFO_SYM(void)
|
||
|
{
|
||
|
return &serial_plugin;
|
||
|
}
|
||
|
#else
|
||
|
static void __attribute__((constructor))
|
||
|
serial_register_plugin(void)
|
||
|
{
|
||
|
plugin_reg_listener(&serial_plugin);
|
||
|
}
|
||
|
#endif
|
||
|
|