drm/tegra: dp: Track link capabilities alongside settings
authorThierry Reding <treding@nvidia.com>
Tue, 21 Jul 2015 14:33:48 +0000 (16:33 +0200)
committerThierry Reding <treding@nvidia.com>
Mon, 28 Oct 2019 10:18:44 +0000 (11:18 +0100)
Store capabilities in max_* fields and add separate fields for the
currently selected settings.

Signed-off-by: Thierry Reding <treding@nvidia.com>
drivers/gpu/drm/tegra/dp.c
drivers/gpu/drm/tegra/dp.h
drivers/gpu/drm/tegra/dpaux.c
drivers/gpu/drm/tegra/sor.c

index c19060b8753a939a14be09abcea54b82d0ac1ed9..e55efd46a7d98678339d5feeed5da59163e1e6e3 100644 (file)
@@ -14,9 +14,12 @@ static void drm_dp_link_reset(struct drm_dp_link *link)
                return;
 
        link->revision = 0;
-       link->rate = 0;
-       link->num_lanes = 0;
+       link->max_rate = 0;
+       link->max_lanes = 0;
        link->capabilities = 0;
+
+       link->rate = 0;
+       link->lanes = 0;
 }
 
 /**
@@ -42,12 +45,15 @@ int drm_dp_link_probe(struct drm_dp_aux *aux, struct drm_dp_link *link)
                return err;
 
        link->revision = values[0];
-       link->rate = drm_dp_bw_code_to_link_rate(values[1]);
-       link->num_lanes = values[2] & DP_MAX_LANE_COUNT_MASK;
+       link->max_rate = drm_dp_bw_code_to_link_rate(values[1]);
+       link->max_lanes = values[2] & DP_MAX_LANE_COUNT_MASK;
 
        if (values[2] & DP_ENHANCED_FRAME_CAP)
                link->capabilities |= DP_LINK_CAP_ENHANCED_FRAMING;
 
+       link->rate = link->max_rate;
+       link->lanes = link->max_lanes;
+
        return 0;
 }
 
@@ -131,7 +137,7 @@ int drm_dp_link_configure(struct drm_dp_aux *aux, struct drm_dp_link *link)
        int err;
 
        values[0] = drm_dp_link_rate_to_bw_code(link->rate);
-       values[1] = link->num_lanes;
+       values[1] = link->lanes;
 
        if (link->capabilities & DP_LINK_CAP_ENHANCED_FRAMING)
                values[1] |= DP_LANE_COUNT_ENHANCED_FRAME_EN;
index 1cf252e7309a4ad4eb1eaef0d6f4fabc1124a4b4..ec0342d4c95e166b6f1acbf2a6055e014c68dd1e 100644 (file)
@@ -12,17 +12,22 @@ struct drm_dp_aux;
 #define DP_LINK_CAP_ENHANCED_FRAMING (1 << 0)
 
 /**
- * struct drm_dp_link - DP link capabilities
+ * struct drm_dp_link - DP link capabilities and configuration
  * @revision: DP specification revision supported on the link
- * @rate: maximum clock rate supported on the link
- * @num_lanes: maximum number of lanes supported on the link
+ * @max_rate: maximum clock rate supported on the link
+ * @max_lanes: maximum number of lanes supported on the link
  * @capabilities: bitmask of capabilities supported on the link
+ * @rate: currently configured link rate
+ * @lanes: currently configured number of lanes
  */
 struct drm_dp_link {
        unsigned char revision;
-       unsigned int rate;
-       unsigned int num_lanes;
+       unsigned int max_rate;
+       unsigned int max_lanes;
        unsigned long capabilities;
+
+       unsigned int rate;
+       unsigned int lanes;
 };
 
 int drm_dp_link_probe(struct drm_dp_aux *aux, struct drm_dp_link *link);
index 883ed2f025c36f56fa9f64e46c0284c75abe5574..bd3361cea49b1df6ab2f2650a9fc38fbf983f085 100644 (file)
@@ -849,14 +849,14 @@ int drm_dp_aux_train(struct drm_dp_aux *aux, struct drm_dp_link *link,
        if (tp == DP_TRAINING_PATTERN_DISABLE)
                return 0;
 
-       for (i = 0; i < link->num_lanes; i++)
+       for (i = 0; i < link->lanes; i++)
                values[i] = DP_TRAIN_MAX_PRE_EMPHASIS_REACHED |
                            DP_TRAIN_PRE_EMPH_LEVEL_0 |
                            DP_TRAIN_MAX_SWING_REACHED |
                            DP_TRAIN_VOLTAGE_SWING_LEVEL_0;
 
        err = drm_dp_dpcd_write(aux, DP_TRAINING_LANE0_SET, values,
-                               link->num_lanes);
+                               link->lanes);
        if (err < 0)
                return err;
 
@@ -868,13 +868,13 @@ int drm_dp_aux_train(struct drm_dp_aux *aux, struct drm_dp_link *link,
 
        switch (tp) {
        case DP_TRAINING_PATTERN_1:
-               if (!drm_dp_clock_recovery_ok(status, link->num_lanes))
+               if (!drm_dp_clock_recovery_ok(status, link->lanes))
                        return -EAGAIN;
 
                break;
 
        case DP_TRAINING_PATTERN_2:
-               if (!drm_dp_channel_eq_ok(status, link->num_lanes))
+               if (!drm_dp_channel_eq_ok(status, link->lanes))
                        return -EAGAIN;
 
                break;
index 91d5c5041d2c10cd831bbd818f8a1a864b61b0f9..dca71250d88c20e417441c82484e186cd3cbeada 100644 (file)
@@ -650,7 +650,7 @@ static int tegra_sor_dp_train_fast(struct tegra_sor *sor,
        if (err < 0)
                return err;
 
-       for (i = 0, value = 0; i < link->num_lanes; i++) {
+       for (i = 0, value = 0; i < link->lanes; i++) {
                unsigned long lane = SOR_DP_TPG_CHANNEL_CODING |
                                     SOR_DP_TPG_SCRAMBLER_NONE |
                                     SOR_DP_TPG_PATTERN_TRAIN1;
@@ -671,7 +671,7 @@ static int tegra_sor_dp_train_fast(struct tegra_sor *sor,
        value |= SOR_DP_SPARE_MACRO_SOR_CLK;
        tegra_sor_writel(sor, value, SOR_DP_SPARE0);
 
-       for (i = 0, value = 0; i < link->num_lanes; i++) {
+       for (i = 0, value = 0; i < link->lanes; i++) {
                unsigned long lane = SOR_DP_TPG_CHANNEL_CODING |
                                     SOR_DP_TPG_SCRAMBLER_NONE |
                                     SOR_DP_TPG_PATTERN_TRAIN2;
@@ -686,7 +686,7 @@ static int tegra_sor_dp_train_fast(struct tegra_sor *sor,
        if (err < 0)
                return err;
 
-       for (i = 0, value = 0; i < link->num_lanes; i++) {
+       for (i = 0, value = 0; i < link->lanes; i++) {
                unsigned long lane = SOR_DP_TPG_CHANNEL_CODING |
                                     SOR_DP_TPG_SCRAMBLER_GALIOS |
                                     SOR_DP_TPG_PATTERN_NONE;
@@ -913,11 +913,11 @@ static int tegra_sor_compute_config(struct tegra_sor *sor,
        u32 num_syms_per_line;
        unsigned int i;
 
-       if (!link_rate || !link->num_lanes || !pclk || !config->bits_per_pixel)
+       if (!link_rate || !link->lanes || !pclk || !config->bits_per_pixel)
                return -EINVAL;
 
-       output = link_rate * 8 * link->num_lanes;
        input = pclk * config->bits_per_pixel;
+       output = link_rate * 8 * link->lanes;
 
        if (input >= output)
                return -ERANGE;
@@ -960,7 +960,7 @@ static int tegra_sor_compute_config(struct tegra_sor *sor,
        watermark = div_u64(watermark + params.error, f);
        config->watermark = watermark + (config->bits_per_pixel / 8) + 2;
        num_syms_per_line = (mode->hdisplay * config->bits_per_pixel) *
-                           (link->num_lanes * 8);
+                           (link->lanes * 8);
 
        if (config->watermark > 30) {
                config->watermark = 30;
@@ -980,12 +980,12 @@ static int tegra_sor_compute_config(struct tegra_sor *sor,
        if (link->capabilities & DP_LINK_CAP_ENHANCED_FRAMING)
                config->hblank_symbols -= 3;
 
-       config->hblank_symbols -= 12 / link->num_lanes;
+       config->hblank_symbols -= 12 / link->lanes;
 
        /* compute the number of symbols per vertical blanking interval */
        num = (mode->hdisplay - 25) * link_rate;
        config->vblank_symbols = div_u64(num, pclk);
-       config->vblank_symbols -= 36 / link->num_lanes + 4;
+       config->vblank_symbols -= 36 / link->lanes + 4;
 
        dev_dbg(sor->dev, "blank symbols: H:%u V:%u\n", config->hblank_symbols,
                config->vblank_symbols);
@@ -1831,17 +1831,17 @@ static void tegra_sor_edp_enable(struct drm_encoder *encoder)
        /* power DP lanes */
        value = tegra_sor_readl(sor, sor->soc->regs->dp_padctl0);
 
-       if (link.num_lanes <= 2)
+       if (link.lanes <= 2)
                value &= ~(SOR_DP_PADCTL_PD_TXD_3 | SOR_DP_PADCTL_PD_TXD_2);
        else
                value |= SOR_DP_PADCTL_PD_TXD_3 | SOR_DP_PADCTL_PD_TXD_2;
 
-       if (link.num_lanes <= 1)
+       if (link.lanes <= 1)
                value &= ~SOR_DP_PADCTL_PD_TXD_1;
        else
                value |= SOR_DP_PADCTL_PD_TXD_1;
 
-       if (link.num_lanes == 0)
+       if (link.lanes == 0)
                value &= ~SOR_DP_PADCTL_PD_TXD_0;
        else
                value |= SOR_DP_PADCTL_PD_TXD_0;
@@ -1850,7 +1850,7 @@ static void tegra_sor_edp_enable(struct drm_encoder *encoder)
 
        value = tegra_sor_readl(sor, SOR_DP_LINKCTL0);
        value &= ~SOR_DP_LINKCTL_LANE_COUNT_MASK;
-       value |= SOR_DP_LINKCTL_LANE_COUNT(link.num_lanes);
+       value |= SOR_DP_LINKCTL_LANE_COUNT(link.lanes);
        tegra_sor_writel(sor, value, SOR_DP_LINKCTL0);
 
        /* start lane sequencer */
@@ -1907,7 +1907,7 @@ static void tegra_sor_edp_enable(struct drm_encoder *encoder)
                dev_err(sor->dev, "failed to configure eDP link: %d\n", err);
 
        rate = drm_dp_link_rate_to_bw_code(link.rate);
-       lanes = link.num_lanes;
+       lanes = link.lanes;
 
        value = tegra_sor_readl(sor, SOR_CLK_CNTRL);
        value &= ~SOR_CLK_CNTRL_DP_LINK_SPEED_MASK;
@@ -1925,7 +1925,7 @@ static void tegra_sor_edp_enable(struct drm_encoder *encoder)
 
        /* disable training pattern generator */
 
-       for (i = 0; i < link.num_lanes; i++) {
+       for (i = 0; i < link.lanes; i++) {
                unsigned long lane = SOR_DP_TPG_CHANNEL_CODING |
                                     SOR_DP_TPG_SCRAMBLER_GALIOS |
                                     SOR_DP_TPG_PATTERN_NONE;