[PATCH v5 2/3] drm/bridge: ti-sn65dsi83: Add ti,lvds-vod-swing optional properties
Andrej Picej
andrej.picej at norik.com
Thu Dec 12 08:08:03 UTC 2024
On 12. 12. 24 00:04, Dmitry Baryshkov wrote:
> On Wed, Dec 11, 2024 at 08:57:17AM +0100, Andrej Picej wrote:
>>
>>
>> On 10. 12. 24 14:59, Dmitry Baryshkov wrote:
>>> On Tue, Dec 10, 2024 at 02:41:01PM +0100, Andrej Picej wrote:
>>>>
>>>>
>>>> On 10. 12. 24 12:43, Dmitry Baryshkov wrote:
>>>>> On Tue, Dec 10, 2024 at 10:19:00AM +0100, Andrej Picej wrote:
>>>>>> Add a optional properties to change LVDS output voltage. This should not
>>>>>> be static as this depends mainly on the connected display voltage
>>>>>> requirement. We have three properties:
>>>>>> - "ti,lvds-termination-ohms", which sets near end termination,
>>>>>> - "ti,lvds-vod-swing-data-microvolt" and
>>>>>> - "ti,lvds-vod-swing-clock-microvolt" which both set LVDS differential
>>>>>> output voltage for data and clock lanes. They are defined as an array
>>>>>> with min and max values. The appropriate bitfield will be set if
>>>>>> selected constraints can be met.
>>>>>>
>>>>>> If "ti,lvds-termination-ohms" is not defined the default of 200 Ohm near
>>>>>> end termination will be used. Selecting only one:
>>>>>> "ti,lvds-vod-swing-data-microvolt" or
>>>>>> "ti,lvds-vod-swing-clock-microvolt" can be done, but the output voltage
>>>>>> constraint for only data/clock lanes will be met. Setting both is
>>>>>> recommended.
>>>>>>
>>>>>> Signed-off-by: Andrej Picej <andrej.picej at norik.com>
>>>>>> ---
>>>>>> Changes in v5:
>>>>>> - specify default values in sn65dsi83_parse_lvds_endpoint,
>>>>>> - move sn65dsi83_parse_lvds_endpoint for channel B up, outside if,
>>>>>> Changes in v4:
>>>>>> - fix typo in commit message bitfiled -> bitfield
>>>>>> - use arrays (lvds_vod_swing_conf and lvds_term_conf) in private data, instead
>>>>>> of separate variables for channel A/B
>>>>>> - add more checks on return value of "of_property_read_u32_array"
>>>>>> Changes in v3:
>>>>>> - use microvolts for default array values 1000 mV -> 1000000 uV.
>>>>>> Changes in v2:
>>>>>> - use datasheet tables to get the proper configuration
>>>>>> - since major change was done change the authorship to myself
>>>>>> ---
>>>>>> drivers/gpu/drm/bridge/ti-sn65dsi83.c | 142 +++++++++++++++++++++++++-
>>>>>> 1 file changed, 139 insertions(+), 3 deletions(-)
>>>>>>
>>>>>> diff --git a/drivers/gpu/drm/bridge/ti-sn65dsi83.c b/drivers/gpu/drm/bridge/ti-sn65dsi83.c
>>>>>> index 57a7ed13f996..f9578b38da28 100644
>>>>>> --- a/drivers/gpu/drm/bridge/ti-sn65dsi83.c
>>>>>> +++ b/drivers/gpu/drm/bridge/ti-sn65dsi83.c
>>>>>> @@ -132,6 +132,16 @@
>>>>>> #define REG_IRQ_STAT_CHA_SOT_BIT_ERR BIT(2)
>>>>>> #define REG_IRQ_STAT_CHA_PLL_UNLOCK BIT(0)
>>>>>> +enum sn65dsi83_channel {
>>>>>> + CHANNEL_A,
>>>>>> + CHANNEL_B
>>>>>> +};
>>>>>> +
>>>>>> +enum sn65dsi83_lvds_term {
>>>>>> + OHM_100,
>>>>>> + OHM_200
>>>>>> +};
>>>>>> +
>>>>>> enum sn65dsi83_model {
>>>>>> MODEL_SN65DSI83,
>>>>>> MODEL_SN65DSI84,
>>>>>> @@ -147,6 +157,8 @@ struct sn65dsi83 {
>>>>>> struct regulator *vcc;
>>>>>> bool lvds_dual_link;
>>>>>> bool lvds_dual_link_even_odd_swap;
>>>>>> + int lvds_vod_swing_conf[2];
>>>>>> + int lvds_term_conf[2];
>>>>>> };
>>>>>> static const struct regmap_range sn65dsi83_readable_ranges[] = {
>>>>>> @@ -237,6 +249,36 @@ static const struct regmap_config sn65dsi83_regmap_config = {
>>>>>> .max_register = REG_IRQ_STAT,
>>>>>> };
>>>>>> +static const int lvds_vod_swing_data_table[2][4][2] = {
>>>>>> + { /* 100 Ohm */
>>>>>> + { 180000, 313000 },
>>>>>> + { 215000, 372000 },
>>>>>> + { 250000, 430000 },
>>>>>> + { 290000, 488000 },
>>>>>> + },
>>>>>> + { /* 200 Ohm */
>>>>>> + { 150000, 261000 },
>>>>>> + { 200000, 346000 },
>>>>>> + { 250000, 428000 },
>>>>>> + { 300000, 511000 },
>>>>>> + },
>>>>>> +};
>>>>>> +
>>>>>> +static const int lvds_vod_swing_clock_table[2][4][2] = {
>>>>>> + { /* 100 Ohm */
>>>>>> + { 140000, 244000 },
>>>>>> + { 168000, 290000 },
>>>>>> + { 195000, 335000 },
>>>>>> + { 226000, 381000 },
>>>>>> + },
>>>>>> + { /* 200 Ohm */
>>>>>> + { 117000, 204000 },
>>>>>> + { 156000, 270000 },
>>>>>> + { 195000, 334000 },
>>>>>> + { 234000, 399000 },
>>>>>> + },
>>>>>> +};
>>>>>> +
>>>>>> static struct sn65dsi83 *bridge_to_sn65dsi83(struct drm_bridge *bridge)
>>>>>> {
>>>>>> return container_of(bridge, struct sn65dsi83, bridge);
>>>>>> @@ -435,12 +477,16 @@ static void sn65dsi83_atomic_pre_enable(struct drm_bridge *bridge,
>>>>>> val |= REG_LVDS_FMT_LVDS_LINK_CFG;
>>>>>> regmap_write(ctx->regmap, REG_LVDS_FMT, val);
>>>>>> - regmap_write(ctx->regmap, REG_LVDS_VCOM, 0x05);
>>>>>> + regmap_write(ctx->regmap, REG_LVDS_VCOM,
>>>>>> + REG_LVDS_VCOM_CHA_LVDS_VOD_SWING(ctx->lvds_vod_swing_conf[CHANNEL_A]) |
>>>>>> + REG_LVDS_VCOM_CHB_LVDS_VOD_SWING(ctx->lvds_vod_swing_conf[CHANNEL_B]));
>>>>>> regmap_write(ctx->regmap, REG_LVDS_LANE,
>>>>>> (ctx->lvds_dual_link_even_odd_swap ?
>>>>>> REG_LVDS_LANE_EVEN_ODD_SWAP : 0) |
>>>>>> - REG_LVDS_LANE_CHA_LVDS_TERM |
>>>>>> - REG_LVDS_LANE_CHB_LVDS_TERM);
>>>>>> + (ctx->lvds_term_conf[CHANNEL_A] ?
>>>>>> + REG_LVDS_LANE_CHA_LVDS_TERM : 0) |
>>>>>> + (ctx->lvds_term_conf[CHANNEL_B] ?
>>>>>> + REG_LVDS_LANE_CHB_LVDS_TERM : 0));
>>>>>> regmap_write(ctx->regmap, REG_LVDS_CM, 0x00);
>>>>>> le16val = cpu_to_le16(mode->hdisplay);
>>>>>> @@ -576,10 +622,100 @@ static const struct drm_bridge_funcs sn65dsi83_funcs = {
>>>>>> .atomic_get_input_bus_fmts = sn65dsi83_atomic_get_input_bus_fmts,
>>>>>> };
>>>>>> +static int sn65dsi83_select_lvds_vod_swing(struct device *dev,
>>>>>> + u32 lvds_vod_swing_data[2], u32 lvds_vod_swing_clk[2], u8 lvds_term)
>>>>>> +{
>>>>>> + int i;
>>>>>> +
>>>>>> + for (i = 0; i <= 3; i++) {
>>>>>> + if (lvds_vod_swing_data_table[lvds_term][i][0] >= lvds_vod_swing_data[0] &&
>>>>>> + lvds_vod_swing_data_table[lvds_term][i][1] <= lvds_vod_swing_data[1] &&
>>>>>> + lvds_vod_swing_clock_table[lvds_term][i][0] >= lvds_vod_swing_clk[0] &&
>>>>>> + lvds_vod_swing_clock_table[lvds_term][i][1] <= lvds_vod_swing_clk[1])
>>>>>> + return i;
>>>>>> + }
>>>>>> +
>>>>>> + dev_err(dev, "failed to find appropriate LVDS_VOD_SWING configuration\n");
>>>>>> + return -EINVAL;
>>>>>> +}
>>>>>> +
>>>>>> +static int sn65dsi83_parse_lvds_endpoint(struct sn65dsi83 *ctx, int channel)
>>>>>> +{
>>>>>> + struct device *dev = ctx->dev;
>>>>>> + struct device_node *endpoint;
>>>>>> + int endpoint_reg;
>>>>>> + /* Set so the property can be freely selected if not defined */
>>>>>> + u32 lvds_vod_swing_data[2] = { 0, 1000000 };
>>>>>> + u32 lvds_vod_swing_clk[2] = { 0, 1000000 };
>>>>>> + u32 lvds_term;
>>>>>> + u8 lvds_term_conf = 0x1;
>>>>>> + int lvds_vod_swing_conf = 0x1;
>>>>>
>>>>> Magic values
>>>>
>>>> Can you please elaborate.
>>>>
>>>> I can use:
>>>> u8 lvds_term_conf = OHM_200;
>>>>
>>>> What about lvds_vod_swing_conf? Should I create additional define for it?
>>>> But this doesn't solve a hidden meaning? Maybe additional comment above?
>>>> Would like to avoid using voltages for it, since then we are reverse
>>>> engineering the table in datasheet to match the default reg value.
>>>
>>> I think the following example solves both problems:
>>>
>>> lvds_term = 200;
>>> of_property_read_u32(..., &lvds_term);
>>>
>>> if (lvds_term == 100)
>>> ctx->lvds_term_conf[channel] = OHM_100;
>>> else if (lvds_term == 200)
>>> ctx->lvds_term_conf[channel] = OHM_200;
>>> else
>>> return -EINVAL;
>>>
>>> The same approach can be applied to lvds_vod_swing_conf, resulting in
>>> removal of magic values.
>>
>> Sorry, but I think it is not that easy when it comes to the
>> lvds_vod_swing_conf. We should assign default value if
>> "ti,lvds-vod-swing-data-microvolt" and "ti,lvds-vod-swing-clock-microvolt"
>> are not defined. Default value of the lvds_vod_swing_conf is 0x1, but this
>> doesn't have any straight forward meaning like OHM_200 for example.
>>
>> What we can do in that case is that we copy the values from defined
>> datasheet tables to the "lvds_vod_swing_data[2]" and "lvds_vod_swing_clk[2]"
>> arrays and then run the
>> sn65dsi83_select_lvds_vod_swing with it, which will return the default value
>> (0x1).
>>
>> /* If both properties are not defined assign default limits */
>> if (ret_data && ret_clock) {
>> memcpy(lvds_vod_swing_data,
>> lvds_vod_swing_data_table[ctx->lvds_term_conf[channel]][1],
>> sizeof(lvds_vod_swing_data));
>> memcpy(lvds_vod_swing_clk,
>> lvds_vod_swing_clock_table[ctx->lvds_term_conf[channel]][1],
>> sizeof(lvds_vod_swing_clk));
>> }
>> lvds_vod_swing_conf = sn65dsi83_select_lvds_vod_swing(dev,
>> lvds_vod_swing_data, lvds_vod_swing_clk,
>> ctx->lvds_term_conf[channel]);
>> if (lvds_vod_swing_conf < 0) {
>> ret = lvds_vod_swing_conf;
>> goto exit;
>> }
>>
>> ctx->lvds_vod_swing_conf[channel] = lvds_vod_swing_conf;
>>
>> I'm not sure if using this approach gets rid of the problem with magic
>> values.
>> Or maybe I'm not seeing the obvious solution so please bear with me.
>
> Yes, the defaults (0..1000000) should be fixed to result in the same
> value (0x01) as if the property wasn't specified at all
The defaults (0..1000000) is selected because in case if only one
property is defined in dts (ti,lvds-vod-swing-data-microvolt or
ti,lvds-vod-swing-clock-microvolt) the other array values don't effect
the decision which "lvds_vod_swing_conf" is selected. That's why we
initialized the array to be out off bounds of the datasheet tables, all
values in the table match the not defined property, so
lvds_vod_swing_conf is selected purely on the basis of the defined property.
Example:
DTS
ti,lvds-vod-swing-data-microvolt = <250000 428000>;
//ti,lvds-vod-swing-clock-microvolt NOT DEFINED;
After parsing the devicetree we will get:
lvds_vod_swing_data = [ 250000, 428000 ]
lvds_vod_swing_clk = [ 0, 1000000 ]
In sn65dsi83_select_lvds_vod_swing lvds_vod_swing_clk[] values don't
effect the decision making since
lvds_vod_swing_clock_table[lvds_term][i][0] >= lvds_vod_swing_clk[0] &&
lvds_vod_swing_clock_table[lvds_term][i][1] <= lvds_vod_swing_clk[1]
is always true.
>
> I think the following should work:
>
> /* artifical values to select the defaults in both cases */
> u32 lvds_vod_swing_data[2] = { 190000, 330000 };
> u32 lvds_vod_swing_clk[2] = { 150000, 250000 };
This sets the default to 0x0. It should be:
u32 lvds_vod_swing_data[2] = { 200000, 372000 };
u32 lvds_vod_swing_clk[2] = { 156000, 290000 };
This selects the default 0x1 in both cases, if termination is 100 or 200
Ohms.
Nevertheless I think I got your point. But I would still like to give
the user the freedom to only specify one property if maybe connected
panel only has limits on data lanes/clock lane.
So maybe set the arrays lvds_vod_swing_data/clk to [0, 1000000] if
of_property_read_u32_array returns -EINVAL (property does not exist).
What do you say?
>
> Yes, they are artificial, as stated in the comment. Yes, I think it's
> better than special-casing in the property handling.
>
More information about the dri-devel
mailing list