mirror of
https://github.com/golang/go
synced 2024-11-11 21:50:21 -07:00
cmd/compile: optimize len([]rune(string))
Adds a new runtime function to count runes in a string. Modifies the compiler to detect the pattern len([]rune(string)) and replaces it with the new rune counting runtime function. RuneCount/lenruneslice/ASCII 27.8ns ± 2% 14.5ns ± 3% -47.70% (p=0.000 n=10+10) RuneCount/lenruneslice/Japanese 126ns ± 2% 60ns ± 2% -52.03% (p=0.000 n=10+10) RuneCount/lenruneslice/MixedLength 104ns ± 2% 50ns ± 1% -51.71% (p=0.000 n=10+9) Fixes #24923 Change-Id: Ie9c7e7391a4e2cca675c5cdcc1e5ce7d523948b9 Reviewed-on: https://go-review.googlesource.com/108985 Run-TryBot: Martin Möhrmann <moehrmann@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Josh Bleecher Snyder <josharian@gmail.com>
This commit is contained in:
parent
a8a60ac2a7
commit
b9a59d9f2e
@ -46,112 +46,113 @@ var runtimeDecls = [...]struct {
|
|||||||
{"slicerunetostring", funcTag, 42},
|
{"slicerunetostring", funcTag, 42},
|
||||||
{"stringtoslicebyte", funcTag, 43},
|
{"stringtoslicebyte", funcTag, 43},
|
||||||
{"stringtoslicerune", funcTag, 46},
|
{"stringtoslicerune", funcTag, 46},
|
||||||
{"decoderune", funcTag, 47},
|
{"slicecopy", funcTag, 48},
|
||||||
{"slicecopy", funcTag, 49},
|
{"slicestringcopy", funcTag, 49},
|
||||||
{"slicestringcopy", funcTag, 50},
|
{"decoderune", funcTag, 50},
|
||||||
{"convI2I", funcTag, 51},
|
{"countrunes", funcTag, 51},
|
||||||
{"convT2E", funcTag, 52},
|
{"convI2I", funcTag, 52},
|
||||||
{"convT2E16", funcTag, 51},
|
{"convT2E", funcTag, 53},
|
||||||
{"convT2E32", funcTag, 51},
|
{"convT2E16", funcTag, 52},
|
||||||
{"convT2E64", funcTag, 51},
|
{"convT2E32", funcTag, 52},
|
||||||
{"convT2Estring", funcTag, 52},
|
{"convT2E64", funcTag, 52},
|
||||||
{"convT2Eslice", funcTag, 52},
|
{"convT2Estring", funcTag, 53},
|
||||||
{"convT2Enoptr", funcTag, 52},
|
{"convT2Eslice", funcTag, 53},
|
||||||
{"convT2I", funcTag, 52},
|
{"convT2Enoptr", funcTag, 53},
|
||||||
{"convT2I16", funcTag, 51},
|
{"convT2I", funcTag, 53},
|
||||||
{"convT2I32", funcTag, 51},
|
{"convT2I16", funcTag, 52},
|
||||||
{"convT2I64", funcTag, 51},
|
{"convT2I32", funcTag, 52},
|
||||||
{"convT2Istring", funcTag, 52},
|
{"convT2I64", funcTag, 52},
|
||||||
{"convT2Islice", funcTag, 52},
|
{"convT2Istring", funcTag, 53},
|
||||||
{"convT2Inoptr", funcTag, 52},
|
{"convT2Islice", funcTag, 53},
|
||||||
{"assertE2I", funcTag, 51},
|
{"convT2Inoptr", funcTag, 53},
|
||||||
{"assertE2I2", funcTag, 53},
|
{"assertE2I", funcTag, 52},
|
||||||
{"assertI2I", funcTag, 51},
|
{"assertE2I2", funcTag, 54},
|
||||||
{"assertI2I2", funcTag, 53},
|
{"assertI2I", funcTag, 52},
|
||||||
{"panicdottypeE", funcTag, 54},
|
{"assertI2I2", funcTag, 54},
|
||||||
{"panicdottypeI", funcTag, 54},
|
{"panicdottypeE", funcTag, 55},
|
||||||
{"panicnildottype", funcTag, 55},
|
{"panicdottypeI", funcTag, 55},
|
||||||
{"ifaceeq", funcTag, 58},
|
{"panicnildottype", funcTag, 56},
|
||||||
{"efaceeq", funcTag, 58},
|
{"ifaceeq", funcTag, 59},
|
||||||
{"fastrand", funcTag, 60},
|
{"efaceeq", funcTag, 59},
|
||||||
{"makemap64", funcTag, 62},
|
{"fastrand", funcTag, 61},
|
||||||
{"makemap", funcTag, 63},
|
{"makemap64", funcTag, 63},
|
||||||
{"makemap_small", funcTag, 64},
|
{"makemap", funcTag, 64},
|
||||||
{"mapaccess1", funcTag, 65},
|
{"makemap_small", funcTag, 65},
|
||||||
{"mapaccess1_fast32", funcTag, 66},
|
{"mapaccess1", funcTag, 66},
|
||||||
{"mapaccess1_fast64", funcTag, 66},
|
{"mapaccess1_fast32", funcTag, 67},
|
||||||
{"mapaccess1_faststr", funcTag, 66},
|
{"mapaccess1_fast64", funcTag, 67},
|
||||||
{"mapaccess1_fat", funcTag, 67},
|
{"mapaccess1_faststr", funcTag, 67},
|
||||||
{"mapaccess2", funcTag, 68},
|
{"mapaccess1_fat", funcTag, 68},
|
||||||
{"mapaccess2_fast32", funcTag, 69},
|
{"mapaccess2", funcTag, 69},
|
||||||
{"mapaccess2_fast64", funcTag, 69},
|
{"mapaccess2_fast32", funcTag, 70},
|
||||||
{"mapaccess2_faststr", funcTag, 69},
|
{"mapaccess2_fast64", funcTag, 70},
|
||||||
{"mapaccess2_fat", funcTag, 70},
|
{"mapaccess2_faststr", funcTag, 70},
|
||||||
{"mapassign", funcTag, 65},
|
{"mapaccess2_fat", funcTag, 71},
|
||||||
{"mapassign_fast32", funcTag, 66},
|
{"mapassign", funcTag, 66},
|
||||||
{"mapassign_fast32ptr", funcTag, 66},
|
{"mapassign_fast32", funcTag, 67},
|
||||||
{"mapassign_fast64", funcTag, 66},
|
{"mapassign_fast32ptr", funcTag, 67},
|
||||||
{"mapassign_fast64ptr", funcTag, 66},
|
{"mapassign_fast64", funcTag, 67},
|
||||||
{"mapassign_faststr", funcTag, 66},
|
{"mapassign_fast64ptr", funcTag, 67},
|
||||||
{"mapiterinit", funcTag, 71},
|
{"mapassign_faststr", funcTag, 67},
|
||||||
{"mapdelete", funcTag, 71},
|
{"mapiterinit", funcTag, 72},
|
||||||
{"mapdelete_fast32", funcTag, 72},
|
{"mapdelete", funcTag, 72},
|
||||||
{"mapdelete_fast64", funcTag, 72},
|
{"mapdelete_fast32", funcTag, 73},
|
||||||
{"mapdelete_faststr", funcTag, 72},
|
{"mapdelete_fast64", funcTag, 73},
|
||||||
{"mapiternext", funcTag, 73},
|
{"mapdelete_faststr", funcTag, 73},
|
||||||
{"makechan64", funcTag, 75},
|
{"mapiternext", funcTag, 74},
|
||||||
{"makechan", funcTag, 76},
|
{"makechan64", funcTag, 76},
|
||||||
{"chanrecv1", funcTag, 78},
|
{"makechan", funcTag, 77},
|
||||||
{"chanrecv2", funcTag, 79},
|
{"chanrecv1", funcTag, 79},
|
||||||
{"chansend1", funcTag, 81},
|
{"chanrecv2", funcTag, 80},
|
||||||
|
{"chansend1", funcTag, 82},
|
||||||
{"closechan", funcTag, 23},
|
{"closechan", funcTag, 23},
|
||||||
{"writeBarrier", varTag, 83},
|
{"writeBarrier", varTag, 84},
|
||||||
{"typedmemmove", funcTag, 84},
|
{"typedmemmove", funcTag, 85},
|
||||||
{"typedmemclr", funcTag, 85},
|
{"typedmemclr", funcTag, 86},
|
||||||
{"typedslicecopy", funcTag, 86},
|
{"typedslicecopy", funcTag, 87},
|
||||||
{"selectnbsend", funcTag, 87},
|
{"selectnbsend", funcTag, 88},
|
||||||
{"selectnbrecv", funcTag, 88},
|
{"selectnbrecv", funcTag, 89},
|
||||||
{"selectnbrecv2", funcTag, 90},
|
{"selectnbrecv2", funcTag, 91},
|
||||||
{"selectsetpc", funcTag, 55},
|
{"selectsetpc", funcTag, 56},
|
||||||
{"selectgo", funcTag, 91},
|
{"selectgo", funcTag, 92},
|
||||||
{"block", funcTag, 5},
|
{"block", funcTag, 5},
|
||||||
{"makeslice", funcTag, 93},
|
{"makeslice", funcTag, 94},
|
||||||
{"makeslice64", funcTag, 94},
|
{"makeslice64", funcTag, 95},
|
||||||
{"growslice", funcTag, 95},
|
{"growslice", funcTag, 96},
|
||||||
{"memmove", funcTag, 96},
|
{"memmove", funcTag, 97},
|
||||||
{"memclrNoHeapPointers", funcTag, 97},
|
{"memclrNoHeapPointers", funcTag, 98},
|
||||||
{"memclrHasPointers", funcTag, 97},
|
{"memclrHasPointers", funcTag, 98},
|
||||||
{"memequal", funcTag, 98},
|
{"memequal", funcTag, 99},
|
||||||
{"memequal8", funcTag, 99},
|
{"memequal8", funcTag, 100},
|
||||||
{"memequal16", funcTag, 99},
|
{"memequal16", funcTag, 100},
|
||||||
{"memequal32", funcTag, 99},
|
{"memequal32", funcTag, 100},
|
||||||
{"memequal64", funcTag, 99},
|
{"memequal64", funcTag, 100},
|
||||||
{"memequal128", funcTag, 99},
|
{"memequal128", funcTag, 100},
|
||||||
{"int64div", funcTag, 100},
|
{"int64div", funcTag, 101},
|
||||||
{"uint64div", funcTag, 101},
|
{"uint64div", funcTag, 102},
|
||||||
{"int64mod", funcTag, 100},
|
{"int64mod", funcTag, 101},
|
||||||
{"uint64mod", funcTag, 101},
|
{"uint64mod", funcTag, 102},
|
||||||
{"float64toint64", funcTag, 102},
|
{"float64toint64", funcTag, 103},
|
||||||
{"float64touint64", funcTag, 103},
|
{"float64touint64", funcTag, 104},
|
||||||
{"float64touint32", funcTag, 104},
|
{"float64touint32", funcTag, 105},
|
||||||
{"int64tofloat64", funcTag, 105},
|
{"int64tofloat64", funcTag, 106},
|
||||||
{"uint64tofloat64", funcTag, 106},
|
{"uint64tofloat64", funcTag, 107},
|
||||||
{"uint32tofloat64", funcTag, 107},
|
{"uint32tofloat64", funcTag, 108},
|
||||||
{"complex128div", funcTag, 108},
|
{"complex128div", funcTag, 109},
|
||||||
{"racefuncenter", funcTag, 109},
|
{"racefuncenter", funcTag, 110},
|
||||||
{"racefuncexit", funcTag, 5},
|
{"racefuncexit", funcTag, 5},
|
||||||
{"raceread", funcTag, 109},
|
{"raceread", funcTag, 110},
|
||||||
{"racewrite", funcTag, 109},
|
{"racewrite", funcTag, 110},
|
||||||
{"racereadrange", funcTag, 110},
|
{"racereadrange", funcTag, 111},
|
||||||
{"racewriterange", funcTag, 110},
|
{"racewriterange", funcTag, 111},
|
||||||
{"msanread", funcTag, 110},
|
{"msanread", funcTag, 111},
|
||||||
{"msanwrite", funcTag, 110},
|
{"msanwrite", funcTag, 111},
|
||||||
{"support_popcnt", varTag, 11},
|
{"support_popcnt", varTag, 11},
|
||||||
{"support_sse41", varTag, 11},
|
{"support_sse41", varTag, 11},
|
||||||
}
|
}
|
||||||
|
|
||||||
func runtimeTypes() []*types.Type {
|
func runtimeTypes() []*types.Type {
|
||||||
var typs [111]*types.Type
|
var typs [112]*types.Type
|
||||||
typs[0] = types.Bytetype
|
typs[0] = types.Bytetype
|
||||||
typs[1] = types.NewPtr(typs[0])
|
typs[1] = types.NewPtr(typs[0])
|
||||||
typs[2] = types.Types[TANY]
|
typs[2] = types.Types[TANY]
|
||||||
@ -199,69 +200,70 @@ func runtimeTypes() []*types.Type {
|
|||||||
typs[44] = types.NewArray(typs[40], 32)
|
typs[44] = types.NewArray(typs[40], 32)
|
||||||
typs[45] = types.NewPtr(typs[44])
|
typs[45] = types.NewPtr(typs[44])
|
||||||
typs[46] = functype(nil, []*Node{anonfield(typs[45]), anonfield(typs[21])}, []*Node{anonfield(typs[41])})
|
typs[46] = functype(nil, []*Node{anonfield(typs[45]), anonfield(typs[21])}, []*Node{anonfield(typs[41])})
|
||||||
typs[47] = functype(nil, []*Node{anonfield(typs[21]), anonfield(typs[32])}, []*Node{anonfield(typs[40]), anonfield(typs[32])})
|
typs[47] = types.Types[TUINTPTR]
|
||||||
typs[48] = types.Types[TUINTPTR]
|
typs[48] = functype(nil, []*Node{anonfield(typs[2]), anonfield(typs[2]), anonfield(typs[47])}, []*Node{anonfield(typs[32])})
|
||||||
typs[49] = functype(nil, []*Node{anonfield(typs[2]), anonfield(typs[2]), anonfield(typs[48])}, []*Node{anonfield(typs[32])})
|
typs[49] = functype(nil, []*Node{anonfield(typs[2]), anonfield(typs[2])}, []*Node{anonfield(typs[32])})
|
||||||
typs[50] = functype(nil, []*Node{anonfield(typs[2]), anonfield(typs[2])}, []*Node{anonfield(typs[32])})
|
typs[50] = functype(nil, []*Node{anonfield(typs[21]), anonfield(typs[32])}, []*Node{anonfield(typs[40]), anonfield(typs[32])})
|
||||||
typs[51] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2])}, []*Node{anonfield(typs[2])})
|
typs[51] = functype(nil, []*Node{anonfield(typs[21])}, []*Node{anonfield(typs[32])})
|
||||||
typs[52] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3])}, []*Node{anonfield(typs[2])})
|
typs[52] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2])}, []*Node{anonfield(typs[2])})
|
||||||
typs[53] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2])}, []*Node{anonfield(typs[2]), anonfield(typs[11])})
|
typs[53] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3])}, []*Node{anonfield(typs[2])})
|
||||||
typs[54] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[1]), anonfield(typs[1])}, nil)
|
typs[54] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2])}, []*Node{anonfield(typs[2]), anonfield(typs[11])})
|
||||||
typs[55] = functype(nil, []*Node{anonfield(typs[1])}, nil)
|
typs[55] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[1]), anonfield(typs[1])}, nil)
|
||||||
typs[56] = types.NewPtr(typs[48])
|
typs[56] = functype(nil, []*Node{anonfield(typs[1])}, nil)
|
||||||
typs[57] = types.Types[TUNSAFEPTR]
|
typs[57] = types.NewPtr(typs[47])
|
||||||
typs[58] = functype(nil, []*Node{anonfield(typs[56]), anonfield(typs[57]), anonfield(typs[57])}, []*Node{anonfield(typs[11])})
|
typs[58] = types.Types[TUNSAFEPTR]
|
||||||
typs[59] = types.Types[TUINT32]
|
typs[59] = functype(nil, []*Node{anonfield(typs[57]), anonfield(typs[58]), anonfield(typs[58])}, []*Node{anonfield(typs[11])})
|
||||||
typs[60] = functype(nil, nil, []*Node{anonfield(typs[59])})
|
typs[60] = types.Types[TUINT32]
|
||||||
typs[61] = types.NewMap(typs[2], typs[2])
|
typs[61] = functype(nil, nil, []*Node{anonfield(typs[60])})
|
||||||
typs[62] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[3])}, []*Node{anonfield(typs[61])})
|
typs[62] = types.NewMap(typs[2], typs[2])
|
||||||
typs[63] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32]), anonfield(typs[3])}, []*Node{anonfield(typs[61])})
|
typs[63] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[3])}, []*Node{anonfield(typs[62])})
|
||||||
typs[64] = functype(nil, nil, []*Node{anonfield(typs[61])})
|
typs[64] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32]), anonfield(typs[3])}, []*Node{anonfield(typs[62])})
|
||||||
typs[65] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[3])}, []*Node{anonfield(typs[3])})
|
typs[65] = functype(nil, nil, []*Node{anonfield(typs[62])})
|
||||||
typs[66] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[2])}, []*Node{anonfield(typs[3])})
|
typs[66] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[3])}, []*Node{anonfield(typs[3])})
|
||||||
typs[67] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[3]), anonfield(typs[1])}, []*Node{anonfield(typs[3])})
|
typs[67] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[2])}, []*Node{anonfield(typs[3])})
|
||||||
typs[68] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[3])}, []*Node{anonfield(typs[3]), anonfield(typs[11])})
|
typs[68] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[3]), anonfield(typs[1])}, []*Node{anonfield(typs[3])})
|
||||||
typs[69] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[2])}, []*Node{anonfield(typs[3]), anonfield(typs[11])})
|
typs[69] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[3])}, []*Node{anonfield(typs[3]), anonfield(typs[11])})
|
||||||
typs[70] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[3]), anonfield(typs[1])}, []*Node{anonfield(typs[3]), anonfield(typs[11])})
|
typs[70] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[2])}, []*Node{anonfield(typs[3]), anonfield(typs[11])})
|
||||||
typs[71] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[3])}, nil)
|
typs[71] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[3]), anonfield(typs[1])}, []*Node{anonfield(typs[3]), anonfield(typs[11])})
|
||||||
typs[72] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[61]), anonfield(typs[2])}, nil)
|
typs[72] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[3])}, nil)
|
||||||
typs[73] = functype(nil, []*Node{anonfield(typs[3])}, nil)
|
typs[73] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[62]), anonfield(typs[2])}, nil)
|
||||||
typs[74] = types.NewChan(typs[2], types.Cboth)
|
typs[74] = functype(nil, []*Node{anonfield(typs[3])}, nil)
|
||||||
typs[75] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15])}, []*Node{anonfield(typs[74])})
|
typs[75] = types.NewChan(typs[2], types.Cboth)
|
||||||
typs[76] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32])}, []*Node{anonfield(typs[74])})
|
typs[76] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15])}, []*Node{anonfield(typs[75])})
|
||||||
typs[77] = types.NewChan(typs[2], types.Crecv)
|
typs[77] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32])}, []*Node{anonfield(typs[75])})
|
||||||
typs[78] = functype(nil, []*Node{anonfield(typs[77]), anonfield(typs[3])}, nil)
|
typs[78] = types.NewChan(typs[2], types.Crecv)
|
||||||
typs[79] = functype(nil, []*Node{anonfield(typs[77]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
|
typs[79] = functype(nil, []*Node{anonfield(typs[78]), anonfield(typs[3])}, nil)
|
||||||
typs[80] = types.NewChan(typs[2], types.Csend)
|
typs[80] = functype(nil, []*Node{anonfield(typs[78]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
|
||||||
typs[81] = functype(nil, []*Node{anonfield(typs[80]), anonfield(typs[3])}, nil)
|
typs[81] = types.NewChan(typs[2], types.Csend)
|
||||||
typs[82] = types.NewArray(typs[0], 3)
|
typs[82] = functype(nil, []*Node{anonfield(typs[81]), anonfield(typs[3])}, nil)
|
||||||
typs[83] = tostruct([]*Node{namedfield("enabled", typs[11]), namedfield("pad", typs[82]), namedfield("needed", typs[11]), namedfield("cgo", typs[11]), namedfield("alignme", typs[17])})
|
typs[83] = types.NewArray(typs[0], 3)
|
||||||
typs[84] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3]), anonfield(typs[3])}, nil)
|
typs[84] = tostruct([]*Node{namedfield("enabled", typs[11]), namedfield("pad", typs[83]), namedfield("needed", typs[11]), namedfield("cgo", typs[11]), namedfield("alignme", typs[17])})
|
||||||
typs[85] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3])}, nil)
|
typs[85] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3]), anonfield(typs[3])}, nil)
|
||||||
typs[86] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2]), anonfield(typs[2])}, []*Node{anonfield(typs[32])})
|
typs[86] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3])}, nil)
|
||||||
typs[87] = functype(nil, []*Node{anonfield(typs[80]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
|
typs[87] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2]), anonfield(typs[2])}, []*Node{anonfield(typs[32])})
|
||||||
typs[88] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[77])}, []*Node{anonfield(typs[11])})
|
typs[88] = functype(nil, []*Node{anonfield(typs[81]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
|
||||||
typs[89] = types.NewPtr(typs[11])
|
typs[89] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[78])}, []*Node{anonfield(typs[11])})
|
||||||
typs[90] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[89]), anonfield(typs[77])}, []*Node{anonfield(typs[11])})
|
typs[90] = types.NewPtr(typs[11])
|
||||||
typs[91] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[1]), anonfield(typs[32])}, []*Node{anonfield(typs[32]), anonfield(typs[11])})
|
typs[91] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[90]), anonfield(typs[78])}, []*Node{anonfield(typs[11])})
|
||||||
typs[92] = types.NewSlice(typs[2])
|
typs[92] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[1]), anonfield(typs[32])}, []*Node{anonfield(typs[32]), anonfield(typs[11])})
|
||||||
typs[93] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32]), anonfield(typs[32])}, []*Node{anonfield(typs[92])})
|
typs[93] = types.NewSlice(typs[2])
|
||||||
typs[94] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[92])})
|
typs[94] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32]), anonfield(typs[32])}, []*Node{anonfield(typs[93])})
|
||||||
typs[95] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[92]), anonfield(typs[32])}, []*Node{anonfield(typs[92])})
|
typs[95] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[93])})
|
||||||
typs[96] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[48])}, nil)
|
typs[96] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[93]), anonfield(typs[32])}, []*Node{anonfield(typs[93])})
|
||||||
typs[97] = functype(nil, []*Node{anonfield(typs[57]), anonfield(typs[48])}, nil)
|
typs[97] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[47])}, nil)
|
||||||
typs[98] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[48])}, []*Node{anonfield(typs[11])})
|
typs[98] = functype(nil, []*Node{anonfield(typs[58]), anonfield(typs[47])}, nil)
|
||||||
typs[99] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
|
typs[99] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[47])}, []*Node{anonfield(typs[11])})
|
||||||
typs[100] = functype(nil, []*Node{anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[15])})
|
typs[100] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
|
||||||
typs[101] = functype(nil, []*Node{anonfield(typs[17]), anonfield(typs[17])}, []*Node{anonfield(typs[17])})
|
typs[101] = functype(nil, []*Node{anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[15])})
|
||||||
typs[102] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[15])})
|
typs[102] = functype(nil, []*Node{anonfield(typs[17]), anonfield(typs[17])}, []*Node{anonfield(typs[17])})
|
||||||
typs[103] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[17])})
|
typs[103] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[15])})
|
||||||
typs[104] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[59])})
|
typs[104] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[17])})
|
||||||
typs[105] = functype(nil, []*Node{anonfield(typs[15])}, []*Node{anonfield(typs[13])})
|
typs[105] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[60])})
|
||||||
typs[106] = functype(nil, []*Node{anonfield(typs[17])}, []*Node{anonfield(typs[13])})
|
typs[106] = functype(nil, []*Node{anonfield(typs[15])}, []*Node{anonfield(typs[13])})
|
||||||
typs[107] = functype(nil, []*Node{anonfield(typs[59])}, []*Node{anonfield(typs[13])})
|
typs[107] = functype(nil, []*Node{anonfield(typs[17])}, []*Node{anonfield(typs[13])})
|
||||||
typs[108] = functype(nil, []*Node{anonfield(typs[19]), anonfield(typs[19])}, []*Node{anonfield(typs[19])})
|
typs[108] = functype(nil, []*Node{anonfield(typs[60])}, []*Node{anonfield(typs[13])})
|
||||||
typs[109] = functype(nil, []*Node{anonfield(typs[48])}, nil)
|
typs[109] = functype(nil, []*Node{anonfield(typs[19]), anonfield(typs[19])}, []*Node{anonfield(typs[19])})
|
||||||
typs[110] = functype(nil, []*Node{anonfield(typs[48]), anonfield(typs[48])}, nil)
|
typs[110] = functype(nil, []*Node{anonfield(typs[47])}, nil)
|
||||||
|
typs[111] = functype(nil, []*Node{anonfield(typs[47]), anonfield(typs[47])}, nil)
|
||||||
return typs[:]
|
return typs[:]
|
||||||
}
|
}
|
||||||
|
@ -55,10 +55,12 @@ func slicebytetostringtmp([]byte) string
|
|||||||
func slicerunetostring(*[32]byte, []rune) string
|
func slicerunetostring(*[32]byte, []rune) string
|
||||||
func stringtoslicebyte(*[32]byte, string) []byte
|
func stringtoslicebyte(*[32]byte, string) []byte
|
||||||
func stringtoslicerune(*[32]rune, string) []rune
|
func stringtoslicerune(*[32]rune, string) []rune
|
||||||
func decoderune(string, int) (retv rune, retk int)
|
|
||||||
func slicecopy(to any, fr any, wid uintptr) int
|
func slicecopy(to any, fr any, wid uintptr) int
|
||||||
func slicestringcopy(to any, fr any) int
|
func slicestringcopy(to any, fr any) int
|
||||||
|
|
||||||
|
func decoderune(string, int) (retv rune, retk int)
|
||||||
|
func countrunes(string) int
|
||||||
|
|
||||||
// interface conversions
|
// interface conversions
|
||||||
func convI2I(typ *byte, elem any) (ret any)
|
func convI2I(typ *byte, elem any) (ret any)
|
||||||
|
|
||||||
|
@ -1098,7 +1098,14 @@ func (o *Order) expr(n, lhs *Node) *Node {
|
|||||||
OSTRARRAYBYTE,
|
OSTRARRAYBYTE,
|
||||||
OSTRARRAYBYTETMP,
|
OSTRARRAYBYTETMP,
|
||||||
OSTRARRAYRUNE:
|
OSTRARRAYRUNE:
|
||||||
o.call(n)
|
|
||||||
|
if isRuneCount(n) {
|
||||||
|
// len([]rune(s)) is rewritten to runtime.countrunes(s) later.
|
||||||
|
n.Left.Left = o.expr(n.Left.Left, nil)
|
||||||
|
} else {
|
||||||
|
o.call(n)
|
||||||
|
}
|
||||||
|
|
||||||
if lhs == nil || lhs.Op != ONAME || instrumenting {
|
if lhs == nil || lhs.Op != ONAME || instrumenting {
|
||||||
n = o.copyExpr(n, n.Type, false)
|
n = o.copyExpr(n, n.Type, false)
|
||||||
}
|
}
|
||||||
|
@ -538,6 +538,12 @@ opswitch:
|
|||||||
n.Left = walkexpr(n.Left, init)
|
n.Left = walkexpr(n.Left, init)
|
||||||
|
|
||||||
case OLEN, OCAP:
|
case OLEN, OCAP:
|
||||||
|
if isRuneCount(n) {
|
||||||
|
// Replace len([]rune(string)) with runtime.countrunes(string).
|
||||||
|
n = mkcall("countrunes", n.Type, init, conv(n.Left.Left, types.Types[TSTRING]))
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
n.Left = walkexpr(n.Left, init)
|
n.Left = walkexpr(n.Left, init)
|
||||||
|
|
||||||
// replace len(*[10]int) with 10.
|
// replace len(*[10]int) with 10.
|
||||||
@ -4085,3 +4091,9 @@ func canMergeLoads() bool {
|
|||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// isRuneCount reports whether n is of the form len([]rune(string)).
|
||||||
|
// These are optimized into a call to runtime.runecount.
|
||||||
|
func isRuneCount(n *Node) bool {
|
||||||
|
return Debug['N'] == 0 && !instrumenting && n.Op == OLEN && n.Left.Op == OSTRARRAYRUNE
|
||||||
|
}
|
||||||
|
@ -9,6 +9,7 @@ import (
|
|||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
"unicode/utf8"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Strings and slices that don't escape and fit into tmpBuf are stack allocated,
|
// Strings and slices that don't escape and fit into tmpBuf are stack allocated,
|
||||||
@ -110,6 +111,43 @@ var stringdata = []struct{ name, data string }{
|
|||||||
{"MixedLength", "$Ѐࠀက퀀𐀀\U00040000\U0010FFFF"},
|
{"MixedLength", "$Ѐࠀက퀀𐀀\U00040000\U0010FFFF"},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var sinkInt int
|
||||||
|
|
||||||
|
func BenchmarkRuneCount(b *testing.B) {
|
||||||
|
// Each sub-benchmark counts the runes in a string in a different way.
|
||||||
|
b.Run("lenruneslice", func(b *testing.B) {
|
||||||
|
for _, sd := range stringdata {
|
||||||
|
b.Run(sd.name, func(b *testing.B) {
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
sinkInt += len([]rune(sd.data))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("rangeloop", func(b *testing.B) {
|
||||||
|
for _, sd := range stringdata {
|
||||||
|
b.Run(sd.name, func(b *testing.B) {
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
n := 0
|
||||||
|
for range sd.data {
|
||||||
|
n++
|
||||||
|
}
|
||||||
|
sinkInt += n
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("utf8.RuneCountInString", func(b *testing.B) {
|
||||||
|
for _, sd := range stringdata {
|
||||||
|
b.Run(sd.name, func(b *testing.B) {
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
sinkInt += utf8.RuneCountInString(sd.data)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func BenchmarkRuneIterate(b *testing.B) {
|
func BenchmarkRuneIterate(b *testing.B) {
|
||||||
b.Run("range", func(b *testing.B) {
|
b.Run("range", func(b *testing.B) {
|
||||||
for _, sd := range stringdata {
|
for _, sd := range stringdata {
|
||||||
|
@ -39,6 +39,15 @@ const (
|
|||||||
hicb = 0xBF // 1011 1111
|
hicb = 0xBF // 1011 1111
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// countrunes returns the number of runes in s.
|
||||||
|
func countrunes(s string) int {
|
||||||
|
n := 0
|
||||||
|
for range s {
|
||||||
|
n++
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
// decoderune returns the non-ASCII rune at the start of
|
// decoderune returns the non-ASCII rune at the start of
|
||||||
// s[k:] and the index after the rune in s.
|
// s[k:] and the index after the rune in s.
|
||||||
//
|
//
|
||||||
|
@ -212,14 +212,25 @@ func TestSequencing(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that a range loop and a []int conversion visit the same runes.
|
func runtimeRuneCount(s string) int {
|
||||||
|
return len([]rune(s)) // Replaced by gc with call to runtime.countrunes(s).
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that a range loop, len([]rune(string)) optimization and
|
||||||
|
// []rune conversions visit the same runes.
|
||||||
// Not really a test of this package, but the assumption is used here and
|
// Not really a test of this package, but the assumption is used here and
|
||||||
// it's good to verify
|
// it's good to verify.
|
||||||
func TestIntConversion(t *testing.T) {
|
func TestRuntimeConversion(t *testing.T) {
|
||||||
for _, ts := range testStrings {
|
for _, ts := range testStrings {
|
||||||
|
count := RuneCountInString(ts)
|
||||||
|
if n := runtimeRuneCount(ts); n != count {
|
||||||
|
t.Errorf("%q: len([]rune()) counted %d runes; got %d from RuneCountInString", ts, n, count)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
runes := []rune(ts)
|
runes := []rune(ts)
|
||||||
if RuneCountInString(ts) != len(runes) {
|
if n := len(runes); n != count {
|
||||||
t.Errorf("%q: expected %d runes; got %d", ts, len(runes), RuneCountInString(ts))
|
t.Errorf("%q: []rune() has length %d; got %d from RuneCountInString", ts, n, count)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
i := 0
|
i := 0
|
||||||
|
15
test/codegen/strings.go
Normal file
15
test/codegen/strings.go
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// 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
|
||||||
|
|
||||||
|
// 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))
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user