Blame libfreerdp/crypto/base64.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Base64 Encoding & Decoding
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2011-2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *	 http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/crypto/crypto.h>
Packit 1fb8d4
Packit 1fb8d4
static const char base64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
Packit 1fb8d4
Packit 1fb8d4
char* crypto_base64_encode(const BYTE* data, int length)
Packit 1fb8d4
{
Packit 1fb8d4
	int c;
Packit 1fb8d4
	const BYTE* q;
Packit 1fb8d4
	char* p;
Packit 1fb8d4
	char* ret;
Packit 1fb8d4
	int i = 0;
Packit 1fb8d4
	int blocks;
Packit 1fb8d4
Packit 1fb8d4
	q = data;
Packit 1fb8d4
	p = ret = (char*) malloc((length + 3) * 4 / 3 + 1);
Packit 1fb8d4
	if (!p)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	/* b1, b2, b3 are input bytes
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * 0         1         2
Packit 1fb8d4
	 * 012345678901234567890123
Packit 1fb8d4
	 * |  b1  |  b2   |  b3   |
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * [ c1 ]     [  c3 ]
Packit 1fb8d4
	 *      [  c2 ]     [  c4 ]
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * c1, c2, c3, c4 are output chars in base64
Packit 1fb8d4
	 */
Packit 1fb8d4
Packit 1fb8d4
	/* first treat complete blocks */
Packit 1fb8d4
	blocks = length - (length % 3);
Packit 1fb8d4
	for (i = 0; i < blocks; i += 3, q += 3)
Packit 1fb8d4
	{
Packit 1fb8d4
		c = (q[0] << 16) + (q[1] << 8) + q[2];
Packit 1fb8d4
Packit 1fb8d4
		*p++ = base64[(c & 0x00FC0000) >> 18];
Packit 1fb8d4
		*p++ = base64[(c & 0x0003F000) >> 12];
Packit 1fb8d4
		*p++ = base64[(c & 0x00000FC0) >> 6];
Packit 1fb8d4
		*p++ = base64[c & 0x0000003F];
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* then remainder */
Packit 1fb8d4
	switch (length % 3)
Packit 1fb8d4
	{
Packit 1fb8d4
	case 0:
Packit 1fb8d4
		break;
Packit 1fb8d4
	case 1:
Packit 1fb8d4
		c = (q[0] << 16);
Packit 1fb8d4
		*p++ = base64[(c & 0x00FC0000) >> 18];
Packit 1fb8d4
		*p++ = base64[(c & 0x0003F000) >> 12];
Packit 1fb8d4
		*p++ = '=';
Packit 1fb8d4
		*p++ = '=';
Packit 1fb8d4
		break;
Packit 1fb8d4
	case 2:
Packit 1fb8d4
		c = (q[0] << 16) + (q[1] << 8);
Packit 1fb8d4
		*p++ = base64[(c & 0x00FC0000) >> 18];
Packit 1fb8d4
		*p++ = base64[(c & 0x0003F000) >> 12];
Packit 1fb8d4
		*p++ = base64[(c & 0x00000FC0) >> 6];
Packit 1fb8d4
		*p++ = '=';
Packit 1fb8d4
		break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	*p = 0;
Packit 1fb8d4
Packit 1fb8d4
	return ret;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static int base64_decode_char(char c)
Packit 1fb8d4
{
Packit 1fb8d4
	if (c >= 'A' && c <= 'Z')
Packit 1fb8d4
		return c - 'A';
Packit 1fb8d4
Packit 1fb8d4
	if (c >= 'a' && c <= 'z')
Packit 1fb8d4
		return c - 'a' + 26;
Packit 1fb8d4
Packit 1fb8d4
	if (c >= '0' && c <= '9')
Packit 1fb8d4
		return c - '0' + 52;
Packit 1fb8d4
Packit 1fb8d4
	if (c == '+')
Packit 1fb8d4
		return 62;
Packit 1fb8d4
Packit 1fb8d4
	if (c == '/')
Packit 1fb8d4
		return 63;
Packit 1fb8d4
Packit 1fb8d4
	if (c == '=')
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	return -1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void* base64_decode(const char* s, int length, int* data_len)
Packit 1fb8d4
{
Packit 1fb8d4
	int n[4];
Packit 1fb8d4
	BYTE* q;
Packit 1fb8d4
	BYTE* data;
Packit 1fb8d4
	int nBlocks, i, outputLen;
Packit 1fb8d4
Packit 1fb8d4
	if (length % 4)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	q = data = (BYTE*) malloc(length / 4 * 3 + 1);
Packit 1fb8d4
	if (!q)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	/* first treat complete blocks */
Packit 1fb8d4
	nBlocks = (length / 4);
Packit 1fb8d4
	outputLen = 0;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < nBlocks-1; i++, q += 3)
Packit 1fb8d4
	{
Packit 1fb8d4
		n[0] = base64_decode_char(*s++);
Packit 1fb8d4
		n[1] = base64_decode_char(*s++);
Packit 1fb8d4
		n[2] = base64_decode_char(*s++);
Packit 1fb8d4
		n[3] = base64_decode_char(*s++);
Packit 1fb8d4
Packit 1fb8d4
		if ((n[0] == -1) || (n[1] == -1) || (n[2] == -1) || (n[3] == -1))
Packit 1fb8d4
			goto out_free;
Packit 1fb8d4
Packit 1fb8d4
		q[0] = (n[0] << 2) + (n[1] >> 4);
Packit 1fb8d4
		q[1] = ((n[1] & 15) << 4) + (n[2] >> 2);
Packit 1fb8d4
		q[2] = ((n[2] & 3) << 6) + n[3];
Packit 1fb8d4
		outputLen += 3;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* treat last block */
Packit 1fb8d4
	n[0] = base64_decode_char(*s++);
Packit 1fb8d4
	n[1] = base64_decode_char(*s++);
Packit 1fb8d4
	if ((n[0] == -1) || (n[1] == -1))
Packit 1fb8d4
		goto out_free;
Packit 1fb8d4
Packit 1fb8d4
	n[2] = base64_decode_char(*s++);
Packit 1fb8d4
	n[3] = base64_decode_char(*s++);
Packit 1fb8d4
Packit 1fb8d4
	q[0] = (n[0] << 2) + (n[1] >> 4);
Packit 1fb8d4
	if (n[2] == -1)
Packit 1fb8d4
	{
Packit 1fb8d4
		/* XX== */
Packit 1fb8d4
		outputLen += 1;
Packit 1fb8d4
		if (n[3] != -1)
Packit 1fb8d4
			goto out_free;
Packit 1fb8d4
Packit 1fb8d4
		q[1] = ((n[1] & 15) << 4);
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (n[3] == -1)
Packit 1fb8d4
	{
Packit 1fb8d4
		/* yyy= */
Packit 1fb8d4
		outputLen += 2;
Packit 1fb8d4
		q[1] = ((n[1] & 15) << 4) + (n[2] >> 2);
Packit 1fb8d4
		q[2] = ((n[2] & 3) << 6);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		/* XXXX */
Packit 1fb8d4
		outputLen += 3;
Packit 1fb8d4
		q[0] = (n[0] << 2) + (n[1] >> 4);
Packit 1fb8d4
		q[1] = ((n[1] & 15) << 4) + (n[2] >> 2);
Packit 1fb8d4
		q[2] = ((n[2] & 3) << 6) + n[3];
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	*data_len = outputLen;
Packit 1fb8d4
	data[outputLen] = '\0';
Packit 1fb8d4
Packit 1fb8d4
	return data;
Packit 1fb8d4
out_free:
Packit 1fb8d4
	free(data);
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void crypto_base64_decode(const char* enc_data, int length, BYTE** dec_data, int* res_length)
Packit 1fb8d4
{
Packit 1fb8d4
	*dec_data = base64_decode(enc_data, length, res_length);
Packit 1fb8d4
}