aboutsummaryrefslogtreecommitdiffstats
path: root/net/tipc
diff options
context:
space:
mode:
authorRichard Alpe <richard.alpe@ericsson.com>2015-02-09 03:50:11 -0500
committerDavid S. Miller <davem@davemloft.net>2015-02-09 16:20:48 -0500
commit487d2a3a1326d339ce273ffbcd03247f2b7b052e (patch)
tree6ffa649e50216a6a894cf4fe0143b587ee2e190d /net/tipc
parent44a8ae94fd5525aa06a8c71cb52efbc418fb8b7c (diff)
tipc: convert legacy nl socket dump to nl compat
Convert socket (port) listing to compat dumpit call. If a socket (port) has publications a second dumpit call is issued to collect them and format then into the legacy buffer before continuing to process the sockets (ports). Command converted in this patch: TIPC_CMD_SHOW_PORTS Signed-off-by: Richard Alpe <richard.alpe@ericsson.com> Reviewed-by: Erik Hugne <erik.hugne@ericsson.com> Reviewed-by: Ying Xue <ying.xue@windriver.com> Reviewed-by: Jon Maloy <jon.maloy@ericsson.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/tipc')
-rw-r--r--net/tipc/config.c3
-rw-r--r--net/tipc/netlink_compat.c111
-rw-r--r--net/tipc/socket.c85
-rw-r--r--net/tipc/socket.h1
4 files changed, 111 insertions, 89 deletions
diff --git a/net/tipc/config.c b/net/tipc/config.c
index 7b053fcc4b87..6e4c215879c8 100644
--- a/net/tipc/config.c
+++ b/net/tipc/config.c
@@ -212,9 +212,6 @@ struct sk_buff *tipc_cfg_do_cmd(struct net *net, u32 orig_node, u16 cmd,
212 case TIPC_CMD_GET_MEDIA_NAMES: 212 case TIPC_CMD_GET_MEDIA_NAMES:
213 rep_tlv_buf = tipc_media_get_names(); 213 rep_tlv_buf = tipc_media_get_names();
214 break; 214 break;
215 case TIPC_CMD_SHOW_PORTS:
216 rep_tlv_buf = tipc_sk_socks_show(net);
217 break;
218 case TIPC_CMD_SHOW_STATS: 215 case TIPC_CMD_SHOW_STATS:
219 rep_tlv_buf = tipc_show_stats(); 216 rep_tlv_buf = tipc_show_stats();
220 break; 217 break;
diff --git a/net/tipc/netlink_compat.c b/net/tipc/netlink_compat.c
index 40c24ea31231..48e15a4a36d8 100644
--- a/net/tipc/netlink_compat.c
+++ b/net/tipc/netlink_compat.c
@@ -36,6 +36,7 @@
36#include "bearer.h" 36#include "bearer.h"
37#include "link.h" 37#include "link.h"
38#include "name_table.h" 38#include "name_table.h"
39#include "socket.h"
39#include <net/genetlink.h> 40#include <net/genetlink.h>
40#include <linux/tipc_config.h> 41#include <linux/tipc_config.h>
41 42
@@ -718,6 +719,109 @@ out:
718 return 0; 719 return 0;
719} 720}
720 721
722static int __tipc_nl_compat_publ_dump(struct tipc_nl_compat_msg *msg,
723 struct nlattr **attrs)
724{
725 u32 type, lower, upper;
726 struct nlattr *publ[TIPC_NLA_PUBL_MAX + 1];
727
728 nla_parse_nested(publ, TIPC_NLA_PUBL_MAX, attrs[TIPC_NLA_PUBL], NULL);
729
730 type = nla_get_u32(publ[TIPC_NLA_PUBL_TYPE]);
731 lower = nla_get_u32(publ[TIPC_NLA_PUBL_LOWER]);
732 upper = nla_get_u32(publ[TIPC_NLA_PUBL_UPPER]);
733
734 if (lower == upper)
735 tipc_tlv_sprintf(msg->rep, " {%u,%u}", type, lower);
736 else
737 tipc_tlv_sprintf(msg->rep, " {%u,%u,%u}", type, lower, upper);
738
739 return 0;
740}
741
742static int tipc_nl_compat_publ_dump(struct tipc_nl_compat_msg *msg, u32 sock)
743{
744 int err;
745 void *hdr;
746 struct nlattr *nest;
747 struct sk_buff *args;
748 struct tipc_nl_compat_cmd_dump dump;
749
750 args = nlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
751 if (!args)
752 return -ENOMEM;
753
754 hdr = genlmsg_put(args, 0, 0, &tipc_genl_family, NLM_F_MULTI,
755 TIPC_NL_PUBL_GET);
756
757 nest = nla_nest_start(args, TIPC_NLA_SOCK);
758 if (!nest) {
759 kfree_skb(args);
760 return -EMSGSIZE;
761 }
762
763 if (nla_put_u32(args, TIPC_NLA_SOCK_REF, sock)) {
764 kfree_skb(args);
765 return -EMSGSIZE;
766 }
767
768 nla_nest_end(args, nest);
769 genlmsg_end(args, hdr);
770
771 dump.dumpit = tipc_nl_publ_dump;
772 dump.format = __tipc_nl_compat_publ_dump;
773
774 err = __tipc_nl_compat_dumpit(&dump, msg, args);
775
776 kfree_skb(args);
777
778 return err;
779}
780
781static int tipc_nl_compat_sk_dump(struct tipc_nl_compat_msg *msg,
782 struct nlattr **attrs)
783{
784 int err;
785 u32 sock_ref;
786 struct nlattr *sock[TIPC_NLA_SOCK_MAX + 1];
787
788 nla_parse_nested(sock, TIPC_NLA_SOCK_MAX, attrs[TIPC_NLA_SOCK], NULL);
789
790 sock_ref = nla_get_u32(sock[TIPC_NLA_SOCK_REF]);
791 tipc_tlv_sprintf(msg->rep, "%u:", sock_ref);
792
793 if (sock[TIPC_NLA_SOCK_CON]) {
794 u32 node;
795 struct nlattr *con[TIPC_NLA_CON_MAX + 1];
796
797 nla_parse_nested(con, TIPC_NLA_CON_MAX, sock[TIPC_NLA_SOCK_CON],
798 NULL);
799
800 node = nla_get_u32(con[TIPC_NLA_CON_NODE]);
801 tipc_tlv_sprintf(msg->rep, " connected to <%u.%u.%u:%u>",
802 tipc_zone(node),
803 tipc_cluster(node),
804 tipc_node(node),
805 nla_get_u32(con[TIPC_NLA_CON_SOCK]));
806
807 if (con[TIPC_NLA_CON_FLAG])
808 tipc_tlv_sprintf(msg->rep, " via {%u,%u}\n",
809 nla_get_u32(con[TIPC_NLA_CON_TYPE]),
810 nla_get_u32(con[TIPC_NLA_CON_INST]));
811 else
812 tipc_tlv_sprintf(msg->rep, "\n");
813 } else if (sock[TIPC_NLA_SOCK_HAS_PUBL]) {
814 tipc_tlv_sprintf(msg->rep, " bound to");
815
816 err = tipc_nl_compat_publ_dump(msg, sock_ref);
817 if (err)
818 return err;
819 }
820 tipc_tlv_sprintf(msg->rep, "\n");
821
822 return 0;
823}
824
721static int tipc_nl_compat_handle(struct tipc_nl_compat_msg *msg) 825static int tipc_nl_compat_handle(struct tipc_nl_compat_msg *msg)
722{ 826{
723 struct tipc_nl_compat_cmd_dump dump; 827 struct tipc_nl_compat_cmd_dump dump;
@@ -775,6 +879,12 @@ static int tipc_nl_compat_handle(struct tipc_nl_compat_msg *msg)
775 dump.dumpit = tipc_nl_name_table_dump; 879 dump.dumpit = tipc_nl_name_table_dump;
776 dump.format = tipc_nl_compat_name_table_dump; 880 dump.format = tipc_nl_compat_name_table_dump;
777 return tipc_nl_compat_dumpit(&dump, msg); 881 return tipc_nl_compat_dumpit(&dump, msg);
882 case TIPC_CMD_SHOW_PORTS:
883 msg->rep_size = ULTRA_STRING_MAX_LEN;
884 msg->rep_type = TIPC_TLV_ULTRA_STRING;
885 dump.dumpit = tipc_nl_sk_dump;
886 dump.format = tipc_nl_compat_sk_dump;
887 return tipc_nl_compat_dumpit(&dump, msg);
778 } 888 }
779 889
780 return -EOPNOTSUPP; 890 return -EOPNOTSUPP;
@@ -881,6 +991,7 @@ static int tipc_nl_compat_tmp_wrap(struct sk_buff *skb, struct genl_info *info)
881 case TIPC_CMD_SET_LINK_WINDOW: 991 case TIPC_CMD_SET_LINK_WINDOW:
882 case TIPC_CMD_RESET_LINK_STATS: 992 case TIPC_CMD_RESET_LINK_STATS:
883 case TIPC_CMD_SHOW_NAME_TABLE: 993 case TIPC_CMD_SHOW_NAME_TABLE:
994 case TIPC_CMD_SHOW_PORTS:
884 return tipc_nl_compat_recv(skb, info); 995 return tipc_nl_compat_recv(skb, info);
885 } 996 }
886 997
diff --git a/net/tipc/socket.c b/net/tipc/socket.c
index d76c171f7b7e..e77d738bb771 100644
--- a/net/tipc/socket.c
+++ b/net/tipc/socket.c
@@ -2281,91 +2281,6 @@ static int tipc_sk_withdraw(struct tipc_sock *tsk, uint scope,
2281 return rc; 2281 return rc;
2282} 2282}
2283 2283
2284static int tipc_sk_show(struct tipc_sock *tsk, char *buf,
2285 int len, int full_id)
2286{
2287 struct net *net = sock_net(&tsk->sk);
2288 struct tipc_net *tn = net_generic(net, tipc_net_id);
2289 struct publication *publ;
2290 int ret;
2291
2292 if (full_id)
2293 ret = tipc_snprintf(buf, len, "<%u.%u.%u:%u>:",
2294 tipc_zone(tn->own_addr),
2295 tipc_cluster(tn->own_addr),
2296 tipc_node(tn->own_addr), tsk->portid);
2297 else
2298 ret = tipc_snprintf(buf, len, "%-10u:", tsk->portid);
2299
2300 if (tsk->connected) {
2301 u32 dport = tsk_peer_port(tsk);
2302 u32 destnode = tsk_peer_node(tsk);
2303
2304 ret += tipc_snprintf(buf + ret, len - ret,
2305 " connected to <%u.%u.%u:%u>",
2306 tipc_zone(destnode),
2307 tipc_cluster(destnode),
2308 tipc_node(destnode), dport);
2309 if (tsk->conn_type != 0)
2310 ret += tipc_snprintf(buf + ret, len - ret,
2311 " via {%u,%u}", tsk->conn_type,
2312 tsk->conn_instance);
2313 } else if (tsk->published) {
2314 ret += tipc_snprintf(buf + ret, len - ret, " bound to");
2315 list_for_each_entry(publ, &tsk->publications, pport_list) {
2316 if (publ->lower == publ->upper)
2317 ret += tipc_snprintf(buf + ret, len - ret,
2318 " {%u,%u}", publ->type,
2319 publ->lower);
2320 else
2321 ret += tipc_snprintf(buf + ret, len - ret,
2322 " {%u,%u,%u}", publ->type,
2323 publ->lower, publ->upper);
2324 }
2325 }
2326 ret += tipc_snprintf(buf + ret, len - ret, "\n");
2327 return ret;
2328}
2329
2330struct sk_buff *tipc_sk_socks_show(struct net *net)
2331{
2332 struct tipc_net *tn = net_generic(net, tipc_net_id);
2333 const struct bucket_table *tbl;
2334 struct rhash_head *pos;
2335 struct sk_buff *buf;
2336 struct tlv_desc *rep_tlv;
2337 char *pb;
2338 int pb_len;
2339 struct tipc_sock *tsk;
2340 int str_len = 0;
2341 int i;
2342
2343 buf = tipc_cfg_reply_alloc(TLV_SPACE(ULTRA_STRING_MAX_LEN));
2344 if (!buf)
2345 return NULL;
2346 rep_tlv = (struct tlv_desc *)buf->data;
2347 pb = TLV_DATA(rep_tlv);
2348 pb_len = ULTRA_STRING_MAX_LEN;
2349
2350 rcu_read_lock();
2351 tbl = rht_dereference_rcu((&tn->sk_rht)->tbl, &tn->sk_rht);
2352 for (i = 0; i < tbl->size; i++) {
2353 rht_for_each_entry_rcu(tsk, pos, tbl, i, node) {
2354 spin_lock_bh(&tsk->sk.sk_lock.slock);
2355 str_len += tipc_sk_show(tsk, pb + str_len,
2356 pb_len - str_len, 0);
2357 spin_unlock_bh(&tsk->sk.sk_lock.slock);
2358 }
2359 }
2360 rcu_read_unlock();
2361
2362 str_len += 1; /* for "\0" */
2363 skb_put(buf, TLV_SPACE(str_len));
2364 TLV_SET(rep_tlv, TIPC_TLV_ULTRA_STRING, NULL, str_len);
2365
2366 return buf;
2367}
2368
2369/* tipc_sk_reinit: set non-zero address in all existing sockets 2284/* tipc_sk_reinit: set non-zero address in all existing sockets
2370 * when we go from standalone to network mode. 2285 * when we go from standalone to network mode.
2371 */ 2286 */
diff --git a/net/tipc/socket.h b/net/tipc/socket.h
index 8be0da7df8fc..238f1b7bd9bd 100644
--- a/net/tipc/socket.h
+++ b/net/tipc/socket.h
@@ -49,7 +49,6 @@ void tipc_sock_release_local(struct socket *sock);
49int tipc_sock_accept_local(struct socket *sock, struct socket **newsock, 49int tipc_sock_accept_local(struct socket *sock, struct socket **newsock,
50 int flags); 50 int flags);
51int tipc_sk_rcv(struct net *net, struct sk_buff_head *inputq); 51int tipc_sk_rcv(struct net *net, struct sk_buff_head *inputq);
52struct sk_buff *tipc_sk_socks_show(struct net *net);
53void tipc_sk_mcast_rcv(struct net *net, struct sk_buff_head *arrvq, 52void tipc_sk_mcast_rcv(struct net *net, struct sk_buff_head *arrvq,
54 struct sk_buff_head *inputq); 53 struct sk_buff_head *inputq);
55void tipc_sk_reinit(struct net *net); 54void tipc_sk_reinit(struct net *net);