mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-17 10:46:33 +00:00
fs: dlm: remove dlm_node_addrs lookup list
This patch merges the dlm_node_addrs lookup list to the connection structure. It is a per node mapping to some configuration setup by configfs. We don't need two lookup structures. The connection hash has now a lifetime like the dlm_node_addrs entries. Means we add only new entries when configure cluster and not while new connections are coming in, remove connection when a node got fenced and cleanup all connection when the dlm exits. It should work the same and even will show more issues because we don't try to somehow keep those two data structures in sync with the current cluster configuration. Signed-off-by: Alexander Aring <aahringo@redhat.com> Signed-off-by: David Teigland <teigland@redhat.com>
This commit is contained in:
parent
c51c9cd8ad
commit
6f0b0b5d7a
@ -93,6 +93,11 @@ struct connection {
|
|||||||
unsigned char *rx_buf;
|
unsigned char *rx_buf;
|
||||||
int rx_buflen;
|
int rx_buflen;
|
||||||
int rx_leftover;
|
int rx_leftover;
|
||||||
|
int mark;
|
||||||
|
int addr_count;
|
||||||
|
int curr_addr_index;
|
||||||
|
struct sockaddr_storage addr[DLM_MAX_ADDR_COUNT];
|
||||||
|
spinlock_t addrs_lock;
|
||||||
struct rcu_head rcu;
|
struct rcu_head rcu;
|
||||||
};
|
};
|
||||||
#define sock2con(x) ((struct connection *)(x)->sk_user_data)
|
#define sock2con(x) ((struct connection *)(x)->sk_user_data)
|
||||||
@ -131,15 +136,6 @@ struct dlm_msg {
|
|||||||
struct kref ref;
|
struct kref ref;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct dlm_node_addr {
|
|
||||||
struct list_head list;
|
|
||||||
int nodeid;
|
|
||||||
int mark;
|
|
||||||
int addr_count;
|
|
||||||
int curr_addr_index;
|
|
||||||
struct sockaddr_storage *addr[DLM_MAX_ADDR_COUNT];
|
|
||||||
};
|
|
||||||
|
|
||||||
struct dlm_proto_ops {
|
struct dlm_proto_ops {
|
||||||
bool try_new_addr;
|
bool try_new_addr;
|
||||||
const char *name;
|
const char *name;
|
||||||
@ -161,9 +157,6 @@ static struct listen_sock_callbacks {
|
|||||||
void (*sk_write_space)(struct sock *);
|
void (*sk_write_space)(struct sock *);
|
||||||
} listen_sock;
|
} listen_sock;
|
||||||
|
|
||||||
static LIST_HEAD(dlm_node_addrs);
|
|
||||||
static DEFINE_SPINLOCK(dlm_node_addrs_spin);
|
|
||||||
|
|
||||||
static struct listen_connection listen_con;
|
static struct listen_connection listen_con;
|
||||||
static struct sockaddr_storage dlm_local_addr[DLM_MAX_ADDR_COUNT];
|
static struct sockaddr_storage dlm_local_addr[DLM_MAX_ADDR_COUNT];
|
||||||
static int dlm_local_count;
|
static int dlm_local_count;
|
||||||
@ -306,17 +299,6 @@ static void foreach_conn(void (*conn_func)(struct connection *c))
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct dlm_node_addr *find_node_addr(int nodeid)
|
|
||||||
{
|
|
||||||
struct dlm_node_addr *na;
|
|
||||||
|
|
||||||
list_for_each_entry(na, &dlm_node_addrs, list) {
|
|
||||||
if (na->nodeid == nodeid)
|
|
||||||
return na;
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int addr_compare(const struct sockaddr_storage *x,
|
static int addr_compare(const struct sockaddr_storage *x,
|
||||||
const struct sockaddr_storage *y)
|
const struct sockaddr_storage *y)
|
||||||
{
|
{
|
||||||
@ -350,38 +332,45 @@ static int nodeid_to_addr(int nodeid, struct sockaddr_storage *sas_out,
|
|||||||
unsigned int *mark)
|
unsigned int *mark)
|
||||||
{
|
{
|
||||||
struct sockaddr_storage sas;
|
struct sockaddr_storage sas;
|
||||||
struct dlm_node_addr *na;
|
struct connection *con;
|
||||||
|
int idx;
|
||||||
|
|
||||||
if (!dlm_local_count)
|
if (!dlm_local_count)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
spin_lock(&dlm_node_addrs_spin);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
na = find_node_addr(nodeid);
|
con = nodeid2con(nodeid, 0);
|
||||||
if (na && na->addr_count) {
|
if (!con) {
|
||||||
memcpy(&sas, na->addr[na->curr_addr_index],
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
sizeof(struct sockaddr_storage));
|
|
||||||
|
|
||||||
if (try_new_addr) {
|
|
||||||
na->curr_addr_index++;
|
|
||||||
if (na->curr_addr_index == na->addr_count)
|
|
||||||
na->curr_addr_index = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
|
||||||
|
|
||||||
if (!na)
|
|
||||||
return -EEXIST;
|
|
||||||
|
|
||||||
if (!na->addr_count)
|
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
*mark = na->mark;
|
spin_lock(&con->addrs_lock);
|
||||||
|
if (!con->addr_count) {
|
||||||
|
spin_unlock(&con->addrs_lock);
|
||||||
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(&sas, &con->addr[con->curr_addr_index],
|
||||||
|
sizeof(struct sockaddr_storage));
|
||||||
|
|
||||||
|
if (try_new_addr) {
|
||||||
|
con->curr_addr_index++;
|
||||||
|
if (con->curr_addr_index == con->addr_count)
|
||||||
|
con->curr_addr_index = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
*mark = con->mark;
|
||||||
|
spin_unlock(&con->addrs_lock);
|
||||||
|
|
||||||
if (sas_out)
|
if (sas_out)
|
||||||
memcpy(sas_out, &sas, sizeof(struct sockaddr_storage));
|
memcpy(sas_out, &sas, sizeof(struct sockaddr_storage));
|
||||||
|
|
||||||
if (!sa_out)
|
if (!sa_out) {
|
||||||
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (dlm_local_addr[0].ss_family == AF_INET) {
|
if (dlm_local_addr[0].ss_family == AF_INET) {
|
||||||
struct sockaddr_in *in4 = (struct sockaddr_in *) &sas;
|
struct sockaddr_in *in4 = (struct sockaddr_in *) &sas;
|
||||||
@ -393,43 +382,46 @@ static int nodeid_to_addr(int nodeid, struct sockaddr_storage *sas_out,
|
|||||||
ret6->sin6_addr = in6->sin6_addr;
|
ret6->sin6_addr = in6->sin6_addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int addr_to_nodeid(struct sockaddr_storage *addr, int *nodeid,
|
static int addr_to_nodeid(struct sockaddr_storage *addr, int *nodeid,
|
||||||
unsigned int *mark)
|
unsigned int *mark)
|
||||||
{
|
{
|
||||||
struct dlm_node_addr *na;
|
struct connection *con;
|
||||||
int rv = -EEXIST;
|
int i, idx, addr_i;
|
||||||
int addr_i;
|
|
||||||
|
|
||||||
spin_lock(&dlm_node_addrs_spin);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
list_for_each_entry(na, &dlm_node_addrs, list) {
|
for (i = 0; i < CONN_HASH_SIZE; i++) {
|
||||||
if (!na->addr_count)
|
hlist_for_each_entry_rcu(con, &connection_hash[i], list) {
|
||||||
continue;
|
WARN_ON_ONCE(!con->addr_count);
|
||||||
|
|
||||||
for (addr_i = 0; addr_i < na->addr_count; addr_i++) {
|
spin_lock(&con->addrs_lock);
|
||||||
if (addr_compare(na->addr[addr_i], addr)) {
|
for (addr_i = 0; addr_i < con->addr_count; addr_i++) {
|
||||||
*nodeid = na->nodeid;
|
if (addr_compare(&con->addr[addr_i], addr)) {
|
||||||
*mark = na->mark;
|
*nodeid = con->nodeid;
|
||||||
rv = 0;
|
*mark = con->mark;
|
||||||
goto unlock;
|
spin_unlock(&con->addrs_lock);
|
||||||
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
spin_unlock(&con->addrs_lock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
unlock:
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
|
||||||
return rv;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* caller need to held dlm_node_addrs_spin lock */
|
static bool dlm_lowcomms_con_has_addr(const struct connection *con,
|
||||||
static bool dlm_lowcomms_na_has_addr(const struct dlm_node_addr *na,
|
const struct sockaddr_storage *addr)
|
||||||
const struct sockaddr_storage *addr)
|
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < na->addr_count; i++) {
|
for (i = 0; i < con->addr_count; i++) {
|
||||||
if (addr_compare(na->addr[i], addr))
|
if (addr_compare(&con->addr[i], addr))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -438,52 +430,42 @@ static bool dlm_lowcomms_na_has_addr(const struct dlm_node_addr *na,
|
|||||||
|
|
||||||
int dlm_lowcomms_addr(int nodeid, struct sockaddr_storage *addr, int len)
|
int dlm_lowcomms_addr(int nodeid, struct sockaddr_storage *addr, int len)
|
||||||
{
|
{
|
||||||
struct sockaddr_storage *new_addr;
|
struct connection *con;
|
||||||
struct dlm_node_addr *new_node, *na;
|
bool ret, idx;
|
||||||
bool ret;
|
|
||||||
|
|
||||||
new_node = kzalloc(sizeof(struct dlm_node_addr), GFP_NOFS);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
if (!new_node)
|
con = nodeid2con(nodeid, GFP_NOFS);
|
||||||
return -ENOMEM;
|
if (!con) {
|
||||||
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
new_addr = kzalloc(sizeof(struct sockaddr_storage), GFP_NOFS);
|
|
||||||
if (!new_addr) {
|
|
||||||
kfree(new_node);
|
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(new_addr, addr, len);
|
spin_lock(&con->addrs_lock);
|
||||||
|
if (!con->addr_count) {
|
||||||
spin_lock(&dlm_node_addrs_spin);
|
memcpy(&con->addr[0], addr, sizeof(*addr));
|
||||||
na = find_node_addr(nodeid);
|
con->addr_count = 1;
|
||||||
if (!na) {
|
con->mark = dlm_config.ci_mark;
|
||||||
new_node->nodeid = nodeid;
|
spin_unlock(&con->addrs_lock);
|
||||||
new_node->addr[0] = new_addr;
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
new_node->addr_count = 1;
|
|
||||||
new_node->mark = dlm_config.ci_mark;
|
|
||||||
list_add(&new_node->list, &dlm_node_addrs);
|
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = dlm_lowcomms_na_has_addr(na, addr);
|
ret = dlm_lowcomms_con_has_addr(con, addr);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
spin_unlock(&con->addrs_lock);
|
||||||
kfree(new_addr);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
kfree(new_node);
|
|
||||||
return -EEXIST;
|
return -EEXIST;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (na->addr_count >= DLM_MAX_ADDR_COUNT) {
|
if (con->addr_count >= DLM_MAX_ADDR_COUNT) {
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
spin_unlock(&con->addrs_lock);
|
||||||
kfree(new_addr);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
kfree(new_node);
|
|
||||||
return -ENOSPC;
|
return -ENOSPC;
|
||||||
}
|
}
|
||||||
|
|
||||||
na->addr[na->addr_count++] = new_addr;
|
memcpy(&con->addr[con->addr_count++], addr, sizeof(*addr));
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
kfree(new_node);
|
spin_unlock(&con->addrs_lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -558,10 +540,10 @@ int dlm_lowcomms_connect_node(int nodeid)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
idx = srcu_read_lock(&connections_srcu);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
con = nodeid2con(nodeid, GFP_NOFS);
|
con = nodeid2con(nodeid, 0);
|
||||||
if (!con) {
|
if (WARN_ON_ONCE(!con)) {
|
||||||
srcu_read_unlock(&connections_srcu, idx);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
return -ENOMEM;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
lowcomms_connect_sock(con);
|
lowcomms_connect_sock(con);
|
||||||
@ -572,18 +554,20 @@ int dlm_lowcomms_connect_node(int nodeid)
|
|||||||
|
|
||||||
int dlm_lowcomms_nodes_set_mark(int nodeid, unsigned int mark)
|
int dlm_lowcomms_nodes_set_mark(int nodeid, unsigned int mark)
|
||||||
{
|
{
|
||||||
struct dlm_node_addr *na;
|
struct connection *con;
|
||||||
|
int idx;
|
||||||
|
|
||||||
spin_lock(&dlm_node_addrs_spin);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
na = find_node_addr(nodeid);
|
con = nodeid2con(nodeid, 0);
|
||||||
if (!na) {
|
if (!con) {
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
na->mark = mark;
|
spin_lock(&con->addrs_lock);
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
con->mark = mark;
|
||||||
|
spin_unlock(&con->addrs_lock);
|
||||||
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -960,10 +944,10 @@ static int accept_from_sock(struct listen_connection *con)
|
|||||||
* In this case we store the incoming one in "othercon"
|
* In this case we store the incoming one in "othercon"
|
||||||
*/
|
*/
|
||||||
idx = srcu_read_lock(&connections_srcu);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
newcon = nodeid2con(nodeid, GFP_NOFS);
|
newcon = nodeid2con(nodeid, 0);
|
||||||
if (!newcon) {
|
if (WARN_ON_ONCE(!newcon)) {
|
||||||
srcu_read_unlock(&connections_srcu, idx);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
result = -ENOMEM;
|
result = -ENOENT;
|
||||||
goto accept_err;
|
goto accept_err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1210,8 +1194,8 @@ struct dlm_msg *dlm_lowcomms_new_msg(int nodeid, int len, gfp_t allocation,
|
|||||||
}
|
}
|
||||||
|
|
||||||
idx = srcu_read_lock(&connections_srcu);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
con = nodeid2con(nodeid, allocation);
|
con = nodeid2con(nodeid, 0);
|
||||||
if (!con) {
|
if (WARN_ON_ONCE(!con)) {
|
||||||
srcu_read_unlock(&connections_srcu, idx);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@ -1376,36 +1360,45 @@ static void clean_one_writequeue(struct connection *con)
|
|||||||
spin_unlock(&con->writequeue_lock);
|
spin_unlock(&con->writequeue_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void connection_release(struct rcu_head *rcu)
|
||||||
|
{
|
||||||
|
struct connection *con = container_of(rcu, struct connection, rcu);
|
||||||
|
|
||||||
|
kfree(con->rx_buf);
|
||||||
|
kfree(con);
|
||||||
|
}
|
||||||
|
|
||||||
/* Called from recovery when it knows that a node has
|
/* Called from recovery when it knows that a node has
|
||||||
left the cluster */
|
left the cluster */
|
||||||
int dlm_lowcomms_close(int nodeid)
|
int dlm_lowcomms_close(int nodeid)
|
||||||
{
|
{
|
||||||
struct connection *con;
|
struct connection *con;
|
||||||
struct dlm_node_addr *na;
|
|
||||||
int idx;
|
int idx;
|
||||||
|
|
||||||
log_print("closing connection to node %d", nodeid);
|
log_print("closing connection to node %d", nodeid);
|
||||||
|
|
||||||
idx = srcu_read_lock(&connections_srcu);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
con = nodeid2con(nodeid, 0);
|
con = nodeid2con(nodeid, 0);
|
||||||
if (con) {
|
if (WARN_ON_ONCE(!con)) {
|
||||||
set_bit(CF_CLOSE, &con->flags);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
close_connection(con, true, true, true);
|
return -ENOENT;
|
||||||
clean_one_writequeue(con);
|
}
|
||||||
|
|
||||||
|
spin_lock(&connections_lock);
|
||||||
|
hlist_del_rcu(&con->list);
|
||||||
|
spin_unlock(&connections_lock);
|
||||||
|
|
||||||
|
close_connection(con, true, true, true);
|
||||||
|
|
||||||
|
clean_one_writequeue(con);
|
||||||
|
call_srcu(&connections_srcu, &con->rcu, connection_release);
|
||||||
|
if (con->othercon) {
|
||||||
|
clean_one_writequeue(con->othercon);
|
||||||
if (con->othercon)
|
if (con->othercon)
|
||||||
clean_one_writequeue(con->othercon);
|
call_srcu(&connections_srcu, &con->othercon->rcu, connection_release);
|
||||||
}
|
}
|
||||||
srcu_read_unlock(&connections_srcu, idx);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
|
|
||||||
spin_lock(&dlm_node_addrs_spin);
|
|
||||||
na = find_node_addr(nodeid);
|
|
||||||
if (na) {
|
|
||||||
list_del(&na->list);
|
|
||||||
while (na->addr_count--)
|
|
||||||
kfree(na->addr[na->addr_count]);
|
|
||||||
kfree(na);
|
|
||||||
}
|
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1607,27 +1600,9 @@ static void stop_conn(struct connection *con)
|
|||||||
_stop_conn(con, true);
|
_stop_conn(con, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void connection_release(struct rcu_head *rcu)
|
|
||||||
{
|
|
||||||
struct connection *con = container_of(rcu, struct connection, rcu);
|
|
||||||
|
|
||||||
kfree(con->rx_buf);
|
|
||||||
kfree(con);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void free_conn(struct connection *con)
|
static void free_conn(struct connection *con)
|
||||||
{
|
{
|
||||||
close_connection(con, true, true, true);
|
close_connection(con, true, true, true);
|
||||||
spin_lock(&connections_lock);
|
|
||||||
hlist_del_rcu(&con->list);
|
|
||||||
spin_unlock(&connections_lock);
|
|
||||||
if (con->othercon) {
|
|
||||||
clean_one_writequeue(con->othercon);
|
|
||||||
call_srcu(&connections_srcu, &con->othercon->rcu,
|
|
||||||
connection_release);
|
|
||||||
}
|
|
||||||
clean_one_writequeue(con);
|
|
||||||
call_srcu(&connections_srcu, &con->rcu, connection_release);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void work_flush(void)
|
static void work_flush(void)
|
||||||
@ -1922,14 +1897,21 @@ void dlm_lowcomms_init(void)
|
|||||||
|
|
||||||
void dlm_lowcomms_exit(void)
|
void dlm_lowcomms_exit(void)
|
||||||
{
|
{
|
||||||
struct dlm_node_addr *na, *safe;
|
struct connection *con;
|
||||||
|
int i, idx;
|
||||||
|
|
||||||
spin_lock(&dlm_node_addrs_spin);
|
idx = srcu_read_lock(&connections_srcu);
|
||||||
list_for_each_entry_safe(na, safe, &dlm_node_addrs, list) {
|
for (i = 0; i < CONN_HASH_SIZE; i++) {
|
||||||
list_del(&na->list);
|
hlist_for_each_entry_rcu(con, &connection_hash[i], list) {
|
||||||
while (na->addr_count--)
|
spin_lock(&connections_lock);
|
||||||
kfree(na->addr[na->addr_count]);
|
hlist_del_rcu(&con->list);
|
||||||
kfree(na);
|
spin_unlock(&connections_lock);
|
||||||
|
|
||||||
|
if (con->othercon)
|
||||||
|
call_srcu(&connections_srcu, &con->othercon->rcu,
|
||||||
|
connection_release);
|
||||||
|
call_srcu(&connections_srcu, &con->rcu, connection_release);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
spin_unlock(&dlm_node_addrs_spin);
|
srcu_read_unlock(&connections_srcu, idx);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user