blob: f9bb53dbeb578528319dc3991076397601d18163 [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2019-2020, The Linux foundation. All rights reserved.
*/
#include <linux/cdev.h>
#include <linux/circ_buf.h>
#include <linux/clk.h>
#include <linux/debugfs.h>
#include <linux/delay.h>
#include <linux/dma-mapping.h>
#include <linux/err.h>
#include <linux/gpio.h>
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/ioctl.h>
#include <linux/kernel.h>
#include <linux/kfifo.h>
#include <linux/ktime.h>
#include <linux/list.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_device.h>
#include <linux/of_gpio.h>
#include <linux/platform_device.h>
#include <linux/pm_runtime.h>
#include <linux/poll.h>
#include <linux/spi/spi.h>
#include <linux/time.h>
#include <linux/types.h>
#include <linux/uaccess.h>
#include <uapi/misc/wigig_sensing_uapi.h>
#include "wigig_sensing.h"
#define DRV_NAME "wigig_sensing"
#define CLEAR_LOW_23_BITS ~(BIT(24) - 1)
#define NUM_RETRIES 5
#define FW_TIMEOUT_MSECS (1000)
#define circ_cnt(circ, size) \
((CIRC_CNT((circ)->head, (circ)->tail, size)) & ~3)
#define circ_cnt_to_end(circ, size) \
((CIRC_CNT_TO_END((circ)->head, (circ)->tail, size)) & ~3)
#define circ_space(circ, size) \
((CIRC_SPACE((circ)->head, (circ)->tail, size)) & ~3)
#define circ_space_to_end(circ, size) \
((CIRC_SPACE_TO_END((circ)->head, (circ)->tail, size)) & ~3)
#ifdef CONFIG_DEBUG_FS
#define SPI_STATS_MEAS_INIT(ctx, idx, name_str) \
do { \
strlcpy((ctx)->spi_stats[idx].name, name_str, \
SPI_STATS_MAX_NAME_LEN); \
atomic64_set(&(ctx)->spi_stats[idx].min, U64_MAX); \
} while (0)
#define SPI_STATS_MEAS_START(ctx, idx) (ctx)->spi_stats[idx].start = ktime_get()
#define SPI_STATS_MEAS_STOP(ctx, idx) \
do { \
struct spi_stats *ss = &(ctx)->spi_stats[idx]; \
u64 min = atomic64_read(&ss->min); \
u64 max = atomic64_read(&ss->max); \
\
ss->delta = ktime_sub(ktime_get(), ss->start); \
atomic64_set(&ss->min, (ktime_to_us(ss->delta) != 0) ? \
min_t(u64, min, ss->delta) : min); \
atomic64_set(&ss->max, max_t(u64, max, ss->delta)); \
atomic64_add(ss->delta, &ss->acc); \
atomic_inc(&ss->num_meas); \
} while (0)
#else /* CONFIG_DEBUG_FS */
#define SPI_STATS_MEAS_INIT(ctx, idx, name_str)
#define SPI_STATS_MEAS_START(ctx, idx)
#define SPI_STATS_MEAS_STOP(ctx, idx)
#endif /* CONFIG_DEBUG_FS */
struct wigig_sensing_platform_data {
struct gpio_desc *dri_gpio;
};
struct wigig_sensing_ctx *ctx;
static int spis_reset(struct spi_device *spi)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = { 0xDA, 0xBA, 0x00, 0x0 };
int rc;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc = spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_clock_request(struct spi_device *spi)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = { 0xDA, 0xBA, 0x80, 0x0 };
int rc;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc = spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_nop(struct spi_device *spi)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = { 0x0 };
int rc;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc = spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_write_enable(struct spi_device *spi)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = { 0x6 };
int rc;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc = spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_extended_reset(struct spi_device *spi)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[SPIS_EXTENDED_RESET_COMMAND_LEN] = { 0xDA, 0xBA };
int rc;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc = spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_read_internal_reg(struct spi_device *spi, u8 address, u32 *val)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = { 0x81, address, 0, 0 };
struct spi_transfer tx_xfer = {
.tx_buf = ctx->cmd_buf,
.rx_buf = NULL,
.len = 4,
.bits_per_word = 8,
};
struct spi_transfer rx_xfer = {
.tx_buf = NULL,
.rx_buf = ctx->cmd_reply_buf,
.len = 4,
.bits_per_word = 8,
};
int rc;
struct spi_message msg;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
spi_message_init(&msg);
spi_message_add_tail(&tx_xfer, &msg);
spi_message_add_tail(&rx_xfer, &msg);
rc = spi_sync(ctx->spi_dev, &msg);
*val = be32_to_cpu(*(u32 *)(ctx->cmd_reply_buf));
return rc;
}
static int spis_read_mem(struct spi_device *spi, u32 address, u32 *val)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = {
0x83,
(address >> 16) & 0xFF,
(address >> 8) & 0xFF,
(address >> 0) & 0xFF,
0, 0, 0, 0
};
struct spi_transfer tx_xfer = {
.tx_buf = ctx->cmd_buf,
.rx_buf = NULL,
.len = 8,
.bits_per_word = 8,
};
struct spi_transfer rx_xfer = {
.tx_buf = NULL,
.rx_buf = ctx->cmd_reply_buf,
.len = 4,
.bits_per_word = 8,
};
int rc;
struct spi_message msg;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
spi_message_init(&msg);
spi_message_add_tail(&tx_xfer, &msg);
spi_message_add_tail(&rx_xfer, &msg);
rc = spi_sync(ctx->spi_dev, &msg);
*val = be32_to_cpu(*(u32 *)(ctx->cmd_reply_buf));
return rc;
}
static int spis_write_internal_reg(struct spi_device *spi, u8 address, u32 val)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = { 0x82, address, 0, 0, 0, 0 };
int rc = spis_write_enable(ctx->spi_dev);
cmd[2] = (val >> 24) & 0xFF;
cmd[3] = (val >> 16) & 0xFF;
cmd[4] = (val >> 8) & 0xFF;
cmd[5] = (val >> 0) & 0xFF;
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc |= spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_write_mem(struct spi_device *spi, u32 address, u32 val)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
u8 cmd[] = {
0x02,
(address >> 16) & 0xFF,
(address >> 8) & 0xFF,
(address >> 0) & 0xFF,
(val >> 24) & 0xFF,
(val >> 16) & 0xFF,
(val >> 8) & 0xFF,
(val >> 0) & 0xFF
};
int rc = spis_write_enable(ctx->spi_dev);
memcpy(ctx->cmd_buf, cmd, sizeof(cmd));
rc |= spi_write(spi, ctx->cmd_buf, sizeof(cmd));
return rc;
}
static int spis_write_reg(struct spi_device *spi, u32 addr, u32 val)
{
int rc;
if (addr < 256)
rc = spis_write_internal_reg(spi, addr, val);
else
rc = spis_write_mem(spi, addr, val);
pr_debug("write reg 0x%X val 0x%X\n", addr, val);
return rc;
}
static int spis_block_read_mem(struct spi_device *spi,
u32 address,
u8 *data,
u32 length,
u32 last_read_length)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
int rc;
int overhead = OPCODE_WIDTH + ADDR_WIDTH + DUMMY_BYTES_WIDTH;
int sz = overhead + length;
struct spi_transfer xfer = {
.tx_buf = ctx->tx_buf,
.rx_buf = ctx->rx_buf,
.len = sz,
.bits_per_word = 32,
};
u32 frame = 0xB << 24;
frame |= address & 0xFFFFFF;
frame = cpu_to_le32(frame);
/* Read length must be in 32 bit units */
if (length & 3) {
pr_err("Read length must be a multiple of 32 bits\n");
return -EINVAL;
}
if (length > SPI_MAX_TRANSACTION_SIZE) {
pr_err("Read length too large\n");
return -EINVAL;
}
/* Write transfer length to SPI core */
if (length != last_read_length) {
rc = spis_write_reg(ctx->spi_dev,
SPIS_TRNS_LEN_REG_ADDR,
length << 16);
if (rc) {
pr_err("Failed setting SPIS_TRNS_LEN_REG_ADDR\n");
return rc;
}
ctx->last_read_length = length;
}
memcpy(ctx->tx_buf, &frame, sizeof(frame));
/* Execute transaction */
rc = spi_sync_transfer(spi, &xfer, 1);
if (rc) {
pr_err("SPI transaction failed, rc = %d\n", rc);
return rc;
}
memcpy(data, ctx->rx_buf + overhead, length);
pr_debug("Sent block_read_mem command, addr=0x%X, length=%u\n",
address, length);
return rc;
}
static int spis_read_reg(struct spi_device *spi, u32 addr, u32 *val)
{
int rc;
if (addr < 256)
rc = spis_read_internal_reg(spi, addr, val);
else
rc = spis_read_mem(spi, addr, val);
pr_debug("read reg 0x%X, val = 0x%X\n", addr, *val);
return rc;
}
#define RGF_SPI_FIFO_CONTROL_ADDR (0x8800A0)
#define RGF_SPI_FIFO_WR_PTR_ADDR (0x88009C)
#define RGF_SPI_FIFO_RD_PTR_ADDR (0x880098)
#define RGF_SPI_FIFO_BASE_ADDR_ADDR (0x880094)
#define RGF_SPI_CONTROL_ADDR (0x880090)
#define RGF_SPI_CONFIG_ADDR (0x88008C)
static int cache_fifo_regs(struct wigig_sensing_ctx *ctx)
{
int rc;
struct spi_device *spi = ctx->spi_dev;
struct spi_fifo *f = &ctx->spi_fifo;
rc = spis_read_reg(spi, RGF_SPI_CONFIG_ADDR, &f->config.v);
rc |= spis_read_reg(spi, RGF_SPI_CONTROL_ADDR, &f->control.v);
rc |= spis_read_reg(spi, RGF_SPI_FIFO_BASE_ADDR_ADDR, &f->base_addr);
rc |= spis_read_reg(spi, RGF_SPI_FIFO_RD_PTR_ADDR, &f->rd_ptr);
rc |= spis_read_reg(spi, RGF_SPI_FIFO_WR_PTR_ADDR, &f->wr_ptr);
if (rc) {
pr_err("%s failed, rc=%u\n", __func__, rc);
return -EFAULT;
}
/* Update read pointer to be the FIFO base address */
f->rd_ptr = ctx->spi_fifo.base_addr;
pr_debug("SPI FIFO base address = 0x%X\n", f->base_addr);
pr_debug("SPI FIFO size = 0x%X\n", f->config.b.size);
pr_debug("SPI FIFO enable = %u\n", f->config.b.enable);
pr_debug("SPI FIFO read pointer = 0x%X\n", f->rd_ptr);
pr_debug("SPI FIFO write pointer = 0x%X\n", f->wr_ptr);
return 0;
}
static int spis_init(struct wigig_sensing_ctx *ctx)
{
int rc;
u32 spis_sanity_reg = 0;
u32 jtag_id = 0;
struct spi_device *spi = ctx->spi_dev;
/* Initialize SPI */
spi->max_speed_hz = 32e6;
spi->bits_per_word = 8;
spi->mode = SPI_MODE_0;
rc = spi_setup(spi);
if (rc) {
pr_err("spi_setup() failed (%d)\n", rc);
return rc;
}
rc = spis_reset(spi);
rc |= spis_nop(spi);
rc |= spis_clock_request(spi);
rc |= spis_nop(spi);
rc |= spis_read_internal_reg(spi, SPIS_SANITY_REG_ADDR,
&spis_sanity_reg);
pr_debug("sanity = 0x%x\n", spis_sanity_reg);
/* use 4 dummy bytes to make block read/writes 32-bit aligned */
rc |= spis_write_reg(spi, SPIS_CFG_REG_ADDR, SPIS_CONFIG_REG_OPT_VAL);
rc |= spis_read_mem(spi, JTAG_ID_REG_ADDR, &jtag_id);
pr_debug("jtag_id = 0x%x\n", jtag_id);
if (rc || spis_sanity_reg != SPIS_SANITY_REG_VAL ||
jtag_id != JTAG_ID) {
pr_err("SPI init failed, sanity=0x%X, jtag_id=0x%X\n",
spis_sanity_reg, jtag_id);
return -EFAULT;
}
/* Copy FIFO register values from the HW */
rc = cache_fifo_regs(ctx);
if (rc)
pr_err("cache_fifo_regs() failed\n");
return rc;
}
static enum wigig_sensing_stm_e convert_mode_to_state(
enum wigig_sensing_mode mode)
{
switch (mode) {
case WIGIG_SENSING_MODE_SEARCH:
return WIGIG_SENSING_STATE_SEARCH;
case WIGIG_SENSING_MODE_FACIAL_RECOGNITION:
return WIGIG_SENSING_STATE_FACIAL;
case WIGIG_SENSING_MODE_GESTURE_DETECTION:
return WIGIG_SENSING_STATE_GESTURE;
case WIGIG_SENSING_MODE_STOP:
return WIGIG_SENSING_STATE_READY_STOPPED;
case WIGIG_SENSING_MODE_CUSTOM:
return WIGIG_SENSING_STATE_CUSTOM;
default:
return WIGIG_SENSING_STATE_MIN;
}
}
static int wigig_sensing_send_change_mode_command(
struct wigig_sensing_ctx *ctx,
enum wigig_sensing_mode mode,
u32 channel)
{
int rc = 0;
union user_rgf_spi_mbox_inb inb_reg = { { 0 } };
pr_debug("mode=%d, channel=%d\n", mode, channel);
inb_reg.b.mode = mode;
inb_reg.b.channel_request = channel;
ctx->inb_cmd = inb_reg;
mutex_lock(&ctx->spi_lock);
rc = spis_extended_reset(ctx->spi_dev);
pr_debug("Sent extended reset command, rc = %d\n", rc);
mutex_unlock(&ctx->spi_lock);
if (rc) {
pr_err("failed to send extended reset\n");
return -EFAULT;
}
ctx->stm.waiting_for_deep_sleep_exit = true;
return 0;
}
static int wigig_sensing_change_state(struct wigig_sensing_ctx *ctx,
struct wigig_sensing_stm *state,
enum wigig_sensing_stm_e new_state)
{
enum wigig_sensing_stm_e curr_state;
bool transition_allowed = true;
int rc = 0;
if (!state) {
pr_err("state is NULL\n");
return -EINVAL;
}
if (new_state <= WIGIG_SENSING_STATE_MIN ||
new_state >= WIGIG_SENSING_STATE_MAX) {
pr_err("new_state (%d) is invalid\n", new_state);
return -EINVAL;
}
curr_state = state->state;
/* Moving to SYS_ASSEERT state is always allowed */
if (new_state == WIGIG_SENSING_STATE_SYS_ASSERT)
goto skip;
/*
* Moving from INITIALIZED state is allowed only to READY_STOPPED state
* and only when spi_ready is set
*/
else if (curr_state == WIGIG_SENSING_STATE_INITIALIZED &&
(new_state != WIGIG_SENSING_STATE_READY_STOPPED ||
!ctx->stm.spi_ready)) {
transition_allowed = false;
rc = -EFAULT;
}
/*
* Moving to GET_PARAMS state is allowed only from READY_STOPPED state
*/
else if (curr_state != WIGIG_SENSING_STATE_READY_STOPPED &&
new_state == WIGIG_SENSING_STATE_GET_PARAMS) {
transition_allowed = false;
rc = -EFAULT;
}
/*
* Moving from GET_PARAMS state is allowed only to READY_STOPPED state
*/
else if (curr_state == WIGIG_SENSING_STATE_GET_PARAMS &&
new_state != WIGIG_SENSING_STATE_READY_STOPPED) {
transition_allowed = false;
rc = -EFAULT;
}
/*
* Moving from SYS_ASSERT state is allowed only to READY_STOPPED state
*/
else if (curr_state == WIGIG_SENSING_STATE_SYS_ASSERT &&
new_state != WIGIG_SENSING_STATE_READY_STOPPED) {
transition_allowed = false;
rc = -ENODEV;
}
skip:
if (transition_allowed) {
pr_info("state transition (%d) --> (%d)\n", curr_state,
new_state);
state->state = new_state;
} else {
pr_err("state transition rejected (%d) xx> (%d)\n",
curr_state, new_state);
}
return rc;
}
static int wigig_sensing_ioc_set_auto_recovery(struct wigig_sensing_ctx *ctx)
{
pr_info("Handling WIGIG_SENSING_IOCTL_SET_AUTO_RECOVERY\n");
pr_info("NOT SUPPORTED\n");
ctx->stm.auto_recovery = true;
return 0;
}
static int wigig_sensing_ioc_get_mode(struct wigig_sensing_ctx *ctx)
{
return ctx->stm.mode;
}
static int wigig_sensing_ioc_change_mode(struct wigig_sensing_ctx *ctx,
struct wigig_sensing_change_mode *req)
{
struct wigig_sensing_stm sim_state;
int rc;
u32 ch;
if (req == NULL)
return -EINVAL;
pr_info("mode = %d, channel = %d, has_channel = %d\n",
req->mode, req->channel, req->has_channel);
if (!ctx)
return -EINVAL;
/* Save the request for later use */
ctx->stm.mode_request = req->mode;
/* Simulate a state change */
ctx->stm.state_request = convert_mode_to_state(req->mode);
sim_state = ctx->stm;
rc = wigig_sensing_change_state(ctx, &sim_state,
ctx->stm.state_request);
if (rc) {
pr_err("State change not allowed\n");
goto End;
}
/* Send command to FW */
mutex_lock(&ctx->dri_lock);
ctx->stm.change_mode_in_progress = true;
ch = req->has_channel ? req->channel : 0;
ctx->stm.channel_request = ch;
ctx->stm.burst_size_ready = false;
/* Change mode command must not be called during DRI processing */
rc = wigig_sensing_send_change_mode_command(ctx, req->mode, ch);
mutex_unlock(&ctx->dri_lock);
if (rc) {
pr_err("wigig_sensing_send_change_mode_command() failed, err %d\n",
rc);
goto End;
}
/* Put the calling process to sleep until we get a response from FW */
rc = wait_event_interruptible_timeout(
ctx->cmd_wait_q,
ctx->stm.burst_size_ready,
msecs_to_jiffies(FW_TIMEOUT_MSECS));
if (rc < 0) {
/* Interrupted by a signal */
pr_err("wait_event_interruptible_timeout() interrupted by a signal (%d)\n",
rc);
goto End;
} else if (rc == 0) {
/* Timeout, FW did not respond in time */
pr_err("wait_event_interruptible_timeout() timed out\n");
rc = -ETIME;
goto End;
} else {
/* rc > 0, this is fine, set rc to 0 */
rc = 0;
}
if (ctx->stm.state != ctx->stm.state_request) {
pr_err("%s() failed\n", __func__);
if (ctx->stm.state == WIGIG_SENSING_STATE_SYS_ASSERT)
rc = -ENODEV;
else
rc = -EFAULT;
}
End:
ctx->stm.state_request = WIGIG_SENSING_STATE_MIN;
ctx->stm.channel_request = 0;
ctx->stm.mode_request = WIGIG_SENSING_MODE_STOP;
req->burst_size = ctx->stm.burst_size;
return rc;
}
static int wigig_sensing_ioc_clear_data(struct wigig_sensing_ctx *ctx)
{
struct cir_data *d = &ctx->cir_data;
if (mutex_lock_interruptible(&d->lock))
return -ERESTARTSYS;
d->b.tail = d->b.head = 0;
/* Make sure that the write above is visible to other threads */
wmb();
mutex_unlock(&d->lock);
return 0;
}
static int wigig_sensing_ioc_get_num_dropped_bursts(
struct wigig_sensing_ctx *ctx)
{
return ctx->dropped_bursts;
}
static int wigig_sensing_ioc_get_num_avail_bursts(
struct wigig_sensing_ctx *ctx)
{
if (ctx->stm.burst_size)
return circ_cnt(&ctx->cir_data.b, ctx->cir_data.size_bytes) /
ctx->stm.burst_size;
else
return 0;
}
static int wigig_sensing_ioc_get_event(struct wigig_sensing_ctx *ctx,
enum wigig_sensing_event *event)
{
u32 copied;
if (!ctx->event_pending)
return -EINVAL;
if (kfifo_len(&ctx->events_fifo) == 1)
ctx->event_pending = false;
return kfifo_to_user(&ctx->events_fifo, event,
sizeof(enum wigig_sensing_event), &copied);
}
static int wigig_sensing_open(struct inode *inode, struct file *filp)
{
/* forbid opening more then one instance at a time */
if (mutex_lock_interruptible(&ctx->file_lock))
return -ERESTARTSYS;
if (ctx->opened) {
mutex_unlock(&ctx->file_lock);
return -EBUSY;
}
filp->private_data = ctx;
ctx->opened = true;
mutex_unlock(&ctx->file_lock);
return 0;
}
static unsigned int wigig_sensing_poll(struct file *filp, poll_table *wait)
{
struct wigig_sensing_ctx *ctx = filp->private_data;
unsigned int mask = 0;
if (!ctx->opened)
return -ENODEV;
poll_wait(filp, &ctx->data_wait_q, wait);
if (!ctx->stm.change_mode_in_progress &&
circ_cnt(&ctx->cir_data.b, ctx->cir_data.size_bytes))
mask |= (POLLIN | POLLRDNORM);
if (ctx->event_pending)
mask |= (POLLPRI);
return mask;
}
static ssize_t wigig_sensing_read(struct file *filp, char __user *buf,
size_t count, loff_t *f_pos)
{
int rc = 0;
struct wigig_sensing_ctx *ctx = filp->private_data;
u32 copy_size, size_to_end;
int tail;
struct cir_data *d = &ctx->cir_data;
/* Driver not ready to send data */
if (!ctx || !ctx->spi_dev || !d->b.buf ||
ctx->stm.state == WIGIG_SENSING_STATE_SYS_ASSERT)
return -ENODEV;
if (ctx->stm.change_mode_in_progress)
return -EINVAL;
/* Read buffer too small */
if (count < ctx->stm.burst_size) {
pr_err("Read buffer must be larger than burst size\n");
return -EINVAL;
}
/* No data in the buffer */
while (circ_cnt(&d->b, d->size_bytes) == 0) {
if (filp->f_flags & O_NONBLOCK)
return -EAGAIN;
if (wait_event_interruptible(ctx->data_wait_q,
circ_cnt(&d->b, d->size_bytes) != 0))
return -ERESTARTSYS;
}
if (mutex_lock_interruptible(&d->lock))
return -ERESTARTSYS;
copy_size = min_t(u32, circ_cnt(&d->b, d->size_bytes), count);
copy_size -= copy_size % ctx->stm.burst_size;
size_to_end = circ_cnt_to_end(&d->b, d->size_bytes);
tail = d->b.tail;
pr_debug("copy_size=%u, size_to_end=%u, head=%u, tail=%u\n",
copy_size, size_to_end, d->b.head, tail);
if (copy_size <= size_to_end) {
rc = copy_to_user(buf, &d->b.buf[tail], copy_size);
if (rc) {
pr_err("copy_to_user() failed.\n");
rc = -EFAULT;
goto bail_out;
}
} else {
rc = copy_to_user(buf, &d->b.buf[tail], size_to_end);
if (rc) {
pr_err("copy_to_user() failed.\n");
rc = -EFAULT;
goto bail_out;
}
rc = copy_to_user(buf + size_to_end, &d->b.buf[0],
copy_size - size_to_end);
if (rc) {
pr_err("copy_to_user() failed.\n");
rc = -EFAULT;
goto bail_out;
}
}
/* Increment tail pointer */
d->b.tail = (d->b.tail + copy_size) & (d->size_bytes - 1);
bail_out:
mutex_unlock(&d->lock);
return (rc == 0) ? copy_size : rc;
}
static int wigig_sensing_release(struct inode *inode, struct file *filp)
{
struct wigig_sensing_ctx *ctx = filp->private_data;
if (!ctx || !ctx->spi_dev)
return -ENODEV;
mutex_lock(&ctx->file_lock);
if (!ctx->opened) {
mutex_unlock(&ctx->file_lock);
return -ENODEV;
}
filp->private_data = NULL;
ctx->opened = false;
mutex_unlock(&ctx->file_lock);
return 0;
}
static long wigig_sensing_ioctl(struct file *file, unsigned int cmd,
__user unsigned long arg)
{
int rc;
struct wigig_sensing_ctx *ctx = file->private_data;
if (!ctx || !ctx->spi_dev)
return -ENODEV;
if (mutex_lock_interruptible(&ctx->ioctl_lock))
return -ERESTARTSYS;
if (!ctx->opened) {
mutex_unlock(&ctx->ioctl_lock);
return -ENODEV;
}
/* Check type and command number */
if (_IOC_TYPE(cmd) != WIGIG_SENSING_IOC_MAGIC) {
mutex_unlock(&ctx->ioctl_lock);
return -ENOTTY;
}
switch (_IOC_NR(cmd)) {
case WIGIG_SENSING_IOCTL_SET_AUTO_RECOVERY:
pr_info("Received WIGIG_SENSING_IOCTL_SET_AUTO_RECOVERY command\n");
rc = wigig_sensing_ioc_set_auto_recovery(ctx);
break;
case WIGIG_SENSING_IOCTL_GET_MODE:
pr_info("Received WIGIG_SENSING_IOCTL_GET_MODE command\n");
rc = wigig_sensing_ioc_get_mode(ctx);
break;
case WIGIG_SENSING_IOCTL_CHANGE_MODE:
{
struct wigig_sensing_change_mode req;
pr_info("Received WIGIG_SENSING_IOCTL_CHANGE_MODE command\n");
if (copy_from_user(&req, (void *)arg, sizeof(req)))
return -EFAULT;
SPI_STATS_MEAS_START(ctx, SPI_STATS_MEAS_CHANGE_MODE);
rc = wigig_sensing_ioc_change_mode(ctx, &req);
if (copy_to_user((void *)arg, &req, sizeof(req)))
return -EFAULT;
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_CHANGE_MODE);
break;
}
case WIGIG_SENSING_IOCTL_CLEAR_DATA:
pr_info("Received WIGIG_SENSING_IOCTL_CLEAR_DATA command\n");
rc = wigig_sensing_ioc_clear_data(ctx);
break;
case WIGIG_SENSING_IOCTL_GET_NUM_DROPPED_BURSTS:
pr_info("Received WIGIG_SENSING_IOCTL_GET_NUM_DROPPED_BURSTS command\n");
rc = wigig_sensing_ioc_get_num_dropped_bursts(ctx);
break;
case WIGIG_SENSING_IOCTL_GET_EVENT:
pr_info("Received WIGIG_SENSING_IOCTL_GET_EVENT command\n");
rc = wigig_sensing_ioc_get_event(ctx,
(enum wigig_sensing_event *)arg);
break;
case WIGIG_SENSING_IOCTL_GET_NUM_AVAIL_BURSTS:
pr_info("Received WIGIG_SENSING_IOCTL_GET_NUM_AVAIL_BURSTS command\n");
rc = wigig_sensing_ioc_get_num_avail_bursts(ctx);
break;
default:
rc = -EINVAL;
break;
}
mutex_unlock(&ctx->ioctl_lock);
return rc;
}
static const struct file_operations wigig_sensing_fops = {
.owner = THIS_MODULE,
.llseek = no_llseek,
.open = wigig_sensing_open,
.poll = wigig_sensing_poll,
.read = wigig_sensing_read,
.release = wigig_sensing_release,
.unlocked_ioctl = wigig_sensing_ioctl,
};
static int wigig_sensing_alloc_buffer(struct wigig_sensing_ctx *ctx,
u32 buffer_size_bytes)
{
/* Allocate CIR data_buffer */
ctx->cir_data.b.buf = vmalloc(buffer_size_bytes);
if (!ctx->cir_data.b.buf)
return -ENOMEM;
ctx->cir_data.size_bytes = buffer_size_bytes;
mutex_init(&ctx->cir_data.lock);
return 0;
}
static int wigig_sensing_deassert_dri(
struct wigig_sensing_ctx *ctx,
union user_rgf_spi_mbox_inb additional_cmd)
{
union user_rgf_spi_mbox_inb inb_reg;
int rc;
inb_reg = additional_cmd;
inb_reg.b.deassert_dri = 1;
mutex_lock(&ctx->spi_lock);
rc = spis_write_reg(ctx->spi_dev, RGF_USER_SPI_SPI_MBOX_INB, inb_reg.v);
mutex_unlock(&ctx->spi_lock);
if (rc)
pr_err("Fail to write RGF_USER_SPI_SPI_MBOX_INB, err %d\n",
rc);
return rc;
}
/*
* Calculate SPI transaction size so that the size is between the minimum size
* and two times the minimum size. The size must also divide the burst size.
* The motivaion for using equal sized transactions is to prevent reprogramming
* of the length register for every transaction.
*/
static u32 calc_spi_transaction_size(u32 burst_size,
u32 min_spi_transaction_size)
{
u32 i, res;
if (burst_size <= min_spi_transaction_size)
return burst_size;
for (i = 2; i < MAX_SPI_READ_CHUNKS; i++) {
res = burst_size / i;
if (burst_size % res == 0 &&
res >= min_spi_transaction_size &&
res < 2 * min_spi_transaction_size)
return res;
}
return min_spi_transaction_size;
}
static int wigig_sensing_handle_fifo_ready_dri(struct wigig_sensing_ctx *ctx)
{
int rc = 0;
u32 burst_size = 0;
mutex_lock(&ctx->spi_lock);
/* Read burst size over SPI */
rc = spis_read_reg(ctx->spi_dev, RGF_USER_SPI_SPI_EXT_MBOX_OUTB,
&burst_size);
if (rc) {
pr_err("Fail to read RGF_USER_SPI_SPI_EXT_MBOX_OUTB, err %d\n",
rc);
goto End;
}
ctx->stm.burst_size = burst_size;
pr_info_ratelimited("burst_size = %u\n", burst_size);
if (ctx->stm.state >= WIGIG_SENSING_STATE_SYS_ASSERT ||
ctx->stm.state < WIGIG_SENSING_STATE_READY_STOPPED) {
pr_err("Received burst_size in an unexpected state (%d)\n",
ctx->stm.state);
rc = -EFAULT;
goto End;
}
/* Program burst size into the transfer length register */
ctx->spi_transaction_size =
calc_spi_transaction_size(burst_size, SPI_MIN_TRANSACTION_SIZE);
pr_info_ratelimited("spi_transaction_size = %u\n",
ctx->spi_transaction_size);
rc = spis_write_reg(ctx->spi_dev, SPIS_TRNS_LEN_REG_ADDR,
ctx->spi_transaction_size << 16);
if (rc) {
pr_err("Failed setting SPIS_TRNS_LEN_REG_ADDR\n");
goto End;
}
ctx->last_read_length = ctx->spi_transaction_size;
ctx->stm.burst_size_ready = true;
/*
* Allocate a temporary buffer to be used in case of cir_data buffer
* wrap around
*/
vfree(ctx->temp_buffer);
ctx->temp_buffer = 0;
if (burst_size != 0) {
ctx->temp_buffer = vmalloc(burst_size);
if (!ctx->temp_buffer) {
rc = -ENOMEM;
goto End;
}
}
/* Change internal state */
rc = wigig_sensing_change_state(ctx, &ctx->stm, ctx->stm.state_request);
if (rc) {
pr_err("wigig_sensing_change_state() failed\n");
goto End;
}
/* Initialize head and tail pointers to 0 */
wigig_sensing_ioc_clear_data(ctx);
ctx->dropped_bursts = 0;
ctx->stm.channel = ctx->stm.channel_request;
ctx->stm.mode = ctx->stm.mode_request;
End:
ctx->stm.change_mode_in_progress = false;
mutex_unlock(&ctx->spi_lock);
wake_up_interruptible(&ctx->cmd_wait_q);
return rc;
}
static int wigig_sensing_chip_data_ready_internal(struct wigig_sensing_ctx *ctx,
u16 fill_level, u32 *offset)
{
int rc = 0;
struct spi_fifo *spi_fifo = &ctx->spi_fifo;
struct cir_data *d = &ctx->cir_data;
struct circ_buf local;
u32 bytes_to_read;
u32 available_space_to_end;
/*
* Make sure that fill_level is in 32 bit units
*/
fill_level = fill_level & ~0x3;
local = d->b;
local.head = (local.head + *offset) & (d->size_bytes - 1);
mutex_lock(&ctx->spi_lock);
while (fill_level > 0) {
if (ctx->stm.change_mode_in_progress) {
rc = -EFAULT;
break;
}
bytes_to_read = (fill_level < SPI_MAX_TRANSACTION_SIZE) ?
fill_level : SPI_MAX_TRANSACTION_SIZE;
available_space_to_end =
circ_space_to_end(&local, d->size_bytes);
pr_debug("fill_level=%u, bytes_to_read=%u, offset=%u, available_space_to_end = %u\n",
fill_level, bytes_to_read, *offset,
available_space_to_end);
/* Determine transaction type */
if (available_space_to_end >= bytes_to_read) {
rc = spis_block_read_mem(ctx->spi_dev,
spi_fifo->base_addr + *offset,
&d->b.buf[local.head],
bytes_to_read,
ctx->last_read_length);
if (rc)
break;
} else {
/*
* There is not enough place in the CIR buffer, copy to
* a temporay buffer and then split
*/
rc = spis_block_read_mem(ctx->spi_dev,
spi_fifo->base_addr + *offset,
ctx->temp_buffer,
bytes_to_read,
ctx->last_read_length);
if (rc)
break;
memcpy(&d->b.buf[local.head], ctx->temp_buffer,
available_space_to_end);
memcpy(&d->b.buf[0],
&ctx->temp_buffer[available_space_to_end],
bytes_to_read - available_space_to_end);
}
fill_level -= bytes_to_read;
*offset += bytes_to_read;
local.head = (local.head + bytes_to_read) & (d->size_bytes - 1);
}
mutex_unlock(&ctx->spi_lock);
return rc;
}
static int wigig_sensing_chip_data_ready(struct wigig_sensing_ctx *ctx,
u16 fill_level,
u32 burst_size)
{
int rc = 0;
u32 read_bytes = 0;
enum wigig_sensing_stm_e stm_state = ctx->stm.state;
struct cir_data *d = &ctx->cir_data;
union user_rgf_spi_status spi_status;
if (stm_state == WIGIG_SENSING_STATE_INITIALIZED ||
stm_state == WIGIG_SENSING_STATE_READY_STOPPED ||
stm_state == WIGIG_SENSING_STATE_SYS_ASSERT) {
pr_err("Received data ready interrupt in an unexpected state, disregarding\n");
return 0;
}
if (!ctx->cir_data.b.buf)
return -EFAULT;
/*
* In case there is not enough space in the buffer, discard an old
* burst
*/
if (circ_space(&d->b, d->size_bytes) < burst_size) {
mutex_lock(&d->lock);
if (circ_space(&d->b, d->size_bytes) < burst_size) {
pr_debug("Buffer full, dropping burst\n");
d->b.tail = (d->b.tail + burst_size) &
(d->size_bytes - 1);
ctx->dropped_bursts++;
}
mutex_unlock(&d->lock);
}
while (read_bytes < burst_size) {
if (fill_level >= ctx->spi_transaction_size) {
u32 txn_size = (fill_level >= burst_size) ? burst_size :
ctx->spi_transaction_size;
rc = wigig_sensing_chip_data_ready_internal(
ctx, txn_size, &read_bytes);
if (rc) {
if (ctx->stm.change_mode_in_progress)
pr_err("change_mode_in_progress, aborting SPI transactions\n");
else
pr_err("wigig_sensing_chip_data_ready_internal failed, err %d\n",
rc);
return rc;
}
}
if (ctx->stm.change_mode_in_progress) {
read_bytes = 0;
break;
}
if (read_bytes == burst_size)
break;
/* Read fill_level again */
pr_debug("Reading RGF_USER_SPI_SPI_MBOX_FILL_STATUS register\n");
SPI_STATS_MEAS_START(ctx, SPI_STATS_MEAS_MBOX_FILL_STATUS);
mutex_lock(&ctx->spi_lock);
rc = spis_read_reg(ctx->spi_dev,
RGF_USER_SPI_SPI_MBOX_FILL_STATUS,
&spi_status.v);
mutex_unlock(&ctx->spi_lock);
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_MBOX_FILL_STATUS);
if (rc) {
pr_err("Fail to read RGF_USER_SPI_SPI_MBOX_FILL_STATUS, err %d\n",
rc);
return rc;
}
fill_level = spi_status.b.fill_level;
}
/* Increment destination rd_ptr */
mutex_lock(&d->lock);
d->b.head = (d->b.head + read_bytes) & (d->size_bytes - 1);
pr_debug("head=%u, tail=%u\n", d->b.head, d->b.tail);
mutex_unlock(&d->lock);
wake_up_interruptible(&ctx->data_wait_q);
return 0;
}
static int wigig_sensing_spi_init(struct wigig_sensing_ctx *ctx)
{
int rc;
/* Allocate buffers for SPI transactions */
ctx->cmd_buf = devm_kzalloc(ctx->dev, SPI_CMD_BUFFER_SIZE, GFP_KERNEL);
if (!ctx->cmd_buf)
return -ENOMEM;
ctx->cmd_reply_buf = devm_kzalloc(ctx->dev, SPI_CMD_BUFFER_SIZE,
GFP_KERNEL);
if (!ctx->cmd_reply_buf) {
rc = -ENOMEM;
goto cmd_reply_buf_alloc_failed;
}
ctx->rx_buf = devm_kzalloc(ctx->dev, SPI_BUFFER_SIZE, GFP_KERNEL);
if (!ctx->rx_buf) {
rc = -ENOMEM;
goto rx_buf_alloc_failed;
}
ctx->tx_buf = devm_kzalloc(ctx->dev, SPI_BUFFER_SIZE, GFP_KERNEL);
if (!ctx->tx_buf) {
rc = -ENOMEM;
goto tx_buf_alloc_failed;
}
/* Initialize SPI slave device */
mutex_lock(&ctx->spi_lock);
rc = spis_init(ctx);
mutex_unlock(&ctx->spi_lock);
if (rc) {
rc = -EFAULT;
goto spis_init_failed;
}
return 0;
spis_init_failed:
devm_kfree(ctx->dev, ctx->tx_buf);
ctx->tx_buf = NULL;
tx_buf_alloc_failed:
devm_kfree(ctx->dev, ctx->rx_buf);
ctx->rx_buf = NULL;
rx_buf_alloc_failed:
devm_kfree(ctx->dev, ctx->cmd_reply_buf);
ctx->cmd_reply_buf = NULL;
cmd_reply_buf_alloc_failed:
devm_kfree(ctx->dev, ctx->cmd_buf);
ctx->cmd_buf = NULL;
return rc;
}
static int wigig_sensing_send_event(struct wigig_sensing_ctx *ctx,
enum wigig_sensing_event event)
{
if (kfifo_is_full(&ctx->events_fifo)) {
pr_err("events fifo is full, unable to send event\n");
return -EFAULT;
}
kfifo_in(&ctx->events_fifo, &event, 1);
ctx->event_pending = true;
wake_up_interruptible(&ctx->cmd_wait_q);
return 0;
}
static irqreturn_t wigig_sensing_dri_isr_thread(int irq, void *cookie)
{
struct wigig_sensing_ctx *ctx = cookie;
union user_rgf_spi_status spi_status;
int rc;
u32 sanity_reg = 0;
union user_rgf_spi_mbox_inb additional_inb_command;
u8 num_retries = 0;
bool dont_deassert = false;
mutex_lock(&ctx->dri_lock);
pr_debug("Process DRI signal, ctx->stm.state == %d\n", ctx->stm.state);
memset(&additional_inb_command, 0, sizeof(additional_inb_command));
if (ctx->stm.waiting_for_deep_sleep_exit) {
pr_debug("waiting_for_deep_sleep_exit is set, sending NOP\n");
ctx->stm.waiting_for_deep_sleep_exit_first_pass = true;
mutex_lock(&ctx->spi_lock);
rc = spis_nop(ctx->spi_dev);
/* use 4 dummy bytes to make block read/writes 32-bit aligned */
rc |= spis_write_reg(ctx->spi_dev, SPIS_CFG_REG_ADDR,
SPIS_CONFIG_REG_OPT_VAL);
mutex_unlock(&ctx->spi_lock);
}
while (num_retries < NUM_RETRIES) {
if (ctx->stm.state == WIGIG_SENSING_STATE_INITIALIZED ||
ctx->stm.spi_malfunction) {
pr_debug("Initializing SPI for the first time, or SPI malfunction\n");
rc = wigig_sensing_spi_init(ctx);
if (rc) {
pr_err("wigig_sensing_spi_init() failed\n");
goto bail_out;
}
ctx->stm.spi_malfunction = false;
if (ctx->stm.state == WIGIG_SENSING_STATE_INITIALIZED) {
wigig_sensing_change_state(ctx, &ctx->stm,
WIGIG_SENSING_STATE_READY_STOPPED);
ctx->stm.spi_ready = true;
}
}
pr_debug("Reading SANITY register\n");
SPI_STATS_MEAS_START(ctx, SPI_STATS_MEAS_SANITY);
mutex_lock(&ctx->spi_lock);
rc = spis_read_reg(ctx->spi_dev, SPIS_SANITY_REG_ADDR,
&sanity_reg);
mutex_unlock(&ctx->spi_lock);
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_SANITY);
if (rc || sanity_reg != SPIS_SANITY_REG_VAL) {
pr_err("Fail to read SANITY, expected 0x%X found 0x%X err %d\n",
SPIS_SANITY_REG_VAL, sanity_reg, (int)rc);
ctx->stm.spi_malfunction = true;
} else {
pr_debug("SANITY OK\n");
ctx->stm.spi_malfunction = false;
break;
}
num_retries++;
if (num_retries == NUM_RETRIES) {
pr_err("SPI bus malfunction, bailing out\n");
goto bail_out;
}
}
pr_debug("Reading RGF_USER_SPI_SPI_MBOX_FILL_STATUS register\n");
SPI_STATS_MEAS_START(ctx, SPI_STATS_MEAS_MBOX_FILL_STATUS);
mutex_lock(&ctx->spi_lock);
rc = spis_read_reg(ctx->spi_dev, RGF_USER_SPI_SPI_MBOX_FILL_STATUS,
&spi_status.v);
mutex_unlock(&ctx->spi_lock);
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_MBOX_FILL_STATUS);
if (rc) {
pr_err("Fail to read RGF_USER_SPI_SPI_MBOX_FILL_STATUS, err %d\n",
rc);
goto bail_out;
}
if (spi_status.b.int_dont_deassert) {
dont_deassert = true;
spi_status.v &= ~INT_DONT_DEASSERT;
}
if (spi_status.b.int_sysassert) {
enum wigig_sensing_stm_e old_state = ctx->stm.state;
pr_info_ratelimited("SYSASSERT INTERRUPT\n");
ctx->stm.fw_is_ready = false;
wigig_sensing_change_state(ctx, &ctx->stm,
WIGIG_SENSING_STATE_SYS_ASSERT);
/* Send asynchronous RESET event to application */
if (old_state != WIGIG_SENSING_STATE_READY_STOPPED)
wigig_sensing_send_event(ctx,
WIGIG_SENSING_EVENT_RESET);
ctx->stm.spi_malfunction = true;
ctx->stm.spi_ready = false;
memset(&ctx->inb_cmd, 0, sizeof(ctx->inb_cmd));
spi_status.v &= ~INT_SYSASSERT;
goto deassert_and_bail_out;
}
if (spi_status.b.int_fw_ready) {
pr_info_ratelimited("FW READY INTERRUPT\n");
ctx->stm.fw_is_ready = true;
ctx->stm.channel_request = 0;
ctx->stm.burst_size = 0;
ctx->stm.mode = WIGIG_SENSING_MODE_STOP;
wigig_sensing_change_state(ctx, &ctx->stm,
WIGIG_SENSING_STATE_READY_STOPPED);
/* Send asynchronous FW_READY event to application */
wigig_sensing_send_event(ctx, WIGIG_SENSING_EVENT_FW_READY);
spi_status.v &= ~INT_FW_READY;
}
if (spi_status.b.int_data_ready) {
SPI_STATS_MEAS_START(ctx, SPI_STATS_MEAS_DATA_READY);
pr_debug("DATA READY INTERRUPT\n");
/*
* Data ready interrupt is received when waiting for deep sleep
* exit, break and exit deep sleep
*/
if (ctx->stm.waiting_for_deep_sleep_exit) {
additional_inb_command = ctx->inb_cmd;
memset(&ctx->inb_cmd, 0, sizeof(ctx->inb_cmd));
pr_err("Received data ready interrupt when waiting for deep sleep exit, treating it so\n");
ctx->stm.waiting_for_deep_sleep_exit = false;
ctx->stm.waiting_for_deep_sleep_exit_first_pass = false;
spi_status.v &= ~INT_DEEP_SLEEP_EXIT;
goto finish_data_ready;
}
if (!ctx->stm.change_mode_in_progress)
wigig_sensing_chip_data_ready(
ctx, spi_status.b.fill_level, ctx->stm.burst_size);
else
pr_debug("Change mode in progress, aborting data processing\n");
finish_data_ready:
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_DATA_READY);
spi_status.v &= ~INT_DATA_READY;
}
if (spi_status.b.int_deep_sleep_exit ||
(ctx->stm.waiting_for_deep_sleep_exit &&
ctx->stm.waiting_for_deep_sleep_exit_first_pass)) {
if (spi_status.b.int_deep_sleep_exit)
pr_info_ratelimited("DEEP SLEEP EXIT INTERRUPT\n");
if (ctx->stm.waiting_for_deep_sleep_exit) {
additional_inb_command = ctx->inb_cmd;
memset(&ctx->inb_cmd, 0, sizeof(ctx->inb_cmd));
} else {
pr_err("Got deep sleep exit DRI when ctx->stm.waiting_for_deep_sleep_exit is false\n");
}
ctx->stm.waiting_for_deep_sleep_exit = false;
ctx->stm.waiting_for_deep_sleep_exit_first_pass = false;
spi_status.v &= ~INT_DEEP_SLEEP_EXIT;
}
if (spi_status.b.int_fifo_ready) {
pr_info_ratelimited("FIFO READY INTERRUPT\n");
wigig_sensing_handle_fifo_ready_dri(ctx);
spi_status.v &= ~INT_FIFO_READY;
}
/*
* Check if there are unexpected interrupts, lowest 23 bits needs to be
* cleared
*/
if ((spi_status.v & CLEAR_LOW_23_BITS) != 0)
pr_err("Unexpected interrupt received, spi_status=0x%X\n",
spi_status.v & CLEAR_LOW_23_BITS);
deassert_and_bail_out:
/* Notify FW we are done with interrupt handling */
if (!dont_deassert || additional_inb_command.b.mode != 0) {
SPI_STATS_MEAS_START(ctx, SPI_STATS_MEAS_DEASSERT);
rc = wigig_sensing_deassert_dri(ctx, additional_inb_command);
if (rc)
pr_err("wigig_sensing_deassert_dri() failed, rc=%d\n",
rc);
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_DEASSERT);
}
bail_out:
mutex_unlock(&ctx->dri_lock);
SPI_STATS_MEAS_STOP(ctx, SPI_STATS_MEAS_DRI_PROC);
return IRQ_HANDLED;
}
static int wigig_sensing_debugfs_spi_stats_show(struct seq_file *s, void *data)
{
struct wigig_sensing_ctx *ctx = s->private;
int i;
if (ctx == NULL)
return -ENODEV;
seq_printf(s, "|%18s|%10s|%10s|%15s|%10s|%10s|\n",
"Name", "Min [uS]", "Max [uS]", "Acc [uS]", "#", "Avg [uS]");
for (i = 0; i < SPI_STATS_MEAS_MAX; i++) {
struct spi_stats *ss = &ctx->spi_stats[i];
u64 min = atomic64_read(&ss->min);
u64 max = atomic64_read(&ss->max);
u64 acc = atomic64_read(&ss->acc);
u64 num_meas = atomic_read(&ss->num_meas);
seq_printf(s, "|%18s|%10lu|%10lu|%15llu|%10lu|%10lu|\n",
ss->name,
ktime_to_us(min),
ktime_to_us(max),
ktime_to_us(acc),
num_meas,
ktime_to_us((num_meas != 0) ? acc / num_meas : 0));
}
return 0;
}
static int wigig_sensing_seq_spi_stats_open(struct inode *inode,
struct file *file)
{
return single_open(file, wigig_sensing_debugfs_spi_stats_show,
inode->i_private);
}
static const struct file_operations debugfs_spi_stats_fops = {
.open = wigig_sensing_seq_spi_stats_open,
.release = single_release,
.read = seq_read,
.llseek = seq_lseek,
};
static int wigig_sensing_debugfs_spi_stats_init_open(struct seq_file *s,
void *data)
{
struct wigig_sensing_ctx *ctx = s->private;
int i;
if (ctx == NULL)
return -ENODEV;
for (i = 0; i < SPI_STATS_MEAS_MAX; i++) {
struct spi_stats *ss = &ctx->spi_stats[i];
atomic64_set(&ss->min, U64_MAX);
atomic64_set(&ss->max, 0);
atomic64_set(&ss->acc, 0);
atomic_set(&ss->num_meas, 0);
ss->start = 0;
ss->delta = 0;
}
return 0;
}
static int wigig_sensing_seq_spi_stats_init_open(struct inode *inode,
struct file *file)
{
return single_open(file, wigig_sensing_debugfs_spi_stats_init_open,
inode->i_private);
}
static const struct file_operations debugfs_spi_stats_init_fops = {
.open = wigig_sensing_seq_spi_stats_init_open,
.release = single_release,
.read = seq_read,
.llseek = seq_lseek,
};
static int wigig_sensing_debugfs_init(struct wigig_sensing_ctx *ctx)
{
SPI_STATS_MEAS_INIT(ctx, SPI_STATS_MEAS_SANITY, "Sanity");
SPI_STATS_MEAS_INIT(ctx, SPI_STATS_MEAS_DEASSERT, "Deassert DRI");
SPI_STATS_MEAS_INIT(ctx, SPI_STATS_MEAS_DRI_PROC, "DRI proc");
SPI_STATS_MEAS_INIT(ctx, SPI_STATS_MEAS_MBOX_FILL_STATUS,
"MBOX FILL STATUS");
SPI_STATS_MEAS_INIT(ctx, SPI_STATS_MEAS_CHANGE_MODE, "CHANGE MODE");
SPI_STATS_MEAS_INIT(ctx, SPI_STATS_MEAS_DATA_READY, "Data Ready");
ctx->debugfs_dent = debugfs_create_dir("wigig_sensing", NULL);
debugfs_create_file("spi_stats", 0644, ctx->debugfs_dent, ctx,
&debugfs_spi_stats_fops);
debugfs_create_file("spi_stats_init", 0644, ctx->debugfs_dent, ctx,
&debugfs_spi_stats_init_fops);
return 0;
}
static irqreturn_t wigig_sensing_dri_isr_hard(int irq, void *cookie)
{
SPI_STATS_MEAS_START((struct wigig_sensing_ctx *)cookie,
SPI_STATS_MEAS_DRI_PROC);
return IRQ_WAKE_THREAD;
}
static int wigig_sensing_probe(struct spi_device *spi)
{
int rc = 0;
struct wigig_sensing_platform_data pdata;
/* Allocate driver context */
ctx = devm_kzalloc(&spi->dev, sizeof(struct wigig_sensing_ctx),
GFP_KERNEL);
if (!ctx)
return -ENOMEM;
/* Initialize driver context */
spi_set_drvdata(spi, ctx);
ctx->spi_dev = spi;
ctx->opened = false;
mutex_init(&ctx->ioctl_lock);
mutex_init(&ctx->file_lock);
mutex_init(&ctx->spi_lock);
mutex_init(&ctx->dri_lock);
init_waitqueue_head(&ctx->cmd_wait_q);
init_waitqueue_head(&ctx->data_wait_q);
ctx->stm.state = WIGIG_SENSING_STATE_INITIALIZED;
INIT_KFIFO(ctx->events_fifo);
/* Allocate memory for the CIRs */
/* Allocate a 2MB == 2^21 buffer for CIR data */
rc = wigig_sensing_alloc_buffer(ctx, 1 << 21);
if (rc) {
pr_err("wigig_sensing_alloc_buffer() failed (%d)\n", rc);
return rc;
}
/* Create device node */
rc = alloc_chrdev_region(&ctx->wigig_sensing_dev, 0, 1, DRV_NAME);
if (rc < 0) {
pr_err("alloc_chrdev_region() failed\n");
return rc;
}
ctx->class = class_create(THIS_MODULE, DRV_NAME);
if (IS_ERR(ctx->class)) {
rc = PTR_ERR(ctx->class);
pr_err("Error creating ctx->class: %d\n", rc);
goto fail_class_create;
}
ctx->dev = device_create(ctx->class, NULL, ctx->wigig_sensing_dev, ctx,
DRV_NAME);
if (IS_ERR(ctx->dev)) {
rc = PTR_ERR(ctx->dev);
pr_err("device_create failed: %d\n", rc);
goto fail_device_create;
}
cdev_init(&ctx->cdev, &wigig_sensing_fops);
ctx->cdev.owner = THIS_MODULE;
ctx->cdev.ops = &wigig_sensing_fops;
rc = cdev_add(&ctx->cdev, ctx->wigig_sensing_dev, 1);
if (rc) {
pr_err("Error calling cdev_add: %d\n", rc);
goto fail_cdev_add;
}
/* Setup DRI - interrupt */
pdata.dri_gpio = devm_gpiod_get(&spi->dev, "dri", GPIOD_IN);
if (IS_ERR(pdata.dri_gpio)) {
pr_err("Could not find dri gpio\n");
rc = -EFAULT;
goto fail_gpiod_get;
}
ctx->dri_gpio = pdata.dri_gpio;
ctx->dri_irq = gpiod_to_irq(ctx->dri_gpio);
pr_debug("dri_irq = %d\n", ctx->dri_irq);
rc = devm_request_threaded_irq(&spi->dev, ctx->dri_irq,
wigig_sensing_dri_isr_hard, wigig_sensing_dri_isr_thread,
IRQF_TRIGGER_RISING, "wigig_sensing_dri", ctx);
if (rc) {
pr_err("devm_request_threaded_irq() failed (%d)\n", rc);
goto fail_gpiod_get;
}
/* Initialize debugfs */
wigig_sensing_debugfs_init(ctx);
return 0;
fail_gpiod_get:
cdev_del(&ctx->cdev);
fail_cdev_add:
device_destroy(ctx->class, ctx->wigig_sensing_dev);
fail_device_create:
class_destroy(ctx->class);
fail_class_create:
unregister_chrdev_region(ctx->wigig_sensing_dev, 1);
return rc;
}
static int wigig_sensing_remove(struct spi_device *spi)
{
struct wigig_sensing_ctx *ctx = spi_get_drvdata(spi);
struct wigig_sensing_change_mode req = {
.mode = WIGIG_SENSING_MODE_STOP,
.has_channel = false,
.channel = 0,
.burst_size = 0,
};
/* Make sure that FW is in STOP mode */
wigig_sensing_ioc_change_mode(ctx, &req);
debugfs_remove_recursive(ctx->debugfs_dent);
device_destroy(ctx->class, ctx->wigig_sensing_dev);
unregister_chrdev_region(ctx->wigig_sensing_dev, 1);
class_destroy(ctx->class);
cdev_del(&ctx->cdev);
vfree(ctx->cir_data.b.buf);
spi_set_drvdata(ctx->spi_dev, NULL);
return 0;
}
static const struct of_device_id wigig_sensing_match_table[] = {
{ .compatible = DRV_NAME },
{}
};
static struct spi_driver radar_spi_driver = {
.driver = {
.name = DRV_NAME,
.of_match_table = of_match_ptr(wigig_sensing_match_table),
},
.probe = wigig_sensing_probe,
.remove = wigig_sensing_remove,
};
module_spi_driver(radar_spi_driver);
MODULE_DESCRIPTION("Wigig sensing slave SPI Driver");
MODULE_LICENSE("GPL v2");