internal/radix51: test all combinations of argument and receiver aliasing

This commit is contained in:
Filippo Valsorda 2019-03-02 18:35:44 -05:00 committed by George Tankersley
parent 5758cbf76c
commit a68796f011
2 changed files with 132 additions and 2 deletions

View File

@ -0,0 +1,129 @@
// Copyright (c) 2019 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package radix51
import (
"testing"
"testing/quick"
)
func checkAliasingOneArg(f func(v, x *FieldElement) *FieldElement) func(v, x FieldElement) bool {
return func(v, x FieldElement) bool {
x1, v1 := x, x
// Calculate a reference f(x) without aliasing.
if out := f(&v, &x); out != &v {
return false
}
// Test aliasing the argument and the receiver.
if out := f(&v1, &v1); out != &v1 || v1 != v {
return false
}
// Ensure the arguments was not modified.
return x == x1
}
}
func checkAliasingTwoArgs(f func(v, x, y *FieldElement) *FieldElement) func(v, x, y FieldElement) bool {
return func(v, x, y FieldElement) bool {
x1, y1, v1 := x, y, FieldElement{}
// Calculate a reference f(x, y) without aliasing.
if out := f(&v, &x, &y); out != &v {
return false
}
// Test aliasing the first argument and the receiver.
v1 = x
if out := f(&v1, &v1, &y); out != &v1 || v1 != v {
return false
}
// Test aliasing the second argument and the receiver.
v1 = y
if out := f(&v1, &x, &v1); out != &v1 || v1 != v {
return false
}
// Calculate a reference f(x, x) without aliasing.
if out := f(&v, &x, &x); out != &v {
return false
}
// Test aliasing the first argument and the receiver.
v1 = x
if out := f(&v1, &v1, &x); out != &v1 || v1 != v {
return false
}
// Test aliasing the second argument and the receiver.
v1 = x
if out := f(&v1, &x, &v1); out != &v1 || v1 != v {
return false
}
// Test aliasing both arguments and the receiver.
v1 = x
if out := f(&v1, &v1, &v1); out != &v1 || v1 != v {
return false
}
// Ensure the arguments were not modified.
return x == x1 && y == y1
}
}
func TestAliasing(t *testing.T) {
type target struct {
name string
oneArgF func(v, x *FieldElement) *FieldElement
twoArgsF func(v, x, y *FieldElement) *FieldElement
}
for _, tt := range []target{
{name: "Abs", oneArgF: (*FieldElement).Abs},
{name: "Invert", oneArgF: (*FieldElement).Invert},
{name: "Neg", oneArgF: (*FieldElement).Neg},
{name: "Reduce", oneArgF: (*FieldElement).Reduce},
{name: "Set", oneArgF: (*FieldElement).Set},
{name: "Square", oneArgF: (*FieldElement).Square},
{
name: "CondNeg0",
oneArgF: func(v, x *FieldElement) *FieldElement {
return (*FieldElement).CondNeg(v, x, 0)
},
},
{
name: "CondNeg1",
oneArgF: func(v, x *FieldElement) *FieldElement {
return (*FieldElement).CondNeg(v, x, 1)
},
},
{name: "Mul", twoArgsF: (*FieldElement).Mul},
{name: "Add", twoArgsF: (*FieldElement).Add},
{name: "Sub", twoArgsF: (*FieldElement).Sub},
{
name: "Select0",
twoArgsF: func(v, x, y *FieldElement) *FieldElement {
return (*FieldElement).Select(v, x, y, 0)
},
},
{
name: "Select1",
twoArgsF: func(v, x, y *FieldElement) *FieldElement {
return (*FieldElement).Select(v, x, y, 1)
},
},
} {
var err error
switch {
case tt.oneArgF != nil:
quick.Check(checkAliasingOneArg(tt.oneArgF), quickCheckConfig)
case tt.twoArgsF != nil:
quick.Check(checkAliasingTwoArgs(tt.twoArgsF), quickCheckConfig)
}
if err != nil {
t.Errorf("%v: %v", tt.name, err)
}
}
}

View File

@ -14,8 +14,9 @@ import (
"testing/quick"
)
var quickCheckScaleFactor = uint8(3)
var quickCheckConfig = &quick.Config{MaxCount: (1 << (12 + quickCheckScaleFactor))}
// quickCheckConfig will make each quickcheck test run (256 * -quickchecks)
// times. The default value of -quickchecks is 100.
var quickCheckConfig = &quick.Config{MaxCountScale: 1 << 8}
func generateFieldElement(rand *mathrand.Rand) FieldElement {
// Generation strategy: generate random limb values bounded by