diff options
| author | Vladimir Davydov <vdavydov@parallels.com> | 2014-08-08 17:21:56 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-08-08 18:57:23 -0400 |
| commit | 41f727fde1fe40efeb4fef6fdce74ff794be5aeb (patch) | |
| tree | b4946b74270cfe286580f44659000aef0d204277 /kernel | |
| parent | 8f053ac11f96cc6edcabcbb154c9cf06c5d63333 (diff) | |
fork/exec: cleanup mm initialization
mm initialization on fork/exec is spread all over the place, which makes
the code look inconsistent.
We have mm_init(), which is supposed to init/nullify mm's internals, but
it doesn't init all the fields it should:
- on fork ->mmap,mm_rb,vmacache_seqnum,map_count,mm_cpumask,locked_vm
are zeroed in dup_mmap();
- on fork ->pmd_huge_pte is zeroed in dup_mm(), immediately before
calling mm_init();
- ->cpu_vm_mask_var ptr is initialized by mm_init_cpumask(), which is
called before mm_init() on both fork and exec;
- ->context is initialized by init_new_context(), which is called after
mm_init() on both fork and exec;
Let's consolidate all the initializations in mm_init() to make the code
look cleaner.
Signed-off-by: Vladimir Davydov <vdavydov@parallels.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: David Rientjes <rientjes@google.com>
Cc: Christoph Lameter <cl@linux.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'kernel')
| -rw-r--r-- | kernel/fork.c | 47 |
1 files changed, 20 insertions, 27 deletions
diff --git a/kernel/fork.c b/kernel/fork.c index f6f5086c9e7d..418b52a9ec6a 100644 --- a/kernel/fork.c +++ b/kernel/fork.c | |||
| @@ -374,12 +374,6 @@ static int dup_mmap(struct mm_struct *mm, struct mm_struct *oldmm) | |||
| 374 | */ | 374 | */ |
| 375 | down_write_nested(&mm->mmap_sem, SINGLE_DEPTH_NESTING); | 375 | down_write_nested(&mm->mmap_sem, SINGLE_DEPTH_NESTING); |
| 376 | 376 | ||
| 377 | mm->locked_vm = 0; | ||
| 378 | mm->mmap = NULL; | ||
| 379 | mm->vmacache_seqnum = 0; | ||
| 380 | mm->map_count = 0; | ||
| 381 | cpumask_clear(mm_cpumask(mm)); | ||
| 382 | mm->mm_rb = RB_ROOT; | ||
| 383 | rb_link = &mm->mm_rb.rb_node; | 377 | rb_link = &mm->mm_rb.rb_node; |
| 384 | rb_parent = NULL; | 378 | rb_parent = NULL; |
| 385 | pprev = &mm->mmap; | 379 | pprev = &mm->mmap; |
| @@ -538,17 +532,27 @@ static void mm_init_aio(struct mm_struct *mm) | |||
| 538 | 532 | ||
| 539 | static struct mm_struct *mm_init(struct mm_struct *mm, struct task_struct *p) | 533 | static struct mm_struct *mm_init(struct mm_struct *mm, struct task_struct *p) |
| 540 | { | 534 | { |
| 535 | mm->mmap = NULL; | ||
| 536 | mm->mm_rb = RB_ROOT; | ||
| 537 | mm->vmacache_seqnum = 0; | ||
| 541 | atomic_set(&mm->mm_users, 1); | 538 | atomic_set(&mm->mm_users, 1); |
| 542 | atomic_set(&mm->mm_count, 1); | 539 | atomic_set(&mm->mm_count, 1); |
| 543 | init_rwsem(&mm->mmap_sem); | 540 | init_rwsem(&mm->mmap_sem); |
| 544 | INIT_LIST_HEAD(&mm->mmlist); | 541 | INIT_LIST_HEAD(&mm->mmlist); |
| 545 | mm->core_state = NULL; | 542 | mm->core_state = NULL; |
| 546 | atomic_long_set(&mm->nr_ptes, 0); | 543 | atomic_long_set(&mm->nr_ptes, 0); |
| 544 | mm->map_count = 0; | ||
| 545 | mm->locked_vm = 0; | ||
| 547 | memset(&mm->rss_stat, 0, sizeof(mm->rss_stat)); | 546 | memset(&mm->rss_stat, 0, sizeof(mm->rss_stat)); |
| 548 | spin_lock_init(&mm->page_table_lock); | 547 | spin_lock_init(&mm->page_table_lock); |
| 548 | mm_init_cpumask(mm); | ||
| 549 | mm_init_aio(mm); | 549 | mm_init_aio(mm); |
| 550 | mm_init_owner(mm, p); | 550 | mm_init_owner(mm, p); |
| 551 | mmu_notifier_mm_init(mm); | ||
| 551 | clear_tlb_flush_pending(mm); | 552 | clear_tlb_flush_pending(mm); |
| 553 | #if defined(CONFIG_TRANSPARENT_HUGEPAGE) && !USE_SPLIT_PMD_PTLOCKS | ||
| 554 | mm->pmd_huge_pte = NULL; | ||
| 555 | #endif | ||
| 552 | 556 | ||
| 553 | if (current->mm) { | 557 | if (current->mm) { |
| 554 | mm->flags = current->mm->flags & MMF_INIT_MASK; | 558 | mm->flags = current->mm->flags & MMF_INIT_MASK; |
| @@ -558,11 +562,17 @@ static struct mm_struct *mm_init(struct mm_struct *mm, struct task_struct *p) | |||
| 558 | mm->def_flags = 0; | 562 | mm->def_flags = 0; |
| 559 | } | 563 | } |
| 560 | 564 | ||
| 561 | if (likely(!mm_alloc_pgd(mm))) { | 565 | if (mm_alloc_pgd(mm)) |
| 562 | mmu_notifier_mm_init(mm); | 566 | goto fail_nopgd; |
| 563 | return mm; | 567 | |
| 564 | } | 568 | if (init_new_context(p, mm)) |
| 569 | goto fail_nocontext; | ||
| 565 | 570 | ||
| 571 | return mm; | ||
| 572 | |||
| 573 | fail_nocontext: | ||
| 574 | mm_free_pgd(mm); | ||
| 575 | fail_nopgd: | ||
| 566 | free_mm(mm); | 576 | free_mm(mm); |
| 567 | return NULL; | 577 | return NULL; |
| 568 | } | 578 | } |
| @@ -596,7 +606,6 @@ struct mm_struct *mm_alloc(void) | |||
| 596 | return NULL; | 606 | return NULL; |
| 597 | 607 | ||
| 598 | memset(mm, 0, sizeof(*mm)); | 608 | memset(mm, 0, sizeof(*mm)); |
| 599 | mm_init_cpumask(mm); | ||
| 600 | return mm_init(mm, current); | 609 | return mm_init(mm, current); |
| 601 | } | 610 | } |
| 602 | 611 | ||
| @@ -828,17 +837,10 @@ static struct mm_struct *dup_mm(struct task_struct *tsk) | |||
| 828 | goto fail_nomem; | 837 | goto fail_nomem; |
| 829 | 838 | ||
| 830 | memcpy(mm, oldmm, sizeof(*mm)); | 839 | memcpy(mm, oldmm, sizeof(*mm)); |
| 831 | mm_init_cpumask(mm); | ||
| 832 | 840 | ||
| 833 | #if defined(CONFIG_TRANSPARENT_HUGEPAGE) && !USE_SPLIT_PMD_PTLOCKS | ||
| 834 | mm->pmd_huge_pte = NULL; | ||
| 835 | #endif | ||
| 836 | if (!mm_init(mm, tsk)) | 841 | if (!mm_init(mm, tsk)) |
| 837 | goto fail_nomem; | 842 | goto fail_nomem; |
| 838 | 843 | ||
| 839 | if (init_new_context(tsk, mm)) | ||
| 840 | goto fail_nocontext; | ||
| 841 | |||
| 842 | dup_mm_exe_file(oldmm, mm); | 844 | dup_mm_exe_file(oldmm, mm); |
| 843 | 845 | ||
| 844 | err = dup_mmap(mm, oldmm); | 846 | err = dup_mmap(mm, oldmm); |
| @@ -860,15 +862,6 @@ free_pt: | |||
| 860 | 862 | ||
| 861 | fail_nomem: | 863 | fail_nomem: |
| 862 | return NULL; | 864 | return NULL; |
| 863 | |||
| 864 | fail_nocontext: | ||
| 865 | /* | ||
| 866 | * If init_new_context() failed, we cannot use mmput() to free the mm | ||
| 867 | * because it calls destroy_context() | ||
| 868 | */ | ||
| 869 | mm_free_pgd(mm); | ||
| 870 | free_mm(mm); | ||
| 871 | return NULL; | ||
| 872 | } | 865 | } |
| 873 | 866 | ||
| 874 | static int copy_mm(unsigned long clone_flags, struct task_struct *tsk) | 867 | static int copy_mm(unsigned long clone_flags, struct task_struct *tsk) |
