| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- <?php
- /**
- * Class GenericGFPoly
- *
- * @filesource GenericGFPoly.php
- * @created 16.01.2021
- * @package chillerlan\QRCode\Common
- * @author ZXing Authors
- * @author Smiley <smiley@chillerlan.net>
- * @copyright 2021 Smiley
- * @license Apache-2.0
- */
- namespace chillerlan\QRCode\Common;
- use InvalidArgumentException;
- use function array_fill, array_slice, array_splice, count;
- /**
- * <p>Represents a polynomial whose coefficients are elements of a GF.
- * Instances of this class are immutable.</p>
- *
- * <p>Much credit is due to William Rucklidge since portions of this code are an indirect
- * port of his C++ Reed-Solomon implementation.</p>
- *
- * @author Sean Owen
- */
- final class GenericGFPoly{
- private array $coefficients;
- /**
- * @param array $coefficients array coefficients as ints representing elements of GF(size), arranged
- * from most significant (highest-power term) coefficient to least significant
- * @param int|null $degree
- *
- * @throws \InvalidArgumentException if argument is null or empty, or if leading coefficient is 0 and this is not a
- * constant polynomial (that is, it is not the monomial "0")
- */
- public function __construct(array $coefficients, int $degree = null){
- $degree ??= 0;
- if(empty($coefficients)){
- throw new InvalidArgumentException('arg $coefficients is empty');
- }
- if($degree < 0){
- throw new InvalidArgumentException('negative degree');
- }
- $coefficientsLength = count($coefficients);
- // Leading term must be non-zero for anything except the constant polynomial "0"
- $firstNonZero = 0;
- while($firstNonZero < $coefficientsLength && $coefficients[$firstNonZero] === 0){
- $firstNonZero++;
- }
- if($firstNonZero === $coefficientsLength){
- $this->coefficients = [0];
- }
- else{
- $this->coefficients = array_fill(0, $coefficientsLength - $firstNonZero + $degree, 0);
- for($i = 0; $i < $coefficientsLength - $firstNonZero; $i++){
- $this->coefficients[$i] = $coefficients[$i + $firstNonZero];
- }
- }
- }
- /**
- * @return int $coefficient of x^degree term in this polynomial
- */
- public function getCoefficient(int $degree):int{
- return $this->coefficients[count($this->coefficients) - 1 - $degree];
- }
- /**
- * @return int[]
- */
- public function getCoefficients():array{
- return $this->coefficients;
- }
- /**
- * @return int $degree of this polynomial
- */
- public function getDegree():int{
- return count($this->coefficients) - 1;
- }
- /**
- * @return bool true if this polynomial is the monomial "0"
- */
- public function isZero():bool{
- return $this->coefficients[0] === 0;
- }
- /**
- * @return int evaluation of this polynomial at a given point
- */
- public function evaluateAt(int $a):int{
- if($a === 0){
- // Just return the x^0 coefficient
- return $this->getCoefficient(0);
- }
- $result = 0;
- foreach($this->coefficients as $c){
- // if $a === 1 just the sum of the coefficients
- $result = GF256::addOrSubtract(($a === 1 ? $result : GF256::multiply($a, $result)), $c);
- }
- return $result;
- }
- /**
- * @param \chillerlan\QRCode\Common\GenericGFPoly $other
- *
- * @return \chillerlan\QRCode\Common\GenericGFPoly
- */
- public function multiply(GenericGFPoly $other):GenericGFPoly{
- if($this->isZero() || $other->isZero()){
- return new self([0]);
- }
- $product = array_fill(0, count($this->coefficients) + count($other->coefficients) - 1, 0);
- foreach($this->coefficients as $i => $aCoeff){
- foreach($other->coefficients as $j => $bCoeff){
- $product[$i + $j] ^= GF256::multiply($aCoeff, $bCoeff);
- }
- }
- return new self($product);
- }
- /**
- * @param \chillerlan\QRCode\Common\GenericGFPoly $other
- *
- * @return \chillerlan\QRCode\Common\GenericGFPoly[] [quotient, remainder]
- */
- public function divide(GenericGFPoly $other):array{
- if($other->isZero()){
- throw new InvalidArgumentException('Division by 0');
- }
- $quotient = new self([0]);
- $remainder = clone $this;
- $denominatorLeadingTerm = $other->getCoefficient($other->getDegree());
- $inverseDenominatorLeadingTerm = GF256::inverse($denominatorLeadingTerm);
- while($remainder->getDegree() >= $other->getDegree() && !$remainder->isZero()){
- $scale = GF256::multiply($remainder->getCoefficient($remainder->getDegree()), $inverseDenominatorLeadingTerm);
- $diff = $remainder->getDegree() - $other->getDegree();
- $quotient = $quotient->addOrSubtract(GF256::buildMonomial($diff, $scale));
- $remainder = $remainder->addOrSubtract($other->multiplyByMonomial($diff, $scale));
- }
- return [$quotient, $remainder];
- }
- /**
- * @param int $scalar
- *
- * @return \chillerlan\QRCode\Common\GenericGFPoly
- */
- public function multiplyInt(int $scalar):GenericGFPoly{
- if($scalar === 0){
- return new self([0]);
- }
- if($scalar === 1){
- return $this;
- }
- $product = array_fill(0, count($this->coefficients), 0);
- foreach($this->coefficients as $i => $c){
- $product[$i] = GF256::multiply($c, $scalar);
- }
- return new self($product);
- }
- /**
- * @param int $degree
- * @param int $coefficient
- *
- * @return \chillerlan\QRCode\Common\GenericGFPoly
- */
- public function multiplyByMonomial(int $degree, int $coefficient):GenericGFPoly{
- if($degree < 0){
- throw new InvalidArgumentException();
- }
- if($coefficient === 0){
- return new self([0]);
- }
- $product = array_fill(0, count($this->coefficients) + $degree, 0);
- foreach($this->coefficients as $i => $c){
- $product[$i] = GF256::multiply($c, $coefficient);
- }
- return new self($product);
- }
- /**
- * @param \chillerlan\QRCode\Common\GenericGFPoly $other
- *
- * @return \chillerlan\QRCode\Common\GenericGFPoly
- */
- public function mod(GenericGFPoly $other):GenericGFPoly{
- if(count($this->coefficients) - count($other->coefficients) < 0){
- return $this;
- }
- $ratio = GF256::log($this->coefficients[0]) - GF256::log($other->coefficients[0]);
- foreach($other->coefficients as $i => $c){
- $this->coefficients[$i] ^= GF256::exp(GF256::log($c) + $ratio);
- }
- return (new self($this->coefficients))->mod($other);
- }
- /**
- * @param \chillerlan\QRCode\Common\GenericGFPoly $other
- *
- * @return \chillerlan\QRCode\Common\GenericGFPoly
- */
- public function addOrSubtract(GenericGFPoly $other):GenericGFPoly{
- if($this->isZero()){
- return $other;
- }
- if($other->isZero()){
- return $this;
- }
- $smallerCoefficients = $this->coefficients;
- $largerCoefficients = $other->coefficients;
- if(count($smallerCoefficients) > count($largerCoefficients)){
- $temp = $smallerCoefficients;
- $smallerCoefficients = $largerCoefficients;
- $largerCoefficients = $temp;
- }
- $sumDiff = array_fill(0, count($largerCoefficients), 0);
- $lengthDiff = count($largerCoefficients) - count($smallerCoefficients);
- // Copy high-order terms only found in higher-degree polynomial's coefficients
- array_splice($sumDiff, 0, $lengthDiff, array_slice($largerCoefficients, 0, $lengthDiff));
- $countLargerCoefficients = count($largerCoefficients);
- for($i = $lengthDiff; $i < $countLargerCoefficients; $i++){
- $sumDiff[$i] = GF256::addOrSubtract($smallerCoefficients[$i - $lengthDiff], $largerCoefficients[$i]);
- }
- return new self($sumDiff);
- }
- }
|