Passed
Push — gh-pages ( 20c441...dd59e5 )
by
unknown
02:54 queued 01:00
created

GenericGFPoly::evaluateAt()   A

Complexity

Conditions 4
Paths 3

Size

Total Lines 15
Code Lines 6

Duplication

Lines 0
Ratio 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 4
eloc 6
nc 3
nop 1
dl 0
loc 15
rs 10
c 1
b 0
f 0
1
<?php
2
/**
3
 * Class GenericGFPoly
4
 *
5
 * @created      16.01.2021
6
 * @author       ZXing Authors
7
 * @author       Smiley <[email protected]>
8
 * @copyright    2021 Smiley
9
 * @license      Apache-2.0
10
 */
11
12
namespace chillerlan\QRCode\Common;
13
14
use InvalidArgumentException;
15
16
use function array_fill, array_slice, array_splice, count;
17
18
/**
19
 * <p>Represents a polynomial whose coefficients are elements of a GF.
20
 * Instances of this class are immutable.</p>
21
 *
22
 * <p>Much credit is due to William Rucklidge since portions of this code are an indirect
23
 * port of his C++ Reed-Solomon implementation.</p>
24
 *
25
 * @author Sean Owen
26
 */
27
final class GenericGFPoly{
28
29
	private array $coefficients;
30
31
	/**
32
	 * @param array      $coefficients array coefficients as ints representing elements of GF(size), arranged
33
	 *                                 from most significant (highest-power term) coefficient to least significant
34
	 * @param int|null   $degree
35
	 *
36
	 * @throws \InvalidArgumentException if argument is null or empty, or if leading coefficient is 0 and this is not a
37
	 *                                  constant polynomial (that is, it is not the monomial "0")
38
	 */
39
	public function __construct(array $coefficients, int $degree = null){
40
		$degree ??= 0;
41
42
		if(empty($coefficients)){
43
			throw new InvalidArgumentException('arg $coefficients is empty');
44
		}
45
46
		if($degree < 0){
47
			throw new InvalidArgumentException('negative degree');
48
		}
49
50
		$coefficientsLength = count($coefficients);
51
52
		// Leading term must be non-zero for anything except the constant polynomial "0"
53
		$firstNonZero = 0;
54
55
		while($firstNonZero < $coefficientsLength && $coefficients[$firstNonZero] === 0){
56
			$firstNonZero++;
57
		}
58
59
		if($firstNonZero === $coefficientsLength){
60
			$this->coefficients = [0];
61
		}
62
		else{
63
			$this->coefficients = array_fill(0, $coefficientsLength - $firstNonZero + $degree, 0);
64
65
			for($i = 0; $i < $coefficientsLength - $firstNonZero; $i++){
66
				$this->coefficients[$i] = $coefficients[$i + $firstNonZero];
67
			}
68
		}
69
	}
70
71
	/**
72
	 * @return int $coefficient of x^degree term in this polynomial
73
	 */
74
	public function getCoefficient(int $degree):int{
75
		return $this->coefficients[count($this->coefficients) - 1 - $degree];
76
	}
77
78
	/**
79
	 * @return int[]
80
	 */
81
	public function getCoefficients():array{
82
		return $this->coefficients;
83
	}
84
85
	/**
86
	 * @return int $degree of this polynomial
87
	 */
88
	public function getDegree():int{
89
		return count($this->coefficients) - 1;
90
	}
91
92
	/**
93
	 * @return bool true if this polynomial is the monomial "0"
94
	 */
95
	public function isZero():bool{
96
		return $this->coefficients[0] === 0;
97
	}
98
99
	/**
100
	 * @return int evaluation of this polynomial at a given point
101
	 */
102
	public function evaluateAt(int $a):int{
103
104
		if($a === 0){
105
			// Just return the x^0 coefficient
106
			return $this->getCoefficient(0);
107
		}
108
109
		$result = 0;
110
111
		foreach($this->coefficients as $c){
112
			// if $a === 1 just the sum of the coefficients
113
			$result = GF256::addOrSubtract(($a === 1 ? $result : GF256::multiply($a, $result)), $c);
114
		}
115
116
		return $result;
117
	}
118
119
	/**
120
	 *
121
	 */
122
	public function multiply(GenericGFPoly $other):self{
123
124
		if($this->isZero() || $other->isZero()){
125
			return new self([0]);
126
		}
127
128
		$product = array_fill(0, count($this->coefficients) + count($other->coefficients) - 1, 0);
129
130
		foreach($this->coefficients as $i => $aCoeff){
131
			foreach($other->coefficients as $j => $bCoeff){
132
				$product[$i + $j] ^= GF256::multiply($aCoeff, $bCoeff);
133
			}
134
		}
135
136
		return new self($product);
137
	}
138
139
	/**
140
	 * @return \chillerlan\QRCode\Common\GenericGFPoly[] [quotient, remainder]
141
	 */
142
	public function divide(GenericGFPoly $other):array{
143
144
		if($other->isZero()){
145
			throw new InvalidArgumentException('Division by 0');
146
		}
147
148
		$quotient  = new self([0]);
149
		$remainder = clone $this;
150
151
		$denominatorLeadingTerm        = $other->getCoefficient($other->getDegree());
152
		$inverseDenominatorLeadingTerm = GF256::inverse($denominatorLeadingTerm);
153
154
		while($remainder->getDegree() >= $other->getDegree() && !$remainder->isZero()){
155
			$scale     = GF256::multiply($remainder->getCoefficient($remainder->getDegree()), $inverseDenominatorLeadingTerm);
156
			$diff      = $remainder->getDegree() - $other->getDegree();
157
			$quotient  = $quotient->addOrSubtract(GF256::buildMonomial($diff, $scale));
158
			$remainder = $remainder->addOrSubtract($other->multiplyByMonomial($diff, $scale));
159
		}
160
161
		return [$quotient, $remainder];
162
163
	}
164
165
	/**
166
	 *
167
	 */
168
	public function multiplyInt(int $scalar):self{
169
170
		if($scalar === 0){
171
			return new self([0]);
172
		}
173
174
		if($scalar === 1){
175
			return $this;
176
		}
177
178
		$product = array_fill(0, count($this->coefficients), 0);
179
180
		foreach($this->coefficients as $i => $c){
181
			$product[$i] = GF256::multiply($c, $scalar);
182
		}
183
184
		return new self($product);
185
	}
186
187
	/**
188
	 *
189
	 */
190
	public function multiplyByMonomial(int $degree, int $coefficient):self{
191
192
		if($degree < 0){
193
			throw new InvalidArgumentException();
194
		}
195
196
		if($coefficient === 0){
197
			return new self([0]);
198
		}
199
200
		$product = array_fill(0, count($this->coefficients) + $degree, 0);
201
202
		foreach($this->coefficients as $i => $c){
203
			$product[$i] = GF256::multiply($c, $coefficient);
204
		}
205
206
		return new self($product);
207
	}
208
209
	/**
210
	 *
211
	 */
212
	public function mod(GenericGFPoly $other):self{
213
214
		if(count($this->coefficients) - count($other->coefficients) < 0){
215
			return $this;
216
		}
217
218
		$ratio = GF256::log($this->coefficients[0]) - GF256::log($other->coefficients[0]);
219
220
		foreach($other->coefficients as $i => $c){
221
			$this->coefficients[$i] ^= GF256::exp(GF256::log($c) + $ratio);
222
		}
223
224
		return (new self($this->coefficients))->mod($other);
225
	}
226
227
	/**
228
	 *
229
	 */
230
	public function addOrSubtract(GenericGFPoly $other):self{
231
232
		if($this->isZero()){
233
			return $other;
234
		}
235
236
		if($other->isZero()){
237
			return $this;
238
		}
239
240
		$smallerCoefficients = $this->coefficients;
241
		$largerCoefficients  = $other->coefficients;
242
243
		if(count($smallerCoefficients) > count($largerCoefficients)){
244
			$temp                = $smallerCoefficients;
245
			$smallerCoefficients = $largerCoefficients;
246
			$largerCoefficients  = $temp;
247
		}
248
249
		$sumDiff    = array_fill(0, count($largerCoefficients), 0);
250
		$lengthDiff = count($largerCoefficients) - count($smallerCoefficients);
251
		// Copy high-order terms only found in higher-degree polynomial's coefficients
252
		array_splice($sumDiff, 0, $lengthDiff, array_slice($largerCoefficients, 0, $lengthDiff));
253
254
		$countLargerCoefficients = count($largerCoefficients);
255
256
		for($i = $lengthDiff; $i < $countLargerCoefficients; $i++){
257
			$sumDiff[$i] = GF256::addOrSubtract($smallerCoefficients[$i - $lengthDiff], $largerCoefficients[$i]);
258
		}
259
260
		return new self($sumDiff);
261
	}
262
263
}
264