2019-06-01 11:08:55 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2012-05-09 17:27:19 +04:00
/*
* Industry - pack bus support functions .
*
2012-11-16 19:19:58 +04:00
* Copyright ( C ) 2011 - 2012 CERN ( www . cern . ch )
* Author : Samuel Iglesias Gonsalvez < siglesias @ igalia . com >
2012-05-09 17:27:19 +04:00
*/
# include <linux/module.h>
2012-05-18 13:10:05 +04:00
# include <linux/slab.h>
2012-05-25 12:03:01 +04:00
# include <linux/idr.h>
2012-11-12 00:41:12 +04:00
# include <linux/io.h>
2012-11-16 22:33:45 +04:00
# include <linux/ipack.h>
2012-05-09 17:27:19 +04:00
# define to_ipack_dev(device) container_of(device, struct ipack_device, dev)
# define to_ipack_driver(drv) container_of(drv, struct ipack_driver, driver)
2012-05-25 12:03:01 +04:00
static DEFINE_IDA ( ipack_ida ) ;
2012-05-09 17:27:19 +04:00
2012-05-18 13:10:05 +04:00
static void ipack_device_release ( struct device * dev )
{
struct ipack_device * device = to_ipack_dev ( dev ) ;
2012-09-04 19:01:14 +04:00
kfree ( device - > id ) ;
2012-09-27 14:37:26 +04:00
device - > release ( device ) ;
2012-05-18 13:10:05 +04:00
}
2012-09-04 19:01:19 +04:00
static inline const struct ipack_device_id *
ipack_match_one_device ( const struct ipack_device_id * id ,
const struct ipack_device * device )
2012-05-09 17:27:19 +04:00
{
2012-09-07 12:29:19 +04:00
if ( ( id - > format = = IPACK_ANY_FORMAT | |
id - > format = = device - > id_format ) & &
2012-09-04 19:01:19 +04:00
( id - > vendor = = IPACK_ANY_ID | | id - > vendor = = device - > id_vendor ) & &
( id - > device = = IPACK_ANY_ID | | id - > device = = device - > id_device ) )
return id ;
return NULL ;
}
2012-05-09 17:27:19 +04:00
2012-09-04 19:01:19 +04:00
static const struct ipack_device_id *
ipack_match_id ( const struct ipack_device_id * ids , struct ipack_device * idev )
{
if ( ids ) {
while ( ids - > vendor | | ids - > device ) {
if ( ipack_match_one_device ( ids , idev ) )
return ids ;
ids + + ;
}
}
return NULL ;
}
static int ipack_bus_match ( struct device * dev , struct device_driver * drv )
{
struct ipack_device * idev = to_ipack_dev ( dev ) ;
struct ipack_driver * idrv = to_ipack_driver ( drv ) ;
const struct ipack_device_id * found_id ;
2012-05-09 17:27:19 +04:00
2012-09-04 19:01:19 +04:00
found_id = ipack_match_id ( idrv - > id_table , idev ) ;
2012-09-11 15:34:59 +04:00
return found_id ? 1 : 0 ;
2012-05-09 17:27:19 +04:00
}
static int ipack_bus_probe ( struct device * device )
{
struct ipack_device * dev = to_ipack_dev ( device ) ;
2012-09-11 15:34:59 +04:00
struct ipack_driver * drv = to_ipack_driver ( device - > driver ) ;
2012-05-09 17:27:19 +04:00
2012-09-11 15:34:59 +04:00
return drv - > ops - > probe ( dev ) ;
2012-05-09 17:27:19 +04:00
}
2021-07-13 22:35:22 +03:00
static void ipack_bus_remove ( struct device * device )
2012-05-09 17:27:19 +04:00
{
struct ipack_device * dev = to_ipack_dev ( device ) ;
2012-09-11 15:34:59 +04:00
struct ipack_driver * drv = to_ipack_driver ( device - > driver ) ;
2012-05-09 17:27:19 +04:00
2021-02-08 00:55:56 +03:00
if ( drv - > ops - > remove )
drv - > ops - > remove ( dev ) ;
2012-05-09 17:27:19 +04:00
}
2023-01-11 14:30:17 +03:00
static int ipack_uevent ( const struct device * dev , struct kobj_uevent_env * env )
2012-09-04 19:01:20 +04:00
{
2023-01-11 14:30:17 +03:00
const struct ipack_device * idev ;
2012-09-04 19:01:20 +04:00
if ( ! dev )
return - ENODEV ;
idev = to_ipack_dev ( dev ) ;
if ( add_uevent_var ( env ,
" MODALIAS=ipack:f%02Xv%08Xd%08X " , idev - > id_format ,
idev - > id_vendor , idev - > id_device ) )
return - ENOMEM ;
return 0 ;
}
2012-09-04 19:01:15 +04:00
# define ipack_device_attr(field, format_string) \
static ssize_t \
field # # _show ( struct device * dev , struct device_attribute * attr , \
char * buf ) \
{ \
struct ipack_device * idev = to_ipack_dev ( dev ) ; \
return sprintf ( buf , format_string , idev - > field ) ; \
}
2012-09-04 19:01:21 +04:00
static ssize_t id_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
unsigned int i , c , l , s ;
struct ipack_device * idev = to_ipack_dev ( dev ) ;
switch ( idev - > id_format ) {
case IPACK_ID_VERSION_1 :
l = 0x7 ; s = 1 ; break ;
case IPACK_ID_VERSION_2 :
l = 0xf ; s = 2 ; break ;
default :
return - EIO ;
}
c = 0 ;
for ( i = 0 ; i < idev - > id_avail ; i + + ) {
if ( i > 0 ) {
if ( ( i & l ) = = 0 )
buf [ c + + ] = ' \n ' ;
else if ( ( i & s ) = = 0 )
buf [ c + + ] = ' ' ;
}
sprintf ( & buf [ c ] , " %02x " , idev - > id [ i ] ) ;
c + = 2 ;
}
buf [ c + + ] = ' \n ' ;
return c ;
}
2012-09-04 19:01:15 +04:00
static ssize_t
id_vendor_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct ipack_device * idev = to_ipack_dev ( dev ) ;
switch ( idev - > id_format ) {
case IPACK_ID_VERSION_1 :
return sprintf ( buf , " 0x%02x \n " , idev - > id_vendor ) ;
case IPACK_ID_VERSION_2 :
return sprintf ( buf , " 0x%06x \n " , idev - > id_vendor ) ;
default :
return - EIO ;
}
}
static ssize_t
id_device_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct ipack_device * idev = to_ipack_dev ( dev ) ;
switch ( idev - > id_format ) {
case IPACK_ID_VERSION_1 :
return sprintf ( buf , " 0x%02x \n " , idev - > id_device ) ;
case IPACK_ID_VERSION_2 :
return sprintf ( buf , " 0x%04x \n " , idev - > id_device ) ;
default :
return - EIO ;
}
}
2012-09-04 19:01:20 +04:00
static ssize_t modalias_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
struct ipack_device * idev = to_ipack_dev ( dev ) ;
return sprintf ( buf , " ipac:f%02Xv%08Xd%08X " , idev - > id_format ,
idev - > id_vendor , idev - > id_device ) ;
}
2016-10-28 03:47:07 +03:00
ipack_device_attr ( id_format , " 0x%hhx \n " ) ;
2012-09-04 19:01:15 +04:00
2013-10-08 05:27:36 +04:00
static DEVICE_ATTR_RO ( id ) ;
static DEVICE_ATTR_RO ( id_device ) ;
static DEVICE_ATTR_RO ( id_format ) ;
static DEVICE_ATTR_RO ( id_vendor ) ;
static DEVICE_ATTR_RO ( modalias ) ;
static struct attribute * ipack_attrs [ ] = {
& dev_attr_id . attr ,
& dev_attr_id_device . attr ,
& dev_attr_id_format . attr ,
& dev_attr_id_vendor . attr ,
& dev_attr_modalias . attr ,
NULL ,
2012-09-04 19:01:15 +04:00
} ;
2013-10-08 05:27:36 +04:00
ATTRIBUTE_GROUPS ( ipack ) ;
2012-09-04 19:01:15 +04:00
2012-05-09 17:27:19 +04:00
static struct bus_type ipack_bus_type = {
2012-09-04 19:01:15 +04:00
. name = " ipack " ,
. probe = ipack_bus_probe ,
. match = ipack_bus_match ,
. remove = ipack_bus_remove ,
2013-10-08 05:27:36 +04:00
. dev_groups = ipack_groups ,
2012-09-04 19:01:20 +04:00
. uevent = ipack_uevent ,
2012-05-09 17:27:19 +04:00
} ;
2012-05-18 13:10:05 +04:00
struct ipack_bus_device * ipack_bus_register ( struct device * parent , int slots ,
2014-09-02 19:31:40 +04:00
const struct ipack_bus_ops * ops ,
struct module * owner )
2012-05-09 17:27:19 +04:00
{
int bus_nr ;
2012-05-18 13:10:05 +04:00
struct ipack_bus_device * bus ;
2017-05-15 12:08:10 +03:00
bus = kzalloc ( sizeof ( * bus ) , GFP_KERNEL ) ;
2012-05-18 13:10:05 +04:00
if ( ! bus )
return NULL ;
2012-05-09 17:27:19 +04:00
2023-11-01 12:41:17 +03:00
bus_nr = ida_alloc ( & ipack_ida , GFP_KERNEL ) ;
2012-05-18 13:10:05 +04:00
if ( bus_nr < 0 ) {
kfree ( bus ) ;
return NULL ;
}
2012-05-09 17:27:19 +04:00
bus - > bus_nr = bus_nr ;
2012-05-18 13:10:05 +04:00
bus - > parent = parent ;
bus - > slots = slots ;
bus - > ops = ops ;
2014-09-02 19:31:40 +04:00
bus - > owner = owner ;
2012-05-18 13:10:05 +04:00
return bus ;
2012-05-09 17:27:19 +04:00
}
EXPORT_SYMBOL_GPL ( ipack_bus_register ) ;
2012-09-11 15:35:09 +04:00
static int ipack_unregister_bus_member ( struct device * dev , void * data )
{
struct ipack_device * idev = to_ipack_dev ( dev ) ;
struct ipack_bus_device * bus = data ;
2012-09-27 14:37:25 +04:00
if ( idev - > bus = = bus )
2013-03-08 12:21:47 +04:00
ipack_device_del ( idev ) ;
2012-09-11 15:35:09 +04:00
return 1 ;
}
2012-05-09 17:27:19 +04:00
int ipack_bus_unregister ( struct ipack_bus_device * bus )
{
2012-11-12 00:41:12 +04:00
bus_for_each_dev ( & ipack_bus_type , NULL , bus ,
ipack_unregister_bus_member ) ;
2023-11-01 12:41:17 +03:00
ida_free ( & ipack_ida , bus - > bus_nr ) ;
2012-05-18 13:10:05 +04:00
kfree ( bus ) ;
2012-05-09 17:27:19 +04:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( ipack_bus_unregister ) ;
2012-05-18 13:10:05 +04:00
int ipack_driver_register ( struct ipack_driver * edrv , struct module * owner ,
2012-09-10 22:14:01 +04:00
const char * name )
2012-05-09 17:27:19 +04:00
{
2021-02-08 00:55:55 +03:00
if ( ! edrv - > ops - > probe )
return - EINVAL ;
2012-05-18 13:10:05 +04:00
edrv - > driver . owner = owner ;
edrv - > driver . name = name ;
2012-05-09 17:27:19 +04:00
edrv - > driver . bus = & ipack_bus_type ;
return driver_register ( & edrv - > driver ) ;
}
EXPORT_SYMBOL_GPL ( ipack_driver_register ) ;
void ipack_driver_unregister ( struct ipack_driver * edrv )
{
driver_unregister ( & edrv - > driver ) ;
}
EXPORT_SYMBOL_GPL ( ipack_driver_unregister ) ;
2012-09-11 15:35:03 +04:00
static u16 ipack_crc_byte ( u16 crc , u8 c )
{
int i ;
crc ^ = c < < 8 ;
for ( i = 0 ; i < 8 ; i + + )
crc = ( crc < < 1 ) ^ ( ( crc & 0x8000 ) ? 0x1021 : 0 ) ;
return crc ;
}
/*
* The algorithm in lib / crc - ccitt . c does not seem to apply since it uses the
* opposite bit ordering .
*/
static u8 ipack_calc_crc1 ( struct ipack_device * dev )
{
u8 c ;
u16 crc ;
unsigned int i ;
crc = 0xffff ;
for ( i = 0 ; i < dev - > id_avail ; i + + ) {
c = ( i ! = 11 ) ? dev - > id [ i ] : 0 ;
crc = ipack_crc_byte ( crc , c ) ;
}
crc = ~ crc ;
return crc & 0xff ;
}
static u16 ipack_calc_crc2 ( struct ipack_device * dev )
{
u8 c ;
u16 crc ;
unsigned int i ;
crc = 0xffff ;
for ( i = 0 ; i < dev - > id_avail ; i + + ) {
c = ( ( i ! = 0x18 ) & & ( i ! = 0x19 ) ) ? dev - > id [ i ] : 0 ;
crc = ipack_crc_byte ( crc , c ) ;
}
crc = ~ crc ;
return crc ;
}
2012-09-04 19:01:15 +04:00
static void ipack_parse_id1 ( struct ipack_device * dev )
{
u8 * id = dev - > id ;
2012-09-11 15:35:03 +04:00
u8 crc ;
2012-09-04 19:01:15 +04:00
dev - > id_vendor = id [ 4 ] ;
dev - > id_device = id [ 5 ] ;
2012-09-11 15:34:57 +04:00
dev - > speed_8mhz = 1 ;
dev - > speed_32mhz = ( id [ 7 ] = = ' H ' ) ;
2012-09-11 15:35:03 +04:00
crc = ipack_calc_crc1 ( dev ) ;
dev - > id_crc_correct = ( crc = = id [ 11 ] ) ;
if ( ! dev - > id_crc_correct ) {
dev_warn ( & dev - > dev , " ID CRC invalid found 0x%x, expected 0x%x. \n " ,
id [ 11 ] , crc ) ;
}
2012-09-04 19:01:15 +04:00
}
static void ipack_parse_id2 ( struct ipack_device * dev )
{
__be16 * id = ( __be16 * ) dev - > id ;
2012-09-11 15:35:03 +04:00
u16 flags , crc ;
2012-09-04 19:01:15 +04:00
dev - > id_vendor = ( ( be16_to_cpu ( id [ 3 ] ) & 0xff ) < < 16 )
+ be16_to_cpu ( id [ 4 ] ) ;
dev - > id_device = be16_to_cpu ( id [ 5 ] ) ;
2012-09-11 15:34:57 +04:00
flags = be16_to_cpu ( id [ 10 ] ) ;
dev - > speed_8mhz = ! ! ( flags & 2 ) ;
dev - > speed_32mhz = ! ! ( flags & 4 ) ;
2012-09-11 15:35:03 +04:00
crc = ipack_calc_crc2 ( dev ) ;
dev - > id_crc_correct = ( crc = = be16_to_cpu ( id [ 12 ] ) ) ;
if ( ! dev - > id_crc_correct ) {
dev_warn ( & dev - > dev , " ID CRC invalid found 0x%x, expected 0x%x. \n " ,
id [ 11 ] , crc ) ;
}
2012-09-04 19:01:15 +04:00
}
2012-09-04 19:01:14 +04:00
static int ipack_device_read_id ( struct ipack_device * dev )
{
u8 __iomem * idmem ;
int i ;
int ret = 0 ;
2012-09-27 14:37:34 +04:00
idmem = ioremap ( dev - > region [ IPACK_ID_SPACE ] . start ,
dev - > region [ IPACK_ID_SPACE ] . size ) ;
if ( ! idmem ) {
2012-09-04 19:01:14 +04:00
dev_err ( & dev - > dev , " error mapping memory \n " ) ;
2012-09-27 14:37:41 +04:00
return - ENOMEM ;
2012-09-04 19:01:14 +04:00
}
/* Determine ID PROM Data Format. If we find the ids "IPAC" or "IPAH"
* we are dealing with a IndustryPack format 1 device . If we detect
* " VITA4 " ( 16 bit big endian formatted ) we are dealing with a
* IndustryPack format 2 device */
if ( ( ioread8 ( idmem + 1 ) = = ' I ' ) & &
( ioread8 ( idmem + 3 ) = = ' P ' ) & &
( ioread8 ( idmem + 5 ) = = ' A ' ) & &
( ( ioread8 ( idmem + 7 ) = = ' C ' ) | |
( ioread8 ( idmem + 7 ) = = ' H ' ) ) ) {
dev - > id_format = IPACK_ID_VERSION_1 ;
dev - > id_avail = ioread8 ( idmem + 0x15 ) ;
if ( ( dev - > id_avail < 0x0c ) | | ( dev - > id_avail > 0x40 ) ) {
dev_warn ( & dev - > dev , " invalid id size " ) ;
dev - > id_avail = 0x0c ;
}
} else if ( ( ioread8 ( idmem + 0 ) = = ' I ' ) & &
( ioread8 ( idmem + 1 ) = = ' V ' ) & &
( ioread8 ( idmem + 2 ) = = ' A ' ) & &
( ioread8 ( idmem + 3 ) = = ' T ' ) & &
( ioread8 ( idmem + 4 ) = = ' ' ) & &
( ioread8 ( idmem + 5 ) = = ' 4 ' ) ) {
dev - > id_format = IPACK_ID_VERSION_2 ;
dev - > id_avail = ioread16be ( idmem + 0x16 ) ;
if ( ( dev - > id_avail < 0x1a ) | | ( dev - > id_avail > 0x40 ) ) {
dev_warn ( & dev - > dev , " invalid id size " ) ;
dev - > id_avail = 0x1a ;
}
} else {
dev - > id_format = IPACK_ID_VERSION_INVALID ;
dev - > id_avail = 0 ;
}
if ( ! dev - > id_avail ) {
ret = - ENODEV ;
goto out ;
}
/* Obtain the amount of memory required to store a copy of the complete
* ID ROM contents */
dev - > id = kmalloc ( dev - > id_avail , GFP_KERNEL ) ;
if ( ! dev - > id ) {
ret = - ENOMEM ;
goto out ;
}
for ( i = 0 ; i < dev - > id_avail ; i + + ) {
if ( dev - > id_format = = IPACK_ID_VERSION_1 )
dev - > id [ i ] = ioread8 ( idmem + ( i < < 1 ) + 1 ) ;
else
dev - > id [ i ] = ioread8 ( idmem + i ) ;
}
2012-09-04 19:01:15 +04:00
/* now we can finally work with the copy */
switch ( dev - > id_format ) {
case IPACK_ID_VERSION_1 :
ipack_parse_id1 ( dev ) ;
break ;
case IPACK_ID_VERSION_2 :
ipack_parse_id2 ( dev ) ;
break ;
}
2012-09-04 19:01:14 +04:00
out :
2012-09-27 14:37:34 +04:00
iounmap ( idmem ) ;
2012-09-04 19:01:14 +04:00
return ret ;
}
2013-03-08 12:21:47 +04:00
int ipack_device_init ( struct ipack_device * dev )
2012-05-09 17:27:19 +04:00
{
int ret ;
dev - > dev . bus = & ipack_bus_type ;
dev - > dev . release = ipack_device_release ;
2012-09-27 14:37:26 +04:00
dev - > dev . parent = dev - > bus - > parent ;
2022-08-05 12:10:57 +03:00
ret = dev_set_name ( & dev - > dev ,
2012-09-27 14:37:25 +04:00
" ipack-dev.%u.%u " , dev - > bus - > bus_nr , dev - > slot ) ;
2022-08-05 12:10:57 +03:00
if ( ret )
return ret ;
2013-03-08 12:21:47 +04:00
device_initialize ( & dev - > dev ) ;
2012-05-09 17:27:19 +04:00
2012-09-27 14:37:26 +04:00
if ( dev - > bus - > ops - > set_clockrate ( dev , 8 ) )
2012-09-11 15:35:01 +04:00
dev_warn ( & dev - > dev , " failed to switch to 8 MHz operation for reading of device ID. \n " ) ;
2012-09-27 14:37:26 +04:00
if ( dev - > bus - > ops - > reset_timeout ( dev ) )
2012-09-11 15:35:02 +04:00
dev_warn ( & dev - > dev , " failed to reset potential timeout. " ) ;
2012-09-11 15:35:01 +04:00
2012-09-04 19:01:14 +04:00
ret = ipack_device_read_id ( dev ) ;
if ( ret < 0 ) {
dev_err ( & dev - > dev , " error reading device id section. \n " ) ;
2012-09-27 14:37:26 +04:00
return ret ;
2012-09-04 19:01:14 +04:00
}
2012-09-11 15:34:58 +04:00
/* if the device supports 32 MHz operation, use it. */
2012-09-11 15:35:01 +04:00
if ( dev - > speed_32mhz ) {
2012-09-27 14:37:26 +04:00
ret = dev - > bus - > ops - > set_clockrate ( dev , 32 ) ;
2012-09-11 15:35:01 +04:00
if ( ret < 0 )
dev_err ( & dev - > dev , " failed to switch to 32 MHz operation. \n " ) ;
}
2012-09-11 15:34:58 +04:00
2013-03-08 12:21:47 +04:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( ipack_device_init ) ;
2012-05-09 17:27:19 +04:00
2013-03-08 12:21:47 +04:00
int ipack_device_add ( struct ipack_device * dev )
{
return device_add ( & dev - > dev ) ;
2012-05-09 17:27:19 +04:00
}
2013-03-08 12:21:47 +04:00
EXPORT_SYMBOL_GPL ( ipack_device_add ) ;
2012-05-09 17:27:19 +04:00
2013-03-08 12:21:47 +04:00
void ipack_device_del ( struct ipack_device * dev )
2012-05-09 17:27:19 +04:00
{
2013-03-08 12:21:47 +04:00
device_del ( & dev - > dev ) ;
ipack_put_device ( dev ) ;
2012-05-09 17:27:19 +04:00
}
2013-03-08 12:21:47 +04:00
EXPORT_SYMBOL_GPL ( ipack_device_del ) ;
2012-05-09 17:27:19 +04:00
2013-03-08 12:21:46 +04:00
void ipack_get_device ( struct ipack_device * dev )
{
get_device ( & dev - > dev ) ;
}
EXPORT_SYMBOL_GPL ( ipack_get_device ) ;
void ipack_put_device ( struct ipack_device * dev )
{
put_device ( & dev - > dev ) ;
}
EXPORT_SYMBOL_GPL ( ipack_put_device ) ;
2012-05-09 17:27:19 +04:00
static int __init ipack_init ( void )
{
2012-05-25 12:03:01 +04:00
ida_init ( & ipack_ida ) ;
2012-05-09 17:27:19 +04:00
return bus_register ( & ipack_bus_type ) ;
}
static void __exit ipack_exit ( void )
{
bus_unregister ( & ipack_bus_type ) ;
2012-05-25 12:03:01 +04:00
ida_destroy ( & ipack_ida ) ;
2012-05-09 17:27:19 +04:00
}
module_init ( ipack_init ) ;
module_exit ( ipack_exit ) ;
MODULE_AUTHOR ( " Samuel Iglesias Gonsalvez <siglesias@igalia.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Industry-pack bus core " ) ;