blob: 373aed856091e12efdfb48ce34a7a98b1885b6a3 [file] [log] [blame]
/*
* Copyright 2019 AT&T Intellectual Property
* Copyright 2019 Nokia
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* This source code is part of the near-RT RIC (RAN Intelligent Controller)
* platform project (RICP).
*/
//
// Created by adi ENZEL on 9/26/19.
//
#include "base64.h"
int base64::encode(const unsigned char *src, int srcLen, char unsigned *dst, long &dstLen) {
unsigned char *pos;
const unsigned char *end, *in;
if (dstLen <= 0 || srcLen <= 0) {
mdclog_write(MDCLOG_ERR, "source or destination length are 0. dst =%ld source = %d",
dstLen, srcLen);
return -1;
}
if (dstLen < (srcLen * 4 / 3)) {
mdclog_write(MDCLOG_ERR, "Destination size %ld must be at least 140 percent from source size %d",
dstLen, srcLen);
return -1;
}
if (dst == nullptr) {
mdclog_write(MDCLOG_ERR, "Destination must be allocated and freed by caller the function not allocate the memory");
return -1;
}
if (src == nullptr) {
mdclog_write(MDCLOG_ERR, "source is null pointer");
return -1;
}
end = src + srcLen;
in = src;
pos = dst;
while (end - in >= 3) {
*pos++ = base64_table[in[0] >> (unsigned int)2];
*pos++ = base64_table[((in[0] & 0x03) << (unsigned int)4) | (in[1] >> (unsigned int)4)];
*pos++ = base64_table[((in[1] & 0x0f) << (unsigned int)2) | (in[2] >> (unsigned int)6)];
*pos++ = base64_table[in[2] & (unsigned int)0x3f];
in += 3;
}
if (end - in) {
*pos++ = base64_table[in[0] >> (unsigned int)2];
if (end - in == 1) {
*pos++ = base64_table[(in[0] & 0x03) << (unsigned int)4];
*pos++ = '=';
} else {
*pos++ = base64_table[((in[0] & 0x03) << 4) | (in[1] >> (unsigned int)4)];
*pos++ = base64_table[(in[1] & 0x0f) << 2];
}
*pos++ = '=';
}
*pos = '\0';
dstLen = pos - dst;return 0;
}
int base64::decode(const unsigned char *src, int srcLen, char unsigned *dst, long dstLen) {
unsigned char inv_table[INVERSE_TABLE_SIZE];
memset(inv_table, 0x80, INVERSE_TABLE_SIZE);
for (ulong i = 0; i < sizeof(base64_table) - 1; i++) {
inv_table[base64_table[i]] = (unsigned char) i;
}
inv_table['='] = 0;
if (dstLen == 0 || dstLen < (int)(srcLen / 4 * 3)) {
mdclog_write(MDCLOG_ERR, "Destination size %ld can be up to 40 smaller then source size %d",
dstLen, srcLen);
return -1;
}
if (dst == nullptr) {
mdclog_write(MDCLOG_ERR, "Destination must be allocated and freed by caller the function not allocate the memory");
return -1;
}
unsigned char *pos, block[4], tmp;
long i;
int pad = 0;
size_t count = 0;
for (i = 0; i < srcLen; i++) {
if (inv_table[src[i]] != 0x80) {
count++;
}
}
if (count == 0 || count % 4)
return -1;
pos = dst;
count = 0;
for (i = 0; i < srcLen; i++) {
tmp = inv_table[src[i]];
if (tmp == 0x80) {
continue;
}
block[count] = tmp;
if (src[i] == '=') {
pad++;
}
count++;
if (count == 4) {
*pos++ = (block[0] << 2) | ((unsigned char)block[1] >> (unsigned int)4);
*pos++ = (block[1] << 4) | ((unsigned char)block[2] >> (unsigned int)2);
*pos++ = (block[2] << 6) | block[3];
count = 0;
if (pad) {
if (pad == 1) {
pos--;
}
else if (pad == 2) {
pos -= 2;
}
else {
return -1;
}
break;
}
}
}
dstLen = pos - dst;
return 0;
}