libpurple/ciphers/aes.c

Wed, 21 Aug 2013 14:59:29 +0200

author
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
date
Wed, 21 Aug 2013 14:59:29 +0200
changeset 34304
faf0414a8b51
parent 34275
a6ddc53d1c84
permissions
-rw-r--r--

Fix most of libpurple warnings about -Wsign-compare

34182
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
1 /*
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
2 * purple
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
3 *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
4 * Purple is the legal property of its developers, whose names are too numerous
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
5 * to list here. Please refer to the COPYRIGHT file distributed with this
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
6 * source distribution.
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
7 *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
8 * This program is free software; you can redistribute it and/or modify
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
9 * it under the terms of the GNU General Public License as published by
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
10 * the Free Software Foundation; either version 2 of the License, or
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
11 * (at your option) any later version.
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
12 *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
13 * This program is distributed in the hope that it will be useful,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
16 * GNU General Public License for more details.
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
17 *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
18 * You should have received a copy of the GNU General Public License
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
19 * along with this program; if not, write to the Free Software
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
21 *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
22 * Written by Tomek Wasilczyk <tomkiewicz@cpw.pidgin.im>
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
23 */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
24
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
25 #include "internal.h"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
26 #include "cipher.h"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
27 #include "ciphers.h"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
28 #include "debug.h"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
29
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
30 #if defined(HAVE_GNUTLS)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
31 # define PURPLE_AES_USE_GNUTLS 1
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
32 # include <gnutls/gnutls.h>
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
33 # include <gnutls/crypto.h>
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
34 #elif defined(HAVE_NSS)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
35 # define PURPLE_AES_USE_NSS 1
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
36 # include <nss.h>
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
37 # include <pk11pub.h>
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
38 # include <prerror.h>
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
39 #else
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
40 # error "No GnuTLS or NSS support"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
41 #endif
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
42
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
43 /* 128bit */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
44 #define PURPLE_AES_BLOCK_SIZE 16
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
45
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
46 typedef struct
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
47 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
48 guchar iv[PURPLE_AES_BLOCK_SIZE];
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
49 guchar key[32];
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
50 guint key_size;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
51 gboolean failure;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
52 } AESContext;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
53
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
54 typedef gboolean (*purple_aes_crypt_func)(
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
55 const guchar *input, guchar *output, size_t len,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
56 guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32], guint key_size);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
57
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
58 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
59 purple_aes_init(PurpleCipherContext *context, void *extra)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
60 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
61 AESContext *ctx_data;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
62
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
63 ctx_data = g_new0(AESContext, 1);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
64 purple_cipher_context_set_data(context, ctx_data);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
65
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
66 purple_cipher_context_reset(context, extra);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
67 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
68
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
69 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
70 purple_aes_uninit(PurpleCipherContext *context)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
71 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
72 AESContext *ctx_data;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
73
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
74 purple_cipher_context_reset(context, NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
75
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
76 ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
77 g_free(ctx_data);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
78 purple_cipher_context_set_data(context, NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
79 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
80
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
81 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
82 purple_aes_reset(PurpleCipherContext *context, void *extra)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
83 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
84 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
85
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
86 g_return_if_fail(ctx_data != NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
87
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
88 memset(ctx_data->iv, 0, sizeof(ctx_data->iv));
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
89 memset(ctx_data->key, 0, sizeof(ctx_data->key));
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
90 ctx_data->key_size = 32; /* 256bit */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
91 ctx_data->failure = FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
92 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
93
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
94 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
95 purple_aes_set_option(PurpleCipherContext *context, const gchar *name,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
96 void *value)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
97 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
98 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
99
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
100 purple_debug_error("cipher-aes", "set_option not supported\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
101 ctx_data->failure = TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
102 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
103
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
104 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
105 purple_aes_set_iv(PurpleCipherContext *context, guchar *iv, size_t len)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
106 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
107 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
108
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
109 if ((len > 0 && iv == NULL) ||
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
110 (len != 0 && len != sizeof(ctx_data->iv))) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
111 purple_debug_error("cipher-aes", "invalid IV length\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
112 ctx_data->failure = TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
113 return;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
114 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
115
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
116 if (len == 0)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
117 memset(ctx_data->iv, 0, sizeof(ctx_data->iv));
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
118 else
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
119 memcpy(ctx_data->iv, iv, len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
120 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
121
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
122 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
123 purple_aes_set_key(PurpleCipherContext *context, const guchar *key, size_t len)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
124 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
125 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
126
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
127 if ((len > 0 && key == NULL) ||
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
128 (len != 0 && len != 16 && len != 24 && len != 32)) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
129 purple_debug_error("cipher-aes", "invalid key length\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
130 ctx_data->failure = TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
131 return;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
132 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
133
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
134 ctx_data->key_size = len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
135 memset(ctx_data->key, 0, sizeof(ctx_data->key));
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
136 if (len > 0)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
137 memcpy(ctx_data->key, key, len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
138 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
139
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
140 static guchar *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
141 purple_aes_pad_pkcs7(const guchar input[], size_t in_len, size_t *out_len)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
142 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
143 int padding_len, total_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
144 guchar *padded;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
145
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
146 g_return_val_if_fail(input != NULL, NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
147 g_return_val_if_fail(out_len != NULL, NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
148
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
149 padding_len = PURPLE_AES_BLOCK_SIZE - (in_len % PURPLE_AES_BLOCK_SIZE);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
150 total_len = in_len + padding_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
151 g_assert((total_len % PURPLE_AES_BLOCK_SIZE) == 0);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
152
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
153 padded = g_new(guchar, total_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
154 *out_len = total_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
155
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
156 memcpy(padded, input, in_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
157 memset(padded + in_len, padding_len, padding_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
158
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
159 return padded;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
160 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
161
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
162 static ssize_t
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
163 purple_aes_unpad_pkcs7(guchar input[], size_t in_len)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
164 {
34304
faf0414a8b51 Fix most of libpurple warnings about -Wsign-compare
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents: 34275
diff changeset
165 guchar padding_len, i;
34182
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
166 size_t out_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
167
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
168 g_return_val_if_fail(input != NULL, -1);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
169 g_return_val_if_fail(in_len > 0, -1);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
170
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
171 padding_len = input[in_len - 1];
34304
faf0414a8b51 Fix most of libpurple warnings about -Wsign-compare
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents: 34275
diff changeset
172 if (padding_len == 0 || padding_len > PURPLE_AES_BLOCK_SIZE ||
34182
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
173 padding_len > in_len) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
174 purple_debug_warning("cipher-aes",
34275
a6ddc53d1c84 Fix minor printf format warning.
Elliott Sales de Andrade <qulogic@pidgin.im>
parents: 34250
diff changeset
175 "Invalid padding length: %d (total %" G_GSIZE_FORMAT ") - "
34182
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
176 "most probably, the key was invalid\n",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
177 padding_len, in_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
178 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
179 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
180
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
181 out_len = in_len - padding_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
182 for (i = 0; i < padding_len; i++) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
183 if (input[out_len + i] != padding_len) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
184 purple_debug_warning("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
185 "Padding doesn't match at pos %d (found %02x, "
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
186 "expected %02x) - "
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
187 "most probably, the key was invalid\n",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
188 i, input[out_len + i], padding_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
189 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
190 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
191 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
192
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
193 memset(input + out_len, 0, padding_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
194 return out_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
195 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
196
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
197 #ifdef PURPLE_AES_USE_GNUTLS
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
198
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
199 static gnutls_cipher_hd_t
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
200 purple_aes_crypt_gnutls_init(guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32],
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
201 guint key_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
202 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
203 gnutls_cipher_hd_t handle;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
204 gnutls_cipher_algorithm_t algorithm;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
205 gnutls_datum_t key_info, iv_info;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
206 int ret;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
207
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
208 if (key_size == 16)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
209 algorithm = GNUTLS_CIPHER_AES_128_CBC;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
210 else if (key_size == 24)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
211 algorithm = GNUTLS_CIPHER_AES_192_CBC;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
212 else if (key_size == 32)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
213 algorithm = GNUTLS_CIPHER_AES_256_CBC;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
214 else
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
215 g_return_val_if_reached(NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
216
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
217 key_info.data = key;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
218 key_info.size = key_size;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
219
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
220 iv_info.data = iv;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
221 iv_info.size = PURPLE_AES_BLOCK_SIZE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
222
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
223 ret = gnutls_cipher_init(&handle, algorithm, &key_info, &iv_info);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
224 if (ret != 0) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
225 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
226 "gnutls_cipher_init failed: %d\n", ret);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
227 return NULL;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
228 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
229
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
230 return handle;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
231 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
232
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
233 static gboolean
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
234 purple_aes_encrypt_gnutls(const guchar *input, guchar *output, size_t len,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
235 guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32], guint key_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
236 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
237 gnutls_cipher_hd_t handle;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
238 int ret;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
239
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
240 handle = purple_aes_crypt_gnutls_init(iv, key, key_size);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
241 if (handle == NULL)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
242 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
243
34248
28750ae09061 fix warning with gnutls 2.12
Daniel Atallah <datallah@pidgin.im>
parents: 34182
diff changeset
244 ret = gnutls_cipher_encrypt2(handle, (void *)input, len, output, len);
34182
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
245 gnutls_cipher_deinit(handle);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
246
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
247 if (ret != 0) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
248 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
249 "gnutls_cipher_encrypt2 failed: %d\n", ret);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
250 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
251 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
252
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
253 return TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
254 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
255
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
256 static gboolean
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
257 purple_aes_decrypt_gnutls(const guchar *input, guchar *output, size_t len,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
258 guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32], guint key_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
259 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
260 gnutls_cipher_hd_t handle;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
261 int ret;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
262
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
263 handle = purple_aes_crypt_gnutls_init(iv, key, key_size);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
264 if (handle == NULL)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
265 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
266
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
267 ret = gnutls_cipher_decrypt2(handle, input, len, output, len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
268 gnutls_cipher_deinit(handle);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
269
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
270 if (ret != 0) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
271 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
272 "gnutls_cipher_decrypt2 failed: %d\n", ret);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
273 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
274 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
275
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
276 return TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
277 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
278
34250
1625de486023 Fix pre-processing directives to consistently only inclde code that will be used.
Daniel Atallah <datallah@pidgin.im>
parents: 34248
diff changeset
279 #elif defined(PURPLE_AES_USE_NSS)
34182
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
280
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
281 typedef struct
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
282 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
283 PK11SlotInfo *slot;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
284 PK11SymKey *sym_key;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
285 SECItem *sec_param;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
286 PK11Context *enc_context;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
287 } purple_aes_encrypt_nss_context;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
288
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
289 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
290 purple_aes_encrypt_nss_context_cleanup(purple_aes_encrypt_nss_context *ctx)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
291 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
292 g_return_if_fail(ctx != NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
293
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
294 if (ctx->enc_context != NULL)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
295 PK11_DestroyContext(ctx->enc_context, TRUE);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
296 if (ctx->sec_param != NULL)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
297 SECITEM_FreeItem(ctx->sec_param, TRUE);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
298 if (ctx->sym_key != NULL)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
299 PK11_FreeSymKey(ctx->sym_key);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
300 if (ctx->slot != NULL)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
301 PK11_FreeSlot(ctx->slot);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
302
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
303 memset(ctx, 0, sizeof(purple_aes_encrypt_nss_context));
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
304 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
305
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
306 static gboolean
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
307 purple_aes_crypt_nss(const guchar *input, guchar *output, size_t len,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
308 guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32], guint key_size,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
309 CK_ATTRIBUTE_TYPE operation)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
310 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
311 purple_aes_encrypt_nss_context ctx;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
312 CK_MECHANISM_TYPE cipher_mech = CKM_AES_CBC;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
313 SECItem key_item, iv_item;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
314 SECStatus ret;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
315 int outlen = 0;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
316 unsigned int outlen_tmp = 0;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
317
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
318 memset(&ctx, 0, sizeof(purple_aes_encrypt_nss_context));
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
319
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
320 if (NSS_NoDB_Init(NULL) != SECSuccess) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
321 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
322 "NSS_NoDB_Init failed: %d\n", PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
323 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
324 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
325
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
326 ctx.slot = PK11_GetBestSlot(cipher_mech, NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
327 if (ctx.slot == NULL) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
328 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
329 "PK11_GetBestSlot failed: %d\n", PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
330 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
331 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
332
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
333 key_item.type = siBuffer;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
334 key_item.data = key;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
335 key_item.len = key_size;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
336 ctx.sym_key = PK11_ImportSymKey(ctx.slot, cipher_mech,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
337 PK11_OriginUnwrap, CKA_ENCRYPT, &key_item, NULL);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
338 if (ctx.sym_key == NULL) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
339 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
340 "PK11_ImportSymKey failed: %d\n", PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
341 purple_aes_encrypt_nss_context_cleanup(&ctx);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
342 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
343 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
344
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
345 iv_item.type = siBuffer;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
346 iv_item.data = iv;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
347 iv_item.len = PURPLE_AES_BLOCK_SIZE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
348 ctx.sec_param = PK11_ParamFromIV(cipher_mech, &iv_item);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
349 if (ctx.sec_param == NULL) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
350 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
351 "PK11_ParamFromIV failed: %d\n", PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
352 purple_aes_encrypt_nss_context_cleanup(&ctx);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
353 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
354 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
355
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
356 ctx.enc_context = PK11_CreateContextBySymKey(cipher_mech, operation,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
357 ctx.sym_key, ctx.sec_param);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
358 if (ctx.enc_context == NULL) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
359 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
360 "PK11_CreateContextBySymKey failed: %d\n",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
361 PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
362 purple_aes_encrypt_nss_context_cleanup(&ctx);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
363 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
364 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
365
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
366 ret = PK11_CipherOp(ctx.enc_context, output, &outlen, len, input, len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
367 if (ret != SECSuccess) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
368 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
369 "PK11_CipherOp failed: %d\n", PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
370 purple_aes_encrypt_nss_context_cleanup(&ctx);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
371 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
372 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
373
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
374 ret = PK11_DigestFinal(ctx.enc_context, output + outlen, &outlen_tmp,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
375 len - outlen);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
376 if (ret != SECSuccess) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
377 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
378 "PK11_DigestFinal failed: %d\n", PR_GetError());
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
379 purple_aes_encrypt_nss_context_cleanup(&ctx);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
380 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
381 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
382
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
383 purple_aes_encrypt_nss_context_cleanup(&ctx);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
384
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
385 outlen += outlen_tmp;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
386 if (outlen != len) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
387 purple_debug_error("cipher-aes",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
388 "resulting length doesn't match: %d (expected: %d)\n",
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
389 outlen, len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
390 return FALSE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
391 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
392
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
393 return TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
394 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
395
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
396 static gboolean
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
397 purple_aes_encrypt_nss(const guchar *input, guchar *output, size_t len,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
398 guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32], guint key_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
399 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
400 return purple_aes_crypt_nss(input, output, len, iv, key, key_size,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
401 CKA_ENCRYPT);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
402 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
403
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
404 static gboolean
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
405 purple_aes_decrypt_nss(const guchar *input, guchar *output, size_t len,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
406 guchar iv[PURPLE_AES_BLOCK_SIZE], guchar key[32], guint key_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
407 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
408 return purple_aes_crypt_nss(input, output, len, iv, key, key_size,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
409 CKA_DECRYPT);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
410 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
411
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
412 #endif /* PURPLE_AES_USE_NSS */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
413
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
414 static ssize_t
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
415 purple_aes_encrypt(PurpleCipherContext *context, const guchar input[],
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
416 size_t in_len, guchar output[], size_t out_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
417 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
418 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
419 purple_aes_crypt_func encrypt_func;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
420 guchar *input_padded;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
421 size_t out_len = 0;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
422 gboolean succ;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
423
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
424 if (ctx_data->failure)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
425 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
426
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
427 input_padded = purple_aes_pad_pkcs7(input, in_len, &out_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
428
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
429 if (out_len > out_size) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
430 purple_debug_error("cipher-aes", "Output buffer too small\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
431 memset(input_padded, 0, out_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
432 g_free(input_padded);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
433 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
434 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
435
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
436 #if defined(PURPLE_AES_USE_GNUTLS)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
437 encrypt_func = purple_aes_encrypt_gnutls;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
438 #elif defined(PURPLE_AES_USE_NSS)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
439 encrypt_func = purple_aes_encrypt_nss;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
440 #else
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
441 # error "No matching encrypt_func"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
442 #endif
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
443
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
444 succ = encrypt_func(input_padded, output, out_len, ctx_data->iv,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
445 ctx_data->key, ctx_data->key_size);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
446
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
447 memset(input_padded, 0, out_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
448 g_free(input_padded);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
449
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
450 if (!succ) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
451 memset(output, 0, out_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
452 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
453 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
454
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
455 return out_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
456 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
457
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
458 static ssize_t
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
459 purple_aes_decrypt(PurpleCipherContext *context, const guchar input[],
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
460 size_t in_len, guchar output[], size_t out_size)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
461 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
462 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
463 purple_aes_crypt_func decrypt_func;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
464 gboolean succ;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
465 ssize_t out_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
466
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
467 if (ctx_data->failure)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
468 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
469
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
470 if (in_len > out_size) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
471 purple_debug_error("cipher-aes", "Output buffer too small\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
472 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
473 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
474
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
475 if ((in_len % PURPLE_AES_BLOCK_SIZE) != 0 || in_len == 0) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
476 purple_debug_error("cipher-aes", "Malformed data\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
477 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
478 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
479
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
480 #if defined(PURPLE_AES_USE_GNUTLS)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
481 decrypt_func = purple_aes_decrypt_gnutls;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
482 #elif defined(PURPLE_AES_USE_NSS)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
483 decrypt_func = purple_aes_decrypt_nss;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
484 #else
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
485 # error "No matching encrypt_func"
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
486 #endif
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
487
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
488 succ = decrypt_func(input, output, in_len, ctx_data->iv, ctx_data->key,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
489 ctx_data->key_size);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
490
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
491 if (!succ) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
492 memset(output, 0, in_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
493 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
494 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
495
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
496 out_len = purple_aes_unpad_pkcs7(output, in_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
497 if (out_len < 0) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
498 memset(output, 0, in_len);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
499 return -1;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
500 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
501
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
502 return out_len;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
503 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
504
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
505 static size_t
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
506 purple_aes_get_key_size(PurpleCipherContext *context)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
507 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
508 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
509
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
510 return ctx_data->key_size;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
511 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
512
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
513 static void
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
514 purple_aes_set_batch_mode(PurpleCipherContext *context,
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
515 PurpleCipherBatchMode mode)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
516 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
517 AESContext *ctx_data = purple_cipher_context_get_data(context);
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
518
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
519 if (mode == PURPLE_CIPHER_BATCH_MODE_CBC)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
520 return;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
521
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
522 purple_debug_error("cipher-aes", "unsupported batch mode\n");
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
523 ctx_data->failure = TRUE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
524 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
525
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
526 static PurpleCipherBatchMode
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
527 purple_aes_get_batch_mode(PurpleCipherContext *context)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
528 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
529 return PURPLE_CIPHER_BATCH_MODE_CBC;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
530 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
531
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
532 static size_t
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
533 purple_aes_get_block_size(PurpleCipherContext *context)
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
534 {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
535 return PURPLE_AES_BLOCK_SIZE;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
536 }
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
537
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
538 static PurpleCipherOps AESOps = {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
539 purple_aes_set_option, /* set_option */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
540 NULL, /* get_option */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
541 purple_aes_init, /* init */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
542 purple_aes_reset, /* reset */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
543 NULL, /* reset_state */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
544 purple_aes_uninit, /* uninit */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
545 purple_aes_set_iv, /* set_iv */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
546 NULL, /* append */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
547 NULL, /* digest */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
548 NULL, /* get_digest_size */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
549 purple_aes_encrypt, /* encrypt */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
550 purple_aes_decrypt, /* decrypt */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
551 NULL, /* set_salt */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
552 NULL, /* get_salt_size */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
553 purple_aes_set_key, /* set_key */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
554 purple_aes_get_key_size, /* get_key_size */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
555 purple_aes_set_batch_mode, /* set_batch_mode */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
556 purple_aes_get_batch_mode, /* get_batch_mode */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
557 purple_aes_get_block_size, /* get_block_size */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
558 NULL, NULL, NULL, NULL /* reserved */
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
559 };
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
560
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
561 PurpleCipherOps *
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
562 purple_aes_cipher_get_ops(void) {
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
563 return &AESOps;
087c0fbac984 AES support
Tomasz Wasilczyk <tomkiewicz@cpw.pidgin.im>
parents:
diff changeset
564 }

mercurial