2015-09-22 15:47:14 +03:00
/*
* Intel ( R ) Trace Hub data structures
*
* Copyright ( C ) 2014 - 2015 Intel Corporation .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*/
# ifndef __INTEL_TH_H__
# define __INTEL_TH_H__
/* intel_th_device device types */
enum {
/* Devices that generate trace data */
INTEL_TH_SOURCE = 0 ,
/* Output ports (MSC, PTI) */
INTEL_TH_OUTPUT ,
/* Switch, the Global Trace Hub (GTH) */
INTEL_TH_SWITCH ,
} ;
/**
* struct intel_th_output - descriptor INTEL_TH_OUTPUT type devices
* @ port : output port number , assigned by the switch
* @ type : GTH_ { MSU , CTP , PTI }
2016-02-15 19:11:55 +02:00
* @ scratchpad : scratchpad bits to flag when this output is enabled
2015-09-22 15:47:14 +03:00
* @ multiblock : true for multiblock output configuration
* @ active : true when this output is enabled
*
* Output port descriptor , used by switch driver to tell which output
* port this output device corresponds to . Filled in at output device ' s
* probe time by switch : : assign ( ) . Passed from output device driver to
* switch related code to enable / disable its port .
*/
struct intel_th_output {
int port ;
unsigned int type ;
2016-02-15 19:11:55 +02:00
unsigned int scratchpad ;
2015-09-22 15:47:14 +03:00
bool multiblock ;
bool active ;
} ;
2017-08-18 17:57:35 +03:00
/**
* struct intel_th_drvdata - describes hardware capabilities and quirks
* @ tscu_enable : device needs SW to enable time stamping unit
*/
struct intel_th_drvdata {
unsigned int tscu_enable : 1 ;
} ;
# define INTEL_TH_CAP(_th, _cap) ((_th)->drvdata ? (_th)->drvdata->_cap : 0)
2015-09-22 15:47:14 +03:00
/**
* struct intel_th_device - device on the intel_th bus
* @ dev : device
2017-08-18 17:57:35 +03:00
* @ drvdata : hardware capabilities / quirks
2015-09-22 15:47:14 +03:00
* @ resource : array of resources available to this device
* @ num_resources : number of resources in @ resource array
* @ type : INTEL_TH_ { SOURCE , OUTPUT , SWITCH }
* @ id : device instance or - 1
2016-09-19 17:07:47 +03:00
* @ host_mode : Intel TH is controlled by an external debug host
2015-09-22 15:47:14 +03:00
* @ output : output descriptor for INTEL_TH_OUTPUT devices
* @ name : device name to match the driver
*/
struct intel_th_device {
2017-08-18 17:57:35 +03:00
struct device dev ;
struct intel_th_drvdata * drvdata ;
struct resource * resource ;
unsigned int num_resources ;
unsigned int type ;
int id ;
2015-09-22 15:47:14 +03:00
2016-09-19 17:07:47 +03:00
/* INTEL_TH_SWITCH specific */
bool host_mode ;
2015-09-22 15:47:14 +03:00
/* INTEL_TH_OUTPUT specific */
struct intel_th_output output ;
char name [ ] ;
} ;
# define to_intel_th_device(_d) \
container_of ( ( _d ) , struct intel_th_device , dev )
/**
* intel_th_device_get_resource ( ) - obtain @ num ' th resource of type @ type
* @ thdev : the device to search the resource for
* @ type : resource type
* @ num : number of the resource
*/
static inline struct resource *
intel_th_device_get_resource ( struct intel_th_device * thdev , unsigned int type ,
unsigned int num )
{
int i ;
for ( i = 0 ; i < thdev - > num_resources ; i + + )
if ( resource_type ( & thdev - > resource [ i ] ) = = type & & ! num - - )
return & thdev - > resource [ i ] ;
return NULL ;
}
2016-11-18 14:51:05 +02:00
/*
* GTH , output ports configuration
*/
enum {
GTH_NONE = 0 ,
GTH_MSU , /* memory/usb */
GTH_CTP , /* Common Trace Port */
2016-11-11 12:09:11 +02:00
GTH_LPP , /* Low Power Path */
GTH_PTI , /* MIPI-PTI */
2016-11-18 14:51:05 +02:00
} ;
2015-09-22 15:47:14 +03:00
/**
* intel_th_output_assigned ( ) - if an output device is assigned to a switch port
* @ thdev : the output device
*
* Return : true if the device is INTEL_TH_OUTPUT * and * is assigned a port
*/
static inline bool
intel_th_output_assigned ( struct intel_th_device * thdev )
{
return thdev - > type = = INTEL_TH_OUTPUT & &
2016-11-18 14:51:05 +02:00
( thdev - > output . port > = 0 | |
thdev - > output . type = = GTH_NONE ) ;
2015-09-22 15:47:14 +03:00
}
/**
* struct intel_th_driver - driver for an intel_th_device device
* @ driver : generic driver
* @ probe : probe method
* @ remove : remove method
* @ assign : match a given output type device against available outputs
* @ unassign : deassociate an output type device from an output port
* @ enable : enable tracing for a given output device
* @ disable : disable tracing for a given output device
2016-06-29 19:35:22 +03:00
* @ irq : interrupt callback
* @ activate : enable tracing on the output ' s side
* @ deactivate : disable tracing on the output ' s side
2015-09-22 15:47:14 +03:00
* @ fops : file operations for device nodes
2016-03-04 19:42:48 +02:00
* @ attr_group : attributes provided by the driver
2015-09-22 15:47:14 +03:00
*
* Callbacks @ probe and @ remove are required for all device types .
* Switch device driver needs to fill in @ assign , @ enable and @ disable
* callbacks .
*/
struct intel_th_driver {
struct device_driver driver ;
int ( * probe ) ( struct intel_th_device * thdev ) ;
void ( * remove ) ( struct intel_th_device * thdev ) ;
/* switch (GTH) ops */
int ( * assign ) ( struct intel_th_device * thdev ,
struct intel_th_device * othdev ) ;
void ( * unassign ) ( struct intel_th_device * thdev ,
struct intel_th_device * othdev ) ;
void ( * enable ) ( struct intel_th_device * thdev ,
struct intel_th_output * output ) ;
void ( * disable ) ( struct intel_th_device * thdev ,
struct intel_th_output * output ) ;
/* output ops */
void ( * irq ) ( struct intel_th_device * thdev ) ;
int ( * activate ) ( struct intel_th_device * thdev ) ;
void ( * deactivate ) ( struct intel_th_device * thdev ) ;
/* file_operations for those who want a device node */
const struct file_operations * fops ;
2016-03-04 19:42:48 +02:00
/* optional attributes */
struct attribute_group * attr_group ;
2015-09-22 15:47:14 +03:00
/* source ops */
int ( * set_output ) ( struct intel_th_device * thdev ,
unsigned int master ) ;
} ;
# define to_intel_th_driver(_d) \
container_of ( ( _d ) , struct intel_th_driver , driver )
2016-03-07 17:04:45 +02:00
# define to_intel_th_driver_or_null(_d) \
( ( _d ) ? to_intel_th_driver ( _d ) : NULL )
2016-11-18 15:36:39 +02:00
/*
* Subdevice tree structure is as follows :
* + struct intel_th device ( pci ; dev_ { get , set } _drvdata ( )
* + struct intel_th_device INTEL_TH_SWITCH ( GTH )
* + struct intel_th_device INTEL_TH_OUTPUT ( MSU , PTI )
* + struct intel_th_device INTEL_TH_SOURCE ( STH )
*
* In other words , INTEL_TH_OUTPUT devices are children of INTEL_TH_SWITCH ;
* INTEL_TH_SWITCH and INTEL_TH_SOURCE are children of the intel_th device .
*/
2015-09-22 15:47:14 +03:00
static inline struct intel_th_device *
2016-11-18 15:05:01 +02:00
to_intel_th_parent ( struct intel_th_device * thdev )
2015-09-22 15:47:14 +03:00
{
struct device * parent = thdev - > dev . parent ;
if ( ! parent )
return NULL ;
return to_intel_th_device ( parent ) ;
}
2016-11-18 15:05:01 +02:00
static inline struct intel_th * to_intel_th ( struct intel_th_device * thdev )
{
2016-11-18 15:36:39 +02:00
if ( thdev - > type = = INTEL_TH_OUTPUT )
thdev = to_intel_th_parent ( thdev ) ;
2016-11-18 15:05:01 +02:00
2016-11-18 15:36:39 +02:00
if ( WARN_ON_ONCE ( ! thdev | | thdev - > type = = INTEL_TH_OUTPUT ) )
2016-11-18 15:05:01 +02:00
return NULL ;
return dev_get_drvdata ( thdev - > dev . parent ) ;
}
2015-09-22 15:47:14 +03:00
struct intel_th *
2017-08-18 17:57:35 +03:00
intel_th_alloc ( struct device * dev , struct intel_th_drvdata * drvdata ,
struct resource * devres , unsigned int ndevres , int irq ) ;
2015-09-22 15:47:14 +03:00
void intel_th_free ( struct intel_th * th ) ;
int intel_th_driver_register ( struct intel_th_driver * thdrv ) ;
void intel_th_driver_unregister ( struct intel_th_driver * thdrv ) ;
int intel_th_trace_enable ( struct intel_th_device * thdev ) ;
int intel_th_trace_disable ( struct intel_th_device * thdev ) ;
int intel_th_set_output ( struct intel_th_device * thdev ,
unsigned int master ) ;
2017-08-10 18:28:38 +03:00
int intel_th_output_enable ( struct intel_th * th , unsigned int otype ) ;
2015-09-22 15:47:14 +03:00
enum {
TH_MMIO_CONFIG = 0 ,
TH_MMIO_SW = 2 ,
TH_MMIO_END ,
} ;
# define TH_POSSIBLE_OUTPUTS 8
2017-08-10 18:28:38 +03:00
/* Total number of possible subdevices: outputs + GTH + STH */
# define TH_SUBDEVICE_MAX (TH_POSSIBLE_OUTPUTS + 2)
2015-09-22 15:47:14 +03:00
# define TH_CONFIGURABLE_MASTERS 256
# define TH_MSC_MAX 2
/**
* struct intel_th - Intel TH controller
* @ dev : driver core ' s device
* @ thdev : subdevices
* @ hub : " switch " subdevice ( GTH )
2017-08-10 18:28:38 +03:00
* @ resource : resources of the entire controller
* @ num_thdevs : number of devices in the @ thdev array
* @ num_resources : number or resources in the @ resource array
* @ irq : irq number
2015-09-22 15:47:14 +03:00
* @ id : this Intel TH controller ' s device ID in the system
* @ major : device node major for output devices
*/
struct intel_th {
struct device * dev ;
struct intel_th_device * thdev [ TH_SUBDEVICE_MAX ] ;
struct intel_th_device * hub ;
2017-08-18 17:57:35 +03:00
struct intel_th_drvdata * drvdata ;
2015-09-22 15:47:14 +03:00
2017-08-10 18:28:38 +03:00
struct resource * resource ;
unsigned int num_thdevs ;
unsigned int num_resources ;
int irq ;
2015-09-22 15:47:14 +03:00
int id ;
int major ;
2016-06-30 11:51:44 +03:00
# ifdef CONFIG_MODULES
struct work_struct request_module_work ;
# endif /* CONFIG_MODULES */
2015-09-22 15:47:14 +03:00
# ifdef CONFIG_INTEL_TH_DEBUG
struct dentry * dbg ;
# endif
} ;
2016-11-18 15:36:39 +02:00
static inline struct intel_th_device *
to_intel_th_hub ( struct intel_th_device * thdev )
{
if ( thdev - > type = = INTEL_TH_SWITCH )
return thdev ;
else if ( thdev - > type = = INTEL_TH_OUTPUT )
return to_intel_th_parent ( thdev ) ;
return to_intel_th ( thdev ) - > hub ;
}
2015-09-22 15:47:14 +03:00
/*
* Register windows
*/
enum {
/* Global Trace Hub (GTH) */
REG_GTH_OFFSET = 0x0000 ,
REG_GTH_LENGTH = 0x2000 ,
/* Software Trace Hub (STH) [0x4000..0x4fff] */
REG_STH_OFFSET = 0x4000 ,
REG_STH_LENGTH = 0x2000 ,
/* Memory Storage Unit (MSU) [0xa0000..0xa1fff] */
REG_MSU_OFFSET = 0xa0000 ,
REG_MSU_LENGTH = 0x02000 ,
/* Internal MSU trace buffer [0x80000..0x9ffff] */
BUF_MSU_OFFSET = 0x80000 ,
BUF_MSU_LENGTH = 0x20000 ,
/* PTI output == same window as GTH */
REG_PTI_OFFSET = REG_GTH_OFFSET ,
REG_PTI_LENGTH = REG_GTH_LENGTH ,
/* DCI Handler (DCIH) == some window as MSU */
REG_DCIH_OFFSET = REG_MSU_OFFSET ,
REG_DCIH_LENGTH = REG_MSU_LENGTH ,
} ;
2016-02-15 19:11:55 +02:00
/*
* Scratchpad bits : tell firmware and external debuggers
* what we are up to .
*/
enum {
/* Memory is the primary destination */
SCRPD_MEM_IS_PRIM_DEST = BIT ( 0 ) ,
/* XHCI DbC is the primary destination */
SCRPD_DBC_IS_PRIM_DEST = BIT ( 1 ) ,
/* PTI is the primary destination */
SCRPD_PTI_IS_PRIM_DEST = BIT ( 2 ) ,
/* BSSB is the primary destination */
SCRPD_BSSB_IS_PRIM_DEST = BIT ( 3 ) ,
/* PTI is the alternate destination */
SCRPD_PTI_IS_ALT_DEST = BIT ( 4 ) ,
/* BSSB is the alternate destination */
SCRPD_BSSB_IS_ALT_DEST = BIT ( 5 ) ,
/* DeepSx exit occurred */
SCRPD_DEEPSX_EXIT = BIT ( 6 ) ,
/* S4 exit occurred */
SCRPD_S4_EXIT = BIT ( 7 ) ,
/* S5 exit occurred */
SCRPD_S5_EXIT = BIT ( 8 ) ,
/* MSU controller 0/1 is enabled */
SCRPD_MSC0_IS_ENABLED = BIT ( 9 ) ,
SCRPD_MSC1_IS_ENABLED = BIT ( 10 ) ,
/* Sx exit occurred */
SCRPD_SX_EXIT = BIT ( 11 ) ,
/* Trigger Unit is enabled */
SCRPD_TRIGGER_IS_ENABLED = BIT ( 12 ) ,
SCRPD_ODLA_IS_ENABLED = BIT ( 13 ) ,
SCRPD_SOCHAP_IS_ENABLED = BIT ( 14 ) ,
SCRPD_STH_IS_ENABLED = BIT ( 15 ) ,
SCRPD_DCIH_IS_ENABLED = BIT ( 16 ) ,
SCRPD_VER_IS_ENABLED = BIT ( 17 ) ,
/* External debugger is using Intel TH */
SCRPD_DEBUGGER_IN_USE = BIT ( 24 ) ,
} ;
2015-09-22 15:47:14 +03:00
# endif