gloox  0.9.9.12
base64.cpp
1 /*
2  Copyright (c) 2005-2008 by Jakob Schroeter <js@camaya.net>
3  This file is part of the gloox library. http://camaya.net/gloox
4 
5  This software is distributed under a license. The full license
6  agreement can be found in the file LICENSE in this distribution.
7  This software may not be copied, modified, sold or distributed
8  other than expressed in the named license agreement.
9 
10  This software is distributed without any warranty.
11 */
12 
13 
14 #include "base64.h"
15 
16 namespace gloox
17 {
18 
19  const std::string Base64::alphabet64( "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/" );
20  const char Base64::pad = '=';
21 
22  const std::string::size_type Base64::np = std::string::npos;
23  const std::string::size_type Base64::table64[] =
24  {
25  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
26  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
27  np, np, np, 62, np, np, np, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, np, np,
28  np, np, np, np, np, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
29  15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, np, np, np, np, np, np, 26, 27, 28,
30  29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48,
31  49, 50, 51, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
32  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
33  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
34  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
35  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
36  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np,
37  np, np, np, np, np, np, np, np, np, np, np, np, np, np, np, np
38  };
39 
40  const std::string Base64::encode64( const std::string& input )
41  {
42  std::string encoded;
43  char c;
44  const std::string::size_type length = input.length();
45 
46  encoded.reserve( length * 2 );
47 
48  for( std::string::size_type i = 0; i < length; ++i )
49  {
50  c = ( input[i] >> 2 ) & 0x3f;
51  encoded.append( 1, alphabet64[c] );
52 
53  c = ( input[i] << 4 ) & 0x3f;
54  if( ++i < length )
55  c |= ( ( input[i] >> 4 ) & 0x0f );
56  encoded.append( 1, alphabet64[c] );
57 
58  if( i < length )
59  {
60  c = ( input[i] << 2 ) & 0x3c;
61  if( ++i < length )
62  c |= ( input[i] >> 6 ) & 0x03;
63  encoded.append( 1, alphabet64[c] );
64  }
65  else
66  {
67  ++i;
68  encoded.append( 1, pad );
69  }
70 
71  if( i < length )
72  {
73  c = input[i] & 0x3f;
74  encoded.append( 1, alphabet64[c] );
75  }
76  else
77  {
78  encoded.append( 1, pad );
79  }
80  }
81 
82  return encoded;
83  }
84 
85  const std::string Base64::decode64( const std::string& input )
86  {
87  char c, d;
88  const std::string::size_type length = input.length();
89  std::string decoded;
90 
91  decoded.reserve( length );
92 
93  for( std::string::size_type i = 0; i < length; ++i )
94  {
95  c = (char)table64[(unsigned char)input[i]];
96  ++i;
97  d = (char)table64[(unsigned char)input[i]];
98  c = ( c << 2 ) | ( ( d >> 4 ) & 0x3 );
99  decoded.append( 1, c );
100  if( ++i < length )
101  {
102  c = input[i];
103  if( pad == c )
104  break;
105 
106  c = (char)table64[(unsigned char)input[i]];
107  d = ( ( d << 4 ) & 0xf0 ) | ( ( c >> 2 ) & 0xf );
108  decoded.append( 1, d );
109  }
110 
111  if( ++i < length )
112  {
113  d = input[i];
114  if( pad == d )
115  break;
116 
117  d = (char)table64[(unsigned char)input[i]];
118  c = ( ( c << 6 ) & 0xc0 ) | d;
119  decoded.append( 1, c );
120  }
121  }
122 
123  return decoded;
124  }
125 
126 }