2022-01-14 14:06:37 -08:00
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright ( c ) 2021 , Google LLC .
* Pasha Tatashin < pasha . tatashin @ soleen . com >
*/
2022-11-01 22:14:09 +01:00
# include <linux/kstrtox.h>
2022-01-14 14:06:37 -08:00
# include <linux/mm.h>
# include <linux/page_table_check.h>
# undef pr_fmt
# define pr_fmt(fmt) "page_table_check: " fmt
struct page_table_check {
atomic_t anon_map_count ;
atomic_t file_map_count ;
} ;
static bool __page_table_check_enabled __initdata =
IS_ENABLED ( CONFIG_PAGE_TABLE_CHECK_ENFORCED ) ;
DEFINE_STATIC_KEY_TRUE ( page_table_check_disabled ) ;
EXPORT_SYMBOL ( page_table_check_disabled ) ;
static int __init early_page_table_check_param ( char * buf )
{
2022-11-01 22:14:09 +01:00
return kstrtobool ( buf , & __page_table_check_enabled ) ;
2022-01-14 14:06:37 -08:00
}
early_param ( " page_table_check " , early_page_table_check_param ) ;
static bool __init need_page_table_check ( void )
{
return __page_table_check_enabled ;
}
static void __init init_page_table_check ( void )
{
if ( ! __page_table_check_enabled )
return ;
static_branch_disable ( & page_table_check_disabled ) ;
}
struct page_ext_operations page_table_check_ops = {
. size = sizeof ( struct page_table_check ) ,
. need = need_page_table_check ,
. init = init_page_table_check ,
2023-01-13 15:42:53 +00:00
. need_shared_flags = false ,
2022-01-14 14:06:37 -08:00
} ;
static struct page_table_check * get_page_table_check ( struct page_ext * page_ext )
{
BUG_ON ( ! page_ext ) ;
2023-07-18 22:58:11 +08:00
return page_ext_data ( page_ext , & page_table_check_ops ) ;
2022-01-14 14:06:37 -08:00
}
/*
2022-09-16 17:04:34 +08:00
* An entry is removed from the page table , decrement the counters for that page
2022-01-14 14:06:37 -08:00
* verify that it is of correct type and counters do not become negative .
*/
2023-07-14 01:26:29 +08:00
static void page_table_check_clear ( unsigned long pfn , unsigned long pgcnt )
2022-01-14 14:06:37 -08:00
{
struct page_ext * page_ext ;
struct page * page ;
2022-02-03 20:49:15 -08:00
unsigned long i ;
2022-01-14 14:06:37 -08:00
bool anon ;
if ( ! pfn_valid ( pfn ) )
return ;
page = pfn_to_page ( pfn ) ;
2022-08-18 19:20:00 +05:30
page_ext = page_ext_get ( page ) ;
2023-05-15 21:09:58 +08:00
BUG_ON ( PageSlab ( page ) ) ;
2022-01-14 14:06:37 -08:00
anon = PageAnon ( page ) ;
for ( i = 0 ; i < pgcnt ; i + + ) {
struct page_table_check * ptc = get_page_table_check ( page_ext ) ;
if ( anon ) {
BUG_ON ( atomic_read ( & ptc - > file_map_count ) ) ;
BUG_ON ( atomic_dec_return ( & ptc - > anon_map_count ) < 0 ) ;
} else {
BUG_ON ( atomic_read ( & ptc - > anon_map_count ) ) ;
BUG_ON ( atomic_dec_return ( & ptc - > file_map_count ) < 0 ) ;
}
page_ext = page_ext_next ( page_ext ) ;
}
2022-08-18 19:20:00 +05:30
page_ext_put ( page_ext ) ;
2022-01-14 14:06:37 -08:00
}
/*
2022-09-16 17:04:34 +08:00
* A new entry is added to the page table , increment the counters for that page
2022-01-14 14:06:37 -08:00
* verify that it is of correct type and is not being mapped with a different
* type to a different process .
*/
2023-07-14 01:26:30 +08:00
static void page_table_check_set ( unsigned long pfn , unsigned long pgcnt ,
2022-01-14 14:06:37 -08:00
bool rw )
{
struct page_ext * page_ext ;
struct page * page ;
2022-02-03 20:49:15 -08:00
unsigned long i ;
2022-01-14 14:06:37 -08:00
bool anon ;
if ( ! pfn_valid ( pfn ) )
return ;
page = pfn_to_page ( pfn ) ;
2022-08-18 19:20:00 +05:30
page_ext = page_ext_get ( page ) ;
2023-05-15 21:09:58 +08:00
BUG_ON ( PageSlab ( page ) ) ;
2022-01-14 14:06:37 -08:00
anon = PageAnon ( page ) ;
for ( i = 0 ; i < pgcnt ; i + + ) {
struct page_table_check * ptc = get_page_table_check ( page_ext ) ;
if ( anon ) {
BUG_ON ( atomic_read ( & ptc - > file_map_count ) ) ;
BUG_ON ( atomic_inc_return ( & ptc - > anon_map_count ) > 1 & & rw ) ;
} else {
BUG_ON ( atomic_read ( & ptc - > anon_map_count ) ) ;
BUG_ON ( atomic_inc_return ( & ptc - > file_map_count ) < 0 ) ;
}
page_ext = page_ext_next ( page_ext ) ;
}
2022-08-18 19:20:00 +05:30
page_ext_put ( page_ext ) ;
2022-01-14 14:06:37 -08:00
}
/*
* page is on free list , or is being allocated , verify that counters are zeroes
* crash if they are not .
*/
void __page_table_check_zero ( struct page * page , unsigned int order )
{
2022-08-18 19:20:00 +05:30
struct page_ext * page_ext ;
2022-02-03 20:49:15 -08:00
unsigned long i ;
2022-01-14 14:06:37 -08:00
2023-05-15 21:09:58 +08:00
BUG_ON ( PageSlab ( page ) ) ;
2022-08-18 19:20:00 +05:30
page_ext = page_ext_get ( page ) ;
2022-01-14 14:06:37 -08:00
BUG_ON ( ! page_ext ) ;
2022-02-03 20:49:15 -08:00
for ( i = 0 ; i < ( 1ul < < order ) ; i + + ) {
2022-01-14 14:06:37 -08:00
struct page_table_check * ptc = get_page_table_check ( page_ext ) ;
BUG_ON ( atomic_read ( & ptc - > anon_map_count ) ) ;
BUG_ON ( atomic_read ( & ptc - > file_map_count ) ) ;
page_ext = page_ext_next ( page_ext ) ;
}
2022-08-18 19:20:00 +05:30
page_ext_put ( page_ext ) ;
2022-01-14 14:06:37 -08:00
}
2023-07-14 01:26:31 +08:00
void __page_table_check_pte_clear ( struct mm_struct * mm , pte_t pte )
2022-01-14 14:06:37 -08:00
{
if ( & init_mm = = mm )
return ;
if ( pte_user_accessible_page ( pte ) ) {
2023-07-14 01:26:29 +08:00
page_table_check_clear ( pte_pfn ( pte ) , PAGE_SIZE > > PAGE_SHIFT ) ;
2022-01-14 14:06:37 -08:00
}
}
EXPORT_SYMBOL ( __page_table_check_pte_clear ) ;
2023-07-14 01:26:32 +08:00
void __page_table_check_pmd_clear ( struct mm_struct * mm , pmd_t pmd )
2022-01-14 14:06:37 -08:00
{
if ( & init_mm = = mm )
return ;
if ( pmd_user_accessible_page ( pmd ) ) {
2023-07-14 01:26:29 +08:00
page_table_check_clear ( pmd_pfn ( pmd ) , PMD_SIZE > > PAGE_SHIFT ) ;
2022-01-14 14:06:37 -08:00
}
}
EXPORT_SYMBOL ( __page_table_check_pmd_clear ) ;
2023-07-14 01:26:33 +08:00
void __page_table_check_pud_clear ( struct mm_struct * mm , pud_t pud )
2022-01-14 14:06:37 -08:00
{
if ( & init_mm = = mm )
return ;
if ( pud_user_accessible_page ( pud ) ) {
2023-07-14 01:26:29 +08:00
page_table_check_clear ( pud_pfn ( pud ) , PUD_SIZE > > PAGE_SHIFT ) ;
2022-01-14 14:06:37 -08:00
}
}
EXPORT_SYMBOL ( __page_table_check_pud_clear ) ;
2023-08-02 16:13:30 +01:00
void __page_table_check_ptes_set ( struct mm_struct * mm , pte_t * ptep , pte_t pte ,
unsigned int nr )
2022-01-14 14:06:37 -08:00
{
2023-08-02 16:13:30 +01:00
unsigned int i ;
2022-01-14 14:06:37 -08:00
if ( & init_mm = = mm )
return ;
2023-08-02 16:13:30 +01:00
for ( i = 0 ; i < nr ; i + + )
__page_table_check_pte_clear ( mm , ptep_get ( ptep + i ) ) ;
if ( pte_user_accessible_page ( pte ) )
page_table_check_set ( pte_pfn ( pte ) , nr , pte_write ( pte ) ) ;
2022-01-14 14:06:37 -08:00
}
2023-08-02 16:13:30 +01:00
EXPORT_SYMBOL ( __page_table_check_ptes_set ) ;
2022-01-14 14:06:37 -08:00
2023-07-14 01:26:35 +08:00
void __page_table_check_pmd_set ( struct mm_struct * mm , pmd_t * pmdp , pmd_t pmd )
2022-01-14 14:06:37 -08:00
{
if ( & init_mm = = mm )
return ;
2023-07-14 01:26:32 +08:00
__page_table_check_pmd_clear ( mm , * pmdp ) ;
2022-01-14 14:06:37 -08:00
if ( pmd_user_accessible_page ( pmd ) ) {
2023-07-14 01:26:30 +08:00
page_table_check_set ( pmd_pfn ( pmd ) , PMD_SIZE > > PAGE_SHIFT ,
2022-01-14 14:06:37 -08:00
pmd_write ( pmd ) ) ;
}
}
EXPORT_SYMBOL ( __page_table_check_pmd_set ) ;
2023-07-14 01:26:36 +08:00
void __page_table_check_pud_set ( struct mm_struct * mm , pud_t * pudp , pud_t pud )
2022-01-14 14:06:37 -08:00
{
if ( & init_mm = = mm )
return ;
2023-07-14 01:26:33 +08:00
__page_table_check_pud_clear ( mm , * pudp ) ;
2022-01-14 14:06:37 -08:00
if ( pud_user_accessible_page ( pud ) ) {
2023-07-14 01:26:30 +08:00
page_table_check_set ( pud_pfn ( pud ) , PUD_SIZE > > PAGE_SHIFT ,
2022-01-14 14:06:37 -08:00
pud_write ( pud ) ) ;
}
}
EXPORT_SYMBOL ( __page_table_check_pud_set ) ;
2022-02-03 20:49:24 -08:00
void __page_table_check_pte_clear_range ( struct mm_struct * mm ,
unsigned long addr ,
pmd_t pmd )
{
if ( & init_mm = = mm )
return ;
if ( ! pmd_bad ( pmd ) & & ! pmd_leaf ( pmd ) ) {
pte_t * ptep = pte_offset_map ( & pmd , addr ) ;
unsigned long i ;
2023-06-08 18:27:52 -07:00
if ( WARN_ON ( ! ptep ) )
return ;
2022-02-03 20:49:24 -08:00
for ( i = 0 ; i < PTRS_PER_PTE ; i + + ) {
2023-07-14 01:26:31 +08:00
__page_table_check_pte_clear ( mm , ptep_get ( ptep ) ) ;
2022-02-03 20:49:24 -08:00
addr + = PAGE_SIZE ;
ptep + + ;
}
2022-05-26 19:33:50 +08:00
pte_unmap ( ptep - PTRS_PER_PTE ) ;
2022-02-03 20:49:24 -08:00
}
}