Commit f01e4948 authored by Tao Zhang's avatar Tao Zhang Committed by Suzuki K Poulose

coresight-tpdm: Initialize DSB subunit configuration

DSB is used for monitoring “events”. Events are something that
occurs at some point in time. It could be a state decode, the
act of writing/reading a particular address, a FIFO being empty,
etc. This decoding of the event desired is done outside TPDM.
DSB subunit need to be configured in enablement and disablement.
A struct that specifics associated to dsb dataset is needed. It
saves the configuration and parameters of the dsb datasets. This
change is to add this struct and initialize the configuration of
DSB subunit.
Signed-off-by: default avatarTao Zhang <quic_taozha@quicinc.com>
Signed-off-by: default avatarSuzuki K Poulose <suzuki.poulose@arm.com>
Link: https://lore.kernel.org/r/1695882586-10306-6-git-send-email-quic_taozha@quicinc.com
parent 57e7235a
...@@ -20,23 +20,57 @@ ...@@ -20,23 +20,57 @@
DEFINE_CORESIGHT_DEVLIST(tpdm_devs, "tpdm"); DEFINE_CORESIGHT_DEVLIST(tpdm_devs, "tpdm");
static bool tpdm_has_dsb_dataset(struct tpdm_drvdata *drvdata)
{
return (drvdata->datasets & TPDM_PIDR0_DS_DSB);
}
static void tpdm_reset_datasets(struct tpdm_drvdata *drvdata)
{
if (tpdm_has_dsb_dataset(drvdata)) {
memset(drvdata->dsb, 0, sizeof(struct dsb_dataset));
drvdata->dsb->trig_ts = true;
drvdata->dsb->trig_type = false;
}
}
static void tpdm_enable_dsb(struct tpdm_drvdata *drvdata) static void tpdm_enable_dsb(struct tpdm_drvdata *drvdata)
{ {
u32 val; u32 val;
/* Set the enable bit of DSB control register to 1 */ val = readl_relaxed(drvdata->base + TPDM_DSB_TIER);
/* Set trigger timestamp */
if (drvdata->dsb->trig_ts)
val |= TPDM_DSB_TIER_XTRIG_TSENAB;
else
val &= ~TPDM_DSB_TIER_XTRIG_TSENAB;
writel_relaxed(val, drvdata->base + TPDM_DSB_TIER);
val = readl_relaxed(drvdata->base + TPDM_DSB_CR); val = readl_relaxed(drvdata->base + TPDM_DSB_CR);
/* Set trigger type */
if (drvdata->dsb->trig_type)
val |= TPDM_DSB_CR_TRIG_TYPE;
else
val &= ~TPDM_DSB_CR_TRIG_TYPE;
/* Set the enable bit of DSB control register to 1 */
val |= TPDM_DSB_CR_ENA; val |= TPDM_DSB_CR_ENA;
writel_relaxed(val, drvdata->base + TPDM_DSB_CR); writel_relaxed(val, drvdata->base + TPDM_DSB_CR);
} }
/* TPDM enable operations */ /*
* TPDM enable operations
* The TPDM or Monitor serves as data collection component for various
* dataset types. It covers Basic Counts(BC), Tenure Counts(TC),
* Continuous Multi-Bit(CMB), Multi-lane CMB(MCMB) and Discrete Single
* Bit(DSB). This function will initialize the configuration according
* to the dataset type supported by the TPDM.
*/
static void __tpdm_enable(struct tpdm_drvdata *drvdata) static void __tpdm_enable(struct tpdm_drvdata *drvdata)
{ {
CS_UNLOCK(drvdata->base); CS_UNLOCK(drvdata->base);
/* Check if DSB datasets is present for TPDM. */ if (tpdm_has_dsb_dataset(drvdata))
if (drvdata->datasets & TPDM_PIDR0_DS_DSB)
tpdm_enable_dsb(drvdata); tpdm_enable_dsb(drvdata);
CS_LOCK(drvdata->base); CS_LOCK(drvdata->base);
...@@ -76,8 +110,7 @@ static void __tpdm_disable(struct tpdm_drvdata *drvdata) ...@@ -76,8 +110,7 @@ static void __tpdm_disable(struct tpdm_drvdata *drvdata)
{ {
CS_UNLOCK(drvdata->base); CS_UNLOCK(drvdata->base);
/* Check if DSB datasets is present for TPDM. */ if (tpdm_has_dsb_dataset(drvdata))
if (drvdata->datasets & TPDM_PIDR0_DS_DSB)
tpdm_disable_dsb(drvdata); tpdm_disable_dsb(drvdata);
CS_LOCK(drvdata->base); CS_LOCK(drvdata->base);
...@@ -110,13 +143,23 @@ static const struct coresight_ops tpdm_cs_ops = { ...@@ -110,13 +143,23 @@ static const struct coresight_ops tpdm_cs_ops = {
.source_ops = &tpdm_source_ops, .source_ops = &tpdm_source_ops,
}; };
static void tpdm_init_default_data(struct tpdm_drvdata *drvdata) static int tpdm_datasets_setup(struct tpdm_drvdata *drvdata)
{ {
u32 pidr; u32 pidr;
/* Get the datasets present on the TPDM. */ /* Get the datasets present on the TPDM. */
pidr = readl_relaxed(drvdata->base + CORESIGHT_PERIPHIDR0); pidr = readl_relaxed(drvdata->base + CORESIGHT_PERIPHIDR0);
drvdata->datasets |= pidr & GENMASK(TPDM_DATASETS - 1, 0); drvdata->datasets |= pidr & GENMASK(TPDM_DATASETS - 1, 0);
if (tpdm_has_dsb_dataset(drvdata) && (!drvdata->dsb)) {
drvdata->dsb = devm_kzalloc(drvdata->dev,
sizeof(*drvdata->dsb), GFP_KERNEL);
if (!drvdata->dsb)
return -ENOMEM;
}
tpdm_reset_datasets(drvdata);
return 0;
} }
/* /*
...@@ -179,6 +222,7 @@ static int tpdm_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -179,6 +222,7 @@ static int tpdm_probe(struct amba_device *adev, const struct amba_id *id)
struct coresight_platform_data *pdata; struct coresight_platform_data *pdata;
struct tpdm_drvdata *drvdata; struct tpdm_drvdata *drvdata;
struct coresight_desc desc = { 0 }; struct coresight_desc desc = { 0 };
int ret;
pdata = coresight_get_platform_data(dev); pdata = coresight_get_platform_data(dev);
if (IS_ERR(pdata)) if (IS_ERR(pdata))
...@@ -198,6 +242,10 @@ static int tpdm_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -198,6 +242,10 @@ static int tpdm_probe(struct amba_device *adev, const struct amba_id *id)
drvdata->base = base; drvdata->base = base;
ret = tpdm_datasets_setup(drvdata);
if (ret)
return ret;
/* Set up coresight component description */ /* Set up coresight component description */
desc.name = coresight_alloc_device_name(&tpdm_devs, dev); desc.name = coresight_alloc_device_name(&tpdm_devs, dev);
if (!desc.name) if (!desc.name)
...@@ -214,7 +262,7 @@ static int tpdm_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -214,7 +262,7 @@ static int tpdm_probe(struct amba_device *adev, const struct amba_id *id)
return PTR_ERR(drvdata->csdev); return PTR_ERR(drvdata->csdev);
spin_lock_init(&drvdata->spinlock); spin_lock_init(&drvdata->spinlock);
tpdm_init_default_data(drvdata);
/* Decrease pm refcount when probe is done.*/ /* Decrease pm refcount when probe is done.*/
pm_runtime_put(&adev->dev); pm_runtime_put(&adev->dev);
......
...@@ -11,8 +11,14 @@ ...@@ -11,8 +11,14 @@
/* DSB Subunit Registers */ /* DSB Subunit Registers */
#define TPDM_DSB_CR (0x780) #define TPDM_DSB_CR (0x780)
#define TPDM_DSB_TIER (0x784)
/* Enable bit for DSB subunit */ /* Enable bit for DSB subunit */
#define TPDM_DSB_CR_ENA BIT(0) #define TPDM_DSB_CR_ENA BIT(0)
/* Enable bit for DSB subunit trigger type */
#define TPDM_DSB_CR_TRIG_TYPE BIT(12)
/* Enable bit for DSB subunit trigger timestamp */
#define TPDM_DSB_TIER_XTRIG_TSENAB BIT(1)
/* TPDM integration test registers */ /* TPDM integration test registers */
#define TPDM_ITATBCNTRL (0xEF0) #define TPDM_ITATBCNTRL (0xEF0)
...@@ -40,6 +46,16 @@ ...@@ -40,6 +46,16 @@
#define TPDM_PIDR0_DS_IMPDEF BIT(0) #define TPDM_PIDR0_DS_IMPDEF BIT(0)
#define TPDM_PIDR0_DS_DSB BIT(1) #define TPDM_PIDR0_DS_DSB BIT(1)
/**
* struct dsb_dataset - specifics associated to dsb dataset
* @trig_ts: Enable/Disable trigger timestamp.
* @trig_type: Enable/Disable trigger type.
*/
struct dsb_dataset {
bool trig_ts;
bool trig_type;
};
/** /**
* struct tpdm_drvdata - specifics associated to an TPDM component * struct tpdm_drvdata - specifics associated to an TPDM component
* @base: memory mapped base address for this component. * @base: memory mapped base address for this component.
...@@ -48,6 +64,7 @@ ...@@ -48,6 +64,7 @@
* @spinlock: lock for the drvdata value. * @spinlock: lock for the drvdata value.
* @enable: enable status of the component. * @enable: enable status of the component.
* @datasets: The datasets types present of the TPDM. * @datasets: The datasets types present of the TPDM.
* @dsb Specifics associated to TPDM DSB.
*/ */
struct tpdm_drvdata { struct tpdm_drvdata {
...@@ -57,6 +74,7 @@ struct tpdm_drvdata { ...@@ -57,6 +74,7 @@ struct tpdm_drvdata {
spinlock_t spinlock; spinlock_t spinlock;
bool enable; bool enable;
unsigned long datasets; unsigned long datasets;
struct dsb_dataset *dsb;
}; };
#endif /* _CORESIGHT_CORESIGHT_TPDM_H */ #endif /* _CORESIGHT_CORESIGHT_TPDM_H */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment