Lines Matching +full:cec +full:- +full:irq
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /* drivers/media/platform/s5p-cec/s5p_cec.c
4 * Samsung S5P CEC driver
8 * This driver is based on the "cec interface driver for exynos soc" by
23 #include <media/cec.h>
24 #include <media/cec-notifier.h>
27 #include "regs-cec.h"
30 #define CEC_NAME "s5p-cec"
34 MODULE_PARM_DESC(debug, "debug level (0-2)");
39 struct s5p_cec_dev *cec = cec_get_drvdata(adap); in s5p_cec_adap_enable() local
42 ret = pm_runtime_resume_and_get(cec->dev); in s5p_cec_adap_enable()
46 s5p_cec_reset(cec); in s5p_cec_adap_enable()
48 s5p_cec_set_divider(cec); in s5p_cec_adap_enable()
49 s5p_cec_threshold(cec); in s5p_cec_adap_enable()
51 s5p_cec_unmask_tx_interrupts(cec); in s5p_cec_adap_enable()
52 s5p_cec_unmask_rx_interrupts(cec); in s5p_cec_adap_enable()
53 s5p_cec_enable_rx(cec); in s5p_cec_adap_enable()
55 s5p_cec_mask_tx_interrupts(cec); in s5p_cec_adap_enable()
56 s5p_cec_mask_rx_interrupts(cec); in s5p_cec_adap_enable()
57 pm_runtime_put(cec->dev); in s5p_cec_adap_enable()
65 struct s5p_cec_dev *cec = cec_get_drvdata(adap); in s5p_cec_adap_log_addr() local
67 s5p_cec_set_addr(cec, addr); in s5p_cec_adap_log_addr()
74 struct s5p_cec_dev *cec = cec_get_drvdata(adap); in s5p_cec_adap_transmit() local
80 s5p_cec_copy_packet(cec, msg->msg, msg->len, max(1, attempts - 1)); in s5p_cec_adap_transmit()
84 static irqreturn_t s5p_cec_irq_handler(int irq, void *priv) in s5p_cec_irq_handler() argument
86 struct s5p_cec_dev *cec = priv; in s5p_cec_irq_handler() local
89 status = s5p_cec_get_status(cec); in s5p_cec_irq_handler()
91 dev_dbg(cec->dev, "irq received\n"); in s5p_cec_irq_handler()
95 dev_dbg(cec->dev, "CEC_STATUS_TX_NACK set\n"); in s5p_cec_irq_handler()
96 cec->tx = STATE_NACK; in s5p_cec_irq_handler()
98 dev_dbg(cec->dev, "CEC_STATUS_TX_ERROR set\n"); in s5p_cec_irq_handler()
99 cec->tx = STATE_ERROR; in s5p_cec_irq_handler()
101 dev_dbg(cec->dev, "CEC_STATUS_TX_DONE\n"); in s5p_cec_irq_handler()
102 cec->tx = STATE_DONE; in s5p_cec_irq_handler()
104 s5p_clr_pending_tx(cec); in s5p_cec_irq_handler()
109 dev_dbg(cec->dev, "CEC_STATUS_RX_ERROR set\n"); in s5p_cec_irq_handler()
110 s5p_cec_rx_reset(cec); in s5p_cec_irq_handler()
111 s5p_cec_enable_rx(cec); in s5p_cec_irq_handler()
113 dev_dbg(cec->dev, "CEC_STATUS_RX_DONE set\n"); in s5p_cec_irq_handler()
114 if (cec->rx != STATE_IDLE) in s5p_cec_irq_handler()
115 dev_dbg(cec->dev, "Buffer overrun (worker did not process previous message)\n"); in s5p_cec_irq_handler()
116 cec->rx = STATE_BUSY; in s5p_cec_irq_handler()
117 cec->msg.len = status >> 24; in s5p_cec_irq_handler()
118 if (cec->msg.len > CEC_MAX_MSG_SIZE) in s5p_cec_irq_handler()
119 cec->msg.len = CEC_MAX_MSG_SIZE; in s5p_cec_irq_handler()
120 cec->msg.rx_status = CEC_RX_STATUS_OK; in s5p_cec_irq_handler()
121 s5p_cec_get_rx_buf(cec, cec->msg.len, in s5p_cec_irq_handler()
122 cec->msg.msg); in s5p_cec_irq_handler()
123 cec->rx = STATE_DONE; in s5p_cec_irq_handler()
124 s5p_cec_enable_rx(cec); in s5p_cec_irq_handler()
127 s5p_clr_pending_rx(cec); in s5p_cec_irq_handler()
132 static irqreturn_t s5p_cec_irq_handler_thread(int irq, void *priv) in s5p_cec_irq_handler_thread() argument
134 struct s5p_cec_dev *cec = priv; in s5p_cec_irq_handler_thread() local
136 dev_dbg(cec->dev, "irq processing thread\n"); in s5p_cec_irq_handler_thread()
137 switch (cec->tx) { in s5p_cec_irq_handler_thread()
139 cec_transmit_done(cec->adap, CEC_TX_STATUS_OK, 0, 0, 0, 0); in s5p_cec_irq_handler_thread()
140 cec->tx = STATE_IDLE; in s5p_cec_irq_handler_thread()
143 cec_transmit_done(cec->adap, in s5p_cec_irq_handler_thread()
146 cec->tx = STATE_IDLE; in s5p_cec_irq_handler_thread()
149 cec_transmit_done(cec->adap, in s5p_cec_irq_handler_thread()
152 cec->tx = STATE_IDLE; in s5p_cec_irq_handler_thread()
155 dev_err(cec->dev, "state set to busy, this should not occur here\n"); in s5p_cec_irq_handler_thread()
161 switch (cec->rx) { in s5p_cec_irq_handler_thread()
163 cec_received_msg(cec->adap, &cec->msg); in s5p_cec_irq_handler_thread()
164 cec->rx = STATE_IDLE; in s5p_cec_irq_handler_thread()
181 struct device *dev = &pdev->dev; in s5p_cec_probe()
183 struct s5p_cec_dev *cec; in s5p_cec_probe() local
184 bool needs_hpd = of_property_read_bool(pdev->dev.of_node, "needs-hpd"); in s5p_cec_probe()
192 cec = devm_kzalloc(&pdev->dev, sizeof(*cec), GFP_KERNEL); in s5p_cec_probe()
193 if (!cec) in s5p_cec_probe()
194 return -ENOMEM; in s5p_cec_probe()
196 cec->dev = dev; in s5p_cec_probe()
198 cec->irq = platform_get_irq(pdev, 0); in s5p_cec_probe()
199 if (cec->irq < 0) in s5p_cec_probe()
200 return cec->irq; in s5p_cec_probe()
202 ret = devm_request_threaded_irq(dev, cec->irq, s5p_cec_irq_handler, in s5p_cec_probe()
203 s5p_cec_irq_handler_thread, 0, pdev->name, cec); in s5p_cec_probe()
207 cec->clk = devm_clk_get(dev, "hdmicec"); in s5p_cec_probe()
208 if (IS_ERR(cec->clk)) in s5p_cec_probe()
209 return PTR_ERR(cec->clk); in s5p_cec_probe()
211 cec->pmu = syscon_regmap_lookup_by_phandle(dev->of_node, in s5p_cec_probe()
212 "samsung,syscon-phandle"); in s5p_cec_probe()
213 if (IS_ERR(cec->pmu)) in s5p_cec_probe()
214 return -EPROBE_DEFER; in s5p_cec_probe()
216 cec->reg = devm_platform_ioremap_resource(pdev, 0); in s5p_cec_probe()
217 if (IS_ERR(cec->reg)) in s5p_cec_probe()
218 return PTR_ERR(cec->reg); in s5p_cec_probe()
220 cec->adap = cec_allocate_adapter(&s5p_cec_adap_ops, cec, CEC_NAME, in s5p_cec_probe()
223 ret = PTR_ERR_OR_ZERO(cec->adap); in s5p_cec_probe()
227 cec->notifier = cec_notifier_cec_adap_register(hdmi_dev, NULL, in s5p_cec_probe()
228 cec->adap); in s5p_cec_probe()
229 if (!cec->notifier) { in s5p_cec_probe()
230 ret = -ENOMEM; in s5p_cec_probe()
234 ret = cec_register_adapter(cec->adap, &pdev->dev); in s5p_cec_probe()
238 platform_set_drvdata(pdev, cec); in s5p_cec_probe()
245 cec_notifier_cec_adap_unregister(cec->notifier, cec->adap); in s5p_cec_probe()
248 cec_delete_adapter(cec->adap); in s5p_cec_probe()
254 struct s5p_cec_dev *cec = platform_get_drvdata(pdev); in s5p_cec_remove() local
256 cec_notifier_cec_adap_unregister(cec->notifier, cec->adap); in s5p_cec_remove()
257 cec_unregister_adapter(cec->adap); in s5p_cec_remove()
258 pm_runtime_disable(&pdev->dev); in s5p_cec_remove()
263 struct s5p_cec_dev *cec = dev_get_drvdata(dev); in s5p_cec_runtime_suspend() local
265 clk_disable_unprepare(cec->clk); in s5p_cec_runtime_suspend()
271 struct s5p_cec_dev *cec = dev_get_drvdata(dev); in s5p_cec_runtime_resume() local
274 ret = clk_prepare_enable(cec->clk); in s5p_cec_runtime_resume()
289 .compatible = "samsung,s5p-cec",
309 MODULE_DESCRIPTION("Samsung S5P CEC driver");