Skip to content

Commit

Permalink
llist: fix/simplify llist_add() and llist_add_batch()
Browse files Browse the repository at this point in the history
1. This is mostly theoretical, but llist_add*() need ACCESS_ONCE().

   Otherwise it is not guaranteed that the first cmpxchg() uses the
   same value for old_entry and new_last->next.

2. These helpers cache the result of cmpxchg() and read the initial
   value of head->first before the main loop. I do not think this
   makes sense. In the likely case cmpxchg() succeeds, otherwise
   it doesn't hurt to reload head->first.

   I think it would be better to simplify the code and simply read
   ->first before cmpxchg().

Signed-off-by: Oleg Nesterov <[email protected]>
Cc: Al Viro <[email protected]>
Cc: Andrey Vagin <[email protected]>
Cc: "Eric W. Biederman" <[email protected]>
Cc: David Howells <[email protected]>
Cc: Huang Ying <[email protected]>
Cc: Peter Zijlstra <[email protected]>
Signed-off-by: Andrew Morton <[email protected]>
Signed-off-by: Al Viro <[email protected]>
  • Loading branch information
oleg-nesterov authored and Al Viro committed Jul 13, 2013
1 parent 4f5e65a commit fb4214d
Show file tree
Hide file tree
Showing 2 changed files with 12 additions and 22 deletions.
19 changes: 7 additions & 12 deletions include/linux/llist.h
Original file line number Diff line number Diff line change
Expand Up @@ -151,18 +151,13 @@ static inline struct llist_node *llist_next(struct llist_node *node)
*/
static inline bool llist_add(struct llist_node *new, struct llist_head *head)
{
struct llist_node *entry, *old_entry;

entry = head->first;
for (;;) {
old_entry = entry;
new->next = entry;
entry = cmpxchg(&head->first, old_entry, new);
if (entry == old_entry)
break;
}

return old_entry == NULL;
struct llist_node *first;

do {
new->next = first = ACCESS_ONCE(head->first);
} while (cmpxchg(&head->first, first, new) != first);

return !first;
}

/**
Expand Down
15 changes: 5 additions & 10 deletions lib/llist.c
Original file line number Diff line number Diff line change
Expand Up @@ -39,18 +39,13 @@
bool llist_add_batch(struct llist_node *new_first, struct llist_node *new_last,
struct llist_head *head)
{
struct llist_node *entry, *old_entry;
struct llist_node *first;

entry = head->first;
for (;;) {
old_entry = entry;
new_last->next = entry;
entry = cmpxchg(&head->first, old_entry, new_first);
if (entry == old_entry)
break;
}
do {
new_last->next = first = ACCESS_ONCE(head->first);
} while (cmpxchg(&head->first, first, new_first) != first);

return old_entry == NULL;
return !first;
}
EXPORT_SYMBOL_GPL(llist_add_batch);

Expand Down

0 comments on commit fb4214d

Please sign in to comment.