1
/* Copyright (C) 2003 MySQL AB
3
This program is free software; you can redistribute it and/or modify
4
it under the terms of the GNU General Public License as published by
5
the Free Software Foundation; version 2 of the License.
7
This program is distributed in the hope that it will be useful,
8
but WITHOUT ANY WARRANTY; without even the implied warranty of
9
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
GNU General Public License for more details.
12
You should have received a copy of the GNU General Public License
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
16
#include <drizzled/global.h>
17
#include <mystrings/m_string.h> /* strchr() */
18
#include <mystrings/m_ctype.h> /* my_isspace() */
19
#include <mysys/base64.h>
25
template <class T, class U>
26
inline void skip_space(T src, U i, const U size)
28
while (i < size && my_isspace(&my_charset_utf8_general_ci, * src))
42
static char base64_table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
43
"abcdefghijklmnopqrstuvwxyz"
48
base64_needed_encoded_length(int length_of_data)
51
nb_base64_chars= (length_of_data + 2) / 3 * 4;
54
nb_base64_chars + /* base64 char incl padding */
55
(nb_base64_chars - 1)/ 76 + /* newlines */
56
1; /* NUL termination of string */
61
base64_needed_decoded_length(int length_of_encoded_data)
63
return (int) ceil((double)(length_of_encoded_data * 3 / 4));
68
Encode a data as base64.
70
Note: We require that dst is pre-allocated to correct size.
71
See base64_needed_encoded_length().
75
base64_encode(const void *src, size_t src_len, char *dst)
77
const unsigned char *s= (const unsigned char*)src;
81
for (; i < src_len; len += 4)
103
*dst++= base64_table[(c >> 18) & 0x3f];
104
*dst++= base64_table[(c >> 12) & 0x3f];
106
if (i > (src_len + 1))
109
*dst++= base64_table[(c >> 6) & 0x3f];
114
*dst++= base64_table[(c >> 0) & 0x3f];
125
return (uint32_t) (strchr(base64_table, c) - base64_table);
130
Decode a base64 string
134
src Pointer to base64-encoded string
135
len Length of string at 'src'
136
dst Pointer to location where decoded data will be stored
137
end_ptr Pointer to variable that will refer to the character
138
after the end of the encoded data that were decoded. Can
143
The base64-encoded data in the range ['src','*end_ptr') will be
144
decoded and stored starting at 'dst'. The decoding will stop
145
after 'len' characters have been read from 'src', or when padding
146
occurs in the base64-encoded data. In either case: if 'end_ptr' is
147
non-null, '*end_ptr' will be set to point to the character after
148
the last read character, even in the presence of error.
151
We require that 'dst' is pre-allocated to correct size.
154
base64_needed_decoded_length().
157
Number of bytes written at 'dst' or -1 in case of failure
160
base64_decode(const char *src_base, size_t len,
161
void *dst, const char **end_ptr)
165
char *dst_base= (char *)dst;
166
char const *src= src_base;
175
skip_space(src, i, len);
181
skip_space(src, i, len);
187
skip_space(src, i, len);
193
src += 2; /* There should be two bytes padding */
202
skip_space(src, i, len);
208
src += 1; /* There should be one byte padding */
216
b[0]= (c >> 16) & 0xff;
217
b[1]= (c >> 8) & 0xff;
218
b[2]= (c >> 0) & 0xff;
220
for (j=0; j<3-mark; j++)
228
The variable 'i' is set to 'len' when padding has been read, so it
229
does not actually reflect the number of bytes read from 'src'.
231
return i != len ? -1 : d - dst_base;
237
#define require(b) { \
239
printf("Require failed at %s:%d\n", __FILE__, __LINE__); \
252
size_t needed_length;
254
for (i= 0; i < 500; i++)
256
/* Create source data */
257
const size_t src_len= rand() % 1000 + 1;
259
char * src= (char *) malloc(src_len);
265
for (j= 0; j<src_len; j++)
272
needed_length= base64_needed_encoded_length(src_len);
273
str= (char *) malloc(needed_length);
275
for (k= 0; k < needed_length; k++)
276
str[k]= 0xff; /* Fill memory to check correct NUL termination */
277
require(base64_encode(src, src_len, str) == 0);
278
require(needed_length == strlen(str) + 1);
281
dst= (char *) malloc(base64_needed_decoded_length(strlen(str)));
283
dst_len= base64_decode(str, strlen(str), dst, NULL);
284
require(dst_len == src_len);
286
if (memcmp(src, dst, src_len) != 0)
288
printf(" --------- src --------- --------- dst ---------\n");
289
for (k= 0; k<src_len; k+=8)
291
printf("%.4x ", (uint32_t) k);
292
for (l=0; l<8 && k+l<src_len; l++)
294
unsigned char c= src[k+l];
295
printf("%.2x ", (unsigned)c);
300
for (l=0; l<8 && k+l<dst_len; l++)
302
unsigned char c= dst[k+l];
303
printf("%.2x ", (unsigned)c);
307
printf("src length: %.8x, dst length: %.8x\n",
308
(uint32_t) src_len, (uint32_t) dst_len);
312
printf("Test succeeded.\n");