diff options
author | Jack Morgenstein <jackm@dev.mellanox.co.il> | 2012-08-03 04:40:39 -0400 |
---|---|---|
committer | Roland Dreier <roland@purestorage.com> | 2012-09-30 23:33:30 -0400 |
commit | 73aaa7418f8069103ca56fc620b3cd16c5a37d6e (patch) | |
tree | 0110a685c949f7ce92e804ddff1cf6f34482d152 /drivers | |
parent | ff7166c447df23a61e4f51bf748319dc6728dc74 (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')
-rw-r--r-- | drivers/infiniband/core/cache.c | 32 |
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 | } |
200 | EXPORT_SYMBOL(ib_find_cached_pkey); | 200 | EXPORT_SYMBOL(ib_find_cached_pkey); |
201 | 201 | ||
202 | int 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 | } | ||
232 | EXPORT_SYMBOL(ib_find_exact_cached_pkey); | ||
233 | |||
202 | int ib_get_cached_lmc(struct ib_device *device, | 234 | int ib_get_cached_lmc(struct ib_device *device, |
203 | u8 port_num, | 235 | u8 port_num, |
204 | u8 *lmc) | 236 | u8 *lmc) |