384 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			384 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/**
 | 
						|
 * @file AD_touch.c
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#include "AD_touch.h"
 | 
						|
 | 
						|
#if USE_AD_TOUCH
 | 
						|
 | 
						|
#include LV_DRV_INDEV_INCLUDE
 | 
						|
#include LV_DRV_DELAY_INCLUDE
 | 
						|
 | 
						|
#define SAMPLE_POINTS   4
 | 
						|
 | 
						|
#define CALIBRATIONINSET 20 // range 0 <= CALIBRATIONINSET <= 40
 | 
						|
 | 
						|
#define RESISTIVETOUCH_AUTO_SAMPLE_MODE
 | 
						|
#define TOUCHSCREEN_RESISTIVE_PRESS_THRESHOLD 350 // between 0-0x03ff the lesser this value
 | 
						|
 | 
						|
 | 
						|
// Current ADC values for X and Y channels
 | 
						|
int16_t adcX = 0;
 | 
						|
int16_t adcY = 0;
 | 
						|
volatile unsigned int adcTC = 0;
 | 
						|
 | 
						|
// coefficient values
 | 
						|
volatile long _trA;
 | 
						|
volatile long _trB;
 | 
						|
volatile long _trC;
 | 
						|
volatile long _trD;
 | 
						|
 | 
						|
volatile int16_t xRawTouch[SAMPLE_POINTS] = {TOUCHCAL_ULX, TOUCHCAL_URX, TOUCHCAL_LRX, TOUCHCAL_LLX};
 | 
						|
volatile int16_t yRawTouch[SAMPLE_POINTS] = {TOUCHCAL_ULY, TOUCHCAL_URY, TOUCHCAL_LRY, TOUCHCAL_LLY};
 | 
						|
 | 
						|
#define TOUCHSCREEN_RESISTIVE_CALIBRATION_SCALE_FACTOR 8
 | 
						|
 | 
						|
// use this scale factor to avoid working in floating point numbers
 | 
						|
#define SCALE_FACTOR (1 << TOUCHSCREEN_RESISTIVE_CALIBRATION_SCALE_FACTOR)
 | 
						|
 | 
						|
typedef enum {
 | 
						|
    IDLE, //0
 | 
						|
    SET_X, //1
 | 
						|
    RUN_X, //2
 | 
						|
    GET_X, //3
 | 
						|
    RUN_CHECK_X, //4
 | 
						|
    CHECK_X, //5
 | 
						|
    SET_Y, //6
 | 
						|
    RUN_Y, //7
 | 
						|
    GET_Y, //8
 | 
						|
    CHECK_Y, //9
 | 
						|
    SET_VALUES, //10
 | 
						|
    GET_POT, //11
 | 
						|
    RUN_POT //12
 | 
						|
} TOUCH_STATES;
 | 
						|
 | 
						|
volatile TOUCH_STATES state = IDLE;
 | 
						|
 | 
						|
#define CAL_X_INSET (((GetMaxX() + 1) * (CALIBRATIONINSET >> 1)) / 100)
 | 
						|
#define CAL_Y_INSET (((GetMaxY() + 1) * (CALIBRATIONINSET >> 1)) / 100)
 | 
						|
 | 
						|
int stat;
 | 
						|
int16_t temp_x, temp_y;
 | 
						|
 | 
						|
 | 
						|
static int16_t TouchGetX(void);
 | 
						|
static int16_t TouchGetRawX(void);
 | 
						|
static int16_t TouchGetY(void);
 | 
						|
static int16_t TouchGetRawY(void);
 | 
						|
static int16_t TouchDetectPosition(void);
 | 
						|
static void TouchCalculateCalPoints(void);
 | 
						|
 | 
						|
 | 
						|
/********************************************************************/
 | 
						|
void ad_touch_init(void)
 | 
						|
{
 | 
						|
    // Initialize ADC for auto sampling mode
 | 
						|
    AD1CON1 = 0;      // reset
 | 
						|
    AD1CON2 = 0;      // AVdd, AVss, int every conversion, MUXA only
 | 
						|
    AD1CON3 = 0x1FFF; // 31 Tad auto-sample, Tad = 256*Tcy
 | 
						|
    AD1CON1 = 0x80E0; // Turn on A/D module, use auto-convert
 | 
						|
 | 
						|
 | 
						|
    ADPCFG_XPOS = RESISTIVETOUCH_ANALOG;
 | 
						|
    ADPCFG_YPOS = RESISTIVETOUCH_ANALOG;
 | 
						|
 | 
						|
    AD1CSSL = 0; // No scanned inputs
 | 
						|
 | 
						|
    state = SET_X; // set the state of the state machine to start the sampling
 | 
						|
 | 
						|
    /*Load calibration data*/
 | 
						|
    xRawTouch[0] = TOUCHCAL_ULX;
 | 
						|
    yRawTouch[0] = TOUCHCAL_ULY;
 | 
						|
    xRawTouch[1] = TOUCHCAL_URX;
 | 
						|
    yRawTouch[1] = TOUCHCAL_URY;
 | 
						|
    xRawTouch[3] = TOUCHCAL_LLX;
 | 
						|
    yRawTouch[3] = TOUCHCAL_LLY;
 | 
						|
    xRawTouch[2] = TOUCHCAL_LRX;
 | 
						|
    yRawTouch[2] = TOUCHCAL_LRY;
 | 
						|
 | 
						|
    TouchCalculateCalPoints();
 | 
						|
}
 | 
						|
 | 
						|
/*Use this in lv_indev_drv*/
 | 
						|
bool ad_touch_read(lv_indev_drv_t * indev_drv, lv_indev_data_t * data)
 | 
						|
{
 | 
						|
    static int16_t last_x = 0;
 | 
						|
    static int16_t last_y = 0;
 | 
						|
 | 
						|
    int16_t x, y;
 | 
						|
 | 
						|
    x = TouchGetX();
 | 
						|
    y = TouchGetY();
 | 
						|
 | 
						|
    if((x > 0) && (y > 0)) {
 | 
						|
        data->point.x = x;
 | 
						|
        data->point.y = y;
 | 
						|
        last_x = data->point.x;
 | 
						|
        last_y = data->point.y;
 | 
						|
        data->state = LV_INDEV_STATE_PR;
 | 
						|
    } else {
 | 
						|
        data->point.x = last_x;
 | 
						|
        data->point.y = last_y;
 | 
						|
        data->state = LV_INDEV_STATE_REL;
 | 
						|
    }
 | 
						|
 | 
						|
    return false;
 | 
						|
}
 | 
						|
 | 
						|
/* Call periodically (e.g. in every 1 ms) to handle reading with ADC*/
 | 
						|
int16_t ad_touch_handler(void)
 | 
						|
{
 | 
						|
    static int16_t tempX, tempY;
 | 
						|
    int16_t temp;
 | 
						|
 | 
						|
    switch(state) {
 | 
						|
        case IDLE:
 | 
						|
            adcX = 0;
 | 
						|
            adcY = 0;
 | 
						|
            break;
 | 
						|
 | 
						|
        case SET_VALUES:
 | 
						|
            if(!TOUCH_ADC_DONE)
 | 
						|
                break;
 | 
						|
            if((WORD)TOUCHSCREEN_RESISTIVE_PRESS_THRESHOLD < (WORD)ADC1BUF0) {
 | 
						|
                adcX = 0;
 | 
						|
                adcY = 0;
 | 
						|
            } else {
 | 
						|
                adcX = tempX;
 | 
						|
                adcY = tempY;
 | 
						|
            }
 | 
						|
            state = SET_X;
 | 
						|
            return 1; // touch screen acquisition is done
 | 
						|
 | 
						|
        case SET_X:
 | 
						|
            TOUCH_ADC_INPUT_SEL = ADC_XPOS;
 | 
						|
 | 
						|
            ResistiveTouchScreen_XPlus_Config_As_Input();
 | 
						|
            ResistiveTouchScreen_YPlus_Config_As_Input();
 | 
						|
            ResistiveTouchScreen_XMinus_Config_As_Input();
 | 
						|
            ResistiveTouchScreen_YMinus_Drive_Low();
 | 
						|
            ResistiveTouchScreen_YMinus_Config_As_Output();
 | 
						|
 | 
						|
            ADPCFG_YPOS = RESISTIVETOUCH_DIGITAL; // set to digital pin
 | 
						|
            ADPCFG_XPOS = RESISTIVETOUCH_ANALOG; // set to analog pin
 | 
						|
 | 
						|
            TOUCH_ADC_START = 1; // run conversion
 | 
						|
            state = CHECK_X;
 | 
						|
            break;
 | 
						|
 | 
						|
        case CHECK_X:
 | 
						|
        case CHECK_Y:
 | 
						|
 | 
						|
            if(TOUCH_ADC_DONE == 0) {
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            if((WORD)TOUCHSCREEN_RESISTIVE_PRESS_THRESHOLD > (WORD)ADC1BUF0) {
 | 
						|
                if(state == CHECK_X) {
 | 
						|
                    ResistiveTouchScreen_YPlus_Drive_High();
 | 
						|
                    ResistiveTouchScreen_YPlus_Config_As_Output();
 | 
						|
                    tempX = 0;
 | 
						|
                    state = RUN_X;
 | 
						|
                } else {
 | 
						|
                    ResistiveTouchScreen_XPlus_Drive_High();
 | 
						|
                    ResistiveTouchScreen_XPlus_Config_As_Output();
 | 
						|
                    tempY = 0;
 | 
						|
                    state = RUN_Y;
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                adcX = 0;
 | 
						|
                adcY = 0;
 | 
						|
                state = SET_X;
 | 
						|
                return 1; // touch screen acquisition is done
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
        case RUN_X:
 | 
						|
        case RUN_Y:
 | 
						|
            TOUCH_ADC_START = 1;
 | 
						|
            state = (state == RUN_X) ? GET_X : GET_Y;
 | 
						|
            // no break needed here since the next state is either GET_X or GET_Y
 | 
						|
            break;
 | 
						|
 | 
						|
        case GET_X:
 | 
						|
        case GET_Y:
 | 
						|
            if(!TOUCH_ADC_DONE)
 | 
						|
                break;
 | 
						|
 | 
						|
            temp = ADC1BUF0;
 | 
						|
            if(state == GET_X) {
 | 
						|
                if(temp != tempX) {
 | 
						|
                    tempX =  temp;
 | 
						|
                    state = RUN_X;
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                if(temp != tempY) {
 | 
						|
                    tempY = temp;
 | 
						|
                    state = RUN_Y;
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            if(state == GET_X)
 | 
						|
                ResistiveTouchScreen_YPlus_Config_As_Input();
 | 
						|
            else
 | 
						|
                ResistiveTouchScreen_XPlus_Config_As_Input();
 | 
						|
            TOUCH_ADC_START = 1;
 | 
						|
            state = (state == GET_X) ? SET_Y : SET_VALUES;
 | 
						|
            break;
 | 
						|
 | 
						|
        case SET_Y:
 | 
						|
            if(!TOUCH_ADC_DONE)
 | 
						|
                break;
 | 
						|
 | 
						|
            if((WORD)TOUCHSCREEN_RESISTIVE_PRESS_THRESHOLD < (WORD)ADC1BUF0) {
 | 
						|
                adcX = 0;
 | 
						|
                adcY = 0;
 | 
						|
                state = SET_X;
 | 
						|
                return 1; // touch screen acquisition is done
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            TOUCH_ADC_INPUT_SEL = ADC_YPOS;
 | 
						|
 | 
						|
            ResistiveTouchScreen_XPlus_Config_As_Input();
 | 
						|
            ResistiveTouchScreen_YPlus_Config_As_Input();
 | 
						|
            ResistiveTouchScreen_XMinus_Drive_Low();
 | 
						|
            ResistiveTouchScreen_XMinus_Config_As_Output();
 | 
						|
            ResistiveTouchScreen_YMinus_Config_As_Input();
 | 
						|
 | 
						|
            ADPCFG_YPOS = RESISTIVETOUCH_ANALOG; // set to analog pin
 | 
						|
            ADPCFG_XPOS = RESISTIVETOUCH_DIGITAL; // set to digital pin
 | 
						|
            TOUCH_ADC_START = 1; // run conversion
 | 
						|
            state = CHECK_Y;
 | 
						|
            break;
 | 
						|
 | 
						|
        default:
 | 
						|
            state = SET_X;
 | 
						|
            return 1; // touch screen acquisition is done
 | 
						|
    }
 | 
						|
    stat = state;
 | 
						|
    temp_x = adcX;
 | 
						|
    temp_y = adcY;
 | 
						|
 | 
						|
    return 0; // touch screen acquisition is not done
 | 
						|
}
 | 
						|
 | 
						|
/**********************
 | 
						|
 *   STATIC FUNCTIONS
 | 
						|
 **********************/
 | 
						|
 | 
						|
/********************************************************************/
 | 
						|
static int16_t TouchGetX(void)
 | 
						|
{
 | 
						|
    long result;
 | 
						|
 | 
						|
    result = TouchGetRawX();
 | 
						|
 | 
						|
    if(result > 0) {
 | 
						|
        result = (long)((((long)_trC * result) + _trD) >> TOUCHSCREEN_RESISTIVE_CALIBRATION_SCALE_FACTOR);
 | 
						|
 | 
						|
    }
 | 
						|
    return ((int16_t)result);
 | 
						|
}
 | 
						|
/********************************************************************/
 | 
						|
static int16_t TouchGetRawX(void)
 | 
						|
{
 | 
						|
#ifdef TOUCHSCREEN_RESISTIVE_SWAP_XY
 | 
						|
    return adcY;
 | 
						|
#else
 | 
						|
    return adcX;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
/********************************************************************/
 | 
						|
static int16_t TouchGetY(void)
 | 
						|
{
 | 
						|
 | 
						|
    long result;
 | 
						|
 | 
						|
    result = TouchGetRawY();
 | 
						|
 | 
						|
    if(result > 0) {
 | 
						|
        result = (long)((((long)_trA * result) + (long)_trB) >> TOUCHSCREEN_RESISTIVE_CALIBRATION_SCALE_FACTOR);
 | 
						|
 | 
						|
    }
 | 
						|
    return ((int16_t)result);
 | 
						|
}
 | 
						|
 | 
						|
/********************************************************************/
 | 
						|
static int16_t TouchGetRawY(void)
 | 
						|
{
 | 
						|
#ifdef TOUCHSCREEN_RESISTIVE_SWAP_XY
 | 
						|
    return adcX;
 | 
						|
#else
 | 
						|
    return adcY;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void TouchCalculateCalPoints(void)
 | 
						|
{
 | 
						|
    long trA, trB, trC, trD; // variables for the coefficients
 | 
						|
    long trAhold, trBhold, trChold, trDhold;
 | 
						|
    long test1, test2; // temp variables (must be signed type)
 | 
						|
 | 
						|
    int16_t xPoint[SAMPLE_POINTS], yPoint[SAMPLE_POINTS];
 | 
						|
 | 
						|
    yPoint[0] = yPoint[1] = CAL_Y_INSET;
 | 
						|
    yPoint[2] = yPoint[3] = (GetMaxY() - CAL_Y_INSET);
 | 
						|
    xPoint[0] = xPoint[3] = CAL_X_INSET;
 | 
						|
    xPoint[1] = xPoint[2] = (GetMaxX() - CAL_X_INSET);
 | 
						|
 | 
						|
    // calculate points transfer functiona
 | 
						|
    // based on two simultaneous equations solve for the
 | 
						|
    // constants
 | 
						|
 | 
						|
    // use sample points 1 and 4
 | 
						|
    // Dy1 = aTy1 + b; Dy4 = aTy4 + b
 | 
						|
    // Dx1 = cTx1 + d; Dy4 = aTy4 + b
 | 
						|
 | 
						|
    test1 = (long)yPoint[0] - (long)yPoint[3];
 | 
						|
    test2 = (long)yRawTouch[0] - (long)yRawTouch[3];
 | 
						|
 | 
						|
    trA = ((long)((long)test1 * SCALE_FACTOR) / test2);
 | 
						|
    trB = ((long)((long)yPoint[0] * SCALE_FACTOR) - (trA * (long)yRawTouch[0]));
 | 
						|
 | 
						|
    test1 = (long)xPoint[0] - (long)xPoint[2];
 | 
						|
    test2 = (long)xRawTouch[0] - (long)xRawTouch[2];
 | 
						|
 | 
						|
    trC = ((long)((long)test1 * SCALE_FACTOR) / test2);
 | 
						|
    trD = ((long)((long)xPoint[0] * SCALE_FACTOR) - (trC * (long)xRawTouch[0]));
 | 
						|
 | 
						|
    trAhold = trA;
 | 
						|
    trBhold = trB;
 | 
						|
    trChold = trC;
 | 
						|
    trDhold = trD;
 | 
						|
 | 
						|
    // use sample points 2 and 3
 | 
						|
    // Dy2 = aTy2 + b; Dy3 = aTy3 + b
 | 
						|
    // Dx2 = cTx2 + d; Dy3 = aTy3 + b
 | 
						|
 | 
						|
    test1 = (long)yPoint[1] - (long)yPoint[2];
 | 
						|
    test2 = (long)yRawTouch[1] - (long)yRawTouch[2];
 | 
						|
 | 
						|
    trA = ((long)(test1 * SCALE_FACTOR) / test2);
 | 
						|
    trB = ((long)((long)yPoint[1] * SCALE_FACTOR) - (trA * (long)yRawTouch[1]));
 | 
						|
 | 
						|
    test1 = (long)xPoint[1] - (long)xPoint[3];
 | 
						|
    test2 = (long)xRawTouch[1] - (long)xRawTouch[3];
 | 
						|
 | 
						|
    trC = ((long)((long)test1 * SCALE_FACTOR) / test2);
 | 
						|
    trD = ((long)((long)xPoint[1] * SCALE_FACTOR) - (trC * (long)xRawTouch[1]));
 | 
						|
 | 
						|
    // get the average and use the average
 | 
						|
    _trA = (trA + trAhold) >> 1;
 | 
						|
    _trB = (trB + trBhold) >> 1;
 | 
						|
    _trC = (trC + trChold) >> 1;
 | 
						|
    _trD = (trD + trDhold) >> 1;
 | 
						|
}
 | 
						|
 | 
						|
#endif
 |