2019-05-30 02:57:55 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2017-03-27 12:45:14 +03:00
/**
* Host side test driver to test endpoint functionality
*
* Copyright ( C ) 2017 Texas Instruments
* Author : Kishon Vijay Abraham I < kishon @ ti . com >
*/
# include <linux/crc32.h>
# include <linux/delay.h>
# include <linux/fs.h>
# include <linux/io.h>
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/miscdevice.h>
# include <linux/module.h>
# include <linux/mutex.h>
# include <linux/random.h>
# include <linux/slab.h>
2020-03-16 14:24:24 +03:00
# include <linux/uaccess.h>
2017-03-27 12:45:14 +03:00
# include <linux/pci.h>
# include <linux/pci_ids.h>
# include <linux/pci_regs.h>
# include <uapi/linux/pcitest.h>
2018-07-19 11:32:17 +03:00
# define DRV_MODULE_NAME "pci-endpoint-test"
2018-07-19 11:32:20 +03:00
# define IRQ_TYPE_UNDEFINED -1
2018-07-19 11:32:17 +03:00
# define IRQ_TYPE_LEGACY 0
# define IRQ_TYPE_MSI 1
2018-07-19 11:32:19 +03:00
# define IRQ_TYPE_MSIX 2
2018-07-19 11:32:17 +03:00
# define PCI_ENDPOINT_TEST_MAGIC 0x0
# define PCI_ENDPOINT_TEST_COMMAND 0x4
# define COMMAND_RAISE_LEGACY_IRQ BIT(0)
# define COMMAND_RAISE_MSI_IRQ BIT(1)
2018-07-19 11:32:19 +03:00
# define COMMAND_RAISE_MSIX_IRQ BIT(2)
2018-07-19 11:32:17 +03:00
# define COMMAND_READ BIT(3)
# define COMMAND_WRITE BIT(4)
# define COMMAND_COPY BIT(5)
# define PCI_ENDPOINT_TEST_STATUS 0x8
# define STATUS_READ_SUCCESS BIT(0)
# define STATUS_READ_FAIL BIT(1)
# define STATUS_WRITE_SUCCESS BIT(2)
# define STATUS_WRITE_FAIL BIT(3)
# define STATUS_COPY_SUCCESS BIT(4)
# define STATUS_COPY_FAIL BIT(5)
# define STATUS_IRQ_RAISED BIT(6)
# define STATUS_SRC_ADDR_INVALID BIT(7)
# define STATUS_DST_ADDR_INVALID BIT(8)
# define PCI_ENDPOINT_TEST_LOWER_SRC_ADDR 0x0c
2017-03-27 12:45:14 +03:00
# define PCI_ENDPOINT_TEST_UPPER_SRC_ADDR 0x10
# define PCI_ENDPOINT_TEST_LOWER_DST_ADDR 0x14
# define PCI_ENDPOINT_TEST_UPPER_DST_ADDR 0x18
2018-07-19 11:32:17 +03:00
# define PCI_ENDPOINT_TEST_SIZE 0x1c
# define PCI_ENDPOINT_TEST_CHECKSUM 0x20
# define PCI_ENDPOINT_TEST_IRQ_TYPE 0x24
# define PCI_ENDPOINT_TEST_IRQ_NUMBER 0x28
2017-03-27 12:45:14 +03:00
2020-03-16 14:24:24 +03:00
# define PCI_ENDPOINT_TEST_FLAGS 0x2c
# define FLAG_USE_DMA BIT(0)
2020-07-22 14:03:16 +03:00
# define PCI_DEVICE_ID_TI_J721E 0xb00d
2019-03-25 12:39:46 +03:00
# define PCI_DEVICE_ID_TI_AM654 0xb00c
2020-09-18 11:00:23 +03:00
# define PCI_DEVICE_ID_LS1088A 0x80c0
2019-03-25 12:39:46 +03:00
# define is_am654_pci_dev(pdev) \
( ( pdev ) - > device = = PCI_DEVICE_ID_TI_AM654 )
2020-08-14 20:30:34 +03:00
# define PCI_DEVICE_ID_RENESAS_R8A774A1 0x0028
# define PCI_DEVICE_ID_RENESAS_R8A774B1 0x002b
2020-05-15 01:03:29 +03:00
# define PCI_DEVICE_ID_RENESAS_R8A774C0 0x002d
2020-09-04 13:38:51 +03:00
# define PCI_DEVICE_ID_RENESAS_R8A774E1 0x0025
2020-05-15 01:03:29 +03:00
2017-03-27 12:45:14 +03:00
static DEFINE_IDA ( pci_endpoint_test_ida ) ;
# define to_endpoint_test(priv) container_of((priv), struct pci_endpoint_test, \
miscdev )
2017-08-18 17:58:09 +03:00
static bool no_msi ;
module_param ( no_msi , bool , 0444 ) ;
MODULE_PARM_DESC ( no_msi , " Disable MSI interrupt in pci_endpoint_test " ) ;
2018-07-19 11:32:18 +03:00
static int irq_type = IRQ_TYPE_MSI ;
module_param ( irq_type , int , 0444 ) ;
2018-07-19 11:32:19 +03:00
MODULE_PARM_DESC ( irq_type , " IRQ mode selection in pci_endpoint_test (0 - Legacy, 1 - MSI, 2 - MSI-X) " ) ;
2018-07-19 11:32:18 +03:00
2017-03-27 12:45:14 +03:00
enum pci_barno {
BAR_0 ,
BAR_1 ,
BAR_2 ,
BAR_3 ,
BAR_4 ,
BAR_5 ,
} ;
struct pci_endpoint_test {
struct pci_dev * pdev ;
void __iomem * base ;
2019-09-28 02:43:08 +03:00
void __iomem * bar [ PCI_STD_NUM_BARS ] ;
2017-03-27 12:45:14 +03:00
struct completion irq_raised ;
int last_irq ;
2017-10-11 11:44:38 +03:00
int num_irqs ;
2020-03-17 13:01:54 +03:00
int irq_type ;
2017-03-27 12:45:14 +03:00
/* mutex to protect the ioctls */
struct mutex mutex ;
struct miscdevice miscdev ;
2017-08-18 17:58:05 +03:00
enum pci_barno test_reg_bar ;
2017-08-18 17:58:06 +03:00
size_t alignment ;
2020-03-17 13:01:58 +03:00
const char * name ;
2017-03-27 12:45:14 +03:00
} ;
2017-08-18 17:58:05 +03:00
struct pci_endpoint_test_data {
enum pci_barno test_reg_bar ;
2017-08-18 17:58:06 +03:00
size_t alignment ;
2018-07-19 11:32:18 +03:00
int irq_type ;
2017-08-18 17:58:05 +03:00
} ;
2017-03-27 12:45:14 +03:00
static inline u32 pci_endpoint_test_readl ( struct pci_endpoint_test * test ,
u32 offset )
{
return readl ( test - > base + offset ) ;
}
static inline void pci_endpoint_test_writel ( struct pci_endpoint_test * test ,
u32 offset , u32 value )
{
writel ( value , test - > base + offset ) ;
}
static inline u32 pci_endpoint_test_bar_readl ( struct pci_endpoint_test * test ,
int bar , int offset )
{
return readl ( test - > bar [ bar ] + offset ) ;
}
static inline void pci_endpoint_test_bar_writel ( struct pci_endpoint_test * test ,
int bar , u32 offset , u32 value )
{
writel ( value , test - > bar [ bar ] + offset ) ;
}
static irqreturn_t pci_endpoint_test_irqhandler ( int irq , void * dev_id )
{
struct pci_endpoint_test * test = dev_id ;
u32 reg ;
reg = pci_endpoint_test_readl ( test , PCI_ENDPOINT_TEST_STATUS ) ;
if ( reg & STATUS_IRQ_RAISED ) {
test - > last_irq = irq ;
complete ( & test - > irq_raised ) ;
reg & = ~ STATUS_IRQ_RAISED ;
}
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_STATUS ,
reg ) ;
return IRQ_HANDLED ;
}
2018-07-19 11:32:20 +03:00
static void pci_endpoint_test_free_irq_vectors ( struct pci_endpoint_test * test )
{
struct pci_dev * pdev = test - > pdev ;
pci_free_irq_vectors ( pdev ) ;
2020-03-17 13:01:54 +03:00
test - > irq_type = IRQ_TYPE_UNDEFINED ;
2018-07-19 11:32:20 +03:00
}
static bool pci_endpoint_test_alloc_irq_vectors ( struct pci_endpoint_test * test ,
int type )
{
int irq = - 1 ;
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
bool res = true ;
switch ( type ) {
case IRQ_TYPE_LEGACY :
irq = pci_alloc_irq_vectors ( pdev , 1 , 1 , PCI_IRQ_LEGACY ) ;
if ( irq < 0 )
dev_err ( dev , " Failed to get Legacy interrupt \n " ) ;
break ;
case IRQ_TYPE_MSI :
irq = pci_alloc_irq_vectors ( pdev , 1 , 32 , PCI_IRQ_MSI ) ;
if ( irq < 0 )
dev_err ( dev , " Failed to get MSI interrupts \n " ) ;
break ;
case IRQ_TYPE_MSIX :
irq = pci_alloc_irq_vectors ( pdev , 1 , 2048 , PCI_IRQ_MSIX ) ;
if ( irq < 0 )
dev_err ( dev , " Failed to get MSI-X interrupts \n " ) ;
break ;
default :
dev_err ( dev , " Invalid IRQ type selected \n " ) ;
}
if ( irq < 0 ) {
irq = 0 ;
res = false ;
}
2020-03-17 13:01:54 +03:00
test - > irq_type = type ;
2018-07-19 11:32:20 +03:00
test - > num_irqs = irq ;
return res ;
}
static void pci_endpoint_test_release_irq ( struct pci_endpoint_test * test )
{
int i ;
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
for ( i = 0 ; i < test - > num_irqs ; i + + )
devm_free_irq ( dev , pci_irq_vector ( pdev , i ) , test ) ;
test - > num_irqs = 0 ;
}
static bool pci_endpoint_test_request_irq ( struct pci_endpoint_test * test )
{
int i ;
int err ;
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
for ( i = 0 ; i < test - > num_irqs ; i + + ) {
err = devm_request_irq ( dev , pci_irq_vector ( pdev , i ) ,
pci_endpoint_test_irqhandler ,
2020-03-17 13:01:58 +03:00
IRQF_SHARED , test - > name , test ) ;
2018-07-19 11:32:20 +03:00
if ( err )
goto fail ;
}
return true ;
fail :
switch ( irq_type ) {
case IRQ_TYPE_LEGACY :
dev_err ( dev , " Failed to request IRQ %d for Legacy \n " ,
pci_irq_vector ( pdev , i ) ) ;
break ;
case IRQ_TYPE_MSI :
dev_err ( dev , " Failed to request IRQ %d for MSI %d \n " ,
pci_irq_vector ( pdev , i ) ,
i + 1 ) ;
break ;
case IRQ_TYPE_MSIX :
dev_err ( dev , " Failed to request IRQ %d for MSI-X %d \n " ,
pci_irq_vector ( pdev , i ) ,
i + 1 ) ;
break ;
}
return false ;
}
2017-03-27 12:45:14 +03:00
static bool pci_endpoint_test_bar ( struct pci_endpoint_test * test ,
enum pci_barno barno )
{
int j ;
u32 val ;
int size ;
2017-08-18 17:58:08 +03:00
struct pci_dev * pdev = test - > pdev ;
2017-03-27 12:45:14 +03:00
if ( ! test - > bar [ barno ] )
return false ;
2017-08-18 17:58:08 +03:00
size = pci_resource_len ( pdev , barno ) ;
2017-03-27 12:45:14 +03:00
2017-08-18 17:58:05 +03:00
if ( barno = = test - > test_reg_bar )
size = 0x4 ;
2017-03-27 12:45:14 +03:00
for ( j = 0 ; j < size ; j + = 4 )
pci_endpoint_test_bar_writel ( test , barno , j , 0xA0A0A0A0 ) ;
for ( j = 0 ; j < size ; j + = 4 ) {
val = pci_endpoint_test_bar_readl ( test , barno , j ) ;
if ( val ! = 0xA0A0A0A0 )
return false ;
}
return true ;
}
static bool pci_endpoint_test_legacy_irq ( struct pci_endpoint_test * test )
{
u32 val ;
2018-07-19 11:32:17 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_TYPE ,
IRQ_TYPE_LEGACY ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_NUMBER , 0 ) ;
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_COMMAND ,
COMMAND_RAISE_LEGACY_IRQ ) ;
val = wait_for_completion_timeout ( & test - > irq_raised ,
msecs_to_jiffies ( 1000 ) ) ;
if ( ! val )
return false ;
return true ;
}
static bool pci_endpoint_test_msi_irq ( struct pci_endpoint_test * test ,
2018-07-19 11:32:19 +03:00
u16 msi_num , bool msix )
2017-03-27 12:45:14 +03:00
{
u32 val ;
struct pci_dev * pdev = test - > pdev ;
2018-07-19 11:32:17 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_TYPE ,
2018-07-19 11:32:19 +03:00
msix = = false ? IRQ_TYPE_MSI :
IRQ_TYPE_MSIX ) ;
2018-07-19 11:32:17 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_NUMBER , msi_num ) ;
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_COMMAND ,
2018-07-19 11:32:19 +03:00
msix = = false ? COMMAND_RAISE_MSI_IRQ :
COMMAND_RAISE_MSIX_IRQ ) ;
2017-03-27 12:45:14 +03:00
val = wait_for_completion_timeout ( & test - > irq_raised ,
msecs_to_jiffies ( 1000 ) ) ;
if ( ! val )
return false ;
2018-05-14 20:27:48 +03:00
if ( pci_irq_vector ( pdev , msi_num - 1 ) = = test - > last_irq )
2017-03-27 12:45:14 +03:00
return true ;
return false ;
}
2020-03-16 14:24:24 +03:00
static bool pci_endpoint_test_copy ( struct pci_endpoint_test * test ,
unsigned long arg )
2017-03-27 12:45:14 +03:00
{
2020-03-16 14:24:24 +03:00
struct pci_endpoint_test_xfer_param param ;
2017-03-27 12:45:14 +03:00
bool ret = false ;
void * src_addr ;
void * dst_addr ;
2020-03-16 14:24:24 +03:00
u32 flags = 0 ;
bool use_dma ;
size_t size ;
2017-03-27 12:45:14 +03:00
dma_addr_t src_phys_addr ;
dma_addr_t dst_phys_addr ;
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
2017-08-18 17:58:06 +03:00
void * orig_src_addr ;
dma_addr_t orig_src_phys_addr ;
void * orig_dst_addr ;
dma_addr_t orig_dst_phys_addr ;
size_t offset ;
size_t alignment = test - > alignment ;
2020-03-17 13:01:54 +03:00
int irq_type = test - > irq_type ;
2017-03-27 12:45:14 +03:00
u32 src_crc32 ;
u32 dst_crc32 ;
2020-03-16 14:24:24 +03:00
int err ;
2017-03-27 12:45:14 +03:00
2020-03-16 14:24:24 +03:00
err = copy_from_user ( & param , ( void __user * ) arg , sizeof ( param ) ) ;
if ( err ) {
dev_err ( dev , " Failed to get transfer param \n " ) ;
return false ;
}
size = param . size ;
2017-09-30 11:15:52 +03:00
if ( size > SIZE_MAX - alignment )
goto err ;
2020-03-16 14:24:24 +03:00
use_dma = ! ! ( param . flags & PCITEST_FLAGS_USE_DMA ) ;
if ( use_dma )
flags | = FLAG_USE_DMA ;
2018-07-19 11:32:20 +03:00
if ( irq_type < IRQ_TYPE_LEGACY | | irq_type > IRQ_TYPE_MSIX ) {
dev_err ( dev , " Invalid IRQ type option \n " ) ;
goto err ;
}
2020-03-16 14:24:22 +03:00
orig_src_addr = kzalloc ( size + alignment , GFP_KERNEL ) ;
2017-08-18 17:58:06 +03:00
if ( ! orig_src_addr ) {
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Failed to allocate source buffer \n " ) ;
2017-03-27 12:45:14 +03:00
ret = false ;
goto err ;
}
2020-03-16 14:24:22 +03:00
get_random_bytes ( orig_src_addr , size + alignment ) ;
orig_src_phys_addr = dma_map_single ( dev , orig_src_addr ,
size + alignment , DMA_TO_DEVICE ) ;
if ( dma_mapping_error ( dev , orig_src_phys_addr ) ) {
dev_err ( dev , " failed to map source buffer address \n " ) ;
ret = false ;
goto err_src_phys_addr ;
}
2017-08-18 17:58:06 +03:00
if ( alignment & & ! IS_ALIGNED ( orig_src_phys_addr , alignment ) ) {
src_phys_addr = PTR_ALIGN ( orig_src_phys_addr , alignment ) ;
offset = src_phys_addr - orig_src_phys_addr ;
src_addr = orig_src_addr + offset ;
} else {
src_phys_addr = orig_src_phys_addr ;
src_addr = orig_src_addr ;
}
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_LOWER_SRC_ADDR ,
lower_32_bits ( src_phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_UPPER_SRC_ADDR ,
upper_32_bits ( src_phys_addr ) ) ;
src_crc32 = crc32_le ( ~ 0 , src_addr , size ) ;
2020-03-16 14:24:22 +03:00
orig_dst_addr = kzalloc ( size + alignment , GFP_KERNEL ) ;
2017-08-18 17:58:06 +03:00
if ( ! orig_dst_addr ) {
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Failed to allocate destination address \n " ) ;
2017-03-27 12:45:14 +03:00
ret = false ;
2020-03-16 14:24:22 +03:00
goto err_dst_addr ;
}
orig_dst_phys_addr = dma_map_single ( dev , orig_dst_addr ,
size + alignment , DMA_FROM_DEVICE ) ;
if ( dma_mapping_error ( dev , orig_dst_phys_addr ) ) {
dev_err ( dev , " failed to map destination buffer address \n " ) ;
ret = false ;
goto err_dst_phys_addr ;
2017-08-18 17:58:06 +03:00
}
if ( alignment & & ! IS_ALIGNED ( orig_dst_phys_addr , alignment ) ) {
dst_phys_addr = PTR_ALIGN ( orig_dst_phys_addr , alignment ) ;
offset = dst_phys_addr - orig_dst_phys_addr ;
dst_addr = orig_dst_addr + offset ;
} else {
dst_phys_addr = orig_dst_phys_addr ;
dst_addr = orig_dst_addr ;
2017-03-27 12:45:14 +03:00
}
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_LOWER_DST_ADDR ,
lower_32_bits ( dst_phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_UPPER_DST_ADDR ,
upper_32_bits ( dst_phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_SIZE ,
size ) ;
2020-03-16 14:24:24 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_FLAGS , flags ) ;
2018-07-19 11:32:18 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_TYPE , irq_type ) ;
2018-07-19 11:32:17 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_NUMBER , 1 ) ;
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_COMMAND ,
2018-07-19 11:32:17 +03:00
COMMAND_COPY ) ;
2017-03-27 12:45:14 +03:00
wait_for_completion ( & test - > irq_raised ) ;
2020-03-16 14:24:22 +03:00
dma_unmap_single ( dev , orig_dst_phys_addr , size + alignment ,
DMA_FROM_DEVICE ) ;
2017-03-27 12:45:14 +03:00
dst_crc32 = crc32_le ( ~ 0 , dst_addr , size ) ;
if ( dst_crc32 = = src_crc32 )
ret = true ;
2020-03-16 14:24:22 +03:00
err_dst_phys_addr :
kfree ( orig_dst_addr ) ;
2017-03-27 12:45:14 +03:00
2020-03-16 14:24:22 +03:00
err_dst_addr :
dma_unmap_single ( dev , orig_src_phys_addr , size + alignment ,
DMA_TO_DEVICE ) ;
err_src_phys_addr :
kfree ( orig_src_addr ) ;
2017-03-27 12:45:14 +03:00
err :
return ret ;
}
2020-03-16 14:24:24 +03:00
static bool pci_endpoint_test_write ( struct pci_endpoint_test * test ,
unsigned long arg )
2017-03-27 12:45:14 +03:00
{
2020-03-16 14:24:24 +03:00
struct pci_endpoint_test_xfer_param param ;
2017-03-27 12:45:14 +03:00
bool ret = false ;
2020-03-16 14:24:24 +03:00
u32 flags = 0 ;
bool use_dma ;
2017-03-27 12:45:14 +03:00
u32 reg ;
void * addr ;
dma_addr_t phys_addr ;
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
2017-08-18 17:58:06 +03:00
void * orig_addr ;
dma_addr_t orig_phys_addr ;
size_t offset ;
size_t alignment = test - > alignment ;
2020-03-17 13:01:54 +03:00
int irq_type = test - > irq_type ;
2020-03-16 14:24:24 +03:00
size_t size ;
2017-03-27 12:45:14 +03:00
u32 crc32 ;
2020-03-16 14:24:24 +03:00
int err ;
2017-03-27 12:45:14 +03:00
2020-03-16 14:24:24 +03:00
err = copy_from_user ( & param , ( void __user * ) arg , sizeof ( param ) ) ;
if ( err ! = 0 ) {
dev_err ( dev , " Failed to get transfer param \n " ) ;
return false ;
}
size = param . size ;
2017-09-30 11:15:52 +03:00
if ( size > SIZE_MAX - alignment )
goto err ;
2020-03-16 14:24:24 +03:00
use_dma = ! ! ( param . flags & PCITEST_FLAGS_USE_DMA ) ;
if ( use_dma )
flags | = FLAG_USE_DMA ;
2018-07-19 11:32:20 +03:00
if ( irq_type < IRQ_TYPE_LEGACY | | irq_type > IRQ_TYPE_MSIX ) {
dev_err ( dev , " Invalid IRQ type option \n " ) ;
goto err ;
}
2020-03-16 14:24:22 +03:00
orig_addr = kzalloc ( size + alignment , GFP_KERNEL ) ;
2017-08-18 17:58:06 +03:00
if ( ! orig_addr ) {
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Failed to allocate address \n " ) ;
2017-03-27 12:45:14 +03:00
ret = false ;
goto err ;
}
2020-03-16 14:24:22 +03:00
get_random_bytes ( orig_addr , size + alignment ) ;
orig_phys_addr = dma_map_single ( dev , orig_addr , size + alignment ,
DMA_TO_DEVICE ) ;
if ( dma_mapping_error ( dev , orig_phys_addr ) ) {
dev_err ( dev , " failed to map source buffer address \n " ) ;
ret = false ;
goto err_phys_addr ;
}
2017-08-18 17:58:06 +03:00
if ( alignment & & ! IS_ALIGNED ( orig_phys_addr , alignment ) ) {
phys_addr = PTR_ALIGN ( orig_phys_addr , alignment ) ;
offset = phys_addr - orig_phys_addr ;
addr = orig_addr + offset ;
} else {
phys_addr = orig_phys_addr ;
addr = orig_addr ;
}
2017-03-27 12:45:14 +03:00
crc32 = crc32_le ( ~ 0 , addr , size ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_CHECKSUM ,
crc32 ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_LOWER_SRC_ADDR ,
lower_32_bits ( phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_UPPER_SRC_ADDR ,
upper_32_bits ( phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_SIZE , size ) ;
2020-03-16 14:24:24 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_FLAGS , flags ) ;
2018-07-19 11:32:18 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_TYPE , irq_type ) ;
2018-07-19 11:32:17 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_NUMBER , 1 ) ;
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_COMMAND ,
2018-07-19 11:32:17 +03:00
COMMAND_READ ) ;
2017-03-27 12:45:14 +03:00
wait_for_completion ( & test - > irq_raised ) ;
reg = pci_endpoint_test_readl ( test , PCI_ENDPOINT_TEST_STATUS ) ;
if ( reg & STATUS_READ_SUCCESS )
ret = true ;
2020-03-16 14:24:22 +03:00
dma_unmap_single ( dev , orig_phys_addr , size + alignment ,
DMA_TO_DEVICE ) ;
err_phys_addr :
kfree ( orig_addr ) ;
2017-03-27 12:45:14 +03:00
err :
return ret ;
}
2020-03-16 14:24:24 +03:00
static bool pci_endpoint_test_read ( struct pci_endpoint_test * test ,
unsigned long arg )
2017-03-27 12:45:14 +03:00
{
2020-03-16 14:24:24 +03:00
struct pci_endpoint_test_xfer_param param ;
2017-03-27 12:45:14 +03:00
bool ret = false ;
2020-03-16 14:24:24 +03:00
u32 flags = 0 ;
bool use_dma ;
size_t size ;
2017-03-27 12:45:14 +03:00
void * addr ;
dma_addr_t phys_addr ;
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
2017-08-18 17:58:06 +03:00
void * orig_addr ;
dma_addr_t orig_phys_addr ;
size_t offset ;
size_t alignment = test - > alignment ;
2020-03-17 13:01:54 +03:00
int irq_type = test - > irq_type ;
2017-03-27 12:45:14 +03:00
u32 crc32 ;
2020-03-16 14:24:24 +03:00
int err ;
2017-03-27 12:45:14 +03:00
2020-03-16 14:24:24 +03:00
err = copy_from_user ( & param , ( void __user * ) arg , sizeof ( param ) ) ;
if ( err ) {
dev_err ( dev , " Failed to get transfer param \n " ) ;
return false ;
}
size = param . size ;
2017-09-30 11:15:52 +03:00
if ( size > SIZE_MAX - alignment )
goto err ;
2020-03-16 14:24:24 +03:00
use_dma = ! ! ( param . flags & PCITEST_FLAGS_USE_DMA ) ;
if ( use_dma )
flags | = FLAG_USE_DMA ;
2018-07-19 11:32:20 +03:00
if ( irq_type < IRQ_TYPE_LEGACY | | irq_type > IRQ_TYPE_MSIX ) {
dev_err ( dev , " Invalid IRQ type option \n " ) ;
goto err ;
}
2020-03-16 14:24:22 +03:00
orig_addr = kzalloc ( size + alignment , GFP_KERNEL ) ;
2017-08-18 17:58:06 +03:00
if ( ! orig_addr ) {
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Failed to allocate destination address \n " ) ;
2017-03-27 12:45:14 +03:00
ret = false ;
goto err ;
}
2020-03-16 14:24:22 +03:00
orig_phys_addr = dma_map_single ( dev , orig_addr , size + alignment ,
DMA_FROM_DEVICE ) ;
if ( dma_mapping_error ( dev , orig_phys_addr ) ) {
dev_err ( dev , " failed to map source buffer address \n " ) ;
ret = false ;
goto err_phys_addr ;
}
2017-08-18 17:58:06 +03:00
if ( alignment & & ! IS_ALIGNED ( orig_phys_addr , alignment ) ) {
phys_addr = PTR_ALIGN ( orig_phys_addr , alignment ) ;
offset = phys_addr - orig_phys_addr ;
addr = orig_addr + offset ;
} else {
phys_addr = orig_phys_addr ;
addr = orig_addr ;
}
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_LOWER_DST_ADDR ,
lower_32_bits ( phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_UPPER_DST_ADDR ,
upper_32_bits ( phys_addr ) ) ;
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_SIZE , size ) ;
2020-03-16 14:24:24 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_FLAGS , flags ) ;
2018-07-19 11:32:18 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_TYPE , irq_type ) ;
2018-07-19 11:32:17 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_IRQ_NUMBER , 1 ) ;
2017-03-27 12:45:14 +03:00
pci_endpoint_test_writel ( test , PCI_ENDPOINT_TEST_COMMAND ,
2018-07-19 11:32:17 +03:00
COMMAND_WRITE ) ;
2017-03-27 12:45:14 +03:00
wait_for_completion ( & test - > irq_raised ) ;
2020-03-16 14:24:22 +03:00
dma_unmap_single ( dev , orig_phys_addr , size + alignment ,
DMA_FROM_DEVICE ) ;
2017-03-27 12:45:14 +03:00
crc32 = crc32_le ( ~ 0 , addr , size ) ;
if ( crc32 = = pci_endpoint_test_readl ( test , PCI_ENDPOINT_TEST_CHECKSUM ) )
ret = true ;
2020-03-16 14:24:22 +03:00
err_phys_addr :
kfree ( orig_addr ) ;
2017-03-27 12:45:14 +03:00
err :
return ret ;
}
2020-03-17 13:01:55 +03:00
static bool pci_endpoint_test_clear_irq ( struct pci_endpoint_test * test )
{
pci_endpoint_test_release_irq ( test ) ;
pci_endpoint_test_free_irq_vectors ( test ) ;
return true ;
}
2018-07-19 11:32:20 +03:00
static bool pci_endpoint_test_set_irq ( struct pci_endpoint_test * test ,
int req_irq_type )
{
struct pci_dev * pdev = test - > pdev ;
struct device * dev = & pdev - > dev ;
if ( req_irq_type < IRQ_TYPE_LEGACY | | req_irq_type > IRQ_TYPE_MSIX ) {
dev_err ( dev , " Invalid IRQ type option \n " ) ;
return false ;
}
2020-03-17 13:01:54 +03:00
if ( test - > irq_type = = req_irq_type )
2018-07-19 11:32:20 +03:00
return true ;
pci_endpoint_test_release_irq ( test ) ;
pci_endpoint_test_free_irq_vectors ( test ) ;
if ( ! pci_endpoint_test_alloc_irq_vectors ( test , req_irq_type ) )
goto err ;
if ( ! pci_endpoint_test_request_irq ( test ) )
goto err ;
return true ;
err :
pci_endpoint_test_free_irq_vectors ( test ) ;
return false ;
}
2017-03-27 12:45:14 +03:00
static long pci_endpoint_test_ioctl ( struct file * file , unsigned int cmd ,
unsigned long arg )
{
int ret = - EINVAL ;
enum pci_barno bar ;
struct pci_endpoint_test * test = to_endpoint_test ( file - > private_data ) ;
2019-03-25 12:39:46 +03:00
struct pci_dev * pdev = test - > pdev ;
2017-03-27 12:45:14 +03:00
mutex_lock ( & test - > mutex ) ;
switch ( cmd ) {
case PCITEST_BAR :
bar = arg ;
2020-10-22 00:28:10 +03:00
if ( bar > BAR_5 )
2017-03-27 12:45:14 +03:00
goto ret ;
2019-03-25 12:39:46 +03:00
if ( is_am654_pci_dev ( pdev ) & & bar = = BAR_0 )
goto ret ;
2017-03-27 12:45:14 +03:00
ret = pci_endpoint_test_bar ( test , bar ) ;
break ;
case PCITEST_LEGACY_IRQ :
ret = pci_endpoint_test_legacy_irq ( test ) ;
break ;
case PCITEST_MSI :
2018-07-19 11:32:19 +03:00
case PCITEST_MSIX :
ret = pci_endpoint_test_msi_irq ( test , arg , cmd = = PCITEST_MSIX ) ;
2017-03-27 12:45:14 +03:00
break ;
case PCITEST_WRITE :
ret = pci_endpoint_test_write ( test , arg ) ;
break ;
case PCITEST_READ :
ret = pci_endpoint_test_read ( test , arg ) ;
break ;
case PCITEST_COPY :
ret = pci_endpoint_test_copy ( test , arg ) ;
break ;
2018-07-19 11:32:20 +03:00
case PCITEST_SET_IRQTYPE :
ret = pci_endpoint_test_set_irq ( test , arg ) ;
break ;
case PCITEST_GET_IRQTYPE :
ret = irq_type ;
break ;
2020-03-17 13:01:55 +03:00
case PCITEST_CLEAR_IRQ :
ret = pci_endpoint_test_clear_irq ( test ) ;
break ;
2017-03-27 12:45:14 +03:00
}
ret :
mutex_unlock ( & test - > mutex ) ;
return ret ;
}
static const struct file_operations pci_endpoint_test_fops = {
. owner = THIS_MODULE ,
. unlocked_ioctl = pci_endpoint_test_ioctl ,
} ;
static int pci_endpoint_test_probe ( struct pci_dev * pdev ,
const struct pci_device_id * ent )
{
int err ;
int id ;
2020-03-17 13:01:57 +03:00
char name [ 24 ] ;
2017-03-27 12:45:14 +03:00
enum pci_barno bar ;
void __iomem * base ;
struct device * dev = & pdev - > dev ;
struct pci_endpoint_test * test ;
2017-08-18 17:58:05 +03:00
struct pci_endpoint_test_data * data ;
enum pci_barno test_reg_bar = BAR_0 ;
2017-03-27 12:45:14 +03:00
struct miscdevice * misc_device ;
if ( pci_is_bridge ( pdev ) )
return - ENODEV ;
test = devm_kzalloc ( dev , sizeof ( * test ) , GFP_KERNEL ) ;
if ( ! test )
return - ENOMEM ;
2017-08-18 17:58:05 +03:00
test - > test_reg_bar = 0 ;
2017-08-18 17:58:06 +03:00
test - > alignment = 0 ;
2017-03-27 12:45:14 +03:00
test - > pdev = pdev ;
2020-03-17 13:01:54 +03:00
test - > irq_type = IRQ_TYPE_UNDEFINED ;
2017-08-18 17:58:05 +03:00
2018-07-19 11:32:18 +03:00
if ( no_msi )
irq_type = IRQ_TYPE_LEGACY ;
2017-08-18 17:58:05 +03:00
data = ( struct pci_endpoint_test_data * ) ent - > driver_data ;
2017-08-18 17:58:06 +03:00
if ( data ) {
2017-08-18 17:58:05 +03:00
test_reg_bar = data - > test_reg_bar ;
2019-03-25 12:39:47 +03:00
test - > test_reg_bar = test_reg_bar ;
2017-08-18 17:58:06 +03:00
test - > alignment = data - > alignment ;
2018-07-19 11:32:18 +03:00
irq_type = data - > irq_type ;
2017-08-18 17:58:06 +03:00
}
2017-08-18 17:58:05 +03:00
2017-03-27 12:45:14 +03:00
init_completion ( & test - > irq_raised ) ;
mutex_init ( & test - > mutex ) ;
2020-03-16 14:24:22 +03:00
if ( ( dma_set_mask_and_coherent ( & pdev - > dev , DMA_BIT_MASK ( 48 ) ) ! = 0 ) & &
dma_set_mask_and_coherent ( & pdev - > dev , DMA_BIT_MASK ( 32 ) ) ! = 0 ) {
dev_err ( dev , " Cannot set DMA mask \n " ) ;
return - EINVAL ;
}
2017-03-27 12:45:14 +03:00
err = pci_enable_device ( pdev ) ;
if ( err ) {
dev_err ( dev , " Cannot enable PCI device \n " ) ;
return err ;
}
err = pci_request_regions ( pdev , DRV_MODULE_NAME ) ;
if ( err ) {
dev_err ( dev , " Cannot obtain PCI resources \n " ) ;
goto err_disable_pdev ;
}
pci_set_master ( pdev ) ;
2020-11-19 15:49:18 +03:00
if ( ! pci_endpoint_test_alloc_irq_vectors ( test , irq_type ) ) {
err = - EINVAL ;
2018-07-19 11:32:20 +03:00
goto err_disable_irq ;
2020-11-19 15:49:18 +03:00
}
2017-03-27 12:45:14 +03:00
2019-09-28 02:43:08 +03:00
for ( bar = 0 ; bar < PCI_STD_NUM_BARS ; bar + + ) {
2018-03-28 14:50:17 +03:00
if ( pci_resource_flags ( pdev , bar ) & IORESOURCE_MEM ) {
base = pci_ioremap_bar ( pdev , bar ) ;
if ( ! base ) {
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Failed to read BAR%d \n " , bar ) ;
2018-03-28 14:50:17 +03:00
WARN_ON ( bar = = test_reg_bar ) ;
}
test - > bar [ bar ] = base ;
2017-03-27 12:45:14 +03:00
}
}
2017-08-18 17:58:05 +03:00
test - > base = test - > bar [ test_reg_bar ] ;
2017-03-27 12:45:14 +03:00
if ( ! test - > base ) {
2017-10-11 11:44:36 +03:00
err = - ENOMEM ;
2017-08-18 17:58:05 +03:00
dev_err ( dev , " Cannot perform PCI test without BAR%d \n " ,
test_reg_bar ) ;
2017-03-27 12:45:14 +03:00
goto err_iounmap ;
}
pci_set_drvdata ( pdev , test ) ;
id = ida_simple_get ( & pci_endpoint_test_ida , 0 , 0 , GFP_KERNEL ) ;
if ( id < 0 ) {
2017-10-11 11:44:36 +03:00
err = id ;
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Unable to get id \n " ) ;
2017-03-27 12:45:14 +03:00
goto err_iounmap ;
}
snprintf ( name , sizeof ( name ) , DRV_MODULE_NAME " .%d " , id ) ;
2020-03-17 13:01:58 +03:00
test - > name = kstrdup ( name , GFP_KERNEL ) ;
if ( ! test - > name ) {
err = - ENOMEM ;
goto err_ida_remove ;
}
2020-11-19 15:49:18 +03:00
if ( ! pci_endpoint_test_request_irq ( test ) ) {
err = - EINVAL ;
2020-03-17 13:01:58 +03:00
goto err_kfree_test_name ;
2020-11-19 15:49:18 +03:00
}
2020-03-17 13:01:58 +03:00
2017-03-27 12:45:14 +03:00
misc_device = & test - > miscdev ;
misc_device - > minor = MISC_DYNAMIC_MINOR ;
2017-10-11 11:44:37 +03:00
misc_device - > name = kstrdup ( name , GFP_KERNEL ) ;
if ( ! misc_device - > name ) {
err = - ENOMEM ;
2020-03-17 13:01:58 +03:00
goto err_release_irq ;
2017-10-11 11:44:37 +03:00
}
2017-03-27 12:45:14 +03:00
misc_device - > fops = & pci_endpoint_test_fops ,
err = misc_register ( misc_device ) ;
if ( err ) {
2018-05-14 19:56:23 +03:00
dev_err ( dev , " Failed to register device \n " ) ;
2017-10-11 11:44:37 +03:00
goto err_kfree_name ;
2017-03-27 12:45:14 +03:00
}
return 0 ;
2017-10-11 11:44:37 +03:00
err_kfree_name :
kfree ( misc_device - > name ) ;
2020-03-17 13:01:58 +03:00
err_release_irq :
pci_endpoint_test_release_irq ( test ) ;
err_kfree_test_name :
kfree ( test - > name ) ;
2017-03-27 12:45:14 +03:00
err_ida_remove :
ida_simple_remove ( & pci_endpoint_test_ida , id ) ;
err_iounmap :
2019-09-28 02:43:08 +03:00
for ( bar = 0 ; bar < PCI_STD_NUM_BARS ; bar + + ) {
2017-03-27 12:45:14 +03:00
if ( test - > bar [ bar ] )
pci_iounmap ( pdev , test - > bar [ bar ] ) ;
}
2018-07-19 11:32:20 +03:00
err_disable_irq :
pci_endpoint_test_free_irq_vectors ( test ) ;
2017-03-27 12:45:14 +03:00
pci_release_regions ( pdev ) ;
err_disable_pdev :
pci_disable_device ( pdev ) ;
return err ;
}
static void pci_endpoint_test_remove ( struct pci_dev * pdev )
{
int id ;
enum pci_barno bar ;
struct pci_endpoint_test * test = pci_get_drvdata ( pdev ) ;
struct miscdevice * misc_device = & test - > miscdev ;
if ( sscanf ( misc_device - > name , DRV_MODULE_NAME " .%d " , & id ) ! = 1 )
return ;
2017-09-30 11:16:51 +03:00
if ( id < 0 )
return ;
2017-03-27 12:45:14 +03:00
misc_deregister ( & test - > miscdev ) ;
2017-10-11 11:44:37 +03:00
kfree ( misc_device - > name ) ;
2020-03-17 13:01:58 +03:00
kfree ( test - > name ) ;
2017-03-27 12:45:14 +03:00
ida_simple_remove ( & pci_endpoint_test_ida , id ) ;
2019-09-28 02:43:08 +03:00
for ( bar = 0 ; bar < PCI_STD_NUM_BARS ; bar + + ) {
2017-03-27 12:45:14 +03:00
if ( test - > bar [ bar ] )
pci_iounmap ( pdev , test - > bar [ bar ] ) ;
}
2018-07-19 11:32:20 +03:00
pci_endpoint_test_release_irq ( test ) ;
pci_endpoint_test_free_irq_vectors ( test ) ;
2017-03-27 12:45:14 +03:00
pci_release_regions ( pdev ) ;
pci_disable_device ( pdev ) ;
}
2020-03-16 14:24:22 +03:00
static const struct pci_endpoint_test_data default_data = {
. test_reg_bar = BAR_0 ,
. alignment = SZ_4K ,
. irq_type = IRQ_TYPE_MSI ,
} ;
2019-03-25 12:39:46 +03:00
static const struct pci_endpoint_test_data am654_data = {
. test_reg_bar = BAR_2 ,
. alignment = SZ_64K ,
. irq_type = IRQ_TYPE_MSI ,
} ;
2020-07-22 14:03:16 +03:00
static const struct pci_endpoint_test_data j721e_data = {
. alignment = 256 ,
. irq_type = IRQ_TYPE_MSI ,
} ;
2017-03-27 12:45:14 +03:00
static const struct pci_device_id pci_endpoint_test_tbl [ ] = {
2020-03-16 14:24:22 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_TI , PCI_DEVICE_ID_TI_DRA74x ) ,
. driver_data = ( kernel_ulong_t ) & default_data ,
} ,
{ PCI_DEVICE ( PCI_VENDOR_ID_TI , PCI_DEVICE_ID_TI_DRA72x ) ,
. driver_data = ( kernel_ulong_t ) & default_data ,
} ,
2020-09-18 11:00:24 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_FREESCALE , 0x81c0 ) ,
. driver_data = ( kernel_ulong_t ) & default_data ,
} ,
{ PCI_DEVICE ( PCI_VENDOR_ID_FREESCALE , PCI_DEVICE_ID_LS1088A ) ,
. driver_data = ( kernel_ulong_t ) & default_data ,
} ,
2019-06-04 16:29:25 +03:00
{ PCI_DEVICE_DATA ( SYNOPSYS , EDDA , NULL ) } ,
2019-03-25 12:39:46 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_TI , PCI_DEVICE_ID_TI_AM654 ) ,
. driver_data = ( kernel_ulong_t ) & am654_data
} ,
2020-08-14 20:30:34 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_RENESAS , PCI_DEVICE_ID_RENESAS_R8A774A1 ) , } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_RENESAS , PCI_DEVICE_ID_RENESAS_R8A774B1 ) , } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_RENESAS , PCI_DEVICE_ID_RENESAS_R8A774C0 ) , } ,
2020-09-04 13:38:51 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_RENESAS , PCI_DEVICE_ID_RENESAS_R8A774E1 ) , } ,
2020-07-22 14:03:16 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_TI , PCI_DEVICE_ID_TI_J721E ) ,
. driver_data = ( kernel_ulong_t ) & j721e_data ,
} ,
2017-03-27 12:45:14 +03:00
{ }
} ;
MODULE_DEVICE_TABLE ( pci , pci_endpoint_test_tbl ) ;
static struct pci_driver pci_endpoint_test_driver = {
. name = DRV_MODULE_NAME ,
. id_table = pci_endpoint_test_tbl ,
. probe = pci_endpoint_test_probe ,
. remove = pci_endpoint_test_remove ,
} ;
module_pci_driver ( pci_endpoint_test_driver ) ;
MODULE_DESCRIPTION ( " PCI ENDPOINT TEST HOST DRIVER " ) ;
MODULE_AUTHOR ( " Kishon Vijay Abraham I <kishon@ti.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;