Message ID | 20240531042745.494222-7-lcherian@marvell.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | Coresight for Kernel panic and watchdog reset | expand |
On 31/05/2024 05:27, Linu Cherian wrote: > Configure TMC ETR and ETF to flush and stop trace capture > on FlIn event based on sysfs attribute, > /sys/bus/coresight/devices/tmc_etXn/stop_on_flush. > > Signed-off-by: Linu Cherian <lcherian@marvell.com> > --- > Changelog from v7: > * Honour stop_on_flush sysfs attribute for perf sessions as well. > This allows user to start trace sessions using perf method as > well to use the Coresight kernel panic support. > > .../hwtracing/coresight/coresight-tmc-core.c | 31 +++++++++++++++++++ > .../hwtracing/coresight/coresight-tmc-etf.c | 18 ++++++++--- > .../hwtracing/coresight/coresight-tmc-etr.c | 18 ++++++++--- > drivers/hwtracing/coresight/coresight-tmc.h | 4 +++ > 4 files changed, 61 insertions(+), 10 deletions(-) > > diff --git a/drivers/hwtracing/coresight/coresight-tmc-core.c b/drivers/hwtracing/coresight/coresight-tmc-core.c > index aaa9de9f4cd2..24edd9d750c3 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc-core.c > +++ b/drivers/hwtracing/coresight/coresight-tmc-core.c > @@ -452,9 +452,40 @@ static ssize_t buffer_size_store(struct device *dev, > > static DEVICE_ATTR_RW(buffer_size); > > +static ssize_t stop_on_flush_show(struct device *dev, > + struct device_attribute *attr, char *buf) > +{ > + struct tmc_drvdata *drvdata = dev_get_drvdata(dev->parent); > + > + return sprintf(buf, "%#x\n", drvdata->stop_on_flush); > +} > + > +static ssize_t stop_on_flush_store(struct device *dev, > + struct device_attribute *attr, > + const char *buf, size_t size) > +{ > + int ret; > + u8 val; > + struct tmc_drvdata *drvdata = dev_get_drvdata(dev->parent); > + > + ret = kstrtou8(buf, 0, &val); > + if (ret) > + return ret; > + if (val) > + drvdata->stop_on_flush = true; > + else > + drvdata->stop_on_flush = false; > + > + return size; > +} > + > +static DEVICE_ATTR_RW(stop_on_flush); > + > + > static struct attribute *coresight_tmc_attrs[] = { > &dev_attr_trigger_cntr.attr, > &dev_attr_buffer_size.attr, > + &dev_attr_stop_on_flush.attr, > NULL, > }; > > diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c b/drivers/hwtracing/coresight/coresight-tmc-etf.c > index 655c0c0ba54b..1529b9cc210a 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c > +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c > @@ -19,6 +19,7 @@ static int tmc_set_etf_buffer(struct coresight_device *csdev, > static int __tmc_etb_enable_hw(struct tmc_drvdata *drvdata) > { > int rc = 0; > + u32 ffcr; > > CS_UNLOCK(drvdata->base); > > @@ -32,10 +33,12 @@ static int __tmc_etb_enable_hw(struct tmc_drvdata *drvdata) > } > > writel_relaxed(TMC_MODE_CIRCULAR_BUFFER, drvdata->base + TMC_MODE); > - writel_relaxed(TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | > - TMC_FFCR_FON_FLIN | TMC_FFCR_FON_TRIG_EVT | > - TMC_FFCR_TRIGON_TRIGIN, > - drvdata->base + TMC_FFCR); > + > + ffcr = TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | TMC_FFCR_FON_FLIN | > + TMC_FFCR_FON_TRIG_EVT | TMC_FFCR_TRIGON_TRIGIN; > + if (drvdata->stop_on_flush_en) > + ffcr |= TMC_FFCR_STOP_ON_FLUSH; > + writel_relaxed(ffcr, drvdata->base + TMC_FFCR); > > writel_relaxed(drvdata->trigger_cntr, drvdata->base + TMC_TRG); > tmc_enable_hw(drvdata); > @@ -225,7 +228,8 @@ static int tmc_enable_etf_sink_sysfs(struct coresight_device *csdev) > used = true; > drvdata->buf = buf; > } > - > + if (drvdata->stop_on_flush) > + drvdata->stop_on_flush_en = true; Does this do anything different than only using stop_on_flush in __tmc_etr_enable_hw()? Seems like stop_on_flush_en == stop_on_flush whenever stop_on_flush_en is read, so you might as well just read stop_on_flush and get rid of the _en copy? > ret = tmc_etb_enable_hw(drvdata); > if (!ret) { > coresight_set_mode(csdev, CS_MODE_SYSFS); > @@ -288,6 +292,8 @@ static int tmc_enable_etf_sink_perf(struct coresight_device *csdev, void *data) > break; > } > > + if (drvdata->stop_on_flush) > + drvdata->stop_on_flush_en = true; > ret = tmc_etb_enable_hw(drvdata); > if (!ret) { > /* Associate with monitored process. */ > @@ -349,6 +355,8 @@ static int tmc_disable_etf_sink(struct coresight_device *csdev) > tmc_etb_disable_hw(drvdata); > /* Dissociate from monitored process. */ > drvdata->pid = -1; > + /* Reset stop on flush */ > + drvdata->stop_on_flush_en = false; > coresight_set_mode(csdev, CS_MODE_DISABLED); > > spin_unlock_irqrestore(&drvdata->spinlock, flags); > diff --git a/drivers/hwtracing/coresight/coresight-tmc-etr.c b/drivers/hwtracing/coresight/coresight-tmc-etr.c > index 74f53972ce3c..bf2d0932e205 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc-etr.c > +++ b/drivers/hwtracing/coresight/coresight-tmc-etr.c > @@ -1059,7 +1059,7 @@ static void tmc_sync_etr_buf(struct tmc_drvdata *drvdata) > > static int __tmc_etr_enable_hw(struct tmc_drvdata *drvdata) > { > - u32 axictl, sts; > + u32 axictl, sts, ffcr; > struct etr_buf *etr_buf = drvdata->etr_buf; > int rc = 0; > > @@ -1105,10 +1105,12 @@ static int __tmc_etr_enable_hw(struct tmc_drvdata *drvdata) > writel_relaxed(sts, drvdata->base + TMC_STS); > } > > - writel_relaxed(TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | > - TMC_FFCR_FON_FLIN | TMC_FFCR_FON_TRIG_EVT | > - TMC_FFCR_TRIGON_TRIGIN, > - drvdata->base + TMC_FFCR); > + ffcr = TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | TMC_FFCR_FON_FLIN | > + TMC_FFCR_FON_TRIG_EVT | TMC_FFCR_TRIGON_TRIGIN; > + if (drvdata->stop_on_flush_en) > + ffcr |= TMC_FFCR_STOP_ON_FLUSH; > + writel_relaxed(ffcr, drvdata->base + TMC_FFCR); > + > writel_relaxed(drvdata->trigger_cntr, drvdata->base + TMC_TRG); > tmc_enable_hw(drvdata); > > @@ -1313,6 +1315,8 @@ static int tmc_enable_etr_sink_sysfs(struct coresight_device *csdev) > goto out; > } > > + if (drvdata->stop_on_flush) > + drvdata->stop_on_flush_en = true; > ret = tmc_etr_enable_hw(drvdata, sysfs_buf); > if (!ret) { > coresight_set_mode(csdev, CS_MODE_SYSFS); > @@ -1758,6 +1762,8 @@ static int tmc_enable_etr_sink_perf(struct coresight_device *csdev, void *data) > goto unlock_out; > } > > + if (drvdata->stop_on_flush) > + drvdata->stop_on_flush_en = true; > rc = tmc_etr_enable_hw(drvdata, etr_perf->etr_buf); > if (!rc) { > /* Associate with monitored process. */ > @@ -1808,6 +1814,8 @@ static int tmc_disable_etr_sink(struct coresight_device *csdev) > tmc_etr_disable_hw(drvdata); > /* Dissociate from monitored process. */ > drvdata->pid = -1; > + /* Reset stop on flush */ > + drvdata->stop_on_flush_en = false; > coresight_set_mode(csdev, CS_MODE_DISABLED); > /* Reset perf specific data */ > drvdata->perf_buf = NULL; > diff --git a/drivers/hwtracing/coresight/coresight-tmc.h b/drivers/hwtracing/coresight/coresight-tmc.h > index 528174283ecc..81eadb384b83 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc.h > +++ b/drivers/hwtracing/coresight/coresight-tmc.h > @@ -205,6 +205,8 @@ struct tmc_resrv_buf { > * @spinlock: only one at a time pls. > * @pid: Process ID of the process being monitored by the session > * that is using this component. > + * @stop_on_flush: Stop on flush trigger user configuration. > + * @stop_on_flush_en: Stop on flush enable flag > * @buf: Snapshot of the trace data for ETF/ETB. > * @etr_buf: details of buffer used in TMC-ETR > * @len: size of the available trace for ETF/ETB. > @@ -238,6 +240,8 @@ struct tmc_drvdata { > spinlock_t spinlock; > pid_t pid; > bool reading; > + bool stop_on_flush; > + bool stop_on_flush_en; > union { > char *buf; /* TMC ETB */ > struct etr_buf *etr_buf; /* TMC ETR */
Hi James, > -----Original Message----- > From: James Clark <james.clark@arm.com> > Sent: Friday, May 31, 2024 3:47 PM > To: Linu Cherian <lcherian@marvell.com> > Cc: linux-arm-kernel@lists.infradead.org; coresight@lists.linaro.org; linux- > kernel@vger.kernel.org; robh+dt@kernel.org; > krzysztof.kozlowski+dt@linaro.org; conor+dt@kernel.org; > devicetree@vger.kernel.org; Sunil Kovvuri Goutham > <sgoutham@marvell.com>; George Cherian <gcherian@marvell.com>; > suzuki.poulose@arm.com; mike.leach@linaro.org > Subject: [EXTERNAL] Re: [PATCH v8 6/7] coresight: tmc: Stop trace capture on > FlIn > > Prioritize security for external emails: Confirm sender and content safety > before clicking links or opening attachments > > ---------------------------------------------------------------------- > > > On 31/05/2024 05:27, Linu Cherian wrote: > > Configure TMC ETR and ETF to flush and stop trace capture on FlIn > > event based on sysfs attribute, > > /sys/bus/coresight/devices/tmc_etXn/stop_on_flush. > > > > Signed-off-by: Linu Cherian <lcherian@marvell.com> > > --- > > Changelog from v7: > > * Honour stop_on_flush sysfs attribute for perf sessions as well. > > This allows user to start trace sessions using perf method as > > well to use the Coresight kernel panic support. > > > > .../hwtracing/coresight/coresight-tmc-core.c | 31 > +++++++++++++++++++ > > .../hwtracing/coresight/coresight-tmc-etf.c | 18 ++++++++--- > > .../hwtracing/coresight/coresight-tmc-etr.c | 18 ++++++++--- > > drivers/hwtracing/coresight/coresight-tmc.h | 4 +++ > > 4 files changed, 61 insertions(+), 10 deletions(-) > > > > diff --git a/drivers/hwtracing/coresight/coresight-tmc-core.c > > b/drivers/hwtracing/coresight/coresight-tmc-core.c > > index aaa9de9f4cd2..24edd9d750c3 100644 > > --- a/drivers/hwtracing/coresight/coresight-tmc-core.c > > +++ b/drivers/hwtracing/coresight/coresight-tmc-core.c > > @@ -452,9 +452,40 @@ static ssize_t buffer_size_store(struct device > > *dev, > > > > static DEVICE_ATTR_RW(buffer_size); > > > > +static ssize_t stop_on_flush_show(struct device *dev, > > + struct device_attribute *attr, char *buf) { > > + struct tmc_drvdata *drvdata = dev_get_drvdata(dev->parent); > > + > > + return sprintf(buf, "%#x\n", drvdata->stop_on_flush); } > > + > > +static ssize_t stop_on_flush_store(struct device *dev, > > + struct device_attribute *attr, > > + const char *buf, size_t size) > > +{ > > + int ret; > > + u8 val; > > + struct tmc_drvdata *drvdata = dev_get_drvdata(dev->parent); > > + > > + ret = kstrtou8(buf, 0, &val); > > + if (ret) > > + return ret; > > + if (val) > > + drvdata->stop_on_flush = true; > > + else > > + drvdata->stop_on_flush = false; > > + > > + return size; > > +} > > + > > +static DEVICE_ATTR_RW(stop_on_flush); > > + > > + > > static struct attribute *coresight_tmc_attrs[] = { > > &dev_attr_trigger_cntr.attr, > > &dev_attr_buffer_size.attr, > > + &dev_attr_stop_on_flush.attr, > > NULL, > > }; > > > > diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c > > b/drivers/hwtracing/coresight/coresight-tmc-etf.c > > index 655c0c0ba54b..1529b9cc210a 100644 > > --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c > > +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c > > @@ -19,6 +19,7 @@ static int tmc_set_etf_buffer(struct > > coresight_device *csdev, static int __tmc_etb_enable_hw(struct > > tmc_drvdata *drvdata) { > > int rc = 0; > > + u32 ffcr; > > > > CS_UNLOCK(drvdata->base); > > > > @@ -32,10 +33,12 @@ static int __tmc_etb_enable_hw(struct > tmc_drvdata *drvdata) > > } > > > > writel_relaxed(TMC_MODE_CIRCULAR_BUFFER, drvdata->base + > TMC_MODE); > > - writel_relaxed(TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | > > - TMC_FFCR_FON_FLIN | TMC_FFCR_FON_TRIG_EVT | > > - TMC_FFCR_TRIGON_TRIGIN, > > - drvdata->base + TMC_FFCR); > > + > > + ffcr = TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | > TMC_FFCR_FON_FLIN | > > + TMC_FFCR_FON_TRIG_EVT | TMC_FFCR_TRIGON_TRIGIN; > > + if (drvdata->stop_on_flush_en) > > + ffcr |= TMC_FFCR_STOP_ON_FLUSH; > > + writel_relaxed(ffcr, drvdata->base + TMC_FFCR); > > > > writel_relaxed(drvdata->trigger_cntr, drvdata->base + TMC_TRG); > > tmc_enable_hw(drvdata); > > @@ -225,7 +228,8 @@ static int tmc_enable_etf_sink_sysfs(struct > coresight_device *csdev) > > used = true; > > drvdata->buf = buf; > > } > > - > > + if (drvdata->stop_on_flush) > > + drvdata->stop_on_flush_en = true; > > Does this do anything different than only using stop_on_flush in > __tmc_etr_enable_hw()? > > Seems like stop_on_flush_en == stop_on_flush whenever > stop_on_flush_en is read, so you might as well just read stop_on_flush and > get rid of the _en copy? > Ack.
diff --git a/drivers/hwtracing/coresight/coresight-tmc-core.c b/drivers/hwtracing/coresight/coresight-tmc-core.c index aaa9de9f4cd2..24edd9d750c3 100644 --- a/drivers/hwtracing/coresight/coresight-tmc-core.c +++ b/drivers/hwtracing/coresight/coresight-tmc-core.c @@ -452,9 +452,40 @@ static ssize_t buffer_size_store(struct device *dev, static DEVICE_ATTR_RW(buffer_size); +static ssize_t stop_on_flush_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct tmc_drvdata *drvdata = dev_get_drvdata(dev->parent); + + return sprintf(buf, "%#x\n", drvdata->stop_on_flush); +} + +static ssize_t stop_on_flush_store(struct device *dev, + struct device_attribute *attr, + const char *buf, size_t size) +{ + int ret; + u8 val; + struct tmc_drvdata *drvdata = dev_get_drvdata(dev->parent); + + ret = kstrtou8(buf, 0, &val); + if (ret) + return ret; + if (val) + drvdata->stop_on_flush = true; + else + drvdata->stop_on_flush = false; + + return size; +} + +static DEVICE_ATTR_RW(stop_on_flush); + + static struct attribute *coresight_tmc_attrs[] = { &dev_attr_trigger_cntr.attr, &dev_attr_buffer_size.attr, + &dev_attr_stop_on_flush.attr, NULL, }; diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c b/drivers/hwtracing/coresight/coresight-tmc-etf.c index 655c0c0ba54b..1529b9cc210a 100644 --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c @@ -19,6 +19,7 @@ static int tmc_set_etf_buffer(struct coresight_device *csdev, static int __tmc_etb_enable_hw(struct tmc_drvdata *drvdata) { int rc = 0; + u32 ffcr; CS_UNLOCK(drvdata->base); @@ -32,10 +33,12 @@ static int __tmc_etb_enable_hw(struct tmc_drvdata *drvdata) } writel_relaxed(TMC_MODE_CIRCULAR_BUFFER, drvdata->base + TMC_MODE); - writel_relaxed(TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | - TMC_FFCR_FON_FLIN | TMC_FFCR_FON_TRIG_EVT | - TMC_FFCR_TRIGON_TRIGIN, - drvdata->base + TMC_FFCR); + + ffcr = TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | TMC_FFCR_FON_FLIN | + TMC_FFCR_FON_TRIG_EVT | TMC_FFCR_TRIGON_TRIGIN; + if (drvdata->stop_on_flush_en) + ffcr |= TMC_FFCR_STOP_ON_FLUSH; + writel_relaxed(ffcr, drvdata->base + TMC_FFCR); writel_relaxed(drvdata->trigger_cntr, drvdata->base + TMC_TRG); tmc_enable_hw(drvdata); @@ -225,7 +228,8 @@ static int tmc_enable_etf_sink_sysfs(struct coresight_device *csdev) used = true; drvdata->buf = buf; } - + if (drvdata->stop_on_flush) + drvdata->stop_on_flush_en = true; ret = tmc_etb_enable_hw(drvdata); if (!ret) { coresight_set_mode(csdev, CS_MODE_SYSFS); @@ -288,6 +292,8 @@ static int tmc_enable_etf_sink_perf(struct coresight_device *csdev, void *data) break; } + if (drvdata->stop_on_flush) + drvdata->stop_on_flush_en = true; ret = tmc_etb_enable_hw(drvdata); if (!ret) { /* Associate with monitored process. */ @@ -349,6 +355,8 @@ static int tmc_disable_etf_sink(struct coresight_device *csdev) tmc_etb_disable_hw(drvdata); /* Dissociate from monitored process. */ drvdata->pid = -1; + /* Reset stop on flush */ + drvdata->stop_on_flush_en = false; coresight_set_mode(csdev, CS_MODE_DISABLED); spin_unlock_irqrestore(&drvdata->spinlock, flags); diff --git a/drivers/hwtracing/coresight/coresight-tmc-etr.c b/drivers/hwtracing/coresight/coresight-tmc-etr.c index 74f53972ce3c..bf2d0932e205 100644 --- a/drivers/hwtracing/coresight/coresight-tmc-etr.c +++ b/drivers/hwtracing/coresight/coresight-tmc-etr.c @@ -1059,7 +1059,7 @@ static void tmc_sync_etr_buf(struct tmc_drvdata *drvdata) static int __tmc_etr_enable_hw(struct tmc_drvdata *drvdata) { - u32 axictl, sts; + u32 axictl, sts, ffcr; struct etr_buf *etr_buf = drvdata->etr_buf; int rc = 0; @@ -1105,10 +1105,12 @@ static int __tmc_etr_enable_hw(struct tmc_drvdata *drvdata) writel_relaxed(sts, drvdata->base + TMC_STS); } - writel_relaxed(TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | - TMC_FFCR_FON_FLIN | TMC_FFCR_FON_TRIG_EVT | - TMC_FFCR_TRIGON_TRIGIN, - drvdata->base + TMC_FFCR); + ffcr = TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI | TMC_FFCR_FON_FLIN | + TMC_FFCR_FON_TRIG_EVT | TMC_FFCR_TRIGON_TRIGIN; + if (drvdata->stop_on_flush_en) + ffcr |= TMC_FFCR_STOP_ON_FLUSH; + writel_relaxed(ffcr, drvdata->base + TMC_FFCR); + writel_relaxed(drvdata->trigger_cntr, drvdata->base + TMC_TRG); tmc_enable_hw(drvdata); @@ -1313,6 +1315,8 @@ static int tmc_enable_etr_sink_sysfs(struct coresight_device *csdev) goto out; } + if (drvdata->stop_on_flush) + drvdata->stop_on_flush_en = true; ret = tmc_etr_enable_hw(drvdata, sysfs_buf); if (!ret) { coresight_set_mode(csdev, CS_MODE_SYSFS); @@ -1758,6 +1762,8 @@ static int tmc_enable_etr_sink_perf(struct coresight_device *csdev, void *data) goto unlock_out; } + if (drvdata->stop_on_flush) + drvdata->stop_on_flush_en = true; rc = tmc_etr_enable_hw(drvdata, etr_perf->etr_buf); if (!rc) { /* Associate with monitored process. */ @@ -1808,6 +1814,8 @@ static int tmc_disable_etr_sink(struct coresight_device *csdev) tmc_etr_disable_hw(drvdata); /* Dissociate from monitored process. */ drvdata->pid = -1; + /* Reset stop on flush */ + drvdata->stop_on_flush_en = false; coresight_set_mode(csdev, CS_MODE_DISABLED); /* Reset perf specific data */ drvdata->perf_buf = NULL; diff --git a/drivers/hwtracing/coresight/coresight-tmc.h b/drivers/hwtracing/coresight/coresight-tmc.h index 528174283ecc..81eadb384b83 100644 --- a/drivers/hwtracing/coresight/coresight-tmc.h +++ b/drivers/hwtracing/coresight/coresight-tmc.h @@ -205,6 +205,8 @@ struct tmc_resrv_buf { * @spinlock: only one at a time pls. * @pid: Process ID of the process being monitored by the session * that is using this component. + * @stop_on_flush: Stop on flush trigger user configuration. + * @stop_on_flush_en: Stop on flush enable flag * @buf: Snapshot of the trace data for ETF/ETB. * @etr_buf: details of buffer used in TMC-ETR * @len: size of the available trace for ETF/ETB. @@ -238,6 +240,8 @@ struct tmc_drvdata { spinlock_t spinlock; pid_t pid; bool reading; + bool stop_on_flush; + bool stop_on_flush_en; union { char *buf; /* TMC ETB */ struct etr_buf *etr_buf; /* TMC ETR */
Configure TMC ETR and ETF to flush and stop trace capture on FlIn event based on sysfs attribute, /sys/bus/coresight/devices/tmc_etXn/stop_on_flush. Signed-off-by: Linu Cherian <lcherian@marvell.com> --- Changelog from v7: * Honour stop_on_flush sysfs attribute for perf sessions as well. This allows user to start trace sessions using perf method as well to use the Coresight kernel panic support. .../hwtracing/coresight/coresight-tmc-core.c | 31 +++++++++++++++++++ .../hwtracing/coresight/coresight-tmc-etf.c | 18 ++++++++--- .../hwtracing/coresight/coresight-tmc-etr.c | 18 ++++++++--- drivers/hwtracing/coresight/coresight-tmc.h | 4 +++ 4 files changed, 61 insertions(+), 10 deletions(-)