| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2008 (c) Intel Corporation | 
					
						
							|  |  |  |  *   Jesse Barnes <jbarnes@virtuousgeek.org> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a | 
					
						
							|  |  |  |  * copy of this software and associated documentation files (the | 
					
						
							|  |  |  |  * "Software"), to deal in the Software without restriction, including | 
					
						
							|  |  |  |  * without limitation the rights to use, copy, modify, merge, publish, | 
					
						
							|  |  |  |  * distribute, sub license, and/or sell copies of the Software, and to | 
					
						
							|  |  |  |  * permit persons to whom the Software is furnished to do so, subject to | 
					
						
							|  |  |  |  * the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice (including the | 
					
						
							|  |  |  |  * next paragraph) shall be included in all copies or substantial portions | 
					
						
							|  |  |  |  * of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | 
					
						
							|  |  |  |  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | 
					
						
							|  |  |  |  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. | 
					
						
							|  |  |  |  * IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS BE LIABLE FOR | 
					
						
							|  |  |  |  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, | 
					
						
							|  |  |  |  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE | 
					
						
							|  |  |  |  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-02 18:01:07 +01:00
										 |  |  | #include <drm/drmP.h>
 | 
					
						
							|  |  |  | #include <drm/i915_drm.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:56:30 -08:00
										 |  |  | #include "intel_drv.h"
 | 
					
						
							| 
									
										
										
										
											2012-01-07 23:40:34 -02:00
										 |  |  | #include "i915_reg.h"
 | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static u8 i915_read_indexed(struct drm_device *dev, u16 index_port, u16 data_port, u8 reg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	I915_WRITE8(index_port, reg); | 
					
						
							|  |  |  | 	return I915_READ8(data_port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static u8 i915_read_ar(struct drm_device *dev, u16 st01, u8 reg, u16 palette_enable) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	I915_READ8(st01); | 
					
						
							|  |  |  | 	I915_WRITE8(VGA_AR_INDEX, palette_enable | reg); | 
					
						
							|  |  |  | 	return I915_READ8(VGA_AR_DATA_READ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void i915_write_ar(struct drm_device *dev, u16 st01, u8 reg, u8 val, u16 palette_enable) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	I915_READ8(st01); | 
					
						
							|  |  |  | 	I915_WRITE8(VGA_AR_INDEX, palette_enable | reg); | 
					
						
							|  |  |  | 	I915_WRITE8(VGA_AR_DATA_WRITE, val); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void i915_write_indexed(struct drm_device *dev, u16 index_port, u16 data_port, u8 reg, u8 val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	I915_WRITE8(index_port, reg); | 
					
						
							|  |  |  | 	I915_WRITE8(data_port, val); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void i915_save_vga(struct drm_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	u16 cr_index, cr_data, st01; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 	/* VGA state */ | 
					
						
							|  |  |  | 	dev_priv->regfile.saveVGA0 = I915_READ(VGA0); | 
					
						
							|  |  |  | 	dev_priv->regfile.saveVGA1 = I915_READ(VGA1); | 
					
						
							|  |  |  | 	dev_priv->regfile.saveVGA_PD = I915_READ(VGA_PD); | 
					
						
							| 
									
										
										
										
											2013-01-25 21:44:46 +02:00
										 |  |  | 	dev_priv->regfile.saveVGACNTRL = I915_READ(i915_vgacntrl_reg(dev)); | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	/* VGA color palette registers */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveDACMASK = I915_READ8(VGA_DACMASK); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* MSR bits */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveMSR = I915_READ8(VGA_MSR_READ); | 
					
						
							|  |  |  | 	if (dev_priv->regfile.saveMSR & VGA_MSR_CGA_MODE) { | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 		cr_index = VGA_CR_INDEX_CGA; | 
					
						
							|  |  |  | 		cr_data = VGA_CR_DATA_CGA; | 
					
						
							|  |  |  | 		st01 = VGA_ST01_CGA; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		cr_index = VGA_CR_INDEX_MDA; | 
					
						
							|  |  |  | 		cr_data = VGA_CR_DATA_MDA; | 
					
						
							|  |  |  | 		st01 = VGA_ST01_MDA; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* CRT controller regs */ | 
					
						
							|  |  |  | 	i915_write_indexed(dev, cr_index, cr_data, 0x11, | 
					
						
							|  |  |  | 			   i915_read_indexed(dev, cr_index, cr_data, 0x11) & | 
					
						
							|  |  |  | 			   (~0x80)); | 
					
						
							|  |  |  | 	for (i = 0; i <= 0x24; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.saveCR[i] = | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 			i915_read_indexed(dev, cr_index, cr_data, i); | 
					
						
							|  |  |  | 	/* Make sure we don't turn off CR group 0 writes */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveCR[0x11] &= ~0x80; | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Attribute controller registers */ | 
					
						
							|  |  |  | 	I915_READ8(st01); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveAR_INDEX = I915_READ8(VGA_AR_INDEX); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	for (i = 0; i <= 0x14; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.saveAR[i] = i915_read_ar(dev, st01, i, 0); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	I915_READ8(st01); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	I915_WRITE8(VGA_AR_INDEX, dev_priv->regfile.saveAR_INDEX); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	I915_READ8(st01); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Graphics controller registers */ | 
					
						
							|  |  |  | 	for (i = 0; i < 9; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.saveGR[i] = | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 			i915_read_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, i); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveGR[0x10] = | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 		i915_read_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, 0x10); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveGR[0x11] = | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 		i915_read_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, 0x11); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveGR[0x18] = | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 		i915_read_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, 0x18); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Sequencer registers */ | 
					
						
							|  |  |  | 	for (i = 0; i < 8; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.saveSR[i] = | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 			i915_read_indexed(dev, VGA_SR_INDEX, VGA_SR_DATA, i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void i915_restore_vga(struct drm_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	u16 cr_index, cr_data, st01; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 	/* VGA state */ | 
					
						
							| 
									
										
										
										
											2013-01-25 21:44:46 +02:00
										 |  |  | 	I915_WRITE(i915_vgacntrl_reg(dev), dev_priv->regfile.saveVGACNTRL); | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	I915_WRITE(VGA0, dev_priv->regfile.saveVGA0); | 
					
						
							|  |  |  | 	I915_WRITE(VGA1, dev_priv->regfile.saveVGA1); | 
					
						
							|  |  |  | 	I915_WRITE(VGA_PD, dev_priv->regfile.saveVGA_PD); | 
					
						
							|  |  |  | 	POSTING_READ(VGA_PD); | 
					
						
							|  |  |  | 	udelay(150); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	/* MSR bits */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	I915_WRITE8(VGA_MSR_WRITE, dev_priv->regfile.saveMSR); | 
					
						
							|  |  |  | 	if (dev_priv->regfile.saveMSR & VGA_MSR_CGA_MODE) { | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 		cr_index = VGA_CR_INDEX_CGA; | 
					
						
							|  |  |  | 		cr_data = VGA_CR_DATA_CGA; | 
					
						
							|  |  |  | 		st01 = VGA_ST01_CGA; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		cr_index = VGA_CR_INDEX_MDA; | 
					
						
							|  |  |  | 		cr_data = VGA_CR_DATA_MDA; | 
					
						
							|  |  |  | 		st01 = VGA_ST01_MDA; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Sequencer registers, don't write SR07 */ | 
					
						
							|  |  |  | 	for (i = 0; i < 7; i++) | 
					
						
							|  |  |  | 		i915_write_indexed(dev, VGA_SR_INDEX, VGA_SR_DATA, i, | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 				   dev_priv->regfile.saveSR[i]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* CRT controller regs */ | 
					
						
							|  |  |  | 	/* Enable CR group 0 writes */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	i915_write_indexed(dev, cr_index, cr_data, 0x11, dev_priv->regfile.saveCR[0x11]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	for (i = 0; i <= 0x24; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		i915_write_indexed(dev, cr_index, cr_data, i, dev_priv->regfile.saveCR[i]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Graphics controller regs */ | 
					
						
							|  |  |  | 	for (i = 0; i < 9; i++) | 
					
						
							|  |  |  | 		i915_write_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, i, | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 				   dev_priv->regfile.saveGR[i]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	i915_write_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, 0x10, | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			   dev_priv->regfile.saveGR[0x10]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	i915_write_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, 0x11, | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			   dev_priv->regfile.saveGR[0x11]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	i915_write_indexed(dev, VGA_GR_INDEX, VGA_GR_DATA, 0x18, | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			   dev_priv->regfile.saveGR[0x18]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Attribute controller registers */ | 
					
						
							|  |  |  | 	I915_READ8(st01); /* switch back to index mode */ | 
					
						
							|  |  |  | 	for (i = 0; i <= 0x14; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		i915_write_ar(dev, st01, i, dev_priv->regfile.saveAR[i], 0); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	I915_READ8(st01); /* switch back to index mode */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	I915_WRITE8(VGA_AR_INDEX, dev_priv->regfile.saveAR_INDEX | 0x20); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	I915_READ8(st01); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* VGA color palette registers */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	I915_WRITE8(VGA_DACMASK, dev_priv->regfile.saveDACMASK); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | static void i915_save_display(struct drm_device *dev) | 
					
						
							| 
									
										
										
										
											2009-07-08 14:13:14 +08:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Display arbitration control */ | 
					
						
							| 
									
										
										
										
											2013-01-18 18:29:03 -02:00
										 |  |  | 	if (INTEL_INFO(dev)->gen <= 4) | 
					
						
							|  |  |  | 		dev_priv->regfile.saveDSPARB = I915_READ(DSPARB); | 
					
						
							| 
									
										
										
										
											2009-07-08 14:13:14 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* This is only meaningful in non-KMS mode */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	/* Don't regfile.save them in KMS mode */ | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:19 +01:00
										 |  |  | 	if (!drm_core_check_feature(dev, DRIVER_MODESET)) | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:20 +01:00
										 |  |  | 		i915_save_display_reg(dev); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	/* LVDS state */ | 
					
						
							| 
									
										
										
										
											2014-11-12 17:01:10 +02:00
										 |  |  | 	if (HAS_PCH_IBX(dev) || HAS_PCH_CPT(dev)) | 
					
						
							|  |  |  | 		dev_priv->regfile.saveLVDS = I915_READ(PCH_LVDS); | 
					
						
							|  |  |  | 	else if (INTEL_INFO(dev)->gen <= 4 && IS_MOBILE(dev) && !IS_I830(dev)) | 
					
						
							|  |  |  | 		dev_priv->regfile.saveLVDS = I915_READ(LVDS); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-12 17:01:10 +02:00
										 |  |  | 	/* Panel power sequencer */ | 
					
						
							| 
									
										
										
										
											2010-08-14 14:41:23 +01:00
										 |  |  | 	if (HAS_PCH_SPLIT(dev)) { | 
					
						
							| 
									
										
										
										
											2014-11-12 17:01:10 +02:00
										 |  |  | 		dev_priv->regfile.savePP_CONTROL = I915_READ(PCH_PP_CONTROL); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.savePP_ON_DELAYS = I915_READ(PCH_PP_ON_DELAYS); | 
					
						
							|  |  |  | 		dev_priv->regfile.savePP_OFF_DELAYS = I915_READ(PCH_PP_OFF_DELAYS); | 
					
						
							|  |  |  | 		dev_priv->regfile.savePP_DIVISOR = I915_READ(PCH_PP_DIVISOR); | 
					
						
							| 
									
										
										
										
											2014-11-11 16:48:03 +02:00
										 |  |  | 	} else if (!IS_VALLEYVIEW(dev)) { | 
					
						
							| 
									
										
										
										
											2014-11-12 17:01:10 +02:00
										 |  |  | 		dev_priv->regfile.savePP_CONTROL = I915_READ(PP_CONTROL); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.savePP_ON_DELAYS = I915_READ(PP_ON_DELAYS); | 
					
						
							|  |  |  | 		dev_priv->regfile.savePP_OFF_DELAYS = I915_READ(PP_OFF_DELAYS); | 
					
						
							|  |  |  | 		dev_priv->regfile.savePP_DIVISOR = I915_READ(PP_DIVISOR); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-23 16:49:15 +02:00
										 |  |  | 	/* save FBC interval */ | 
					
						
							|  |  |  | 	if (HAS_FBC(dev) && INTEL_INFO(dev)->gen <= 4 && !IS_G4X(dev)) | 
					
						
							|  |  |  | 		dev_priv->regfile.saveFBC_CONTROL = I915_READ(FBC_CONTROL); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 	if (!drm_core_check_feature(dev, DRIVER_MODESET)) | 
					
						
							|  |  |  | 		i915_save_vga(dev); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | static void i915_restore_display(struct drm_device *dev) | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							| 
									
										
										
										
											2013-02-19 12:11:38 -08:00
										 |  |  | 	u32 mask = 0xffffffff; | 
					
						
							| 
									
										
										
										
											2008-11-18 12:39:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-02 23:08:44 -08:00
										 |  |  | 	/* Display arbitration */ | 
					
						
							| 
									
										
										
										
											2013-01-18 18:29:03 -02:00
										 |  |  | 	if (INTEL_INFO(dev)->gen <= 4) | 
					
						
							|  |  |  | 		I915_WRITE(DSPARB, dev_priv->regfile.saveDSPARB); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:19 +01:00
										 |  |  | 	if (!drm_core_check_feature(dev, DRIVER_MODESET)) | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:20 +01:00
										 |  |  | 		i915_restore_display_reg(dev); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-19 12:11:38 -08:00
										 |  |  | 	if (drm_core_check_feature(dev, DRIVER_MODESET)) | 
					
						
							|  |  |  | 		mask = ~LVDS_PORT_EN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-12 17:01:10 +02:00
										 |  |  | 	/* LVDS state */ | 
					
						
							| 
									
										
										
										
											2013-03-06 20:03:20 -03:00
										 |  |  | 	if (HAS_PCH_IBX(dev) || HAS_PCH_CPT(dev)) | 
					
						
							| 
									
										
										
										
											2013-02-19 12:11:38 -08:00
										 |  |  | 		I915_WRITE(PCH_LVDS, dev_priv->regfile.saveLVDS & mask); | 
					
						
							| 
									
										
										
										
											2013-03-06 20:03:20 -03:00
										 |  |  | 	else if (INTEL_INFO(dev)->gen <= 4 && IS_MOBILE(dev) && !IS_I830(dev)) | 
					
						
							| 
									
										
										
										
											2013-02-19 12:11:38 -08:00
										 |  |  | 		I915_WRITE(LVDS, dev_priv->regfile.saveLVDS & mask); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-12 17:01:10 +02:00
										 |  |  | 	/* Panel power sequencer */ | 
					
						
							| 
									
										
										
										
											2010-08-14 14:41:23 +01:00
										 |  |  | 	if (HAS_PCH_SPLIT(dev)) { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		I915_WRITE(PCH_PP_ON_DELAYS, dev_priv->regfile.savePP_ON_DELAYS); | 
					
						
							|  |  |  | 		I915_WRITE(PCH_PP_OFF_DELAYS, dev_priv->regfile.savePP_OFF_DELAYS); | 
					
						
							|  |  |  | 		I915_WRITE(PCH_PP_DIVISOR, dev_priv->regfile.savePP_DIVISOR); | 
					
						
							|  |  |  | 		I915_WRITE(PCH_PP_CONTROL, dev_priv->regfile.savePP_CONTROL); | 
					
						
							| 
									
										
										
										
											2014-11-12 16:25:43 +02:00
										 |  |  | 	} else if (!IS_VALLEYVIEW(dev)) { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		I915_WRITE(PP_ON_DELAYS, dev_priv->regfile.savePP_ON_DELAYS); | 
					
						
							|  |  |  | 		I915_WRITE(PP_OFF_DELAYS, dev_priv->regfile.savePP_OFF_DELAYS); | 
					
						
							|  |  |  | 		I915_WRITE(PP_DIVISOR, dev_priv->regfile.savePP_DIVISOR); | 
					
						
							|  |  |  | 		I915_WRITE(PP_CONTROL, dev_priv->regfile.savePP_CONTROL); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 17:05:10 +08:00
										 |  |  | 	/* only restore FBC info on the platform that supports FBC*/ | 
					
						
							| 
									
										
										
										
											2014-12-08 14:09:10 -02:00
										 |  |  | 	intel_fbc_disable(dev); | 
					
						
							| 
									
										
										
										
											2014-01-23 16:49:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* restore FBC interval */ | 
					
						
							|  |  |  | 	if (HAS_FBC(dev) && INTEL_INFO(dev)->gen <= 4 && !IS_G4X(dev)) | 
					
						
							|  |  |  | 		I915_WRITE(FBC_CONTROL, dev_priv->regfile.saveFBC_CONTROL); | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 	if (!drm_core_check_feature(dev, DRIVER_MODESET)) | 
					
						
							|  |  |  | 		i915_restore_vga(dev); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2013-01-25 17:53:21 +01:00
										 |  |  | 		i915_redisable_vga(dev); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int i915_save_state(struct drm_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | 	mutex_lock(&dev->struct_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 	i915_save_display(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 	if (!drm_core_check_feature(dev, DRIVER_MODESET)) { | 
					
						
							|  |  |  | 		/* Interrupt state */ | 
					
						
							|  |  |  | 		if (HAS_PCH_SPLIT(dev)) { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			dev_priv->regfile.saveDEIER = I915_READ(DEIER); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveDEIMR = I915_READ(DEIMR); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveGTIER = I915_READ(GTIER); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveGTIMR = I915_READ(GTIMR); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveFDI_RXA_IMR = I915_READ(_FDI_RXA_IMR); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveFDI_RXB_IMR = I915_READ(_FDI_RXB_IMR); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveMCHBAR_RENDER_STANDBY = | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 				I915_READ(RSTDBYCTL); | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			dev_priv->regfile.savePCH_PORT_HOTPLUG = I915_READ(PCH_PORT_HOTPLUG); | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			dev_priv->regfile.saveIER = I915_READ(IER); | 
					
						
							|  |  |  | 			dev_priv->regfile.saveIMR = I915_READ(IMR); | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-10 12:16:05 -08:00
										 |  |  | 	if (IS_GEN4(dev)) | 
					
						
							|  |  |  | 		pci_read_config_word(dev->pdev, GCDGMBUS, | 
					
						
							|  |  |  | 				     &dev_priv->regfile.saveGCDGMBUS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 	/* Cache mode state */ | 
					
						
							| 
									
										
										
										
											2013-10-11 12:09:29 -07:00
										 |  |  | 	if (INTEL_INFO(dev)->gen < 7) | 
					
						
							|  |  |  | 		dev_priv->regfile.saveCACHE_MODE_0 = I915_READ(CACHE_MODE_0); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Memory Arbitration state */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	dev_priv->regfile.saveMI_ARB_STATE = I915_READ(MI_ARB_STATE); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Scratch space */ | 
					
						
							|  |  |  | 	for (i = 0; i < 16; i++) { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.saveSWF0[i] = I915_READ(SWF00 + (i << 2)); | 
					
						
							|  |  |  | 		dev_priv->regfile.saveSWF1[i] = I915_READ(SWF10 + (i << 2)); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	for (i = 0; i < 3; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		dev_priv->regfile.saveSWF2[i] = I915_READ(SWF30 + (i << 2)); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | 	mutex_unlock(&dev->struct_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int i915_restore_state(struct drm_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_i915_private *dev_priv = dev->dev_private; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | 	mutex_lock(&dev->struct_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-12 10:15:12 +01:00
										 |  |  | 	i915_gem_restore_fences(dev); | 
					
						
							| 
									
										
										
										
											2014-12-10 12:16:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (IS_GEN4(dev)) | 
					
						
							|  |  |  | 		pci_write_config_word(dev->pdev, GCDGMBUS, | 
					
						
							|  |  |  | 				      dev_priv->regfile.saveGCDGMBUS); | 
					
						
							| 
									
										
										
										
											2009-09-14 17:48:42 -04:00
										 |  |  | 	i915_restore_display(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 	if (!drm_core_check_feature(dev, DRIVER_MODESET)) { | 
					
						
							|  |  |  | 		/* Interrupt state */ | 
					
						
							|  |  |  | 		if (HAS_PCH_SPLIT(dev)) { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			I915_WRITE(DEIER, dev_priv->regfile.saveDEIER); | 
					
						
							|  |  |  | 			I915_WRITE(DEIMR, dev_priv->regfile.saveDEIMR); | 
					
						
							|  |  |  | 			I915_WRITE(GTIER, dev_priv->regfile.saveGTIER); | 
					
						
							|  |  |  | 			I915_WRITE(GTIMR, dev_priv->regfile.saveGTIMR); | 
					
						
							|  |  |  | 			I915_WRITE(_FDI_RXA_IMR, dev_priv->regfile.saveFDI_RXA_IMR); | 
					
						
							|  |  |  | 			I915_WRITE(_FDI_RXB_IMR, dev_priv->regfile.saveFDI_RXB_IMR); | 
					
						
							|  |  |  | 			I915_WRITE(PCH_PORT_HOTPLUG, dev_priv->regfile.savePCH_PORT_HOTPLUG); | 
					
						
							| 
									
										
										
										
											2014-11-11 16:48:04 +02:00
										 |  |  | 			I915_WRITE(RSTDBYCTL, | 
					
						
							|  |  |  | 				   dev_priv->regfile.saveMCHBAR_RENDER_STANDBY); | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 			I915_WRITE(IER, dev_priv->regfile.saveIER); | 
					
						
							|  |  |  | 			I915_WRITE(IMR, dev_priv->regfile.saveIMR); | 
					
						
							| 
									
										
										
										
											2012-10-17 11:32:56 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-10-21 15:27:01 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	/* Cache mode state */ | 
					
						
							| 
									
										
										
										
											2013-10-11 12:09:29 -07:00
										 |  |  | 	if (INTEL_INFO(dev)->gen < 7) | 
					
						
							|  |  |  | 		I915_WRITE(CACHE_MODE_0, dev_priv->regfile.saveCACHE_MODE_0 | | 
					
						
							|  |  |  | 			   0xffff0000); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Memory arbitration state */ | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 	I915_WRITE(MI_ARB_STATE, dev_priv->regfile.saveMI_ARB_STATE | 0xffff0000); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < 16; i++) { | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		I915_WRITE(SWF00 + (i << 2), dev_priv->regfile.saveSWF0[i]); | 
					
						
							|  |  |  | 		I915_WRITE(SWF10 + (i << 2), dev_priv->regfile.saveSWF1[i]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	for (i = 0; i < 3; i++) | 
					
						
							| 
									
										
										
										
											2012-11-02 19:55:02 +01:00
										 |  |  | 		I915_WRITE(SWF30 + (i << 2), dev_priv->regfile.saveSWF2[i]); | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-29 00:30:34 -07:00
										 |  |  | 	mutex_unlock(&dev->struct_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-20 15:44:45 -07:00
										 |  |  | 	intel_i2c_reset(dev); | 
					
						
							| 
									
										
										
										
											2009-12-01 11:56:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-25 15:11:06 -07:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } |