mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-06 17:18:29 +03:00
273 lines
6.3 KiB
C
273 lines
6.3 KiB
C
/*
|
|
* Copyright (C) 2009 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#define _FILE_OFFSET_BITS 64
|
|
|
|
#include <netinet/in.h>
|
|
#include <sys/un.h>
|
|
#include <sys/socket.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <configure.h>
|
|
#include <libdevmapper.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include "locking.h"
|
|
#include "lvm-logging.h"
|
|
#include "clvm.h"
|
|
#include "clvmd-comms.h"
|
|
#include "lvm-functions.h"
|
|
#include "clvmd.h"
|
|
|
|
static const char SINGLENODE_CLVMD_SOCKNAME[] = "\0singlenode_clvmd";
|
|
static int listen_fd = -1;
|
|
|
|
static int init_comms()
|
|
{
|
|
struct sockaddr_un addr;
|
|
|
|
listen_fd = socket(PF_UNIX, SOCK_STREAM, 0);
|
|
if (listen_fd < 0) {
|
|
DEBUGLOG("Can't create local socket: %s\n", strerror(errno));
|
|
return -1;
|
|
}
|
|
/* Set Close-on-exec */
|
|
fcntl(listen_fd, F_SETFD, 1);
|
|
|
|
memset(&addr, 0, sizeof(addr));
|
|
memcpy(addr.sun_path, SINGLENODE_CLVMD_SOCKNAME,
|
|
sizeof(SINGLENODE_CLVMD_SOCKNAME));
|
|
addr.sun_family = AF_UNIX;
|
|
|
|
if (bind(listen_fd, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
|
|
DEBUGLOG("Can't bind local socket: %s\n", strerror(errno));
|
|
close(listen_fd);
|
|
return -1;
|
|
}
|
|
if (listen(listen_fd, 10) < 0) {
|
|
DEBUGLOG("Can't listen local socket: %s\n", strerror(errno));
|
|
close(listen_fd);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _init_cluster(void)
|
|
{
|
|
int r;
|
|
|
|
r = init_comms();
|
|
if (r)
|
|
return r;
|
|
|
|
DEBUGLOG("Single-node cluster initialised.\n");
|
|
return 0;
|
|
}
|
|
|
|
static void _cluster_closedown(void)
|
|
{
|
|
close(listen_fd);
|
|
|
|
DEBUGLOG("cluster_closedown\n");
|
|
destroy_lvhash();
|
|
}
|
|
|
|
static void _get_our_csid(char *csid)
|
|
{
|
|
int nodeid = 1;
|
|
memcpy(csid, &nodeid, sizeof(int));
|
|
}
|
|
|
|
static int _csid_from_name(char *csid, const char *name)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int _name_from_csid(const char *csid, char *name)
|
|
{
|
|
sprintf(name, "%x", 0xdead);
|
|
return 0;
|
|
}
|
|
|
|
static int _get_num_nodes()
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
/* Node is now known to be running a clvmd */
|
|
static void _add_up_node(const char *csid)
|
|
{
|
|
}
|
|
|
|
/* Call a callback for each node, so the caller knows whether it's up or down */
|
|
static int _cluster_do_node_callback(struct local_client *master_client,
|
|
void (*callback)(struct local_client *,
|
|
const char *csid, int node_up))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int _lock_file(const char *file, uint32_t flags);
|
|
|
|
static int *_locks = NULL;
|
|
static char **_resources = NULL;
|
|
static int _lock_max = 1;
|
|
static pthread_mutex_t _lock_mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
/* Real locking */
|
|
static int _lock_resource(const char *resource, int mode, int flags, int *lockid)
|
|
{
|
|
int *_locks_1;
|
|
char **_resources_1;
|
|
int i, j;
|
|
|
|
DEBUGLOG("lock_resource '%s', flags=%d, mode=%d\n",
|
|
resource, flags, mode);
|
|
|
|
retry:
|
|
pthread_mutex_lock(&_lock_mutex);
|
|
|
|
/* look for an existing lock for this resource */
|
|
for (i = 1; i < _lock_max; ++i) {
|
|
if (!_resources[i])
|
|
break;
|
|
if (!strcmp(_resources[i], resource)) {
|
|
if ((_locks[i] & LCK_WRITE) || (_locks[i] & LCK_EXCL)) {
|
|
DEBUGLOG("%s already write/exclusively locked...\n", resource);
|
|
goto maybe_retry;
|
|
}
|
|
if ((mode & LCK_WRITE) || (mode & LCK_EXCL)) {
|
|
DEBUGLOG("%s already locked and WRITE/EXCL lock requested...\n",
|
|
resource);
|
|
goto maybe_retry;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (i == _lock_max) { /* out of lock slots, extend */
|
|
_locks_1 = dm_realloc(_locks, 2 * _lock_max * sizeof(int));
|
|
if (!_locks_1)
|
|
return 1; /* fail */
|
|
_locks = _locks_1;
|
|
_resources_1 = dm_realloc(_resources, 2 * _lock_max * sizeof(char *));
|
|
if (!_resources_1) {
|
|
/* _locks may get realloc'd twice, but that should be safe */
|
|
return 1; /* fail */
|
|
}
|
|
_resources = _resources_1;
|
|
/* clear the new resource entries */
|
|
for (j = _lock_max; j < 2 * _lock_max; ++j)
|
|
_resources[j] = NULL;
|
|
_lock_max = 2 * _lock_max;
|
|
}
|
|
|
|
/* resource is not currently locked, grab it */
|
|
|
|
*lockid = i;
|
|
_locks[i] = mode;
|
|
_resources[i] = dm_strdup(resource);
|
|
|
|
DEBUGLOG("%s locked -> %d\n", resource, i);
|
|
|
|
pthread_mutex_unlock(&_lock_mutex);
|
|
return 0;
|
|
maybe_retry:
|
|
pthread_mutex_unlock(&_lock_mutex);
|
|
if (!(flags & LCK_NONBLOCK)) {
|
|
usleep(10000);
|
|
goto retry;
|
|
}
|
|
|
|
return 1; /* fail */
|
|
}
|
|
|
|
static int _unlock_resource(const char *resource, int lockid)
|
|
{
|
|
DEBUGLOG("unlock_resource: %s lockid: %x\n", resource, lockid);
|
|
if(!_resources[lockid]) {
|
|
DEBUGLOG("(%s) %d not locked\n", resource, lockid);
|
|
return 1;
|
|
}
|
|
if(strcmp(_resources[lockid], resource)) {
|
|
DEBUGLOG("%d has wrong resource (requested %s, got %s)\n",
|
|
lockid, resource, _resources[lockid]);
|
|
return 1;
|
|
}
|
|
|
|
dm_free(_resources[lockid]);
|
|
_resources[lockid] = 0;
|
|
return 0;
|
|
}
|
|
|
|
static int _is_quorate()
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int _get_main_cluster_fd(void)
|
|
{
|
|
return listen_fd;
|
|
}
|
|
|
|
static int _cluster_fd_callback(struct local_client *fd, char *buf, int len,
|
|
const char *csid,
|
|
struct local_client **new_client)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int _cluster_send_message(const void *buf, int msglen,
|
|
const char *csid,
|
|
const char *errtext)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int _get_cluster_name(char *buf, int buflen)
|
|
{
|
|
strncpy(buf, "localcluster", buflen);
|
|
buf[buflen - 1] = 0;
|
|
return 0;
|
|
}
|
|
|
|
static struct cluster_ops _cluster_singlenode_ops = {
|
|
.cluster_init_completed = NULL,
|
|
.cluster_send_message = _cluster_send_message,
|
|
.name_from_csid = _name_from_csid,
|
|
.csid_from_name = _csid_from_name,
|
|
.get_num_nodes = _get_num_nodes,
|
|
.cluster_fd_callback = _cluster_fd_callback,
|
|
.get_main_cluster_fd = _get_main_cluster_fd,
|
|
.cluster_do_node_callback = _cluster_do_node_callback,
|
|
.is_quorate = _is_quorate,
|
|
.get_our_csid = _get_our_csid,
|
|
.add_up_node = _add_up_node,
|
|
.reread_config = NULL,
|
|
.cluster_closedown = _cluster_closedown,
|
|
.get_cluster_name = _get_cluster_name,
|
|
.sync_lock = _lock_resource,
|
|
.sync_unlock = _unlock_resource,
|
|
};
|
|
|
|
struct cluster_ops *init_singlenode_cluster(void)
|
|
{
|
|
if (!_init_cluster())
|
|
return &_cluster_singlenode_ops;
|
|
else
|
|
return NULL;
|
|
}
|