18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Maxim Integrated MAX3355 USB OTG chip extcon driver
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Copyright (C)  2014-2015 Cogent Embedded, Inc.
68c2ecf20Sopenharmony_ci * Author: Sergei Shtylyov <sergei.shtylyov@cogentembedded.com>
78c2ecf20Sopenharmony_ci */
88c2ecf20Sopenharmony_ci
98c2ecf20Sopenharmony_ci#include <linux/extcon-provider.h>
108c2ecf20Sopenharmony_ci#include <linux/gpio.h>
118c2ecf20Sopenharmony_ci#include <linux/gpio/consumer.h>
128c2ecf20Sopenharmony_ci#include <linux/interrupt.h>
138c2ecf20Sopenharmony_ci#include <linux/module.h>
148c2ecf20Sopenharmony_ci#include <linux/mod_devicetable.h>
158c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
168c2ecf20Sopenharmony_ci
178c2ecf20Sopenharmony_cistruct max3355_data {
188c2ecf20Sopenharmony_ci	struct extcon_dev *edev;
198c2ecf20Sopenharmony_ci	struct gpio_desc *id_gpiod;
208c2ecf20Sopenharmony_ci	struct gpio_desc *shdn_gpiod;
218c2ecf20Sopenharmony_ci};
228c2ecf20Sopenharmony_ci
238c2ecf20Sopenharmony_cistatic const unsigned int max3355_cable[] = {
248c2ecf20Sopenharmony_ci	EXTCON_USB,
258c2ecf20Sopenharmony_ci	EXTCON_USB_HOST,
268c2ecf20Sopenharmony_ci	EXTCON_NONE,
278c2ecf20Sopenharmony_ci};
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_cistatic irqreturn_t max3355_id_irq(int irq, void *dev_id)
308c2ecf20Sopenharmony_ci{
318c2ecf20Sopenharmony_ci	struct max3355_data *data = dev_id;
328c2ecf20Sopenharmony_ci	int id = gpiod_get_value_cansleep(data->id_gpiod);
338c2ecf20Sopenharmony_ci
348c2ecf20Sopenharmony_ci	if (id) {
358c2ecf20Sopenharmony_ci		/*
368c2ecf20Sopenharmony_ci		 * ID = 1 means USB HOST cable detached.
378c2ecf20Sopenharmony_ci		 * As we don't have event for USB peripheral cable attached,
388c2ecf20Sopenharmony_ci		 * we simulate USB peripheral attach here.
398c2ecf20Sopenharmony_ci		 */
408c2ecf20Sopenharmony_ci		extcon_set_state_sync(data->edev, EXTCON_USB_HOST, false);
418c2ecf20Sopenharmony_ci		extcon_set_state_sync(data->edev, EXTCON_USB, true);
428c2ecf20Sopenharmony_ci	} else {
438c2ecf20Sopenharmony_ci		/*
448c2ecf20Sopenharmony_ci		 * ID = 0 means USB HOST cable attached.
458c2ecf20Sopenharmony_ci		 * As we don't have event for USB peripheral cable detached,
468c2ecf20Sopenharmony_ci		 * we simulate USB peripheral detach here.
478c2ecf20Sopenharmony_ci		 */
488c2ecf20Sopenharmony_ci		extcon_set_state_sync(data->edev, EXTCON_USB, false);
498c2ecf20Sopenharmony_ci		extcon_set_state_sync(data->edev, EXTCON_USB_HOST, true);
508c2ecf20Sopenharmony_ci	}
518c2ecf20Sopenharmony_ci
528c2ecf20Sopenharmony_ci	return IRQ_HANDLED;
538c2ecf20Sopenharmony_ci}
548c2ecf20Sopenharmony_ci
558c2ecf20Sopenharmony_cistatic int max3355_probe(struct platform_device *pdev)
568c2ecf20Sopenharmony_ci{
578c2ecf20Sopenharmony_ci	struct max3355_data *data;
588c2ecf20Sopenharmony_ci	struct gpio_desc *gpiod;
598c2ecf20Sopenharmony_ci	int irq, err;
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci	data = devm_kzalloc(&pdev->dev, sizeof(struct max3355_data),
628c2ecf20Sopenharmony_ci			    GFP_KERNEL);
638c2ecf20Sopenharmony_ci	if (!data)
648c2ecf20Sopenharmony_ci		return -ENOMEM;
658c2ecf20Sopenharmony_ci
668c2ecf20Sopenharmony_ci	gpiod = devm_gpiod_get(&pdev->dev, "id", GPIOD_IN);
678c2ecf20Sopenharmony_ci	if (IS_ERR(gpiod)) {
688c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "failed to get ID_OUT GPIO\n");
698c2ecf20Sopenharmony_ci		return PTR_ERR(gpiod);
708c2ecf20Sopenharmony_ci	}
718c2ecf20Sopenharmony_ci	data->id_gpiod = gpiod;
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_ci	gpiod = devm_gpiod_get(&pdev->dev, "maxim,shdn", GPIOD_OUT_HIGH);
748c2ecf20Sopenharmony_ci	if (IS_ERR(gpiod)) {
758c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "failed to get SHDN# GPIO\n");
768c2ecf20Sopenharmony_ci		return PTR_ERR(gpiod);
778c2ecf20Sopenharmony_ci	}
788c2ecf20Sopenharmony_ci	data->shdn_gpiod = gpiod;
798c2ecf20Sopenharmony_ci
808c2ecf20Sopenharmony_ci	data->edev = devm_extcon_dev_allocate(&pdev->dev, max3355_cable);
818c2ecf20Sopenharmony_ci	if (IS_ERR(data->edev)) {
828c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "failed to allocate extcon device\n");
838c2ecf20Sopenharmony_ci		return PTR_ERR(data->edev);
848c2ecf20Sopenharmony_ci	}
858c2ecf20Sopenharmony_ci
868c2ecf20Sopenharmony_ci	err = devm_extcon_dev_register(&pdev->dev, data->edev);
878c2ecf20Sopenharmony_ci	if (err < 0) {
888c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "failed to register extcon device\n");
898c2ecf20Sopenharmony_ci		return err;
908c2ecf20Sopenharmony_ci	}
918c2ecf20Sopenharmony_ci
928c2ecf20Sopenharmony_ci	irq = gpiod_to_irq(data->id_gpiod);
938c2ecf20Sopenharmony_ci	if (irq < 0) {
948c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "failed to translate ID_OUT GPIO to IRQ\n");
958c2ecf20Sopenharmony_ci		return irq;
968c2ecf20Sopenharmony_ci	}
978c2ecf20Sopenharmony_ci
988c2ecf20Sopenharmony_ci	err = devm_request_threaded_irq(&pdev->dev, irq, NULL, max3355_id_irq,
998c2ecf20Sopenharmony_ci					IRQF_ONESHOT | IRQF_NO_SUSPEND |
1008c2ecf20Sopenharmony_ci					IRQF_TRIGGER_RISING |
1018c2ecf20Sopenharmony_ci					IRQF_TRIGGER_FALLING,
1028c2ecf20Sopenharmony_ci					pdev->name, data);
1038c2ecf20Sopenharmony_ci	if (err < 0) {
1048c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "failed to request ID_OUT IRQ\n");
1058c2ecf20Sopenharmony_ci		return err;
1068c2ecf20Sopenharmony_ci	}
1078c2ecf20Sopenharmony_ci
1088c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, data);
1098c2ecf20Sopenharmony_ci
1108c2ecf20Sopenharmony_ci	/* Perform initial detection */
1118c2ecf20Sopenharmony_ci	max3355_id_irq(irq, data);
1128c2ecf20Sopenharmony_ci
1138c2ecf20Sopenharmony_ci	return 0;
1148c2ecf20Sopenharmony_ci}
1158c2ecf20Sopenharmony_ci
1168c2ecf20Sopenharmony_cistatic int max3355_remove(struct platform_device *pdev)
1178c2ecf20Sopenharmony_ci{
1188c2ecf20Sopenharmony_ci	struct max3355_data *data = platform_get_drvdata(pdev);
1198c2ecf20Sopenharmony_ci
1208c2ecf20Sopenharmony_ci	gpiod_set_value_cansleep(data->shdn_gpiod, 0);
1218c2ecf20Sopenharmony_ci
1228c2ecf20Sopenharmony_ci	return 0;
1238c2ecf20Sopenharmony_ci}
1248c2ecf20Sopenharmony_ci
1258c2ecf20Sopenharmony_cistatic const struct of_device_id max3355_match_table[] = {
1268c2ecf20Sopenharmony_ci	{ .compatible = "maxim,max3355", },
1278c2ecf20Sopenharmony_ci	{ }
1288c2ecf20Sopenharmony_ci};
1298c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, max3355_match_table);
1308c2ecf20Sopenharmony_ci
1318c2ecf20Sopenharmony_cistatic struct platform_driver max3355_driver = {
1328c2ecf20Sopenharmony_ci	.probe		= max3355_probe,
1338c2ecf20Sopenharmony_ci	.remove		= max3355_remove,
1348c2ecf20Sopenharmony_ci	.driver		= {
1358c2ecf20Sopenharmony_ci		.name	= "extcon-max3355",
1368c2ecf20Sopenharmony_ci		.of_match_table = max3355_match_table,
1378c2ecf20Sopenharmony_ci	},
1388c2ecf20Sopenharmony_ci};
1398c2ecf20Sopenharmony_ci
1408c2ecf20Sopenharmony_cimodule_platform_driver(max3355_driver);
1418c2ecf20Sopenharmony_ci
1428c2ecf20Sopenharmony_ciMODULE_AUTHOR("Sergei Shtylyov <sergei.shtylyov@cogentembedded.com>");
1438c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Maxim MAX3355 extcon driver");
1448c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2");
145