123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276 |
|
/*******************************************************************************
copyright: Copyright (c) 2010 Ulrik Mikaelsson. All rights reserved
license: BSD style: $(LICENSE)
author: Ulrik Mikaelsson
standards: rfc3548, rfc4648
*******************************************************************************/
/*******************************************************************************
This module is used to decode and encode hex char[] arrays.
Example:
---
char[] blah = "Hello there, my name is Jeff.";
scope encodebuf = new char[allocateEncodeSize(cast(ubyte[])blah)];
char[] encoded = encode(cast(ubyte[])blah, encodebuf);
scope decodebuf = new ubyte[encoded.length];
if (cast(char[])decode(encoded, decodebuf) == "Hello there, my name is Jeff.")
Stdout("yay").newline;
---
Since v1.0
*******************************************************************************/
module tango.util.encode.Base16;
/*******************************************************************************
calculates and returns the size needed to encode the length of the
array passed.
Params:
data = An array that will be encoded
*******************************************************************************/
size_t allocateEncodeSize(const(ubyte[]) data)
{
return allocateEncodeSize(data.length);
}
/*******************************************************************************
calculates and returns the size needed to encode the length passed.
Params:
length = Number of bytes to be encoded
*******************************************************************************/
size_t allocateEncodeSize(size_t length)
{
return length*2;
}
/*******************************************************************************
encodes data and returns as an ASCII hex string.
Params:
data = what is to be encoded
buff = buffer large enough to hold encoded data
Example:
---
char[512] encodebuf;
char[] myEncodedString = encode(cast(ubyte[])"Hello, how are you today?", encodebuf);
Stdout(myEncodedString).newline; // 48656C6C6F2C20686F772061726520796F7520746F6461793F
---
*******************************************************************************/
char[] encode(const(ubyte[]) data, char[] buff)
in
{
assert(data);
assert(buff.length >= allocateEncodeSize(data));
}
body
{
size_t i;
foreach (ubyte j; data) {
buff[i++] = _encodeTable[j >> 4];
buff[i++] = _encodeTable[j & 0b0000_1111];
}
return buff[0..i];
}
/*******************************************************************************
encodes data and returns as an ASCII hex string.
Params:
data = what is to be encoded
Example:
---
char[] myEncodedString = encode(cast(ubyte[])"Hello, how are you today?");
Stdout(myEncodedString).newline; // 48656C6C6F2C20686F772061726520796F7520746F6461793F
---
*******************************************************************************/
char[] encode(const(ubyte[]) data)
in
{
assert(data);
}
body
{
auto rtn = new char[allocateEncodeSize(data)];
return encode(data, rtn);
}
/*******************************************************************************
decodes an ASCII hex string and returns it as ubyte[] data. Pre-allocates
the size of the array.
This decoder will ignore non-hex characters. So:
SGVsbG8sIGhvd
yBhcmUgeW91IH
RvZGF5Pw==
Is valid.
Params:
data = what is to be decoded
Example:
---
char[] myDecodedString = cast(char[])decode("48656C6C6F2C20686F772061726520796F7520746F6461793F");
Stdout(myDecodeString).newline; // Hello, how are you today?
---
*******************************************************************************/
ubyte[] decode(const(char[]) data)
in
{
assert(data);
}
body
{
auto rtn = new ubyte[data.length+1/2];
return decode(data, rtn);
}
/*******************************************************************************
decodes an ASCII hex string and returns it as ubyte[] data.
This decoder will ignore non-hex characters. So:
SGVsbG8sIGhvd
yBhcmUgeW91IH
RvZGF5Pw==
Is valid.
Params:
data = what is to be decoded
buff = a big enough array to hold the decoded data
Example:
---
ubyte[512] decodebuf;
char[] myDecodedString = cast(char[])decode("48656C6C6F2C20686F772061726520796F7520746F6461793F", decodebuf);
Stdout(myDecodeString).newline; // Hello, how are you today?
---
*******************************************************************************/
ubyte[] decode(const(char[]) data, ubyte[] buff)
in
{
assert(data);
}
body
{
bool even=true;
size_t i;
foreach (c; data) {
auto val = _decodeTable[c];
if (val & 0b1000_0000)
continue;
if (even) {
buff[i] = cast(ubyte) (val << 4); // Store val in high for bits
} else {
buff[i] |= val; // OR-in low 4 bits,
i += 1; // and move on to next
}
even = !even; // Switch mode for next iteration
}
assert(even, "Non-even amount of hex characters in input.");
return buff[0..i];
}
debug (UnitTest)
{
unittest
{
immutable immutable(char)[][] testRaw = [
"",
"A",
"AB",
"BAC",
"BACD",
"Hello, how are you today?",
"AbCdEfGhIjKlMnOpQrStUvXyZ",
];
immutable immutable(char)[][] testEnc = [
"",
"41",
"4142",
"424143",
"42414344",
"48656C6C6F2C20686F772061726520796F7520746F6461793F",
"4162436445664768496A4B6C4D6E4F7051725374557658795A",
];
for (size_t i; i < testRaw.length; i++) {
auto resultChars = encode(cast(ubyte[])testRaw[i]);
assert(resultChars == testEnc[i],
testRaw[i]~": ("~resultChars~") != ("~testEnc[i]~")");
auto resultBytes = decode(testEnc[i]);
assert(resultBytes == cast(ubyte[])testRaw[i],
testEnc[i]~": ("~cast(char[])resultBytes~") != ("~testRaw[i]~")");
}
}
}
private:
/*
Static immutable tables used for fast lookups to
encode and decode data.
*/
immutable ubyte hex_PAD = '=';
immutable char[] _encodeTable = "0123456789ABCDEF";
immutable ubyte[] _decodeTable = [
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0x00,0x01,0x02,0x03, 0x04,0x05,0x06,0x07, 0x08,0x09,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0x0A,0x0B,0x0C, 0x0D,0x0E,0x0F,0x1F, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0x0A,0x0B,0x0C, 0x0D,0x0E,0x0F,0x1F, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF, 0xFF,0xFF,0xFF,0xFF,
];
|