blob: 72686171b2f195cbf586363ac49109ac8124651a [file] [log] [blame]
Kirill A. Shutemov026abc32012-03-08 16:02:20 +00001/**************************************************************************
2 * Copyright (c) 2011, Intel Corporation.
3 * All Rights Reserved.
4 *
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms and conditions of the GNU General Public License,
7 * version 2, as published by the Free Software Foundation.
8 *
9 * This program is distributed in the hope it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
12 * more details.
13 *
14 * You should have received a copy of the GNU General Public License along with
15 * this program; if not, write to the Free Software Foundation, Inc.,
16 * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
17 *
18 **************************************************************************/
19
20#include "psb_drv.h"
21#include "mid_bios.h"
22#include "mdfld_output.h"
23#include "mdfld_dsi_output.h"
24#include "tc35876x-dsi-lvds.h"
25
26#include <asm/intel_scu_ipc.h>
27
28#ifdef CONFIG_BACKLIGHT_CLASS_DEVICE
29
30#define MRST_BLC_MAX_PWM_REG_FREQ 0xFFFF
31#define BLC_PWM_PRECISION_FACTOR 100 /* 10000000 */
32#define BLC_PWM_FREQ_CALC_CONSTANT 32
33#define MHz 1000000
34#define BRIGHTNESS_MIN_LEVEL 1
35#define BRIGHTNESS_MAX_LEVEL 100
36#define BRIGHTNESS_MASK 0xFF
37#define BLC_POLARITY_NORMAL 0
38#define BLC_POLARITY_INVERSE 1
39#define BLC_ADJUSTMENT_MAX 100
40
41#define MDFLD_BLC_PWM_PRECISION_FACTOR 10
42#define MDFLD_BLC_MAX_PWM_REG_FREQ 0xFFFE
43#define MDFLD_BLC_MIN_PWM_REG_FREQ 0x2
44
45#define MDFLD_BACKLIGHT_PWM_POLARITY_BIT_CLEAR (0xFFFE)
46#define MDFLD_BACKLIGHT_PWM_CTL_SHIFT (16)
47
48static struct backlight_device *mdfld_backlight_device;
49
50int mdfld_set_brightness(struct backlight_device *bd)
51{
52 struct drm_device *dev =
53 (struct drm_device *)bl_get_data(mdfld_backlight_device);
54 struct drm_psb_private *dev_priv = dev->dev_private;
55 int level = bd->props.brightness;
56
57 DRM_DEBUG_DRIVER("backlight level set to %d\n", level);
58
59 /* Perform value bounds checking */
60 if (level < BRIGHTNESS_MIN_LEVEL)
61 level = BRIGHTNESS_MIN_LEVEL;
62
63 if (gma_power_begin(dev, false)) {
64 u32 adjusted_level = 0;
65
66 /*
67 * Adjust the backlight level with the percent in
68 * dev_priv->blc_adj2
69 */
70 adjusted_level = level * dev_priv->blc_adj2;
71 adjusted_level = adjusted_level / BLC_ADJUSTMENT_MAX;
72 dev_priv->brightness_adjusted = adjusted_level;
73
74 if (mdfld_get_panel_type(dev, 0) == TC35876X) {
75 if (dev_priv->dpi_panel_on[0] ||
76 dev_priv->dpi_panel_on[2])
77 tc35876x_brightness_control(dev,
78 dev_priv->brightness_adjusted);
79 } else {
80 if (dev_priv->dpi_panel_on[0])
81 mdfld_dsi_brightness_control(dev, 0,
82 dev_priv->brightness_adjusted);
83 }
84
85 if (dev_priv->dpi_panel_on[2])
86 mdfld_dsi_brightness_control(dev, 2,
87 dev_priv->brightness_adjusted);
88 gma_power_end(dev);
89 }
90
91 /* cache the brightness for later use */
92 dev_priv->brightness = level;
93 return 0;
94}
95
Kirill A. Shutemov1c6a6262012-03-08 16:11:14 +000096static int mdfld_get_brightness(struct backlight_device *bd)
Kirill A. Shutemov026abc32012-03-08 16:02:20 +000097{
98 struct drm_device *dev =
99 (struct drm_device *)bl_get_data(mdfld_backlight_device);
100 struct drm_psb_private *dev_priv = dev->dev_private;
101
102 DRM_DEBUG_DRIVER("brightness = 0x%x \n", dev_priv->brightness);
103
104 /* return locally cached var instead of HW read (due to DPST etc.) */
105 return dev_priv->brightness;
106}
107
108static const struct backlight_ops mdfld_ops = {
109 .get_brightness = mdfld_get_brightness,
110 .update_status = mdfld_set_brightness,
111};
112
113static int device_backlight_init(struct drm_device *dev)
114{
115 struct drm_psb_private *dev_priv = (struct drm_psb_private *)
116 dev->dev_private;
117
118 dev_priv->blc_adj1 = BLC_ADJUSTMENT_MAX;
119 dev_priv->blc_adj2 = BLC_ADJUSTMENT_MAX;
120
121 return 0;
122}
123
Kirill A. Shutemov1c6a6262012-03-08 16:11:14 +0000124static int mdfld_backlight_init(struct drm_device *dev)
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000125{
126 struct backlight_properties props;
127 int ret = 0;
128
129 memset(&props, 0, sizeof(struct backlight_properties));
130 props.max_brightness = BRIGHTNESS_MAX_LEVEL;
131 props.type = BACKLIGHT_PLATFORM;
132 mdfld_backlight_device = backlight_device_register("mdfld-bl",
133 NULL, (void *)dev, &mdfld_ops, &props);
134
135 if (IS_ERR(mdfld_backlight_device))
136 return PTR_ERR(mdfld_backlight_device);
137
138 ret = device_backlight_init(dev);
139 if (ret)
140 return ret;
141
142 mdfld_backlight_device->props.brightness = BRIGHTNESS_MAX_LEVEL;
143 mdfld_backlight_device->props.max_brightness = BRIGHTNESS_MAX_LEVEL;
144 backlight_update_status(mdfld_backlight_device);
145 return 0;
146}
147#endif
148
149struct backlight_device *mdfld_get_backlight_device(void)
150{
151#ifdef CONFIG_BACKLIGHT_CLASS_DEVICE
152 return mdfld_backlight_device;
153#else
154 return NULL;
155#endif
156}
157
158/*
159 * mdfld_save_display_registers
160 *
161 * Description: We are going to suspend so save current display
162 * register state.
163 *
164 * Notes: FIXME_JLIU7 need to add the support for DPI MIPI & HDMI audio
165 */
Alan Cox62563042012-05-11 11:30:16 +0100166static int mdfld_save_display_registers(struct drm_device *dev, int pipenum)
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000167{
168 struct drm_psb_private *dev_priv = dev->dev_private;
169 struct medfield_state *regs = &dev_priv->regs.mdfld;
Alan Cox62563042012-05-11 11:30:16 +0100170 struct psb_pipe *pipe = &dev_priv->regs.pipe[pipenum];
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000171 int i;
172
173 /* register */
174 u32 dpll_reg = MRST_DPLL_A;
175 u32 fp_reg = MRST_FPA0;
176 u32 pipeconf_reg = PIPEACONF;
177 u32 htot_reg = HTOTAL_A;
178 u32 hblank_reg = HBLANK_A;
179 u32 hsync_reg = HSYNC_A;
180 u32 vtot_reg = VTOTAL_A;
181 u32 vblank_reg = VBLANK_A;
182 u32 vsync_reg = VSYNC_A;
183 u32 pipesrc_reg = PIPEASRC;
184 u32 dspstride_reg = DSPASTRIDE;
185 u32 dsplinoff_reg = DSPALINOFF;
186 u32 dsptileoff_reg = DSPATILEOFF;
187 u32 dspsize_reg = DSPASIZE;
188 u32 dsppos_reg = DSPAPOS;
189 u32 dspsurf_reg = DSPASURF;
190 u32 mipi_reg = MIPI;
191 u32 dspcntr_reg = DSPACNTR;
192 u32 dspstatus_reg = PIPEASTAT;
193 u32 palette_reg = PALETTE_A;
194
195 /* pointer to values */
Alan Cox62563042012-05-11 11:30:16 +0100196 u32 *dpll_val = &pipe->dpll;
197 u32 *fp_val = &pipe->fp0;
198 u32 *pipeconf_val = &pipe->conf;
199 u32 *htot_val = &pipe->htotal;
200 u32 *hblank_val = &pipe->hblank;
201 u32 *hsync_val = &pipe->hsync;
202 u32 *vtot_val = &pipe->vtotal;
203 u32 *vblank_val = &pipe->vblank;
204 u32 *vsync_val = &pipe->vsync;
205 u32 *pipesrc_val = &pipe->src;
206 u32 *dspstride_val = &pipe->stride;
207 u32 *dsplinoff_val = &pipe->linoff;
208 u32 *dsptileoff_val = &pipe->tileoff;
209 u32 *dspsize_val = &pipe->size;
210 u32 *dsppos_val = &pipe->pos;
211 u32 *dspsurf_val = &pipe->surf;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000212 u32 *mipi_val = &regs->saveMIPI;
Alan Cox62563042012-05-11 11:30:16 +0100213 u32 *dspcntr_val = &pipe->cntr;
214 u32 *dspstatus_val = &pipe->status;
215 u32 *palette_val = pipe->palette;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000216
Alan Cox62563042012-05-11 11:30:16 +0100217 switch (pipenum) {
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000218 case 0:
219 break;
220 case 1:
221 /* regester */
222 dpll_reg = MDFLD_DPLL_B;
223 fp_reg = MDFLD_DPLL_DIV0;
224 pipeconf_reg = PIPEBCONF;
225 htot_reg = HTOTAL_B;
226 hblank_reg = HBLANK_B;
227 hsync_reg = HSYNC_B;
228 vtot_reg = VTOTAL_B;
229 vblank_reg = VBLANK_B;
230 vsync_reg = VSYNC_B;
231 pipesrc_reg = PIPEBSRC;
232 dspstride_reg = DSPBSTRIDE;
233 dsplinoff_reg = DSPBLINOFF;
234 dsptileoff_reg = DSPBTILEOFF;
235 dspsize_reg = DSPBSIZE;
236 dsppos_reg = DSPBPOS;
237 dspsurf_reg = DSPBSURF;
238 dspcntr_reg = DSPBCNTR;
239 dspstatus_reg = PIPEBSTAT;
240 palette_reg = PALETTE_B;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000241 break;
242 case 2:
243 /* register */
244 pipeconf_reg = PIPECCONF;
245 htot_reg = HTOTAL_C;
246 hblank_reg = HBLANK_C;
247 hsync_reg = HSYNC_C;
248 vtot_reg = VTOTAL_C;
249 vblank_reg = VBLANK_C;
250 vsync_reg = VSYNC_C;
251 pipesrc_reg = PIPECSRC;
252 dspstride_reg = DSPCSTRIDE;
253 dsplinoff_reg = DSPCLINOFF;
254 dsptileoff_reg = DSPCTILEOFF;
255 dspsize_reg = DSPCSIZE;
256 dsppos_reg = DSPCPOS;
257 dspsurf_reg = DSPCSURF;
258 mipi_reg = MIPI_C;
259 dspcntr_reg = DSPCCNTR;
260 dspstatus_reg = PIPECSTAT;
261 palette_reg = PALETTE_C;
262
263 /* pointer to values */
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000264 mipi_val = &regs->saveMIPI_C;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000265 break;
266 default:
267 DRM_ERROR("%s, invalid pipe number.\n", __func__);
268 return -EINVAL;
269 }
270
271 /* Pipe & plane A info */
272 *dpll_val = PSB_RVDC32(dpll_reg);
273 *fp_val = PSB_RVDC32(fp_reg);
274 *pipeconf_val = PSB_RVDC32(pipeconf_reg);
275 *htot_val = PSB_RVDC32(htot_reg);
276 *hblank_val = PSB_RVDC32(hblank_reg);
277 *hsync_val = PSB_RVDC32(hsync_reg);
278 *vtot_val = PSB_RVDC32(vtot_reg);
279 *vblank_val = PSB_RVDC32(vblank_reg);
280 *vsync_val = PSB_RVDC32(vsync_reg);
281 *pipesrc_val = PSB_RVDC32(pipesrc_reg);
282 *dspstride_val = PSB_RVDC32(dspstride_reg);
283 *dsplinoff_val = PSB_RVDC32(dsplinoff_reg);
284 *dsptileoff_val = PSB_RVDC32(dsptileoff_reg);
285 *dspsize_val = PSB_RVDC32(dspsize_reg);
286 *dsppos_val = PSB_RVDC32(dsppos_reg);
287 *dspsurf_val = PSB_RVDC32(dspsurf_reg);
288 *dspcntr_val = PSB_RVDC32(dspcntr_reg);
289 *dspstatus_val = PSB_RVDC32(dspstatus_reg);
290
291 /*save palette (gamma) */
292 for (i = 0; i < 256; i++)
293 palette_val[i] = PSB_RVDC32(palette_reg + (i << 2));
294
Alan Cox62563042012-05-11 11:30:16 +0100295 if (pipenum == 1) {
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000296 regs->savePFIT_CONTROL = PSB_RVDC32(PFIT_CONTROL);
297 regs->savePFIT_PGM_RATIOS = PSB_RVDC32(PFIT_PGM_RATIOS);
298
299 regs->saveHDMIPHYMISCCTL = PSB_RVDC32(HDMIPHYMISCCTL);
300 regs->saveHDMIB_CONTROL = PSB_RVDC32(HDMIB_CONTROL);
301 return 0;
302 }
303
304 *mipi_val = PSB_RVDC32(mipi_reg);
305 return 0;
306}
307
308/*
309 * mdfld_restore_display_registers
310 *
311 * Description: We are going to resume so restore display register state.
312 *
313 * Notes: FIXME_JLIU7 need to add the support for DPI MIPI & HDMI audio
314 */
Alan Cox62563042012-05-11 11:30:16 +0100315static int mdfld_restore_display_registers(struct drm_device *dev, int pipenum)
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000316{
317 /* To get panel out of ULPS mode. */
318 u32 temp = 0;
319 u32 device_ready_reg = DEVICE_READY_REG;
320 struct drm_psb_private *dev_priv = dev->dev_private;
321 struct mdfld_dsi_config *dsi_config = NULL;
322 struct medfield_state *regs = &dev_priv->regs.mdfld;
Alan Cox62563042012-05-11 11:30:16 +0100323 struct psb_pipe *pipe = &dev_priv->regs.pipe[pipenum];
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000324 u32 i = 0;
325 u32 dpll = 0;
326 u32 timeout = 0;
327
Alan Cox62563042012-05-11 11:30:16 +0100328 /* register */
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000329 u32 dpll_reg = MRST_DPLL_A;
330 u32 fp_reg = MRST_FPA0;
331 u32 pipeconf_reg = PIPEACONF;
332 u32 htot_reg = HTOTAL_A;
333 u32 hblank_reg = HBLANK_A;
334 u32 hsync_reg = HSYNC_A;
335 u32 vtot_reg = VTOTAL_A;
336 u32 vblank_reg = VBLANK_A;
337 u32 vsync_reg = VSYNC_A;
338 u32 pipesrc_reg = PIPEASRC;
339 u32 dspstride_reg = DSPASTRIDE;
340 u32 dsplinoff_reg = DSPALINOFF;
341 u32 dsptileoff_reg = DSPATILEOFF;
342 u32 dspsize_reg = DSPASIZE;
343 u32 dsppos_reg = DSPAPOS;
344 u32 dspsurf_reg = DSPASURF;
345 u32 dspstatus_reg = PIPEASTAT;
346 u32 mipi_reg = MIPI;
347 u32 dspcntr_reg = DSPACNTR;
348 u32 palette_reg = PALETTE_A;
349
350 /* values */
Alan Cox62563042012-05-11 11:30:16 +0100351 u32 dpll_val = pipe->dpll;
352 u32 fp_val = pipe->fp0;
353 u32 pipeconf_val = pipe->conf;
354 u32 htot_val = pipe->htotal;
355 u32 hblank_val = pipe->hblank;
356 u32 hsync_val = pipe->hsync;
357 u32 vtot_val = pipe->vtotal;
358 u32 vblank_val = pipe->vblank;
359 u32 vsync_val = pipe->vsync;
360 u32 pipesrc_val = pipe->src;
361 u32 dspstride_val = pipe->stride;
362 u32 dsplinoff_val = pipe->linoff;
363 u32 dsptileoff_val = pipe->tileoff;
364 u32 dspsize_val = pipe->size;
365 u32 dsppos_val = pipe->pos;
366 u32 dspsurf_val = pipe->surf;
367 u32 dspstatus_val = pipe->status;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000368 u32 mipi_val = regs->saveMIPI;
Alan Cox62563042012-05-11 11:30:16 +0100369 u32 dspcntr_val = pipe->cntr;
370 u32 *palette_val = pipe->palette;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000371
Alan Cox62563042012-05-11 11:30:16 +0100372 switch (pipenum) {
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000373 case 0:
Alan Cox62563042012-05-11 11:30:16 +0100374 dpll_val &= ~DPLL_VCO_ENABLE;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000375 dsi_config = dev_priv->dsi_configs[0];
376 break;
377 case 1:
Alan Cox62563042012-05-11 11:30:16 +0100378 /* register */
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000379 dpll_reg = MDFLD_DPLL_B;
380 fp_reg = MDFLD_DPLL_DIV0;
381 pipeconf_reg = PIPEBCONF;
382 htot_reg = HTOTAL_B;
383 hblank_reg = HBLANK_B;
384 hsync_reg = HSYNC_B;
385 vtot_reg = VTOTAL_B;
386 vblank_reg = VBLANK_B;
387 vsync_reg = VSYNC_B;
388 pipesrc_reg = PIPEBSRC;
389 dspstride_reg = DSPBSTRIDE;
390 dsplinoff_reg = DSPBLINOFF;
391 dsptileoff_reg = DSPBTILEOFF;
392 dspsize_reg = DSPBSIZE;
393 dsppos_reg = DSPBPOS;
394 dspsurf_reg = DSPBSURF;
395 dspcntr_reg = DSPBCNTR;
396 dspstatus_reg = PIPEBSTAT;
397 palette_reg = PALETTE_B;
398
399 /* values */
Alan Cox62563042012-05-11 11:30:16 +0100400 dpll_val &= ~DPLL_VCO_ENABLE;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000401 break;
402 case 2:
Alan Cox62563042012-05-11 11:30:16 +0100403 /* register */
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000404 pipeconf_reg = PIPECCONF;
405 htot_reg = HTOTAL_C;
406 hblank_reg = HBLANK_C;
407 hsync_reg = HSYNC_C;
408 vtot_reg = VTOTAL_C;
409 vblank_reg = VBLANK_C;
410 vsync_reg = VSYNC_C;
411 pipesrc_reg = PIPECSRC;
412 dspstride_reg = DSPCSTRIDE;
413 dsplinoff_reg = DSPCLINOFF;
414 dsptileoff_reg = DSPCTILEOFF;
415 dspsize_reg = DSPCSIZE;
416 dsppos_reg = DSPCPOS;
417 dspsurf_reg = DSPCSURF;
418 mipi_reg = MIPI_C;
419 dspcntr_reg = DSPCCNTR;
420 dspstatus_reg = PIPECSTAT;
421 palette_reg = PALETTE_C;
422
423 /* values */
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000424 mipi_val = regs->saveMIPI_C;
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000425 dsi_config = dev_priv->dsi_configs[1];
426 break;
427 default:
428 DRM_ERROR("%s, invalid pipe number.\n", __func__);
429 return -EINVAL;
430 }
431
432 /*make sure VGA plane is off. it initializes to on after reset!*/
433 PSB_WVDC32(0x80000000, VGACNTRL);
434
Alan Cox62563042012-05-11 11:30:16 +0100435 if (pipenum == 1) {
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000436 PSB_WVDC32(dpll_val & ~DPLL_VCO_ENABLE, dpll_reg);
437 PSB_RVDC32(dpll_reg);
438
439 PSB_WVDC32(fp_val, fp_reg);
440 } else {
441
442 dpll = PSB_RVDC32(dpll_reg);
443
444 if (!(dpll & DPLL_VCO_ENABLE)) {
445
446 /* When ungating power of DPLL, needs to wait 0.5us
447 before enable the VCO */
448 if (dpll & MDFLD_PWR_GATE_EN) {
449 dpll &= ~MDFLD_PWR_GATE_EN;
450 PSB_WVDC32(dpll, dpll_reg);
451 /* FIXME_MDFLD PO - change 500 to 1 after PO */
452 udelay(500);
453 }
454
455 PSB_WVDC32(fp_val, fp_reg);
456 PSB_WVDC32(dpll_val, dpll_reg);
457 /* FIXME_MDFLD PO - change 500 to 1 after PO */
458 udelay(500);
459
460 dpll_val |= DPLL_VCO_ENABLE;
461 PSB_WVDC32(dpll_val, dpll_reg);
462 PSB_RVDC32(dpll_reg);
463
464 /* wait for DSI PLL to lock */
465 while (timeout < 20000 &&
466 !(PSB_RVDC32(pipeconf_reg) & PIPECONF_DSIPLL_LOCK)) {
467 udelay(150);
468 timeout++;
469 }
470
471 if (timeout == 20000) {
472 DRM_ERROR("%s, can't lock DSIPLL.\n",
473 __func__);
474 return -EINVAL;
475 }
476 }
477 }
478 /* Restore mode */
479 PSB_WVDC32(htot_val, htot_reg);
480 PSB_WVDC32(hblank_val, hblank_reg);
481 PSB_WVDC32(hsync_val, hsync_reg);
482 PSB_WVDC32(vtot_val, vtot_reg);
483 PSB_WVDC32(vblank_val, vblank_reg);
484 PSB_WVDC32(vsync_val, vsync_reg);
485 PSB_WVDC32(pipesrc_val, pipesrc_reg);
486 PSB_WVDC32(dspstatus_val, dspstatus_reg);
487
488 /*set up the plane*/
489 PSB_WVDC32(dspstride_val, dspstride_reg);
490 PSB_WVDC32(dsplinoff_val, dsplinoff_reg);
491 PSB_WVDC32(dsptileoff_val, dsptileoff_reg);
492 PSB_WVDC32(dspsize_val, dspsize_reg);
493 PSB_WVDC32(dsppos_val, dsppos_reg);
494 PSB_WVDC32(dspsurf_val, dspsurf_reg);
495
Alan Cox62563042012-05-11 11:30:16 +0100496 if (pipenum == 1) {
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000497 /* restore palette (gamma) */
498 /*DRM_UDELAY(50000); */
499 for (i = 0; i < 256; i++)
500 PSB_WVDC32(palette_val[i], palette_reg + (i << 2));
501
502 PSB_WVDC32(regs->savePFIT_CONTROL, PFIT_CONTROL);
503 PSB_WVDC32(regs->savePFIT_PGM_RATIOS, PFIT_PGM_RATIOS);
504
505 /*TODO: resume HDMI port */
506
507 /*TODO: resume pipe*/
508
509 /*enable the plane*/
510 PSB_WVDC32(dspcntr_val & ~DISPLAY_PLANE_ENABLE, dspcntr_reg);
511
512 return 0;
513 }
514
515 /*set up pipe related registers*/
516 PSB_WVDC32(mipi_val, mipi_reg);
517
518 /*setup MIPI adapter + MIPI IP registers*/
519 if (dsi_config)
Alan Cox62563042012-05-11 11:30:16 +0100520 mdfld_dsi_controller_init(dsi_config, pipenum);
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000521
522 if (in_atomic() || in_interrupt())
523 mdelay(20);
524 else
525 msleep(20);
526
527 /*enable the plane*/
528 PSB_WVDC32(dspcntr_val, dspcntr_reg);
529
530 if (in_atomic() || in_interrupt())
531 mdelay(20);
532 else
533 msleep(20);
534
535 /* LP Hold Release */
536 temp = REG_READ(mipi_reg);
537 temp |= LP_OUTPUT_HOLD_RELEASE;
538 REG_WRITE(mipi_reg, temp);
539 mdelay(1);
540
541
542 /* Set DSI host to exit from Utra Low Power State */
543 temp = REG_READ(device_ready_reg);
544 temp &= ~ULPS_MASK;
545 temp |= 0x3;
546 temp |= EXIT_ULPS_DEV_READY;
547 REG_WRITE(device_ready_reg, temp);
548 mdelay(1);
549
550 temp = REG_READ(device_ready_reg);
551 temp &= ~ULPS_MASK;
552 temp |= EXITING_ULPS;
553 REG_WRITE(device_ready_reg, temp);
554 mdelay(1);
555
556 /*enable the pipe*/
557 PSB_WVDC32(pipeconf_val, pipeconf_reg);
558
559 /* restore palette (gamma) */
560 /*DRM_UDELAY(50000); */
561 for (i = 0; i < 256; i++)
562 PSB_WVDC32(palette_val[i], palette_reg + (i << 2));
563
564 return 0;
565}
566
567static int mdfld_save_registers(struct drm_device *dev)
568{
569 /* mdfld_save_cursor_overlay_registers(dev); */
570 mdfld_save_display_registers(dev, 0);
571 mdfld_save_display_registers(dev, 2);
572 mdfld_disable_crtc(dev, 0);
573 mdfld_disable_crtc(dev, 2);
574
575 return 0;
576}
577
578static int mdfld_restore_registers(struct drm_device *dev)
579{
580 mdfld_restore_display_registers(dev, 2);
581 mdfld_restore_display_registers(dev, 0);
582 /* mdfld_restore_cursor_overlay_registers(dev); */
583
584 return 0;
585}
586
587static int mdfld_power_down(struct drm_device *dev)
588{
589 /* FIXME */
590 return 0;
591}
592
593static int mdfld_power_up(struct drm_device *dev)
594{
595 /* FIXME */
596 return 0;
597}
598
599const struct psb_ops mdfld_chip_ops = {
600 .name = "mdfld",
601 .accel_2d = 0,
602 .pipes = 3,
603 .crtcs = 3,
Patrik Jakobsson87438492012-04-28 23:20:42 +0200604 .lvds_mask = (1 << 1),
605 .hdmi_mask = (1 << 1),
Kirill A. Shutemov026abc32012-03-08 16:02:20 +0000606 .sgx_offset = MRST_SGX_OFFSET,
607
608 .chip_setup = mid_chip_setup,
609 .crtc_helper = &mdfld_helper_funcs,
610 .crtc_funcs = &psb_intel_crtc_funcs,
611
612 .output_init = mdfld_output_init,
613
614#ifdef CONFIG_BACKLIGHT_CLASS_DEVICE
615 .backlight_init = mdfld_backlight_init,
616#endif
617
618 .save_regs = mdfld_save_registers,
619 .restore_regs = mdfld_restore_registers,
620 .power_down = mdfld_power_down,
621 .power_up = mdfld_power_up,
622};