/*++ * linux/drivers/video/wmt/dev-cec.c * WonderMedia video post processor (VPP) driver * * Copyright c 2013 WonderMedia Technologies, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * WonderMedia Technologies, Inc. * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C --*/ #define DEV_CEC_C /* #define DEBUG */ #include #include #include #include #include #include "cec.h" #define DRIVER_NAME "wmt-cec" #define TXNORSP 0x8000 /* Tx no response interrupt */ #define TXLSARB 0x2000 /* Tx lose arbitration */ #define TXDONE 0x0100 /* Tx done */ #define RXFAIL 0x0002 /* Rx fail, details in Rx10 */ #define RXDONE 0x0001 /* Rx done */ #define MAX_RETRY 3 #define MAX_TIMEOUT 5000 static int tx_state; static struct cdev *wmt_cec_cdev; static DEFINE_SEMAPHORE(wmt_cec_sem); struct wmt_cec_msg recv_msg; static void wmt_cec_do_rx_notify(struct work_struct *ptr) { vpp_netlink_notify(USER_PID, DEVICE_RX_DATA, (int)&recv_msg); } DECLARE_DELAYED_WORK(wmt_cec_rx_work, wmt_cec_do_rx_notify); /* receive message interrupt handling */ static void wmt_cec_do_recv(void) { memset(&recv_msg, 0, sizeof(recv_msg)); /* get received data */ recv_msg.msglen = wmt_cec_rx_data(recv_msg.msgdata); if (recv_msg.msglen > MAX_MSG_BYTE) recv_msg.msglen = MAX_MSG_BYTE; DBGMSG("read a received byte! msglen: %d\n", recv_msg.msglen); /* clear receive blockage */ /* notify AP the received message, let AP decide what to response */ schedule_delayed_work(&wmt_cec_rx_work, HZ / 10); } /* check if logic address is valid */ static int bvalidaddr(char addr) { if (addr > 15) return 0; return 1; } /* make sure cec line is not busy */ static int tx_get_cecline(void) { int timeout = 400; while (timeout-- > 0) { /* if not busy */ if (1) return 0; msleep(1); } return -ETIMEDOUT; } /* transfer a time*/ DECLARE_COMPLETION(txcomp); static int wmt_cec_do_xfer_one(char *msgdata, int msglen) { int ret; unsigned long jiffies; ret = tx_get_cecline(); if (ret != 0) ret = -EAGAIN; wmt_cec_tx_data(msgdata, msglen); jiffies = msecs_to_jiffies((unsigned int)MAX_TIMEOUT); jiffies = wait_for_completion_timeout(&txcomp, jiffies); if (jiffies == 0) return -EAGAIN; if (tx_state == TXLSARB) return -EAGAIN; return tx_state; } /* transfer a message, including retransmission if needed*/ static int wmt_cec_do_xfer(char *msgdata, int msglen) { int retry; int ret; char srcaddr, tgtaddr; if (msglen < 1) { DPRINT("[CEC] xfer: invalid message, msglen is less than 1.\n"); return -1; } srcaddr = (msgdata[0] & 0xf0) >> 4; tgtaddr = (msgdata[0] & 0x0f); if (!bvalidaddr(srcaddr) || !bvalidaddr(tgtaddr)) { DPRINT("[CEC] xfer: invalid logic address in msg data.\n"); return -1; } for (retry = 0; retry < MAX_RETRY; retry++) { ret = wmt_cec_do_xfer_one(msgdata, msglen); if (ret != -EAGAIN) goto out; DPRINT("[CEC] Retrying transmission (%d)\n", retry); udelay(100); } return -EAGAIN; out: /* if polling message ret is no-response, set logical address register, and enable slave mode */ if (srcaddr == tgtaddr && msglen == 1) { if (ret == TXNORSP) { DBGMSG("[CEC] logic addr register is 0x%x\n", tgtaddr); return 0; } else return -1; } else if (ret == TXDONE) return 0; else return -1; } /* irq handling function */ irqreturn_t wmt_cec_do_irq(int irq, void *dev_id) { int sts; sts = wmt_cec_get_int(); wmt_cec_clr_int(sts); if (sts & BIT0) { /* tx done */ DBGMSG("[CEC] write ok int\n"); complete(&txcomp); tx_state = TXDONE; } if (sts & BIT1) { /* rx done */ DBGMSG("[CEC] read ok int\n"); wmt_cec_do_recv(); } if (sts & BIT2) { /* rx error */ DBGMSG("[CEC] read error int\n"); } if (sts & BIT3) { /* tx arb fail */ DBGMSG("[CEC] wr arb fail int\n"); complete(&txcomp); tx_state = TXLSARB; } if (sts & BIT4) { /* tx no ack */ /* DBGMSG("[CEC] write no ack int(addr not match)\n"); */ complete(&txcomp); tx_state = TXNORSP; } return IRQ_HANDLED; } static int wmt_cec_open( struct inode *inode, struct file *filp ) { int ret = 0; DBGMSG("[CEC] wmt_cec_open\n"); down(&wmt_cec_sem); wmt_cec_rx_enable(1); up(&wmt_cec_sem); return ret; } /* End of videodecoder_open() */ static int wmt_cec_release( struct inode *inode, struct file *filp ) { int ret = 0; DBGMSG("[CEC] wmt_cec_release\n"); down(&wmt_cec_sem); wmt_cec_rx_enable(0); up(&wmt_cec_sem); return ret; } /* End of videodecoder_release() */ static long wmt_cec_ioctl( struct file *filp, unsigned int cmd, unsigned long arg ) { int ret = -EINVAL; DBGMSG("[CEC] wmt_cec_ioctl 0x%x,0x%x\n", cmd, arg); /* check type and number, if fail return ENOTTY */ if (_IOC_TYPE(cmd) != WMT_CEC_IOC_MAGIC) return -ENOTTY; if (_IOC_NR(cmd) > WMT_CEC_IOC_MAXNR) return -ENOTTY; /* check argument area */ if (_IOC_DIR(cmd) & _IOC_READ) ret = !access_ok(VERIFY_WRITE, (void __user *) arg, _IOC_SIZE(cmd)); else if (_IOC_DIR(cmd) & _IOC_WRITE) ret = !access_ok(VERIFY_READ, (void __user *) arg, _IOC_SIZE(cmd)); else ret = 0; if (ret) return -EFAULT; down(&wmt_cec_sem); switch (cmd) { case CECIO_TX_DATA: { struct wmt_cec_msg msg; ret = copy_from_user((void *) &msg, (const void *)arg, sizeof(struct wmt_cec_msg)); wmt_cec_do_xfer(msg.msgdata, msg.msglen); } break; case CECIO_TX_LOGADDR: wmt_cec_set_logical_addr((arg & 0xFF00) >> 8, arg & 0xFF, 1); break; case CECIO_RX_PHYADDR: { wmt_phy_addr_t parm; parm.phy_addr = edid_get_hdmi_phy_addr(); ret = copy_to_user((void *)arg, (void *)&parm, sizeof(wmt_phy_addr_t)); } break; default: DBGMSG("[CEC] *W* wmt_cec_ioctl cmd 0x%x\n", cmd); break; } up(&wmt_cec_sem); return ret; } /* End of videodecoder_ioctl() */ static int wmt_cec_mmap( struct file *filp, struct vm_area_struct *vma ) { int ret = -EINVAL; down(&wmt_cec_sem); up(&wmt_cec_sem); return ret; } static ssize_t wmt_cec_read( struct file *filp, char __user *buf, size_t count, loff_t *f_pos ) { ssize_t ret = 0; down(&wmt_cec_sem); up(&wmt_cec_sem); return ret; } /* videodecoder_read */ static ssize_t wmt_cec_write( struct file *filp, const char __user *buf, size_t count, loff_t *f_pos ) { ssize_t ret = 0; down(&wmt_cec_sem); up(&wmt_cec_sem); return ret; } /* End of videodecoder_write() */ struct file_operations wmt_cec_fops = { .owner = THIS_MODULE, .open = wmt_cec_open, .release = wmt_cec_release, .read = wmt_cec_read, .write = wmt_cec_write, .unlocked_ioctl = wmt_cec_ioctl, .mmap = wmt_cec_mmap, }; static int __init wmt_cec_probe ( struct platform_device *dev /*!<; // a pointer to struct device */ ) { dev_t dev_no; int ret; DBGMSG("[CEC] Enter wmt_cec_probe\n"); wmt_cec_init_hw(); wmt_cec_set_logical_addr(4, 0xf, 1); /* default set boardcast address */ wmt_cec_rx_enable(0); /* wmt_cec_enable_loopback(1); */ dev_no = MKDEV(CEC_MAJOR, 0); ret = register_chrdev_region(dev_no, 1, "wmtcec"); if (ret < 0) { DPRINT("can't get %s dev major %d\n", DRIVER_NAME, CEC_MAJOR); return ret; } /* register char device */ wmt_cec_cdev = cdev_alloc(); if (!wmt_cec_cdev) { DPRINT("alloc dev error.\n"); return -ENOMEM; } cdev_init(wmt_cec_cdev, &wmt_cec_fops); ret = cdev_add(wmt_cec_cdev, dev_no, 1); if (ret) { DPRINT("reg char dev error(%d).\n", ret); cdev_del(wmt_cec_cdev); return ret; } if (vpp_request_irq(IRQ_VPP_IRQ20, wmt_cec_do_irq, SA_INTERRUPT, "cec", (void *) 0)) DPRINT("*E* request CEC ISR fail\n"); vppif_reg32_out(REG_CEC_INT_ENABLE, 0x1f); DBGMSG("[CEC] Exit wmt_cec_probe(0x%x)\n", dev_no); return 0; } /* End of wmt_cec_probe */ static int wmt_cec_remove ( struct platform_device *dev /*!<; // a pointer point to struct device */ ) { return 0; } /* End of wmt_cec_remove */ #ifdef CONFIG_PM static int wmt_cec_suspend ( struct platform_device *pDev, /*!<; // a pointer to struct device */ pm_message_t state /*!<; // suspend state */ ) { DPRINT("Enter wmt_cec_suspend\n"); wmt_cec_do_suspend(); return 0; } /* End of wmt_cec_suspend */ static int wmt_cec_resume ( struct platform_device *pDev /*!<; // a pointer to struct device */ ) { DPRINT("Enter wmt_cec_resume\n"); wmt_cec_do_resume(); return 0; } /* End of wmt_cec_resume */ #else #define wmt_cec_suspend NULL #define wmt_cec_resume NULL #endif /*************************************************************************** device driver struct define ****************************************************************************/ static struct platform_driver wmt_cec_driver = { .driver.name = "wmtcec", /* equal to platform device name. */ .driver.bus = &platform_bus_type, .probe = wmt_cec_probe, .remove = wmt_cec_remove, .suspend = wmt_cec_suspend, .resume = wmt_cec_resume, }; /*************************************************************************** platform device struct define ****************************************************************************/ static u64 wmt_cec_dma_mask = 0xffffffffUL; static struct platform_device wmt_cec_device = { .name = "wmtcec", .dev = { .dma_mask = &wmt_cec_dma_mask, .coherent_dma_mask = ~0, }, #if 0 .id = 0, .dev = { .release = wmt_cec_platform_release, }, .num_resources = 0, /* ARRAY_SIZE(wmt_cec_resources), */ .resource = NULL, /* wmt_cec_resources, */ #endif }; static int __init wmt_cec_init(void) { int ret; char buf[100]; int varlen = 100; unsigned int cec_enable = 0; if (wmt_getsyspara("wmt.display.cec", buf, &varlen) == 0) vpp_parse_param(buf, &cec_enable, 1, 0); if (cec_enable == 0) return 0; DBGMSG(KERN_ALERT "Enter wmt_cec_init\n"); ret = platform_driver_register(&wmt_cec_driver); if (!ret) { ret = platform_device_register(&wmt_cec_device); if (ret) platform_driver_unregister(&wmt_cec_driver); } return ret; } static void __exit wmt_cec_exit(void) { dev_t dev_no; DBGMSG(KERN_ALERT "Enter wmt_cec_exit\n"); platform_driver_unregister(&wmt_cec_driver); platform_device_unregister(&wmt_cec_device); dev_no = MKDEV(CEC_MAJOR, 0); unregister_chrdev_region(dev_no, 1); return; } module_init(wmt_cec_init); module_exit(wmt_cec_exit); MODULE_LICENSE("GPL"); MODULE_DESCRIPTION("WMT CEC driver"); MODULE_AUTHOR("WMT TECH"); MODULE_VERSION("1.0.0");