 b07aaf8849
			
		
	
	
	b07aaf8849
	
	
	
		
			
			The vbt on my Fujitsu-Siemens Lifebook S6010 provides two 800x600 modes, 60Hz and 56Hz. The magic register values we have correspond to the 60Hz mode, and as I don't know how one would trick the VGA BIOS to set up the 56Hz mode we can't get the magic values for the orther mode. So when checking whether a mode is valid also check the pixel clock so that we filter out the 56Hz variant. Signed-off-by: Ville Syrjälä <ville.syrjala@linux.intel.com> Reviewed-by: Thomas Richter <richter@rus.uni-stuttgart.de> Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
		
			
				
	
	
		
			631 lines
		
	
	
	
		
			21 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			631 lines
		
	
	
	
		
			21 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *
 | |
|  * Copyright (c) 2012 Gilles Dartiguelongue, Thomas Richter
 | |
|  *
 | |
|  * All Rights Reserved.
 | |
|  *
 | |
|  * 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 THE AUTHOR 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.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include "dvo.h"
 | |
| #include "i915_reg.h"
 | |
| #include "i915_drv.h"
 | |
| 
 | |
| #define NS2501_VID 0x1305
 | |
| #define NS2501_DID 0x6726
 | |
| 
 | |
| #define NS2501_VID_LO 0x00
 | |
| #define NS2501_VID_HI 0x01
 | |
| #define NS2501_DID_LO 0x02
 | |
| #define NS2501_DID_HI 0x03
 | |
| #define NS2501_REV 0x04
 | |
| #define NS2501_RSVD 0x05
 | |
| #define NS2501_FREQ_LO 0x06
 | |
| #define NS2501_FREQ_HI 0x07
 | |
| 
 | |
| #define NS2501_REG8 0x08
 | |
| #define NS2501_8_VEN (1<<5)
 | |
| #define NS2501_8_HEN (1<<4)
 | |
| #define NS2501_8_DSEL (1<<3)
 | |
| #define NS2501_8_BPAS (1<<2)
 | |
| #define NS2501_8_RSVD (1<<1)
 | |
| #define NS2501_8_PD (1<<0)
 | |
| 
 | |
| #define NS2501_REG9 0x09
 | |
| #define NS2501_9_VLOW (1<<7)
 | |
| #define NS2501_9_MSEL_MASK (0x7<<4)
 | |
| #define NS2501_9_TSEL (1<<3)
 | |
| #define NS2501_9_RSEN (1<<2)
 | |
| #define NS2501_9_RSVD (1<<1)
 | |
| #define NS2501_9_MDI (1<<0)
 | |
| 
 | |
| #define NS2501_REGC 0x0c
 | |
| 
 | |
| enum {
 | |
| 	MODE_640x480,
 | |
| 	MODE_800x600,
 | |
| 	MODE_1024x768,
 | |
| };
 | |
| 
 | |
| struct ns2501_reg {
 | |
| 	 uint8_t offset;
 | |
| 	 uint8_t value;
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * Magic values based on what the BIOS on
 | |
|  * Fujitsu-Siemens Lifebook S6010 programs (1024x768 panel).
 | |
|  */
 | |
| static const struct ns2501_reg regs_1024x768[][86] = {
 | |
| 	[MODE_640x480] = {
 | |
| 		[0] = { .offset = 0x0a, .value = 0x81, },
 | |
| 		[1] = { .offset = 0x18, .value = 0x07, },
 | |
| 		[2] = { .offset = 0x19, .value = 0x00, },
 | |
| 		[3] = { .offset = 0x1a, .value = 0x00, },
 | |
| 		[4] = { .offset = 0x1b, .value = 0x11, },
 | |
| 		[5] = { .offset = 0x1c, .value = 0x54, },
 | |
| 		[6] = { .offset = 0x1d, .value = 0x03, },
 | |
| 		[7] = { .offset = 0x1e, .value = 0x02, },
 | |
| 		[8] = { .offset = 0xf3, .value = 0x90, },
 | |
| 		[9] = { .offset = 0xf9, .value = 0x00, },
 | |
| 		[10] = { .offset = 0xc1, .value = 0x90, },
 | |
| 		[11] = { .offset = 0xc2, .value = 0x00, },
 | |
| 		[12] = { .offset = 0xc3, .value = 0x0f, },
 | |
| 		[13] = { .offset = 0xc4, .value = 0x03, },
 | |
| 		[14] = { .offset = 0xc5, .value = 0x16, },
 | |
| 		[15] = { .offset = 0xc6, .value = 0x00, },
 | |
| 		[16] = { .offset = 0xc7, .value = 0x02, },
 | |
| 		[17] = { .offset = 0xc8, .value = 0x02, },
 | |
| 		[18] = { .offset = 0xf4, .value = 0x00, },
 | |
| 		[19] = { .offset = 0x80, .value = 0xff, },
 | |
| 		[20] = { .offset = 0x81, .value = 0x07, },
 | |
| 		[21] = { .offset = 0x82, .value = 0x3d, },
 | |
| 		[22] = { .offset = 0x83, .value = 0x05, },
 | |
| 		[23] = { .offset = 0x94, .value = 0x00, },
 | |
| 		[24] = { .offset = 0x95, .value = 0x00, },
 | |
| 		[25] = { .offset = 0x96, .value = 0x05, },
 | |
| 		[26] = { .offset = 0x97, .value = 0x00, },
 | |
| 		[27] = { .offset = 0x9a, .value = 0x88, },
 | |
| 		[28] = { .offset = 0x9b, .value = 0x00, },
 | |
| 		[29] = { .offset = 0x98, .value = 0x00, },
 | |
| 		[30] = { .offset = 0x99, .value = 0x00, },
 | |
| 		[31] = { .offset = 0xf7, .value = 0x88, },
 | |
| 		[32] = { .offset = 0xf8, .value = 0x0a, },
 | |
| 		[33] = { .offset = 0x9c, .value = 0x24, },
 | |
| 		[34] = { .offset = 0x9d, .value = 0x00, },
 | |
| 		[35] = { .offset = 0x9e, .value = 0x25, },
 | |
| 		[36] = { .offset = 0x9f, .value = 0x03, },
 | |
| 		[37] = { .offset = 0xa0, .value = 0x28, },
 | |
| 		[38] = { .offset = 0xa1, .value = 0x01, },
 | |
| 		[39] = { .offset = 0xa2, .value = 0x28, },
 | |
| 		[40] = { .offset = 0xa3, .value = 0x05, },
 | |
| 		[41] = { .offset = 0xb6, .value = 0x09, },
 | |
| 		[42] = { .offset = 0xb8, .value = 0x00, },
 | |
| 		[43] = { .offset = 0xb9, .value = 0xa0, },
 | |
| 		[44] = { .offset = 0xba, .value = 0x00, },
 | |
| 		[45] = { .offset = 0xbb, .value = 0x20, },
 | |
| 		[46] = { .offset = 0x10, .value = 0x00, },
 | |
| 		[47] = { .offset = 0x11, .value = 0xa0, },
 | |
| 		[48] = { .offset = 0x12, .value = 0x02, },
 | |
| 		[49] = { .offset = 0x20, .value = 0x00, },
 | |
| 		[50] = { .offset = 0x22, .value = 0x00, },
 | |
| 		[51] = { .offset = 0x23, .value = 0x00, },
 | |
| 		[52] = { .offset = 0x24, .value = 0x00, },
 | |
| 		[53] = { .offset = 0x25, .value = 0x00, },
 | |
| 		[54] = { .offset = 0x8c, .value = 0x10, },
 | |
| 		[55] = { .offset = 0x8d, .value = 0x02, },
 | |
| 		[56] = { .offset = 0x8e, .value = 0x10, },
 | |
| 		[57] = { .offset = 0x8f, .value = 0x00, },
 | |
| 		[58] = { .offset = 0x90, .value = 0xff, },
 | |
| 		[59] = { .offset = 0x91, .value = 0x07, },
 | |
| 		[60] = { .offset = 0x92, .value = 0xa0, },
 | |
| 		[61] = { .offset = 0x93, .value = 0x02, },
 | |
| 		[62] = { .offset = 0xa5, .value = 0x00, },
 | |
| 		[63] = { .offset = 0xa6, .value = 0x00, },
 | |
| 		[64] = { .offset = 0xa7, .value = 0x00, },
 | |
| 		[65] = { .offset = 0xa8, .value = 0x00, },
 | |
| 		[66] = { .offset = 0xa9, .value = 0x04, },
 | |
| 		[67] = { .offset = 0xaa, .value = 0x70, },
 | |
| 		[68] = { .offset = 0xab, .value = 0x4f, },
 | |
| 		[69] = { .offset = 0xac, .value = 0x00, },
 | |
| 		[70] = { .offset = 0xa4, .value = 0x84, },
 | |
| 		[71] = { .offset = 0x7e, .value = 0x18, },
 | |
| 		[72] = { .offset = 0x84, .value = 0x00, },
 | |
| 		[73] = { .offset = 0x85, .value = 0x00, },
 | |
| 		[74] = { .offset = 0x86, .value = 0x00, },
 | |
| 		[75] = { .offset = 0x87, .value = 0x00, },
 | |
| 		[76] = { .offset = 0x88, .value = 0x00, },
 | |
| 		[77] = { .offset = 0x89, .value = 0x00, },
 | |
| 		[78] = { .offset = 0x8a, .value = 0x00, },
 | |
| 		[79] = { .offset = 0x8b, .value = 0x00, },
 | |
| 		[80] = { .offset = 0x26, .value = 0x00, },
 | |
| 		[81] = { .offset = 0x27, .value = 0x00, },
 | |
| 		[82] = { .offset = 0xad, .value = 0x00, },
 | |
| 		[83] = { .offset = 0x08, .value = 0x30, }, /* 0x31 */
 | |
| 		[84] = { .offset = 0x41, .value = 0x00, },
 | |
| 		[85] = { .offset = 0xc0, .value = 0x05, },
 | |
| 	},
 | |
| 	[MODE_800x600] = {
 | |
| 		[0] = { .offset = 0x0a, .value = 0x81, },
 | |
| 		[1] = { .offset = 0x18, .value = 0x07, },
 | |
| 		[2] = { .offset = 0x19, .value = 0x00, },
 | |
| 		[3] = { .offset = 0x1a, .value = 0x00, },
 | |
| 		[4] = { .offset = 0x1b, .value = 0x19, },
 | |
| 		[5] = { .offset = 0x1c, .value = 0x64, },
 | |
| 		[6] = { .offset = 0x1d, .value = 0x02, },
 | |
| 		[7] = { .offset = 0x1e, .value = 0x02, },
 | |
| 		[8] = { .offset = 0xf3, .value = 0x90, },
 | |
| 		[9] = { .offset = 0xf9, .value = 0x00, },
 | |
| 		[10] = { .offset = 0xc1, .value = 0xd7, },
 | |
| 		[11] = { .offset = 0xc2, .value = 0x00, },
 | |
| 		[12] = { .offset = 0xc3, .value = 0xf8, },
 | |
| 		[13] = { .offset = 0xc4, .value = 0x03, },
 | |
| 		[14] = { .offset = 0xc5, .value = 0x1a, },
 | |
| 		[15] = { .offset = 0xc6, .value = 0x00, },
 | |
| 		[16] = { .offset = 0xc7, .value = 0x73, },
 | |
| 		[17] = { .offset = 0xc8, .value = 0x02, },
 | |
| 		[18] = { .offset = 0xf4, .value = 0x00, },
 | |
| 		[19] = { .offset = 0x80, .value = 0x27, },
 | |
| 		[20] = { .offset = 0x81, .value = 0x03, },
 | |
| 		[21] = { .offset = 0x82, .value = 0x41, },
 | |
| 		[22] = { .offset = 0x83, .value = 0x05, },
 | |
| 		[23] = { .offset = 0x94, .value = 0x00, },
 | |
| 		[24] = { .offset = 0x95, .value = 0x00, },
 | |
| 		[25] = { .offset = 0x96, .value = 0x05, },
 | |
| 		[26] = { .offset = 0x97, .value = 0x00, },
 | |
| 		[27] = { .offset = 0x9a, .value = 0x88, },
 | |
| 		[28] = { .offset = 0x9b, .value = 0x00, },
 | |
| 		[29] = { .offset = 0x98, .value = 0x00, },
 | |
| 		[30] = { .offset = 0x99, .value = 0x00, },
 | |
| 		[31] = { .offset = 0xf7, .value = 0x88, },
 | |
| 		[32] = { .offset = 0xf8, .value = 0x06, },
 | |
| 		[33] = { .offset = 0x9c, .value = 0x23, },
 | |
| 		[34] = { .offset = 0x9d, .value = 0x00, },
 | |
| 		[35] = { .offset = 0x9e, .value = 0x25, },
 | |
| 		[36] = { .offset = 0x9f, .value = 0x03, },
 | |
| 		[37] = { .offset = 0xa0, .value = 0x28, },
 | |
| 		[38] = { .offset = 0xa1, .value = 0x01, },
 | |
| 		[39] = { .offset = 0xa2, .value = 0x28, },
 | |
| 		[40] = { .offset = 0xa3, .value = 0x05, },
 | |
| 		[41] = { .offset = 0xb6, .value = 0x09, },
 | |
| 		[42] = { .offset = 0xb8, .value = 0x30, },
 | |
| 		[43] = { .offset = 0xb9, .value = 0xc8, },
 | |
| 		[44] = { .offset = 0xba, .value = 0x00, },
 | |
| 		[45] = { .offset = 0xbb, .value = 0x20, },
 | |
| 		[46] = { .offset = 0x10, .value = 0x20, },
 | |
| 		[47] = { .offset = 0x11, .value = 0xc8, },
 | |
| 		[48] = { .offset = 0x12, .value = 0x02, },
 | |
| 		[49] = { .offset = 0x20, .value = 0x00, },
 | |
| 		[50] = { .offset = 0x22, .value = 0x00, },
 | |
| 		[51] = { .offset = 0x23, .value = 0x00, },
 | |
| 		[52] = { .offset = 0x24, .value = 0x00, },
 | |
| 		[53] = { .offset = 0x25, .value = 0x00, },
 | |
| 		[54] = { .offset = 0x8c, .value = 0x10, },
 | |
| 		[55] = { .offset = 0x8d, .value = 0x02, },
 | |
| 		[56] = { .offset = 0x8e, .value = 0x04, },
 | |
| 		[57] = { .offset = 0x8f, .value = 0x00, },
 | |
| 		[58] = { .offset = 0x90, .value = 0xff, },
 | |
| 		[59] = { .offset = 0x91, .value = 0x07, },
 | |
| 		[60] = { .offset = 0x92, .value = 0xa0, },
 | |
| 		[61] = { .offset = 0x93, .value = 0x02, },
 | |
| 		[62] = { .offset = 0xa5, .value = 0x00, },
 | |
| 		[63] = { .offset = 0xa6, .value = 0x00, },
 | |
| 		[64] = { .offset = 0xa7, .value = 0x00, },
 | |
| 		[65] = { .offset = 0xa8, .value = 0x00, },
 | |
| 		[66] = { .offset = 0xa9, .value = 0x83, },
 | |
| 		[67] = { .offset = 0xaa, .value = 0x40, },
 | |
| 		[68] = { .offset = 0xab, .value = 0x32, },
 | |
| 		[69] = { .offset = 0xac, .value = 0x00, },
 | |
| 		[70] = { .offset = 0xa4, .value = 0x80, },
 | |
| 		[71] = { .offset = 0x7e, .value = 0x18, },
 | |
| 		[72] = { .offset = 0x84, .value = 0x00, },
 | |
| 		[73] = { .offset = 0x85, .value = 0x00, },
 | |
| 		[74] = { .offset = 0x86, .value = 0x00, },
 | |
| 		[75] = { .offset = 0x87, .value = 0x00, },
 | |
| 		[76] = { .offset = 0x88, .value = 0x00, },
 | |
| 		[77] = { .offset = 0x89, .value = 0x00, },
 | |
| 		[78] = { .offset = 0x8a, .value = 0x00, },
 | |
| 		[79] = { .offset = 0x8b, .value = 0x00, },
 | |
| 		[80] = { .offset = 0x26, .value = 0x00, },
 | |
| 		[81] = { .offset = 0x27, .value = 0x00, },
 | |
| 		[82] = { .offset = 0xad, .value = 0x00, },
 | |
| 		[83] = { .offset = 0x08, .value = 0x30, }, /* 0x31 */
 | |
| 		[84] = { .offset = 0x41, .value = 0x00, },
 | |
| 		[85] = { .offset = 0xc0, .value = 0x07, },
 | |
| 	},
 | |
| 	[MODE_1024x768] = {
 | |
| 		[0] = { .offset = 0x0a, .value = 0x81, },
 | |
| 		[1] = { .offset = 0x18, .value = 0x07, },
 | |
| 		[2] = { .offset = 0x19, .value = 0x00, },
 | |
| 		[3] = { .offset = 0x1a, .value = 0x00, },
 | |
| 		[4] = { .offset = 0x1b, .value = 0x11, },
 | |
| 		[5] = { .offset = 0x1c, .value = 0x54, },
 | |
| 		[6] = { .offset = 0x1d, .value = 0x03, },
 | |
| 		[7] = { .offset = 0x1e, .value = 0x02, },
 | |
| 		[8] = { .offset = 0xf3, .value = 0x90, },
 | |
| 		[9] = { .offset = 0xf9, .value = 0x00, },
 | |
| 		[10] = { .offset = 0xc1, .value = 0x90, },
 | |
| 		[11] = { .offset = 0xc2, .value = 0x00, },
 | |
| 		[12] = { .offset = 0xc3, .value = 0x0f, },
 | |
| 		[13] = { .offset = 0xc4, .value = 0x03, },
 | |
| 		[14] = { .offset = 0xc5, .value = 0x16, },
 | |
| 		[15] = { .offset = 0xc6, .value = 0x00, },
 | |
| 		[16] = { .offset = 0xc7, .value = 0x02, },
 | |
| 		[17] = { .offset = 0xc8, .value = 0x02, },
 | |
| 		[18] = { .offset = 0xf4, .value = 0x00, },
 | |
| 		[19] = { .offset = 0x80, .value = 0xff, },
 | |
| 		[20] = { .offset = 0x81, .value = 0x07, },
 | |
| 		[21] = { .offset = 0x82, .value = 0x3d, },
 | |
| 		[22] = { .offset = 0x83, .value = 0x05, },
 | |
| 		[23] = { .offset = 0x94, .value = 0x00, },
 | |
| 		[24] = { .offset = 0x95, .value = 0x00, },
 | |
| 		[25] = { .offset = 0x96, .value = 0x05, },
 | |
| 		[26] = { .offset = 0x97, .value = 0x00, },
 | |
| 		[27] = { .offset = 0x9a, .value = 0x88, },
 | |
| 		[28] = { .offset = 0x9b, .value = 0x00, },
 | |
| 		[29] = { .offset = 0x98, .value = 0x00, },
 | |
| 		[30] = { .offset = 0x99, .value = 0x00, },
 | |
| 		[31] = { .offset = 0xf7, .value = 0x88, },
 | |
| 		[32] = { .offset = 0xf8, .value = 0x0a, },
 | |
| 		[33] = { .offset = 0x9c, .value = 0x24, },
 | |
| 		[34] = { .offset = 0x9d, .value = 0x00, },
 | |
| 		[35] = { .offset = 0x9e, .value = 0x25, },
 | |
| 		[36] = { .offset = 0x9f, .value = 0x03, },
 | |
| 		[37] = { .offset = 0xa0, .value = 0x28, },
 | |
| 		[38] = { .offset = 0xa1, .value = 0x01, },
 | |
| 		[39] = { .offset = 0xa2, .value = 0x28, },
 | |
| 		[40] = { .offset = 0xa3, .value = 0x05, },
 | |
| 		[41] = { .offset = 0xb6, .value = 0x09, },
 | |
| 		[42] = { .offset = 0xb8, .value = 0x00, },
 | |
| 		[43] = { .offset = 0xb9, .value = 0xa0, },
 | |
| 		[44] = { .offset = 0xba, .value = 0x00, },
 | |
| 		[45] = { .offset = 0xbb, .value = 0x20, },
 | |
| 		[46] = { .offset = 0x10, .value = 0x00, },
 | |
| 		[47] = { .offset = 0x11, .value = 0xa0, },
 | |
| 		[48] = { .offset = 0x12, .value = 0x02, },
 | |
| 		[49] = { .offset = 0x20, .value = 0x00, },
 | |
| 		[50] = { .offset = 0x22, .value = 0x00, },
 | |
| 		[51] = { .offset = 0x23, .value = 0x00, },
 | |
| 		[52] = { .offset = 0x24, .value = 0x00, },
 | |
| 		[53] = { .offset = 0x25, .value = 0x00, },
 | |
| 		[54] = { .offset = 0x8c, .value = 0x10, },
 | |
| 		[55] = { .offset = 0x8d, .value = 0x02, },
 | |
| 		[56] = { .offset = 0x8e, .value = 0x10, },
 | |
| 		[57] = { .offset = 0x8f, .value = 0x00, },
 | |
| 		[58] = { .offset = 0x90, .value = 0xff, },
 | |
| 		[59] = { .offset = 0x91, .value = 0x07, },
 | |
| 		[60] = { .offset = 0x92, .value = 0xa0, },
 | |
| 		[61] = { .offset = 0x93, .value = 0x02, },
 | |
| 		[62] = { .offset = 0xa5, .value = 0x00, },
 | |
| 		[63] = { .offset = 0xa6, .value = 0x00, },
 | |
| 		[64] = { .offset = 0xa7, .value = 0x00, },
 | |
| 		[65] = { .offset = 0xa8, .value = 0x00, },
 | |
| 		[66] = { .offset = 0xa9, .value = 0x04, },
 | |
| 		[67] = { .offset = 0xaa, .value = 0x70, },
 | |
| 		[68] = { .offset = 0xab, .value = 0x4f, },
 | |
| 		[69] = { .offset = 0xac, .value = 0x00, },
 | |
| 		[70] = { .offset = 0xa4, .value = 0x84, },
 | |
| 		[71] = { .offset = 0x7e, .value = 0x18, },
 | |
| 		[72] = { .offset = 0x84, .value = 0x00, },
 | |
| 		[73] = { .offset = 0x85, .value = 0x00, },
 | |
| 		[74] = { .offset = 0x86, .value = 0x00, },
 | |
| 		[75] = { .offset = 0x87, .value = 0x00, },
 | |
| 		[76] = { .offset = 0x88, .value = 0x00, },
 | |
| 		[77] = { .offset = 0x89, .value = 0x00, },
 | |
| 		[78] = { .offset = 0x8a, .value = 0x00, },
 | |
| 		[79] = { .offset = 0x8b, .value = 0x00, },
 | |
| 		[80] = { .offset = 0x26, .value = 0x00, },
 | |
| 		[81] = { .offset = 0x27, .value = 0x00, },
 | |
| 		[82] = { .offset = 0xad, .value = 0x00, },
 | |
| 		[83] = { .offset = 0x08, .value = 0x34, }, /* 0x35 */
 | |
| 		[84] = { .offset = 0x41, .value = 0x00, },
 | |
| 		[85] = { .offset = 0xc0, .value = 0x01, },
 | |
| 	},
 | |
| };
 | |
| 
 | |
| static const struct ns2501_reg regs_init[] = {
 | |
| 	[0] = { .offset = 0x35, .value = 0xff, },
 | |
| 	[1] = { .offset = 0x34, .value = 0x00, },
 | |
| 	[2] = { .offset = 0x08, .value = 0x30, },
 | |
| };
 | |
| 
 | |
| struct ns2501_priv {
 | |
| 	bool quiet;
 | |
| 	const struct ns2501_reg *regs;
 | |
| };
 | |
| 
 | |
| #define NSPTR(d) ((NS2501Ptr)(d->DriverPrivate.ptr))
 | |
| 
 | |
| /*
 | |
|  * For reasons unclear to me, the ns2501 at least on the Fujitsu/Siemens
 | |
|  * laptops does not react on the i2c bus unless
 | |
|  * both the PLL is running and the display is configured in its native
 | |
|  * resolution.
 | |
|  * This function forces the DVO on, and stores the registers it touches.
 | |
|  * Afterwards, registers are restored to regular values.
 | |
|  *
 | |
|  * This is pretty much a hack, though it works.
 | |
|  * Without that, ns2501_readb and ns2501_writeb fail
 | |
|  * when switching the resolution.
 | |
|  */
 | |
| 
 | |
| /*
 | |
| ** Read a register from the ns2501.
 | |
| ** Returns true if successful, false otherwise.
 | |
| ** If it returns false, it might be wise to enable the
 | |
| ** DVO with the above function.
 | |
| */
 | |
| static bool ns2501_readb(struct intel_dvo_device *dvo, int addr, uint8_t * ch)
 | |
| {
 | |
| 	struct ns2501_priv *ns = dvo->dev_priv;
 | |
| 	struct i2c_adapter *adapter = dvo->i2c_bus;
 | |
| 	u8 out_buf[2];
 | |
| 	u8 in_buf[2];
 | |
| 
 | |
| 	struct i2c_msg msgs[] = {
 | |
| 		{
 | |
| 		 .addr = dvo->slave_addr,
 | |
| 		 .flags = 0,
 | |
| 		 .len = 1,
 | |
| 		 .buf = out_buf,
 | |
| 		 },
 | |
| 		{
 | |
| 		 .addr = dvo->slave_addr,
 | |
| 		 .flags = I2C_M_RD,
 | |
| 		 .len = 1,
 | |
| 		 .buf = in_buf,
 | |
| 		 }
 | |
| 	};
 | |
| 
 | |
| 	out_buf[0] = addr;
 | |
| 	out_buf[1] = 0;
 | |
| 
 | |
| 	if (i2c_transfer(adapter, msgs, 2) == 2) {
 | |
| 		*ch = in_buf[0];
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	if (!ns->quiet) {
 | |
| 		DRM_DEBUG_KMS
 | |
| 		    ("Unable to read register 0x%02x from %s:0x%02x.\n", addr,
 | |
| 		     adapter->name, dvo->slave_addr);
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Write a register to the ns2501.
 | |
| ** Returns true if successful, false otherwise.
 | |
| ** If it returns false, it might be wise to enable the
 | |
| ** DVO with the above function.
 | |
| */
 | |
| static bool ns2501_writeb(struct intel_dvo_device *dvo, int addr, uint8_t ch)
 | |
| {
 | |
| 	struct ns2501_priv *ns = dvo->dev_priv;
 | |
| 	struct i2c_adapter *adapter = dvo->i2c_bus;
 | |
| 	uint8_t out_buf[2];
 | |
| 
 | |
| 	struct i2c_msg msg = {
 | |
| 		.addr = dvo->slave_addr,
 | |
| 		.flags = 0,
 | |
| 		.len = 2,
 | |
| 		.buf = out_buf,
 | |
| 	};
 | |
| 
 | |
| 	out_buf[0] = addr;
 | |
| 	out_buf[1] = ch;
 | |
| 
 | |
| 	if (i2c_transfer(adapter, &msg, 1) == 1) {
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	if (!ns->quiet) {
 | |
| 		DRM_DEBUG_KMS("Unable to write register 0x%02x to %s:%d\n",
 | |
| 			      addr, adapter->name, dvo->slave_addr);
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| /* National Semiconductor 2501 driver for chip on i2c bus
 | |
|  * scan for the chip on the bus.
 | |
|  * Hope the VBIOS initialized the PLL correctly so we can
 | |
|  * talk to it. If not, it will not be seen and not detected.
 | |
|  * Bummer!
 | |
|  */
 | |
| static bool ns2501_init(struct intel_dvo_device *dvo,
 | |
| 			struct i2c_adapter *adapter)
 | |
| {
 | |
| 	/* this will detect the NS2501 chip on the specified i2c bus */
 | |
| 	struct ns2501_priv *ns;
 | |
| 	unsigned char ch;
 | |
| 
 | |
| 	ns = kzalloc(sizeof(struct ns2501_priv), GFP_KERNEL);
 | |
| 	if (ns == NULL)
 | |
| 		return false;
 | |
| 
 | |
| 	dvo->i2c_bus = adapter;
 | |
| 	dvo->dev_priv = ns;
 | |
| 	ns->quiet = true;
 | |
| 
 | |
| 	if (!ns2501_readb(dvo, NS2501_VID_LO, &ch))
 | |
| 		goto out;
 | |
| 
 | |
| 	if (ch != (NS2501_VID & 0xff)) {
 | |
| 		DRM_DEBUG_KMS("ns2501 not detected got %d: from %s Slave %d.\n",
 | |
| 			      ch, adapter->name, dvo->slave_addr);
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (!ns2501_readb(dvo, NS2501_DID_LO, &ch))
 | |
| 		goto out;
 | |
| 
 | |
| 	if (ch != (NS2501_DID & 0xff)) {
 | |
| 		DRM_DEBUG_KMS("ns2501 not detected got %d: from %s Slave %d.\n",
 | |
| 			      ch, adapter->name, dvo->slave_addr);
 | |
| 		goto out;
 | |
| 	}
 | |
| 	ns->quiet = false;
 | |
| 
 | |
| 	DRM_DEBUG_KMS("init ns2501 dvo controller successfully!\n");
 | |
| 
 | |
| 	return true;
 | |
| 
 | |
| out:
 | |
| 	kfree(ns);
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| static enum drm_connector_status ns2501_detect(struct intel_dvo_device *dvo)
 | |
| {
 | |
| 	/*
 | |
| 	 * This is a Laptop display, it doesn't have hotplugging.
 | |
| 	 * Even if not, the detection bit of the 2501 is unreliable as
 | |
| 	 * it only works for some display types.
 | |
| 	 * It is even more unreliable as the PLL must be active for
 | |
| 	 * allowing reading from the chiop.
 | |
| 	 */
 | |
| 	return connector_status_connected;
 | |
| }
 | |
| 
 | |
| static enum drm_mode_status ns2501_mode_valid(struct intel_dvo_device *dvo,
 | |
| 					      struct drm_display_mode *mode)
 | |
| {
 | |
| 	DRM_DEBUG_KMS
 | |
| 	    ("is mode valid (hdisplay=%d,htotal=%d,vdisplay=%d,vtotal=%d)\n",
 | |
| 	     mode->hdisplay, mode->htotal, mode->vdisplay, mode->vtotal);
 | |
| 
 | |
| 	/*
 | |
| 	 * Currently, these are all the modes I have data from.
 | |
| 	 * More might exist. Unclear how to find the native resolution
 | |
| 	 * of the panel in here so we could always accept it
 | |
| 	 * by disabling the scaler.
 | |
| 	 */
 | |
| 	if ((mode->hdisplay == 640 && mode->vdisplay == 480 && mode->clock == 25175) ||
 | |
| 	    (mode->hdisplay == 800 && mode->vdisplay == 600 && mode->clock == 40000) ||
 | |
| 	    (mode->hdisplay == 1024 && mode->vdisplay == 768 && mode->clock == 65000)) {
 | |
| 		return MODE_OK;
 | |
| 	} else {
 | |
| 		return MODE_ONE_SIZE;	/* Is this a reasonable error? */
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void ns2501_mode_set(struct intel_dvo_device *dvo,
 | |
| 			    struct drm_display_mode *mode,
 | |
| 			    struct drm_display_mode *adjusted_mode)
 | |
| {
 | |
| 	struct ns2501_priv *ns = (struct ns2501_priv *)(dvo->dev_priv);
 | |
| 	int mode_idx, i;
 | |
| 
 | |
| 	DRM_DEBUG_KMS
 | |
| 	    ("set mode (hdisplay=%d,htotal=%d,vdisplay=%d,vtotal=%d).\n",
 | |
| 	     mode->hdisplay, mode->htotal, mode->vdisplay, mode->vtotal);
 | |
| 
 | |
| 	if (mode->hdisplay == 640 && mode->vdisplay == 480)
 | |
| 		mode_idx = MODE_640x480;
 | |
| 	else if (mode->hdisplay == 800 && mode->vdisplay == 600)
 | |
| 		mode_idx = MODE_800x600;
 | |
| 	else if (mode->hdisplay == 1024 && mode->vdisplay == 768)
 | |
| 		mode_idx = MODE_1024x768;
 | |
| 	else
 | |
| 		return;
 | |
| 
 | |
| 	/* Hopefully doing it every time won't hurt... */
 | |
| 	for (i = 0; i < ARRAY_SIZE(regs_init); i++)
 | |
| 		ns2501_writeb(dvo, regs_init[i].offset, regs_init[i].value);
 | |
| 
 | |
| 	ns->regs = regs_1024x768[mode_idx];
 | |
| 
 | |
| 	for (i = 0; i < 84; i++)
 | |
| 		ns2501_writeb(dvo, ns->regs[i].offset, ns->regs[i].value);
 | |
| }
 | |
| 
 | |
| /* set the NS2501 power state */
 | |
| static bool ns2501_get_hw_state(struct intel_dvo_device *dvo)
 | |
| {
 | |
| 	unsigned char ch;
 | |
| 
 | |
| 	if (!ns2501_readb(dvo, NS2501_REG8, &ch))
 | |
| 		return false;
 | |
| 
 | |
| 	return ch & NS2501_8_PD;
 | |
| }
 | |
| 
 | |
| /* set the NS2501 power state */
 | |
| static void ns2501_dpms(struct intel_dvo_device *dvo, bool enable)
 | |
| {
 | |
| 	struct ns2501_priv *ns = (struct ns2501_priv *)(dvo->dev_priv);
 | |
| 
 | |
| 	DRM_DEBUG_KMS("Trying set the dpms of the DVO to %i\n", enable);
 | |
| 
 | |
| 	if (enable) {
 | |
| 		if (WARN_ON(ns->regs[83].offset != 0x08 ||
 | |
| 			    ns->regs[84].offset != 0x41 ||
 | |
| 			    ns->regs[85].offset != 0xc0))
 | |
| 			return;
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0xc0, ns->regs[85].value | 0x08);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0x41, ns->regs[84].value);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0x34, 0x01);
 | |
| 		msleep(15);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0x08, 0x35);
 | |
| 		if (!(ns->regs[83].value & NS2501_8_BPAS))
 | |
| 			ns2501_writeb(dvo, 0x08, 0x31);
 | |
| 		msleep(200);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0x34, 0x03);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0xc0, ns->regs[85].value);
 | |
| 	} else {
 | |
| 		ns2501_writeb(dvo, 0x34, 0x01);
 | |
| 		msleep(200);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0x08, 0x34);
 | |
| 		msleep(15);
 | |
| 
 | |
| 		ns2501_writeb(dvo, 0x34, 0x00);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void ns2501_destroy(struct intel_dvo_device *dvo)
 | |
| {
 | |
| 	struct ns2501_priv *ns = dvo->dev_priv;
 | |
| 
 | |
| 	if (ns) {
 | |
| 		kfree(ns);
 | |
| 		dvo->dev_priv = NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| struct intel_dvo_dev_ops ns2501_ops = {
 | |
| 	.init = ns2501_init,
 | |
| 	.detect = ns2501_detect,
 | |
| 	.mode_valid = ns2501_mode_valid,
 | |
| 	.mode_set = ns2501_mode_set,
 | |
| 	.dpms = ns2501_dpms,
 | |
| 	.get_hw_state = ns2501_get_hw_state,
 | |
| 	.destroy = ns2501_destroy,
 | |
| };
 |