35
35
/**
36
36
* A large number polynomial representation using sparse limbs of signed
37
37
* long (64-bit) values. Limb values will always fit within a long, so inputs
38
- * to multiplication must be less than 32 bits. All IntegerPolynomial
39
- * implementations allow at most one addition before multiplication. Additions
40
- * after that will result in an ArithmeticException.
38
+ * to multiplication must be less than 32 bits.
41
39
*
42
40
* The following element operations are branch-free for all subclasses:
43
41
*
@@ -553,16 +551,22 @@ public MutableElement mutable() {
553
551
return new MutableElement (limbs .clone (), numAdds );
554
552
}
555
553
556
- protected boolean isSummand () {
557
- return numAdds < maxAdds ;
558
- }
559
-
560
554
@ Override
561
555
public ImmutableElement add (IntegerModuloP genB ) {
562
556
assert IntegerPolynomial .this == genB .getField ();
563
- Element b = (Element ) genB ;
564
- if (!(isSummand () && b .isSummand ())) {
565
- throw new ArithmeticException ("Not a valid summand" );
557
+ Element b = (Element )genB ;
558
+
559
+ // Reduce if required.
560
+ // if (numAdds >= maxAdds) {
561
+ if (numAdds > 32 - bitsPerLimb ) {
562
+ reduce (limbs );
563
+ numAdds = 0 ;
564
+ }
565
+
566
+ // if (b.numAdds >= maxAdds) {
567
+ if (b .numAdds > 32 - bitsPerLimb ) {
568
+ reduce (b .limbs );
569
+ b .numAdds = 0 ;
566
570
}
567
571
568
572
long [] newLimbs = new long [limbs .length ];
@@ -597,7 +601,18 @@ protected void copyLow(long[] limbs, long[] out) {
597
601
@ Override
598
602
public ImmutableElement multiply (IntegerModuloP genB ) {
599
603
assert IntegerPolynomial .this == genB .getField ();
600
- Element b = (Element ) genB ;
604
+ Element b = (Element )genB ;
605
+
606
+ // Reduce if required.
607
+ if (numAdds > maxAdds ) {
608
+ reduce (limbs );
609
+ numAdds = 0 ;
610
+ }
611
+
612
+ if (b .numAdds > maxAdds ) {
613
+ reduce (b .limbs );
614
+ b .numAdds = 0 ;
615
+ }
601
616
602
617
long [] newLimbs = new long [limbs .length ];
603
618
mult (limbs , b .limbs , newLimbs );
@@ -606,24 +621,42 @@ public ImmutableElement multiply(IntegerModuloP genB) {
606
621
607
622
@ Override
608
623
public ImmutableElement square () {
624
+ // Reduce if required.
625
+ if (numAdds > maxAdds ) {
626
+ reduce (limbs );
627
+ numAdds = 0 ;
628
+ }
629
+
609
630
long [] newLimbs = new long [limbs .length ];
610
631
IntegerPolynomial .this .square (limbs , newLimbs );
611
632
return new ImmutableElement (newLimbs , 0 );
612
633
}
613
634
614
635
public void addModPowerTwo (IntegerModuloP arg , byte [] result ) {
615
636
assert IntegerPolynomial .this == arg .getField ();
616
- Element other = (Element ) arg ;
617
- if (!(isSummand () && other .isSummand ())) {
618
- throw new ArithmeticException ("Not a valid summand" );
637
+ Element other = (Element )arg ;
638
+
639
+ // Reduce if required.
640
+ if (numAdds > 32 - bitsPerLimb ) {
641
+ reduce (limbs );
642
+ numAdds = 0 ;
619
643
}
644
+
645
+ if (other .numAdds > 32 - bitsPerLimb ) {
646
+ reduce (other .limbs );
647
+ other .numAdds = 0 ;
648
+ }
649
+
620
650
addLimbsModPowerTwo (limbs , other .limbs , result );
621
651
}
622
652
623
653
public void asByteArray (byte [] result ) {
624
- if (!isSummand ()) {
625
- throw new ArithmeticException ("Not a valid summand" );
654
+ // Reduce if required.
655
+ if (numAdds != 0 ) {
656
+ reduce (limbs );
657
+ numAdds = 0 ;
626
658
}
659
+
627
660
limbsToByteArray (limbs , result );
628
661
}
629
662
@@ -698,15 +731,33 @@ public MutableElement setValue(ByteBuffer buf, int length,
698
731
@ Override
699
732
public MutableElement setProduct (IntegerModuloP genB ) {
700
733
assert IntegerPolynomial .this == genB .getField ();
701
- Element b = (Element ) genB ;
734
+ Element b = (Element )genB ;
735
+
736
+ // Reduce if required.
737
+ if (numAdds > maxAdds ) {
738
+ reduce (limbs );
739
+ numAdds = 0 ;
740
+ }
741
+
742
+ if (b .numAdds > maxAdds ) {
743
+ reduce (b .limbs );
744
+ b .numAdds = 0 ;
745
+ }
746
+
702
747
mult (limbs , b .limbs , limbs );
703
748
numAdds = 0 ;
704
749
return this ;
705
750
}
706
751
707
752
@ Override
708
753
public MutableElement setProduct (SmallValue v ) {
709
- int value = ((Limb ) v ).value ;
754
+ // Reduce if required.
755
+ if (numAdds > maxAdds ) {
756
+ reduce (limbs );
757
+ numAdds = 0 ;
758
+ }
759
+
760
+ int value = ((Limb )v ).value ;
710
761
multByInt (limbs , value );
711
762
numAdds = 0 ;
712
763
return this ;
@@ -715,9 +766,19 @@ public MutableElement setProduct(SmallValue v) {
715
766
@ Override
716
767
public MutableElement setSum (IntegerModuloP genB ) {
717
768
assert IntegerPolynomial .this == genB .getField ();
718
- Element b = (Element ) genB ;
719
- if (!(isSummand () && b .isSummand ())) {
720
- throw new ArithmeticException ("Not a valid summand" );
769
+ Element b = (Element )genB ;
770
+
771
+ // Reduce if required.
772
+ // if (numAdds >= maxAdds) {
773
+ if (numAdds > 32 - bitsPerLimb ) {
774
+ reduce (limbs );
775
+ numAdds = 0 ;
776
+ }
777
+
778
+ // if (b.numAdds >= maxAdds) {
779
+ if (b .numAdds > 32 - bitsPerLimb ) {
780
+ reduce (b .limbs );
781
+ b .numAdds = 0 ;
721
782
}
722
783
723
784
for (int i = 0 ; i < limbs .length ; i ++) {
@@ -731,9 +792,19 @@ public MutableElement setSum(IntegerModuloP genB) {
731
792
@ Override
732
793
public MutableElement setDifference (IntegerModuloP genB ) {
733
794
assert IntegerPolynomial .this == genB .getField ();
734
- Element b = (Element ) genB ;
735
- if (!(isSummand () && b .isSummand ())) {
736
- throw new ArithmeticException ("Not a valid summand" );
795
+ Element b = (Element )genB ;
796
+
797
+ // Reduce if required.
798
+ // if (numAdds >= maxAdds) {
799
+ if (numAdds > 32 - bitsPerLimb ) {
800
+ reduce (limbs );
801
+ numAdds = 0 ;
802
+ }
803
+
804
+ // if (b.numAdds >= maxAdds) {
805
+ if (b .numAdds > 32 - bitsPerLimb ) {
806
+ reduce (b .limbs );
807
+ b .numAdds = 0 ;
737
808
}
738
809
739
810
for (int i = 0 ; i < limbs .length ; i ++) {
@@ -746,6 +817,12 @@ public MutableElement setDifference(IntegerModuloP genB) {
746
817
747
818
@ Override
748
819
public MutableElement setSquare () {
820
+ // Reduce if required.
821
+ if (numAdds > maxAdds ) {
822
+ reduce (limbs );
823
+ numAdds = 0 ;
824
+ }
825
+
749
826
IntegerPolynomial .this .square (limbs , limbs );
750
827
numAdds = 0 ;
751
828
return this ;
@@ -758,13 +835,6 @@ public MutableElement setAdditiveInverse() {
758
835
}
759
836
return this ;
760
837
}
761
-
762
- @ Override
763
- public MutableElement setReduced () {
764
- reduce (limbs );
765
- numAdds = 0 ;
766
- return this ;
767
- }
768
838
}
769
839
770
840
class ImmutableElement extends Element implements ImmutableIntegerModuloP {
@@ -795,6 +865,5 @@ static class Limb implements SmallValue {
795
865
this .value = value ;
796
866
}
797
867
}
798
-
799
-
800
868
}
869
+
0 commit comments