bignum: make mpi_init() and mpi_free() accept a single argument
[tropicssl.git] / library / dhm.c
blob07327142bc36867fe6ccc6379c0f7f214124b8d1
1 /*
2 * Diffie-Hellman-Merkle key exchange
4 * Based on XySSL: Copyright (C) 2006-2008 Christophe Devine
6 * Copyright (C) 2009 Paul Bakker <polarssl_maintainer at polarssl dot org>
8 * All rights reserved.
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
12 * are met:
14 * * Redistributions of source code must retain the above copyright
15 * notice, this list of conditions and the following disclaimer.
16 * * Redistributions in binary form must reproduce the above copyright
17 * notice, this list of conditions and the following disclaimer in the
18 * documentation and/or other materials provided with the distribution.
19 * * Neither the names of PolarSSL or XySSL nor the names of its contributors
20 * may be used to endorse or promote products derived from this software
21 * without specific prior written permission.
23 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
25 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
26 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
27 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
28 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
29 * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
30 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
31 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
32 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
33 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36 * Reference:
38 * http://www.cacr.math.uwaterloo.ca/hac/ (chapter 12)
41 #include "tropicssl/config.h"
43 #if defined(TROPICSSL_DHM_C)
45 #include "tropicssl/dhm.h"
47 #include <string.h>
50 * helper to validate the mpi size and import it
52 static int dhm_read_bignum(mpi * X, unsigned char **p, const unsigned char *end)
54 int ret, n;
56 if (end - *p < 2)
57 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
59 n = ((*p)[0] << 8) | (*p)[1];
60 (*p) += 2;
62 if ((int)(end - *p) < n)
63 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
65 if ((ret = mpi_read_binary(X, *p, n)) != 0)
66 return (TROPICSSL_ERR_DHM_READ_PARAMS_FAILED | ret);
68 (*p) += n;
70 return (0);
74 * Parse the ServerKeyExchange parameters
76 int dhm_read_params(dhm_context * ctx, unsigned char **p, const unsigned char *end)
78 int ret, n;
80 memset(ctx, 0, sizeof(dhm_context));
82 if ((ret = dhm_read_bignum(&ctx->P, p, end)) != 0 ||
83 (ret = dhm_read_bignum(&ctx->G, p, end)) != 0 ||
84 (ret = dhm_read_bignum(&ctx->GY, p, end)) != 0)
85 return (ret);
87 ctx->len = mpi_size(&ctx->P);
89 if (end - *p < 2)
90 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
92 n = ((*p)[0] << 8) | (*p)[1];
93 (*p) += 2;
95 if (end != *p + n)
96 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
98 return (0);
102 * Setup and write the ServerKeyExchange parameters
104 int dhm_make_params(dhm_context * ctx, int x_size,
105 unsigned char *output, int *olen,
106 int (*f_rng) (void *), void *p_rng)
108 int i, ret, n, n1, n2, n3;
109 unsigned char *p;
112 * generate X and calculate GX = G^X mod P
114 n = x_size / sizeof(t_int);
115 MPI_CHK(mpi_grow(&ctx->X, n));
116 MPI_CHK(mpi_lset(&ctx->X, 0));
118 n = x_size >> 3;
119 p = (unsigned char *)ctx->X.p;
120 for (i = 0; i < n; i++)
121 *p++ = (unsigned char)f_rng(p_rng);
123 while (mpi_cmp_mpi(&ctx->X, &ctx->P) >= 0)
124 mpi_shift_r(&ctx->X, 1);
126 MPI_CHK(mpi_exp_mod(&ctx->GX, &ctx->G, &ctx->X, &ctx->P, &ctx->RP));
129 * export P, G, GX
131 #define DHM_MPI_EXPORT(X,n) \
132 MPI_CHK( mpi_write_binary( X, p + 2, n ) ); \
133 *p++ = (unsigned char)( n >> 8 ); \
134 *p++ = (unsigned char)( n ); p += n;
136 n1 = mpi_size(&ctx->P);
137 n2 = mpi_size(&ctx->G);
138 n3 = mpi_size(&ctx->GX);
140 p = output;
141 DHM_MPI_EXPORT(&ctx->P, n1);
142 DHM_MPI_EXPORT(&ctx->G, n2);
143 DHM_MPI_EXPORT(&ctx->GX, n3);
145 *olen = p - output;
147 ctx->len = n1;
149 cleanup:
151 if (ret != 0)
152 return (ret | TROPICSSL_ERR_DHM_MAKE_PARAMS_FAILED);
154 return (0);
158 * Import the peer's public value G^Y
160 int dhm_read_public(dhm_context * ctx, const unsigned char *input, int ilen)
162 int ret;
164 if (ctx == NULL || ilen < 1 || ilen > ctx->len)
165 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
167 if ((ret = mpi_read_binary(&ctx->GY, input, ilen)) != 0)
168 return (TROPICSSL_ERR_DHM_READ_PUBLIC_FAILED | ret);
170 return (0);
174 * Create own private value X and export G^X
176 int dhm_make_public(dhm_context * ctx, int x_size,
177 unsigned char *output, int olen,
178 int (*f_rng) (void *), void *p_rng)
180 int ret, i, n;
181 unsigned char *p;
183 if (ctx == NULL || olen < 1 || olen > ctx->len)
184 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
187 * generate X and calculate GX = G^X mod P
189 n = x_size / sizeof(t_int);
190 MPI_CHK(mpi_grow(&ctx->X, n));
191 MPI_CHK(mpi_lset(&ctx->X, 0));
193 n = x_size >> 3;
194 p = (unsigned char *)ctx->X.p;
195 for (i = 0; i < n; i++)
196 *p++ = (unsigned char)f_rng(p_rng);
198 while (mpi_cmp_mpi(&ctx->X, &ctx->P) >= 0)
199 mpi_shift_r(&ctx->X, 1);
201 MPI_CHK(mpi_exp_mod(&ctx->GX, &ctx->G, &ctx->X, &ctx->P, &ctx->RP));
203 MPI_CHK(mpi_write_binary(&ctx->GX, output, olen));
205 cleanup:
207 if (ret != 0)
208 return (TROPICSSL_ERR_DHM_MAKE_PUBLIC_FAILED | ret);
210 return (0);
214 * Derive and export the shared secret (G^Y)^X mod P
216 int dhm_calc_secret(dhm_context * ctx, unsigned char *output, int *olen)
218 int ret;
220 if (ctx == NULL || *olen < ctx->len)
221 return (TROPICSSL_ERR_DHM_BAD_INPUT_DATA);
223 MPI_CHK(mpi_exp_mod(&ctx->K, &ctx->GY, &ctx->X, &ctx->P, &ctx->RP));
225 *olen = mpi_size(&ctx->K);
227 MPI_CHK(mpi_write_binary(&ctx->K, output, *olen));
229 cleanup:
231 if (ret != 0)
232 return (TROPICSSL_ERR_DHM_CALC_SECRET_FAILED | ret);
234 return (0);
238 * Free the components of a DHM key
240 void dhm_free(dhm_context * ctx)
242 mpi_free(&ctx->RP); mpi_free(&ctx->K); mpi_free(&ctx->GY);
243 mpi_free(&ctx->GX); mpi_free(&ctx->X); mpi_free(&ctx->G);
244 mpi_free(&ctx->P);
247 #if defined(TROPICSSL_SELF_TEST)
250 * Checkup routine
252 int dhm_self_test(int verbose)
254 return (verbose++);
257 #endif
259 #endif