2d7f034871e0b478b0758e34c079d6ed8b42e094
[metze/wireshark/wip.git] / epan / base64.c
1 /* base64.c
2  * Base-64 conversion
3  *
4  * $Id$
5  *
6  * Ethereal - Network traffic analyzer
7  * By Gerald Combs <gerald@ethereal.com>
8  * Copyright 1998 Gerald Combs
9  *
10  * This program is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU General Public License
12  * as published by the Free Software Foundation; either version 2
13  * of the License, or (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
23  */
24
25 #ifdef HAVE_CONFIG_H
26 # include "config.h"
27 #endif
28
29 #include <string.h>
30 #include "base64.h"
31
32 /* Decode a base64 string in-place - simple and slow algorithm.
33    Return length of result. Taken from rproxy/librsync/base64.c by
34    Andrew Tridgell. */
35
36 size_t epan_base64_decode(char *s)
37 {
38         static const char b64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
39         int bit_offset, byte_offset, idx, i, n;
40         unsigned char *d = (unsigned char *)s;
41         char *p;
42
43         n=i=0;
44
45         while (*s && (p=strchr(b64, *s))) {
46                 idx = (int)(p - b64);
47                 byte_offset = (i*6)/8;
48                 bit_offset = (i*6)%8;
49                 d[byte_offset] &= ~((1<<(8-bit_offset))-1);
50                 if (bit_offset < 3) {
51                         d[byte_offset] |= (idx << (2-bit_offset));
52                         n = byte_offset+1;
53                 } else {
54                         d[byte_offset] |= (idx >> (bit_offset-2));
55                         d[byte_offset+1] = 0;
56                         d[byte_offset+1] |= (idx << (8-(bit_offset-2))) & 0xFF;
57                         n = byte_offset+2;
58                 }
59                 s++; i++;
60         }
61
62         return n;
63 }