1 | <?php |
||||||
2 | declare(strict_types=1); |
||||||
3 | |||||||
4 | /*********************************************************************** |
||||||
5 | Copyright (C) 2012 Matyas Danter |
||||||
6 | |||||||
7 | Permission is hereby granted, free of charge, to any person obtaining |
||||||
8 | a copy of this software and associated documentation files (the "Software"), |
||||||
9 | to deal in the Software without restriction, including without limitation |
||||||
10 | the rights to use, copy, modify, merge, publish, distribute, sublicense, |
||||||
11 | and/or sell copies of the Software, and to permit persons to whom the |
||||||
12 | Software is furnished to do so, subject to the following conditions: |
||||||
13 | |||||||
14 | The above copyright notice and this permission notice shall be included |
||||||
15 | in all copies or substantial portions of the Software. |
||||||
16 | |||||||
17 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
||||||
18 | OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||||
19 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
||||||
20 | THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES |
||||||
21 | OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, |
||||||
22 | ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR |
||||||
23 | OTHER DEALINGS IN THE SOFTWARE. |
||||||
24 | *************************************************************************/ |
||||||
25 | namespace Mdanter\Ecc\Primitives; |
||||||
26 | |||||||
27 | use Mdanter\Ecc\Exception\PointRecoveryException; |
||||||
28 | use Mdanter\Ecc\Exception\SquareRootException; |
||||||
29 | use Mdanter\Ecc\Math\GmpMathInterface; |
||||||
30 | use Mdanter\Ecc\Math\ModularArithmetic; |
||||||
31 | use Mdanter\Ecc\Random\RandomNumberGeneratorInterface; |
||||||
32 | |||||||
33 | /** |
||||||
34 | * This class is a representation of an EC over a field modulo a prime number |
||||||
35 | * |
||||||
36 | * Important objectives for this class are: |
||||||
37 | * - Does the curve contain a point? |
||||||
38 | * - Comparison of two curves. |
||||||
39 | */ |
||||||
40 | class CurveFp implements CurveFpInterface |
||||||
41 | { |
||||||
42 | |||||||
43 | /** |
||||||
44 | * @var CurveParameters |
||||||
45 | */ |
||||||
46 | protected $parameters; |
||||||
47 | |||||||
48 | /** |
||||||
49 | * |
||||||
50 | * @var GmpMathInterface |
||||||
51 | */ |
||||||
52 | protected $adapter = null; |
||||||
53 | |||||||
54 | /** |
||||||
55 | * |
||||||
56 | * @var ModularArithmetic |
||||||
57 | */ |
||||||
58 | protected $modAdapter = null; |
||||||
59 | |||||||
60 | /** |
||||||
61 | * Constructor that sets up the instance variables. |
||||||
62 | * |
||||||
63 | * @param CurveParameters $parameters |
||||||
64 | * @param GmpMathInterface $adapter |
||||||
65 | */ |
||||||
66 | public function __construct(CurveParameters $parameters, GmpMathInterface $adapter) |
||||||
67 | { |
||||||
68 | $this->parameters = $parameters; |
||||||
69 | $this->adapter = $adapter; |
||||||
70 | $this->modAdapter = new ModularArithmetic($this->adapter, $this->parameters->getPrime()); |
||||||
71 | } |
||||||
72 | |||||||
73 | /** |
||||||
74 | * {@inheritDoc} |
||||||
75 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getModAdapter() |
||||||
76 | */ |
||||||
77 | public function getModAdapter(): ModularArithmetic |
||||||
78 | { |
||||||
79 | return $this->modAdapter; |
||||||
80 | } |
||||||
81 | |||||||
82 | /** |
||||||
83 | * {@inheritDoc} |
||||||
84 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getPoint() |
||||||
85 | */ |
||||||
86 | public function getPoint(\GMP $x, \GMP $y, \GMP $order = null): PointInterface |
||||||
87 | { |
||||||
88 | return new Point($this->adapter, $this, $x, $y, $order); |
||||||
89 | } |
||||||
90 | |||||||
91 | /** |
||||||
92 | * {@inheritDoc} |
||||||
93 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getInfinity() |
||||||
94 | */ |
||||||
95 | public function getInfinity(): PointInterface |
||||||
96 | { |
||||||
97 | return new Point($this->adapter, $this, gmp_init(0, 10), gmp_init(0, 10), null, true); |
||||||
0 ignored issues
–
show
Bug
introduced
by
![]() It seems like
gmp_init(0, 10) can also be of type resource ; however, parameter $y of Mdanter\Ecc\Primitives\Point::__construct() does only seem to accept GMP , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
98 | } |
||||||
99 | |||||||
100 | /** |
||||||
101 | * {@inheritDoc} |
||||||
102 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getGenerator() |
||||||
103 | */ |
||||||
104 | public function getGenerator(\GMP $x, \GMP $y, \GMP $order, RandomNumberGeneratorInterface $randomGenerator = null): GeneratorPoint |
||||||
105 | { |
||||||
106 | return new GeneratorPoint($this->adapter, $this, $x, $y, $order, $randomGenerator); |
||||||
107 | } |
||||||
108 | |||||||
109 | /** |
||||||
110 | * @param bool $wasOdd |
||||||
111 | * @param \GMP $xCoord |
||||||
112 | * @return \GMP |
||||||
113 | */ |
||||||
114 | public function recoverYfromX(bool $wasOdd, \GMP $xCoord): \GMP |
||||||
115 | { |
||||||
116 | $math = $this->adapter; |
||||||
117 | $prime = $this->getPrime(); |
||||||
118 | |||||||
119 | try { |
||||||
120 | $root = $this->adapter->getNumberTheory()->squareRootModP( |
||||||
121 | $math->add( |
||||||
122 | $math->add( |
||||||
123 | $this->modAdapter->pow($xCoord, gmp_init(3, 10)), |
||||||
0 ignored issues
–
show
It seems like
gmp_init(3, 10) can also be of type resource ; however, parameter $exponent of Mdanter\Ecc\Math\ModularArithmetic::pow() does only seem to accept GMP , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
124 | $math->mul($this->getA(), $xCoord) |
||||||
125 | ), |
||||||
126 | $this->getB() |
||||||
127 | ), |
||||||
128 | $prime |
||||||
129 | ); |
||||||
130 | } catch (SquareRootException $e) { |
||||||
131 | throw new PointRecoveryException("Failed to recover y coordinate for point", 0, $e); |
||||||
132 | } |
||||||
133 | |||||||
134 | if ($math->equals($math->mod($root, gmp_init(2, 10)), gmp_init(1)) === $wasOdd) { |
||||||
0 ignored issues
–
show
It seems like
gmp_init(1) can also be of type resource ; however, parameter $other of Mdanter\Ecc\Math\GmpMathInterface::equals() does only seem to accept GMP , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() It seems like
gmp_init(2, 10) can also be of type resource ; however, parameter $modulus of Mdanter\Ecc\Math\GmpMathInterface::mod() does only seem to accept GMP , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
135 | return $root; |
||||||
136 | } else { |
||||||
137 | return $math->sub($prime, $root); |
||||||
138 | } |
||||||
139 | } |
||||||
140 | /** |
||||||
141 | * {@inheritDoc} |
||||||
142 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::contains() |
||||||
143 | */ |
||||||
144 | public function contains(\GMP $x, \GMP $y): bool |
||||||
145 | { |
||||||
146 | $math = $this->adapter; |
||||||
147 | |||||||
148 | $eq_zero = $math->equals( |
||||||
149 | $this->modAdapter->sub( |
||||||
150 | $math->pow($y, 2), |
||||||
151 | $math->add( |
||||||
152 | $math->add( |
||||||
153 | $math->pow($x, 3), |
||||||
154 | $math->mul($this->getA(), $x) |
||||||
155 | ), |
||||||
156 | $this->getB() |
||||||
157 | ) |
||||||
158 | ), |
||||||
159 | gmp_init(0, 10) |
||||||
0 ignored issues
–
show
It seems like
gmp_init(0, 10) can also be of type resource ; however, parameter $other of Mdanter\Ecc\Math\GmpMathInterface::equals() does only seem to accept GMP , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
160 | ); |
||||||
161 | |||||||
162 | return $eq_zero; |
||||||
163 | } |
||||||
164 | |||||||
165 | /** |
||||||
166 | * {@inheritDoc} |
||||||
167 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getA() |
||||||
168 | */ |
||||||
169 | public function getA(): \GMP |
||||||
170 | { |
||||||
171 | return $this->parameters->getA(); |
||||||
172 | } |
||||||
173 | |||||||
174 | /** |
||||||
175 | * {@inheritDoc} |
||||||
176 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getB() |
||||||
177 | */ |
||||||
178 | public function getB(): \GMP |
||||||
179 | { |
||||||
180 | return $this->parameters->getB(); |
||||||
181 | } |
||||||
182 | |||||||
183 | /** |
||||||
184 | * {@inheritDoc} |
||||||
185 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::getPrime() |
||||||
186 | */ |
||||||
187 | public function getPrime(): \GMP |
||||||
188 | { |
||||||
189 | return $this->parameters->getPrime(); |
||||||
190 | } |
||||||
191 | |||||||
192 | /** |
||||||
193 | * @return int |
||||||
194 | */ |
||||||
195 | public function getSize(): int |
||||||
196 | { |
||||||
197 | return $this->parameters->getSize(); |
||||||
198 | } |
||||||
199 | |||||||
200 | /** |
||||||
201 | * {@inheritDoc} |
||||||
202 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::cmp() |
||||||
203 | */ |
||||||
204 | public function cmp(CurveFpInterface $other): int |
||||||
205 | { |
||||||
206 | $math = $this->adapter; |
||||||
207 | |||||||
208 | $equal = $math->equals($this->getA(), $other->getA()); |
||||||
209 | $equal &= $math->equals($this->getB(), $other->getB()); |
||||||
210 | $equal &= $math->equals($this->getPrime(), $other->getPrime()); |
||||||
211 | |||||||
212 | return ($equal) ? 0 : 1; |
||||||
213 | } |
||||||
214 | |||||||
215 | /** |
||||||
216 | * {@inheritDoc} |
||||||
217 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::equals() |
||||||
218 | */ |
||||||
219 | public function equals(CurveFpInterface $other): bool |
||||||
220 | { |
||||||
221 | return $this->cmp($other) == 0; |
||||||
222 | } |
||||||
223 | |||||||
224 | /** |
||||||
225 | * {@inheritDoc} |
||||||
226 | * @see \Mdanter\Ecc\Primitives\CurveFpInterface::__toString() |
||||||
227 | */ |
||||||
228 | public function __toString(): string |
||||||
229 | { |
||||||
230 | return 'curve(' . $this->adapter->toString($this->getA()) . ', ' . $this->adapter->toString($this->getB()) . ', ' . $this->adapter->toString($this->getPrime()) . ')'; |
||||||
231 | } |
||||||
232 | |||||||
233 | /** |
||||||
234 | * @return array |
||||||
235 | */ |
||||||
236 | public function __debugInfo() |
||||||
237 | { |
||||||
238 | return [ |
||||||
239 | 'a' => $this->adapter->toString($this->getA()), |
||||||
240 | 'b' => $this->adapter->toString($this->getB()), |
||||||
241 | 'prime' => $this->adapter->toString($this->getPrime()) |
||||||
242 | ]; |
||||||
243 | } |
||||||
244 | } |
||||||
245 |