2011-11-09 15:31:15 +04:00
/*
* Copyright ( C ) 2010 - 2012 Advanced Micro Devices , Inc .
2015-02-04 18:12:55 +03:00
* Author : Joerg Roedel < jroedel @ suse . de >
2011-11-09 15:31:15 +04: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 .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
2011-11-24 19:21:52 +04:00
# include <linux/mmu_notifier.h>
2011-11-23 20:30:39 +04:00
# include <linux/amd-iommu.h>
# include <linux/mm_types.h>
2011-11-24 19:21:52 +04:00
# include <linux/profile.h>
2011-11-09 15:31:15 +04:00
# include <linux/module.h>
2011-11-24 13:41:57 +04:00
# include <linux/sched.h>
2011-11-23 20:30:39 +04:00
# include <linux/iommu.h>
2011-11-24 15:48:13 +04:00
# include <linux/wait.h>
2011-11-23 20:30:39 +04:00
# include <linux/pci.h>
# include <linux/gfp.h>
2011-11-24 15:48:13 +04:00
# include "amd_iommu_types.h"
2011-11-23 20:30:39 +04:00
# include "amd_iommu_proto.h"
2011-11-09 15:31:15 +04:00
MODULE_LICENSE ( " GPL v2 " ) ;
2015-02-04 18:12:55 +03:00
MODULE_AUTHOR ( " Joerg Roedel <jroedel@suse.de> " ) ;
2011-11-09 15:31:15 +04:00
2011-11-23 20:30:39 +04:00
# define MAX_DEVICES 0x10000
# define PRI_QUEUE_SIZE 512
struct pri_queue {
atomic_t inflight ;
bool finish ;
2011-11-24 15:48:13 +04:00
int status ;
2011-11-23 20:30:39 +04:00
} ;
struct pasid_state {
struct list_head list ; /* For global state-list */
atomic_t count ; /* Reference count */
2014-06-20 18:14:22 +04:00
unsigned mmu_notifier_count ; /* Counting nested mmu_notifier
2014-05-21 01:18:26 +04:00
calls */
2011-11-23 20:30:39 +04:00
struct mm_struct * mm ; /* mm_struct for the faults */
2014-07-08 14:49:50 +04:00
struct mmu_notifier mn ; /* mmu_notifier handle */
2011-11-23 20:30:39 +04:00
struct pri_queue pri [ PRI_QUEUE_SIZE ] ; /* PRI tag states */
struct device_state * device_state ; /* Link to our device_state */
int pasid ; /* PASID index */
2014-07-09 17:43:11 +04:00
bool invalid ; /* Used during setup and
teardown of the pasid */
2014-06-20 18:14:22 +04:00
spinlock_t lock ; /* Protect pri_queues and
mmu_notifer_count */
2011-11-24 15:48:13 +04:00
wait_queue_head_t wq ; /* To wait for count == 0 */
2011-11-23 20:30:39 +04:00
} ;
struct device_state {
2014-05-21 01:18:23 +04:00
struct list_head list ;
u16 devid ;
2011-11-23 20:30:39 +04:00
atomic_t count ;
struct pci_dev * pdev ;
struct pasid_state * * states ;
struct iommu_domain * domain ;
int pasid_levels ;
int max_pasids ;
2011-11-28 17:36:36 +04:00
amd_iommu_invalid_ppr_cb inv_ppr_cb ;
2011-12-07 15:24:42 +04:00
amd_iommu_invalidate_ctx inv_ctx_cb ;
2011-11-23 20:30:39 +04:00
spinlock_t lock ;
2011-11-24 15:48:13 +04:00
wait_queue_head_t wq ;
} ;
struct fault {
struct work_struct work ;
struct device_state * dev_state ;
struct pasid_state * state ;
struct mm_struct * mm ;
u64 address ;
u16 devid ;
u16 pasid ;
u16 tag ;
u16 finish ;
u16 flags ;
2011-11-23 20:30:39 +04:00
} ;
2014-05-21 01:18:23 +04:00
static LIST_HEAD ( state_list ) ;
2011-11-23 20:30:39 +04:00
static spinlock_t state_lock ;
2011-11-24 15:48:13 +04:00
static struct workqueue_struct * iommu_wq ;
2011-11-24 13:41:57 +04:00
static void free_pasid_states ( struct device_state * dev_state ) ;
2011-11-23 20:30:39 +04:00
static u16 device_id ( struct pci_dev * pdev )
{
u16 devid ;
devid = pdev - > bus - > number ;
devid = ( devid < < 8 ) | pdev - > devfn ;
return devid ;
}
2014-05-21 01:18:22 +04:00
static struct device_state * __get_device_state ( u16 devid )
{
2014-05-21 01:18:23 +04:00
struct device_state * dev_state ;
list_for_each_entry ( dev_state , & state_list , list ) {
if ( dev_state - > devid = = devid )
return dev_state ;
}
return NULL ;
2014-05-21 01:18:22 +04:00
}
2011-11-23 20:30:39 +04:00
static struct device_state * get_device_state ( u16 devid )
{
struct device_state * dev_state ;
unsigned long flags ;
spin_lock_irqsave ( & state_lock , flags ) ;
2014-05-21 01:18:22 +04:00
dev_state = __get_device_state ( devid ) ;
2011-11-23 20:30:39 +04:00
if ( dev_state ! = NULL )
atomic_inc ( & dev_state - > count ) ;
spin_unlock_irqrestore ( & state_lock , flags ) ;
return dev_state ;
}
static void free_device_state ( struct device_state * dev_state )
{
2015-07-28 17:58:47 +03:00
struct iommu_group * group ;
2011-11-24 13:41:57 +04:00
/*
* First detach device from domain - No more PRI requests will arrive
* from that device after it is unbound from the IOMMUv2 domain .
*/
2015-07-28 17:58:47 +03:00
group = iommu_group_get ( & dev_state - > pdev - > dev ) ;
if ( WARN_ON ( ! group ) )
return ;
iommu_detach_group ( dev_state - > domain , group ) ;
iommu_group_put ( group ) ;
2011-11-24 13:41:57 +04:00
/* Everything is down now, free the IOMMUv2 domain */
2011-11-23 20:30:39 +04:00
iommu_domain_free ( dev_state - > domain ) ;
2011-11-24 13:41:57 +04:00
/* Finally get rid of the device-state */
2011-11-23 20:30:39 +04:00
kfree ( dev_state ) ;
}
static void put_device_state ( struct device_state * dev_state )
{
if ( atomic_dec_and_test ( & dev_state - > count ) )
2011-11-24 15:48:13 +04:00
wake_up ( & dev_state - > wq ) ;
2011-11-23 20:30:39 +04:00
}
2011-11-24 13:41:57 +04:00
/* Must be called under dev_state->lock */
static struct pasid_state * * __get_pasid_state_ptr ( struct device_state * dev_state ,
int pasid , bool alloc )
{
struct pasid_state * * root , * * ptr ;
int level , index ;
level = dev_state - > pasid_levels ;
root = dev_state - > states ;
while ( true ) {
index = ( pasid > > ( 9 * level ) ) & 0x1ff ;
ptr = & root [ index ] ;
if ( level = = 0 )
break ;
if ( * ptr = = NULL ) {
if ( ! alloc )
return NULL ;
* ptr = ( void * ) get_zeroed_page ( GFP_ATOMIC ) ;
if ( * ptr = = NULL )
return NULL ;
}
root = ( struct pasid_state * * ) * ptr ;
level - = 1 ;
}
return ptr ;
}
static int set_pasid_state ( struct device_state * dev_state ,
struct pasid_state * pasid_state ,
int pasid )
{
struct pasid_state * * ptr ;
unsigned long flags ;
int ret ;
spin_lock_irqsave ( & dev_state - > lock , flags ) ;
ptr = __get_pasid_state_ptr ( dev_state , pasid , true ) ;
ret = - ENOMEM ;
if ( ptr = = NULL )
goto out_unlock ;
ret = - ENOMEM ;
if ( * ptr ! = NULL )
goto out_unlock ;
* ptr = pasid_state ;
ret = 0 ;
out_unlock :
spin_unlock_irqrestore ( & dev_state - > lock , flags ) ;
return ret ;
}
static void clear_pasid_state ( struct device_state * dev_state , int pasid )
{
struct pasid_state * * ptr ;
unsigned long flags ;
spin_lock_irqsave ( & dev_state - > lock , flags ) ;
ptr = __get_pasid_state_ptr ( dev_state , pasid , true ) ;
if ( ptr = = NULL )
goto out_unlock ;
* ptr = NULL ;
out_unlock :
spin_unlock_irqrestore ( & dev_state - > lock , flags ) ;
}
static struct pasid_state * get_pasid_state ( struct device_state * dev_state ,
int pasid )
{
struct pasid_state * * ptr , * ret = NULL ;
unsigned long flags ;
spin_lock_irqsave ( & dev_state - > lock , flags ) ;
ptr = __get_pasid_state_ptr ( dev_state , pasid , false ) ;
if ( ptr = = NULL )
goto out_unlock ;
ret = * ptr ;
if ( ret )
atomic_inc ( & ret - > count ) ;
out_unlock :
spin_unlock_irqrestore ( & dev_state - > lock , flags ) ;
return ret ;
}
static void free_pasid_state ( struct pasid_state * pasid_state )
{
kfree ( pasid_state ) ;
}
static void put_pasid_state ( struct pasid_state * pasid_state )
{
2014-11-10 13:21:39 +03:00
if ( atomic_dec_and_test ( & pasid_state - > count ) )
2011-11-24 15:48:13 +04:00
wake_up ( & pasid_state - > wq ) ;
2011-11-24 13:41:57 +04:00
}
2011-11-24 15:48:13 +04:00
static void put_pasid_state_wait ( struct pasid_state * pasid_state )
{
2015-04-16 17:08:44 +03:00
atomic_dec ( & pasid_state - > count ) ;
2015-02-04 17:50:38 +03:00
wait_event ( pasid_state - > wq , ! atomic_read ( & pasid_state - > count ) ) ;
2011-11-24 15:48:13 +04:00
free_pasid_state ( pasid_state ) ;
}
2014-07-08 16:19:35 +04:00
static void unbind_pasid ( struct pasid_state * pasid_state )
2011-11-24 19:21:52 +04:00
{
struct iommu_domain * domain ;
domain = pasid_state - > device_state - > domain ;
2014-07-08 17:01:43 +04:00
/*
* Mark pasid_state as invalid , no more faults will we added to the
* work queue after this is visible everywhere .
*/
pasid_state - > invalid = true ;
/* Make sure this is visible */
smp_wmb ( ) ;
/* After this the device/pasid can't access the mm anymore */
2011-11-24 19:21:52 +04:00
amd_iommu_domain_clear_gcr3 ( domain , pasid_state - > pasid ) ;
/* Make sure no more pending faults are in the queue */
flush_workqueue ( iommu_wq ) ;
}
2011-11-24 13:41:57 +04:00
static void free_pasid_states_level1 ( struct pasid_state * * tbl )
{
int i ;
for ( i = 0 ; i < 512 ; + + i ) {
if ( tbl [ i ] = = NULL )
continue ;
free_page ( ( unsigned long ) tbl [ i ] ) ;
}
}
static void free_pasid_states_level2 ( struct pasid_state * * tbl )
{
struct pasid_state * * ptr ;
int i ;
for ( i = 0 ; i < 512 ; + + i ) {
if ( tbl [ i ] = = NULL )
continue ;
ptr = ( struct pasid_state * * ) tbl [ i ] ;
free_pasid_states_level1 ( ptr ) ;
}
}
static void free_pasid_states ( struct device_state * dev_state )
{
struct pasid_state * pasid_state ;
int i ;
for ( i = 0 ; i < dev_state - > max_pasids ; + + i ) {
pasid_state = get_pasid_state ( dev_state , i ) ;
if ( pasid_state = = NULL )
continue ;
put_pasid_state ( pasid_state ) ;
2014-05-21 01:18:24 +04:00
/*
* This will call the mn_release function and
* unbind the PASID
*/
mmu_notifier_unregister ( & pasid_state - > mn , pasid_state - > mm ) ;
2014-07-08 16:15:45 +04:00
put_pasid_state_wait ( pasid_state ) ; /* Reference taken in
2014-07-30 18:04:40 +04:00
amd_iommu_bind_pasid */
2014-07-30 18:04:39 +04:00
/* Drop reference taken in amd_iommu_bind_pasid */
put_device_state ( dev_state ) ;
2011-11-24 13:41:57 +04:00
}
if ( dev_state - > pasid_levels = = 2 )
free_pasid_states_level2 ( dev_state - > states ) ;
else if ( dev_state - > pasid_levels = = 1 )
free_pasid_states_level1 ( dev_state - > states ) ;
2015-08-13 12:15:13 +03:00
else
BUG_ON ( dev_state - > pasid_levels ! = 0 ) ;
2011-11-24 13:41:57 +04:00
free_page ( ( unsigned long ) dev_state - > states ) ;
}
2011-11-24 19:21:52 +04:00
static struct pasid_state * mn_to_state ( struct mmu_notifier * mn )
{
return container_of ( mn , struct pasid_state , mn ) ;
}
static void __mn_flush_page ( struct mmu_notifier * mn ,
unsigned long address )
{
struct pasid_state * pasid_state ;
struct device_state * dev_state ;
pasid_state = mn_to_state ( mn ) ;
dev_state = pasid_state - > device_state ;
amd_iommu_flush_page ( dev_state - > domain , pasid_state - > pasid , address ) ;
}
static int mn_clear_flush_young ( struct mmu_notifier * mn ,
struct mm_struct * mm ,
2014-09-23 01:54:42 +04:00
unsigned long start ,
unsigned long end )
2011-11-24 19:21:52 +04:00
{
2014-09-23 01:54:42 +04:00
for ( ; start < end ; start + = PAGE_SIZE )
__mn_flush_page ( mn , start ) ;
2011-11-24 19:21:52 +04:00
return 0 ;
}
static void mn_invalidate_page ( struct mmu_notifier * mn ,
struct mm_struct * mm ,
unsigned long address )
{
__mn_flush_page ( mn , address ) ;
}
2014-11-13 05:46:09 +03:00
static void mn_invalidate_range ( struct mmu_notifier * mn ,
struct mm_struct * mm ,
unsigned long start , unsigned long end )
2011-11-24 19:21:52 +04:00
{
struct pasid_state * pasid_state ;
struct device_state * dev_state ;
pasid_state = mn_to_state ( mn ) ;
dev_state = pasid_state - > device_state ;
2014-11-13 05:46:09 +03:00
if ( ( start ^ ( end - 1 ) ) < PAGE_SIZE )
amd_iommu_flush_page ( dev_state - > domain , pasid_state - > pasid ,
start ) ;
else
amd_iommu_flush_tlb ( dev_state - > domain , pasid_state - > pasid ) ;
2011-11-24 19:21:52 +04:00
}
2014-05-21 01:18:24 +04:00
static void mn_release ( struct mmu_notifier * mn , struct mm_struct * mm )
{
struct pasid_state * pasid_state ;
struct device_state * dev_state ;
2014-07-09 17:43:11 +04:00
bool run_inv_ctx_cb ;
2014-05-21 01:18:24 +04:00
might_sleep ( ) ;
2014-07-09 17:43:11 +04:00
pasid_state = mn_to_state ( mn ) ;
dev_state = pasid_state - > device_state ;
run_inv_ctx_cb = ! pasid_state - > invalid ;
2014-05-21 01:18:24 +04:00
2015-02-20 13:52:01 +03:00
if ( run_inv_ctx_cb & & dev_state - > inv_ctx_cb )
2014-05-21 01:18:24 +04:00
dev_state - > inv_ctx_cb ( dev_state - > pdev , pasid_state - > pasid ) ;
2014-07-08 16:19:35 +04:00
unbind_pasid ( pasid_state ) ;
2014-05-21 01:18:24 +04:00
}
2011-11-24 19:21:52 +04:00
static struct mmu_notifier_ops iommu_mn = {
2014-05-21 01:18:24 +04:00
. release = mn_release ,
2011-11-24 19:21:52 +04:00
. clear_flush_young = mn_clear_flush_young ,
. invalidate_page = mn_invalidate_page ,
2014-11-13 05:46:09 +03:00
. invalidate_range = mn_invalidate_range ,
2011-11-24 19:21:52 +04:00
} ;
2011-11-24 15:48:13 +04:00
static void set_pri_tag_status ( struct pasid_state * pasid_state ,
u16 tag , int status )
{
unsigned long flags ;
spin_lock_irqsave ( & pasid_state - > lock , flags ) ;
pasid_state - > pri [ tag ] . status = status ;
spin_unlock_irqrestore ( & pasid_state - > lock , flags ) ;
}
static void finish_pri_tag ( struct device_state * dev_state ,
struct pasid_state * pasid_state ,
u16 tag )
{
unsigned long flags ;
spin_lock_irqsave ( & pasid_state - > lock , flags ) ;
if ( atomic_dec_and_test ( & pasid_state - > pri [ tag ] . inflight ) & &
pasid_state - > pri [ tag ] . finish ) {
amd_iommu_complete_ppr ( dev_state - > pdev , pasid_state - > pasid ,
pasid_state - > pri [ tag ] . status , tag ) ;
pasid_state - > pri [ tag ] . finish = false ;
pasid_state - > pri [ tag ] . status = PPR_SUCCESS ;
}
spin_unlock_irqrestore ( & pasid_state - > lock , flags ) ;
}
2014-12-13 03:55:30 +03:00
static void handle_fault_error ( struct fault * fault )
{
int status ;
if ( ! fault - > dev_state - > inv_ppr_cb ) {
set_pri_tag_status ( fault - > state , fault - > tag , PPR_INVALID ) ;
return ;
}
status = fault - > dev_state - > inv_ppr_cb ( fault - > dev_state - > pdev ,
fault - > pasid ,
fault - > address ,
fault - > flags ) ;
switch ( status ) {
case AMD_IOMMU_INV_PRI_RSP_SUCCESS :
set_pri_tag_status ( fault - > state , fault - > tag , PPR_SUCCESS ) ;
break ;
case AMD_IOMMU_INV_PRI_RSP_INVALID :
set_pri_tag_status ( fault - > state , fault - > tag , PPR_INVALID ) ;
break ;
case AMD_IOMMU_INV_PRI_RSP_FAIL :
set_pri_tag_status ( fault - > state , fault - > tag , PPR_FAILURE ) ;
break ;
default :
BUG ( ) ;
}
}
2015-11-17 18:11:36 +03:00
static bool access_error ( struct vm_area_struct * vma , struct fault * fault )
{
unsigned long requested = 0 ;
if ( fault - > flags & PPR_FAULT_EXEC )
requested | = VM_EXEC ;
if ( fault - > flags & PPR_FAULT_READ )
requested | = VM_READ ;
if ( fault - > flags & PPR_FAULT_WRITE )
requested | = VM_WRITE ;
return ( requested & ~ vma - > vm_flags ) ! = 0 ;
}
2011-11-24 15:48:13 +04:00
static void do_fault ( struct work_struct * work )
{
struct fault * fault = container_of ( work , struct fault , work ) ;
2014-12-13 03:55:30 +03:00
struct mm_struct * mm ;
struct vm_area_struct * vma ;
u64 address ;
int ret , write ;
2011-11-24 15:48:13 +04:00
write = ! ! ( fault - > flags & PPR_FAULT_WRITE ) ;
2014-12-13 03:55:30 +03:00
mm = fault - > state - > mm ;
address = fault - > address ;
down_read ( & mm - > mmap_sem ) ;
vma = find_extend_vma ( mm , address ) ;
if ( ! vma | | address < vma - > vm_start ) {
/* failed to get a vma in the right range */
up_read ( & mm - > mmap_sem ) ;
handle_fault_error ( fault ) ;
goto out ;
2011-11-28 17:36:36 +04:00
}
2011-11-24 15:48:13 +04:00
2015-11-17 18:11:36 +03:00
/* Check if we have the right permissions on the vma */
if ( access_error ( vma , fault ) ) {
2015-09-16 22:10:03 +03:00
up_read ( & mm - > mmap_sem ) ;
handle_fault_error ( fault ) ;
goto out ;
}
2014-12-13 03:55:30 +03:00
ret = handle_mm_fault ( mm , vma , address , write ) ;
if ( ret & VM_FAULT_ERROR ) {
/* failed to service fault */
up_read ( & mm - > mmap_sem ) ;
handle_fault_error ( fault ) ;
goto out ;
}
up_read ( & mm - > mmap_sem ) ;
out :
2011-11-24 15:48:13 +04:00
finish_pri_tag ( fault - > dev_state , fault - > state , fault - > tag ) ;
put_pasid_state ( fault - > state ) ;
kfree ( fault ) ;
}
static int ppr_notifier ( struct notifier_block * nb , unsigned long e , void * data )
{
struct amd_iommu_fault * iommu_fault ;
struct pasid_state * pasid_state ;
struct device_state * dev_state ;
unsigned long flags ;
struct fault * fault ;
bool finish ;
u16 tag ;
int ret ;
iommu_fault = data ;
tag = iommu_fault - > tag & 0x1ff ;
finish = ( iommu_fault - > tag > > 9 ) & 1 ;
ret = NOTIFY_DONE ;
dev_state = get_device_state ( iommu_fault - > device_id ) ;
if ( dev_state = = NULL )
goto out ;
pasid_state = get_pasid_state ( dev_state , iommu_fault - > pasid ) ;
2014-07-08 17:01:43 +04:00
if ( pasid_state = = NULL | | pasid_state - > invalid ) {
2011-11-24 15:48:13 +04:00
/* We know the device but not the PASID -> send INVALID */
amd_iommu_complete_ppr ( dev_state - > pdev , iommu_fault - > pasid ,
PPR_INVALID , tag ) ;
goto out_drop_state ;
}
spin_lock_irqsave ( & pasid_state - > lock , flags ) ;
atomic_inc ( & pasid_state - > pri [ tag ] . inflight ) ;
if ( finish )
pasid_state - > pri [ tag ] . finish = true ;
spin_unlock_irqrestore ( & pasid_state - > lock , flags ) ;
fault = kzalloc ( sizeof ( * fault ) , GFP_ATOMIC ) ;
if ( fault = = NULL ) {
/* We are OOM - send success and let the device re-fault */
finish_pri_tag ( dev_state , pasid_state , tag ) ;
goto out_drop_state ;
}
fault - > dev_state = dev_state ;
fault - > address = iommu_fault - > address ;
fault - > state = pasid_state ;
fault - > tag = tag ;
fault - > finish = finish ;
2014-07-08 18:30:16 +04:00
fault - > pasid = iommu_fault - > pasid ;
2011-11-24 15:48:13 +04:00
fault - > flags = iommu_fault - > flags ;
INIT_WORK ( & fault - > work , do_fault ) ;
queue_work ( iommu_wq , & fault - > work ) ;
ret = NOTIFY_OK ;
out_drop_state :
2014-07-08 16:55:10 +04:00
if ( ret ! = NOTIFY_OK & & pasid_state )
put_pasid_state ( pasid_state ) ;
2011-11-24 15:48:13 +04:00
put_device_state ( dev_state ) ;
out :
return ret ;
}
static struct notifier_block ppr_nb = {
. notifier_call = ppr_notifier ,
} ;
2011-11-24 13:41:57 +04:00
int amd_iommu_bind_pasid ( struct pci_dev * pdev , int pasid ,
struct task_struct * task )
{
struct pasid_state * pasid_state ;
struct device_state * dev_state ;
2014-07-08 17:15:07 +04:00
struct mm_struct * mm ;
2011-11-24 13:41:57 +04:00
u16 devid ;
int ret ;
might_sleep ( ) ;
if ( ! amd_iommu_v2_supported ( ) )
return - ENODEV ;
devid = device_id ( pdev ) ;
dev_state = get_device_state ( devid ) ;
if ( dev_state = = NULL )
return - EINVAL ;
ret = - EINVAL ;
if ( pasid < 0 | | pasid > = dev_state - > max_pasids )
goto out ;
ret = - ENOMEM ;
pasid_state = kzalloc ( sizeof ( * pasid_state ) , GFP_KERNEL ) ;
if ( pasid_state = = NULL )
goto out ;
2014-07-08 17:15:07 +04:00
2011-11-24 13:41:57 +04:00
atomic_set ( & pasid_state - > count , 1 ) ;
2011-11-24 15:48:13 +04:00
init_waitqueue_head ( & pasid_state - > wq ) ;
2012-07-19 12:56:10 +04:00
spin_lock_init ( & pasid_state - > lock ) ;
2014-07-08 17:15:07 +04:00
mm = get_task_mm ( task ) ;
pasid_state - > mm = mm ;
2011-11-24 13:41:57 +04:00
pasid_state - > device_state = dev_state ;
pasid_state - > pasid = pasid ;
2014-07-09 17:43:11 +04:00
pasid_state - > invalid = true ; /* Mark as valid only if we are
done with setting up the pasid */
2011-11-24 19:21:52 +04:00
pasid_state - > mn . ops = & iommu_mn ;
2011-11-24 13:41:57 +04:00
if ( pasid_state - > mm = = NULL )
goto out_free ;
2014-07-08 17:15:07 +04:00
mmu_notifier_register ( & pasid_state - > mn , mm ) ;
2011-11-24 19:21:52 +04:00
2011-11-24 13:41:57 +04:00
ret = set_pasid_state ( dev_state , pasid_state , pasid ) ;
if ( ret )
2011-11-24 19:21:52 +04:00
goto out_unregister ;
2011-11-24 13:41:57 +04:00
ret = amd_iommu_domain_set_gcr3 ( dev_state - > domain , pasid ,
__pa ( pasid_state - > mm - > pgd ) ) ;
if ( ret )
goto out_clear_state ;
2014-07-09 17:43:11 +04:00
/* Now we are ready to handle faults */
pasid_state - > invalid = false ;
2014-07-08 17:15:07 +04:00
/*
* Drop the reference to the mm_struct here . We rely on the
* mmu_notifier release call - back to inform us when the mm
* is going away .
*/
mmput ( mm ) ;
2011-11-24 13:41:57 +04:00
return 0 ;
out_clear_state :
clear_pasid_state ( dev_state , pasid ) ;
2011-11-24 19:21:52 +04:00
out_unregister :
2014-07-08 17:15:07 +04:00
mmu_notifier_unregister ( & pasid_state - > mn , mm ) ;
2011-11-24 19:21:52 +04:00
2011-11-24 13:41:57 +04:00
out_free :
2014-07-08 17:15:07 +04:00
mmput ( mm ) ;
2011-11-24 15:48:13 +04:00
free_pasid_state ( pasid_state ) ;
2011-11-24 13:41:57 +04:00
out :
put_device_state ( dev_state ) ;
return ret ;
}
EXPORT_SYMBOL ( amd_iommu_bind_pasid ) ;
void amd_iommu_unbind_pasid ( struct pci_dev * pdev , int pasid )
{
2014-05-21 01:18:24 +04:00
struct pasid_state * pasid_state ;
2011-11-24 13:41:57 +04:00
struct device_state * dev_state ;
u16 devid ;
might_sleep ( ) ;
if ( ! amd_iommu_v2_supported ( ) )
return ;
devid = device_id ( pdev ) ;
dev_state = get_device_state ( devid ) ;
if ( dev_state = = NULL )
return ;
if ( pasid < 0 | | pasid > = dev_state - > max_pasids )
goto out ;
2014-05-21 01:18:24 +04:00
pasid_state = get_pasid_state ( dev_state , pasid ) ;
if ( pasid_state = = NULL )
goto out ;
/*
* Drop reference taken here . We are safe because we still hold
* the reference taken in the amd_iommu_bind_pasid function .
*/
put_pasid_state ( pasid_state ) ;
2014-07-08 17:01:43 +04:00
/* Clear the pasid state so that the pasid can be re-used */
clear_pasid_state ( dev_state , pasid_state - > pasid ) ;
2014-07-08 17:15:07 +04:00
/*
2014-07-30 18:04:37 +04:00
* Call mmu_notifier_unregister to drop our reference
* to pasid_state - > mm
2014-07-08 17:15:07 +04:00
*/
2014-07-30 18:04:37 +04:00
mmu_notifier_unregister ( & pasid_state - > mn , pasid_state - > mm ) ;
2011-11-24 13:41:57 +04:00
2014-07-08 16:15:45 +04:00
put_pasid_state_wait ( pasid_state ) ; /* Reference taken in
2014-07-30 18:04:40 +04:00
amd_iommu_bind_pasid */
2011-11-24 13:41:57 +04:00
out :
2014-07-30 18:04:39 +04:00
/* Drop reference taken in this function */
put_device_state ( dev_state ) ;
/* Drop reference taken in amd_iommu_bind_pasid */
2011-11-24 13:41:57 +04:00
put_device_state ( dev_state ) ;
}
EXPORT_SYMBOL ( amd_iommu_unbind_pasid ) ;
2011-11-23 20:30:39 +04:00
int amd_iommu_init_device ( struct pci_dev * pdev , int pasids )
{
struct device_state * dev_state ;
2015-07-28 17:58:47 +03:00
struct iommu_group * group ;
2011-11-23 20:30:39 +04:00
unsigned long flags ;
int ret , tmp ;
u16 devid ;
might_sleep ( ) ;
if ( ! amd_iommu_v2_supported ( ) )
return - ENODEV ;
if ( pasids < = 0 | | pasids > ( PASID_MASK + 1 ) )
return - EINVAL ;
devid = device_id ( pdev ) ;
dev_state = kzalloc ( sizeof ( * dev_state ) , GFP_KERNEL ) ;
if ( dev_state = = NULL )
return - ENOMEM ;
spin_lock_init ( & dev_state - > lock ) ;
2011-11-24 15:48:13 +04:00
init_waitqueue_head ( & dev_state - > wq ) ;
2014-05-21 01:18:23 +04:00
dev_state - > pdev = pdev ;
dev_state - > devid = devid ;
2011-11-23 20:30:39 +04:00
tmp = pasids ;
for ( dev_state - > pasid_levels = 0 ; ( tmp - 1 ) & ~ 0x1ff ; tmp > > = 9 )
dev_state - > pasid_levels + = 1 ;
atomic_set ( & dev_state - > count , 1 ) ;
dev_state - > max_pasids = pasids ;
ret = - ENOMEM ;
dev_state - > states = ( void * ) get_zeroed_page ( GFP_KERNEL ) ;
if ( dev_state - > states = = NULL )
goto out_free_dev_state ;
dev_state - > domain = iommu_domain_alloc ( & pci_bus_type ) ;
if ( dev_state - > domain = = NULL )
goto out_free_states ;
amd_iommu_domain_direct_map ( dev_state - > domain ) ;
ret = amd_iommu_domain_enable_v2 ( dev_state - > domain , pasids ) ;
if ( ret )
goto out_free_domain ;
2015-07-28 17:58:47 +03:00
group = iommu_group_get ( & pdev - > dev ) ;
if ( ! group )
2011-11-23 20:30:39 +04:00
goto out_free_domain ;
2015-07-28 17:58:47 +03:00
ret = iommu_attach_group ( dev_state - > domain , group ) ;
if ( ret ! = 0 )
goto out_drop_group ;
iommu_group_put ( group ) ;
2011-11-23 20:30:39 +04:00
spin_lock_irqsave ( & state_lock , flags ) ;
2014-05-21 01:18:23 +04:00
if ( __get_device_state ( devid ) ! = NULL ) {
2011-11-23 20:30:39 +04:00
spin_unlock_irqrestore ( & state_lock , flags ) ;
ret = - EBUSY ;
goto out_free_domain ;
}
2014-05-21 01:18:23 +04:00
list_add_tail ( & dev_state - > list , & state_list ) ;
2011-11-23 20:30:39 +04:00
spin_unlock_irqrestore ( & state_lock , flags ) ;
return 0 ;
2015-07-28 17:58:47 +03:00
out_drop_group :
iommu_group_put ( group ) ;
2011-11-23 20:30:39 +04:00
out_free_domain :
iommu_domain_free ( dev_state - > domain ) ;
out_free_states :
free_page ( ( unsigned long ) dev_state - > states ) ;
out_free_dev_state :
kfree ( dev_state ) ;
return ret ;
}
EXPORT_SYMBOL ( amd_iommu_init_device ) ;
void amd_iommu_free_device ( struct pci_dev * pdev )
{
struct device_state * dev_state ;
unsigned long flags ;
u16 devid ;
if ( ! amd_iommu_v2_supported ( ) )
return ;
devid = device_id ( pdev ) ;
spin_lock_irqsave ( & state_lock , flags ) ;
2014-05-21 01:18:22 +04:00
dev_state = __get_device_state ( devid ) ;
2011-11-23 20:30:39 +04:00
if ( dev_state = = NULL ) {
spin_unlock_irqrestore ( & state_lock , flags ) ;
return ;
}
2014-05-21 01:18:23 +04:00
list_del ( & dev_state - > list ) ;
2011-11-23 20:30:39 +04:00
spin_unlock_irqrestore ( & state_lock , flags ) ;
2011-11-24 13:41:57 +04:00
/* Get rid of any remaining pasid states */
free_pasid_states ( dev_state ) ;
2015-02-03 15:25:51 +03:00
put_device_state ( dev_state ) ;
/*
* Wait until the last reference is dropped before freeing
* the device state .
*/
wait_event ( dev_state - > wq , ! atomic_read ( & dev_state - > count ) ) ;
free_device_state ( dev_state ) ;
2011-11-23 20:30:39 +04:00
}
EXPORT_SYMBOL ( amd_iommu_free_device ) ;
2011-11-28 17:36:36 +04:00
int amd_iommu_set_invalid_ppr_cb ( struct pci_dev * pdev ,
amd_iommu_invalid_ppr_cb cb )
{
struct device_state * dev_state ;
unsigned long flags ;
u16 devid ;
int ret ;
if ( ! amd_iommu_v2_supported ( ) )
return - ENODEV ;
devid = device_id ( pdev ) ;
spin_lock_irqsave ( & state_lock , flags ) ;
ret = - EINVAL ;
2014-05-21 01:18:22 +04:00
dev_state = __get_device_state ( devid ) ;
2011-11-28 17:36:36 +04:00
if ( dev_state = = NULL )
goto out_unlock ;
dev_state - > inv_ppr_cb = cb ;
ret = 0 ;
out_unlock :
spin_unlock_irqrestore ( & state_lock , flags ) ;
return ret ;
}
EXPORT_SYMBOL ( amd_iommu_set_invalid_ppr_cb ) ;
2011-12-07 15:24:42 +04:00
int amd_iommu_set_invalidate_ctx_cb ( struct pci_dev * pdev ,
amd_iommu_invalidate_ctx cb )
{
struct device_state * dev_state ;
unsigned long flags ;
u16 devid ;
int ret ;
if ( ! amd_iommu_v2_supported ( ) )
return - ENODEV ;
devid = device_id ( pdev ) ;
spin_lock_irqsave ( & state_lock , flags ) ;
ret = - EINVAL ;
2014-05-21 01:18:22 +04:00
dev_state = __get_device_state ( devid ) ;
2011-12-07 15:24:42 +04:00
if ( dev_state = = NULL )
goto out_unlock ;
dev_state - > inv_ctx_cb = cb ;
ret = 0 ;
out_unlock :
spin_unlock_irqrestore ( & state_lock , flags ) ;
return ret ;
}
EXPORT_SYMBOL ( amd_iommu_set_invalidate_ctx_cb ) ;
2011-11-09 15:31:15 +04:00
static int __init amd_iommu_v2_init ( void )
{
2011-11-24 15:48:13 +04:00
int ret ;
2011-11-23 20:30:39 +04:00
2015-02-04 18:12:55 +03:00
pr_info ( " AMD IOMMUv2 driver by Joerg Roedel <jroedel@suse.de> \n " ) ;
2012-03-15 15:46:40 +04:00
if ( ! amd_iommu_v2_supported ( ) ) {
2012-07-21 21:21:32 +04:00
pr_info ( " AMD IOMMUv2 functionality not available on this system \n " ) ;
2012-03-15 15:46:40 +04:00
/*
* Load anyway to provide the symbols to other modules
* which may use AMD IOMMUv2 optionally .
*/
return 0 ;
}
2011-11-09 15:31:15 +04:00
2011-11-23 20:30:39 +04:00
spin_lock_init ( & state_lock ) ;
2011-11-24 15:48:13 +04:00
ret = - ENOMEM ;
iommu_wq = create_workqueue ( " amd_iommu_v2 " ) ;
2011-11-24 19:21:52 +04:00
if ( iommu_wq = = NULL )
2014-05-21 01:18:23 +04:00
goto out ;
2011-11-24 19:21:52 +04:00
2011-11-24 15:48:13 +04:00
amd_iommu_register_ppr_notifier ( & ppr_nb ) ;
2011-11-09 15:31:15 +04:00
return 0 ;
2011-11-24 15:48:13 +04:00
2014-05-21 01:18:23 +04:00
out :
2011-11-24 15:48:13 +04:00
return ret ;
2011-11-09 15:31:15 +04:00
}
static void __exit amd_iommu_v2_exit ( void )
{
2011-11-23 20:30:39 +04:00
struct device_state * dev_state ;
int i ;
2012-03-15 15:46:40 +04:00
if ( ! amd_iommu_v2_supported ( ) )
return ;
2011-11-24 15:48:13 +04:00
amd_iommu_unregister_ppr_notifier ( & ppr_nb ) ;
flush_workqueue ( iommu_wq ) ;
/*
* The loop below might call flush_workqueue ( ) , so call
* destroy_workqueue ( ) after it
*/
2011-11-23 20:30:39 +04:00
for ( i = 0 ; i < MAX_DEVICES ; + + i ) {
dev_state = get_device_state ( i ) ;
if ( dev_state = = NULL )
continue ;
WARN_ON_ONCE ( 1 ) ;
put_device_state ( dev_state ) ;
2011-11-24 15:48:13 +04:00
amd_iommu_free_device ( dev_state - > pdev ) ;
2011-11-23 20:30:39 +04:00
}
2011-11-24 15:48:13 +04:00
destroy_workqueue ( iommu_wq ) ;
2011-11-09 15:31:15 +04:00
}
module_init ( amd_iommu_v2_init ) ;
module_exit ( amd_iommu_v2_exit ) ;