2011-11-17 07:31:31 +02:00
/*
2014-04-16 09:24:44 +02:00
* Copyright ( C ) 2011 - 2014 NVIDIA CORPORATION . All rights reserved .
2011-11-17 07:31:31 +02:00
*
2014-04-16 09:24:44 +02:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
2011-11-17 07:31:31 +02:00
*/
2015-03-27 11:07:27 +01:00
# include <linux/bitops.h>
2015-01-23 09:49:25 +01:00
# include <linux/debugfs.h>
2013-01-21 11:09:06 +01:00
# include <linux/err.h>
2011-11-17 07:31:31 +02:00
# include <linux/iommu.h>
2014-04-16 09:24:44 +02:00
# include <linux/kernel.h>
2012-06-25 14:23:55 +03:00
# include <linux/of.h>
2014-04-16 09:24:44 +02:00
# include <linux/of_device.h>
# include <linux/platform_device.h>
# include <linux/slab.h>
2014-07-17 13:17:24 +02:00
# include <soc/tegra/ahb.h>
2014-04-16 09:24:44 +02:00
# include <soc/tegra/mc.h>
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
struct tegra_smmu {
void __iomem * regs ;
struct device * dev ;
2012-09-04 16:36:15 -06:00
2014-04-16 09:24:44 +02:00
struct tegra_mc * mc ;
const struct tegra_smmu_soc * soc ;
2012-08-02 11:46:40 +03:00
2015-03-27 11:07:27 +01:00
unsigned long pfn_mask ;
2014-04-16 09:24:44 +02:00
unsigned long * asids ;
struct mutex lock ;
2012-08-02 11:46:40 +03:00
2014-04-16 09:24:44 +02:00
struct list_head list ;
2015-01-23 09:49:25 +01:00
struct dentry * debugfs ;
2011-11-17 07:31:31 +02:00
} ;
2014-04-16 09:24:44 +02:00
struct tegra_smmu_as {
2015-03-26 13:43:12 +01:00
struct iommu_domain domain ;
2014-04-16 09:24:44 +02:00
struct tegra_smmu * smmu ;
unsigned int use_count ;
2015-07-27 13:29:31 +01:00
u32 * count ;
2015-07-27 13:29:26 +01:00
struct page * * pts ;
2014-04-16 09:24:44 +02:00
struct page * pd ;
unsigned id ;
u32 attr ;
2011-11-17 07:31:31 +02:00
} ;
2015-03-26 13:43:12 +01:00
static struct tegra_smmu_as * to_smmu_as ( struct iommu_domain * dom )
{
return container_of ( dom , struct tegra_smmu_as , domain ) ;
}
2014-04-16 09:24:44 +02:00
static inline void smmu_writel ( struct tegra_smmu * smmu , u32 value ,
unsigned long offset )
{
writel ( value , smmu - > regs + offset ) ;
}
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
static inline u32 smmu_readl ( struct tegra_smmu * smmu , unsigned long offset )
{
return readl ( smmu - > regs + offset ) ;
}
2012-09-14 10:22:00 -06:00
2014-04-16 09:24:44 +02:00
# define SMMU_CONFIG 0x010
# define SMMU_CONFIG_ENABLE (1 << 0)
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_TLB_CONFIG 0x14
# define SMMU_TLB_CONFIG_HIT_UNDER_MISS (1 << 29)
# define SMMU_TLB_CONFIG_ROUND_ROBIN_ARBITRATION (1 << 28)
# define SMMU_TLB_CONFIG_ACTIVE_LINES(x) ((x) & 0x3f)
2012-06-25 14:23:55 +03:00
2014-04-16 09:24:44 +02:00
# define SMMU_PTC_CONFIG 0x18
# define SMMU_PTC_CONFIG_ENABLE (1 << 29)
# define SMMU_PTC_CONFIG_REQ_LIMIT(x) (((x) & 0x0f) << 24)
# define SMMU_PTC_CONFIG_INDEX_MAP(x) ((x) & 0x3f)
2012-08-02 11:46:40 +03:00
2014-04-16 09:24:44 +02:00
# define SMMU_PTB_ASID 0x01c
# define SMMU_PTB_ASID_VALUE(x) ((x) & 0x7f)
2012-06-25 14:23:56 +03:00
2014-04-16 09:24:44 +02:00
# define SMMU_PTB_DATA 0x020
# define SMMU_PTB_DATA_VALUE(page, attr) (page_to_phys(page) >> 12 | (attr))
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_MK_PDE(page, attr) (page_to_phys(page) >> SMMU_PTE_SHIFT | (attr))
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_TLB_FLUSH 0x030
# define SMMU_TLB_FLUSH_VA_MATCH_ALL (0 << 0)
# define SMMU_TLB_FLUSH_VA_MATCH_SECTION (2 << 0)
# define SMMU_TLB_FLUSH_VA_MATCH_GROUP (3 << 0)
# define SMMU_TLB_FLUSH_ASID(x) (((x) & 0x7f) << 24)
# define SMMU_TLB_FLUSH_VA_SECTION(addr) ((((addr) & 0xffc00000) >> 12) | \
SMMU_TLB_FLUSH_VA_MATCH_SECTION )
# define SMMU_TLB_FLUSH_VA_GROUP(addr) ((((addr) & 0xffffc000) >> 12) | \
SMMU_TLB_FLUSH_VA_MATCH_GROUP )
# define SMMU_TLB_FLUSH_ASID_MATCH (1 << 31)
2013-01-31 10:14:10 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_PTC_FLUSH 0x034
# define SMMU_PTC_FLUSH_TYPE_ALL (0 << 0)
# define SMMU_PTC_FLUSH_TYPE_ADR (1 << 0)
2013-01-31 10:14:10 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_PTC_FLUSH_HI 0x9b8
# define SMMU_PTC_FLUSH_HI_MASK 0x3
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
/* per-SWGROUP SMMU_*_ASID register */
# define SMMU_ASID_ENABLE (1 << 31)
# define SMMU_ASID_MASK 0x7f
# define SMMU_ASID_VALUE(x) ((x) & SMMU_ASID_MASK)
2013-01-31 10:14:10 +02:00
2014-04-16 09:24:44 +02:00
/* page table definitions */
# define SMMU_NUM_PDE 1024
# define SMMU_NUM_PTE 1024
2013-01-31 10:14:10 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_SIZE_PD (SMMU_NUM_PDE * 4)
# define SMMU_SIZE_PT (SMMU_NUM_PTE * 4)
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_PDE_SHIFT 22
# define SMMU_PTE_SHIFT 12
2013-02-04 20:40:58 +01:00
2014-04-16 09:24:44 +02:00
# define SMMU_PD_READABLE (1 << 31)
# define SMMU_PD_WRITABLE (1 << 30)
# define SMMU_PD_NONSECURE (1 << 29)
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_PDE_READABLE (1 << 31)
# define SMMU_PDE_WRITABLE (1 << 30)
# define SMMU_PDE_NONSECURE (1 << 29)
# define SMMU_PDE_NEXT (1 << 28)
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_PTE_READABLE (1 << 31)
# define SMMU_PTE_WRITABLE (1 << 30)
# define SMMU_PTE_NONSECURE (1 << 29)
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
# define SMMU_PDE_ATTR (SMMU_PDE_READABLE | SMMU_PDE_WRITABLE | \
SMMU_PDE_NONSECURE )
# define SMMU_PTE_ATTR (SMMU_PTE_READABLE | SMMU_PTE_WRITABLE | \
SMMU_PTE_NONSECURE )
2011-11-17 07:31:31 +02:00
2015-07-27 13:29:16 +01:00
static unsigned int iova_pd_index ( unsigned long iova )
{
return ( iova > > SMMU_PDE_SHIFT ) & ( SMMU_NUM_PDE - 1 ) ;
}
static unsigned int iova_pt_index ( unsigned long iova )
{
return ( iova > > SMMU_PTE_SHIFT ) & ( SMMU_NUM_PTE - 1 ) ;
}
2014-04-16 09:24:44 +02:00
static inline void smmu_flush_ptc ( struct tegra_smmu * smmu , struct page * page ,
unsigned long offset )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
phys_addr_t phys = page ? page_to_phys ( page ) : 0 ;
u32 value ;
if ( page ) {
offset & = ~ ( smmu - > mc - > soc - > atom_size - 1 ) ;
if ( smmu - > mc - > soc - > num_address_bits > 32 ) {
# ifdef CONFIG_PHYS_ADDR_T_64BIT
value = ( phys > > 32 ) & SMMU_PTC_FLUSH_HI_MASK ;
# else
value = 0 ;
# endif
smmu_writel ( smmu , value , SMMU_PTC_FLUSH_HI ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
value = ( phys + offset ) | SMMU_PTC_FLUSH_TYPE_ADR ;
} else {
value = SMMU_PTC_FLUSH_TYPE_ALL ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
smmu_writel ( smmu , value , SMMU_PTC_FLUSH ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static inline void smmu_flush_tlb ( struct tegra_smmu * smmu )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
smmu_writel ( smmu , SMMU_TLB_FLUSH_VA_MATCH_ALL , SMMU_TLB_FLUSH ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static inline void smmu_flush_tlb_asid ( struct tegra_smmu * smmu ,
unsigned long asid )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
u32 value ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
value = SMMU_TLB_FLUSH_ASID_MATCH | SMMU_TLB_FLUSH_ASID ( asid ) |
SMMU_TLB_FLUSH_VA_MATCH_ALL ;
smmu_writel ( smmu , value , SMMU_TLB_FLUSH ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static inline void smmu_flush_tlb_section ( struct tegra_smmu * smmu ,
unsigned long asid ,
unsigned long iova )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
u32 value ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
value = SMMU_TLB_FLUSH_ASID_MATCH | SMMU_TLB_FLUSH_ASID ( asid ) |
SMMU_TLB_FLUSH_VA_SECTION ( iova ) ;
smmu_writel ( smmu , value , SMMU_TLB_FLUSH ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static inline void smmu_flush_tlb_group ( struct tegra_smmu * smmu ,
unsigned long asid ,
unsigned long iova )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
u32 value ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
value = SMMU_TLB_FLUSH_ASID_MATCH | SMMU_TLB_FLUSH_ASID ( asid ) |
SMMU_TLB_FLUSH_VA_GROUP ( iova ) ;
smmu_writel ( smmu , value , SMMU_TLB_FLUSH ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static inline void smmu_flush ( struct tegra_smmu * smmu )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
smmu_readl ( smmu , SMMU_CONFIG ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static int tegra_smmu_alloc_asid ( struct tegra_smmu * smmu , unsigned int * idp )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
unsigned long id ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
mutex_lock ( & smmu - > lock ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
id = find_first_zero_bit ( smmu - > asids , smmu - > soc - > num_asids ) ;
if ( id > = smmu - > soc - > num_asids ) {
mutex_unlock ( & smmu - > lock ) ;
return - ENOSPC ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
set_bit ( id , smmu - > asids ) ;
* idp = id ;
mutex_unlock ( & smmu - > lock ) ;
return 0 ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static void tegra_smmu_free_asid ( struct tegra_smmu * smmu , unsigned int id )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
mutex_lock ( & smmu - > lock ) ;
clear_bit ( id , smmu - > asids ) ;
mutex_unlock ( & smmu - > lock ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static bool tegra_smmu_capable ( enum iommu_cap cap )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
return false ;
2011-11-17 07:31:31 +02:00
}
2015-03-26 13:43:12 +01:00
static struct iommu_domain * tegra_smmu_domain_alloc ( unsigned type )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
struct tegra_smmu_as * as ;
unsigned int i ;
uint32_t * pd ;
2011-11-17 07:31:31 +02:00
2015-03-26 13:43:12 +01:00
if ( type ! = IOMMU_DOMAIN_UNMANAGED )
return NULL ;
2014-04-16 09:24:44 +02:00
as = kzalloc ( sizeof ( * as ) , GFP_KERNEL ) ;
if ( ! as )
2015-03-26 13:43:12 +01:00
return NULL ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
as - > attr = SMMU_PD_READABLE | SMMU_PD_WRITABLE | SMMU_PD_NONSECURE ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
as - > pd = alloc_page ( GFP_KERNEL | __GFP_DMA ) ;
if ( ! as - > pd ) {
kfree ( as ) ;
2015-03-26 13:43:12 +01:00
return NULL ;
2011-11-17 07:31:31 +02:00
}
2012-07-02 14:26:38 +03:00
2015-07-27 13:29:31 +01:00
as - > count = kcalloc ( SMMU_NUM_PDE , sizeof ( u32 ) , GFP_KERNEL ) ;
2014-04-16 09:24:44 +02:00
if ( ! as - > count ) {
__free_page ( as - > pd ) ;
kfree ( as ) ;
2015-03-26 13:43:12 +01:00
return NULL ;
2011-11-17 07:31:31 +02:00
}
2012-07-02 14:26:38 +03:00
2015-07-27 13:29:26 +01:00
as - > pts = kcalloc ( SMMU_NUM_PDE , sizeof ( * as - > pts ) , GFP_KERNEL ) ;
if ( ! as - > pts ) {
2015-07-27 13:29:31 +01:00
kfree ( as - > count ) ;
2015-07-27 13:29:26 +01:00
__free_page ( as - > pd ) ;
kfree ( as ) ;
return NULL ;
}
2014-04-16 09:24:44 +02:00
/* clear PDEs */
pd = page_address ( as - > pd ) ;
SetPageReserved ( as - > pd ) ;
2012-07-02 14:26:38 +03:00
2014-04-16 09:24:44 +02:00
for ( i = 0 ; i < SMMU_NUM_PDE ; i + + )
pd [ i ] = 0 ;
2012-07-02 14:26:38 +03:00
2015-03-27 11:07:25 +01:00
/* setup aperture */
2015-04-02 13:33:19 +02:00
as - > domain . geometry . aperture_start = 0 ;
as - > domain . geometry . aperture_end = 0xffffffff ;
as - > domain . geometry . force_aperture = true ;
2012-07-17 11:47:14 +02:00
2015-03-26 13:43:12 +01:00
return & as - > domain ;
2011-11-17 07:31:31 +02:00
}
2015-03-26 13:43:12 +01:00
static void tegra_smmu_domain_free ( struct iommu_domain * domain )
2011-11-17 07:31:31 +02:00
{
2015-03-26 13:43:12 +01:00
struct tegra_smmu_as * as = to_smmu_as ( domain ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
/* TODO: free page directory and page tables */
ClearPageReserved ( as - > pd ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
kfree ( as ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static const struct tegra_smmu_swgroup *
tegra_smmu_find_swgroup ( struct tegra_smmu * smmu , unsigned int swgroup )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
const struct tegra_smmu_swgroup * group = NULL ;
unsigned int i ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
for ( i = 0 ; i < smmu - > soc - > num_swgroups ; i + + ) {
if ( smmu - > soc - > swgroups [ i ] . swgroup = = swgroup ) {
group = & smmu - > soc - > swgroups [ i ] ;
break ;
}
}
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
return group ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static void tegra_smmu_enable ( struct tegra_smmu * smmu , unsigned int swgroup ,
unsigned int asid )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
const struct tegra_smmu_swgroup * group ;
unsigned int i ;
u32 value ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
for ( i = 0 ; i < smmu - > soc - > num_clients ; i + + ) {
const struct tegra_mc_client * client = & smmu - > soc - > clients [ i ] ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
if ( client - > swgroup ! = swgroup )
continue ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
value = smmu_readl ( smmu , client - > smmu . reg ) ;
value | = BIT ( client - > smmu . bit ) ;
smmu_writel ( smmu , value , client - > smmu . reg ) ;
}
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
group = tegra_smmu_find_swgroup ( smmu , swgroup ) ;
if ( group ) {
value = smmu_readl ( smmu , group - > reg ) ;
value & = ~ SMMU_ASID_MASK ;
value | = SMMU_ASID_VALUE ( asid ) ;
value | = SMMU_ASID_ENABLE ;
smmu_writel ( smmu , value , group - > reg ) ;
}
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static void tegra_smmu_disable ( struct tegra_smmu * smmu , unsigned int swgroup ,
unsigned int asid )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
const struct tegra_smmu_swgroup * group ;
unsigned int i ;
u32 value ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
group = tegra_smmu_find_swgroup ( smmu , swgroup ) ;
if ( group ) {
value = smmu_readl ( smmu , group - > reg ) ;
value & = ~ SMMU_ASID_MASK ;
value | = SMMU_ASID_VALUE ( asid ) ;
value & = ~ SMMU_ASID_ENABLE ;
smmu_writel ( smmu , value , group - > reg ) ;
}
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
for ( i = 0 ; i < smmu - > soc - > num_clients ; i + + ) {
const struct tegra_mc_client * client = & smmu - > soc - > clients [ i ] ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
if ( client - > swgroup ! = swgroup )
continue ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
value = smmu_readl ( smmu , client - > smmu . reg ) ;
value & = ~ BIT ( client - > smmu . bit ) ;
smmu_writel ( smmu , value , client - > smmu . reg ) ;
}
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static int tegra_smmu_as_prepare ( struct tegra_smmu * smmu ,
struct tegra_smmu_as * as )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
u32 value ;
2011-11-17 07:31:31 +02:00
int err ;
2014-04-16 09:24:44 +02:00
if ( as - > use_count > 0 ) {
as - > use_count + + ;
return 0 ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
err = tegra_smmu_alloc_asid ( smmu , & as - > id ) ;
if ( err < 0 )
return err ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
smmu - > soc - > ops - > flush_dcache ( as - > pd , 0 , SMMU_SIZE_PD ) ;
smmu_flush_ptc ( smmu , as - > pd , 0 ) ;
smmu_flush_tlb_asid ( smmu , as - > id ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
smmu_writel ( smmu , as - > id & 0x7f , SMMU_PTB_ASID ) ;
value = SMMU_PTB_DATA_VALUE ( as - > pd , as - > attr ) ;
smmu_writel ( smmu , value , SMMU_PTB_DATA ) ;
smmu_flush ( smmu ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
as - > smmu = smmu ;
as - > use_count + + ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
return 0 ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static void tegra_smmu_as_unprepare ( struct tegra_smmu * smmu ,
struct tegra_smmu_as * as )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
if ( - - as - > use_count > 0 )
return ;
tegra_smmu_free_asid ( smmu , as - > id ) ;
as - > smmu = NULL ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static int tegra_smmu_attach_dev ( struct iommu_domain * domain ,
struct device * dev )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
struct tegra_smmu * smmu = dev - > archdata . iommu ;
2015-03-26 13:43:12 +01:00
struct tegra_smmu_as * as = to_smmu_as ( domain ) ;
2014-04-16 09:24:44 +02:00
struct device_node * np = dev - > of_node ;
struct of_phandle_args args ;
unsigned int index = 0 ;
int err = 0 ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
while ( ! of_parse_phandle_with_args ( np , " iommus " , " #iommu-cells " , index ,
& args ) ) {
unsigned int swgroup = args . args [ 0 ] ;
2012-07-30 08:39:18 +03:00
2014-04-16 09:24:44 +02:00
if ( args . np ! = smmu - > dev - > of_node ) {
of_node_put ( args . np ) ;
2012-07-30 08:39:18 +03:00
continue ;
2014-04-16 09:24:44 +02:00
}
2012-07-30 08:39:18 +03:00
2014-04-16 09:24:44 +02:00
of_node_put ( args . np ) ;
2012-07-30 08:39:18 +03:00
2014-04-16 09:24:44 +02:00
err = tegra_smmu_as_prepare ( smmu , as ) ;
if ( err < 0 )
return err ;
tegra_smmu_enable ( smmu , swgroup , as - > id ) ;
index + + ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
if ( index = = 0 )
return - ENODEV ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
return 0 ;
}
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
static void tegra_smmu_detach_dev ( struct iommu_domain * domain , struct device * dev )
{
2015-03-26 13:43:12 +01:00
struct tegra_smmu_as * as = to_smmu_as ( domain ) ;
2014-04-16 09:24:44 +02:00
struct device_node * np = dev - > of_node ;
struct tegra_smmu * smmu = as - > smmu ;
struct of_phandle_args args ;
unsigned int index = 0 ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
while ( ! of_parse_phandle_with_args ( np , " iommus " , " #iommu-cells " , index ,
& args ) ) {
unsigned int swgroup = args . args [ 0 ] ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
if ( args . np ! = smmu - > dev - > of_node ) {
of_node_put ( args . np ) ;
continue ;
}
2012-01-26 19:40:57 +01:00
2014-04-16 09:24:44 +02:00
of_node_put ( args . np ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
tegra_smmu_disable ( smmu , swgroup , as - > id ) ;
tegra_smmu_as_unprepare ( smmu , as ) ;
index + + ;
}
2011-11-17 07:31:31 +02:00
}
2015-07-27 13:29:21 +01:00
static u32 * tegra_smmu_pte_offset ( struct page * pt_page , unsigned long iova )
{
u32 * pt = page_address ( pt_page ) ;
return pt + iova_pt_index ( iova ) ;
}
static u32 * tegra_smmu_pte_lookup ( struct tegra_smmu_as * as , unsigned long iova ,
struct page * * pagep )
{
unsigned int pd_index = iova_pd_index ( iova ) ;
struct page * pt_page ;
2015-07-27 13:29:26 +01:00
pt_page = as - > pts [ pd_index ] ;
if ( ! pt_page )
2015-07-27 13:29:21 +01:00
return NULL ;
* pagep = pt_page ;
return tegra_smmu_pte_offset ( pt_page , iova ) ;
}
2014-04-16 09:24:44 +02:00
static u32 * as_get_pte ( struct tegra_smmu_as * as , dma_addr_t iova ,
struct page * * pagep )
2011-11-17 07:31:31 +02:00
{
2015-07-27 13:29:31 +01:00
u32 * pd = page_address ( as - > pd ) , * pt ;
2015-07-27 13:29:16 +01:00
unsigned int pde = iova_pd_index ( iova ) ;
2014-04-16 09:24:44 +02:00
struct tegra_smmu * smmu = as - > smmu ;
struct page * page ;
unsigned int i ;
2015-07-27 13:29:26 +01:00
if ( ! as - > pts [ pde ] ) {
2014-04-16 09:24:44 +02:00
page = alloc_page ( GFP_KERNEL | __GFP_DMA ) ;
if ( ! page )
return NULL ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
pt = page_address ( page ) ;
SetPageReserved ( page ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
for ( i = 0 ; i < SMMU_NUM_PTE ; i + + )
pt [ i ] = 0 ;
2011-11-17 07:31:31 +02:00
2015-07-27 13:29:26 +01:00
as - > pts [ pde ] = page ;
2014-04-16 09:24:44 +02:00
smmu - > soc - > ops - > flush_dcache ( page , 0 , SMMU_SIZE_PT ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
pd [ pde ] = SMMU_MK_PDE ( page , SMMU_PDE_ATTR | SMMU_PDE_NEXT ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
smmu - > soc - > ops - > flush_dcache ( as - > pd , pde < < 2 , 4 ) ;
smmu_flush_ptc ( smmu , as - > pd , pde < < 2 ) ;
smmu_flush_tlb_section ( smmu , as - > id , iova ) ;
smmu_flush ( smmu ) ;
} else {
2015-07-27 13:29:26 +01:00
page = as - > pts [ pde ] ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
* pagep = page ;
2011-11-17 07:31:31 +02:00
2015-07-27 13:29:21 +01:00
pt = page_address ( page ) ;
2014-04-16 09:24:44 +02:00
/* Keep track of entries in this page table. */
2015-07-27 13:29:21 +01:00
if ( pt [ iova_pt_index ( iova ) ] = = 0 )
2015-07-27 13:29:31 +01:00
as - > count [ pde ] + + ;
2011-11-17 07:31:31 +02:00
2015-07-27 13:29:21 +01:00
return tegra_smmu_pte_offset ( page , iova ) ;
2014-04-16 09:24:44 +02:00
}
2012-08-02 11:46:40 +03:00
2015-07-27 13:29:05 +01:00
static void tegra_smmu_pte_put_use ( struct tegra_smmu_as * as , unsigned long iova )
2012-08-02 11:46:40 +03:00
{
2015-07-27 13:29:05 +01:00
struct tegra_smmu * smmu = as - > smmu ;
2015-07-27 13:29:16 +01:00
unsigned int pde = iova_pd_index ( iova ) ;
2015-07-27 13:29:05 +01:00
u32 * pd = page_address ( as - > pd ) ;
2015-07-27 13:29:26 +01:00
struct page * page = as - > pts [ pde ] ;
2012-08-02 11:46:40 +03:00
2014-04-16 09:24:44 +02:00
/*
* When no entries in this page table are used anymore , return the
* memory page to the system .
*/
2015-07-27 13:29:31 +01:00
if ( - - as - > count [ pde ] = = 0 ) {
2015-07-27 13:29:05 +01:00
unsigned int offset = pde * sizeof ( * pd ) ;
2012-08-02 11:46:40 +03:00
2015-07-27 13:29:05 +01:00
/* Clear the page directory entry first */
pd [ pde ] = 0 ;
/* Flush the page directory entry */
smmu - > soc - > ops - > flush_dcache ( as - > pd , offset , sizeof ( * pd ) ) ;
smmu_flush_ptc ( smmu , as - > pd , offset ) ;
smmu_flush_tlb_section ( smmu , as - > id , iova ) ;
smmu_flush ( smmu ) ;
/* Finally, free the page */
ClearPageReserved ( page ) ;
__free_page ( page ) ;
2015-07-27 13:29:26 +01:00
as - > pts [ pde ] = NULL ;
2012-08-02 11:46:40 +03:00
}
}
2015-07-27 13:29:10 +01:00
static void tegra_smmu_set_pte ( struct tegra_smmu_as * as , unsigned long iova ,
u32 * pte , struct page * pte_page , u32 val )
{
struct tegra_smmu * smmu = as - > smmu ;
unsigned long offset = offset_in_page ( pte ) ;
* pte = val ;
smmu - > soc - > ops - > flush_dcache ( pte_page , offset , 4 ) ;
smmu_flush_ptc ( smmu , pte_page , offset ) ;
smmu_flush_tlb_group ( smmu , as - > id , iova ) ;
smmu_flush ( smmu ) ;
}
2014-04-16 09:24:44 +02:00
static int tegra_smmu_map ( struct iommu_domain * domain , unsigned long iova ,
phys_addr_t paddr , size_t size , int prot )
2012-08-02 11:46:40 +03:00
{
2015-03-26 13:43:12 +01:00
struct tegra_smmu_as * as = to_smmu_as ( domain ) ;
2014-04-16 09:24:44 +02:00
struct page * page ;
u32 * pte ;
2012-08-02 11:46:40 +03:00
2014-04-16 09:24:44 +02:00
pte = as_get_pte ( as , iova , & page ) ;
if ( ! pte )
return - ENOMEM ;
2012-08-02 11:46:40 +03:00
2015-07-27 13:29:10 +01:00
tegra_smmu_set_pte ( as , iova , pte , page ,
__phys_to_pfn ( paddr ) | SMMU_PTE_ATTR ) ;
2012-08-02 11:46:40 +03:00
return 0 ;
}
2014-04-16 09:24:44 +02:00
static size_t tegra_smmu_unmap ( struct iommu_domain * domain , unsigned long iova ,
size_t size )
2012-08-02 11:46:40 +03:00
{
2015-03-26 13:43:12 +01:00
struct tegra_smmu_as * as = to_smmu_as ( domain ) ;
2015-07-27 13:29:21 +01:00
struct page * pte_page ;
2014-04-16 09:24:44 +02:00
u32 * pte ;
2012-08-02 11:46:40 +03:00
2015-07-27 13:29:21 +01:00
pte = tegra_smmu_pte_lookup ( as , iova , & pte_page ) ;
2015-07-27 13:29:05 +01:00
if ( ! pte | | ! * pte )
2014-04-16 09:24:44 +02:00
return 0 ;
2012-08-02 11:46:40 +03:00
2015-07-27 13:29:21 +01:00
tegra_smmu_set_pte ( as , iova , pte , pte_page , 0 ) ;
2015-07-27 13:29:05 +01:00
tegra_smmu_pte_put_use ( as , iova ) ;
2014-04-16 09:24:44 +02:00
return size ;
2012-08-02 11:46:40 +03:00
}
2014-04-16 09:24:44 +02:00
static phys_addr_t tegra_smmu_iova_to_phys ( struct iommu_domain * domain ,
dma_addr_t iova )
2012-08-02 11:46:40 +03:00
{
2015-03-26 13:43:12 +01:00
struct tegra_smmu_as * as = to_smmu_as ( domain ) ;
2015-07-27 13:29:21 +01:00
struct page * pte_page ;
2014-04-16 09:24:44 +02:00
unsigned long pfn ;
u32 * pte ;
2012-08-02 11:46:40 +03:00
2015-07-27 13:29:21 +01:00
pte = tegra_smmu_pte_lookup ( as , iova , & pte_page ) ;
2015-07-27 13:29:00 +01:00
if ( ! pte | | ! * pte )
return 0 ;
2015-03-27 11:07:27 +01:00
pfn = * pte & as - > smmu - > pfn_mask ;
2012-08-02 11:46:40 +03:00
2014-04-16 09:24:44 +02:00
return PFN_PHYS ( pfn ) ;
2012-08-02 11:46:40 +03:00
}
2014-04-16 09:24:44 +02:00
static struct tegra_smmu * tegra_smmu_find ( struct device_node * np )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
struct platform_device * pdev ;
struct tegra_mc * mc ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
pdev = of_find_device_by_node ( np ) ;
if ( ! pdev )
return NULL ;
mc = platform_get_drvdata ( pdev ) ;
if ( ! mc )
return NULL ;
return mc - > smmu ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static int tegra_smmu_add_device ( struct device * dev )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
struct device_node * np = dev - > of_node ;
struct of_phandle_args args ;
unsigned int index = 0 ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
while ( of_parse_phandle_with_args ( np , " iommus " , " #iommu-cells " , index ,
& args ) = = 0 ) {
struct tegra_smmu * smmu ;
smmu = tegra_smmu_find ( args . np ) ;
if ( smmu ) {
/*
* Only a single IOMMU master interface is currently
* supported by the Linux kernel , so abort after the
* first match .
*/
dev - > archdata . iommu = smmu ;
break ;
}
index + + ;
}
return 0 ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
static void tegra_smmu_remove_device ( struct device * dev )
2011-11-17 07:31:31 +02:00
{
2014-04-16 09:24:44 +02:00
dev - > archdata . iommu = NULL ;
}
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
static const struct iommu_ops tegra_smmu_ops = {
. capable = tegra_smmu_capable ,
2015-03-26 13:43:12 +01:00
. domain_alloc = tegra_smmu_domain_alloc ,
. domain_free = tegra_smmu_domain_free ,
2014-04-16 09:24:44 +02:00
. attach_dev = tegra_smmu_attach_dev ,
. detach_dev = tegra_smmu_detach_dev ,
. add_device = tegra_smmu_add_device ,
. remove_device = tegra_smmu_remove_device ,
. map = tegra_smmu_map ,
. unmap = tegra_smmu_unmap ,
. map_sg = default_iommu_map_sg ,
. iova_to_phys = tegra_smmu_iova_to_phys ,
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
. pgsize_bitmap = SZ_4K ,
} ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
static void tegra_smmu_ahb_enable ( void )
{
static const struct of_device_id ahb_match [ ] = {
{ . compatible = " nvidia,tegra30-ahb " , } ,
{ }
} ;
struct device_node * ahb ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
ahb = of_find_matching_node ( NULL , ahb_match ) ;
if ( ahb ) {
tegra_ahb_enable_smmu ( ahb ) ;
of_node_put ( ahb ) ;
2011-11-17 07:31:31 +02:00
}
2014-04-16 09:24:44 +02:00
}
2011-11-17 07:31:31 +02:00
2015-01-23 09:49:25 +01:00
static int tegra_smmu_swgroups_show ( struct seq_file * s , void * data )
{
struct tegra_smmu * smmu = s - > private ;
unsigned int i ;
u32 value ;
seq_printf ( s , " swgroup enabled ASID \n " ) ;
seq_printf ( s , " ------------------------ \n " ) ;
for ( i = 0 ; i < smmu - > soc - > num_swgroups ; i + + ) {
const struct tegra_smmu_swgroup * group = & smmu - > soc - > swgroups [ i ] ;
const char * status ;
unsigned int asid ;
value = smmu_readl ( smmu , group - > reg ) ;
if ( value & SMMU_ASID_ENABLE )
status = " yes " ;
else
status = " no " ;
asid = value & SMMU_ASID_MASK ;
seq_printf ( s , " %-9s %-7s %#04x \n " , group - > name , status ,
asid ) ;
}
return 0 ;
}
static int tegra_smmu_swgroups_open ( struct inode * inode , struct file * file )
{
return single_open ( file , tegra_smmu_swgroups_show , inode - > i_private ) ;
}
static const struct file_operations tegra_smmu_swgroups_fops = {
. open = tegra_smmu_swgroups_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static int tegra_smmu_clients_show ( struct seq_file * s , void * data )
{
struct tegra_smmu * smmu = s - > private ;
unsigned int i ;
u32 value ;
seq_printf ( s , " client enabled \n " ) ;
seq_printf ( s , " -------------------- \n " ) ;
for ( i = 0 ; i < smmu - > soc - > num_clients ; i + + ) {
const struct tegra_mc_client * client = & smmu - > soc - > clients [ i ] ;
const char * status ;
value = smmu_readl ( smmu , client - > smmu . reg ) ;
if ( value & BIT ( client - > smmu . bit ) )
status = " yes " ;
else
status = " no " ;
seq_printf ( s , " %-12s %s \n " , client - > name , status ) ;
}
return 0 ;
}
static int tegra_smmu_clients_open ( struct inode * inode , struct file * file )
{
return single_open ( file , tegra_smmu_clients_show , inode - > i_private ) ;
}
static const struct file_operations tegra_smmu_clients_fops = {
. open = tegra_smmu_clients_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static void tegra_smmu_debugfs_init ( struct tegra_smmu * smmu )
{
smmu - > debugfs = debugfs_create_dir ( " smmu " , NULL ) ;
if ( ! smmu - > debugfs )
return ;
debugfs_create_file ( " swgroups " , S_IRUGO , smmu - > debugfs , smmu ,
& tegra_smmu_swgroups_fops ) ;
debugfs_create_file ( " clients " , S_IRUGO , smmu - > debugfs , smmu ,
& tegra_smmu_clients_fops ) ;
}
static void tegra_smmu_debugfs_exit ( struct tegra_smmu * smmu )
{
debugfs_remove_recursive ( smmu - > debugfs ) ;
}
2014-04-16 09:24:44 +02:00
struct tegra_smmu * tegra_smmu_probe ( struct device * dev ,
const struct tegra_smmu_soc * soc ,
struct tegra_mc * mc )
{
struct tegra_smmu * smmu ;
size_t size ;
u32 value ;
int err ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
/* This can happen on Tegra20 which doesn't have an SMMU */
if ( ! soc )
return NULL ;
2012-06-25 14:23:55 +03:00
2014-04-16 09:24:44 +02:00
smmu = devm_kzalloc ( dev , sizeof ( * smmu ) , GFP_KERNEL ) ;
if ( ! smmu )
return ERR_PTR ( - ENOMEM ) ;
2012-06-25 14:23:55 +03:00
2014-04-16 09:24:44 +02:00
/*
* This is a bit of a hack . Ideally we ' d want to simply return this
* value . However the IOMMU registration process will attempt to add
* all devices to the IOMMU when bus_set_iommu ( ) is called . In order
* not to rely on global variables to track the IOMMU instance , we
* set it here so that it can be looked up from the . add_device ( )
* callback via the IOMMU device ' s . drvdata field .
*/
mc - > smmu = smmu ;
2012-06-25 14:23:55 +03:00
2014-04-16 09:24:44 +02:00
size = BITS_TO_LONGS ( soc - > num_asids ) * sizeof ( long ) ;
2012-06-25 14:23:55 +03:00
2014-04-16 09:24:44 +02:00
smmu - > asids = devm_kzalloc ( dev , size , GFP_KERNEL ) ;
if ( ! smmu - > asids )
return ERR_PTR ( - ENOMEM ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
mutex_init ( & smmu - > lock ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
smmu - > regs = mc - > regs ;
smmu - > soc = soc ;
smmu - > dev = dev ;
smmu - > mc = mc ;
2011-11-17 07:31:31 +02:00
2015-03-27 11:07:27 +01:00
smmu - > pfn_mask = BIT_MASK ( mc - > soc - > num_address_bits - PAGE_SHIFT ) - 1 ;
dev_dbg ( dev , " address bits: %u, PFN mask: %#lx \n " ,
mc - > soc - > num_address_bits , smmu - > pfn_mask ) ;
2014-04-16 09:24:44 +02:00
value = SMMU_PTC_CONFIG_ENABLE | SMMU_PTC_CONFIG_INDEX_MAP ( 0x3f ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
if ( soc - > supports_request_limit )
value | = SMMU_PTC_CONFIG_REQ_LIMIT ( 8 ) ;
2012-08-02 11:46:40 +03:00
2014-04-16 09:24:44 +02:00
smmu_writel ( smmu , value , SMMU_PTC_CONFIG ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
value = SMMU_TLB_CONFIG_HIT_UNDER_MISS |
SMMU_TLB_CONFIG_ACTIVE_LINES ( 0x20 ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
if ( soc - > supports_round_robin_arbitration )
value | = SMMU_TLB_CONFIG_ROUND_ROBIN_ARBITRATION ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
smmu_writel ( smmu , value , SMMU_TLB_CONFIG ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
smmu_flush_ptc ( smmu , NULL , 0 ) ;
smmu_flush_tlb ( smmu ) ;
smmu_writel ( smmu , SMMU_CONFIG_ENABLE , SMMU_CONFIG ) ;
smmu_flush ( smmu ) ;
tegra_smmu_ahb_enable ( ) ;
2011-11-17 07:31:31 +02:00
2014-04-16 09:24:44 +02:00
err = bus_set_iommu ( & platform_bus_type , & tegra_smmu_ops ) ;
if ( err < 0 )
return ERR_PTR ( err ) ;
2011-11-17 07:31:31 +02:00
2015-01-23 09:49:25 +01:00
if ( IS_ENABLED ( CONFIG_DEBUG_FS ) )
tegra_smmu_debugfs_init ( smmu ) ;
2014-04-16 09:24:44 +02:00
return smmu ;
}
2015-01-23 09:49:25 +01:00
void tegra_smmu_remove ( struct tegra_smmu * smmu )
{
if ( IS_ENABLED ( CONFIG_DEBUG_FS ) )
tegra_smmu_debugfs_exit ( smmu ) ;
}