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