~zooko/cryptopp/trunk

1 by weidai
Initial revision
1
// gf2_32.cpp - written and placed in the public domain by Wei Dai
2
3
#include "pch.h"
4
#include "misc.h"
5
#include "gf2_32.h"
6
7
NAMESPACE_BEGIN(CryptoPP)
8
9
GF2_32::Element GF2_32::Multiply(Element a, Element b) const
10
{
11
	word32 table[4];
12
	table[0] = 0;
13
	table[1] = m_modulus;
14
	if (a & 0x80000000)
15
	{
16
		table[2] = m_modulus ^ (a<<1);
17
		table[3] = a<<1;
18
	}
19
	else
20
	{
21
		table[2] = a<<1;
22
		table[3] = m_modulus ^ (a<<1);
23
	}
24
284 by weidai
optimizations
25
#if CRYPTOPP_FAST_ROTATE(32)
1 by weidai
Initial revision
26
	b = rotrFixed(b, 30U);
27
	word32 result = table[b&2];
28
29
	for (int i=29; i>=0; --i)
30
	{
31
		b = rotlFixed(b, 1U);
32
		result = (result<<1) ^ table[(b&2) + (result>>31)];
33
	}
34
35
	return (b&1) ? result ^ a : result;
36
#else
37
	word32 result = table[(b>>30) & 2];
38
39
	for (int i=29; i>=0; --i)
40
		result = (result<<1) ^ table[((b>>i)&2) + (result>>31)];
41
42
	return (b&1) ? result ^ a : result;
43
#endif
44
}
45
46
GF2_32::Element GF2_32::MultiplicativeInverse(Element a) const
47
{
48
	if (a <= 1)		// 1 is a special case
49
		return a;
50
51
	// warning - don't try to adapt this algorithm for another situation
52
	word32 g0=m_modulus, g1=a, g2=a;
53
	word32 v0=0, v1=1, v2=1;
54
55
	assert(g1);
56
57
	while (!(g2 & 0x80000000))
58
	{
59
		g2 <<= 1;
60
		v2 <<= 1;
61
	}
62
63
	g2 <<= 1;
64
	v2 <<= 1;
65
66
	g0 ^= g2;
67
	v0 ^= v2;
68
69
	while (g0 != 1)
70
	{
71
		if (g1 < g0 || ((g0^g1) < g0 && (g0^g1) < g1))
72
		{
73
			assert(BitPrecision(g1) <= BitPrecision(g0));
74
			g2 = g1;
75
			v2 = v1;
76
		}
77
		else
78
		{
79
			assert(BitPrecision(g1) > BitPrecision(g0));
80
			g2 = g0; g0 = g1; g1 = g2;
81
			v2 = v0; v0 = v1; v1 = v2;
82
		}
83
84
		while ((g0^g2) >= g2)
85
		{
86
			assert(BitPrecision(g0) > BitPrecision(g2));
87
			g2 <<= 1;
88
			v2 <<= 1;
89
		}
90
91
		assert(BitPrecision(g0) == BitPrecision(g2));
92
		g0 ^= g2;
93
		v0 ^= v2;
94
	}
95
96
	return v0;
97
}
98
99
NAMESPACE_END