xref: /inferno-os/libkern/u64.c (revision 37da2899f40661e3e9631e497da8dc59b971cbd0)
1*37da2899SCharles.Forsyth #include <lib9.h>
2*37da2899SCharles.Forsyth 
3*37da2899SCharles.Forsyth enum {
4*37da2899SCharles.Forsyth 	INVAL=	255
5*37da2899SCharles.Forsyth };
6*37da2899SCharles.Forsyth 
7*37da2899SCharles.Forsyth static uchar t64d[256] = {
8*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
9*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
10*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,   62,INVAL,INVAL,INVAL,   63,
11*37da2899SCharles.Forsyth       52,   53,   54,   55,   56,   57,   58,   59,   60,   61,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
12*37da2899SCharles.Forsyth    INVAL,    0,    1,    2,    3,    4,    5,    6,    7,    8,    9,   10,   11,   12,   13,   14,
13*37da2899SCharles.Forsyth       15,   16,   17,   18,   19,   20,   21,   22,   23,   24,   25,INVAL,INVAL,INVAL,INVAL,INVAL,
14*37da2899SCharles.Forsyth    INVAL,   26,   27,   28,   29,   30,   31,   32,   33,   34,   35,   36,   37,   38,   39,   40,
15*37da2899SCharles.Forsyth       41,   42,   43,   44,   45,   46,   47,   48,   49,   50,   51,INVAL,INVAL,INVAL,INVAL,INVAL,
16*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
17*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
18*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
19*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
20*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
21*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
22*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,
23*37da2899SCharles.Forsyth    INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL,INVAL
24*37da2899SCharles.Forsyth };
25*37da2899SCharles.Forsyth static char t64e[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
26*37da2899SCharles.Forsyth 
27*37da2899SCharles.Forsyth int
dec64(uchar * out,int lim,char * in,int n)28*37da2899SCharles.Forsyth dec64(uchar *out, int lim, char *in, int n)
29*37da2899SCharles.Forsyth {
30*37da2899SCharles.Forsyth 	ulong b24;
31*37da2899SCharles.Forsyth 	uchar *start = out;
32*37da2899SCharles.Forsyth 	uchar *e = out + lim;
33*37da2899SCharles.Forsyth 	int i, c;
34*37da2899SCharles.Forsyth 
35*37da2899SCharles.Forsyth 	b24 = 0;
36*37da2899SCharles.Forsyth 	i = 0;
37*37da2899SCharles.Forsyth 	while(n-- > 0){
38*37da2899SCharles.Forsyth 
39*37da2899SCharles.Forsyth 		c = t64d[*(uchar*)in++];
40*37da2899SCharles.Forsyth 		if(c == INVAL)
41*37da2899SCharles.Forsyth 			continue;
42*37da2899SCharles.Forsyth 		switch(i){
43*37da2899SCharles.Forsyth 		case 0:
44*37da2899SCharles.Forsyth 			b24 = c<<18;
45*37da2899SCharles.Forsyth 			break;
46*37da2899SCharles.Forsyth 		case 1:
47*37da2899SCharles.Forsyth 			b24 |= c<<12;
48*37da2899SCharles.Forsyth 			break;
49*37da2899SCharles.Forsyth 		case 2:
50*37da2899SCharles.Forsyth 			b24 |= c<<6;
51*37da2899SCharles.Forsyth 			break;
52*37da2899SCharles.Forsyth 		case 3:
53*37da2899SCharles.Forsyth 			if(out + 3 > e)
54*37da2899SCharles.Forsyth 				goto exhausted;
55*37da2899SCharles.Forsyth 
56*37da2899SCharles.Forsyth 			b24 |= c;
57*37da2899SCharles.Forsyth 			*out++ = b24>>16;
58*37da2899SCharles.Forsyth 			*out++ = b24>>8;
59*37da2899SCharles.Forsyth 			*out++ = b24;
60*37da2899SCharles.Forsyth 			i = -1;
61*37da2899SCharles.Forsyth 			break;
62*37da2899SCharles.Forsyth 		}
63*37da2899SCharles.Forsyth 		i++;
64*37da2899SCharles.Forsyth 	}
65*37da2899SCharles.Forsyth 	switch(i){
66*37da2899SCharles.Forsyth 	case 2:
67*37da2899SCharles.Forsyth 		if(out + 1 > e)
68*37da2899SCharles.Forsyth 			goto exhausted;
69*37da2899SCharles.Forsyth 		*out++ = b24>>16;
70*37da2899SCharles.Forsyth 		break;
71*37da2899SCharles.Forsyth 	case 3:
72*37da2899SCharles.Forsyth 		if(out + 2 > e)
73*37da2899SCharles.Forsyth 			goto exhausted;
74*37da2899SCharles.Forsyth 		*out++ = b24>>16;
75*37da2899SCharles.Forsyth 		*out++ = b24>>8;
76*37da2899SCharles.Forsyth 		break;
77*37da2899SCharles.Forsyth 	}
78*37da2899SCharles.Forsyth exhausted:
79*37da2899SCharles.Forsyth 	return out - start;
80*37da2899SCharles.Forsyth }
81*37da2899SCharles.Forsyth 
82*37da2899SCharles.Forsyth int
enc64(char * out,int lim,uchar * in,int n)83*37da2899SCharles.Forsyth enc64(char *out, int lim, uchar *in, int n)
84*37da2899SCharles.Forsyth {
85*37da2899SCharles.Forsyth 	int i;
86*37da2899SCharles.Forsyth 	ulong b24;
87*37da2899SCharles.Forsyth 	char *start = out;
88*37da2899SCharles.Forsyth 	char *e = out + lim;
89*37da2899SCharles.Forsyth 
90*37da2899SCharles.Forsyth 	for(i = n/3; i > 0; i--){
91*37da2899SCharles.Forsyth 		b24 = (*in++)<<16;
92*37da2899SCharles.Forsyth 		b24 |= (*in++)<<8;
93*37da2899SCharles.Forsyth 		b24 |= *in++;
94*37da2899SCharles.Forsyth 		if(out + 4 >= e)
95*37da2899SCharles.Forsyth 			goto exhausted;
96*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>18)];
97*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>12)&0x3f];
98*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>6)&0x3f];
99*37da2899SCharles.Forsyth 		*out++ = t64e[(b24)&0x3f];
100*37da2899SCharles.Forsyth 	}
101*37da2899SCharles.Forsyth 
102*37da2899SCharles.Forsyth 	switch(n%3){
103*37da2899SCharles.Forsyth 	case 2:
104*37da2899SCharles.Forsyth 		b24 = (*in++)<<16;
105*37da2899SCharles.Forsyth 		b24 |= (*in)<<8;
106*37da2899SCharles.Forsyth 		if(out + 4 >= e)
107*37da2899SCharles.Forsyth 			goto exhausted;
108*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>18)];
109*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>12)&0x3f];
110*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>6)&0x3f];
111*37da2899SCharles.Forsyth 		*out++ = '=';
112*37da2899SCharles.Forsyth 		break;
113*37da2899SCharles.Forsyth 	case 1:
114*37da2899SCharles.Forsyth 		b24 = (*in)<<16;
115*37da2899SCharles.Forsyth 		if(out + 4 >= e)
116*37da2899SCharles.Forsyth 			goto exhausted;
117*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>18)];
118*37da2899SCharles.Forsyth 		*out++ = t64e[(b24>>12)&0x3f];
119*37da2899SCharles.Forsyth 		*out++ = '=';
120*37da2899SCharles.Forsyth 		*out++ = '=';
121*37da2899SCharles.Forsyth 		break;
122*37da2899SCharles.Forsyth 	}
123*37da2899SCharles.Forsyth exhausted:
124*37da2899SCharles.Forsyth 	*out = 0;
125*37da2899SCharles.Forsyth 	return out - start;
126*37da2899SCharles.Forsyth }
127