Blame libfreerdp/crypto/base64.c

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