intel_acpi.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /*
  2. * Intel ACPI functions
  3. *
  4. * _DSM related code stolen from nouveau_acpi.c.
  5. */
  6. #include <linux/pci.h>
  7. #include <linux/acpi.h>
  8. #include <linux/vga_switcheroo.h>
  9. #include <acpi/acpi_drivers.h>
  10. #include <drm/drmP.h>
  11. #include "i915_drv.h"
  12. #define INTEL_DSM_REVISION_ID 1 /* For Calpella anyway... */
  13. #define INTEL_DSM_FN_SUPPORTED_FUNCTIONS 0 /* No args */
  14. #define INTEL_DSM_FN_PLATFORM_MUX_INFO 1 /* No args */
  15. static struct intel_dsm_priv {
  16. acpi_handle dhandle;
  17. } intel_dsm_priv;
  18. static const u8 intel_dsm_guid[] = {
  19. 0xd3, 0x73, 0xd8, 0x7e,
  20. 0xd0, 0xc2,
  21. 0x4f, 0x4e,
  22. 0xa8, 0x54,
  23. 0x0f, 0x13, 0x17, 0xb0, 0x1c, 0x2c
  24. };
  25. static int intel_dsm(acpi_handle handle, int func)
  26. {
  27. struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
  28. struct acpi_object_list input;
  29. union acpi_object params[4];
  30. union acpi_object *obj;
  31. u32 result;
  32. int ret = 0;
  33. input.count = 4;
  34. input.pointer = params;
  35. params[0].type = ACPI_TYPE_BUFFER;
  36. params[0].buffer.length = sizeof(intel_dsm_guid);
  37. params[0].buffer.pointer = (char *)intel_dsm_guid;
  38. params[1].type = ACPI_TYPE_INTEGER;
  39. params[1].integer.value = INTEL_DSM_REVISION_ID;
  40. params[2].type = ACPI_TYPE_INTEGER;
  41. params[2].integer.value = func;
  42. params[3].type = ACPI_TYPE_PACKAGE;
  43. params[3].package.count = 0;
  44. params[3].package.elements = NULL;
  45. ret = acpi_evaluate_object(handle, "_DSM", &input, &output);
  46. if (ret) {
  47. DRM_DEBUG_DRIVER("failed to evaluate _DSM: %d\n", ret);
  48. return ret;
  49. }
  50. obj = (union acpi_object *)output.pointer;
  51. result = 0;
  52. switch (obj->type) {
  53. case ACPI_TYPE_INTEGER:
  54. result = obj->integer.value;
  55. break;
  56. case ACPI_TYPE_BUFFER:
  57. if (obj->buffer.length == 4) {
  58. result = (obj->buffer.pointer[0] |
  59. (obj->buffer.pointer[1] << 8) |
  60. (obj->buffer.pointer[2] << 16) |
  61. (obj->buffer.pointer[3] << 24));
  62. break;
  63. }
  64. default:
  65. ret = -EINVAL;
  66. break;
  67. }
  68. if (result == 0x80000002)
  69. ret = -ENODEV;
  70. kfree(output.pointer);
  71. return ret;
  72. }
  73. static char *intel_dsm_port_name(u8 id)
  74. {
  75. switch (id) {
  76. case 0:
  77. return "Reserved";
  78. case 1:
  79. return "Analog VGA";
  80. case 2:
  81. return "LVDS";
  82. case 3:
  83. return "Reserved";
  84. case 4:
  85. return "HDMI/DVI_B";
  86. case 5:
  87. return "HDMI/DVI_C";
  88. case 6:
  89. return "HDMI/DVI_D";
  90. case 7:
  91. return "DisplayPort_A";
  92. case 8:
  93. return "DisplayPort_B";
  94. case 9:
  95. return "DisplayPort_C";
  96. case 0xa:
  97. return "DisplayPort_D";
  98. case 0xb:
  99. case 0xc:
  100. case 0xd:
  101. return "Reserved";
  102. case 0xe:
  103. return "WiDi";
  104. default:
  105. return "bad type";
  106. }
  107. }
  108. static char *intel_dsm_mux_type(u8 type)
  109. {
  110. switch (type) {
  111. case 0:
  112. return "unknown";
  113. case 1:
  114. return "No MUX, iGPU only";
  115. case 2:
  116. return "No MUX, dGPU only";
  117. case 3:
  118. return "MUXed between iGPU and dGPU";
  119. default:
  120. return "bad type";
  121. }
  122. }
  123. static void intel_dsm_platform_mux_info(void)
  124. {
  125. struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
  126. struct acpi_object_list input;
  127. union acpi_object params[4];
  128. union acpi_object *pkg;
  129. int i, ret;
  130. input.count = 4;
  131. input.pointer = params;
  132. params[0].type = ACPI_TYPE_BUFFER;
  133. params[0].buffer.length = sizeof(intel_dsm_guid);
  134. params[0].buffer.pointer = (char *)intel_dsm_guid;
  135. params[1].type = ACPI_TYPE_INTEGER;
  136. params[1].integer.value = INTEL_DSM_REVISION_ID;
  137. params[2].type = ACPI_TYPE_INTEGER;
  138. params[2].integer.value = INTEL_DSM_FN_PLATFORM_MUX_INFO;
  139. params[3].type = ACPI_TYPE_PACKAGE;
  140. params[3].package.count = 0;
  141. params[3].package.elements = NULL;
  142. ret = acpi_evaluate_object(intel_dsm_priv.dhandle, "_DSM", &input,
  143. &output);
  144. if (ret) {
  145. DRM_DEBUG_DRIVER("failed to evaluate _DSM: %d\n", ret);
  146. goto out;
  147. }
  148. pkg = (union acpi_object *)output.pointer;
  149. if (pkg->type == ACPI_TYPE_PACKAGE) {
  150. union acpi_object *connector_count = &pkg->package.elements[0];
  151. DRM_DEBUG_DRIVER("MUX info connectors: %lld\n",
  152. (unsigned long long)connector_count->integer.value);
  153. for (i = 1; i < pkg->package.count; i++) {
  154. union acpi_object *obj = &pkg->package.elements[i];
  155. union acpi_object *connector_id =
  156. &obj->package.elements[0];
  157. union acpi_object *info = &obj->package.elements[1];
  158. DRM_DEBUG_DRIVER("Connector id: 0x%016llx\n",
  159. (unsigned long long)connector_id->integer.value);
  160. DRM_DEBUG_DRIVER(" port id: %s\n",
  161. intel_dsm_port_name(info->buffer.pointer[0]));
  162. DRM_DEBUG_DRIVER(" display mux info: %s\n",
  163. intel_dsm_mux_type(info->buffer.pointer[1]));
  164. DRM_DEBUG_DRIVER(" aux/dc mux info: %s\n",
  165. intel_dsm_mux_type(info->buffer.pointer[2]));
  166. DRM_DEBUG_DRIVER(" hpd mux info: %s\n",
  167. intel_dsm_mux_type(info->buffer.pointer[3]));
  168. }
  169. }
  170. out:
  171. kfree(output.pointer);
  172. }
  173. static bool intel_dsm_pci_probe(struct pci_dev *pdev)
  174. {
  175. acpi_handle dhandle, intel_handle;
  176. acpi_status status;
  177. int ret;
  178. dhandle = DEVICE_ACPI_HANDLE(&pdev->dev);
  179. if (!dhandle)
  180. return false;
  181. status = acpi_get_handle(dhandle, "_DSM", &intel_handle);
  182. if (ACPI_FAILURE(status)) {
  183. DRM_DEBUG_KMS("no _DSM method for intel device\n");
  184. return false;
  185. }
  186. ret = intel_dsm(dhandle, INTEL_DSM_FN_SUPPORTED_FUNCTIONS);
  187. if (ret < 0) {
  188. DRM_DEBUG_KMS("failed to get supported _DSM functions\n");
  189. return false;
  190. }
  191. intel_dsm_priv.dhandle = dhandle;
  192. intel_dsm_platform_mux_info();
  193. return true;
  194. }
  195. static bool intel_dsm_detect(void)
  196. {
  197. char acpi_method_name[255] = { 0 };
  198. struct acpi_buffer buffer = {sizeof(acpi_method_name), acpi_method_name};
  199. struct pci_dev *pdev = NULL;
  200. bool has_dsm = false;
  201. int vga_count = 0;
  202. while ((pdev = pci_get_class(PCI_CLASS_DISPLAY_VGA << 8, pdev)) != NULL) {
  203. vga_count++;
  204. has_dsm |= intel_dsm_pci_probe(pdev);
  205. }
  206. if (vga_count == 2 && has_dsm) {
  207. acpi_get_name(intel_dsm_priv.dhandle, ACPI_FULL_PATHNAME, &buffer);
  208. DRM_DEBUG_DRIVER("VGA switcheroo: detected DSM switching method %s handle\n",
  209. acpi_method_name);
  210. return true;
  211. }
  212. return false;
  213. }
  214. void intel_register_dsm_handler(void)
  215. {
  216. if (!intel_dsm_detect())
  217. return;
  218. }
  219. void intel_unregister_dsm_handler(void)
  220. {
  221. }