OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [gnu-dev/] [or1k-gcc/] [libgo/] [go/] [math/] [big/] [hilbert_test.go] - Rev 833

Go to most recent revision | Compare with Previous | Blame | View Log

// Copyright 2009 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.

// A little test program and benchmark for rational arithmetics.
// Computes a Hilbert matrix, its inverse, multiplies them
// and verifies that the product is the identity matrix.

package big

import (
        "fmt"
        "testing"
)

type matrix struct {
        n, m int
        a    []*Rat
}

func (a *matrix) at(i, j int) *Rat {
        if !(0 <= i && i < a.n && 0 <= j && j < a.m) {
                panic("index out of range")
        }
        return a.a[i*a.m+j]
}

func (a *matrix) set(i, j int, x *Rat) {
        if !(0 <= i && i < a.n && 0 <= j && j < a.m) {
                panic("index out of range")
        }
        a.a[i*a.m+j] = x
}

func newMatrix(n, m int) *matrix {
        if !(0 <= n && 0 <= m) {
                panic("illegal matrix")
        }
        a := new(matrix)
        a.n = n
        a.m = m
        a.a = make([]*Rat, n*m)
        return a
}

func newUnit(n int) *matrix {
        a := newMatrix(n, n)
        for i := 0; i < n; i++ {
                for j := 0; j < n; j++ {
                        x := NewRat(0, 1)
                        if i == j {
                                x.SetInt64(1)
                        }
                        a.set(i, j, x)
                }
        }
        return a
}

func newHilbert(n int) *matrix {
        a := newMatrix(n, n)
        for i := 0; i < n; i++ {
                for j := 0; j < n; j++ {
                        a.set(i, j, NewRat(1, int64(i+j+1)))
                }
        }
        return a
}

func newInverseHilbert(n int) *matrix {
        a := newMatrix(n, n)
        for i := 0; i < n; i++ {
                for j := 0; j < n; j++ {
                        x1 := new(Rat).SetInt64(int64(i + j + 1))
                        x2 := new(Rat).SetInt(new(Int).Binomial(int64(n+i), int64(n-j-1)))
                        x3 := new(Rat).SetInt(new(Int).Binomial(int64(n+j), int64(n-i-1)))
                        x4 := new(Rat).SetInt(new(Int).Binomial(int64(i+j), int64(i)))

                        x1.Mul(x1, x2)
                        x1.Mul(x1, x3)
                        x1.Mul(x1, x4)
                        x1.Mul(x1, x4)

                        if (i+j)&1 != 0 {
                                x1.Neg(x1)
                        }

                        a.set(i, j, x1)
                }
        }
        return a
}

func (a *matrix) mul(b *matrix) *matrix {
        if a.m != b.n {
                panic("illegal matrix multiply")
        }
        c := newMatrix(a.n, b.m)
        for i := 0; i < c.n; i++ {
                for j := 0; j < c.m; j++ {
                        x := NewRat(0, 1)
                        for k := 0; k < a.m; k++ {
                                x.Add(x, new(Rat).Mul(a.at(i, k), b.at(k, j)))
                        }
                        c.set(i, j, x)
                }
        }
        return c
}

func (a *matrix) eql(b *matrix) bool {
        if a.n != b.n || a.m != b.m {
                return false
        }
        for i := 0; i < a.n; i++ {
                for j := 0; j < a.m; j++ {
                        if a.at(i, j).Cmp(b.at(i, j)) != 0 {
                                return false
                        }
                }
        }
        return true
}

func (a *matrix) String() string {
        s := ""
        for i := 0; i < a.n; i++ {
                for j := 0; j < a.m; j++ {
                        s += fmt.Sprintf("\t%s", a.at(i, j))
                }
                s += "\n"
        }
        return s
}

func doHilbert(t *testing.T, n int) {
        a := newHilbert(n)
        b := newInverseHilbert(n)
        I := newUnit(n)
        ab := a.mul(b)
        if !ab.eql(I) {
                if t == nil {
                        panic("Hilbert failed")
                }
                t.Errorf("a   = %s\n", a)
                t.Errorf("b   = %s\n", b)
                t.Errorf("a*b = %s\n", ab)
                t.Errorf("I   = %s\n", I)
        }
}

func TestHilbert(t *testing.T) {
        doHilbert(t, 10)
}

func BenchmarkHilbert(b *testing.B) {
        for i := 0; i < b.N; i++ {
                doHilbert(nil, 10)
        }
}

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.