2018-05-09 12:06:04 -06:00
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright ( c ) 2012 , The Linux Foundation . All rights reserved .
2014-11-03 11:07:35 -07:00
*/
2019-06-19 13:53:06 -06:00
# include <linux/acpi.h>
2014-11-03 11:07:35 -07:00
# include <linux/types.h>
# include <linux/err.h>
# include <linux/slab.h>
# include <linux/clk.h>
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/of_graph.h>
# include <linux/of_platform.h>
# include <linux/platform_device.h>
# include <linux/amba/bus.h>
# include <linux/coresight.h>
2015-03-30 14:13:34 -06:00
# include <linux/cpumask.h>
2014-11-03 11:07:35 -07:00
# include <asm/smp_plat.h>
2019-06-19 13:53:01 -06:00
# include "coresight-priv.h"
2019-06-19 13:52:52 -06:00
/*
* coresight_alloc_conns : Allocate connections record for each output
* port from the device .
*/
static int coresight_alloc_conns ( struct device * dev ,
struct coresight_platform_data * pdata )
{
if ( pdata - > nr_outport ) {
pdata - > conns = devm_kzalloc ( dev , pdata - > nr_outport *
sizeof ( * pdata - > conns ) ,
GFP_KERNEL ) ;
if ( ! pdata - > conns )
return - ENOMEM ;
}
return 0 ;
}
2019-07-12 12:24:03 -07:00
int coresight_device_fwnode_match ( struct device * dev , const void * fwnode )
2014-11-03 11:07:35 -07:00
{
2019-06-19 13:53:02 -06:00
return dev_fwnode ( dev ) = = fwnode ;
2014-11-03 11:07:35 -07:00
}
static struct device *
2019-06-19 13:53:02 -06:00
coresight_find_device_by_fwnode ( struct fwnode_handle * fwnode )
2014-11-03 11:07:35 -07:00
{
struct device * dev = NULL ;
/*
2015-05-19 10:55:20 -06:00
* If we have a non - configurable replicator , it will be found on the
2014-11-03 11:07:35 -07:00
* platform bus .
*/
dev = bus_find_device ( & platform_bus_type , NULL ,
2019-06-19 13:53:02 -06:00
fwnode , coresight_device_fwnode_match ) ;
2014-11-03 11:07:35 -07:00
if ( dev )
return dev ;
/*
* We have a configurable component - circle through the AMBA bus
* looking for the device that matches the endpoint node .
*/
return bus_find_device ( & amba_bustype , NULL ,
2019-06-19 13:53:02 -06:00
fwnode , coresight_device_fwnode_match ) ;
2014-11-03 11:07:35 -07:00
}
2019-06-19 13:53:02 -06:00
# ifdef CONFIG_OF
2018-09-20 13:17:43 -06:00
static inline bool of_coresight_legacy_ep_is_input ( struct device_node * ep )
2018-09-20 13:17:41 -06:00
{
return of_property_read_bool ( ep , " slave-mode " ) ;
}
2018-09-20 13:17:43 -06:00
static void of_coresight_get_ports_legacy ( const struct device_node * node ,
int * nr_inport , int * nr_outport )
2014-11-03 11:07:35 -07:00
{
struct device_node * ep = NULL ;
int in = 0 , out = 0 ;
do {
2014-12-01 13:32:32 +01:00
ep = of_graph_get_next_endpoint ( node , ep ) ;
2014-11-03 11:07:35 -07:00
if ( ! ep )
break ;
2018-09-20 13:17:43 -06:00
if ( of_coresight_legacy_ep_is_input ( ep ) )
2014-11-03 11:07:35 -07:00
in + + ;
else
out + + ;
} while ( ep ) ;
* nr_inport = in ;
* nr_outport = out ;
}
2018-09-20 13:17:43 -06:00
static struct device_node * of_coresight_get_port_parent ( struct device_node * ep )
{
struct device_node * parent = of_graph_get_port_parent ( ep ) ;
/*
* Skip one - level up to the real device node , if we
* are using the new bindings .
*/
2019-02-05 16:24:55 -07:00
if ( of_node_name_eq ( parent , " in-ports " ) | |
of_node_name_eq ( parent , " out-ports " ) )
2018-09-20 13:17:43 -06:00
parent = of_get_next_parent ( parent ) ;
return parent ;
}
static inline struct device_node *
of_coresight_get_input_ports_node ( const struct device_node * node )
{
return of_get_child_by_name ( node , " in-ports " ) ;
}
static inline struct device_node *
of_coresight_get_output_ports_node ( const struct device_node * node )
{
return of_get_child_by_name ( node , " out-ports " ) ;
}
static inline int
of_coresight_count_ports ( struct device_node * port_parent )
{
int i = 0 ;
struct device_node * ep = NULL ;
while ( ( ep = of_graph_get_next_endpoint ( port_parent , ep ) ) )
i + + ;
return i ;
}
static void of_coresight_get_ports ( const struct device_node * node ,
int * nr_inport , int * nr_outport )
{
struct device_node * input_ports = NULL , * output_ports = NULL ;
input_ports = of_coresight_get_input_ports_node ( node ) ;
output_ports = of_coresight_get_output_ports_node ( node ) ;
if ( input_ports | | output_ports ) {
if ( input_ports ) {
* nr_inport = of_coresight_count_ports ( input_ports ) ;
of_node_put ( input_ports ) ;
}
if ( output_ports ) {
* nr_outport = of_coresight_count_ports ( output_ports ) ;
of_node_put ( output_ports ) ;
}
} else {
/* Fall back to legacy DT bindings parsing */
of_coresight_get_ports_legacy ( node , nr_inport , nr_outport ) ;
}
}
2019-06-19 13:52:55 -06:00
static int of_coresight_get_cpu ( struct device * dev )
2017-06-05 14:15:15 -06:00
{
int cpu ;
2018-01-02 11:25:28 +00:00
struct device_node * dn ;
2017-06-05 14:15:15 -06:00
2019-06-19 13:52:55 -06:00
if ( ! dev - > of_node )
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2019-06-19 13:52:55 -06:00
dn = of_parse_phandle ( dev - > of_node , " cpu " , 0 ) ;
2017-06-05 14:15:15 -06:00
if ( ! dn )
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2018-01-02 11:25:28 +00:00
cpu = of_cpu_node_to_id ( dn ) ;
2017-06-05 14:15:15 -06:00
of_node_put ( dn ) ;
2019-07-04 15:23:05 +05:30
return cpu ;
2017-06-05 14:15:15 -06:00
}
2018-09-20 13:17:37 -06:00
/*
* of_coresight_parse_endpoint : Parse the given output endpoint @ ep
2018-09-20 13:17:42 -06:00
* and fill the connection information in @ conn
2018-09-20 13:17:37 -06:00
*
* Parses the local port , remote device name and the remote port .
*
* Returns :
* 1 - If the parsing is successful and a connection record
* was created for an output connection .
* 0 - If the parsing completed without any fatal errors .
* - Errno - Fatal error , abort the scanning .
*/
static int of_coresight_parse_endpoint ( struct device * dev ,
struct device_node * ep ,
2018-09-20 13:17:42 -06:00
struct coresight_connection * conn )
2018-09-20 13:17:37 -06:00
{
int ret = 0 ;
struct of_endpoint endpoint , rendpoint ;
struct device_node * rparent = NULL ;
2018-09-20 13:17:40 -06:00
struct device_node * rep = NULL ;
2018-09-20 13:17:37 -06:00
struct device * rdev = NULL ;
2019-06-19 13:53:02 -06:00
struct fwnode_handle * rdev_fwnode ;
2018-09-20 13:17:37 -06:00
do {
/* Parse the local port details */
if ( of_graph_parse_endpoint ( ep , & endpoint ) )
break ;
/*
2018-09-20 13:17:40 -06:00
* Get a handle on the remote endpoint and the device it is
* attached to .
2018-09-20 13:17:37 -06:00
*/
2018-09-20 13:17:40 -06:00
rep = of_graph_get_remote_endpoint ( ep ) ;
if ( ! rep )
2018-09-20 13:17:37 -06:00
break ;
2018-09-20 13:17:43 -06:00
rparent = of_coresight_get_port_parent ( rep ) ;
2018-09-20 13:17:40 -06:00
if ( ! rparent )
2018-09-20 13:17:37 -06:00
break ;
2018-09-20 13:17:40 -06:00
if ( of_graph_parse_endpoint ( rep , & rendpoint ) )
2018-09-20 13:17:37 -06:00
break ;
2019-06-19 13:53:02 -06:00
rdev_fwnode = of_fwnode_handle ( rparent ) ;
2018-09-20 13:17:37 -06:00
/* If the remote device is not available, defer probing */
2019-06-19 13:53:02 -06:00
rdev = coresight_find_device_by_fwnode ( rdev_fwnode ) ;
2018-09-20 13:17:37 -06:00
if ( ! rdev ) {
ret = - EPROBE_DEFER ;
break ;
}
2018-09-20 13:17:42 -06:00
conn - > outport = endpoint . port ;
2019-06-19 13:53:03 -06:00
/*
* Hold the refcount to the target device . This could be
* released via :
* 1 ) coresight_release_platform_data ( ) if the probe fails or
* this device is unregistered .
* 2 ) While removing the target device via
* coresight_remove_match ( )
*/
conn - > child_fwnode = fwnode_handle_get ( rdev_fwnode ) ;
2018-09-20 13:17:42 -06:00
conn - > child_port = rendpoint . port ;
2018-09-20 13:17:37 -06:00
/* Connection record updated */
ret = 1 ;
} while ( 0 ) ;
2018-09-20 13:18:19 -06:00
of_node_put ( rparent ) ;
of_node_put ( rep ) ;
put_device ( rdev ) ;
2018-09-20 13:17:38 -06:00
2018-09-20 13:17:37 -06:00
return ret ;
}
2019-06-19 13:52:54 -06:00
static int of_get_coresight_platform_data ( struct device * dev ,
struct coresight_platform_data * pdata )
2014-11-03 11:07:35 -07:00
{
2018-09-20 13:17:42 -06:00
int ret = 0 ;
struct coresight_connection * conn ;
2014-11-03 11:07:35 -07:00
struct device_node * ep = NULL ;
2018-09-20 13:17:43 -06:00
const struct device_node * parent = NULL ;
bool legacy_binding = false ;
2019-06-19 13:52:54 -06:00
struct device_node * node = dev - > of_node ;
2014-11-03 11:07:35 -07:00
/* Get the number of input and output port for this component */
of_coresight_get_ports ( node , & pdata - > nr_inport , & pdata - > nr_outport ) ;
2018-09-20 13:17:37 -06:00
/* If there are no output connections, we are done */
if ( ! pdata - > nr_outport )
2019-06-19 13:52:54 -06:00
return 0 ;
2018-09-20 13:17:37 -06:00
2019-06-19 13:52:52 -06:00
ret = coresight_alloc_conns ( dev , pdata ) ;
2018-09-20 13:17:37 -06:00
if ( ret )
2019-06-19 13:52:54 -06:00
return ret ;
2018-09-20 13:17:37 -06:00
2018-09-20 13:17:43 -06:00
parent = of_coresight_get_output_ports_node ( node ) ;
/*
* If the DT uses obsoleted bindings , the ports are listed
* under the device and we need to filter out the input
* ports .
*/
if ( ! parent ) {
legacy_binding = true ;
parent = node ;
dev_warn_once ( dev , " Uses obsolete Coresight DT bindings \n " ) ;
}
2018-09-20 13:17:42 -06:00
conn = pdata - > conns ;
2018-09-20 13:17:43 -06:00
/* Iterate through each output port to discover topology */
while ( ( ep = of_graph_get_next_endpoint ( parent , ep ) ) ) {
2018-09-20 13:17:37 -06:00
/*
2018-09-20 13:17:43 -06:00
* Legacy binding mixes input / output ports under the
* same parent . So , skip the input ports if we are dealing
* with legacy binding , as they processed with their
* connected output ports .
2018-09-20 13:17:37 -06:00
*/
2018-09-20 13:17:43 -06:00
if ( legacy_binding & & of_coresight_legacy_ep_is_input ( ep ) )
2018-09-20 13:17:37 -06:00
continue ;
2018-09-20 13:17:42 -06:00
ret = of_coresight_parse_endpoint ( dev , ep , conn ) ;
2018-09-20 13:17:37 -06:00
switch ( ret ) {
case 1 :
2018-09-20 13:17:42 -06:00
conn + + ; /* Fall through */
2018-09-20 13:17:37 -06:00
case 0 :
break ;
default :
2019-06-19 13:52:54 -06:00
return ret ;
2018-09-20 13:17:37 -06:00
}
2014-11-03 11:07:35 -07:00
}
2019-06-19 13:52:54 -06:00
return 0 ;
}
# else
static inline int
of_get_coresight_platform_data ( struct device * dev ,
struct coresight_platform_data * pdata )
{
return - ENOENT ;
2014-11-03 11:07:35 -07:00
}
2019-06-19 13:53:09 -06:00
static inline int of_coresight_get_cpu ( struct device * dev )
{
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2019-06-19 13:53:09 -06:00
}
2019-06-19 13:52:51 -06:00
# endif
2019-06-19 13:52:54 -06:00
2019-06-19 13:53:06 -06:00
# ifdef CONFIG_ACPI
# include <acpi/actypes.h>
# include <acpi/processor.h>
/* ACPI Graph _DSD UUID : "ab02a46b-74c7-45a2-bd68-f7d344ef2153" */
static const guid_t acpi_graph_uuid = GUID_INIT ( 0xab02a46b , 0x74c7 , 0x45a2 ,
0xbd , 0x68 , 0xf7 , 0xd3 ,
0x44 , 0xef , 0x21 , 0x53 ) ;
/* Coresight ACPI Graph UUID : "3ecbc8b6-1d0e-4fb3-8107-e627f805c6cd" */
static const guid_t coresight_graph_uuid = GUID_INIT ( 0x3ecbc8b6 , 0x1d0e , 0x4fb3 ,
0x81 , 0x07 , 0xe6 , 0x27 ,
0xf8 , 0x05 , 0xc6 , 0xcd ) ;
# define ACPI_CORESIGHT_LINK_SLAVE 0
# define ACPI_CORESIGHT_LINK_MASTER 1
static inline bool is_acpi_guid ( const union acpi_object * obj )
{
return ( obj - > type = = ACPI_TYPE_BUFFER ) & & ( obj - > buffer . length = = 16 ) ;
}
/*
* acpi_guid_matches - Checks if the given object is a GUID object and
* that it matches the supplied the GUID .
*/
static inline bool acpi_guid_matches ( const union acpi_object * obj ,
const guid_t * guid )
{
return is_acpi_guid ( obj ) & &
guid_equal ( ( guid_t * ) obj - > buffer . pointer , guid ) ;
}
static inline bool is_acpi_dsd_graph_guid ( const union acpi_object * obj )
{
return acpi_guid_matches ( obj , & acpi_graph_uuid ) ;
}
static inline bool is_acpi_coresight_graph_guid ( const union acpi_object * obj )
{
return acpi_guid_matches ( obj , & coresight_graph_uuid ) ;
}
static inline bool is_acpi_coresight_graph ( const union acpi_object * obj )
{
const union acpi_object * graphid , * guid , * links ;
if ( obj - > type ! = ACPI_TYPE_PACKAGE | |
obj - > package . count < 3 )
return false ;
graphid = & obj - > package . elements [ 0 ] ;
guid = & obj - > package . elements [ 1 ] ;
links = & obj - > package . elements [ 2 ] ;
if ( graphid - > type ! = ACPI_TYPE_INTEGER | |
links - > type ! = ACPI_TYPE_INTEGER )
return false ;
return is_acpi_coresight_graph_guid ( guid ) ;
}
/*
* acpi_validate_dsd_graph - Make sure the given _DSD graph conforms
* to the ACPI _DSD Graph specification .
*
* ACPI Devices Graph property has the following format :
* {
* Revision - Integer , must be 0
* NumberOfGraphs - Integer , N indicating the following list .
* Graph [ 1 ] ,
* . . .
* Graph [ N ]
* }
*
* And each Graph entry has the following format :
* {
* GraphID - Integer , identifying a graph the device belongs to .
* UUID - UUID identifying the specification that governs
* this graph . ( e . g , see is_acpi_coresight_graph ( ) )
* NumberOfLinks - Number " N " of connections on this node of the graph .
* Links [ 1 ]
* . . .
* Links [ N ]
* }
*
* Where each " Links " entry has the following format :
*
* {
* SourcePortAddress - Integer
* DestinationPortAddress - Integer
* DestinationDeviceName - Reference to another device
* ( - - - CoreSight specific extensions below - - - )
* DirectionOfFlow - Integer 1 for output ( master )
* 0 for input ( slave )
* }
*
* e . g :
* For a Funnel device
*
* Device ( MFUN ) {
* . . .
*
* Name ( _DSD , Package ( ) {
* // DSD Package contains tuples of { Proeprty_Type_UUID, Package() }
* ToUUID ( " daffd814-6eba-4d8c-8a91-bc9bbf4aa301 " ) , //Std. Property UUID
* Package ( ) {
* Package ( 2 ) { " property-name " , < property - value > }
* } ,
*
* ToUUID ( " ab02a46b-74c7-45a2-bd68-f7d344ef2153 " ) , // ACPI Graph UUID
* Package ( ) {
* 0 , // Revision
* 1 , // NumberOfGraphs.
* Package ( ) { // Graph[0] Package
* 1 , // GraphID
* // Coresight Graph UUID
* ToUUID ( " 3ecbc8b6-1d0e-4fb3-8107-e627f805c6cd " ) ,
* 3 , // NumberOfLinks aka ports
* // Link[0]: Output_0 -> Replicator:Input_0
* Package ( ) { 0 , 0 , \ _SB_ . RPL0 , 1 } ,
* // Link[1]: Input_0 <- Cluster0_Funnel0:Output_0
* Package ( ) { 0 , 0 , \ _SB_ . CLU0 . FUN0 , 0 } ,
* // Link[2]: Input_1 <- Cluster1_Funnel0:Output_0
* Package ( ) { 1 , 0 , \ _SB_ . CLU1 . FUN0 , 0 } ,
* } // End of Graph[0] Package
*
* } , // End of ACPI Graph Property
* } )
*/
static inline bool acpi_validate_dsd_graph ( const union acpi_object * graph )
{
int i , n ;
const union acpi_object * rev , * nr_graphs ;
/* The graph must contain at least the Revision and Number of Graphs */
if ( graph - > package . count < 2 )
return false ;
rev = & graph - > package . elements [ 0 ] ;
nr_graphs = & graph - > package . elements [ 1 ] ;
if ( rev - > type ! = ACPI_TYPE_INTEGER | |
nr_graphs - > type ! = ACPI_TYPE_INTEGER )
return false ;
/* We only support revision 0 */
if ( rev - > integer . value ! = 0 )
return false ;
n = nr_graphs - > integer . value ;
/* CoreSight devices are only part of a single Graph */
if ( n ! = 1 )
return false ;
/* Make sure the ACPI graph package has right number of elements */
if ( graph - > package . count ! = ( n + 2 ) )
return false ;
/*
* Each entry must be a graph package with at least 3 members :
* { GraphID , UUID , NumberOfLinks ( n ) , Links [ . ] , . . . }
*/
for ( i = 2 ; i < n + 2 ; i + + ) {
const union acpi_object * obj = & graph - > package . elements [ i ] ;
if ( obj - > type ! = ACPI_TYPE_PACKAGE | |
obj - > package . count < 3 )
return false ;
}
return true ;
}
/* acpi_get_dsd_graph - Find the _DSD Graph property for the given device. */
const union acpi_object *
acpi_get_dsd_graph ( struct acpi_device * adev )
{
int i ;
struct acpi_buffer buf = { ACPI_ALLOCATE_BUFFER } ;
acpi_status status ;
const union acpi_object * dsd ;
status = acpi_evaluate_object_typed ( adev - > handle , " _DSD " , NULL ,
& buf , ACPI_TYPE_PACKAGE ) ;
if ( ACPI_FAILURE ( status ) )
return NULL ;
dsd = buf . pointer ;
/*
* _DSD property consists tuples { Prop_UUID , Package ( ) }
* Iterate through all the packages and find the Graph .
*/
for ( i = 0 ; i + 1 < dsd - > package . count ; i + = 2 ) {
const union acpi_object * guid , * package ;
guid = & dsd - > package . elements [ i ] ;
package = & dsd - > package . elements [ i + 1 ] ;
/* All _DSD elements must have a UUID and a Package */
if ( ! is_acpi_guid ( guid ) | | package - > type ! = ACPI_TYPE_PACKAGE )
break ;
/* Skip the non-Graph _DSD packages */
if ( ! is_acpi_dsd_graph_guid ( guid ) )
continue ;
if ( acpi_validate_dsd_graph ( package ) )
return package ;
/* Invalid graph format, continue */
dev_warn ( & adev - > dev , " Invalid Graph _DSD property \n " ) ;
}
return NULL ;
}
static inline bool
acpi_validate_coresight_graph ( const union acpi_object * cs_graph )
{
int nlinks ;
nlinks = cs_graph - > package . elements [ 2 ] . integer . value ;
/*
* Graph must have the following fields :
* { GraphID , GraphUUID , NumberOfLinks , Links . . . }
*/
if ( cs_graph - > package . count ! = ( nlinks + 3 ) )
return false ;
/* The links are validated in acpi_coresight_parse_link() */
return true ;
}
/*
* acpi_get_coresight_graph - Parse the device _DSD tables and find
* the Graph property matching the CoreSight Graphs .
*
* Returns the pointer to the CoreSight Graph Package when found . Otherwise
* returns NULL .
*/
const union acpi_object *
acpi_get_coresight_graph ( struct acpi_device * adev )
{
const union acpi_object * graph_list , * graph ;
int i , nr_graphs ;
graph_list = acpi_get_dsd_graph ( adev ) ;
if ( ! graph_list )
return graph_list ;
nr_graphs = graph_list - > package . elements [ 1 ] . integer . value ;
for ( i = 2 ; i < nr_graphs + 2 ; i + + ) {
graph = & graph_list - > package . elements [ i ] ;
if ( ! is_acpi_coresight_graph ( graph ) )
continue ;
if ( acpi_validate_coresight_graph ( graph ) )
return graph ;
/* Invalid graph format */
break ;
}
return NULL ;
}
/*
* acpi_coresight_parse_link - Parse the given Graph connection
* of the device and populate the coresight_connection for an output
* connection .
*
* CoreSight Graph specification mandates that the direction of the data
* flow must be specified in the link . i . e ,
*
* SourcePortAddress , // Integer
* DestinationPortAddress , // Integer
* DestinationDeviceName , // Reference to another device
* DirectionOfFlow , // 1 for output(master), 0 for input(slave)
*
* Returns the direction of the data flow [ Input ( slave ) or Output ( master ) ]
* upon success .
* Returns an negative error number otherwise .
*/
static int acpi_coresight_parse_link ( struct acpi_device * adev ,
const union acpi_object * link ,
struct coresight_connection * conn )
{
int rc , dir ;
const union acpi_object * fields ;
struct acpi_device * r_adev ;
struct device * rdev ;
if ( link - > type ! = ACPI_TYPE_PACKAGE | |
link - > package . count ! = 4 )
return - EINVAL ;
fields = link - > package . elements ;
if ( fields [ 0 ] . type ! = ACPI_TYPE_INTEGER | |
fields [ 1 ] . type ! = ACPI_TYPE_INTEGER | |
fields [ 2 ] . type ! = ACPI_TYPE_LOCAL_REFERENCE | |
fields [ 3 ] . type ! = ACPI_TYPE_INTEGER )
return - EINVAL ;
rc = acpi_bus_get_device ( fields [ 2 ] . reference . handle , & r_adev ) ;
if ( rc )
return rc ;
dir = fields [ 3 ] . integer . value ;
if ( dir = = ACPI_CORESIGHT_LINK_MASTER ) {
conn - > outport = fields [ 0 ] . integer . value ;
conn - > child_port = fields [ 1 ] . integer . value ;
rdev = coresight_find_device_by_fwnode ( & r_adev - > fwnode ) ;
if ( ! rdev )
return - EPROBE_DEFER ;
/*
* Hold the refcount to the target device . This could be
* released via :
* 1 ) coresight_release_platform_data ( ) if the probe fails or
* this device is unregistered .
* 2 ) While removing the target device via
* coresight_remove_match ( ) .
*/
conn - > child_fwnode = fwnode_handle_get ( & r_adev - > fwnode ) ;
}
return dir ;
}
/*
* acpi_coresight_parse_graph - Parse the _DSD CoreSight graph
* connection information and populate the supplied coresight_platform_data
* instance .
*/
static int acpi_coresight_parse_graph ( struct acpi_device * adev ,
struct coresight_platform_data * pdata )
{
int rc , i , nlinks ;
const union acpi_object * graph ;
struct coresight_connection * conns , * ptr ;
pdata - > nr_inport = pdata - > nr_outport = 0 ;
graph = acpi_get_coresight_graph ( adev ) ;
if ( ! graph )
return - ENOENT ;
nlinks = graph - > package . elements [ 2 ] . integer . value ;
if ( ! nlinks )
return 0 ;
/*
* To avoid scanning the table twice ( once for finding the number of
* output links and then later for parsing the output links ) ,
* cache the links information in one go and then later copy
* it to the pdata .
*/
conns = devm_kcalloc ( & adev - > dev , nlinks , sizeof ( * conns ) , GFP_KERNEL ) ;
if ( ! conns )
return - ENOMEM ;
ptr = conns ;
for ( i = 0 ; i < nlinks ; i + + ) {
const union acpi_object * link = & graph - > package . elements [ 3 + i ] ;
int dir ;
dir = acpi_coresight_parse_link ( adev , link , ptr ) ;
if ( dir < 0 )
return dir ;
if ( dir = = ACPI_CORESIGHT_LINK_MASTER ) {
pdata - > nr_outport + + ;
ptr + + ;
} else {
pdata - > nr_inport + + ;
}
}
rc = coresight_alloc_conns ( & adev - > dev , pdata ) ;
if ( rc )
return rc ;
/* Copy the connection information to the final location */
for ( i = 0 ; i < pdata - > nr_outport ; i + + )
pdata - > conns [ i ] = conns [ i ] ;
devm_kfree ( & adev - > dev , conns ) ;
return 0 ;
}
/*
* acpi_handle_to_logical_cpuid - Map a given acpi_handle to the
* logical CPU id of the corresponding CPU device .
*
* Returns the logical CPU id when found . Otherwise returns > = nr_cpus_id .
*/
static int
acpi_handle_to_logical_cpuid ( acpi_handle handle )
{
int i ;
struct acpi_processor * pr ;
for_each_possible_cpu ( i ) {
pr = per_cpu ( processors , i ) ;
if ( pr & & pr - > handle = = handle )
break ;
}
return i ;
}
/*
* acpi_coresigh_get_cpu - Find the logical CPU id of the CPU associated
* with this coresight device . With ACPI bindings , the CoreSight components
* are listed as child device of the associated CPU .
*
* Returns the logical CPU id when found . Otherwise returns 0.
*/
static int acpi_coresight_get_cpu ( struct device * dev )
{
int cpu ;
acpi_handle cpu_handle ;
acpi_status status ;
struct acpi_device * adev = ACPI_COMPANION ( dev ) ;
if ( ! adev )
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2019-06-19 13:53:06 -06:00
status = acpi_get_parent ( adev - > handle , & cpu_handle ) ;
if ( ACPI_FAILURE ( status ) )
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2019-06-19 13:53:06 -06:00
cpu = acpi_handle_to_logical_cpuid ( cpu_handle ) ;
if ( cpu > = nr_cpu_ids )
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2019-06-19 13:53:06 -06:00
return cpu ;
}
static int
acpi_get_coresight_platform_data ( struct device * dev ,
struct coresight_platform_data * pdata )
{
struct acpi_device * adev ;
adev = ACPI_COMPANION ( dev ) ;
if ( ! adev )
return - EINVAL ;
return acpi_coresight_parse_graph ( adev , pdata ) ;
}
# else
static inline int
acpi_get_coresight_platform_data ( struct device * dev ,
struct coresight_platform_data * pdata )
{
return - ENOENT ;
}
static inline int acpi_coresight_get_cpu ( struct device * dev )
{
2019-07-04 15:23:05 +05:30
return - ENODEV ;
2019-06-19 13:53:06 -06:00
}
# endif
2019-06-19 13:52:55 -06:00
int coresight_get_cpu ( struct device * dev )
{
if ( is_of_node ( dev - > fwnode ) )
return of_coresight_get_cpu ( dev ) ;
2019-06-19 13:53:06 -06:00
else if ( is_acpi_device_node ( dev - > fwnode ) )
return acpi_coresight_get_cpu ( dev ) ;
2019-06-19 13:52:55 -06:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( coresight_get_cpu ) ;
2019-06-19 13:52:54 -06:00
struct coresight_platform_data *
coresight_get_platform_data ( struct device * dev )
{
int ret = - ENOENT ;
2019-06-19 13:53:01 -06:00
struct coresight_platform_data * pdata = NULL ;
2019-06-19 13:52:54 -06:00
struct fwnode_handle * fwnode = dev_fwnode ( dev ) ;
if ( IS_ERR_OR_NULL ( fwnode ) )
goto error ;
pdata = devm_kzalloc ( dev , sizeof ( * pdata ) , GFP_KERNEL ) ;
if ( ! pdata ) {
ret = - ENOMEM ;
goto error ;
}
if ( is_of_node ( fwnode ) )
ret = of_get_coresight_platform_data ( dev , pdata ) ;
2019-06-19 13:53:06 -06:00
else if ( is_acpi_device_node ( fwnode ) )
ret = acpi_get_coresight_platform_data ( dev , pdata ) ;
2019-06-19 13:52:54 -06:00
if ( ! ret )
return pdata ;
error :
2019-06-19 13:53:01 -06:00
if ( ! IS_ERR_OR_NULL ( pdata ) )
/* Cleanup the connection information */
coresight_release_platform_data ( pdata ) ;
2019-06-19 13:52:54 -06:00
return ERR_PTR ( ret ) ;
}
EXPORT_SYMBOL_GPL ( coresight_get_platform_data ) ;