2014-10-08 19:55:02 +11:00
/*
* Copyright 2014 IBM Corp .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/spinlock.h>
# include <linux/sched.h>
# include <linux/slab.h>
# include <linux/sched.h>
# include <linux/mutex.h>
# include <linux/mm.h>
# include <linux/uaccess.h>
2016-04-22 14:57:49 +10:00
# include <linux/delay.h>
2014-10-08 19:55:02 +11:00
# include <asm/synch.h>
2015-05-27 16:07:16 +10:00
# include <misc/cxl-base.h>
2014-10-08 19:55:02 +11:00
# include "cxl.h"
2015-01-09 20:34:36 +11:00
# include "trace.h"
2014-10-08 19:55:02 +11:00
static int afu_control ( struct cxl_afu * afu , u64 command ,
u64 result , u64 mask , bool enabled )
{
u64 AFU_Cntl = cxl_p2n_read ( afu , CXL_AFU_Cntl_An ) ;
unsigned long timeout = jiffies + ( HZ * CXL_TIMEOUT ) ;
2015-01-09 20:34:36 +11:00
int rc = 0 ;
2014-10-08 19:55:02 +11:00
spin_lock ( & afu - > afu_cntl_lock ) ;
pr_devel ( " AFU command starting: %llx \n " , command ) ;
2015-01-09 20:34:36 +11:00
trace_cxl_afu_ctrl ( afu , command ) ;
2014-10-08 19:55:02 +11:00
cxl_p2n_write ( afu , CXL_AFU_Cntl_An , AFU_Cntl | command ) ;
AFU_Cntl = cxl_p2n_read ( afu , CXL_AFU_Cntl_An ) ;
while ( ( AFU_Cntl & mask ) ! = result ) {
if ( time_after_eq ( jiffies , timeout ) ) {
dev_warn ( & afu - > dev , " WARNING: AFU control timed out! \n " ) ;
2015-01-09 20:34:36 +11:00
rc = - EBUSY ;
goto out ;
2014-10-08 19:55:02 +11:00
}
2015-08-14 17:41:18 +10:00
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) {
2015-08-14 17:41:18 +10:00
afu - > enabled = enabled ;
rc = - EIO ;
goto out ;
}
2015-06-11 13:27:52 +02:00
pr_devel_ratelimited ( " AFU control... (0x%016llx) \n " ,
2014-10-08 19:55:02 +11:00
AFU_Cntl | command ) ;
cpu_relax ( ) ;
AFU_Cntl = cxl_p2n_read ( afu , CXL_AFU_Cntl_An ) ;
} ;
pr_devel ( " AFU command complete: %llx \n " , command ) ;
afu - > enabled = enabled ;
2015-01-09 20:34:36 +11:00
out :
trace_cxl_afu_ctrl_done ( afu , command , rc ) ;
2014-10-08 19:55:02 +11:00
spin_unlock ( & afu - > afu_cntl_lock ) ;
2015-01-09 20:34:36 +11:00
return rc ;
2014-10-08 19:55:02 +11:00
}
static int afu_enable ( struct cxl_afu * afu )
{
pr_devel ( " AFU enable request \n " ) ;
return afu_control ( afu , CXL_AFU_Cntl_An_E ,
CXL_AFU_Cntl_An_ES_Enabled ,
CXL_AFU_Cntl_An_ES_MASK , true ) ;
}
int cxl_afu_disable ( struct cxl_afu * afu )
{
pr_devel ( " AFU disable request \n " ) ;
return afu_control ( afu , 0 , CXL_AFU_Cntl_An_ES_Disabled ,
CXL_AFU_Cntl_An_ES_MASK , false ) ;
}
/* This will disable as well as reset */
2016-03-04 12:26:29 +01:00
static int native_afu_reset ( struct cxl_afu * afu )
2014-10-08 19:55:02 +11:00
{
pr_devel ( " AFU reset request \n " ) ;
return afu_control ( afu , CXL_AFU_Cntl_An_RA ,
CXL_AFU_Cntl_An_RS_Complete | CXL_AFU_Cntl_An_ES_Disabled ,
CXL_AFU_Cntl_An_RS_MASK | CXL_AFU_Cntl_An_ES_MASK ,
false ) ;
}
2016-03-04 12:26:29 +01:00
static int native_afu_check_and_enable ( struct cxl_afu * afu )
2014-10-08 19:55:02 +11:00
{
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) {
2015-08-14 17:41:18 +10:00
WARN ( 1 , " Refusing to enable afu while link down! \n " ) ;
return - EIO ;
}
2014-10-08 19:55:02 +11:00
if ( afu - > enabled )
return 0 ;
return afu_enable ( afu ) ;
}
int cxl_psl_purge ( struct cxl_afu * afu )
{
u64 PSL_CNTL = cxl_p1n_read ( afu , CXL_PSL_SCNTL_An ) ;
u64 AFU_Cntl = cxl_p2n_read ( afu , CXL_AFU_Cntl_An ) ;
u64 dsisr , dar ;
u64 start , end ;
unsigned long timeout = jiffies + ( HZ * CXL_TIMEOUT ) ;
2015-01-09 20:34:36 +11:00
int rc = 0 ;
trace_cxl_psl_ctrl ( afu , CXL_PSL_SCNTL_An_Pc ) ;
2014-10-08 19:55:02 +11:00
pr_devel ( " PSL purge request \n " ) ;
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) {
2015-08-14 17:41:18 +10:00
dev_warn ( & afu - > dev , " PSL Purge called with link down, ignoring \n " ) ;
rc = - EIO ;
goto out ;
}
2014-10-08 19:55:02 +11:00
if ( ( AFU_Cntl & CXL_AFU_Cntl_An_ES_MASK ) ! = CXL_AFU_Cntl_An_ES_Disabled ) {
WARN ( 1 , " psl_purge request while AFU not disabled! \n " ) ;
cxl_afu_disable ( afu ) ;
}
cxl_p1n_write ( afu , CXL_PSL_SCNTL_An ,
PSL_CNTL | CXL_PSL_SCNTL_An_Pc ) ;
start = local_clock ( ) ;
PSL_CNTL = cxl_p1n_read ( afu , CXL_PSL_SCNTL_An ) ;
while ( ( PSL_CNTL & CXL_PSL_SCNTL_An_Ps_MASK )
= = CXL_PSL_SCNTL_An_Ps_Pending ) {
if ( time_after_eq ( jiffies , timeout ) ) {
dev_warn ( & afu - > dev , " WARNING: PSL Purge timed out! \n " ) ;
2015-01-09 20:34:36 +11:00
rc = - EBUSY ;
goto out ;
2014-10-08 19:55:02 +11:00
}
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) {
2015-08-14 17:41:18 +10:00
rc = - EIO ;
goto out ;
}
2014-10-08 19:55:02 +11:00
dsisr = cxl_p2n_read ( afu , CXL_PSL_DSISR_An ) ;
2015-06-11 13:27:52 +02:00
pr_devel_ratelimited ( " PSL purging... PSL_CNTL: 0x%016llx PSL_DSISR: 0x%016llx \n " , PSL_CNTL , dsisr ) ;
2014-10-08 19:55:02 +11:00
if ( dsisr & CXL_PSL_DSISR_TRANS ) {
dar = cxl_p2n_read ( afu , CXL_PSL_DAR_An ) ;
2015-06-11 13:27:52 +02:00
dev_notice ( & afu - > dev , " PSL purge terminating pending translation, DSISR: 0x%016llx, DAR: 0x%016llx \n " , dsisr , dar ) ;
2014-10-08 19:55:02 +11:00
cxl_p2n_write ( afu , CXL_PSL_TFC_An , CXL_PSL_TFC_An_AE ) ;
} else if ( dsisr ) {
2015-06-11 13:27:52 +02:00
dev_notice ( & afu - > dev , " PSL purge acknowledging pending non-translation fault, DSISR: 0x%016llx \n " , dsisr ) ;
2014-10-08 19:55:02 +11:00
cxl_p2n_write ( afu , CXL_PSL_TFC_An , CXL_PSL_TFC_An_A ) ;
} else {
cpu_relax ( ) ;
}
PSL_CNTL = cxl_p1n_read ( afu , CXL_PSL_SCNTL_An ) ;
} ;
end = local_clock ( ) ;
pr_devel ( " PSL purged in %lld ns \n " , end - start ) ;
cxl_p1n_write ( afu , CXL_PSL_SCNTL_An ,
PSL_CNTL & ~ CXL_PSL_SCNTL_An_Pc ) ;
2015-01-09 20:34:36 +11:00
out :
trace_cxl_psl_ctrl_done ( afu , CXL_PSL_SCNTL_An_Pc , rc ) ;
return rc ;
2014-10-08 19:55:02 +11:00
}
static int spa_max_procs ( int spa_size )
{
/*
* From the CAIA :
* end_of_SPA_area = SPA_Base + ( ( n + 4 ) * 128 ) + ( ( ( ( n * 8 ) + 127 ) > > 7 ) * 128 ) + 255
* Most of that junk is really just an overly - complicated way of saying
* the last 256 bytes are __aligned ( 128 ) , so it ' s really :
* end_of_SPA_area = end_of_PSL_queue_area + __aligned ( 128 ) 255
* and
* end_of_PSL_queue_area = SPA_Base + ( ( n + 4 ) * 128 ) + ( n * 8 ) - 1
* so
* sizeof ( SPA ) = ( ( n + 4 ) * 128 ) + ( n * 8 ) + __aligned ( 128 ) 256
* Ignore the alignment ( which is safe in this case as long as we are
* careful with our rounding ) and solve for n :
*/
return ( ( spa_size / 8 ) - 96 ) / 17 ;
}
2015-08-14 17:41:19 +10:00
int cxl_alloc_spa ( struct cxl_afu * afu )
2014-10-08 19:55:02 +11:00
{
2016-05-04 14:46:30 +10:00
unsigned spa_size ;
2014-10-08 19:55:02 +11:00
/* Work out how many pages to allocate */
2016-03-04 12:26:35 +01:00
afu - > native - > spa_order = 0 ;
2014-10-08 19:55:02 +11:00
do {
2016-03-04 12:26:35 +01:00
afu - > native - > spa_order + + ;
2016-05-04 14:46:30 +10:00
spa_size = ( 1 < < afu - > native - > spa_order ) * PAGE_SIZE ;
if ( spa_size > 0x100000 ) {
dev_warn ( & afu - > dev , " num_of_processes too large for the SPA, limiting to %i (0x%x) \n " ,
afu - > native - > spa_max_procs , afu - > native - > spa_size ) ;
afu - > num_procs = afu - > native - > spa_max_procs ;
break ;
}
afu - > native - > spa_size = spa_size ;
2016-03-04 12:26:35 +01:00
afu - > native - > spa_max_procs = spa_max_procs ( afu - > native - > spa_size ) ;
} while ( afu - > native - > spa_max_procs < afu - > num_procs ) ;
2014-10-08 19:55:02 +11:00
2016-03-04 12:26:35 +01:00
if ( ! ( afu - > native - > spa = ( struct cxl_process_element * )
__get_free_pages ( GFP_KERNEL | __GFP_ZERO , afu - > native - > spa_order ) ) ) {
2014-10-08 19:55:02 +11:00
pr_err ( " cxl_alloc_spa: Unable to allocate scheduled process area \n " ) ;
return - ENOMEM ;
}
pr_devel ( " spa pages: %i afu->spa_max_procs: %i afu->num_procs: %i \n " ,
2016-03-04 12:26:35 +01:00
1 < < afu - > native - > spa_order , afu - > native - > spa_max_procs , afu - > num_procs ) ;
2014-10-08 19:55:02 +11:00
2015-08-14 17:41:19 +10:00
return 0 ;
}
static void attach_spa ( struct cxl_afu * afu )
{
u64 spap ;
2016-03-04 12:26:35 +01:00
afu - > native - > sw_command_status = ( __be64 * ) ( ( char * ) afu - > native - > spa +
( ( afu - > native - > spa_max_procs + 3 ) * 128 ) ) ;
2014-10-08 19:55:02 +11:00
2016-03-04 12:26:35 +01:00
spap = virt_to_phys ( afu - > native - > spa ) & CXL_PSL_SPAP_Addr ;
spap | = ( ( afu - > native - > spa_size > > ( 12 - CXL_PSL_SPAP_Size_Shift ) ) - 1 ) & CXL_PSL_SPAP_Size ;
2014-10-08 19:55:02 +11:00
spap | = CXL_PSL_SPAP_V ;
2016-03-04 12:26:35 +01:00
pr_devel ( " cxl: SPA allocated at 0x%p. Max processes: %i, sw_command_status: 0x%p CXL_PSL_SPAP_An=0x%016llx \n " ,
afu - > native - > spa , afu - > native - > spa_max_procs ,
afu - > native - > sw_command_status , spap ) ;
2014-10-08 19:55:02 +11:00
cxl_p1n_write ( afu , CXL_PSL_SPAP_An , spap ) ;
}
2015-08-14 17:41:19 +10:00
static inline void detach_spa ( struct cxl_afu * afu )
2014-10-08 19:55:02 +11:00
{
2014-12-08 19:18:00 +11:00
cxl_p1n_write ( afu , CXL_PSL_SPAP_An , 0 ) ;
2015-08-14 17:41:19 +10:00
}
void cxl_release_spa ( struct cxl_afu * afu )
{
2016-03-04 12:26:35 +01:00
if ( afu - > native - > spa ) {
free_pages ( ( unsigned long ) afu - > native - > spa ,
afu - > native - > spa_order ) ;
afu - > native - > spa = NULL ;
2015-08-14 17:41:19 +10:00
}
2014-10-08 19:55:02 +11:00
}
int cxl_tlb_slb_invalidate ( struct cxl * adapter )
{
unsigned long timeout = jiffies + ( HZ * CXL_TIMEOUT ) ;
pr_devel ( " CXL adapter wide TLBIA & SLBIA \n " ) ;
cxl_p1_write ( adapter , CXL_PSL_AFUSEL , CXL_PSL_AFUSEL_A ) ;
cxl_p1_write ( adapter , CXL_PSL_TLBIA , CXL_TLB_SLB_IQ_ALL ) ;
while ( cxl_p1_read ( adapter , CXL_PSL_TLBIA ) & CXL_TLB_SLB_P ) {
if ( time_after_eq ( jiffies , timeout ) ) {
dev_warn ( & adapter - > dev , " WARNING: CXL adapter wide TLBIA timed out! \n " ) ;
return - EBUSY ;
}
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( adapter , NULL ) )
2015-08-14 17:41:18 +10:00
return - EIO ;
2014-10-08 19:55:02 +11:00
cpu_relax ( ) ;
}
cxl_p1_write ( adapter , CXL_PSL_SLBIA , CXL_TLB_SLB_IQ_ALL ) ;
while ( cxl_p1_read ( adapter , CXL_PSL_SLBIA ) & CXL_TLB_SLB_P ) {
if ( time_after_eq ( jiffies , timeout ) ) {
dev_warn ( & adapter - > dev , " WARNING: CXL adapter wide SLBIA timed out! \n " ) ;
return - EBUSY ;
}
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( adapter , NULL ) )
2015-08-14 17:41:18 +10:00
return - EIO ;
2014-10-08 19:55:02 +11:00
cpu_relax ( ) ;
}
return 0 ;
}
static int cxl_write_sstp ( struct cxl_afu * afu , u64 sstp0 , u64 sstp1 )
{
int rc ;
/* 1. Disable SSTP by writing 0 to SSTP1[V] */
cxl_p2n_write ( afu , CXL_SSTP1_An , 0 ) ;
/* 2. Invalidate all SLB entries */
if ( ( rc = cxl_afu_slbia ( afu ) ) )
return rc ;
/* 3. Set SSTP0_An */
cxl_p2n_write ( afu , CXL_SSTP0_An , sstp0 ) ;
/* 4. Set SSTP1_An */
cxl_p2n_write ( afu , CXL_SSTP1_An , sstp1 ) ;
return 0 ;
}
/* Using per slice version may improve performance here. (ie. SLBIA_An) */
static void slb_invalid ( struct cxl_context * ctx )
{
struct cxl * adapter = ctx - > afu - > adapter ;
u64 slbia ;
2016-03-04 12:26:35 +01:00
WARN_ON ( ! mutex_is_locked ( & ctx - > afu - > native - > spa_mutex ) ) ;
2014-10-08 19:55:02 +11:00
cxl_p1_write ( adapter , CXL_PSL_LBISEL ,
( ( u64 ) be32_to_cpu ( ctx - > elem - > common . pid ) < < 32 ) |
be32_to_cpu ( ctx - > elem - > lpid ) ) ;
cxl_p1_write ( adapter , CXL_PSL_SLBIA , CXL_TLB_SLB_IQ_LPIDPID ) ;
while ( 1 ) {
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( adapter , NULL ) )
2015-08-14 17:41:18 +10:00
break ;
2014-10-08 19:55:02 +11:00
slbia = cxl_p1_read ( adapter , CXL_PSL_SLBIA ) ;
if ( ! ( slbia & CXL_TLB_SLB_P ) )
break ;
cpu_relax ( ) ;
}
}
static int do_process_element_cmd ( struct cxl_context * ctx ,
u64 cmd , u64 pe_state )
{
u64 state ;
2014-12-08 19:17:56 +11:00
unsigned long timeout = jiffies + ( HZ * CXL_TIMEOUT ) ;
2015-01-09 20:34:36 +11:00
int rc = 0 ;
trace_cxl_llcmd ( ctx , cmd ) ;
2014-10-08 19:55:02 +11:00
WARN_ON ( ! ctx - > afu - > enabled ) ;
ctx - > elem - > software_state = cpu_to_be32 ( pe_state ) ;
smp_wmb ( ) ;
2016-03-04 12:26:35 +01:00
* ( ctx - > afu - > native - > sw_command_status ) = cpu_to_be64 ( cmd | 0 | ctx - > pe ) ;
2014-10-08 19:55:02 +11:00
smp_mb ( ) ;
cxl_p1n_write ( ctx - > afu , CXL_PSL_LLCMD_An , cmd | ctx - > pe ) ;
while ( 1 ) {
2014-12-08 19:17:56 +11:00
if ( time_after_eq ( jiffies , timeout ) ) {
dev_warn ( & ctx - > afu - > dev , " WARNING: Process Element Command timed out! \n " ) ;
2015-01-09 20:34:36 +11:00
rc = - EBUSY ;
goto out ;
2014-12-08 19:17:56 +11:00
}
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( ctx - > afu - > adapter , ctx - > afu ) ) {
2015-08-14 17:41:18 +10:00
dev_warn ( & ctx - > afu - > dev , " WARNING: Device link down, aborting Process Element Command! \n " ) ;
rc = - EIO ;
goto out ;
}
2016-03-04 12:26:35 +01:00
state = be64_to_cpup ( ctx - > afu - > native - > sw_command_status ) ;
2014-10-08 19:55:02 +11:00
if ( state = = ~ 0ULL ) {
pr_err ( " cxl: Error adding process element to AFU \n " ) ;
2015-01-09 20:34:36 +11:00
rc = - 1 ;
goto out ;
2014-10-08 19:55:02 +11:00
}
if ( ( state & ( CXL_SPA_SW_CMD_MASK | CXL_SPA_SW_STATE_MASK | CXL_SPA_SW_LINK_MASK ) ) = =
( cmd | ( cmd > > 16 ) | ctx - > pe ) )
break ;
/*
* The command won ' t finish in the PSL if there are
* outstanding DSIs . Hence we need to yield here in
* case there are outstanding DSIs that we need to
* service . Tuning possiblity : we could wait for a
* while before sched
*/
schedule ( ) ;
}
2015-01-09 20:34:36 +11:00
out :
trace_cxl_llcmd_done ( ctx , cmd , rc ) ;
return rc ;
2014-10-08 19:55:02 +11:00
}
static int add_process_element ( struct cxl_context * ctx )
{
int rc = 0 ;
2016-03-04 12:26:35 +01:00
mutex_lock ( & ctx - > afu - > native - > spa_mutex ) ;
2014-10-08 19:55:02 +11:00
pr_devel ( " %s Adding pe: %i started \n " , __func__ , ctx - > pe ) ;
if ( ! ( rc = do_process_element_cmd ( ctx , CXL_SPA_SW_CMD_ADD , CXL_PE_SOFTWARE_STATE_V ) ) )
ctx - > pe_inserted = true ;
pr_devel ( " %s Adding pe: %i finished \n " , __func__ , ctx - > pe ) ;
2016-03-04 12:26:35 +01:00
mutex_unlock ( & ctx - > afu - > native - > spa_mutex ) ;
2014-10-08 19:55:02 +11:00
return rc ;
}
static int terminate_process_element ( struct cxl_context * ctx )
{
int rc = 0 ;
/* fast path terminate if it's already invalid */
if ( ! ( ctx - > elem - > software_state & cpu_to_be32 ( CXL_PE_SOFTWARE_STATE_V ) ) )
return rc ;
2016-03-04 12:26:35 +01:00
mutex_lock ( & ctx - > afu - > native - > spa_mutex ) ;
2014-10-08 19:55:02 +11:00
pr_devel ( " %s Terminate pe: %i started \n " , __func__ , ctx - > pe ) ;
2015-08-14 17:41:18 +10:00
/* We could be asked to terminate when the hw is down. That
* should always succeed : it ' s not running if the hw has gone
* away and is being reset .
*/
2016-03-04 12:26:41 +01:00
if ( cxl_ops - > link_ok ( ctx - > afu - > adapter , ctx - > afu ) )
2015-08-14 17:41:18 +10:00
rc = do_process_element_cmd ( ctx , CXL_SPA_SW_CMD_TERMINATE ,
CXL_PE_SOFTWARE_STATE_V | CXL_PE_SOFTWARE_STATE_T ) ;
2014-10-08 19:55:02 +11:00
ctx - > elem - > software_state = 0 ; /* Remove Valid bit */
pr_devel ( " %s Terminate pe: %i finished \n " , __func__ , ctx - > pe ) ;
2016-03-04 12:26:35 +01:00
mutex_unlock ( & ctx - > afu - > native - > spa_mutex ) ;
2014-10-08 19:55:02 +11:00
return rc ;
}
static int remove_process_element ( struct cxl_context * ctx )
{
int rc = 0 ;
2016-03-04 12:26:35 +01:00
mutex_lock ( & ctx - > afu - > native - > spa_mutex ) ;
2014-10-08 19:55:02 +11:00
pr_devel ( " %s Remove pe: %i started \n " , __func__ , ctx - > pe ) ;
2015-08-14 17:41:18 +10:00
/* We could be asked to remove when the hw is down. Again, if
* the hw is down , the PE is gone , so we succeed .
*/
2016-03-04 12:26:41 +01:00
if ( cxl_ops - > link_ok ( ctx - > afu - > adapter , ctx - > afu ) )
2015-08-14 17:41:18 +10:00
rc = do_process_element_cmd ( ctx , CXL_SPA_SW_CMD_REMOVE , 0 ) ;
if ( ! rc )
2014-10-08 19:55:02 +11:00
ctx - > pe_inserted = false ;
slb_invalid ( ctx ) ;
pr_devel ( " %s Remove pe: %i finished \n " , __func__ , ctx - > pe ) ;
2016-03-04 12:26:35 +01:00
mutex_unlock ( & ctx - > afu - > native - > spa_mutex ) ;
2014-10-08 19:55:02 +11:00
return rc ;
}
2015-05-27 16:07:10 +10:00
void cxl_assign_psn_space ( struct cxl_context * ctx )
2014-10-08 19:55:02 +11:00
{
if ( ! ctx - > afu - > pp_size | | ctx - > master ) {
ctx - > psn_phys = ctx - > afu - > psn_phys ;
ctx - > psn_size = ctx - > afu - > adapter - > ps_size ;
} else {
ctx - > psn_phys = ctx - > afu - > psn_phys +
2016-03-04 12:26:35 +01:00
( ctx - > afu - > native - > pp_offset + ctx - > afu - > pp_size * ctx - > pe ) ;
2014-10-08 19:55:02 +11:00
ctx - > psn_size = ctx - > afu - > pp_size ;
}
}
static int activate_afu_directed ( struct cxl_afu * afu )
{
int rc ;
dev_info ( & afu - > dev , " Activating AFU directed mode \n " ) ;
2015-10-07 16:07:40 +11:00
afu - > num_procs = afu - > max_procs_virtualised ;
2016-03-04 12:26:35 +01:00
if ( afu - > native - > spa = = NULL ) {
2015-08-14 17:41:19 +10:00
if ( cxl_alloc_spa ( afu ) )
return - ENOMEM ;
}
attach_spa ( afu ) ;
2014-10-08 19:55:02 +11:00
cxl_p1n_write ( afu , CXL_PSL_SCNTL_An , CXL_PSL_SCNTL_An_PM_AFU ) ;
cxl_p1n_write ( afu , CXL_PSL_AMOR_An , 0xFFFFFFFFFFFFFFFFULL ) ;
cxl_p1n_write ( afu , CXL_PSL_ID_An , CXL_PSL_ID_An_F | CXL_PSL_ID_An_L ) ;
afu - > current_mode = CXL_MODE_DIRECTED ;
if ( ( rc = cxl_chardev_m_afu_add ( afu ) ) )
return rc ;
if ( ( rc = cxl_sysfs_afu_m_add ( afu ) ) )
goto err ;
if ( ( rc = cxl_chardev_s_afu_add ( afu ) ) )
goto err1 ;
return 0 ;
err1 :
cxl_sysfs_afu_m_remove ( afu ) ;
err :
cxl_chardev_afu_remove ( afu ) ;
return rc ;
}
# ifdef CONFIG_CPU_LITTLE_ENDIAN
# define set_endian(sr) ((sr) |= CXL_PSL_SR_An_LE)
# else
# define set_endian(sr) ((sr) &= ~(CXL_PSL_SR_An_LE))
# endif
2015-05-27 16:07:13 +10:00
static u64 calculate_sr ( struct cxl_context * ctx )
{
u64 sr = 0 ;
2015-12-07 14:34:40 +01:00
set_endian ( sr ) ;
2015-05-27 16:07:13 +10:00
if ( ctx - > master )
sr | = CXL_PSL_SR_An_MP ;
if ( mfspr ( SPRN_LPCR ) & LPCR_TC )
sr | = CXL_PSL_SR_An_TC ;
if ( ctx - > kernel ) {
2016-05-06 17:46:36 +10:00
if ( ! ctx - > real_mode )
sr | = CXL_PSL_SR_An_R ;
sr | = ( mfmsr ( ) & MSR_SF ) | CXL_PSL_SR_An_HV ;
2015-05-27 16:07:13 +10:00
} else {
sr | = CXL_PSL_SR_An_PR | CXL_PSL_SR_An_R ;
sr & = ~ ( CXL_PSL_SR_An_HV ) ;
if ( ! test_tsk_thread_flag ( current , TIF_32BIT ) )
sr | = CXL_PSL_SR_An_SF ;
}
return sr ;
}
2014-10-08 19:55:02 +11:00
static int attach_afu_directed ( struct cxl_context * ctx , u64 wed , u64 amr )
{
2015-05-27 16:07:13 +10:00
u32 pid ;
2014-10-08 19:55:02 +11:00
int r , result ;
2015-05-27 16:07:10 +10:00
cxl_assign_psn_space ( ctx ) ;
2014-10-08 19:55:02 +11:00
ctx - > elem - > ctxtime = 0 ; /* disable */
ctx - > elem - > lpid = cpu_to_be32 ( mfspr ( SPRN_LPID ) ) ;
ctx - > elem - > haurp = 0 ; /* disable */
ctx - > elem - > sdr = cpu_to_be64 ( mfspr ( SPRN_SDR1 ) ) ;
2015-05-27 16:07:13 +10:00
pid = current - > pid ;
if ( ctx - > kernel )
pid = 0 ;
2014-10-08 19:55:02 +11:00
ctx - > elem - > common . tid = 0 ;
2015-05-27 16:07:13 +10:00
ctx - > elem - > common . pid = cpu_to_be32 ( pid ) ;
ctx - > elem - > sr = cpu_to_be64 ( calculate_sr ( ctx ) ) ;
2014-10-08 19:55:02 +11:00
ctx - > elem - > common . csrp = 0 ; /* disable */
ctx - > elem - > common . aurp0 = 0 ; /* disable */
ctx - > elem - > common . aurp1 = 0 ; /* disable */
cxl_prefault ( ctx , wed ) ;
ctx - > elem - > common . sstp0 = cpu_to_be64 ( ctx - > sstp0 ) ;
ctx - > elem - > common . sstp1 = cpu_to_be64 ( ctx - > sstp1 ) ;
2016-05-04 14:52:58 +10:00
/*
* Ensure we have the multiplexed PSL interrupt set up to take faults
* for kernel contexts that may not have allocated any AFU IRQs at all :
*/
if ( ctx - > irqs . range [ 0 ] = = 0 ) {
ctx - > irqs . offset [ 0 ] = ctx - > afu - > native - > psl_hwirq ;
ctx - > irqs . range [ 0 ] = 1 ;
}
2014-10-08 19:55:02 +11:00
for ( r = 0 ; r < CXL_IRQ_RANGES ; r + + ) {
ctx - > elem - > ivte_offsets [ r ] = cpu_to_be16 ( ctx - > irqs . offset [ r ] ) ;
ctx - > elem - > ivte_ranges [ r ] = cpu_to_be16 ( ctx - > irqs . range [ r ] ) ;
}
ctx - > elem - > common . amr = cpu_to_be64 ( amr ) ;
ctx - > elem - > common . wed = cpu_to_be64 ( wed ) ;
/* first guy needs to enable */
2016-03-04 12:26:28 +01:00
if ( ( result = cxl_ops - > afu_check_and_enable ( ctx - > afu ) ) )
2014-10-08 19:55:02 +11:00
return result ;
2015-07-29 14:07:22 +10:00
return add_process_element ( ctx ) ;
2014-10-08 19:55:02 +11:00
}
static int deactivate_afu_directed ( struct cxl_afu * afu )
{
dev_info ( & afu - > dev , " Deactivating AFU directed mode \n " ) ;
afu - > current_mode = 0 ;
afu - > num_procs = 0 ;
cxl_sysfs_afu_m_remove ( afu ) ;
cxl_chardev_afu_remove ( afu ) ;
2016-03-04 12:26:28 +01:00
cxl_ops - > afu_reset ( afu ) ;
2014-10-08 19:55:02 +11:00
cxl_afu_disable ( afu ) ;
cxl_psl_purge ( afu ) ;
return 0 ;
}
static int activate_dedicated_process ( struct cxl_afu * afu )
{
dev_info ( & afu - > dev , " Activating dedicated process mode \n " ) ;
cxl_p1n_write ( afu , CXL_PSL_SCNTL_An , CXL_PSL_SCNTL_An_PM_Process ) ;
cxl_p1n_write ( afu , CXL_PSL_CtxTime_An , 0 ) ; /* disable */
cxl_p1n_write ( afu , CXL_PSL_SPAP_An , 0 ) ; /* disable */
cxl_p1n_write ( afu , CXL_PSL_AMOR_An , 0xFFFFFFFFFFFFFFFFULL ) ;
cxl_p1n_write ( afu , CXL_PSL_LPID_An , mfspr ( SPRN_LPID ) ) ;
cxl_p1n_write ( afu , CXL_HAURP_An , 0 ) ; /* disable */
cxl_p1n_write ( afu , CXL_PSL_SDR_An , mfspr ( SPRN_SDR1 ) ) ;
cxl_p2n_write ( afu , CXL_CSRP_An , 0 ) ; /* disable */
cxl_p2n_write ( afu , CXL_AURP0_An , 0 ) ; /* disable */
cxl_p2n_write ( afu , CXL_AURP1_An , 0 ) ; /* disable */
afu - > current_mode = CXL_MODE_DEDICATED ;
afu - > num_procs = 1 ;
return cxl_chardev_d_afu_add ( afu ) ;
}
static int attach_dedicated ( struct cxl_context * ctx , u64 wed , u64 amr )
{
struct cxl_afu * afu = ctx - > afu ;
2015-05-27 16:07:13 +10:00
u64 pid ;
2014-10-08 19:55:02 +11:00
int rc ;
2015-05-27 16:07:13 +10:00
pid = ( u64 ) current - > pid < < 32 ;
if ( ctx - > kernel )
pid = 0 ;
cxl_p2n_write ( afu , CXL_PSL_PID_TID_An , pid ) ;
cxl_p1n_write ( afu , CXL_PSL_SR_An , calculate_sr ( ctx ) ) ;
2014-10-08 19:55:02 +11:00
if ( ( rc = cxl_write_sstp ( afu , ctx - > sstp0 , ctx - > sstp1 ) ) )
return rc ;
cxl_prefault ( ctx , wed ) ;
cxl_p1n_write ( afu , CXL_PSL_IVTE_Offset_An ,
( ( ( u64 ) ctx - > irqs . offset [ 0 ] & 0xffff ) < < 48 ) |
( ( ( u64 ) ctx - > irqs . offset [ 1 ] & 0xffff ) < < 32 ) |
( ( ( u64 ) ctx - > irqs . offset [ 2 ] & 0xffff ) < < 16 ) |
( ( u64 ) ctx - > irqs . offset [ 3 ] & 0xffff ) ) ;
cxl_p1n_write ( afu , CXL_PSL_IVTE_Limit_An , ( u64 )
( ( ( u64 ) ctx - > irqs . range [ 0 ] & 0xffff ) < < 48 ) |
( ( ( u64 ) ctx - > irqs . range [ 1 ] & 0xffff ) < < 32 ) |
( ( ( u64 ) ctx - > irqs . range [ 2 ] & 0xffff ) < < 16 ) |
( ( u64 ) ctx - > irqs . range [ 3 ] & 0xffff ) ) ;
cxl_p2n_write ( afu , CXL_PSL_AMR_An , amr ) ;
/* master only context for dedicated */
2015-05-27 16:07:10 +10:00
cxl_assign_psn_space ( ctx ) ;
2014-10-08 19:55:02 +11:00
2016-03-04 12:26:28 +01:00
if ( ( rc = cxl_ops - > afu_reset ( afu ) ) )
2014-10-08 19:55:02 +11:00
return rc ;
cxl_p2n_write ( afu , CXL_PSL_WED_An , wed ) ;
return afu_enable ( afu ) ;
}
static int deactivate_dedicated_process ( struct cxl_afu * afu )
{
dev_info ( & afu - > dev , " Deactivating dedicated process mode \n " ) ;
afu - > current_mode = 0 ;
afu - > num_procs = 0 ;
cxl_chardev_afu_remove ( afu ) ;
return 0 ;
}
2016-03-04 12:26:29 +01:00
static int native_afu_deactivate_mode ( struct cxl_afu * afu , int mode )
2014-10-08 19:55:02 +11:00
{
if ( mode = = CXL_MODE_DIRECTED )
return deactivate_afu_directed ( afu ) ;
if ( mode = = CXL_MODE_DEDICATED )
return deactivate_dedicated_process ( afu ) ;
return 0 ;
}
2016-03-04 12:26:29 +01:00
static int native_afu_activate_mode ( struct cxl_afu * afu , int mode )
2014-10-08 19:55:02 +11:00
{
if ( ! mode )
return 0 ;
if ( ! ( mode & afu - > modes_supported ) )
return - EINVAL ;
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) {
2015-08-14 17:41:18 +10:00
WARN ( 1 , " Device link is down, refusing to activate! \n " ) ;
return - EIO ;
}
2014-10-08 19:55:02 +11:00
if ( mode = = CXL_MODE_DIRECTED )
return activate_afu_directed ( afu ) ;
if ( mode = = CXL_MODE_DEDICATED )
return activate_dedicated_process ( afu ) ;
return - EINVAL ;
}
2016-03-04 12:26:29 +01:00
static int native_attach_process ( struct cxl_context * ctx , bool kernel ,
u64 wed , u64 amr )
2014-10-08 19:55:02 +11:00
{
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( ctx - > afu - > adapter , ctx - > afu ) ) {
2015-08-14 17:41:18 +10:00
WARN ( 1 , " Device link is down, refusing to attach process! \n " ) ;
return - EIO ;
}
2014-10-08 19:55:02 +11:00
ctx - > kernel = kernel ;
if ( ctx - > afu - > current_mode = = CXL_MODE_DIRECTED )
return attach_afu_directed ( ctx , wed , amr ) ;
if ( ctx - > afu - > current_mode = = CXL_MODE_DEDICATED )
return attach_dedicated ( ctx , wed , amr ) ;
return - EINVAL ;
}
static inline int detach_process_native_dedicated ( struct cxl_context * ctx )
{
2016-03-04 12:26:28 +01:00
cxl_ops - > afu_reset ( ctx - > afu ) ;
2014-10-08 19:55:02 +11:00
cxl_afu_disable ( ctx - > afu ) ;
cxl_psl_purge ( ctx - > afu ) ;
return 0 ;
}
static inline int detach_process_native_afu_directed ( struct cxl_context * ctx )
{
if ( ! ctx - > pe_inserted )
return 0 ;
if ( terminate_process_element ( ctx ) )
return - 1 ;
if ( remove_process_element ( ctx ) )
return - 1 ;
return 0 ;
}
2016-03-04 12:26:29 +01:00
static int native_detach_process ( struct cxl_context * ctx )
2014-10-08 19:55:02 +11:00
{
2015-01-09 20:34:36 +11:00
trace_cxl_detach ( ctx ) ;
2014-10-08 19:55:02 +11:00
if ( ctx - > afu - > current_mode = = CXL_MODE_DEDICATED )
return detach_process_native_dedicated ( ctx ) ;
return detach_process_native_afu_directed ( ctx ) ;
}
2016-03-04 12:26:29 +01:00
static int native_get_irq_info ( struct cxl_afu * afu , struct cxl_irq_info * info )
2014-10-08 19:55:02 +11:00
{
u64 pidtid ;
2015-08-14 17:41:18 +10:00
/* If the adapter has gone away, we can't get any meaningful
* information .
*/
2016-03-04 12:26:41 +01:00
if ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) )
2015-08-14 17:41:18 +10:00
return - EIO ;
2014-11-14 17:37:50 +11:00
info - > dsisr = cxl_p2n_read ( afu , CXL_PSL_DSISR_An ) ;
info - > dar = cxl_p2n_read ( afu , CXL_PSL_DAR_An ) ;
info - > dsr = cxl_p2n_read ( afu , CXL_PSL_DSR_An ) ;
pidtid = cxl_p2n_read ( afu , CXL_PSL_PID_TID_An ) ;
2014-10-08 19:55:02 +11:00
info - > pid = pidtid > > 32 ;
info - > tid = pidtid & 0xffffffff ;
2014-11-14 17:37:50 +11:00
info - > afu_err = cxl_p2n_read ( afu , CXL_AFU_ERR_An ) ;
info - > errstat = cxl_p2n_read ( afu , CXL_PSL_ErrStat_An ) ;
2016-03-04 12:26:34 +01:00
info - > proc_handle = 0 ;
2014-10-08 19:55:02 +11:00
return 0 ;
}
2016-03-04 12:26:29 +01:00
static irqreturn_t native_handle_psl_slice_error ( struct cxl_context * ctx ,
u64 dsisr , u64 errstat )
2016-03-04 12:26:26 +01:00
{
u64 fir1 , fir2 , fir_slice , serr , afu_debug ;
fir1 = cxl_p1_read ( ctx - > afu - > adapter , CXL_PSL_FIR1 ) ;
fir2 = cxl_p1_read ( ctx - > afu - > adapter , CXL_PSL_FIR2 ) ;
fir_slice = cxl_p1n_read ( ctx - > afu , CXL_PSL_FIR_SLICE_An ) ;
serr = cxl_p1n_read ( ctx - > afu , CXL_PSL_SERR_An ) ;
afu_debug = cxl_p1n_read ( ctx - > afu , CXL_AFU_DEBUG_An ) ;
dev_crit ( & ctx - > afu - > dev , " PSL ERROR STATUS: 0x%016llx \n " , errstat ) ;
dev_crit ( & ctx - > afu - > dev , " PSL_FIR1: 0x%016llx \n " , fir1 ) ;
dev_crit ( & ctx - > afu - > dev , " PSL_FIR2: 0x%016llx \n " , fir2 ) ;
dev_crit ( & ctx - > afu - > dev , " PSL_SERR_An: 0x%016llx \n " , serr ) ;
dev_crit ( & ctx - > afu - > dev , " PSL_FIR_SLICE_An: 0x%016llx \n " , fir_slice ) ;
dev_crit ( & ctx - > afu - > dev , " CXL_PSL_AFU_DEBUG_An: 0x%016llx \n " , afu_debug ) ;
dev_crit ( & ctx - > afu - > dev , " STOPPING CXL TRACE \n " ) ;
cxl_stop_trace ( ctx - > afu - > adapter ) ;
2016-03-04 12:26:28 +01:00
return cxl_ops - > ack_irq ( ctx , 0 , errstat ) ;
2016-03-04 12:26:26 +01:00
}
static irqreturn_t fail_psl_irq ( struct cxl_afu * afu , struct cxl_irq_info * irq_info )
{
if ( irq_info - > dsisr & CXL_PSL_DSISR_TRANS )
cxl_p2n_write ( afu , CXL_PSL_TFC_An , CXL_PSL_TFC_An_AE ) ;
else
cxl_p2n_write ( afu , CXL_PSL_TFC_An , CXL_PSL_TFC_An_A ) ;
return IRQ_HANDLED ;
}
2016-03-04 12:26:29 +01:00
static irqreturn_t native_irq_multiplexed ( int irq , void * data )
2016-03-04 12:26:26 +01:00
{
struct cxl_afu * afu = data ;
struct cxl_context * ctx ;
struct cxl_irq_info irq_info ;
int ph = cxl_p2n_read ( afu , CXL_PSL_PEHandle_An ) & 0xffff ;
int ret ;
2016-03-04 12:26:29 +01:00
if ( ( ret = native_get_irq_info ( afu , & irq_info ) ) ) {
2016-03-04 12:26:26 +01:00
WARN ( 1 , " Unable to get CXL IRQ Info: %i \n " , ret ) ;
return fail_psl_irq ( afu , & irq_info ) ;
}
rcu_read_lock ( ) ;
ctx = idr_find ( & afu - > contexts_idr , ph ) ;
if ( ctx ) {
ret = cxl_irq ( irq , ctx , & irq_info ) ;
rcu_read_unlock ( ) ;
return ret ;
}
rcu_read_unlock ( ) ;
WARN ( 1 , " Unable to demultiplex CXL PSL IRQ for PE %i DSISR %016llx DAR "
" %016llx \n (Possible AFU HW issue - was a term/remove acked "
" with outstanding transactions?) \n " , ph , irq_info . dsisr ,
irq_info . dar ) ;
return fail_psl_irq ( afu , & irq_info ) ;
}
2016-04-22 14:57:49 +10:00
void native_irq_wait ( struct cxl_context * ctx )
{
u64 dsisr ;
int timeout = 1000 ;
int ph ;
/*
* Wait until no further interrupts are presented by the PSL
* for this context .
*/
while ( timeout - - ) {
ph = cxl_p2n_read ( ctx - > afu , CXL_PSL_PEHandle_An ) & 0xffff ;
if ( ph ! = ctx - > pe )
return ;
dsisr = cxl_p2n_read ( ctx - > afu , CXL_PSL_DSISR_An ) ;
if ( ( dsisr & CXL_PSL_DSISR_PENDING ) = = 0 )
return ;
/*
* We are waiting for the workqueue to process our
* irq , so need to let that run here .
*/
msleep ( 1 ) ;
}
dev_warn ( & ctx - > afu - > dev , " WARNING: waiting on DSI for PE %i "
" DSISR %016llx! \n " , ph , dsisr ) ;
return ;
}
2016-03-04 12:26:29 +01:00
static irqreturn_t native_slice_irq_err ( int irq , void * data )
2016-03-04 12:26:26 +01:00
{
struct cxl_afu * afu = data ;
u64 fir_slice , errstat , serr , afu_debug ;
WARN ( irq , " CXL SLICE ERROR interrupt %i \n " , irq ) ;
serr = cxl_p1n_read ( afu , CXL_PSL_SERR_An ) ;
fir_slice = cxl_p1n_read ( afu , CXL_PSL_FIR_SLICE_An ) ;
errstat = cxl_p2n_read ( afu , CXL_PSL_ErrStat_An ) ;
afu_debug = cxl_p1n_read ( afu , CXL_AFU_DEBUG_An ) ;
dev_crit ( & afu - > dev , " PSL_SERR_An: 0x%016llx \n " , serr ) ;
dev_crit ( & afu - > dev , " PSL_FIR_SLICE_An: 0x%016llx \n " , fir_slice ) ;
dev_crit ( & afu - > dev , " CXL_PSL_ErrStat_An: 0x%016llx \n " , errstat ) ;
dev_crit ( & afu - > dev , " CXL_PSL_AFU_DEBUG_An: 0x%016llx \n " , afu_debug ) ;
cxl_p1n_write ( afu , CXL_PSL_SERR_An , serr ) ;
return IRQ_HANDLED ;
}
2016-03-04 12:26:29 +01:00
static irqreturn_t native_irq_err ( int irq , void * data )
2016-03-04 12:26:26 +01:00
{
struct cxl * adapter = data ;
u64 fir1 , fir2 , err_ivte ;
WARN ( 1 , " CXL ERROR interrupt %i \n " , irq ) ;
err_ivte = cxl_p1_read ( adapter , CXL_PSL_ErrIVTE ) ;
dev_crit ( & adapter - > dev , " PSL_ErrIVTE: 0x%016llx \n " , err_ivte ) ;
dev_crit ( & adapter - > dev , " STOPPING CXL TRACE \n " ) ;
cxl_stop_trace ( adapter ) ;
fir1 = cxl_p1_read ( adapter , CXL_PSL_FIR1 ) ;
fir2 = cxl_p1_read ( adapter , CXL_PSL_FIR2 ) ;
dev_crit ( & adapter - > dev , " PSL_FIR1: 0x%016llx \n PSL_FIR2: 0x%016llx \n " , fir1 , fir2 ) ;
return IRQ_HANDLED ;
}
2016-03-04 12:26:29 +01:00
int cxl_native_register_psl_err_irq ( struct cxl * adapter )
2016-03-04 12:26:26 +01:00
{
int rc ;
adapter - > irq_name = kasprintf ( GFP_KERNEL , " cxl-%s-err " ,
dev_name ( & adapter - > dev ) ) ;
if ( ! adapter - > irq_name )
return - ENOMEM ;
2016-03-04 12:26:29 +01:00
if ( ( rc = cxl_register_one_irq ( adapter , native_irq_err , adapter ,
2016-03-04 12:26:35 +01:00
& adapter - > native - > err_hwirq ,
& adapter - > native - > err_virq ,
2016-03-04 12:26:26 +01:00
adapter - > irq_name ) ) ) {
kfree ( adapter - > irq_name ) ;
adapter - > irq_name = NULL ;
return rc ;
}
2016-03-04 12:26:35 +01:00
cxl_p1_write ( adapter , CXL_PSL_ErrIVTE , adapter - > native - > err_hwirq & 0xffff ) ;
2016-03-04 12:26:26 +01:00
return 0 ;
}
2016-03-04 12:26:29 +01:00
void cxl_native_release_psl_err_irq ( struct cxl * adapter )
2016-03-04 12:26:26 +01:00
{
2016-03-04 12:26:35 +01:00
if ( adapter - > native - > err_virq ! = irq_find_mapping ( NULL , adapter - > native - > err_hwirq ) )
2016-03-04 12:26:26 +01:00
return ;
cxl_p1_write ( adapter , CXL_PSL_ErrIVTE , 0x0000000000000000 ) ;
2016-03-04 12:26:35 +01:00
cxl_unmap_irq ( adapter - > native - > err_virq , adapter ) ;
cxl_ops - > release_one_irq ( adapter , adapter - > native - > err_hwirq ) ;
2016-03-04 12:26:26 +01:00
kfree ( adapter - > irq_name ) ;
}
2016-03-04 12:26:29 +01:00
int cxl_native_register_serr_irq ( struct cxl_afu * afu )
2016-03-04 12:26:26 +01:00
{
u64 serr ;
int rc ;
afu - > err_irq_name = kasprintf ( GFP_KERNEL , " cxl-%s-err " ,
dev_name ( & afu - > dev ) ) ;
if ( ! afu - > err_irq_name )
return - ENOMEM ;
2016-03-04 12:26:29 +01:00
if ( ( rc = cxl_register_one_irq ( afu - > adapter , native_slice_irq_err , afu ,
2016-03-04 12:26:26 +01:00
& afu - > serr_hwirq ,
& afu - > serr_virq , afu - > err_irq_name ) ) ) {
kfree ( afu - > err_irq_name ) ;
afu - > err_irq_name = NULL ;
return rc ;
}
serr = cxl_p1n_read ( afu , CXL_PSL_SERR_An ) ;
serr = ( serr & 0x00ffffffffff0000ULL ) | ( afu - > serr_hwirq & 0xffff ) ;
cxl_p1n_write ( afu , CXL_PSL_SERR_An , serr ) ;
return 0 ;
}
2016-03-04 12:26:29 +01:00
void cxl_native_release_serr_irq ( struct cxl_afu * afu )
2016-03-04 12:26:26 +01:00
{
if ( afu - > serr_virq ! = irq_find_mapping ( NULL , afu - > serr_hwirq ) )
return ;
cxl_p1n_write ( afu , CXL_PSL_SERR_An , 0x0000000000000000 ) ;
cxl_unmap_irq ( afu - > serr_virq , afu ) ;
2016-03-04 12:26:28 +01:00
cxl_ops - > release_one_irq ( afu - > adapter , afu - > serr_hwirq ) ;
2016-03-04 12:26:26 +01:00
kfree ( afu - > err_irq_name ) ;
}
2016-03-04 12:26:29 +01:00
int cxl_native_register_psl_irq ( struct cxl_afu * afu )
2016-03-04 12:26:26 +01:00
{
int rc ;
afu - > psl_irq_name = kasprintf ( GFP_KERNEL , " cxl-%s " ,
dev_name ( & afu - > dev ) ) ;
if ( ! afu - > psl_irq_name )
return - ENOMEM ;
2016-03-04 12:26:35 +01:00
if ( ( rc = cxl_register_one_irq ( afu - > adapter , native_irq_multiplexed ,
afu , & afu - > native - > psl_hwirq , & afu - > native - > psl_virq ,
2016-03-04 12:26:26 +01:00
afu - > psl_irq_name ) ) ) {
kfree ( afu - > psl_irq_name ) ;
afu - > psl_irq_name = NULL ;
}
return rc ;
}
2016-03-04 12:26:29 +01:00
void cxl_native_release_psl_irq ( struct cxl_afu * afu )
2016-03-04 12:26:26 +01:00
{
2016-03-04 12:26:35 +01:00
if ( afu - > native - > psl_virq ! = irq_find_mapping ( NULL , afu - > native - > psl_hwirq ) )
2016-03-04 12:26:26 +01:00
return ;
2016-03-04 12:26:35 +01:00
cxl_unmap_irq ( afu - > native - > psl_virq , afu ) ;
cxl_ops - > release_one_irq ( afu - > adapter , afu - > native - > psl_hwirq ) ;
2016-03-04 12:26:26 +01:00
kfree ( afu - > psl_irq_name ) ;
}
2014-10-08 19:55:02 +11:00
static void recover_psl_err ( struct cxl_afu * afu , u64 errstat )
{
u64 dsisr ;
2015-06-11 13:27:52 +02:00
pr_devel ( " RECOVERING FROM PSL ERROR... (0x%016llx) \n " , errstat ) ;
2014-10-08 19:55:02 +11:00
/* Clear PSL_DSISR[PE] */
dsisr = cxl_p2n_read ( afu , CXL_PSL_DSISR_An ) ;
cxl_p2n_write ( afu , CXL_PSL_DSISR_An , dsisr & ~ CXL_PSL_DSISR_An_PE ) ;
/* Write 1s to clear error status bits */
cxl_p2n_write ( afu , CXL_PSL_ErrStat_An , errstat ) ;
}
2016-03-04 12:26:29 +01:00
static int native_ack_irq ( struct cxl_context * ctx , u64 tfc , u64 psl_reset_mask )
2014-10-08 19:55:02 +11:00
{
2015-01-09 20:34:36 +11:00
trace_cxl_psl_irq_ack ( ctx , tfc ) ;
2014-10-08 19:55:02 +11:00
if ( tfc )
cxl_p2n_write ( ctx - > afu , CXL_PSL_TFC_An , tfc ) ;
if ( psl_reset_mask )
recover_psl_err ( ctx - > afu , psl_reset_mask ) ;
return 0 ;
}
int cxl_check_error ( struct cxl_afu * afu )
{
return ( cxl_p1n_read ( afu , CXL_PSL_SCNTL_An ) = = ~ 0ULL ) ;
}
2016-03-04 12:26:26 +01:00
2016-03-04 12:26:37 +01:00
static bool native_support_attributes ( const char * attr_name ,
enum cxl_attrs type )
{
return true ;
}
2016-03-04 12:26:29 +01:00
static int native_afu_cr_read64 ( struct cxl_afu * afu , int cr , u64 off , u64 * out )
2016-03-04 12:26:26 +01:00
{
2016-03-04 12:26:41 +01:00
if ( unlikely ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) )
2016-03-04 12:26:28 +01:00
return - EIO ;
if ( unlikely ( off > = afu - > crs_len ) )
return - ERANGE ;
2016-03-04 12:26:35 +01:00
* out = in_le64 ( afu - > native - > afu_desc_mmio + afu - > crs_offset +
2016-03-04 12:26:28 +01:00
( cr * afu - > crs_len ) + off ) ;
return 0 ;
2016-03-04 12:26:26 +01:00
}
2016-03-04 12:26:29 +01:00
static int native_afu_cr_read32 ( struct cxl_afu * afu , int cr , u64 off , u32 * out )
2016-03-04 12:26:26 +01:00
{
2016-03-04 12:26:41 +01:00
if ( unlikely ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) )
2016-03-04 12:26:28 +01:00
return - EIO ;
if ( unlikely ( off > = afu - > crs_len ) )
return - ERANGE ;
2016-03-04 12:26:35 +01:00
* out = in_le32 ( afu - > native - > afu_desc_mmio + afu - > crs_offset +
2016-03-04 12:26:28 +01:00
( cr * afu - > crs_len ) + off ) ;
return 0 ;
2016-03-04 12:26:26 +01:00
}
2016-03-04 12:26:29 +01:00
static int native_afu_cr_read16 ( struct cxl_afu * afu , int cr , u64 off , u16 * out )
2016-03-04 12:26:26 +01:00
{
u64 aligned_off = off & ~ 0x3L ;
u32 val ;
2016-03-04 12:26:28 +01:00
int rc ;
2016-03-04 12:26:26 +01:00
2016-03-04 12:26:29 +01:00
rc = native_afu_cr_read32 ( afu , cr , aligned_off , & val ) ;
2016-03-04 12:26:28 +01:00
if ( ! rc )
* out = ( val > > ( ( off & 0x3 ) * 8 ) ) & 0xffff ;
return rc ;
2016-03-04 12:26:26 +01:00
}
2016-03-04 12:26:29 +01:00
static int native_afu_cr_read8 ( struct cxl_afu * afu , int cr , u64 off , u8 * out )
2016-03-04 12:26:26 +01:00
{
u64 aligned_off = off & ~ 0x3L ;
u32 val ;
2016-03-04 12:26:28 +01:00
int rc ;
2016-03-04 12:26:26 +01:00
2016-03-04 12:26:29 +01:00
rc = native_afu_cr_read32 ( afu , cr , aligned_off , & val ) ;
2016-03-04 12:26:28 +01:00
if ( ! rc )
* out = ( val > > ( ( off & 0x3 ) * 8 ) ) & 0xff ;
return rc ;
2016-03-04 12:26:26 +01:00
}
2016-03-04 12:26:28 +01:00
2016-03-04 12:26:40 +01:00
static int native_afu_cr_write32 ( struct cxl_afu * afu , int cr , u64 off , u32 in )
{
2016-03-04 12:26:41 +01:00
if ( unlikely ( ! cxl_ops - > link_ok ( afu - > adapter , afu ) ) )
2016-03-04 12:26:40 +01:00
return - EIO ;
if ( unlikely ( off > = afu - > crs_len ) )
return - ERANGE ;
out_le32 ( afu - > native - > afu_desc_mmio + afu - > crs_offset +
( cr * afu - > crs_len ) + off , in ) ;
return 0 ;
}
static int native_afu_cr_write16 ( struct cxl_afu * afu , int cr , u64 off , u16 in )
{
u64 aligned_off = off & ~ 0x3L ;
u32 val32 , mask , shift ;
int rc ;
rc = native_afu_cr_read32 ( afu , cr , aligned_off , & val32 ) ;
if ( rc )
return rc ;
shift = ( off & 0x3 ) * 8 ;
WARN_ON ( shift = = 24 ) ;
mask = 0xffff < < shift ;
val32 = ( val32 & ~ mask ) | ( in < < shift ) ;
rc = native_afu_cr_write32 ( afu , cr , aligned_off , val32 ) ;
return rc ;
}
static int native_afu_cr_write8 ( struct cxl_afu * afu , int cr , u64 off , u8 in )
{
u64 aligned_off = off & ~ 0x3L ;
u32 val32 , mask , shift ;
int rc ;
rc = native_afu_cr_read32 ( afu , cr , aligned_off , & val32 ) ;
if ( rc )
return rc ;
shift = ( off & 0x3 ) * 8 ;
mask = 0xff < < shift ;
val32 = ( val32 & ~ mask ) | ( in < < shift ) ;
rc = native_afu_cr_write32 ( afu , cr , aligned_off , val32 ) ;
return rc ;
}
2016-03-04 12:26:28 +01:00
const struct cxl_backend_ops cxl_native_ops = {
. module = THIS_MODULE ,
2016-03-04 12:26:29 +01:00
. adapter_reset = cxl_pci_reset ,
. alloc_one_irq = cxl_pci_alloc_one_irq ,
. release_one_irq = cxl_pci_release_one_irq ,
. alloc_irq_ranges = cxl_pci_alloc_irq_ranges ,
. release_irq_ranges = cxl_pci_release_irq_ranges ,
. setup_irq = cxl_pci_setup_irq ,
. handle_psl_slice_error = native_handle_psl_slice_error ,
2016-03-04 12:26:28 +01:00
. psl_interrupt = NULL ,
2016-03-04 12:26:29 +01:00
. ack_irq = native_ack_irq ,
2016-04-22 14:57:49 +10:00
. irq_wait = native_irq_wait ,
2016-03-04 12:26:29 +01:00
. attach_process = native_attach_process ,
. detach_process = native_detach_process ,
2016-03-04 12:26:37 +01:00
. support_attributes = native_support_attributes ,
2016-03-04 12:26:28 +01:00
. link_ok = cxl_adapter_link_ok ,
2016-03-04 12:26:29 +01:00
. release_afu = cxl_pci_release_afu ,
. afu_read_err_buffer = cxl_pci_afu_read_err_buffer ,
. afu_check_and_enable = native_afu_check_and_enable ,
. afu_activate_mode = native_afu_activate_mode ,
. afu_deactivate_mode = native_afu_deactivate_mode ,
. afu_reset = native_afu_reset ,
. afu_cr_read8 = native_afu_cr_read8 ,
. afu_cr_read16 = native_afu_cr_read16 ,
. afu_cr_read32 = native_afu_cr_read32 ,
. afu_cr_read64 = native_afu_cr_read64 ,
2016-03-04 12:26:40 +01:00
. afu_cr_write8 = native_afu_cr_write8 ,
. afu_cr_write16 = native_afu_cr_write16 ,
. afu_cr_write32 = native_afu_cr_write32 ,
. read_adapter_vpd = cxl_pci_read_adapter_vpd ,
2016-03-04 12:26:28 +01:00
} ;