This repository was archived by the owner on May 18, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 51
/
Copy pathmodp_b16.c
125 lines (119 loc) · 3.39 KB
/
modp_b16.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/**
* \file
* <PRE>
* MODP_B16 - High performance base16 encoder/decoder
* https://github.com/client9/stringencoders
*
* Copyright © 2005-2016 Nick Galbreath
* All rights reserved.
* Released under MIT license. See LICENSE for details.
* </PRE>
*/
#include "modp_b16.h"
#include "config.h"
#include "modp_b16_data.h"
#include "modp_stdint.h"
size_t modp_b16_encode(char* dest, const char* str, size_t len)
{
size_t i;
const size_t buckets = len >> 2; /* i.e. i / 4 */
const size_t leftover = len & 0x03; /* i.e. i % 4 */
const uint8_t* srcChar;
uint8_t* p = (uint8_t*)dest;
uint8_t t1, t2, t3, t4;
const uint32_t* srcInt = (const uint32_t*)str;
uint32_t x;
for (i = 0; i < buckets; ++i) {
x = *srcInt++;
/* t1 = *s++; t2 = *s++; t3 = *s++; t4 = *s++; */
#ifdef WORDS_BIGENDIAN
t1 = (uint8_t)(x >> 24);
t2 = (uint8_t)(x >> 16);
t3 = (uint8_t)(x >> 8);
t4 = (uint8_t)x;
#else
t4 = (uint8_t)(x >> 24);
t3 = (uint8_t)(x >> 16);
t2 = (uint8_t)(x >> 8);
t1 = (uint8_t)x;
#endif
*p++ = gsHexEncodeC1[t1];
*p++ = gsHexEncodeC2[t1];
*p++ = gsHexEncodeC1[t2];
*p++ = gsHexEncodeC2[t2];
*p++ = gsHexEncodeC1[t3];
*p++ = gsHexEncodeC2[t3];
*p++ = gsHexEncodeC1[t4];
*p++ = gsHexEncodeC2[t4];
}
srcChar = (const uint8_t*)srcInt;
switch (leftover) {
case 0:
break;
case 1:
t1 = (uint8_t)*srcChar;
*p++ = gsHexEncodeC1[t1];
*p++ = gsHexEncodeC2[t1];
break;
case 2:
t1 = (uint8_t)*srcChar++;
t2 = (uint8_t)*srcChar;
*p++ = gsHexEncodeC1[t1];
*p++ = gsHexEncodeC2[t1];
*p++ = gsHexEncodeC1[t2];
*p++ = gsHexEncodeC2[t2];
break;
default: /* case 3 */
t1 = (uint8_t)*srcChar++;
t2 = (uint8_t)*srcChar++;
t3 = (uint8_t)*srcChar;
*p++ = gsHexEncodeC1[t1];
*p++ = gsHexEncodeC2[t1];
*p++ = gsHexEncodeC1[t2];
*p++ = gsHexEncodeC2[t2];
*p++ = gsHexEncodeC1[t3];
*p++ = gsHexEncodeC2[t3];
}
*p = '\0';
return (size_t)(p - (uint8_t*)dest);
}
size_t modp_b16_decode(char* dest, const char* str, size_t len)
{
size_t i;
uint8_t t0, t1, t2, t3;
uint8_t* p = (uint8_t*)dest;
uint32_t val1, val2;
const uint8_t* s = (const uint8_t*)str;
const size_t buckets = len >> 2; /* i.e. len / 4 */
const size_t leftover = len & 0x03; /* i.e. len % 4 */
if (leftover & 0x01) { /* i.e if leftover is odd, */
/* leftover==1 || leftover == 3 */
return (size_t)-1;
}
/* read 4 bytes, output 2.
* Note on PPC G4, GCC 4.0, it's quite a bit faster to
* NOT use t0,t1,t2,t3, and just put the *s++ in the gsHexDecodeMap
* lookup
*/
for (i = 0; i < buckets; ++i) {
t0 = *s++;
t1 = *s++;
t2 = *s++;
t3 = *s++;
val1 = gsHexDecodeD2[t0] | gsHexDecodeMap[t1];
val2 = gsHexDecodeD2[t2] | gsHexDecodeMap[t3];
if (val1 > 0xff || val2 > 0xff) {
return (size_t)-1;
}
*p++ = (uint8_t)val1;
*p++ = (uint8_t)val2;
}
if (leftover == 2) {
val1 = gsHexDecodeD2[s[0]] | gsHexDecodeMap[s[1]];
if (val1 > 0xff) {
return (size_t)-1;
}
*p++ = (uint8_t)val1;
}
return (size_t)(p - (uint8_t*)dest);
}