Skip to content

Commit 870994a

Browse files
David Herrmannniclimcy
David Herrmann
authored andcommitted
mm: allow drivers to prevent new writable mappings
This patch (of 6): The i_mmap_writable field counts existing writable mappings of an address_space. To allow drivers to prevent new writable mappings, make this counter signed and prevent new writable mappings if it is negative. This is modelled after i_writecount and DENYWRITE. This will be required by the shmem-sealing infrastructure to prevent any new writable mappings after the WRITE seal has been set. In case there exists a writable mapping, this operation will fail with EBUSY. Note that we rely on the fact that iff you already own a writable mapping, you can increase the counter without using the helpers. This is the same that we do for i_writecount. Signed-off-by: David Herrmann <dh.herrmann@gmail.com> Acked-by: Hugh Dickins <hughd@google.com> Cc: Michael Kerrisk <mtk.manpages@gmail.com> Cc: Ryan Lortie <desrt@desrt.ca> Cc: Lennart Poettering <lennart@poettering.net> Cc: Daniel Mack <zonque@gmail.com> Cc: Andy Lutomirski <luto@amacapital.net> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org> Change-Id: I329bb3347b76c8b1f06a591a7ff3c8b078541353
1 parent 6cae088 commit 870994a

File tree

5 files changed

+54
-9
lines changed

5 files changed

+54
-9
lines changed

fs/inode.c

+1
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,7 @@ int inode_init_always(struct super_block *sb, struct inode *inode)
164164
mapping->a_ops = &empty_aops;
165165
mapping->host = inode;
166166
mapping->flags = 0;
167+
atomic_set(&mapping->i_mmap_writable, 0);
167168
mapping_set_gfp_mask(mapping, GFP_HIGHUSER_MOVABLE);
168169
mapping->private_data = NULL;
169170
mapping->backing_dev_info = &default_backing_dev_info;

include/linux/fs.h

+27-2
Original file line numberDiff line numberDiff line change
@@ -422,7 +422,7 @@ struct address_space {
422422
struct inode *host; /* owner: inode, block_device */
423423
struct radix_tree_root page_tree; /* radix tree of all pages */
424424
spinlock_t tree_lock; /* and lock protecting it */
425-
unsigned int i_mmap_writable;/* count VM_SHARED mappings */
425+
atomic_t i_mmap_writable;/* count VM_SHARED mappings */
426426
struct rb_root i_mmap; /* tree of private and shared mappings */
427427
struct list_head i_mmap_nonlinear;/*list VM_NONLINEAR mappings */
428428
struct mutex i_mmap_mutex; /* protect tree, count, list */
@@ -504,10 +504,35 @@ static inline int mapping_mapped(struct address_space *mapping)
504504
* Note that i_mmap_writable counts all VM_SHARED vmas: do_mmap_pgoff
505505
* marks vma as VM_SHARED if it is shared, and the file was opened for
506506
* writing i.e. vma may be mprotected writable even if now readonly.
507+
*
508+
* If i_mmap_writable is negative, no new writable mappings are allowed. You
509+
* can only deny writable mappings, if none exists right now.
507510
*/
508511
static inline int mapping_writably_mapped(struct address_space *mapping)
509512
{
510-
return mapping->i_mmap_writable != 0;
513+
return atomic_read(&mapping->i_mmap_writable) > 0;
514+
}
515+
516+
static inline int mapping_map_writable(struct address_space *mapping)
517+
{
518+
return atomic_inc_unless_negative(&mapping->i_mmap_writable) ?
519+
0 : -EPERM;
520+
}
521+
522+
static inline void mapping_unmap_writable(struct address_space *mapping)
523+
{
524+
atomic_dec(&mapping->i_mmap_writable);
525+
}
526+
527+
static inline int mapping_deny_writable(struct address_space *mapping)
528+
{
529+
return atomic_dec_unless_positive(&mapping->i_mmap_writable) ?
530+
0 : -EBUSY;
531+
}
532+
533+
static inline void mapping_allow_writable(struct address_space *mapping)
534+
{
535+
atomic_inc(&mapping->i_mmap_writable);
511536
}
512537

513538
/*

kernel/fork.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -444,7 +444,7 @@ static int dup_mmap(struct mm_struct *mm, struct mm_struct *oldmm)
444444
atomic_dec(&inode->i_writecount);
445445
mutex_lock(&mapping->i_mmap_mutex);
446446
if (tmp->vm_flags & VM_SHARED)
447-
mapping->i_mmap_writable++;
447+
atomic_inc(&mapping->i_mmap_writable);
448448
flush_dcache_mmap_lock(mapping);
449449
/* insert tmp into the share list, just after mpnt */
450450
if (unlikely(tmp->vm_flags & VM_NONLINEAR))

mm/mmap.c

+24-6
Original file line numberDiff line numberDiff line change
@@ -225,7 +225,7 @@ static void __remove_shared_vm_struct(struct vm_area_struct *vma,
225225
if (vma->vm_flags & VM_DENYWRITE)
226226
atomic_inc(&file_inode(file)->i_writecount);
227227
if (vma->vm_flags & VM_SHARED)
228-
mapping->i_mmap_writable--;
228+
mapping_unmap_writable(mapping);
229229

230230
flush_dcache_mmap_lock(mapping);
231231
if (unlikely(vma->vm_flags & VM_NONLINEAR))
@@ -640,7 +640,7 @@ static void __vma_link_file(struct vm_area_struct *vma)
640640
if (vma->vm_flags & VM_DENYWRITE)
641641
atomic_dec(&file_inode(file)->i_writecount);
642642
if (vma->vm_flags & VM_SHARED)
643-
mapping->i_mmap_writable++;
643+
atomic_inc(&mapping->i_mmap_writable);
644644

645645
flush_dcache_mmap_lock(mapping);
646646
if (unlikely(vma->vm_flags & VM_NONLINEAR))
@@ -1582,6 +1582,17 @@ unsigned long mmap_region(struct file *file, unsigned long addr,
15821582
if (error)
15831583
goto free_vma;
15841584
}
1585+
if (vm_flags & VM_SHARED) {
1586+
error = mapping_map_writable(file->f_mapping);
1587+
if (error)
1588+
goto allow_write_and_free_vma;
1589+
}
1590+
1591+
/* ->mmap() can change vma->vm_file, but must guarantee that
1592+
* vma_link() below can deny write-access if VM_DENYWRITE is set
1593+
* and map writably if VM_SHARED is set. This usually means the
1594+
* new file must not have been exposed to user-space, yet.
1595+
*/
15851596
vma->vm_file = get_file(file);
15861597
error = file->f_op->mmap(file, vma);
15871598
if (error)
@@ -1622,8 +1633,12 @@ unsigned long mmap_region(struct file *file, unsigned long addr,
16221633

16231634
vma_link(mm, vma, prev, rb_link, rb_parent);
16241635
/* Once vma denies write, undo our temporary denial count */
1625-
if (vm_flags & VM_DENYWRITE)
1626-
allow_write_access(file);
1636+
if (file) {
1637+
if (vm_flags & VM_SHARED)
1638+
mapping_unmap_writable(file->f_mapping);
1639+
if (vm_flags & VM_DENYWRITE)
1640+
allow_write_access(file);
1641+
}
16271642
file = vma->vm_file;
16281643
out:
16291644
perf_event_mmap(vma);
@@ -1643,14 +1658,17 @@ unsigned long mmap_region(struct file *file, unsigned long addr,
16431658
return addr;
16441659

16451660
unmap_and_free_vma:
1646-
if (vm_flags & VM_DENYWRITE)
1647-
allow_write_access(file);
16481661
vma->vm_file = NULL;
16491662
fput(file);
16501663

16511664
/* Undo any partial mapping done by a device driver. */
16521665
unmap_region(mm, vma, prev, vma->vm_start, vma->vm_end);
16531666
charged = 0;
1667+
if (vm_flags & VM_SHARED)
1668+
mapping_unmap_writable(file->f_mapping);
1669+
allow_write_and_free_vma:
1670+
if (vm_flags & VM_DENYWRITE)
1671+
allow_write_access(file);
16541672
free_vma:
16551673
kmem_cache_free(vm_area_cachep, vma);
16561674
unacct_error:

mm/swap_state.c

+1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ static struct backing_dev_info swap_backing_dev_info = {
3939
struct address_space swapper_spaces[MAX_SWAPFILES] = {
4040
[0 ... MAX_SWAPFILES - 1] = {
4141
.page_tree = RADIX_TREE_INIT(GFP_ATOMIC|__GFP_NOWARN),
42+
.i_mmap_writable = ATOMIC_INIT(0),
4243
.a_ops = &swap_aops,
4344
.backing_dev_info = &swap_backing_dev_info,
4445
}

0 commit comments

Comments
 (0)