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