2015-05-13 15:26:42 +03:00
/**
* ulpi . c - USB ULPI PHY bus
*
* Copyright ( C ) 2015 Intel Corporation
*
* Author : Heikki Krogerus < heikki . krogerus @ linux . intel . com >
*
* 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 .
*/
# include <linux/ulpi/interface.h>
# include <linux/ulpi/driver.h>
# include <linux/ulpi/regs.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/acpi.h>
2016-12-28 14:56:49 -08:00
# include <linux/of.h>
# include <linux/of_device.h>
# include <linux/clk/clk-conf.h>
2015-05-13 15:26:42 +03:00
/* -------------------------------------------------------------------------- */
int ulpi_read ( struct ulpi * ulpi , u8 addr )
{
2016-08-16 19:04:50 +03:00
return ulpi - > ops - > read ( ulpi - > dev . parent , addr ) ;
2015-05-13 15:26:42 +03:00
}
EXPORT_SYMBOL_GPL ( ulpi_read ) ;
int ulpi_write ( struct ulpi * ulpi , u8 addr , u8 val )
{
2016-08-16 19:04:50 +03:00
return ulpi - > ops - > write ( ulpi - > dev . parent , addr , val ) ;
2015-05-13 15:26:42 +03:00
}
EXPORT_SYMBOL_GPL ( ulpi_write ) ;
/* -------------------------------------------------------------------------- */
static int ulpi_match ( struct device * dev , struct device_driver * driver )
{
struct ulpi_driver * drv = to_ulpi_driver ( driver ) ;
struct ulpi * ulpi = to_ulpi_dev ( dev ) ;
const struct ulpi_device_id * id ;
2016-12-28 14:56:49 -08:00
/* Some ULPI devices don't have a vendor id so rely on OF match */
if ( ulpi - > id . vendor = = 0 )
return of_driver_match_device ( dev , driver ) ;
2015-05-13 15:26:42 +03:00
for ( id = drv - > id_table ; id - > vendor ; id + + )
if ( id - > vendor = = ulpi - > id . vendor & &
id - > product = = ulpi - > id . product )
return 1 ;
return 0 ;
}
static int ulpi_uevent ( struct device * dev , struct kobj_uevent_env * env )
{
struct ulpi * ulpi = to_ulpi_dev ( dev ) ;
2016-12-28 14:56:49 -08:00
int ret ;
ret = of_device_uevent_modalias ( dev , env ) ;
if ( ret ! = - ENODEV )
return ret ;
2015-05-13 15:26:42 +03:00
if ( add_uevent_var ( env , " MODALIAS=ulpi:v%04xp%04x " ,
ulpi - > id . vendor , ulpi - > id . product ) )
return - ENOMEM ;
return 0 ;
}
static int ulpi_probe ( struct device * dev )
{
struct ulpi_driver * drv = to_ulpi_driver ( dev - > driver ) ;
2016-12-28 14:56:49 -08:00
int ret ;
ret = of_clk_set_defaults ( dev - > of_node , false ) ;
if ( ret < 0 )
return ret ;
2015-05-13 15:26:42 +03:00
return drv - > probe ( to_ulpi_dev ( dev ) ) ;
}
static int ulpi_remove ( struct device * dev )
{
struct ulpi_driver * drv = to_ulpi_driver ( dev - > driver ) ;
if ( drv - > remove )
drv - > remove ( to_ulpi_dev ( dev ) ) ;
return 0 ;
}
static struct bus_type ulpi_bus = {
. name = " ulpi " ,
. match = ulpi_match ,
. uevent = ulpi_uevent ,
. probe = ulpi_probe ,
. remove = ulpi_remove ,
} ;
/* -------------------------------------------------------------------------- */
static ssize_t modalias_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
2016-12-28 14:56:49 -08:00
int len ;
2015-05-13 15:26:42 +03:00
struct ulpi * ulpi = to_ulpi_dev ( dev ) ;
2017-03-22 09:16:27 -05:00
len = of_device_modalias ( dev , buf , PAGE_SIZE ) ;
2016-12-28 14:56:49 -08:00
if ( len ! = - ENODEV )
return len ;
2015-05-13 15:26:42 +03:00
return sprintf ( buf , " ulpi:v%04xp%04x \n " ,
ulpi - > id . vendor , ulpi - > id . product ) ;
}
static DEVICE_ATTR_RO ( modalias ) ;
static struct attribute * ulpi_dev_attrs [ ] = {
& dev_attr_modalias . attr ,
NULL
} ;
static struct attribute_group ulpi_dev_attr_group = {
. attrs = ulpi_dev_attrs ,
} ;
static const struct attribute_group * ulpi_dev_attr_groups [ ] = {
& ulpi_dev_attr_group ,
NULL
} ;
static void ulpi_dev_release ( struct device * dev )
{
kfree ( to_ulpi_dev ( dev ) ) ;
}
static struct device_type ulpi_dev_type = {
. name = " ulpi_device " ,
. groups = ulpi_dev_attr_groups ,
. release = ulpi_dev_release ,
} ;
/* -------------------------------------------------------------------------- */
/**
* ulpi_register_driver - register a driver with the ULPI bus
* @ drv : driver being registered
*
* Registers a driver with the ULPI bus .
*/
2016-06-25 22:38:21 -07:00
int __ulpi_register_driver ( struct ulpi_driver * drv , struct module * module )
2015-05-13 15:26:42 +03:00
{
if ( ! drv - > probe )
return - EINVAL ;
2016-06-25 22:38:21 -07:00
drv - > driver . owner = module ;
2015-05-13 15:26:42 +03:00
drv - > driver . bus = & ulpi_bus ;
return driver_register ( & drv - > driver ) ;
}
2016-06-25 22:38:21 -07:00
EXPORT_SYMBOL_GPL ( __ulpi_register_driver ) ;
2015-05-13 15:26:42 +03:00
/**
* ulpi_unregister_driver - unregister a driver with the ULPI bus
* @ drv : driver to unregister
*
* Unregisters a driver with the ULPI bus .
*/
void ulpi_unregister_driver ( struct ulpi_driver * drv )
{
driver_unregister ( & drv - > driver ) ;
}
EXPORT_SYMBOL_GPL ( ulpi_unregister_driver ) ;
/* -------------------------------------------------------------------------- */
2016-12-28 14:56:49 -08:00
static int ulpi_of_register ( struct ulpi * ulpi )
2015-05-13 15:26:42 +03:00
{
2016-12-28 14:56:49 -08:00
struct device_node * np = NULL , * child ;
struct device * parent ;
/* Find a ulpi bus underneath the parent or the grandparent */
parent = ulpi - > dev . parent ;
if ( parent - > of_node )
np = of_find_node_by_name ( parent - > of_node , " ulpi " ) ;
else if ( parent - > parent & & parent - > parent - > of_node )
np = of_find_node_by_name ( parent - > parent - > of_node , " ulpi " ) ;
if ( ! np )
return 0 ;
child = of_get_next_available_child ( np , NULL ) ;
of_node_put ( np ) ;
if ( ! child )
return - EINVAL ;
2015-05-13 15:26:42 +03:00
2016-12-28 14:56:49 -08:00
ulpi - > dev . of_node = child ;
return 0 ;
}
static int ulpi_read_id ( struct ulpi * ulpi )
{
int ret ;
2016-08-16 19:04:46 +03:00
2015-05-13 15:26:42 +03:00
/* Test the interface */
ret = ulpi_write ( ulpi , ULPI_SCRATCH , 0xaa ) ;
if ( ret < 0 )
2016-12-28 14:56:49 -08:00
goto err ;
2015-05-13 15:26:42 +03:00
ret = ulpi_read ( ulpi , ULPI_SCRATCH ) ;
if ( ret < 0 )
return ret ;
if ( ret ! = 0xaa )
2016-12-28 14:56:49 -08:00
goto err ;
2015-05-13 15:26:42 +03:00
ulpi - > id . vendor = ulpi_read ( ulpi , ULPI_VENDOR_ID_LOW ) ;
ulpi - > id . vendor | = ulpi_read ( ulpi , ULPI_VENDOR_ID_HIGH ) < < 8 ;
ulpi - > id . product = ulpi_read ( ulpi , ULPI_PRODUCT_ID_LOW ) ;
ulpi - > id . product | = ulpi_read ( ulpi , ULPI_PRODUCT_ID_HIGH ) < < 8 ;
2016-12-28 14:56:49 -08:00
/* Some ULPI devices don't have a vendor id so rely on OF match */
if ( ulpi - > id . vendor = = 0 )
goto err ;
request_module ( " ulpi:v%04xp%04x " , ulpi - > id . vendor , ulpi - > id . product ) ;
return 0 ;
err :
of_device_request_module ( & ulpi - > dev ) ;
return 0 ;
}
static int ulpi_register ( struct device * dev , struct ulpi * ulpi )
{
int ret ;
ulpi - > dev . parent = dev ; /* needed early for ops */
2015-05-13 15:26:42 +03:00
ulpi - > dev . bus = & ulpi_bus ;
ulpi - > dev . type = & ulpi_dev_type ;
dev_set_name ( & ulpi - > dev , " %s.ulpi " , dev_name ( dev ) ) ;
ACPI_COMPANION_SET ( & ulpi - > dev , ACPI_COMPANION ( dev ) ) ;
2016-12-28 14:56:49 -08:00
ret = ulpi_of_register ( ulpi ) ;
if ( ret )
return ret ;
ret = ulpi_read_id ( ulpi ) ;
if ( ret )
return ret ;
2015-05-13 15:26:42 +03:00
ret = device_register ( & ulpi - > dev ) ;
if ( ret )
return ret ;
dev_dbg ( & ulpi - > dev , " registered ULPI PHY: vendor %04x, product %04x \n " ,
ulpi - > id . vendor , ulpi - > id . product ) ;
return 0 ;
}
/**
* ulpi_register_interface - instantiate new ULPI device
* @ dev : USB controller ' s device interface
* @ ops : ULPI register access
*
* Allocates and registers a ULPI device and an interface for it . Called from
* the USB controller that provides the ULPI interface .
*/
2016-08-16 19:04:52 +03:00
struct ulpi * ulpi_register_interface ( struct device * dev ,
const struct ulpi_ops * ops )
2015-05-13 15:26:42 +03:00
{
struct ulpi * ulpi ;
int ret ;
ulpi = kzalloc ( sizeof ( * ulpi ) , GFP_KERNEL ) ;
if ( ! ulpi )
return ERR_PTR ( - ENOMEM ) ;
ulpi - > ops = ops ;
ret = ulpi_register ( dev , ulpi ) ;
if ( ret ) {
kfree ( ulpi ) ;
return ERR_PTR ( ret ) ;
}
return ulpi ;
}
EXPORT_SYMBOL_GPL ( ulpi_register_interface ) ;
/**
* ulpi_unregister_interface - unregister ULPI interface
* @ intrf : struct ulpi_interface
*
* Unregisters a ULPI device and it ' s interface that was created with
* ulpi_create_interface ( ) .
*/
void ulpi_unregister_interface ( struct ulpi * ulpi )
{
2016-12-28 14:56:49 -08:00
of_node_put ( ulpi - > dev . of_node ) ;
2015-05-13 15:26:42 +03:00
device_unregister ( & ulpi - > dev ) ;
}
EXPORT_SYMBOL_GPL ( ulpi_unregister_interface ) ;
/* -------------------------------------------------------------------------- */
static int __init ulpi_init ( void )
{
return bus_register ( & ulpi_bus ) ;
}
2015-06-04 20:44:14 +08:00
subsys_initcall ( ulpi_init ) ;
2015-05-13 15:26:42 +03:00
static void __exit ulpi_exit ( void )
{
bus_unregister ( & ulpi_bus ) ;
}
module_exit ( ulpi_exit ) ;
MODULE_AUTHOR ( " Intel Corporation " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_DESCRIPTION ( " USB ULPI PHY bus " ) ;