go.tools/go/types: fun with Hilbert: a const arithmetic test

This test generates a program that declares the constant
elements of an n*n Hilbert matrix, its inverse, and the
constant elements of the explicit product of the two.
The product should be the identity matrix; that check is
also expressed as a constant expression. Type-checking
verifies that the product is indeed the identity matrix
by asserting the result of the identity check (using the
assert built-in which is available for type-check tests).

The test is run for n = 5. Other values can be tested via
the -H flag, say: go test -run=Hilbert -H=100

The generated program can be written to a file for testing
the constant arithmetic of a compiler: go test -out=test.go

Because of the mathematically precise constant arithmetic
of go/types, this test should always succeed and is only
limited by the size of the matrix. It does run successfully
from n = 0 to values larger than 100.

The Hilbert matrix is famous for being ill-conditioned and
thus exposes arithmetic imprecision very quickly. The gc
compiler only produces a correct result for n = 0 (trivially),
and n = 1 at the moment.

R=adonovan, rsc
CC=golang-dev
https://golang.org/cl/35840043
This commit is contained in:
Robert Griesemer 2013-12-03 13:36:57 -08:00
parent 62a3fc7538
commit 14cf5b0a28
1 changed files with 232 additions and 0 deletions

232
go/types/hilbert_test.go Normal file
View File

@ -0,0 +1,232 @@
// Copyright 2013 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 types_test
import (
"bytes"
"flag"
"fmt"
"go/ast"
"go/parser"
"go/token"
"io/ioutil"
"testing"
. "code.google.com/p/go.tools/go/types"
)
var (
H = flag.Int("H", 5, "Hilbert matrix size")
out = flag.String("out", "", "write generated program to out")
)
func TestHilbert(t *testing.T) {
// generate source
src := program(*H, *out)
if *out != "" {
ioutil.WriteFile(*out, src, 0666)
return
}
// parse source
fset := token.NewFileSet()
f, err := parser.ParseFile(fset, "hilbert.go", src, 0)
if err != nil {
t.Fatal(err)
}
// type-check file
DefPredeclaredTestFuncs() // define assert built-in
_, err = Check(f.Name.Name, fset, []*ast.File{f})
if err != nil {
t.Fatal(err)
}
}
func program(n int, out string) []byte {
var g gen
g.p(`// WARNING: GENERATED FILE - DO NOT MODIFY MANUALLY!
// (To generate, in go/types directory: go test -run=Hilbert -H=%d -out=%q)
// This program tests arbitrary precision constant arithmetic
// by generating the constant elements of a Hilbert matrix H,
// its inverse I, and the product P = H*I. The product should
// be the identity matrix.
package main
func main() {
if !ok {
printProduct()
return
}
println("PASS")
}
`, n, out)
g.hilbert(n)
g.inverse(n)
g.product(n)
g.verify(n)
g.printProduct(n)
g.binomials(2*n - 1)
g.factorials(2*n - 1)
return g.Bytes()
}
type gen struct {
bytes.Buffer
}
func (g *gen) p(format string, args ...interface{}) {
fmt.Fprintf(&g.Buffer, format, args...)
}
func (g *gen) hilbert(n int) {
g.p(`// Hilbert matrix, n = %d
const (
`, n)
for i := 0; i < n; i++ {
g.p("\t")
for j := 0; j < n; j++ {
if j > 0 {
g.p(", ")
}
g.p("h%d_%d", i, j)
}
if i == 0 {
g.p(" = ")
for j := 0; j < n; j++ {
if j > 0 {
g.p(", ")
}
g.p("1.0/(iota + %d)", j+1)
}
}
g.p("\n")
}
g.p(")\n\n")
}
func (g *gen) inverse(n int) {
g.p(`// Inverse Hilbert matrix
const (
`)
for i := 0; i < n; i++ {
for j := 0; j < n; j++ {
s := "+"
if (i+j)&1 != 0 {
s = "-"
}
g.p("\ti%d_%d = %s%d * b%d_%d * b%d_%d * b%d_%d * b%d_%d\n",
i, j, s, i+j+1, n+i, n-j-1, n+j, n-i-1, i+j, i, i+j, i)
}
g.p("\n")
}
g.p(")\n\n")
}
func (g *gen) product(n int) {
g.p(`// Product matrix
const (
`)
for i := 0; i < n; i++ {
for j := 0; j < n; j++ {
g.p("\tp%d_%d = ", i, j)
for k := 0; k < n; k++ {
if k > 0 {
g.p(" + ")
}
g.p("h%d_%d*i%d_%d", i, k, k, j)
}
g.p("\n")
}
g.p("\n")
}
g.p(")\n\n")
}
func (g *gen) verify(n int) {
g.p(`// Verify that product is the identity matrix
const ok =
`)
for i := 0; i < n; i++ {
for j := 0; j < n; j++ {
if j == 0 {
g.p("\t")
} else {
g.p(" && ")
}
v := 0
if i == j {
v = 1
}
g.p("p%d_%d == %d", i, j, v)
}
g.p(" &&\n")
}
g.p("\ttrue\n\n")
// verify ok at type-check time
if *out == "" {
g.p("const _ = assert(ok)\n\n")
}
}
func (g *gen) printProduct(n int) {
g.p("func printProduct() {\n")
for i := 0; i < n; i++ {
g.p("\tprintln(")
for j := 0; j < n; j++ {
if j > 0 {
g.p(", ")
}
g.p("p%d_%d", i, j)
}
g.p(")\n")
}
g.p("}\n\n")
}
func (g *gen) mulRange(a, b int) {
if a > b {
g.p("1")
return
}
for i := a; i <= b; i++ {
if i > a {
g.p("*")
}
g.p("%d", i)
}
}
func (g *gen) binomials(n int) {
g.p(`// Binomials
const (
`)
for j := 0; j <= n; j++ {
if j > 0 {
g.p("\n")
}
for k := 0; k <= j; k++ {
g.p("\tb%d_%d = f%d / (f%d*f%d)\n", j, k, j, k, j-k)
}
}
g.p(")\n\n")
}
func (g *gen) factorials(n int) {
g.p(`// Factorials
const (
f0 = 1
f1 = 1
`)
for i := 2; i <= n; i++ {
g.p("\tf%d = f%d * %d\n", i, i-1, i)
}
g.p(")\n\n")
}