git: 9front

ref: c6b681c48e27d4d5d258c393567878b55d1a43e2
dir: /sys/src/libscribble/li_recognizer.c/

View raw version
/*
 *  li_recognizer.c
 *
 *	Copyright 2000 Compaq Computer Corporation.
 *	Copying or modifying this code for any purpose is permitted,
 *	provided that this copyright notice is preserved in its entirety
 *	in all copies or modifications.
 *	COMPAQ COMPUTER CORPORATION MAKES NO WARRANTIES, EXPRESSED OR
 *	IMPLIED, AS TO THE USEFULNESS OR CORRECTNESS OF THIS CODE OR
 *
 *
 *  Adapted from cmu_recognizer.c by Jay Kistler.
 *  
 *  Where is the CMU copyright???? Gotta track it down - Jim Gettys
 *
 *  Credit to Dean Rubine, Jim Kempf, and Ari Rapkin.
 */

#include <u.h>
#include <libc.h>
#include <stdio.h>
#include <draw.h>
#include <scribble.h>
#include "scribbleimpl.h"

#include "hre_internal.h"
#include "li_recognizer_internal.h"

int lidebug = 0;

#define LI_MAGIC 0xACCBADDD

#define CHECK_LI_MAGIC(_a) \
  ((_a) != nil && ((li_recognizer*)(_a))->li_magic == LI_MAGIC)


static void lialg_initialize(rClassifier *);
static int lialg_read_classifier_digest(rClassifier *);
static int lialg_canonicalize_examples(rClassifier *);
static char *lialg_recognize_stroke(rClassifier *, point_list *);
static void lialg_compute_lpf_parameters(void);


char* li_err_msg = nil;

#define bcopy(s1,s2,n) memmove(s2,s1,n)

/*Freeing classifier*/

static void
free_rClassifier(rClassifier* rc);

/*
 * Point List Support
*/

static point_list*
add_example(point_list* l,int npts,pen_point* pts)
{
	pen_point* lpts = mallocz(npts*sizeof(pen_point), 1);
	point_list *p = malloc(sizeof(point_list));

	p->npts = npts;
	p->pts = lpts;
	p->next = l;       /*Order doesn't matter, so we stick on end.*/

	/*Copy points.*/

	bcopy(pts, lpts, npts * sizeof(pen_point));

	return(p);
}
	

static void
delete_examples(point_list* l)
{
	point_list* p;

	for( ; l != nil; l = p ) {
		p = l->next;
		free(l->pts);
		free(l);
	}
}

/*
 * Local functions
 */

/*
 * recognize_internal-Form Vector, use Classifier to classify, return char.
 */

static char*
recognize_internal(rClassifier* rec, Stroke* str, int*)
{
	char *res;
	point_list *stroke;

	stroke = add_example(nil, str->npts, str->pts);
	if (stroke == nil) return(nil);

	res = lialg_recognize_stroke(rec, stroke);

	delete_examples(stroke);
	return(res);
}

/*
 * file_path-Construct pathname, check for proper extension.
 */

static int
  file_path(char* dir,char* filename,char* pathname)
{
	char* dot;
	
	/*Check for proper extension on file name.*/
	
	dot = strrchr(filename,'.');
	
	if( dot == nil ) {
		return(-1);
	}

	/*Determine whether a gesture or character classifier.*/

	if( strcmp(dot,LI_CLASSIFIER_EXTENSION) != 0 ) {
		return(-1);
	}

	/*Concatenate directory and filename into pathname.*/
	
	strcpy(pathname,dir);
	strcat(pathname,"/");
	strcat(pathname,filename);
	
	return(0);
}

/*read_classifier_points-Read points so classifier can be extended.*/

static int 
read_classifier_points(FILE* fd,int nclss,point_list** ex,char** cnames)
{
	int i,j,k;
	char buf[BUFSIZ];
	int nex = 0;
	char* names[MAXSCLASSES];
	point_list* examples[MAXSCLASSES];
	pen_point* pts;
	int npts;

	/*Initialize*/

		for( i = 0; i < MAXSCLASSES; i++ ) {
				names[i] = nil;
				examples[i] = nil;
		}

		/*Go thru classes.*/

		for( k = 0; k < nclss; k++ ) {

				/*Read class name and number of examples.*/
		
				if( fscanf(fd,"%d %s",&nex,buf) != 2 )
					goto unallocate;
		
				/*Save class name*/
		
				names[k] = strdup(buf);
		
				/*Read examples.*/
				
				for( i = 0; i < nex; i++ ) {
					
					/*Read number of points.*/
					
					if( fscanf(fd,"%d",&npts) != 1 )
								goto unallocate; /*Boy would I like exceptions!*/
					
					/*Allocate array for points.*/
					
					if( (pts = mallocz(npts*sizeof(pen_point), 1)) == nil )
								goto unallocate;
					
					/*Read in points.*/
					
					for( j = 0; j < npts; j++ ) {
								int x,y;
								if( fscanf(fd,"%d %d",&x,&y) != 2 ) {
									delete_pen_point_array(pts);
									goto unallocate;
								}
								pts[j].Point = Pt(x, y);
					}
					
					/*Add example*/
					
					if( (examples[k] = add_example(examples[k],npts,pts)) == nil ) {
								delete_pen_point_array(pts);
								goto unallocate;
					}
					
					delete_pen_point_array(pts);
			
				}
		}

/* ari -- end of list of classes */
/* fprint(2, "]\n"); */

		/*Transfer to recognizer.*/

		bcopy(examples,ex,sizeof(examples));
		bcopy(names,cnames,sizeof(names));

		return(0);

		/*Error. Deallocate memory and return.*/

unallocate:
		for( ; k >= 0; k-- ) {
				delete_examples(examples[k]);
				free(names[k]);
		}

		return(-1);
}

/*read_classifier-Read a classifier file.*/

static int read_classifier(FILE* fd,rClassifier* rc)
{
	
	li_err_msg = nil;

	/*Read in classifier file.*/

	if(fscanf(fd, "%d", &rc->nclasses) != 1)
		return -1;

	/*Read in the example points, so classifier can be extended.*/

	if( read_classifier_points(fd,rc->nclasses,rc->ex,rc->cnames) != 0 )
		return -1;

	return(0);
}

/*
 * Extension Functions
*/

/* getClasses and clearState are by Ari */

static int
recognizer_getClasses (recognizer r, char ***list, int *nc)
{
	int i, nclasses;
	li_recognizer* rec;
	char **ret;

		rec = (li_recognizer*)r->recognizer_specific;

		/*Check for LI recognizer.*/

		if( !CHECK_LI_MAGIC(rec) ) {
				li_err_msg = "Not a LI recognizer";
				return(-1);
	}
	
	*nc = nclasses = rec->li_rc.nclasses;
	ret = malloc(nclasses*sizeof(char*));

	for (i = 0; i < nclasses; i++)
				ret[i] = rec->li_rc.cnames[i];   /* only the 1st char of the cname */
	*list = ret;
		return 0;
}

static int
recognizer_clearState (recognizer)
{
  /*This operation isn't supported by the LI recognizer.*/

  li_err_msg = "Clearing state is not supported by the LI recognizer";

  return(-1);
}

static bool isa_li(recognizer r)
{ return(CHECK_LI_MAGIC(r)); }

static int
recognizer_train(recognizer, rc*, uint, Stroke*, rec_element*, bool)
{
  /*This operation isn't supported by the LI recognizer.*/

  li_err_msg = "Training is not supported by the LI recognizer";

  return(-1);
}

int
li_recognizer_get_example (recognizer r,
						   int		class, 
						   int		instance,
						   char		**name, 
						   pen_point	**points,
						   int		*npts)
{
	li_recognizer   *rec = (li_recognizer*)r->recognizer_specific;
	int nclasses = rec->li_rc.nclasses;
	point_list	    *pl;
	
		if( !CHECK_LI_MAGIC(rec) ) {
				li_err_msg = "Not a LI recognizer";
				return(-1);
		}
		if (class > nclasses)
				return -1;
		pl = rec->li_rc.canonex[class];
		while (instance && pl)
		{
				pl = pl->next;
				instance--;
		}
		if (!pl)
				return -1;
		*name = rec->li_rc.cnames[class];
		*points = pl->pts;
		*npts = pl->npts;
		return pl->npts; /* I hope [sjm] */
}

/*
 * API Functions
 */


/*li_recognizer_load-Load a classifier file.*/

static int li_recognizer_load(recognizer r, char* dir, char* filename)
{
		FILE *fd;
		char* pathname;
		li_recognizer* rec;
		rClassifier* rc;
		
		rec = (li_recognizer*)r->recognizer_specific;

		/*Make sure recognizer's OK*/

		if( !CHECK_LI_MAGIC(rec) ) {
				li_err_msg = "Not a LI recognizer";
				return(-1);
		}

		rc = &(rec->li_rc);

		/*Check parameters.*/

		if( filename == nil ) {
				li_err_msg = "Invalid parameters";
				return(-1);
		}

		/*We let the directory be null.*/

		if( dir == nil || (int)strlen(dir) <= 0 ) {
				dir = ".";
		}

		if(0)fprint(2, "dir = %s, filename = %s\n",
				dir, filename);

		/*Make full pathname and check filename*/

		pathname = malloc((strlen(dir) + strlen(filename) + 2)*sizeof(char));
		if( file_path(dir, filename, pathname) == -1 ) {
				free(pathname);
				li_err_msg = "Not a LI recognizer classifier file";
				return -1;
		}

		/* Try to short-circuit the full classifier-file processing. */
		rc->file_name = pathname;
		if (lialg_read_classifier_digest(rc) == 0)
				return(0);
		rc->file_name = nil;

		/*Open the file*/

		if( (fd = fopen(pathname,"r")) == nil ) {
				li_err_msg = "Can't open classifier file";
				if(0)fprint(2, "Can't open %s.\n", pathname);
				free(pathname);
				return(-1);
		}

		/*If rClassifier is OK, then delete it first.*/

		if( rc->file_name != nil ) {
				free_rClassifier(rc);
		}

		/*Read classifier.*/
		
		if( read_classifier(fd,rc) < 0 ) {
				free(pathname);
				return(-1);
		}

		/*Close file.*/

		fclose(fd);

		/*Add classifier name.*/

		rc->file_name = pathname;

		/* Canonicalize examples. */
		if (lialg_canonicalize_examples(rc) != 0) {
				free(pathname);
				rc->file_name = nil;
				return -1;
		}

		return(0);
}

/*li_recognizer_save-Save a classifier file.*/

static int li_recognizer_save(recognizer, char*, char*)
{ 
		/*This operation isn't supported by the LI recognizer.*/

		li_err_msg = "Saving is not supported by the LI recognizer";
		return -1;
}

static wordset
li_recognizer_load_dictionary(recognizer, char*, char*)
{
		/*This operation isn't supported by the LI recognizer.*/

		li_err_msg = "Dictionaries are not supported by the LI recognizer";
		return nil;
}

static int
li_recognizer_save_dictionary(recognizer, char*, char*, wordset)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Dictionaries are not supported by the LI recognizer";
		return -1;
}

static int
li_recognizer_free_dictionary(recognizer, wordset)
{
  /*This operation isn't supported by the LI recognizer.*/

  li_err_msg = "Dictionaries are not supported by the LI recognizer";

  return -1;

}

static int
li_recognizer_add_to_dictionary(recognizer, letterset*, wordset)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Dictionaries are not supported by the LI recognizer";
		return -1;
}

static int
li_recognizer_delete_from_dictionary(recognizer, letterset*, wordset)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Dictionaries are not supported by the LI recognizer";
		return -1;
}

static char*
li_recognizer_error(recognizer rec)
{
	char* ret = li_err_msg;

	/*Check for LI recognizer.*/

	if( !CHECK_LI_MAGIC(rec->recognizer_specific) ) {
				li_err_msg = "Not a LI recognizer";
				return nil;
	}
	li_err_msg = nil;
	return ret;
}

static int 
li_recognizer_clear(recognizer r, bool)
{
		li_recognizer* rec; 

		rec = (li_recognizer*)r->recognizer_specific;
		/*Check for LI recognizer.*/
		if( !CHECK_LI_MAGIC(rec) ) {
				li_err_msg = "Not a LI recognizer";
				return 0;
		}
		return 0;
}

static int 
li_recognizer_set_context(recognizer, rc*)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Contexts are not supported by the LI recognizer";
		return -1;
}

static rc*
li_recognizer_get_context(recognizer)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Contexts are not supported by the LI recognizer";
		return nil;
}

static int 
li_recognizer_get_buffer(recognizer, uint*, Stroke**)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Buffer get/set are not supported by the LI recognizer";
		return -1;
}

static int 
li_recognizer_set_buffer(recognizer, uint, Stroke*)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Buffer get/set are not supported by the LI recognizer";
		return -1;
}

static int
li_recognizer_translate(recognizer r, uint ncs, Stroke* tps, bool, int* nret, rec_alternative** ret)
{
	char* clss;
	li_recognizer* rec; 
	int conf;
	rClassifier* rc;
	  
	rec = (li_recognizer*)r->recognizer_specific;

	*nret = 0;
	*ret = nil;

	/*Check for LI recognizer.*/

	if( !CHECK_LI_MAGIC(rec) ) {
				li_err_msg = "Not a LI recognizer";
				return(-1);
	}

		rc = &(rec->li_rc);

	/*Check for valid parameters.*/
	if (ncs < 1) {
				li_err_msg = "Invalid parameters: ncs";
				return(-1);
	}
	if( tps == nil) {
				li_err_msg = "Invalid parameters: tps";
				return(-1);
	}
	if( nret == nil) {
				li_err_msg = "Invalid parameters: nret";
				return(-1);
	}
	if( ret == nil) {
				li_err_msg = "Invalid parameters: ret";
				return(-1);
	}

	/*
	 * Go through the stroke array and recognize. Since this is a single
	 *   stroke recognizer, each stroke is treated as a separate
	 *   character or gesture. We allow only characters or gestures
	 *   to be recognized at one time, since otherwise, handling
	 *   the display of segmentation would be difficult. 
	*/
	clss = recognize_internal(rc,tps,&conf);
	if (clss == nil) {
				*nret = 1;
				return(0);
	}

	/*Return values.*/
	*nret = 1;
	return(*clss);
}


static rec_fn*
li_recognizer_get_extension_functions(recognizer rec)
{
	rec_fn* ret;

	/*Check for LI recognizer.*/

	if( !CHECK_LI_MAGIC(rec->recognizer_specific) ) {
				li_err_msg = "Not a LI recognizer";
				return(nil);
	}

	ret = make_rec_fn_array(LI_NUM_EX_FNS);

/* ari -- clearState & getClasses are mine */
	ret[LI_GET_CLASSES] =	(rec_fn)recognizer_getClasses;
	ret[LI_CLEAR] =			(rec_fn)recognizer_clearState;
	ret[LI_ISA_LI] =		(rec_fn)isa_li;
	ret[LI_TRAIN] =			(rec_fn)recognizer_train;

	return(ret);
}

static char**
li_recognizer_get_gesture_names(recognizer)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Gestures are not supported by the LI recognizer";
		return nil;
}

static xgesture
li_recognizer_set_gesture_action(recognizer, char*, xgesture, void*)
{
		/*This operation isn't supported by the LI recognizer.*/
		li_err_msg = "Gestures are not supported by the LI recognizer";
		return nil;
}

/*
 * Exported Functions
*/

/*RECOGNIZER_INITIALIZE-Initialize the recognizer.*/

/* note from ari:  this expands via pre-processor to
 *
 * recognizer __recognizer_internal_initialize(rec_info* ri)
 */

RECOGNIZER_INITIALIZE(ri)
{
	recognizer r;
	li_recognizer* rec;
	int i;

	/*Check that locale matches.*/

	if( strcmp(ri->ri_locale,LI_SUPPORTED_LOCALE) != 0 ) {
		li_err_msg = "Not a supported locale";
/* fprint(2, "Locale error.\n");*/
		return(nil);
	}

	/*
	 * Check that character sets match. Note that this is only approximate,
	 * since the classifier file will have more information.
	*/

	if( ri->ri_subset != nil ) {
	  for(i = 0; ri->ri_subset[i] != nil; i++ ) {

		if( strcmp(ri->ri_subset[i],UPPERCASE) != 0 &&
			strcmp(ri->ri_subset[i],LOWERCASE) != 0  &&
			strcmp(ri->ri_subset[i],DIGITS) != 0 &&
			strcmp(ri->ri_subset[i],GESTURE) != 0 ) {
		  li_err_msg = "Not a supported character set";
/* fprint(2, "charset error.\n"); */

		  return(nil);
		}
	  }
	}
			 
/* ari */
	r = make_recognizer(ri);
/* fprint(2, "past make_recognizer.\n"); */

	if( r == nil ) {
		li_err_msg = "Can't allocate storage";

		return(nil);
	}

	/*Make a LI recognizer structure.*/


	/* rec = (li_recognizer*)safe_malloc(sizeof(li_recognizer))) == nil ); */

	rec = malloc(sizeof(li_recognizer));

	r->recognizer_specific = rec;

	rec->li_rc.file_name = nil;
	rec->li_rc.nclasses = 0;

	/*Initialize the recognizer struct.*/

	r->recognizer_load_state = li_recognizer_load;
	r->recognizer_save_state = li_recognizer_save;
	r->recognizer_load_dictionary = li_recognizer_load_dictionary;
	r->recognizer_save_dictionary = li_recognizer_save_dictionary;
	r->recognizer_free_dictionary = li_recognizer_free_dictionary;
	r->recognizer_add_to_dictionary = li_recognizer_add_to_dictionary;
	r->recognizer_delete_from_dictionary = li_recognizer_delete_from_dictionary;
	r->recognizer_error = li_recognizer_error;
	r->recognizer_translate = li_recognizer_translate;
	r->recognizer_get_context = li_recognizer_get_context;
	r->recognizer_set_context = li_recognizer_set_context;
	r->recognizer_get_buffer = li_recognizer_get_buffer;
	r->recognizer_set_buffer = li_recognizer_set_buffer;
	r->recognizer_clear = li_recognizer_clear;
	r->recognizer_get_extension_functions = 
	  li_recognizer_get_extension_functions;
	r->recognizer_get_gesture_names = li_recognizer_get_gesture_names;
	r->recognizer_set_gesture_action = 
	  li_recognizer_set_gesture_action;

	/*Initialize LI Magic Number.*/

	rec->li_magic = LI_MAGIC;

	/*Initialize rClassifier.*/

	rec->li_rc.file_name = nil;

	for( i = 0; i < MAXSCLASSES; i++ ) {
		rec->li_rc.ex[i] = nil;
		rec->li_rc.cnames[i] = nil;
	}

	lialg_initialize(&rec->li_rc);

	/*Get rid of error message. Not needed here.*/
	li_err_msg = nil;

	return(r);
}

/*free_rClassifier-Free the rClassifier.*/

static void
free_rClassifier(rClassifier* rc)
{
	int i;

	if( rc->file_name != nil) {
		free(rc->file_name);
	}

	for( i = 0; rc->ex[i] != nil; i++) {
		delete_examples(rc->ex[i]);
		free(rc->cnames[i]);
	}

}

/*RECOGNIZER_FINALIZE-Deallocate the recognizer, finalize.*/

RECOGNIZER_FINALIZE(r)
{
		li_recognizer* rec = (li_recognizer*)r->recognizer_specific;

		/*Make sure this is a li_recognizer first*/
		if( !CHECK_LI_MAGIC(rec) ) {
				li_err_msg = "Not a LI recognizer";
				return(-1);
	}

		/*Deallocate rClassifier resources.*/

		free_rClassifier(&(rec->li_rc));

		/*Deallocate the li_recognizer struct.*/
		free(rec);

		/*Deallocate the recognizer*/
		delete_recognizer(r);

		return(0);
}


/* **************************************************

  Implementation of the Li/Yeung recognition algorithm

************************************************** */

#define	WORST_SCORE	0x7fffffff

/* Dynamic programming parameters */
#define	DP_BAND		3
#define	MIN_SIM		0
#define	MAX_DIST	0x7fffffff
#define	SIM_THLD	60	/* x 100 */
#define	DIST_THLD	3200	/* x 100 */

/* Low-pass filter parameters -- empirically derived */
#define	LP_FILTER_WIDTH	6
#define	LP_FILTER_ITERS	8
#define	LP_FILTER_THLD	250	/* x 100 */
#define	LP_FILTER_MIN	5

/* Pseudo-extrema parameters -- empirically derived */
#define	PE_AL_THLD	1500	/* x 100 */
#define	PE_ATCR_THLD	135	/* x 100 */

/* Contour-angle derivation parameters */
#define	T_ONE		1
#define	T_TWO		20

/* Pre-processing and canonicalization parameters */
#define	CANONICAL_X	108
#define	CANONICAL_Y	128
#define	DIST_SQ_THRESHOLD   (3*3)	/* copied from fv.h */
#define	NCANONICAL	50

/* Tap-handling parameters */
#define	TAP_CHAR	"."
#define	TAP_TIME_THLD	150	    /* msec */
#define	TAP_DIST_THLD	75	    /* dx * dx + dy * dy */
#define	TAP_PATHLEN	1000	    /* x 100 */


/* region types */
#define	RGN_CONVEX  0
#define	RGN_CONCAVE 1
#define	RGN_PLAIN   2
#define	RGN_PSEUDO  3


typedef struct RegionList {
	int start;
	int end;
	int type;
	struct RegionList *next;
} region_list;


/* direction-code table; indexed by dx, dy */
static int lialg_dctbl[3][3] = {{1, 0, 7}, {2, 0x7FFFFFFF, 6}, {3, 4, 5}};

/* low-pass filter weights */
static int lialg_lpfwts[2 * LP_FILTER_WIDTH + 1];
static int lialg_lpfconst = -1;


static int lialg_preprocess_stroke(point_list *);
static point_list *lialg_compute_dominant_points(point_list *);
static point_list *lialg_interpolate_points(point_list *);
static void lialg_bresline(pen_point *, pen_point *, point_list *, int *);
static void lialg_compute_chain_code(point_list *);
static void lialg_compute_unit_chain_code(point_list *);
static region_list *lialg_compute_regions(point_list *);
static point_list *lialg_compute_dompts(point_list *, region_list *);
static int *lialg_compute_contour_angle_set(point_list *, region_list *);
static void lialg_score_stroke(point_list *, point_list *, int *, int *);
static int lialg_compute_similarity(point_list *, point_list *);
static int lialg_compute_distance(point_list *, point_list *);

static int lialg_read_classifier_digest(rClassifier *);

static int lialg_canonicalize_examples(rClassifier *);
static int lialg_canonicalize_example_stroke(point_list *);
static int lialg_compute_equipoints(point_list *);

static int lialg_compute_pathlen(point_list *);
static int lialg_compute_pathlen_subset(point_list *, int, int);
static int lialg_filter_points(point_list *);
static int lialg_translate_points(point_list *, int, int, int, int);
static void lialg_get_bounding_box(point_list *, int *, int *, int *, int *);
static void lialg_compute_lpf_parameters();
static int isqrt(int);
static int likeatan(int, int);
static int quadr(int);


/*************************************************************

  Core routines for the Li/Yeung recognition algorithm

 *************************************************************/

static void lialg_initialize(rClassifier *rec) {
	int i;

	/* Initialize the dompts arrays. */
	for (i = 0; i < MAXSCLASSES; i++) {
		rec->dompts[i] = nil;
	}
}


/*
 *  Main recognition routine -- called by HRE API.
 */
static char *lialg_recognize_stroke(rClassifier *rec, point_list *stroke) {
		int i;
		char *name = nil;
		point_list *input_dompts = nil;
		char *best_name = nil;
		int best_score = WORST_SCORE;
		char *curr_name;
		point_list *curr_dompts;

		/*    (void)gettimeofday(&stv, nil);*/

		if (stroke->npts < 1) goto done;

		/* Check for tap. */

		/* First thing is to filter out ``close points.'' */
		if (lialg_filter_points(stroke) != 0) return(nil);

		/* Unfortunately, we don't have the actual time that each point */
		/* was recorded (i.e., dt is invalid).  Hence, we have to use a */
		/* heuristic based on total distance and the number of points. */
		if (stroke->npts == 1 || lialg_compute_pathlen(stroke) < TAP_PATHLEN)
			return(TAP_CHAR);

		/* Pre-process input stroke. */
		if (lialg_preprocess_stroke(stroke) != 0) goto done;

		/* Compute its dominant points. */
		input_dompts = lialg_compute_dominant_points(stroke);
		if (input_dompts == nil) goto done;

		/* Score input stroke against every class in classifier. */
		for (i = 0, curr_name = rec->cnames[i], curr_dompts = rec->dompts[i];
				i < MAXSCLASSES && curr_name != nil && curr_dompts != nil;
				i++, curr_name = rec->cnames[i], curr_dompts = rec->dompts[i]) {
				int sim;
				int dist;
				int curr_score;

				lialg_score_stroke(input_dompts, curr_dompts, &sim, &dist);
				curr_score = dist;

				if (lidebug && curr_score < DIST_THLD)
				fprint(2, "(%s, %d, %d) ", curr_name, sim, dist);

				/* Is it the best so far? */
				if (curr_score < best_score && curr_score <= DIST_THLD) {
				best_score = curr_score;
				best_name = curr_name;
				}
		}

		if (lidebug)
				fprint(2, "\n");

		/* No errors. */
		name = best_name;

done:
		delete_examples(input_dompts);
		return(name);
}


static int lialg_preprocess_stroke(point_list *points) {
	int minx, miny, maxx, maxy, xrange, yrange, scale, xoff, yoff;

	/* Filter out points that are too close. */
	/* We did this earlier, when we checked for a tap. */
/*
	if (lialg_filter_points(points) != 0) return(-1);
*/

/*    assert(points->npts > 0);*/

	/* Scale up to avoid conversion errors. */
	lialg_get_bounding_box(points, &minx, &miny, &maxx, &maxy);
	xrange = maxx - minx;
	yrange = maxy - miny;
	scale = ( ((100 * xrange + CANONICAL_X / 2) / CANONICAL_X) > 
			  ((100 * yrange + CANONICAL_Y / 2) / CANONICAL_Y))
	  ? (100 * CANONICAL_X + xrange / 2) / xrange
	  : (100 * CANONICAL_Y + yrange / 2) / yrange;
	if (lialg_translate_points(points, minx, miny, scale, scale) != 0)
		return(-1);

	/* Center the stroke. */
	lialg_get_bounding_box(points, &minx, &miny, &maxx, &maxy);
	xrange = maxx - minx;
	yrange = maxy - miny;
	xoff = -((CANONICAL_X - xrange + 1) / 2);
	yoff = -((CANONICAL_Y - yrange + 1) / 2);
	if (lialg_translate_points(points, xoff, yoff, 100, 100) != 0) return(-1);

	/* Store the x and y ranges in the point list. */
	xrange = maxx - minx;
	yrange = maxy - miny;
	points->xrange = xrange;
	points->yrange = yrange;

	if (lidebug) {
		int i;
		fprint(2, "After pre-processing:   %d %d %d %d\n",
				minx, miny, maxx, maxy);
		for (i = 0; i < points->npts; i++)
			fprint(2, "      (%P)\n", points->pts[i].Point);
		fflush(stderr);
	}

	return(0);
}


static point_list *lialg_compute_dominant_points(point_list *points) {
	point_list *ipts;
	region_list *regions;
	point_list *dpts;

	/* Interpolate points. */
	ipts = lialg_interpolate_points(points);
	if (ipts == nil) return(nil);
	if (lidebug) {
				int j;
				fprint(2, "After interpolation:  %d ipts\n", ipts->npts);
				for (j = 0; j < ipts->npts; j++) {
					fprint(2, "  (%P), %lud\n", ipts->pts[j].Point, ipts->pts[j].chaincode);
				}
				fflush(stderr);
	}

	/* Compute regions. */
	regions = lialg_compute_regions(ipts);
/*    assert(regions != nil);*/

	/* Compute dominant points. */
	dpts = lialg_compute_dompts(ipts, regions);
	if (lidebug) {
				int j;
				fprint(2, "Dominant points:  ");
				for (j = 0; j < dpts->npts; j++) {
					fprint(2, "%P (%lud)  ", dpts->pts[j].Point, dpts->pts[j].chaincode);
				}
				fprint(2, "\n");
				fflush(stderr);
	}

	/* Delete region data structure. */
	{
				region_list *curr, *next;
				for (curr = regions; curr != nil; ) {
					next = curr->next;
					free(curr);
					curr = next;
				}
	}
	delete_examples(ipts);
	return(dpts);
}

/* Input points are assumed to be integer-valued! */
static point_list *lialg_interpolate_points(point_list *points) {
	int i, j;
	int maxpts;
	point_list *newpts;

	/* Compute an upper-bound on the number of interpolated points. */
	maxpts = 0;
	for (i = 0; i < (points->npts - 1); i++) {
				pen_point *pta = &(points->pts[i]);
				pen_point *ptb = &(points->pts[i+1]);
				maxpts += abs(pta->x - ptb->x) + abs(pta->y - ptb->y);
	}

	/* Allocate an array of the requisite size. */
	maxpts += points->npts;
	/* newpts = (point_list *)safe_malloc(sizeof(point_list)); */
	newpts = malloc(sizeof(point_list));
	newpts->pts = mallocz(maxpts*sizeof(pen_point), 1);
	if (newpts->pts == nil) {
				free(newpts);
				return(nil);
	}
	newpts->npts = maxpts;
	newpts->next = nil;

	/* Interpolate each of the segments. */
	j = 0;
	for (i = 0; i < (points->npts - 1); i++) {
				pen_point *startpt = &(points->pts[i]);
				pen_point *endpt = &(points->pts[i+1]);

				lialg_bresline(startpt, endpt, newpts, &j);

				j--;	/* end point gets recorded as start point of next segment! */
	}

	/* Add-in last point. */
	newpts->pts[j++] = points->pts[points->npts - 1];
	newpts->npts = j;

	/* Compute the chain code for P (the list of points). */
	lialg_compute_unit_chain_code(newpts);

	return(newpts);
}


/* This implementation is due to Kenny Hoff. */
static void lialg_bresline(pen_point *startpt, pen_point *endpt,
							point_list *newpts, int *j) {
	int Ax, Ay, Bx, By, dX, dY, Xincr, Yincr;

	Ax = startpt->x;
	Ay = startpt->y;
	Bx = endpt->x;
	By = endpt->y;

	/* INITIALIZE THE COMPONENTS OF THE ALGORITHM THAT ARE NOT AFFECTED */
	/* BY THE SLOPE OR DIRECTION OF THE	LINE */
	dX = abs(Bx-Ax);	/* store the change in X and Y of the line endpoints */
	dY = abs(By-Ay);

	/* DETERMINE "DIRECTIONS" TO INCREMENT X AND Y (REGARDLESS OF DECISION) */
	if (Ax > Bx) { Xincr=-1; } else { Xincr=1; }    /* which direction in X? */
	if (Ay > By) { Yincr=-1; } else { Yincr=1; }    /* which direction in Y? */

	/* DETERMINE INDEPENDENT VARIABLE (ONE THAT ALWAYS INCREMENTS BY 1 (OR -1) ) */
	/* AND INITIATE APPROPRIATE LINE DRAWING ROUTINE (BASED ON FIRST OCTANT */
	/* ALWAYS). THE X AND Y'S MAY BE FLIPPED IF Y IS THE INDEPENDENT VARIABLE. */
	if (dX >= dY) {	    /* if X is the independent variable */
				int dPr	= dY<<1;	    /* amount to increment decision if right is chosen (always) */
				int dPru = dPr - (dX<<1);   /* amount to increment decision if up is chosen */
				int P =	dPr - dX;	    /* decision variable start value */
		
				/* process each point in the line one at a time (just use dX) */
				for (; dX>=0; dX--) {
					newpts->pts[*j].x = Ax;
					newpts->pts[*j].y = Ay;
					(*j)++;
		
					if (P > 0) {	/* is the pixel	going right AND	up? */
								Ax+=Xincr;	/* increment independent variable */
								Ay+=Yincr;	/* increment dependent variable */
								P+=dPru;	/* increment decision (for up) */
					} else {		/* is the pixel just going right? */
								Ax+=Xincr;	/* increment independent variable */
								P+=dPr;		/* increment decision (for right) */
					}
				}
	} else {		    /* if Y is the independent variable */
				int dPr	= dX<<1;	    /* amount to increment decision if right is chosen (always) */
				int dPru = dPr - (dY<<1);   /* amount to increment decision if up is chosen */
				int P  = dPr - dY;	    /* decision variable start value */
		
				/* process each point in the line one at a time (just use dY) */
				for (; dY>=0; dY--) {
					newpts->pts[*j].x = Ax;
					newpts->pts[*j].y = Ay;
					(*j)++;
		
					if (P > 0) {	/* is the pixel going up AND right? */
								Ax+=Xincr;	/* increment dependent variable */
								Ay+=Yincr;	/* increment independent variable */
								P+=dPru;	/* increment decision (for up) */
					} else {		/* is the pixel just going up? */
								Ay+=Yincr;	/* increment independent variable */
								P+=dPr;		/* increment decision (for right) */
					}
				}
	}
}

static void lialg_compute_chain_code(point_list *pts) {
	int i;

	for (i = 0; i < (pts->npts - 1); i++) {
				pen_point *startpt = &(pts->pts[i]);
				pen_point *endpt = &(pts->pts[i+1]);
				int dx = endpt->x - startpt->x;
				int dy = endpt->y - startpt->y;
				int tmp = quadr(likeatan(dy, dx));
				int dircode = (12 - tmp) % 8;
		
				startpt->chaincode = dircode;
	}
}


static void lialg_compute_unit_chain_code(point_list *pts) {
	int i;

	for (i = 0; i < (pts->npts - 1); i++) {
				pen_point *startpt = &(pts->pts[i]);
				pen_point *endpt = &(pts->pts[i+1]);
				int dx = endpt->x - startpt->x;
				int dy = endpt->y - startpt->y;
				int dircode = lialg_dctbl[dx+1][dy+1];

				startpt->chaincode = dircode;
	}
}


static region_list *lialg_compute_regions(point_list *pts) {
		region_list *regions;
		region_list *curr_reg;
		int *R[2 + LP_FILTER_ITERS];
		int *junk;
		int *curr, *next;
		int i, j;

		/* Initialize low-pass filter parameters if necessary. */
		if (lialg_lpfconst == -1)
				lialg_compute_lpf_parameters();

	/* Allocate a 2 x pts->npts array for use in computing the (filtered) Angle set, A_n. */
	junk = malloc((2 + LP_FILTER_ITERS) * pts->npts*sizeof(int));
	for (i = 0; i < (2 + LP_FILTER_ITERS); i++)
		R[i] = junk + (i * pts->npts);
	curr = R[0];

	/* Compute the Angle set, A, in the first element of array R. */
	/* Values in R are in degrees, x 100. */
	curr[0] = 18000;				/* a_0 */
	for (i = 1; i < (pts->npts - 1); i++) {
				int d_i = pts->pts[i].chaincode;
				int d_iminusone = pts->pts[i-1].chaincode;
				int a_i;
		
				if (d_iminusone < d_i)
					d_iminusone += 8;
		
				a_i = (d_iminusone - d_i) % 8;
		
				/* convert to degrees, x 100 */
				curr[i] = ((12 - a_i) % 8) * 45 * 100;
	}
	curr[pts->npts - 1]	= 18000;		/* a_L-1 */

	/* Perform a number of filtering iterations. */
	next = R[1];
	for (j = 0; j < LP_FILTER_ITERS; j++, curr = R[j], next = R[j+1]) {
				for (i = 0; i < pts->npts; i++) {
					int k;
		
					next[i] = 0;
		
					for (k = i - LP_FILTER_WIDTH; k <= i + LP_FILTER_WIDTH; k++) {
						int oldval = (k < 0 || k >= pts->npts) ? 18000 : curr[k];
						next[i]	+= oldval * lialg_lpfwts[k - (i	- LP_FILTER_WIDTH)];	/* overflow? */
					}
		
					next[i] /= lialg_lpfconst;
				}
	}

	/* Do final thresholding around PI. */
	/* curr and next are set-up correctly at end of previous loop! */
	for (i = 0; i < pts->npts; i++)
				next[i] = (abs(curr[i] - 18000) < LP_FILTER_THLD) ? 18000 : curr[i];
	curr = next;

	/* Debugging. */
	if (lidebug > 1) {
				for (i = 0; i < pts->npts; i++) {
					fprint(2, "%3d:  (%P)  %lud  ",
								i, pts->pts[i].Point, pts->pts[i].chaincode);
					for (j = 0; j < 2 + LP_FILTER_ITERS; j++)
						fprint(2, "%d  ", R[j][i]);
					fprint(2, "\n");
				}
	}

	/* Do the region segmentation. */
	{
				int start, end;
				int currtype;
		
#define	RGN_TYPE(val) (((val)==18000)?RGN_PLAIN:((val)<18000?RGN_CONCAVE:RGN_CONVEX))
		
				start = 0;
				currtype = RGN_TYPE(curr[0]);
				regions = malloc(sizeof(region_list));
				curr_reg = regions;
				curr_reg->start = start;
				curr_reg->end = 0;
				curr_reg->type = currtype;
				curr_reg->next = nil;
				for (i = 1; i < pts->npts; i++) {
					int nexttype = RGN_TYPE(curr[i]);
		
					if (nexttype != currtype) {
								region_list *next_reg;
				
								end = i - 1;
								curr_reg->end = end;
								if (lidebug > 1)
									fprint(2, "  (%d, %d), %d\n", start, end, currtype);
				
								start = i;
								currtype = nexttype;
								next_reg = malloc(sizeof(region_list));
								next_reg->start = start;
								next_reg->end = 0;
								next_reg->type = nexttype;
								next_reg->next = nil;
				
								curr_reg->next = next_reg;
								curr_reg = next_reg;
					}
				}
				end = i - 1;
				curr_reg->end = end;
				if (lidebug > 1)
					fprint(2, "  (%d, %d), %d\n", start, end, currtype);

				/* Filter out convex/concave regions that are too short. */
				for (curr_reg = regions; curr_reg; curr_reg = curr_reg->next)
					if (curr_reg->type == RGN_PLAIN) {
								region_list *next_reg;
				
								for (next_reg = curr_reg->next;
									 next_reg != nil &&
									   (next_reg->end - next_reg->start) < LP_FILTER_MIN;
									 next_reg = curr_reg->next) {
									/* next_reg must not be plain, and it must be followed by a plain */
									/* assert(next_reg->type != RGN_PLAIN); */
									/* assert(next_reg->next != nil && (next_reg->next)->type == RGN_PLAIN); */
				
									curr_reg->next = (next_reg->next)->next;
									curr_reg->end = (next_reg->next)->end;
				
									free(next_reg->next);
									free(next_reg);
								}
					}
		
				/* Add-in pseudo-extremes. */
				{
					region_list *tmp, *prev_reg;
		
					tmp = regions;
					regions = nil;
					prev_reg = nil;
					for (curr_reg = tmp; curr_reg; curr_reg = curr_reg->next) {
						if (curr_reg->type == RGN_PLAIN) {
							int arclen = lialg_compute_pathlen_subset(pts,
																		curr_reg->start,
																		curr_reg->end);
							int dx = pts->pts[curr_reg->end].x -
							  pts->pts[curr_reg->start].x;
							int dy = pts->pts[curr_reg->end].y -
							  pts->pts[curr_reg->start].y;
							int chordlen = isqrt(10000 * (dx * dx + dy * dy));
							int atcr = (chordlen == 0) ? 0 : (100 * arclen + chordlen / 2) / chordlen;

							if (lidebug)
								fprint(2, "%d, %d, %d\n", arclen, chordlen, atcr);
		
							/* Split region if necessary. */
							if (arclen >= PE_AL_THLD && atcr >= PE_ATCR_THLD) {
								int mid = curr_reg->start + (curr_reg->end - curr_reg->start) / 2;
								int end = curr_reg->end;
								region_list *saved_next = curr_reg->next;
		
								curr_reg->end = mid - 1;
								if (prev_reg == nil)
									regions = curr_reg;
								else
									prev_reg->next = curr_reg;
								prev_reg = curr_reg;
		
								/* curr_reg = (region_list *)safe_malloc(sizeof(region_list));*/
								curr_reg = malloc(sizeof(region_list));
								curr_reg->start = mid;
								curr_reg->end = mid;
								curr_reg->type = RGN_PSEUDO;
								curr_reg->next = nil;
								prev_reg->next = curr_reg;
								prev_reg = curr_reg;
		
								/* curr_reg = (region_list *)malloc(sizeof(region_list)); */
								curr_reg = malloc(sizeof(region_list));
								curr_reg->start = mid + 1;
								curr_reg->end = end;
								curr_reg->type = RGN_PLAIN;
								curr_reg->next = nil;
								prev_reg->next = curr_reg;
								prev_reg = curr_reg;
		
								curr_reg->next = saved_next;
								continue;
							}
						}
		
						if (prev_reg == nil)
							regions = curr_reg;
						else
							prev_reg->next = curr_reg;
						prev_reg = curr_reg;
					}
				}
		}

	free(junk);
	return(regions);
}


static point_list *lialg_compute_dompts(point_list *pts, region_list *regions) {
	point_list *dpts;
	int ndpts;
	int *cas;
	int nonplain;
	region_list *r;
		region_list *curr;
		int dp;
		int previx;
		int currix;

	/* Compute contour angle set. */
	cas = lialg_compute_contour_angle_set(pts, regions);

	/* Dominant points include:  start_pt, end_pt, extrema_of_non_plain_regions, midpts of the preceding. */
	nonplain = 0;
	for (r = regions; r != nil; r = r->next)
				if (r->type != RGN_PLAIN)
						nonplain++;
	ndpts = 2 * (2 + nonplain) - 1;
	/* dpts = (point_list *)safe_malloc(sizeof(point_list)); */
	dpts = malloc(sizeof(point_list));
	dpts->pts = mallocz(ndpts*sizeof(pen_point), 1);
	if (dpts->pts == nil) {
				free(dpts);
				return(nil);
	}
	dpts->npts = ndpts;
	dpts->next = nil;

	/* Pick out dominant points. */

		/* Record start point. */
		dp = 0;
		previx = 0;
		dpts->pts[dp++] = pts->pts[previx];

		for (curr = regions; curr != nil; curr = curr->next)
			if (curr->type != RGN_PLAIN) {
						int max_v = 0;
						int min_v = 0x7fffffff;	/* maxint */
						int max_ix = -1;
						int min_ix = -1;
						int i;
		
						for (i = curr->start; i <= curr->end; i++) {
							int v = cas[i];
							if (v > max_v) { max_v = v; max_ix = i; }
							if (v < min_v) { min_v = v; min_ix = i; }
							if (lidebug > 1)
								fprint(2, "  %d\n", v);
						}
		
						currix = (curr->type == RGN_CONVEX ? max_ix : min_ix);
		
						/* Record midpoint. */
						dpts->pts[dp++] = pts->pts[previx + (currix - previx) / 2];
		
						/* Record extreme point. */
						dpts->pts[dp++] = pts->pts[currix];
		
						previx = currix;
			}

		/* Record last mid-point and end point. */
		currix = pts->npts - 1;
		dpts->pts[dp++] = pts->pts[previx + (currix - previx) / 2];
		dpts->pts[dp] = pts->pts[currix];

	/* Compute chain-code. */
	lialg_compute_chain_code(dpts);

	free(cas);
	return(dpts);
}


static int *lialg_compute_contour_angle_set(point_list *pts,
											   region_list *regions) {
		int *V;
		region_list *curr_reg;
		int i;

		V = malloc(pts->npts*sizeof(int));

		V[0] = 18000;
		for (curr_reg = regions; curr_reg != nil; curr_reg = curr_reg->next) {
				for (i = curr_reg->start; i <= curr_reg->end; i++) {
					if (curr_reg->type == RGN_PLAIN) {
								V[i] = 18000;
					} else {
								/* For now, simply choose the mid-point. */
								int isMidPt = i == (curr_reg->start +
													 (curr_reg->end - curr_reg->start) / 2);
								V[i] = (curr_reg->type == RGN_CONVEX)
								  ? (isMidPt ? 18000 : 0)
								  : (isMidPt ? 0 : 18000);
					}
				}
	}
	V[pts->npts - 1] = 18000;

	return(V);
}


/*
 *  First compute the similarity between the two strings.
 *  If it's above a threshold, compute the distance between
 *  the two and return it as the ``score.''
 *  Otherwise, return the constant WORST_SCORE.
 *
 */
static void lialg_score_stroke(point_list *input_dompts, point_list *curr_dompts, int *sim, int *dist) {
	*sim = MIN_SIM;
	*dist = MAX_DIST;

	*sim = lialg_compute_similarity(input_dompts, curr_dompts);
	if (*sim < SIM_THLD) goto done;

	*dist = lialg_compute_distance(input_dompts, curr_dompts);

done:
	if (lidebug)
		fprint(2, "%d, %d\n", *sim, *dist);
}


static int lialg_compute_similarity(point_list *input_dompts, point_list *curr_dompts) {
	int sim;
	point_list *A, *B;
	int N, M;
	int **G;
	int *junk;
	int i, j;

	/* A is the	longer sequence, length	N. */
	/* B is the shorter sequence, length M. */
		if (input_dompts->npts >= curr_dompts->npts) {
				A = input_dompts;
				N = input_dompts->npts;
				B = curr_dompts;
				M = curr_dompts->npts;
	} else {
				A = curr_dompts;
				N = curr_dompts->npts;
				B = input_dompts;
				M = input_dompts->npts;
		}

		/* Allocate and initialize the Gain matrix, G. */
		/* The size of G is M x (N + 1). */
		/* Note that row 0 is unused. */
		/* Similarities are x 10. */
		G = malloc(M*sizeof(int *));
		junk = malloc(M * (N + 1) * sizeof(int));
		for (i = 0; i < M; i++)
			G[i] = junk + (i * (N + 1));

		for (i = 1; i < M; i++) {
			int bval = B->pts[i-1].chaincode;

			/* Source column. */
			G[i][0] = 0;

			for (j = 1; j < N; j++) {
				int aval = A->pts[j-1].chaincode;
				int diff = abs(bval - aval);
				if (diff > 4) diff = 8 - diff;

				G[i][j] = (diff == 0)
				  ? 10
				  : (diff == 1)
				  ? 6
				  : 0;
			}

			/* Sink column. */
			G[i][N] = 0;
		}

	/* Do the DP algorithm. */
	/* Proceed in column order, from highest column to the lowest. */
	/* Within each column, proceed from the highest row to the lowest. */
	/* Skip the highest column. */
		for (j = N - 1; j >= 0; j--)
			for (i = M - 1; i > 0; i--) {
				int max = G[i][j + 1];

				if (i < (M - 1)) {
					int tmp = G[i + 1][j + 1];
					if (tmp > max) max = tmp;
				}

				G[i][j] += max;
		}

		sim = (10 * G[1][0] + (N - 1) / 2) / (N - 1);

		if (G != nil)
				free(G);
		if (junk != nil)
				free(junk);
		return(sim);
}


static int lialg_compute_distance(point_list *input_dompts,
								   point_list *curr_dompts) {
		int dist;
		point_list *A, *B;
		int N, M;
		int **C;
		int *junk;
		int *BE;
		int *TE;
		int i, j;

		/* A is the	longer sequence, length	N. */
		/* B is the shorter sequence, length M. */
		if (input_dompts->npts >= curr_dompts->npts) {
				A = input_dompts;
				N = input_dompts->npts;
				B = curr_dompts;
				M = curr_dompts->npts;
		}
		else {
				A = curr_dompts;
				N = curr_dompts->npts;
				B = input_dompts;
				M = input_dompts->npts;
		}

		/* Construct the helper vectors, BE and TE, which say for each column */
		/* what are the ``bottom'' and ``top'' rows of interest. */
		BE = malloc((N + 1)*sizeof(int));
		TE = malloc((N + 1)*sizeof(int));

		for (j = 1; j <= N; j++) {
				int bot, top;

				bot = j + (M - DP_BAND);
				if (bot > M) bot = M;
				BE[j] = bot;

				top = j - (N - DP_BAND);
				if (top < 1) top = 1;
				TE[j] = top;
		}

		/* Allocate and initialize the Cost matrix, C. */
		/* The size of C is (M + 1) x (N + 1). */
		/* Note that row and column 0 are unused. */
		/* Costs are x 100. */
		/*	C = (int **)safe_malloc((M + 1) * sizeof(int *)); */
		C = malloc((M + 1)*sizeof( int *));
		junk = malloc((M + 1) * (N + 1)*sizeof(int));
		for (i = 0; i <= M; i++)
				C[i] = junk + (i * (N + 1));

		for (i = 1; i <= M; i++) {
				int bx = B->pts[i-1].x;
				int by = B->pts[i-1].y;

				for (j = 1; j <= N; j++) {
						int ax = A->pts[j-1].x;
						int ay = A->pts[j-1].y;
						int dx = bx - ax;
						int dy = by - ay;
						int dist = isqrt(10000 * (dx * dx + dy * dy));
				
						C[i][j] = dist;
				}
		}

		/* Do the DP algorithm. */
		/* Proceed in column order, from highest column to the lowest. */
		/* Within each column, proceed from the highest row to the lowest. */
		for (j = N; j > 0; j--)
				for (i = M; i > 0; i--) {
						int min = MAX_DIST;
		
						if (i > BE[j] || i < TE[j] || (j == N && i == M))
								continue;
		
						if (j < N) {
								if (i >= TE[j+1]) {
										int tmp = C[i][j+1];
										if (tmp < min)
												min = tmp;
								}
		
								if (i < M) {
										int tmp = C[i+1][j+1];
										if (tmp < min)
												min = tmp;
								}
						}
		
						if (i < BE[j]) {
								int tmp = C[i+1][j];
								if (tmp < min) min = tmp;
						}
		
						C[i][j] += min;
				}

		dist = (C[1][1] + N / 2) / N;

		if (C != nil) free(C);
		if (junk != nil) free(junk);
		if (BE != nil) free(BE);
		if (TE != nil) free(TE);
		return(dist);
}


/*************************************************************

  Digest-processing routines

 *************************************************************/

static int lialg_read_classifier_digest(rClassifier *rec) {
	int nclasses;
	FILE *fp;

	/* Try to open the corresponding digest file. */
	{
				char *clx_path;
				char *dot;
		
				/* Get a copy of the filename, with some room on the end. */
				/*	clx_path = safe_malloc(strlen(rec->file_name) + 5); */
				clx_path = malloc((strlen(rec->file_name) + 5) *sizeof(char));
				strcpy(clx_path, rec->file_name);
		
				/* Truncate the path after the last dot. */
				dot = strrchr(clx_path, '.');
				if (dot == nil) { free(clx_path); return(-1); }
				*(dot + 1) = 0;
		
				/* Append the classifier-digest extension. */
				strcat(clx_path, "clx");
		
				fp = fopen(clx_path, "r");
				if (fp == nil) {
						free(clx_path);
						return(-1);
				}
		
				free(clx_path);
	}

	/* Read-in the name and dominant points for each class. */
	for (nclasses = 0; !feof(fp); nclasses++) {
		point_list *dpts = nil;
		char class[BUFSIZ];
		int npts;
		int j;

		if (fscanf(fp, "%s %d", class, &npts) != 2) {
			if (feof(fp)) break;

			goto failed;
		}
		rec->cnames[nclasses] = strdup(class);

		/* Allocate a dominant-points list. */
		/* dpts = (point_list *)safe_malloc(sizeof(point_list)); */
		dpts = malloc(sizeof(point_list));
		dpts->pts = mallocz(npts*sizeof(pen_point), 1);
		if (dpts->pts == nil) goto failed;
		dpts->npts = npts;
		dpts->next = nil;

		/* Read in each point. */
		for (j = 0; j < npts; j++) {
			int x, y;

			if (fscanf(fp, "%d %d", &x, &y) != 2) goto failed;
			dpts->pts[j].x = x;
			dpts->pts[j].y = y;
		}

		/* Compute the chain-code. */
		lialg_compute_chain_code(dpts);

		/* Store the list in the rec data structure. */
		rec->dompts[nclasses] = dpts;

		continue;

failed:
		fprint(2, "read_classifier_digest failed...\n");
		for (; nclasses >= 0; nclasses--) {
			if (rec->cnames[nclasses] != nil) {
				free(rec->cnames[nclasses]);
				rec->cnames[nclasses] = nil;
			}
			if (rec->dompts[nclasses] != nil) {
				delete_examples(rec->dompts[nclasses]);
				rec->dompts[nclasses] = nil;
			}
		}
		if (dpts != nil)
			delete_examples(dpts);
		fclose(fp);
		return(-1);
	}

	fclose(fp);
	return(0);
}


/*************************************************************

  Canonicalization routines

 *************************************************************/

static int lialg_canonicalize_examples(rClassifier *rec) {
	int i;
	int nclasses;

	if (lidebug) {
		fprint(2, "lialg_canonicalize_examples working on %s\n",
				rec->file_name);
	}
	/* Initialize canonical-example arrays. */
	for (i = 0; i < MAXSCLASSES; i++) {
		rec->canonex[i] = nil;
	}

	/* Figure out number of classes. */
	for (nclasses = 0;
		  nclasses < MAXSCLASSES && rec->cnames[nclasses] != nil;
		  nclasses++)
		;

	/* Canonicalize the examples for each class. */
	for (i = 0; i < nclasses; i++) {
		int j, k;
		int nex;
		point_list *pts, *tmp, *avg;
		int maxxrange, maxyrange;
		int minx, miny, maxx, maxy;
		int avgxrange, avgyrange, avgxoff, avgyoff, avgscale;

		
		if (lidebug) {
			fprint(2, "lialg_canonicalize_examples working on class %s\n",
					rec->cnames[i]);
		}
		/* Make a copy of the examples. */
		pts = nil;
		tmp = rec->ex[i];
		for (nex = 0; tmp != nil; nex++, tmp = tmp->next) {
			if ((pts = add_example(pts, tmp->npts, tmp->pts)) == nil) {
				delete_examples(pts);
				return(-1);
			}
		}

		/* Canonicalize each example, and derive the max x and y ranges. */
		maxxrange = 0;
		maxyrange = 0;
		for (j = 0, tmp = pts; j < nex; j++, tmp = tmp->next) {
			if (lialg_canonicalize_example_stroke(tmp) != 0) {
  	        if (lidebug) {
					fprint(2, "lialg_canonicalize_example_stroke returned error\n");
				}
				return(-1);
			}

			if (tmp->xrange > maxxrange) maxxrange = tmp->xrange;
			if (tmp->yrange > maxyrange) maxyrange = tmp->yrange;
		}

		/* Normalize max ranges. */
		if (((100 * maxxrange + CANONICAL_X / 2) / CANONICAL_X) >
			((100 * maxyrange + CANONICAL_Y / 2) / CANONICAL_Y)) {
			maxyrange = (maxyrange * CANONICAL_X + maxxrange / 2) / maxxrange;
			maxxrange = CANONICAL_X;
		}
		else {
			maxxrange = (maxxrange * CANONICAL_Y + maxyrange / 2) / maxyrange;
			maxyrange = CANONICAL_Y;
		}

		/* Re-scale each example to max ranges. */
		for (j = 0, tmp = pts; j < nex; j++, tmp = tmp->next) {
			int scalex = (tmp->xrange == 0) ? 100 : (100 * maxxrange + tmp->xrange / 2) / tmp->xrange;
			int scaley = (tmp->yrange == 0) ? 100 : (100 * maxyrange + tmp->yrange / 2) / tmp->yrange;
			if (lialg_translate_points(tmp, 0, 0, scalex, scaley) != 0) {
				delete_examples(pts);
				return(-1);
			}
		}

		/* Average the examples; leave average in first example. */
		avg = pts;				/* careful aliasing!! */
		for (k = 0; k < NCANONICAL; k++) {
			int xsum = 0;
			int ysum = 0;

			for (j = 0, tmp = pts; j < nex; j++, tmp = tmp->next) {
						xsum += tmp->pts[k].x;
						ysum += tmp->pts[k].y;
			}

			avg->pts[k].x = (xsum + j / 2) / j;
			avg->pts[k].y = (ysum + j / 2) / j;
		}

		/* Compute BB of averaged stroke and re-scale. */
		lialg_get_bounding_box(avg, &minx, &miny, &maxx, &maxy);
		avgxrange = maxx - minx;
		avgyrange = maxy - miny;
		avgscale = (((100 * avgxrange + CANONICAL_X / 2) / CANONICAL_X) >
					((100 * avgyrange + CANONICAL_Y / 2) / CANONICAL_Y))
		  ? (100 * CANONICAL_X + avgxrange / 2) / avgxrange
		  : (100 * CANONICAL_Y + avgyrange / 2) / avgyrange;
		if (lialg_translate_points(avg, minx, miny, avgscale, avgscale) != 0) {
			delete_examples(pts);
			return(-1);
		}

		/* Re-compute the x and y ranges and center the stroke. */
		lialg_get_bounding_box(avg, &minx, &miny, &maxx, &maxy);
		avgxrange = maxx - minx;
		avgyrange = maxy - miny;
		avgxoff = -((CANONICAL_X - avgxrange + 1) / 2);
		avgyoff = -((CANONICAL_Y - avgyrange + 1) / 2);
		if (lialg_translate_points(avg, avgxoff, avgyoff, 100, 100) != 0) {
			delete_examples(pts);
			return(-1);
		}

		/* Create a point list to serve as the ``canonical representation. */
		if ((rec->canonex[i] = add_example(nil, avg->npts, avg->pts)) == nil) {
			delete_examples(pts);
			return(-1);
		}
		(rec->canonex[i])->xrange = maxx - minx;
		(rec->canonex[i])->yrange = maxy - miny;

		if (lidebug) {
			fprint(2, "%s, avgpts = %d\n", rec->cnames[i], avg->npts);
			for (j = 0; j < avg->npts; j++) {
						fprint(2, "  (%P)\n", avg->pts[j].Point);
			}
		}

		/* Compute dominant points of canonical representation. */
		rec->dompts[i] = lialg_compute_dominant_points(avg);

		/* Clean up. */
		delete_examples(pts);
	}

	/* Sanity check. */
	for (i = 0; i < nclasses; i++) {
		char *best_name = lialg_recognize_stroke(rec, rec->canonex[i]);

		if (best_name != rec->cnames[i])
			fprint(2, "%s, best = %s\n", rec->cnames[i], best_name);
	}

	return(0);
}


static int lialg_canonicalize_example_stroke(point_list *points) {
	int minx, miny, maxx, maxy, xrange, yrange, scale;

	/* Filter out points that are too close. */
	if (lialg_filter_points(points) != 0) return(-1);

	/* Must be at least two points! */
	if (points->npts < 2) {
		if (lidebug) {
			fprint(2, "lialg_canonicalize_example_stroke: npts=%d\n",
					points->npts);
		}
		return(-1);
	}

	/* Scale up to avoid conversion errors. */
	lialg_get_bounding_box(points, &minx, &miny, &maxx, &maxy);
	xrange = maxx - minx;
	yrange = maxy - miny;
	scale = (((100 * xrange + CANONICAL_X / 2) / CANONICAL_X) >
			 ((100 * yrange + CANONICAL_Y / 2) / CANONICAL_Y))
	  ? (100 * CANONICAL_X + xrange / 2) / xrange
	  : (100 * CANONICAL_Y + yrange / 2) / yrange;
	if (lialg_translate_points(points, minx, miny, scale, scale) != 0) {
		if (lidebug) {
			fprint(2, "lialg_translate_points (minx=%d,miny=%d,scale=%d) returned error\n", minx, miny, scale);
		}
		return(-1);
	}

	/* Compute an equivalent stroke with equi-distant points. */
	if (lialg_compute_equipoints(points) != 0) return(-1);

	/* Re-translate the points to the origin. */
	lialg_get_bounding_box(points, &minx, &miny, &maxx, &maxy);
	if (lialg_translate_points(points, minx, miny, 100, 100) != 0) {
		if (lidebug) {
			fprint(2, "lialg_translate_points (minx=%d,miny=%d) returned error\n", minx, miny);
		}
		return(-1);
	}

	/* Store the x and y ranges in the point list. */
	xrange = maxx - minx;
	yrange = maxy - miny;
	points->xrange = xrange;
	points->yrange = yrange;

	if (lidebug) {
		int i;
		fprint(2, "Canonicalized:   %d, %d, %d, %d\n", minx, miny, maxx, maxy);
		for (i = 0; i < points->npts; i++)
			fprint(2, "      (%P)\n", points->pts[i].Point);
		fflush(stderr);
	}

	return(0);
}


static int lialg_compute_equipoints(point_list *points) {
	pen_point *equipoints = mallocz(NCANONICAL*sizeof(pen_point), 1);
	int nequipoints = 0;
	int pathlen = lialg_compute_pathlen(points);
	int equidist = (pathlen + (NCANONICAL - 1) / 2) / (NCANONICAL - 1);
	int i;
	int dist_since_last_eqpt;
	int remaining_seglen;
	int dist_to_next_eqpt;

	if (equipoints == nil) {
		fprint(2, "can't allocate memory in lialg_compute_equipoints");
		return(-1);
	}

	if (lidebug) {
		fprint(2, "compute_equipoints:  npts = %d, pathlen = %d, equidist = %d\n",
				points->npts, pathlen, equidist);
		fflush(stderr);
	}

	/* First original point is an equipoint. */
	equipoints[0] = points->pts[0];
	nequipoints++;
	dist_since_last_eqpt = 0;

	for (i = 1; i < points->npts; i++) {
		int dx1 = points->pts[i].x - points->pts[i-1].x;
		int dy1 = points->pts[i].y - points->pts[i-1].y;
		int endx = 100 * points->pts[i-1].x;
		int endy = 100 * points->pts[i-1].y;
		remaining_seglen = isqrt(10000 * (dx1 * dx1 + dy1 * dy1));
		dist_to_next_eqpt = equidist - dist_since_last_eqpt;

		while (remaining_seglen >= dist_to_next_eqpt) {
			if (dx1 == 0) {
				/* x-coordinate stays the same */
				if (dy1 >= 0)
					endy += dist_to_next_eqpt;
				else
					endy -= dist_to_next_eqpt;
			}
			else {
				int slope = (100 * dy1 + dx1 / 2) / dx1;
				int tmp = isqrt(10000 + slope * slope);
				int dx = (100 * dist_to_next_eqpt + tmp / 2) / tmp;
				int dy = (slope * dx + 50) / 100;

				if (dy < 0) dy = -dy;
				if (dx1 >= 0)
					endx += dx;
				else
					endx -= dx;
				if (dy1 >= 0)
					endy += dy;
				else
					endy -= dy;
			}

			equipoints[nequipoints].x = (endx + 50) / 100;
			equipoints[nequipoints].y = (endy + 50) / 100;
			nequipoints++;
/*	    assert(nequipoints <= NCANONICAL);*/
			dist_since_last_eqpt = 0;
			remaining_seglen -= dist_to_next_eqpt;
			dist_to_next_eqpt = equidist;
		}

		dist_since_last_eqpt += remaining_seglen;
	}

	/* Take care of last equipoint. */
	if (nequipoints == NCANONICAL) {
		/* Good. */
	} else if (nequipoints == (NCANONICAL - 1)) {
		/* Make last original point the last equipoint. */
		equipoints[nequipoints] = points->pts[points->npts - 1];
	} else {
	  if (lidebug) {
		fprint(2,"lialg_compute_equipoints: nequipoints = %d\n", 
				nequipoints);
	  }
/*	assert(false);*/
		return(-1);
	}

	points->npts = NCANONICAL;
	delete_pen_point_array(points->pts);
	points->pts = equipoints;
	return(0);
}


/*************************************************************

  Utility routines

 *************************************************************/

/* Result is x 100. */
static int lialg_compute_pathlen(point_list *points) {
	return(lialg_compute_pathlen_subset(points, 0, points->npts - 1));
}


/* Result is x 100. */
static int lialg_compute_pathlen_subset(point_list *points,
										   int start, int end) {
	int pathlen;
	int i;

	pathlen = 0;
	for (i = start + 1; i <= end; i++) {
		int dx = points->pts[i].x - points->pts[i-1].x;
		int dy = points->pts[i].y - points->pts[i-1].y;
		int dist = isqrt(10000 * (dx * dx + dy * dy));
		pathlen += dist;
	}

	return(pathlen);
}


/* Note that this does NOT update points->xrange and points->yrange! */
static int lialg_filter_points(point_list *points) {
	int filtered_npts;
	pen_point *filtered_pts = mallocz(points->npts*sizeof(pen_point), 1);
	int i;

	if (filtered_pts == nil) {
		fprint(2, "can't allocate memory in lialg_filter_points");
		return(-1);
	}

	filtered_pts[0] = points->pts[0];
	filtered_npts = 1;
	for (i = 1; i < points->npts; i++) {
		int j = filtered_npts - 1;
		int dx = points->pts[i].x - filtered_pts[j].x;
		int dy = points->pts[i].y - filtered_pts[j].y;
		int magsq = dx * dx + dy * dy;

		if (magsq >= DIST_SQ_THRESHOLD) {
			filtered_pts[filtered_npts] = points->pts[i];
			filtered_npts++;
		}
	}

	points->npts = filtered_npts;
	delete_pen_point_array(points->pts);
	points->pts = filtered_pts;
	return(0);
}


/* scalex and scaley are x 100. */
/* Note that this does NOT update points->xrange and points->yrange! */
static int lialg_translate_points(point_list *points,
								   int minx, int miny,
								   int scalex, int scaley) {
	int i;

	for (i = 0; i < points->npts; i++) {
				points->pts[i].x = ((points->pts[i].x - minx) * scalex + 50) / 100;
				points->pts[i].y = ((points->pts[i].y - miny) * scaley + 50) / 100;
	}

	return(0);
}


static void lialg_get_bounding_box(point_list *points,
									int *pminx, int *pminy,
									int *pmaxx, int *pmaxy) {
	int minx, miny, maxx, maxy;
	int i;

	minx = maxx = points->pts[0].x;
	miny = maxy = points->pts[0].y;
	for (i = 1; i < points->npts; i++) {
				pen_point *pt = &(points->pts[i]);
				if (pt->x < minx) minx = pt->x;
				else if (pt->x > maxx) maxx = pt->x;
				if (pt->y < miny) miny = pt->y;
				else if (pt->y > maxy) maxy = pt->y;
	}

	*pminx = minx;
	*pminy = miny;
	*pmaxx = maxx;
	*pmaxy = maxy;
}


int wtvals[] = {100, 104, 117, 143, 189, 271, 422};

static void lialg_compute_lpf_parameters(void) {
	int i;

		for (i = LP_FILTER_WIDTH; i >= 0; i--) {
//		double x = 0.04 * (i * i);
//		double tmp = 100.0 * exp(x);
//		int wt = floor((double)tmp);
				int wt = wtvals[i];
				lialg_lpfwts[LP_FILTER_WIDTH - i] = wt;
				lialg_lpfwts[LP_FILTER_WIDTH + i] = wt;
		}
		lialg_lpfconst = 0;
		for (i = 0; i < (2 * LP_FILTER_WIDTH + 1); i++) {
				lialg_lpfconst += lialg_lpfwts[i];
		}
}


/* Code from Joseph Hall (jnhall@sat.mot.com). */
static int isqrt(int n) {
		register int i;
		register long k0, k1, nn;

		for (nn = i = n, k0 = 2; i > 0; i >>= 2, k0 <<= 1)
				;
		nn <<= 2;
	for (;;) {
				k1 = (nn / k0 + k0) >> 1;
				if (((k0 ^ k1) & ~1) == 0)
				break;
				k0 = k1;
		}
		return (int) ((k1 + 1) >> 1);
}


/* Helper routines from Mark Hayter. */
static int likeatan(int tantop, int tanbot) { 
		int t;
		/* Use tan(theta)=top/bot --> order for t */
		/* t in range 0..0x40000 */

		if ((tantop == 0) && (tanbot == 0)) 
				t = 0;
	else
		{
				t = (tantop << 16) / (abs(tantop) + abs(tanbot));
				if (tanbot < 0) 
						t = 0x20000 - t;
				else 
						if (tantop < 0) t = 0x40000 + t;
		}
		return t;
}


static int quadr(int t) {
	return (8 - (((t + 0x4000) >> 15) & 7)) & 7;
}