1*aa96afeeSriastradh /* $NetBSD: intel_panel.c,v 1.5 2021/12/26 21:00:51 riastradh Exp $ */
24e390cabSriastradh
34e390cabSriastradh /*
44e390cabSriastradh * Copyright © 2006-2010 Intel Corporation
54e390cabSriastradh * Copyright (c) 2006 Dave Airlie <airlied@linux.ie>
64e390cabSriastradh *
74e390cabSriastradh * Permission is hereby granted, free of charge, to any person obtaining a
84e390cabSriastradh * copy of this software and associated documentation files (the "Software"),
94e390cabSriastradh * to deal in the Software without restriction, including without limitation
104e390cabSriastradh * the rights to use, copy, modify, merge, publish, distribute, sublicense,
114e390cabSriastradh * and/or sell copies of the Software, and to permit persons to whom the
124e390cabSriastradh * Software is furnished to do so, subject to the following conditions:
134e390cabSriastradh *
144e390cabSriastradh * The above copyright notice and this permission notice (including the next
154e390cabSriastradh * paragraph) shall be included in all copies or substantial portions of the
164e390cabSriastradh * Software.
174e390cabSriastradh *
184e390cabSriastradh * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
194e390cabSriastradh * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
204e390cabSriastradh * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
214e390cabSriastradh * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
224e390cabSriastradh * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
234e390cabSriastradh * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
244e390cabSriastradh * DEALINGS IN THE SOFTWARE.
254e390cabSriastradh *
264e390cabSriastradh * Authors:
274e390cabSriastradh * Eric Anholt <eric@anholt.net>
284e390cabSriastradh * Dave Airlie <airlied@linux.ie>
294e390cabSriastradh * Jesse Barnes <jesse.barnes@intel.com>
304e390cabSriastradh * Chris Wilson <chris@chris-wilson.co.uk>
314e390cabSriastradh */
324e390cabSriastradh
334e390cabSriastradh #include <sys/cdefs.h>
34*aa96afeeSriastradh __KERNEL_RCSID(0, "$NetBSD: intel_panel.c,v 1.5 2021/12/26 21:00:51 riastradh Exp $");
354e390cabSriastradh
364e390cabSriastradh #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
374e390cabSriastradh
384e390cabSriastradh #include <linux/kernel.h>
394e390cabSriastradh #include <linux/moduleparam.h>
404e390cabSriastradh #include <linux/pwm.h>
414e390cabSriastradh
424e390cabSriastradh #include "intel_connector.h"
434e390cabSriastradh #include "intel_display_types.h"
444e390cabSriastradh #include "intel_dp_aux_backlight.h"
454e390cabSriastradh #include "intel_dsi_dcs_backlight.h"
464e390cabSriastradh #include "intel_panel.h"
474e390cabSriastradh
484e390cabSriastradh #define CRC_PMIC_PWM_PERIOD_NS 21333
494e390cabSriastradh
504e390cabSriastradh void
intel_fixed_panel_mode(const struct drm_display_mode * fixed_mode,struct drm_display_mode * adjusted_mode)514e390cabSriastradh intel_fixed_panel_mode(const struct drm_display_mode *fixed_mode,
524e390cabSriastradh struct drm_display_mode *adjusted_mode)
534e390cabSriastradh {
544e390cabSriastradh drm_mode_copy(adjusted_mode, fixed_mode);
554e390cabSriastradh
564e390cabSriastradh drm_mode_set_crtcinfo(adjusted_mode, 0);
574e390cabSriastradh }
584e390cabSriastradh
is_downclock_mode(const struct drm_display_mode * downclock_mode,const struct drm_display_mode * fixed_mode)594e390cabSriastradh static bool is_downclock_mode(const struct drm_display_mode *downclock_mode,
604e390cabSriastradh const struct drm_display_mode *fixed_mode)
614e390cabSriastradh {
624e390cabSriastradh return drm_mode_match(downclock_mode, fixed_mode,
634e390cabSriastradh DRM_MODE_MATCH_TIMINGS |
644e390cabSriastradh DRM_MODE_MATCH_FLAGS |
654e390cabSriastradh DRM_MODE_MATCH_3D_FLAGS) &&
664e390cabSriastradh downclock_mode->clock < fixed_mode->clock;
674e390cabSriastradh }
684e390cabSriastradh
694e390cabSriastradh struct drm_display_mode *
intel_panel_edid_downclock_mode(struct intel_connector * connector,const struct drm_display_mode * fixed_mode)704e390cabSriastradh intel_panel_edid_downclock_mode(struct intel_connector *connector,
714e390cabSriastradh const struct drm_display_mode *fixed_mode)
724e390cabSriastradh {
734e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
744e390cabSriastradh const struct drm_display_mode *scan, *best_mode = NULL;
754e390cabSriastradh struct drm_display_mode *downclock_mode;
764e390cabSriastradh int best_clock = fixed_mode->clock;
774e390cabSriastradh
784e390cabSriastradh list_for_each_entry(scan, &connector->base.probed_modes, head) {
794e390cabSriastradh /*
804e390cabSriastradh * If one mode has the same resolution with the fixed_panel
814e390cabSriastradh * mode while they have the different refresh rate, it means
824e390cabSriastradh * that the reduced downclock is found. In such
834e390cabSriastradh * case we can set the different FPx0/1 to dynamically select
844e390cabSriastradh * between low and high frequency.
854e390cabSriastradh */
864e390cabSriastradh if (is_downclock_mode(scan, fixed_mode) &&
874e390cabSriastradh scan->clock < best_clock) {
884e390cabSriastradh /*
894e390cabSriastradh * The downclock is already found. But we
904e390cabSriastradh * expect to find the lower downclock.
914e390cabSriastradh */
924e390cabSriastradh best_clock = scan->clock;
934e390cabSriastradh best_mode = scan;
944e390cabSriastradh }
954e390cabSriastradh }
964e390cabSriastradh
974e390cabSriastradh if (!best_mode)
984e390cabSriastradh return NULL;
994e390cabSriastradh
1004e390cabSriastradh downclock_mode = drm_mode_duplicate(&dev_priv->drm, best_mode);
1014e390cabSriastradh if (!downclock_mode)
1024e390cabSriastradh return NULL;
1034e390cabSriastradh
1044e390cabSriastradh DRM_DEBUG_KMS("[CONNECTOR:%d:%s] using downclock mode from EDID: ",
1054e390cabSriastradh connector->base.base.id, connector->base.name);
1064e390cabSriastradh drm_mode_debug_printmodeline(downclock_mode);
1074e390cabSriastradh
1084e390cabSriastradh return downclock_mode;
1094e390cabSriastradh }
1104e390cabSriastradh
1114e390cabSriastradh struct drm_display_mode *
intel_panel_edid_fixed_mode(struct intel_connector * connector)1124e390cabSriastradh intel_panel_edid_fixed_mode(struct intel_connector *connector)
1134e390cabSriastradh {
1144e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
1154e390cabSriastradh const struct drm_display_mode *scan;
1164e390cabSriastradh struct drm_display_mode *fixed_mode;
1174e390cabSriastradh
1184e390cabSriastradh if (list_empty(&connector->base.probed_modes))
1194e390cabSriastradh return NULL;
1204e390cabSriastradh
1214e390cabSriastradh /* prefer fixed mode from EDID if available */
1224e390cabSriastradh list_for_each_entry(scan, &connector->base.probed_modes, head) {
1234e390cabSriastradh if ((scan->type & DRM_MODE_TYPE_PREFERRED) == 0)
1244e390cabSriastradh continue;
1254e390cabSriastradh
1264e390cabSriastradh fixed_mode = drm_mode_duplicate(&dev_priv->drm, scan);
1274e390cabSriastradh if (!fixed_mode)
1284e390cabSriastradh return NULL;
1294e390cabSriastradh
1304e390cabSriastradh DRM_DEBUG_KMS("[CONNECTOR:%d:%s] using preferred mode from EDID: ",
1314e390cabSriastradh connector->base.base.id, connector->base.name);
1324e390cabSriastradh drm_mode_debug_printmodeline(fixed_mode);
1334e390cabSriastradh
1344e390cabSriastradh return fixed_mode;
1354e390cabSriastradh }
1364e390cabSriastradh
1374e390cabSriastradh scan = list_first_entry(&connector->base.probed_modes,
1384e390cabSriastradh typeof(*scan), head);
1394e390cabSriastradh
1404e390cabSriastradh fixed_mode = drm_mode_duplicate(&dev_priv->drm, scan);
1414e390cabSriastradh if (!fixed_mode)
1424e390cabSriastradh return NULL;
1434e390cabSriastradh
1444e390cabSriastradh fixed_mode->type |= DRM_MODE_TYPE_PREFERRED;
1454e390cabSriastradh
1464e390cabSriastradh DRM_DEBUG_KMS("[CONNECTOR:%d:%s] using first mode from EDID: ",
1474e390cabSriastradh connector->base.base.id, connector->base.name);
1484e390cabSriastradh drm_mode_debug_printmodeline(fixed_mode);
1494e390cabSriastradh
1504e390cabSriastradh return fixed_mode;
1514e390cabSriastradh }
1524e390cabSriastradh
1534e390cabSriastradh struct drm_display_mode *
intel_panel_vbt_fixed_mode(struct intel_connector * connector)1544e390cabSriastradh intel_panel_vbt_fixed_mode(struct intel_connector *connector)
1554e390cabSriastradh {
1564e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
1574e390cabSriastradh struct drm_display_info *info = &connector->base.display_info;
1584e390cabSriastradh struct drm_display_mode *fixed_mode;
1594e390cabSriastradh
1604e390cabSriastradh if (!dev_priv->vbt.lfp_lvds_vbt_mode)
1614e390cabSriastradh return NULL;
1624e390cabSriastradh
1634e390cabSriastradh fixed_mode = drm_mode_duplicate(&dev_priv->drm,
1644e390cabSriastradh dev_priv->vbt.lfp_lvds_vbt_mode);
1654e390cabSriastradh if (!fixed_mode)
1664e390cabSriastradh return NULL;
1674e390cabSriastradh
1684e390cabSriastradh fixed_mode->type |= DRM_MODE_TYPE_PREFERRED;
1694e390cabSriastradh
1704e390cabSriastradh DRM_DEBUG_KMS("[CONNECTOR:%d:%s] using mode from VBT: ",
1714e390cabSriastradh connector->base.base.id, connector->base.name);
1724e390cabSriastradh drm_mode_debug_printmodeline(fixed_mode);
1734e390cabSriastradh
1744e390cabSriastradh info->width_mm = fixed_mode->width_mm;
1754e390cabSriastradh info->height_mm = fixed_mode->height_mm;
1764e390cabSriastradh
1774e390cabSriastradh return fixed_mode;
1784e390cabSriastradh }
1794e390cabSriastradh
1804e390cabSriastradh /* adjusted_mode has been preset to be the panel's fixed mode */
1814e390cabSriastradh void
intel_pch_panel_fitting(struct intel_crtc * intel_crtc,struct intel_crtc_state * pipe_config,int fitting_mode)1824e390cabSriastradh intel_pch_panel_fitting(struct intel_crtc *intel_crtc,
1834e390cabSriastradh struct intel_crtc_state *pipe_config,
1844e390cabSriastradh int fitting_mode)
1854e390cabSriastradh {
1864e390cabSriastradh const struct drm_display_mode *adjusted_mode = &pipe_config->hw.adjusted_mode;
1874e390cabSriastradh int x = 0, y = 0, width = 0, height = 0;
1884e390cabSriastradh
1894e390cabSriastradh /* Native modes don't need fitting */
1904e390cabSriastradh if (adjusted_mode->crtc_hdisplay == pipe_config->pipe_src_w &&
1914e390cabSriastradh adjusted_mode->crtc_vdisplay == pipe_config->pipe_src_h &&
1924e390cabSriastradh pipe_config->output_format != INTEL_OUTPUT_FORMAT_YCBCR420)
1934e390cabSriastradh goto done;
1944e390cabSriastradh
1954e390cabSriastradh switch (fitting_mode) {
1964e390cabSriastradh case DRM_MODE_SCALE_CENTER:
1974e390cabSriastradh width = pipe_config->pipe_src_w;
1984e390cabSriastradh height = pipe_config->pipe_src_h;
1994e390cabSriastradh x = (adjusted_mode->crtc_hdisplay - width + 1)/2;
2004e390cabSriastradh y = (adjusted_mode->crtc_vdisplay - height + 1)/2;
2014e390cabSriastradh break;
2024e390cabSriastradh
2034e390cabSriastradh case DRM_MODE_SCALE_ASPECT:
2044e390cabSriastradh /* Scale but preserve the aspect ratio */
2054e390cabSriastradh {
2064e390cabSriastradh u32 scaled_width = adjusted_mode->crtc_hdisplay
2074e390cabSriastradh * pipe_config->pipe_src_h;
2084e390cabSriastradh u32 scaled_height = pipe_config->pipe_src_w
2094e390cabSriastradh * adjusted_mode->crtc_vdisplay;
2104e390cabSriastradh if (scaled_width > scaled_height) { /* pillar */
2114e390cabSriastradh width = scaled_height / pipe_config->pipe_src_h;
2124e390cabSriastradh if (width & 1)
2134e390cabSriastradh width++;
2144e390cabSriastradh x = (adjusted_mode->crtc_hdisplay - width + 1) / 2;
2154e390cabSriastradh y = 0;
2164e390cabSriastradh height = adjusted_mode->crtc_vdisplay;
2174e390cabSriastradh } else if (scaled_width < scaled_height) { /* letter */
2184e390cabSriastradh height = scaled_width / pipe_config->pipe_src_w;
2194e390cabSriastradh if (height & 1)
2204e390cabSriastradh height++;
2214e390cabSriastradh y = (adjusted_mode->crtc_vdisplay - height + 1) / 2;
2224e390cabSriastradh x = 0;
2234e390cabSriastradh width = adjusted_mode->crtc_hdisplay;
2244e390cabSriastradh } else {
2254e390cabSriastradh x = y = 0;
2264e390cabSriastradh width = adjusted_mode->crtc_hdisplay;
2274e390cabSriastradh height = adjusted_mode->crtc_vdisplay;
2284e390cabSriastradh }
2294e390cabSriastradh }
2304e390cabSriastradh break;
2314e390cabSriastradh
2324e390cabSriastradh case DRM_MODE_SCALE_FULLSCREEN:
2334e390cabSriastradh x = y = 0;
2344e390cabSriastradh width = adjusted_mode->crtc_hdisplay;
2354e390cabSriastradh height = adjusted_mode->crtc_vdisplay;
2364e390cabSriastradh break;
2374e390cabSriastradh
2384e390cabSriastradh default:
2394e390cabSriastradh WARN(1, "bad panel fit mode: %d\n", fitting_mode);
2404e390cabSriastradh return;
2414e390cabSriastradh }
2424e390cabSriastradh
2434e390cabSriastradh done:
2444e390cabSriastradh pipe_config->pch_pfit.pos = (x << 16) | y;
2454e390cabSriastradh pipe_config->pch_pfit.size = (width << 16) | height;
2464e390cabSriastradh pipe_config->pch_pfit.enabled = pipe_config->pch_pfit.size != 0;
2474e390cabSriastradh }
2484e390cabSriastradh
2494e390cabSriastradh static void
centre_horizontally(struct drm_display_mode * adjusted_mode,int width)2504e390cabSriastradh centre_horizontally(struct drm_display_mode *adjusted_mode,
2514e390cabSriastradh int width)
2524e390cabSriastradh {
2534e390cabSriastradh u32 border, sync_pos, blank_width, sync_width;
2544e390cabSriastradh
2554e390cabSriastradh /* keep the hsync and hblank widths constant */
2564e390cabSriastradh sync_width = adjusted_mode->crtc_hsync_end - adjusted_mode->crtc_hsync_start;
2574e390cabSriastradh blank_width = adjusted_mode->crtc_hblank_end - adjusted_mode->crtc_hblank_start;
2584e390cabSriastradh sync_pos = (blank_width - sync_width + 1) / 2;
2594e390cabSriastradh
2604e390cabSriastradh border = (adjusted_mode->crtc_hdisplay - width + 1) / 2;
2614e390cabSriastradh border += border & 1; /* make the border even */
2624e390cabSriastradh
2634e390cabSriastradh adjusted_mode->crtc_hdisplay = width;
2644e390cabSriastradh adjusted_mode->crtc_hblank_start = width + border;
2654e390cabSriastradh adjusted_mode->crtc_hblank_end = adjusted_mode->crtc_hblank_start + blank_width;
2664e390cabSriastradh
2674e390cabSriastradh adjusted_mode->crtc_hsync_start = adjusted_mode->crtc_hblank_start + sync_pos;
2684e390cabSriastradh adjusted_mode->crtc_hsync_end = adjusted_mode->crtc_hsync_start + sync_width;
2694e390cabSriastradh }
2704e390cabSriastradh
2714e390cabSriastradh static void
centre_vertically(struct drm_display_mode * adjusted_mode,int height)2724e390cabSriastradh centre_vertically(struct drm_display_mode *adjusted_mode,
2734e390cabSriastradh int height)
2744e390cabSriastradh {
2754e390cabSriastradh u32 border, sync_pos, blank_width, sync_width;
2764e390cabSriastradh
2774e390cabSriastradh /* keep the vsync and vblank widths constant */
2784e390cabSriastradh sync_width = adjusted_mode->crtc_vsync_end - adjusted_mode->crtc_vsync_start;
2794e390cabSriastradh blank_width = adjusted_mode->crtc_vblank_end - adjusted_mode->crtc_vblank_start;
2804e390cabSriastradh sync_pos = (blank_width - sync_width + 1) / 2;
2814e390cabSriastradh
2824e390cabSriastradh border = (adjusted_mode->crtc_vdisplay - height + 1) / 2;
2834e390cabSriastradh
2844e390cabSriastradh adjusted_mode->crtc_vdisplay = height;
2854e390cabSriastradh adjusted_mode->crtc_vblank_start = height + border;
2864e390cabSriastradh adjusted_mode->crtc_vblank_end = adjusted_mode->crtc_vblank_start + blank_width;
2874e390cabSriastradh
2884e390cabSriastradh adjusted_mode->crtc_vsync_start = adjusted_mode->crtc_vblank_start + sync_pos;
2894e390cabSriastradh adjusted_mode->crtc_vsync_end = adjusted_mode->crtc_vsync_start + sync_width;
2904e390cabSriastradh }
2914e390cabSriastradh
panel_fitter_scaling(u32 source,u32 target)2924e390cabSriastradh static inline u32 panel_fitter_scaling(u32 source, u32 target)
2934e390cabSriastradh {
2944e390cabSriastradh /*
2954e390cabSriastradh * Floating point operation is not supported. So the FACTOR
2964e390cabSriastradh * is defined, which can avoid the floating point computation
2974e390cabSriastradh * when calculating the panel ratio.
2984e390cabSriastradh */
2994e390cabSriastradh #define ACCURACY 12
3004e390cabSriastradh #define FACTOR (1 << ACCURACY)
3014e390cabSriastradh u32 ratio = source * FACTOR / target;
3024e390cabSriastradh return (FACTOR * ratio + FACTOR/2) / FACTOR;
3034e390cabSriastradh }
3044e390cabSriastradh
i965_scale_aspect(struct intel_crtc_state * pipe_config,u32 * pfit_control)3054e390cabSriastradh static void i965_scale_aspect(struct intel_crtc_state *pipe_config,
3064e390cabSriastradh u32 *pfit_control)
3074e390cabSriastradh {
3084e390cabSriastradh const struct drm_display_mode *adjusted_mode = &pipe_config->hw.adjusted_mode;
3094e390cabSriastradh u32 scaled_width = adjusted_mode->crtc_hdisplay *
3104e390cabSriastradh pipe_config->pipe_src_h;
3114e390cabSriastradh u32 scaled_height = pipe_config->pipe_src_w *
3124e390cabSriastradh adjusted_mode->crtc_vdisplay;
3134e390cabSriastradh
3144e390cabSriastradh /* 965+ is easy, it does everything in hw */
3154e390cabSriastradh if (scaled_width > scaled_height)
3164e390cabSriastradh *pfit_control |= PFIT_ENABLE |
3174e390cabSriastradh PFIT_SCALING_PILLAR;
3184e390cabSriastradh else if (scaled_width < scaled_height)
3194e390cabSriastradh *pfit_control |= PFIT_ENABLE |
3204e390cabSriastradh PFIT_SCALING_LETTER;
3214e390cabSriastradh else if (adjusted_mode->crtc_hdisplay != pipe_config->pipe_src_w)
3224e390cabSriastradh *pfit_control |= PFIT_ENABLE | PFIT_SCALING_AUTO;
3234e390cabSriastradh }
3244e390cabSriastradh
i9xx_scale_aspect(struct intel_crtc_state * pipe_config,u32 * pfit_control,u32 * pfit_pgm_ratios,u32 * border)3254e390cabSriastradh static void i9xx_scale_aspect(struct intel_crtc_state *pipe_config,
3264e390cabSriastradh u32 *pfit_control, u32 *pfit_pgm_ratios,
3274e390cabSriastradh u32 *border)
3284e390cabSriastradh {
3294e390cabSriastradh struct drm_display_mode *adjusted_mode = &pipe_config->hw.adjusted_mode;
3304e390cabSriastradh u32 scaled_width = adjusted_mode->crtc_hdisplay *
3314e390cabSriastradh pipe_config->pipe_src_h;
3324e390cabSriastradh u32 scaled_height = pipe_config->pipe_src_w *
3334e390cabSriastradh adjusted_mode->crtc_vdisplay;
3344e390cabSriastradh u32 bits;
3354e390cabSriastradh
3364e390cabSriastradh /*
3374e390cabSriastradh * For earlier chips we have to calculate the scaling
3384e390cabSriastradh * ratio by hand and program it into the
3394e390cabSriastradh * PFIT_PGM_RATIO register
3404e390cabSriastradh */
3414e390cabSriastradh if (scaled_width > scaled_height) { /* pillar */
3424e390cabSriastradh centre_horizontally(adjusted_mode,
3434e390cabSriastradh scaled_height /
3444e390cabSriastradh pipe_config->pipe_src_h);
3454e390cabSriastradh
3464e390cabSriastradh *border = LVDS_BORDER_ENABLE;
3474e390cabSriastradh if (pipe_config->pipe_src_h != adjusted_mode->crtc_vdisplay) {
3484e390cabSriastradh bits = panel_fitter_scaling(pipe_config->pipe_src_h,
3494e390cabSriastradh adjusted_mode->crtc_vdisplay);
3504e390cabSriastradh
3514e390cabSriastradh *pfit_pgm_ratios |= (bits << PFIT_HORIZ_SCALE_SHIFT |
3524e390cabSriastradh bits << PFIT_VERT_SCALE_SHIFT);
3534e390cabSriastradh *pfit_control |= (PFIT_ENABLE |
3544e390cabSriastradh VERT_INTERP_BILINEAR |
3554e390cabSriastradh HORIZ_INTERP_BILINEAR);
3564e390cabSriastradh }
3574e390cabSriastradh } else if (scaled_width < scaled_height) { /* letter */
3584e390cabSriastradh centre_vertically(adjusted_mode,
3594e390cabSriastradh scaled_width /
3604e390cabSriastradh pipe_config->pipe_src_w);
3614e390cabSriastradh
3624e390cabSriastradh *border = LVDS_BORDER_ENABLE;
3634e390cabSriastradh if (pipe_config->pipe_src_w != adjusted_mode->crtc_hdisplay) {
3644e390cabSriastradh bits = panel_fitter_scaling(pipe_config->pipe_src_w,
3654e390cabSriastradh adjusted_mode->crtc_hdisplay);
3664e390cabSriastradh
3674e390cabSriastradh *pfit_pgm_ratios |= (bits << PFIT_HORIZ_SCALE_SHIFT |
3684e390cabSriastradh bits << PFIT_VERT_SCALE_SHIFT);
3694e390cabSriastradh *pfit_control |= (PFIT_ENABLE |
3704e390cabSriastradh VERT_INTERP_BILINEAR |
3714e390cabSriastradh HORIZ_INTERP_BILINEAR);
3724e390cabSriastradh }
3734e390cabSriastradh } else {
3744e390cabSriastradh /* Aspects match, Let hw scale both directions */
3754e390cabSriastradh *pfit_control |= (PFIT_ENABLE |
3764e390cabSriastradh VERT_AUTO_SCALE | HORIZ_AUTO_SCALE |
3774e390cabSriastradh VERT_INTERP_BILINEAR |
3784e390cabSriastradh HORIZ_INTERP_BILINEAR);
3794e390cabSriastradh }
3804e390cabSriastradh }
3814e390cabSriastradh
intel_gmch_panel_fitting(struct intel_crtc * intel_crtc,struct intel_crtc_state * pipe_config,int fitting_mode)3824e390cabSriastradh void intel_gmch_panel_fitting(struct intel_crtc *intel_crtc,
3834e390cabSriastradh struct intel_crtc_state *pipe_config,
3844e390cabSriastradh int fitting_mode)
3854e390cabSriastradh {
3864e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(intel_crtc->base.dev);
3874e390cabSriastradh u32 pfit_control = 0, pfit_pgm_ratios = 0, border = 0;
3884e390cabSriastradh struct drm_display_mode *adjusted_mode = &pipe_config->hw.adjusted_mode;
3894e390cabSriastradh
3904e390cabSriastradh /* Native modes don't need fitting */
3914e390cabSriastradh if (adjusted_mode->crtc_hdisplay == pipe_config->pipe_src_w &&
3924e390cabSriastradh adjusted_mode->crtc_vdisplay == pipe_config->pipe_src_h)
3934e390cabSriastradh goto out;
3944e390cabSriastradh
3954e390cabSriastradh switch (fitting_mode) {
3964e390cabSriastradh case DRM_MODE_SCALE_CENTER:
3974e390cabSriastradh /*
3984e390cabSriastradh * For centered modes, we have to calculate border widths &
3994e390cabSriastradh * heights and modify the values programmed into the CRTC.
4004e390cabSriastradh */
4014e390cabSriastradh centre_horizontally(adjusted_mode, pipe_config->pipe_src_w);
4024e390cabSriastradh centre_vertically(adjusted_mode, pipe_config->pipe_src_h);
4034e390cabSriastradh border = LVDS_BORDER_ENABLE;
4044e390cabSriastradh break;
4054e390cabSriastradh case DRM_MODE_SCALE_ASPECT:
4064e390cabSriastradh /* Scale but preserve the aspect ratio */
4074e390cabSriastradh if (INTEL_GEN(dev_priv) >= 4)
4084e390cabSriastradh i965_scale_aspect(pipe_config, &pfit_control);
4094e390cabSriastradh else
4104e390cabSriastradh i9xx_scale_aspect(pipe_config, &pfit_control,
4114e390cabSriastradh &pfit_pgm_ratios, &border);
4124e390cabSriastradh break;
4134e390cabSriastradh case DRM_MODE_SCALE_FULLSCREEN:
4144e390cabSriastradh /*
4154e390cabSriastradh * Full scaling, even if it changes the aspect ratio.
4164e390cabSriastradh * Fortunately this is all done for us in hw.
4174e390cabSriastradh */
4184e390cabSriastradh if (pipe_config->pipe_src_h != adjusted_mode->crtc_vdisplay ||
4194e390cabSriastradh pipe_config->pipe_src_w != adjusted_mode->crtc_hdisplay) {
4204e390cabSriastradh pfit_control |= PFIT_ENABLE;
4214e390cabSriastradh if (INTEL_GEN(dev_priv) >= 4)
4224e390cabSriastradh pfit_control |= PFIT_SCALING_AUTO;
4234e390cabSriastradh else
4244e390cabSriastradh pfit_control |= (VERT_AUTO_SCALE |
4254e390cabSriastradh VERT_INTERP_BILINEAR |
4264e390cabSriastradh HORIZ_AUTO_SCALE |
4274e390cabSriastradh HORIZ_INTERP_BILINEAR);
4284e390cabSriastradh }
4294e390cabSriastradh break;
4304e390cabSriastradh default:
4314e390cabSriastradh WARN(1, "bad panel fit mode: %d\n", fitting_mode);
4324e390cabSriastradh return;
4334e390cabSriastradh }
4344e390cabSriastradh
4354e390cabSriastradh /* 965+ wants fuzzy fitting */
4364e390cabSriastradh /* FIXME: handle multiple panels by failing gracefully */
4374e390cabSriastradh if (INTEL_GEN(dev_priv) >= 4)
4384e390cabSriastradh pfit_control |= ((intel_crtc->pipe << PFIT_PIPE_SHIFT) |
4394e390cabSriastradh PFIT_FILTER_FUZZY);
4404e390cabSriastradh
4414e390cabSriastradh out:
4424e390cabSriastradh if ((pfit_control & PFIT_ENABLE) == 0) {
4434e390cabSriastradh pfit_control = 0;
4444e390cabSriastradh pfit_pgm_ratios = 0;
4454e390cabSriastradh }
4464e390cabSriastradh
4474e390cabSriastradh /* Make sure pre-965 set dither correctly for 18bpp panels. */
4484e390cabSriastradh if (INTEL_GEN(dev_priv) < 4 && pipe_config->pipe_bpp == 18)
4494e390cabSriastradh pfit_control |= PANEL_8TO6_DITHER_ENABLE;
4504e390cabSriastradh
4514e390cabSriastradh pipe_config->gmch_pfit.control = pfit_control;
4524e390cabSriastradh pipe_config->gmch_pfit.pgm_ratios = pfit_pgm_ratios;
4534e390cabSriastradh pipe_config->gmch_pfit.lvds_border_bits = border;
4544e390cabSriastradh }
4554e390cabSriastradh
4564e390cabSriastradh /**
4574e390cabSriastradh * scale - scale values from one range to another
4584e390cabSriastradh * @source_val: value in range [@source_min..@source_max]
4594e390cabSriastradh * @source_min: minimum legal value for @source_val
4604e390cabSriastradh * @source_max: maximum legal value for @source_val
4614e390cabSriastradh * @target_min: corresponding target value for @source_min
4624e390cabSriastradh * @target_max: corresponding target value for @source_max
4634e390cabSriastradh *
4644e390cabSriastradh * Return @source_val in range [@source_min..@source_max] scaled to range
4654e390cabSriastradh * [@target_min..@target_max].
4664e390cabSriastradh */
scale(u32 source_val,u32 source_min,u32 source_max,u32 target_min,u32 target_max)4674e390cabSriastradh static u32 scale(u32 source_val,
4684e390cabSriastradh u32 source_min, u32 source_max,
4694e390cabSriastradh u32 target_min, u32 target_max)
4704e390cabSriastradh {
4714e390cabSriastradh u64 target_val;
4724e390cabSriastradh
4734e390cabSriastradh WARN_ON(source_min > source_max);
4744e390cabSriastradh WARN_ON(target_min > target_max);
4754e390cabSriastradh
4764e390cabSriastradh /* defensive */
4774e390cabSriastradh source_val = clamp(source_val, source_min, source_max);
4784e390cabSriastradh
4794e390cabSriastradh /* avoid overflows */
4804e390cabSriastradh target_val = mul_u32_u32(source_val - source_min,
4814e390cabSriastradh target_max - target_min);
4824e390cabSriastradh target_val = DIV_ROUND_CLOSEST_ULL(target_val, source_max - source_min);
4834e390cabSriastradh target_val += target_min;
4844e390cabSriastradh
4854e390cabSriastradh return target_val;
4864e390cabSriastradh }
4874e390cabSriastradh
48841ec0267Sriastradh #if IS_ENABLED(CONFIG_BACKLIGHT_CLASS_DEVICE)
4894e390cabSriastradh /* Scale user_level in range [0..user_max] to [hw_min..hw_max]. */
scale_user_to_hw(struct intel_connector * connector,u32 user_level,u32 user_max)4904e390cabSriastradh static inline u32 scale_user_to_hw(struct intel_connector *connector,
4914e390cabSriastradh u32 user_level, u32 user_max)
4924e390cabSriastradh {
4934e390cabSriastradh struct intel_panel *panel = &connector->panel;
4944e390cabSriastradh
4954e390cabSriastradh return scale(user_level, 0, user_max,
4964e390cabSriastradh panel->backlight.min, panel->backlight.max);
4974e390cabSriastradh }
49841ec0267Sriastradh #endif
4994e390cabSriastradh
5004e390cabSriastradh /* Scale user_level in range [0..user_max] to [0..hw_max], clamping the result
5014e390cabSriastradh * to [hw_min..hw_max]. */
clamp_user_to_hw(struct intel_connector * connector,u32 user_level,u32 user_max)5024e390cabSriastradh static inline u32 clamp_user_to_hw(struct intel_connector *connector,
5034e390cabSriastradh u32 user_level, u32 user_max)
5044e390cabSriastradh {
5054e390cabSriastradh struct intel_panel *panel = &connector->panel;
5064e390cabSriastradh u32 hw_level;
5074e390cabSriastradh
5084e390cabSriastradh hw_level = scale(user_level, 0, user_max, 0, panel->backlight.max);
5094e390cabSriastradh hw_level = clamp(hw_level, panel->backlight.min, panel->backlight.max);
5104e390cabSriastradh
5114e390cabSriastradh return hw_level;
5124e390cabSriastradh }
5134e390cabSriastradh
51441ec0267Sriastradh #if IS_ENABLED(CONFIG_BACKLIGHT_CLASS_DEVICE)
5154e390cabSriastradh /* Scale hw_level in range [hw_min..hw_max] to [0..user_max]. */
scale_hw_to_user(struct intel_connector * connector,u32 hw_level,u32 user_max)5164e390cabSriastradh static inline u32 scale_hw_to_user(struct intel_connector *connector,
5174e390cabSriastradh u32 hw_level, u32 user_max)
5184e390cabSriastradh {
5194e390cabSriastradh struct intel_panel *panel = &connector->panel;
5204e390cabSriastradh
5214e390cabSriastradh return scale(hw_level, panel->backlight.min, panel->backlight.max,
5224e390cabSriastradh 0, user_max);
5234e390cabSriastradh }
52441ec0267Sriastradh #endif
5254e390cabSriastradh
intel_panel_compute_brightness(struct intel_connector * connector,u32 val)5264e390cabSriastradh static u32 intel_panel_compute_brightness(struct intel_connector *connector,
5274e390cabSriastradh u32 val)
5284e390cabSriastradh {
5294e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
5304e390cabSriastradh struct intel_panel *panel = &connector->panel;
5314e390cabSriastradh
5324e390cabSriastradh WARN_ON(panel->backlight.max == 0);
5334e390cabSriastradh
5344e390cabSriastradh if (i915_modparams.invert_brightness < 0)
5354e390cabSriastradh return val;
5364e390cabSriastradh
5374e390cabSriastradh if (i915_modparams.invert_brightness > 0 ||
5384e390cabSriastradh dev_priv->quirks & QUIRK_INVERT_BRIGHTNESS) {
5394e390cabSriastradh return panel->backlight.max - val + panel->backlight.min;
5404e390cabSriastradh }
5414e390cabSriastradh
5424e390cabSriastradh return val;
5434e390cabSriastradh }
5444e390cabSriastradh
lpt_get_backlight(struct intel_connector * connector)5454e390cabSriastradh static u32 lpt_get_backlight(struct intel_connector *connector)
5464e390cabSriastradh {
5474e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
5484e390cabSriastradh
5494e390cabSriastradh return I915_READ(BLC_PWM_PCH_CTL2) & BACKLIGHT_DUTY_CYCLE_MASK;
5504e390cabSriastradh }
5514e390cabSriastradh
pch_get_backlight(struct intel_connector * connector)5524e390cabSriastradh static u32 pch_get_backlight(struct intel_connector *connector)
5534e390cabSriastradh {
5544e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
5554e390cabSriastradh
5564e390cabSriastradh return I915_READ(BLC_PWM_CPU_CTL) & BACKLIGHT_DUTY_CYCLE_MASK;
5574e390cabSriastradh }
5584e390cabSriastradh
i9xx_get_backlight(struct intel_connector * connector)5594e390cabSriastradh static u32 i9xx_get_backlight(struct intel_connector *connector)
5604e390cabSriastradh {
5614e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
5624e390cabSriastradh struct intel_panel *panel = &connector->panel;
5634e390cabSriastradh u32 val;
5644e390cabSriastradh
5654e390cabSriastradh val = I915_READ(BLC_PWM_CTL) & BACKLIGHT_DUTY_CYCLE_MASK;
5664e390cabSriastradh if (INTEL_GEN(dev_priv) < 4)
5674e390cabSriastradh val >>= 1;
5684e390cabSriastradh
5694e390cabSriastradh if (panel->backlight.combination_mode) {
5704e390cabSriastradh u8 lbpc;
5714e390cabSriastradh
5724e390cabSriastradh pci_read_config_byte(dev_priv->drm.pdev, LBPC, &lbpc);
5734e390cabSriastradh val *= lbpc;
5744e390cabSriastradh }
5754e390cabSriastradh
5764e390cabSriastradh return val;
5774e390cabSriastradh }
5784e390cabSriastradh
_vlv_get_backlight(struct drm_i915_private * dev_priv,enum pipe pipe)5794e390cabSriastradh static u32 _vlv_get_backlight(struct drm_i915_private *dev_priv, enum pipe pipe)
5804e390cabSriastradh {
5814e390cabSriastradh if (WARN_ON(pipe != PIPE_A && pipe != PIPE_B))
5824e390cabSriastradh return 0;
5834e390cabSriastradh
5844e390cabSriastradh return I915_READ(VLV_BLC_PWM_CTL(pipe)) & BACKLIGHT_DUTY_CYCLE_MASK;
5854e390cabSriastradh }
5864e390cabSriastradh
vlv_get_backlight(struct intel_connector * connector)5874e390cabSriastradh static u32 vlv_get_backlight(struct intel_connector *connector)
5884e390cabSriastradh {
5894e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
5904e390cabSriastradh enum pipe pipe = intel_connector_get_pipe(connector);
5914e390cabSriastradh
5924e390cabSriastradh return _vlv_get_backlight(dev_priv, pipe);
5934e390cabSriastradh }
5944e390cabSriastradh
bxt_get_backlight(struct intel_connector * connector)5954e390cabSriastradh static u32 bxt_get_backlight(struct intel_connector *connector)
5964e390cabSriastradh {
5974e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
5984e390cabSriastradh struct intel_panel *panel = &connector->panel;
5994e390cabSriastradh
6004e390cabSriastradh return I915_READ(BXT_BLC_PWM_DUTY(panel->backlight.controller));
6014e390cabSriastradh }
6024e390cabSriastradh
60341ec0267Sriastradh #ifndef __NetBSD__ /* XXX mipi */
pwm_get_backlight(struct intel_connector * connector)6044e390cabSriastradh static u32 pwm_get_backlight(struct intel_connector *connector)
6054e390cabSriastradh {
6064e390cabSriastradh struct intel_panel *panel = &connector->panel;
6074e390cabSriastradh int duty_ns;
6084e390cabSriastradh
6094e390cabSriastradh duty_ns = pwm_get_duty_cycle(panel->backlight.pwm);
6104e390cabSriastradh return DIV_ROUND_UP(duty_ns * 100, CRC_PMIC_PWM_PERIOD_NS);
6114e390cabSriastradh }
61241ec0267Sriastradh #endif
6134e390cabSriastradh
lpt_set_backlight(const struct drm_connector_state * conn_state,u32 level)6144e390cabSriastradh static void lpt_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6154e390cabSriastradh {
6164e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
6174e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
6184e390cabSriastradh
6194e390cabSriastradh u32 val = I915_READ(BLC_PWM_PCH_CTL2) & ~BACKLIGHT_DUTY_CYCLE_MASK;
6204e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL2, val | level);
6214e390cabSriastradh }
6224e390cabSriastradh
pch_set_backlight(const struct drm_connector_state * conn_state,u32 level)6234e390cabSriastradh static void pch_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6244e390cabSriastradh {
6254e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
6264e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
6274e390cabSriastradh u32 tmp;
6284e390cabSriastradh
6294e390cabSriastradh tmp = I915_READ(BLC_PWM_CPU_CTL) & ~BACKLIGHT_DUTY_CYCLE_MASK;
6304e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL, tmp | level);
6314e390cabSriastradh }
6324e390cabSriastradh
i9xx_set_backlight(const struct drm_connector_state * conn_state,u32 level)6334e390cabSriastradh static void i9xx_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6344e390cabSriastradh {
6354e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
6364e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
6374e390cabSriastradh struct intel_panel *panel = &connector->panel;
6384e390cabSriastradh u32 tmp, mask;
6394e390cabSriastradh
6404e390cabSriastradh WARN_ON(panel->backlight.max == 0);
6414e390cabSriastradh
6424e390cabSriastradh if (panel->backlight.combination_mode) {
6434e390cabSriastradh u8 lbpc;
6444e390cabSriastradh
6454e390cabSriastradh lbpc = level * 0xfe / panel->backlight.max + 1;
6464e390cabSriastradh level /= lbpc;
6474e390cabSriastradh pci_write_config_byte(dev_priv->drm.pdev, LBPC, lbpc);
6484e390cabSriastradh }
6494e390cabSriastradh
6504e390cabSriastradh if (IS_GEN(dev_priv, 4)) {
6514e390cabSriastradh mask = BACKLIGHT_DUTY_CYCLE_MASK;
6524e390cabSriastradh } else {
6534e390cabSriastradh level <<= 1;
6544e390cabSriastradh mask = BACKLIGHT_DUTY_CYCLE_MASK_PNV;
6554e390cabSriastradh }
6564e390cabSriastradh
6574e390cabSriastradh tmp = I915_READ(BLC_PWM_CTL) & ~mask;
6584e390cabSriastradh I915_WRITE(BLC_PWM_CTL, tmp | level);
6594e390cabSriastradh }
6604e390cabSriastradh
vlv_set_backlight(const struct drm_connector_state * conn_state,u32 level)6614e390cabSriastradh static void vlv_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6624e390cabSriastradh {
6634e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
6644e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
6654e390cabSriastradh enum pipe pipe = to_intel_crtc(conn_state->crtc)->pipe;
6664e390cabSriastradh u32 tmp;
6674e390cabSriastradh
6684e390cabSriastradh tmp = I915_READ(VLV_BLC_PWM_CTL(pipe)) & ~BACKLIGHT_DUTY_CYCLE_MASK;
6694e390cabSriastradh I915_WRITE(VLV_BLC_PWM_CTL(pipe), tmp | level);
6704e390cabSriastradh }
6714e390cabSriastradh
bxt_set_backlight(const struct drm_connector_state * conn_state,u32 level)6724e390cabSriastradh static void bxt_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6734e390cabSriastradh {
6744e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
6754e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
6764e390cabSriastradh struct intel_panel *panel = &connector->panel;
6774e390cabSriastradh
6784e390cabSriastradh I915_WRITE(BXT_BLC_PWM_DUTY(panel->backlight.controller), level);
6794e390cabSriastradh }
6804e390cabSriastradh
68141ec0267Sriastradh #ifndef __NetBSD__ /* XXX mipi */
pwm_set_backlight(const struct drm_connector_state * conn_state,u32 level)6824e390cabSriastradh static void pwm_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6834e390cabSriastradh {
6844e390cabSriastradh struct intel_panel *panel = &to_intel_connector(conn_state->connector)->panel;
6854e390cabSriastradh int duty_ns = DIV_ROUND_UP(level * CRC_PMIC_PWM_PERIOD_NS, 100);
6864e390cabSriastradh
6874e390cabSriastradh pwm_config(panel->backlight.pwm, duty_ns, CRC_PMIC_PWM_PERIOD_NS);
6884e390cabSriastradh }
68941ec0267Sriastradh #endif
6904e390cabSriastradh
6914e390cabSriastradh static void
intel_panel_actually_set_backlight(const struct drm_connector_state * conn_state,u32 level)6924e390cabSriastradh intel_panel_actually_set_backlight(const struct drm_connector_state *conn_state, u32 level)
6934e390cabSriastradh {
6944e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
6954e390cabSriastradh struct intel_panel *panel = &connector->panel;
6964e390cabSriastradh
6974e390cabSriastradh DRM_DEBUG_DRIVER("set backlight PWM = %d\n", level);
6984e390cabSriastradh
6994e390cabSriastradh level = intel_panel_compute_brightness(connector, level);
7004e390cabSriastradh panel->backlight.set(conn_state, level);
7014e390cabSriastradh }
7024e390cabSriastradh
7034e390cabSriastradh /* set backlight brightness to level in range [0..max], assuming hw min is
7044e390cabSriastradh * respected.
7054e390cabSriastradh */
intel_panel_set_backlight_acpi(const struct drm_connector_state * conn_state,u32 user_level,u32 user_max)7064e390cabSriastradh void intel_panel_set_backlight_acpi(const struct drm_connector_state *conn_state,
7074e390cabSriastradh u32 user_level, u32 user_max)
7084e390cabSriastradh {
7094e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
7104e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
7114e390cabSriastradh struct intel_panel *panel = &connector->panel;
7124e390cabSriastradh u32 hw_level;
7134e390cabSriastradh
7144e390cabSriastradh /*
7154e390cabSriastradh * Lack of crtc may occur during driver init because
7164e390cabSriastradh * connection_mutex isn't held across the entire backlight
7174e390cabSriastradh * setup + modeset readout, and the BIOS can issue the
7184e390cabSriastradh * requests at any time.
7194e390cabSriastradh */
7204e390cabSriastradh if (!panel->backlight.present || !conn_state->crtc)
7214e390cabSriastradh return;
7224e390cabSriastradh
7234e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
7244e390cabSriastradh
7254e390cabSriastradh WARN_ON(panel->backlight.max == 0);
7264e390cabSriastradh
7274e390cabSriastradh hw_level = clamp_user_to_hw(connector, user_level, user_max);
7284e390cabSriastradh panel->backlight.level = hw_level;
7294e390cabSriastradh
73041ec0267Sriastradh #ifndef __NetBSD__ /* XXX backlight */
7314e390cabSriastradh if (panel->backlight.device)
7324e390cabSriastradh panel->backlight.device->props.brightness =
7334e390cabSriastradh scale_hw_to_user(connector,
7344e390cabSriastradh panel->backlight.level,
7354e390cabSriastradh panel->backlight.device->props.max_brightness);
73641ec0267Sriastradh #endif
7374e390cabSriastradh
7384e390cabSriastradh if (panel->backlight.enabled)
7394e390cabSriastradh intel_panel_actually_set_backlight(conn_state, hw_level);
7404e390cabSriastradh
7414e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
7424e390cabSriastradh }
7434e390cabSriastradh
lpt_disable_backlight(const struct drm_connector_state * old_conn_state)7444e390cabSriastradh static void lpt_disable_backlight(const struct drm_connector_state *old_conn_state)
7454e390cabSriastradh {
7464e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
7474e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
7484e390cabSriastradh u32 tmp;
7494e390cabSriastradh
7504e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
7514e390cabSriastradh
7524e390cabSriastradh /*
7534e390cabSriastradh * Although we don't support or enable CPU PWM with LPT/SPT based
7544e390cabSriastradh * systems, it may have been enabled prior to loading the
7554e390cabSriastradh * driver. Disable to avoid warnings on LCPLL disable.
7564e390cabSriastradh *
7574e390cabSriastradh * This needs rework if we need to add support for CPU PWM on PCH split
7584e390cabSriastradh * platforms.
7594e390cabSriastradh */
7604e390cabSriastradh tmp = I915_READ(BLC_PWM_CPU_CTL2);
7614e390cabSriastradh if (tmp & BLM_PWM_ENABLE) {
7624e390cabSriastradh DRM_DEBUG_KMS("cpu backlight was enabled, disabling\n");
7634e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL2, tmp & ~BLM_PWM_ENABLE);
7644e390cabSriastradh }
7654e390cabSriastradh
7664e390cabSriastradh tmp = I915_READ(BLC_PWM_PCH_CTL1);
7674e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, tmp & ~BLM_PCH_PWM_ENABLE);
7684e390cabSriastradh }
7694e390cabSriastradh
pch_disable_backlight(const struct drm_connector_state * old_conn_state)7704e390cabSriastradh static void pch_disable_backlight(const struct drm_connector_state *old_conn_state)
7714e390cabSriastradh {
7724e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
7734e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
7744e390cabSriastradh u32 tmp;
7754e390cabSriastradh
7764e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
7774e390cabSriastradh
7784e390cabSriastradh tmp = I915_READ(BLC_PWM_CPU_CTL2);
7794e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL2, tmp & ~BLM_PWM_ENABLE);
7804e390cabSriastradh
7814e390cabSriastradh tmp = I915_READ(BLC_PWM_PCH_CTL1);
7824e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, tmp & ~BLM_PCH_PWM_ENABLE);
7834e390cabSriastradh }
7844e390cabSriastradh
i9xx_disable_backlight(const struct drm_connector_state * old_conn_state)7854e390cabSriastradh static void i9xx_disable_backlight(const struct drm_connector_state *old_conn_state)
7864e390cabSriastradh {
7874e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
7884e390cabSriastradh }
7894e390cabSriastradh
i965_disable_backlight(const struct drm_connector_state * old_conn_state)7904e390cabSriastradh static void i965_disable_backlight(const struct drm_connector_state *old_conn_state)
7914e390cabSriastradh {
7924e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(old_conn_state->connector->dev);
7934e390cabSriastradh u32 tmp;
7944e390cabSriastradh
7954e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
7964e390cabSriastradh
7974e390cabSriastradh tmp = I915_READ(BLC_PWM_CTL2);
7984e390cabSriastradh I915_WRITE(BLC_PWM_CTL2, tmp & ~BLM_PWM_ENABLE);
7994e390cabSriastradh }
8004e390cabSriastradh
vlv_disable_backlight(const struct drm_connector_state * old_conn_state)8014e390cabSriastradh static void vlv_disable_backlight(const struct drm_connector_state *old_conn_state)
8024e390cabSriastradh {
8034e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
8044e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
8054e390cabSriastradh enum pipe pipe = to_intel_crtc(old_conn_state->crtc)->pipe;
8064e390cabSriastradh u32 tmp;
8074e390cabSriastradh
8084e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
8094e390cabSriastradh
8104e390cabSriastradh tmp = I915_READ(VLV_BLC_PWM_CTL2(pipe));
8114e390cabSriastradh I915_WRITE(VLV_BLC_PWM_CTL2(pipe), tmp & ~BLM_PWM_ENABLE);
8124e390cabSriastradh }
8134e390cabSriastradh
bxt_disable_backlight(const struct drm_connector_state * old_conn_state)8144e390cabSriastradh static void bxt_disable_backlight(const struct drm_connector_state *old_conn_state)
8154e390cabSriastradh {
8164e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
8174e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
8184e390cabSriastradh struct intel_panel *panel = &connector->panel;
8194e390cabSriastradh u32 tmp, val;
8204e390cabSriastradh
8214e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
8224e390cabSriastradh
8234e390cabSriastradh tmp = I915_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
8244e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller),
8254e390cabSriastradh tmp & ~BXT_BLC_PWM_ENABLE);
8264e390cabSriastradh
8274e390cabSriastradh if (panel->backlight.controller == 1) {
8284e390cabSriastradh val = I915_READ(UTIL_PIN_CTL);
8294e390cabSriastradh val &= ~UTIL_PIN_ENABLE;
8304e390cabSriastradh I915_WRITE(UTIL_PIN_CTL, val);
8314e390cabSriastradh }
8324e390cabSriastradh }
8334e390cabSriastradh
cnp_disable_backlight(const struct drm_connector_state * old_conn_state)8344e390cabSriastradh static void cnp_disable_backlight(const struct drm_connector_state *old_conn_state)
8354e390cabSriastradh {
8364e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
8374e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
8384e390cabSriastradh struct intel_panel *panel = &connector->panel;
8394e390cabSriastradh u32 tmp;
8404e390cabSriastradh
8414e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
8424e390cabSriastradh
8434e390cabSriastradh tmp = I915_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
8444e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller),
8454e390cabSriastradh tmp & ~BXT_BLC_PWM_ENABLE);
8464e390cabSriastradh }
8474e390cabSriastradh
848*aa96afeeSriastradh #ifndef __NetBSD__ /* XXX mipi */
pwm_disable_backlight(const struct drm_connector_state * old_conn_state)8494e390cabSriastradh static void pwm_disable_backlight(const struct drm_connector_state *old_conn_state)
8504e390cabSriastradh {
8514e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
8524e390cabSriastradh struct intel_panel *panel = &connector->panel;
8534e390cabSriastradh
8544e390cabSriastradh /* Disable the backlight */
8554e390cabSriastradh intel_panel_actually_set_backlight(old_conn_state, 0);
8564e390cabSriastradh usleep_range(2000, 3000);
8574e390cabSriastradh pwm_disable(panel->backlight.pwm);
8584e390cabSriastradh }
85941ec0267Sriastradh #endif
8604e390cabSriastradh
intel_panel_disable_backlight(const struct drm_connector_state * old_conn_state)8614e390cabSriastradh void intel_panel_disable_backlight(const struct drm_connector_state *old_conn_state)
8624e390cabSriastradh {
8634e390cabSriastradh struct intel_connector *connector = to_intel_connector(old_conn_state->connector);
8644e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
8654e390cabSriastradh struct intel_panel *panel = &connector->panel;
8664e390cabSriastradh
8674e390cabSriastradh if (!panel->backlight.present)
8684e390cabSriastradh return;
8694e390cabSriastradh
8704e390cabSriastradh /*
8714e390cabSriastradh * Do not disable backlight on the vga_switcheroo path. When switching
8724e390cabSriastradh * away from i915, the other client may depend on i915 to handle the
8734e390cabSriastradh * backlight. This will leave the backlight on unnecessarily when
8744e390cabSriastradh * another client is not activated.
8754e390cabSriastradh */
8764e390cabSriastradh if (dev_priv->drm.switch_power_state == DRM_SWITCH_POWER_CHANGING) {
8774e390cabSriastradh DRM_DEBUG_DRIVER("Skipping backlight disable on vga switch\n");
8784e390cabSriastradh return;
8794e390cabSriastradh }
8804e390cabSriastradh
8814e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
8824e390cabSriastradh
88341ec0267Sriastradh #ifndef __NetBSD__ /* XXX backlight */
8844e390cabSriastradh if (panel->backlight.device)
8854e390cabSriastradh panel->backlight.device->props.power = FB_BLANK_POWERDOWN;
88641ec0267Sriastradh #endif
8874e390cabSriastradh panel->backlight.enabled = false;
8884e390cabSriastradh panel->backlight.disable(old_conn_state);
8894e390cabSriastradh
8904e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
8914e390cabSriastradh }
8924e390cabSriastradh
lpt_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)8934e390cabSriastradh static void lpt_enable_backlight(const struct intel_crtc_state *crtc_state,
8944e390cabSriastradh const struct drm_connector_state *conn_state)
8954e390cabSriastradh {
8964e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
8974e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
8984e390cabSriastradh struct intel_panel *panel = &connector->panel;
8994e390cabSriastradh u32 pch_ctl1, pch_ctl2, schicken;
9004e390cabSriastradh
9014e390cabSriastradh pch_ctl1 = I915_READ(BLC_PWM_PCH_CTL1);
9024e390cabSriastradh if (pch_ctl1 & BLM_PCH_PWM_ENABLE) {
9034e390cabSriastradh DRM_DEBUG_KMS("pch backlight already enabled\n");
9044e390cabSriastradh pch_ctl1 &= ~BLM_PCH_PWM_ENABLE;
9054e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1);
9064e390cabSriastradh }
9074e390cabSriastradh
9084e390cabSriastradh if (HAS_PCH_LPT(dev_priv)) {
9094e390cabSriastradh schicken = I915_READ(SOUTH_CHICKEN2);
9104e390cabSriastradh if (panel->backlight.alternate_pwm_increment)
9114e390cabSriastradh schicken |= LPT_PWM_GRANULARITY;
9124e390cabSriastradh else
9134e390cabSriastradh schicken &= ~LPT_PWM_GRANULARITY;
9144e390cabSriastradh I915_WRITE(SOUTH_CHICKEN2, schicken);
9154e390cabSriastradh } else {
9164e390cabSriastradh schicken = I915_READ(SOUTH_CHICKEN1);
9174e390cabSriastradh if (panel->backlight.alternate_pwm_increment)
9184e390cabSriastradh schicken |= SPT_PWM_GRANULARITY;
9194e390cabSriastradh else
9204e390cabSriastradh schicken &= ~SPT_PWM_GRANULARITY;
9214e390cabSriastradh I915_WRITE(SOUTH_CHICKEN1, schicken);
9224e390cabSriastradh }
9234e390cabSriastradh
9244e390cabSriastradh pch_ctl2 = panel->backlight.max << 16;
9254e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL2, pch_ctl2);
9264e390cabSriastradh
9274e390cabSriastradh pch_ctl1 = 0;
9284e390cabSriastradh if (panel->backlight.active_low_pwm)
9294e390cabSriastradh pch_ctl1 |= BLM_PCH_POLARITY;
9304e390cabSriastradh
9314e390cabSriastradh /* After LPT, override is the default. */
9324e390cabSriastradh if (HAS_PCH_LPT(dev_priv))
9334e390cabSriastradh pch_ctl1 |= BLM_PCH_OVERRIDE_ENABLE;
9344e390cabSriastradh
9354e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1);
9364e390cabSriastradh POSTING_READ(BLC_PWM_PCH_CTL1);
9374e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1 | BLM_PCH_PWM_ENABLE);
9384e390cabSriastradh
9394e390cabSriastradh /* This won't stick until the above enable. */
9404e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
9414e390cabSriastradh }
9424e390cabSriastradh
pch_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)9434e390cabSriastradh static void pch_enable_backlight(const struct intel_crtc_state *crtc_state,
9444e390cabSriastradh const struct drm_connector_state *conn_state)
9454e390cabSriastradh {
9464e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
9474e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
9484e390cabSriastradh struct intel_panel *panel = &connector->panel;
9494e390cabSriastradh enum transcoder cpu_transcoder = crtc_state->cpu_transcoder;
9504e390cabSriastradh u32 cpu_ctl2, pch_ctl1, pch_ctl2;
9514e390cabSriastradh
9524e390cabSriastradh cpu_ctl2 = I915_READ(BLC_PWM_CPU_CTL2);
9534e390cabSriastradh if (cpu_ctl2 & BLM_PWM_ENABLE) {
9544e390cabSriastradh DRM_DEBUG_KMS("cpu backlight already enabled\n");
9554e390cabSriastradh cpu_ctl2 &= ~BLM_PWM_ENABLE;
9564e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL2, cpu_ctl2);
9574e390cabSriastradh }
9584e390cabSriastradh
9594e390cabSriastradh pch_ctl1 = I915_READ(BLC_PWM_PCH_CTL1);
9604e390cabSriastradh if (pch_ctl1 & BLM_PCH_PWM_ENABLE) {
9614e390cabSriastradh DRM_DEBUG_KMS("pch backlight already enabled\n");
9624e390cabSriastradh pch_ctl1 &= ~BLM_PCH_PWM_ENABLE;
9634e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1);
9644e390cabSriastradh }
9654e390cabSriastradh
9664e390cabSriastradh if (cpu_transcoder == TRANSCODER_EDP)
9674e390cabSriastradh cpu_ctl2 = BLM_TRANSCODER_EDP;
9684e390cabSriastradh else
9694e390cabSriastradh cpu_ctl2 = BLM_PIPE(cpu_transcoder);
9704e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL2, cpu_ctl2);
9714e390cabSriastradh POSTING_READ(BLC_PWM_CPU_CTL2);
9724e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL2, cpu_ctl2 | BLM_PWM_ENABLE);
9734e390cabSriastradh
9744e390cabSriastradh /* This won't stick until the above enable. */
9754e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
9764e390cabSriastradh
9774e390cabSriastradh pch_ctl2 = panel->backlight.max << 16;
9784e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL2, pch_ctl2);
9794e390cabSriastradh
9804e390cabSriastradh pch_ctl1 = 0;
9814e390cabSriastradh if (panel->backlight.active_low_pwm)
9824e390cabSriastradh pch_ctl1 |= BLM_PCH_POLARITY;
9834e390cabSriastradh
9844e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1);
9854e390cabSriastradh POSTING_READ(BLC_PWM_PCH_CTL1);
9864e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1 | BLM_PCH_PWM_ENABLE);
9874e390cabSriastradh }
9884e390cabSriastradh
i9xx_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)9894e390cabSriastradh static void i9xx_enable_backlight(const struct intel_crtc_state *crtc_state,
9904e390cabSriastradh const struct drm_connector_state *conn_state)
9914e390cabSriastradh {
9924e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
9934e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
9944e390cabSriastradh struct intel_panel *panel = &connector->panel;
9954e390cabSriastradh u32 ctl, freq;
9964e390cabSriastradh
9974e390cabSriastradh ctl = I915_READ(BLC_PWM_CTL);
9984e390cabSriastradh if (ctl & BACKLIGHT_DUTY_CYCLE_MASK_PNV) {
9994e390cabSriastradh DRM_DEBUG_KMS("backlight already enabled\n");
10004e390cabSriastradh I915_WRITE(BLC_PWM_CTL, 0);
10014e390cabSriastradh }
10024e390cabSriastradh
10034e390cabSriastradh freq = panel->backlight.max;
10044e390cabSriastradh if (panel->backlight.combination_mode)
10054e390cabSriastradh freq /= 0xff;
10064e390cabSriastradh
10074e390cabSriastradh ctl = freq << 17;
10084e390cabSriastradh if (panel->backlight.combination_mode)
10094e390cabSriastradh ctl |= BLM_LEGACY_MODE;
10104e390cabSriastradh if (IS_PINEVIEW(dev_priv) && panel->backlight.active_low_pwm)
10114e390cabSriastradh ctl |= BLM_POLARITY_PNV;
10124e390cabSriastradh
10134e390cabSriastradh I915_WRITE(BLC_PWM_CTL, ctl);
10144e390cabSriastradh POSTING_READ(BLC_PWM_CTL);
10154e390cabSriastradh
10164e390cabSriastradh /* XXX: combine this into above write? */
10174e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
10184e390cabSriastradh
10194e390cabSriastradh /*
10204e390cabSriastradh * Needed to enable backlight on some 855gm models. BLC_HIST_CTL is
10214e390cabSriastradh * 855gm only, but checking for gen2 is safe, as 855gm is the only gen2
10224e390cabSriastradh * that has backlight.
10234e390cabSriastradh */
10244e390cabSriastradh if (IS_GEN(dev_priv, 2))
10254e390cabSriastradh I915_WRITE(BLC_HIST_CTL, BLM_HISTOGRAM_ENABLE);
10264e390cabSriastradh }
10274e390cabSriastradh
i965_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)10284e390cabSriastradh static void i965_enable_backlight(const struct intel_crtc_state *crtc_state,
10294e390cabSriastradh const struct drm_connector_state *conn_state)
10304e390cabSriastradh {
10314e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
10324e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
10334e390cabSriastradh struct intel_panel *panel = &connector->panel;
10344e390cabSriastradh enum pipe pipe = to_intel_crtc(conn_state->crtc)->pipe;
10354e390cabSriastradh u32 ctl, ctl2, freq;
10364e390cabSriastradh
10374e390cabSriastradh ctl2 = I915_READ(BLC_PWM_CTL2);
10384e390cabSriastradh if (ctl2 & BLM_PWM_ENABLE) {
10394e390cabSriastradh DRM_DEBUG_KMS("backlight already enabled\n");
10404e390cabSriastradh ctl2 &= ~BLM_PWM_ENABLE;
10414e390cabSriastradh I915_WRITE(BLC_PWM_CTL2, ctl2);
10424e390cabSriastradh }
10434e390cabSriastradh
10444e390cabSriastradh freq = panel->backlight.max;
10454e390cabSriastradh if (panel->backlight.combination_mode)
10464e390cabSriastradh freq /= 0xff;
10474e390cabSriastradh
10484e390cabSriastradh ctl = freq << 16;
10494e390cabSriastradh I915_WRITE(BLC_PWM_CTL, ctl);
10504e390cabSriastradh
10514e390cabSriastradh ctl2 = BLM_PIPE(pipe);
10524e390cabSriastradh if (panel->backlight.combination_mode)
10534e390cabSriastradh ctl2 |= BLM_COMBINATION_MODE;
10544e390cabSriastradh if (panel->backlight.active_low_pwm)
10554e390cabSriastradh ctl2 |= BLM_POLARITY_I965;
10564e390cabSriastradh I915_WRITE(BLC_PWM_CTL2, ctl2);
10574e390cabSriastradh POSTING_READ(BLC_PWM_CTL2);
10584e390cabSriastradh I915_WRITE(BLC_PWM_CTL2, ctl2 | BLM_PWM_ENABLE);
10594e390cabSriastradh
10604e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
10614e390cabSriastradh }
10624e390cabSriastradh
vlv_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)10634e390cabSriastradh static void vlv_enable_backlight(const struct intel_crtc_state *crtc_state,
10644e390cabSriastradh const struct drm_connector_state *conn_state)
10654e390cabSriastradh {
10664e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
10674e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
10684e390cabSriastradh struct intel_panel *panel = &connector->panel;
10694e390cabSriastradh enum pipe pipe = to_intel_crtc(crtc_state->uapi.crtc)->pipe;
10704e390cabSriastradh u32 ctl, ctl2;
10714e390cabSriastradh
10724e390cabSriastradh ctl2 = I915_READ(VLV_BLC_PWM_CTL2(pipe));
10734e390cabSriastradh if (ctl2 & BLM_PWM_ENABLE) {
10744e390cabSriastradh DRM_DEBUG_KMS("backlight already enabled\n");
10754e390cabSriastradh ctl2 &= ~BLM_PWM_ENABLE;
10764e390cabSriastradh I915_WRITE(VLV_BLC_PWM_CTL2(pipe), ctl2);
10774e390cabSriastradh }
10784e390cabSriastradh
10794e390cabSriastradh ctl = panel->backlight.max << 16;
10804e390cabSriastradh I915_WRITE(VLV_BLC_PWM_CTL(pipe), ctl);
10814e390cabSriastradh
10824e390cabSriastradh /* XXX: combine this into above write? */
10834e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
10844e390cabSriastradh
10854e390cabSriastradh ctl2 = 0;
10864e390cabSriastradh if (panel->backlight.active_low_pwm)
10874e390cabSriastradh ctl2 |= BLM_POLARITY_I965;
10884e390cabSriastradh I915_WRITE(VLV_BLC_PWM_CTL2(pipe), ctl2);
10894e390cabSriastradh POSTING_READ(VLV_BLC_PWM_CTL2(pipe));
10904e390cabSriastradh I915_WRITE(VLV_BLC_PWM_CTL2(pipe), ctl2 | BLM_PWM_ENABLE);
10914e390cabSriastradh }
10924e390cabSriastradh
bxt_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)10934e390cabSriastradh static void bxt_enable_backlight(const struct intel_crtc_state *crtc_state,
10944e390cabSriastradh const struct drm_connector_state *conn_state)
10954e390cabSriastradh {
10964e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
10974e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
10984e390cabSriastradh struct intel_panel *panel = &connector->panel;
10994e390cabSriastradh enum pipe pipe = to_intel_crtc(crtc_state->uapi.crtc)->pipe;
11004e390cabSriastradh u32 pwm_ctl, val;
11014e390cabSriastradh
11024e390cabSriastradh /* Controller 1 uses the utility pin. */
11034e390cabSriastradh if (panel->backlight.controller == 1) {
11044e390cabSriastradh val = I915_READ(UTIL_PIN_CTL);
11054e390cabSriastradh if (val & UTIL_PIN_ENABLE) {
11064e390cabSriastradh DRM_DEBUG_KMS("util pin already enabled\n");
11074e390cabSriastradh val &= ~UTIL_PIN_ENABLE;
11084e390cabSriastradh I915_WRITE(UTIL_PIN_CTL, val);
11094e390cabSriastradh }
11104e390cabSriastradh
11114e390cabSriastradh val = 0;
11124e390cabSriastradh if (panel->backlight.util_pin_active_low)
11134e390cabSriastradh val |= UTIL_PIN_POLARITY;
11144e390cabSriastradh I915_WRITE(UTIL_PIN_CTL, val | UTIL_PIN_PIPE(pipe) |
11154e390cabSriastradh UTIL_PIN_MODE_PWM | UTIL_PIN_ENABLE);
11164e390cabSriastradh }
11174e390cabSriastradh
11184e390cabSriastradh pwm_ctl = I915_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
11194e390cabSriastradh if (pwm_ctl & BXT_BLC_PWM_ENABLE) {
11204e390cabSriastradh DRM_DEBUG_KMS("backlight already enabled\n");
11214e390cabSriastradh pwm_ctl &= ~BXT_BLC_PWM_ENABLE;
11224e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller),
11234e390cabSriastradh pwm_ctl);
11244e390cabSriastradh }
11254e390cabSriastradh
11264e390cabSriastradh I915_WRITE(BXT_BLC_PWM_FREQ(panel->backlight.controller),
11274e390cabSriastradh panel->backlight.max);
11284e390cabSriastradh
11294e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
11304e390cabSriastradh
11314e390cabSriastradh pwm_ctl = 0;
11324e390cabSriastradh if (panel->backlight.active_low_pwm)
11334e390cabSriastradh pwm_ctl |= BXT_BLC_PWM_POLARITY;
11344e390cabSriastradh
11354e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller), pwm_ctl);
11364e390cabSriastradh POSTING_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
11374e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller),
11384e390cabSriastradh pwm_ctl | BXT_BLC_PWM_ENABLE);
11394e390cabSriastradh }
11404e390cabSriastradh
cnp_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)11414e390cabSriastradh static void cnp_enable_backlight(const struct intel_crtc_state *crtc_state,
11424e390cabSriastradh const struct drm_connector_state *conn_state)
11434e390cabSriastradh {
11444e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
11454e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
11464e390cabSriastradh struct intel_panel *panel = &connector->panel;
11474e390cabSriastradh u32 pwm_ctl;
11484e390cabSriastradh
11494e390cabSriastradh pwm_ctl = I915_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
11504e390cabSriastradh if (pwm_ctl & BXT_BLC_PWM_ENABLE) {
11514e390cabSriastradh DRM_DEBUG_KMS("backlight already enabled\n");
11524e390cabSriastradh pwm_ctl &= ~BXT_BLC_PWM_ENABLE;
11534e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller),
11544e390cabSriastradh pwm_ctl);
11554e390cabSriastradh }
11564e390cabSriastradh
11574e390cabSriastradh I915_WRITE(BXT_BLC_PWM_FREQ(panel->backlight.controller),
11584e390cabSriastradh panel->backlight.max);
11594e390cabSriastradh
11604e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
11614e390cabSriastradh
11624e390cabSriastradh pwm_ctl = 0;
11634e390cabSriastradh if (panel->backlight.active_low_pwm)
11644e390cabSriastradh pwm_ctl |= BXT_BLC_PWM_POLARITY;
11654e390cabSriastradh
11664e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller), pwm_ctl);
11674e390cabSriastradh POSTING_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
11684e390cabSriastradh I915_WRITE(BXT_BLC_PWM_CTL(panel->backlight.controller),
11694e390cabSriastradh pwm_ctl | BXT_BLC_PWM_ENABLE);
11704e390cabSriastradh }
11714e390cabSriastradh
1172*aa96afeeSriastradh #ifndef __NetBSD__ /* XXX mipi */
pwm_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)11734e390cabSriastradh static void pwm_enable_backlight(const struct intel_crtc_state *crtc_state,
11744e390cabSriastradh const struct drm_connector_state *conn_state)
11754e390cabSriastradh {
11764e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
11774e390cabSriastradh struct intel_panel *panel = &connector->panel;
11784e390cabSriastradh
11794e390cabSriastradh pwm_enable(panel->backlight.pwm);
11804e390cabSriastradh intel_panel_actually_set_backlight(conn_state, panel->backlight.level);
11814e390cabSriastradh }
118241ec0267Sriastradh #endif
11834e390cabSriastradh
__intel_panel_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)11844e390cabSriastradh static void __intel_panel_enable_backlight(const struct intel_crtc_state *crtc_state,
11854e390cabSriastradh const struct drm_connector_state *conn_state)
11864e390cabSriastradh {
11874e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
11884e390cabSriastradh struct intel_panel *panel = &connector->panel;
11894e390cabSriastradh
11904e390cabSriastradh WARN_ON(panel->backlight.max == 0);
11914e390cabSriastradh
11924e390cabSriastradh if (panel->backlight.level <= panel->backlight.min) {
11934e390cabSriastradh panel->backlight.level = panel->backlight.max;
119441ec0267Sriastradh #ifndef __NetBSD__ /* XXX backlight */
11954e390cabSriastradh if (panel->backlight.device)
11964e390cabSriastradh panel->backlight.device->props.brightness =
11974e390cabSriastradh scale_hw_to_user(connector,
11984e390cabSriastradh panel->backlight.level,
11994e390cabSriastradh panel->backlight.device->props.max_brightness);
120041ec0267Sriastradh #endif
12014e390cabSriastradh }
12024e390cabSriastradh
12034e390cabSriastradh panel->backlight.enable(crtc_state, conn_state);
12044e390cabSriastradh panel->backlight.enabled = true;
120541ec0267Sriastradh #ifndef __NetBSD__ /* XXX backlight */
12064e390cabSriastradh if (panel->backlight.device)
12074e390cabSriastradh panel->backlight.device->props.power = FB_BLANK_UNBLANK;
120841ec0267Sriastradh #endif
12094e390cabSriastradh }
12104e390cabSriastradh
intel_panel_enable_backlight(const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)12114e390cabSriastradh void intel_panel_enable_backlight(const struct intel_crtc_state *crtc_state,
12124e390cabSriastradh const struct drm_connector_state *conn_state)
12134e390cabSriastradh {
12144e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
12154e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
12164e390cabSriastradh struct intel_panel *panel = &connector->panel;
12174e390cabSriastradh enum pipe pipe = to_intel_crtc(crtc_state->uapi.crtc)->pipe;
12184e390cabSriastradh
12194e390cabSriastradh if (!panel->backlight.present)
12204e390cabSriastradh return;
12214e390cabSriastradh
12224e390cabSriastradh DRM_DEBUG_KMS("pipe %c\n", pipe_name(pipe));
12234e390cabSriastradh
12244e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
12254e390cabSriastradh
12264e390cabSriastradh __intel_panel_enable_backlight(crtc_state, conn_state);
12274e390cabSriastradh
12284e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
12294e390cabSriastradh }
12304e390cabSriastradh
12314e390cabSriastradh #if IS_ENABLED(CONFIG_BACKLIGHT_CLASS_DEVICE)
intel_panel_get_backlight(struct intel_connector * connector)12324e390cabSriastradh static u32 intel_panel_get_backlight(struct intel_connector *connector)
12334e390cabSriastradh {
12344e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
12354e390cabSriastradh struct intel_panel *panel = &connector->panel;
12364e390cabSriastradh u32 val = 0;
12374e390cabSriastradh
12384e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
12394e390cabSriastradh
12404e390cabSriastradh if (panel->backlight.enabled) {
12414e390cabSriastradh val = panel->backlight.get(connector);
12424e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
12434e390cabSriastradh }
12444e390cabSriastradh
12454e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
12464e390cabSriastradh
12474e390cabSriastradh DRM_DEBUG_DRIVER("get backlight PWM = %d\n", val);
12484e390cabSriastradh return val;
12494e390cabSriastradh }
12504e390cabSriastradh
12514e390cabSriastradh /* set backlight brightness to level in range [0..max], scaling wrt hw min */
intel_panel_set_backlight(const struct drm_connector_state * conn_state,u32 user_level,u32 user_max)12524e390cabSriastradh static void intel_panel_set_backlight(const struct drm_connector_state *conn_state,
12534e390cabSriastradh u32 user_level, u32 user_max)
12544e390cabSriastradh {
12554e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
12564e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
12574e390cabSriastradh struct intel_panel *panel = &connector->panel;
12584e390cabSriastradh u32 hw_level;
12594e390cabSriastradh
12604e390cabSriastradh if (!panel->backlight.present)
12614e390cabSriastradh return;
12624e390cabSriastradh
12634e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
12644e390cabSriastradh
12654e390cabSriastradh WARN_ON(panel->backlight.max == 0);
12664e390cabSriastradh
12674e390cabSriastradh hw_level = scale_user_to_hw(connector, user_level, user_max);
12684e390cabSriastradh panel->backlight.level = hw_level;
12694e390cabSriastradh
12704e390cabSriastradh if (panel->backlight.enabled)
12714e390cabSriastradh intel_panel_actually_set_backlight(conn_state, hw_level);
12724e390cabSriastradh
12734e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
12744e390cabSriastradh }
12754e390cabSriastradh
intel_backlight_device_update_status(struct backlight_device * bd)12764e390cabSriastradh static int intel_backlight_device_update_status(struct backlight_device *bd)
12774e390cabSriastradh {
12784e390cabSriastradh struct intel_connector *connector = bl_get_data(bd);
12794e390cabSriastradh struct intel_panel *panel = &connector->panel;
12804e390cabSriastradh struct drm_device *dev = connector->base.dev;
12814e390cabSriastradh
12824e390cabSriastradh drm_modeset_lock(&dev->mode_config.connection_mutex, NULL);
12834e390cabSriastradh DRM_DEBUG_KMS("updating intel_backlight, brightness=%d/%d\n",
12844e390cabSriastradh bd->props.brightness, bd->props.max_brightness);
12854e390cabSriastradh intel_panel_set_backlight(connector->base.state, bd->props.brightness,
12864e390cabSriastradh bd->props.max_brightness);
12874e390cabSriastradh
12884e390cabSriastradh /*
12894e390cabSriastradh * Allow flipping bl_power as a sub-state of enabled. Sadly the
12904e390cabSriastradh * backlight class device does not make it easy to to differentiate
12914e390cabSriastradh * between callbacks for brightness and bl_power, so our backlight_power
12924e390cabSriastradh * callback needs to take this into account.
12934e390cabSriastradh */
12944e390cabSriastradh if (panel->backlight.enabled) {
12954e390cabSriastradh if (panel->backlight.power) {
12964e390cabSriastradh bool enable = bd->props.power == FB_BLANK_UNBLANK &&
12974e390cabSriastradh bd->props.brightness != 0;
12984e390cabSriastradh panel->backlight.power(connector, enable);
12994e390cabSriastradh }
13004e390cabSriastradh } else {
13014e390cabSriastradh bd->props.power = FB_BLANK_POWERDOWN;
13024e390cabSriastradh }
13034e390cabSriastradh
13044e390cabSriastradh drm_modeset_unlock(&dev->mode_config.connection_mutex);
13054e390cabSriastradh return 0;
13064e390cabSriastradh }
13074e390cabSriastradh
intel_backlight_device_get_brightness(struct backlight_device * bd)13084e390cabSriastradh static int intel_backlight_device_get_brightness(struct backlight_device *bd)
13094e390cabSriastradh {
13104e390cabSriastradh struct intel_connector *connector = bl_get_data(bd);
13114e390cabSriastradh struct drm_device *dev = connector->base.dev;
13124e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(dev);
13134e390cabSriastradh intel_wakeref_t wakeref;
13144e390cabSriastradh int ret = 0;
13154e390cabSriastradh
13164e390cabSriastradh with_intel_runtime_pm(&dev_priv->runtime_pm, wakeref) {
13174e390cabSriastradh u32 hw_level;
13184e390cabSriastradh
13194e390cabSriastradh drm_modeset_lock(&dev->mode_config.connection_mutex, NULL);
13204e390cabSriastradh
13214e390cabSriastradh hw_level = intel_panel_get_backlight(connector);
13224e390cabSriastradh ret = scale_hw_to_user(connector,
13234e390cabSriastradh hw_level, bd->props.max_brightness);
13244e390cabSriastradh
13254e390cabSriastradh drm_modeset_unlock(&dev->mode_config.connection_mutex);
13264e390cabSriastradh }
13274e390cabSriastradh
13284e390cabSriastradh return ret;
13294e390cabSriastradh }
13304e390cabSriastradh
13314e390cabSriastradh static const struct backlight_ops intel_backlight_device_ops = {
13324e390cabSriastradh .update_status = intel_backlight_device_update_status,
13334e390cabSriastradh .get_brightness = intel_backlight_device_get_brightness,
13344e390cabSriastradh };
13354e390cabSriastradh
intel_backlight_device_register(struct intel_connector * connector)13364e390cabSriastradh int intel_backlight_device_register(struct intel_connector *connector)
13374e390cabSriastradh {
13384e390cabSriastradh struct intel_panel *panel = &connector->panel;
13394e390cabSriastradh struct backlight_properties props;
13404e390cabSriastradh
13414e390cabSriastradh if (WARN_ON(panel->backlight.device))
13424e390cabSriastradh return -ENODEV;
13434e390cabSriastradh
13444e390cabSriastradh if (!panel->backlight.present)
13454e390cabSriastradh return 0;
13464e390cabSriastradh
13474e390cabSriastradh WARN_ON(panel->backlight.max == 0);
13484e390cabSriastradh
13494e390cabSriastradh memset(&props, 0, sizeof(props));
13504e390cabSriastradh props.type = BACKLIGHT_RAW;
13514e390cabSriastradh
13524e390cabSriastradh /*
13534e390cabSriastradh * Note: Everything should work even if the backlight device max
13544e390cabSriastradh * presented to the userspace is arbitrarily chosen.
13554e390cabSriastradh */
13564e390cabSriastradh props.max_brightness = panel->backlight.max;
13574e390cabSriastradh props.brightness = scale_hw_to_user(connector,
13584e390cabSriastradh panel->backlight.level,
13594e390cabSriastradh props.max_brightness);
13604e390cabSriastradh
13614e390cabSriastradh if (panel->backlight.enabled)
13624e390cabSriastradh props.power = FB_BLANK_UNBLANK;
13634e390cabSriastradh else
13644e390cabSriastradh props.power = FB_BLANK_POWERDOWN;
13654e390cabSriastradh
13664e390cabSriastradh /*
13674e390cabSriastradh * Note: using the same name independent of the connector prevents
13684e390cabSriastradh * registration of multiple backlight devices in the driver.
13694e390cabSriastradh */
13704e390cabSriastradh panel->backlight.device =
13714e390cabSriastradh backlight_device_register("intel_backlight",
13724e390cabSriastradh connector->base.kdev,
13734e390cabSriastradh connector,
13744e390cabSriastradh &intel_backlight_device_ops, &props);
13754e390cabSriastradh
13764e390cabSriastradh if (IS_ERR(panel->backlight.device)) {
13774e390cabSriastradh DRM_ERROR("Failed to register backlight: %ld\n",
13784e390cabSriastradh PTR_ERR(panel->backlight.device));
13794e390cabSriastradh panel->backlight.device = NULL;
13804e390cabSriastradh return -ENODEV;
13814e390cabSriastradh }
13824e390cabSriastradh
13834e390cabSriastradh DRM_DEBUG_KMS("Connector %s backlight sysfs interface registered\n",
13844e390cabSriastradh connector->base.name);
13854e390cabSriastradh
13864e390cabSriastradh return 0;
13874e390cabSriastradh }
13884e390cabSriastradh
intel_backlight_device_unregister(struct intel_connector * connector)13894e390cabSriastradh void intel_backlight_device_unregister(struct intel_connector *connector)
13904e390cabSriastradh {
13914e390cabSriastradh struct intel_panel *panel = &connector->panel;
13924e390cabSriastradh
13934e390cabSriastradh if (panel->backlight.device) {
13944e390cabSriastradh backlight_device_unregister(panel->backlight.device);
13954e390cabSriastradh panel->backlight.device = NULL;
13964e390cabSriastradh }
13974e390cabSriastradh }
13984e390cabSriastradh #endif /* CONFIG_BACKLIGHT_CLASS_DEVICE */
13994e390cabSriastradh
14004e390cabSriastradh /*
14014e390cabSriastradh * CNP: PWM clock frequency is 19.2 MHz or 24 MHz.
14024e390cabSriastradh * PWM increment = 1
14034e390cabSriastradh */
cnp_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14044e390cabSriastradh static u32 cnp_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14054e390cabSriastradh {
14064e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
14074e390cabSriastradh
14084e390cabSriastradh return DIV_ROUND_CLOSEST(KHz(dev_priv->rawclk_freq), pwm_freq_hz);
14094e390cabSriastradh }
14104e390cabSriastradh
14114e390cabSriastradh /*
14124e390cabSriastradh * BXT: PWM clock frequency = 19.2 MHz.
14134e390cabSriastradh */
bxt_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14144e390cabSriastradh static u32 bxt_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14154e390cabSriastradh {
14164e390cabSriastradh return DIV_ROUND_CLOSEST(KHz(19200), pwm_freq_hz);
14174e390cabSriastradh }
14184e390cabSriastradh
14194e390cabSriastradh /*
14204e390cabSriastradh * SPT: This value represents the period of the PWM stream in clock periods
14214e390cabSriastradh * multiplied by 16 (default increment) or 128 (alternate increment selected in
14224e390cabSriastradh * SCHICKEN_1 bit 0). PWM clock is 24 MHz.
14234e390cabSriastradh */
spt_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14244e390cabSriastradh static u32 spt_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14254e390cabSriastradh {
14264e390cabSriastradh struct intel_panel *panel = &connector->panel;
14274e390cabSriastradh u32 mul;
14284e390cabSriastradh
14294e390cabSriastradh if (panel->backlight.alternate_pwm_increment)
14304e390cabSriastradh mul = 128;
14314e390cabSriastradh else
14324e390cabSriastradh mul = 16;
14334e390cabSriastradh
14344e390cabSriastradh return DIV_ROUND_CLOSEST(MHz(24), pwm_freq_hz * mul);
14354e390cabSriastradh }
14364e390cabSriastradh
14374e390cabSriastradh /*
14384e390cabSriastradh * LPT: This value represents the period of the PWM stream in clock periods
14394e390cabSriastradh * multiplied by 128 (default increment) or 16 (alternate increment, selected in
14404e390cabSriastradh * LPT SOUTH_CHICKEN2 register bit 5).
14414e390cabSriastradh */
lpt_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14424e390cabSriastradh static u32 lpt_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14434e390cabSriastradh {
14444e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
14454e390cabSriastradh struct intel_panel *panel = &connector->panel;
14464e390cabSriastradh u32 mul, clock;
14474e390cabSriastradh
14484e390cabSriastradh if (panel->backlight.alternate_pwm_increment)
14494e390cabSriastradh mul = 16;
14504e390cabSriastradh else
14514e390cabSriastradh mul = 128;
14524e390cabSriastradh
14534e390cabSriastradh if (HAS_PCH_LPT_H(dev_priv))
14544e390cabSriastradh clock = MHz(135); /* LPT:H */
14554e390cabSriastradh else
14564e390cabSriastradh clock = MHz(24); /* LPT:LP */
14574e390cabSriastradh
14584e390cabSriastradh return DIV_ROUND_CLOSEST(clock, pwm_freq_hz * mul);
14594e390cabSriastradh }
14604e390cabSriastradh
14614e390cabSriastradh /*
14624e390cabSriastradh * ILK/SNB/IVB: This value represents the period of the PWM stream in PCH
14634e390cabSriastradh * display raw clocks multiplied by 128.
14644e390cabSriastradh */
pch_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14654e390cabSriastradh static u32 pch_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14664e390cabSriastradh {
14674e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
14684e390cabSriastradh
14694e390cabSriastradh return DIV_ROUND_CLOSEST(KHz(dev_priv->rawclk_freq), pwm_freq_hz * 128);
14704e390cabSriastradh }
14714e390cabSriastradh
14724e390cabSriastradh /*
14734e390cabSriastradh * Gen2: This field determines the number of time base events (display core
14744e390cabSriastradh * clock frequency/32) in total for a complete cycle of modulated backlight
14754e390cabSriastradh * control.
14764e390cabSriastradh *
14774e390cabSriastradh * Gen3: A time base event equals the display core clock ([DevPNV] HRAW clock)
14784e390cabSriastradh * divided by 32.
14794e390cabSriastradh */
i9xx_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14804e390cabSriastradh static u32 i9xx_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14814e390cabSriastradh {
14824e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
14834e390cabSriastradh int clock;
14844e390cabSriastradh
14854e390cabSriastradh if (IS_PINEVIEW(dev_priv))
14864e390cabSriastradh clock = KHz(dev_priv->rawclk_freq);
14874e390cabSriastradh else
14884e390cabSriastradh clock = KHz(dev_priv->cdclk.hw.cdclk);
14894e390cabSriastradh
14904e390cabSriastradh return DIV_ROUND_CLOSEST(clock, pwm_freq_hz * 32);
14914e390cabSriastradh }
14924e390cabSriastradh
14934e390cabSriastradh /*
14944e390cabSriastradh * Gen4: This value represents the period of the PWM stream in display core
14954e390cabSriastradh * clocks ([DevCTG] HRAW clocks) multiplied by 128.
14964e390cabSriastradh *
14974e390cabSriastradh */
i965_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)14984e390cabSriastradh static u32 i965_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
14994e390cabSriastradh {
15004e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
15014e390cabSriastradh int clock;
15024e390cabSriastradh
15034e390cabSriastradh if (IS_G4X(dev_priv))
15044e390cabSriastradh clock = KHz(dev_priv->rawclk_freq);
15054e390cabSriastradh else
15064e390cabSriastradh clock = KHz(dev_priv->cdclk.hw.cdclk);
15074e390cabSriastradh
15084e390cabSriastradh return DIV_ROUND_CLOSEST(clock, pwm_freq_hz * 128);
15094e390cabSriastradh }
15104e390cabSriastradh
15114e390cabSriastradh /*
15124e390cabSriastradh * VLV: This value represents the period of the PWM stream in display core
15134e390cabSriastradh * clocks ([DevCTG] 200MHz HRAW clocks) multiplied by 128 or 25MHz S0IX clocks
15144e390cabSriastradh * multiplied by 16. CHV uses a 19.2MHz S0IX clock.
15154e390cabSriastradh */
vlv_hz_to_pwm(struct intel_connector * connector,u32 pwm_freq_hz)15164e390cabSriastradh static u32 vlv_hz_to_pwm(struct intel_connector *connector, u32 pwm_freq_hz)
15174e390cabSriastradh {
15184e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
15194e390cabSriastradh int mul, clock;
15204e390cabSriastradh
15214e390cabSriastradh if ((I915_READ(CBR1_VLV) & CBR_PWM_CLOCK_MUX_SELECT) == 0) {
15224e390cabSriastradh if (IS_CHERRYVIEW(dev_priv))
15234e390cabSriastradh clock = KHz(19200);
15244e390cabSriastradh else
15254e390cabSriastradh clock = MHz(25);
15264e390cabSriastradh mul = 16;
15274e390cabSriastradh } else {
15284e390cabSriastradh clock = KHz(dev_priv->rawclk_freq);
15294e390cabSriastradh mul = 128;
15304e390cabSriastradh }
15314e390cabSriastradh
15324e390cabSriastradh return DIV_ROUND_CLOSEST(clock, pwm_freq_hz * mul);
15334e390cabSriastradh }
15344e390cabSriastradh
get_backlight_max_vbt(struct intel_connector * connector)15354e390cabSriastradh static u32 get_backlight_max_vbt(struct intel_connector *connector)
15364e390cabSriastradh {
15374e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
15384e390cabSriastradh struct intel_panel *panel = &connector->panel;
15394e390cabSriastradh u16 pwm_freq_hz = dev_priv->vbt.backlight.pwm_freq_hz;
15404e390cabSriastradh u32 pwm;
15414e390cabSriastradh
15424e390cabSriastradh if (!panel->backlight.hz_to_pwm) {
15434e390cabSriastradh DRM_DEBUG_KMS("backlight frequency conversion not supported\n");
15444e390cabSriastradh return 0;
15454e390cabSriastradh }
15464e390cabSriastradh
15474e390cabSriastradh if (pwm_freq_hz) {
15484e390cabSriastradh DRM_DEBUG_KMS("VBT defined backlight frequency %u Hz\n",
15494e390cabSriastradh pwm_freq_hz);
15504e390cabSriastradh } else {
15514e390cabSriastradh pwm_freq_hz = 200;
15524e390cabSriastradh DRM_DEBUG_KMS("default backlight frequency %u Hz\n",
15534e390cabSriastradh pwm_freq_hz);
15544e390cabSriastradh }
15554e390cabSriastradh
15564e390cabSriastradh pwm = panel->backlight.hz_to_pwm(connector, pwm_freq_hz);
15574e390cabSriastradh if (!pwm) {
15584e390cabSriastradh DRM_DEBUG_KMS("backlight frequency conversion failed\n");
15594e390cabSriastradh return 0;
15604e390cabSriastradh }
15614e390cabSriastradh
15624e390cabSriastradh return pwm;
15634e390cabSriastradh }
15644e390cabSriastradh
15654e390cabSriastradh /*
15664e390cabSriastradh * Note: The setup hooks can't assume pipe is set!
15674e390cabSriastradh */
get_backlight_min_vbt(struct intel_connector * connector)15684e390cabSriastradh static u32 get_backlight_min_vbt(struct intel_connector *connector)
15694e390cabSriastradh {
15704e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
15714e390cabSriastradh struct intel_panel *panel = &connector->panel;
15724e390cabSriastradh int min;
15734e390cabSriastradh
15744e390cabSriastradh WARN_ON(panel->backlight.max == 0);
15754e390cabSriastradh
15764e390cabSriastradh /*
15774e390cabSriastradh * XXX: If the vbt value is 255, it makes min equal to max, which leads
15784e390cabSriastradh * to problems. There are such machines out there. Either our
15794e390cabSriastradh * interpretation is wrong or the vbt has bogus data. Or both. Safeguard
15804e390cabSriastradh * against this by letting the minimum be at most (arbitrarily chosen)
15814e390cabSriastradh * 25% of the max.
15824e390cabSriastradh */
15834e390cabSriastradh min = clamp_t(int, dev_priv->vbt.backlight.min_brightness, 0, 64);
15844e390cabSriastradh if (min != dev_priv->vbt.backlight.min_brightness) {
15854e390cabSriastradh DRM_DEBUG_KMS("clamping VBT min backlight %d/255 to %d/255\n",
15864e390cabSriastradh dev_priv->vbt.backlight.min_brightness, min);
15874e390cabSriastradh }
15884e390cabSriastradh
15894e390cabSriastradh /* vbt value is a coefficient in range [0..255] */
15904e390cabSriastradh return scale(min, 0, 255, 0, panel->backlight.max);
15914e390cabSriastradh }
15924e390cabSriastradh
lpt_setup_backlight(struct intel_connector * connector,enum pipe unused)15934e390cabSriastradh static int lpt_setup_backlight(struct intel_connector *connector, enum pipe unused)
15944e390cabSriastradh {
15954e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
15964e390cabSriastradh struct intel_panel *panel = &connector->panel;
15974e390cabSriastradh u32 cpu_ctl2, pch_ctl1, pch_ctl2, val;
15984e390cabSriastradh bool alt, cpu_mode;
15994e390cabSriastradh
16004e390cabSriastradh if (HAS_PCH_LPT(dev_priv))
16014e390cabSriastradh alt = I915_READ(SOUTH_CHICKEN2) & LPT_PWM_GRANULARITY;
16024e390cabSriastradh else
16034e390cabSriastradh alt = I915_READ(SOUTH_CHICKEN1) & SPT_PWM_GRANULARITY;
16044e390cabSriastradh panel->backlight.alternate_pwm_increment = alt;
16054e390cabSriastradh
16064e390cabSriastradh pch_ctl1 = I915_READ(BLC_PWM_PCH_CTL1);
16074e390cabSriastradh panel->backlight.active_low_pwm = pch_ctl1 & BLM_PCH_POLARITY;
16084e390cabSriastradh
16094e390cabSriastradh pch_ctl2 = I915_READ(BLC_PWM_PCH_CTL2);
16104e390cabSriastradh panel->backlight.max = pch_ctl2 >> 16;
16114e390cabSriastradh
16124e390cabSriastradh cpu_ctl2 = I915_READ(BLC_PWM_CPU_CTL2);
16134e390cabSriastradh
16144e390cabSriastradh if (!panel->backlight.max)
16154e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
16164e390cabSriastradh
16174e390cabSriastradh if (!panel->backlight.max)
16184e390cabSriastradh return -ENODEV;
16194e390cabSriastradh
16204e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
16214e390cabSriastradh
16224e390cabSriastradh panel->backlight.enabled = pch_ctl1 & BLM_PCH_PWM_ENABLE;
16234e390cabSriastradh
16244e390cabSriastradh cpu_mode = panel->backlight.enabled && HAS_PCH_LPT(dev_priv) &&
16254e390cabSriastradh !(pch_ctl1 & BLM_PCH_OVERRIDE_ENABLE) &&
16264e390cabSriastradh (cpu_ctl2 & BLM_PWM_ENABLE);
16274e390cabSriastradh if (cpu_mode)
16284e390cabSriastradh val = pch_get_backlight(connector);
16294e390cabSriastradh else
16304e390cabSriastradh val = lpt_get_backlight(connector);
16314e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
16324e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
16334e390cabSriastradh panel->backlight.max);
16344e390cabSriastradh
16354e390cabSriastradh if (cpu_mode) {
16364e390cabSriastradh DRM_DEBUG_KMS("CPU backlight register was enabled, switching to PCH override\n");
16374e390cabSriastradh
16384e390cabSriastradh /* Write converted CPU PWM value to PCH override register */
16394e390cabSriastradh lpt_set_backlight(connector->base.state, panel->backlight.level);
16404e390cabSriastradh I915_WRITE(BLC_PWM_PCH_CTL1, pch_ctl1 | BLM_PCH_OVERRIDE_ENABLE);
16414e390cabSriastradh
16424e390cabSriastradh I915_WRITE(BLC_PWM_CPU_CTL2, cpu_ctl2 & ~BLM_PWM_ENABLE);
16434e390cabSriastradh }
16444e390cabSriastradh
16454e390cabSriastradh return 0;
16464e390cabSriastradh }
16474e390cabSriastradh
pch_setup_backlight(struct intel_connector * connector,enum pipe unused)16484e390cabSriastradh static int pch_setup_backlight(struct intel_connector *connector, enum pipe unused)
16494e390cabSriastradh {
16504e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
16514e390cabSriastradh struct intel_panel *panel = &connector->panel;
16524e390cabSriastradh u32 cpu_ctl2, pch_ctl1, pch_ctl2, val;
16534e390cabSriastradh
16544e390cabSriastradh pch_ctl1 = I915_READ(BLC_PWM_PCH_CTL1);
16554e390cabSriastradh panel->backlight.active_low_pwm = pch_ctl1 & BLM_PCH_POLARITY;
16564e390cabSriastradh
16574e390cabSriastradh pch_ctl2 = I915_READ(BLC_PWM_PCH_CTL2);
16584e390cabSriastradh panel->backlight.max = pch_ctl2 >> 16;
16594e390cabSriastradh
16604e390cabSriastradh if (!panel->backlight.max)
16614e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
16624e390cabSriastradh
16634e390cabSriastradh if (!panel->backlight.max)
16644e390cabSriastradh return -ENODEV;
16654e390cabSriastradh
16664e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
16674e390cabSriastradh
16684e390cabSriastradh val = pch_get_backlight(connector);
16694e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
16704e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
16714e390cabSriastradh panel->backlight.max);
16724e390cabSriastradh
16734e390cabSriastradh cpu_ctl2 = I915_READ(BLC_PWM_CPU_CTL2);
16744e390cabSriastradh panel->backlight.enabled = (cpu_ctl2 & BLM_PWM_ENABLE) &&
16754e390cabSriastradh (pch_ctl1 & BLM_PCH_PWM_ENABLE);
16764e390cabSriastradh
16774e390cabSriastradh return 0;
16784e390cabSriastradh }
16794e390cabSriastradh
i9xx_setup_backlight(struct intel_connector * connector,enum pipe unused)16804e390cabSriastradh static int i9xx_setup_backlight(struct intel_connector *connector, enum pipe unused)
16814e390cabSriastradh {
16824e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
16834e390cabSriastradh struct intel_panel *panel = &connector->panel;
16844e390cabSriastradh u32 ctl, val;
16854e390cabSriastradh
16864e390cabSriastradh ctl = I915_READ(BLC_PWM_CTL);
16874e390cabSriastradh
16884e390cabSriastradh if (IS_GEN(dev_priv, 2) || IS_I915GM(dev_priv) || IS_I945GM(dev_priv))
16894e390cabSriastradh panel->backlight.combination_mode = ctl & BLM_LEGACY_MODE;
16904e390cabSriastradh
16914e390cabSriastradh if (IS_PINEVIEW(dev_priv))
16924e390cabSriastradh panel->backlight.active_low_pwm = ctl & BLM_POLARITY_PNV;
16934e390cabSriastradh
16944e390cabSriastradh panel->backlight.max = ctl >> 17;
16954e390cabSriastradh
16964e390cabSriastradh if (!panel->backlight.max) {
16974e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
16984e390cabSriastradh panel->backlight.max >>= 1;
16994e390cabSriastradh }
17004e390cabSriastradh
17014e390cabSriastradh if (!panel->backlight.max)
17024e390cabSriastradh return -ENODEV;
17034e390cabSriastradh
17044e390cabSriastradh if (panel->backlight.combination_mode)
17054e390cabSriastradh panel->backlight.max *= 0xff;
17064e390cabSriastradh
17074e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
17084e390cabSriastradh
17094e390cabSriastradh val = i9xx_get_backlight(connector);
17104e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
17114e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
17124e390cabSriastradh panel->backlight.max);
17134e390cabSriastradh
17144e390cabSriastradh panel->backlight.enabled = val != 0;
17154e390cabSriastradh
17164e390cabSriastradh return 0;
17174e390cabSriastradh }
17184e390cabSriastradh
i965_setup_backlight(struct intel_connector * connector,enum pipe unused)17194e390cabSriastradh static int i965_setup_backlight(struct intel_connector *connector, enum pipe unused)
17204e390cabSriastradh {
17214e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
17224e390cabSriastradh struct intel_panel *panel = &connector->panel;
17234e390cabSriastradh u32 ctl, ctl2, val;
17244e390cabSriastradh
17254e390cabSriastradh ctl2 = I915_READ(BLC_PWM_CTL2);
17264e390cabSriastradh panel->backlight.combination_mode = ctl2 & BLM_COMBINATION_MODE;
17274e390cabSriastradh panel->backlight.active_low_pwm = ctl2 & BLM_POLARITY_I965;
17284e390cabSriastradh
17294e390cabSriastradh ctl = I915_READ(BLC_PWM_CTL);
17304e390cabSriastradh panel->backlight.max = ctl >> 16;
17314e390cabSriastradh
17324e390cabSriastradh if (!panel->backlight.max)
17334e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
17344e390cabSriastradh
17354e390cabSriastradh if (!panel->backlight.max)
17364e390cabSriastradh return -ENODEV;
17374e390cabSriastradh
17384e390cabSriastradh if (panel->backlight.combination_mode)
17394e390cabSriastradh panel->backlight.max *= 0xff;
17404e390cabSriastradh
17414e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
17424e390cabSriastradh
17434e390cabSriastradh val = i9xx_get_backlight(connector);
17444e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
17454e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
17464e390cabSriastradh panel->backlight.max);
17474e390cabSriastradh
17484e390cabSriastradh panel->backlight.enabled = ctl2 & BLM_PWM_ENABLE;
17494e390cabSriastradh
17504e390cabSriastradh return 0;
17514e390cabSriastradh }
17524e390cabSriastradh
vlv_setup_backlight(struct intel_connector * connector,enum pipe pipe)17534e390cabSriastradh static int vlv_setup_backlight(struct intel_connector *connector, enum pipe pipe)
17544e390cabSriastradh {
17554e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
17564e390cabSriastradh struct intel_panel *panel = &connector->panel;
17574e390cabSriastradh u32 ctl, ctl2, val;
17584e390cabSriastradh
17594e390cabSriastradh if (WARN_ON(pipe != PIPE_A && pipe != PIPE_B))
17604e390cabSriastradh return -ENODEV;
17614e390cabSriastradh
17624e390cabSriastradh ctl2 = I915_READ(VLV_BLC_PWM_CTL2(pipe));
17634e390cabSriastradh panel->backlight.active_low_pwm = ctl2 & BLM_POLARITY_I965;
17644e390cabSriastradh
17654e390cabSriastradh ctl = I915_READ(VLV_BLC_PWM_CTL(pipe));
17664e390cabSriastradh panel->backlight.max = ctl >> 16;
17674e390cabSriastradh
17684e390cabSriastradh if (!panel->backlight.max)
17694e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
17704e390cabSriastradh
17714e390cabSriastradh if (!panel->backlight.max)
17724e390cabSriastradh return -ENODEV;
17734e390cabSriastradh
17744e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
17754e390cabSriastradh
17764e390cabSriastradh val = _vlv_get_backlight(dev_priv, pipe);
17774e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
17784e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
17794e390cabSriastradh panel->backlight.max);
17804e390cabSriastradh
17814e390cabSriastradh panel->backlight.enabled = ctl2 & BLM_PWM_ENABLE;
17824e390cabSriastradh
17834e390cabSriastradh return 0;
17844e390cabSriastradh }
17854e390cabSriastradh
17864e390cabSriastradh static int
bxt_setup_backlight(struct intel_connector * connector,enum pipe unused)17874e390cabSriastradh bxt_setup_backlight(struct intel_connector *connector, enum pipe unused)
17884e390cabSriastradh {
17894e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
17904e390cabSriastradh struct intel_panel *panel = &connector->panel;
17914e390cabSriastradh u32 pwm_ctl, val;
17924e390cabSriastradh
17934e390cabSriastradh panel->backlight.controller = dev_priv->vbt.backlight.controller;
17944e390cabSriastradh
17954e390cabSriastradh pwm_ctl = I915_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
17964e390cabSriastradh
17974e390cabSriastradh /* Controller 1 uses the utility pin. */
17984e390cabSriastradh if (panel->backlight.controller == 1) {
17994e390cabSriastradh val = I915_READ(UTIL_PIN_CTL);
18004e390cabSriastradh panel->backlight.util_pin_active_low =
18014e390cabSriastradh val & UTIL_PIN_POLARITY;
18024e390cabSriastradh }
18034e390cabSriastradh
18044e390cabSriastradh panel->backlight.active_low_pwm = pwm_ctl & BXT_BLC_PWM_POLARITY;
18054e390cabSriastradh panel->backlight.max =
18064e390cabSriastradh I915_READ(BXT_BLC_PWM_FREQ(panel->backlight.controller));
18074e390cabSriastradh
18084e390cabSriastradh if (!panel->backlight.max)
18094e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
18104e390cabSriastradh
18114e390cabSriastradh if (!panel->backlight.max)
18124e390cabSriastradh return -ENODEV;
18134e390cabSriastradh
18144e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
18154e390cabSriastradh
18164e390cabSriastradh val = bxt_get_backlight(connector);
18174e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
18184e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
18194e390cabSriastradh panel->backlight.max);
18204e390cabSriastradh
18214e390cabSriastradh panel->backlight.enabled = pwm_ctl & BXT_BLC_PWM_ENABLE;
18224e390cabSriastradh
18234e390cabSriastradh return 0;
18244e390cabSriastradh }
18254e390cabSriastradh
18264e390cabSriastradh static int
cnp_setup_backlight(struct intel_connector * connector,enum pipe unused)18274e390cabSriastradh cnp_setup_backlight(struct intel_connector *connector, enum pipe unused)
18284e390cabSriastradh {
18294e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
18304e390cabSriastradh struct intel_panel *panel = &connector->panel;
18314e390cabSriastradh u32 pwm_ctl, val;
18324e390cabSriastradh
18334e390cabSriastradh /*
18344e390cabSriastradh * CNP has the BXT implementation of backlight, but with only one
18354e390cabSriastradh * controller. TODO: ICP has multiple controllers but we only use
18364e390cabSriastradh * controller 0 for now.
18374e390cabSriastradh */
18384e390cabSriastradh panel->backlight.controller = 0;
18394e390cabSriastradh
18404e390cabSriastradh pwm_ctl = I915_READ(BXT_BLC_PWM_CTL(panel->backlight.controller));
18414e390cabSriastradh
18424e390cabSriastradh panel->backlight.active_low_pwm = pwm_ctl & BXT_BLC_PWM_POLARITY;
18434e390cabSriastradh panel->backlight.max =
18444e390cabSriastradh I915_READ(BXT_BLC_PWM_FREQ(panel->backlight.controller));
18454e390cabSriastradh
18464e390cabSriastradh if (!panel->backlight.max)
18474e390cabSriastradh panel->backlight.max = get_backlight_max_vbt(connector);
18484e390cabSriastradh
18494e390cabSriastradh if (!panel->backlight.max)
18504e390cabSriastradh return -ENODEV;
18514e390cabSriastradh
18524e390cabSriastradh panel->backlight.min = get_backlight_min_vbt(connector);
18534e390cabSriastradh
18544e390cabSriastradh val = bxt_get_backlight(connector);
18554e390cabSriastradh val = intel_panel_compute_brightness(connector, val);
18564e390cabSriastradh panel->backlight.level = clamp(val, panel->backlight.min,
18574e390cabSriastradh panel->backlight.max);
18584e390cabSriastradh
18594e390cabSriastradh panel->backlight.enabled = pwm_ctl & BXT_BLC_PWM_ENABLE;
18604e390cabSriastradh
18614e390cabSriastradh return 0;
18624e390cabSriastradh }
18634e390cabSriastradh
186441ec0267Sriastradh #ifndef __NetBSD__ /* XXX mipi */
pwm_setup_backlight(struct intel_connector * connector,enum pipe pipe)18654e390cabSriastradh static int pwm_setup_backlight(struct intel_connector *connector,
18664e390cabSriastradh enum pipe pipe)
18674e390cabSriastradh {
18684e390cabSriastradh struct drm_device *dev = connector->base.dev;
18694e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(dev);
18704e390cabSriastradh struct intel_panel *panel = &connector->panel;
18714e390cabSriastradh const char *desc;
18724e390cabSriastradh int retval;
18734e390cabSriastradh
18744e390cabSriastradh /* Get the right PWM chip for DSI backlight according to VBT */
18754e390cabSriastradh if (dev_priv->vbt.dsi.config->pwm_blc == PPS_BLC_PMIC) {
18764e390cabSriastradh panel->backlight.pwm = pwm_get(dev->dev, "pwm_pmic_backlight");
18774e390cabSriastradh desc = "PMIC";
18784e390cabSriastradh } else {
18794e390cabSriastradh panel->backlight.pwm = pwm_get(dev->dev, "pwm_soc_backlight");
18804e390cabSriastradh desc = "SoC";
18814e390cabSriastradh }
18824e390cabSriastradh
18834e390cabSriastradh if (IS_ERR(panel->backlight.pwm)) {
18844e390cabSriastradh DRM_ERROR("Failed to get the %s PWM chip\n", desc);
18854e390cabSriastradh panel->backlight.pwm = NULL;
18864e390cabSriastradh return -ENODEV;
18874e390cabSriastradh }
18884e390cabSriastradh
18894e390cabSriastradh /*
18904e390cabSriastradh * FIXME: pwm_apply_args() should be removed when switching to
18914e390cabSriastradh * the atomic PWM API.
18924e390cabSriastradh */
18934e390cabSriastradh pwm_apply_args(panel->backlight.pwm);
18944e390cabSriastradh
18954e390cabSriastradh retval = pwm_config(panel->backlight.pwm, CRC_PMIC_PWM_PERIOD_NS,
18964e390cabSriastradh CRC_PMIC_PWM_PERIOD_NS);
18974e390cabSriastradh if (retval < 0) {
18984e390cabSriastradh DRM_ERROR("Failed to configure the pwm chip\n");
18994e390cabSriastradh pwm_put(panel->backlight.pwm);
19004e390cabSriastradh panel->backlight.pwm = NULL;
19014e390cabSriastradh return retval;
19024e390cabSriastradh }
19034e390cabSriastradh
19044e390cabSriastradh panel->backlight.min = 0; /* 0% */
19054e390cabSriastradh panel->backlight.max = 100; /* 100% */
19064e390cabSriastradh panel->backlight.level = DIV_ROUND_UP(
19074e390cabSriastradh pwm_get_duty_cycle(panel->backlight.pwm) * 100,
19084e390cabSriastradh CRC_PMIC_PWM_PERIOD_NS);
19094e390cabSriastradh panel->backlight.enabled = panel->backlight.level != 0;
19104e390cabSriastradh
19114e390cabSriastradh DRM_INFO("Using %s PWM for LCD backlight control\n", desc);
19124e390cabSriastradh return 0;
19134e390cabSriastradh }
191441ec0267Sriastradh #endif
19154e390cabSriastradh
intel_panel_update_backlight(struct intel_encoder * encoder,const struct intel_crtc_state * crtc_state,const struct drm_connector_state * conn_state)19164e390cabSriastradh void intel_panel_update_backlight(struct intel_encoder *encoder,
19174e390cabSriastradh const struct intel_crtc_state *crtc_state,
19184e390cabSriastradh const struct drm_connector_state *conn_state)
19194e390cabSriastradh {
19204e390cabSriastradh struct intel_connector *connector = to_intel_connector(conn_state->connector);
19214e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
19224e390cabSriastradh struct intel_panel *panel = &connector->panel;
19234e390cabSriastradh
19244e390cabSriastradh if (!panel->backlight.present)
19254e390cabSriastradh return;
19264e390cabSriastradh
19274e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
19284e390cabSriastradh if (!panel->backlight.enabled)
19294e390cabSriastradh __intel_panel_enable_backlight(crtc_state, conn_state);
19304e390cabSriastradh
19314e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
19324e390cabSriastradh }
19334e390cabSriastradh
intel_panel_setup_backlight(struct drm_connector * connector,enum pipe pipe)19344e390cabSriastradh int intel_panel_setup_backlight(struct drm_connector *connector, enum pipe pipe)
19354e390cabSriastradh {
19364e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->dev);
19374e390cabSriastradh struct intel_connector *intel_connector = to_intel_connector(connector);
19384e390cabSriastradh struct intel_panel *panel = &intel_connector->panel;
19394e390cabSriastradh int ret;
19404e390cabSriastradh
19414e390cabSriastradh if (!dev_priv->vbt.backlight.present) {
19424e390cabSriastradh if (dev_priv->quirks & QUIRK_BACKLIGHT_PRESENT) {
19434e390cabSriastradh DRM_DEBUG_KMS("no backlight present per VBT, but present per quirk\n");
19444e390cabSriastradh } else {
19454e390cabSriastradh DRM_DEBUG_KMS("no backlight present per VBT\n");
19464e390cabSriastradh return 0;
19474e390cabSriastradh }
19484e390cabSriastradh }
19494e390cabSriastradh
19504e390cabSriastradh /* ensure intel_panel has been initialized first */
19514e390cabSriastradh if (WARN_ON(!panel->backlight.setup))
19524e390cabSriastradh return -ENODEV;
19534e390cabSriastradh
19544e390cabSriastradh /* set level and max in panel struct */
19554e390cabSriastradh mutex_lock(&dev_priv->backlight_lock);
19564e390cabSriastradh ret = panel->backlight.setup(intel_connector, pipe);
19574e390cabSriastradh mutex_unlock(&dev_priv->backlight_lock);
19584e390cabSriastradh
19594e390cabSriastradh if (ret) {
19604e390cabSriastradh DRM_DEBUG_KMS("failed to setup backlight for connector %s\n",
19614e390cabSriastradh connector->name);
19624e390cabSriastradh return ret;
19634e390cabSriastradh }
19644e390cabSriastradh
19654e390cabSriastradh panel->backlight.present = true;
19664e390cabSriastradh
19674e390cabSriastradh DRM_DEBUG_KMS("Connector %s backlight initialized, %s, brightness %u/%u\n",
19684e390cabSriastradh connector->name,
19694e390cabSriastradh enableddisabled(panel->backlight.enabled),
19704e390cabSriastradh panel->backlight.level, panel->backlight.max);
19714e390cabSriastradh
19724e390cabSriastradh return 0;
19734e390cabSriastradh }
19744e390cabSriastradh
intel_panel_destroy_backlight(struct intel_panel * panel)19754e390cabSriastradh static void intel_panel_destroy_backlight(struct intel_panel *panel)
19764e390cabSriastradh {
197741ec0267Sriastradh #ifndef __NetBSD__ /* XXX mipi */
19784e390cabSriastradh /* dispose of the pwm */
19794e390cabSriastradh if (panel->backlight.pwm)
19804e390cabSriastradh pwm_put(panel->backlight.pwm);
198141ec0267Sriastradh #endif
19824e390cabSriastradh
19834e390cabSriastradh panel->backlight.present = false;
19844e390cabSriastradh }
19854e390cabSriastradh
19864e390cabSriastradh /* Set up chip specific backlight functions */
19874e390cabSriastradh static void
intel_panel_init_backlight_funcs(struct intel_panel * panel)19884e390cabSriastradh intel_panel_init_backlight_funcs(struct intel_panel *panel)
19894e390cabSriastradh {
19904e390cabSriastradh struct intel_connector *connector =
19914e390cabSriastradh container_of(panel, struct intel_connector, panel);
19924e390cabSriastradh struct drm_i915_private *dev_priv = to_i915(connector->base.dev);
19934e390cabSriastradh
19944e390cabSriastradh if (connector->base.connector_type == DRM_MODE_CONNECTOR_eDP &&
19954e390cabSriastradh intel_dp_aux_init_backlight_funcs(connector) == 0)
19964e390cabSriastradh return;
19974e390cabSriastradh
19984e390cabSriastradh if (connector->base.connector_type == DRM_MODE_CONNECTOR_DSI &&
19994e390cabSriastradh intel_dsi_dcs_init_backlight_funcs(connector) == 0)
20004e390cabSriastradh return;
20014e390cabSriastradh
20024e390cabSriastradh if (IS_GEN9_LP(dev_priv)) {
20034e390cabSriastradh panel->backlight.setup = bxt_setup_backlight;
20044e390cabSriastradh panel->backlight.enable = bxt_enable_backlight;
20054e390cabSriastradh panel->backlight.disable = bxt_disable_backlight;
20064e390cabSriastradh panel->backlight.set = bxt_set_backlight;
20074e390cabSriastradh panel->backlight.get = bxt_get_backlight;
20084e390cabSriastradh panel->backlight.hz_to_pwm = bxt_hz_to_pwm;
20094e390cabSriastradh } else if (INTEL_PCH_TYPE(dev_priv) >= PCH_CNP) {
20104e390cabSriastradh panel->backlight.setup = cnp_setup_backlight;
20114e390cabSriastradh panel->backlight.enable = cnp_enable_backlight;
20124e390cabSriastradh panel->backlight.disable = cnp_disable_backlight;
20134e390cabSriastradh panel->backlight.set = bxt_set_backlight;
20144e390cabSriastradh panel->backlight.get = bxt_get_backlight;
20154e390cabSriastradh panel->backlight.hz_to_pwm = cnp_hz_to_pwm;
20164e390cabSriastradh } else if (INTEL_PCH_TYPE(dev_priv) >= PCH_LPT) {
20174e390cabSriastradh panel->backlight.setup = lpt_setup_backlight;
20184e390cabSriastradh panel->backlight.enable = lpt_enable_backlight;
20194e390cabSriastradh panel->backlight.disable = lpt_disable_backlight;
20204e390cabSriastradh panel->backlight.set = lpt_set_backlight;
20214e390cabSriastradh panel->backlight.get = lpt_get_backlight;
20224e390cabSriastradh if (HAS_PCH_LPT(dev_priv))
20234e390cabSriastradh panel->backlight.hz_to_pwm = lpt_hz_to_pwm;
20244e390cabSriastradh else
20254e390cabSriastradh panel->backlight.hz_to_pwm = spt_hz_to_pwm;
20264e390cabSriastradh } else if (HAS_PCH_SPLIT(dev_priv)) {
20274e390cabSriastradh panel->backlight.setup = pch_setup_backlight;
20284e390cabSriastradh panel->backlight.enable = pch_enable_backlight;
20294e390cabSriastradh panel->backlight.disable = pch_disable_backlight;
20304e390cabSriastradh panel->backlight.set = pch_set_backlight;
20314e390cabSriastradh panel->backlight.get = pch_get_backlight;
20324e390cabSriastradh panel->backlight.hz_to_pwm = pch_hz_to_pwm;
20334e390cabSriastradh } else if (IS_VALLEYVIEW(dev_priv) || IS_CHERRYVIEW(dev_priv)) {
20344e390cabSriastradh if (connector->base.connector_type == DRM_MODE_CONNECTOR_DSI) {
203541ec0267Sriastradh #ifdef __NetBSD__
203641ec0267Sriastradh panic("no muppets supported, sorry");
203741ec0267Sriastradh #else
20384e390cabSriastradh panel->backlight.setup = pwm_setup_backlight;
20394e390cabSriastradh panel->backlight.enable = pwm_enable_backlight;
20404e390cabSriastradh panel->backlight.disable = pwm_disable_backlight;
20414e390cabSriastradh panel->backlight.set = pwm_set_backlight;
20424e390cabSriastradh panel->backlight.get = pwm_get_backlight;
204341ec0267Sriastradh #endif
20444e390cabSriastradh } else {
20454e390cabSriastradh panel->backlight.setup = vlv_setup_backlight;
20464e390cabSriastradh panel->backlight.enable = vlv_enable_backlight;
20474e390cabSriastradh panel->backlight.disable = vlv_disable_backlight;
20484e390cabSriastradh panel->backlight.set = vlv_set_backlight;
20494e390cabSriastradh panel->backlight.get = vlv_get_backlight;
20504e390cabSriastradh panel->backlight.hz_to_pwm = vlv_hz_to_pwm;
20514e390cabSriastradh }
20524e390cabSriastradh } else if (IS_GEN(dev_priv, 4)) {
20534e390cabSriastradh panel->backlight.setup = i965_setup_backlight;
20544e390cabSriastradh panel->backlight.enable = i965_enable_backlight;
20554e390cabSriastradh panel->backlight.disable = i965_disable_backlight;
20564e390cabSriastradh panel->backlight.set = i9xx_set_backlight;
20574e390cabSriastradh panel->backlight.get = i9xx_get_backlight;
20584e390cabSriastradh panel->backlight.hz_to_pwm = i965_hz_to_pwm;
20594e390cabSriastradh } else {
20604e390cabSriastradh panel->backlight.setup = i9xx_setup_backlight;
20614e390cabSriastradh panel->backlight.enable = i9xx_enable_backlight;
20624e390cabSriastradh panel->backlight.disable = i9xx_disable_backlight;
20634e390cabSriastradh panel->backlight.set = i9xx_set_backlight;
20644e390cabSriastradh panel->backlight.get = i9xx_get_backlight;
20654e390cabSriastradh panel->backlight.hz_to_pwm = i9xx_hz_to_pwm;
20664e390cabSriastradh }
20674e390cabSriastradh }
20684e390cabSriastradh
intel_panel_init(struct intel_panel * panel,struct drm_display_mode * fixed_mode,struct drm_display_mode * downclock_mode)20694e390cabSriastradh int intel_panel_init(struct intel_panel *panel,
20704e390cabSriastradh struct drm_display_mode *fixed_mode,
20714e390cabSriastradh struct drm_display_mode *downclock_mode)
20724e390cabSriastradh {
20734e390cabSriastradh intel_panel_init_backlight_funcs(panel);
20744e390cabSriastradh
20754e390cabSriastradh panel->fixed_mode = fixed_mode;
20764e390cabSriastradh panel->downclock_mode = downclock_mode;
20774e390cabSriastradh
20784e390cabSriastradh return 0;
20794e390cabSriastradh }
20804e390cabSriastradh
intel_panel_fini(struct intel_panel * panel)20814e390cabSriastradh void intel_panel_fini(struct intel_panel *panel)
20824e390cabSriastradh {
20834e390cabSriastradh struct intel_connector *intel_connector =
20844e390cabSriastradh container_of(panel, struct intel_connector, panel);
20854e390cabSriastradh
20864e390cabSriastradh intel_panel_destroy_backlight(panel);
20874e390cabSriastradh
20884e390cabSriastradh if (panel->fixed_mode)
20894e390cabSriastradh drm_mode_destroy(intel_connector->base.dev, panel->fixed_mode);
20904e390cabSriastradh
20914e390cabSriastradh if (panel->downclock_mode)
20924e390cabSriastradh drm_mode_destroy(intel_connector->base.dev,
20934e390cabSriastradh panel->downclock_mode);
20944e390cabSriastradh }
2095