mm: avoid repeated anon_vma lock/unlock sequences in unlink_anon_vmas()
This matches the anon_vma_clone() case, and uses the same lock helper functions. Because of the need to potentially release the anon_vma's, it's a bit more complex, though. We traverse the 'vma->anon_vma_chain' in two phases: the first loop gets the anon_vma lock (with the helper function that only takes the lock once for the whole loop), and removes any entries that don't need any more processing. The second phase just traverses the remaining list entries (without holding the anon_vma lock), and does any actual freeing of the anon_vma's that is required. Signed-off-by: Peter Zijlstra <peterz@infradead.org> Tested-by: Hugh Dickins <hughd@google.com> Tested-by: Tim Chen <tim.c.chen@linux.intel.com> Cc: Andi Kleen <ak@linux.intel.com> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
bb4aa39676
commit
eee2acbae9
1 changed files with 28 additions and 21 deletions
49
mm/rmap.c
49
mm/rmap.c
|
@ -324,36 +324,43 @@ int anon_vma_fork(struct vm_area_struct *vma, struct vm_area_struct *pvma)
|
|||
return -ENOMEM;
|
||||
}
|
||||
|
||||
static void anon_vma_unlink(struct anon_vma_chain *anon_vma_chain)
|
||||
{
|
||||
struct anon_vma *anon_vma = anon_vma_chain->anon_vma;
|
||||
int empty;
|
||||
|
||||
/* If anon_vma_fork fails, we can get an empty anon_vma_chain. */
|
||||
if (!anon_vma)
|
||||
return;
|
||||
|
||||
anon_vma_lock(anon_vma);
|
||||
list_del(&anon_vma_chain->same_anon_vma);
|
||||
|
||||
/* We must garbage collect the anon_vma if it's empty */
|
||||
empty = list_empty(&anon_vma->head);
|
||||
anon_vma_unlock(anon_vma);
|
||||
|
||||
if (empty)
|
||||
put_anon_vma(anon_vma);
|
||||
}
|
||||
|
||||
void unlink_anon_vmas(struct vm_area_struct *vma)
|
||||
{
|
||||
struct anon_vma_chain *avc, *next;
|
||||
struct anon_vma *root = NULL;
|
||||
|
||||
/*
|
||||
* Unlink each anon_vma chained to the VMA. This list is ordered
|
||||
* from newest to oldest, ensuring the root anon_vma gets freed last.
|
||||
*/
|
||||
list_for_each_entry_safe(avc, next, &vma->anon_vma_chain, same_vma) {
|
||||
anon_vma_unlink(avc);
|
||||
struct anon_vma *anon_vma = avc->anon_vma;
|
||||
|
||||
root = lock_anon_vma_root(root, anon_vma);
|
||||
list_del(&avc->same_anon_vma);
|
||||
|
||||
/*
|
||||
* Leave empty anon_vmas on the list - we'll need
|
||||
* to free them outside the lock.
|
||||
*/
|
||||
if (list_empty(&anon_vma->head))
|
||||
continue;
|
||||
|
||||
list_del(&avc->same_vma);
|
||||
anon_vma_chain_free(avc);
|
||||
}
|
||||
unlock_anon_vma_root(root);
|
||||
|
||||
/*
|
||||
* Iterate the list once more, it now only contains empty and unlinked
|
||||
* anon_vmas, destroy them. Could not do before due to __put_anon_vma()
|
||||
* needing to acquire the anon_vma->root->mutex.
|
||||
*/
|
||||
list_for_each_entry_safe(avc, next, &vma->anon_vma_chain, same_vma) {
|
||||
struct anon_vma *anon_vma = avc->anon_vma;
|
||||
|
||||
put_anon_vma(anon_vma);
|
||||
|
||||
list_del(&avc->same_vma);
|
||||
anon_vma_chain_free(avc);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue