Merge remote-tracking branch 'iommu/next'
[deliverable/linux.git] / drivers / input / touchscreen / atmel_mxt_ts.c
index 7376c42d23703f52bd2fca8eefdef99d59d894fd..beede8fe511cefc69d027639bea1984c7b77eb67 100644 (file)
@@ -135,8 +135,13 @@ struct t9_range {
 /* MXT_DEBUG_DIAGNOSTIC_T37 */
 #define MXT_DIAGNOSTIC_PAGEUP  0x01
 #define MXT_DIAGNOSTIC_DELTAS  0x10
+#define MXT_DIAGNOSTIC_REFS    0x11
 #define MXT_DIAGNOSTIC_SIZE    128
 
+#define MXT_FAMILY_1386                        160
+#define MXT1386_COLUMNS                        3
+#define MXT1386_PAGES_PER_COLUMN       8
+
 struct t37_debug {
 #ifdef CONFIG_TOUCHSCREEN_ATMEL_MXT_T37
        u8 mode;
@@ -245,6 +250,12 @@ struct mxt_dbg {
        int input;
 };
 
+enum v4l_dbg_inputs {
+       MXT_V4L_INPUT_DELTAS,
+       MXT_V4L_INPUT_REFS,
+       MXT_V4L_INPUT_MAX,
+};
+
 static const struct v4l2_file_operations mxt_video_fops = {
        .owner = THIS_MODULE,
        .open = v4l2_fh_open,
@@ -2140,13 +2151,27 @@ recheck:
 static u16 mxt_get_debug_value(struct mxt_data *data, unsigned int x,
                               unsigned int y)
 {
+       struct mxt_info *info = &data->info;
        struct mxt_dbg *dbg = &data->dbg;
        unsigned int ofs, page;
+       unsigned int col = 0;
+       unsigned int col_width;
 
-       ofs = (y + (x * data->info.matrix_ysize)) * sizeof(u16);
+       if (info->family_id == MXT_FAMILY_1386) {
+               col_width = info->matrix_ysize / MXT1386_COLUMNS;
+               col = y / col_width;
+               y = y % col_width;
+       } else {
+               col_width = info->matrix_ysize;
+       }
+
+       ofs = (y + (x * col_width)) * sizeof(u16);
        page = ofs / MXT_DIAGNOSTIC_SIZE;
        ofs %= MXT_DIAGNOSTIC_SIZE;
 
+       if (info->family_id == MXT_FAMILY_1386)
+               page += col * MXT1386_PAGES_PER_COLUMN;
+
        return get_unaligned_le16(&dbg->t37_buf[page].data[ofs]);
 }
 
@@ -2255,6 +2280,7 @@ static void mxt_buffer_queue(struct vb2_buffer *vb)
        struct mxt_data *data = vb2_get_drv_priv(vb->vb2_queue);
        u16 *ptr;
        int ret;
+       u8 mode;
 
        ptr = vb2_plane_vaddr(vb, 0);
        if (!ptr) {
@@ -2262,7 +2288,18 @@ static void mxt_buffer_queue(struct vb2_buffer *vb)
                goto fault;
        }
 
-       ret = mxt_read_diagnostic_debug(data, MXT_DIAGNOSTIC_DELTAS, ptr);
+       switch (data->dbg.input) {
+       case MXT_V4L_INPUT_DELTAS:
+       default:
+               mode = MXT_DIAGNOSTIC_DELTAS;
+               break;
+
+       case MXT_V4L_INPUT_REFS:
+               mode = MXT_DIAGNOSTIC_REFS;
+               break;
+       }
+
+       ret = mxt_read_diagnostic_debug(data, mode, ptr);
        if (ret)
                goto fault;
 
@@ -2307,11 +2344,21 @@ static int mxt_vidioc_querycap(struct file *file, void *priv,
 static int mxt_vidioc_enum_input(struct file *file, void *priv,
                                   struct v4l2_input *i)
 {
-       if (i->index > 0)
+       if (i->index >= MXT_V4L_INPUT_MAX)
                return -EINVAL;
 
        i->type = V4L2_INPUT_TYPE_TOUCH;
-       strlcpy(i->name, "Mutual Capacitance Deltas", sizeof(i->name));
+
+       switch (i->index) {
+       case MXT_V4L_INPUT_REFS:
+               strlcpy(i->name, "Mutual Capacitance References",
+                       sizeof(i->name));
+               break;
+       case MXT_V4L_INPUT_DELTAS:
+               strlcpy(i->name, "Mutual Capacitance Deltas", sizeof(i->name));
+               break;
+       }
+
        return 0;
 }
 
@@ -2319,12 +2366,16 @@ static int mxt_set_input(struct mxt_data *data, unsigned int i)
 {
        struct v4l2_pix_format *f = &data->dbg.format;
 
-       if (i > 0)
+       if (i >= MXT_V4L_INPUT_MAX)
                return -EINVAL;
 
+       if (i == MXT_V4L_INPUT_DELTAS)
+               f->pixelformat = V4L2_TCH_FMT_DELTA_TD16;
+       else
+               f->pixelformat = V4L2_TCH_FMT_TU16;
+
        f->width = data->xy_switch ? data->ysize : data->xsize;
        f->height = data->xy_switch ? data->xsize : data->ysize;
-       f->pixelformat = V4L2_TCH_FMT_DELTA_TD16;
        f->field = V4L2_FIELD_NONE;
        f->colorspace = V4L2_COLORSPACE_RAW;
        f->bytesperline = f->width * sizeof(u16);
@@ -2365,7 +2416,19 @@ static int mxt_vidioc_enum_fmt(struct file *file, void *priv,
        if (fmt->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
                return -EINVAL;
 
-       fmt->pixelformat = V4L2_TCH_FMT_DELTA_TD16;
+       switch (fmt->index) {
+       case 0:
+               fmt->pixelformat = V4L2_TCH_FMT_TU16;
+               break;
+
+       case 1:
+               fmt->pixelformat = V4L2_TCH_FMT_DELTA_TD16;
+               break;
+
+       default:
+               return -EINVAL;
+       }
+
        return 0;
 }
 
@@ -2416,6 +2479,7 @@ static const struct video_device mxt_video_device = {
 
 static void mxt_debug_init(struct mxt_data *data)
 {
+       struct mxt_info *info = &data->info;
        struct mxt_dbg *dbg = &data->dbg;
        struct mxt_object *object;
        int error;
@@ -2439,8 +2503,14 @@ static void mxt_debug_init(struct mxt_data *data)
 
        /* Calculate size of data and allocate buffer */
        dbg->t37_nodes = data->xsize * data->ysize;
-       dbg->t37_pages = DIV_ROUND_UP(data->xsize * data->info.matrix_ysize *
-                                     sizeof(u16), sizeof(dbg->t37_buf->data));
+
+       if (info->family_id == MXT_FAMILY_1386)
+               dbg->t37_pages = MXT1386_COLUMNS * MXT1386_PAGES_PER_COLUMN;
+       else
+               dbg->t37_pages = DIV_ROUND_UP(data->xsize *
+                                             data->info.matrix_ysize *
+                                             sizeof(u16),
+                                             sizeof(dbg->t37_buf->data));
 
        dbg->t37_buf = devm_kmalloc_array(&data->client->dev, dbg->t37_pages,
                                          sizeof(struct t37_debug), GFP_KERNEL);
This page took 0.026341 seconds and 5 git commands to generate.