2022-09-16 11:41:49 +02:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Apple DART page table allocator .
*
* Copyright ( C ) 2022 The Asahi Linux Contributors
*
* Based on io - pgtable - arm .
*
* Copyright ( C ) 2014 ARM Limited
*
* Author : Will Deacon < will . deacon @ arm . com >
*/
# define pr_fmt(fmt) "dart io-pgtable: " fmt
# include <linux/atomic.h>
2022-09-16 11:41:50 +02:00
# include <linux/bitfield.h>
2022-09-16 11:41:49 +02:00
# include <linux/bitops.h>
# include <linux/io-pgtable.h>
# include <linux/kernel.h>
# include <linux/sizes.h>
# include <linux/slab.h>
# include <linux/types.h>
# include <asm/barrier.h>
# define DART1_MAX_ADDR_BITS 36
# define DART_MAX_TABLES 4
# define DART_LEVELS 2
/* Struct accessors */
# define io_pgtable_to_data(x) \
container_of ( ( x ) , struct dart_io_pgtable , iop )
# define io_pgtable_ops_to_data(x) \
io_pgtable_to_data ( io_pgtable_ops_to_pgtable ( x ) )
# define DART_GRANULE(d) \
( sizeof ( dart_iopte ) < < ( d ) - > bits_per_level )
# define DART_PTES_PER_TABLE(d) \
( DART_GRANULE ( d ) > > ilog2 ( sizeof ( dart_iopte ) ) )
2022-09-16 11:41:50 +02:00
# define APPLE_DART_PTE_SUBPAGE_START GENMASK_ULL(63, 52)
# define APPLE_DART_PTE_SUBPAGE_END GENMASK_ULL(51, 40)
2022-09-16 11:41:49 +02:00
# define APPLE_DART1_PADDR_MASK GENMASK_ULL(35, 12)
2022-09-16 11:41:51 +02:00
# define APPLE_DART2_PADDR_MASK GENMASK_ULL(37, 10)
# define APPLE_DART2_PADDR_SHIFT (4)
2022-09-16 11:41:49 +02:00
/* Apple DART1 protection bits */
# define APPLE_DART1_PTE_PROT_NO_READ BIT(8)
# define APPLE_DART1_PTE_PROT_NO_WRITE BIT(7)
# define APPLE_DART1_PTE_PROT_SP_DIS BIT(1)
2022-09-16 11:41:51 +02:00
/* Apple DART2 protection bits */
# define APPLE_DART2_PTE_PROT_NO_READ BIT(3)
# define APPLE_DART2_PTE_PROT_NO_WRITE BIT(2)
# define APPLE_DART2_PTE_PROT_NO_CACHE BIT(1)
2022-09-16 11:41:49 +02:00
/* marks PTE as valid */
# define APPLE_DART_PTE_VALID BIT(0)
/* IOPTE accessors */
# define iopte_deref(pte, d) __va(iopte_to_paddr(pte, d))
struct dart_io_pgtable {
struct io_pgtable iop ;
int tbl_bits ;
int bits_per_level ;
void * pgd [ DART_MAX_TABLES ] ;
} ;
typedef u64 dart_iopte ;
static dart_iopte paddr_to_iopte ( phys_addr_t paddr ,
struct dart_io_pgtable * data )
{
2022-09-16 11:41:51 +02:00
dart_iopte pte ;
if ( data - > iop . fmt = = APPLE_DART )
return paddr & APPLE_DART1_PADDR_MASK ;
/* format is APPLE_DART2 */
pte = paddr > > APPLE_DART2_PADDR_SHIFT ;
pte & = APPLE_DART2_PADDR_MASK ;
return pte ;
2022-09-16 11:41:49 +02:00
}
static phys_addr_t iopte_to_paddr ( dart_iopte pte ,
struct dart_io_pgtable * data )
{
2022-09-16 11:41:51 +02:00
u64 paddr ;
if ( data - > iop . fmt = = APPLE_DART )
return pte & APPLE_DART1_PADDR_MASK ;
/* format is APPLE_DART2 */
paddr = pte & APPLE_DART2_PADDR_MASK ;
paddr < < = APPLE_DART2_PADDR_SHIFT ;
return paddr ;
2022-09-16 11:41:49 +02:00
}
static void * __dart_alloc_pages ( size_t size , gfp_t gfp ,
struct io_pgtable_cfg * cfg )
{
int order = get_order ( size ) ;
struct page * p ;
VM_BUG_ON ( ( gfp & __GFP_HIGHMEM ) ) ;
p = alloc_pages ( gfp | __GFP_ZERO , order ) ;
if ( ! p )
return NULL ;
return page_address ( p ) ;
}
static int dart_init_pte ( struct dart_io_pgtable * data ,
unsigned long iova , phys_addr_t paddr ,
dart_iopte prot , int num_entries ,
dart_iopte * ptep )
{
int i ;
dart_iopte pte = prot ;
size_t sz = data - > iop . cfg . pgsize_bitmap ;
for ( i = 0 ; i < num_entries ; i + + )
if ( ptep [ i ] & APPLE_DART_PTE_VALID ) {
/* We require an unmap first */
WARN_ON ( ptep [ i ] & APPLE_DART_PTE_VALID ) ;
return - EEXIST ;
}
2022-09-16 11:41:50 +02:00
/* subpage protection: always allow access to the entire page */
pte | = FIELD_PREP ( APPLE_DART_PTE_SUBPAGE_START , 0 ) ;
pte | = FIELD_PREP ( APPLE_DART_PTE_SUBPAGE_END , 0xfff ) ;
2022-09-16 11:41:49 +02:00
pte | = APPLE_DART1_PTE_PROT_SP_DIS ;
pte | = APPLE_DART_PTE_VALID ;
for ( i = 0 ; i < num_entries ; i + + )
ptep [ i ] = pte | paddr_to_iopte ( paddr + i * sz , data ) ;
return 0 ;
}
static dart_iopte dart_install_table ( dart_iopte * table ,
dart_iopte * ptep ,
dart_iopte curr ,
struct dart_io_pgtable * data )
{
dart_iopte old , new ;
new = paddr_to_iopte ( __pa ( table ) , data ) | APPLE_DART_PTE_VALID ;
/*
* Ensure the table itself is visible before its PTE can be .
* Whilst we could get away with cmpxchg64_release below , this
* doesn ' t have any ordering semantics when ! CONFIG_SMP .
*/
dma_wmb ( ) ;
old = cmpxchg64_relaxed ( ptep , curr , new ) ;
return old ;
}
static int dart_get_table ( struct dart_io_pgtable * data , unsigned long iova )
{
return ( iova > > ( 3 * data - > bits_per_level + ilog2 ( sizeof ( dart_iopte ) ) ) ) &
( ( 1 < < data - > tbl_bits ) - 1 ) ;
}
static int dart_get_l1_index ( struct dart_io_pgtable * data , unsigned long iova )
{
return ( iova > > ( 2 * data - > bits_per_level + ilog2 ( sizeof ( dart_iopte ) ) ) ) &
( ( 1 < < data - > bits_per_level ) - 1 ) ;
}
static int dart_get_l2_index ( struct dart_io_pgtable * data , unsigned long iova )
{
return ( iova > > ( data - > bits_per_level + ilog2 ( sizeof ( dart_iopte ) ) ) ) &
( ( 1 < < data - > bits_per_level ) - 1 ) ;
}
static dart_iopte * dart_get_l2 ( struct dart_io_pgtable * data , unsigned long iova )
{
dart_iopte pte , * ptep ;
int tbl = dart_get_table ( data , iova ) ;
ptep = data - > pgd [ tbl ] ;
if ( ! ptep )
return NULL ;
ptep + = dart_get_l1_index ( data , iova ) ;
pte = READ_ONCE ( * ptep ) ;
/* Valid entry? */
if ( ! pte )
return NULL ;
/* Deref to get level 2 table */
return iopte_deref ( pte , data ) ;
}
static dart_iopte dart_prot_to_pte ( struct dart_io_pgtable * data ,
int prot )
{
dart_iopte pte = 0 ;
2022-09-16 11:41:51 +02:00
if ( data - > iop . fmt = = APPLE_DART ) {
if ( ! ( prot & IOMMU_WRITE ) )
pte | = APPLE_DART1_PTE_PROT_NO_WRITE ;
if ( ! ( prot & IOMMU_READ ) )
pte | = APPLE_DART1_PTE_PROT_NO_READ ;
}
if ( data - > iop . fmt = = APPLE_DART2 ) {
if ( ! ( prot & IOMMU_WRITE ) )
pte | = APPLE_DART2_PTE_PROT_NO_WRITE ;
if ( ! ( prot & IOMMU_READ ) )
pte | = APPLE_DART2_PTE_PROT_NO_READ ;
if ( ! ( prot & IOMMU_CACHE ) )
pte | = APPLE_DART2_PTE_PROT_NO_CACHE ;
}
2022-09-16 11:41:49 +02:00
return pte ;
}
static int dart_map_pages ( struct io_pgtable_ops * ops , unsigned long iova ,
phys_addr_t paddr , size_t pgsize , size_t pgcount ,
int iommu_prot , gfp_t gfp , size_t * mapped )
{
struct dart_io_pgtable * data = io_pgtable_ops_to_data ( ops ) ;
struct io_pgtable_cfg * cfg = & data - > iop . cfg ;
size_t tblsz = DART_GRANULE ( data ) ;
int ret = 0 , tbl , num_entries , max_entries , map_idx_start ;
dart_iopte pte , * cptep , * ptep ;
dart_iopte prot ;
if ( WARN_ON ( pgsize ! = cfg - > pgsize_bitmap ) )
return - EINVAL ;
if ( WARN_ON ( paddr > > cfg - > oas ) )
return - ERANGE ;
/* If no access, then nothing to do */
if ( ! ( iommu_prot & ( IOMMU_READ | IOMMU_WRITE ) ) )
return 0 ;
tbl = dart_get_table ( data , iova ) ;
ptep = data - > pgd [ tbl ] ;
ptep + = dart_get_l1_index ( data , iova ) ;
pte = READ_ONCE ( * ptep ) ;
/* no L2 table present */
if ( ! pte ) {
cptep = __dart_alloc_pages ( tblsz , gfp , cfg ) ;
if ( ! cptep )
return - ENOMEM ;
pte = dart_install_table ( cptep , ptep , 0 , data ) ;
if ( pte )
free_pages ( ( unsigned long ) cptep , get_order ( tblsz ) ) ;
/* L2 table is present (now) */
pte = READ_ONCE ( * ptep ) ;
}
ptep = iopte_deref ( pte , data ) ;
/* install a leaf entries into L2 table */
prot = dart_prot_to_pte ( data , iommu_prot ) ;
map_idx_start = dart_get_l2_index ( data , iova ) ;
max_entries = DART_PTES_PER_TABLE ( data ) - map_idx_start ;
num_entries = min_t ( int , pgcount , max_entries ) ;
ptep + = map_idx_start ;
ret = dart_init_pte ( data , iova , paddr , prot , num_entries , ptep ) ;
if ( ! ret & & mapped )
* mapped + = num_entries * pgsize ;
/*
* Synchronise all PTE updates for the new mapping before there ' s
* a chance for anything to kick off a table walk for the new iova .
*/
wmb ( ) ;
return ret ;
}
static size_t dart_unmap_pages ( struct io_pgtable_ops * ops , unsigned long iova ,
size_t pgsize , size_t pgcount ,
struct iommu_iotlb_gather * gather )
{
struct dart_io_pgtable * data = io_pgtable_ops_to_data ( ops ) ;
struct io_pgtable_cfg * cfg = & data - > iop . cfg ;
int i = 0 , num_entries , max_entries , unmap_idx_start ;
dart_iopte pte , * ptep ;
if ( WARN_ON ( pgsize ! = cfg - > pgsize_bitmap | | ! pgcount ) )
return 0 ;
ptep = dart_get_l2 ( data , iova ) ;
/* Valid L2 IOPTE pointer? */
if ( WARN_ON ( ! ptep ) )
return 0 ;
unmap_idx_start = dart_get_l2_index ( data , iova ) ;
ptep + = unmap_idx_start ;
max_entries = DART_PTES_PER_TABLE ( data ) - unmap_idx_start ;
num_entries = min_t ( int , pgcount , max_entries ) ;
while ( i < num_entries ) {
pte = READ_ONCE ( * ptep ) ;
if ( WARN_ON ( ! pte ) )
break ;
/* clear pte */
* ptep = 0 ;
if ( ! iommu_iotlb_gather_queued ( gather ) )
io_pgtable_tlb_add_page ( & data - > iop , gather ,
iova + i * pgsize , pgsize ) ;
ptep + + ;
i + + ;
}
return i * pgsize ;
}
static phys_addr_t dart_iova_to_phys ( struct io_pgtable_ops * ops ,
unsigned long iova )
{
struct dart_io_pgtable * data = io_pgtable_ops_to_data ( ops ) ;
dart_iopte pte , * ptep ;
ptep = dart_get_l2 ( data , iova ) ;
/* Valid L2 IOPTE pointer? */
if ( ! ptep )
return 0 ;
ptep + = dart_get_l2_index ( data , iova ) ;
pte = READ_ONCE ( * ptep ) ;
/* Found translation */
if ( pte ) {
iova & = ( data - > iop . cfg . pgsize_bitmap - 1 ) ;
return iopte_to_paddr ( pte , data ) | iova ;
}
/* Ran out of page tables to walk */
return 0 ;
}
static struct dart_io_pgtable *
dart_alloc_pgtable ( struct io_pgtable_cfg * cfg )
{
struct dart_io_pgtable * data ;
int tbl_bits , bits_per_level , va_bits , pg_shift ;
pg_shift = __ffs ( cfg - > pgsize_bitmap ) ;
bits_per_level = pg_shift - ilog2 ( sizeof ( dart_iopte ) ) ;
va_bits = cfg - > ias - pg_shift ;
tbl_bits = max_t ( int , 0 , va_bits - ( bits_per_level * DART_LEVELS ) ) ;
if ( ( 1 < < tbl_bits ) > DART_MAX_TABLES )
return NULL ;
data = kzalloc ( sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data )
return NULL ;
data - > tbl_bits = tbl_bits ;
data - > bits_per_level = bits_per_level ;
data - > iop . ops = ( struct io_pgtable_ops ) {
. map_pages = dart_map_pages ,
. unmap_pages = dart_unmap_pages ,
. iova_to_phys = dart_iova_to_phys ,
} ;
return data ;
}
static struct io_pgtable *
apple_dart_alloc_pgtable ( struct io_pgtable_cfg * cfg , void * cookie )
{
struct dart_io_pgtable * data ;
int i ;
if ( ! cfg - > coherent_walk )
return NULL ;
2022-09-16 11:41:51 +02:00
if ( cfg - > oas ! = 36 & & cfg - > oas ! = 42 )
2022-09-16 11:41:49 +02:00
return NULL ;
if ( cfg - > ias > cfg - > oas )
return NULL ;
if ( ! ( cfg - > pgsize_bitmap = = SZ_4K | | cfg - > pgsize_bitmap = = SZ_16K ) )
return NULL ;
data = dart_alloc_pgtable ( cfg ) ;
if ( ! data )
return NULL ;
cfg - > apple_dart_cfg . n_ttbrs = 1 < < data - > tbl_bits ;
for ( i = 0 ; i < cfg - > apple_dart_cfg . n_ttbrs ; + + i ) {
data - > pgd [ i ] = __dart_alloc_pages ( DART_GRANULE ( data ) , GFP_KERNEL ,
cfg ) ;
if ( ! data - > pgd [ i ] )
goto out_free_data ;
cfg - > apple_dart_cfg . ttbr [ i ] = virt_to_phys ( data - > pgd [ i ] ) ;
}
return & data - > iop ;
out_free_data :
while ( - - i > = 0 )
free_pages ( ( unsigned long ) data - > pgd [ i ] ,
get_order ( DART_GRANULE ( data ) ) ) ;
kfree ( data ) ;
return NULL ;
}
static void apple_dart_free_pgtable ( struct io_pgtable * iop )
{
struct dart_io_pgtable * data = io_pgtable_to_data ( iop ) ;
dart_iopte * ptep , * end ;
int i ;
for ( i = 0 ; i < ( 1 < < data - > tbl_bits ) & & data - > pgd [ i ] ; + + i ) {
ptep = data - > pgd [ i ] ;
end = ( void * ) ptep + DART_GRANULE ( data ) ;
while ( ptep ! = end ) {
dart_iopte pte = * ptep + + ;
if ( pte ) {
unsigned long page =
( unsigned long ) iopte_deref ( pte , data ) ;
free_pages ( page , get_order ( DART_GRANULE ( data ) ) ) ;
}
}
free_pages ( ( unsigned long ) data - > pgd [ i ] ,
get_order ( DART_GRANULE ( data ) ) ) ;
}
kfree ( data ) ;
}
struct io_pgtable_init_fns io_pgtable_apple_dart_init_fns = {
. alloc = apple_dart_alloc_pgtable ,
. free = apple_dart_free_pgtable ,
} ;