blob: c089196627045181b07b2df41ec823e4d6a60d15 [file] [log] [blame]
Mike Rapoportd18edf52018-03-21 21:22:39 +02001.. _split_page_table_lock:
2
3=====================
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -08004Split page table lock
5=====================
6
7Originally, mm->page_table_lock spinlock protected all page tables of the
8mm_struct. But this approach leads to poor page fault scalability of
9multi-threaded applications due high contention on the lock. To improve
10scalability, split page table lock was introduced.
11
12With split page table lock we have separate per-table lock to serialize
13access to the table. At the moment we use split lock for PTE and PMD
14tables. Access to higher level tables protected by mm->page_table_lock.
15
16There are helpers to lock/unlock a table and other accessor functions:
Mike Rapoportd18edf52018-03-21 21:22:39 +020017
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080018 - pte_offset_map_lock()
19 maps pte and takes PTE table lock, returns pointer to the taken
20 lock;
21 - pte_unmap_unlock()
22 unlocks and unmaps PTE table;
23 - pte_alloc_map_lock()
24 allocates PTE table if needed and take the lock, returns pointer
25 to taken lock or NULL if allocation failed;
26 - pte_lockptr()
27 returns pointer to PTE table lock;
28 - pmd_lock()
29 takes PMD table lock, returns pointer to taken lock;
30 - pmd_lockptr()
31 returns pointer to PMD table lock;
32
33Split page table lock for PTE tables is enabled compile-time if
34CONFIG_SPLIT_PTLOCK_CPUS (usually 4) is less or equal to NR_CPUS.
Rolf Eike Beer96c0f7c2021-01-12 14:19:36 +010035If split lock is disabled, all tables are guarded by mm->page_table_lock.
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080036
37Split page table lock for PMD tables is enabled, if it's enabled for PTE
38tables and the architecture supports it (see below).
39
40Hugetlb and split page table lock
Mike Rapoportd18edf52018-03-21 21:22:39 +020041=================================
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080042
43Hugetlb can support several page sizes. We use split lock only for PMD
44level, but not for PUD.
45
46Hugetlb-specific helpers:
Mike Rapoportd18edf52018-03-21 21:22:39 +020047
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080048 - huge_pte_lock()
49 takes pmd split lock for PMD_SIZE page, mm->page_table_lock
50 otherwise;
51 - huge_pte_lockptr()
52 returns pointer to table lock;
53
54Support of split page table lock by an architecture
Mike Rapoportd18edf52018-03-21 21:22:39 +020055===================================================
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080056
Mark Rutlandb4ed71f2019-09-25 16:49:46 -070057There's no need in special enabling of PTE split page table lock: everything
58required is done by pgtable_pte_page_ctor() and pgtable_pte_page_dtor(), which
59must be called on PTE table allocation / freeing.
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080060
61Make sure the architecture doesn't use slab allocator for page table
Kirill A. Shutemov1d798ca2015-11-06 16:29:54 -080062allocation: slab uses page->slab_cache for its pages.
63This field shares storage with page->ptl.
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080064
65PMD split lock only makes sense if you have more than two page table
66levels.
67
68PMD split lock enabling requires pgtable_pmd_page_ctor() call on PMD table
69allocation and pgtable_pmd_page_dtor() on freeing.
70
Kirill A. Shutemovc2836102013-11-21 14:32:09 -080071Allocation usually happens in pmd_alloc_one(), freeing in pmd_free() and
72pmd_free_tlb(), but make sure you cover all PMD table allocation / freeing
73paths: i.e X86_PAE preallocate few PMDs on pgd_alloc().
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080074
75With everything in place you can set CONFIG_ARCH_ENABLE_SPLIT_PMD_PTLOCK.
76
Mark Rutlandb4ed71f2019-09-25 16:49:46 -070077NOTE: pgtable_pte_page_ctor() and pgtable_pmd_page_ctor() can fail -- it must
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080078be handled properly.
79
80page->ptl
Mike Rapoportd18edf52018-03-21 21:22:39 +020081=========
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080082
83page->ptl is used to access split page table lock, where 'page' is struct
84page of page containing the table. It shares storage with page->private
85(and few other fields in union).
86
87To avoid increasing size of struct page and have best performance, we use a
88trick:
Mike Rapoportd18edf52018-03-21 21:22:39 +020089
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080090 - if spinlock_t fits into long, we use page->ptr as spinlock, so we
91 can avoid indirect access and save a cache line.
92 - if size of spinlock_t is bigger then size of long, we use page->ptl as
93 pointer to spinlock_t and allocate it dynamically. This allows to use
94 split lock with enabled DEBUG_SPINLOCK or DEBUG_LOCK_ALLOC, but costs
95 one more cache line for indirect access;
96
Mark Rutlandb4ed71f2019-09-25 16:49:46 -070097The spinlock_t allocated in pgtable_pte_page_ctor() for PTE table and in
Kirill A. Shutemov49076ec2013-11-14 14:31:51 -080098pgtable_pmd_page_ctor() for PMD table.
99
100Please, never access page->ptl directly -- use appropriate helper.