2018-04-24 15:13:08 +02:00
|
|
|
// asmcheck
|
|
|
|
|
|
|
|
|
|
// Copyright 2018 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 codegen
|
|
|
|
|
|
2025-09-04 09:08:14 +08:00
|
|
|
import "strings"
|
|
|
|
|
|
2018-04-24 15:13:08 +02:00
|
|
|
// This file contains code generation tests related to the handling of
|
|
|
|
|
// string types.
|
|
|
|
|
|
|
|
|
|
func CountRunes(s string) int { // Issue #24923
|
|
|
|
|
// amd64:`.*countrunes`
|
|
|
|
|
return len([]rune(s))
|
|
|
|
|
}
|
cmd/compile: make []byte("...") more efficient
Do []byte(string) conversions more efficiently when the string
is a constant. Instead of calling stringtobyteslice, allocate
just the space we need and encode the initialization directly.
[]byte("foo") rewrites to the following pseudocode:
var s [3]byte // on heap or stack, depending on whether b escapes
s = *(*[3]byte)(&"foo"[0]) // initialize s from the string
b = s[:]
which generates this assembly:
0x001d 00029 (tmp1.go:9) LEAQ type.[3]uint8(SB), AX
0x0024 00036 (tmp1.go:9) MOVQ AX, (SP)
0x0028 00040 (tmp1.go:9) CALL runtime.newobject(SB)
0x002d 00045 (tmp1.go:9) MOVQ 8(SP), AX
0x0032 00050 (tmp1.go:9) MOVBLZX go.string."foo"+2(SB), CX
0x0039 00057 (tmp1.go:9) MOVWLZX go.string."foo"(SB), DX
0x0040 00064 (tmp1.go:9) MOVW DX, (AX)
0x0043 00067 (tmp1.go:9) MOVB CL, 2(AX)
// Then the slice is b = {AX, 3, 3}
The generated code is still not optimal, as it still does load/store
from read-only memory instead of constant stores. Next CL...
Update #26498
Fixes #10170
Change-Id: I4b990b19f9a308f60c8f4f148934acffefe0a5bd
Reviewed-on: https://go-review.googlesource.com/c/140698
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
2018-10-08 17:46:45 -07:00
|
|
|
|
2023-05-23 11:23:48 +07:00
|
|
|
func CountBytes(s []byte) int {
|
|
|
|
|
// amd64:-`.*runtime.slicebytetostring`
|
|
|
|
|
return len(string(s))
|
|
|
|
|
}
|
|
|
|
|
|
cmd/compile: make []byte("...") more efficient
Do []byte(string) conversions more efficiently when the string
is a constant. Instead of calling stringtobyteslice, allocate
just the space we need and encode the initialization directly.
[]byte("foo") rewrites to the following pseudocode:
var s [3]byte // on heap or stack, depending on whether b escapes
s = *(*[3]byte)(&"foo"[0]) // initialize s from the string
b = s[:]
which generates this assembly:
0x001d 00029 (tmp1.go:9) LEAQ type.[3]uint8(SB), AX
0x0024 00036 (tmp1.go:9) MOVQ AX, (SP)
0x0028 00040 (tmp1.go:9) CALL runtime.newobject(SB)
0x002d 00045 (tmp1.go:9) MOVQ 8(SP), AX
0x0032 00050 (tmp1.go:9) MOVBLZX go.string."foo"+2(SB), CX
0x0039 00057 (tmp1.go:9) MOVWLZX go.string."foo"(SB), DX
0x0040 00064 (tmp1.go:9) MOVW DX, (AX)
0x0043 00067 (tmp1.go:9) MOVB CL, 2(AX)
// Then the slice is b = {AX, 3, 3}
The generated code is still not optimal, as it still does load/store
from read-only memory instead of constant stores. Next CL...
Update #26498
Fixes #10170
Change-Id: I4b990b19f9a308f60c8f4f148934acffefe0a5bd
Reviewed-on: https://go-review.googlesource.com/c/140698
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
2018-10-08 17:46:45 -07:00
|
|
|
func ToByteSlice() []byte { // Issue #24698
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`LEAQ type:\[3\]uint8`
|
2025-11-13 17:04:54 -05:00
|
|
|
// amd64:`CALL runtime\.(newobject|mallocTiny3)`
|
cmd/compile: make []byte("...") more efficient
Do []byte(string) conversions more efficiently when the string
is a constant. Instead of calling stringtobyteslice, allocate
just the space we need and encode the initialization directly.
[]byte("foo") rewrites to the following pseudocode:
var s [3]byte // on heap or stack, depending on whether b escapes
s = *(*[3]byte)(&"foo"[0]) // initialize s from the string
b = s[:]
which generates this assembly:
0x001d 00029 (tmp1.go:9) LEAQ type.[3]uint8(SB), AX
0x0024 00036 (tmp1.go:9) MOVQ AX, (SP)
0x0028 00040 (tmp1.go:9) CALL runtime.newobject(SB)
0x002d 00045 (tmp1.go:9) MOVQ 8(SP), AX
0x0032 00050 (tmp1.go:9) MOVBLZX go.string."foo"+2(SB), CX
0x0039 00057 (tmp1.go:9) MOVWLZX go.string."foo"(SB), DX
0x0040 00064 (tmp1.go:9) MOVW DX, (AX)
0x0043 00067 (tmp1.go:9) MOVB CL, 2(AX)
// Then the slice is b = {AX, 3, 3}
The generated code is still not optimal, as it still does load/store
from read-only memory instead of constant stores. Next CL...
Update #26498
Fixes #10170
Change-Id: I4b990b19f9a308f60c8f4f148934acffefe0a5bd
Reviewed-on: https://go-review.googlesource.com/c/140698
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
2018-10-08 17:46:45 -07:00
|
|
|
// amd64:-`.*runtime.stringtoslicebyte`
|
|
|
|
|
return []byte("foo")
|
|
|
|
|
}
|
2018-10-09 22:55:36 -07:00
|
|
|
|
2023-09-13 12:44:17 +03:00
|
|
|
func ConvertToByteSlice(a, b, c string) []byte {
|
|
|
|
|
// amd64:`.*runtime.concatbyte3`
|
|
|
|
|
return []byte(a + b + c)
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-09 22:55:36 -07:00
|
|
|
// Loading from read-only symbols should get transformed into constants.
|
|
|
|
|
func ConstantLoad() {
|
|
|
|
|
// 12592 = 0x3130
|
|
|
|
|
// 50 = 0x32
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVW \$12592, \(`,`MOVB \$50, 2\(`
|
|
|
|
|
// 386:`MOVW \$12592, \(`,`MOVB \$50, 2\(`
|
|
|
|
|
// arm:`MOVW \$48`,`MOVW \$49`,`MOVW \$50`
|
|
|
|
|
// arm64:`MOVD \$12592`,`MOVD \$50`
|
|
|
|
|
// loong64:`MOVV \$12592`,`MOVV \$50`
|
|
|
|
|
// wasm:`I64Const \$12592`,`I64Store16 \$0`,`I64Const \$50`,`I64Store8 \$2`
|
|
|
|
|
// mips64:`MOVV \$48`,`MOVV \$49`,`MOVV \$50`
|
2018-10-09 22:55:36 -07:00
|
|
|
bsink = []byte("012")
|
|
|
|
|
|
|
|
|
|
// 858927408 = 0x33323130
|
|
|
|
|
// 13620 = 0x3534
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVL \$858927408`,`MOVW \$13620, 4\(`
|
|
|
|
|
// 386:`MOVL \$858927408`,`MOVW \$13620, 4\(`
|
|
|
|
|
// arm64:`MOVD \$858927408`,`MOVD \$13620`
|
|
|
|
|
// loong64:`MOVV \$858927408`,`MOVV \$13620`
|
|
|
|
|
// wasm:`I64Const \$858927408`,`I64Store32 \$0`,`I64Const \$13620`,`I64Store16 \$4`
|
2018-10-09 22:55:36 -07:00
|
|
|
bsink = []byte("012345")
|
|
|
|
|
|
|
|
|
|
// 3978425819141910832 = 0x3736353433323130
|
|
|
|
|
// 7306073769690871863 = 0x6564636261393837
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVQ \$3978425819141910832`,`MOVQ \$7306073769690871863`
|
|
|
|
|
// 386:`MOVL \$858927408, \(`,`DUFFCOPY`
|
|
|
|
|
// arm64:`MOVD \$3978425819141910832`,`MOVD \$7306073769690871863`,`MOVD \$15`
|
|
|
|
|
// loong64:`MOVV \$3978425819141910832`,`MOVV \$7306073769690871863`,`MOVV \$15`
|
|
|
|
|
// wasm:`I64Const \$3978425819141910832`,`I64Store \$0`,`I64Const \$7306073769690871863`,`I64Store \$7`
|
2018-10-09 22:55:36 -07:00
|
|
|
bsink = []byte("0123456789abcde")
|
2018-10-30 12:58:50 -07:00
|
|
|
|
|
|
|
|
// 56 = 0x38
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVQ \$3978425819141910832`,`MOVB \$56`
|
|
|
|
|
// loong64:`MOVV \$3978425819141910832`,`MOVV \$56`
|
2018-10-30 12:58:50 -07:00
|
|
|
bsink = []byte("012345678")
|
|
|
|
|
|
|
|
|
|
// 14648 = 0x3938
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVQ \$3978425819141910832`,`MOVW \$14648`
|
|
|
|
|
// loong64:`MOVV \$3978425819141910832`,`MOVV \$14648`
|
2018-10-30 12:58:50 -07:00
|
|
|
bsink = []byte("0123456789")
|
|
|
|
|
|
|
|
|
|
// 1650538808 = 0x62613938
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVQ \$3978425819141910832`,`MOVL \$1650538808`
|
|
|
|
|
// loong64:`MOVV \$3978425819141910832`,`MOVV \$1650538808`
|
2018-10-30 12:58:50 -07:00
|
|
|
bsink = []byte("0123456789ab")
|
2018-10-09 22:55:36 -07:00
|
|
|
}
|
|
|
|
|
|
2023-06-15 09:14:12 -07:00
|
|
|
// self-equality is always true. See issue 60777.
|
|
|
|
|
func EqualSelf(s string) bool {
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`MOVL \$1, AX`,-`.*memequal.*`
|
2023-06-15 09:14:12 -07:00
|
|
|
return s == s
|
|
|
|
|
}
|
|
|
|
|
func NotEqualSelf(s string) bool {
|
2025-10-26 22:51:14 -04:00
|
|
|
// amd64:`XORL AX, AX`,-`.*memequal.*`
|
2023-06-15 09:14:12 -07:00
|
|
|
return s != s
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-09 22:55:36 -07:00
|
|
|
var bsink []byte
|
2025-09-04 09:08:14 +08:00
|
|
|
|
|
|
|
|
func HasPrefix3(s string) bool {
|
|
|
|
|
// amd64:-`.*memequal.*`
|
|
|
|
|
return strings.HasPrefix(s, "str")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func HasPrefix5(s string) bool {
|
|
|
|
|
// amd64:-`.*memequal.*`
|
|
|
|
|
return strings.HasPrefix(s, "strin")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func HasPrefix6(s string) bool {
|
|
|
|
|
// amd64:-`.*memequal.*`
|
|
|
|
|
return strings.HasPrefix(s, "string")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func HasPrefix7(s string) bool {
|
|
|
|
|
// amd64:-`.*memequal.*`
|
|
|
|
|
return strings.HasPrefix(s, "strings")
|
|
|
|
|
}
|