| // run |
| |
| // 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. |
| |
| // Semi-exhaustive test for the copy predeclared function. |
| |
| package main |
| |
| import ( |
| "fmt" |
| "os" |
| ) |
| |
| const N = 40 |
| |
| var input8 = make([]uint8, N) |
| var output8 = make([]uint8, N) |
| var input16 = make([]uint16, N) |
| var output16 = make([]uint16, N) |
| var input32 = make([]uint32, N) |
| var output32 = make([]uint32, N) |
| var input64 = make([]uint64, N) |
| var output64 = make([]uint64, N) |
| var inputS string |
| var outputS = make([]uint8, N) |
| |
| type my8 []uint8 |
| type my16 []uint16 |
| type my32 []uint32 |
| type my32b []uint32 |
| type my64 []uint64 |
| type myS string |
| |
| func u8(i int) uint8 { |
| i = 'a' + i%26 |
| return uint8(i) |
| } |
| |
| func u16(ii int) uint16 { |
| var i = uint16(ii) |
| i = 'a' + i%26 |
| i |= i << 8 |
| return i |
| } |
| |
| func u32(ii int) uint32 { |
| var i = uint32(ii) |
| i = 'a' + i%26 |
| i |= i << 8 |
| i |= i << 16 |
| return i |
| } |
| |
| func u64(ii int) uint64 { |
| var i = uint64(ii) |
| i = 'a' + i%26 |
| i |= i << 8 |
| i |= i << 16 |
| i |= i << 32 |
| return i |
| } |
| |
| func reset() { |
| // swap in and out to exercise copy-up and copy-down |
| input8, output8 = output8, input8 |
| input16, output16 = output16, input16 |
| input32, output32 = output32, input32 |
| input64, output64 = output64, input64 |
| in := 0 |
| out := 13 |
| for i := range input8 { |
| input8[i] = u8(in) |
| output8[i] = u8(out) |
| outputS[i] = u8(out) |
| input16[i] = u16(in) |
| output16[i] = u16(out) |
| input32[i] = u32(in) |
| output32[i] = u32(out) |
| input64[i] = u64(in) |
| output64[i] = u64(out) |
| in++ |
| out++ |
| } |
| inputS = string(input8) |
| } |
| |
| func clamp(n int) int { |
| if n > N { |
| return N |
| } |
| return n |
| } |
| |
| func ncopied(length, in, out int) int { |
| n := length |
| if in+n > N { |
| n = N - in |
| } |
| if out+n > N { |
| n = N - out |
| } |
| return n |
| } |
| |
| func doAllSlices(length, in, out int) { |
| reset() |
| n := copy(my8(output8[out:clamp(out+length)]), input8[in:clamp(in+length)]) |
| verify8(length, in, out, n) |
| n = copy(my8(outputS[out:clamp(out+length)]), myS(inputS[in:clamp(in+length)])) |
| verifyS(length, in, out, n) |
| n = copy(my16(output16[out:clamp(out+length)]), input16[in:clamp(in+length)]) |
| verify16(length, in, out, n) |
| n = copy(my32(output32[out:clamp(out+length)]), my32b(input32[in:clamp(in+length)])) |
| verify32(length, in, out, n) |
| n = copy(my64(output64[out:clamp(out+length)]), input64[in:clamp(in+length)]) |
| verify64(length, in, out, n) |
| } |
| |
| func bad8(state string, i, length, in, out int) { |
| fmt.Printf("%s bad(%d %d %d): %c not %c:\n\t%s\n\t%s\n", |
| state, |
| length, in, out, |
| output8[i], |
| uint8(i+13), |
| input8, output8) |
| os.Exit(1) |
| } |
| |
| func verify8(length, in, out, m int) { |
| n := ncopied(length, in, out) |
| if m != n { |
| fmt.Printf("count bad(%d %d %d): %d not %d\n", length, in, out, m, n) |
| os.Exit(1) |
| return |
| } |
| // before |
| var i int |
| for i = 0; i < out; i++ { |
| if output8[i] != u8(i+13) { |
| bad8("before8", i, length, in, out) |
| return |
| } |
| } |
| // copied part |
| for ; i < out+n; i++ { |
| if output8[i] != u8(i+in-out) { |
| bad8("copied8", i, length, in, out) |
| return |
| } |
| } |
| // after |
| for ; i < len(output8); i++ { |
| if output8[i] != u8(i+13) { |
| bad8("after8", i, length, in, out) |
| return |
| } |
| } |
| } |
| |
| func badS(state string, i, length, in, out int) { |
| fmt.Printf("%s bad(%d %d %d): %c not %c:\n\t%s\n\t%s\n", |
| state, |
| length, in, out, |
| outputS[i], |
| uint8(i+13), |
| inputS, outputS) |
| os.Exit(1) |
| } |
| |
| func verifyS(length, in, out, m int) { |
| n := ncopied(length, in, out) |
| if m != n { |
| fmt.Printf("count bad(%d %d %d): %d not %d\n", length, in, out, m, n) |
| os.Exit(1) |
| return |
| } |
| // before |
| var i int |
| for i = 0; i < out; i++ { |
| if outputS[i] != u8(i+13) { |
| badS("beforeS", i, length, in, out) |
| return |
| } |
| } |
| // copied part |
| for ; i < out+n; i++ { |
| if outputS[i] != u8(i+in-out) { |
| badS("copiedS", i, length, in, out) |
| return |
| } |
| } |
| // after |
| for ; i < len(outputS); i++ { |
| if outputS[i] != u8(i+13) { |
| badS("afterS", i, length, in, out) |
| return |
| } |
| } |
| } |
| |
| func bad16(state string, i, length, in, out int) { |
| fmt.Printf("%s bad(%d %d %d): %x not %x:\n\t%v\n\t%v\n", |
| state, |
| length, in, out, |
| output16[i], |
| uint16(i+13), |
| input16, output16) |
| os.Exit(1) |
| } |
| |
| func verify16(length, in, out, m int) { |
| n := ncopied(length, in, out) |
| if m != n { |
| fmt.Printf("count bad(%d %d %d): %d not %d\n", length, in, out, m, n) |
| os.Exit(1) |
| return |
| } |
| // before |
| var i int |
| for i = 0; i < out; i++ { |
| if output16[i] != u16(i+13) { |
| bad16("before16", i, length, in, out) |
| return |
| } |
| } |
| // copied part |
| for ; i < out+n; i++ { |
| if output16[i] != u16(i+in-out) { |
| bad16("copied16", i, length, in, out) |
| return |
| } |
| } |
| // after |
| for ; i < len(output16); i++ { |
| if output16[i] != u16(i+13) { |
| bad16("after16", i, length, in, out) |
| return |
| } |
| } |
| } |
| |
| func bad32(state string, i, length, in, out int) { |
| fmt.Printf("%s bad(%d %d %d): %x not %x:\n\t%v\n\t%v\n", |
| state, |
| length, in, out, |
| output32[i], |
| uint32(i+13), |
| input32, output32) |
| os.Exit(1) |
| } |
| |
| func verify32(length, in, out, m int) { |
| n := ncopied(length, in, out) |
| if m != n { |
| fmt.Printf("count bad(%d %d %d): %d not %d\n", length, in, out, m, n) |
| os.Exit(1) |
| return |
| } |
| // before |
| var i int |
| for i = 0; i < out; i++ { |
| if output32[i] != u32(i+13) { |
| bad32("before32", i, length, in, out) |
| return |
| } |
| } |
| // copied part |
| for ; i < out+n; i++ { |
| if output32[i] != u32(i+in-out) { |
| bad32("copied32", i, length, in, out) |
| return |
| } |
| } |
| // after |
| for ; i < len(output32); i++ { |
| if output32[i] != u32(i+13) { |
| bad32("after32", i, length, in, out) |
| return |
| } |
| } |
| } |
| |
| func bad64(state string, i, length, in, out int) { |
| fmt.Printf("%s bad(%d %d %d): %x not %x:\n\t%v\n\t%v\n", |
| state, |
| length, in, out, |
| output64[i], |
| uint64(i+13), |
| input64, output64) |
| os.Exit(1) |
| } |
| |
| func verify64(length, in, out, m int) { |
| n := ncopied(length, in, out) |
| if m != n { |
| fmt.Printf("count bad(%d %d %d): %d not %d\n", length, in, out, m, n) |
| os.Exit(1) |
| return |
| } |
| // before |
| var i int |
| for i = 0; i < out; i++ { |
| if output64[i] != u64(i+13) { |
| bad64("before64", i, length, in, out) |
| return |
| } |
| } |
| // copied part |
| for ; i < out+n; i++ { |
| if output64[i] != u64(i+in-out) { |
| bad64("copied64", i, length, in, out) |
| return |
| } |
| } |
| // after |
| for ; i < len(output64); i++ { |
| if output64[i] != u64(i+13) { |
| bad64("after64", i, length, in, out) |
| return |
| } |
| } |
| } |
| |
| func slice() { |
| for length := 0; length < N; length++ { |
| for in := 0; in <= 32; in++ { |
| for out := 0; out <= 32; out++ { |
| doAllSlices(length, in, out) |
| } |
| } |
| } |
| } |
| |
| // Array test. Can be much simpler. It's only checking for correct handling of [0:]. |
| func array() { |
| var array [N]uint8 |
| reset() |
| copy(array[0:], input8) |
| for i := 0; i < N; i++ { |
| output8[i] = 0 |
| } |
| copy(output8, array[0:]) |
| verify8(N, 0, 0, N) |
| } |
| |
| func main() { |
| slice() |
| array() |
| } |