162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * CLPS711X GPIO driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2012,2013 Alexander Shiyan <shc_work@mail.ru> 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include <linux/err.h> 962306a36Sopenharmony_ci#include <linux/module.h> 1062306a36Sopenharmony_ci#include <linux/gpio/driver.h> 1162306a36Sopenharmony_ci#include <linux/platform_device.h> 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_cistatic int clps711x_gpio_probe(struct platform_device *pdev) 1462306a36Sopenharmony_ci{ 1562306a36Sopenharmony_ci struct device_node *np = pdev->dev.of_node; 1662306a36Sopenharmony_ci void __iomem *dat, *dir; 1762306a36Sopenharmony_ci struct gpio_chip *gc; 1862306a36Sopenharmony_ci int err, id; 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci if (!np) 2162306a36Sopenharmony_ci return -ENODEV; 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci id = of_alias_get_id(np, "gpio"); 2462306a36Sopenharmony_ci if ((id < 0) || (id > 4)) 2562306a36Sopenharmony_ci return -ENODEV; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci gc = devm_kzalloc(&pdev->dev, sizeof(*gc), GFP_KERNEL); 2862306a36Sopenharmony_ci if (!gc) 2962306a36Sopenharmony_ci return -ENOMEM; 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_ci dat = devm_platform_ioremap_resource(pdev, 0); 3262306a36Sopenharmony_ci if (IS_ERR(dat)) 3362306a36Sopenharmony_ci return PTR_ERR(dat); 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_ci dir = devm_platform_ioremap_resource(pdev, 1); 3662306a36Sopenharmony_ci if (IS_ERR(dir)) 3762306a36Sopenharmony_ci return PTR_ERR(dir); 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_ci switch (id) { 4062306a36Sopenharmony_ci case 3: 4162306a36Sopenharmony_ci /* PORTD is inverted logic for direction register */ 4262306a36Sopenharmony_ci err = bgpio_init(gc, &pdev->dev, 1, dat, NULL, NULL, 4362306a36Sopenharmony_ci NULL, dir, 0); 4462306a36Sopenharmony_ci break; 4562306a36Sopenharmony_ci default: 4662306a36Sopenharmony_ci err = bgpio_init(gc, &pdev->dev, 1, dat, NULL, NULL, 4762306a36Sopenharmony_ci dir, NULL, 0); 4862306a36Sopenharmony_ci break; 4962306a36Sopenharmony_ci } 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci if (err) 5262306a36Sopenharmony_ci return err; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci switch (id) { 5562306a36Sopenharmony_ci case 4: 5662306a36Sopenharmony_ci /* PORTE is 3 lines only */ 5762306a36Sopenharmony_ci gc->ngpio = 3; 5862306a36Sopenharmony_ci break; 5962306a36Sopenharmony_ci default: 6062306a36Sopenharmony_ci break; 6162306a36Sopenharmony_ci } 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci gc->base = -1; 6462306a36Sopenharmony_ci gc->owner = THIS_MODULE; 6562306a36Sopenharmony_ci platform_set_drvdata(pdev, gc); 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci return devm_gpiochip_add_data(&pdev->dev, gc, NULL); 6862306a36Sopenharmony_ci} 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_cistatic const struct of_device_id clps711x_gpio_ids[] = { 7162306a36Sopenharmony_ci { .compatible = "cirrus,ep7209-gpio" }, 7262306a36Sopenharmony_ci { } 7362306a36Sopenharmony_ci}; 7462306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, clps711x_gpio_ids); 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_cistatic struct platform_driver clps711x_gpio_driver = { 7762306a36Sopenharmony_ci .driver = { 7862306a36Sopenharmony_ci .name = "clps711x-gpio", 7962306a36Sopenharmony_ci .of_match_table = clps711x_gpio_ids, 8062306a36Sopenharmony_ci }, 8162306a36Sopenharmony_ci .probe = clps711x_gpio_probe, 8262306a36Sopenharmony_ci}; 8362306a36Sopenharmony_cimodule_platform_driver(clps711x_gpio_driver); 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 8662306a36Sopenharmony_ciMODULE_AUTHOR("Alexander Shiyan <shc_work@mail.ru>"); 8762306a36Sopenharmony_ciMODULE_DESCRIPTION("CLPS711X GPIO driver"); 8862306a36Sopenharmony_ciMODULE_ALIAS("platform:clps711x-gpio"); 89