name_distr.c 9.15 KB
Newer Older
Per Liden's avatar
Per Liden committed
1
2
/*
 * net/tipc/name_distr.c: TIPC name distribution code
3
 *
4
 * Copyright (c) 2000-2006, Ericsson AB
5
 * Copyright (c) 2005, 2010-2011, Wind River Systems
Per Liden's avatar
Per Liden committed
6
7
 * All rights reserved.
 *
Per Liden's avatar
Per Liden committed
8
 * Redistribution and use in source and binary forms, with or without
Per Liden's avatar
Per Liden committed
9
10
 * modification, are permitted provided that the following conditions are met:
 *
Per Liden's avatar
Per Liden committed
11
12
13
14
15
16
17
18
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. Neither the names of the copyright holders nor the names of its
 *    contributors may be used to endorse or promote products derived from
 *    this software without specific prior written permission.
Per Liden's avatar
Per Liden committed
19
 *
Per Liden's avatar
Per Liden committed
20
21
22
23
24
25
26
27
28
29
30
31
32
33
 * Alternatively, this software may be distributed under the terms of the
 * GNU General Public License ("GPL") version 2 as published by the Free
 * Software Foundation.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
Per Liden's avatar
Per Liden committed
34
35
36
37
38
39
40
41
 * POSSIBILITY OF SUCH DAMAGE.
 */

#include "core.h"
#include "link.h"
#include "name_distr.h"

/**
42
43
44
 * struct publ_list - list of publications made by this node
 * @list: circular list of publications
 * @list_size: number of entries in list
Per Liden's avatar
Per Liden committed
45
 */
46
47
48
49
struct publ_list {
	struct list_head list;
	u32 size;
};
Per Liden's avatar
Per Liden committed
50

51
52
53
54
55
static struct publ_list publ_zone = {
	.list = LIST_HEAD_INIT(publ_zone.list),
	.size = 0,
};

56
57
58
59
static struct publ_list publ_cluster = {
	.list = LIST_HEAD_INIT(publ_cluster.list),
	.size = 0,
};
Per Liden's avatar
Per Liden committed
60

61
62
63
64
65
66
67
68
69
70
71
72
73
static struct publ_list publ_node = {
	.list = LIST_HEAD_INIT(publ_node.list),
	.size = 0,
};

static struct publ_list *publ_lists[] = {
	NULL,
	&publ_zone,	/* publ_lists[TIPC_ZONE_SCOPE]		*/
	&publ_cluster,	/* publ_lists[TIPC_CLUSTER_SCOPE]	*/
	&publ_node	/* publ_lists[TIPC_NODE_SCOPE]		*/
};


Per Liden's avatar
Per Liden committed
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
/**
 * publ_to_item - add publication info to a publication message
 */
static void publ_to_item(struct distr_item *i, struct publication *p)
{
	i->type = htonl(p->type);
	i->lower = htonl(p->lower);
	i->upper = htonl(p->upper);
	i->ref = htonl(p->ref);
	i->key = htonl(p->key);
}

/**
 * named_prepare_buf - allocate & initialize a publication message
 */
static struct sk_buff *named_prepare_buf(u32 type, u32 size, u32 dest)
{
91
	struct sk_buff *buf = tipc_buf_acquire(INT_H_SIZE + size);
Per Liden's avatar
Per Liden committed
92
93
94
95
	struct tipc_msg *msg;

	if (buf != NULL) {
		msg = buf_msg(buf);
96
97
		tipc_msg_init(msg, NAME_DISTRIBUTOR, type, INT_H_SIZE, dest);
		msg_set_size(msg, INT_H_SIZE + size);
Per Liden's avatar
Per Liden committed
98
99
100
101
	}
	return buf;
}

102
void named_cluster_distribute(struct sk_buff *buf)
103
{
104
105
106
	struct sk_buff *obuf;
	struct tipc_node *node;
	u32 dnode;
107

108
	rcu_read_lock();
109
110
111
112
113
114
115
116
117
118
	list_for_each_entry_rcu(node, &tipc_node_list, list) {
		dnode = node->addr;
		if (in_own_node(dnode))
			continue;
		if (!tipc_node_active_links(node))
			continue;
		obuf = skb_copy(buf, GFP_ATOMIC);
		if (!obuf)
			break;
		msg_set_destnode(buf_msg(obuf), dnode);
119
		tipc_link_xmit(obuf, dnode, dnode);
120
	}
121
	rcu_read_unlock();
122

123
	kfree_skb(buf);
124
125
}

Per Liden's avatar
Per Liden committed
126
/**
127
 * tipc_named_publish - tell other nodes about a new publication by this node
Per Liden's avatar
Per Liden committed
128
 */
129
struct sk_buff *tipc_named_publish(struct publication *publ)
Per Liden's avatar
Per Liden committed
130
131
132
133
{
	struct sk_buff *buf;
	struct distr_item *item;

134
135
	list_add_tail(&publ->local_list, &publ_lists[publ->scope]->list);
	publ_lists[publ->scope]->size++;
Per Liden's avatar
Per Liden committed
136

137
	if (publ->scope == TIPC_NODE_SCOPE)
138
		return NULL;
139

Per Liden's avatar
Per Liden committed
140
141
	buf = named_prepare_buf(PUBLICATION, ITEM_SIZE, 0);
	if (!buf) {
142
		pr_warn("Publication distribution failure\n");
143
		return NULL;
Per Liden's avatar
Per Liden committed
144
145
146
147
	}

	item = (struct distr_item *)msg_data(buf_msg(buf));
	publ_to_item(item, publ);
148
	return buf;
Per Liden's avatar
Per Liden committed
149
150
151
}

/**
152
 * tipc_named_withdraw - tell other nodes about a withdrawn publication by this node
Per Liden's avatar
Per Liden committed
153
 */
154
struct sk_buff *tipc_named_withdraw(struct publication *publ)
Per Liden's avatar
Per Liden committed
155
156
157
158
159
{
	struct sk_buff *buf;
	struct distr_item *item;

	list_del(&publ->local_list);
160
	publ_lists[publ->scope]->size--;
Per Liden's avatar
Per Liden committed
161

162
	if (publ->scope == TIPC_NODE_SCOPE)
163
		return NULL;
164

Per Liden's avatar
Per Liden committed
165
166
	buf = named_prepare_buf(WITHDRAWAL, ITEM_SIZE, 0);
	if (!buf) {
167
		pr_warn("Withdrawal distribution failure\n");
168
		return NULL;
Per Liden's avatar
Per Liden committed
169
170
171
172
	}

	item = (struct distr_item *)msg_data(buf_msg(buf));
	publ_to_item(item, publ);
173
	return buf;
Per Liden's avatar
Per Liden committed
174
175
}

176
/**
177
 * named_distribute - prepare name info for bulk distribution to another node
178
179
180
 * @msg_list: list of messages (buffers) to be returned from this function
 * @dnode: node to be updated
 * @pls: linked list of publication items to be packed into buffer chain
181
 */
182
183
static void named_distribute(struct list_head *msg_list, u32 dnode,
			     struct publ_list *pls)
184
185
186
187
{
	struct publication *publ;
	struct sk_buff *buf = NULL;
	struct distr_item *item = NULL;
188
189
190
191
	uint dsz = pls->size * ITEM_SIZE;
	uint msg_dsz = (tipc_node_get_mtu(dnode, 0) / ITEM_SIZE) * ITEM_SIZE;
	uint rem = dsz;
	uint msg_rem = 0;
192
193

	list_for_each_entry(publ, &pls->list, local_list) {
194
		/* Prepare next buffer: */
195
		if (!buf) {
196
197
198
			msg_rem = min_t(uint, rem, msg_dsz);
			rem -= msg_rem;
			buf = named_prepare_buf(PUBLICATION, msg_rem, dnode);
199
			if (!buf) {
200
				pr_warn("Bulk publication failure\n");
201
202
203
204
				return;
			}
			item = (struct distr_item *)msg_data(buf_msg(buf));
		}
205
206

		/* Pack publication into message: */
207
208
		publ_to_item(item, publ);
		item++;
209
210
211
212
213
		msg_rem -= ITEM_SIZE;

		/* Append full buffer to list: */
		if (!msg_rem) {
			list_add_tail((struct list_head *)buf, msg_list);
214
215
216
217
218
			buf = NULL;
		}
	}
}

Per Liden's avatar
Per Liden committed
219
/**
220
 * tipc_named_node_up - tell specified node about all publications by this node
Per Liden's avatar
Per Liden committed
221
 */
222
void tipc_named_node_up(u32 dnode)
Per Liden's avatar
Per Liden committed
223
{
224
225
	LIST_HEAD(msg_list);
	struct sk_buff *buf_chain;
226

227
	read_lock_bh(&tipc_nametbl_lock);
228
229
	named_distribute(&msg_list, dnode, &publ_cluster);
	named_distribute(&msg_list, dnode, &publ_zone);
230
	read_unlock_bh(&tipc_nametbl_lock);
231

232
233
234
	/* Convert circular list to linear list and send: */
	buf_chain = (struct sk_buff *)msg_list.next;
	((struct sk_buff *)msg_list.prev)->next = NULL;
235
	tipc_link_xmit(buf_chain, dnode, dnode);
Per Liden's avatar
Per Liden committed
236
237
238
}

/**
239
 * named_purge_publ - remove publication associated with a failed node
240
241
 *
 * Invoked for each publication issued by a newly failed node.
Per Liden's avatar
Per Liden committed
242
243
 * Removes publication structure from name table & deletes it.
 */
244
static void named_purge_publ(struct publication *publ)
Per Liden's avatar
Per Liden committed
245
246
{
	struct publication *p;
247

248
249
	write_lock_bh(&tipc_nametbl_lock);
	p = tipc_nametbl_remove_publ(publ->type, publ->lower,
250
				     publ->node, publ->ref, publ->key);
251
252
	if (p)
		tipc_nodesub_unsubscribe(&p->subscr);
253
	write_unlock_bh(&tipc_nametbl_lock);
254

255
	if (p != publ) {
256
257
258
259
		pr_err("Unable to remove publication from failed node\n"
		       " (type=%u, lower=%u, node=0x%x, ref=%u, key=%u)\n",
		       publ->type, publ->lower, publ->node, publ->ref,
		       publ->key);
260
261
	}

262
	kfree(p);
Per Liden's avatar
Per Liden committed
263
264
}

265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
/**
 * tipc_update_nametbl - try to process a nametable update and notify
 *			 subscribers
 *
 * tipc_nametbl_lock must be held.
 * Returns the publication item if successful, otherwise NULL.
 */
struct publication *tipc_update_nametbl(struct distr_item *i, u32 node,
					u32 dtype)
{
	struct publication *publ = NULL;

	if (dtype == PUBLICATION) {
		publ = tipc_nametbl_insert_publ(ntohl(i->type), ntohl(i->lower),
						ntohl(i->upper),
						TIPC_CLUSTER_SCOPE, node,
						ntohl(i->ref), ntohl(i->key));
		if (publ) {
			tipc_nodesub_subscribe(&publ->subscr, node, publ,
					       (net_ev_handler)
					       named_purge_publ);
		}
	} else if (dtype == WITHDRAWAL) {
		publ = tipc_nametbl_remove_publ(ntohl(i->type), ntohl(i->lower),
						node, ntohl(i->ref),
						ntohl(i->key));
		if (publ) {
			tipc_nodesub_unsubscribe(&publ->subscr);
			kfree(publ);
		}
	} else {
		pr_warn("Unrecognized name table message received\n");
	}
	return publ;
}

Per Liden's avatar
Per Liden committed
301
/**
302
 * tipc_named_rcv - process name table update message sent by another node
Per Liden's avatar
Per Liden committed
303
 */
304
void tipc_named_rcv(struct sk_buff *buf)
Per Liden's avatar
Per Liden committed
305
306
307
308
309
{
	struct tipc_msg *msg = buf_msg(buf);
	struct distr_item *item = (struct distr_item *)msg_data(msg);
	u32 count = msg_data_sz(msg) / ITEM_SIZE;

310
	write_lock_bh(&tipc_nametbl_lock);
Per Liden's avatar
Per Liden committed
311
	while (count--) {
312
313
		tipc_update_nametbl(item, msg_orignode(msg),
				    msg_type(msg));
Per Liden's avatar
Per Liden committed
314
315
		item++;
	}
316
	write_unlock_bh(&tipc_nametbl_lock);
317
	kfree_skb(buf);
Per Liden's avatar
Per Liden committed
318
319
320
}

/**
321
 * tipc_named_reinit - re-initialize local publications
322
 *
323
 * This routine is called whenever TIPC networking is enabled.
324
325
 * All name table entries published by this node are updated to reflect
 * the node's new network address.
Per Liden's avatar
Per Liden committed
326
 */
327
void tipc_named_reinit(void)
Per Liden's avatar
Per Liden committed
328
329
{
	struct publication *publ;
330
	int scope;
Per Liden's avatar
Per Liden committed
331

332
	write_lock_bh(&tipc_nametbl_lock);
333

334
	for (scope = TIPC_ZONE_SCOPE; scope <= TIPC_NODE_SCOPE; scope++)
335
336
		list_for_each_entry(publ, &publ_lists[scope]->list, local_list)
			publ->node = tipc_own_addr;
337

338
	write_unlock_bh(&tipc_nametbl_lock);
Per Liden's avatar
Per Liden committed
339
}