Commit 90c73cb2 authored by Mauro Carvalho Chehab's avatar Mauro Carvalho Chehab Committed by Paul E. McKenney

docs: RCU: Convert rcuref.txt to ReST

- Add a SPDX header;
- Adjust document title;
- Some whitespace fixes and new line breaks;
- Mark literal blocks as such;
- Add it to RCU/index.rst.
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+huawei@kernel.org>
Signed-off-by: default avatarPaul E. McKenney <paulmck@kernel.org>
parent 43cb5451
...@@ -18,6 +18,7 @@ RCU concepts ...@@ -18,6 +18,7 @@ RCU concepts
whatisRCU whatisRCU
rcu rcu
rculist_nulls rculist_nulls
rcuref
torture torture
listRCU listRCU
NMI-RCU NMI-RCU
......
Reference-count design for elements of lists/arrays protected by RCU. .. SPDX-License-Identifier: GPL-2.0
====================================================================
Reference-count design for elements of lists/arrays protected by RCU
====================================================================
Please note that the percpu-ref feature is likely your first Please note that the percpu-ref feature is likely your first
...@@ -12,10 +16,11 @@ please read on. ...@@ -12,10 +16,11 @@ please read on.
Reference counting on elements of lists which are protected by traditional Reference counting on elements of lists which are protected by traditional
reader/writer spinlocks or semaphores are straightforward: reader/writer spinlocks or semaphores are straightforward:
CODE LISTING A: CODE LISTING A::
1. 2.
add() search_and_reference() 1. 2.
{ { add() search_and_reference()
{ {
alloc_object read_lock(&list_lock); alloc_object read_lock(&list_lock);
... search_for_element ... search_for_element
atomic_set(&el->rc, 1); atomic_inc(&el->rc); atomic_set(&el->rc, 1); atomic_inc(&el->rc);
...@@ -23,16 +28,16 @@ add() search_and_reference() ...@@ -23,16 +28,16 @@ add() search_and_reference()
add_element read_unlock(&list_lock); add_element read_unlock(&list_lock);
... ... ... ...
write_unlock(&list_lock); } write_unlock(&list_lock); }
} }
3. 4. 3. 4.
release_referenced() delete() release_referenced() delete()
{ { { {
... write_lock(&list_lock); ... write_lock(&list_lock);
if(atomic_dec_and_test(&el->rc)) ... if(atomic_dec_and_test(&el->rc)) ...
kfree(el); kfree(el);
... remove_element ... remove_element
} write_unlock(&list_lock); } write_unlock(&list_lock);
... ...
if (atomic_dec_and_test(&el->rc)) if (atomic_dec_and_test(&el->rc))
kfree(el); kfree(el);
...@@ -46,10 +51,11 @@ search_and_reference() could potentially hold reference to an element which ...@@ -46,10 +51,11 @@ search_and_reference() could potentially hold reference to an element which
has already been deleted from the list/array. Use atomic_inc_not_zero() has already been deleted from the list/array. Use atomic_inc_not_zero()
in this scenario as follows: in this scenario as follows:
CODE LISTING B: CODE LISTING B::
1. 2.
add() search_and_reference() 1. 2.
{ { add() search_and_reference()
{ {
alloc_object rcu_read_lock(); alloc_object rcu_read_lock();
... search_for_element ... search_for_element
atomic_set(&el->rc, 1); if (!atomic_inc_not_zero(&el->rc)) { atomic_set(&el->rc, 1); if (!atomic_inc_not_zero(&el->rc)) {
...@@ -58,15 +64,15 @@ add() search_and_reference() ...@@ -58,15 +64,15 @@ add() search_and_reference()
add_element } add_element }
... ... ... ...
spin_unlock(&list_lock); rcu_read_unlock(); spin_unlock(&list_lock); rcu_read_unlock();
} } } }
3. 4. 3. 4.
release_referenced() delete() release_referenced() delete()
{ { { {
... spin_lock(&list_lock); ... spin_lock(&list_lock);
if (atomic_dec_and_test(&el->rc)) ... if (atomic_dec_and_test(&el->rc)) ...
call_rcu(&el->head, el_free); remove_element call_rcu(&el->head, el_free); remove_element
... spin_unlock(&list_lock); ... spin_unlock(&list_lock);
} ... } ...
if (atomic_dec_and_test(&el->rc)) if (atomic_dec_and_test(&el->rc))
call_rcu(&el->head, el_free); call_rcu(&el->head, el_free);
... ...
...@@ -82,10 +88,11 @@ search_and_reference() code path. In such cases, the ...@@ -82,10 +88,11 @@ search_and_reference() code path. In such cases, the
atomic_dec_and_test() may be moved from delete() to el_free() atomic_dec_and_test() may be moved from delete() to el_free()
as follows: as follows:
CODE LISTING C: CODE LISTING C::
1. 2.
add() search_and_reference() 1. 2.
{ { add() search_and_reference()
{ {
alloc_object rcu_read_lock(); alloc_object rcu_read_lock();
... search_for_element ... search_for_element
atomic_set(&el->rc, 1); atomic_inc(&el->rc); atomic_set(&el->rc, 1); atomic_inc(&el->rc);
...@@ -94,20 +101,20 @@ add() search_and_reference() ...@@ -94,20 +101,20 @@ add() search_and_reference()
add_element rcu_read_unlock(); add_element rcu_read_unlock();
... } ... }
spin_unlock(&list_lock); 4. spin_unlock(&list_lock); 4.
} delete() } delete()
3. { 3. {
release_referenced() spin_lock(&list_lock); release_referenced() spin_lock(&list_lock);
{ ... { ...
... remove_element ... remove_element
if (atomic_dec_and_test(&el->rc)) spin_unlock(&list_lock); if (atomic_dec_and_test(&el->rc)) spin_unlock(&list_lock);
kfree(el); ... kfree(el); ...
... call_rcu(&el->head, el_free); ... call_rcu(&el->head, el_free);
} ... } ...
5. } 5. }
void el_free(struct rcu_head *rhp) void el_free(struct rcu_head *rhp)
{ {
release_referenced(); release_referenced();
} }
The key point is that the initial reference added by add() is not removed The key point is that the initial reference added by add() is not removed
until after a grace period has elapsed following removal. This means that until after a grace period has elapsed following removal. This means that
...@@ -130,11 +137,11 @@ the eventual invocation of kfree(), which is usually not a problem on ...@@ -130,11 +137,11 @@ the eventual invocation of kfree(), which is usually not a problem on
modern computer systems, even the small ones. modern computer systems, even the small ones.
In cases where delete() can sleep, synchronize_rcu() can be called from In cases where delete() can sleep, synchronize_rcu() can be called from
delete(), so that el_free() can be subsumed into delete as follows: delete(), so that el_free() can be subsumed into delete as follows::
4. 4.
delete() delete()
{ {
spin_lock(&list_lock); spin_lock(&list_lock);
... ...
remove_element remove_element
...@@ -144,7 +151,7 @@ delete() ...@@ -144,7 +151,7 @@ delete()
if (atomic_dec_and_test(&el->rc)) if (atomic_dec_and_test(&el->rc))
kfree(el); kfree(el);
... ...
} }
As additional examples in the kernel, the pattern in listing C is used by As additional examples in the kernel, the pattern in listing C is used by
reference counting of struct pid, while the pattern in listing B is used by reference counting of struct pid, while the pattern in listing B is used by
......
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