summaryrefslogtreecommitdiff
path: root/isl_gmp.c
blob: 2dbce62ef95a83e7662457b8a329bed408a99410 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#include <stdio.h>
/*
 * Copyright 2008-2009 Katholieke Universiteit Leuven
 *
 * Use of this software is governed by the MIT license
 *
 * Written by Sven Verdoolaege, K.U.Leuven, Departement
 * Computerwetenschappen, Celestijnenlaan 200A, B-3001 Leuven, Belgium
 */

#include <isl_int.h>

uint32_t isl_gmp_hash(mpz_t v, uint32_t hash)
{
	int sa = v[0]._mp_size;
	int abs_sa = sa < 0 ? -sa : sa;
	unsigned char *data = (unsigned char *)v[0]._mp_d;
	unsigned char *end = data + abs_sa * sizeof(v[0]._mp_d[0]);

	if (sa < 0)
		isl_hash_byte(hash, 0xFF);
	for (; data < end; ++data)
		isl_hash_byte(hash, *data);
	return hash;
}

/* This function tries to produce outputs that do not depend on
 * the version of GMP that is being used.
 *
 * In particular, when computing the extended gcd of -1 and 9,
 * some versions will produce
 *
 *	1 = -1 * -1 + 0 * 9
 *
 * while other versions will produce
 *
 *	1 = 8 * -1 + 1 * 9
 *
 * If configure detects that we are in the former case, then
 * mpz_gcdext will be called directly.  Otherwise, this function
 * is called and then we try to mimic the behavior of the other versions.
 */
void isl_gmp_gcdext(mpz_t G, mpz_t S, mpz_t T, mpz_t A, mpz_t B)
{
	if (mpz_divisible_p(B, A)) {
		mpz_set_si(S, mpz_sgn(A));
		mpz_set_si(T, 0);
		mpz_abs(G, A);
		return;
	}
	if (mpz_divisible_p(A, B)) {
		mpz_set_si(S, 0);
		mpz_set_si(T, mpz_sgn(B));
		mpz_abs(G, B);
		return;
	}
	mpz_gcdext(G, S, T, A, B);
}