xref: /plan9-contrib/sys/src/9/port/latin1.c (revision e94a8e9b44de11e62dfd4566245554f8bf8e0295)
1*e94a8e9bSDavid du Colombier #include	"u.h"
2*e94a8e9bSDavid du Colombier #include	"../port/lib.h"
3219b2ee8SDavid du Colombier /*
4219b2ee8SDavid du Colombier  * The code makes two assumptions: strlen(ld) is 1 or 2; latintab[i].ld can be a
5219b2ee8SDavid du Colombier  * prefix of latintab[j].ld only when j<i.
6219b2ee8SDavid du Colombier  */
7219b2ee8SDavid du Colombier struct cvlist
83e12c5d1SDavid du Colombier {
9219b2ee8SDavid du Colombier 	char	*ld;		/* must be seen before using this conversion */
10219b2ee8SDavid du Colombier 	char	*si;		/* options for last input characters */
11219b2ee8SDavid du Colombier 	Rune	*so;		/* the corresponding Rune for each si entry */
123e12c5d1SDavid du Colombier } latintab[] = {
137dd7cddfSDavid du Colombier #include "../port/latin1.h"
14219b2ee8SDavid du Colombier 	0,	0,		0
153e12c5d1SDavid du Colombier };
163e12c5d1SDavid du Colombier 
17219b2ee8SDavid du Colombier /*
18*e94a8e9bSDavid du Colombier  * Given n characters k[0]..k[n-1], find the rune or return -1 for failure.
19219b2ee8SDavid du Colombier  */
203e12c5d1SDavid du Colombier long
unicode(Rune * k,int n)21*e94a8e9bSDavid du Colombier unicode(Rune *k, int n)
223e12c5d1SDavid du Colombier {
23*e94a8e9bSDavid du Colombier 	long c;
24*e94a8e9bSDavid du Colombier 	Rune *r;
253e12c5d1SDavid du Colombier 
263e12c5d1SDavid du Colombier 	c = 0;
27*e94a8e9bSDavid du Colombier 	for(r = &k[1]; r<&k[n]; r++){		/* +1 to skip [Xx] */
283e12c5d1SDavid du Colombier 		c <<= 4;
29*e94a8e9bSDavid du Colombier 		if('0'<=*r && *r<='9')
30*e94a8e9bSDavid du Colombier 			c += *r-'0';
31*e94a8e9bSDavid du Colombier 		else if('a'<=*r && *r<='f')
32*e94a8e9bSDavid du Colombier 			c += 10 + *r-'a';
33*e94a8e9bSDavid du Colombier 		else if('A'<=*r && *r<='F')
34*e94a8e9bSDavid du Colombier 			c += 10 + *r-'A';
353e12c5d1SDavid du Colombier 		else
363e12c5d1SDavid du Colombier 			return -1;
373e12c5d1SDavid du Colombier 	}
383e12c5d1SDavid du Colombier 	return c;
393e12c5d1SDavid du Colombier }
40219b2ee8SDavid du Colombier 
41219b2ee8SDavid du Colombier /*
42219b2ee8SDavid du Colombier  * Given n characters k[0]..k[n-1], find the corresponding rune or return -1 for
43219b2ee8SDavid du Colombier  * failure, or something < -1 if n is too small.  In the latter case, the result
44219b2ee8SDavid du Colombier  * is minus the required n.
45219b2ee8SDavid du Colombier  */
46219b2ee8SDavid du Colombier long
latin1(Rune * k,int n)477dd7cddfSDavid du Colombier latin1(Rune *k, int n)
48219b2ee8SDavid du Colombier {
49219b2ee8SDavid du Colombier 	struct cvlist *l;
50219b2ee8SDavid du Colombier 	int c;
51219b2ee8SDavid du Colombier 	char* p;
52219b2ee8SDavid du Colombier 
53219b2ee8SDavid du Colombier 	if(k[0] == 'X')
54219b2ee8SDavid du Colombier 		if(n>=5)
55*e94a8e9bSDavid du Colombier 			return unicode(k, 5);
56219b2ee8SDavid du Colombier 		else
57219b2ee8SDavid du Colombier 			return -5;
58*e94a8e9bSDavid du Colombier 	if(k[0] == 'x')
59*e94a8e9bSDavid du Colombier 		if(n>=UTFmax*2+1)
60*e94a8e9bSDavid du Colombier 			return unicode(k, UTFmax*2+1);
61*e94a8e9bSDavid du Colombier 		else
62*e94a8e9bSDavid du Colombier 			return -(UTFmax+1);
63219b2ee8SDavid du Colombier 	for(l=latintab; l->ld!=0; l++)
64219b2ee8SDavid du Colombier 		if(k[0] == l->ld[0]){
65219b2ee8SDavid du Colombier 			if(n == 1)
66219b2ee8SDavid du Colombier 				return -2;
67219b2ee8SDavid du Colombier 			if(l->ld[1] == 0)
68219b2ee8SDavid du Colombier 				c = k[1];
69219b2ee8SDavid du Colombier 			else if(l->ld[1] != k[1])
70219b2ee8SDavid du Colombier 				continue;
71219b2ee8SDavid du Colombier 			else if(n == 2)
72219b2ee8SDavid du Colombier 				return -3;
73219b2ee8SDavid du Colombier 			else
74219b2ee8SDavid du Colombier 				c = k[2];
75219b2ee8SDavid du Colombier 			for(p=l->si; *p!=0; p++)
76219b2ee8SDavid du Colombier 				if(*p == c)
77219b2ee8SDavid du Colombier 					return l->so[p - l->si];
78219b2ee8SDavid du Colombier 			return -1;
79219b2ee8SDavid du Colombier 		}
80219b2ee8SDavid du Colombier 	return -1;
81219b2ee8SDavid du Colombier }
82