xref: /netbsd-src/sys/external/bsd/drm2/dist/drm/i915/display/intel_panel.c (revision aa96afee193559fe4ebe9ee0d88e3193dbac4c01)
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