Lines Matching +full:cec +full:- +full:irq

2  * Driver for Amlogic Meson AO CEC Controller
8 * SPDX-License-Identifier: GPL-2.0+
24 #include <media/cec.h>
25 #include <media/cec-notifier.h>
27 /* CEC Registers */
31 * - 0 = Disable clk (Power-off mode)
32 * - 1 = Enable gated clock (Normal mode)
33 * - 2 = Enable free-run clk (Debug mode)
47 * - 0 = Read
48 * - 1 = Write
71 /* CEC Commands */
235 while (readl_relaxed(ao_cec->base + CEC_RW_REG) & CEC_RW_BUS_BUSY) { in meson_ao_cec_wait_busy()
237 return -ETIMEDOUT; in meson_ao_cec_wait_busy()
254 spin_lock_irqsave(&ao_cec->cec_reg_lock, flags); in meson_ao_cec_read()
260 writel_relaxed(reg, ao_cec->base + CEC_RW_REG); in meson_ao_cec_read()
267 readl_relaxed(ao_cec->base + CEC_RW_REG)); in meson_ao_cec_read()
270 spin_unlock_irqrestore(&ao_cec->cec_reg_lock, flags); in meson_ao_cec_read()
289 spin_lock_irqsave(&ao_cec->cec_reg_lock, flags); in meson_ao_cec_write()
295 writel_relaxed(reg, ao_cec->base + CEC_RW_REG); in meson_ao_cec_write()
298 spin_unlock_irqrestore(&ao_cec->cec_reg_lock, flags); in meson_ao_cec_write()
310 ao_cec->base + CEC_INTR_MASKN_REG); in meson_ao_cec_irq_setup()
371 static irqreturn_t meson_ao_cec_irq(int irq, void *data) in meson_ao_cec_irq() argument
374 u32 stat = readl_relaxed(ao_cec->base + CEC_INTR_STAT_REG); in meson_ao_cec_irq()
412 writel_relaxed(CEC_INTR_TX, ao_cec->base + CEC_INTR_CLR_REG); in meson_ao_cec_irq_tx()
419 cec_transmit_attempt_done(ao_cec->adap, tx_status); in meson_ao_cec_irq_tx()
423 cec_transmit_attempt_done(ao_cec->adap, CEC_TX_STATUS_ERROR); in meson_ao_cec_irq_tx()
441 ao_cec->rx_msg.len = reg + 1; in meson_ao_cec_irq_rx()
442 if (ao_cec->rx_msg.len > CEC_MAX_MSG_SIZE) in meson_ao_cec_irq_rx()
443 ao_cec->rx_msg.len = CEC_MAX_MSG_SIZE; in meson_ao_cec_irq_rx()
445 for (i = 0; i < ao_cec->rx_msg.len; i++) { in meson_ao_cec_irq_rx()
450 ao_cec->rx_msg.msg[i] = byte; in meson_ao_cec_irq_rx()
456 cec_received_msg(ao_cec->adap, &ao_cec->rx_msg); in meson_ao_cec_irq_rx()
460 writel_relaxed(CEC_INTR_RX, ao_cec->base + CEC_INTR_CLR_REG); in meson_ao_cec_irq_rx()
471 static irqreturn_t meson_ao_cec_irq_thread(int irq, void *data) in meson_ao_cec_irq_thread() argument
474 u32 stat = readl_relaxed(ao_cec->base + CEC_INTR_STAT_REG); in meson_ao_cec_irq_thread()
486 struct meson_ao_cec_device *ao_cec = adap->priv; in meson_ao_cec_set_log_addr()
518 struct meson_ao_cec_device *ao_cec = adap->priv; in meson_ao_cec_transmit()
527 dev_dbg(&ao_cec->pdev->dev, "%s: busy TX: aborting\n", in meson_ao_cec_transmit()
532 for (i = 0; i < msg->len; i++) { in meson_ao_cec_transmit()
534 msg->msg[i], &ret); in meson_ao_cec_transmit()
537 meson_ao_cec_write(ao_cec, CEC_TX_MSG_LENGTH, msg->len - 1, &ret); in meson_ao_cec_transmit()
545 struct meson_ao_cec_device *ao_cec = adap->priv; in meson_ao_cec_adap_enable()
551 ao_cec->base + CEC_GEN_CNTL_REG); in meson_ao_cec_adap_enable()
560 ao_cec->base + CEC_GEN_CNTL_REG); in meson_ao_cec_adap_enable()
566 ao_cec->base + CEC_GEN_CNTL_REG); in meson_ao_cec_adap_enable()
573 /* CEC arbitration 3/5/7 bit time set. */ in meson_ao_cec_adap_enable()
605 int ret, irq; in meson_ao_cec_probe() local
607 hdmi_dev = cec_notifier_parse_hdmi_phandle(&pdev->dev); in meson_ao_cec_probe()
612 ao_cec = devm_kzalloc(&pdev->dev, sizeof(*ao_cec), GFP_KERNEL); in meson_ao_cec_probe()
614 return -ENOMEM; in meson_ao_cec_probe()
616 spin_lock_init(&ao_cec->cec_reg_lock); in meson_ao_cec_probe()
618 ao_cec->adap = cec_allocate_adapter(&meson_ao_cec_ops, ao_cec, in meson_ao_cec_probe()
623 if (IS_ERR(ao_cec->adap)) in meson_ao_cec_probe()
624 return PTR_ERR(ao_cec->adap); in meson_ao_cec_probe()
626 ao_cec->adap->owner = THIS_MODULE; in meson_ao_cec_probe()
628 ao_cec->base = devm_platform_ioremap_resource(pdev, 0); in meson_ao_cec_probe()
629 if (IS_ERR(ao_cec->base)) { in meson_ao_cec_probe()
630 ret = PTR_ERR(ao_cec->base); in meson_ao_cec_probe()
634 irq = platform_get_irq(pdev, 0); in meson_ao_cec_probe()
635 ret = devm_request_threaded_irq(&pdev->dev, irq, in meson_ao_cec_probe()
640 dev_err(&pdev->dev, "irq request failed\n"); in meson_ao_cec_probe()
644 ao_cec->core = devm_clk_get(&pdev->dev, "core"); in meson_ao_cec_probe()
645 if (IS_ERR(ao_cec->core)) { in meson_ao_cec_probe()
646 dev_err(&pdev->dev, "core clock request failed\n"); in meson_ao_cec_probe()
647 ret = PTR_ERR(ao_cec->core); in meson_ao_cec_probe()
651 ret = clk_prepare_enable(ao_cec->core); in meson_ao_cec_probe()
653 dev_err(&pdev->dev, "core clock enable failed\n"); in meson_ao_cec_probe()
657 ret = clk_set_rate(ao_cec->core, CEC_CLK_RATE); in meson_ao_cec_probe()
659 dev_err(&pdev->dev, "core clock set rate failed\n"); in meson_ao_cec_probe()
663 device_reset_optional(&pdev->dev); in meson_ao_cec_probe()
665 ao_cec->pdev = pdev; in meson_ao_cec_probe()
668 ao_cec->notify = cec_notifier_cec_adap_register(hdmi_dev, NULL, in meson_ao_cec_probe()
669 ao_cec->adap); in meson_ao_cec_probe()
670 if (!ao_cec->notify) { in meson_ao_cec_probe()
671 ret = -ENOMEM; in meson_ao_cec_probe()
675 ret = cec_register_adapter(ao_cec->adap, &pdev->dev); in meson_ao_cec_probe()
681 ao_cec->base + CEC_GEN_CNTL_REG); in meson_ao_cec_probe()
686 cec_notifier_cec_adap_unregister(ao_cec->notify, ao_cec->adap); in meson_ao_cec_probe()
689 clk_disable_unprepare(ao_cec->core); in meson_ao_cec_probe()
692 cec_delete_adapter(ao_cec->adap); in meson_ao_cec_probe()
694 dev_err(&pdev->dev, "CEC controller registration failed\n"); in meson_ao_cec_probe()
703 clk_disable_unprepare(ao_cec->core); in meson_ao_cec_remove()
705 cec_notifier_cec_adap_unregister(ao_cec->notify, ao_cec->adap); in meson_ao_cec_remove()
706 cec_unregister_adapter(ao_cec->adap); in meson_ao_cec_remove()
710 { .compatible = "amlogic,meson-gx-ao-cec", },
719 .name = "meson-ao-cec",
726 MODULE_DESCRIPTION("Meson AO CEC Controller driver");