]> Git Repo - linux.git/commit - net/xfrm/xfrm_state.c
xfrm: state: fix out-of-bounds read during lookup
authorFlorian Westphal <[email protected]>
Thu, 28 Nov 2024 14:26:25 +0000 (15:26 +0100)
committerSteffen Klassert <[email protected]>
Thu, 5 Dec 2024 08:43:02 +0000 (09:43 +0100)
commite952837f3ddb0ff726d5b582aa1aad9aa38d024d
tree34aefa56fd76e7f87511379a6e6ceda14724d2c9
parentc05c5e5aa163f4682ca97a2f0536575fc7dbdecb
xfrm: state: fix out-of-bounds read during lookup

lookup and resize can run in parallel.

The xfrm_state_hash_generation seqlock ensures a retry, but the hash
functions can observe a hmask value that is too large for the new hlist
array.

rehash does:
  rcu_assign_pointer(net->xfrm.state_bydst, ndst) [..]
  net->xfrm.state_hmask = nhashmask;

While state lookup does:
  h = xfrm_dst_hash(net, daddr, saddr, tmpl->reqid, encap_family);
  hlist_for_each_entry_rcu(x, net->xfrm.state_bydst + h, bydst) {

This is only safe in case the update to state_bydst is larger than
net->xfrm.xfrm_state_hmask (or if the lookup function gets
serialized via state spinlock again).

Fix this by prefetching state_hmask and the associated pointers.
The xfrm_state_hash_generation seqlock retry will ensure that the pointer
and the hmask will be consistent.

The existing helpers, like xfrm_dst_hash(), are now unsafe for RCU side,
add lockdep assertions to document that they are only safe for insert
side.

xfrm_state_lookup_byaddr() uses the spinlock rather than RCU.
AFAICS this is an oversight from back when state lookup was converted to
RCU, this lock should be replaced with RCU in a future patch.

Reported-by: [email protected]
Closes: https://lore.kernel.org/netdev/CACT4Y+azwfrE3uz6A5ZErov5YN2LYBN5KrsymBerT36VU8qzBA@mail.gmail.com/
Diagnosed-by: Dmitry Vyukov <[email protected]>
Fixes: c2f672fc9464 ("xfrm: state lookup can be lockless")
Signed-off-by: Florian Westphal <[email protected]>
Signed-off-by: Steffen Klassert <[email protected]>
net/xfrm/xfrm_state.c
This page took 0.051116 seconds and 4 git commands to generate.