/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * Defines the PN80T spidev device and platform wrappers consumed in * the common code. */ #include <fcntl.h> #include <limits.h> #include <linux/spi/spidev.h> #include <stdint.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/ioctl.h> #include <unistd.h> #include "../include/ese/hw/nxp/pn80t/common.h" #include "../include/ese/hw/nxp/spi_board.h" struct Handle { int spi_fd; struct NxpSpiBoard *board; }; int gpio_set(int num, int val) { char val_path[256]; char val_chr = (val ? '1' : '0'); int fd; if (num < 0) { return 0; } if (snprintf(val_path, sizeof(val_path), "/sys/class/gpio/gpio%d/value", num) >= (int)sizeof(val_path)) { return -1; } printf("Gpio @ %s\n", val_path); fd = open(val_path, O_WRONLY); if (fd < 0) { return -1; } if (write(fd, &val_chr, 1) < 0) { close(fd); return -1; } close(fd); return 0; } int platform_toggle_ven(void *blob, int val) { struct Handle *handle = blob; printf("Toggling VEN: %d\n", val); return gpio_set(handle->board->gpios[kBoardGpioNfcVen], val); } int platform_toggle_reset(void *blob, int val) { struct Handle *handle = blob; printf("Toggling RST: %d\n", val); return gpio_set(handle->board->gpios[kBoardGpioEseRst], val); } int platform_toggle_power_req(void *blob, int val) { struct Handle *handle = blob; printf("Toggling SVDD_PWR_REQ: %d\n", val); return gpio_set(handle->board->gpios[kBoardGpioEseSvddPwrReq], val); } int gpio_configure(int num, int out, int val) { char dir_path[256]; char numstr[8]; char dir[5]; int fd; /* <0 is unmapped. No work to do! */ if (num < 0) { return 0; } if (snprintf(dir, sizeof(dir), "%s", (out ? "out" : "in")) >= (int)sizeof(dir)) { return -1; } if (snprintf(dir_path, sizeof(dir_path), "/sys/class/gpio/gpio%d/direction", num) >= (int)sizeof(dir_path)) { return -1; } if (snprintf(numstr, sizeof(numstr), "%d", num) >= (int)sizeof(numstr)) { return -1; } fd = open("/sys/class/gpio/export", O_WRONLY); if (fd < 0) { return -1; } /* Exporting can only happen once, so instead of stat()ing, just ignore * errors. */ (void)write(fd, numstr, strlen(numstr)); close(fd); fd = open(dir_path, O_WRONLY); if (fd < 0) { return -1; } if (write(fd, dir, strlen(dir)) < 0) { close(fd); return -1; } close(fd); return gpio_set(num, val); } void *platform_init(void *hwopts) { struct NxpSpiBoard *board = hwopts; struct Handle *handle; int gpio = 0; handle = malloc(sizeof(*handle)); if (!handle) { return NULL; } handle->board = board; /* Initialize the mapped GPIOs */ for (; gpio < kBoardGpioMax; ++gpio) { if (gpio_configure(board->gpios[gpio], 1, 1) < 0) { free(handle); return NULL; } } handle->spi_fd = open(board->dev_path, O_RDWR); if (handle->spi_fd < 0) { free(handle); return NULL; } /* If we need anything fancier, we'll need MODE32 in the headers. */ if (ioctl(handle->spi_fd, SPI_IOC_WR_MODE, &board->mode) < 0) { close(handle->spi_fd); free(handle); return NULL; } if (ioctl(handle->spi_fd, SPI_IOC_WR_BITS_PER_WORD, &board->bits) < 0) { close(handle->spi_fd); free(handle); return NULL; } if (ioctl(handle->spi_fd, SPI_IOC_WR_MAX_SPEED_HZ, &board->speed) < 0) { close(handle->spi_fd); free(handle); return NULL; } printf("Linux SPIDev initialized\n"); return (void *)handle; } int platform_release(void *blob) { struct Handle *handle = blob; close(handle->spi_fd); free(handle); /* Note, we don't unconfigure the GPIOs. */ return 0; } int platform_wait(void *blob __attribute__((unused)), long usec) { return usleep((useconds_t)usec); } uint32_t spidev_transmit(struct EseInterface *ese, const uint8_t *buf, uint32_t len, int complete) { struct NxpState *ns = NXP_PN80T_STATE(ese); struct Handle *handle = ns->handle; struct spi_ioc_transfer tr = { .tx_buf = (unsigned long)buf, .rx_buf = 0, .len = (uint32_t)len, .delay_usecs = 0, .speed_hz = 0, .bits_per_word = 0, .cs_change = !!complete, }; ssize_t ret = -1; ALOGV("spidev:%s: called [%d]", __func__, len); if (len > INT_MAX) { ese_set_error(ese, kNxpPn80tErrorTransmitSize); ALOGE("Unexpectedly large transfer attempted: %u", len); return 0; } ret = ioctl(handle->spi_fd, SPI_IOC_MESSAGE(1), &tr); if (ret < 1) { ese_set_error(ese, kNxpPn80tErrorTransmit); ALOGE("%s: failed to write to hw (ret=%zd)", __func__, ret); return 0; } return len; } uint32_t spidev_receive(struct EseInterface *ese, uint8_t *buf, uint32_t len, int complete) { struct NxpState *ns = NXP_PN80T_STATE(ese); struct Handle *handle = ns->handle; ssize_t ret = -1; struct spi_ioc_transfer tr = { .tx_buf = 0, .rx_buf = (unsigned long)buf, .len = (uint32_t)len, .delay_usecs = 0, .speed_hz = 0, .bits_per_word = 0, .cs_change = !!complete, }; ALOGV("spidev:%s: called [%d]", __func__, len); if (len > INT_MAX) { ese_set_error(ese, kNxpPn80tErrorReceiveSize); ALOGE("Unexpectedly large receive attempted: %u", len); return 0; } ret = ioctl(handle->spi_fd, SPI_IOC_MESSAGE(1), &tr); if (ret < 1) { ALOGE("%s: failed to read from hw (ret=%zd)", __func__, ret); ese_set_error(ese, kNxpPn80tErrorReceive); return 0; } ALOGV("%s: read bytes: %zd", __func__, len); return len; } static const struct Pn80tPlatform kPn80tLinuxSpidevPlatform = { .initialize = &platform_init, .release = &platform_release, .toggle_reset = &platform_toggle_reset, .toggle_ven = &platform_toggle_ven, .toggle_power_req = &platform_toggle_power_req, .wait = &platform_wait, }; static const struct EseOperations ops = { .name = "NXP PN80T/PN81A (PN553)", .open = &nxp_pn80t_open, .hw_receive = &spidev_receive, .hw_transmit = &spidev_transmit, .hw_reset = &nxp_pn80t_reset, .transceive = &nxp_pn80t_transceive, .poll = &nxp_pn80t_poll, .close = &nxp_pn80t_close, .opts = &kPn80tLinuxSpidevPlatform, .errors = kNxpPn80tErrorMessages, .errors_count = kNxpPn80tErrorMax, }; __attribute__((visibility("default"))) ESE_DEFINE_HW_OPS(ESE_HW_NXP_PN80T_SPIDEV, ops);