| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Intel ACPI functions | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * _DSM related code stolen from nouveau_acpi.c. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <linux/pci.h>
 | 
					
						
							|  |  |  | #include <linux/acpi.h>
 | 
					
						
							|  |  |  | #include <linux/vga_switcheroo.h>
 | 
					
						
							| 
									
										
										
										
											2012-10-02 18:01:07 +01:00
										 |  |  | #include <drm/drmP.h>
 | 
					
						
							| 
									
										
										
										
											2012-04-16 14:07:40 -07:00
										 |  |  | #include "i915_drv.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define INTEL_DSM_REVISION_ID 1 /* For Calpella anyway... */
 | 
					
						
							|  |  |  | #define INTEL_DSM_FN_PLATFORM_MUX_INFO 1 /* No args */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct intel_dsm_priv { | 
					
						
							|  |  |  | 	acpi_handle dhandle; | 
					
						
							|  |  |  | } intel_dsm_priv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const u8 intel_dsm_guid[] = { | 
					
						
							|  |  |  | 	0xd3, 0x73, 0xd8, 0x7e, | 
					
						
							|  |  |  | 	0xd0, 0xc2, | 
					
						
							|  |  |  | 	0x4f, 0x4e, | 
					
						
							|  |  |  | 	0xa8, 0x54, | 
					
						
							|  |  |  | 	0x0f, 0x13, 0x17, 0xb0, 0x1c, 0x2c | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *intel_dsm_port_name(u8 id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (id) { | 
					
						
							|  |  |  | 	case 0: | 
					
						
							|  |  |  | 		return "Reserved"; | 
					
						
							|  |  |  | 	case 1: | 
					
						
							|  |  |  | 		return "Analog VGA"; | 
					
						
							|  |  |  | 	case 2: | 
					
						
							|  |  |  | 		return "LVDS"; | 
					
						
							|  |  |  | 	case 3: | 
					
						
							|  |  |  | 		return "Reserved"; | 
					
						
							|  |  |  | 	case 4: | 
					
						
							|  |  |  | 		return "HDMI/DVI_B"; | 
					
						
							|  |  |  | 	case 5: | 
					
						
							|  |  |  | 		return "HDMI/DVI_C"; | 
					
						
							|  |  |  | 	case 6: | 
					
						
							|  |  |  | 		return "HDMI/DVI_D"; | 
					
						
							|  |  |  | 	case 7: | 
					
						
							|  |  |  | 		return "DisplayPort_A"; | 
					
						
							|  |  |  | 	case 8: | 
					
						
							|  |  |  | 		return "DisplayPort_B"; | 
					
						
							|  |  |  | 	case 9: | 
					
						
							|  |  |  | 		return "DisplayPort_C"; | 
					
						
							|  |  |  | 	case 0xa: | 
					
						
							|  |  |  | 		return "DisplayPort_D"; | 
					
						
							|  |  |  | 	case 0xb: | 
					
						
							|  |  |  | 	case 0xc: | 
					
						
							|  |  |  | 	case 0xd: | 
					
						
							|  |  |  | 		return "Reserved"; | 
					
						
							|  |  |  | 	case 0xe: | 
					
						
							|  |  |  | 		return "WiDi"; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return "bad type"; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *intel_dsm_mux_type(u8 type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (type) { | 
					
						
							|  |  |  | 	case 0: | 
					
						
							|  |  |  | 		return "unknown"; | 
					
						
							|  |  |  | 	case 1: | 
					
						
							|  |  |  | 		return "No MUX, iGPU only"; | 
					
						
							|  |  |  | 	case 2: | 
					
						
							|  |  |  | 		return "No MUX, dGPU only"; | 
					
						
							|  |  |  | 	case 3: | 
					
						
							|  |  |  | 		return "MUXed between iGPU and dGPU"; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return "bad type"; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void intel_dsm_platform_mux_info(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-12-19 20:38:20 +08:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 	union acpi_object *pkg, *connector_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pkg = acpi_evaluate_dsm_typed(intel_dsm_priv.dhandle, intel_dsm_guid, | 
					
						
							|  |  |  | 			INTEL_DSM_REVISION_ID, INTEL_DSM_FN_PLATFORM_MUX_INFO, | 
					
						
							|  |  |  | 			NULL, ACPI_TYPE_PACKAGE); | 
					
						
							|  |  |  | 	if (!pkg) { | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("failed to evaluate _DSM\n"); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-19 20:38:20 +08:00
										 |  |  | 	connector_count = &pkg->package.elements[0]; | 
					
						
							|  |  |  | 	DRM_DEBUG_DRIVER("MUX info connectors: %lld\n", | 
					
						
							|  |  |  | 		  (unsigned long long)connector_count->integer.value); | 
					
						
							|  |  |  | 	for (i = 1; i < pkg->package.count; i++) { | 
					
						
							|  |  |  | 		union acpi_object *obj = &pkg->package.elements[i]; | 
					
						
							|  |  |  | 		union acpi_object *connector_id = &obj->package.elements[0]; | 
					
						
							|  |  |  | 		union acpi_object *info = &obj->package.elements[1]; | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("Connector id: 0x%016llx\n", | 
					
						
							|  |  |  | 			  (unsigned long long)connector_id->integer.value); | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("  port id: %s\n", | 
					
						
							|  |  |  | 		       intel_dsm_port_name(info->buffer.pointer[0])); | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("  display mux info: %s\n", | 
					
						
							|  |  |  | 		       intel_dsm_mux_type(info->buffer.pointer[1])); | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("  aux/dc mux info: %s\n", | 
					
						
							|  |  |  | 		       intel_dsm_mux_type(info->buffer.pointer[2])); | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("  hpd mux info: %s\n", | 
					
						
							|  |  |  | 		       intel_dsm_mux_type(info->buffer.pointer[3])); | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-19 20:38:20 +08:00
										 |  |  | 	ACPI_FREE(pkg); | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool intel_dsm_pci_probe(struct pci_dev *pdev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-03 08:31:59 +08:00
										 |  |  | 	acpi_handle dhandle; | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-14 23:17:21 +01:00
										 |  |  | 	dhandle = ACPI_HANDLE(&pdev->dev); | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 	if (!dhandle) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-19 20:38:20 +08:00
										 |  |  | 	if (!acpi_check_dsm(dhandle, intel_dsm_guid, INTEL_DSM_REVISION_ID, | 
					
						
							|  |  |  | 			    1 << INTEL_DSM_FN_PLATFORM_MUX_INFO)) { | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 		DRM_DEBUG_KMS("no _DSM method for intel device\n"); | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	intel_dsm_priv.dhandle = dhandle; | 
					
						
							|  |  |  | 	intel_dsm_platform_mux_info(); | 
					
						
							| 
									
										
										
										
											2013-12-19 20:38:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-07 16:01:13 -07:00
										 |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool intel_dsm_detect(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char acpi_method_name[255] = { 0 }; | 
					
						
							|  |  |  | 	struct acpi_buffer buffer = {sizeof(acpi_method_name), acpi_method_name}; | 
					
						
							|  |  |  | 	struct pci_dev *pdev = NULL; | 
					
						
							|  |  |  | 	bool has_dsm = false; | 
					
						
							|  |  |  | 	int vga_count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ((pdev = pci_get_class(PCI_CLASS_DISPLAY_VGA << 8, pdev)) != NULL) { | 
					
						
							|  |  |  | 		vga_count++; | 
					
						
							|  |  |  | 		has_dsm |= intel_dsm_pci_probe(pdev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (vga_count == 2 && has_dsm) { | 
					
						
							|  |  |  | 		acpi_get_name(intel_dsm_priv.dhandle, ACPI_FULL_PATHNAME, &buffer); | 
					
						
							|  |  |  | 		DRM_DEBUG_DRIVER("VGA switcheroo: detected DSM switching method %s handle\n", | 
					
						
							|  |  |  | 				 acpi_method_name); | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void intel_register_dsm_handler(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!intel_dsm_detect()) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void intel_unregister_dsm_handler(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } |