diff options
Diffstat (limited to 'net/tipc')
| -rw-r--r-- | net/tipc/bcast.c | 5 | ||||
| -rw-r--r-- | net/tipc/bcast.h | 1 | ||||
| -rw-r--r-- | net/tipc/bearer.c | 18 | ||||
| -rw-r--r-- | net/tipc/link.c | 167 | ||||
| -rw-r--r-- | net/tipc/link.h | 7 | ||||
| -rw-r--r-- | net/tipc/name_table.c | 20 | ||||
| -rw-r--r-- | net/tipc/net.c | 7 | ||||
| -rw-r--r-- | net/tipc/netlink.c | 69 | ||||
| -rw-r--r-- | net/tipc/netlink.h | 11 | ||||
| -rw-r--r-- | net/tipc/netlink_compat.c | 2 | ||||
| -rw-r--r-- | net/tipc/node.c | 130 | ||||
| -rw-r--r-- | net/tipc/server.c | 4 | ||||
| -rw-r--r-- | net/tipc/socket.c | 9 | ||||
| -rw-r--r-- | net/tipc/subscr.c | 132 | ||||
| -rw-r--r-- | net/tipc/subscr.h | 11 | ||||
| -rw-r--r-- | net/tipc/udp_media.c | 44 | 
16 files changed, 331 insertions, 306 deletions
| diff --git a/net/tipc/bcast.c b/net/tipc/bcast.c index e401108360a2..ae469b37d852 100644 --- a/net/tipc/bcast.c +++ b/net/tipc/bcast.c @@ -412,11 +412,6 @@ enomem:  	return -ENOMEM;  } -void tipc_bcast_reinit(struct net *net) -{ -	tipc_link_reinit(tipc_bc_sndlink(net), tipc_own_addr(net)); -} -  void tipc_bcast_stop(struct net *net)  {  	struct tipc_net *tn = net_generic(net, tipc_net_id); diff --git a/net/tipc/bcast.h b/net/tipc/bcast.h index 1944c6c00bb9..d5e79b3767fd 100644 --- a/net/tipc/bcast.h +++ b/net/tipc/bcast.h @@ -46,7 +46,6 @@ struct tipc_node_map;  extern const char tipc_bclink_name[];  int tipc_bcast_init(struct net *net); -void tipc_bcast_reinit(struct net *net);  void tipc_bcast_stop(struct net *net);  void tipc_bcast_add_peer(struct net *net, struct tipc_link *l,  			 struct sk_buff_head *xmitq); diff --git a/net/tipc/bearer.c b/net/tipc/bearer.c index 802ffad3200d..27a5406213c6 100644 --- a/net/tipc/bearer.c +++ b/net/tipc/bearer.c @@ -40,6 +40,7 @@  #include "link.h"  #include "discover.h"  #include "bcast.h" +#include "netlink.h"  #define MAX_ADDR_STR 60 @@ -54,23 +55,6 @@ static struct tipc_media * const media_info_array[] = {  	NULL  }; -static const struct nla_policy -tipc_nl_bearer_policy[TIPC_NLA_BEARER_MAX + 1]	= { -	[TIPC_NLA_BEARER_UNSPEC]		= { .type = NLA_UNSPEC }, -	[TIPC_NLA_BEARER_NAME] = { -		.type = NLA_STRING, -		.len = TIPC_MAX_BEARER_NAME -	}, -	[TIPC_NLA_BEARER_PROP]			= { .type = NLA_NESTED }, -	[TIPC_NLA_BEARER_DOMAIN]		= { .type = NLA_U32 } -}; - -static const struct nla_policy tipc_nl_media_policy[TIPC_NLA_MEDIA_MAX + 1] = { -	[TIPC_NLA_MEDIA_UNSPEC]		= { .type = NLA_UNSPEC }, -	[TIPC_NLA_MEDIA_NAME]		= { .type = NLA_STRING }, -	[TIPC_NLA_MEDIA_PROP]		= { .type = NLA_NESTED } -}; -  static void bearer_disable(struct net *net, struct tipc_bearer *b);  /** diff --git a/net/tipc/link.c b/net/tipc/link.c index 347cdc99ed09..7d2bb3e70baa 100644 --- a/net/tipc/link.c +++ b/net/tipc/link.c @@ -1,7 +1,7 @@  /*   * net/tipc/link.c: TIPC link code   * - * Copyright (c) 1996-2007, 2012-2015, Ericsson AB + * Copyright (c) 1996-2007, 2012-2016, Ericsson AB   * Copyright (c) 2004-2007, 2010-2013, Wind River Systems   * All rights reserved.   * @@ -123,11 +123,11 @@ struct tipc_stats {  struct tipc_link {  	u32 addr;  	char name[TIPC_MAX_LINK_NAME]; -	struct tipc_media_addr *media_addr;  	struct net *net;  	/* Management and link supervision data */  	u32 peer_session; +	u32 session;  	u32 peer_bearer_id;  	u32 bearer_id;  	u32 tolerance; @@ -137,11 +137,7 @@ struct tipc_link {  	u16 peer_caps;  	bool active;  	u32 silent_intv_cnt; -	struct { -		unchar hdr[INT_H_SIZE]; -		unchar body[TIPC_MAX_IF_NAME]; -	} proto_msg; -	struct tipc_msg *pmsg; +	char if_name[TIPC_MAX_IF_NAME];  	u32 priority;  	char net_plane; @@ -196,14 +192,6 @@ struct tipc_link {  static const char *link_co_err = "Link tunneling error, ";  static const char *link_rst_msg = "Resetting link "; -/* Properties valid for media, bearar and link */ -static const struct nla_policy tipc_nl_prop_policy[TIPC_NLA_PROP_MAX + 1] = { -	[TIPC_NLA_PROP_UNSPEC]		= { .type = NLA_UNSPEC }, -	[TIPC_NLA_PROP_PRIO]		= { .type = NLA_U32 }, -	[TIPC_NLA_PROP_TOL]		= { .type = NLA_U32 }, -	[TIPC_NLA_PROP_WIN]		= { .type = NLA_U32 } -}; -  /* Send states for broadcast NACKs   */  enum { @@ -216,10 +204,11 @@ enum {   * Interval between NACKs when packets arrive out of order   */  #define TIPC_NACK_INTV (TIPC_MIN_LINK_WIN * 2) -/* - * Out-of-range value for link session numbers + +/* Wildcard value for link session numbers. When it is known that + * peer endpoint is down, any session number must be accepted.   */ -#define WILDCARD_SESSION 0x10000 +#define ANY_SESSION 0x10000  /* Link FSM states:   */ @@ -399,16 +388,6 @@ char *tipc_link_name(struct tipc_link *l)  	return l->name;  } -static u32 link_own_addr(struct tipc_link *l) -{ -	return msg_prevnode(l->pmsg); -} - -void tipc_link_reinit(struct tipc_link *l, u32 addr) -{ -	msg_set_prevnode(l->pmsg, addr); -} -  /**   * tipc_link_create - create a new link   * @n: pointer to associated node @@ -442,29 +421,22 @@ bool tipc_link_create(struct net *net, char *if_name, int bearer_id,  		      struct tipc_link **link)  {  	struct tipc_link *l; -	struct tipc_msg *hdr;  	l = kzalloc(sizeof(*l), GFP_ATOMIC);  	if (!l)  		return false;  	*link = l; -	l->pmsg = (struct tipc_msg *)&l->proto_msg; -	hdr = l->pmsg; -	tipc_msg_init(ownnode, hdr, LINK_PROTOCOL, RESET_MSG, INT_H_SIZE, peer); -	msg_set_size(hdr, sizeof(l->proto_msg)); -	msg_set_session(hdr, session); -	msg_set_bearer_id(hdr, l->bearer_id); +	l->session = session;  	/* Note: peer i/f name is completed by reset/activate message */  	sprintf(l->name, "%u.%u.%u:%s-%u.%u.%u:unknown",  		tipc_zone(ownnode), tipc_cluster(ownnode), tipc_node(ownnode),  		if_name, tipc_zone(peer), tipc_cluster(peer), tipc_node(peer)); -	strcpy((char *)msg_data(hdr), if_name); - +	strcpy(l->if_name, if_name);  	l->addr = peer;  	l->peer_caps = peer_caps;  	l->net = net; -	l->peer_session = WILDCARD_SESSION; +	l->peer_session = ANY_SESSION;  	l->bearer_id = bearer_id;  	l->tolerance = tolerance;  	l->net_plane = net_plane; @@ -791,7 +763,7 @@ static int link_schedule_user(struct tipc_link *link, struct sk_buff_head *list)  	struct tipc_msg *msg = buf_msg(skb_peek(list));  	int imp = msg_importance(msg);  	u32 oport = msg_origport(msg); -	u32 addr = link_own_addr(link); +	u32 addr = tipc_own_addr(link->net);  	struct sk_buff *skb;  	/* This really cannot happen...  */ @@ -840,16 +812,9 @@ void link_prepare_wakeup(struct tipc_link *l)  void tipc_link_reset(struct tipc_link *l)  { -	/* Link is down, accept any session */ -	l->peer_session = WILDCARD_SESSION; - -	/* If peer is up, it only accepts an incremented session number */ -	msg_set_session(l->pmsg, msg_session(l->pmsg) + 1); - -	/* Prepare for renewed mtu size negotiation */ +	l->peer_session = ANY_SESSION; +	l->session++;  	l->mtu = l->advertised_mtu; - -	/* Clean up all queues and counters: */  	__skb_queue_purge(&l->transmq);  	__skb_queue_purge(&l->deferdq);  	skb_queue_splice_init(&l->wakeupq, l->inputq); @@ -904,8 +869,10 @@ int tipc_link_xmit(struct tipc_link *l, struct sk_buff_head *list,  		if (unlikely(l->backlog[i].len >= l->backlog[i].limit))  			return link_schedule_user(l, list);  	} -	if (unlikely(msg_size(hdr) > mtu)) +	if (unlikely(msg_size(hdr) > mtu)) { +		skb_queue_purge(list);  		return -EMSGSIZE; +	}  	/* Prepare each packet for sending, and add to relevant queue: */  	while (skb_queue_len(list)) { @@ -917,8 +884,10 @@ int tipc_link_xmit(struct tipc_link *l, struct sk_buff_head *list,  		if (likely(skb_queue_len(transmq) < maxwin)) {  			_skb = skb_clone(skb, GFP_ATOMIC); -			if (!_skb) +			if (!_skb) { +				skb_queue_purge(list);  				return -ENOBUFS; +			}  			__skb_dequeue(list);  			__skb_queue_tail(transmq, skb);  			__skb_queue_tail(xmitq, _skb); @@ -1153,7 +1122,7 @@ int tipc_link_build_ack_msg(struct tipc_link *l, struct sk_buff_head *xmitq)  	/* Broadcast ACK must be sent via a unicast link => defer to caller */  	if (link_is_bc_rcvlink(l)) { -		if (((l->rcv_nxt ^ link_own_addr(l)) & 0xf) != 0xf) +		if (((l->rcv_nxt ^ tipc_own_addr(l->net)) & 0xf) != 0xf)  			return 0;  		l->rcv_unacked = 0;  		return TIPC_LINK_SND_BC_ACK; @@ -1261,39 +1230,34 @@ drop:  	return rc;  } -/* - * Send protocol message to the other endpoint. - */ -static void tipc_link_proto_xmit(struct tipc_link *l, u32 msg_typ, -				 int probe_msg, u32 gap, u32 tolerance, -				 u32 priority) -{ -	struct sk_buff *skb = NULL; -	struct sk_buff_head xmitq; - -	__skb_queue_head_init(&xmitq); -	tipc_link_build_proto_msg(l, msg_typ, probe_msg, gap, -				  tolerance, priority, &xmitq); -	skb = __skb_dequeue(&xmitq); -	if (!skb) -		return; -	tipc_bearer_xmit_skb(l->net, l->bearer_id, skb, l->media_addr); -	l->rcv_unacked = 0; -} -  static void tipc_link_build_proto_msg(struct tipc_link *l, int mtyp, bool probe,  				      u16 rcvgap, int tolerance, int priority,  				      struct sk_buff_head *xmitq)  { -	struct sk_buff *skb = NULL; -	struct tipc_msg *hdr = l->pmsg; +	struct sk_buff *skb; +	struct tipc_msg *hdr; +	struct sk_buff_head *dfq = &l->deferdq;  	bool node_up = link_is_up(l->bc_rcvlink);  	/* Don't send protocol message during reset or link failover */  	if (tipc_link_is_blocked(l))  		return; -	msg_set_type(hdr, mtyp); +	if (!tipc_link_is_up(l) && (mtyp == STATE_MSG)) +		return; + +	if (!skb_queue_empty(dfq)) +		rcvgap = buf_seqno(skb_peek(dfq)) - l->rcv_nxt; + +	skb = tipc_msg_create(LINK_PROTOCOL, mtyp, INT_H_SIZE, +			      TIPC_MAX_IF_NAME, l->addr, +			      tipc_own_addr(l->net), 0, 0, 0); +	if (!skb) +		return; + +	hdr = buf_msg(skb); +	msg_set_session(hdr, l->session); +	msg_set_bearer_id(hdr, l->bearer_id);  	msg_set_net_plane(hdr, l->net_plane);  	msg_set_next_sent(hdr, l->snd_nxt);  	msg_set_ack(hdr, l->rcv_nxt - 1); @@ -1303,36 +1267,23 @@ static void tipc_link_build_proto_msg(struct tipc_link *l, int mtyp, bool probe,  	msg_set_linkprio(hdr, priority);  	msg_set_redundant_link(hdr, node_up);  	msg_set_seq_gap(hdr, 0); - -	/* Compatibility: created msg must not be in sequence with pkt flow */  	msg_set_seqno(hdr, l->snd_nxt + U16_MAX / 2);  	if (mtyp == STATE_MSG) { -		if (!tipc_link_is_up(l)) -			return; - -		/* Override rcvgap if there are packets in deferred queue */ -		if (!skb_queue_empty(&l->deferdq)) -			rcvgap = buf_seqno(skb_peek(&l->deferdq)) - l->rcv_nxt; -		if (rcvgap) { -			msg_set_seq_gap(hdr, rcvgap); -			l->stats.sent_nacks++; -		} +		msg_set_seq_gap(hdr, rcvgap); +		msg_set_size(hdr, INT_H_SIZE);  		msg_set_probe(hdr, probe); -		if (probe) -			l->stats.sent_probes++;  		l->stats.sent_states++;  		l->rcv_unacked = 0;  	} else {  		/* RESET_MSG or ACTIVATE_MSG */  		msg_set_max_pkt(hdr, l->advertised_mtu); -		msg_set_ack(hdr, l->rcv_nxt - 1); -		msg_set_next_sent(hdr, 1); +		strcpy(msg_data(hdr), l->if_name);  	} -	skb = tipc_buf_acquire(msg_size(hdr)); -	if (!skb) -		return; -	skb_copy_to_linear_data(skb, hdr, msg_size(hdr)); +	if (probe) +		l->stats.sent_probes++; +	if (rcvgap) +		l->stats.sent_nacks++;  	skb->priority = TC_PRIO_CONTROL;  	__skb_queue_tail(xmitq, skb);  } @@ -1357,7 +1308,7 @@ void tipc_link_tnl_prepare(struct tipc_link *l, struct tipc_link *tnl,  	/* At least one packet required for safe algorithm => add dummy */  	skb = tipc_msg_create(TIPC_LOW_IMPORTANCE, TIPC_DIRECT_MSG, -			      BASIC_H_SIZE, 0, l->addr, link_own_addr(l), +			      BASIC_H_SIZE, 0, l->addr, tipc_own_addr(l->net),  			      0, 0, TIPC_ERR_NO_PORT);  	if (!skb) {  		pr_warn("%sunable to create tunnel packet\n", link_co_err); @@ -1368,7 +1319,7 @@ void tipc_link_tnl_prepare(struct tipc_link *l, struct tipc_link *tnl,  	__skb_queue_purge(&tmpxq);  	/* Initialize reusable tunnel packet header */ -	tipc_msg_init(link_own_addr(l), &tnlhdr, TUNNEL_PROTOCOL, +	tipc_msg_init(tipc_own_addr(l->net), &tnlhdr, TUNNEL_PROTOCOL,  		      mtyp, INT_H_SIZE, l->addr);  	pktcnt = skb_queue_len(&l->transmq) + skb_queue_len(&l->backlogq);  	msg_set_msgcnt(&tnlhdr, pktcnt); @@ -1427,7 +1378,7 @@ static int tipc_link_proto_rcv(struct tipc_link *l, struct sk_buff *skb,  	if (tipc_link_is_blocked(l) || !xmitq)  		goto exit; -	if (link_own_addr(l) > msg_prevnode(hdr)) +	if (tipc_own_addr(l->net) > msg_prevnode(hdr))  		l->net_plane = msg_net_plane(hdr);  	switch (mtyp) { @@ -1435,7 +1386,7 @@ static int tipc_link_proto_rcv(struct tipc_link *l, struct sk_buff *skb,  		/* Ignore duplicate RESET with old session number */  		if ((less_eq(msg_session(hdr), l->peer_session)) && -		    (l->peer_session != WILDCARD_SESSION)) +		    (l->peer_session != ANY_SESSION))  			break;  		/* fall thru' */ @@ -1479,6 +1430,12 @@ static int tipc_link_proto_rcv(struct tipc_link *l, struct sk_buff *skb,  		if (in_range(peers_tol, TIPC_MIN_LINK_TOL, TIPC_MAX_LINK_TOL))  			l->tolerance = peers_tol; +		if (peers_prio && in_range(peers_prio, TIPC_MIN_LINK_PRI, +					   TIPC_MAX_LINK_PRI)) { +			l->priority = peers_prio; +			rc = tipc_link_fsm_evt(l, LINK_FAILURE_EVT); +		} +  		l->silent_intv_cnt = 0;  		l->stats.recv_states++;  		if (msg_probe(hdr)) @@ -1526,7 +1483,7 @@ static bool tipc_link_build_bc_proto_msg(struct tipc_link *l, bool bcast,  	u16 gap_to = peers_snd_nxt - 1;  	skb = tipc_msg_create(BCAST_PROTOCOL, STATE_MSG, INT_H_SIZE, -			      0, l->addr, link_own_addr(l), 0, 0, 0); +			      0, l->addr, tipc_own_addr(l->net), 0, 0, 0);  	if (!skb)  		return false;  	hdr = buf_msg(skb); @@ -1681,7 +1638,7 @@ int tipc_link_bc_nack_rcv(struct tipc_link *l, struct sk_buff *skb,  	if (mtyp != STATE_MSG)  		return 0; -	if (dnode == link_own_addr(l)) { +	if (dnode == tipc_own_addr(l->net)) {  		tipc_link_bc_ack_rcv(l, acked, xmitq);  		rc = tipc_link_retrans(l->bc_sndlink, from, to, xmitq);  		l->stats.recv_nacks++; @@ -2023,16 +1980,18 @@ msg_full:  	return -EMSGSIZE;  } -void tipc_link_set_tolerance(struct tipc_link *l, u32 tol) +void tipc_link_set_tolerance(struct tipc_link *l, u32 tol, +			     struct sk_buff_head *xmitq)  {  	l->tolerance = tol; -	tipc_link_proto_xmit(l, STATE_MSG, 0, 0, tol, 0); +	tipc_link_build_proto_msg(l, STATE_MSG, 0, 0, tol, 0, xmitq);  } -void tipc_link_set_prio(struct tipc_link *l, u32 prio) +void tipc_link_set_prio(struct tipc_link *l, u32 prio, +			struct sk_buff_head *xmitq)  {  	l->priority = prio; -	tipc_link_proto_xmit(l, STATE_MSG, 0, 0, 0, prio); +	tipc_link_build_proto_msg(l, STATE_MSG, 0, 0, 0, prio, xmitq);  }  void tipc_link_set_abort_limit(struct tipc_link *l, u32 limit) diff --git a/net/tipc/link.h b/net/tipc/link.h index b2ae0f4276af..6a94175ee20a 100644 --- a/net/tipc/link.h +++ b/net/tipc/link.h @@ -86,7 +86,6 @@ bool tipc_link_bc_create(struct net *net, u32 ownnode, u32 peer,  			 struct sk_buff_head *namedq,  			 struct tipc_link *bc_sndlink,  			 struct tipc_link **link); -void tipc_link_reinit(struct tipc_link *l, u32 addr);  void tipc_link_tnl_prepare(struct tipc_link *l, struct tipc_link *tnl,  			   int mtyp, struct sk_buff_head *xmitq);  void tipc_link_build_reset_msg(struct tipc_link *l, struct sk_buff_head *xmitq); @@ -112,8 +111,10 @@ char tipc_link_plane(struct tipc_link *l);  int tipc_link_prio(struct tipc_link *l);  int tipc_link_window(struct tipc_link *l);  unsigned long tipc_link_tolerance(struct tipc_link *l); -void tipc_link_set_tolerance(struct tipc_link *l, u32 tol); -void tipc_link_set_prio(struct tipc_link *l, u32 prio); +void tipc_link_set_tolerance(struct tipc_link *l, u32 tol, +			     struct sk_buff_head *xmitq); +void tipc_link_set_prio(struct tipc_link *l, u32 prio, +			struct sk_buff_head *xmitq);  void tipc_link_set_abort_limit(struct tipc_link *l, u32 limit);  void tipc_link_set_queue_limits(struct tipc_link *l, u32 window);  int __tipc_nl_add_link(struct net *net, struct tipc_nl_msg *msg, diff --git a/net/tipc/name_table.c b/net/tipc/name_table.c index 91fce70291a8..e190460fe0d3 100644 --- a/net/tipc/name_table.c +++ b/net/tipc/name_table.c @@ -47,12 +47,6 @@  #define TIPC_NAMETBL_SIZE 1024		/* must be a power of 2 */ -static const struct nla_policy -tipc_nl_name_table_policy[TIPC_NLA_NAME_TABLE_MAX + 1] = { -	[TIPC_NLA_NAME_TABLE_UNSPEC]	= { .type = NLA_UNSPEC }, -	[TIPC_NLA_NAME_TABLE_PUBL]	= { .type = NLA_NESTED } -}; -  /**   * struct name_info - name sequence publication info   * @node_list: circular list of publications made by own node @@ -418,6 +412,9 @@ static void tipc_nameseq_subscribe(struct name_seq *nseq,  				   struct tipc_subscription *s)  {  	struct sub_seq *sseq = nseq->sseqs; +	struct tipc_name_seq ns; + +	tipc_subscrp_convert_seq(&s->evt.s.seq, s->swap, &ns);  	list_add(&s->nameseq_list, &nseq->subscriptions); @@ -425,7 +422,7 @@ static void tipc_nameseq_subscribe(struct name_seq *nseq,  		return;  	while (sseq != &nseq->sseqs[nseq->first_free]) { -		if (tipc_subscrp_check_overlap(s, sseq->lower, sseq->upper)) { +		if (tipc_subscrp_check_overlap(&ns, sseq->lower, sseq->upper)) {  			struct publication *crs;  			struct name_info *info = sseq->info;  			int must_report = 1; @@ -722,9 +719,10 @@ int tipc_nametbl_withdraw(struct net *net, u32 type, u32 lower, u32 ref,  void tipc_nametbl_subscribe(struct tipc_subscription *s)  {  	struct tipc_net *tn = net_generic(s->net, tipc_net_id); -	u32 type = s->seq.type; +	u32 type = tipc_subscrp_convert_seq_type(s->evt.s.seq.type, s->swap);  	int index = hash(type);  	struct name_seq *seq; +	struct tipc_name_seq ns;  	spin_lock_bh(&tn->nametbl_lock);  	seq = nametbl_find_seq(s->net, type); @@ -735,8 +733,9 @@ void tipc_nametbl_subscribe(struct tipc_subscription *s)  		tipc_nameseq_subscribe(seq, s);  		spin_unlock_bh(&seq->lock);  	} else { +		tipc_subscrp_convert_seq(&s->evt.s.seq, s->swap, &ns);  		pr_warn("Failed to create subscription for {%u,%u,%u}\n", -			s->seq.type, s->seq.lower, s->seq.upper); +			ns.type, ns.lower, ns.upper);  	}  	spin_unlock_bh(&tn->nametbl_lock);  } @@ -748,9 +747,10 @@ void tipc_nametbl_unsubscribe(struct tipc_subscription *s)  {  	struct tipc_net *tn = net_generic(s->net, tipc_net_id);  	struct name_seq *seq; +	u32 type = tipc_subscrp_convert_seq_type(s->evt.s.seq.type, s->swap);  	spin_lock_bh(&tn->nametbl_lock); -	seq = nametbl_find_seq(s->net, s->seq.type); +	seq = nametbl_find_seq(s->net, type);  	if (seq != NULL) {  		spin_lock_bh(&seq->lock);  		list_del_init(&s->nameseq_list); diff --git a/net/tipc/net.c b/net/tipc/net.c index 77bf9113c7a7..28bf4feeb81c 100644 --- a/net/tipc/net.c +++ b/net/tipc/net.c @@ -41,11 +41,7 @@  #include "socket.h"  #include "node.h"  #include "bcast.h" - -static const struct nla_policy tipc_nl_net_policy[TIPC_NLA_NET_MAX + 1] = { -	[TIPC_NLA_NET_UNSPEC]	= { .type = NLA_UNSPEC }, -	[TIPC_NLA_NET_ID]	= { .type = NLA_U32 } -}; +#include "netlink.h"  /*   * The TIPC locking policy is designed to ensure a very fine locking @@ -116,7 +112,6 @@ int tipc_net_start(struct net *net, u32 addr)  	tn->own_addr = addr;  	tipc_named_reinit(net);  	tipc_sk_reinit(net); -	tipc_bcast_reinit(net);  	tipc_nametbl_publish(net, TIPC_CFG_SRV, tn->own_addr, tn->own_addr,  			     TIPC_ZONE_SCOPE, 0, tn->own_addr); diff --git a/net/tipc/netlink.c b/net/tipc/netlink.c index 8975b0135b76..56935df2167a 100644 --- a/net/tipc/netlink.c +++ b/net/tipc/netlink.c @@ -55,6 +55,75 @@ static const struct nla_policy tipc_nl_policy[TIPC_NLA_MAX + 1] = {  	[TIPC_NLA_NAME_TABLE]	= { .type = NLA_NESTED, }  }; +const struct nla_policy +tipc_nl_name_table_policy[TIPC_NLA_NAME_TABLE_MAX + 1] = { +	[TIPC_NLA_NAME_TABLE_UNSPEC]	= { .type = NLA_UNSPEC }, +	[TIPC_NLA_NAME_TABLE_PUBL]	= { .type = NLA_NESTED } +}; + +const struct nla_policy tipc_nl_sock_policy[TIPC_NLA_SOCK_MAX + 1] = { +	[TIPC_NLA_SOCK_UNSPEC]		= { .type = NLA_UNSPEC }, +	[TIPC_NLA_SOCK_ADDR]		= { .type = NLA_U32 }, +	[TIPC_NLA_SOCK_REF]		= { .type = NLA_U32 }, +	[TIPC_NLA_SOCK_CON]		= { .type = NLA_NESTED }, +	[TIPC_NLA_SOCK_HAS_PUBL]	= { .type = NLA_FLAG } +}; + +const struct nla_policy tipc_nl_net_policy[TIPC_NLA_NET_MAX + 1] = { +	[TIPC_NLA_NET_UNSPEC]		= { .type = NLA_UNSPEC }, +	[TIPC_NLA_NET_ID]		= { .type = NLA_U32 } +}; + +const struct nla_policy tipc_nl_link_policy[TIPC_NLA_LINK_MAX + 1] = { +	[TIPC_NLA_LINK_UNSPEC]		= { .type = NLA_UNSPEC }, +	[TIPC_NLA_LINK_NAME]		= { .type = NLA_STRING, +					    .len = TIPC_MAX_LINK_NAME }, +	[TIPC_NLA_LINK_MTU]		= { .type = NLA_U32 }, +	[TIPC_NLA_LINK_BROADCAST]	= { .type = NLA_FLAG }, +	[TIPC_NLA_LINK_UP]		= { .type = NLA_FLAG }, +	[TIPC_NLA_LINK_ACTIVE]		= { .type = NLA_FLAG }, +	[TIPC_NLA_LINK_PROP]		= { .type = NLA_NESTED }, +	[TIPC_NLA_LINK_STATS]		= { .type = NLA_NESTED }, +	[TIPC_NLA_LINK_RX]		= { .type = NLA_U32 }, +	[TIPC_NLA_LINK_TX]		= { .type = NLA_U32 } +}; + +const struct nla_policy tipc_nl_node_policy[TIPC_NLA_NODE_MAX + 1] = { +	[TIPC_NLA_NODE_UNSPEC]		= { .type = NLA_UNSPEC }, +	[TIPC_NLA_NODE_ADDR]		= { .type = NLA_U32 }, +	[TIPC_NLA_NODE_UP]		= { .type = NLA_FLAG } +}; + +/* Properties valid for media, bearer and link */ +const struct nla_policy tipc_nl_prop_policy[TIPC_NLA_PROP_MAX + 1] = { +	[TIPC_NLA_PROP_UNSPEC]		= { .type = NLA_UNSPEC }, +	[TIPC_NLA_PROP_PRIO]		= { .type = NLA_U32 }, +	[TIPC_NLA_PROP_TOL]		= { .type = NLA_U32 }, +	[TIPC_NLA_PROP_WIN]		= { .type = NLA_U32 } +}; + +const struct nla_policy tipc_nl_bearer_policy[TIPC_NLA_BEARER_MAX + 1]	= { +	[TIPC_NLA_BEARER_UNSPEC]	= { .type = NLA_UNSPEC }, +	[TIPC_NLA_BEARER_NAME]		= { .type = NLA_STRING, +					    .len = TIPC_MAX_BEARER_NAME }, +	[TIPC_NLA_BEARER_PROP]		= { .type = NLA_NESTED }, +	[TIPC_NLA_BEARER_DOMAIN]	= { .type = NLA_U32 } +}; + +const struct nla_policy tipc_nl_media_policy[TIPC_NLA_MEDIA_MAX + 1] = { +	[TIPC_NLA_MEDIA_UNSPEC]		= { .type = NLA_UNSPEC }, +	[TIPC_NLA_MEDIA_NAME]		= { .type = NLA_STRING }, +	[TIPC_NLA_MEDIA_PROP]		= { .type = NLA_NESTED } +}; + +const struct nla_policy tipc_nl_udp_policy[TIPC_NLA_UDP_MAX + 1] = { +	[TIPC_NLA_UDP_UNSPEC]	= {.type = NLA_UNSPEC}, +	[TIPC_NLA_UDP_LOCAL]	= {.type = NLA_BINARY, +				   .len = sizeof(struct sockaddr_storage)}, +	[TIPC_NLA_UDP_REMOTE]	= {.type = NLA_BINARY, +				   .len = sizeof(struct sockaddr_storage)}, +}; +  /* Users of the legacy API (tipc-config) can't handle that we add operations,   * so we have a separate genl handling for the new API.   */ diff --git a/net/tipc/netlink.h b/net/tipc/netlink.h index 08a1db67b927..ed1dbcb4afbd 100644 --- a/net/tipc/netlink.h +++ b/net/tipc/netlink.h @@ -35,6 +35,7 @@  #ifndef _TIPC_NETLINK_H  #define _TIPC_NETLINK_H +#include <net/netlink.h>  extern struct genl_family tipc_genl_family;  int tipc_nlmsg_parse(const struct nlmsghdr *nlh, struct nlattr ***buf); @@ -45,6 +46,16 @@ struct tipc_nl_msg {  	u32 seq;  }; +extern const struct nla_policy tipc_nl_name_table_policy[]; +extern const struct nla_policy tipc_nl_sock_policy[]; +extern const struct nla_policy tipc_nl_net_policy[]; +extern const struct nla_policy tipc_nl_link_policy[]; +extern const struct nla_policy tipc_nl_node_policy[]; +extern const struct nla_policy tipc_nl_prop_policy[]; +extern const struct nla_policy tipc_nl_bearer_policy[]; +extern const struct nla_policy tipc_nl_media_policy[]; +extern const struct nla_policy tipc_nl_udp_policy[]; +  int tipc_netlink_start(void);  int tipc_netlink_compat_start(void);  void tipc_netlink_stop(void); diff --git a/net/tipc/netlink_compat.c b/net/tipc/netlink_compat.c index 2c016fdefe97..d7d050f44fc1 100644 --- a/net/tipc/netlink_compat.c +++ b/net/tipc/netlink_compat.c @@ -1104,8 +1104,8 @@ static int tipc_nl_compat_recv(struct sk_buff *skb, struct genl_info *info)  	req_nlh = (struct nlmsghdr *)skb->data;  	msg.req = nlmsg_data(req_nlh) + GENL_HDRLEN + TIPC_GENL_HDRLEN;  	msg.cmd = req_userhdr->cmd; -	msg.dst_sk = info->dst_sk;  	msg.net = genl_info_net(info); +	msg.dst_sk = skb->sk;  	if ((msg.cmd & 0xC000) && (!netlink_net_capable(skb, CAP_NET_ADMIN))) {  		msg.rep = tipc_get_err_tlv(TIPC_CFG_NOT_NET_ADMIN); diff --git a/net/tipc/node.c b/net/tipc/node.c index 9d7a16fc5ca4..ace178fd3850 100644 --- a/net/tipc/node.c +++ b/net/tipc/node.c @@ -41,6 +41,7 @@  #include "socket.h"  #include "bcast.h"  #include "discover.h" +#include "netlink.h"  #define INVALID_NODE_SIG	0x10000 @@ -164,28 +165,6 @@ struct tipc_sock_conn {  	struct list_head list;  }; -static const struct nla_policy tipc_nl_link_policy[TIPC_NLA_LINK_MAX + 1] = { -	[TIPC_NLA_LINK_UNSPEC]		= { .type = NLA_UNSPEC }, -	[TIPC_NLA_LINK_NAME] = { -		.type = NLA_STRING, -		.len = TIPC_MAX_LINK_NAME -	}, -	[TIPC_NLA_LINK_MTU]		= { .type = NLA_U32 }, -	[TIPC_NLA_LINK_BROADCAST]	= { .type = NLA_FLAG }, -	[TIPC_NLA_LINK_UP]		= { .type = NLA_FLAG }, -	[TIPC_NLA_LINK_ACTIVE]		= { .type = NLA_FLAG }, -	[TIPC_NLA_LINK_PROP]		= { .type = NLA_NESTED }, -	[TIPC_NLA_LINK_STATS]		= { .type = NLA_NESTED }, -	[TIPC_NLA_LINK_RX]		= { .type = NLA_U32 }, -	[TIPC_NLA_LINK_TX]		= { .type = NLA_U32 } -}; - -static const struct nla_policy tipc_nl_node_policy[TIPC_NLA_NODE_MAX + 1] = { -	[TIPC_NLA_NODE_UNSPEC]		= { .type = NLA_UNSPEC }, -	[TIPC_NLA_NODE_ADDR]		= { .type = NLA_U32 }, -	[TIPC_NLA_NODE_UP]		= { .type = NLA_FLAG } -}; -  static struct tipc_link *node_active_link(struct tipc_node *n, int sel)  {  	int bearer_id = n->active_links[sel & 1]; @@ -225,9 +204,10 @@ static unsigned int tipc_hashfn(u32 addr)  static void tipc_node_kref_release(struct kref *kref)  { -	struct tipc_node *node = container_of(kref, struct tipc_node, kref); +	struct tipc_node *n = container_of(kref, struct tipc_node, kref); -	tipc_node_delete(node); +	kfree(n->bc_entry.link); +	kfree_rcu(n, rcu);  }  static void tipc_node_put(struct tipc_node *node) @@ -245,23 +225,23 @@ static void tipc_node_get(struct tipc_node *node)   */  static struct tipc_node *tipc_node_find(struct net *net, u32 addr)  { -	struct tipc_net *tn = net_generic(net, tipc_net_id); +	struct tipc_net *tn = tipc_net(net);  	struct tipc_node *node; +	unsigned int thash = tipc_hashfn(addr);  	if (unlikely(!in_own_cluster_exact(net, addr)))  		return NULL;  	rcu_read_lock(); -	hlist_for_each_entry_rcu(node, &tn->node_htable[tipc_hashfn(addr)], -				 hash) { -		if (node->addr == addr) { -			tipc_node_get(node); -			rcu_read_unlock(); -			return node; -		} +	hlist_for_each_entry_rcu(node, &tn->node_htable[thash], hash) { +		if (node->addr != addr) +			continue; +		if (!kref_get_unless_zero(&node->kref)) +			node = NULL; +		break;  	}  	rcu_read_unlock(); -	return NULL; +	return node;  }  static void tipc_node_read_lock(struct tipc_node *n) @@ -395,21 +375,20 @@ static void tipc_node_delete(struct tipc_node *node)  {  	list_del_rcu(&node->list);  	hlist_del_rcu(&node->hash); -	kfree(node->bc_entry.link); -	kfree_rcu(node, rcu); +	tipc_node_put(node); + +	del_timer_sync(&node->timer); +	tipc_node_put(node);  }  void tipc_node_stop(struct net *net)  { -	struct tipc_net *tn = net_generic(net, tipc_net_id); +	struct tipc_net *tn = tipc_net(net);  	struct tipc_node *node, *t_node;  	spin_lock_bh(&tn->node_list_lock); -	list_for_each_entry_safe(node, t_node, &tn->node_list, list) { -		if (del_timer(&node->timer)) -			tipc_node_put(node); -		tipc_node_put(node); -	} +	list_for_each_entry_safe(node, t_node, &tn->node_list, list) +		tipc_node_delete(node);  	spin_unlock_bh(&tn->node_list_lock);  } @@ -530,9 +509,7 @@ static void tipc_node_timeout(unsigned long data)  		if (rc & TIPC_LINK_DOWN_EVT)  			tipc_node_link_down(n, bearer_id, false);  	} -	if (!mod_timer(&n->timer, jiffies + n->keepalive_intv)) -		tipc_node_get(n); -	tipc_node_put(n); +	mod_timer(&n->timer, jiffies + n->keepalive_intv);  }  /** @@ -845,7 +822,7 @@ void tipc_node_check_dest(struct net *net, u32 onode,  	memcpy(&le->maddr, maddr, sizeof(*maddr));  exit:  	tipc_node_write_unlock(n); -	if (reset && !tipc_link_is_reset(l)) +	if (reset && l && !tipc_link_is_reset(l))  		tipc_node_link_down(n, b->identity, false);  	tipc_node_put(n);  } @@ -1166,7 +1143,7 @@ msg_full:   * @dnode: address of destination node   * @selector: a number used for deterministic link selection   * Consumes the buffer chain, except when returning -ELINKCONG - * Returns 0 if success, otherwise errno: -ELINKCONG,-EHOSTUNREACH,-EMSGSIZE + * Returns 0 if success, otherwise: -ELINKCONG,-EHOSTUNREACH,-EMSGSIZE,-ENOBUF   */  int tipc_node_xmit(struct net *net, struct sk_buff_head *list,  		   u32 dnode, int selector) @@ -1174,33 +1151,43 @@ int tipc_node_xmit(struct net *net, struct sk_buff_head *list,  	struct tipc_link_entry *le = NULL;  	struct tipc_node *n;  	struct sk_buff_head xmitq; -	int bearer_id = -1; -	int rc = -EHOSTUNREACH; +	int bearer_id; +	int rc; + +	if (in_own_node(net, dnode)) { +		tipc_sk_rcv(net, list); +		return 0; +	} -	__skb_queue_head_init(&xmitq);  	n = tipc_node_find(net, dnode); -	if (likely(n)) { -		tipc_node_read_lock(n); -		bearer_id = n->active_links[selector & 1]; -		if (bearer_id >= 0) { -			le = &n->links[bearer_id]; -			spin_lock_bh(&le->lock); -			rc = tipc_link_xmit(le->link, list, &xmitq); -			spin_unlock_bh(&le->lock); -		} +	if (unlikely(!n)) { +		skb_queue_purge(list); +		return -EHOSTUNREACH; +	} + +	tipc_node_read_lock(n); +	bearer_id = n->active_links[selector & 1]; +	if (unlikely(bearer_id == INVALID_BEARER_ID)) {  		tipc_node_read_unlock(n); -		if (likely(!rc)) -			tipc_bearer_xmit(net, bearer_id, &xmitq, &le->maddr); -		else if (rc == -ENOBUFS) -			tipc_node_link_down(n, bearer_id, false);  		tipc_node_put(n); -		return rc; +		skb_queue_purge(list); +		return -EHOSTUNREACH;  	} -	if (likely(in_own_node(net, dnode))) { -		tipc_sk_rcv(net, list); -		return 0; -	} +	__skb_queue_head_init(&xmitq); +	le = &n->links[bearer_id]; +	spin_lock_bh(&le->lock); +	rc = tipc_link_xmit(le->link, list, &xmitq); +	spin_unlock_bh(&le->lock); +	tipc_node_read_unlock(n); + +	if (likely(rc == 0)) +		tipc_bearer_xmit(net, bearer_id, &xmitq, &le->maddr); +	else if (rc == -ENOBUFS) +		tipc_node_link_down(n, bearer_id, false); + +	tipc_node_put(n); +  	return rc;  } @@ -1637,9 +1624,12 @@ int tipc_nl_node_set_link(struct sk_buff *skb, struct genl_info *info)  	char *name;  	struct tipc_link *link;  	struct tipc_node *node; +	struct sk_buff_head xmitq;  	struct nlattr *attrs[TIPC_NLA_LINK_MAX + 1];  	struct net *net = sock_net(skb->sk); +	__skb_queue_head_init(&xmitq); +  	if (!info->attrs[TIPC_NLA_LINK])  		return -EINVAL; @@ -1683,13 +1673,13 @@ int tipc_nl_node_set_link(struct sk_buff *skb, struct genl_info *info)  			u32 tol;  			tol = nla_get_u32(props[TIPC_NLA_PROP_TOL]); -			tipc_link_set_tolerance(link, tol); +			tipc_link_set_tolerance(link, tol, &xmitq);  		}  		if (props[TIPC_NLA_PROP_PRIO]) {  			u32 prio;  			prio = nla_get_u32(props[TIPC_NLA_PROP_PRIO]); -			tipc_link_set_prio(link, prio); +			tipc_link_set_prio(link, prio, &xmitq);  		}  		if (props[TIPC_NLA_PROP_WIN]) {  			u32 win; @@ -1701,7 +1691,7 @@ int tipc_nl_node_set_link(struct sk_buff *skb, struct genl_info *info)  out:  	tipc_node_read_unlock(node); - +	tipc_bearer_xmit(net, bearer_id, &xmitq, &node->links[bearer_id].maddr);  	return res;  } diff --git a/net/tipc/server.c b/net/tipc/server.c index 922e04a43396..2446bfbaa309 100644 --- a/net/tipc/server.c +++ b/net/tipc/server.c @@ -571,13 +571,13 @@ static void tipc_work_stop(struct tipc_server *s)  static int tipc_work_start(struct tipc_server *s)  { -	s->rcv_wq = alloc_workqueue("tipc_rcv", WQ_UNBOUND, 1); +	s->rcv_wq = alloc_ordered_workqueue("tipc_rcv", 0);  	if (!s->rcv_wq) {  		pr_err("can't start tipc receive workqueue\n");  		return -ENOMEM;  	} -	s->send_wq = alloc_workqueue("tipc_send", WQ_UNBOUND, 1); +	s->send_wq = alloc_ordered_workqueue("tipc_send", 0);  	if (!s->send_wq) {  		pr_err("can't start tipc send workqueue\n");  		destroy_workqueue(s->rcv_wq); diff --git a/net/tipc/socket.c b/net/tipc/socket.c index 4d420bb27396..3eeb50a27b89 100644 --- a/net/tipc/socket.c +++ b/net/tipc/socket.c @@ -42,6 +42,7 @@  #include "name_distr.h"  #include "socket.h"  #include "bcast.h" +#include "netlink.h"  #define SS_LISTENING		-1	/* socket is listening */  #define SS_READY		-2	/* socket is connectionless */ @@ -126,14 +127,6 @@ static const struct proto_ops stream_ops;  static const struct proto_ops msg_ops;  static struct proto tipc_proto; -static const struct nla_policy tipc_nl_sock_policy[TIPC_NLA_SOCK_MAX + 1] = { -	[TIPC_NLA_SOCK_UNSPEC]		= { .type = NLA_UNSPEC }, -	[TIPC_NLA_SOCK_ADDR]		= { .type = NLA_U32 }, -	[TIPC_NLA_SOCK_REF]		= { .type = NLA_U32 }, -	[TIPC_NLA_SOCK_CON]		= { .type = NLA_NESTED }, -	[TIPC_NLA_SOCK_HAS_PUBL]	= { .type = NLA_FLAG } -}; -  static const struct rhashtable_params tsk_rht_params;  /* diff --git a/net/tipc/subscr.c b/net/tipc/subscr.c index f9ff73a8d815..e6cb386fbf34 100644 --- a/net/tipc/subscr.c +++ b/net/tipc/subscr.c @@ -92,25 +92,42 @@ static void tipc_subscrp_send_event(struct tipc_subscription *sub,   *   * Returns 1 if there is overlap, otherwise 0.   */ -int tipc_subscrp_check_overlap(struct tipc_subscription *sub, u32 found_lower, +int tipc_subscrp_check_overlap(struct tipc_name_seq *seq, u32 found_lower,  			       u32 found_upper)  { -	if (found_lower < sub->seq.lower) -		found_lower = sub->seq.lower; -	if (found_upper > sub->seq.upper) -		found_upper = sub->seq.upper; +	if (found_lower < seq->lower) +		found_lower = seq->lower; +	if (found_upper > seq->upper) +		found_upper = seq->upper;  	if (found_lower > found_upper)  		return 0;  	return 1;  } +u32 tipc_subscrp_convert_seq_type(u32 type, int swap) +{ +	return htohl(type, swap); +} + +void tipc_subscrp_convert_seq(struct tipc_name_seq *in, int swap, +			      struct tipc_name_seq *out) +{ +	out->type = htohl(in->type, swap); +	out->lower = htohl(in->lower, swap); +	out->upper = htohl(in->upper, swap); +} +  void tipc_subscrp_report_overlap(struct tipc_subscription *sub, u32 found_lower,  				 u32 found_upper, u32 event, u32 port_ref,  				 u32 node, int must)  { -	if (!tipc_subscrp_check_overlap(sub, found_lower, found_upper)) +	struct tipc_name_seq seq; + +	tipc_subscrp_convert_seq(&sub->evt.s.seq, sub->swap, &seq); +	if (!tipc_subscrp_check_overlap(&seq, found_lower, found_upper))  		return; -	if (!must && !(sub->filter & TIPC_SUB_PORTS)) +	if (!must && +	    !(htohl(sub->evt.s.filter, sub->swap) & TIPC_SUB_PORTS))  		return;  	tipc_subscrp_send_event(sub, found_lower, found_upper, event, port_ref, @@ -171,12 +188,14 @@ static struct tipc_subscriber *tipc_subscrb_create(int conid)  static void tipc_subscrb_delete(struct tipc_subscriber *subscriber)  {  	struct tipc_subscription *sub, *temp; +	u32 timeout;  	spin_lock_bh(&subscriber->lock);  	/* Destroy any existing subscriptions for subscriber */  	list_for_each_entry_safe(sub, temp, &subscriber->subscrp_list,  				 subscrp_list) { -		if (del_timer(&sub->timer)) { +		timeout = htohl(sub->evt.s.timeout, sub->swap); +		if ((timeout == TIPC_WAIT_FOREVER) || del_timer(&sub->timer)) {  			tipc_subscrp_delete(sub);  			tipc_subscrb_put(subscriber);  		} @@ -200,13 +219,16 @@ static void tipc_subscrp_cancel(struct tipc_subscr *s,  				struct tipc_subscriber *subscriber)  {  	struct tipc_subscription *sub, *temp; +	u32 timeout;  	spin_lock_bh(&subscriber->lock);  	/* Find first matching subscription, exit if not found */  	list_for_each_entry_safe(sub, temp, &subscriber->subscrp_list,  				 subscrp_list) {  		if (!memcmp(s, &sub->evt.s, sizeof(struct tipc_subscr))) { -			if (del_timer(&sub->timer)) { +			timeout = htohl(sub->evt.s.timeout, sub->swap); +			if ((timeout == TIPC_WAIT_FOREVER) || +			    del_timer(&sub->timer)) {  				tipc_subscrp_delete(sub);  				tipc_subscrb_put(subscriber);  			} @@ -216,66 +238,67 @@ static void tipc_subscrp_cancel(struct tipc_subscr *s,  	spin_unlock_bh(&subscriber->lock);  } -static int tipc_subscrp_create(struct net *net, struct tipc_subscr *s, -			       struct tipc_subscriber *subscriber, -			       struct tipc_subscription **sub_p) +static struct tipc_subscription *tipc_subscrp_create(struct net *net, +						     struct tipc_subscr *s, +						     int swap)  {  	struct tipc_net *tn = net_generic(net, tipc_net_id);  	struct tipc_subscription *sub; -	int swap; - -	/* Determine subscriber's endianness */ -	swap = !(s->filter & (TIPC_SUB_PORTS | TIPC_SUB_SERVICE)); - -	/* Detect & process a subscription cancellation request */ -	if (s->filter & htohl(TIPC_SUB_CANCEL, swap)) { -		s->filter &= ~htohl(TIPC_SUB_CANCEL, swap); -		tipc_subscrp_cancel(s, subscriber); -		return 0; -	} +	u32 filter = htohl(s->filter, swap);  	/* Refuse subscription if global limit exceeded */  	if (atomic_read(&tn->subscription_count) >= TIPC_MAX_SUBSCRIPTIONS) {  		pr_warn("Subscription rejected, limit reached (%u)\n",  			TIPC_MAX_SUBSCRIPTIONS); -		return -EINVAL; +		return NULL;  	}  	/* Allocate subscription object */  	sub = kmalloc(sizeof(*sub), GFP_ATOMIC);  	if (!sub) {  		pr_warn("Subscription rejected, no memory\n"); -		return -ENOMEM; +		return NULL;  	}  	/* Initialize subscription object */  	sub->net = net; -	sub->seq.type = htohl(s->seq.type, swap); -	sub->seq.lower = htohl(s->seq.lower, swap); -	sub->seq.upper = htohl(s->seq.upper, swap); -	sub->timeout = msecs_to_jiffies(htohl(s->timeout, swap)); -	sub->filter = htohl(s->filter, swap); -	if ((!(sub->filter & TIPC_SUB_PORTS) == -	     !(sub->filter & TIPC_SUB_SERVICE)) || -	    (sub->seq.lower > sub->seq.upper)) { +	if (((filter & TIPC_SUB_PORTS) && (filter & TIPC_SUB_SERVICE)) || +	    (htohl(s->seq.lower, swap) > htohl(s->seq.upper, swap))) {  		pr_warn("Subscription rejected, illegal request\n");  		kfree(sub); -		return -EINVAL; +		return NULL;  	} -	spin_lock_bh(&subscriber->lock); -	list_add(&sub->subscrp_list, &subscriber->subscrp_list); -	spin_unlock_bh(&subscriber->lock); -	sub->subscriber = subscriber; +  	sub->swap = swap;  	memcpy(&sub->evt.s, s, sizeof(*s));  	atomic_inc(&tn->subscription_count); +	return sub; +} + +static void tipc_subscrp_subscribe(struct net *net, struct tipc_subscr *s, +				   struct tipc_subscriber *subscriber, int swap) +{ +	struct tipc_net *tn = net_generic(net, tipc_net_id); +	struct tipc_subscription *sub = NULL; +	u32 timeout; + +	sub = tipc_subscrp_create(net, s, swap); +	if (!sub) +		return tipc_conn_terminate(tn->topsrv, subscriber->conid); + +	spin_lock_bh(&subscriber->lock); +	list_add(&sub->subscrp_list, &subscriber->subscrp_list); +	tipc_subscrb_get(subscriber); +	sub->subscriber = subscriber; +	tipc_nametbl_subscribe(sub); +	spin_unlock_bh(&subscriber->lock); + +	timeout = htohl(sub->evt.s.timeout, swap); +	if (timeout == TIPC_WAIT_FOREVER) +		return; +  	setup_timer(&sub->timer, tipc_subscrp_timeout, (unsigned long)sub); -	if (sub->timeout != TIPC_WAIT_FOREVER) -		sub->timeout += jiffies; -	if (!mod_timer(&sub->timer, sub->timeout)) -		tipc_subscrb_get(subscriber); -	*sub_p = sub; -	return 0; +	mod_timer(&sub->timer, jiffies + msecs_to_jiffies(timeout));  }  /* Handle one termination request for the subscriber */ @@ -289,15 +312,22 @@ static void tipc_subscrb_rcv_cb(struct net *net, int conid,  				struct sockaddr_tipc *addr, void *usr_data,  				void *buf, size_t len)  { -	struct tipc_subscriber *subscrb = usr_data; -	struct tipc_subscription *sub = NULL; -	struct tipc_net *tn = net_generic(net, tipc_net_id); +	struct tipc_subscriber *subscriber = usr_data; +	struct tipc_subscr *s = (struct tipc_subscr *)buf; +	int swap; -	if (tipc_subscrp_create(net, (struct tipc_subscr *)buf, subscrb, &sub)) -		return tipc_conn_terminate(tn->topsrv, subscrb->conid); +	/* Determine subscriber's endianness */ +	swap = !(s->filter & (TIPC_SUB_PORTS | TIPC_SUB_SERVICE | +			      TIPC_SUB_CANCEL)); + +	/* Detect & process a subscription cancellation request */ +	if (s->filter & htohl(TIPC_SUB_CANCEL, swap)) { +		s->filter &= ~htohl(TIPC_SUB_CANCEL, swap); +		return tipc_subscrp_cancel(s, subscriber); +	} -	if (sub) -		tipc_nametbl_subscribe(sub); +	if (s) +		tipc_subscrp_subscribe(net, s, subscriber, swap);  }  /* Handle one request to establish a new subscriber */ diff --git a/net/tipc/subscr.h b/net/tipc/subscr.h index 92ee18cc5fe6..be60103082c9 100644 --- a/net/tipc/subscr.h +++ b/net/tipc/subscr.h @@ -50,21 +50,15 @@ struct tipc_subscriber;   * @subscriber: pointer to its subscriber   * @seq: name sequence associated with subscription   * @net: point to network namespace - * @timeout: duration of subscription (in ms) - * @filter: event filtering to be done for subscription   * @timer: timer governing subscription duration (optional)   * @nameseq_list: adjacent subscriptions in name sequence's subscription list   * @subscrp_list: adjacent subscriptions in subscriber's subscription list - * @server_ref: object reference of server port associated with subscription   * @swap: indicates if subscriber uses opposite endianness in its messages   * @evt: template for events generated by subscription   */  struct tipc_subscription {  	struct tipc_subscriber *subscriber; -	struct tipc_name_seq seq;  	struct net *net; -	unsigned long timeout; -	u32 filter;  	struct timer_list timer;  	struct list_head nameseq_list;  	struct list_head subscrp_list; @@ -72,11 +66,14 @@ struct tipc_subscription {  	struct tipc_event evt;  }; -int tipc_subscrp_check_overlap(struct tipc_subscription *sub, u32 found_lower, +int tipc_subscrp_check_overlap(struct tipc_name_seq *seq, u32 found_lower,  			       u32 found_upper);  void tipc_subscrp_report_overlap(struct tipc_subscription *sub,  				 u32 found_lower, u32 found_upper, u32 event,  				 u32 port_ref, u32 node, int must); +void tipc_subscrp_convert_seq(struct tipc_name_seq *in, int swap, +			      struct tipc_name_seq *out); +u32 tipc_subscrp_convert_seq_type(u32 type, int swap);  int tipc_topsrv_start(struct net *net);  void tipc_topsrv_stop(struct net *net); diff --git a/net/tipc/udp_media.c b/net/tipc/udp_media.c index d63a911e7fe2..c9cf2be3674a 100644 --- a/net/tipc/udp_media.c +++ b/net/tipc/udp_media.c @@ -48,19 +48,12 @@  #include <linux/tipc_netlink.h>  #include "core.h"  #include "bearer.h" +#include "netlink.h"  /* IANA assigned UDP port */  #define UDP_PORT_DEFAULT	6118 -#define UDP_MIN_HEADROOM        28 - -static const struct nla_policy tipc_nl_udp_policy[TIPC_NLA_UDP_MAX + 1] = { -	[TIPC_NLA_UDP_UNSPEC]	= {.type = NLA_UNSPEC}, -	[TIPC_NLA_UDP_LOCAL]	= {.type = NLA_BINARY, -				   .len = sizeof(struct sockaddr_storage)}, -	[TIPC_NLA_UDP_REMOTE]	= {.type = NLA_BINARY, -				   .len = sizeof(struct sockaddr_storage)}, -}; +#define UDP_MIN_HEADROOM        48  /**   * struct udp_media_addr - IP/UDP addressing information @@ -181,6 +174,8 @@ static int tipc_udp_send_msg(struct net *net, struct sk_buff *skb,  			err = PTR_ERR(rt);  			goto tx_error;  		} + +		skb->dev = rt->dst.dev;  		ttl = ip4_dst_hoplimit(&rt->dst);  		udp_tunnel_xmit_skb(rt, ub->ubsock->sk, skb, src->ipv4.s_addr,  				    dst->ipv4.s_addr, 0, ttl, 0, src->udp_port, @@ -201,7 +196,7 @@ static int tipc_udp_send_msg(struct net *net, struct sk_buff *skb,  		ttl = ip6_dst_hoplimit(ndst);  		err = udp_tunnel6_xmit_skb(ndst, ub->ubsock->sk, skb,  					   ndst->dev, &src->ipv6, -					   &dst->ipv6, 0, ttl, src->udp_port, +					   &dst->ipv6, 0, ttl, 0, src->udp_port,  					   dst->udp_port, false);  #endif  	} @@ -274,7 +269,7 @@ static int parse_options(struct nlattr *attrs[], struct udp_bearer *ub,  			 struct udp_media_addr *remote)  {  	struct nlattr *opts[TIPC_NLA_UDP_MAX + 1]; -	struct sockaddr_storage *sa_local, *sa_remote; +	struct sockaddr_storage sa_local, sa_remote;  	if (!attrs[TIPC_NLA_BEARER_UDP_OPTS])  		goto err; @@ -283,41 +278,48 @@ static int parse_options(struct nlattr *attrs[], struct udp_bearer *ub,  			     tipc_nl_udp_policy))  		goto err;  	if (opts[TIPC_NLA_UDP_LOCAL] && opts[TIPC_NLA_UDP_REMOTE]) { -		sa_local = nla_data(opts[TIPC_NLA_UDP_LOCAL]); -		sa_remote = nla_data(opts[TIPC_NLA_UDP_REMOTE]); +		nla_memcpy(&sa_local, opts[TIPC_NLA_UDP_LOCAL], +			   sizeof(sa_local)); +		nla_memcpy(&sa_remote, opts[TIPC_NLA_UDP_REMOTE], +			   sizeof(sa_remote));  	} else {  err:  		pr_err("Invalid UDP bearer configuration");  		return -EINVAL;  	} -	if ((sa_local->ss_family & sa_remote->ss_family) == AF_INET) { +	if ((sa_local.ss_family & sa_remote.ss_family) == AF_INET) {  		struct sockaddr_in *ip4; -		ip4 = (struct sockaddr_in *)sa_local; +		ip4 = (struct sockaddr_in *)&sa_local;  		local->proto = htons(ETH_P_IP);  		local->udp_port = ip4->sin_port;  		local->ipv4.s_addr = ip4->sin_addr.s_addr; -		ip4 = (struct sockaddr_in *)sa_remote; +		ip4 = (struct sockaddr_in *)&sa_remote;  		remote->proto = htons(ETH_P_IP);  		remote->udp_port = ip4->sin_port;  		remote->ipv4.s_addr = ip4->sin_addr.s_addr;  		return 0;  #if IS_ENABLED(CONFIG_IPV6) -	} else if ((sa_local->ss_family & sa_remote->ss_family) == AF_INET6) { +	} else if ((sa_local.ss_family & sa_remote.ss_family) == AF_INET6) { +		int atype;  		struct sockaddr_in6 *ip6; -		ip6 = (struct sockaddr_in6 *)sa_local; +		ip6 = (struct sockaddr_in6 *)&sa_local; +		atype = ipv6_addr_type(&ip6->sin6_addr); +		if (__ipv6_addr_needs_scope_id(atype) && !ip6->sin6_scope_id) +			return -EINVAL; +  		local->proto = htons(ETH_P_IPV6);  		local->udp_port = ip6->sin6_port; -		local->ipv6 = ip6->sin6_addr; +		memcpy(&local->ipv6, &ip6->sin6_addr, sizeof(struct in6_addr));  		ub->ifindex = ip6->sin6_scope_id; -		ip6 = (struct sockaddr_in6 *)sa_remote; +		ip6 = (struct sockaddr_in6 *)&sa_remote;  		remote->proto = htons(ETH_P_IPV6);  		remote->udp_port = ip6->sin6_port; -		remote->ipv6 = ip6->sin6_addr; +		memcpy(&remote->ipv6, &ip6->sin6_addr, sizeof(struct in6_addr));  		return 0;  #endif  	} | 
