diff options
author | root <root@no.no.james.local> | 2015-12-28 11:06:03 +0000 |
---|---|---|
committer | root <root@no.no.james.local> | 2015-12-28 11:06:03 +0000 |
commit | 5a84d0ece0af652f3453088e1b2524c36730dff7 (patch) | |
tree | d5d3ec766db6002d876c468d7d18132f469cb719 /master/marek-f198907e2b4dda2b548baf557b32d94c1f5e7a1a | |
download | xorg-input-kobomultitouch-pq-master.tar.gz xorg-input-kobomultitouch-pq-master.tar.bz2 xorg-input-kobomultitouch-pq-master.zip |
Diffstat (limited to 'master/marek-f198907e2b4dda2b548baf557b32d94c1f5e7a1a')
-rw-r--r-- | master/marek-f198907e2b4dda2b548baf557b32d94c1f5e7a1a | 281 |
1 files changed, 281 insertions, 0 deletions
diff --git a/master/marek-f198907e2b4dda2b548baf557b32d94c1f5e7a1a b/master/marek-f198907e2b4dda2b548baf557b32d94c1f5e7a1a new file mode 100644 index 0000000..e8aa141 --- /dev/null +++ b/master/marek-f198907e2b4dda2b548baf557b32d94c1f5e7a1a @@ -0,0 +1,281 @@ +diff --git a/Makefile b/Makefile +index 7efc80d..505a2cd 100644 +--- a/Makefile ++++ b/Makefile +@@ -6,7 +6,7 @@ LIBRARY = kobomultitouch_drv.so + MODULES = src + XMODULES = driver + +-o_src = capabilities hwstate mtstate memory mtouch gestures ++o_src = capabilities hwstate mtstate memory mtouch gestures variance_filter jitter_filter + + o_driver= kobomultitouch_drv + +diff --git a/include/jitter_filter.h b/include/jitter_filter.h +new file mode 100644 +index 0000000..00acf98 +--- /dev/null ++++ b/include/jitter_filter.h +@@ -0,0 +1,8 @@ ++#ifndef __JITTER_FILTER_H__ ++#define __JITTER_FILTER_H__ ++ ++#include "mtstate.h" ++ ++int jitter_filter(struct MTState *state); ++ ++#endif +diff --git a/include/variance_filter.h b/include/variance_filter.h +new file mode 100644 +index 0000000..3aad910 +--- /dev/null ++++ b/include/variance_filter.h +@@ -0,0 +1,8 @@ ++#ifndef __VARIANCE_FILTER_H__ ++#define __VARIANCE_FILTER_H__ ++ ++#include "mtstate.h" ++ ++int variance_filter(struct MTState *state); ++ ++#endif +diff --git a/src/jitter_filter.c b/src/jitter_filter.c +new file mode 100644 +index 0000000..2c50adc +--- /dev/null ++++ b/src/jitter_filter.c +@@ -0,0 +1,185 @@ ++/* ++ * Taken and modified from: ++ * ++ * tslib/plugins/dejitter.c ++ * ++ * Copyright (C) 2001 Russell King. ++ * ++ * This file is placed under the LGPL. Please see the file ++ * COPYING for more details. ++ * ++ * ++ * Problem: some touchscreens read the X/Y values from ADC with a ++ * great level of noise in their lowest bits. This produces "jitter" ++ * in touchscreen output, e.g. even if we hold the stylus still, ++ * we get a great deal of X/Y coordinate pairs that are close enough ++ * but not equal. Also if we try to draw a straight line in a painter ++ * program, we'll get a line full of spikes. ++ * ++ * Solution: we apply a smoothing filter on the last several values ++ * thus excluding spikes from output. If we detect a substantial change ++ * in coordinates, we reset the backlog of pen positions, thus avoiding ++ * smoothing coordinates that are not supposed to be smoothed. This ++ * supposes all noise has been filtered by the lower-level filter, ++ * e.g. by the "variance" module. ++ */ ++#include "jitter_filter.h" ++ ++/** ++ * This filter works as follows: we keep track of latest N samples, ++ * and average them with certain weights. The oldest samples have the ++ * least weight and the most recent samples have the most weight. ++ * This helps remove the jitter and at the same time doesn't influence ++ * responsivity because for each input sample we generate one output ++ * sample; pen movement becomes just somehow more smooth. ++ */ ++ ++#define NR_SAMPHISTLEN 8 ++#define MAX_FINGERS 2 ++ ++/* To keep things simple (avoiding division) we ensure that ++ * SUM(weight) = power-of-two. Also we must know how to approximate ++ * measurements when we have less than NR_SAMPHISTLEN samples. ++ */ ++static const unsigned char weight [NR_SAMPHISTLEN - 1][NR_SAMPHISTLEN + 1] = ++{ ++ /* The last element is pow2(SUM(0..3)) */ ++ { 5, 3, 0, 0, 0, 0, 0, 0, 3 }, /* When we have 2 samples ... */ ++ { 8, 5, 3, 0, 0, 0, 0, 0, 4 }, /* When we have 3 samples ... */ ++ { 6, 4, 3, 3, 0, 0, 0, 0, 4 }, /* When we have 4 samples ... */ ++ { 10, 8, 5, 5, 4, 0, 0, 0, 5 }, /* When we have 5 samples ... */ ++ { 9, 7, 5, 4, 4, 3, 0, 0, 5 }, /* When we have 6 samples ... */ ++ { 9, 6, 5, 4, 3, 3, 2, 0, 5 }, /* When we have 7 samples ... */ ++ { 9, 5, 4, 3, 3, 3, 3, 2, 5 }, /* When we have 8 samples ... */ ++}; ++ ++struct ts_hist { ++ int x; ++ int y; ++ unsigned int p; ++}; ++ ++struct tslib_dejitter { ++ int delta; ++ int x; ++ int y; ++ int down; ++ int nr; ++ int head; ++ struct ts_hist hist[NR_SAMPHISTLEN]; ++}; ++ ++static struct tslib_dejitter *djts = NULL; ++ ++static int sqr (int x) ++{ ++ return x * x; ++} ++ ++static void average(struct tslib_dejitter *djt, struct FingerState *samp) ++{ ++ const unsigned char *w; ++ int sn = djt->head; ++ int i, x = 0, y = 0; ++ unsigned int p = 0; ++ ++ w = weight[djt->nr - 2]; ++ ++ for (i = 0; i < djt->nr; i++) { ++ x += djt->hist [sn].x * w [i]; ++ y += djt->hist [sn].y * w [i]; ++ p += djt->hist [sn].p * w [i]; ++ sn = (sn - 1) & (NR_SAMPHISTLEN - 1); ++ } ++ ++ samp->position_x = x >> w [NR_SAMPHISTLEN]; ++ samp->position_y = y >> w [NR_SAMPHISTLEN]; ++ samp->pressure = p >> w [NR_SAMPHISTLEN]; ++//#ifdef DEBUG ++ //fprintf(stderr,"DEJITTER----------------> %d %d %d (nr: %d)\n", ++ // samp->position_x, samp->position_y, samp->pressure, djt->nr); ++//#endif ++} ++ ++static void init() ++{ ++ int i; ++ ++ djts = malloc(sizeof(struct tslib_dejitter) * MAX_FINGERS); ++ memset(djts, 0, sizeof(struct tslib_dejitter) * MAX_FINGERS); ++ ++ for (i = 0; i < MAX_FINGERS; i++) { ++ djts[i].delta = 15; ++ djts[i].head = 0; ++ } ++} ++ ++void jitter_finger(struct FingerState *s) ++{ ++ struct tslib_dejitter *djt; ++ ++ djt = &djts[s->tracking_id]; ++ ++ /* If the pen moves too fast, reset the backlog. */ ++ if (djt->nr) { ++ int prev = (djt->head - 1) & (NR_SAMPHISTLEN - 1); ++ if (sqr (s->position_x - djt->hist [prev].x) + ++ sqr (s->position_y - djt->hist [prev].y) > djt->delta) { ++#ifdef DEBUG ++ //fprintf (stderr, "DEJITTER: pen movement exceeds threshold\n"); ++#endif ++ djt->nr = 0; ++ } ++ } ++ ++ djt->hist[djt->head].x = s->position_x; ++ djt->hist[djt->head].y = s->position_y; ++ djt->hist[djt->head].p = s->pressure; ++ if (djt->nr < NR_SAMPHISTLEN) ++ djt->nr++; ++ ++ /* We'll pass through the very first sample since ++ * we can't average it (no history yet). ++ */ ++ if (djt->nr > 1) { ++ average (djt, s); ++ } ++ ++ djt->head = (djt->head + 1) & (NR_SAMPHISTLEN - 1); ++} ++ ++int jitter_filter(struct MTState *state) ++{ ++ int i; ++ static prev_fingers[MAX_FINGERS]; ++ ++ if (djts == NULL) { ++ init(); ++ for (i = 0; i < MAX_FINGERS; i++) ++ prev_fingers[i] = 0; ++ } ++ ++ for (i = 0; i < state->nfinger; i++) { ++ struct FingerState *s = &state->finger[i]; ++ if (s->tracking_id < 0 || s->tracking_id >= MAX_FINGERS) ++ continue; ++ ++ if (prev_fingers[s->tracking_id] == 0) { ++ // first event - leave it as it is ++ // and place in history ++ djts[s->tracking_id].nr = 0; ++ } ++ ++ jitter_finger(s); ++ } ++ ++ for (i = 0; i < MAX_FINGERS; i++) ++ prev_fingers[i] = 0; ++ for (i = 0; i < state->nfinger; i++) { ++ struct FingerState *s = &state->finger[i]; ++ prev_fingers[s->tracking_id] = 1; ++ } ++ ++ return 1; ++} ++ +diff --git a/src/mtouch.c b/src/mtouch.c +index 388d68e..7b8801e 100644 +--- a/src/mtouch.c ++++ b/src/mtouch.c +@@ -20,6 +20,8 @@ + **************************************************************************/ + + #include "mtouch.h" ++#include "variance_filter.h" ++#include "jitter_filter.h" + + static const int use_grab = 0; + +@@ -44,6 +46,7 @@ int open_mtouch(struct MTouch *mt, int fd) + init_mtstate(&mt->state); + init_memory(&mt->mem); + ++ // we have better filters than included in mtdev + mtdev_set_abs_fuzz(&mt->dev, ABS_MT_POSITION_X, 0); + mtdev_set_abs_fuzz(&mt->dev, ABS_MT_POSITION_Y, 0); + +@@ -78,6 +81,15 @@ int read_packet(struct MTouch *mt, int fd) + if (ret <= 0) + return ret; + extract_mtstate(&mt->state, &mt->hs, &mt->caps); ++ ++ // apply filters ++ ret = variance_filter(&mt->state); ++ if (ret <= 0) ++ return ret; ++ ret = jitter_filter(&mt->state); ++ if (ret <= 0) ++ return ret; ++ + //#if 0 + output_mtstate(&mt->state); + //#endif +diff --git a/src/variance_filter.c b/src/variance_filter.c +new file mode 100644 +index 0000000..e420bfe +--- /dev/null ++++ b/src/variance_filter.c +@@ -0,0 +1,6 @@ ++#include "variance_filter.h" ++ ++int variance_filter(struct MTState *state) ++{ ++ return 1; ++} |