2017-10-11 18:48:27 +05:00
|
|
|
#include <nntpchan/base64.hpp>
|
2016-10-15 21:37:59 +05:00
|
|
|
|
2017-10-17 19:29:56 +05:00
|
|
|
// taken from i2pd
|
2016-10-15 21:37:59 +05:00
|
|
|
namespace i2p
|
|
|
|
{
|
|
|
|
namespace data
|
|
|
|
{
|
2017-10-24 17:51:21 +05:00
|
|
|
|
|
|
|
static const char T32[32] = {
|
|
|
|
'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', '2', '3', '4', '5', '6', '7',
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *GetBase32SubstitutionTable() { return T32; }
|
|
|
|
|
2017-10-17 19:29:56 +05:00
|
|
|
static void iT64Build(void);
|
|
|
|
|
|
|
|
/*
|
2018-05-06 17:10:20 +05:00
|
|
|
*
|
|
|
|
* BASE64 Substitution Table
|
|
|
|
* -------------------------
|
|
|
|
*
|
|
|
|
* Direct Substitution Table
|
|
|
|
*/
|
2017-10-17 19:29:56 +05:00
|
|
|
|
|
|
|
static const char T64[64] = {'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',
|
2017-10-24 17:51:21 +05:00
|
|
|
'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-', '~'};
|
|
|
|
|
|
|
|
const char *GetBase64SubstitutionTable() { return T64; }
|
2017-10-17 19:29:56 +05:00
|
|
|
|
|
|
|
/*
|
2018-05-06 17:10:20 +05:00
|
|
|
* Reverse Substitution Table (built in run time)
|
|
|
|
*/
|
2017-10-17 19:29:56 +05:00
|
|
|
|
|
|
|
static char iT64[256];
|
|
|
|
static int isFirstTime = 1;
|
|
|
|
|
|
|
|
/*
|
2018-05-06 17:10:20 +05:00
|
|
|
* Padding
|
|
|
|
*/
|
2017-10-17 19:29:56 +05:00
|
|
|
|
|
|
|
static char P64 = '=';
|
|
|
|
|
|
|
|
/*
|
2018-05-06 17:10:20 +05:00
|
|
|
*
|
|
|
|
* ByteStreamToBase64
|
|
|
|
* ------------------
|
|
|
|
*
|
|
|
|
* Converts binary encoded data to BASE64 format.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
size_t /* Number of bytes in the encoded buffer */
|
|
|
|
ByteStreamToBase64(const uint8_t *InBuffer, /* Input buffer, binary data */
|
|
|
|
size_t InCount, /* Number of bytes in the input buffer */
|
|
|
|
char *OutBuffer, /* output buffer */
|
|
|
|
size_t len /* length of output buffer */
|
|
|
|
)
|
2016-10-15 21:37:59 +05:00
|
|
|
|
2017-10-17 19:29:56 +05:00
|
|
|
{
|
|
|
|
unsigned char *ps;
|
|
|
|
unsigned char *pd;
|
|
|
|
unsigned char acc_1;
|
|
|
|
unsigned char acc_2;
|
|
|
|
int i;
|
|
|
|
int n;
|
|
|
|
int m;
|
|
|
|
size_t outCount;
|
|
|
|
|
|
|
|
ps = (unsigned char *)InBuffer;
|
|
|
|
n = InCount / 3;
|
|
|
|
m = InCount % 3;
|
|
|
|
if (!m)
|
|
|
|
outCount = 4 * n;
|
|
|
|
else
|
|
|
|
outCount = 4 * (n + 1);
|
|
|
|
if (outCount > len)
|
|
|
|
return 0;
|
|
|
|
pd = (unsigned char *)OutBuffer;
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
{
|
|
|
|
acc_1 = *ps++;
|
|
|
|
acc_2 = (acc_1 << 4) & 0x30;
|
|
|
|
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];
|
|
|
|
}
|
|
|
|
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++;
|
|
|
|
acc_2 = (acc_1 << 4) & 0x3f;
|
|
|
|
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;
|
|
|
|
}
|
2016-10-15 21:37:59 +05:00
|
|
|
|
2017-10-17 19:29:56 +05:00
|
|
|
return outCount;
|
2016-10-15 21:37:59 +05:00
|
|
|
}
|
|
|
|
|
2017-10-17 19:29:56 +05:00
|
|
|
/*
|
2018-05-06 17:10:20 +05:00
|
|
|
*
|
|
|
|
* 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 */
|
|
|
|
Base64ToByteStream(const char *InBuffer, /* BASE64 encoded buffer */
|
|
|
|
size_t InCount, /* Number of input bytes */
|
|
|
|
uint8_t *OutBuffer, /* output buffer length */
|
|
|
|
size_t len /* length of output buffer */
|
|
|
|
)
|
2016-10-15 21:37:59 +05:00
|
|
|
{
|
2017-10-17 19:29:56 +05:00
|
|
|
unsigned char *ps;
|
|
|
|
unsigned char *pd;
|
|
|
|
unsigned char acc_1;
|
|
|
|
unsigned char acc_2;
|
|
|
|
int i;
|
|
|
|
int n;
|
|
|
|
int m;
|
|
|
|
size_t outCount;
|
|
|
|
|
|
|
|
if (isFirstTime)
|
|
|
|
iT64Build();
|
|
|
|
n = InCount / 4;
|
|
|
|
m = InCount % 4;
|
|
|
|
if (InCount && !m)
|
|
|
|
outCount = 3 * n;
|
|
|
|
else
|
2016-10-15 21:37:59 +05:00
|
|
|
{
|
2017-10-17 19:29:56 +05:00
|
|
|
outCount = 0;
|
|
|
|
return 0;
|
2016-10-15 21:37:59 +05:00
|
|
|
}
|
|
|
|
|
2017-10-17 19:29:56 +05:00
|
|
|
ps = (unsigned char *)(InBuffer + InCount - 1);
|
|
|
|
while (*ps-- == P64)
|
|
|
|
outCount--;
|
|
|
|
ps = (unsigned char *)InBuffer;
|
|
|
|
|
|
|
|
if (outCount > len)
|
|
|
|
return -1;
|
|
|
|
pd = OutBuffer;
|
|
|
|
auto endOfOutBuffer = OutBuffer + outCount;
|
|
|
|
for (i = 0; i < n; i++)
|
2016-10-15 21:37:59 +05:00
|
|
|
{
|
2017-10-17 19:29:56 +05:00
|
|
|
acc_1 = iT64[*ps++];
|
|
|
|
acc_2 = iT64[*ps++];
|
|
|
|
acc_1 <<= 2;
|
|
|
|
acc_1 |= acc_2 >> 4;
|
|
|
|
*pd++ = acc_1;
|
|
|
|
if (pd >= endOfOutBuffer)
|
|
|
|
break;
|
|
|
|
|
|
|
|
acc_2 <<= 4;
|
|
|
|
acc_1 = iT64[*ps++];
|
|
|
|
acc_2 |= acc_1 >> 2;
|
|
|
|
*pd++ = acc_2;
|
|
|
|
if (pd >= endOfOutBuffer)
|
|
|
|
break;
|
|
|
|
|
|
|
|
acc_2 = iT64[*ps++];
|
|
|
|
acc_2 |= acc_1 << 6;
|
|
|
|
*pd++ = acc_2;
|
2016-10-15 21:37:59 +05:00
|
|
|
}
|
2017-10-17 19:29:56 +05:00
|
|
|
|
|
|
|
return outCount;
|
|
|
|
}
|
|
|
|
|
2017-10-24 17:51:21 +05:00
|
|
|
size_t Base64EncodingBufferSize(const size_t input_size)
|
2017-10-17 19:29:56 +05:00
|
|
|
{
|
|
|
|
auto d = div(input_size, 3);
|
|
|
|
if (d.rem)
|
|
|
|
d.quot++;
|
|
|
|
return 4 * d.quot;
|
|
|
|
}
|
|
|
|
|
2017-10-24 17:51:21 +05:00
|
|
|
size_t Base32EncodingBufferSize(const size_t input_size)
|
|
|
|
{
|
|
|
|
auto d = div(input_size, 5);
|
|
|
|
if (d.rem)
|
|
|
|
d.quot++;
|
|
|
|
return 8 * d.quot;
|
|
|
|
}
|
2017-10-17 19:29:56 +05:00
|
|
|
/*
|
2018-05-06 17:10:20 +05:00
|
|
|
*
|
|
|
|
* iT64
|
|
|
|
* ----
|
|
|
|
* Reverse table builder. P64 character is replaced with 0
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
2017-10-17 19:29:56 +05:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2017-10-24 17:51:21 +05:00
|
|
|
|
|
|
|
size_t Base32ToByteStream(const char *inBuf, size_t len, uint8_t *outBuf, size_t outLen)
|
|
|
|
{
|
|
|
|
int tmp = 0, bits = 0;
|
|
|
|
size_t ret = 0;
|
|
|
|
for (size_t i = 0; i < len; i++)
|
|
|
|
{
|
|
|
|
char ch = inBuf[i];
|
|
|
|
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
|
|
|
|
|
|
|
|
tmp |= ch;
|
|
|
|
bits += 5;
|
|
|
|
if (bits >= 8)
|
|
|
|
{
|
|
|
|
if (ret >= outLen)
|
|
|
|
return ret;
|
|
|
|
outBuf[ret] = tmp >> (bits - 8);
|
|
|
|
bits -= 8;
|
|
|
|
ret++;
|
|
|
|
}
|
|
|
|
tmp <<= 5;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
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))
|
|
|
|
{
|
|
|
|
if (bits < 5)
|
|
|
|
{
|
|
|
|
if (pos < len)
|
|
|
|
{
|
|
|
|
tmp <<= 8;
|
|
|
|
tmp |= inBuf[pos] & 0xFF;
|
|
|
|
pos++;
|
|
|
|
bits += 8;
|
|
|
|
}
|
|
|
|
else // last byte
|
|
|
|
{
|
|
|
|
tmp <<= (5 - bits);
|
|
|
|
bits = 5;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bits -= 5;
|
|
|
|
int ind = (tmp >> bits) & 0x1F;
|
|
|
|
outBuf[ret] = (ind < 26) ? (ind + 'a') : ((ind - 26) + '2');
|
|
|
|
ret++;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2017-10-17 19:29:56 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace nntpchan
|
|
|
|
{
|
|
|
|
std::string B64Encode(const uint8_t *data, const std::size_t l)
|
|
|
|
{
|
|
|
|
std::string out;
|
|
|
|
out.resize(i2p::data::Base64EncodingBufferSize(l));
|
|
|
|
i2p::data::ByteStreamToBase64(data, l, &out[0], out.size());
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool B64Decode(const std::string &data, std::vector<uint8_t> &out)
|
|
|
|
{
|
|
|
|
out.resize(data.size());
|
2017-10-24 17:51:21 +05:00
|
|
|
if (i2p::data::Base64ToByteStream(data.c_str(), data.size(), &out[0], out.size()))
|
|
|
|
{
|
|
|
|
out.shrink_to_fit();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string B32Encode(const uint8_t *data, const std::size_t l)
|
|
|
|
{
|
|
|
|
std::string out;
|
|
|
|
out.resize(i2p::data::Base32EncodingBufferSize(l));
|
|
|
|
i2p::data::ByteStreamToBase32(data, l, &out[0], out.size());
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool B32Decode(const std::string &data, std::vector<uint8_t> &out)
|
|
|
|
{
|
|
|
|
out.resize(data.size());
|
|
|
|
if (i2p::data::Base32ToByteStream(data.c_str(), data.size(), &out[0], out.size()))
|
|
|
|
{
|
|
|
|
out.shrink_to_fit();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2017-10-17 19:29:56 +05:00
|
|
|
}
|
2016-10-15 21:37:59 +05:00
|
|
|
}
|