151 lines
4.7 KiB
C
151 lines
4.7 KiB
C
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
/*
|
||
|
* Driver to power on the Analogix ANX7428 USB Type-C crosspoint switch
|
||
|
* on MeeGoPad top-set boxes.
|
||
|
*
|
||
|
* The MeeGoPad T8 and T9 are Cherry Trail top-set boxes which
|
||
|
* use an ANX7428 to provide a Type-C port with USB3.1 Gen 1 and
|
||
|
* DisplayPort over Type-C alternate mode support.
|
||
|
*
|
||
|
* The ANX7428 has a microcontroller which takes care of the PD
|
||
|
* negotiation and automatically sets the builtin Crosspoint Switch
|
||
|
* to send the right signal to the 4 highspeed pairs of the Type-C
|
||
|
* connector. It also takes care of HPD and AUX channel routing for
|
||
|
* DP alternate mode.
|
||
|
*
|
||
|
* IOW the ANX7428 operates fully autonomous and to the x5-Z8350 SoC
|
||
|
* things look like there simply is a USB-3 Type-A connector and a
|
||
|
* separate DisplayPort connector. Except that the BIOS does not
|
||
|
* power on the ANX7428 at boot. This driver takes care of powering
|
||
|
* on the ANX7428.
|
||
|
*
|
||
|
* It should be possible to tell the micro-controller which data- and/or
|
||
|
* power-role to negotiate and to swap the role(s) after negotiation
|
||
|
* but the MeeGoPad top-set boxes always draw their power from a separate
|
||
|
* power-connector and they only support USB host-mode. So this functionality
|
||
|
* is unnecessary and due to lack of documentation this is tricky to support.
|
||
|
*
|
||
|
* For a more complete ANX7428 driver see drivers/usb/misc/anx7418/ of
|
||
|
* the LineageOS kernel for the LG G5 (International) aka the LG H850:
|
||
|
* https://github.com/LineageOS/android_kernel_lge_msm8996/
|
||
|
*
|
||
|
* (C) Copyright 2024 Hans de Goede <hansg@kernel.org>
|
||
|
*/
|
||
|
|
||
|
#include <linux/acpi.h>
|
||
|
#include <linux/bits.h>
|
||
|
#include <linux/delay.h>
|
||
|
#include <linux/dev_printk.h>
|
||
|
#include <linux/dmi.h>
|
||
|
#include <linux/err.h>
|
||
|
#include <linux/gpio/consumer.h>
|
||
|
#include <linux/i2c.h>
|
||
|
#include <linux/iopoll.h>
|
||
|
#include <linux/module.h>
|
||
|
#include <linux/types.h>
|
||
|
|
||
|
/* Register addresses and fields */
|
||
|
#define VENDOR_ID 0x00
|
||
|
#define DEVICE_ID 0x02
|
||
|
|
||
|
#define TX_STATUS 0x16
|
||
|
#define STATUS_SUCCESS BIT(0)
|
||
|
#define STATUS_ERROR BIT(1)
|
||
|
#define OCM_STARTUP BIT(7)
|
||
|
|
||
|
static bool force;
|
||
|
module_param(force, bool, 0444);
|
||
|
MODULE_PARM_DESC(force, "Force the driver to probe on unknown boards");
|
||
|
|
||
|
static const struct acpi_gpio_params enable_gpio = { 0, 0, false };
|
||
|
static const struct acpi_gpio_params reset_gpio = { 1, 0, true };
|
||
|
|
||
|
static const struct acpi_gpio_mapping meegopad_anx7428_gpios[] = {
|
||
|
{ "enable-gpios", &enable_gpio, 1 },
|
||
|
{ "reset-gpios", &reset_gpio, 1 },
|
||
|
{ }
|
||
|
};
|
||
|
|
||
|
static const struct dmi_system_id meegopad_anx7428_ids[] = {
|
||
|
{
|
||
|
/* Meegopad T08 */
|
||
|
.matches = {
|
||
|
DMI_MATCH(DMI_SYS_VENDOR, "Default string"),
|
||
|
DMI_MATCH(DMI_PRODUCT_NAME, "Default string"),
|
||
|
DMI_MATCH(DMI_BOARD_NAME, "T3 MRD"),
|
||
|
DMI_MATCH(DMI_BOARD_VERSION, "V1.1"),
|
||
|
},
|
||
|
},
|
||
|
{ }
|
||
|
};
|
||
|
|
||
|
static int anx7428_probe(struct i2c_client *client)
|
||
|
{
|
||
|
struct device *dev = &client->dev;
|
||
|
struct gpio_desc *gpio;
|
||
|
int ret, val;
|
||
|
|
||
|
if (!dmi_check_system(meegopad_anx7428_ids) && !force) {
|
||
|
dev_warn(dev, "Not probing unknown board, pass meegopad_anx7428.force=1 to probe");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
ret = devm_acpi_dev_add_driver_gpios(dev, meegopad_anx7428_gpios);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
/*
|
||
|
* Set GPIOs to desired values while getting them, they are not needed
|
||
|
* afterwards. Ordering and delays come from android_kernel_lge_msm8996.
|
||
|
*/
|
||
|
gpio = devm_gpiod_get(dev, "enable", GPIOD_OUT_HIGH);
|
||
|
if (IS_ERR(gpio))
|
||
|
return dev_err_probe(dev, PTR_ERR(gpio), "getting enable GPIO\n");
|
||
|
|
||
|
fsleep(10000);
|
||
|
|
||
|
gpio = devm_gpiod_get(dev, "reset", GPIOD_OUT_LOW);
|
||
|
if (IS_ERR(gpio))
|
||
|
return dev_err_probe(dev, PTR_ERR(gpio), "getting reset GPIO\n");
|
||
|
|
||
|
/* Wait for the OCM (On Chip Microcontroller) to start */
|
||
|
ret = read_poll_timeout(i2c_smbus_read_byte_data, val,
|
||
|
val >= 0 && (val & OCM_STARTUP),
|
||
|
5000, 50000, true, client, TX_STATUS);
|
||
|
if (ret)
|
||
|
return dev_err_probe(dev, ret,
|
||
|
"On Chip Microcontroller did not start, status: 0x%02x\n",
|
||
|
val);
|
||
|
|
||
|
ret = i2c_smbus_read_word_data(client, VENDOR_ID);
|
||
|
if (ret < 0)
|
||
|
return dev_err_probe(dev, ret, "reading vendor-id register\n");
|
||
|
val = ret;
|
||
|
|
||
|
ret = i2c_smbus_read_word_data(client, DEVICE_ID);
|
||
|
if (ret < 0)
|
||
|
return dev_err_probe(dev, ret, "reading device-id register\n");
|
||
|
|
||
|
dev_dbg(dev, "Powered on ANX7428 id %04x:%04x\n", val, ret);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static const struct acpi_device_id anx7428_acpi_match[] = {
|
||
|
{ "ANXO7418" }, /* ACPI says 7418 (max 2 DP lanes version) but HW is 7428 */
|
||
|
{ }
|
||
|
};
|
||
|
MODULE_DEVICE_TABLE(acpi, anx7428_acpi_match);
|
||
|
|
||
|
static struct i2c_driver anx7428_driver = {
|
||
|
.driver = {
|
||
|
.name = "meegopad_anx7428",
|
||
|
.acpi_match_table = anx7428_acpi_match,
|
||
|
},
|
||
|
.probe = anx7428_probe,
|
||
|
};
|
||
|
module_i2c_driver(anx7428_driver);
|
||
|
|
||
|
MODULE_AUTHOR("Hans de Goede <hansg@kernel.org>");
|
||
|
MODULE_DESCRIPTION("MeeGoPad ANX7428 driver");
|
||
|
MODULE_LICENSE("GPL");
|