7a9389330e
gcc/: * gcc.c (default_compilers): Add entry for ".go". * common.opt: Add -static-libgo as a driver option. * doc/install.texi (Configuration): Mention libgo as an option for --enable-shared. Mention go as an option for --enable-languages. * doc/invoke.texi (Overall Options): Mention .go as a file name suffix. Mention go as a -x option. * doc/frontends.texi (G++ and GCC): Mention Go as a supported language. * doc/sourcebuild.texi (Top Level): Mention libgo. * doc/standards.texi (Standards): Add section on Go language. Move references for other languages into their own section. * doc/contrib.texi (Contributors): Mention that I contributed the Go frontend. gcc/testsuite/: * lib/go.exp: New file. * lib/go-dg.exp: New file. * lib/go-torture.exp: New file. * lib/target-supports.exp (check_compile): Match // Go. From-SVN: r167407
334 lines
6.6 KiB
Go
334 lines
6.6 KiB
Go
// 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.
|
|
|
|
package path
|
|
|
|
import (
|
|
"os"
|
|
"testing"
|
|
)
|
|
|
|
type CleanTest struct {
|
|
path, clean string
|
|
}
|
|
|
|
var cleantests = []CleanTest{
|
|
// Already clean
|
|
{"", "."},
|
|
{"abc", "abc"},
|
|
{"abc/def", "abc/def"},
|
|
{"a/b/c", "a/b/c"},
|
|
{".", "."},
|
|
{"..", ".."},
|
|
{"../..", "../.."},
|
|
{"../../abc", "../../abc"},
|
|
{"/abc", "/abc"},
|
|
{"/", "/"},
|
|
|
|
// Remove trailing slash
|
|
{"abc/", "abc"},
|
|
{"abc/def/", "abc/def"},
|
|
{"a/b/c/", "a/b/c"},
|
|
{"./", "."},
|
|
{"../", ".."},
|
|
{"../../", "../.."},
|
|
{"/abc/", "/abc"},
|
|
|
|
// Remove doubled slash
|
|
{"abc//def//ghi", "abc/def/ghi"},
|
|
{"//abc", "/abc"},
|
|
{"///abc", "/abc"},
|
|
{"//abc//", "/abc"},
|
|
{"abc//", "abc"},
|
|
|
|
// Remove . elements
|
|
{"abc/./def", "abc/def"},
|
|
{"/./abc/def", "/abc/def"},
|
|
{"abc/.", "abc"},
|
|
|
|
// Remove .. elements
|
|
{"abc/def/ghi/../jkl", "abc/def/jkl"},
|
|
{"abc/def/../ghi/../jkl", "abc/jkl"},
|
|
{"abc/def/..", "abc"},
|
|
{"abc/def/../..", "."},
|
|
{"/abc/def/../..", "/"},
|
|
{"abc/def/../../..", ".."},
|
|
{"/abc/def/../../..", "/"},
|
|
{"abc/def/../../../ghi/jkl/../../../mno", "../../mno"},
|
|
|
|
// Combinations
|
|
{"abc/./../def", "def"},
|
|
{"abc//./../def", "def"},
|
|
{"abc/../../././../def", "../../def"},
|
|
}
|
|
|
|
func TestClean(t *testing.T) {
|
|
for _, test := range cleantests {
|
|
if s := Clean(test.path); s != test.clean {
|
|
t.Errorf("Clean(%q) = %q, want %q", test.path, s, test.clean)
|
|
}
|
|
}
|
|
}
|
|
|
|
type SplitTest struct {
|
|
path, dir, file string
|
|
}
|
|
|
|
var splittests = []SplitTest{
|
|
{"a/b", "a/", "b"},
|
|
{"a/b/", "a/b/", ""},
|
|
{"a/", "a/", ""},
|
|
{"a", "", "a"},
|
|
{"/", "/", ""},
|
|
}
|
|
|
|
func TestSplit(t *testing.T) {
|
|
for _, test := range splittests {
|
|
if d, f := Split(test.path); d != test.dir || f != test.file {
|
|
t.Errorf("Split(%q) = %q, %q, want %q, %q", test.path, d, f, test.dir, test.file)
|
|
}
|
|
}
|
|
}
|
|
|
|
type JoinTest struct {
|
|
elem []string
|
|
path string
|
|
}
|
|
|
|
var jointests = []JoinTest{
|
|
// zero parameters
|
|
{[]string{}, ""},
|
|
|
|
// one parameter
|
|
{[]string{""}, ""},
|
|
{[]string{"a"}, "a"},
|
|
|
|
// two parameters
|
|
{[]string{"a", "b"}, "a/b"},
|
|
{[]string{"a", ""}, "a"},
|
|
{[]string{"", "b"}, "b"},
|
|
{[]string{"/", "a"}, "/a"},
|
|
{[]string{"/", ""}, "/"},
|
|
{[]string{"a/", "b"}, "a/b"},
|
|
{[]string{"a/", ""}, "a"},
|
|
{[]string{"", ""}, ""},
|
|
}
|
|
|
|
// join takes a []string and passes it to Join.
|
|
func join(elem []string, args ...string) string {
|
|
args = elem
|
|
return Join(args...)
|
|
}
|
|
|
|
func TestJoin(t *testing.T) {
|
|
for _, test := range jointests {
|
|
if p := join(test.elem); p != test.path {
|
|
t.Errorf("join(%q) = %q, want %q", test.elem, p, test.path)
|
|
}
|
|
}
|
|
}
|
|
|
|
type ExtTest struct {
|
|
path, ext string
|
|
}
|
|
|
|
var exttests = []ExtTest{
|
|
{"path.go", ".go"},
|
|
{"path.pb.go", ".go"},
|
|
{"a.dir/b", ""},
|
|
{"a.dir/b.go", ".go"},
|
|
{"a.dir/", ""},
|
|
}
|
|
|
|
func TestExt(t *testing.T) {
|
|
for _, test := range exttests {
|
|
if x := Ext(test.path); x != test.ext {
|
|
t.Errorf("Ext(%q) = %q, want %q", test.path, x, test.ext)
|
|
}
|
|
}
|
|
}
|
|
|
|
type Node struct {
|
|
name string
|
|
entries []*Node // nil if the entry is a file
|
|
mark int
|
|
}
|
|
|
|
var tree = &Node{
|
|
"testdata",
|
|
[]*Node{
|
|
&Node{"a", nil, 0},
|
|
&Node{"b", []*Node{}, 0},
|
|
&Node{"c", nil, 0},
|
|
&Node{
|
|
"d",
|
|
[]*Node{
|
|
&Node{"x", nil, 0},
|
|
&Node{"y", []*Node{}, 0},
|
|
&Node{
|
|
"z",
|
|
[]*Node{
|
|
&Node{"u", nil, 0},
|
|
&Node{"v", nil, 0},
|
|
},
|
|
0,
|
|
},
|
|
},
|
|
0,
|
|
},
|
|
},
|
|
0,
|
|
}
|
|
|
|
func walkTree(n *Node, path string, f func(path string, n *Node)) {
|
|
f(path, n)
|
|
for _, e := range n.entries {
|
|
walkTree(e, Join(path, e.name), f)
|
|
}
|
|
}
|
|
|
|
func makeTree(t *testing.T) {
|
|
walkTree(tree, tree.name, func(path string, n *Node) {
|
|
if n.entries == nil {
|
|
fd, err := os.Open(path, os.O_CREAT, 0660)
|
|
if err != nil {
|
|
t.Errorf("makeTree: %v", err)
|
|
}
|
|
fd.Close()
|
|
} else {
|
|
os.Mkdir(path, 0770)
|
|
}
|
|
})
|
|
}
|
|
|
|
func markTree(n *Node) { walkTree(n, "", func(path string, n *Node) { n.mark++ }) }
|
|
|
|
func checkMarks(t *testing.T) {
|
|
walkTree(tree, tree.name, func(path string, n *Node) {
|
|
if n.mark != 1 {
|
|
t.Errorf("node %s mark = %d; expected 1", path, n.mark)
|
|
}
|
|
n.mark = 0
|
|
})
|
|
}
|
|
|
|
// Assumes that each node name is unique. Good enough for a test.
|
|
func mark(name string) {
|
|
walkTree(tree, tree.name, func(path string, n *Node) {
|
|
if n.name == name {
|
|
n.mark++
|
|
}
|
|
})
|
|
}
|
|
|
|
type TestVisitor struct{}
|
|
|
|
func (v *TestVisitor) VisitDir(path string, f *os.FileInfo) bool {
|
|
mark(f.Name)
|
|
return true
|
|
}
|
|
|
|
func (v *TestVisitor) VisitFile(path string, f *os.FileInfo) {
|
|
mark(f.Name)
|
|
}
|
|
|
|
func TestWalk(t *testing.T) {
|
|
makeTree(t)
|
|
|
|
// 1) ignore error handling, expect none
|
|
v := &TestVisitor{}
|
|
Walk(tree.name, v, nil)
|
|
checkMarks(t)
|
|
|
|
// 2) handle errors, expect none
|
|
errors := make(chan os.Error, 64)
|
|
Walk(tree.name, v, errors)
|
|
if err, ok := <-errors; ok {
|
|
t.Errorf("no error expected, found: s", err)
|
|
}
|
|
checkMarks(t)
|
|
|
|
if os.Getuid() != 0 {
|
|
// introduce 2 errors: chmod top-level directories to 0
|
|
os.Chmod(Join(tree.name, tree.entries[1].name), 0)
|
|
os.Chmod(Join(tree.name, tree.entries[3].name), 0)
|
|
// mark respective subtrees manually
|
|
markTree(tree.entries[1])
|
|
markTree(tree.entries[3])
|
|
// correct double-marking of directory itself
|
|
tree.entries[1].mark--
|
|
tree.entries[3].mark--
|
|
|
|
// 3) handle errors, expect two
|
|
errors = make(chan os.Error, 64)
|
|
os.Chmod(Join(tree.name, tree.entries[1].name), 0)
|
|
Walk(tree.name, v, errors)
|
|
for i := 1; i <= 2; i++ {
|
|
if _, ok := <-errors; !ok {
|
|
t.Errorf("%d. error expected, none found", i)
|
|
break
|
|
}
|
|
}
|
|
if err, ok := <-errors; ok {
|
|
t.Errorf("only two errors expected, found 3rd: %v", err)
|
|
}
|
|
// the inaccessible subtrees were marked manually
|
|
checkMarks(t)
|
|
}
|
|
|
|
// cleanup
|
|
os.Chmod(Join(tree.name, tree.entries[1].name), 0770)
|
|
os.Chmod(Join(tree.name, tree.entries[3].name), 0770)
|
|
if err := os.RemoveAll(tree.name); err != nil {
|
|
t.Errorf("removeTree: %v", err)
|
|
}
|
|
}
|
|
|
|
var basetests = []CleanTest{
|
|
// Already clean
|
|
{"", "."},
|
|
{".", "."},
|
|
{"/.", "."},
|
|
{"/", "/"},
|
|
{"////", "/"},
|
|
{"x/", "x"},
|
|
{"abc", "abc"},
|
|
{"abc/def", "def"},
|
|
{"a/b/.x", ".x"},
|
|
{"a/b/c.", "c."},
|
|
{"a/b/c.x", "c.x"},
|
|
}
|
|
|
|
func TestBase(t *testing.T) {
|
|
for _, test := range basetests {
|
|
if s := Base(test.path); s != test.clean {
|
|
t.Errorf("Base(%q) = %q, want %q", test.path, s, test.clean)
|
|
}
|
|
}
|
|
}
|
|
|
|
type IsAbsTest struct {
|
|
path string
|
|
isAbs bool
|
|
}
|
|
|
|
var isAbsTests = []IsAbsTest{
|
|
{"", false},
|
|
{"/", true},
|
|
{"/usr/bin/gcc", true},
|
|
{"..", false},
|
|
{"/a/../bb", true},
|
|
{".", false},
|
|
{"./", false},
|
|
{"lala", false},
|
|
}
|
|
|
|
func TestIsAbs(t *testing.T) {
|
|
for _, test := range isAbsTests {
|
|
if r := IsAbs(test.path); r != test.isAbs {
|
|
t.Errorf("IsAbs(%q) = %v, want %v", test.path, r, test.isAbs)
|
|
}
|
|
}
|
|
}
|