From df0515d85203d6e4690bd078d3b6b4382941e1d3 Mon Sep 17 00:00:00 2001 From: hyperion70 Date: Mon, 4 Feb 2019 23:14:23 +0300 Subject: [PATCH] Add NFC files --- arch/arm64/configs/N1_defconfig | 4 + drivers/input/fingerprint/Makefile | 2 +- drivers/input/fingerprint/gx556/Kconfig | 7 +- drivers/input/fingerprint/gx556/Makefile | 1 + drivers/input/fingerprint/gx556/gf_spi.h | 4 +- drivers/misc/mediatek/nfc/mt6757/Makefile | 9 +- drivers/misc/mediatek/nfc/mt6757/pn544.c | 1209 +++++++++++++++++++++ drivers/spi/mediatek/mt6757/mtk_spi.h | 2 +- include/linux/nfc/pn547.h | 30 + 9 files changed, 1257 insertions(+), 11 deletions(-) create mode 100644 drivers/misc/mediatek/nfc/mt6757/pn544.c create mode 100644 include/linux/nfc/pn547.h diff --git a/arch/arm64/configs/N1_defconfig b/arch/arm64/configs/N1_defconfig index ecf4189e2..58d4c5cb2 100755 --- a/arch/arm64/configs/N1_defconfig +++ b/arch/arm64/configs/N1_defconfig @@ -576,9 +576,13 @@ CONFIG_GTP_DRIVER_SEND_CFG=y CONFIG_GTP_CUSTOM_CFG=y CONFIG_GTP_CREATE_WR_NODE=y +CONFIG_MTK_NFC_SUPPORT=y + CONFIG_DEBUG_KMEMLEAK=y CONFIG_DEBUG_KMEMLEAK_EARLY_LOG_SIZE=3200 CONFIG_DEBUG_KMEMLEAK_DEFAULT_OFF=y CONFIG_MTK_RCU_MONITOR=y + +CONFIG_LOG_BUF_SHIFT=21 # [hyperion70 add end] diff --git a/drivers/input/fingerprint/Makefile b/drivers/input/fingerprint/Makefile index 135f96fff..0eb1cc60c 100644 --- a/drivers/input/fingerprint/Makefile +++ b/drivers/input/fingerprint/Makefile @@ -3,5 +3,5 @@ # obj-$(CONFIG_GOODIX_FINGERPRINT) += goodix/ -obj-$(CONFIG_FPC_FINGERPRINT) += fpc/ +obj-$(CONFIG_GX556_FINGERPRINT) += gx556/ diff --git a/drivers/input/fingerprint/gx556/Kconfig b/drivers/input/fingerprint/gx556/Kconfig index 651512aa9..3c3862481 100755 --- a/drivers/input/fingerprint/gx556/Kconfig +++ b/drivers/input/fingerprint/gx556/Kconfig @@ -1,7 +1,4 @@ config FINGERPRINT_GT556 - bool "gf5216(goodix fingerprint sensor) for dewav" + bool "gt556(goodix fingerprint sensor) for dewav" default n - help - It support different type sensor - in this platform. If this option - is set, it will support fingerprint gf3208 silead sensor . + diff --git a/drivers/input/fingerprint/gx556/Makefile b/drivers/input/fingerprint/gx556/Makefile index dbde40cdf..52eaaea1e 100755 --- a/drivers/input/fingerprint/gx556/Makefile +++ b/drivers/input/fingerprint/gx556/Makefile @@ -1,6 +1,7 @@ #include $(srctree)/drivers/misc/mediatek/Makefile.custom # Makefile for the fingerprint device. ccflags-y += -I$(srctree)/drivers/spi/mediatek/$(MTK_PLATFORM) +ccflags-y += -I$(srctree)/drivers/misc/mediatek/include ccflags-y += -I$(srctree)/drivers/misc/mediatek/include/mt-plat ccflags-y += -I$(srctree)/drivers/misc/mediatek/include/mt-plat/$(MTK_PLATFORM)/include subdir-ccflags-y += -Werror diff --git a/drivers/input/fingerprint/gx556/gf_spi.h b/drivers/input/fingerprint/gx556/gf_spi.h index 4d994325b..f174a4633 100755 --- a/drivers/input/fingerprint/gx556/gf_spi.h +++ b/drivers/input/fingerprint/gx556/gf_spi.h @@ -5,8 +5,8 @@ #include #include - -#include "../../../../spi/mediatek/mt6757/mtk_spi.h" +#include "mtk_spi.h" +//#include "../../../../spi/mediatek/mt6757/mtk_spi.h" /**********************************************************/ /**********************GF ops****************************/ diff --git a/drivers/misc/mediatek/nfc/mt6757/Makefile b/drivers/misc/mediatek/nfc/mt6757/Makefile index b44e0ce57..bb50b304e 100644 --- a/drivers/misc/mediatek/nfc/mt6757/Makefile +++ b/drivers/misc/mediatek/nfc/mt6757/Makefile @@ -1,3 +1,8 @@ -obj-$(CONFIG_MTK_NFC_SUPPORT) += mt6605.o -ccflags-$(CONFIG_MTK_NFC_SUPPORT) += -DDEBUG +ifeq ($(TARGET_BUILD_VARIANT),user) +ccflags-y += -D_MTK_USER_ +endif + +#obj-$(CONFIG_MTK_NFC_SUPPORT) += mt6605.o +#ccflags-$(CONFIG_MTK_NFC_SUPPORT) += -DDEBUG +obj-y += pn544.o diff --git a/drivers/misc/mediatek/nfc/mt6757/pn544.c b/drivers/misc/mediatek/nfc/mt6757/pn544.c new file mode 100644 index 000000000..a63f5a075 --- /dev/null +++ b/drivers/misc/mediatek/nfc/mt6757/pn544.c @@ -0,0 +1,1209 @@ +/*************************************************************************** + * Filename: + * --------- + * pn544.c + * + * Project: + * -------- + * + * Description: + * ------------ + * + * Author: + * ------- + * LiangChi Huang, ext 25609, LiangChi.Huang@mediatek.com, 2012-08-09 + * + *****************************************************************************/ + +#define pr_fmt(fmt) "["KBUILD_MODNAME"]" fmt +/***************************************************************************** + * Include + *****************************************************************************/ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#ifdef CONFIG_OF +#include +#include +#include +#endif + +/* #include */ +/* #include */ + +#ifndef CONFIG_MTK_FPGA +/* #include for clock buffer */ +#endif + +/* #include */ +/* #include */ + +#if defined(CONFIG_MTK_LEGACY) +#include +#include +#endif + +/***************************************************************************** + * Define + *****************************************************************************/ + +/* #define NFC_I2C_BUSNUM I2C_NFC_CHANNEL */ + +#ifndef _MTK_USER_ +#define NFC_DEBUG 1 +#else +#define NFC_DEBUG 0 +#endif + +#define I2C_ID_NAME "pn544" + +#define MAX_BUFFER_SIZE 512 + +#define NXP_KR_READ_IRQ_MODIFY + +#ifdef NXP_KR_READ_IRQ_MODIFY +static bool do_reading; +static bool cancle_read; +#endif + +#define NFC_CLIENT_TIMING 400 /* I2C speed */ + +enum { + MTK_NFC_GPIO_EN_B = 0x0, + MTK_NFC_GPIO_SYSRST_B, + MTK_NFC_GPIO_EINT, + MTK_NFC_GPIO_IRQ, + MTK_NFC_GPIO_IOCTL, + MTK_NFC_GPIO_MAX_NUM +}; + +enum { + MTK_NFC_IOCTL_READ = 0x0, + MTK_NFC_IOCTL_WRITE, + MTK_NFC_IOCTL_MAX_NUM +}; + +enum { + MTK_NFC_IOCTL_CMD_CLOCK_BUF_ENABLE = 0x0, + MTK_NFC_IOCTL_CMD_CLOCK_BUF_DISABLE, + MTK_NFC_IOCTL_CMD_EXIT_EINT, + MTK_NFC_IOCTL_CMD_GET_CHIP_ID, + MTK_NFC_IOCTL_CMD_READ_DATA, + MTK_NFC_IOCTL_CMD_MAX_NUM +}; + +enum { + MTK_NFC_PULL_LOW = 0x0, + MTK_NFC_PULL_HIGH, + MTK_NFC_PULL_INVALID, +}; + +enum { + MTK_NFC_GPIO_DIR_IN = 0x0, + MTK_NFC_GPIO_DIR_OUT, + MTK_NFC_GPIO_DIR_INVALID, +}; + +/***************************************************************************** + * Global Variable + *****************************************************************************/ +struct pn544_dev *pn544_dev_ptr = NULL; +struct pn544_dev _gpn544_dev; + +#if !defined(CONFIG_MTK_LEGACY) +struct platform_device *nfc_plt_dev = NULL; +struct pinctrl *gpctrl = NULL; +struct pinctrl_state *st_ven_h = NULL; +struct pinctrl_state *st_ven_l = NULL; +struct pinctrl_state *st_rst_h = NULL; +struct pinctrl_state *st_rst_l = NULL; +struct pinctrl_state *st_eint_h = NULL; +struct pinctrl_state *st_eint_l = NULL; +struct pinctrl_state *st_irq_init = NULL; +struct pinctrl_state *st_osc_init = NULL; +#endif + +/* static struct i2c_board_info nfc_board_info __initdata = */ +/* { I2C_BOARD_INFO(I2C_ID_NAME, I2C_NFC_SLAVE_7_BIT_ADDR) }; */ + +/* For DMA */ +#ifdef CONFIG_MTK_I2C_EXTENSION +static char *I2CDMAWriteBuf; /*= NULL;*//* unnecessary initialise */ +static unsigned int I2CDMAWriteBuf_pa; /* = NULL; */ +static char *I2CDMAReadBuf; /*= NULL;*//* unnecessary initialise */ +static unsigned int I2CDMAReadBuf_pa; /* = NULL; */ +#else +static char I2CDMAWriteBuf[MAX_BUFFER_SIZE]; +static char I2CDMAReadBuf[MAX_BUFFER_SIZE]; +#endif +//static int fgNfcChip; /*= 0;*//* unnecessary initialise */ +int forceExitBlockingRead = 0; + +/* NFC IRQ */ +static u32 nfc_irq; + +/***************************************************************************** + * Function Prototype + *****************************************************************************/ +static int pn544_probe(struct i2c_client *client, + const struct i2c_device_id *id); +static int pn544_remove(struct i2c_client *client); +/* static int pn544_detect(struct i2c_client *client, */ +/* int kind, struct i2c_board_info *info); */ +static int pn544_dev_open(struct inode *inode, struct file *filp); +static long pn544_dev_unlocked_ioctl(struct file *filp, unsigned int cmd, + unsigned long arg); +static ssize_t pn544_dev_read(struct file *filp, char __user *buf, + size_t count, loff_t *offset); +static ssize_t pn544_dev_write(struct file *filp, const char __user *buf, + size_t count, loff_t *offset); + +static int mt_nfc_probe(struct platform_device *pdev); +static int mt_nfc_remove(struct platform_device *pdev); + +/* void pn544_dev_irq_handler(void); */ +static irqreturn_t pn544_dev_irq_handler(int irq, void *data); /*IRQ handler */ + +/* static void pn544_disable_irq(struct pn544_dev *pn544_dev); */ +#if !defined(CONFIG_MTK_LEGACY) +static int mt_nfc_pinctrl_init(struct platform_device *pdev); +static int mt_nfc_pinctrl_select(struct pinctrl *p, struct pinctrl_state *s); +static int mt_nfc_gpio_init(void); +#endif +static int mt_nfc_get_gpio_value(int gpio_num); +static int mt_nfc_get_gpio_dir(int gpio_num); + +/***************************************************************************** + * Data Structure + *****************************************************************************/ + +struct pn544_dev { + wait_queue_head_t read_wq; + struct mutex read_mutex; + struct i2c_client *client; + struct miscdevice pn544_device; + unsigned int ven_gpio; + unsigned int sysrstb_gpio; + unsigned int irq_gpio; /* Chip inform Host */ + unsigned int eint_gpio; /* Host inform Chip */ + bool irq_enabled; + struct mutex irq_enabled_lock; + /* spinlock_t irq_enabled_lock; */ +}; +struct wake_lock nfc_wake_lock; + +struct pn544_i2c_platform_data { + unsigned int irq_gpio; /* Chip inform Host */ + unsigned int ven_gpio; + unsigned int sysrstb_gpio; + unsigned int eint_gpio; /* Host inform Chip */ + unsigned int osc_en; +}; + +static const struct i2c_device_id pn544_id[] = { + {I2C_ID_NAME, 0}, + {} +}; + +#ifdef CONFIG_OF +static const struct of_device_id nfc_switch_of_match[] = { + {.compatible = "mediatek,nfc"}, + {}, +}; +#endif + +/* #if (LINUX_VERSION_CODE < KERNEL_VERSION(2,6,36)) */ +/* static struct i2c_client_address_data addr_data = { .forces = forces,}; */ +/* #endif */ +static struct i2c_driver pn544_dev_driver = { + .id_table = pn544_id, + .probe = pn544_probe, + .remove = pn544_remove, + /* .detect = pn544_detect, */ + .driver = { + .name = "pn544", + .owner = THIS_MODULE, +#ifdef CONFIG_OF + .of_match_table = nfc_switch_of_match, +#endif + }, +}; + +/* platform driver */ +static const struct of_device_id nfc_dev_of_match[] = { + {.compatible = "mediatek,nfc-gpio-v2",}, + {}, +}; + +static struct platform_driver mtk_nfc_platform_driver = { + .probe = mt_nfc_probe, + .remove = mt_nfc_remove, + .driver = { + .name = I2C_ID_NAME, + .owner = THIS_MODULE, +#ifdef CONFIG_OF + .of_match_table = nfc_dev_of_match, +#endif + }, +}; + +#if !defined(CONFIG_MTK_LEGACY) +struct pn544_i2c_platform_data pn544_platform_data; +#else +static struct pn544_i2c_platform_data pn544_platform_data = { + .irq_gpio = GPIO_IRQ_NFC_PIN, + .ven_gpio = GPIO_NFC_VENB_PIN, + .sysrstb_gpio = GPIO_NFC_RST_PIN, + .eint_gpio = GPIO_NFC_EINT_PIN, +}; +#endif + +static const struct file_operations pn544_dev_fops = { + .owner = THIS_MODULE, + .llseek = no_llseek, + .read = pn544_dev_read, + .write = pn544_dev_write, + .open = pn544_dev_open, +#ifdef CONFIG_COMPAT + .compat_ioctl = pn544_dev_unlocked_ioctl, +#endif + .unlocked_ioctl = pn544_dev_unlocked_ioctl, +}; + +/***************************************************************************** + * Extern Area + *****************************************************************************/ + +/***************************************************************************** + * Function + *****************************************************************************/ +static void pn544_enable_irq(u32 irq_line,struct pn544_dev *pn544_dev) +{ +#if NFC_DEBUG + pr_debug("pn544_enable_irq() irq_line=%d, irq_enabled=%d.\n", irq_line, pn544_dev->irq_enabled); +#endif + + mutex_lock(&pn544_dev->irq_enabled_lock); + enable_irq(irq_line); + + enable_irq_wake(irq_line); + + pn544_dev->irq_enabled = true; + mutex_unlock(&pn544_dev->irq_enabled_lock); +} + +static void pn544_disable_irq(u32 irq_line,struct pn544_dev *pn544_dev) +{ +#if NFC_DEBUG + pr_debug("pn544_disable_irq() irq_line=%d, irq_enabled=%d.\n", irq_line, pn544_dev->irq_enabled); +#endif + + mutex_lock(&pn544_dev->irq_enabled_lock); + if (pn544_dev->irq_enabled) { + disable_irq_nosync(irq_line); + pn544_dev->irq_enabled = false; + } else { + pr_debug("%s : disable irq fail.\n", __func__); + } + mutex_unlock(&pn544_dev->irq_enabled_lock); +} + +static int pn544_probe(struct i2c_client *client, + const struct i2c_device_id *id) +{ + int ret = 0; + struct pn544_i2c_platform_data *platform_data; + struct device_node *node; + + pr_debug("pn544_dev_probe\n"); + + platform_data = &pn544_platform_data; + + if (platform_data == NULL) { + pr_err("%s : nfc probe fail\n", __func__); + return -ENODEV; + } + + if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) { + pr_err("%s : need I2C_FUNC_I2C\n", __func__); + return -ENODEV; + } +#if defined(CONFIG_MTK_LEGACY) + mt_set_gpio_mode(platform_data->irq_gpio, GPIO_IRQ_NFC_PIN_M_GPIO); + mt_set_gpio_dir(platform_data->irq_gpio, GPIO_DIR_IN); + mt_set_gpio_pull_enable(platform_data->irq_gpio, GPIO_PULL_ENABLE); + mt_set_gpio_pull_select(platform_data->irq_gpio, GPIO_PULL_DOWN); + + /* ven_gpio */ + mt_set_gpio_mode(platform_data->ven_gpio, GPIO_NFC_VENB_PIN_M_GPIO); + mt_set_gpio_dir(platform_data->ven_gpio, GPIO_DIR_OUT); + mt_set_gpio_out(platform_data->ven_gpio, GPIO_OUT_ONE); + usleep_range(900, 1000); /* debug */ + + /* firm_gpio */ + mt_set_gpio_mode(platform_data->sysrstb_gpio, GPIO_NFC_RST_PIN_M_GPIO); + mt_set_gpio_dir(platform_data->sysrstb_gpio, GPIO_DIR_OUT); + mt_set_gpio_out(platform_data->sysrstb_gpio, GPIO_OUT_ONE); + usleep_range(900, 1000); /* debug */ + + /* EINT_gpio */ + mt_set_gpio_mode(platform_data->eint_gpio, GPIO_NFC_EINT_PIN_M_GPIO); + mt_set_gpio_dir(platform_data->eint_gpio, GPIO_DIR_OUT); + /* Set output High */ + mt_set_gpio_out(platform_data->eint_gpio, GPIO_OUT_ZERO); + + _gpn544_dev.irq_gpio = platform_data->irq_gpio; + _gpn544_dev.ven_gpio = platform_data->ven_gpio; + _gpn544_dev.sysrstb_gpio = platform_data->sysrstb_gpio; + _gpn544_dev.eint_gpio = platform_data->eint_gpio; +#endif + + _gpn544_dev.client = client; + /* init mutex and queues */ + init_waitqueue_head(&_gpn544_dev.read_wq); + mutex_init(&_gpn544_dev.read_mutex); + /* spin_lock_init(&pn544_dev->irq_enabled_lock); */ + mutex_init(&_gpn544_dev.irq_enabled_lock); + +#if 0 + _gpn544_dev.pn544_device.minor = MISC_DYNAMIC_MINOR; + _gpn544_dev.pn544_device.name = "pn544"; + _gpn544_dev.pn544_device.fops = &pn544_dev_fops; + + ret = misc_register(&_gpn544_dev.pn544_device); + if (ret) { + pr_err("%s : misc_register failed\n", __FILE__); + goto err_misc_register; + } +#endif + +#ifdef CONFIG_MTK_I2C_EXTENSION +#ifdef CONFIG_64BIT + I2CDMAWriteBuf = + (char *)dma_alloc_coherent(&client->dev, MAX_BUFFER_SIZE, + (dma_addr_t *) &I2CDMAWriteBuf_pa, + GFP_KERNEL); +#else + I2CDMAWriteBuf = + (char *)dma_alloc_coherent(NULL, MAX_BUFFER_SIZE, + (dma_addr_t *) &I2CDMAWriteBuf_pa, + GFP_KERNEL); +#endif + + if (I2CDMAWriteBuf == NULL) { + pr_err("%s : failed to allocate dma buffer\n", __func__); + mutex_destroy(&_gpn544_dev.read_mutex); + gpio_free(platform_data->sysrstb_gpio); + return ret; + } +#ifdef CONFIG_64BIT + I2CDMAReadBuf = + (char *)dma_alloc_coherent(&client->dev, MAX_BUFFER_SIZE, + (dma_addr_t *) &I2CDMAReadBuf_pa, + GFP_KERNEL); +#else + I2CDMAReadBuf = + (char *)dma_alloc_coherent(NULL, MAX_BUFFER_SIZE, + (dma_addr_t *) &I2CDMAReadBuf_pa, + GFP_KERNEL); +#endif + + if (I2CDMAReadBuf == NULL) { + pr_err("%s : failed to allocate dma buffer\n", __func__); + mutex_destroy(&_gpn544_dev.read_mutex); + gpio_free(platform_data->sysrstb_gpio); + return ret; + } + pr_debug("%s :I2CDMAWriteBuf_pa %d, I2CDMAReadBuf_pa,%d\n", __func__, + I2CDMAWriteBuf_pa, I2CDMAReadBuf_pa); +#else + memset(I2CDMAWriteBuf, 0x00, sizeof(I2CDMAWriteBuf)); + memset(I2CDMAReadBuf, 0x00, sizeof(I2CDMAReadBuf)); +#endif + /* request irq. the irq is set whenever the chip has data available + * for reading. it is cleared when all data has been read. + */ + + /* client->irq = CUST_EINT_IRQ_NFC_NUM; */ + /* pr_debug("%s : requesting IRQ %d\n", __func__, client->irq); */ + /* pr_debug("pn544_Prob2\n"); */ + /* mt_eint_set_hw_debounce(CUST_EINT_IRQ_NFC_NUM, */ + /* CUST_EINT_IRQ_NFC_DEBOUNCE_CN); */ + /* mt_eint_registration(CUST_EINT_IRQ_NFC_NUM, */ + /* CUST_EINT_IRQ_NFC_TYPE, pn544_dev_irq_handler, */ + /* 0); */ + + /* NFC IRQ settings */ + node = of_find_compatible_node(NULL, NULL, "mediatek,nfc-gpio-v2"); + + if (node) { + + nfc_irq = irq_of_parse_and_map(node, 0); + + client->irq = nfc_irq; + + _gpn544_dev.irq_enabled = true; + + ret = + request_irq(nfc_irq, pn544_dev_irq_handler, + IRQF_TRIGGER_RISING, "irq_nfc-eint", NULL); + + if (ret) { + + pr_err("%s : EINT IRQ LINE NOT AVAILABLE, ret = %d\n", __func__, ret); + + } else { + + pr_debug("%s : set EINT finished, nfc_irq=%d", __func__, + nfc_irq); + //pn544_enable_irq(nfc_irq);//TEMP! + //enable_irq_wake(nfc_irq); + pn544_disable_irq(nfc_irq,&_gpn544_dev); + + } + + } else { + pr_err("%s : can not find NFC eint compatible node\n", + __func__); + } + + i2c_set_clientdata(client, &_gpn544_dev); + + wake_lock_init(&nfc_wake_lock, WAKE_LOCK_SUSPEND, "NFCWAKE"); + + forceExitBlockingRead = 0; + + return 0; +} + +static int pn544_remove(struct i2c_client *client) +{ + /* struct pn544_dev *pn544_dev; */ + + pr_debug("pn544_remove\n"); + +#ifdef CONFIG_MTK_I2C_EXTENSION + if (I2CDMAWriteBuf) { + #ifdef CONFIG_64BIT + dma_free_coherent(&client->dev, MAX_BUFFER_SIZE, I2CDMAWriteBuf, + I2CDMAWriteBuf_pa); + #else + dma_free_coherent(NULL, MAX_BUFFER_SIZE, I2CDMAWriteBuf, + I2CDMAWriteBuf_pa); + #endif + I2CDMAWriteBuf = NULL; + I2CDMAWriteBuf_pa = 0; + } + + if (I2CDMAReadBuf) { + #ifdef CONFIG_64BIT + dma_free_coherent(&client->dev, MAX_BUFFER_SIZE, I2CDMAReadBuf, + I2CDMAReadBuf_pa); + #else + dma_free_coherent(NULL, MAX_BUFFER_SIZE, I2CDMAReadBuf, + I2CDMAReadBuf_pa); + #endif + I2CDMAReadBuf = NULL; + I2CDMAReadBuf_pa = 0; + } +#endif + + /* pn544_dev = i2c_get_clientdata(client); */ + /* free_irq(client->irq, &_gpn544_dev); */ + wake_lock_destroy(&nfc_wake_lock); + free_irq(nfc_irq, &_gpn544_dev); + misc_deregister(&_gpn544_dev.pn544_device); + mutex_destroy(&_gpn544_dev.read_mutex); + +#if defined(CONFIG_MTK_LEGACY) + gpio_free(_gpn544_dev.irq_gpio); + gpio_free(_gpn544_dev.ven_gpio); + gpio_free(_gpn544_dev.sysrstb_gpio); + gpio_free(_gpn544_dev.eint_gpio); +#endif + + /* kfree(pn544_dev); */ + + return 0; +} + +static int mt_nfc_probe(struct platform_device *pdev) +{ + int ret = 0; + +#if !defined(CONFIG_MTK_LEGACY) + + nfc_plt_dev = pdev; + + pr_debug("%s : &nfc_plt_dev=%p\n", __func__, nfc_plt_dev); + + /* pinctrl init */ + ret = mt_nfc_pinctrl_init(pdev); + + /* gpio init */ + if (mt_nfc_gpio_init() != 0) + pr_debug("%s : mt_nfc_gpio_init err.\n", __func__); + +#endif + + return 0; +} + +static int mt_nfc_remove(struct platform_device *pdev) +{ + pr_debug("%s : &pdev=%p\n", __func__, pdev); + return 0; +} + +/* void pn544_dev_irq_handler(void) */ +irqreturn_t pn544_dev_irq_handler(int irq, void *data) +{ + struct pn544_dev *pn544_dev = pn544_dev_ptr; + /* pr_debug("%s : &pn544_dev=%p\n", __func__, pn544_dev); */ + + if (pn544_dev == NULL) { + pr_debug("pn544_dev NULL.\n"); + return IRQ_HANDLED; + } + + + if(!mt_nfc_get_gpio_value(pn544_dev->irq_gpio)) { + #if NFC_DEBUG + pr_err("%s, irq_gpio = %d\n", __func__, mt_nfc_get_gpio_value(pn544_dev->irq_gpio)); + #endif + return IRQ_HANDLED; + } + + #ifdef NXP_KR_READ_IRQ_MODIFY + do_reading = true; + #endif + + //pn544_disable_irq(nfc_irq,pn544_dev);//Temp!!! + + + wake_lock_timeout(&nfc_wake_lock,2*HZ); + + + wake_up(&pn544_dev->read_wq); + + #if NFC_DEBUG + pr_info("%s, IRQ_HANDLED! wake lock time out 2sec\n", __func__); + #endif + + //wake_lock_timeout(&pn544_dev->nfc_wake_lock, 10 * HZ); + + return IRQ_HANDLED; +} + +static ssize_t pn544_dev_read(struct file *filp, char __user *buf, + size_t count, loff_t *offset) +{ + struct pn544_dev *pn544_dev = filp->private_data; + int ret = 0; + int read_retry = 5; + + if (count > MAX_BUFFER_SIZE) + count = MAX_BUFFER_SIZE; + +#if NFC_DEBUG + dev_info(&pn544_dev->client->dev, "pn544 : + r\n"); +#endif + + mutex_lock(&pn544_dev->read_mutex); + +#if NFC_DEBUG + pr_debug("%s : enter wait and wake unlock, enable_irq %d, irq status=%d\n", + __func__, + nfc_irq, mt_nfc_get_gpio_value(pn544_dev->irq_gpio)); +#endif + + if (!mt_nfc_get_gpio_value(pn544_dev->irq_gpio)) { + #ifdef NXP_KR_READ_IRQ_MODIFY + do_reading = false; + #endif + + if (filp->f_flags & O_NONBLOCK) { + ret = -EAGAIN; + pr_debug("%s : goto fail.\n", __func__); + goto fail; + } + + wake_unlock(&nfc_wake_lock); + + #if NFC_DEBUG + dev_info(&pn544_dev->client->dev, "wait_event_interruptible : in\n"); + #endif + ret = wait_event_interruptible(pn544_dev->read_wq, do_reading); + #if NFC_DEBUG + dev_info(&pn544_dev->client->dev, "wait_event_interruptible : out\n"); + #endif + + #ifdef NXP_KR_READ_IRQ_MODIFY + if (cancle_read == true) { + cancle_read = false; + ret = -1; + goto fail; + } + #endif + + if (ret) + { + pr_info("%s, wait_event_interruptible return ret=%d\n", __func__,ret); + goto fail; + } + } + else + { +#if NFC_DEBUG + pr_info("%s, irq_gpio is high, wake lock timeout 2HZ by continue read\n", __func__); +#endif + wake_lock_timeout(&nfc_wake_lock,2*HZ); + } + #ifdef CONFIG_MTK_I2C_EXTENSION + pn544_dev->client->addr = (pn544_dev->client->addr & I2C_MASK_FLAG); + pn544_dev->client->ext_flag |= I2C_DMA_FLAG; + /* pn544_dev->client->ext_flag |= I2C_DIRECTION_FLAG; */ + /* pn544_dev->client->ext_flag |= I2C_A_FILTER_MSG; */ + pn544_dev->client->timing = NFC_CLIENT_TIMING; + + /* Read data */ + ret = + i2c_master_recv(pn544_dev->client, + (unsigned char *)(uintptr_t) I2CDMAReadBuf_pa, + count); + + #else + + while (read_retry) { + ret = + i2c_master_recv(pn544_dev->client, + (unsigned char *)(uintptr_t) I2CDMAReadBuf, + count); + + /* mutex_unlock(&pn544_dev->read_mutex); */ + + /*pr_debug("%s : i2c_master_recv returned=%d, irq status=%d\n", __func__, + ret, mt_nfc_get_gpio_value(pn544_dev->irq_gpio));*/ + + if (ret < 0) { + pr_debug("%s: i2c_master_recv failed: %d, read_retry: %d\n", + __func__, ret, read_retry); + read_retry--; + usleep_range(900, 1000); + continue; + } + break; + } + #endif + if (ret < 0) { + pr_err("%s: i2c_master_recv failed: %d, read_retry: %d\n", + __func__, ret, read_retry); + } + + if (ret > count) { + pr_debug("%s: received too many bytes from i2c (%d)\n", + __func__, ret); + ret = -EIO; + } + + if (copy_to_user(buf, I2CDMAReadBuf, ret)) { + pr_debug("%s : failed to copy to user space\n", __func__); + ret = -EFAULT; + } + + if(mt_nfc_get_gpio_value(pn544_dev->irq_gpio) == 1) + { +#if NFC_DEBUG + pr_info("%s, irq_gpio is high, wake lock timeout 2HZ by next read\n", __func__); +#endif + wake_lock_timeout(&nfc_wake_lock,2*HZ); + } + + mutex_unlock(&pn544_dev->read_mutex); +#if NFC_DEBUG + if(ret > 0) + { + int i; + for (i = 0; i < ret; i++) + dev_info(&pn544_dev->client->dev, "Read_buf[%d] = 0x%x\n", i, I2CDMAReadBuf[i]); + } + dev_info(&pn544_dev->client->dev, "pn544 : - r\n"); +#endif + + return ret; + +fail: + if(mt_nfc_get_gpio_value(pn544_dev->irq_gpio)) + { +#if NFC_DEBUG + pr_info("%s, Read fail irq_gpio is high, wake lock timeout 2HZ by retry\n", __func__); +#endif + wake_lock_timeout(&nfc_wake_lock,2*HZ); + } + else + { +#if NFC_DEBUG + pr_info("%s, Read fail irq_gpio is low, wake unlock timeout 2HZ\n", __func__); +#endif + wake_unlock(&nfc_wake_lock); + } + + mutex_unlock(&pn544_dev->read_mutex); + pr_debug("%s: return, fail: %d\n", __func__, ret); + return ret; + +} + +static ssize_t pn544_dev_write(struct file *filp, const char __user *buf, + size_t count, loff_t *offset) +{ + struct pn544_dev *pn544_dev; + int ret = 0, ret_tmp = 0, count_ori = 0, count_remain = 0, idx = 0; + +#if NFC_DEBUG + int i; +// char *tmp; +#endif + + pn544_dev = filp->private_data; + count_ori = count; + count_remain = count_ori; + +#if NFC_DEBUG + dev_info(&pn544_dev->client->dev, "pn544 : + w\n"); + for (i = 0; i < count; i++) + dev_info(&pn544_dev->client->dev, "buf[%d] = 0x%x\n", i, buf[i]); +#endif + + + if (count > MAX_BUFFER_SIZE) { + count = MAX_BUFFER_SIZE; + count_remain -= count; + } + + pn544_disable_irq(nfc_irq,pn544_dev); + + while (1) + { + if (copy_from_user + (I2CDMAWriteBuf, &buf[(idx * MAX_BUFFER_SIZE)], count)) { + pr_err("%s : failed to copy from user space.\n", + __func__); + ret = -EFAULT; + break; + } + + /*pr_debug("%s : writing %zu bytes, remain bytes %d.\n", __func__, + count, count_remain); */ + + /* Write data */ + #ifdef CONFIG_MTK_I2C_EXTENSION + pn544_dev->client->addr = + (pn544_dev->client->addr & I2C_MASK_FLAG); + + pn544_dev->client->ext_flag |= I2C_DMA_FLAG; + /* pn544_dev->client->ext_flag |= I2C_DIRECTION_FLAG; */ + /* pn544_dev->client->ext_flag |= I2C_A_FILTER_MSG; */ + pn544_dev->client->timing = NFC_CLIENT_TIMING; + + ret_tmp = + i2c_master_send(pn544_dev->client, + (unsigned char *)(uintptr_t) + I2CDMAWriteBuf_pa, count); + #else + ret_tmp = + i2c_master_send(pn544_dev->client, + (unsigned char *)(uintptr_t) + I2CDMAWriteBuf, count); + #endif + + if (ret_tmp != count) { + pr_debug("%s : i2c_master_send returned %d\n", __func__, + ret); + ret = -EIO; + break; + } + + ret += ret_tmp; + /* pr_debug("%s : ret_tmp=%d,ret=%d,count_ori=%d\n", __func__, + ret_tmp, ret, count_ori); */ + + if (ret == count_ori) { + /* pr_debug("%s : ret == count_ori\n", __func__); */ + dev_info(&pn544_dev->client->dev, "ret == count_ori"); + break; + } + + if (count_remain > MAX_BUFFER_SIZE) { + count = MAX_BUFFER_SIZE; + count_remain -= MAX_BUFFER_SIZE; + } else { + count = count_remain; + count_remain = 0; + } + idx++; + } + + pn544_enable_irq(nfc_irq,pn544_dev); + +#if NFC_DEBUG + pr_debug("%s : writing %d bytes. Status %d\n", __func__, count_ori, + ret); + + dev_info(&pn544_dev->client->dev, "pn544 : - w\n"); +#endif + + return ret; +} + +static int pn544_dev_open(struct inode *inode, struct file *filp) +{ + + struct pn544_dev *pn544_dev = + container_of(filp->private_data, struct pn544_dev, pn544_device); + + filp->private_data = pn544_dev; + pn544_dev_ptr = pn544_dev; + + pr_debug("pn544_dev_open,%s : %d,%d, &pn544_dev_open=%p\n", __func__, + imajor(inode), iminor(inode), pn544_dev_ptr); + + forceExitBlockingRead = 0; + return 0; +} + +static long pn544_dev_unlocked_ioctl(struct file *filp, unsigned int cmd, + unsigned long arg) +{ + + struct pn544_dev *pn544_dev = filp->private_data; + int ret = 0;//[VY36 M] + + + switch (cmd) { + case PN544_SET_PWR: + if (arg == 2) { + /* power on with firmware download (requires hw reset) */ + + ret = mt_nfc_pinctrl_select(gpctrl, st_ven_h); + ret = mt_nfc_pinctrl_select(gpctrl, st_eint_h); + usleep_range(10000, 10000); + ret = mt_nfc_pinctrl_select(gpctrl, st_ven_l); + usleep_range(10000, 10000); + ret = mt_nfc_pinctrl_select(gpctrl, st_ven_h); + usleep_range(10000, 10000); + /* + if (atomic_read(&pn544_dev->irq_enabled) == 0) { + atomic_set(&pn544_dev->irq_enabled, 1); + pn544_enable_irq(nfc_irq); + }*/ + pn544_enable_irq(nfc_irq,pn544_dev); + dev_info(&pn544_dev->client->dev, "%s power on with firmware, irq=%d\n", __func__, pn544_dev->irq_enabled); + + } + else if (arg == 1) + { + /* power on */ + + ret = mt_nfc_pinctrl_select(gpctrl, st_eint_l); + ret = mt_nfc_pinctrl_select(gpctrl, st_ven_h); + usleep_range(10000, 10000); + /* + if (atomic_read(&pn544_dev->irq_enabled) == 0) { + atomic_set(&pn544_dev->irq_enabled, 1); + pn544_enable_irq(nfc_irq); + }*/ + pn544_enable_irq(nfc_irq,pn544_dev); + dev_info(&pn544_dev->client->dev, "%s power on, irq=%d\n", __func__, pn544_dev->irq_enabled); + } + else if (arg == 0) + { + /* power off */ + /* + if (atomic_read(&pn544_dev->irq_enabled) == 1) { + pn544_disable_irq(nfc_irq); + atomic_set(&pn544_dev->irq_enabled, 0); + }*/ + dev_info(&pn544_dev->client->dev, "%s power off, irq=%d\n", __func__, pn544_dev->irq_enabled); + ret = mt_nfc_pinctrl_select(gpctrl, st_eint_l); + ret = mt_nfc_pinctrl_select(gpctrl, st_ven_l); + + pn544_disable_irq(nfc_irq,pn544_dev); + wake_unlock(&nfc_wake_lock); + + } +#ifdef NXP_KR_READ_IRQ_MODIFY + else if (arg == 3) { + pr_info("%s Read Cancle\n", __func__); + cancle_read = true; + do_reading = true; + wake_up(&pn544_dev->read_wq); + } +#endif + else + { + dev_err(&pn544_dev->client->dev, "%s bad arg %lu\n", __func__, arg); + return -EINVAL; + } + + return ret;//[VY36 M] JackBB 2015/10/12 Kernel standardization + + break; + default: + dev_err(&pn544_dev->client->dev, "%s bad ioctl %u\n", __func__, cmd); + return -EINVAL; + } + + return 0; +} + +#if !defined(CONFIG_MTK_LEGACY) +static int mt_nfc_pinctrl_select(struct pinctrl *p, struct pinctrl_state *s) +{ + int ret = 0; + + if (p != NULL && s != NULL) { + ret = pinctrl_select_state(p, s); + } else { + pr_debug("%s : pinctrl_select err\n", __func__); + ret = -1; + } + return ret; +} + +static int mt_nfc_pinctrl_init(struct platform_device *pdev) +{ + int ret = 0; + + gpctrl = devm_pinctrl_get(&pdev->dev); + if (IS_ERR(gpctrl)) { + dev_err(&pdev->dev, "Cannot find pinctrl!"); + ret = PTR_ERR(gpctrl); + goto end; + } + + st_ven_h = pinctrl_lookup_state(gpctrl, "ven_high"); + if (IS_ERR(st_ven_h)) { + ret = PTR_ERR(st_ven_h); + pr_debug("%s : pinctrl err, ven_high\n", __func__); + } + if (st_ven_h == NULL) { + pr_err("%s : st_ven_h is NULL\n", __func__); + goto end; + } + + st_ven_l = pinctrl_lookup_state(gpctrl, "ven_low"); + if (IS_ERR(st_ven_l)) { + ret = PTR_ERR(st_ven_l); + pr_debug("%s : pinctrl err, ven_low\n", __func__); + } + if (st_ven_l == NULL) { + pr_err("%s : st_ven_l is NULL\n", __func__); + goto end; + } + + st_rst_h = pinctrl_lookup_state(gpctrl, "rst_high"); + if (IS_ERR(st_rst_h)) { + ret = PTR_ERR(st_rst_h); + pr_debug("%s : pinctrl err, rst_high\n", __func__); + } + if (st_rst_h == NULL) { + pr_err("%s : st_rst_h is NULL\n", __func__); + goto end; + } + + st_rst_l = pinctrl_lookup_state(gpctrl, "rst_low"); + if (IS_ERR(st_rst_l)) { + ret = PTR_ERR(st_rst_l); + pr_debug("%s : pinctrl err, rst_low\n", __func__); + } + if (st_rst_l == NULL) { + pr_err("%s : st_rst_l is NULL\n", __func__); + goto end; + } + + st_eint_h = pinctrl_lookup_state(gpctrl, "eint_high"); + if (IS_ERR(st_eint_h)) { + ret = PTR_ERR(st_eint_h); + pr_debug("%s : pinctrl err, eint_high\n", __func__); + } + if (st_eint_h == NULL) { + pr_err("%s : st_eint_h is NULL\n", __func__); + goto end; + } + + st_eint_l = pinctrl_lookup_state(gpctrl, "eint_low"); + if (IS_ERR(st_eint_l)) { + ret = PTR_ERR(st_eint_l); + pr_debug("%s : pinctrl err, eint_low\n", __func__); + } + if (st_eint_l == NULL) { + pr_err("%s : st_eint_l is NULL\n", __func__); + goto end; + } + + st_irq_init = pinctrl_lookup_state(gpctrl, "irq_init"); + if (IS_ERR(st_irq_init)) { + ret = PTR_ERR(st_irq_init); + pr_debug("%s : pinctrl err, irq_init\n", __func__); + } + if (st_irq_init == NULL) { + pr_err("%s : st_irq_init is NULL\n", __func__); + goto end; + } +#if 0 + st_osc_init = pinctrl_lookup_state(gpctrl, "osc_init"); + if (IS_ERR(st_osc_init)) { + ret = PTR_ERR(st_osc_init); + pr_debug("%s : pinctrl err, osc_init\n", __func__); + } + if (st_osc_init == NULL) { + pr_err("%s : st_osc_init is NULL\n", __func__); + goto end; + } +#endif + /* select state */ +#if 0 + ret = mt_nfc_pinctrl_select(gpctrl, st_osc_init); + usleep_range(900, 1000); +#endif + ret = mt_nfc_pinctrl_select(gpctrl, st_irq_init); + usleep_range(900, 1000); + + ret = mt_nfc_pinctrl_select(gpctrl, st_ven_l); + usleep_range(900, 1000); + + pr_info("pn544 : set st_ven_l in mt_nfc_pinctrl_init()\n"); + + ret = mt_nfc_pinctrl_select(gpctrl, st_rst_l); + usleep_range(900, 1000); + + ret = mt_nfc_pinctrl_select(gpctrl, st_eint_l); + +end: + + return ret; +} + +static int mt_nfc_gpio_init(void) +{ + + struct device_node *node; + + node = of_find_compatible_node(NULL, NULL, "mediatek,nfc-gpio-v2"); + + if (node) { + pn544_platform_data.ven_gpio = of_get_named_gpio(node, "gpio-ven", 0); + pn544_platform_data.sysrstb_gpio = of_get_named_gpio(node, "gpio-rst", 0); + pn544_platform_data.eint_gpio = of_get_named_gpio(node, "gpio-eint", 0); + pn544_platform_data.irq_gpio = of_get_named_gpio(node, "gpio-irq", 0); + } else { + pr_debug("%s : get gpio num err.\n", __func__); + return -1; + } + + _gpn544_dev.irq_gpio = pn544_platform_data.irq_gpio; + _gpn544_dev.ven_gpio = pn544_platform_data.ven_gpio; + _gpn544_dev.sysrstb_gpio = pn544_platform_data.sysrstb_gpio; + _gpn544_dev.eint_gpio = pn544_platform_data.eint_gpio; + + return 0; +} +#endif + +static int mt_nfc_get_gpio_value(int gpio_num) +{ + int value = 0; + + pr_info("%s : gpio_num = %d \n", __func__ , gpio_num);//BBTEST + + if (mt_nfc_get_gpio_dir(gpio_num) != MTK_NFC_GPIO_DIR_INVALID) { +#if !defined(CONFIG_MTK_LEGACY) + value = __gpio_get_value(gpio_num); +#else + value = mt_get_gpio_in(gpio_num); +#endif + } + + return value; +} + +static int mt_nfc_get_gpio_dir(int gpio_num) +{ + if (gpio_num == pn544_platform_data.irq_gpio) { + return MTK_NFC_GPIO_DIR_IN; /* input */ + + } else if ((gpio_num == pn544_platform_data.ven_gpio) || + (gpio_num == pn544_platform_data.sysrstb_gpio) || + (gpio_num == pn544_platform_data.eint_gpio)) { + return MTK_NFC_GPIO_DIR_OUT; /* output */ + + } else { + return MTK_NFC_GPIO_DIR_INVALID; + + } +} + +/* + * module load/unload record keeping + */ +static int __init pn544_dev_init(void) +{ + int ret; + + pr_debug("pn544_dev_init\n"); + /* i2c_register_board_info(NFC_I2C_BUSNUM, &nfc_board_info, 1); */ + /* pr_debug("pn544_dev_init2\n"); */ + + platform_driver_register(&mtk_nfc_platform_driver); + + i2c_add_driver(&pn544_dev_driver); + + _gpn544_dev.pn544_device.minor = MISC_DYNAMIC_MINOR; + _gpn544_dev.pn544_device.name = "pn54x"; + _gpn544_dev.pn544_device.fops = &pn544_dev_fops; + + ret = misc_register(&_gpn544_dev.pn544_device); + if (ret) { + pr_debug("%s : misc_register failed\n", __FILE__); + return ret; + } + + pr_debug("pn544_dev_init success\n"); + return 0; +} + +static void __exit pn544_dev_exit(void) +{ + pr_debug("pn544_dev_exit\n"); + + i2c_del_driver(&pn544_dev_driver); +} + +module_init(pn544_dev_init); +module_exit(pn544_dev_exit); + +MODULE_AUTHOR("LiangChi Huang"); +MODULE_DESCRIPTION("MTK NFC driver"); +MODULE_LICENSE("GPL"); diff --git a/drivers/spi/mediatek/mt6757/mtk_spi.h b/drivers/spi/mediatek/mt6757/mtk_spi.h index ce6b0f54b..aa9039f36 100644 --- a/drivers/spi/mediatek/mt6757/mtk_spi.h +++ b/drivers/spi/mediatek/mt6757/mtk_spi.h @@ -109,7 +109,7 @@ struct mt_chip_conf { enum spi_tckdly tckdly; }; -void secspi_enable_clk(struct spi_device *spidev); +//void secspi_enable_clk(struct spi_device *spidev); #ifdef CONFIG_TRUSTONIC_TEE_SUPPORT int secspi_session_open(u32 spinum); int secspi_execute(u32 cmd, tciSpiMessage_t *param); diff --git a/include/linux/nfc/pn547.h b/include/linux/nfc/pn547.h new file mode 100644 index 000000000..ef2416b12 --- /dev/null +++ b/include/linux/nfc/pn547.h @@ -0,0 +1,30 @@ +/* + * Copyright (C) 2010 Trusted Logic S.A. + * + * 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, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#define PN544_MAGIC 0xE9 + +/* + * PN544 power control via ioctl + * PN544_SET_PWR(0): power off + * PN544_SET_PWR(1): power on + * PN544_SET_PWR(2): reset and power on with firmware download enabled + */ +#define PN544_SET_PWR _IOW(PN544_MAGIC, 0x01, long) + +/* Still use the devices name "pn544" due to following libnfc original design*/ +//#define NFC_I2C_DEV_NAME "pn544"