1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-15 01:20:10 +00:00
i2pd/libi2pd/Base.cpp

306 lines
6.5 KiB
C++
Raw Normal View History

2013-12-10 08:08:35 -05:00
#include <stdlib.h>
#include <string.h>
2015-11-03 09:15:49 -05:00
#include "Base.h"
2013-12-10 08:08:35 -05:00
namespace i2p
{
namespace data
{
static const char T32[32] = {
'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h',
2016-03-14 13:33:51 -04:00
'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p',
'q', 'r', 's', 't', 'u', 'v', 'w', 'x',
'y', 'z', '2', '3', '4', '5', '6', '7',
};
const char * GetBase32SubstitutionTable ()
{
return T32;
}
2013-12-10 08:08:35 -05:00
static void iT64Build(void);
/*
*
* BASE64 Substitution Table
* -------------------------
*
* Direct Substitution Table
*/
static const char T64[64] = {
2013-12-10 08:08:35 -05:00
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H',
'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X',
'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n',
'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
'w', 'x', 'y', 'z', '0', '1', '2', '3',
'4', '5', '6', '7', '8', '9', '-', '~'
};
2014-01-29 17:38:53 -05:00
const char * GetBase64SubstitutionTable ()
{
return T64;
2018-01-06 11:48:51 +08:00
}
2013-12-10 08:08:35 -05:00
/*
* Reverse Substitution Table (built in run time)
*/
static char iT64[256];
static int isFirstTime = 1;
/*
2018-01-06 11:48:51 +08:00
* Padding
2013-12-10 08:08:35 -05:00
*/
2018-01-06 11:48:51 +08:00
static char P64 = '=';
2013-12-10 08:08:35 -05:00
/*
*
* ByteStreamToBase64
* ------------------
*
* Converts binary encoded data to BASE64 format.
*
*/
size_t /* Number of bytes in the encoded buffer */
2018-01-06 11:48:51 +08:00
ByteStreamToBase64 (
2013-12-10 08:08:35 -05:00
const uint8_t * InBuffer, /* Input buffer, binary data */
2018-01-06 11:48:51 +08:00
size_t InCount, /* Number of bytes in the input buffer */
2013-12-10 08:08:35 -05:00
char * OutBuffer, /* output buffer */
2018-01-06 11:48:51 +08:00
size_t len /* length of output buffer */
2013-12-10 08:08:35 -05:00
)
{
unsigned char * ps;
unsigned char * pd;
unsigned char acc_1;
unsigned char acc_2;
2018-01-06 11:48:51 +08:00
int i;
int n;
int m;
2013-12-10 08:08:35 -05:00
size_t outCount;
ps = (unsigned char *)InBuffer;
n = InCount/3;
m = InCount%3;
if (!m)
outCount = 4*n;
else
outCount = 4*(n+1);
2014-12-04 19:28:20 -05:00
if (outCount > len) return 0;
2013-12-10 08:08:35 -05:00
pd = (unsigned char *)OutBuffer;
for ( i = 0; i<n; i++ ){
acc_1 = *ps++;
2018-01-06 11:48:51 +08:00
acc_2 = (acc_1<<4)&0x30;
2013-12-10 08:08:35 -05:00
acc_1 >>= 2; /* base64 digit #1 */
*pd++ = T64[acc_1];
acc_1 = *ps++;
acc_2 |= acc_1 >> 4; /* base64 digit #2 */
*pd++ = T64[acc_2];
acc_1 &= 0x0f;
acc_1 <<=2;
acc_2 = *ps++;
acc_1 |= acc_2>>6; /* base64 digit #3 */
*pd++ = T64[acc_1];
acc_2 &= 0x3f; /* base64 digit #4 */
*pd++ = T64[acc_2];
2018-01-06 11:48:51 +08:00
}
2013-12-10 08:08:35 -05:00
if ( m == 1 ){
acc_1 = *ps++;
acc_2 = (acc_1<<4)&0x3f; /* base64 digit #2 */
acc_1 >>= 2; /* base64 digit #1 */
*pd++ = T64[acc_1];
*pd++ = T64[acc_2];
*pd++ = P64;
*pd++ = P64;
}
else if ( m == 2 ){
acc_1 = *ps++;
2018-01-06 11:48:51 +08:00
acc_2 = (acc_1<<4)&0x3f;
2013-12-10 08:08:35 -05:00
acc_1 >>= 2; /* base64 digit #1 */
*pd++ = T64[acc_1];
acc_1 = *ps++;
acc_2 |= acc_1 >> 4; /* base64 digit #2 */
*pd++ = T64[acc_2];
acc_1 &= 0x0f;
acc_1 <<=2; /* base64 digit #3 */
*pd++ = T64[acc_1];
*pd++ = P64;
}
2018-01-06 11:48:51 +08:00
2013-12-10 08:08:35 -05:00
return outCount;
}
/*
*
* Base64ToByteStream
* ------------------
*
* Converts BASE64 encoded data to binary format. If input buffer is
* not properly padded, buffer of negative length is returned
*
*/
size_t /* Number of output bytes */
2018-01-06 11:48:51 +08:00
Base64ToByteStream (
2013-12-10 08:08:35 -05:00
const char * InBuffer, /* BASE64 encoded buffer */
size_t InCount, /* Number of input bytes */
2018-01-06 11:48:51 +08:00
uint8_t * OutBuffer, /* output buffer length */
2013-12-10 08:08:35 -05:00
size_t len /* length of output buffer */
)
{
unsigned char * ps;
unsigned char * pd;
unsigned char acc_1;
unsigned char acc_2;
2018-01-06 11:48:51 +08:00
int i;
int n;
int m;
2013-12-10 08:08:35 -05:00
size_t outCount;
if (isFirstTime) iT64Build();
n = InCount/4;
m = InCount%4;
2018-01-06 11:48:51 +08:00
if (InCount && !m)
2013-12-10 08:08:35 -05:00
outCount = 3*n;
else {
outCount = 0;
2014-12-04 19:28:20 -05:00
return 0;
2013-12-10 08:08:35 -05:00
}
2018-01-06 11:48:51 +08:00
2013-12-10 08:08:35 -05:00
ps = (unsigned char *)(InBuffer + InCount - 1);
while ( *ps-- == P64 ) outCount--;
ps = (unsigned char *)InBuffer;
2018-01-06 11:48:51 +08:00
2013-12-10 08:08:35 -05:00
if (outCount > len) return -1;
2014-09-17 16:24:13 -04:00
pd = OutBuffer;
2018-01-06 11:48:51 +08:00
auto endOfOutBuffer = OutBuffer + outCount;
2013-12-10 08:08:35 -05:00
for ( i = 0; i < n; i++ ){
acc_1 = iT64[*ps++];
acc_2 = iT64[*ps++];
acc_1 <<= 2;
acc_1 |= acc_2>>4;
*pd++ = acc_1;
2014-09-17 16:24:13 -04:00
if (pd >= endOfOutBuffer) break;
2013-12-10 08:08:35 -05:00
acc_2 <<= 4;
acc_1 = iT64[*ps++];
acc_2 |= acc_1 >> 2;
*pd++ = acc_2;
2018-01-06 11:48:51 +08:00
if (pd >= endOfOutBuffer) break;
2013-12-10 08:08:35 -05:00
acc_2 = iT64[*ps++];
acc_2 |= acc_1 << 6;
*pd++ = acc_2;
}
return outCount;
}
2018-01-06 11:48:51 +08:00
size_t Base64EncodingBufferSize (const size_t input_size)
2016-02-07 21:35:06 -05:00
{
auto d = div (input_size, 3);
if (d.rem) d.quot++;
return 4*d.quot;
}
2018-01-06 11:48:51 +08:00
2018-04-06 15:23:56 -04:00
std::string ToBase64Standard (const std::string& in)
{
auto len = Base64EncodingBufferSize (in.length ());
char * str = new char[len+1];
auto l = ByteStreamToBase64 ((const uint8_t *)in.c_str (), in.length (), str, len);
str[l] = 0;
// replace '-' by '+' and '~' by '/'
for (size_t i = 0; i < l; i++)
if (str[i] == '-') str[i] = '+';
else if (str[i] == '~') str[i] = '/';
std::string s(str);
delete[] str;
return s;
}
2013-12-10 08:08:35 -05:00
/*
*
* iT64
* ----
* Reverse table builder. P64 character is replaced with 0
*
*
*/
static void iT64Build()
{
int i;
isFirstTime = 0;
for ( i=0; i<256; i++ ) iT64[i] = -1;
for ( i=0; i<64; i++ ) iT64[(int)T64[i]] = i;
iT64[(int)P64] = 0;
}
2018-01-06 11:48:51 +08:00
size_t Base32ToByteStream (const char * inBuf, size_t len, uint8_t * outBuf, size_t outLen)
2013-12-10 08:08:35 -05:00
{
int tmp = 0, bits = 0;
size_t ret = 0;
for (size_t i = 0; i < len; i++)
{
2018-01-06 11:48:51 +08:00
char ch = inBuf[i];
2013-12-10 08:08:35 -05:00
if (ch >= '2' && ch <= '7') // digit
ch = (ch - '2') + 26; // 26 means a-z
else if (ch >= 'a' && ch <= 'z')
ch = ch - 'a'; // a = 0
else
return 0; // unexpected character
2018-01-06 11:48:51 +08:00
2013-12-10 08:08:35 -05:00
tmp |= ch;
bits += 5;
if (bits >= 8)
{
if (ret >= outLen) return ret;
outBuf[ret] = tmp >> (bits - 8);
bits -= 8;
ret++;
}
tmp <<= 5;
}
return ret;
}
2014-08-21 12:32:03 -04:00
size_t ByteStreamToBase32 (const uint8_t * inBuf, size_t len, char * outBuf, size_t outLen)
{
size_t ret = 0, pos = 1;
int bits = 8, tmp = inBuf[0];
while (ret < outLen && (bits > 0 || pos < len))
2018-01-06 11:48:51 +08:00
{
2014-08-21 12:32:03 -04:00
if (bits < 5)
{
if (pos < len)
{
tmp <<= 8;
2018-01-06 12:01:44 +08:00
tmp |= inBuf[pos] & 0xFF;
2014-08-21 12:32:03 -04:00
pos++;
2018-01-06 12:01:44 +08:00
bits += 8;
2014-08-21 12:32:03 -04:00
}
else // last byte
{
tmp <<= (5 - bits);
2018-01-06 12:01:44 +08:00
bits = 5;
2014-08-21 12:32:03 -04:00
}
2018-01-06 11:48:51 +08:00
}
2014-08-21 12:32:03 -04:00
bits -= 5;
int ind = (tmp >> bits) & 0x1F;
outBuf[ret] = (ind < 26) ? (ind + 'a') : ((ind - 26) + '2');
ret++;
}
return ret;
}
2013-12-10 08:08:35 -05:00
}
}
2015-11-03 09:15:49 -05:00