X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=datapath%2Fbrcompat.c;h=92fcecc216719dce8eb274b96ffb5210c95491ae;hb=e19702a8016df476e8fa476f92d8177257aac0ad;hp=46e7f2b0866170450232dcc1128fdd07632b5fb6;hpb=2e7dd8eca88d131112a76301da24709b0472e381;p=openvswitch diff --git a/datapath/brcompat.c b/datapath/brcompat.c index 46e7f2b0..92fcecc2 100644 --- a/datapath/brcompat.c +++ b/datapath/brcompat.c @@ -9,10 +9,8 @@ #include #include #include -#include #include #include -#include #include #include #include @@ -20,9 +18,7 @@ #include "compat.h" #include "openvswitch/brcompat-netlink.h" #include "brc_procfs.h" -#include "brc_sysfs.h" #include "datapath.h" -#include "dp_dev.h" static struct genl_family brc_genl_family; static struct genl_multicast_group brc_mc_group; @@ -46,36 +42,6 @@ static u32 brc_seq; /* Sequence number for current op. */ static struct sk_buff *brc_send_command(struct sk_buff *, struct nlattr **attrs); static int brc_send_simple_command(struct sk_buff *); -static int -get_dp_ifindices(int *indices, int num) -{ - int i, index = 0; - - rcu_read_lock(); - for (i=0; i < ODP_MAX && index < num; i++) { - struct datapath *dp = get_dp(i); - if (!dp) - continue; - indices[index++] = dp->ports[ODPP_LOCAL]->dev->ifindex; - } - rcu_read_unlock(); - - return index; -} - -static void -get_port_ifindices(struct datapath *dp, int *ifindices, int num) -{ - struct net_bridge_port *p; - - rcu_read_lock(); - list_for_each_entry_rcu (p, &dp->port_list, node) { - if (p->port_no < num) - ifindices[p->port_no] = p->dev->ifindex; - } - rcu_read_unlock(); -} - static struct sk_buff * brc_make_request(int op, const char *bridge, const char *port) { @@ -84,7 +50,8 @@ brc_make_request(int op, const char *bridge, const char *port) goto error; genlmsg_put(skb, 0, 0, &brc_genl_family, 0, op); - NLA_PUT_STRING(skb, BRC_GENL_A_DP_NAME, bridge); + if (bridge) + NLA_PUT_STRING(skb, BRC_GENL_A_DP_NAME, bridge); if (port) NLA_PUT_STRING(skb, BRC_GENL_A_PORT_NAME, port); return skb; @@ -127,26 +94,57 @@ static int brc_add_del_bridge(char __user *uname, int add) return brc_send_simple_command(request); } -static int brc_get_bridges(int __user *uindices, int n) +static int brc_get_indices(int op, const char *br_name, + int __user *uindices, int n) { + struct nlattr *attrs[BRC_GENL_A_MAX + 1]; + struct sk_buff *request, *reply; int *indices; int ret; + int len; + if (n < 0) + return -EINVAL; if (n >= 2048) return -ENOMEM; - indices = kcalloc(n, sizeof(int), GFP_KERNEL); - if (indices == NULL) + request = brc_make_request(op, br_name, NULL); + if (!request) return -ENOMEM; - n = get_dp_ifindices(indices, n); + reply = brc_send_command(request, attrs); + ret = PTR_ERR(reply); + if (IS_ERR(reply)) + goto exit; + + ret = -nla_get_u32(attrs[BRC_GENL_A_ERR_CODE]); + if (ret < 0) + goto exit_free_skb; + + ret = -EINVAL; + if (!attrs[BRC_GENL_A_IFINDEXES]) + goto exit_free_skb; + + len = nla_len(attrs[BRC_GENL_A_IFINDEXES]); + indices = nla_data(attrs[BRC_GENL_A_IFINDEXES]); + if (len % sizeof(int)) + goto exit_free_skb; + n = min_t(int, n, len / sizeof(int)); ret = copy_to_user(uindices, indices, n * sizeof(int)) ? -EFAULT : n; - kfree(indices); +exit_free_skb: + kfree_skb(reply); +exit: return ret; } +/* Called with br_ioctl_mutex. */ +static int brc_get_bridges(int __user *uindices, int n) +{ + return brc_get_indices(BRC_GENL_C_GET_BRIDGES, NULL, uindices, n); +} + /* Legacy deviceless bridge ioctl's. Called with br_ioctl_mutex. */ static int old_deviceless(void __user *uarg) @@ -239,26 +237,14 @@ brc_get_bridge_info(struct net_device *dev, struct __bridge_info __user *ub) static int brc_get_port_list(struct net_device *dev, int __user *uindices, int num) { - struct dp_dev *dp_dev = netdev_priv(dev); - struct datapath *dp = dp_dev->dp; - int *indices; - - if (num < 0) - return -EINVAL; - if (num == 0) - num = 256; - if (num > DP_MAX_PORTS) - num = DP_MAX_PORTS; + int retval; - indices = kcalloc(num, sizeof(int), GFP_KERNEL); - if (indices == NULL) - return -ENOMEM; + rtnl_unlock(); + retval = brc_get_indices(BRC_GENL_C_GET_PORTS, dev->name, + uindices, num); + rtnl_lock(); - get_port_ifindices(dp, indices, num); - if (copy_to_user(uindices, indices, num * sizeof(int))) - num = -EFAULT; - kfree(indices); - return num; + return retval; } /* @@ -526,21 +512,10 @@ error: static int __init brc_init(void) { - int i; int err; printk("Open vSwitch Bridge Compatibility, built "__DATE__" "__TIME__"\n"); - rcu_read_lock(); - for (i=0; i