Loading...
// SPDX-License-Identifier: GPL-2.0+
/*
 * Copyright (c) 2022 Svyatoslav Ryhel <clamor95@gmail.com>
 */

#include <dm.h>
#include <i2c.h>
#include <linux/delay.h>
#include <linux/err.h>
#include <log.h>
#include <extcon.h>
#include <asm/gpio.h>

#define CONTROL_1	0x01
#define SW_CONTROL	0x03

#define ID_200		0x10
#define ADC_EN		0x02
#define CP_EN		0x01

#define DP_USB		0x00
#define DP_UART		0x08
#define DP_AUDIO	0x10
#define DP_OPEN		0x38

#define DM_USB		0x00
#define DM_UART		0x01
#define DM_AUDIO	0x02
#define DM_OPEN		0x07

#define AP_USB		BIT(0)
#define CP_USB		BIT(1)
#define CP_UART		BIT(2)

struct max14526_priv {
	struct gpio_desc usif_gpio;
	struct gpio_desc dp2t_gpio;
	struct gpio_desc ifx_usb_vbus_gpio;
};

static void max14526_set_mode(struct udevice *dev, int mode)
{
	struct max14526_priv *priv = dev_get_priv(dev);
	int ret;

	if ((mode & AP_USB) || (mode & CP_USB)) {
		/* Connect CP UART signals to AP */
		ret = dm_gpio_set_value(&priv->usif_gpio, 0);
		if (ret)
			log_debug("cp-uart > ap failed (%d)\n", ret);
	}

	if (mode & CP_UART) {
		/* Connect CP UART signals to DP2T */
		ret = dm_gpio_set_value(&priv->usif_gpio, 1);
		if (ret)
			log_debug("cp-uart > dp2t failed (%d)\n", ret);
	}

	if (mode & CP_USB) {
		/* Connect CP USB to MUIC UART */
		ret = dm_gpio_set_value(&priv->ifx_usb_vbus_gpio, 1);
		if (ret)
			log_debug("usb-vbus-gpio enable failed (%d)\n", ret);

		ret = dm_gpio_set_value(&priv->dp2t_gpio, 1);
		if (ret)
			log_debug("cp-usb > muic-uart failed (%d)\n", ret);
	}

	if ((mode & AP_USB) || (mode & CP_UART)) {
		/* Connect CP UART to MUIC UART */
		ret = dm_gpio_set_value(&priv->dp2t_gpio, 0);
		if (ret)
			log_debug("cp-uart > muic-uart failed (%d)\n", ret);
	}

	if (mode & AP_USB) {
		/* Enables USB Path */
		ret = dm_i2c_reg_write(dev, SW_CONTROL, DP_USB | DM_USB);
		if (ret)
			log_debug("USB path set failed: %d\n", ret);
	}

	if ((mode & CP_USB) || (mode & CP_UART)) {
		/* Enables UART Path */
		ret = dm_i2c_reg_write(dev, SW_CONTROL, DP_UART | DM_UART);
		if (ret)
			log_debug("UART path set failed: %d\n", ret);
	}

	/* Enables 200K, Charger Pump, and ADC */
	ret = dm_i2c_reg_write(dev, CONTROL_1, ID_200 | ADC_EN | CP_EN);
	if (ret)
		log_debug("200K, Charger Pump, and ADC set failed: %d\n", ret);
}

static int max14526_probe(struct udevice *dev)
{
	struct max14526_priv *priv = dev_get_priv(dev);
	int ret, mode = 0;

	ret = gpio_request_by_name(dev, "usif-gpios", 0,
				   &priv->usif_gpio, GPIOD_IS_OUT);
	if (ret) {
		log_err("could not decode usif-gpios (%d)\n", ret);
		return ret;
	}

	ret = gpio_request_by_name(dev, "dp2t-gpios", 0,
				   &priv->dp2t_gpio, GPIOD_IS_OUT);
	if (ret) {
		log_err("could not decode dp2t-gpios (%d)\n", ret);
		return ret;
	}

	if (dev_read_bool(dev, "maxim,ap-usb"))
		mode |= AP_USB;

	if (dev_read_bool(dev, "maxim,cp-usb")) {
		mode |= CP_USB;

		ret = gpio_request_by_name(dev, "usb-vbus-gpios", 0,
					   &priv->ifx_usb_vbus_gpio, GPIOD_IS_OUT);
		if (ret) {
			log_err("could not decode usb-vbus-gpios (%d)\n", ret);
			return ret;
		}
	}

	if (dev_read_bool(dev, "maxim,cp-uart"))
		mode |= CP_UART;

	max14526_set_mode(dev, mode);

	return 0;
}

static const struct udevice_id max14526_ids[] = {
	{ .compatible = "maxim,max14526-muic" },
	{ }
};

U_BOOT_DRIVER(extcon_max14526) = {
	.name		= "extcon_max14526",
	.id		= UCLASS_EXTCON,
	.of_match	= max14526_ids,
	.probe		= max14526_probe,
	.priv_auto	= sizeof(struct max14526_priv),
};