365 lines
9.2 KiB
C
365 lines
9.2 KiB
C
/*-
|
|
* Copyright (c) 2012 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Paul Fleischer <paul@xpg.dk>
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
#include <sys/cdefs.h>
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/callout.h>
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/bus.h>
|
|
|
|
#include <arm/s3c2xx0/s3c24x0var.h>
|
|
#include <arm/s3c2xx0/s3c2440var.h>
|
|
#include <arm/s3c2xx0/s3c2440reg.h>
|
|
|
|
#include <dev/wscons/wsconsio.h>
|
|
#include <dev/wscons/wsmousevar.h>
|
|
#include <dev/wscons/tpcalibvar.h>
|
|
|
|
#include <lib/libsa/qsort.c>
|
|
|
|
#include <dev/hpc/hpcfbio.h>
|
|
|
|
#define MAX_SAMPLES 20
|
|
|
|
struct sstouch_softc {
|
|
device_t dev;
|
|
|
|
bus_space_tag_t iot;
|
|
bus_space_handle_t ioh;
|
|
|
|
uint32_t next_stylus_intr;
|
|
|
|
device_t wsmousedev;
|
|
|
|
struct tpcalib_softc tpcalib;
|
|
|
|
int sample_count;
|
|
int samples_x[MAX_SAMPLES];
|
|
int samples_y[MAX_SAMPLES];
|
|
|
|
callout_t callout;
|
|
};
|
|
|
|
/* Basic Driver Stuff */
|
|
static int sstouch_match (device_t, cfdata_t, void *);
|
|
static void sstouch_attach (device_t, device_t, void *);
|
|
|
|
CFATTACH_DECL_NEW(sstouch, sizeof(struct sstouch_softc), sstouch_match,
|
|
sstouch_attach, NULL, NULL);
|
|
|
|
/* wsmousedev */
|
|
int sstouch_enable(void *);
|
|
int sstouch_ioctl(void *, u_long, void *, int, struct lwp *);
|
|
void sstouch_disable(void *);
|
|
|
|
const struct wsmouse_accessops sstouch_accessops = {
|
|
sstouch_enable,
|
|
sstouch_ioctl,
|
|
sstouch_disable
|
|
};
|
|
|
|
/* Interrupt Handlers */
|
|
int sstouch_tc_intr(void *arg);
|
|
int sstouch_adc_intr(void *arg);
|
|
|
|
void sstouch_callout(void *arg);
|
|
int sstouch_filter_values(int *vals, int val_count);
|
|
void sstouch_initialize(struct sstouch_softc *sc);
|
|
|
|
#define STYLUS_DOWN 0
|
|
#define STYLUS_UP ADCTSC_UD_SEN
|
|
|
|
static struct wsmouse_calibcoords default_calib = {
|
|
.minx = 0,
|
|
.miny = 0,
|
|
.maxx = 0,
|
|
.maxy = 0,
|
|
.samplelen = WSMOUSE_CALIBCOORDS_RESET
|
|
};
|
|
|
|
/* IMPLEMENTATION PART */
|
|
int
|
|
sstouch_match(device_t parent, cfdata_t match, void *aux)
|
|
{
|
|
/* XXX: Check CPU type? */
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
sstouch_attach(device_t parent, device_t self, void *aux)
|
|
{
|
|
struct sstouch_softc *sc = device_private(self);
|
|
struct s3c2xx0_attach_args *sa = aux;
|
|
struct wsmousedev_attach_args mas;
|
|
|
|
sc->dev = self;
|
|
sc->iot = sa->sa_iot;
|
|
|
|
if (bus_space_map(sc->iot, S3C2440_ADC_BASE,
|
|
S3C2440_ADC_SIZE, 0, &sc->ioh)) {
|
|
aprint_error(": failed to map registers");
|
|
return;
|
|
}
|
|
|
|
sc->next_stylus_intr = STYLUS_DOWN;
|
|
|
|
|
|
/* XXX: Is IPL correct? */
|
|
s3c24x0_intr_establish(S3C2440_INT_TC, IPL_BIO, IST_EDGE_RISING,
|
|
sstouch_tc_intr, sc);
|
|
s3c24x0_intr_establish(S3C2440_INT_ADC, IPL_BIO, IST_EDGE_RISING,
|
|
sstouch_adc_intr, sc);
|
|
|
|
aprint_normal("\n");
|
|
|
|
mas.accessops = &sstouch_accessops;
|
|
mas.accesscookie = sc;
|
|
|
|
sc->wsmousedev = config_found(self, &mas, wsmousedevprint, CFARGS_NONE);
|
|
|
|
tpcalib_init(&sc->tpcalib);
|
|
tpcalib_ioctl(&sc->tpcalib, WSMOUSEIO_SCALIBCOORDS,
|
|
(void*)&default_calib, 0, 0);
|
|
|
|
sc->sample_count = 0;
|
|
|
|
/* Add CALLOUT_MPSAFE to avoid holding the global kernel lock */
|
|
callout_init(&sc->callout, 0);
|
|
callout_setfunc(&sc->callout, sstouch_callout, sc);
|
|
|
|
/* Actual initialization is performed by sstouch_initialize(),
|
|
which is called by sstouch_enable() */
|
|
}
|
|
|
|
/* sstouch_tc_intr is the TC interrupt handler.
|
|
The TC interrupt is generated when the stylus changes up->down,
|
|
or down->up state (depending on configuration of ADC_ADCTSC).
|
|
*/
|
|
int
|
|
sstouch_tc_intr(void *arg)
|
|
{
|
|
struct sstouch_softc *sc = (struct sstouch_softc*)arg;
|
|
uint32_t reg;
|
|
|
|
/*aprint_normal("%s\n", __func__);*/
|
|
|
|
/* Figure out if the stylus was lifted or lowered */
|
|
reg = bus_space_read_4(sc->iot, sc->ioh, ADC_ADCUPDN);
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCUPDN, 0x0);
|
|
if( sc->next_stylus_intr == STYLUS_DOWN && (reg & ADCUPDN_TSC_DN) ) {
|
|
sc->next_stylus_intr = STYLUS_UP;
|
|
|
|
sstouch_callout(sc);
|
|
|
|
} else if (sc->next_stylus_intr == STYLUS_UP && (reg & ADCUPDN_TSC_UP)) {
|
|
uint32_t adctsc = 0;
|
|
sc->next_stylus_intr = STYLUS_DOWN;
|
|
|
|
wsmouse_input(sc->wsmousedev, 0x0, 0, 0, 0, 0, 0);
|
|
|
|
sc->sample_count = 0;
|
|
|
|
adctsc |= ADCTSC_YM_SEN | ADCTSC_YP_SEN | ADCTSC_XP_SEN |
|
|
sc->next_stylus_intr |
|
|
3; /* 3 selects "Waiting for Interrupt Mode" */
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCTSC, adctsc);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* sstouch_adc_intr is ADC interrupt handler.
|
|
ADC interrupt is triggered when the ADC controller has a measurement ready.
|
|
*/
|
|
int
|
|
sstouch_adc_intr(void *arg)
|
|
{
|
|
struct sstouch_softc *sc = (struct sstouch_softc*)arg;
|
|
uint32_t reg;
|
|
uint32_t adctsc = 0;
|
|
int x, y;
|
|
|
|
reg = bus_space_read_4(sc->iot, sc->ioh, ADC_ADCDAT0);
|
|
y = reg & ADCDAT_DATAMASK;
|
|
|
|
reg = bus_space_read_4(sc->iot, sc->ioh, ADC_ADCDAT1);
|
|
x = reg & ADCDAT_DATAMASK;
|
|
|
|
|
|
sc->samples_x[sc->sample_count] = x;
|
|
sc->samples_y[sc->sample_count] = y;
|
|
|
|
sc->sample_count++;
|
|
|
|
x = sstouch_filter_values(sc->samples_x, sc->sample_count);
|
|
y = sstouch_filter_values(sc->samples_y, sc->sample_count);
|
|
|
|
if (x == -1 || y == -1) {
|
|
/* If we do not have enough measurements, make some more. */
|
|
sstouch_callout(sc);
|
|
return 1;
|
|
}
|
|
|
|
sc->sample_count = 0;
|
|
|
|
tpcalib_trans(&sc->tpcalib, x, y, &x, &y);
|
|
|
|
wsmouse_input(sc->wsmousedev, 0x1, x, y, 0, 0,
|
|
WSMOUSE_INPUT_ABSOLUTE_X | WSMOUSE_INPUT_ABSOLUTE_Y);
|
|
|
|
/* Schedule a new adc measurement, unless the stylus has been lifed */
|
|
if (sc->next_stylus_intr == STYLUS_UP) {
|
|
callout_schedule(&sc->callout, hz/50);
|
|
}
|
|
|
|
/* Until measurement is to be performed, listen for stylus up-events */
|
|
adctsc |= ADCTSC_YM_SEN | ADCTSC_YP_SEN | ADCTSC_XP_SEN |
|
|
sc->next_stylus_intr |
|
|
3; /* 3 selects "Waiting for Interrupt Mode" */
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCTSC, adctsc);
|
|
|
|
|
|
return 1;
|
|
}
|
|
|
|
int
|
|
sstouch_enable(void *arg)
|
|
{
|
|
struct sstouch_softc *sc = arg;
|
|
|
|
sstouch_initialize(sc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
sstouch_ioctl(void *v, u_long cmd, void *data, int flag, struct lwp *l)
|
|
{
|
|
struct sstouch_softc *sc = v;
|
|
|
|
aprint_normal("%s\n", __func__);
|
|
|
|
switch (cmd) {
|
|
case WSMOUSEIO_GTYPE:
|
|
*(uint *)data = WSMOUSE_TYPE_PSEUDO;
|
|
break;
|
|
case WSMOUSEIO_GCALIBCOORDS:
|
|
case WSMOUSEIO_SCALIBCOORDS:
|
|
return tpcalib_ioctl(&sc->tpcalib, cmd, data, flag, l);
|
|
default:
|
|
return EPASSTHROUGH;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
sstouch_disable(void *arg)
|
|
{
|
|
struct sstouch_softc *sc = (struct sstouch_softc*)arg;
|
|
|
|
/* By setting ADCCON register to 0, we also disable
|
|
the prescaler, which should disable any interrupts.
|
|
*/
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCCON, 0);
|
|
}
|
|
|
|
void
|
|
sstouch_callout(void *arg)
|
|
{
|
|
struct sstouch_softc *sc = (struct sstouch_softc*)arg;
|
|
|
|
/* If stylus is down, perform a measurement */
|
|
if (sc->next_stylus_intr == STYLUS_UP) {
|
|
uint32_t reg;
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCTSC,
|
|
ADCTSC_YM_SEN | ADCTSC_YP_SEN |
|
|
ADCTSC_XP_SEN | ADCTSC_PULL_UP |
|
|
ADCTSC_AUTO_PST);
|
|
|
|
reg = bus_space_read_4(sc->iot, sc->ioh, ADC_ADCCON);
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCCON,
|
|
reg | ADCCON_ENABLE_START);
|
|
}
|
|
|
|
}
|
|
|
|
/* Do some very simple filtering on the measured values */
|
|
int
|
|
sstouch_filter_values(int *vals, int val_count)
|
|
{
|
|
int sum = 0;
|
|
|
|
if (val_count < 5)
|
|
return -1;
|
|
|
|
for (int i=0; i<val_count; i++) {
|
|
sum += vals[i];
|
|
}
|
|
|
|
return sum/val_count;
|
|
}
|
|
|
|
void
|
|
sstouch_initialize(struct sstouch_softc *sc)
|
|
{
|
|
int prescaler;
|
|
uint32_t adccon = 0;
|
|
uint32_t adctsc = 0;
|
|
|
|
/* ADC Conversion rate is calculated by:
|
|
f(ADC) = PCLK/(prescaler+1)
|
|
|
|
The ADC can operate at a maximum frequency of 2.5MHz for
|
|
500 KSPS.
|
|
*/
|
|
|
|
/* Set f(ADC) = 50MHz / 256 = 1,95MHz */
|
|
prescaler = 0xff;
|
|
|
|
adccon |= ((prescaler<<ADCCON_PRSCVL_SHIFT) &
|
|
ADCCON_PRSCVL_MASK);
|
|
adccon |= ADCCON_PRSCEN;
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCCON, adccon);
|
|
|
|
/* Use Auto Sequential measurement of X and Y positions */
|
|
adctsc |= ADCTSC_YM_SEN | ADCTSC_YP_SEN | ADCTSC_XP_SEN |
|
|
sc->next_stylus_intr |
|
|
3; /* 3 selects "Waiting for Interrupt Mode" */
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCTSC, adctsc);
|
|
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCUPDN, 0x0);
|
|
|
|
/* Time used to measure each X/Y position value? */
|
|
bus_space_write_4(sc->iot, sc->ioh, ADC_ADCDLY, 10000);
|
|
}
|