18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci * (C) Copyright 2009-2010
38c2ecf20Sopenharmony_ci * Nokia Siemens Networks, michael.lawnick.ext@nsn.com
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Portions Copyright (C) 2010 - 2016 Cavium, Inc.
68c2ecf20Sopenharmony_ci *
78c2ecf20Sopenharmony_ci * This is a driver for the i2c adapter in Cavium Networks' OCTEON processors.
88c2ecf20Sopenharmony_ci *
98c2ecf20Sopenharmony_ci * This file is licensed under the terms of the GNU General Public
108c2ecf20Sopenharmony_ci * License version 2. This program is licensed "as is" without any
118c2ecf20Sopenharmony_ci * warranty of any kind, whether express or implied.
128c2ecf20Sopenharmony_ci */
138c2ecf20Sopenharmony_ci
148c2ecf20Sopenharmony_ci#include <linux/atomic.h>
158c2ecf20Sopenharmony_ci#include <linux/delay.h>
168c2ecf20Sopenharmony_ci#include <linux/i2c.h>
178c2ecf20Sopenharmony_ci#include <linux/interrupt.h>
188c2ecf20Sopenharmony_ci#include <linux/io.h>
198c2ecf20Sopenharmony_ci#include <linux/kernel.h>
208c2ecf20Sopenharmony_ci#include <linux/module.h>
218c2ecf20Sopenharmony_ci#include <linux/of.h>
228c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
238c2ecf20Sopenharmony_ci#include <linux/sched.h>
248c2ecf20Sopenharmony_ci#include <linux/slab.h>
258c2ecf20Sopenharmony_ci
268c2ecf20Sopenharmony_ci#include <asm/octeon/octeon.h>
278c2ecf20Sopenharmony_ci#include "i2c-octeon-core.h"
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ci#define DRV_NAME "i2c-octeon"
308c2ecf20Sopenharmony_ci
318c2ecf20Sopenharmony_ci/**
328c2ecf20Sopenharmony_ci * octeon_i2c_int_enable - enable the CORE interrupt
338c2ecf20Sopenharmony_ci * @i2c: The struct octeon_i2c
348c2ecf20Sopenharmony_ci *
358c2ecf20Sopenharmony_ci * The interrupt will be asserted when there is non-STAT_IDLE state in
368c2ecf20Sopenharmony_ci * the SW_TWSI_EOP_TWSI_STAT register.
378c2ecf20Sopenharmony_ci */
388c2ecf20Sopenharmony_cistatic void octeon_i2c_int_enable(struct octeon_i2c *i2c)
398c2ecf20Sopenharmony_ci{
408c2ecf20Sopenharmony_ci	octeon_i2c_write_int(i2c, TWSI_INT_CORE_EN);
418c2ecf20Sopenharmony_ci}
428c2ecf20Sopenharmony_ci
438c2ecf20Sopenharmony_ci/* disable the CORE interrupt */
448c2ecf20Sopenharmony_cistatic void octeon_i2c_int_disable(struct octeon_i2c *i2c)
458c2ecf20Sopenharmony_ci{
468c2ecf20Sopenharmony_ci	/* clear TS/ST/IFLG events */
478c2ecf20Sopenharmony_ci	octeon_i2c_write_int(i2c, 0);
488c2ecf20Sopenharmony_ci}
498c2ecf20Sopenharmony_ci
508c2ecf20Sopenharmony_ci/**
518c2ecf20Sopenharmony_ci * octeon_i2c_int_enable78 - enable the CORE interrupt
528c2ecf20Sopenharmony_ci * @i2c: The struct octeon_i2c
538c2ecf20Sopenharmony_ci *
548c2ecf20Sopenharmony_ci * The interrupt will be asserted when there is non-STAT_IDLE state in the
558c2ecf20Sopenharmony_ci * SW_TWSI_EOP_TWSI_STAT register.
568c2ecf20Sopenharmony_ci */
578c2ecf20Sopenharmony_cistatic void octeon_i2c_int_enable78(struct octeon_i2c *i2c)
588c2ecf20Sopenharmony_ci{
598c2ecf20Sopenharmony_ci	atomic_inc_return(&i2c->int_enable_cnt);
608c2ecf20Sopenharmony_ci	enable_irq(i2c->irq);
618c2ecf20Sopenharmony_ci}
628c2ecf20Sopenharmony_ci
638c2ecf20Sopenharmony_cistatic void __octeon_i2c_irq_disable(atomic_t *cnt, int irq)
648c2ecf20Sopenharmony_ci{
658c2ecf20Sopenharmony_ci	int count;
668c2ecf20Sopenharmony_ci
678c2ecf20Sopenharmony_ci	/*
688c2ecf20Sopenharmony_ci	 * The interrupt can be disabled in two places, but we only
698c2ecf20Sopenharmony_ci	 * want to make the disable_irq_nosync() call once, so keep
708c2ecf20Sopenharmony_ci	 * track with the atomic variable.
718c2ecf20Sopenharmony_ci	 */
728c2ecf20Sopenharmony_ci	count = atomic_dec_if_positive(cnt);
738c2ecf20Sopenharmony_ci	if (count >= 0)
748c2ecf20Sopenharmony_ci		disable_irq_nosync(irq);
758c2ecf20Sopenharmony_ci}
768c2ecf20Sopenharmony_ci
778c2ecf20Sopenharmony_ci/* disable the CORE interrupt */
788c2ecf20Sopenharmony_cistatic void octeon_i2c_int_disable78(struct octeon_i2c *i2c)
798c2ecf20Sopenharmony_ci{
808c2ecf20Sopenharmony_ci	__octeon_i2c_irq_disable(&i2c->int_enable_cnt, i2c->irq);
818c2ecf20Sopenharmony_ci}
828c2ecf20Sopenharmony_ci
838c2ecf20Sopenharmony_ci/**
848c2ecf20Sopenharmony_ci * octeon_i2c_hlc_int_enable78 - enable the ST interrupt
858c2ecf20Sopenharmony_ci * @i2c: The struct octeon_i2c
868c2ecf20Sopenharmony_ci *
878c2ecf20Sopenharmony_ci * The interrupt will be asserted when there is non-STAT_IDLE state in
888c2ecf20Sopenharmony_ci * the SW_TWSI_EOP_TWSI_STAT register.
898c2ecf20Sopenharmony_ci */
908c2ecf20Sopenharmony_cistatic void octeon_i2c_hlc_int_enable78(struct octeon_i2c *i2c)
918c2ecf20Sopenharmony_ci{
928c2ecf20Sopenharmony_ci	atomic_inc_return(&i2c->hlc_int_enable_cnt);
938c2ecf20Sopenharmony_ci	enable_irq(i2c->hlc_irq);
948c2ecf20Sopenharmony_ci}
958c2ecf20Sopenharmony_ci
968c2ecf20Sopenharmony_ci/* disable the ST interrupt */
978c2ecf20Sopenharmony_cistatic void octeon_i2c_hlc_int_disable78(struct octeon_i2c *i2c)
988c2ecf20Sopenharmony_ci{
998c2ecf20Sopenharmony_ci	__octeon_i2c_irq_disable(&i2c->hlc_int_enable_cnt, i2c->hlc_irq);
1008c2ecf20Sopenharmony_ci}
1018c2ecf20Sopenharmony_ci
1028c2ecf20Sopenharmony_ci/* HLC interrupt service routine */
1038c2ecf20Sopenharmony_cistatic irqreturn_t octeon_i2c_hlc_isr78(int irq, void *dev_id)
1048c2ecf20Sopenharmony_ci{
1058c2ecf20Sopenharmony_ci	struct octeon_i2c *i2c = dev_id;
1068c2ecf20Sopenharmony_ci
1078c2ecf20Sopenharmony_ci	i2c->hlc_int_disable(i2c);
1088c2ecf20Sopenharmony_ci	wake_up(&i2c->queue);
1098c2ecf20Sopenharmony_ci
1108c2ecf20Sopenharmony_ci	return IRQ_HANDLED;
1118c2ecf20Sopenharmony_ci}
1128c2ecf20Sopenharmony_ci
1138c2ecf20Sopenharmony_cistatic void octeon_i2c_hlc_int_enable(struct octeon_i2c *i2c)
1148c2ecf20Sopenharmony_ci{
1158c2ecf20Sopenharmony_ci	octeon_i2c_write_int(i2c, TWSI_INT_ST_EN);
1168c2ecf20Sopenharmony_ci}
1178c2ecf20Sopenharmony_ci
1188c2ecf20Sopenharmony_cistatic u32 octeon_i2c_functionality(struct i2c_adapter *adap)
1198c2ecf20Sopenharmony_ci{
1208c2ecf20Sopenharmony_ci	return I2C_FUNC_I2C | (I2C_FUNC_SMBUS_EMUL & ~I2C_FUNC_SMBUS_QUICK) |
1218c2ecf20Sopenharmony_ci	       I2C_FUNC_SMBUS_READ_BLOCK_DATA | I2C_SMBUS_BLOCK_PROC_CALL;
1228c2ecf20Sopenharmony_ci}
1238c2ecf20Sopenharmony_ci
1248c2ecf20Sopenharmony_cistatic const struct i2c_algorithm octeon_i2c_algo = {
1258c2ecf20Sopenharmony_ci	.master_xfer = octeon_i2c_xfer,
1268c2ecf20Sopenharmony_ci	.functionality = octeon_i2c_functionality,
1278c2ecf20Sopenharmony_ci};
1288c2ecf20Sopenharmony_ci
1298c2ecf20Sopenharmony_cistatic const struct i2c_adapter octeon_i2c_ops = {
1308c2ecf20Sopenharmony_ci	.owner = THIS_MODULE,
1318c2ecf20Sopenharmony_ci	.name = "OCTEON adapter",
1328c2ecf20Sopenharmony_ci	.algo = &octeon_i2c_algo,
1338c2ecf20Sopenharmony_ci};
1348c2ecf20Sopenharmony_ci
1358c2ecf20Sopenharmony_cistatic int octeon_i2c_probe(struct platform_device *pdev)
1368c2ecf20Sopenharmony_ci{
1378c2ecf20Sopenharmony_ci	struct device_node *node = pdev->dev.of_node;
1388c2ecf20Sopenharmony_ci	int irq, result = 0, hlc_irq = 0;
1398c2ecf20Sopenharmony_ci	struct octeon_i2c *i2c;
1408c2ecf20Sopenharmony_ci	bool cn78xx_style;
1418c2ecf20Sopenharmony_ci
1428c2ecf20Sopenharmony_ci	cn78xx_style = of_device_is_compatible(node, "cavium,octeon-7890-twsi");
1438c2ecf20Sopenharmony_ci	if (cn78xx_style) {
1448c2ecf20Sopenharmony_ci		hlc_irq = platform_get_irq(pdev, 0);
1458c2ecf20Sopenharmony_ci		if (hlc_irq < 0)
1468c2ecf20Sopenharmony_ci			return hlc_irq;
1478c2ecf20Sopenharmony_ci
1488c2ecf20Sopenharmony_ci		irq = platform_get_irq(pdev, 2);
1498c2ecf20Sopenharmony_ci		if (irq < 0)
1508c2ecf20Sopenharmony_ci			return irq;
1518c2ecf20Sopenharmony_ci	} else {
1528c2ecf20Sopenharmony_ci		/* All adaptors have an irq.  */
1538c2ecf20Sopenharmony_ci		irq = platform_get_irq(pdev, 0);
1548c2ecf20Sopenharmony_ci		if (irq < 0)
1558c2ecf20Sopenharmony_ci			return irq;
1568c2ecf20Sopenharmony_ci	}
1578c2ecf20Sopenharmony_ci
1588c2ecf20Sopenharmony_ci	i2c = devm_kzalloc(&pdev->dev, sizeof(*i2c), GFP_KERNEL);
1598c2ecf20Sopenharmony_ci	if (!i2c) {
1608c2ecf20Sopenharmony_ci		result = -ENOMEM;
1618c2ecf20Sopenharmony_ci		goto out;
1628c2ecf20Sopenharmony_ci	}
1638c2ecf20Sopenharmony_ci	i2c->dev = &pdev->dev;
1648c2ecf20Sopenharmony_ci
1658c2ecf20Sopenharmony_ci	i2c->roff.sw_twsi = 0x00;
1668c2ecf20Sopenharmony_ci	i2c->roff.twsi_int = 0x10;
1678c2ecf20Sopenharmony_ci	i2c->roff.sw_twsi_ext = 0x18;
1688c2ecf20Sopenharmony_ci
1698c2ecf20Sopenharmony_ci	i2c->twsi_base = devm_platform_ioremap_resource(pdev, 0);
1708c2ecf20Sopenharmony_ci	if (IS_ERR(i2c->twsi_base)) {
1718c2ecf20Sopenharmony_ci		result = PTR_ERR(i2c->twsi_base);
1728c2ecf20Sopenharmony_ci		goto out;
1738c2ecf20Sopenharmony_ci	}
1748c2ecf20Sopenharmony_ci
1758c2ecf20Sopenharmony_ci	/*
1768c2ecf20Sopenharmony_ci	 * "clock-rate" is a legacy binding, the official binding is
1778c2ecf20Sopenharmony_ci	 * "clock-frequency".  Try the official one first and then
1788c2ecf20Sopenharmony_ci	 * fall back if it doesn't exist.
1798c2ecf20Sopenharmony_ci	 */
1808c2ecf20Sopenharmony_ci	if (of_property_read_u32(node, "clock-frequency", &i2c->twsi_freq) &&
1818c2ecf20Sopenharmony_ci	    of_property_read_u32(node, "clock-rate", &i2c->twsi_freq)) {
1828c2ecf20Sopenharmony_ci		dev_err(i2c->dev,
1838c2ecf20Sopenharmony_ci			"no I2C 'clock-rate' or 'clock-frequency' property\n");
1848c2ecf20Sopenharmony_ci		result = -ENXIO;
1858c2ecf20Sopenharmony_ci		goto out;
1868c2ecf20Sopenharmony_ci	}
1878c2ecf20Sopenharmony_ci
1888c2ecf20Sopenharmony_ci	i2c->sys_freq = octeon_get_io_clock_rate();
1898c2ecf20Sopenharmony_ci
1908c2ecf20Sopenharmony_ci	init_waitqueue_head(&i2c->queue);
1918c2ecf20Sopenharmony_ci
1928c2ecf20Sopenharmony_ci	i2c->irq = irq;
1938c2ecf20Sopenharmony_ci
1948c2ecf20Sopenharmony_ci	if (cn78xx_style) {
1958c2ecf20Sopenharmony_ci		i2c->hlc_irq = hlc_irq;
1968c2ecf20Sopenharmony_ci
1978c2ecf20Sopenharmony_ci		i2c->int_enable = octeon_i2c_int_enable78;
1988c2ecf20Sopenharmony_ci		i2c->int_disable = octeon_i2c_int_disable78;
1998c2ecf20Sopenharmony_ci		i2c->hlc_int_enable = octeon_i2c_hlc_int_enable78;
2008c2ecf20Sopenharmony_ci		i2c->hlc_int_disable = octeon_i2c_hlc_int_disable78;
2018c2ecf20Sopenharmony_ci
2028c2ecf20Sopenharmony_ci		irq_set_status_flags(i2c->irq, IRQ_NOAUTOEN);
2038c2ecf20Sopenharmony_ci		irq_set_status_flags(i2c->hlc_irq, IRQ_NOAUTOEN);
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_ci		result = devm_request_irq(&pdev->dev, i2c->hlc_irq,
2068c2ecf20Sopenharmony_ci					  octeon_i2c_hlc_isr78, 0,
2078c2ecf20Sopenharmony_ci					  DRV_NAME, i2c);
2088c2ecf20Sopenharmony_ci		if (result < 0) {
2098c2ecf20Sopenharmony_ci			dev_err(i2c->dev, "failed to attach interrupt\n");
2108c2ecf20Sopenharmony_ci			goto out;
2118c2ecf20Sopenharmony_ci		}
2128c2ecf20Sopenharmony_ci	} else {
2138c2ecf20Sopenharmony_ci		i2c->int_enable = octeon_i2c_int_enable;
2148c2ecf20Sopenharmony_ci		i2c->int_disable = octeon_i2c_int_disable;
2158c2ecf20Sopenharmony_ci		i2c->hlc_int_enable = octeon_i2c_hlc_int_enable;
2168c2ecf20Sopenharmony_ci		i2c->hlc_int_disable = octeon_i2c_int_disable;
2178c2ecf20Sopenharmony_ci	}
2188c2ecf20Sopenharmony_ci
2198c2ecf20Sopenharmony_ci	result = devm_request_irq(&pdev->dev, i2c->irq,
2208c2ecf20Sopenharmony_ci				  octeon_i2c_isr, 0, DRV_NAME, i2c);
2218c2ecf20Sopenharmony_ci	if (result < 0) {
2228c2ecf20Sopenharmony_ci		dev_err(i2c->dev, "failed to attach interrupt\n");
2238c2ecf20Sopenharmony_ci		goto out;
2248c2ecf20Sopenharmony_ci	}
2258c2ecf20Sopenharmony_ci
2268c2ecf20Sopenharmony_ci	if (OCTEON_IS_MODEL(OCTEON_CN38XX))
2278c2ecf20Sopenharmony_ci		i2c->broken_irq_check = true;
2288c2ecf20Sopenharmony_ci
2298c2ecf20Sopenharmony_ci	result = octeon_i2c_init_lowlevel(i2c);
2308c2ecf20Sopenharmony_ci	if (result) {
2318c2ecf20Sopenharmony_ci		dev_err(i2c->dev, "init low level failed\n");
2328c2ecf20Sopenharmony_ci		goto  out;
2338c2ecf20Sopenharmony_ci	}
2348c2ecf20Sopenharmony_ci
2358c2ecf20Sopenharmony_ci	octeon_i2c_set_clock(i2c);
2368c2ecf20Sopenharmony_ci
2378c2ecf20Sopenharmony_ci	i2c->adap = octeon_i2c_ops;
2388c2ecf20Sopenharmony_ci	i2c->adap.timeout = msecs_to_jiffies(2);
2398c2ecf20Sopenharmony_ci	i2c->adap.retries = 5;
2408c2ecf20Sopenharmony_ci	i2c->adap.bus_recovery_info = &octeon_i2c_recovery_info;
2418c2ecf20Sopenharmony_ci	i2c->adap.dev.parent = &pdev->dev;
2428c2ecf20Sopenharmony_ci	i2c->adap.dev.of_node = node;
2438c2ecf20Sopenharmony_ci	i2c_set_adapdata(&i2c->adap, i2c);
2448c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, i2c);
2458c2ecf20Sopenharmony_ci
2468c2ecf20Sopenharmony_ci	result = i2c_add_adapter(&i2c->adap);
2478c2ecf20Sopenharmony_ci	if (result < 0)
2488c2ecf20Sopenharmony_ci		goto out;
2498c2ecf20Sopenharmony_ci	dev_info(i2c->dev, "probed\n");
2508c2ecf20Sopenharmony_ci	return 0;
2518c2ecf20Sopenharmony_ci
2528c2ecf20Sopenharmony_ciout:
2538c2ecf20Sopenharmony_ci	return result;
2548c2ecf20Sopenharmony_ci};
2558c2ecf20Sopenharmony_ci
2568c2ecf20Sopenharmony_cistatic int octeon_i2c_remove(struct platform_device *pdev)
2578c2ecf20Sopenharmony_ci{
2588c2ecf20Sopenharmony_ci	struct octeon_i2c *i2c = platform_get_drvdata(pdev);
2598c2ecf20Sopenharmony_ci
2608c2ecf20Sopenharmony_ci	i2c_del_adapter(&i2c->adap);
2618c2ecf20Sopenharmony_ci	return 0;
2628c2ecf20Sopenharmony_ci};
2638c2ecf20Sopenharmony_ci
2648c2ecf20Sopenharmony_cistatic const struct of_device_id octeon_i2c_match[] = {
2658c2ecf20Sopenharmony_ci	{ .compatible = "cavium,octeon-3860-twsi", },
2668c2ecf20Sopenharmony_ci	{ .compatible = "cavium,octeon-7890-twsi", },
2678c2ecf20Sopenharmony_ci	{},
2688c2ecf20Sopenharmony_ci};
2698c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, octeon_i2c_match);
2708c2ecf20Sopenharmony_ci
2718c2ecf20Sopenharmony_cistatic struct platform_driver octeon_i2c_driver = {
2728c2ecf20Sopenharmony_ci	.probe		= octeon_i2c_probe,
2738c2ecf20Sopenharmony_ci	.remove		= octeon_i2c_remove,
2748c2ecf20Sopenharmony_ci	.driver		= {
2758c2ecf20Sopenharmony_ci		.name	= DRV_NAME,
2768c2ecf20Sopenharmony_ci		.of_match_table = octeon_i2c_match,
2778c2ecf20Sopenharmony_ci	},
2788c2ecf20Sopenharmony_ci};
2798c2ecf20Sopenharmony_ci
2808c2ecf20Sopenharmony_cimodule_platform_driver(octeon_i2c_driver);
2818c2ecf20Sopenharmony_ci
2828c2ecf20Sopenharmony_ciMODULE_AUTHOR("Michael Lawnick <michael.lawnick.ext@nsn.com>");
2838c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("I2C-Bus adapter for Cavium OCTEON processors");
2848c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
285