Message ID | 1460483692-25061-10-git-send-email-mathieu.poirier@linaro.org (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
On 12/04/16 18:54, Mathieu Poirier wrote: > Moving tmc_drvdata::enable to a local_t mode. That way the > sink interface is aware of it's orgin and the foundation for > mutual exclusion between the sysFS and Perf interface can be > laid out. > > Signed-off-by: Mathieu Poirier <mathieu.poirier@linaro.org> > --- > drivers/hwtracing/coresight/coresight-tmc-etf.c | 28 ++++++++++++++++++------- > drivers/hwtracing/coresight/coresight-tmc-etr.c | 24 ++++++++++++++++----- > drivers/hwtracing/coresight/coresight-tmc.h | 4 ++-- > 3 files changed, 42 insertions(+), 14 deletions(-) > > diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c b/drivers/hwtracing/coresight/coresight-tmc-etf.c > index 7cb287ef7b9e..5908000e1ae0 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c > +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c > @@ -110,6 +110,7 @@ static int tmc_enable_etf_sink(struct coresight_device *csdev, u32 mode) > { > bool allocated = false; > char *buf = NULL; > + u32 val; > unsigned long flags; > struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent); > > @@ -146,6 +147,15 @@ fast_path: > return -EBUSY; > } > > + val = local_xchg(&drvdata->mode, mode); > + /* > + * In sysFS mode we can have multiple writers per sink. Since this > + * sink is already enabled no memory is needed and the HW need not be > + * touched. > + */ > + if (val == CS_MODE_SYSFS) > + goto out; > + We are not dropping the spinlock in this case. Are we ? > /* > * If drvdata::buf isn't NULL, memory was allocated for a previous > * trace run but wasn't read. If so simply zero-out the memory. > @@ -163,9 +173,9 @@ fast_path: > } > > tmc_etb_enable_hw(drvdata); > - drvdata->enable = true; > spin_unlock_irqrestore(&drvdata->spinlock, flags); > > +out: > /* Free memory outside the spinlock if need be */ > if (!allocated && buf) > kfree(buf); > diff --git a/drivers/hwtracing/coresight/coresight-tmc-etr.c b/drivers/hwtracing/coresight/coresight-tmc-etr.c > index 6022ff26deba..8e6fe267195a 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc-etr.c > +++ b/drivers/hwtracing/coresight/coresight-tmc-etr.c > @@ -85,6 +85,7 @@ static void tmc_etr_disable_hw(struct tmc_drvdata *drvdata) > static int tmc_enable_etr_sink(struct coresight_device *csdev, u32 mode) > { > bool allocated = false; > + u32 val; > unsigned long flags; > void __iomem *vaddr = NULL; > dma_addr_t paddr; > @@ -125,6 +126,15 @@ fast_path: > return -EBUSY; > } > > + val = local_xchg(&drvdata->mode, mode); > + /* > + * In sysFS mode we can have multiple writers per sink. Since this > + * sink is already enabled no memory is needed and the HW need not be > + * touched. > + */ > + if (val == CS_MODE_SYSFS) > + goto out; > + Same as above, we should drop the locks in this case. > @@ -140,9 +150,9 @@ fast_path: > memset(drvdata->vaddr, 0, drvdata->size); > > tmc_etr_enable_hw(drvdata); > - drvdata->enable = true; > spin_unlock_irqrestore(&drvdata->spinlock, flags); > > +out: > /* Free memory outside the spinlock if need be */ > if (!allocated && vaddr) > dma_free_coherent(drvdata->dev, drvdata->size, vaddr, paddr); > @@ -153,6 +163,7 @@ fast_path: > diff --git a/drivers/hwtracing/coresight/coresight-tmc.h b/drivers/hwtracing/coresight/coresight-tmc.h > index 80096fa75326..821bdf150ac9 100644 > --- a/drivers/hwtracing/coresight/coresight-tmc.h > +++ b/drivers/hwtracing/coresight/coresight-tmc.h > @@ -100,7 +100,7 @@ enum tmc_mem_intf_width { > * @paddr: DMA start location in RAM. > * @vaddr: virtual representation of @paddr. > * @size: @buf size. > - * @enable: this TMC is being used. > + * @mode: how this TMC is being used. > * @config_type: TMC variant, must be of type @tmc_config_type. > * @trigger_cntr: amount of words to store after a trigger. > */ > @@ -116,7 +116,7 @@ struct tmc_drvdata { > dma_addr_t paddr; > void __iomem *vaddr; > u32 size; > - bool enable; > + local_t mode; Since we always deal with the mode under the spinlock, do we really need this to be local_t ? Or do we plan to get rid of the lock ? Suzuki
On 19 April 2016 at 07:19, Suzuki K Poulose <Suzuki.Poulose@arm.com> wrote: > On 12/04/16 18:54, Mathieu Poirier wrote: >> >> Moving tmc_drvdata::enable to a local_t mode. That way the >> sink interface is aware of it's orgin and the foundation for >> mutual exclusion between the sysFS and Perf interface can be >> laid out. >> >> Signed-off-by: Mathieu Poirier <mathieu.poirier@linaro.org> >> --- >> drivers/hwtracing/coresight/coresight-tmc-etf.c | 28 >> ++++++++++++++++++------- >> drivers/hwtracing/coresight/coresight-tmc-etr.c | 24 >> ++++++++++++++++----- >> drivers/hwtracing/coresight/coresight-tmc.h | 4 ++-- >> 3 files changed, 42 insertions(+), 14 deletions(-) >> >> diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c >> b/drivers/hwtracing/coresight/coresight-tmc-etf.c >> index 7cb287ef7b9e..5908000e1ae0 100644 >> --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c >> +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c >> @@ -110,6 +110,7 @@ static int tmc_enable_etf_sink(struct coresight_device >> *csdev, u32 mode) >> { >> bool allocated = false; >> char *buf = NULL; >> + u32 val; >> unsigned long flags; >> struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent); >> >> @@ -146,6 +147,15 @@ fast_path: >> return -EBUSY; >> } >> >> + val = local_xchg(&drvdata->mode, mode); >> + /* >> + * In sysFS mode we can have multiple writers per sink. Since >> this >> + * sink is already enabled no memory is needed and the HW need not >> be >> + * touched. >> + */ >> + if (val == CS_MODE_SYSFS) >> + goto out; >> + > > > We are not dropping the spinlock in this case. Are we ? Definitely - thanks for pointing that out. > >> /* >> * If drvdata::buf isn't NULL, memory was allocated for a previous >> * trace run but wasn't read. If so simply zero-out the memory. >> @@ -163,9 +173,9 @@ fast_path: >> } >> >> tmc_etb_enable_hw(drvdata); >> - drvdata->enable = true; >> spin_unlock_irqrestore(&drvdata->spinlock, flags); >> >> +out: >> /* Free memory outside the spinlock if need be */ >> if (!allocated && buf) >> kfree(buf); > > >> diff --git a/drivers/hwtracing/coresight/coresight-tmc-etr.c >> b/drivers/hwtracing/coresight/coresight-tmc-etr.c >> index 6022ff26deba..8e6fe267195a 100644 >> --- a/drivers/hwtracing/coresight/coresight-tmc-etr.c >> +++ b/drivers/hwtracing/coresight/coresight-tmc-etr.c >> @@ -85,6 +85,7 @@ static void tmc_etr_disable_hw(struct tmc_drvdata >> *drvdata) >> static int tmc_enable_etr_sink(struct coresight_device *csdev, u32 mode) >> { >> bool allocated = false; >> + u32 val; >> unsigned long flags; >> void __iomem *vaddr = NULL; >> dma_addr_t paddr; >> @@ -125,6 +126,15 @@ fast_path: >> return -EBUSY; >> } >> >> + val = local_xchg(&drvdata->mode, mode); >> + /* >> + * In sysFS mode we can have multiple writers per sink. Since >> this >> + * sink is already enabled no memory is needed and the HW need not >> be >> + * touched. >> + */ >> + if (val == CS_MODE_SYSFS) >> + goto out; >> + > > > Same as above, we should drop the locks in this case. > >> @@ -140,9 +150,9 @@ fast_path: >> memset(drvdata->vaddr, 0, drvdata->size); >> >> tmc_etr_enable_hw(drvdata); >> - drvdata->enable = true; >> spin_unlock_irqrestore(&drvdata->spinlock, flags); >> >> +out: >> /* Free memory outside the spinlock if need be */ >> if (!allocated && vaddr) >> dma_free_coherent(drvdata->dev, drvdata->size, vaddr, >> paddr); >> @@ -153,6 +163,7 @@ fast_path: > > >> diff --git a/drivers/hwtracing/coresight/coresight-tmc.h >> b/drivers/hwtracing/coresight/coresight-tmc.h >> index 80096fa75326..821bdf150ac9 100644 >> --- a/drivers/hwtracing/coresight/coresight-tmc.h >> +++ b/drivers/hwtracing/coresight/coresight-tmc.h >> @@ -100,7 +100,7 @@ enum tmc_mem_intf_width { >> * @paddr: DMA start location in RAM. >> * @vaddr: virtual representation of @paddr. >> * @size: @buf size. >> - * @enable: this TMC is being used. >> + * @mode: how this TMC is being used. >> * @config_type: TMC variant, must be of type @tmc_config_type. >> * @trigger_cntr: amount of words to store after a trigger. >> */ >> @@ -116,7 +116,7 @@ struct tmc_drvdata { >> dma_addr_t paddr; >> void __iomem *vaddr; >> u32 size; >> - bool enable; >> + local_t mode; > > > Since we always deal with the mode under the spinlock, do we really need > this to > be local_t ? Or do we plan to get rid of the lock ? You are correct, using a local_t variable isn't strictly needed here. But: 1) It greatly simplifies the code. 2) I'm really hoping one day we can do without the spinlocks, though I don't know when that will happen. Get back to me if you're resolute on switching the local_t to a normal type. Mathieu > > Suzuki >
On 19/04/16 16:45, Mathieu Poirier wrote: > On 19 April 2016 at 07:19, Suzuki K Poulose <Suzuki.Poulose@arm.com> wrote: >> On 12/04/16 18:54, Mathieu Poirier wrote: >>> >>> Moving tmc_drvdata::enable to a local_t mode. That way the >>> sink interface is aware of it's orgin and the foundation for >>> mutual exclusion between the sysFS and Perf interface can be >>> laid out. >>> >>> Signed-off-by: Mathieu Poirier <mathieu.poirier@linaro.org> >>> --- >>> drivers/hwtracing/coresight/coresight-tmc-etf.c | 28 >>> ++++++++++++++++++------- >>> drivers/hwtracing/coresight/coresight-tmc-etr.c | 24 >>> ++++++++++++++++----- >>> drivers/hwtracing/coresight/coresight-tmc.h | 4 ++-- >>> 3 files changed, 42 insertions(+), 14 deletions(-) >>> >>> diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c >>> b/drivers/hwtracing/coresight/coresight-tmc-etf.c >>> index 7cb287ef7b9e..5908000e1ae0 100644 >>> --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c >>> +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c >>> @@ -110,6 +110,7 @@ static int tmc_enable_etf_sink(struct coresight_device >>> *csdev, u32 mode) >>> { >>> bool allocated = false; >>> char *buf = NULL; >>> + u32 val; >>> unsigned long flags; >>> struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent); >>> >>> @@ -146,6 +147,15 @@ fast_path: >>> return -EBUSY; >>> } >>> >>> + val = local_xchg(&drvdata->mode, mode); >>> + /* >>> + * In sysFS mode we can have multiple writers per sink. Since >>> this >>> + * sink is already enabled no memory is needed and the HW need not >>> be >>> + * touched. >>> + */ >>> + if (val == CS_MODE_SYSFS) >>> + goto out; >>> + >> >> >> We are not dropping the spinlock in this case. Are we ? > > Definitely - thanks for pointing that out. > >> >>> diff --git a/drivers/hwtracing/coresight/coresight-tmc.h >>> b/drivers/hwtracing/coresight/coresight-tmc.h >>> index 80096fa75326..821bdf150ac9 100644 >>> --- a/drivers/hwtracing/coresight/coresight-tmc.h >>> +++ b/drivers/hwtracing/coresight/coresight-tmc.h >>> @@ -100,7 +100,7 @@ enum tmc_mem_intf_width { >>> * @paddr: DMA start location in RAM. >>> * @vaddr: virtual representation of @paddr. >>> * @size: @buf size. >>> - * @enable: this TMC is being used. >>> + * @mode: how this TMC is being used. >>> * @config_type: TMC variant, must be of type @tmc_config_type. >>> * @trigger_cntr: amount of words to store after a trigger. >>> */ >>> @@ -116,7 +116,7 @@ struct tmc_drvdata { >>> dma_addr_t paddr; >>> void __iomem *vaddr; >>> u32 size; >>> - bool enable; >>> + local_t mode; >> >> >> Since we always deal with the mode under the spinlock, do we really need >> this to >> be local_t ? Or do we plan to get rid of the lock ? > > You are correct, using a local_t variable isn't strictly needed here. But: > > 1) It greatly simplifies the code. +1 > 2) I'm really hoping one day we can do without the spinlocks, though I > don't know when that will happen. > > Get back to me if you're resolute on switching the local_t to a normal type. No, leave it as it is, as you said, it does simplify the code.
diff --git a/drivers/hwtracing/coresight/coresight-tmc-etf.c b/drivers/hwtracing/coresight/coresight-tmc-etf.c index 7cb287ef7b9e..5908000e1ae0 100644 --- a/drivers/hwtracing/coresight/coresight-tmc-etf.c +++ b/drivers/hwtracing/coresight/coresight-tmc-etf.c @@ -110,6 +110,7 @@ static int tmc_enable_etf_sink(struct coresight_device *csdev, u32 mode) { bool allocated = false; char *buf = NULL; + u32 val; unsigned long flags; struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent); @@ -146,6 +147,15 @@ fast_path: return -EBUSY; } + val = local_xchg(&drvdata->mode, mode); + /* + * In sysFS mode we can have multiple writers per sink. Since this + * sink is already enabled no memory is needed and the HW need not be + * touched. + */ + if (val == CS_MODE_SYSFS) + goto out; + /* * If drvdata::buf isn't NULL, memory was allocated for a previous * trace run but wasn't read. If so simply zero-out the memory. @@ -163,9 +173,9 @@ fast_path: } tmc_etb_enable_hw(drvdata); - drvdata->enable = true; spin_unlock_irqrestore(&drvdata->spinlock, flags); +out: /* Free memory outside the spinlock if need be */ if (!allocated && buf) kfree(buf); @@ -176,6 +186,7 @@ fast_path: static void tmc_disable_etf_sink(struct coresight_device *csdev) { + u32 val; unsigned long flags; struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent); @@ -185,8 +196,11 @@ static void tmc_disable_etf_sink(struct coresight_device *csdev) return; } - tmc_etb_disable_hw(drvdata); - drvdata->enable = false; + val = local_xchg(&drvdata->mode, CS_MODE_DISABLED); + /* Disable the TMC only if it needs to */ + if (val != CS_MODE_DISABLED) + tmc_etb_disable_hw(drvdata); + spin_unlock_irqrestore(&drvdata->spinlock, flags); dev_info(drvdata->dev, "TMC-ETB/ETF disabled\n"); @@ -205,7 +219,7 @@ static int tmc_enable_etf_link(struct coresight_device *csdev, } tmc_etf_enable_hw(drvdata); - drvdata->enable = true; + local_set(&drvdata->mode, CS_MODE_SYSFS); spin_unlock_irqrestore(&drvdata->spinlock, flags); dev_info(drvdata->dev, "TMC-ETF enabled\n"); @@ -225,7 +239,7 @@ static void tmc_disable_etf_link(struct coresight_device *csdev, } tmc_etf_disable_hw(drvdata); - drvdata->enable = false; + local_set(&drvdata->mode, CS_MODE_DISABLED); spin_unlock_irqrestore(&drvdata->spinlock, flags); dev_info(drvdata->dev, "TMC disabled\n"); @@ -277,7 +291,7 @@ int tmc_read_prepare_etb(struct tmc_drvdata *drvdata) } /* Disable the TMC if need be */ - if (drvdata->enable) + if (local_read(&drvdata->mode) == CS_MODE_SYSFS) tmc_etb_disable_hw(drvdata); drvdata->reading = true; @@ -308,7 +322,7 @@ int tmc_read_unprepare_etb(struct tmc_drvdata *drvdata) } /* Re-enable the TMC if need be */ - if (drvdata->enable) { + if (local_read(&drvdata->mode) == CS_MODE_SYSFS) { /* * The trace run will continue with the same allocated trace * buffer. As such zero-out the buffer so that we don't end diff --git a/drivers/hwtracing/coresight/coresight-tmc-etr.c b/drivers/hwtracing/coresight/coresight-tmc-etr.c index 6022ff26deba..8e6fe267195a 100644 --- a/drivers/hwtracing/coresight/coresight-tmc-etr.c +++ b/drivers/hwtracing/coresight/coresight-tmc-etr.c @@ -85,6 +85,7 @@ static void tmc_etr_disable_hw(struct tmc_drvdata *drvdata) static int tmc_enable_etr_sink(struct coresight_device *csdev, u32 mode) { bool allocated = false; + u32 val; unsigned long flags; void __iomem *vaddr = NULL; dma_addr_t paddr; @@ -125,6 +126,15 @@ fast_path: return -EBUSY; } + val = local_xchg(&drvdata->mode, mode); + /* + * In sysFS mode we can have multiple writers per sink. Since this + * sink is already enabled no memory is needed and the HW need not be + * touched. + */ + if (val == CS_MODE_SYSFS) + goto out; + /* * If drvdata::buf == NULL, use the memory allocated above. * Otherwise a buffer still exists from a previous session, so @@ -140,9 +150,9 @@ fast_path: memset(drvdata->vaddr, 0, drvdata->size); tmc_etr_enable_hw(drvdata); - drvdata->enable = true; spin_unlock_irqrestore(&drvdata->spinlock, flags); +out: /* Free memory outside the spinlock if need be */ if (!allocated && vaddr) dma_free_coherent(drvdata->dev, drvdata->size, vaddr, paddr); @@ -153,6 +163,7 @@ fast_path: static void tmc_disable_etr_sink(struct coresight_device *csdev) { + u32 val; unsigned long flags; struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent); @@ -162,8 +173,11 @@ static void tmc_disable_etr_sink(struct coresight_device *csdev) return; } - tmc_etr_disable_hw(drvdata); - drvdata->enable = false; + val = local_xchg(&drvdata->mode, CS_MODE_DISABLED); + /* Disable the TMC only if it needs to */ + if (val != CS_MODE_DISABLED) + tmc_etr_disable_hw(drvdata); + spin_unlock_irqrestore(&drvdata->spinlock, flags); dev_info(drvdata->dev, "TMC-ETR disabled\n"); @@ -195,7 +209,7 @@ int tmc_read_prepare_etr(struct tmc_drvdata *drvdata) } /* Disable the TMC if need be */ - if (drvdata->enable) + if (local_read(&drvdata->mode) == CS_MODE_SYSFS) tmc_etr_disable_hw(drvdata); drvdata->reading = true; @@ -217,7 +231,7 @@ int tmc_read_unprepare_etr(struct tmc_drvdata *drvdata) spin_lock_irqsave(&drvdata->spinlock, flags); /* RE-enable the TMC if need be */ - if (drvdata->enable) { + if (local_read(&drvdata->mode) == CS_MODE_SYSFS) { /* * The trace run will continue with the same allocated trace * buffer. As such zero-out the buffer so that we don't end diff --git a/drivers/hwtracing/coresight/coresight-tmc.h b/drivers/hwtracing/coresight/coresight-tmc.h index 80096fa75326..821bdf150ac9 100644 --- a/drivers/hwtracing/coresight/coresight-tmc.h +++ b/drivers/hwtracing/coresight/coresight-tmc.h @@ -100,7 +100,7 @@ enum tmc_mem_intf_width { * @paddr: DMA start location in RAM. * @vaddr: virtual representation of @paddr. * @size: @buf size. - * @enable: this TMC is being used. + * @mode: how this TMC is being used. * @config_type: TMC variant, must be of type @tmc_config_type. * @trigger_cntr: amount of words to store after a trigger. */ @@ -116,7 +116,7 @@ struct tmc_drvdata { dma_addr_t paddr; void __iomem *vaddr; u32 size; - bool enable; + local_t mode; enum tmc_config_type config_type; u32 trigger_cntr; };
Moving tmc_drvdata::enable to a local_t mode. That way the sink interface is aware of it's orgin and the foundation for mutual exclusion between the sysFS and Perf interface can be laid out. Signed-off-by: Mathieu Poirier <mathieu.poirier@linaro.org> --- drivers/hwtracing/coresight/coresight-tmc-etf.c | 28 ++++++++++++++++++------- drivers/hwtracing/coresight/coresight-tmc-etr.c | 24 ++++++++++++++++----- drivers/hwtracing/coresight/coresight-tmc.h | 4 ++-- 3 files changed, 42 insertions(+), 14 deletions(-)