Lines Matching +full:cec +full:- +full:clk
1 // SPDX-License-Identifier: GPL-2.0
3 * STIH4xx CEC driver
7 #include <linux/clk.h>
17 #include <media/cec.h>
18 #include <media/cec-notifier.h>
20 #define CEC_NAME "stih-cec"
22 /* CEC registers */
126 struct clk *clk; member
135 struct stih_cec *cec = cec_get_drvdata(adap); in stih_cec_adap_enable() local
139 unsigned long clk_freq = clk_get_rate(cec->clk); in stih_cec_adap_enable()
142 writel(cec_clk_div, cec->regs + CEC_CLK_DIV); in stih_cec_adap_enable()
146 cec->regs + CEC_BIT_TOUT_THRESH); in stih_cec_adap_enable()
150 cec->regs + CEC_BIT_PULSE_THRESH); in stih_cec_adap_enable()
153 writel(BIT(5) | BIT(7), cec->regs + CEC_TX_CTRL); in stih_cec_adap_enable()
157 cec->regs + CEC_DATA_ARRAY_CTRL); in stih_cec_adap_enable()
159 /* Configuration of the control bits for CEC Transceiver */ in stih_cec_adap_enable()
161 cec->regs + CEC_CTRL); in stih_cec_adap_enable()
164 writel(0, cec->regs + CEC_ADDR_TABLE); in stih_cec_adap_enable()
167 writel(0x0, cec->regs + CEC_STATUS); in stih_cec_adap_enable()
173 cec->regs + CEC_IRQ_CTRL); in stih_cec_adap_enable()
177 writel(0, cec->regs + CEC_ADDR_TABLE); in stih_cec_adap_enable()
180 writel(0x0, cec->regs + CEC_STATUS); in stih_cec_adap_enable()
183 writel(0, cec->regs + CEC_IRQ_CTRL); in stih_cec_adap_enable()
191 struct stih_cec *cec = cec_get_drvdata(adap); in stih_cec_adap_log_addr() local
192 u32 reg = readl(cec->regs + CEC_ADDR_TABLE); in stih_cec_adap_log_addr()
199 writel(reg, cec->regs + CEC_ADDR_TABLE); in stih_cec_adap_log_addr()
207 struct stih_cec *cec = cec_get_drvdata(adap); in stih_cec_adap_transmit() local
211 for (i = 0; i < msg->len; i++) in stih_cec_adap_transmit()
212 writeb(msg->msg[i], cec->regs + CEC_TX_DATA_BASE + i); in stih_cec_adap_transmit()
219 msg->len, cec->regs + CEC_TX_ARRAY_CTRL); in stih_cec_adap_transmit()
224 static void stih_tx_done(struct stih_cec *cec, u32 status) in stih_tx_done() argument
227 cec_transmit_attempt_done(cec->adap, CEC_TX_STATUS_ERROR); in stih_tx_done()
232 cec_transmit_attempt_done(cec->adap, CEC_TX_STATUS_ARB_LOST); in stih_tx_done()
237 cec_transmit_attempt_done(cec->adap, CEC_TX_STATUS_NACK); in stih_tx_done()
241 cec_transmit_attempt_done(cec->adap, CEC_TX_STATUS_OK); in stih_tx_done()
244 static void stih_rx_done(struct stih_cec *cec, u32 status) in stih_rx_done() argument
255 msg.len = readl(cec->regs + CEC_DATA_ARRAY_STATUS) & 0x1f; in stih_rx_done()
264 msg.msg[i] = readl(cec->regs + CEC_RX_DATA_BASE + i); in stih_rx_done()
266 cec_received_msg(cec->adap, &msg); in stih_rx_done()
271 struct stih_cec *cec = priv; in stih_cec_irq_handler_thread() local
273 if (cec->irq_status & CEC_TX_DONE_STS) in stih_cec_irq_handler_thread()
274 stih_tx_done(cec, cec->irq_status); in stih_cec_irq_handler_thread()
276 if (cec->irq_status & CEC_RX_DONE_STS) in stih_cec_irq_handler_thread()
277 stih_rx_done(cec, cec->irq_status); in stih_cec_irq_handler_thread()
279 cec->irq_status = 0; in stih_cec_irq_handler_thread()
286 struct stih_cec *cec = priv; in stih_cec_irq_handler() local
288 cec->irq_status = readl(cec->regs + CEC_STATUS); in stih_cec_irq_handler()
289 writel(cec->irq_status, cec->regs + CEC_STATUS); in stih_cec_irq_handler()
302 struct device *dev = &pdev->dev; in stih_cec_probe()
303 struct stih_cec *cec; in stih_cec_probe() local
312 cec = devm_kzalloc(dev, sizeof(*cec), GFP_KERNEL); in stih_cec_probe()
313 if (!cec) in stih_cec_probe()
314 return -ENOMEM; in stih_cec_probe()
316 cec->dev = dev; in stih_cec_probe()
318 cec->regs = devm_platform_ioremap_resource(pdev, 0); in stih_cec_probe()
319 if (IS_ERR(cec->regs)) in stih_cec_probe()
320 return PTR_ERR(cec->regs); in stih_cec_probe()
322 cec->irq = platform_get_irq(pdev, 0); in stih_cec_probe()
323 if (cec->irq < 0) in stih_cec_probe()
324 return cec->irq; in stih_cec_probe()
326 ret = devm_request_threaded_irq(dev, cec->irq, stih_cec_irq_handler, in stih_cec_probe()
328 pdev->name, cec); in stih_cec_probe()
332 cec->clk = devm_clk_get(dev, "cec-clk"); in stih_cec_probe()
333 if (IS_ERR(cec->clk)) { in stih_cec_probe()
334 dev_err(dev, "Cannot get cec clock\n"); in stih_cec_probe()
335 return PTR_ERR(cec->clk); in stih_cec_probe()
338 cec->adap = cec_allocate_adapter(&sti_cec_adap_ops, cec, CEC_NAME, in stih_cec_probe()
342 ret = PTR_ERR_OR_ZERO(cec->adap); in stih_cec_probe()
346 cec->notifier = cec_notifier_cec_adap_register(hdmi_dev, NULL, in stih_cec_probe()
347 cec->adap); in stih_cec_probe()
348 if (!cec->notifier) { in stih_cec_probe()
349 ret = -ENOMEM; in stih_cec_probe()
353 ret = cec_register_adapter(cec->adap, &pdev->dev); in stih_cec_probe()
357 platform_set_drvdata(pdev, cec); in stih_cec_probe()
361 cec_notifier_cec_adap_unregister(cec->notifier, cec->adap); in stih_cec_probe()
364 cec_delete_adapter(cec->adap); in stih_cec_probe()
370 struct stih_cec *cec = platform_get_drvdata(pdev); in stih_cec_remove() local
372 cec_notifier_cec_adap_unregister(cec->notifier, cec->adap); in stih_cec_remove()
373 cec_unregister_adapter(cec->adap); in stih_cec_remove()
378 .compatible = "st,stih-cec",
397 MODULE_DESCRIPTION("STIH4xx CEC driver");