[RFC 04/14] lib: sbi: Detect number of bits implemented in mhpmcounter

Atish Patra atishp at atishpatra.org
Sun Apr 18 12:45:48 BST 2021


On Wed, Apr 7, 2021 at 8:22 PM Anup Patel <Anup.Patel at wdc.com> wrote:
>
>
>
> > -----Original Message-----
> > From: Atish Patra <atish.patra at wdc.com>
> > Sent: 20 March 2021 03:43
> > To: opensbi at lists.infradead.org
> > Cc: Atish Patra <Atish.Patra at wdc.com>; Anup Patel <Anup.Patel at wdc.com>
> > Subject: [RFC 04/14] lib: sbi: Detect number of bits implemented in
> > mhpmcounter
> >
> > RISC-V privilege specification allows the implementation to have less than 64
> > bits.
> >
> > Add a function to detect the number of implemented bits in mhpmcounter
> > dynamically at runtime.
> >
> > Signed-off-by: Atish Patra <atish.patra at wdc.com>
> > ---
> >  include/sbi/sbi_hart.h |  1 +
> >  lib/sbi/sbi_hart.c     | 49
> > ++++++++++++++++++++++++++++++++++++++++++
> >  2 files changed, 50 insertions(+)
> >
> > diff --git a/include/sbi/sbi_hart.h b/include/sbi/sbi_hart.h index
> > 9e317c52008c..5f23595ccb27 100644
> > --- a/include/sbi/sbi_hart.h
> > +++ b/include/sbi/sbi_hart.h
> > @@ -44,6 +44,7 @@ void sbi_hart_delegation_dump(struct sbi_scratch
> > *scratch,  unsigned int sbi_hart_pmp_count(struct sbi_scratch *scratch);
> > unsigned long sbi_hart_pmp_granularity(struct sbi_scratch *scratch);
> > unsigned int sbi_hart_pmp_addrbits(struct sbi_scratch *scratch);
> > +unsigned int sbi_hart_pmu_event_bits(struct sbi_scratch *scratch);
> >  int sbi_hart_pmp_configure(struct sbi_scratch *scratch);  bool
> > sbi_hart_has_feature(struct sbi_scratch *scratch, unsigned long feature);
> > void sbi_hart_get_features_str(struct sbi_scratch *scratch, diff --git
> > a/lib/sbi/sbi_hart.c b/lib/sbi/sbi_hart.c index ee11b2133d05..006ec830d86c
> > 100644
> > --- a/lib/sbi/sbi_hart.c
> > +++ b/lib/sbi/sbi_hart.c
> > @@ -33,6 +33,7 @@ struct hart_features {
> >       unsigned int pmp_addr_bits;
> >       unsigned long pmp_gran;
> >       unsigned int mhpm_count;
> > +     unsigned int pmu_bits;
>
> Rename pmu_bits to mhpm_bits for consistency.
>
> >  };
> >  static unsigned long hart_features_offset;
> >
> > @@ -177,6 +178,14 @@ unsigned int sbi_hart_pmp_addrbits(struct
> > sbi_scratch *scratch)
> >       return hfeatures->pmp_addr_bits;
> >  }
> >
> > +unsigned int sbi_hart_pmu_event_bits(struct sbi_scratch *scratch) {
>
> Rename sbi_hart_pmu_event_bits() to sbi_hart_mhpm_bits()
> for consistency.
>

Will do.

> > +     struct hart_features *hfeatures =
> > +                     sbi_scratch_offset_ptr(scratch,
> > hart_features_offset);
> > +
> > +     return hfeatures->pmu_bits;
> > +}
> > +
> >  int sbi_hart_pmp_configure(struct sbi_scratch *scratch)  {
> >       struct sbi_domain_memregion *reg;
> > @@ -330,6 +339,37 @@ static unsigned long
> > hart_pmp_get_allowed_addr(void)
> >       return val;
> >  }
> >
> > +static int hart_pmu_get_allowed_bits(void) {
> > +     unsigned long val = ~(0UL);
> > +     struct sbi_trap_info trap = {0};
> > +     int num_bits = 0;
> > +
> > +     /**
> > +      * It is assumed that platforms will implement same number of bits
> > for
> > +      * all the performance counters including mcycle/minstret.
> > +      */
> > +     csr_write_allowed(CSR_MHPMCOUNTER3, (ulong)&trap, val);
> > +     if (!trap.cause) {
> > +             val = csr_read_allowed(CSR_MHPMCOUNTER3,
> > (ulong)&trap);
> > +             if (trap.cause)
> > +                     num_bits = 0;
>
> Return 0 here instead of setting num_bits to zero.
>

ok.

> > +     }
> > +     num_bits = __fls(val) + 1;
> > +#if __riscv_xlen == 32
> > +     csr_write_allowed(CSR_MHPMCOUNTER3H, (ulong)&trap, val);
> > +     if (!trap.cause) {
> > +             val = csr_read_allowed(CSR_MHPMCOUNTER3H,
> > (ulong)&trap);
> > +             if (trap.cause)
> > +                     num_bits = 0;
>
> Return num_bits here.
>
ok.
> > +     }
> > +     num_bits = __fls(val) + 1;
>
> This should be "num_bits += __fls(val) + 1;"
>

Ah yes. Thanks for pointing out this.

> > +
> > +#endif
> > +
> > +     return num_bits;
> > +}
> > +
> >  static void hart_detect_features(struct sbi_scratch *scratch)  {
> >       struct sbi_trap_info trap = {0};
> > @@ -394,10 +434,19 @@ static void hart_detect_features(struct sbi_scratch
> > *scratch)
> >  __pmp_skip:
> >
> >       /* Detect number of MHPM counters */
> > +     trap.cause = 0;
> >       __check_csr(CSR_MHPMCOUNTER3, 0, 1UL, mhpm_count,
> > __mhpm_skip);
> > +     hfeatures->pmu_bits = hart_pmu_get_allowed_bits();
> > +
> >       __check_csr_4(CSR_MHPMCOUNTER4, 0, 1UL, mhpm_count,
> > __mhpm_skip);
> >       __check_csr_8(CSR_MHPMCOUNTER8, 0, 1UL, mhpm_count,
> > __mhpm_skip);
> >       __check_csr_16(CSR_MHPMCOUNTER16, 0, 1UL, mhpm_count,
> > __mhpm_skip);
> > +
> > +     /**
> > +      * No need to check for MHPMCOUNTERH for RV32 as they are
> > expected to be
> > +      * implemented if MHPMCOUNTER is implemented.
> > +      */
> > +
> >  __mhpm_skip:
> >
> >  #undef __check_csr_64
> > --
> > 2.25.1
>
> Regards,
> Anup
>
>
> --
> opensbi mailing list
> opensbi at lists.infradead.org
> http://lists.infradead.org/mailman/listinfo/opensbi



-- 
Regards,
Atish



More information about the opensbi mailing list