net: ipa: support a third pulse register
authorAlex Elder <elder@linaro.org>
Mon, 30 Jan 2023 21:01:57 +0000 (15:01 -0600)
committerJakub Kicinski <kuba@kernel.org>
Wed, 1 Feb 2023 05:45:52 +0000 (21:45 -0800)
The AP has third pulse generator available starting with IPA v5.0.
Redefine ipa_qtime_val() to support that possibility.  Pass the IPA
pointer as an argument so the version can be determined.  And stop
using the sign of the returned tick count to indicate which of two
pulse generators to use.

Instead, have the caller provide the address of a variable that will
hold the selected pulse generator for the Qtime value.  And for
version 5.0, check whether the third pulse generator best represents
the time period.

Add code in ipa_qtime_config() to configure the fourth pulse
generator for IPA v5.0+; in that case configure both the third and
fourth pulse generators to use 10 msec granularity.

Consistently use "ticks" for local variables that represent a tick
count.

Signed-off-by: Alex Elder <elder@linaro.org>
Reviewed-by: Leon Romanovsky <leonro@nvidia.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/ipa/ipa_endpoint.c
drivers/net/ipa/ipa_main.c

index c029209191d41c2e8a821067078f491a6fab1426..798dfa4484d5a19fffccf564e35524ab5839450f 100644 (file)
@@ -922,64 +922,72 @@ static void ipa_endpoint_init_mode(struct ipa_endpoint *endpoint)
        iowrite32(val, ipa->reg_virt + offset);
 }
 
-/* For IPA v4.5+, times are expressed using Qtime.  The AP uses one of two
- * pulse generators (0 and 1) to measure elapsed time.  In ipa_qtime_config()
- * they're configured to have granularity 100 usec and 1 msec, respectively.
- *
- * The return value is the positive or negative Qtime value to use to
- * express the (microsecond) time provided.  A positive return value
- * means pulse generator 0 can be used; otherwise use pulse generator 1.
+/* For IPA v4.5+, times are expressed using Qtime.  A time is represented
+ * at one of several available granularities, which are configured in
+ * ipa_qtime_config().  Three (or, starting with IPA v5.0, four) pulse
+ * generators are set up with different "tick" periods.  A Qtime value
+ * encodes a tick count along with an indication of a pulse generator
+ * (which has a fixed tick period).  Two pulse generators are always
+ * available to the AP; a third is available starting with IPA v5.0.
+ * This function determines which pulse generator most accurately
+ * represents the time period provided, and returns the tick count to
+ * use to represent that time.
  */
-static int ipa_qtime_val(u32 microseconds, u32 max)
-{
-       u32 val;
-
-       /* Use 100 microsecond granularity if possible */
-       val = DIV_ROUND_CLOSEST(microseconds, 100);
-       if (val <= max)
-               return (int)val;
-
-       /* Have to use pulse generator 1 (millisecond granularity) */
-       val = DIV_ROUND_CLOSEST(microseconds, 1000);
-       WARN_ON(val > max);
+static u32
+ipa_qtime_val(struct ipa *ipa, u32 microseconds, u32 max, u32 *select)
+{
+       u32 which = 0;
+       u32 ticks;
+
+       /* Pulse generator 0 has 100 microsecond granularity */
+       ticks = DIV_ROUND_CLOSEST(microseconds, 100);
+       if (ticks <= max)
+               goto out;
+
+       /* Pulse generator 1 has millisecond granularity */
+       which = 1;
+       ticks = DIV_ROUND_CLOSEST(microseconds, 1000);
+       if (ticks <= max)
+               goto out;
+
+       if (ipa->version >= IPA_VERSION_5_0) {
+               /* Pulse generator 2 has 10 millisecond granularity */
+               which = 2;
+               ticks = DIV_ROUND_CLOSEST(microseconds, 100);
+       }
+       WARN_ON(ticks > max);
+out:
+       *select = which;
 
-       return (int)-val;
+       return ticks;
 }
 
 /* Encode the aggregation timer limit (microseconds) based on IPA version */
 static u32 aggr_time_limit_encode(struct ipa *ipa, const struct ipa_reg *reg,
                                  u32 microseconds)
 {
+       u32 ticks;
        u32 max;
-       u32 val;
 
        if (!microseconds)
                return 0;       /* Nothing to compute if time limit is 0 */
 
        max = ipa_reg_field_max(reg, TIME_LIMIT);
        if (ipa->version >= IPA_VERSION_4_5) {
-               u32 gran_sel;
-               int ret;
-
-               /* Compute the Qtime limit value to use */
-               ret = ipa_qtime_val(microseconds, max);
-               if (ret < 0) {
-                       val = -ret;
-                       gran_sel = ipa_reg_encode(reg, AGGR_GRAN_SEL, 1);
-               } else {
-                       val = ret;
-                       gran_sel = 0;
-               }
+               u32 select;
+
+               ticks = ipa_qtime_val(ipa, microseconds, max, &select);
 
-               return gran_sel | ipa_reg_encode(reg, TIME_LIMIT, val);
+               return ipa_reg_encode(reg, AGGR_GRAN_SEL, select) |
+                      ipa_reg_encode(reg, TIME_LIMIT, ticks);
        }
 
        /* We program aggregation granularity in ipa_hardware_config() */
-       val = DIV_ROUND_CLOSEST(microseconds, IPA_AGGR_GRANULARITY);
-       WARN(val > max, "aggr_time_limit too large (%u > %u usec)\n",
+       ticks = DIV_ROUND_CLOSEST(microseconds, IPA_AGGR_GRANULARITY);
+       WARN(ticks > max, "aggr_time_limit too large (%u > %u usec)\n",
             microseconds, max * IPA_AGGR_GRANULARITY);
 
-       return ipa_reg_encode(reg, TIME_LIMIT, val);
+       return ipa_reg_encode(reg, TIME_LIMIT, ticks);
 }
 
 static void ipa_endpoint_init_aggr(struct ipa_endpoint *endpoint)
@@ -1050,20 +1058,13 @@ static u32 hol_block_timer_encode(struct ipa *ipa, const struct ipa_reg *reg,
 
        if (ipa->version >= IPA_VERSION_4_5) {
                u32 max = ipa_reg_field_max(reg, TIMER_LIMIT);
-               u32 gran_sel;
-               int ret;
-
-               /* Compute the Qtime limit value to use */
-               ret = ipa_qtime_val(microseconds, max);
-               if (ret < 0) {
-                       val = -ret;
-                       gran_sel = ipa_reg_encode(reg, TIMER_GRAN_SEL, 1);
-               } else {
-                       val = ret;
-                       gran_sel = 0;
-               }
+               u32 select;
+               u32 ticks;
+
+               ticks = ipa_qtime_val(ipa, microseconds, max, &select);
 
-               return gran_sel | ipa_reg_encode(reg, TIMER_LIMIT, val);
+               return ipa_reg_encode(reg, TIMER_GRAN_SEL, 1) |
+                      ipa_reg_encode(reg, TIMER_LIMIT, ticks);
        }
 
        /* Use 64 bit arithmetic to avoid overflow */
index f3466b913394c6a130c29bb9976604e1d33a87b8..60d7c558163f187873ec31ae4057d22e710e761e 100644 (file)
@@ -390,7 +390,12 @@ static void ipa_qtime_config(struct ipa *ipa)
        reg = ipa_reg(ipa, TIMERS_PULSE_GRAN_CFG);
        val = ipa_reg_encode(reg, PULSE_GRAN_0, IPA_GRAN_100_US);
        val |= ipa_reg_encode(reg, PULSE_GRAN_1, IPA_GRAN_1_MS);
-       val |= ipa_reg_encode(reg, PULSE_GRAN_2, IPA_GRAN_1_MS);
+       if (ipa->version >= IPA_VERSION_5_0) {
+               val |= ipa_reg_encode(reg, PULSE_GRAN_2, IPA_GRAN_10_MS);
+               val |= ipa_reg_encode(reg, PULSE_GRAN_3, IPA_GRAN_10_MS);
+       } else {
+               val |= ipa_reg_encode(reg, PULSE_GRAN_2, IPA_GRAN_1_MS);
+       }
 
        iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));