 70cc374382
			
		
	
	70cc374382
	
	
	
		
			
			git-svn-id: svn://svn.h5l.se/heimdal/trunk/heimdal@21047 ec53bebd-3082-4978-b11e-865c3cabbd6b
		
			
				
	
	
		
			359 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			359 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2006 - 2007 Kungliga Tekniska Högskolan
 | |
|  * (Royal Institute of Technology, Stockholm, Sweden). 
 | |
|  * All rights reserved. 
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without 
 | |
|  * modification, are permitted provided that the following conditions 
 | |
|  * are met: 
 | |
|  *
 | |
|  * 1. Redistributions of source code must retain the above copyright 
 | |
|  *    notice, this list of conditions and the following disclaimer. 
 | |
|  *
 | |
|  * 2. Redistributions in binary form must reproduce the above copyright 
 | |
|  *    notice, this list of conditions and the following disclaimer in the 
 | |
|  *    documentation and/or other materials provided with the distribution. 
 | |
|  *
 | |
|  * 3. Neither the name of the Institute nor the names of its contributors 
 | |
|  *    may be used to endorse or promote products derived from this software 
 | |
|  *    without specific prior written permission. 
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND 
 | |
|  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 
 | |
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 
 | |
|  * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE 
 | |
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
 | |
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 
 | |
|  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 
 | |
|  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
 | |
|  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 
 | |
|  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 
 | |
|  * SUCH DAMAGE. 
 | |
|  */
 | |
| 
 | |
| #ifdef HAVE_CONFIG_H
 | |
| #include <config.h>
 | |
| #endif
 | |
| 
 | |
| #ifdef RCSID
 | |
| RCSID("$Id$");
 | |
| #endif
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <limits.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #include <bn.h>
 | |
| #include <rand.h>
 | |
| 
 | |
| static int
 | |
| set_get(unsigned long num)
 | |
| {
 | |
|     BIGNUM *bn;
 | |
| 
 | |
|     bn = BN_new();
 | |
|     if (!BN_set_word(bn, num))
 | |
| 	return 1;
 | |
| 
 | |
|     if (BN_get_word(bn) != num)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_free(bn);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| #define CHECK(x) do { ret += x; } while(0)
 | |
| 
 | |
| static int
 | |
| test_BN_set_get(void)
 | |
| {
 | |
|     int ret = 0;
 | |
|     CHECK(set_get(0));
 | |
|     CHECK(set_get(1));
 | |
|     CHECK(set_get(0xff));
 | |
|     CHECK(set_get(0x1ff));
 | |
|     CHECK(set_get(0xffff));
 | |
|     CHECK(set_get(0xf000));
 | |
|     CHECK(set_get(ULONG_MAX / 2));
 | |
|     CHECK(set_get(ULONG_MAX - 1));
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| static int
 | |
| test_BN_bit(void)
 | |
| {
 | |
|     BIGNUM *bn;
 | |
|     int ret = 0;
 | |
| 
 | |
|     bn = BN_new();
 | |
| 
 | |
|     /* test setting and getting of "word" */
 | |
|     if (!BN_set_word(bn, 1))
 | |
| 	return 1;
 | |
|     if (!BN_is_bit_set(bn, 0))
 | |
| 	ret += 1;
 | |
|     if (!BN_is_bit_set(bn, 0))
 | |
| 	ret += 1;
 | |
| 
 | |
|     if (!BN_set_word(bn, 2))
 | |
| 	return 1;
 | |
|     if (!BN_is_bit_set(bn, 1))
 | |
| 	ret += 1;
 | |
| 
 | |
|     if (!BN_set_word(bn, 3))
 | |
| 	return 1;
 | |
|     if (!BN_is_bit_set(bn, 0))
 | |
| 	ret += 1;
 | |
|     if (!BN_is_bit_set(bn, 1))
 | |
| 	ret += 1;
 | |
| 
 | |
|     if (!BN_set_word(bn, 0x100))
 | |
| 	return 1;
 | |
|     if (!BN_is_bit_set(bn, 8))
 | |
| 	ret += 1;
 | |
| 
 | |
|     if (!BN_set_word(bn, 0x1000))
 | |
| 	return 1;
 | |
|     if (!BN_is_bit_set(bn, 12))
 | |
| 	ret += 1;
 | |
| 
 | |
|     /* test bitsetting */
 | |
|     if (!BN_set_word(bn, 1))
 | |
| 	return 1;
 | |
|     if (!BN_set_bit(bn, 1))
 | |
| 	return 1;
 | |
|     if (BN_get_word(bn) != 3)
 | |
| 	return 1;
 | |
|     if (!BN_clear_bit(bn, 0))
 | |
| 	return 1;
 | |
|     if (BN_get_word(bn) != 2)
 | |
| 	return 1;
 | |
| 
 | |
|     /* test bitsetting past end of current end */
 | |
|     BN_clear(bn);
 | |
|     if (!BN_set_bit(bn, 12))
 | |
| 	return 1;
 | |
|     if (BN_get_word(bn) != 0x1000)
 | |
| 	return 1;
 | |
| 
 | |
|     /* test bit and byte counting functions */
 | |
|     if (BN_num_bits(bn) != 13)
 | |
| 	return 1;
 | |
|     if (BN_num_bytes(bn) != 2)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_free(bn);
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| struct ietest {
 | |
|     char *data;
 | |
|     size_t len;
 | |
|     unsigned long num;
 | |
| } ietests[] = {
 | |
|     { "", 0, 0 },
 | |
|     { "\x01", 1, 1 },
 | |
|     { "\x02", 1, 2 },
 | |
|     { "\xf2", 1, 0xf2 },
 | |
|     { "\x01\x00", 2, 256 }
 | |
| };
 | |
| 
 | |
| static int
 | |
| test_BN_import_export(void)
 | |
| {
 | |
|     BIGNUM *bn;
 | |
|     int ret = 0;
 | |
|     int i;
 | |
| 
 | |
|     bn = BN_new();
 | |
| 
 | |
|     for (i = 0; i < sizeof(ietests)/sizeof(ietests[0]); i++) {
 | |
| 	size_t len;
 | |
| 	unsigned char *p;
 | |
| 	if (!BN_bin2bn((unsigned char*)ietests[i].data, ietests[i].len, bn))
 | |
| 	    return 1;
 | |
| 	if (BN_get_word(bn) != ietests[i].num)
 | |
| 	    return 1;
 | |
| 	len = BN_num_bytes(bn);
 | |
| 	if (len != ietests[i].len)
 | |
| 	    return 1;
 | |
| 	p = malloc(len + 1);
 | |
| 	p[len] = 0xf4;
 | |
| 	BN_bn2bin(bn, p);
 | |
| 	if (p[len] != 0xf4)
 | |
| 	    return 1;
 | |
| 	if (memcmp(p, ietests[i].data, ietests[i].len) != 0)
 | |
| 	    return 1;
 | |
| 	free(p);
 | |
|     }
 | |
| 
 | |
|     BN_free(bn);
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| static int
 | |
| test_BN_uadd(void)
 | |
| {
 | |
|     BIGNUM *a, *b, *c;
 | |
|     char *p;
 | |
| 
 | |
|     a = BN_new();
 | |
|     b = BN_new();
 | |
|     c = BN_new();
 | |
| 
 | |
|     BN_set_word(a, 1);
 | |
|     BN_set_word(b, 2);
 | |
| 
 | |
|     BN_uadd(c, a, b);
 | |
| 
 | |
|     if (BN_get_word(c) != 3)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_uadd(c, b, a);
 | |
| 
 | |
|     if (BN_get_word(c) != 3)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_set_word(b, 0xff);
 | |
| 
 | |
|     BN_uadd(c, a, b);
 | |
|     if (BN_get_word(c) != 0x100)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_uadd(c, b, a);
 | |
|     if (BN_get_word(c) != 0x100)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_set_word(a, 0xff);
 | |
| 
 | |
|     BN_uadd(c, a, b);
 | |
|     if (BN_get_word(c) != 0x1fe)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_uadd(c, b, a);
 | |
|     if (BN_get_word(c) != 0x1fe)
 | |
| 	return 1;
 | |
| 
 | |
| 
 | |
|     BN_free(a);
 | |
|     BN_free(b);
 | |
| 
 | |
|     BN_hex2bn(&a, "50212A3B611D46642C825A16A354CE0FD4D85DD2");
 | |
|     BN_hex2bn(&b, "84B6C7E8D28ACA1614954DA");
 | |
| 
 | |
|     BN_uadd(c, b, a);
 | |
|     p = BN_bn2hex(c);
 | |
|     if (strcmp(p, "50212A3B611D466434CDC695307D7AB13621B2AC") != 0) {
 | |
| 	free(p);
 | |
| 	return 1;
 | |
|     }
 | |
|     free(p);
 | |
| 
 | |
|     BN_uadd(c, a, b);
 | |
|     p = BN_bn2hex(c);
 | |
|     if (strcmp(p, "50212A3B611D466434CDC695307D7AB13621B2AC") != 0) {
 | |
| 	free(p);
 | |
| 	return 1;
 | |
|     }
 | |
|     free(p);
 | |
| 
 | |
|     BN_free(a);
 | |
|     BN_free(b);
 | |
|     BN_free(c);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| test_BN_cmp(void)
 | |
| {
 | |
|     BIGNUM *a, *b;
 | |
| 
 | |
|     a = BN_new();
 | |
|     b = BN_new();
 | |
| 
 | |
|     if (!BN_set_word(a, 1))
 | |
| 	return 1;
 | |
|     if (!BN_set_word(b, 1))
 | |
| 	return 1;
 | |
| 
 | |
|     if (BN_cmp(a, b) != 0)
 | |
| 	return 1;
 | |
|     if (BN_cmp(b, a) != 0)
 | |
| 	return 1;
 | |
| 
 | |
|     if (!BN_set_word(b, 2))
 | |
| 	return 1;
 | |
| 
 | |
|     if (BN_cmp(a, b) >= 0)
 | |
| 	return 1;
 | |
|     if (BN_cmp(b, a) <= 0)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_set_negative(b, 1);
 | |
| 
 | |
|     if (BN_cmp(a, b) <= 0)
 | |
| 	return 1;
 | |
|     if (BN_cmp(b, a) >= 0)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_free(a);
 | |
|     BN_free(b);
 | |
| 
 | |
|     BN_hex2bn(&a, "50212A3B611D46642C825A16A354CE0FD4D85DD1");
 | |
|     BN_hex2bn(&b, "50212A3B611D46642C825A16A354CE0FD4D85DD2");
 | |
| 
 | |
|     if (BN_cmp(a, b) >= 0)
 | |
| 	return 1;
 | |
|     if (BN_cmp(b, a) <= 0)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_set_negative(b, 1);
 | |
| 
 | |
|     if (BN_cmp(a, b) <= 0)
 | |
| 	return 1;
 | |
|     if (BN_cmp(b, a) >= 0)
 | |
| 	return 1;
 | |
| 
 | |
|     BN_free(a);
 | |
|     BN_free(b);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| test_BN_rand(void)
 | |
| {
 | |
|     BIGNUM *bn;
 | |
| 
 | |
|     if (RAND_status() != 1)
 | |
| 	return 0;
 | |
| 
 | |
|     bn = BN_new();
 | |
|     if (bn == NULL)
 | |
| 	return 1;
 | |
| 
 | |
|     if (!BN_rand(bn, 1024, 0, 0))
 | |
| 	return 1;
 | |
| 
 | |
|     BN_free(bn);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int
 | |
| main(int argc, char **argv)
 | |
| {
 | |
|     int ret = 0;
 | |
| 
 | |
|     ret += test_BN_set_get();
 | |
|     ret += test_BN_bit();
 | |
|     ret += test_BN_import_export();
 | |
|     ret += test_BN_uadd();
 | |
|     ret += test_BN_cmp();
 | |
|     ret += test_BN_rand();
 | |
| 
 | |
|     return ret;
 | |
| }
 |