181 lines
4.6 KiB
C
181 lines
4.6 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
/*
|
|
* apple-gmux.h - microcontroller built into dual GPU MacBook Pro & Mac Pro
|
|
* Copyright (C) 2015 Lukas Wunner <lukas@wunner.de>
|
|
*/
|
|
|
|
#ifndef LINUX_APPLE_GMUX_H
|
|
#define LINUX_APPLE_GMUX_H
|
|
|
|
#include <linux/acpi.h>
|
|
#include <linux/io.h>
|
|
#include <linux/pnp.h>
|
|
|
|
#define GMUX_ACPI_HID "APP000B"
|
|
|
|
/*
|
|
* gmux port offsets. Many of these are not yet used, but may be in the
|
|
* future, and it's useful to have them documented here anyhow.
|
|
*/
|
|
#define GMUX_PORT_VERSION_MAJOR 0x04
|
|
#define GMUX_PORT_VERSION_MINOR 0x05
|
|
#define GMUX_PORT_VERSION_RELEASE 0x06
|
|
#define GMUX_PORT_SWITCH_DISPLAY 0x10
|
|
#define GMUX_PORT_SWITCH_GET_DISPLAY 0x11
|
|
#define GMUX_PORT_INTERRUPT_ENABLE 0x14
|
|
#define GMUX_PORT_INTERRUPT_STATUS 0x16
|
|
#define GMUX_PORT_SWITCH_DDC 0x28
|
|
#define GMUX_PORT_SWITCH_EXTERNAL 0x40
|
|
#define GMUX_PORT_SWITCH_GET_EXTERNAL 0x41
|
|
#define GMUX_PORT_DISCRETE_POWER 0x50
|
|
#define GMUX_PORT_MAX_BRIGHTNESS 0x70
|
|
#define GMUX_PORT_BRIGHTNESS 0x74
|
|
#define GMUX_PORT_VALUE 0xc2
|
|
#define GMUX_PORT_READ 0xd0
|
|
#define GMUX_PORT_WRITE 0xd4
|
|
|
|
#define GMUX_MMIO_PORT_SELECT 0x0e
|
|
#define GMUX_MMIO_COMMAND_SEND 0x0f
|
|
|
|
#define GMUX_MMIO_READ 0x00
|
|
#define GMUX_MMIO_WRITE 0x40
|
|
|
|
#define GMUX_MIN_IO_LEN (GMUX_PORT_BRIGHTNESS + 4)
|
|
|
|
enum apple_gmux_type {
|
|
APPLE_GMUX_TYPE_PIO,
|
|
APPLE_GMUX_TYPE_INDEXED,
|
|
APPLE_GMUX_TYPE_MMIO,
|
|
};
|
|
|
|
#if IS_ENABLED(CONFIG_APPLE_GMUX)
|
|
static inline bool apple_gmux_is_indexed(unsigned long iostart)
|
|
{
|
|
u16 val;
|
|
|
|
outb(0xaa, iostart + 0xcc);
|
|
outb(0x55, iostart + 0xcd);
|
|
outb(0x00, iostart + 0xce);
|
|
|
|
val = inb(iostart + 0xcc) | (inb(iostart + 0xcd) << 8);
|
|
if (val == 0x55aa)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
static inline bool apple_gmux_is_mmio(unsigned long iostart)
|
|
{
|
|
u8 __iomem *iomem_base = ioremap(iostart, 16);
|
|
u8 val;
|
|
|
|
if (!iomem_base)
|
|
return false;
|
|
|
|
/*
|
|
* If this is 0xff, then gmux must not be present, as the gmux would
|
|
* reset it to 0x00, or it would be one of 0x1, 0x4, 0x41, 0x44 if a
|
|
* command is currently being processed.
|
|
*/
|
|
val = ioread8(iomem_base + GMUX_MMIO_COMMAND_SEND);
|
|
iounmap(iomem_base);
|
|
return (val != 0xff);
|
|
}
|
|
|
|
/**
|
|
* apple_gmux_detect() - detect if gmux is built into the machine
|
|
*
|
|
* @pnp_dev: Device to probe or NULL to use the first matching device
|
|
* @type_ret: Returns (by reference) the apple_gmux_type of the device
|
|
*
|
|
* Detect if a supported gmux device is present by actually probing it.
|
|
* This avoids the false positives returned on some models by
|
|
* apple_gmux_present().
|
|
*
|
|
* Return: %true if a supported gmux ACPI device is detected and the kernel
|
|
* was configured with CONFIG_APPLE_GMUX, %false otherwise.
|
|
*/
|
|
static inline bool apple_gmux_detect(struct pnp_dev *pnp_dev, enum apple_gmux_type *type_ret)
|
|
{
|
|
u8 ver_major, ver_minor, ver_release;
|
|
struct device *dev = NULL;
|
|
struct acpi_device *adev;
|
|
struct resource *res;
|
|
enum apple_gmux_type type = APPLE_GMUX_TYPE_PIO;
|
|
bool ret = false;
|
|
|
|
if (!pnp_dev) {
|
|
adev = acpi_dev_get_first_match_dev(GMUX_ACPI_HID, NULL, -1);
|
|
if (!adev)
|
|
return false;
|
|
|
|
dev = get_device(acpi_get_first_physical_node(adev));
|
|
acpi_dev_put(adev);
|
|
if (!dev)
|
|
return false;
|
|
|
|
pnp_dev = to_pnp_dev(dev);
|
|
}
|
|
|
|
res = pnp_get_resource(pnp_dev, IORESOURCE_IO, 0);
|
|
if (res && resource_size(res) >= GMUX_MIN_IO_LEN) {
|
|
/*
|
|
* Invalid version information may indicate either that the gmux
|
|
* device isn't present or that it's a new one that uses indexed io.
|
|
*/
|
|
ver_major = inb(res->start + GMUX_PORT_VERSION_MAJOR);
|
|
ver_minor = inb(res->start + GMUX_PORT_VERSION_MINOR);
|
|
ver_release = inb(res->start + GMUX_PORT_VERSION_RELEASE);
|
|
if (ver_major == 0xff && ver_minor == 0xff && ver_release == 0xff) {
|
|
if (apple_gmux_is_indexed(res->start))
|
|
type = APPLE_GMUX_TYPE_INDEXED;
|
|
else
|
|
goto out;
|
|
}
|
|
} else {
|
|
res = pnp_get_resource(pnp_dev, IORESOURCE_MEM, 0);
|
|
if (res && apple_gmux_is_mmio(res->start))
|
|
type = APPLE_GMUX_TYPE_MMIO;
|
|
else
|
|
goto out;
|
|
}
|
|
|
|
if (type_ret)
|
|
*type_ret = type;
|
|
|
|
ret = true;
|
|
out:
|
|
put_device(dev);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* apple_gmux_present() - check if gmux ACPI device is present
|
|
*
|
|
* Drivers may use this to activate quirks specific to dual GPU MacBook Pros
|
|
* and Mac Pros, e.g. for deferred probing, runtime pm and backlight.
|
|
*
|
|
* Return: %true if gmux ACPI device is present and the kernel was configured
|
|
* with CONFIG_APPLE_GMUX, %false otherwise.
|
|
*/
|
|
static inline bool apple_gmux_present(void)
|
|
{
|
|
return acpi_dev_found(GMUX_ACPI_HID);
|
|
}
|
|
|
|
#else /* !CONFIG_APPLE_GMUX */
|
|
|
|
static inline bool apple_gmux_present(void)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
static inline bool apple_gmux_detect(struct pnp_dev *pnp_dev, bool *indexed_ret)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
#endif /* !CONFIG_APPLE_GMUX */
|
|
|
|
#endif /* LINUX_APPLE_GMUX_H */
|