Hi Uwe,
On 7/7/20 9:50 AM, Uwe Kleine-König wrote:
Hello Hans,
On Sat, Jun 20, 2020 at 02:17:58PM +0200, Hans de Goede wrote:
Now that the PWM drivers which we use have been converted to the atomic PWM API, we can move the i915 panel code over to using the atomic PWM API.
Note that there is no hard dependency, the atomic API should work just fine even with a non-converted driver. (Of course a converted driver behaves better, but the i915 using the atomic API with a non-converted driver is just as good as the legacy API with the same driver.)
So regarding your plan to get this series in soon: There is no hard need to halt the efforts for the drm part if the pwm patches take a bit longer (or vice versa).
I understand, but the main reason to do the conversion to the atomic API, is to be able to skip the step where we force the backlight to 100% brightness (which can look quite ugly during boot).
After this patch the intel-panel code initializes its internal backlight state and the brightness reported under /sys/class/backlight with the "brightness" returned from the PWM-drivers' get_state callback.
Without getting the PWM patches in first I think that things will mostly work, but we will always report an initial brightness value of 0. Lets say it is actually 50% and the user then presses the increase-brightness hotkey, causing userspace to change it from 0% to 5% so instead of increasing it by 1/20th, it just decreased it a lot.
So I do believe it is better to get the whole series in as a whole, since we are at rc4 already (time flies) I guess it might not make it in this cycle, but that is fine.
Talking about merging this, is it ok for me to push the entire series upstream through the intel-drm-next-queued branch, once all the PWM patches have your Ack?
The removes a long standing FIXME and this removes a flicker where the backlight brightness would jump to 100% when i915 loads even if using the fastset path.
Signed-off-by: Hans de Goede hdegoede@redhat.com
.../drm/i915/display/intel_display_types.h | 3 +- drivers/gpu/drm/i915/display/intel_panel.c | 73 +++++++++---------- 2 files changed, 37 insertions(+), 39 deletions(-)
diff --git a/drivers/gpu/drm/i915/display/intel_display_types.h b/drivers/gpu/drm/i915/display/intel_display_types.h index de32f9efb120..4bd9981e70a1 100644 --- a/drivers/gpu/drm/i915/display/intel_display_types.h +++ b/drivers/gpu/drm/i915/display/intel_display_types.h @@ -28,6 +28,7 @@
#include <linux/async.h> #include <linux/i2c.h> +#include <linux/pwm.h> #include <linux/sched/clock.h>
#include <drm/drm_atomic.h> @@ -223,7 +224,7 @@ struct intel_panel { bool util_pin_active_low; /* bxt+ */ u8 controller; /* bxt+ only */ struct pwm_device *pwm;
int pwm_period_ns;
struct pwm_state pwm_state;
/* DPCD backlight */ u8 pwmgen_bit_count;
diff --git a/drivers/gpu/drm/i915/display/intel_panel.c b/drivers/gpu/drm/i915/display/intel_panel.c index cb28b9908ca4..a0f76343f381 100644 --- a/drivers/gpu/drm/i915/display/intel_panel.c +++ b/drivers/gpu/drm/i915/display/intel_panel.c @@ -592,10 +592,11 @@ static u32 bxt_get_backlight(struct intel_connector *connector) static u32 pwm_get_backlight(struct intel_connector *connector) { struct intel_panel *panel = &connector->panel;
- int duty_ns;
int duty_ns, period_ns;
duty_ns = pwm_get_duty_cycle(panel->backlight.pwm);
- return DIV_ROUND_UP(duty_ns * 100, panel->backlight.pwm_period_ns);
- period_ns = pwm_get_period(panel->backlight.pwm);
The transformation is correct, but using
pwm_get_state(pwm, &state); duty_ns = state.duty_cycle; period_ns = state.period;
is a bit more effective as it calls pwm_get_state() only once.
There is a function pwm_get_relative_duty_cycle which is similar (with scale = 100) just used different rounding.
Ah nice, that is better then doing our own stuff. I will switch to that for v4 of this patch-set.
return DIV_ROUND_UP(duty_ns * 100, period_ns); }
static void lpt_set_backlight(const struct drm_connector_state *conn_state, u32 level)
@@ -669,10 +670,10 @@ static void bxt_set_backlight(const struct drm_connector_state *conn_state, u32 static void pwm_set_backlight(const struct drm_connector_state *conn_state, u32 level) { struct intel_panel *panel = &to_intel_connector(conn_state->connector)->panel;
int duty_ns = DIV_ROUND_UP(level * panel->backlight.pwm_period_ns, 100);
pwm_config(panel->backlight.pwm, duty_ns,
panel->backlight.pwm_period_ns);
panel->backlight.pwm_state.duty_cycle =
DIV_ROUND_UP(level * panel->backlight.pwm_state.period, 100);
pwm_apply_state(panel->backlight.pwm, &panel->backlight.pwm_state); }
static void
@@ -841,10 +842,8 @@ static void pwm_disable_backlight(const struct drm_connector_state *old_conn_sta struct intel_connector *connector = to_intel_connector(old_conn_state->connector); struct intel_panel *panel = &connector->panel;
- /* Disable the backlight */
- intel_panel_actually_set_backlight(old_conn_state, 0);
- usleep_range(2000, 3000);
- pwm_disable(panel->backlight.pwm);
- panel->backlight.pwm_state.enabled = false;
- pwm_apply_state(panel->backlight.pwm, &panel->backlight.pwm_state);
Did you drop intel_panel_actually_set_backlight and the sleep on purpose?
Yes, that was on purpose. But I should probably have added a note about this to the commit message.
For v4 of the patchset I will add the following note about this to the commit message for this patch:
"Note that this commit also simplifies pwm_disable_backlight(), by dropping the intel_panel_actually_set_backlight(..., 0) call. This call sets the PWM to 0% duty-cycle. I believe that this call was only present as a workaround for a bug in the pwm-crc.c driver where it failed to clear the PWM_OUTPUT_ENABLE bit. This is fixed by an earlier patch in this series.
After the dropping of this workaround, the usleep call, which seems unnecessary to begin with, has no useful effect anymore, so drop that too."
}
void intel_panel_disable_backlight(const struct drm_connector_state *old_conn_state) [...] @@ -1916,36 +1919,30 @@ static int pwm_setup_backlight(struct intel_connector *connector, return -ENODEV; }
panel->backlight.pwm_period_ns = NSEC_PER_SEC /
get_vbt_pwm_freq(dev_priv);
/*
* FIXME: pwm_apply_args() should be removed when switching to
* the atomic PWM API.
*/
pwm_apply_args(panel->backlight.pwm);
panel->backlight.max = 100; /* 100% */ panel->backlight.min = get_backlight_min_vbt(connector);
level = intel_panel_compute_brightness(connector, 100);
ns = DIV_ROUND_UP(level * panel->backlight.pwm_period_ns, 100);
retval = pwm_config(panel->backlight.pwm, ns,
panel->backlight.pwm_period_ns);
if (retval < 0) {
drm_err(&dev_priv->drm, "Failed to configure the pwm chip\n");
pwm_put(panel->backlight.pwm);
panel->backlight.pwm = NULL;
return retval;
- if (pwm_is_enabled(panel->backlight.pwm) &&
pwm_get_period(panel->backlight.pwm)) {
/* PWM is already enabled, use existing settings */
pwm_get_state(panel->backlight.pwm, &panel->backlight.pwm_state);
level = DIV_ROUND_UP(panel->backlight.pwm_state.duty_cycle *
100, panel->backlight.pwm_state.period);
level = intel_panel_compute_brightness(connector, level);
In pwm_enable_backlight() the order of these two steps is reversed compared to here. Maybe this calculation can be moved into intel_panel_compute_brightness()?
The intel_panel.c code deals with 7 different types of PWM controllers which are built into the GPU + support for external PWM controllers through the kernel's PWM subsystem.
The code this patch changes is for the external PWM controller case, intel_panel_compute_brightness() is used for all supported PWM controllers.
intel_panel_compute_brightness()'s function is to deal with panels where 100% duty-cycle is backlight fully off instead of fully-on. Normally it is called just before setting the duty-cycle, inverting the value/range before sending it to the hardware, since here we are reading back the current value we call it after reading back the value from the controller as the internally cached value is always in 0==min/off 100==max range, so if the panel inverts the range, we need to invert the read-back value to be in our "normalized" internal range.
What we can do here is use pwm_get_relative_duty_cycle as you suggested above. I will change that for v4.
panel->backlight.level = clamp(level, panel->backlight.min,
panel->backlight.max);
panel->backlight.enabled = true;
Best regards Uwe
Regards,
Hans