linux/drivers/media/test-drivers/vivid/vivid-touch-cap.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0-only
   2/*
   3 * vivid-touch-cap.c - touch support functions.
   4 */
   5
   6#include "vivid-core.h"
   7#include "vivid-kthread-touch.h"
   8#include "vivid-vid-common.h"
   9#include "vivid-touch-cap.h"
  10
  11static int touch_cap_queue_setup(struct vb2_queue *vq, unsigned int *nbuffers,
  12                                 unsigned int *nplanes, unsigned int sizes[],
  13                                 struct device *alloc_devs[])
  14{
  15        struct vivid_dev *dev = vb2_get_drv_priv(vq);
  16        struct v4l2_pix_format *f = &dev->tch_format;
  17        unsigned int size = f->sizeimage;
  18
  19        if (*nplanes) {
  20                if (sizes[0] < size)
  21                        return -EINVAL;
  22        } else {
  23                sizes[0] = size;
  24        }
  25
  26        if (vq->num_buffers + *nbuffers < 2)
  27                *nbuffers = 2 - vq->num_buffers;
  28
  29        *nplanes = 1;
  30        return 0;
  31}
  32
  33static int touch_cap_buf_prepare(struct vb2_buffer *vb)
  34{
  35        struct vivid_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
  36        struct v4l2_pix_format *f = &dev->tch_format;
  37        unsigned int size = f->sizeimage;
  38
  39        if (dev->buf_prepare_error) {
  40                /*
  41                 * Error injection: test what happens if buf_prepare() returns
  42                 * an error.
  43                 */
  44                dev->buf_prepare_error = false;
  45                return -EINVAL;
  46        }
  47        if (vb2_plane_size(vb, 0) < size) {
  48                dprintk(dev, 1, "%s data will not fit into plane (%lu < %u)\n",
  49                        __func__, vb2_plane_size(vb, 0), size);
  50                return -EINVAL;
  51        }
  52        vb2_set_plane_payload(vb, 0, size);
  53
  54        return 0;
  55}
  56
  57static void touch_cap_buf_queue(struct vb2_buffer *vb)
  58{
  59        struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb);
  60        struct vivid_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
  61        struct vivid_buffer *buf = container_of(vbuf, struct vivid_buffer, vb);
  62
  63        vbuf->field = V4L2_FIELD_NONE;
  64        spin_lock(&dev->slock);
  65        list_add_tail(&buf->list, &dev->touch_cap_active);
  66        spin_unlock(&dev->slock);
  67}
  68
  69static int touch_cap_start_streaming(struct vb2_queue *vq, unsigned int count)
  70{
  71        struct vivid_dev *dev = vb2_get_drv_priv(vq);
  72        int err;
  73
  74        dev->touch_cap_seq_count = 0;
  75        if (dev->start_streaming_error) {
  76                dev->start_streaming_error = false;
  77                err = -EINVAL;
  78        } else {
  79                err = vivid_start_generating_touch_cap(dev);
  80        }
  81        if (err) {
  82                struct vivid_buffer *buf, *tmp;
  83
  84                list_for_each_entry_safe(buf, tmp,
  85                                         &dev->touch_cap_active, list) {
  86                        list_del(&buf->list);
  87                        vb2_buffer_done(&buf->vb.vb2_buf,
  88                                        VB2_BUF_STATE_QUEUED);
  89                }
  90        }
  91        return err;
  92}
  93
  94/* abort streaming and wait for last buffer */
  95static void touch_cap_stop_streaming(struct vb2_queue *vq)
  96{
  97        struct vivid_dev *dev = vb2_get_drv_priv(vq);
  98
  99        vivid_stop_generating_touch_cap(dev);
 100}
 101
 102static void touch_cap_buf_request_complete(struct vb2_buffer *vb)
 103{
 104        struct vivid_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
 105
 106        v4l2_ctrl_request_complete(vb->req_obj.req, &dev->ctrl_hdl_touch_cap);
 107}
 108
 109const struct vb2_ops vivid_touch_cap_qops = {
 110        .queue_setup            = touch_cap_queue_setup,
 111        .buf_prepare            = touch_cap_buf_prepare,
 112        .buf_queue              = touch_cap_buf_queue,
 113        .start_streaming        = touch_cap_start_streaming,
 114        .stop_streaming         = touch_cap_stop_streaming,
 115        .buf_request_complete   = touch_cap_buf_request_complete,
 116        .wait_prepare           = vb2_ops_wait_prepare,
 117        .wait_finish            = vb2_ops_wait_finish,
 118};
 119
 120int vivid_enum_fmt_tch(struct file *file, void  *priv, struct v4l2_fmtdesc *f)
 121{
 122        if (f->index)
 123                return -EINVAL;
 124
 125        f->pixelformat = V4L2_TCH_FMT_DELTA_TD16;
 126        return 0;
 127}
 128
 129int vivid_g_fmt_tch(struct file *file, void *priv, struct v4l2_format *f)
 130{
 131        struct vivid_dev *dev = video_drvdata(file);
 132
 133        if (dev->multiplanar)
 134                return -ENOTTY;
 135        f->fmt.pix = dev->tch_format;
 136        return 0;
 137}
 138
 139int vivid_g_fmt_tch_mplane(struct file *file, void *priv, struct v4l2_format *f)
 140{
 141        struct vivid_dev *dev = video_drvdata(file);
 142        struct v4l2_format sp_fmt;
 143
 144        if (!dev->multiplanar)
 145                return -ENOTTY;
 146        sp_fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
 147        sp_fmt.fmt.pix = dev->tch_format;
 148        fmt_sp2mp(&sp_fmt, f);
 149        return 0;
 150}
 151
 152int vivid_g_parm_tch(struct file *file, void *priv,
 153                     struct v4l2_streamparm *parm)
 154{
 155        struct vivid_dev *dev = video_drvdata(file);
 156
 157        if (parm->type != (dev->multiplanar ?
 158                           V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE :
 159                           V4L2_BUF_TYPE_VIDEO_CAPTURE))
 160                return -EINVAL;
 161
 162        parm->parm.capture.capability   = V4L2_CAP_TIMEPERFRAME;
 163        parm->parm.capture.timeperframe = dev->timeperframe_tch_cap;
 164        parm->parm.capture.readbuffers  = 1;
 165        return 0;
 166}
 167
 168int vivid_enum_input_tch(struct file *file, void *priv, struct v4l2_input *inp)
 169{
 170        if (inp->index)
 171                return -EINVAL;
 172
 173        inp->type = V4L2_INPUT_TYPE_TOUCH;
 174        strscpy(inp->name, "Vivid Touch", sizeof(inp->name));
 175        inp->capabilities = 0;
 176        return 0;
 177}
 178
 179int vivid_g_input_tch(struct file *file, void *priv, unsigned int *i)
 180{
 181        *i = 0;
 182        return 0;
 183}
 184
 185int vivid_set_touch(struct vivid_dev *dev, unsigned int i)
 186{
 187        struct v4l2_pix_format *f = &dev->tch_format;
 188
 189        if (i)
 190                return -EINVAL;
 191
 192        f->pixelformat = V4L2_TCH_FMT_DELTA_TD16;
 193        f->width =  VIVID_TCH_WIDTH;
 194        f->height = VIVID_TCH_HEIGHT;
 195        f->field = V4L2_FIELD_NONE;
 196        f->colorspace = V4L2_COLORSPACE_RAW;
 197        f->bytesperline = f->width * sizeof(s16);
 198        f->sizeimage = f->width * f->height * sizeof(s16);
 199        return 0;
 200}
 201
 202int vivid_s_input_tch(struct file *file, void *priv, unsigned int i)
 203{
 204        return vivid_set_touch(video_drvdata(file), i);
 205}
 206
 207static void vivid_fill_buff_noise(__s16 *tch_buf, int size)
 208{
 209        int i;
 210
 211        /* Fill 10% of the values within range -3 and 3, zero the others */
 212        for (i = 0; i < size; i++) {
 213                unsigned int rand = get_random_int();
 214
 215                if (rand % 10)
 216                        tch_buf[i] = 0;
 217                else
 218                        tch_buf[i] = (rand / 10) % 7 - 3;
 219        }
 220}
 221
 222static inline int get_random_pressure(void)
 223{
 224        return get_random_int() % VIVID_PRESSURE_LIMIT;
 225}
 226
 227static void vivid_tch_buf_set(struct v4l2_pix_format *f,
 228                              __s16 *tch_buf,
 229                              int index)
 230{
 231        unsigned int x = index % f->width;
 232        unsigned int y = index / f->width;
 233        unsigned int offset = VIVID_MIN_PRESSURE;
 234
 235        tch_buf[index] = offset + get_random_pressure();
 236        offset /= 2;
 237        if (x)
 238                tch_buf[index - 1] = offset + get_random_pressure();
 239        if (x < f->width - 1)
 240                tch_buf[index + 1] = offset + get_random_pressure();
 241        if (y)
 242                tch_buf[index - f->width] = offset + get_random_pressure();
 243        if (y < f->height - 1)
 244                tch_buf[index + f->width] = offset + get_random_pressure();
 245        offset /= 2;
 246        if (x && y)
 247                tch_buf[index - 1 - f->width] = offset + get_random_pressure();
 248        if (x < f->width - 1 && y)
 249                tch_buf[index + 1 - f->width] = offset + get_random_pressure();
 250        if (x && y < f->height - 1)
 251                tch_buf[index - 1 + f->width] = offset + get_random_pressure();
 252        if (x < f->width - 1 && y < f->height - 1)
 253                tch_buf[index + 1 + f->width] = offset + get_random_pressure();
 254}
 255
 256void vivid_fillbuff_tch(struct vivid_dev *dev, struct vivid_buffer *buf)
 257{
 258        struct v4l2_pix_format *f = &dev->tch_format;
 259        int size = f->width * f->height;
 260        int x, y, xstart, ystart, offset_x, offset_y;
 261        unsigned int test_pattern, test_pat_idx, rand;
 262
 263        __s16 *tch_buf = vb2_plane_vaddr(&buf->vb.vb2_buf, 0);
 264
 265        buf->vb.sequence = dev->touch_cap_seq_count;
 266        test_pattern = (buf->vb.sequence / TCH_SEQ_COUNT) % TEST_CASE_MAX;
 267        test_pat_idx = buf->vb.sequence % TCH_SEQ_COUNT;
 268
 269        vivid_fill_buff_noise(tch_buf, size);
 270
 271        if (test_pat_idx >= TCH_PATTERN_COUNT)
 272                return;
 273
 274        if (test_pat_idx == 0)
 275                dev->tch_pat_random = get_random_int();
 276        rand = dev->tch_pat_random;
 277
 278        switch (test_pattern) {
 279        case SINGLE_TAP:
 280                if (test_pat_idx == 2)
 281                        vivid_tch_buf_set(f, tch_buf, rand % size);
 282                break;
 283        case DOUBLE_TAP:
 284                if (test_pat_idx == 2 || test_pat_idx == 4)
 285                        vivid_tch_buf_set(f, tch_buf, rand % size);
 286                break;
 287        case TRIPLE_TAP:
 288                if (test_pat_idx == 2 || test_pat_idx == 4 || test_pat_idx == 6)
 289                        vivid_tch_buf_set(f, tch_buf, rand % size);
 290                break;
 291        case MOVE_LEFT_TO_RIGHT:
 292                vivid_tch_buf_set(f, tch_buf,
 293                                  (rand % f->height) * f->width +
 294                                  test_pat_idx *
 295                                  (f->width / TCH_PATTERN_COUNT));
 296                break;
 297        case ZOOM_IN:
 298                x = f->width / 2;
 299                y = f->height / 2;
 300                offset_x = ((TCH_PATTERN_COUNT - 1 - test_pat_idx) * x) /
 301                                TCH_PATTERN_COUNT;
 302                offset_y = ((TCH_PATTERN_COUNT - 1 - test_pat_idx) * y) /
 303                                TCH_PATTERN_COUNT;
 304                vivid_tch_buf_set(f, tch_buf,
 305                                  (x - offset_x) + f->width * (y - offset_y));
 306                vivid_tch_buf_set(f, tch_buf,
 307                                  (x + offset_x) + f->width * (y + offset_y));
 308                break;
 309        case ZOOM_OUT:
 310                x = f->width / 2;
 311                y = f->height / 2;
 312                offset_x = (test_pat_idx * x) / TCH_PATTERN_COUNT;
 313                offset_y = (test_pat_idx * y) / TCH_PATTERN_COUNT;
 314                vivid_tch_buf_set(f, tch_buf,
 315                                  (x - offset_x) + f->width * (y - offset_y));
 316                vivid_tch_buf_set(f, tch_buf,
 317                                  (x + offset_x) + f->width * (y + offset_y));
 318                break;
 319        case PALM_PRESS:
 320                for (x = 0; x < f->width; x++)
 321                        for (y = f->height / 2; y < f->height; y++)
 322                                tch_buf[x + f->width * y] = VIVID_MIN_PRESSURE +
 323                                                        get_random_pressure();
 324                break;
 325        case MULTIPLE_PRESS:
 326                /* 16 pressure points */
 327                for (y = 0; y < 4; y++) {
 328                        for (x = 0; x < 4; x++) {
 329                                ystart = (y * f->height) / 4 + f->height / 8;
 330                                xstart = (x * f->width) / 4 + f->width / 8;
 331                                vivid_tch_buf_set(f, tch_buf,
 332                                                  ystart * f->width + xstart);
 333                        }
 334                }
 335                break;
 336        }
 337#ifdef __BIG_ENDIAN__
 338        for (x = 0; x < size; x++)
 339                tch_buf[x] = (__force s16)__cpu_to_le16((u16)tch_buf[x]);
 340#endif
 341}
 342