cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

tegra_cec.c (12587B)


      1// SPDX-License-Identifier: GPL-2.0-only
      2/*
      3 * Tegra CEC implementation
      4 *
      5 * The original 3.10 CEC driver using a custom API:
      6 *
      7 * Copyright (c) 2012-2015, NVIDIA CORPORATION.  All rights reserved.
      8 *
      9 * Conversion to the CEC framework and to the mainline kernel:
     10 *
     11 * Copyright 2016-2017 Cisco Systems, Inc. and/or its affiliates. All rights reserved.
     12 */
     13
     14#include <linux/module.h>
     15#include <linux/kernel.h>
     16#include <linux/err.h>
     17#include <linux/errno.h>
     18#include <linux/interrupt.h>
     19#include <linux/slab.h>
     20#include <linux/io.h>
     21#include <linux/clk.h>
     22#include <linux/delay.h>
     23#include <linux/pm.h>
     24#include <linux/of.h>
     25#include <linux/of_platform.h>
     26#include <linux/platform_device.h>
     27#include <linux/clk/tegra.h>
     28
     29#include <media/cec-notifier.h>
     30
     31#include "tegra_cec.h"
     32
     33#define TEGRA_CEC_NAME "tegra-cec"
     34
     35struct tegra_cec {
     36	struct cec_adapter	*adap;
     37	struct device		*dev;
     38	struct clk		*clk;
     39	void __iomem		*cec_base;
     40	struct cec_notifier	*notifier;
     41	int			tegra_cec_irq;
     42	bool			rx_done;
     43	bool			tx_done;
     44	int			tx_status;
     45	u8			rx_buf[CEC_MAX_MSG_SIZE];
     46	u8			rx_buf_cnt;
     47	u32			tx_buf[CEC_MAX_MSG_SIZE];
     48	u8			tx_buf_cur;
     49	u8			tx_buf_cnt;
     50};
     51
     52static inline u32 cec_read(struct tegra_cec *cec, u32 reg)
     53{
     54	return readl(cec->cec_base + reg);
     55}
     56
     57static inline void cec_write(struct tegra_cec *cec, u32 reg, u32 val)
     58{
     59	writel(val, cec->cec_base + reg);
     60}
     61
     62static void tegra_cec_error_recovery(struct tegra_cec *cec)
     63{
     64	u32 hw_ctrl;
     65
     66	hw_ctrl = cec_read(cec, TEGRA_CEC_HW_CONTROL);
     67	cec_write(cec, TEGRA_CEC_HW_CONTROL, 0);
     68	cec_write(cec, TEGRA_CEC_INT_STAT, 0xffffffff);
     69	cec_write(cec, TEGRA_CEC_HW_CONTROL, hw_ctrl);
     70}
     71
     72static irqreturn_t tegra_cec_irq_thread_handler(int irq, void *data)
     73{
     74	struct device *dev = data;
     75	struct tegra_cec *cec = dev_get_drvdata(dev);
     76
     77	if (cec->tx_done) {
     78		cec_transmit_attempt_done(cec->adap, cec->tx_status);
     79		cec->tx_done = false;
     80	}
     81	if (cec->rx_done) {
     82		struct cec_msg msg = {};
     83
     84		msg.len = cec->rx_buf_cnt;
     85		memcpy(msg.msg, cec->rx_buf, msg.len);
     86		cec_received_msg(cec->adap, &msg);
     87		cec->rx_done = false;
     88		cec->rx_buf_cnt = 0;
     89	}
     90	return IRQ_HANDLED;
     91}
     92
     93static irqreturn_t tegra_cec_irq_handler(int irq, void *data)
     94{
     95	struct device *dev = data;
     96	struct tegra_cec *cec = dev_get_drvdata(dev);
     97	u32 status, mask;
     98
     99	status = cec_read(cec, TEGRA_CEC_INT_STAT);
    100	mask = cec_read(cec, TEGRA_CEC_INT_MASK);
    101
    102	status &= mask;
    103
    104	if (!status)
    105		return IRQ_HANDLED;
    106
    107	if (status & TEGRA_CEC_INT_STAT_TX_REGISTER_UNDERRUN) {
    108		dev_err(dev, "TX underrun, interrupt timing issue!\n");
    109
    110		tegra_cec_error_recovery(cec);
    111		cec_write(cec, TEGRA_CEC_INT_MASK,
    112			  mask & ~TEGRA_CEC_INT_MASK_TX_REGISTER_EMPTY);
    113
    114		cec->tx_done = true;
    115		cec->tx_status = CEC_TX_STATUS_ERROR;
    116		return IRQ_WAKE_THREAD;
    117	}
    118
    119	if ((status & TEGRA_CEC_INT_STAT_TX_ARBITRATION_FAILED) ||
    120		   (status & TEGRA_CEC_INT_STAT_TX_BUS_ANOMALY_DETECTED)) {
    121		tegra_cec_error_recovery(cec);
    122		cec_write(cec, TEGRA_CEC_INT_MASK,
    123			  mask & ~TEGRA_CEC_INT_MASK_TX_REGISTER_EMPTY);
    124
    125		cec->tx_done = true;
    126		if (status & TEGRA_CEC_INT_STAT_TX_BUS_ANOMALY_DETECTED)
    127			cec->tx_status = CEC_TX_STATUS_LOW_DRIVE;
    128		else
    129			cec->tx_status = CEC_TX_STATUS_ARB_LOST;
    130		return IRQ_WAKE_THREAD;
    131	}
    132
    133	if (status & TEGRA_CEC_INT_STAT_TX_FRAME_TRANSMITTED) {
    134		cec_write(cec, TEGRA_CEC_INT_STAT,
    135			  TEGRA_CEC_INT_STAT_TX_FRAME_TRANSMITTED);
    136
    137		if (status & TEGRA_CEC_INT_STAT_TX_FRAME_OR_BLOCK_NAKD) {
    138			tegra_cec_error_recovery(cec);
    139
    140			cec->tx_done = true;
    141			cec->tx_status = CEC_TX_STATUS_NACK;
    142		} else {
    143			cec->tx_done = true;
    144			cec->tx_status = CEC_TX_STATUS_OK;
    145		}
    146		return IRQ_WAKE_THREAD;
    147	}
    148
    149	if (status & TEGRA_CEC_INT_STAT_TX_FRAME_OR_BLOCK_NAKD)
    150		dev_warn(dev, "TX NAKed on the fly!\n");
    151
    152	if (status & TEGRA_CEC_INT_STAT_TX_REGISTER_EMPTY) {
    153		if (cec->tx_buf_cur == cec->tx_buf_cnt) {
    154			cec_write(cec, TEGRA_CEC_INT_MASK,
    155				  mask & ~TEGRA_CEC_INT_MASK_TX_REGISTER_EMPTY);
    156		} else {
    157			cec_write(cec, TEGRA_CEC_TX_REGISTER,
    158				  cec->tx_buf[cec->tx_buf_cur++]);
    159			cec_write(cec, TEGRA_CEC_INT_STAT,
    160				  TEGRA_CEC_INT_STAT_TX_REGISTER_EMPTY);
    161		}
    162	}
    163
    164	if (status & TEGRA_CEC_INT_STAT_RX_START_BIT_DETECTED) {
    165		cec_write(cec, TEGRA_CEC_INT_STAT,
    166			  TEGRA_CEC_INT_STAT_RX_START_BIT_DETECTED);
    167		cec->rx_done = false;
    168		cec->rx_buf_cnt = 0;
    169	}
    170	if (status & TEGRA_CEC_INT_STAT_RX_REGISTER_FULL) {
    171		u32 v;
    172
    173		cec_write(cec, TEGRA_CEC_INT_STAT,
    174			  TEGRA_CEC_INT_STAT_RX_REGISTER_FULL);
    175		v = cec_read(cec, TEGRA_CEC_RX_REGISTER);
    176		if (cec->rx_buf_cnt < CEC_MAX_MSG_SIZE)
    177			cec->rx_buf[cec->rx_buf_cnt++] = v & 0xff;
    178		if (v & TEGRA_CEC_RX_REGISTER_EOM) {
    179			cec->rx_done = true;
    180			return IRQ_WAKE_THREAD;
    181		}
    182	}
    183
    184	return IRQ_HANDLED;
    185}
    186
    187static int tegra_cec_adap_enable(struct cec_adapter *adap, bool enable)
    188{
    189	struct tegra_cec *cec = adap->priv;
    190
    191	cec->rx_buf_cnt = 0;
    192	cec->tx_buf_cnt = 0;
    193	cec->tx_buf_cur = 0;
    194
    195	cec_write(cec, TEGRA_CEC_HW_CONTROL, 0);
    196	cec_write(cec, TEGRA_CEC_INT_MASK, 0);
    197	cec_write(cec, TEGRA_CEC_INT_STAT, 0xffffffff);
    198	cec_write(cec, TEGRA_CEC_SW_CONTROL, 0);
    199
    200	if (!enable)
    201		return 0;
    202
    203	cec_write(cec, TEGRA_CEC_INPUT_FILTER, (1U << 31) | 0x20);
    204
    205	cec_write(cec, TEGRA_CEC_RX_TIMING_0,
    206		  (0x7a << TEGRA_CEC_RX_TIM0_START_BIT_MAX_LO_TIME_SHIFT) |
    207		  (0x6d << TEGRA_CEC_RX_TIM0_START_BIT_MIN_LO_TIME_SHIFT) |
    208		  (0x93 << TEGRA_CEC_RX_TIM0_START_BIT_MAX_DURATION_SHIFT) |
    209		  (0x86 << TEGRA_CEC_RX_TIM0_START_BIT_MIN_DURATION_SHIFT));
    210
    211	cec_write(cec, TEGRA_CEC_RX_TIMING_1,
    212		  (0x35 << TEGRA_CEC_RX_TIM1_DATA_BIT_MAX_LO_TIME_SHIFT) |
    213		  (0x21 << TEGRA_CEC_RX_TIM1_DATA_BIT_SAMPLE_TIME_SHIFT) |
    214		  (0x56 << TEGRA_CEC_RX_TIM1_DATA_BIT_MAX_DURATION_SHIFT) |
    215		  (0x40 << TEGRA_CEC_RX_TIM1_DATA_BIT_MIN_DURATION_SHIFT));
    216
    217	cec_write(cec, TEGRA_CEC_RX_TIMING_2,
    218		  (0x50 << TEGRA_CEC_RX_TIM2_END_OF_BLOCK_TIME_SHIFT));
    219
    220	cec_write(cec, TEGRA_CEC_TX_TIMING_0,
    221		  (0x74 << TEGRA_CEC_TX_TIM0_START_BIT_LO_TIME_SHIFT) |
    222		  (0x8d << TEGRA_CEC_TX_TIM0_START_BIT_DURATION_SHIFT) |
    223		  (0x08 << TEGRA_CEC_TX_TIM0_BUS_XITION_TIME_SHIFT) |
    224		  (0x71 << TEGRA_CEC_TX_TIM0_BUS_ERROR_LO_TIME_SHIFT));
    225
    226	cec_write(cec, TEGRA_CEC_TX_TIMING_1,
    227		  (0x2f << TEGRA_CEC_TX_TIM1_LO_DATA_BIT_LO_TIME_SHIFT) |
    228		  (0x13 << TEGRA_CEC_TX_TIM1_HI_DATA_BIT_LO_TIME_SHIFT) |
    229		  (0x4b << TEGRA_CEC_TX_TIM1_DATA_BIT_DURATION_SHIFT) |
    230		  (0x21 << TEGRA_CEC_TX_TIM1_ACK_NAK_BIT_SAMPLE_TIME_SHIFT));
    231
    232	cec_write(cec, TEGRA_CEC_TX_TIMING_2,
    233		  (0x07 << TEGRA_CEC_TX_TIM2_BUS_IDLE_TIME_ADDITIONAL_FRAME_SHIFT) |
    234		  (0x05 << TEGRA_CEC_TX_TIM2_BUS_IDLE_TIME_NEW_FRAME_SHIFT) |
    235		  (0x03 << TEGRA_CEC_TX_TIM2_BUS_IDLE_TIME_RETRY_FRAME_SHIFT));
    236
    237	cec_write(cec, TEGRA_CEC_INT_MASK,
    238		  TEGRA_CEC_INT_MASK_TX_REGISTER_UNDERRUN |
    239		  TEGRA_CEC_INT_MASK_TX_FRAME_OR_BLOCK_NAKD |
    240		  TEGRA_CEC_INT_MASK_TX_ARBITRATION_FAILED |
    241		  TEGRA_CEC_INT_MASK_TX_BUS_ANOMALY_DETECTED |
    242		  TEGRA_CEC_INT_MASK_TX_FRAME_TRANSMITTED |
    243		  TEGRA_CEC_INT_MASK_RX_REGISTER_FULL |
    244		  TEGRA_CEC_INT_MASK_RX_START_BIT_DETECTED);
    245
    246	cec_write(cec, TEGRA_CEC_HW_CONTROL, TEGRA_CEC_HWCTRL_TX_RX_MODE);
    247	return 0;
    248}
    249
    250static int tegra_cec_adap_log_addr(struct cec_adapter *adap, u8 logical_addr)
    251{
    252	struct tegra_cec *cec = adap->priv;
    253	u32 state = cec_read(cec, TEGRA_CEC_HW_CONTROL);
    254
    255	if (logical_addr == CEC_LOG_ADDR_INVALID)
    256		state &= ~TEGRA_CEC_HWCTRL_RX_LADDR_MASK;
    257	else
    258		state |= TEGRA_CEC_HWCTRL_RX_LADDR((1 << logical_addr));
    259
    260	cec_write(cec, TEGRA_CEC_HW_CONTROL, state);
    261	return 0;
    262}
    263
    264static int tegra_cec_adap_monitor_all_enable(struct cec_adapter *adap,
    265					     bool enable)
    266{
    267	struct tegra_cec *cec = adap->priv;
    268	u32 reg = cec_read(cec, TEGRA_CEC_HW_CONTROL);
    269
    270	if (enable)
    271		reg |= TEGRA_CEC_HWCTRL_RX_SNOOP;
    272	else
    273		reg &= ~TEGRA_CEC_HWCTRL_RX_SNOOP;
    274	cec_write(cec, TEGRA_CEC_HW_CONTROL, reg);
    275	return 0;
    276}
    277
    278static int tegra_cec_adap_transmit(struct cec_adapter *adap, u8 attempts,
    279				   u32 signal_free_time_ms, struct cec_msg *msg)
    280{
    281	bool retry_xfer = signal_free_time_ms == CEC_SIGNAL_FREE_TIME_RETRY;
    282	struct tegra_cec *cec = adap->priv;
    283	unsigned int i;
    284	u32 mode = 0;
    285	u32 mask;
    286
    287	if (cec_msg_is_broadcast(msg))
    288		mode = TEGRA_CEC_TX_REG_BCAST;
    289
    290	cec->tx_buf_cur = 0;
    291	cec->tx_buf_cnt = msg->len;
    292
    293	for (i = 0; i < msg->len; i++) {
    294		cec->tx_buf[i] = mode | msg->msg[i];
    295		if (i == 0)
    296			cec->tx_buf[i] |= TEGRA_CEC_TX_REG_START_BIT;
    297		if (i == msg->len - 1)
    298			cec->tx_buf[i] |= TEGRA_CEC_TX_REG_EOM;
    299		if (i == 0 && retry_xfer)
    300			cec->tx_buf[i] |= TEGRA_CEC_TX_REG_RETRY;
    301	}
    302
    303	mask = cec_read(cec, TEGRA_CEC_INT_MASK);
    304	cec_write(cec, TEGRA_CEC_INT_MASK,
    305		  mask | TEGRA_CEC_INT_MASK_TX_REGISTER_EMPTY);
    306
    307	return 0;
    308}
    309
    310static const struct cec_adap_ops tegra_cec_ops = {
    311	.adap_enable = tegra_cec_adap_enable,
    312	.adap_log_addr = tegra_cec_adap_log_addr,
    313	.adap_transmit = tegra_cec_adap_transmit,
    314	.adap_monitor_all_enable = tegra_cec_adap_monitor_all_enable,
    315};
    316
    317static int tegra_cec_probe(struct platform_device *pdev)
    318{
    319	struct device *hdmi_dev;
    320	struct tegra_cec *cec;
    321	struct resource *res;
    322	int ret = 0;
    323
    324	hdmi_dev = cec_notifier_parse_hdmi_phandle(&pdev->dev);
    325
    326	if (IS_ERR(hdmi_dev))
    327		return PTR_ERR(hdmi_dev);
    328
    329	cec = devm_kzalloc(&pdev->dev, sizeof(struct tegra_cec), GFP_KERNEL);
    330
    331	if (!cec)
    332		return -ENOMEM;
    333
    334	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
    335
    336	if (!res) {
    337		dev_err(&pdev->dev,
    338			"Unable to allocate resources for device\n");
    339		return -EBUSY;
    340	}
    341
    342	if (!devm_request_mem_region(&pdev->dev, res->start, resource_size(res),
    343		pdev->name)) {
    344		dev_err(&pdev->dev,
    345			"Unable to request mem region for device\n");
    346		return -EBUSY;
    347	}
    348
    349	cec->tegra_cec_irq = platform_get_irq(pdev, 0);
    350
    351	if (cec->tegra_cec_irq <= 0)
    352		return -EBUSY;
    353
    354	cec->cec_base = devm_ioremap(&pdev->dev, res->start,
    355					     resource_size(res));
    356
    357	if (!cec->cec_base) {
    358		dev_err(&pdev->dev, "Unable to grab IOs for device\n");
    359		return -EBUSY;
    360	}
    361
    362	cec->clk = devm_clk_get(&pdev->dev, "cec");
    363
    364	if (IS_ERR_OR_NULL(cec->clk)) {
    365		dev_err(&pdev->dev, "Can't get clock for CEC\n");
    366		return -ENOENT;
    367	}
    368
    369	ret = clk_prepare_enable(cec->clk);
    370	if (ret) {
    371		dev_err(&pdev->dev, "Unable to prepare clock for CEC\n");
    372		return ret;
    373	}
    374
    375	/* set context info. */
    376	cec->dev = &pdev->dev;
    377
    378	platform_set_drvdata(pdev, cec);
    379
    380	ret = devm_request_threaded_irq(&pdev->dev, cec->tegra_cec_irq,
    381		tegra_cec_irq_handler, tegra_cec_irq_thread_handler,
    382		0, "cec_irq", &pdev->dev);
    383
    384	if (ret) {
    385		dev_err(&pdev->dev,
    386			"Unable to request interrupt for device\n");
    387		goto err_clk;
    388	}
    389
    390	cec->adap = cec_allocate_adapter(&tegra_cec_ops, cec, TEGRA_CEC_NAME,
    391			CEC_CAP_DEFAULTS | CEC_CAP_MONITOR_ALL |
    392			CEC_CAP_CONNECTOR_INFO,
    393			CEC_MAX_LOG_ADDRS);
    394	if (IS_ERR(cec->adap)) {
    395		ret = -ENOMEM;
    396		dev_err(&pdev->dev, "Couldn't create cec adapter\n");
    397		goto err_clk;
    398	}
    399
    400	cec->notifier = cec_notifier_cec_adap_register(hdmi_dev, NULL,
    401						       cec->adap);
    402	if (!cec->notifier) {
    403		ret = -ENOMEM;
    404		goto err_adapter;
    405	}
    406
    407	ret = cec_register_adapter(cec->adap, &pdev->dev);
    408	if (ret) {
    409		dev_err(&pdev->dev, "Couldn't register device\n");
    410		goto err_notifier;
    411	}
    412
    413	return 0;
    414
    415err_notifier:
    416	cec_notifier_cec_adap_unregister(cec->notifier, cec->adap);
    417err_adapter:
    418	cec_delete_adapter(cec->adap);
    419err_clk:
    420	clk_disable_unprepare(cec->clk);
    421	return ret;
    422}
    423
    424static int tegra_cec_remove(struct platform_device *pdev)
    425{
    426	struct tegra_cec *cec = platform_get_drvdata(pdev);
    427
    428	clk_disable_unprepare(cec->clk);
    429
    430	cec_notifier_cec_adap_unregister(cec->notifier, cec->adap);
    431	cec_unregister_adapter(cec->adap);
    432
    433	return 0;
    434}
    435
    436#ifdef CONFIG_PM
    437static int tegra_cec_suspend(struct platform_device *pdev, pm_message_t state)
    438{
    439	struct tegra_cec *cec = platform_get_drvdata(pdev);
    440
    441	clk_disable_unprepare(cec->clk);
    442
    443	dev_notice(&pdev->dev, "suspended\n");
    444	return 0;
    445}
    446
    447static int tegra_cec_resume(struct platform_device *pdev)
    448{
    449	struct tegra_cec *cec = platform_get_drvdata(pdev);
    450
    451	dev_notice(&pdev->dev, "Resuming\n");
    452
    453	return clk_prepare_enable(cec->clk);
    454}
    455#endif
    456
    457static const struct of_device_id tegra_cec_of_match[] = {
    458	{ .compatible = "nvidia,tegra114-cec", },
    459	{ .compatible = "nvidia,tegra124-cec", },
    460	{ .compatible = "nvidia,tegra210-cec", },
    461	{},
    462};
    463
    464static struct platform_driver tegra_cec_driver = {
    465	.driver = {
    466		.name = TEGRA_CEC_NAME,
    467		.of_match_table = of_match_ptr(tegra_cec_of_match),
    468	},
    469	.probe = tegra_cec_probe,
    470	.remove = tegra_cec_remove,
    471
    472#ifdef CONFIG_PM
    473	.suspend = tegra_cec_suspend,
    474	.resume = tegra_cec_resume,
    475#endif
    476};
    477
    478module_platform_driver(tegra_cec_driver);
    479
    480MODULE_DESCRIPTION("Tegra HDMI CEC driver");
    481MODULE_AUTHOR("NVIDIA CORPORATION");
    482MODULE_AUTHOR("Cisco Systems, Inc. and/or its affiliates");
    483MODULE_LICENSE("GPL v2");