aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/infiniband/core/cache.c
diff options
context:
space:
mode:
authorJack Morgenstein <jackm@dev.mellanox.co.il>2012-08-03 04:40:39 -0400
committerRoland Dreier <roland@purestorage.com>2012-09-30 23:33:30 -0400
commit73aaa7418f8069103ca56fc620b3cd16c5a37d6e (patch)
tree0110a685c949f7ce92e804ddff1cf6f34482d152 /drivers/infiniband/core/cache.c
parentff7166c447df23a61e4f51bf748319dc6728dc74 (diff)
IB/core: Add ib_find_exact_cached_pkey()
When P_Key tables potentially contain both full and partial membership copies for the same P_Key, we need a function to find the index for an exact (16-bit) P_Key. This is necessary when the master forwards QP1 MADs sent by guests. If the guest has sent the MAD with a limited membership P_Key, we need to to forward the MAD using the same limited membership P_Key. Since the master may have both the limited and the full member P_Keys in its table, we must make sure to retrieve the limited membership P_Key in this case. Signed-off-by: Jack Morgenstein <jackm@dev.mellanox.co.il> Signed-off-by: Or Gerlitz <ogerlitz@mellanox.com> Signed-off-by: Roland Dreier <roland@purestorage.com>
Diffstat (limited to 'drivers/infiniband/core/cache.c')
-rw-r--r--drivers/infiniband/core/cache.c32
1 files changed, 32 insertions, 0 deletions
diff --git a/drivers/infiniband/core/cache.c b/drivers/infiniband/core/cache.c
index 4da381b74f54..80f6cf2449fb 100644
--- a/drivers/infiniband/core/cache.c
+++ b/drivers/infiniband/core/cache.c
@@ -199,6 +199,38 @@ int ib_find_cached_pkey(struct ib_device *device,
199} 199}
200EXPORT_SYMBOL(ib_find_cached_pkey); 200EXPORT_SYMBOL(ib_find_cached_pkey);
201 201
202int ib_find_exact_cached_pkey(struct ib_device *device,
203 u8 port_num,
204 u16 pkey,
205 u16 *index)
206{
207 struct ib_pkey_cache *cache;
208 unsigned long flags;
209 int i;
210 int ret = -ENOENT;
211
212 if (port_num < start_port(device) || port_num > end_port(device))
213 return -EINVAL;
214
215 read_lock_irqsave(&device->cache.lock, flags);
216
217 cache = device->cache.pkey_cache[port_num - start_port(device)];
218
219 *index = -1;
220
221 for (i = 0; i < cache->table_len; ++i)
222 if (cache->table[i] == pkey) {
223 *index = i;
224 ret = 0;
225 break;
226 }
227
228 read_unlock_irqrestore(&device->cache.lock, flags);
229
230 return ret;
231}
232EXPORT_SYMBOL(ib_find_exact_cached_pkey);
233
202int ib_get_cached_lmc(struct ib_device *device, 234int ib_get_cached_lmc(struct ib_device *device,
203 u8 port_num, 235 u8 port_num,
204 u8 *lmc) 236 u8 *lmc)