/* $NetBSD: intel_acpi.c,v 1.6 2022/02/27 14:20:30 riastradh Exp $ */ // SPDX-License-Identifier: GPL-2.0 /* * Intel ACPI functions * * _DSM related code stolen from nouveau_acpi.c. */ #include __KERNEL_RCSID(0, "$NetBSD: intel_acpi.c,v 1.6 2022/02/27 14:20:30 riastradh Exp $"); #include #include #include "i915_drv.h" #include "intel_acpi.h" #ifdef __NetBSD__ #include #define _COMPONENT ACPI_BUTTON_COMPONENT ACPI_MODULE_NAME("acpi_intel_brightness") #include #define acpi_handle ACPI_HANDLE #define buffer Buffer #define count Count #define elements Elements #define integer Integer #define package Package #define pointer Pointer #define value Value static ACPI_OBJECT * acpi_evaluate_dsm(ACPI_HANDLE handle, const guid_t *uuid, int rev, int func, ACPI_OBJECT *argv4) { ACPI_OBJECT_LIST arg; ACPI_OBJECT params[4]; ACPI_BUFFER buf; ACPI_STATUS rv; if (handle == NULL) handle = ACPI_ROOT_OBJECT; arg.Count = 4; arg.Pointer = params; params[0].Type = ACPI_TYPE_BUFFER; params[0].Buffer.Length = 16; params[0].Buffer.Pointer = (char *)__UNCONST(uuid); params[1].Type = ACPI_TYPE_INTEGER; params[1].Integer.Value = rev; params[2].Type = ACPI_TYPE_INTEGER; params[2].Integer.Value = func; if (argv4 != NULL) { params[3] = *argv4; } else { params[3].Type = ACPI_TYPE_PACKAGE; params[3].Package.Count = 0; params[3].Package.Elements = NULL; } buf.Pointer = NULL; buf.Length = ACPI_ALLOCATE_LOCAL_BUFFER; rv = AcpiEvaluateObject(handle, "_DSM", &arg, &buf); if (ACPI_SUCCESS(rv)) return (ACPI_OBJECT *)buf.Pointer; return NULL; } static inline ACPI_OBJECT * acpi_evaluate_dsm_typed(ACPI_HANDLE handle, const guid_t *uuid, int rev, int func, ACPI_OBJECT *argv4, ACPI_OBJECT_TYPE type) { ACPI_OBJECT *obj; obj = acpi_evaluate_dsm(handle, uuid, rev, func, argv4); if (obj != NULL && obj->Type != type) { ACPI_FREE(obj); obj = NULL; } return obj; } #define ACPI_INIT_DSM_ARGV4(cnt, eles) \ { \ .Package.Type = ACPI_TYPE_PACKAGE, \ .Package.Count = (cnt), \ .Package.Elements = (eles) \ } static bool acpi_check_dsm(ACPI_HANDLE handle, const guid_t *uuid, int rev, uint64_t funcs) { ACPI_OBJECT *obj; uint64_t mask = 0; int i; if (funcs == 0) return false; obj = acpi_evaluate_dsm(handle, uuid, rev, 0, NULL); if (obj == NULL) return false; if (obj->Type == ACPI_TYPE_INTEGER) mask = obj->Integer.Value; else if (obj->Type == ACPI_TYPE_BUFFER) for (i = 0; i < obj->Buffer.Length && i < 8; i++) mask |= (uint64_t)obj->Buffer.Pointer[i] << (i * 8); ACPI_FREE(obj); if ((mask & 0x1) == 0x1 && (mask & funcs) == funcs) return true; return false; } #endif #define INTEL_DSM_REVISION_ID 1 /* For Calpella anyway... */ #define INTEL_DSM_FN_PLATFORM_MUX_INFO 1 /* No args */ static const guid_t intel_dsm_guid = GUID_INIT(0x7ed873d3, 0xc2d0, 0x4e4f, 0xa8, 0x54, 0x0f, 0x13, 0x17, 0xb0, 0x1c, 0x2c); static const 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 const 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(acpi_handle dhandle) { int i; union acpi_object *pkg, *connector_count; pkg = acpi_evaluate_dsm_typed(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; } 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])); } ACPI_FREE(pkg); } #ifdef __NetBSD__ static ACPI_HANDLE intel_dsm_pci_probe(ACPI_HANDLE dhandle) #else static acpi_handle intel_dsm_pci_probe(struct pci_dev *pdev) #endif { #ifndef __NetBSD__ acpi_handle dhandle; dhandle = ACPI_HANDLE(&pdev->dev); if (!dhandle) return NULL; #endif if (!acpi_check_dsm(dhandle, &intel_dsm_guid, INTEL_DSM_REVISION_ID, 1 << INTEL_DSM_FN_PLATFORM_MUX_INFO)) { DRM_DEBUG_KMS("no _DSM method for intel device\n"); return NULL; } intel_dsm_platform_mux_info(dhandle); return dhandle; } #ifdef __NetBSD__ static int vga_count; static ACPI_HANDLE intel_dsm_handle; /* XXX from sys/dev/pci/vga_pcivar.h */ #define DEVICE_IS_VGA_PCI(class, id) \ (((PCI_CLASS(class) == PCI_CLASS_DISPLAY && \ PCI_SUBCLASS(class) == PCI_SUBCLASS_DISPLAY_VGA) || \ (PCI_CLASS(class) == PCI_CLASS_PREHISTORIC && \ PCI_SUBCLASS(class) == PCI_SUBCLASS_PREHISTORIC_VGA)) ? 1 : 0) static int intel_dsm_vga_match(const struct pci_attach_args *pa) { if (!DEVICE_IS_VGA_PCI(pa->pa_class, pa->pa_id)) return 0; vga_count++; struct acpi_devnode *node = acpi_pcidev_find(pci_get_segment(pa->pa_pc), pa->pa_bus, pa->pa_device, pa->pa_function); if (node != NULL && intel_dsm_handle == NULL) intel_dsm_handle = intel_dsm_pci_probe(node->ad_handle); return 0; } static bool intel_dsm_detect(struct drm_device *dev) { char acpi_method_name[255] = { 0 }; vga_count = 0; pci_find_device(&dev->pdev->pd_pa, intel_dsm_vga_match); if (vga_count == 2 && intel_dsm_handle) { const char *name = acpi_name(intel_dsm_handle); strlcpy(acpi_method_name, name, sizeof(acpi_method_name)); DRM_DEBUG_DRIVER("VGA switcheroo: detected DSM switching method %s handle\n", acpi_method_name); return true; } return false; } #else static bool intel_dsm_detect(void) { acpi_handle dhandle = NULL; char acpi_method_name[255] = { 0 }; struct acpi_buffer buffer = {sizeof(acpi_method_name), acpi_method_name}; struct pci_dev *pdev = NULL; int vga_count = 0; while ((pdev = pci_get_class(PCI_CLASS_DISPLAY_VGA << 8, pdev)) != NULL) { vga_count++; dhandle = intel_dsm_pci_probe(pdev) ?: dhandle; } if (vga_count == 2 && dhandle) { acpi_get_name(dhandle, ACPI_FULL_PATHNAME, &buffer); DRM_DEBUG_DRIVER("vga_switcheroo: detected DSM switching method %s handle\n", acpi_method_name); return true; } return false; } #endif #ifdef __NetBSD__ void intel_register_dsm_handler(struct drm_i915_private *i915) { if (!intel_dsm_detect(&i915->drm)) return; } #else void intel_register_dsm_handler(void) { if (!intel_dsm_detect()) return; } #endif void intel_unregister_dsm_handler(void) { }