Commit ef555d8c authored by Pablo Neira Ayuso's avatar Pablo Neira Ayuso Committed by Stefan Bader

rhashtable: add rhashtable_lookup_get_insert_key()

BugLink: https://bugs.launchpad.net/bugs/1818806

commit 5ca8cc5b upstream.

This patch modifies __rhashtable_insert_fast() so it returns the
existing object that clashes with the one that you want to insert.
In case the object is successfully inserted, NULL is returned.
Otherwise, you get an error via ERR_PTR().

This patch adapts the existing callers of __rhashtable_insert_fast()
so they handle this new logic, and it adds a new
rhashtable_lookup_get_insert_key() interface to fetch this existing
object.

nf_tables needs this change to improve handling of EEXIST cases via
honoring the NLM_F_EXCL flag and by checking if the data part of the
mapping matches what we have.

Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Thomas Graf <tgraf@suug.ch>
Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
Acked-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
Signed-off-by: default avatarBen Hutchings <ben.hutchings@codethink.co.uk>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarJuerg Haefliger <juergh@canonical.com>
Signed-off-by: default avatarKhalid Elmously <khalid.elmously@canonical.com>
parent 6950af03
...@@ -343,7 +343,8 @@ int rhashtable_init(struct rhashtable *ht, ...@@ -343,7 +343,8 @@ int rhashtable_init(struct rhashtable *ht,
struct bucket_table *rhashtable_insert_slow(struct rhashtable *ht, struct bucket_table *rhashtable_insert_slow(struct rhashtable *ht,
const void *key, const void *key,
struct rhash_head *obj, struct rhash_head *obj,
struct bucket_table *old_tbl); struct bucket_table *old_tbl,
void **data);
int rhashtable_insert_rehash(struct rhashtable *ht, struct bucket_table *tbl); int rhashtable_insert_rehash(struct rhashtable *ht, struct bucket_table *tbl);
int rhashtable_walk_init(struct rhashtable *ht, struct rhashtable_iter *iter); int rhashtable_walk_init(struct rhashtable *ht, struct rhashtable_iter *iter);
...@@ -562,8 +563,11 @@ static inline void *rhashtable_lookup_fast( ...@@ -562,8 +563,11 @@ static inline void *rhashtable_lookup_fast(
return NULL; return NULL;
} }
/* Internal function, please use rhashtable_insert_fast() instead */ /* Internal function, please use rhashtable_insert_fast() instead. This
static inline int __rhashtable_insert_fast( * function returns the existing element already in hashes in there is a clash,
* otherwise it returns an error via ERR_PTR().
*/
static inline void *__rhashtable_insert_fast(
struct rhashtable *ht, const void *key, struct rhash_head *obj, struct rhashtable *ht, const void *key, struct rhash_head *obj,
const struct rhashtable_params params) const struct rhashtable_params params)
{ {
...@@ -576,6 +580,7 @@ static inline int __rhashtable_insert_fast( ...@@ -576,6 +580,7 @@ static inline int __rhashtable_insert_fast(
spinlock_t *lock; spinlock_t *lock;
unsigned int elasticity; unsigned int elasticity;
unsigned int hash; unsigned int hash;
void *data = NULL;
int err; int err;
restart: restart:
...@@ -600,11 +605,14 @@ static inline int __rhashtable_insert_fast( ...@@ -600,11 +605,14 @@ static inline int __rhashtable_insert_fast(
new_tbl = rht_dereference_rcu(tbl->future_tbl, ht); new_tbl = rht_dereference_rcu(tbl->future_tbl, ht);
if (unlikely(new_tbl)) { if (unlikely(new_tbl)) {
tbl = rhashtable_insert_slow(ht, key, obj, new_tbl); tbl = rhashtable_insert_slow(ht, key, obj, new_tbl, &data);
if (!IS_ERR_OR_NULL(tbl)) if (!IS_ERR_OR_NULL(tbl))
goto slow_path; goto slow_path;
err = PTR_ERR(tbl); err = PTR_ERR(tbl);
if (err == -EEXIST)
err = 0;
goto out; goto out;
} }
...@@ -618,25 +626,25 @@ static inline int __rhashtable_insert_fast( ...@@ -618,25 +626,25 @@ static inline int __rhashtable_insert_fast(
err = rhashtable_insert_rehash(ht, tbl); err = rhashtable_insert_rehash(ht, tbl);
rcu_read_unlock(); rcu_read_unlock();
if (err) if (err)
return err; return ERR_PTR(err);
goto restart; goto restart;
} }
err = -EEXIST; err = 0;
elasticity = ht->elasticity; elasticity = ht->elasticity;
rht_for_each(head, tbl, hash) { rht_for_each(head, tbl, hash) {
if (key && if (key &&
unlikely(!(params.obj_cmpfn ? unlikely(!(params.obj_cmpfn ?
params.obj_cmpfn(&arg, rht_obj(ht, head)) : params.obj_cmpfn(&arg, rht_obj(ht, head)) :
rhashtable_compare(&arg, rht_obj(ht, head))))) rhashtable_compare(&arg, rht_obj(ht, head))))) {
data = rht_obj(ht, head);
goto out; goto out;
}
if (!--elasticity) if (!--elasticity)
goto slow_path; goto slow_path;
} }
err = 0;
head = rht_dereference_bucket(tbl->buckets[hash], tbl, hash); head = rht_dereference_bucket(tbl->buckets[hash], tbl, hash);
RCU_INIT_POINTER(obj->next, head); RCU_INIT_POINTER(obj->next, head);
...@@ -651,7 +659,7 @@ static inline int __rhashtable_insert_fast( ...@@ -651,7 +659,7 @@ static inline int __rhashtable_insert_fast(
spin_unlock_bh(lock); spin_unlock_bh(lock);
rcu_read_unlock(); rcu_read_unlock();
return err; return err ? ERR_PTR(err) : data;
} }
/** /**
...@@ -674,7 +682,13 @@ static inline int rhashtable_insert_fast( ...@@ -674,7 +682,13 @@ static inline int rhashtable_insert_fast(
struct rhashtable *ht, struct rhash_head *obj, struct rhashtable *ht, struct rhash_head *obj,
const struct rhashtable_params params) const struct rhashtable_params params)
{ {
return __rhashtable_insert_fast(ht, NULL, obj, params); void *ret;
ret = __rhashtable_insert_fast(ht, NULL, obj, params);
if (IS_ERR(ret))
return PTR_ERR(ret);
return ret == NULL ? 0 : -EEXIST;
} }
/** /**
...@@ -703,11 +717,15 @@ static inline int rhashtable_lookup_insert_fast( ...@@ -703,11 +717,15 @@ static inline int rhashtable_lookup_insert_fast(
const struct rhashtable_params params) const struct rhashtable_params params)
{ {
const char *key = rht_obj(ht, obj); const char *key = rht_obj(ht, obj);
void *ret;
BUG_ON(ht->p.obj_hashfn); BUG_ON(ht->p.obj_hashfn);
return __rhashtable_insert_fast(ht, key + ht->p.key_offset, obj, ret = __rhashtable_insert_fast(ht, key + ht->p.key_offset, obj, params);
params); if (IS_ERR(ret))
return PTR_ERR(ret);
return ret == NULL ? 0 : -EEXIST;
} }
/** /**
...@@ -735,6 +753,32 @@ static inline int rhashtable_lookup_insert_fast( ...@@ -735,6 +753,32 @@ static inline int rhashtable_lookup_insert_fast(
static inline int rhashtable_lookup_insert_key( static inline int rhashtable_lookup_insert_key(
struct rhashtable *ht, const void *key, struct rhash_head *obj, struct rhashtable *ht, const void *key, struct rhash_head *obj,
const struct rhashtable_params params) const struct rhashtable_params params)
{
void *ret;
BUG_ON(!ht->p.obj_hashfn || !key);
ret = __rhashtable_insert_fast(ht, key, obj, params);
if (IS_ERR(ret))
return PTR_ERR(ret);
return ret == NULL ? 0 : -EEXIST;
}
/**
* rhashtable_lookup_get_insert_key - lookup and insert object into hash table
* @ht: hash table
* @obj: pointer to hash head inside object
* @params: hash table parameters
* @data: pointer to element data already in hashes
*
* Just like rhashtable_lookup_insert_key(), but this function returns the
* object if it exists, NULL if it does not and the insertion was successful,
* and an ERR_PTR otherwise.
*/
static inline void *rhashtable_lookup_get_insert_key(
struct rhashtable *ht, const void *key, struct rhash_head *obj,
const struct rhashtable_params params)
{ {
BUG_ON(!ht->p.obj_hashfn || !key); BUG_ON(!ht->p.obj_hashfn || !key);
......
...@@ -441,7 +441,8 @@ EXPORT_SYMBOL_GPL(rhashtable_insert_rehash); ...@@ -441,7 +441,8 @@ EXPORT_SYMBOL_GPL(rhashtable_insert_rehash);
struct bucket_table *rhashtable_insert_slow(struct rhashtable *ht, struct bucket_table *rhashtable_insert_slow(struct rhashtable *ht,
const void *key, const void *key,
struct rhash_head *obj, struct rhash_head *obj,
struct bucket_table *tbl) struct bucket_table *tbl,
void **data)
{ {
struct rhash_head *head; struct rhash_head *head;
unsigned int hash; unsigned int hash;
...@@ -452,8 +453,11 @@ struct bucket_table *rhashtable_insert_slow(struct rhashtable *ht, ...@@ -452,8 +453,11 @@ struct bucket_table *rhashtable_insert_slow(struct rhashtable *ht,
spin_lock_nested(rht_bucket_lock(tbl, hash), SINGLE_DEPTH_NESTING); spin_lock_nested(rht_bucket_lock(tbl, hash), SINGLE_DEPTH_NESTING);
err = -EEXIST; err = -EEXIST;
if (key && rhashtable_lookup_fast(ht, key, ht->p)) if (key) {
*data = rhashtable_lookup_fast(ht, key, ht->p);
if (*data)
goto exit; goto exit;
}
err = -E2BIG; err = -E2BIG;
if (unlikely(rht_grow_above_max(ht, tbl))) if (unlikely(rht_grow_above_max(ht, tbl)))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment