name_distr.c 10.7 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, 2014, 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
 * POSSIBILITY OF SUCH DAMAGE.
 */

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

41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
int sysctl_tipc_named_timeout __read_mostly = 2000;

/**
 * struct tipc_dist_queue - queue holding deferred name table updates
 */
static struct list_head tipc_dist_queue = LIST_HEAD_INIT(tipc_dist_queue);

struct distr_queue_item {
	struct distr_item i;
	u32 dtype;
	u32 node;
	unsigned long expires;
	struct list_head next;
};

Per Liden's avatar
Per Liden committed
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
/**
 * 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)
{
73
	struct sk_buff *buf = tipc_buf_acquire(INT_H_SIZE + size);
Per Liden's avatar
Per Liden committed
74
75
76
77
	struct tipc_msg *msg;

	if (buf != NULL) {
		msg = buf_msg(buf);
78
79
		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
80
81
82
83
	}
	return buf;
}

84
void named_cluster_distribute(struct sk_buff *skb)
85
{
86
	struct sk_buff *oskb;
87
88
	struct tipc_node *node;
	u32 dnode;
89

90
	rcu_read_lock();
91
92
93
94
95
96
	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;
97
98
		oskb = skb_copy(skb, GFP_ATOMIC);
		if (!oskb)
99
			break;
100
101
		msg_set_destnode(buf_msg(oskb), dnode);
		tipc_link_xmit_skb(oskb, dnode, dnode);
102
	}
103
	rcu_read_unlock();
104

105
	kfree_skb(skb);
106
107
}

Per Liden's avatar
Per Liden committed
108
/**
109
 * tipc_named_publish - tell other nodes about a new publication by this node
Per Liden's avatar
Per Liden committed
110
 */
111
struct sk_buff *tipc_named_publish(struct publication *publ)
Per Liden's avatar
Per Liden committed
112
113
114
115
{
	struct sk_buff *buf;
	struct distr_item *item;

116
117
	list_add_tail_rcu(&publ->local_list,
			  &tipc_nametbl->publ_list[publ->scope]);
Per Liden's avatar
Per Liden committed
118

119
	if (publ->scope == TIPC_NODE_SCOPE)
120
		return NULL;
121

Per Liden's avatar
Per Liden committed
122
123
	buf = named_prepare_buf(PUBLICATION, ITEM_SIZE, 0);
	if (!buf) {
124
		pr_warn("Publication distribution failure\n");
125
		return NULL;
Per Liden's avatar
Per Liden committed
126
127
128
129
	}

	item = (struct distr_item *)msg_data(buf_msg(buf));
	publ_to_item(item, publ);
130
	return buf;
Per Liden's avatar
Per Liden committed
131
132
133
}

/**
134
 * tipc_named_withdraw - tell other nodes about a withdrawn publication by this node
Per Liden's avatar
Per Liden committed
135
 */
136
struct sk_buff *tipc_named_withdraw(struct publication *publ)
Per Liden's avatar
Per Liden committed
137
138
139
140
141
142
{
	struct sk_buff *buf;
	struct distr_item *item;

	list_del(&publ->local_list);

143
	if (publ->scope == TIPC_NODE_SCOPE)
144
		return NULL;
145

Per Liden's avatar
Per Liden committed
146
147
	buf = named_prepare_buf(WITHDRAWAL, ITEM_SIZE, 0);
	if (!buf) {
148
		pr_warn("Withdrawal distribution failure\n");
149
		return NULL;
Per Liden's avatar
Per Liden committed
150
151
152
153
	}

	item = (struct distr_item *)msg_data(buf_msg(buf));
	publ_to_item(item, publ);
154
	return buf;
Per Liden's avatar
Per Liden committed
155
156
}

157
/**
158
 * named_distribute - prepare name info for bulk distribution to another node
159
 * @list: list of messages (buffers) to be returned from this function
160
161
 * @dnode: node to be updated
 * @pls: linked list of publication items to be packed into buffer chain
162
 */
163
static void named_distribute(struct sk_buff_head *list, u32 dnode,
164
			     struct list_head *pls)
165
166
{
	struct publication *publ;
167
	struct sk_buff *skb = NULL;
168
	struct distr_item *item = NULL;
169
	uint msg_dsz = (tipc_node_get_mtu(dnode, 0) / ITEM_SIZE) * ITEM_SIZE;
170
	uint msg_rem = msg_dsz;
171

172
	list_for_each_entry(publ, pls, local_list) {
173
		/* Prepare next buffer: */
174
175
176
		if (!skb) {
			skb = named_prepare_buf(PUBLICATION, msg_rem, dnode);
			if (!skb) {
177
				pr_warn("Bulk publication failure\n");
178
179
				return;
			}
180
			item = (struct distr_item *)msg_data(buf_msg(skb));
181
		}
182
183

		/* Pack publication into message: */
184
185
		publ_to_item(item, publ);
		item++;
186
187
188
189
		msg_rem -= ITEM_SIZE;

		/* Append full buffer to list: */
		if (!msg_rem) {
190
191
			__skb_queue_tail(list, skb);
			skb = NULL;
192
			msg_rem = msg_dsz;
193
194
		}
	}
195
196
197
198
199
	if (skb) {
		msg_set_size(buf_msg(skb), INT_H_SIZE + (msg_dsz - msg_rem));
		skb_trim(skb, INT_H_SIZE + (msg_dsz - msg_rem));
		__skb_queue_tail(list, skb);
	}
200
201
}

Per Liden's avatar
Per Liden committed
202
/**
203
 * tipc_named_node_up - tell specified node about all publications by this node
Per Liden's avatar
Per Liden committed
204
 */
205
void tipc_named_node_up(u32 dnode)
Per Liden's avatar
Per Liden committed
206
{
207
208
209
	struct sk_buff_head head;

	__skb_queue_head_init(&head);
210

211
	rcu_read_lock();
212
213
214
215
	named_distribute(&head, dnode,
			 &tipc_nametbl->publ_list[TIPC_CLUSTER_SCOPE]);
	named_distribute(&head, dnode,
			 &tipc_nametbl->publ_list[TIPC_ZONE_SCOPE]);
216
	rcu_read_unlock();
217

218
	tipc_link_xmit(&head, dnode, dnode);
Per Liden's avatar
Per Liden committed
219
220
}

221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
static void tipc_publ_subscribe(struct publication *publ, u32 addr)
{
	struct tipc_node *node;

	if (in_own_node(addr))
		return;

	node = tipc_node_find(addr);
	if (!node) {
		pr_warn("Node subscription rejected, unknown node 0x%x\n",
			addr);
		return;
	}

	tipc_node_lock(node);
	list_add_tail(&publ->nodesub_list, &node->publ_list);
	tipc_node_unlock(node);
}

static void tipc_publ_unsubscribe(struct publication *publ, u32 addr)
{
	struct tipc_node *node;

	node = tipc_node_find(addr);
	if (!node)
		return;

	tipc_node_lock(node);
	list_del_init(&publ->nodesub_list);
	tipc_node_unlock(node);
}

Per Liden's avatar
Per Liden committed
253
/**
254
 * tipc_publ_purge - remove publication associated with a failed node
255
256
 *
 * Invoked for each publication issued by a newly failed node.
Per Liden's avatar
Per Liden committed
257
258
 * Removes publication structure from name table & deletes it.
 */
259
static void tipc_publ_purge(struct publication *publ, u32 addr)
Per Liden's avatar
Per Liden committed
260
261
{
	struct publication *p;
262

263
	spin_lock_bh(&tipc_nametbl_lock);
264
	p = tipc_nametbl_remove_publ(publ->type, publ->lower,
265
				     publ->node, publ->ref, publ->key);
266
	if (p)
267
		tipc_publ_unsubscribe(p, addr);
268
	spin_unlock_bh(&tipc_nametbl_lock);
269

270
	if (p != publ) {
271
272
273
274
		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);
275
276
	}

277
	kfree_rcu(p, rcu);
Per Liden's avatar
Per Liden committed
278
279
}

280
281
282
283
284
285
286
287
void tipc_publ_notify(struct list_head *nsub_list, u32 addr)
{
	struct publication *publ, *tmp;

	list_for_each_entry_safe(publ, tmp, nsub_list, nodesub_list)
		tipc_publ_purge(publ, addr);
}

288
289
290
291
292
293
294
/**
 * 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.
 */
Erik Hugne's avatar
Erik Hugne committed
295
static bool tipc_update_nametbl(struct distr_item *i, u32 node, u32 dtype)
296
297
298
299
300
301
302
303
304
{
	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) {
305
			tipc_publ_subscribe(publ, node);
Erik Hugne's avatar
Erik Hugne committed
306
			return true;
307
308
309
310
311
312
		}
	} else if (dtype == WITHDRAWAL) {
		publ = tipc_nametbl_remove_publ(ntohl(i->type), ntohl(i->lower),
						node, ntohl(i->ref),
						ntohl(i->key));
		if (publ) {
313
			tipc_publ_unsubscribe(publ, node);
314
			kfree_rcu(publ, rcu);
Erik Hugne's avatar
Erik Hugne committed
315
			return true;
316
317
318
319
		}
	} else {
		pr_warn("Unrecognized name table message received\n");
	}
Erik Hugne's avatar
Erik Hugne committed
320
	return false;
321
322
}

323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
/**
 * tipc_named_add_backlog - add a failed name table update to the backlog
 *
 */
static void tipc_named_add_backlog(struct distr_item *i, u32 type, u32 node)
{
	struct distr_queue_item *e;
	unsigned long now = get_jiffies_64();

	e = kzalloc(sizeof(*e), GFP_ATOMIC);
	if (!e)
		return;
	e->dtype = type;
	e->node = node;
	e->expires = now + msecs_to_jiffies(sysctl_tipc_named_timeout);
	memcpy(e, i, sizeof(*i));
	list_add_tail(&e->next, &tipc_dist_queue);
}

/**
 * tipc_named_process_backlog - try to process any pending name table updates
 * from the network.
 */
void tipc_named_process_backlog(void)
{
	struct distr_queue_item *e, *tmp;
	char addr[16];
	unsigned long now = get_jiffies_64();

	list_for_each_entry_safe(e, tmp, &tipc_dist_queue, next) {
		if (time_after(e->expires, now)) {
			if (!tipc_update_nametbl(&e->i, e->node, e->dtype))
				continue;
		} else {
			tipc_addr_string_fill(addr, e->node);
			pr_warn_ratelimited("Dropping name table update (%d) of {%u, %u, %u} from %s key=%u\n",
					    e->dtype, ntohl(e->i.type),
					    ntohl(e->i.lower),
					    ntohl(e->i.upper),
					    addr, ntohl(e->i.key));
		}
		list_del(&e->next);
		kfree(e);
	}
}

Per Liden's avatar
Per Liden committed
369
/**
370
 * tipc_named_rcv - process name table update message sent by another node
Per Liden's avatar
Per Liden committed
371
 */
372
void tipc_named_rcv(struct sk_buff *buf)
Per Liden's avatar
Per Liden committed
373
374
375
376
{
	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;
377
	u32 node = msg_orignode(msg);
Per Liden's avatar
Per Liden committed
378

379
	spin_lock_bh(&tipc_nametbl_lock);
Per Liden's avatar
Per Liden committed
380
	while (count--) {
381
382
		if (!tipc_update_nametbl(item, node, msg_type(msg)))
			tipc_named_add_backlog(item, msg_type(msg), node);
Per Liden's avatar
Per Liden committed
383
384
		item++;
	}
385
	tipc_named_process_backlog();
386
	spin_unlock_bh(&tipc_nametbl_lock);
387
	kfree_skb(buf);
Per Liden's avatar
Per Liden committed
388
389
390
}

/**
391
 * tipc_named_reinit - re-initialize local publications
392
 *
393
 * This routine is called whenever TIPC networking is enabled.
394
395
 * 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
396
 */
397
void tipc_named_reinit(void)
Per Liden's avatar
Per Liden committed
398
399
{
	struct publication *publ;
400
	int scope;
Per Liden's avatar
Per Liden committed
401

402
	spin_lock_bh(&tipc_nametbl_lock);
403

404
	for (scope = TIPC_ZONE_SCOPE; scope <= TIPC_NODE_SCOPE; scope++)
405
406
		list_for_each_entry_rcu(publ, &tipc_nametbl->publ_list[scope],
					local_list)
407
			publ->node = tipc_own_addr;
408

409
	spin_unlock_bh(&tipc_nametbl_lock);
Per Liden's avatar
Per Liden committed
410
}