RDMA/device: Add ib_device_set_netdev() as an alternative to get_netdev

The associated netdev should not actually be very dynamic, so for most
drivers there is no reason for a callback like this. Provide an API to
inform the core code about the net dev affiliation and use a core
maintained data structure instead.

This allows the core code to be more aware of the ndev relationship which
will allow some new APIs based around this.

This also uses locking that makes some kind of sense, many drivers had a
confusing RCU lock, or missing locking which isn't right.

Signed-off-by: Jason Gunthorpe <jgg@mellanox.com>
This commit is contained in:
Jason Gunthorpe 2019-02-12 21:12:50 -07:00
parent 8faea9fd4a
commit c2261dd76b
6 changed files with 170 additions and 37 deletions

View File

@ -547,21 +547,19 @@ int ib_cache_gid_add(struct ib_device *ib_dev, u8 port,
unsigned long mask; unsigned long mask;
int ret; int ret;
if (ib_dev->ops.get_netdev) { idev = ib_device_get_netdev(ib_dev, port);
idev = ib_dev->ops.get_netdev(ib_dev, port); if (idev && attr->ndev != idev) {
if (idev && attr->ndev != idev) { union ib_gid default_gid;
union ib_gid default_gid;
/* Adding default GIDs in not permitted */ /* Adding default GIDs is not permitted */
make_default_gid(idev, &default_gid); make_default_gid(idev, &default_gid);
if (!memcmp(gid, &default_gid, sizeof(*gid))) { if (!memcmp(gid, &default_gid, sizeof(*gid))) {
dev_put(idev);
return -EPERM;
}
}
if (idev)
dev_put(idev); dev_put(idev);
return -EPERM;
}
} }
if (idev)
dev_put(idev);
mask = GID_ATTR_FIND_MASK_GID | mask = GID_ATTR_FIND_MASK_GID |
GID_ATTR_FIND_MASK_GID_TYPE | GID_ATTR_FIND_MASK_GID_TYPE |

View File

@ -66,6 +66,9 @@ typedef void (*roce_netdev_callback)(struct ib_device *device, u8 port,
typedef bool (*roce_netdev_filter)(struct ib_device *device, u8 port, typedef bool (*roce_netdev_filter)(struct ib_device *device, u8 port,
struct net_device *idev, void *cookie); struct net_device *idev, void *cookie);
struct net_device *ib_device_get_netdev(struct ib_device *ib_dev,
unsigned int port);
void ib_enum_roce_netdev(struct ib_device *ib_dev, void ib_enum_roce_netdev(struct ib_device *ib_dev,
roce_netdev_filter filter, roce_netdev_filter filter,
void *filter_cookie, void *filter_cookie,

View File

@ -134,6 +134,7 @@ static void *xan_find_marked(struct xarray *xa, unsigned long *indexp,
!xa_is_err(entry); \ !xa_is_err(entry); \
(index)++, entry = xan_find_marked(xa, &(index), filter)) (index)++, entry = xan_find_marked(xa, &(index), filter))
static void free_netdevs(struct ib_device *ib_dev);
static int ib_security_change(struct notifier_block *nb, unsigned long event, static int ib_security_change(struct notifier_block *nb, unsigned long event,
void *lsm_data); void *lsm_data);
static void ib_policy_change_task(struct work_struct *work); static void ib_policy_change_task(struct work_struct *work);
@ -290,6 +291,7 @@ static void ib_device_release(struct device *device)
{ {
struct ib_device *dev = container_of(device, struct ib_device, dev); struct ib_device *dev = container_of(device, struct ib_device, dev);
free_netdevs(dev);
WARN_ON(refcount_read(&dev->refcount)); WARN_ON(refcount_read(&dev->refcount));
ib_cache_release_one(dev); ib_cache_release_one(dev);
ib_security_release_port_pkey_list(dev); ib_security_release_port_pkey_list(dev);
@ -371,6 +373,9 @@ EXPORT_SYMBOL(_ib_alloc_device);
*/ */
void ib_dealloc_device(struct ib_device *device) void ib_dealloc_device(struct ib_device *device)
{ {
/* Expedite releasing netdev references */
free_netdevs(device);
WARN_ON(!xa_empty(&device->client_data)); WARN_ON(!xa_empty(&device->client_data));
WARN_ON(refcount_read(&device->refcount)); WARN_ON(refcount_read(&device->refcount));
rdma_restrack_clean(device); rdma_restrack_clean(device);
@ -461,16 +466,16 @@ static void remove_client_context(struct ib_device *device,
up_read(&device->client_data_rwsem); up_read(&device->client_data_rwsem);
} }
static int verify_immutable(const struct ib_device *dev, u8 port) static int alloc_port_data(struct ib_device *device)
{
return WARN_ON(!rdma_cap_ib_mad(dev, port) &&
rdma_max_mad_size(dev, port) != 0);
}
static int setup_port_data(struct ib_device *device)
{ {
unsigned int port; unsigned int port;
int ret;
if (device->port_data)
return 0;
/* This can only be called once the physical port range is defined */
if (WARN_ON(!device->phys_port_cnt))
return -EINVAL;
/* /*
* device->port_data is indexed directly by the port number to make * device->port_data is indexed directly by the port number to make
@ -489,6 +494,28 @@ static int setup_port_data(struct ib_device *device)
spin_lock_init(&pdata->pkey_list_lock); spin_lock_init(&pdata->pkey_list_lock);
INIT_LIST_HEAD(&pdata->pkey_list); INIT_LIST_HEAD(&pdata->pkey_list);
spin_lock_init(&pdata->netdev_lock);
}
return 0;
}
static int verify_immutable(const struct ib_device *dev, u8 port)
{
return WARN_ON(!rdma_cap_ib_mad(dev, port) &&
rdma_max_mad_size(dev, port) != 0);
}
static int setup_port_data(struct ib_device *device)
{
unsigned int port;
int ret;
ret = alloc_port_data(device);
if (ret)
return ret;
rdma_for_each_port (device, port) {
struct ib_port_data *pdata = &device->port_data[port];
ret = device->ops.get_port_immutable(device, port, ret = device->ops.get_port_immutable(device, port,
&pdata->immutable); &pdata->immutable);
@ -682,6 +709,9 @@ static void disable_device(struct ib_device *device)
/* Pairs with refcount_set in enable_device */ /* Pairs with refcount_set in enable_device */
ib_device_put(device); ib_device_put(device);
wait_for_completion(&device->unreg_completion); wait_for_completion(&device->unreg_completion);
/* Expedite removing unregistered pointers from the hash table */
free_netdevs(device);
} }
/* /*
@ -1012,6 +1042,114 @@ int ib_query_port(struct ib_device *device,
} }
EXPORT_SYMBOL(ib_query_port); EXPORT_SYMBOL(ib_query_port);
/**
* ib_device_set_netdev - Associate the ib_dev with an underlying net_device
* @ib_dev: Device to modify
* @ndev: net_device to affiliate, may be NULL
* @port: IB port the net_device is connected to
*
* Drivers should use this to link the ib_device to a netdev so the netdev
* shows up in interfaces like ib_enum_roce_netdev. Only one netdev may be
* affiliated with any port.
*
* The caller must ensure that the given ndev is not unregistered or
* unregistering, and that either the ib_device is unregistered or
* ib_device_set_netdev() is called with NULL when the ndev sends a
* NETDEV_UNREGISTER event.
*/
int ib_device_set_netdev(struct ib_device *ib_dev, struct net_device *ndev,
unsigned int port)
{
struct net_device *old_ndev;
struct ib_port_data *pdata;
unsigned long flags;
int ret;
/*
* Drivers wish to call this before ib_register_driver, so we have to
* setup the port data early.
*/
ret = alloc_port_data(ib_dev);
if (ret)
return ret;
if (!rdma_is_port_valid(ib_dev, port))
return -EINVAL;
pdata = &ib_dev->port_data[port];
spin_lock_irqsave(&pdata->netdev_lock, flags);
if (pdata->netdev == ndev) {
spin_unlock_irqrestore(&pdata->netdev_lock, flags);
return 0;
}
old_ndev = pdata->netdev;
if (ndev)
dev_hold(ndev);
pdata->netdev = ndev;
spin_unlock_irqrestore(&pdata->netdev_lock, flags);
if (old_ndev)
dev_put(old_ndev);
return 0;
}
EXPORT_SYMBOL(ib_device_set_netdev);
static void free_netdevs(struct ib_device *ib_dev)
{
unsigned long flags;
unsigned int port;
rdma_for_each_port (ib_dev, port) {
struct ib_port_data *pdata = &ib_dev->port_data[port];
spin_lock_irqsave(&pdata->netdev_lock, flags);
if (pdata->netdev) {
dev_put(pdata->netdev);
pdata->netdev = NULL;
}
spin_unlock_irqrestore(&pdata->netdev_lock, flags);
}
}
struct net_device *ib_device_get_netdev(struct ib_device *ib_dev,
unsigned int port)
{
struct ib_port_data *pdata;
struct net_device *res;
if (!rdma_is_port_valid(ib_dev, port))
return NULL;
pdata = &ib_dev->port_data[port];
/*
* New drivers should use ib_device_set_netdev() not the legacy
* get_netdev().
*/
if (ib_dev->ops.get_netdev)
res = ib_dev->ops.get_netdev(ib_dev, port);
else {
spin_lock(&pdata->netdev_lock);
res = pdata->netdev;
if (res)
dev_hold(res);
spin_unlock(&pdata->netdev_lock);
}
/*
* If we are starting to unregister expedite things by preventing
* propagation of an unregistering netdev.
*/
if (res && res->reg_state != NETREG_REGISTERED) {
dev_put(res);
return NULL;
}
return res;
}
/** /**
* ib_enum_roce_netdev - enumerate all RoCE ports * ib_enum_roce_netdev - enumerate all RoCE ports
* @ib_dev : IB device we want to query * @ib_dev : IB device we want to query
@ -1034,16 +1172,8 @@ void ib_enum_roce_netdev(struct ib_device *ib_dev,
rdma_for_each_port (ib_dev, port) rdma_for_each_port (ib_dev, port)
if (rdma_protocol_roce(ib_dev, port)) { if (rdma_protocol_roce(ib_dev, port)) {
struct net_device *idev = NULL; struct net_device *idev =
ib_device_get_netdev(ib_dev, port);
if (ib_dev->ops.get_netdev)
idev = ib_dev->ops.get_netdev(ib_dev, port);
if (idev &&
idev->reg_state >= NETREG_UNREGISTERED) {
dev_put(idev);
idev = NULL;
}
if (filter(ib_dev, port, idev, filter_cookie)) if (filter(ib_dev, port, idev, filter_cookie))
cb(ib_dev, port, idev, cookie); cb(ib_dev, port, idev, cookie);

View File

@ -268,9 +268,7 @@ static int fill_port_info(struct sk_buff *msg,
if (nla_put_u8(msg, RDMA_NLDEV_ATTR_PORT_PHYS_STATE, attr.phys_state)) if (nla_put_u8(msg, RDMA_NLDEV_ATTR_PORT_PHYS_STATE, attr.phys_state))
return -EMSGSIZE; return -EMSGSIZE;
if (device->ops.get_netdev) netdev = ib_device_get_netdev(device, port);
netdev = device->ops.get_netdev(device, port);
if (netdev && net_eq(dev_net(netdev), net)) { if (netdev && net_eq(dev_net(netdev), net)) {
ret = nla_put_u32(msg, ret = nla_put_u32(msg,
RDMA_NLDEV_ATTR_NDEV_INDEX, netdev->ifindex); RDMA_NLDEV_ATTR_NDEV_INDEX, netdev->ifindex);

View File

@ -1723,10 +1723,7 @@ int ib_get_eth_speed(struct ib_device *dev, u8 port_num, u8 *speed, u8 *width)
if (rdma_port_get_link_layer(dev, port_num) != IB_LINK_LAYER_ETHERNET) if (rdma_port_get_link_layer(dev, port_num) != IB_LINK_LAYER_ETHERNET)
return -EINVAL; return -EINVAL;
if (!dev->ops.get_netdev) netdev = ib_device_get_netdev(dev, port_num);
return -EOPNOTSUPP;
netdev = dev->ops.get_netdev(dev, port_num);
if (!netdev) if (!netdev)
return -ENODEV; return -ENODEV;

View File

@ -2204,6 +2204,9 @@ struct ib_port_data {
struct list_head pkey_list; struct list_head pkey_list;
struct ib_port_cache cache; struct ib_port_cache cache;
spinlock_t netdev_lock;
struct net_device *netdev;
}; };
/* rdma netdev type - specifies protocol type */ /* rdma netdev type - specifies protocol type */
@ -3996,6 +3999,10 @@ void ib_device_put(struct ib_device *device);
struct net_device *ib_get_net_dev_by_params(struct ib_device *dev, u8 port, struct net_device *ib_get_net_dev_by_params(struct ib_device *dev, u8 port,
u16 pkey, const union ib_gid *gid, u16 pkey, const union ib_gid *gid,
const struct sockaddr *addr); const struct sockaddr *addr);
int ib_device_set_netdev(struct ib_device *ib_dev, struct net_device *ndev,
unsigned int port);
struct net_device *ib_device_netdev(struct ib_device *dev, u8 port);
struct ib_wq *ib_create_wq(struct ib_pd *pd, struct ib_wq *ib_create_wq(struct ib_pd *pd,
struct ib_wq_init_attr *init_attr); struct ib_wq_init_attr *init_attr);
int ib_destroy_wq(struct ib_wq *wq); int ib_destroy_wq(struct ib_wq *wq);