OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [mw/] [src/] [contrib/] [GPL/] [tpcal/] [transform.c] - Rev 1765

Compare with Previous | Blame | View Log

/*
 * transform.c
 * Calculate coefficients for tranformation equation
 * Copyright (C) 1999 Bradley D. LaRonde <brad@ltc.com>
 *
 * This program is free software; you may redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */
#include "windows.h"
#include "mou_tp.h"
#include "transform.h"
 
int CalcTransformationCoefficientsSimple(CALIBRATION_PAIRS *pcp, TRANSFORMATION_COEFFICIENTS *ptc)
{
	/*
	 * This is my simple way of calculating some of the coefficients -
	 * enough to do a simple scale and translate.
	 * It ignores any dependencies between axiis (rotation and/or skew).
	 */
 
	int min_screen_x = pcp->ul.screen.x;
	int min_screen_y = pcp->ul.screen.y;
	int max_screen_x = pcp->lr.screen.x;
	int max_screen_y = pcp->lr.screen.y;
 
	int min_device_x = pcp->ul.device.x;
	int min_device_y = pcp->ul.device.y;
	int max_device_x = pcp->lr.device.x;
	int max_device_y = pcp->lr.device.y;
 
	ptc->s = (1 << 16);
	ptc->a = ptc->s * (min_screen_x - max_screen_x) / (min_device_x - max_device_x);
	ptc->b = 0;
	ptc->c = (ptc->a * -min_device_x) + (ptc->s * min_screen_x);
	ptc->d = 0;
	ptc->e = ptc->s * (min_screen_y - max_screen_y) / (min_device_y - max_device_y);
	ptc->f = (ptc->e * -min_device_y) + (ptc->s * min_screen_y);
 
	return 0;
}
 
int CalcTransformationCoefficientsBetter(CALIBRATION_PAIRS *pcp, TRANSFORMATION_COEFFICIENTS *ptc)
{
	/*
	 * Janus (the man) <janus@place.org> came up with a much better way
	 * to figure the coefficients.  His original algorithm was written in MOO.
	 * Jay Carlson <> did the translation to C.
	 * This way takes into account inter-axis dependency like rotation and skew.
	 */
 
	double vector[3][2] =
	{
		{pcp->ul.screen.x, pcp->ul.screen.y},
		{pcp->ur.screen.x, pcp->ur.screen.y},
		{pcp->lr.screen.x, pcp->lr.screen.y}
	};
 
	double matrix[3][3] =
	{
		{pcp->ul.device.x, pcp->ul.device.y, 1.0},
		{pcp->ur.device.x, pcp->ur.device.y, 1.0},
		{pcp->lr.device.x, pcp->lr.device.y, 1.0}
	};
 
	int i, j, r, k;
	double p, q;
 
	for (i = 0; i < 3; i++) {
          p = matrix[i][i];
 
          for (j = 0; j < 3; j++) {
               matrix[i][j] = matrix[i][j] / p;
          }
 
          for (j = 0; j < 2; j++) {
               vector[i][j] = vector[i][j] / p;
          }
 
          for (r = 0; r < 3; r++) {
               if (r != i) {
                    q = matrix[r][i];
 
                    matrix[r][i] = 0.0;
 
                    for (k = i + 1; k < 3; k++) {
                         matrix[r][k] = matrix[r][k] - (q * matrix[i][k]);
                    }
 
                    for (k = 0; k < 2; k++) {
                         vector[r][k] = vector[r][k] - (q * vector[i][k]);
                    }
               }
          }
	}
 
	ptc->s = 1 << 16;
	ptc->a = vector[0][0] * ptc->s;
	ptc->b = vector[1][0] * ptc->s;
	ptc->c = vector[2][0] * ptc->s;
	ptc->d = vector[0][1] * ptc->s;
	ptc->e = vector[1][1] * ptc->s;
	ptc->f = vector[2][1] * ptc->s;
 
	return 0;
}
 
int CalcTransformationCoefficientsEvenBetter(CALIBRATION_PAIRS *pcp, TRANSFORMATION_COEFFICIENTS *ptc)
{
	/*
	 * Mike Klar <> added the an xy term to correct for trapezoidial distortion.
	 */
 
	double vector[4][2] =
	{
		{pcp->ul.screen.x, pcp->ul.screen.y},
		{pcp->ur.screen.x, pcp->ur.screen.y},
		{pcp->lr.screen.x, pcp->lr.screen.y},
		{pcp->ll.screen.x, pcp->ll.screen.y}
	};
 
	double matrix[4][4] =
	{
		{pcp->ul.device.x, pcp->ul.device.x * pcp->ul.device.y, pcp->ul.device.y, 1.0},
		{pcp->ur.device.x, pcp->ur.device.x * pcp->ur.device.y, pcp->ur.device.y, 1.0},
		{pcp->lr.device.x, pcp->lr.device.x * pcp->lr.device.y, pcp->lr.device.y, 1.0},
		{pcp->ll.device.x, pcp->ll.device.x * pcp->ll.device.y, pcp->ll.device.y, 1.0}
	};
 
	int i, j, r, k;
	double p, q;
 
	for (i = 0; i < 4; i++) {
          p = matrix[i][i];
 
          for (j = 0; j < 4; j++) {
               matrix[i][j] = matrix[i][j] / p;
          }
 
          for (j = 0; j < 2; j++) {
               vector[i][j] = vector[i][j] / p;
          }
 
          for (r = 0; r < 4; r++) {
               if (r != i) {
                    q = matrix[r][i];
 
                    matrix[r][i] = 0.0;
 
                    for (k = i + 1; k < 4; k++) {
                         matrix[r][k] = matrix[r][k] - (q * matrix[i][k]);
                    }
 
                    for (k = 0; k < 2; k++) {
                         vector[r][k] = vector[r][k] - (q * vector[i][k]);
                    }
               }
          }
	}
 
	/* I just drop the xy coefficient since it is so small. */
	ptc->s = 1 << 16;
	ptc->a = vector[0][0] * ptc->s;
	ptc->b = vector[2][0] * ptc->s;
	ptc->c = vector[3][0] * ptc->s;
	ptc->d = vector[0][1] * ptc->s;
	ptc->e = vector[2][1] * ptc->s;
	ptc->f = vector[3][1] * ptc->s;
 
	return 0;
}
 
int CalcTransformationCoefficientsBest(CALIBRATION_PAIRS *pcp, TRANSFORMATION_COEFFICIENTS *ptc)
{
	/*
	 * Mike Klar <> came up with a best-fit solution that works best.
	 */
 
	const int first_point = 0;
	const int last_point = 4;
	int i;
 
	double Sx=0, Sy=0, Sxy=0, Sx2=0, Sy2=0, Sm=0, Sn=0, Smx=0, Smy=0, Snx=0, Sny=0, S=0;
	double t1, t2, t3, t4, t5, t6, q;
 
	/* cast the struct to an array - hacky but ok */
	CALIBRATION_PAIR *cp = (CALIBRATION_PAIR*)pcp;
 
	/*
	 * Do a best-fit calculation for as many points as we want, as
	 * opposed to an exact fit, which can only be done against 3 points.
	 *
	 * The following calculates various sumnations of the sample data
	 * coordinates.  For purposes of naming convention, x and y
	 * refer to device coordinates, m and n refer to screen
	 * coordinates, S means sumnation.  x2 and y2 are x squared and
	 * y squared, S by itself is just a count of points (= sumnation
	 * of 1).
	 */
 
	for (i = first_point; i < last_point + 1; i++) {
		Sx += cp[i].device.x;
		Sy += cp[i].device.y;
		Sxy += cp[i].device.x * cp[i].device.y;
		Sx2 += cp[i].device.x * cp[i].device.x;
		Sy2 += cp[i].device.y * cp[i].device.y;
		Sm += cp[i].screen.x;
		Sn += cp[i].screen.y;
		Smx += cp[i].screen.x * cp[i].device.x;
		Smy += cp[i].screen.x * cp[i].device.y;
		Snx += cp[i].screen.y * cp[i].device.x;
		Sny += cp[i].screen.y * cp[i].device.y;
		S += 1;
	}
 
#if 0
	printf("%f, %f, %f, %f, "
	       "%f, %f, %f, %f, "
	       "%f, %f, %f, %f\n",
	        Sx, Sy, Sxy, Sx2, Sy2, Sm, Sn, Smx, Smy, Snx, Sny, S);
#endif
 
	/*
	 * Next we solve the simultaneous equations (these equations minimize
	 * the sum of the square of the m and n error):
	 *
	 *    | Sx2 Sxy Sx |   | a d |   | Smx Snx |
	 *    | Sxy Sy2 Sy | * | b e | = | Smy Sny |
	 *    | Sx  Sy  S  |   | c f |   | Sm  Sn  |
	 *
	 * We could do the matrix solution in code, but that leads to several
	 * divide by 0 conditions for cases where the data is truly solvable
	 * (becuase those terms cancel out of the final solution), so we just
	 * give the final solution instread.  t1 through t6 and q are just
	 * convenience variables for terms that are used repeatedly - we could
	 * calculate each of the coefficients directly at this point with a
	 * nasty long equation, but that would be extremly inefficient.
	 */
 
	t1 = Sxy * Sy - Sx * Sy2;
	t2 = Sxy * Sx - Sx2 * Sy;
	t3 = Sx2 * Sy2 - Sxy * Sxy;
	t4 = Sy2 * S - Sy * Sy;
	t5 = Sx * Sy - Sxy * S;
	t6 = Sx2 * S - Sx * Sx;
 
	q = t1 * Sx + t2 * Sy + t3 * S;
 
	/*
	 * If q = 0, then the data is unsolvable.  This should only happen
	 * when there are not enough unique data points (less than 3 points
	 * will give infinite solutions), or at least one of the 
	 * coefficients is infinite (which would indicate that the same
	 * device point represents an infinite area of the screen, probably
	 * as a result of the same device data point given for 2 different
	 * screen points).  The first condition should never happen, since
	 * we're always feeding in at least 3 unique screen points.  The
	 * second condition would probably indicate bad user input or the
	 * touchpanel device returning bad data.
	 */
 
	if (q == 0)
		return -1;
 
	ptc->s = 1 << 16;
	ptc->a = ((t4 * Smx + t5 * Smy + t1 * Sm) / q + 0.5/65536) * ptc->s;
	ptc->b = ((t5 * Smx + t6 * Smy + t2 * Sm) / q + 0.5/65536) * ptc->s;
	ptc->c = ((t1 * Smx + t2 * Smy + t3 * Sm) / q + 0.5/65536) * ptc->s;
	ptc->d = ((t4 * Snx + t5 * Sny + t1 * Sn) / q + 0.5/65536) * ptc->s;
	ptc->e = ((t5 * Snx + t6 * Sny + t2 * Sn) / q + 0.5/65536) * ptc->s;
	ptc->f = ((t1 * Snx + t2 * Sny + t3 * Sn) / q + 0.5/65536) * ptc->s;
 
	/*
	 * Finally, we check for overflow on the fp to integer conversion,
	 * which would also probably indicate bad data.
	 */
 
	if ( (ptc->a == 0x80000000) || (ptc->b == 0x80000000) ||
	     (ptc->c == 0x80000000) || (ptc->d == 0x80000000) ||
	     (ptc->e == 0x80000000) || (ptc->f == 0x80000000) )
		return -1;
 
	return 0;
}
 
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.