| 
									
										
										
										
											2018-03-08 10:57:10 +01: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 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This file contains codegen tests related to arithmetic | 
					
						
							| 
									
										
										
										
											2018-03-15 11:15:22 +01:00
										 |  |  | // simplifications and optimizations on integer types. | 
					
						
							|  |  |  | // For codegen tests on float types, see floats.go. | 
					
						
							| 
									
										
										
										
											2018-03-08 10:57:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-15 15:20:56 -05:00
										 |  |  | // ----------------- // | 
					
						
							|  |  |  | //    Addition       // | 
					
						
							|  |  |  | // ----------------- // | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func AddLargeConst(a uint64, out []uint64) { | 
					
						
							|  |  |  | 	// ppc64x/power10:"ADD\t[$]4294967296," | 
					
						
							| 
									
										
										
										
											2023-10-18 13:42:25 -05:00
										 |  |  | 	// ppc64x/power9:"MOVD\t[$]1", "SLD\t[$]32" "ADD\tR[0-9]*" | 
					
						
							|  |  |  | 	// ppc64x/power8:"MOVD\t[$]1", "SLD\t[$]32" "ADD\tR[0-9]*" | 
					
						
							| 
									
										
										
										
											2023-09-15 15:20:56 -05:00
										 |  |  | 	out[0] = a + 0x100000000 | 
					
						
							|  |  |  | 	// ppc64x/power10:"ADD\t[$]-8589934592," | 
					
						
							| 
									
										
										
										
											2023-10-18 13:42:25 -05:00
										 |  |  | 	// ppc64x/power9:"MOVD\t[$]-1", "SLD\t[$]33" "ADD\tR[0-9]*" | 
					
						
							|  |  |  | 	// ppc64x/power8:"MOVD\t[$]-1", "SLD\t[$]33" "ADD\tR[0-9]*" | 
					
						
							| 
									
										
										
										
											2023-09-15 15:20:56 -05:00
										 |  |  | 	out[1] = a + 0xFFFFFFFE00000000 | 
					
						
							| 
									
										
										
										
											2024-02-16 13:29:16 -06:00
										 |  |  | 	// ppc64x/power10:"ADD\t[$]1234567," | 
					
						
							|  |  |  | 	// ppc64x/power9:"ADDIS\t[$]19,", "ADD\t[$]-10617," | 
					
						
							|  |  |  | 	// ppc64x/power8:"ADDIS\t[$]19,", "ADD\t[$]-10617," | 
					
						
							|  |  |  | 	out[2] = a + 1234567 | 
					
						
							|  |  |  | 	// ppc64x/power10:"ADD\t[$]-1234567," | 
					
						
							|  |  |  | 	// ppc64x/power9:"ADDIS\t[$]-19,", "ADD\t[$]10617," | 
					
						
							|  |  |  | 	// ppc64x/power8:"ADDIS\t[$]-19,", "ADD\t[$]10617," | 
					
						
							|  |  |  | 	out[3] = a - 1234567 | 
					
						
							|  |  |  | 	// ppc64x/power10:"ADD\t[$]2147450879," | 
					
						
							|  |  |  | 	// ppc64x/power9:"ADDIS\t[$]32767,", "ADD\t[$]32767," | 
					
						
							|  |  |  | 	// ppc64x/power8:"ADDIS\t[$]32767,", "ADD\t[$]32767," | 
					
						
							|  |  |  | 	out[4] = a + 0x7FFF7FFF | 
					
						
							|  |  |  | 	// ppc64x/power10:"ADD\t[$]-2147483647," | 
					
						
							|  |  |  | 	// ppc64x/power9:"ADDIS\t[$]-32768,", "ADD\t[$]1," | 
					
						
							|  |  |  | 	// ppc64x/power8:"ADDIS\t[$]-32768,", "ADD\t[$]1," | 
					
						
							|  |  |  | 	out[5] = a - 2147483647 | 
					
						
							|  |  |  | 	// ppc64x:"ADDIS\t[$]-32768,", ^"ADD\t" | 
					
						
							|  |  |  | 	out[6] = a - 2147483648 | 
					
						
							|  |  |  | 	// ppc64x:"ADD\t[$]2147450880,", ^"ADDIS\t" | 
					
						
							|  |  |  | 	out[7] = a + 0x7FFF8000 | 
					
						
							|  |  |  | 	// ppc64x:"ADD\t[$]-32768,", ^"ADDIS\t" | 
					
						
							|  |  |  | 	out[8] = a - 32768 | 
					
						
							|  |  |  | 	// ppc64x/power10:"ADD\t[$]-32769," | 
					
						
							|  |  |  | 	// ppc64x/power9:"ADDIS\t[$]-1,", "ADD\t[$]32767," | 
					
						
							|  |  |  | 	// ppc64x/power8:"ADDIS\t[$]-1,", "ADD\t[$]32767," | 
					
						
							|  |  |  | 	out[9] = a - 32769 | 
					
						
							| 
									
										
										
										
											2023-09-15 15:20:56 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-15 09:31:39 +00:00
										 |  |  | // ----------------- // | 
					
						
							|  |  |  | //    Subtraction    // | 
					
						
							|  |  |  | // ----------------- // | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-06 13:13:48 +00:00
										 |  |  | var ef int | 
					
						
							| 
									
										
										
										
											2019-03-13 13:52:17 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-06 13:13:48 +00:00
										 |  |  | func SubMem(arr []int, b, c, d int) int { | 
					
						
							| 
									
										
										
										
											2018-04-29 10:42:14 +00:00
										 |  |  | 	// 386:`SUBL\s[A-Z]+,\s8\([A-Z]+\)` | 
					
						
							| 
									
										
										
										
											2018-06-29 02:11:53 +00:00
										 |  |  | 	// amd64:`SUBQ\s[A-Z]+,\s16\([A-Z]+\)` | 
					
						
							| 
									
										
										
										
											2018-04-29 10:42:14 +00:00
										 |  |  | 	arr[2] -= b | 
					
						
							|  |  |  | 	// 386:`SUBL\s[A-Z]+,\s12\([A-Z]+\)` | 
					
						
							| 
									
										
										
										
											2018-06-29 02:11:53 +00:00
										 |  |  | 	// amd64:`SUBQ\s[A-Z]+,\s24\([A-Z]+\)` | 
					
						
							| 
									
										
										
										
											2018-04-29 10:42:14 +00:00
										 |  |  | 	arr[3] -= b | 
					
						
							| 
									
										
										
										
											2018-06-26 02:58:54 +00:00
										 |  |  | 	// 386:`DECL\s16\([A-Z]+\)` | 
					
						
							|  |  |  | 	arr[4]-- | 
					
						
							|  |  |  | 	// 386:`ADDL\s[$]-20,\s20\([A-Z]+\)` | 
					
						
							|  |  |  | 	arr[5] -= 20 | 
					
						
							| 
									
										
										
										
											2020-03-24 16:46:49 +00:00
										 |  |  | 	// 386:`SUBL\s\([A-Z]+\)\([A-Z]+\*4\),\s[A-Z]+` | 
					
						
							| 
									
										
										
										
											2018-10-06 13:13:48 +00:00
										 |  |  | 	ef -= arr[b] | 
					
						
							| 
									
										
										
										
											2020-03-24 16:46:49 +00:00
										 |  |  | 	// 386:`SUBL\s[A-Z]+,\s\([A-Z]+\)\([A-Z]+\*4\)` | 
					
						
							| 
									
										
										
										
											2018-10-06 13:13:48 +00:00
										 |  |  | 	arr[c] -= b | 
					
						
							| 
									
										
										
										
											2020-03-24 16:46:49 +00:00
										 |  |  | 	// 386:`ADDL\s[$]-15,\s\([A-Z]+\)\([A-Z]+\*4\)` | 
					
						
							| 
									
										
										
										
											2018-10-06 13:13:48 +00:00
										 |  |  | 	arr[d] -= 15 | 
					
						
							| 
									
										
										
										
											2020-03-24 16:46:49 +00:00
										 |  |  | 	// 386:`DECL\s\([A-Z]+\)\([A-Z]+\*4\)` | 
					
						
							| 
									
										
										
										
											2018-10-09 11:01:34 +00:00
										 |  |  | 	arr[b]-- | 
					
						
							| 
									
										
										
										
											2018-10-29 08:34:42 +00:00
										 |  |  | 	// amd64:`DECQ\s64\([A-Z]+\)` | 
					
						
							|  |  |  | 	arr[8]-- | 
					
						
							| 
									
										
										
										
											2018-04-15 09:31:39 +00:00
										 |  |  | 	// 386:"SUBL\t4" | 
					
						
							|  |  |  | 	// amd64:"SUBQ\t8" | 
					
						
							|  |  |  | 	return arr[0] - arr[1] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | func SubFromConst(a int) int { | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `SUBC\tR[0-9]+,\s[$]40,\sR` | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64: "ADDI\t\\$-40","NEG" | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | 	b := 40 - a | 
					
						
							|  |  |  | 	return b | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func SubFromConstNeg(a int) int { | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64: "ADD\t\\$40" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64: "ADDV[U]\t\\$40" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips: "ADD[U]\t\\$40" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64: "ADDV[U]\t\\$40" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `ADD\t[$]40,\sR[0-9]+,\sR` | 
					
						
							| 
									
										
										
											
												cmd/compile/internal/ssa: remove double negation with addition on riscv64
On riscv64, subtraction from a constant is typically implemented as an
ADDI with the negative constant, followed by a negation. However this can
lead to multiple NEG/ADDI/NEG sequences that can be optimised out.
For example, runtime.(*_panic).nextDefer currently contains:
   lbu     t0, 0(t0)
   addi    t0, t0, -8
   neg     t0, t0
   addi    t0, t0, -7
   neg     t0, t0
Which is now optimised to:
   lbu     t0, 0(t0)
   addi    t0, t0, -1
Change-Id: Idf5815e6db2e3705cc4a4811ca9130a064ae3d80
Reviewed-on: https://go-review.googlesource.com/c/go/+/652318
Reviewed-by: Cherry Mui <cherryyz@google.com>
Reviewed-by: Meng Zhuo <mengzhuo1203@gmail.com>
LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
Reviewed-by: Mark Ryan <markdryan@rivosinc.com>
Reviewed-by: David Chase <drchase@google.com>
											
										 
											2025-02-24 00:37:45 +11:00
										 |  |  | 	// riscv64: "ADDI\t\\$40",-"NEG" | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | 	c := 40 - (-a) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func SubSubFromConst(a int) int { | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64: "ADD\t\\$20" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64: "ADDV[U]\t\\$20" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips: "ADD[U]\t\\$20" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64: "ADDV[U]\t\\$20" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `ADD\t[$]20,\sR[0-9]+,\sR` | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64: "ADDI\t\\$20",-"NEG" | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | 	c := 40 - (20 - a) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func AddSubFromConst(a int) int { | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `SUBC\tR[0-9]+,\s[$]60,\sR` | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64: "ADDI\t\\$-60","NEG" | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | 	c := 40 + (20 - a) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NegSubFromConst(a int) int { | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64: "SUB\t\\$20" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64: "ADDV[U]\t\\$-20" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips: "ADD[U]\t\\$-20" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64: "ADDV[U]\t\\$-20" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `ADD\t[$]-20,\sR[0-9]+,\sR` | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64: "ADDI\t\\$-20" | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | 	c := -(20 - a) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NegAddFromConstNeg(a int) int { | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64: "SUB\t\\$40","NEG" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64: "ADDV[U]\t\\$-40","SUBV" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips: "ADD[U]\t\\$-40","SUB" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64: "ADDV[U]\t\\$-40","SUBV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `SUBC\tR[0-9]+,\s[$]40,\sR` | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64: "ADDI\t\\$-40","NEG" | 
					
						
							| 
									
										
										
										
											2020-08-20 15:06:23 -05:00
										 |  |  | 	c := -(-40 + a) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 12:36:17 +02:00
										 |  |  | func SubSubNegSimplify(a, b int) int { | 
					
						
							|  |  |  | 	// amd64:"NEGQ" | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64:"NEG" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64:"SUBV" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips:"SUB" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64:"SUBV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"NEG" | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64:"NEG",-"SUB" | 
					
						
							| 
									
										
										
										
											2021-08-25 12:36:17 +02:00
										 |  |  | 	r := (a - b) - a | 
					
						
							|  |  |  | 	return r | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func SubAddSimplify(a, b int) int { | 
					
						
							|  |  |  | 	// amd64:-"SUBQ",-"ADDQ" | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64:-"SUB",-"ADD" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64:-"SUBV",-"ADDV" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips:-"SUB",-"ADD" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64:-"SUBV",-"ADDV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:-"SUB",-"ADD" | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64:-"SUB",-"ADD" | 
					
						
							| 
									
										
										
										
											2021-08-25 12:36:17 +02:00
										 |  |  | 	r := a + (b - a) | 
					
						
							|  |  |  | 	return r | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-19 07:24:04 +00:00
										 |  |  | func SubAddSimplify2(a, b, c int) (int, int, int, int, int, int) { | 
					
						
							|  |  |  | 	// amd64:-"ADDQ" | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64:-"ADD" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips:"SUB",-"ADD" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64:"SUBV",-"ADDV" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64:"SUBV",-"ADDV" | 
					
						
							| 
									
										
										
										
											2023-03-19 07:24:04 +00:00
										 |  |  | 	r := (a + b) - (a + c) | 
					
						
							|  |  |  | 	// amd64:-"ADDQ" | 
					
						
							|  |  |  | 	r1 := (a + b) - (c + a) | 
					
						
							|  |  |  | 	// amd64:-"ADDQ" | 
					
						
							|  |  |  | 	r2 := (b + a) - (a + c) | 
					
						
							|  |  |  | 	// amd64:-"ADDQ" | 
					
						
							|  |  |  | 	r3 := (b + a) - (c + a) | 
					
						
							|  |  |  | 	// amd64:-"SUBQ" | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64:-"SUB" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips:"ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64:"ADDV",-"SUBV" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64:"ADDV",-"SUBV" | 
					
						
							| 
									
										
										
										
											2023-03-19 07:24:04 +00:00
										 |  |  | 	r4 := (a - c) + (c + b) | 
					
						
							|  |  |  | 	// amd64:-"SUBQ" | 
					
						
							|  |  |  | 	r5 := (a - c) + (b + c) | 
					
						
							|  |  |  | 	return r, r1, r2, r3, r4, r5 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 12:36:17 +02:00
										 |  |  | func SubAddNegSimplify(a, b int) int { | 
					
						
							|  |  |  | 	// amd64:"NEGQ",-"ADDQ",-"SUBQ" | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64:"NEG",-"ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64:"SUBV",-"ADDV" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips:"SUB",-"ADD" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64:"SUBV",-"ADDV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"NEG",-"ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64:"NEG",-"ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2021-08-25 12:36:17 +02:00
										 |  |  | 	r := a - (b + a) | 
					
						
							|  |  |  | 	return r | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func AddAddSubSimplify(a, b, c int) int { | 
					
						
							|  |  |  | 	// amd64:-"SUBQ" | 
					
						
							| 
									
										
										
										
											2025-05-17 14:30:07 +08:00
										 |  |  | 	// arm64:"ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2025-05-14 14:35:41 +08:00
										 |  |  | 	// loong64:"ADDV",-"SUBV" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:27:37 +08:00
										 |  |  | 	// mips:"ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2025-05-17 00:14:31 +08:00
										 |  |  | 	// mips64:"ADDV",-"SUBV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:-"SUB" | 
					
						
							| 
									
										
										
										
											2025-02-24 23:08:46 +11:00
										 |  |  | 	// riscv64:"ADD","ADD",-"SUB" | 
					
						
							| 
									
										
										
										
											2021-08-25 12:36:17 +02:00
										 |  |  | 	r := a + (b + (c - a)) | 
					
						
							|  |  |  | 	return r | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-24 23:24:42 +11:00
										 |  |  | func NegToInt32(a int) int { | 
					
						
							| 
									
										
										
										
											2022-10-12 17:37:23 +11:00
										 |  |  | 	// riscv64: "NEGW",-"MOVW" | 
					
						
							| 
									
										
										
										
											2025-02-24 23:24:42 +11:00
										 |  |  | 	r := int(int32(-a)) | 
					
						
							|  |  |  | 	return r | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | // -------------------- // | 
					
						
							|  |  |  | //    Multiplication    // | 
					
						
							|  |  |  | // -------------------- // | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 10:57:10 +01:00
										 |  |  | func Pow2Muls(n1, n2 int) (int, int) { | 
					
						
							|  |  |  | 	// amd64:"SHLQ\t[$]5",-"IMULQ" | 
					
						
							|  |  |  | 	// 386:"SHLL\t[$]5",-"IMULL" | 
					
						
							|  |  |  | 	// arm:"SLL\t[$]5",-"MUL" | 
					
						
							|  |  |  | 	// arm64:"LSL\t[$]5",-"MUL" | 
					
						
							| 
									
										
										
										
											2025-05-22 16:21:10 +08:00
										 |  |  | 	// loong64:"SLLV\t[$]5",-"MULV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SLD\t[$]5",-"MUL" | 
					
						
							| 
									
										
										
										
											2018-03-08 10:57:10 +01:00
										 |  |  | 	a := n1 * 32 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// amd64:"SHLQ\t[$]6",-"IMULQ" | 
					
						
							|  |  |  | 	// 386:"SHLL\t[$]6",-"IMULL" | 
					
						
							|  |  |  | 	// arm:"SLL\t[$]6",-"MUL" | 
					
						
							| 
									
										
										
										
											2018-09-27 06:21:14 +00:00
										 |  |  | 	// arm64:`NEG\sR[0-9]+<<6,\sR[0-9]+`,-`LSL`,-`MUL` | 
					
						
							| 
									
										
										
										
											2025-05-22 16:21:10 +08:00
										 |  |  | 	// loong64:"SLLV\t[$]6",-"MULV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SLD\t[$]6","NEG\\sR[0-9]+,\\sR[0-9]+",-"MUL" | 
					
						
							| 
									
										
										
										
											2018-03-08 10:57:10 +01:00
										 |  |  | 	b := -64 * n2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												cmd/compile: prefer an add when shifting left by 1
ADD(Q|L) has generally twice the throughput.
Came up in CL 626998.
Throughput by arch:
Zen 4:
SHLL (R64, 1):   0.5
ADD  (R64, R64): 0.25
Intel Alder Lake:
SHLL (R64, 1):   0.5
ADD  (R64, R64): 0.2
Intel Haswell:
SHLL (R64, 1):   0.5
ADD  (R64, R64): 0.25
Also include a minor opt for:
(x + x) << c -> x << (c + 1)
Before this, the code:
func addShift(x int64) int64 {
    return (x + x) << 1
}
emitted two instructions:
        ADDQ    AX, AX
        SHLQ    $1, AX
but we can do it in a single shift:
        SHLQ    $2, AX
Add a codegen test for clearing the last bit.
compilecmp linux/amd64:
math
math.sqrt 243 -> 242  (-0.41%)
math [cmd/compile]
math.sqrt 243 -> 242  (-0.41%)
runtime
runtime.selectgo 5455 -> 5445  (-0.18%)
runtime.sysargs 665 -> 662  (-0.45%)
runtime.isPinned 145 -> 141  (-2.76%)
runtime.atoi64 198 -> 194  (-2.02%)
runtime.setPinned 714 -> 709  (-0.70%)
runtime [cmd/compile]
runtime.sysargs 665 -> 662  (-0.45%)
runtime.setPinned 714 -> 709  (-0.70%)
runtime.atoi64 198 -> 194  (-2.02%)
runtime.isPinned 145 -> 141  (-2.76%)
strconv
strconv.computeBounds 109 -> 107  (-1.83%)
strconv.FormatInt 201 -> 197  (-1.99%)
strconv.ryuFtoaShortest 1298 -> 1266  (-2.47%)
strconv.small 144 -> 134  (-6.94%)
strconv.AppendInt 357 -> 344  (-3.64%)
strconv.ryuDigits32 490 -> 488  (-0.41%)
strconv.AppendUint 342 -> 340  (-0.58%)
strconv [cmd/compile]
strconv.FormatInt 201 -> 197  (-1.99%)
strconv.ryuFtoaShortest 1298 -> 1266  (-2.47%)
strconv.ryuDigits32 490 -> 488  (-0.41%)
strconv.AppendUint 342 -> 340  (-0.58%)
strconv.computeBounds 109 -> 107  (-1.83%)
strconv.small 144 -> 134  (-6.94%)
strconv.AppendInt 357 -> 344  (-3.64%)
image
image.Rectangle.Inset 101 -> 97  (-3.96%)
regexp/syntax
regexp/syntax.inCharClass.func1 111 -> 110  (-0.90%)
regexp/syntax.(*compiler).quest 586 -> 573  (-2.22%)
regexp/syntax.ranges.Less 153 -> 150  (-1.96%)
regexp/syntax.(*compiler).loop 583 -> 568  (-2.57%)
time
time.Time.Before 179 -> 161  (-10.06%)
time.Time.Compare 189 -> 166  (-12.17%)
time.Time.Sub 444 -> 425  (-4.28%)
time.Time.UnixMicro 106 -> 95  (-10.38%)
time.div 592 -> 587  (-0.84%)
time.Time.UnixNano 85 -> 78  (-8.24%)
time.(*Time).UnixMilli 141 -> 140  (-0.71%)
time.Time.UnixMilli 106 -> 95  (-10.38%)
time.(*Time).UnixMicro 141 -> 140  (-0.71%)
time.Time.After 179 -> 161  (-10.06%)
time.Time.Equal 170 -> 150  (-11.76%)
time.Time.AppendBinary 766 -> 757  (-1.17%)
time.Time.IsZero 74 -> 66  (-10.81%)
time.(*Time).UnixNano 124 -> 113  (-8.87%)
time.(*Time).IsZero 113 -> 108  (-4.42%)
regexp
regexp.(*Regexp).FindAllStringSubmatch.func1 590 -> 569  (-3.56%)
regexp.QuoteMeta 485 -> 469  (-3.30%)
regexp/syntax [cmd/compile]
regexp/syntax.inCharClass.func1 111 -> 110  (-0.90%)
regexp/syntax.(*compiler).loop 583 -> 568  (-2.57%)
regexp/syntax.(*compiler).quest 586 -> 573  (-2.22%)
regexp/syntax.ranges.Less 153 -> 150  (-1.96%)
encoding/base64
encoding/base64.decodedLen 92 -> 90  (-2.17%)
encoding/base64.(*Encoding).DecodedLen 99 -> 97  (-2.02%)
time [cmd/compile]
time.(*Time).IsZero 113 -> 108  (-4.42%)
time.Time.IsZero 74 -> 66  (-10.81%)
time.(*Time).UnixNano 124 -> 113  (-8.87%)
time.Time.UnixMilli 106 -> 95  (-10.38%)
time.Time.Equal 170 -> 150  (-11.76%)
time.Time.UnixMicro 106 -> 95  (-10.38%)
time.(*Time).UnixMicro 141 -> 140  (-0.71%)
time.Time.Before 179 -> 161  (-10.06%)
time.Time.UnixNano 85 -> 78  (-8.24%)
time.Time.AppendBinary 766 -> 757  (-1.17%)
time.div 592 -> 587  (-0.84%)
time.Time.After 179 -> 161  (-10.06%)
time.Time.Compare 189 -> 166  (-12.17%)
time.(*Time).UnixMilli 141 -> 140  (-0.71%)
time.Time.Sub 444 -> 425  (-4.28%)
index/suffixarray
index/suffixarray.sais_8_32 1677 -> 1645  (-1.91%)
index/suffixarray.sais_32 1677 -> 1645  (-1.91%)
index/suffixarray.sais_64 1677 -> 1654  (-1.37%)
index/suffixarray.sais_8_64 1677 -> 1654  (-1.37%)
index/suffixarray.writeInt 249 -> 247  (-0.80%)
os
os.Expand 1070 -> 1051  (-1.78%)
os.Chtimes 787 -> 774  (-1.65%)
regexp [cmd/compile]
regexp.(*Regexp).FindAllStringSubmatch.func1 590 -> 569  (-3.56%)
regexp.QuoteMeta 485 -> 469  (-3.30%)
encoding/base64 [cmd/compile]
encoding/base64.decodedLen 92 -> 90  (-2.17%)
encoding/base64.(*Encoding).DecodedLen 99 -> 97  (-2.02%)
encoding/hex
encoding/hex.Encode 138 -> 136  (-1.45%)
encoding/hex.(*decoder).Read 830 -> 824  (-0.72%)
crypto/des
crypto/des.initFeistelBox 235 -> 229  (-2.55%)
crypto/des.cryptBlock 549 -> 538  (-2.00%)
os [cmd/compile]
os.Chtimes 787 -> 774  (-1.65%)
os.Expand 1070 -> 1051  (-1.78%)
math/big
math/big.newFloat 238 -> 223  (-6.30%)
math/big.nat.mul 2138 -> 2122  (-0.75%)
math/big.karatsubaSqr 1372 -> 1369  (-0.22%)
math/big.(*Float).sqrtInverse 895 -> 878  (-1.90%)
math/big.basicSqr 1032 -> 1017  (-1.45%)
cmd/vendor/golang.org/x/sys/unix
cmd/vendor/golang.org/x/sys/unix.TimeToTimespec 72 -> 66  (-8.33%)
encoding/json
encoding/json.Indent 404 -> 403  (-0.25%)
encoding/json.MarshalIndent 303 -> 297  (-1.98%)
testing
testing.(*T).Deadline 84 -> 82  (-2.38%)
testing.(*M).Run 3545 -> 3525  (-0.56%)
archive/zip
archive/zip.headerFileInfo.ModTime 229 -> 223  (-2.62%)
encoding/gob
encoding/gob.(*encoderState).encodeInt 474 -> 469  (-1.05%)
crypto/elliptic
crypto/elliptic.Marshal 728 -> 714  (-1.92%)
debug/buildinfo
debug/buildinfo.readString 325 -> 315  (-3.08%)
image/png
image/png.(*decoder).readImagePass 10866 -> 10834  (-0.29%)
archive/tar
archive/tar.Header.allowedFormats.func3 1768 -> 1736  (-1.81%)
archive/tar.formatPAXTime 389 -> 358  (-7.97%)
archive/tar.(*Writer).writeGNUHeader 741 -> 727  (-1.89%)
archive/tar.readGNUSparseMap0x1 709 -> 695  (-1.97%)
archive/tar.(*Writer).templateV7Plus 915 -> 909  (-0.66%)
crypto/internal/cryptotest
crypto/internal/cryptotest.TestHash.func4 890 -> 879  (-1.24%)
crypto/internal/cryptotest.TestStream.func6.1 646 -> 645  (-0.15%)
crypto/internal/cryptotest.testCipher.func3 1300 -> 1289  (-0.85%)
internal/pkgbits
internal/pkgbits.(*Encoder).Int64 113 -> 103  (-8.85%)
internal/pkgbits.(*Encoder).rawVarint 74 -> 72  (-2.70%)
testing/quick
testing/quick.(*Config).getRand 316 -> 315  (-0.32%)
log/slog
log/slog.TimeValue 489 -> 479  (-2.04%)
runtime/pprof
runtime/pprof.(*profileBuilder).build 2341 -> 2322  (-0.81%)
internal/coverage/cfile
internal/coverage/cfile.(*emitState).openMetaFile 824 -> 822  (-0.24%)
internal/coverage/cfile.(*emitState).openCounterFile 904 -> 892  (-1.33%)
cmd/internal/objabi
cmd/internal/objabi.expandArgs 1177 -> 1169  (-0.68%)
crypto/ecdsa
crypto/ecdsa.pointFromAffine 1162 -> 1144  (-1.55%)
net
net.minNonzeroTime 313 -> 308  (-1.60%)
net.cgoLookupAddrPTR 812 -> 797  (-1.85%)
net.(*IPNet).String 851 -> 827  (-2.82%)
net.IP.AppendText 488 -> 471  (-3.48%)
net.IPMask.String 281 -> 270  (-3.91%)
net.partialDeadline 374 -> 366  (-2.14%)
net.hexString 249 -> 240  (-3.61%)
net.IP.String 454 -> 453  (-0.22%)
internal/fuzz
internal/fuzz.newPcgRand 240 -> 234  (-2.50%)
crypto/x509
crypto/x509.(*Certificate).isValid 2642 -> 2611  (-1.17%)
cmd/internal/obj/s390x
cmd/internal/obj/s390x.buildop 33676 -> 33644  (-0.10%)
encoding/hex [cmd/compile]
encoding/hex.(*decoder).Read 830 -> 824  (-0.72%)
encoding/hex.Encode 138 -> 136  (-1.45%)
cmd/internal/objabi [cmd/compile]
cmd/internal/objabi.expandArgs 1177 -> 1169  (-0.68%)
math/big [cmd/compile]
math/big.(*Float).sqrtInverse 895 -> 878  (-1.90%)
math/big.nat.mul 2138 -> 2122  (-0.75%)
math/big.karatsubaSqr 1372 -> 1369  (-0.22%)
math/big.basicSqr 1032 -> 1017  (-1.45%)
math/big.newFloat 238 -> 223  (-6.30%)
encoding/json [cmd/compile]
encoding/json.MarshalIndent 303 -> 297  (-1.98%)
encoding/json.Indent 404 -> 403  (-0.25%)
cmd/covdata
main.(*metaMerge).emitCounters 985 -> 973  (-1.22%)
runtime/pprof [cmd/compile]
runtime/pprof.(*profileBuilder).build 2341 -> 2322  (-0.81%)
cmd/compile/internal/syntax
cmd/compile/internal/syntax.(*source).fill 722 -> 703  (-2.63%)
cmd/dist
main.runInstall 19081 -> 19049  (-0.17%)
crypto/tls
crypto/tls.extractPadding 176 -> 175  (-0.57%)
slices.Clone[[]crypto/tls.SignatureScheme,crypto/tls.SignatureScheme] 253 -> 247  (-2.37%)
slices.Clone[[]uint16,uint16] 253 -> 247  (-2.37%)
slices.Clone[[]crypto/tls.CurveID,crypto/tls.CurveID] 253 -> 247  (-2.37%)
crypto/tls.(*Config).cipherSuites 335 -> 326  (-2.69%)
slices.DeleteFunc[go.shape.[]crypto/tls.CurveID,go.shape.uint16] 437 -> 434  (-0.69%)
crypto/tls.dial 1349 -> 1339  (-0.74%)
slices.DeleteFunc[go.shape.[]uint16,go.shape.uint16] 437 -> 434  (-0.69%)
internal/pkgbits [cmd/compile]
internal/pkgbits.(*Encoder).Int64 113 -> 103  (-8.85%)
internal/pkgbits.(*Encoder).rawVarint 74 -> 72  (-2.70%)
cmd/compile/internal/syntax [cmd/compile]
cmd/compile/internal/syntax.(*source).fill 722 -> 703  (-2.63%)
cmd/internal/obj/s390x [cmd/compile]
cmd/internal/obj/s390x.buildop 33676 -> 33644  (-0.10%)
cmd/go/internal/trace
cmd/go/internal/trace.Flow 910 -> 886  (-2.64%)
cmd/go/internal/trace.(*Span).Done 311 -> 304  (-2.25%)
cmd/go/internal/trace.StartSpan 620 -> 615  (-0.81%)
cmd/internal/script
cmd/internal/script.(*Engine).Execute.func2 534 -> 528  (-1.12%)
cmd/link/internal/loader
cmd/link/internal/loader.(*Loader).SetSymSect 344 -> 338  (-1.74%)
net/http
net/http.(*Transport).queueForIdleConn 1797 -> 1766  (-1.73%)
net/http.(*Transport).getConn 2149 -> 2131  (-0.84%)
net/http.(*http2ClientConn).tooIdleLocked 207 -> 197  (-4.83%)
net/http.(*http2responseWriter).SetWriteDeadline.func1 520 -> 508  (-2.31%)
net/http.(*Cookie).Valid 837 -> 818  (-2.27%)
net/http.(*http2responseWriter).SetReadDeadline 373 -> 357  (-4.29%)
net/http.checkIfRange 701 -> 690  (-1.57%)
net/http.(*http2SettingsFrame).Value 325 -> 298  (-8.31%)
net/http.(*http2SettingsFrame).HasDuplicates 777 -> 767  (-1.29%)
net/http.(*Server).Serve 1746 -> 1739  (-0.40%)
net/http.http2traceGotConn 569 -> 556  (-2.28%)
net/http/pprof
net/http/pprof.collectProfile 242 -> 239  (-1.24%)
cmd/compile/internal/coverage
cmd/compile/internal/coverage.metaHashAndLen 439 -> 438  (-0.23%)
cmd/vendor/golang.org/x/telemetry/internal/upload
cmd/vendor/golang.org/x/telemetry/internal/upload.(*uploader).findWork 4570 -> 4540  (-0.66%)
cmd/vendor/golang.org/x/telemetry/internal/upload.(*uploader).reports 3604 -> 3572  (-0.89%)
cmd/compile/internal/coverage [cmd/compile]
cmd/compile/internal/coverage.metaHashAndLen 439 -> 438  (-0.23%)
cmd/vendor/golang.org/x/text/language
cmd/vendor/golang.org/x/text/language.regionGroupDist 287 -> 284  (-1.05%)
cmd/go/internal/vcweb
cmd/go/internal/vcweb.(*Server).overview.func1 1045 -> 1041  (-0.38%)
cmd/go/internal/vcs
cmd/go/internal/vcs.expand 761 -> 741  (-2.63%)
cmd/compile/internal/inline/inlheur
slices.stableCmpFunc[go.shape.struct 2300 -> 2284  (-0.70%)
cmd/compile/internal/inline/inlheur [cmd/compile]
slices.stableCmpFunc[go.shape.struct 2300 -> 2284  (-0.70%)
cmd/go/internal/modfetch/codehost
cmd/go/internal/modfetch/codehost.bzrParseStat 2217 -> 2213  (-0.18%)
cmd/link/internal/ld
cmd/link/internal/ld.decodetypeStructFieldCount 157 -> 152  (-3.18%)
cmd/link/internal/ld.(*Link).address 12559 -> 12495  (-0.51%)
cmd/link/internal/ld.(*dodataState).allocateDataSections 18345 -> 18205  (-0.76%)
cmd/link/internal/ld.elfshreloc 618 -> 616  (-0.32%)
cmd/link/internal/ld.(*deadcodePass).decodetypeMethods 794 -> 779  (-1.89%)
cmd/link/internal/ld.(*dodataState).assignDsymsToSection 668 -> 663  (-0.75%)
cmd/link/internal/ld.relocSectFn 285 -> 284  (-0.35%)
cmd/link/internal/ld.decodetypeIfaceMethodCount 146 -> 144  (-1.37%)
cmd/link/internal/ld.decodetypeArrayLen 157 -> 152  (-3.18%)
cmd/link/internal/arm64
cmd/link/internal/arm64.gensymlate.func1 895 -> 888  (-0.78%)
cmd/go/internal/modload
cmd/go/internal/modload.queryProxy.func3 1029 -> 1012  (-1.65%)
cmd/go/internal/load
cmd/go/internal/load.(*Package).setBuildInfo 8453 -> 8447  (-0.07%)
cmd/go/internal/clean
cmd/go/internal/clean.runClean 2120 -> 2104  (-0.75%)
cmd/compile/internal/ssa
cmd/compile/internal/ssa.(*poset).aliasnodes 2010 -> 1978  (-1.59%)
cmd/compile/internal/ssa.rewriteValueARM64_OpARM64MOVHstoreidx2 730 -> 719  (-1.51%)
cmd/compile/internal/ssa.(*debugState).buildLocationLists 3326 -> 3294  (-0.96%)
cmd/compile/internal/ssa.rewriteValueAMD64_OpAMD64ADDLconst 3069 -> 2941  (-4.17%)
cmd/compile/internal/ssa.(*debugState).processValue 9756 -> 9724  (-0.33%)
cmd/compile/internal/ssa.rewriteValueAMD64_OpAMD64ADDQconst 3069 -> 2941  (-4.17%)
cmd/compile/internal/ssa.(*poset).mergeroot 1079 -> 1054  (-2.32%)
cmd/compile/internal/ssa [cmd/compile]
cmd/compile/internal/ssa.rewriteValueARM64_OpARM64MOVHstoreidx2 730 -> 719  (-1.51%)
cmd/compile/internal/ssa.(*poset).aliasnodes 2010 -> 1978  (-1.59%)
cmd/compile/internal/ssa.(*poset).mergeroot 1079 -> 1054  (-2.32%)
cmd/compile/internal/ssa.rewriteValueAMD64_OpAMD64ADDQconst 3069 -> 2941  (-4.17%)
cmd/compile/internal/ssa.rewriteValueAMD64_OpAMD64ADDLconst 3069 -> 2941  (-4.17%)
file                                                before   after    Δ       %
math/bits.s                                         2352     2354     +2      +0.085%
math/bits [cmd/compile].s                           2352     2354     +2      +0.085%
math.s                                              35675    35674    -1      -0.003%
math [cmd/compile].s                                35675    35674    -1      -0.003%
runtime.s                                           577251   577245   -6      -0.001%
runtime [cmd/compile].s                             642419   642438   +19     +0.003%
sort.s                                              37434    37435    +1      +0.003%
strconv.s                                           48391    48343    -48     -0.099%
sort [cmd/compile].s                                37434    37435    +1      +0.003%
bufio.s                                             21386    21418    +32     +0.150%
strconv [cmd/compile].s                             48391    48343    -48     -0.099%
image.s                                             34978    35022    +44     +0.126%
regexp/syntax.s                                     81719    81781    +62     +0.076%
time.s                                              94341    94184    -157    -0.166%
regexp.s                                            60411    60399    -12     -0.020%
bufio [cmd/compile].s                               21512    21544    +32     +0.149%
encoding/binary.s                                   34062    34087    +25     +0.073%
regexp/syntax [cmd/compile].s                       81719    81781    +62     +0.076%
encoding/base64.s                                   11907    11903    -4      -0.034%
time [cmd/compile].s                                94341    94184    -157    -0.166%
index/suffixarray.s                                 41633    41527    -106    -0.255%
os.s                                                101770   101738   -32     -0.031%
regexp [cmd/compile].s                              60411    60399    -12     -0.020%
encoding/binary [cmd/compile].s                     37173    37198    +25     +0.067%
encoding/base64 [cmd/compile].s                     11907    11903    -4      -0.034%
os/exec.s                                           23900    23907    +7      +0.029%
encoding/hex.s                                      6038     6030     -8      -0.132%
crypto/des.s                                        5073     5056     -17     -0.335%
os [cmd/compile].s                                  102030   101998   -32     -0.031%
vendor/golang.org/x/net/http2/hpack.s               22027    22033    +6      +0.027%
math/big.s                                          164808   164753   -55     -0.033%
cmd/vendor/golang.org/x/sys/unix.s                  121450   121444   -6      -0.005%
encoding/json.s                                     110294   110287   -7      -0.006%
testing.s                                           115303   115281   -22     -0.019%
archive/zip.s                                       65329    65325    -4      -0.006%
os/user.s                                           10078    10080    +2      +0.020%
encoding/gob.s                                      143788   143783   -5      -0.003%
crypto/elliptic.s                                   30686    30704    +18     +0.059%
go/doc/comment.s                                    49401    49433    +32     +0.065%
debug/buildinfo.s                                   9095     9085     -10     -0.110%
image/png.s                                         36113    36081    -32     -0.089%
archive/tar.s                                       71994    71897    -97     -0.135%
crypto/internal/cryptotest.s                        60872    60849    -23     -0.038%
internal/pkgbits.s                                  20441    20429    -12     -0.059%
testing/quick.s                                     8236     8235     -1      -0.012%
log/slog.s                                          77568    77558    -10     -0.013%
internal/trace/internal/oldtrace.s                  52885    52896    +11     +0.021%
runtime/pprof.s                                     123978   123969   -9      -0.007%
internal/coverage/cfile.s                           25198    25184    -14     -0.056%
cmd/internal/objabi.s                               19954    19946    -8      -0.040%
crypto/ecdsa.s                                      29159    29141    -18     -0.062%
log/slog/internal/benchmarks.s                      6694     6695     +1      +0.015%
net.s                                               299569   299503   -66     -0.022%
os/exec [cmd/compile].s                             23888    23895    +7      +0.029%
internal/trace.s                                    179226   179240   +14     +0.008%
internal/fuzz.s                                     86190    86191    +1      +0.001%
crypto/x509.s                                       177195   177164   -31     -0.017%
cmd/internal/obj/s390x.s                            121642   121610   -32     -0.026%
cmd/internal/obj/ppc64.s                            140118   140122   +4      +0.003%
encoding/hex [cmd/compile].s                        6149     6141     -8      -0.130%
cmd/internal/objabi [cmd/compile].s                 19954    19946    -8      -0.040%
cmd/internal/obj/arm64.s                            158523   158555   +32     +0.020%
go/doc/comment [cmd/compile].s                      49512    49544    +32     +0.065%
math/big [cmd/compile].s                            166394   166339   -55     -0.033%
encoding/json [cmd/compile].s                       110712   110705   -7      -0.006%
cmd/covdata.s                                       39699    39687    -12     -0.030%
runtime/pprof [cmd/compile].s                       125209   125200   -9      -0.007%
cmd/compile/internal/syntax.s                       181755   181736   -19     -0.010%
cmd/dist.s                                          177893   177861   -32     -0.018%
crypto/tls.s                                        389157   389113   -44     -0.011%
internal/pkgbits [cmd/compile].s                    41644    41632    -12     -0.029%
cmd/compile/internal/syntax [cmd/compile].s         196105   196086   -19     -0.010%
cmd/compile/internal/types.s                        71315    71345    +30     +0.042%
cmd/internal/obj/s390x [cmd/compile].s              121733   121701   -32     -0.026%
cmd/go/internal/trace.s                             4796     4760     -36     -0.751%
cmd/internal/obj/arm64 [cmd/compile].s              168120   168147   +27     +0.016%
cmd/internal/obj/ppc64 [cmd/compile].s              140219   140223   +4      +0.003%
cmd/internal/script.s                               83442    83436    -6      -0.007%
cmd/link/internal/loader.s                          93299    93294    -5      -0.005%
net/http.s                                          620639   620472   -167    -0.027%
net/http/pprof.s                                    35016    35013    -3      -0.009%
cmd/compile/internal/coverage.s                     6668     6667     -1      -0.015%
cmd/vendor/golang.org/x/telemetry/internal/upload.s 34210    34148    -62     -0.181%
cmd/compile/internal/coverage [cmd/compile].s       6664     6663     -1      -0.015%
cmd/vendor/golang.org/x/text/language.s             48077    48074    -3      -0.006%
cmd/go/internal/vcweb.s                             45193    45189    -4      -0.009%
cmd/go/internal/vcs.s                               44749    44729    -20     -0.045%
cmd/compile/internal/inline/inlheur.s               83758    83742    -16     -0.019%
cmd/compile/internal/inline/inlheur [cmd/compile].s 84773    84757    -16     -0.019%
cmd/go/internal/modfetch/codehost.s                 89098    89094    -4      -0.004%
cmd/trace.s                                         257550   257564   +14     +0.005%
cmd/link/internal/ld.s                              641945   641706   -239    -0.037%
cmd/link/internal/arm64.s                           34805    34798    -7      -0.020%
cmd/go/internal/modload.s                           328971   328954   -17     -0.005%
cmd/go/internal/load.s                              178877   178871   -6      -0.003%
cmd/go/internal/clean.s                             11006    10990    -16     -0.145%
cmd/compile/internal/ssa.s                          3552843  3553347  +504    +0.014%
cmd/compile/internal/ssa [cmd/compile].s            3752511  3753123  +612    +0.016%
total                                               36179015 36178687 -328    -0.001%
Change-Id: I251c2898ccf3c9931d162d87dabbd49cf4ec73a5
Reviewed-on: https://go-review.googlesource.com/c/go/+/641757
Reviewed-by: Keith Randall <khr@google.com>
Auto-Submit: Keith Randall <khr@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
Reviewed-by: Cherry Mui <cherryyz@google.com>
LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
											
										 
											2025-01-11 19:26:57 +01:00
										 |  |  | func Mul_2(n1 int32, n2 int64) (int32, int64) { | 
					
						
							|  |  |  | 	// amd64:"ADDL", -"SHLL" | 
					
						
							|  |  |  | 	a := n1 * 2 | 
					
						
							|  |  |  | 	// amd64:"ADDQ", -"SHLQ" | 
					
						
							|  |  |  | 	b := n2 * 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-09 14:24:35 +02:00
										 |  |  | func Mul_96(n int) int { | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// amd64:`SHLQ\t[$]5`,`LEAQ\t\(.*\)\(.*\*2\),`,-`IMULQ` | 
					
						
							|  |  |  | 	// 386:`SHLL\t[$]5`,`LEAL\t\(.*\)\(.*\*2\),`,-`IMULL` | 
					
						
							|  |  |  | 	// arm64:`LSL\t[$]5`,`ADD\sR[0-9]+<<1,\sR[0-9]+`,-`MUL` | 
					
						
							|  |  |  | 	// arm:`SLL\t[$]5`,`ADD\sR[0-9]+<<1,\sR[0-9]+`,-`MUL` | 
					
						
							| 
									
										
										
										
											2025-05-22 16:21:10 +08:00
										 |  |  | 	// loong64:"ADDVU","SLLV\t[$]5",-"MULV" | 
					
						
							| 
									
										
										
										
											2020-06-04 10:55:01 -07:00
										 |  |  | 	// s390x:`SLD\t[$]5`,`SLD\t[$]6`,-`MULLD` | 
					
						
							| 
									
										
										
										
											2018-04-09 14:24:35 +02:00
										 |  |  | 	return n * 96 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 10:55:01 -07:00
										 |  |  | func Mul_n120(n int) int { | 
					
						
							| 
									
										
										
										
											2025-05-22 16:21:10 +08:00
										 |  |  | 	// loong64:"SLLV\t[$]3","SLLV\t[$]7","SUBVU",-"MULV" | 
					
						
							| 
									
										
										
										
											2020-06-04 10:55:01 -07:00
										 |  |  | 	// s390x:`SLD\t[$]3`,`SLD\t[$]7`,-`MULLD` | 
					
						
							|  |  |  | 	return n * -120 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-24 07:04:21 +00:00
										 |  |  | func MulMemSrc(a []uint32, b []float32) { | 
					
						
							|  |  |  | 	// 386:`IMULL\s4\([A-Z]+\),\s[A-Z]+` | 
					
						
							|  |  |  | 	a[0] *= a[1] | 
					
						
							| 
									
										
										
										
											2020-10-06 14:42:15 -07:00
										 |  |  | 	// 386/sse2:`MULSS\s4\([A-Z]+\),\sX[0-9]+` | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// amd64:`MULSS\s4\([A-Z]+\),\sX[0-9]+` | 
					
						
							| 
									
										
										
										
											2018-06-24 07:04:21 +00:00
										 |  |  | 	b[0] *= b[1] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | // Multiplications merging tests | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | func MergeMuls1(n int) int { | 
					
						
							| 
									
										
										
										
											2018-03-09 23:09:46 +03:00
										 |  |  | 	// amd64:"IMUL3Q\t[$]46" | 
					
						
							|  |  |  | 	// 386:"IMUL3L\t[$]46" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"MULLD\t[$]46" | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 	return 15*n + 31*n // 46n | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func MergeMuls2(n int) int { | 
					
						
							| 
									
										
										
										
											2021-04-12 14:00:49 -04:00
										 |  |  | 	// amd64:"IMUL3Q\t[$]23","(ADDQ\t[$]29)|(LEAQ\t29)" | 
					
						
							| 
									
										
										
										
											2018-03-09 23:09:46 +03:00
										 |  |  | 	// 386:"IMUL3L\t[$]23","ADDL\t[$]29" | 
					
						
							| 
									
										
										
										
											2023-01-24 11:38:29 -06:00
										 |  |  | 	// ppc64x/power9:"MADDLD",-"MULLD\t[$]23",-"ADD\t[$]29" | 
					
						
							|  |  |  | 	// ppc64x/power8:"MULLD\t[$]23","ADD\t[$]29" | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 	return 5*n + 7*(n+1) + 11*(n+2) // 23n + 29 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func MergeMuls3(a, n int) int { | 
					
						
							|  |  |  | 	// amd64:"ADDQ\t[$]19",-"IMULQ\t[$]19" | 
					
						
							|  |  |  | 	// 386:"ADDL\t[$]19",-"IMULL\t[$]19" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"ADD\t[$]19",-"MULLD\t[$]19" | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 	return a*n + 19*n // (a+19)n | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func MergeMuls4(n int) int { | 
					
						
							| 
									
										
										
										
											2018-03-09 23:09:46 +03:00
										 |  |  | 	// amd64:"IMUL3Q\t[$]14" | 
					
						
							|  |  |  | 	// 386:"IMUL3L\t[$]14" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"MULLD\t[$]14" | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 	return 23*n - 9*n // 14n | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func MergeMuls5(a, n int) int { | 
					
						
							|  |  |  | 	// amd64:"ADDQ\t[$]-19",-"IMULQ\t[$]19" | 
					
						
							|  |  |  | 	// 386:"ADDL\t[$]-19",-"IMULL\t[$]19" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"ADD\t[$]-19",-"MULLD\t[$]19" | 
					
						
							| 
									
										
										
										
											2018-03-09 14:51:30 +01:00
										 |  |  | 	return a*n - 19*n // (a-19)n | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // -------------- // | 
					
						
							|  |  |  | //    Division    // | 
					
						
							|  |  |  | // -------------- // | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-24 07:04:21 +00:00
										 |  |  | func DivMemSrc(a []float64) { | 
					
						
							| 
									
										
										
										
											2020-10-06 14:42:15 -07:00
										 |  |  | 	// 386/sse2:`DIVSD\s8\([A-Z]+\),\sX[0-9]+` | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// amd64:`DIVSD\s8\([A-Z]+\),\sX[0-9]+` | 
					
						
							| 
									
										
										
										
											2018-06-24 07:04:21 +00:00
										 |  |  | 	a[0] /= a[1] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | func Pow2Divs(n1 uint, n2 int) (uint, int) { | 
					
						
							|  |  |  | 	// 386:"SHRL\t[$]5",-"DIVL" | 
					
						
							|  |  |  | 	// amd64:"SHRQ\t[$]5",-"DIVQ" | 
					
						
							|  |  |  | 	// arm:"SRL\t[$]5",-".*udiv" | 
					
						
							|  |  |  | 	// arm64:"LSR\t[$]5",-"UDIV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SRD" | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	a := n1 / 32 // unsigned | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// amd64:"SARQ\t[$]6",-"IDIVQ" | 
					
						
							|  |  |  | 	// 386:"SARL\t[$]6",-"IDIVL" | 
					
						
							|  |  |  | 	// arm:"SRA\t[$]6",-".*udiv" | 
					
						
							|  |  |  | 	// arm64:"ASR\t[$]6",-"SDIV" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SRAD" | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	b := n2 / 64 // signed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Check that constant divisions get turned into MULs | 
					
						
							|  |  |  | func ConstDivs(n1 uint, n2 int) (uint, int) { | 
					
						
							|  |  |  | 	// amd64:"MOVQ\t[$]-1085102592571150095","MULQ",-"DIVQ" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// 386:"MOVL\t[$]-252645135","MULL",-"DIVL" | 
					
						
							|  |  |  | 	// arm64:`MOVD`,`UMULH`,-`DIV` | 
					
						
							|  |  |  | 	// arm:`MOVW`,`MUL`,-`.*udiv` | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	a := n1 / 17 // unsigned | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// amd64:"MOVQ\t[$]-1085102592571150095","IMULQ",-"IDIVQ" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// 386:"MOVL\t[$]-252645135","IMULL",-"IDIVL" | 
					
						
							| 
									
										
										
										
											2021-06-01 19:29:24 -04:00
										 |  |  | 	// arm64:`SMULH`,-`DIV` | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm:`MOVW`,`MUL`,-`.*udiv` | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	b := n2 / 17 // signed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 10:14:18 +00:00
										 |  |  | func FloatDivs(a []float32) float32 { | 
					
						
							|  |  |  | 	// amd64:`DIVSS\s8\([A-Z]+\),\sX[0-9]+` | 
					
						
							| 
									
										
										
										
											2020-10-06 14:42:15 -07:00
										 |  |  | 	// 386/sse2:`DIVSS\s8\([A-Z]+\),\sX[0-9]+` | 
					
						
							| 
									
										
										
										
											2018-06-21 10:14:18 +00:00
										 |  |  | 	return a[1] / a[2] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | func Pow2Mods(n1 uint, n2 int) (uint, int) { | 
					
						
							|  |  |  | 	// 386:"ANDL\t[$]31",-"DIVL" | 
					
						
							| 
									
										
										
										
											2021-10-10 17:56:16 +02:00
										 |  |  | 	// amd64:"ANDL\t[$]31",-"DIVQ" | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	// arm:"AND\t[$]31",-".*udiv" | 
					
						
							|  |  |  | 	// arm64:"AND\t[$]31",-"UDIV" | 
					
						
							| 
									
										
										
										
											2023-09-27 12:15:04 -05:00
										 |  |  | 	// ppc64x:"RLDICL" | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	a := n1 % 32 // unsigned | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-23 22:04:38 -06:00
										 |  |  | 	// 386:"SHRL",-"IDIVL" | 
					
						
							|  |  |  | 	// amd64:"SHRQ",-"IDIVQ" | 
					
						
							|  |  |  | 	// arm:"SRA",-".*udiv" | 
					
						
							|  |  |  | 	// arm64:"ASR",-"REM" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SRAD" | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	b := n2 % 64 // signed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-23 22:04:38 -06:00
										 |  |  | // Check that signed divisibility checks get converted to AND on low bits | 
					
						
							| 
									
										
										
										
											2019-09-08 21:50:07 -06:00
										 |  |  | func Pow2DivisibleSigned(n1, n2 int) (bool, bool) { | 
					
						
							| 
									
										
										
										
											2019-04-23 22:04:38 -06:00
										 |  |  | 	// 386:"TESTL\t[$]63",-"DIVL",-"SHRL" | 
					
						
							|  |  |  | 	// amd64:"TESTQ\t[$]63",-"DIVQ",-"SHRQ" | 
					
						
							|  |  |  | 	// arm:"AND\t[$]63",-".*udiv",-"SRA" | 
					
						
							| 
									
										
										
										
											2022-07-29 04:05:03 +00:00
										 |  |  | 	// arm64:"TST\t[$]63",-"UDIV",-"ASR",-"AND" | 
					
						
							| 
									
										
										
										
											2024-05-02 15:05:27 -05:00
										 |  |  | 	// ppc64x:"ANDCC",-"RLDICL",-"SRAD",-"CMP" | 
					
						
							| 
									
										
										
										
											2019-09-08 21:50:07 -06:00
										 |  |  | 	a := n1%64 == 0 // signed divisible | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 386:"TESTL\t[$]63",-"DIVL",-"SHRL" | 
					
						
							|  |  |  | 	// amd64:"TESTQ\t[$]63",-"DIVQ",-"SHRQ" | 
					
						
							|  |  |  | 	// arm:"AND\t[$]63",-".*udiv",-"SRA" | 
					
						
							| 
									
										
										
										
											2022-07-29 04:05:03 +00:00
										 |  |  | 	// arm64:"TST\t[$]63",-"UDIV",-"ASR",-"AND" | 
					
						
							| 
									
										
										
										
											2024-05-02 15:05:27 -05:00
										 |  |  | 	// ppc64x:"ANDCC",-"RLDICL",-"SRAD",-"CMP" | 
					
						
							| 
									
										
										
										
											2019-09-08 21:50:07 -06:00
										 |  |  | 	b := n2%64 != 0 // signed indivisible | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							| 
									
										
										
										
											2019-04-23 22:04:38 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | // Check that constant modulo divs get turned into MULs | 
					
						
							|  |  |  | func ConstMods(n1 uint, n2 int) (uint, int) { | 
					
						
							|  |  |  | 	// amd64:"MOVQ\t[$]-1085102592571150095","MULQ",-"DIVQ" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// 386:"MOVL\t[$]-252645135","MULL",-"DIVL" | 
					
						
							|  |  |  | 	// arm64:`MOVD`,`UMULH`,-`DIV` | 
					
						
							|  |  |  | 	// arm:`MOVW`,`MUL`,-`.*udiv` | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	a := n1 % 17 // unsigned | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// amd64:"MOVQ\t[$]-1085102592571150095","IMULQ",-"IDIVQ" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// 386:"MOVL\t[$]-252645135","IMULL",-"IDIVL" | 
					
						
							| 
									
										
										
										
											2021-06-01 19:29:24 -04:00
										 |  |  | 	// arm64:`SMULH`,-`DIV` | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm:`MOVW`,`MUL`,-`.*udiv` | 
					
						
							| 
									
										
										
										
											2018-03-14 11:47:34 +01:00
										 |  |  | 	b := n2 % 17 // signed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a, b | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-09 21:58:16 -07:00
										 |  |  | // Check that divisibility checks x%c==0 are converted to MULs and rotates | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | func DivisibleU(n uint) (bool, bool) { | 
					
						
							| 
									
										
										
										
											2019-03-09 21:58:16 -07:00
										 |  |  | 	// amd64:"MOVQ\t[$]-6148914691236517205","IMULQ","ROLQ\t[$]63",-"DIVQ" | 
					
						
							|  |  |  | 	// 386:"IMUL3L\t[$]-1431655765","ROLL\t[$]31",-"DIVQ" | 
					
						
							| 
									
										
											  
											
												cmd/compile: add late lower pass for last rules to run
Usually optimization rules have corresponding priorities, some need to
be run first, some run next, and some run last, which produces the best
code. But currently our optimization rules have no priority, this CL
adds a late lower pass that runs those rules that need to be run at last,
such as split unreasonable constant folding. This pass can be seen as
the second round of the lower pass.
For example:
func foo(a, b uint64) uint64 {
        d := a+0x1234568
        d1 := b+0x1234568
        return d&d1
}
The code generated by the master branch:
	0x0004 00004        ADD     $19088744, R0, R2 // movz+movk+add
	0x0010 00016        ADD     $19088744, R1, R1 // movz+movk+add
	0x001c 00028        AND     R1, R2, R0
This is because the current constant folding optimization rules do not
take into account the range of constants, causing the constant to be
loaded repeatedly. This CL splits these unreasonable constants folding
in the late lower pass. With this CL the generated code:
	0x0004 00004        MOVD    $19088744, R2 // movz+movk
	0x000c 00012        ADD     R0, R2, R3
	0x0010 00016        ADD     R1, R2, R1
	0x0014 00020        AND     R1, R3, R0
This CL also adds constant folding optimization for ADDS instruction.
In addition, in order not to introduce the codegen regression, an
optimization rule is added to change the addition of a negative number
into a subtraction of a positive number.
go1 benchmarks:
name                     old time/op    new time/op    delta
BinaryTree17-8              1.22s ± 1%     1.24s ± 0%  +1.56%  (p=0.008 n=5+5)
Fannkuch11-8                1.54s ± 0%     1.53s ± 0%  -0.69%  (p=0.016 n=4+5)
FmtFprintfEmpty-8          14.1ns ± 0%    14.1ns ± 0%    ~     (p=0.079 n=4+5)
FmtFprintfString-8         26.0ns ± 0%    26.1ns ± 0%  +0.23%  (p=0.008 n=5+5)
FmtFprintfInt-8            32.3ns ± 0%    32.9ns ± 1%  +1.72%  (p=0.008 n=5+5)
FmtFprintfIntInt-8         54.5ns ± 0%    55.5ns ± 0%  +1.83%  (p=0.008 n=5+5)
FmtFprintfPrefixedInt-8    61.5ns ± 0%    62.0ns ± 0%  +0.93%  (p=0.008 n=5+5)
FmtFprintfFloat-8          72.0ns ± 0%    73.6ns ± 0%  +2.24%  (p=0.008 n=5+5)
FmtManyArgs-8               221ns ± 0%     224ns ± 0%  +1.22%  (p=0.008 n=5+5)
GobDecode-8                1.91ms ± 0%    1.93ms ± 0%  +0.98%  (p=0.008 n=5+5)
GobEncode-8                1.40ms ± 1%    1.39ms ± 0%  -0.79%  (p=0.032 n=5+5)
Gzip-8                      115ms ± 0%     117ms ± 1%  +1.17%  (p=0.008 n=5+5)
Gunzip-8                   19.4ms ± 1%    19.3ms ± 0%  -0.71%  (p=0.016 n=5+4)
HTTPClientServer-8         27.0µs ± 0%    27.3µs ± 0%  +0.80%  (p=0.008 n=5+5)
JSONEncode-8               3.36ms ± 1%    3.33ms ± 0%    ~     (p=0.056 n=5+5)
JSONDecode-8               17.5ms ± 2%    17.8ms ± 0%  +1.71%  (p=0.016 n=5+4)
Mandelbrot200-8            2.29ms ± 0%    2.29ms ± 0%    ~     (p=0.151 n=5+5)
GoParse-8                  1.35ms ± 1%    1.36ms ± 1%    ~     (p=0.056 n=5+5)
RegexpMatchEasy0_32-8      24.5ns ± 0%    24.5ns ± 0%    ~     (p=0.444 n=4+5)
RegexpMatchEasy0_1K-8       131ns ±11%     118ns ± 6%    ~     (p=0.056 n=5+5)
RegexpMatchEasy1_32-8      22.9ns ± 0%    22.9ns ± 0%    ~     (p=0.905 n=4+5)
RegexpMatchEasy1_1K-8       126ns ± 0%     127ns ± 0%    ~     (p=0.063 n=4+5)
RegexpMatchMedium_32-8      486ns ± 5%     483ns ± 0%    ~     (p=0.381 n=5+4)
RegexpMatchMedium_1K-8     15.4µs ± 1%    15.5µs ± 0%    ~     (p=0.151 n=5+5)
RegexpMatchHard_32-8        687ns ± 0%     686ns ± 0%    ~     (p=0.103 n=5+5)
RegexpMatchHard_1K-8       20.7µs ± 0%    20.7µs ± 1%    ~     (p=0.151 n=5+5)
Revcomp-8                   175ms ± 2%     176ms ± 3%    ~     (p=1.000 n=5+5)
Template-8                 20.4ms ± 6%    20.1ms ± 2%    ~     (p=0.151 n=5+5)
TimeParse-8                 112ns ± 0%     113ns ± 0%  +0.97%  (p=0.016 n=5+4)
TimeFormat-8                156ns ± 0%     145ns ± 0%  -7.14%  (p=0.029 n=4+4)
Change-Id: I3ced26e89041f873ac989586514ccc5ee09f13da
Reviewed-on: https://go-review.googlesource.com/c/go/+/425134
Reviewed-by: Keith Randall <khr@google.com>
Reviewed-by: Cherry Mui <cherryyz@google.com>
TryBot-Result: Gopher Robot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
Run-TryBot: Eric Fang <eric.fang@arm.com>
											
										 
											2022-08-17 10:01:17 +00:00
										 |  |  | 	// arm64:"MOVD\t[$]-6148914691236517205","MOVD\t[$]3074457345618258602","MUL","ROR",-"DIV" | 
					
						
							| 
									
										
										
										
											2019-03-09 21:58:16 -07:00
										 |  |  | 	// arm:"MUL","CMP\t[$]715827882",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"MULLD","ROTL\t[$]63" | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	even := n%6 == 0 | 
					
						
							| 
									
										
										
										
											2019-03-09 21:58:16 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// amd64:"MOVQ\t[$]-8737931403336103397","IMULQ",-"ROLQ",-"DIVQ" | 
					
						
							|  |  |  | 	// 386:"IMUL3L\t[$]678152731",-"ROLL",-"DIVQ" | 
					
						
							|  |  |  | 	// arm64:"MOVD\t[$]-8737931403336103397","MUL",-"ROR",-"DIV" | 
					
						
							|  |  |  | 	// arm:"MUL","CMP\t[$]226050910",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"MULLD",-"ROTL" | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	odd := n%19 == 0 | 
					
						
							| 
									
										
										
										
											2019-04-05 14:05:07 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	return even, odd | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func Divisible(n int) (bool, bool) { | 
					
						
							| 
									
										
										
										
											2019-04-05 14:05:07 -06:00
										 |  |  | 	// amd64:"IMULQ","ADD","ROLQ\t[$]63",-"DIVQ" | 
					
						
							|  |  |  | 	// 386:"IMUL3L\t[$]-1431655765","ADDL\t[$]715827882","ROLL\t[$]31",-"DIVQ" | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	// arm64:"MOVD\t[$]-6148914691236517205","MOVD\t[$]3074457345618258602","MUL","ADD\tR","ROR",-"DIV" | 
					
						
							| 
									
										
										
										
											2019-04-05 14:05:07 -06:00
										 |  |  | 	// arm:"MUL","ADD\t[$]715827882",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x/power8:"MULLD","ADD","ROTL\t[$]63" | 
					
						
							|  |  |  | 	// ppc64x/power9:"MADDLD","ROTL\t[$]63" | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	even := n%6 == 0 | 
					
						
							| 
									
										
										
										
											2019-04-05 14:05:07 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// amd64:"IMULQ","ADD",-"ROLQ",-"DIVQ" | 
					
						
							|  |  |  | 	// 386:"IMUL3L\t[$]678152731","ADDL\t[$]113025455",-"ROLL",-"DIVQ" | 
					
						
							| 
									
										
											  
											
												cmd/compile: add late lower pass for last rules to run
Usually optimization rules have corresponding priorities, some need to
be run first, some run next, and some run last, which produces the best
code. But currently our optimization rules have no priority, this CL
adds a late lower pass that runs those rules that need to be run at last,
such as split unreasonable constant folding. This pass can be seen as
the second round of the lower pass.
For example:
func foo(a, b uint64) uint64 {
        d := a+0x1234568
        d1 := b+0x1234568
        return d&d1
}
The code generated by the master branch:
	0x0004 00004        ADD     $19088744, R0, R2 // movz+movk+add
	0x0010 00016        ADD     $19088744, R1, R1 // movz+movk+add
	0x001c 00028        AND     R1, R2, R0
This is because the current constant folding optimization rules do not
take into account the range of constants, causing the constant to be
loaded repeatedly. This CL splits these unreasonable constants folding
in the late lower pass. With this CL the generated code:
	0x0004 00004        MOVD    $19088744, R2 // movz+movk
	0x000c 00012        ADD     R0, R2, R3
	0x0010 00016        ADD     R1, R2, R1
	0x0014 00020        AND     R1, R3, R0
This CL also adds constant folding optimization for ADDS instruction.
In addition, in order not to introduce the codegen regression, an
optimization rule is added to change the addition of a negative number
into a subtraction of a positive number.
go1 benchmarks:
name                     old time/op    new time/op    delta
BinaryTree17-8              1.22s ± 1%     1.24s ± 0%  +1.56%  (p=0.008 n=5+5)
Fannkuch11-8                1.54s ± 0%     1.53s ± 0%  -0.69%  (p=0.016 n=4+5)
FmtFprintfEmpty-8          14.1ns ± 0%    14.1ns ± 0%    ~     (p=0.079 n=4+5)
FmtFprintfString-8         26.0ns ± 0%    26.1ns ± 0%  +0.23%  (p=0.008 n=5+5)
FmtFprintfInt-8            32.3ns ± 0%    32.9ns ± 1%  +1.72%  (p=0.008 n=5+5)
FmtFprintfIntInt-8         54.5ns ± 0%    55.5ns ± 0%  +1.83%  (p=0.008 n=5+5)
FmtFprintfPrefixedInt-8    61.5ns ± 0%    62.0ns ± 0%  +0.93%  (p=0.008 n=5+5)
FmtFprintfFloat-8          72.0ns ± 0%    73.6ns ± 0%  +2.24%  (p=0.008 n=5+5)
FmtManyArgs-8               221ns ± 0%     224ns ± 0%  +1.22%  (p=0.008 n=5+5)
GobDecode-8                1.91ms ± 0%    1.93ms ± 0%  +0.98%  (p=0.008 n=5+5)
GobEncode-8                1.40ms ± 1%    1.39ms ± 0%  -0.79%  (p=0.032 n=5+5)
Gzip-8                      115ms ± 0%     117ms ± 1%  +1.17%  (p=0.008 n=5+5)
Gunzip-8                   19.4ms ± 1%    19.3ms ± 0%  -0.71%  (p=0.016 n=5+4)
HTTPClientServer-8         27.0µs ± 0%    27.3µs ± 0%  +0.80%  (p=0.008 n=5+5)
JSONEncode-8               3.36ms ± 1%    3.33ms ± 0%    ~     (p=0.056 n=5+5)
JSONDecode-8               17.5ms ± 2%    17.8ms ± 0%  +1.71%  (p=0.016 n=5+4)
Mandelbrot200-8            2.29ms ± 0%    2.29ms ± 0%    ~     (p=0.151 n=5+5)
GoParse-8                  1.35ms ± 1%    1.36ms ± 1%    ~     (p=0.056 n=5+5)
RegexpMatchEasy0_32-8      24.5ns ± 0%    24.5ns ± 0%    ~     (p=0.444 n=4+5)
RegexpMatchEasy0_1K-8       131ns ±11%     118ns ± 6%    ~     (p=0.056 n=5+5)
RegexpMatchEasy1_32-8      22.9ns ± 0%    22.9ns ± 0%    ~     (p=0.905 n=4+5)
RegexpMatchEasy1_1K-8       126ns ± 0%     127ns ± 0%    ~     (p=0.063 n=4+5)
RegexpMatchMedium_32-8      486ns ± 5%     483ns ± 0%    ~     (p=0.381 n=5+4)
RegexpMatchMedium_1K-8     15.4µs ± 1%    15.5µs ± 0%    ~     (p=0.151 n=5+5)
RegexpMatchHard_32-8        687ns ± 0%     686ns ± 0%    ~     (p=0.103 n=5+5)
RegexpMatchHard_1K-8       20.7µs ± 0%    20.7µs ± 1%    ~     (p=0.151 n=5+5)
Revcomp-8                   175ms ± 2%     176ms ± 3%    ~     (p=1.000 n=5+5)
Template-8                 20.4ms ± 6%    20.1ms ± 2%    ~     (p=0.151 n=5+5)
TimeParse-8                 112ns ± 0%     113ns ± 0%  +0.97%  (p=0.016 n=5+4)
TimeFormat-8                156ns ± 0%     145ns ± 0%  -7.14%  (p=0.029 n=4+4)
Change-Id: I3ced26e89041f873ac989586514ccc5ee09f13da
Reviewed-on: https://go-review.googlesource.com/c/go/+/425134
Reviewed-by: Keith Randall <khr@google.com>
Reviewed-by: Cherry Mui <cherryyz@google.com>
TryBot-Result: Gopher Robot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
Run-TryBot: Eric Fang <eric.fang@arm.com>
											
										 
											2022-08-17 10:01:17 +00:00
										 |  |  | 	// arm64:"MUL","MOVD\t[$]485440633518672410","ADD",-"ROR",-"DIV" | 
					
						
							| 
									
										
										
										
											2019-04-05 14:05:07 -06:00
										 |  |  | 	// arm:"MUL","ADD\t[$]113025455",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x/power8:"MULLD","ADD",-"ROTL" | 
					
						
							|  |  |  | 	// ppc64x/power9:"MADDLD",-"ROTL" | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	odd := n%19 == 0 | 
					
						
							| 
									
										
										
										
											2019-04-05 14:05:07 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-21 22:22:36 -08:00
										 |  |  | 	return even, odd | 
					
						
							| 
									
										
										
										
											2019-03-09 21:58:16 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | // Check that fix-up code is not generated for divisions where it has been proven that | 
					
						
							|  |  |  | // that the divisor is not -1 or that the dividend is > MinIntNN. | 
					
						
							|  |  |  | func NoFix64A(divr int64) (int64, int64) { | 
					
						
							|  |  |  | 	var d int64 = 42 | 
					
						
							|  |  |  | 	var e int64 = 84 | 
					
						
							|  |  |  | 	if divr > 5 { | 
					
						
							|  |  |  | 		d /= divr // amd64:-"JMP" | 
					
						
							|  |  |  | 		e %= divr // amd64:-"JMP" | 
					
						
							| 
									
										
											  
											
												cmd/compile: use depth first topological sort algorithm for layout
The current layout algorithm tries to put consecutive blocks together,
so the priority of the successor block is higher than the priority of
the zero indegree block. This algorithm is beneficial for subsequent
register allocation, but will result in more branch instructions.
The depth-first topological sorting algorithm is a well-known layout
algorithm, which has applications in many languages, and it helps to
reduce branch instructions. This CL applies it to the layout pass.
The test results show that it helps to reduce the code size.
This CL also includes the following changes:
1, Removed the primary predecessor mechanism. The new layout algorithm is
  not very friendly to register allocator in some cases, in order to adapt
  to the new layout algorithm, a new primary predecessor selection strategy
  is introduced.
2, Since the new layout implementation may place non-loop blocks between
  loop blocks, some adaptive modifications have also been made to looprotate
  pass.
3, The layout also affects the results of codegen, so this CL also adjusted
  several codegen tests accordingly.
It is inevitable that this CL will cause the code size or performance of a
few functions to decrease, but the number of cases it improves is much larger
than the number of cases it drops.
Statistical data from compilecmp on linux/amd64 is as follow:
name                      old time/op       new time/op       delta
Template                        382ms ± 4%        382ms ± 4%    ~     (p=0.497 n=49+50)
Unicode                         170ms ± 9%        169ms ± 8%    ~     (p=0.344 n=48+50)
GoTypes                         2.01s ± 4%        2.01s ± 4%    ~     (p=0.628 n=50+48)
Compiler                        190ms ±10%        189ms ± 9%    ~     (p=0.734 n=50+50)
SSA                             11.8s ± 2%        11.8s ± 3%    ~     (p=0.877 n=50+50)
Flate                           241ms ± 9%        241ms ± 8%    ~     (p=0.897 n=50+49)
GoParser                        366ms ± 3%        361ms ± 4%  -1.21%  (p=0.004 n=47+50)
Reflect                         835ms ± 3%        838ms ± 3%    ~     (p=0.275 n=50+49)
Tar                             336ms ± 4%        335ms ± 3%    ~     (p=0.454 n=48+48)
XML                             433ms ± 4%        431ms ± 3%    ~     (p=0.071 n=49+48)
LinkCompiler                    706ms ± 4%        705ms ± 4%    ~     (p=0.608 n=50+49)
ExternalLinkCompiler            1.85s ± 3%        1.83s ± 2%  -1.47%  (p=0.000 n=49+48)
LinkWithoutDebugCompiler        437ms ± 5%        437ms ± 6%    ~     (p=0.953 n=49+50)
[Geo mean]                      615ms             613ms       -0.37%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 1%       38.7MB ± 1%    ~     (p=0.834 n=50+50)
Unicode                        28.1MB ± 0%       28.1MB ± 0%  -0.22%  (p=0.000 n=49+50)
GoTypes                         168MB ± 1%        168MB ± 1%    ~     (p=0.054 n=47+47)
Compiler                       23.0MB ± 1%       23.0MB ± 1%    ~     (p=0.432 n=50+50)
SSA                            1.54GB ± 0%       1.54GB ± 0%  +0.21%  (p=0.000 n=50+50)
Flate                          23.6MB ± 1%       23.6MB ± 1%    ~     (p=0.153 n=43+46)
GoParser                       35.1MB ± 1%       35.1MB ± 2%    ~     (p=0.202 n=50+50)
Reflect                        84.7MB ± 1%       84.7MB ± 1%    ~     (p=0.333 n=48+49)
Tar                            34.5MB ± 1%       34.5MB ± 1%    ~     (p=0.406 n=46+49)
XML                            44.3MB ± 2%       44.2MB ± 3%    ~     (p=0.981 n=50+50)
LinkCompiler                    131MB ± 0%        128MB ± 0%  -2.74%  (p=0.000 n=50+50)
ExternalLinkCompiler            120MB ± 0%        120MB ± 0%  +0.01%  (p=0.007 n=50+50)
LinkWithoutDebugCompiler       77.3MB ± 0%       77.3MB ± 0%  -0.02%  (p=0.000 n=50+50)
[Geo mean]                     69.3MB            69.1MB       -0.22%
file      before    after     Δ        %
addr2line 4104220   4043684   -60536   -1.475%
api       5342502   5249678   -92824   -1.737%
asm       4973785   4858257   -115528  -2.323%
buildid   2667844   2625660   -42184   -1.581%
cgo       4686849   4616313   -70536   -1.505%
compile   23667431  23268406  -399025  -1.686%
cover     4959676   4874108   -85568   -1.725%
dist      3515934   3450422   -65512   -1.863%
doc       3995581   3925469   -70112   -1.755%
fix       3379202   3318522   -60680   -1.796%
link      6743249   6629913   -113336  -1.681%
nm        4047529   3991777   -55752   -1.377%
objdump   4456151   4388151   -68000   -1.526%
pack      2435040   2398072   -36968   -1.518%
pprof     13804080  13565808  -238272  -1.726%
test2json 2690043   2645987   -44056   -1.638%
trace     10418492  10232716  -185776  -1.783%
vet       7258259   7121259   -137000  -1.888%
total     113145867 111204202 -1941665 -1.716%
The situation on linux/arm64 is as follow:
name                      old time/op       new time/op       delta
Template                        280ms ± 1%        282ms ± 1%  +0.75%  (p=0.000 n=46+48)
Unicode                         124ms ± 2%        124ms ± 2%  +0.37%  (p=0.045 n=50+50)
GoTypes                         1.69s ± 1%        1.70s ± 1%  +0.56%  (p=0.000 n=49+50)
Compiler                        122ms ± 1%        123ms ± 1%  +0.93%  (p=0.000 n=50+50)
SSA                             12.6s ± 1%        12.7s ± 0%  +0.72%  (p=0.000 n=50+50)
Flate                           170ms ± 1%        172ms ± 1%  +0.97%  (p=0.000 n=49+49)
GoParser                        262ms ± 1%        263ms ± 1%  +0.39%  (p=0.000 n=49+48)
Reflect                         639ms ± 1%        650ms ± 1%  +1.63%  (p=0.000 n=49+49)
Tar                             243ms ± 1%        245ms ± 1%  +0.82%  (p=0.000 n=50+50)
XML                             324ms ± 1%        327ms ± 1%  +0.72%  (p=0.000 n=50+49)
LinkCompiler                    597ms ± 1%        596ms ± 1%  -0.27%  (p=0.001 n=48+47)
ExternalLinkCompiler            1.90s ± 1%        1.88s ± 1%  -1.00%  (p=0.000 n=50+50)
LinkWithoutDebugCompiler        364ms ± 1%        363ms ± 1%    ~     (p=0.220 n=49+50)
[Geo mean]                      485ms             488ms       +0.49%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 0%       38.8MB ± 1%    ~     (p=0.093 n=43+49)
Unicode                        28.4MB ± 0%       28.4MB ± 0%  +0.03%  (p=0.000 n=49+45)
GoTypes                         169MB ± 1%        169MB ± 1%  +0.23%  (p=0.010 n=50+50)
Compiler                       23.2MB ± 1%       23.2MB ± 1%  +0.11%  (p=0.000 n=40+44)
SSA                            1.54GB ± 0%       1.55GB ± 0%  +0.45%  (p=0.000 n=47+49)
Flate                          23.8MB ± 2%       23.8MB ± 1%    ~     (p=0.543 n=50+50)
GoParser                       35.3MB ± 1%       35.4MB ± 1%    ~     (p=0.792 n=50+50)
Reflect                        85.2MB ± 1%       85.2MB ± 0%    ~     (p=0.055 n=50+47)
Tar                            34.5MB ± 1%       34.5MB ± 1%  +0.06%  (p=0.015 n=50+50)
XML                            43.8MB ± 2%       43.9MB ± 2%  +0.19%  (p=0.000 n=48+48)
LinkCompiler                    137MB ± 0%        136MB ± 0%  -0.92%  (p=0.000 n=50+50)
ExternalLinkCompiler            127MB ± 0%        127MB ± 0%    ~     (p=0.516 n=50+50)
LinkWithoutDebugCompiler       84.0MB ± 0%       84.0MB ± 0%    ~     (p=0.057 n=50+50)
[Geo mean]                     70.4MB            70.4MB       +0.01%
file      before    after     Δ        %
addr2line 4021557   4002933   -18624   -0.463%
api       5127847   5028503   -99344   -1.937%
asm       5034716   4936836   -97880   -1.944%
buildid   2608118   2594094   -14024   -0.538%
cgo       4488592   4398320   -90272   -2.011%
compile   22501129  22213592  -287537  -1.278%
cover     4742301   4713573   -28728   -0.606%
dist      3388071   3365311   -22760   -0.672%
doc       3802250   3776082   -26168   -0.688%
fix       3306147   3216939   -89208   -2.698%
link      6404483   6363699   -40784   -0.637%
nm        3941026   3921930   -19096   -0.485%
objdump   4383330   4295122   -88208   -2.012%
pack      2404547   2389515   -15032   -0.625%
pprof     12996234  12856818  -139416  -1.073%
test2json 2668500   2586788   -81712   -3.062%
trace     9816276   9609580   -206696  -2.106%
vet       6900682   6787338   -113344  -1.643%
total     108535806 107056973 -1478833 -1.363%
Change-Id: Iaec1cdcaacca8025e9babb0fb8a532fddb70c87d
Reviewed-on: https://go-review.googlesource.com/c/go/+/255239
Reviewed-by: eric fang <eric.fang@arm.com>
Reviewed-by: Keith Randall <khr@golang.org>
Trust: eric fang <eric.fang@arm.com>
											
										 
											2020-07-23 10:24:56 +08:00
										 |  |  | 		// The following statement is to avoid conflict between the above check | 
					
						
							|  |  |  | 		// and the normal JMP generated at the end of the block. | 
					
						
							|  |  |  | 		d += e | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return d, e | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NoFix64B(divd int64) (int64, int64) { | 
					
						
							|  |  |  | 	var d int64 | 
					
						
							|  |  |  | 	var e int64 | 
					
						
							|  |  |  | 	var divr int64 = -1 | 
					
						
							|  |  |  | 	if divd > -9223372036854775808 { | 
					
						
							|  |  |  | 		d = divd / divr // amd64:-"JMP" | 
					
						
							|  |  |  | 		e = divd % divr // amd64:-"JMP" | 
					
						
							| 
									
										
											  
											
												cmd/compile: use depth first topological sort algorithm for layout
The current layout algorithm tries to put consecutive blocks together,
so the priority of the successor block is higher than the priority of
the zero indegree block. This algorithm is beneficial for subsequent
register allocation, but will result in more branch instructions.
The depth-first topological sorting algorithm is a well-known layout
algorithm, which has applications in many languages, and it helps to
reduce branch instructions. This CL applies it to the layout pass.
The test results show that it helps to reduce the code size.
This CL also includes the following changes:
1, Removed the primary predecessor mechanism. The new layout algorithm is
  not very friendly to register allocator in some cases, in order to adapt
  to the new layout algorithm, a new primary predecessor selection strategy
  is introduced.
2, Since the new layout implementation may place non-loop blocks between
  loop blocks, some adaptive modifications have also been made to looprotate
  pass.
3, The layout also affects the results of codegen, so this CL also adjusted
  several codegen tests accordingly.
It is inevitable that this CL will cause the code size or performance of a
few functions to decrease, but the number of cases it improves is much larger
than the number of cases it drops.
Statistical data from compilecmp on linux/amd64 is as follow:
name                      old time/op       new time/op       delta
Template                        382ms ± 4%        382ms ± 4%    ~     (p=0.497 n=49+50)
Unicode                         170ms ± 9%        169ms ± 8%    ~     (p=0.344 n=48+50)
GoTypes                         2.01s ± 4%        2.01s ± 4%    ~     (p=0.628 n=50+48)
Compiler                        190ms ±10%        189ms ± 9%    ~     (p=0.734 n=50+50)
SSA                             11.8s ± 2%        11.8s ± 3%    ~     (p=0.877 n=50+50)
Flate                           241ms ± 9%        241ms ± 8%    ~     (p=0.897 n=50+49)
GoParser                        366ms ± 3%        361ms ± 4%  -1.21%  (p=0.004 n=47+50)
Reflect                         835ms ± 3%        838ms ± 3%    ~     (p=0.275 n=50+49)
Tar                             336ms ± 4%        335ms ± 3%    ~     (p=0.454 n=48+48)
XML                             433ms ± 4%        431ms ± 3%    ~     (p=0.071 n=49+48)
LinkCompiler                    706ms ± 4%        705ms ± 4%    ~     (p=0.608 n=50+49)
ExternalLinkCompiler            1.85s ± 3%        1.83s ± 2%  -1.47%  (p=0.000 n=49+48)
LinkWithoutDebugCompiler        437ms ± 5%        437ms ± 6%    ~     (p=0.953 n=49+50)
[Geo mean]                      615ms             613ms       -0.37%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 1%       38.7MB ± 1%    ~     (p=0.834 n=50+50)
Unicode                        28.1MB ± 0%       28.1MB ± 0%  -0.22%  (p=0.000 n=49+50)
GoTypes                         168MB ± 1%        168MB ± 1%    ~     (p=0.054 n=47+47)
Compiler                       23.0MB ± 1%       23.0MB ± 1%    ~     (p=0.432 n=50+50)
SSA                            1.54GB ± 0%       1.54GB ± 0%  +0.21%  (p=0.000 n=50+50)
Flate                          23.6MB ± 1%       23.6MB ± 1%    ~     (p=0.153 n=43+46)
GoParser                       35.1MB ± 1%       35.1MB ± 2%    ~     (p=0.202 n=50+50)
Reflect                        84.7MB ± 1%       84.7MB ± 1%    ~     (p=0.333 n=48+49)
Tar                            34.5MB ± 1%       34.5MB ± 1%    ~     (p=0.406 n=46+49)
XML                            44.3MB ± 2%       44.2MB ± 3%    ~     (p=0.981 n=50+50)
LinkCompiler                    131MB ± 0%        128MB ± 0%  -2.74%  (p=0.000 n=50+50)
ExternalLinkCompiler            120MB ± 0%        120MB ± 0%  +0.01%  (p=0.007 n=50+50)
LinkWithoutDebugCompiler       77.3MB ± 0%       77.3MB ± 0%  -0.02%  (p=0.000 n=50+50)
[Geo mean]                     69.3MB            69.1MB       -0.22%
file      before    after     Δ        %
addr2line 4104220   4043684   -60536   -1.475%
api       5342502   5249678   -92824   -1.737%
asm       4973785   4858257   -115528  -2.323%
buildid   2667844   2625660   -42184   -1.581%
cgo       4686849   4616313   -70536   -1.505%
compile   23667431  23268406  -399025  -1.686%
cover     4959676   4874108   -85568   -1.725%
dist      3515934   3450422   -65512   -1.863%
doc       3995581   3925469   -70112   -1.755%
fix       3379202   3318522   -60680   -1.796%
link      6743249   6629913   -113336  -1.681%
nm        4047529   3991777   -55752   -1.377%
objdump   4456151   4388151   -68000   -1.526%
pack      2435040   2398072   -36968   -1.518%
pprof     13804080  13565808  -238272  -1.726%
test2json 2690043   2645987   -44056   -1.638%
trace     10418492  10232716  -185776  -1.783%
vet       7258259   7121259   -137000  -1.888%
total     113145867 111204202 -1941665 -1.716%
The situation on linux/arm64 is as follow:
name                      old time/op       new time/op       delta
Template                        280ms ± 1%        282ms ± 1%  +0.75%  (p=0.000 n=46+48)
Unicode                         124ms ± 2%        124ms ± 2%  +0.37%  (p=0.045 n=50+50)
GoTypes                         1.69s ± 1%        1.70s ± 1%  +0.56%  (p=0.000 n=49+50)
Compiler                        122ms ± 1%        123ms ± 1%  +0.93%  (p=0.000 n=50+50)
SSA                             12.6s ± 1%        12.7s ± 0%  +0.72%  (p=0.000 n=50+50)
Flate                           170ms ± 1%        172ms ± 1%  +0.97%  (p=0.000 n=49+49)
GoParser                        262ms ± 1%        263ms ± 1%  +0.39%  (p=0.000 n=49+48)
Reflect                         639ms ± 1%        650ms ± 1%  +1.63%  (p=0.000 n=49+49)
Tar                             243ms ± 1%        245ms ± 1%  +0.82%  (p=0.000 n=50+50)
XML                             324ms ± 1%        327ms ± 1%  +0.72%  (p=0.000 n=50+49)
LinkCompiler                    597ms ± 1%        596ms ± 1%  -0.27%  (p=0.001 n=48+47)
ExternalLinkCompiler            1.90s ± 1%        1.88s ± 1%  -1.00%  (p=0.000 n=50+50)
LinkWithoutDebugCompiler        364ms ± 1%        363ms ± 1%    ~     (p=0.220 n=49+50)
[Geo mean]                      485ms             488ms       +0.49%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 0%       38.8MB ± 1%    ~     (p=0.093 n=43+49)
Unicode                        28.4MB ± 0%       28.4MB ± 0%  +0.03%  (p=0.000 n=49+45)
GoTypes                         169MB ± 1%        169MB ± 1%  +0.23%  (p=0.010 n=50+50)
Compiler                       23.2MB ± 1%       23.2MB ± 1%  +0.11%  (p=0.000 n=40+44)
SSA                            1.54GB ± 0%       1.55GB ± 0%  +0.45%  (p=0.000 n=47+49)
Flate                          23.8MB ± 2%       23.8MB ± 1%    ~     (p=0.543 n=50+50)
GoParser                       35.3MB ± 1%       35.4MB ± 1%    ~     (p=0.792 n=50+50)
Reflect                        85.2MB ± 1%       85.2MB ± 0%    ~     (p=0.055 n=50+47)
Tar                            34.5MB ± 1%       34.5MB ± 1%  +0.06%  (p=0.015 n=50+50)
XML                            43.8MB ± 2%       43.9MB ± 2%  +0.19%  (p=0.000 n=48+48)
LinkCompiler                    137MB ± 0%        136MB ± 0%  -0.92%  (p=0.000 n=50+50)
ExternalLinkCompiler            127MB ± 0%        127MB ± 0%    ~     (p=0.516 n=50+50)
LinkWithoutDebugCompiler       84.0MB ± 0%       84.0MB ± 0%    ~     (p=0.057 n=50+50)
[Geo mean]                     70.4MB            70.4MB       +0.01%
file      before    after     Δ        %
addr2line 4021557   4002933   -18624   -0.463%
api       5127847   5028503   -99344   -1.937%
asm       5034716   4936836   -97880   -1.944%
buildid   2608118   2594094   -14024   -0.538%
cgo       4488592   4398320   -90272   -2.011%
compile   22501129  22213592  -287537  -1.278%
cover     4742301   4713573   -28728   -0.606%
dist      3388071   3365311   -22760   -0.672%
doc       3802250   3776082   -26168   -0.688%
fix       3306147   3216939   -89208   -2.698%
link      6404483   6363699   -40784   -0.637%
nm        3941026   3921930   -19096   -0.485%
objdump   4383330   4295122   -88208   -2.012%
pack      2404547   2389515   -15032   -0.625%
pprof     12996234  12856818  -139416  -1.073%
test2json 2668500   2586788   -81712   -3.062%
trace     9816276   9609580   -206696  -2.106%
vet       6900682   6787338   -113344  -1.643%
total     108535806 107056973 -1478833 -1.363%
Change-Id: Iaec1cdcaacca8025e9babb0fb8a532fddb70c87d
Reviewed-on: https://go-review.googlesource.com/c/go/+/255239
Reviewed-by: eric fang <eric.fang@arm.com>
Reviewed-by: Keith Randall <khr@golang.org>
Trust: eric fang <eric.fang@arm.com>
											
										 
											2020-07-23 10:24:56 +08:00
										 |  |  | 		d += e | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return d, e | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NoFix32A(divr int32) (int32, int32) { | 
					
						
							|  |  |  | 	var d int32 = 42 | 
					
						
							|  |  |  | 	var e int32 = 84 | 
					
						
							|  |  |  | 	if divr > 5 { | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		d /= divr | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		e %= divr | 
					
						
							| 
									
										
											  
											
												cmd/compile: use depth first topological sort algorithm for layout
The current layout algorithm tries to put consecutive blocks together,
so the priority of the successor block is higher than the priority of
the zero indegree block. This algorithm is beneficial for subsequent
register allocation, but will result in more branch instructions.
The depth-first topological sorting algorithm is a well-known layout
algorithm, which has applications in many languages, and it helps to
reduce branch instructions. This CL applies it to the layout pass.
The test results show that it helps to reduce the code size.
This CL also includes the following changes:
1, Removed the primary predecessor mechanism. The new layout algorithm is
  not very friendly to register allocator in some cases, in order to adapt
  to the new layout algorithm, a new primary predecessor selection strategy
  is introduced.
2, Since the new layout implementation may place non-loop blocks between
  loop blocks, some adaptive modifications have also been made to looprotate
  pass.
3, The layout also affects the results of codegen, so this CL also adjusted
  several codegen tests accordingly.
It is inevitable that this CL will cause the code size or performance of a
few functions to decrease, but the number of cases it improves is much larger
than the number of cases it drops.
Statistical data from compilecmp on linux/amd64 is as follow:
name                      old time/op       new time/op       delta
Template                        382ms ± 4%        382ms ± 4%    ~     (p=0.497 n=49+50)
Unicode                         170ms ± 9%        169ms ± 8%    ~     (p=0.344 n=48+50)
GoTypes                         2.01s ± 4%        2.01s ± 4%    ~     (p=0.628 n=50+48)
Compiler                        190ms ±10%        189ms ± 9%    ~     (p=0.734 n=50+50)
SSA                             11.8s ± 2%        11.8s ± 3%    ~     (p=0.877 n=50+50)
Flate                           241ms ± 9%        241ms ± 8%    ~     (p=0.897 n=50+49)
GoParser                        366ms ± 3%        361ms ± 4%  -1.21%  (p=0.004 n=47+50)
Reflect                         835ms ± 3%        838ms ± 3%    ~     (p=0.275 n=50+49)
Tar                             336ms ± 4%        335ms ± 3%    ~     (p=0.454 n=48+48)
XML                             433ms ± 4%        431ms ± 3%    ~     (p=0.071 n=49+48)
LinkCompiler                    706ms ± 4%        705ms ± 4%    ~     (p=0.608 n=50+49)
ExternalLinkCompiler            1.85s ± 3%        1.83s ± 2%  -1.47%  (p=0.000 n=49+48)
LinkWithoutDebugCompiler        437ms ± 5%        437ms ± 6%    ~     (p=0.953 n=49+50)
[Geo mean]                      615ms             613ms       -0.37%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 1%       38.7MB ± 1%    ~     (p=0.834 n=50+50)
Unicode                        28.1MB ± 0%       28.1MB ± 0%  -0.22%  (p=0.000 n=49+50)
GoTypes                         168MB ± 1%        168MB ± 1%    ~     (p=0.054 n=47+47)
Compiler                       23.0MB ± 1%       23.0MB ± 1%    ~     (p=0.432 n=50+50)
SSA                            1.54GB ± 0%       1.54GB ± 0%  +0.21%  (p=0.000 n=50+50)
Flate                          23.6MB ± 1%       23.6MB ± 1%    ~     (p=0.153 n=43+46)
GoParser                       35.1MB ± 1%       35.1MB ± 2%    ~     (p=0.202 n=50+50)
Reflect                        84.7MB ± 1%       84.7MB ± 1%    ~     (p=0.333 n=48+49)
Tar                            34.5MB ± 1%       34.5MB ± 1%    ~     (p=0.406 n=46+49)
XML                            44.3MB ± 2%       44.2MB ± 3%    ~     (p=0.981 n=50+50)
LinkCompiler                    131MB ± 0%        128MB ± 0%  -2.74%  (p=0.000 n=50+50)
ExternalLinkCompiler            120MB ± 0%        120MB ± 0%  +0.01%  (p=0.007 n=50+50)
LinkWithoutDebugCompiler       77.3MB ± 0%       77.3MB ± 0%  -0.02%  (p=0.000 n=50+50)
[Geo mean]                     69.3MB            69.1MB       -0.22%
file      before    after     Δ        %
addr2line 4104220   4043684   -60536   -1.475%
api       5342502   5249678   -92824   -1.737%
asm       4973785   4858257   -115528  -2.323%
buildid   2667844   2625660   -42184   -1.581%
cgo       4686849   4616313   -70536   -1.505%
compile   23667431  23268406  -399025  -1.686%
cover     4959676   4874108   -85568   -1.725%
dist      3515934   3450422   -65512   -1.863%
doc       3995581   3925469   -70112   -1.755%
fix       3379202   3318522   -60680   -1.796%
link      6743249   6629913   -113336  -1.681%
nm        4047529   3991777   -55752   -1.377%
objdump   4456151   4388151   -68000   -1.526%
pack      2435040   2398072   -36968   -1.518%
pprof     13804080  13565808  -238272  -1.726%
test2json 2690043   2645987   -44056   -1.638%
trace     10418492  10232716  -185776  -1.783%
vet       7258259   7121259   -137000  -1.888%
total     113145867 111204202 -1941665 -1.716%
The situation on linux/arm64 is as follow:
name                      old time/op       new time/op       delta
Template                        280ms ± 1%        282ms ± 1%  +0.75%  (p=0.000 n=46+48)
Unicode                         124ms ± 2%        124ms ± 2%  +0.37%  (p=0.045 n=50+50)
GoTypes                         1.69s ± 1%        1.70s ± 1%  +0.56%  (p=0.000 n=49+50)
Compiler                        122ms ± 1%        123ms ± 1%  +0.93%  (p=0.000 n=50+50)
SSA                             12.6s ± 1%        12.7s ± 0%  +0.72%  (p=0.000 n=50+50)
Flate                           170ms ± 1%        172ms ± 1%  +0.97%  (p=0.000 n=49+49)
GoParser                        262ms ± 1%        263ms ± 1%  +0.39%  (p=0.000 n=49+48)
Reflect                         639ms ± 1%        650ms ± 1%  +1.63%  (p=0.000 n=49+49)
Tar                             243ms ± 1%        245ms ± 1%  +0.82%  (p=0.000 n=50+50)
XML                             324ms ± 1%        327ms ± 1%  +0.72%  (p=0.000 n=50+49)
LinkCompiler                    597ms ± 1%        596ms ± 1%  -0.27%  (p=0.001 n=48+47)
ExternalLinkCompiler            1.90s ± 1%        1.88s ± 1%  -1.00%  (p=0.000 n=50+50)
LinkWithoutDebugCompiler        364ms ± 1%        363ms ± 1%    ~     (p=0.220 n=49+50)
[Geo mean]                      485ms             488ms       +0.49%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 0%       38.8MB ± 1%    ~     (p=0.093 n=43+49)
Unicode                        28.4MB ± 0%       28.4MB ± 0%  +0.03%  (p=0.000 n=49+45)
GoTypes                         169MB ± 1%        169MB ± 1%  +0.23%  (p=0.010 n=50+50)
Compiler                       23.2MB ± 1%       23.2MB ± 1%  +0.11%  (p=0.000 n=40+44)
SSA                            1.54GB ± 0%       1.55GB ± 0%  +0.45%  (p=0.000 n=47+49)
Flate                          23.8MB ± 2%       23.8MB ± 1%    ~     (p=0.543 n=50+50)
GoParser                       35.3MB ± 1%       35.4MB ± 1%    ~     (p=0.792 n=50+50)
Reflect                        85.2MB ± 1%       85.2MB ± 0%    ~     (p=0.055 n=50+47)
Tar                            34.5MB ± 1%       34.5MB ± 1%  +0.06%  (p=0.015 n=50+50)
XML                            43.8MB ± 2%       43.9MB ± 2%  +0.19%  (p=0.000 n=48+48)
LinkCompiler                    137MB ± 0%        136MB ± 0%  -0.92%  (p=0.000 n=50+50)
ExternalLinkCompiler            127MB ± 0%        127MB ± 0%    ~     (p=0.516 n=50+50)
LinkWithoutDebugCompiler       84.0MB ± 0%       84.0MB ± 0%    ~     (p=0.057 n=50+50)
[Geo mean]                     70.4MB            70.4MB       +0.01%
file      before    after     Δ        %
addr2line 4021557   4002933   -18624   -0.463%
api       5127847   5028503   -99344   -1.937%
asm       5034716   4936836   -97880   -1.944%
buildid   2608118   2594094   -14024   -0.538%
cgo       4488592   4398320   -90272   -2.011%
compile   22501129  22213592  -287537  -1.278%
cover     4742301   4713573   -28728   -0.606%
dist      3388071   3365311   -22760   -0.672%
doc       3802250   3776082   -26168   -0.688%
fix       3306147   3216939   -89208   -2.698%
link      6404483   6363699   -40784   -0.637%
nm        3941026   3921930   -19096   -0.485%
objdump   4383330   4295122   -88208   -2.012%
pack      2404547   2389515   -15032   -0.625%
pprof     12996234  12856818  -139416  -1.073%
test2json 2668500   2586788   -81712   -3.062%
trace     9816276   9609580   -206696  -2.106%
vet       6900682   6787338   -113344  -1.643%
total     108535806 107056973 -1478833 -1.363%
Change-Id: Iaec1cdcaacca8025e9babb0fb8a532fddb70c87d
Reviewed-on: https://go-review.googlesource.com/c/go/+/255239
Reviewed-by: eric fang <eric.fang@arm.com>
Reviewed-by: Keith Randall <khr@golang.org>
Trust: eric fang <eric.fang@arm.com>
											
										 
											2020-07-23 10:24:56 +08:00
										 |  |  | 		d += e | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return d, e | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NoFix32B(divd int32) (int32, int32) { | 
					
						
							|  |  |  | 	var d int32 | 
					
						
							|  |  |  | 	var e int32 | 
					
						
							|  |  |  | 	var divr int32 = -1 | 
					
						
							|  |  |  | 	if divd > -2147483648 { | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		d = divd / divr | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		e = divd % divr | 
					
						
							| 
									
										
											  
											
												cmd/compile: use depth first topological sort algorithm for layout
The current layout algorithm tries to put consecutive blocks together,
so the priority of the successor block is higher than the priority of
the zero indegree block. This algorithm is beneficial for subsequent
register allocation, but will result in more branch instructions.
The depth-first topological sorting algorithm is a well-known layout
algorithm, which has applications in many languages, and it helps to
reduce branch instructions. This CL applies it to the layout pass.
The test results show that it helps to reduce the code size.
This CL also includes the following changes:
1, Removed the primary predecessor mechanism. The new layout algorithm is
  not very friendly to register allocator in some cases, in order to adapt
  to the new layout algorithm, a new primary predecessor selection strategy
  is introduced.
2, Since the new layout implementation may place non-loop blocks between
  loop blocks, some adaptive modifications have also been made to looprotate
  pass.
3, The layout also affects the results of codegen, so this CL also adjusted
  several codegen tests accordingly.
It is inevitable that this CL will cause the code size or performance of a
few functions to decrease, but the number of cases it improves is much larger
than the number of cases it drops.
Statistical data from compilecmp on linux/amd64 is as follow:
name                      old time/op       new time/op       delta
Template                        382ms ± 4%        382ms ± 4%    ~     (p=0.497 n=49+50)
Unicode                         170ms ± 9%        169ms ± 8%    ~     (p=0.344 n=48+50)
GoTypes                         2.01s ± 4%        2.01s ± 4%    ~     (p=0.628 n=50+48)
Compiler                        190ms ±10%        189ms ± 9%    ~     (p=0.734 n=50+50)
SSA                             11.8s ± 2%        11.8s ± 3%    ~     (p=0.877 n=50+50)
Flate                           241ms ± 9%        241ms ± 8%    ~     (p=0.897 n=50+49)
GoParser                        366ms ± 3%        361ms ± 4%  -1.21%  (p=0.004 n=47+50)
Reflect                         835ms ± 3%        838ms ± 3%    ~     (p=0.275 n=50+49)
Tar                             336ms ± 4%        335ms ± 3%    ~     (p=0.454 n=48+48)
XML                             433ms ± 4%        431ms ± 3%    ~     (p=0.071 n=49+48)
LinkCompiler                    706ms ± 4%        705ms ± 4%    ~     (p=0.608 n=50+49)
ExternalLinkCompiler            1.85s ± 3%        1.83s ± 2%  -1.47%  (p=0.000 n=49+48)
LinkWithoutDebugCompiler        437ms ± 5%        437ms ± 6%    ~     (p=0.953 n=49+50)
[Geo mean]                      615ms             613ms       -0.37%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 1%       38.7MB ± 1%    ~     (p=0.834 n=50+50)
Unicode                        28.1MB ± 0%       28.1MB ± 0%  -0.22%  (p=0.000 n=49+50)
GoTypes                         168MB ± 1%        168MB ± 1%    ~     (p=0.054 n=47+47)
Compiler                       23.0MB ± 1%       23.0MB ± 1%    ~     (p=0.432 n=50+50)
SSA                            1.54GB ± 0%       1.54GB ± 0%  +0.21%  (p=0.000 n=50+50)
Flate                          23.6MB ± 1%       23.6MB ± 1%    ~     (p=0.153 n=43+46)
GoParser                       35.1MB ± 1%       35.1MB ± 2%    ~     (p=0.202 n=50+50)
Reflect                        84.7MB ± 1%       84.7MB ± 1%    ~     (p=0.333 n=48+49)
Tar                            34.5MB ± 1%       34.5MB ± 1%    ~     (p=0.406 n=46+49)
XML                            44.3MB ± 2%       44.2MB ± 3%    ~     (p=0.981 n=50+50)
LinkCompiler                    131MB ± 0%        128MB ± 0%  -2.74%  (p=0.000 n=50+50)
ExternalLinkCompiler            120MB ± 0%        120MB ± 0%  +0.01%  (p=0.007 n=50+50)
LinkWithoutDebugCompiler       77.3MB ± 0%       77.3MB ± 0%  -0.02%  (p=0.000 n=50+50)
[Geo mean]                     69.3MB            69.1MB       -0.22%
file      before    after     Δ        %
addr2line 4104220   4043684   -60536   -1.475%
api       5342502   5249678   -92824   -1.737%
asm       4973785   4858257   -115528  -2.323%
buildid   2667844   2625660   -42184   -1.581%
cgo       4686849   4616313   -70536   -1.505%
compile   23667431  23268406  -399025  -1.686%
cover     4959676   4874108   -85568   -1.725%
dist      3515934   3450422   -65512   -1.863%
doc       3995581   3925469   -70112   -1.755%
fix       3379202   3318522   -60680   -1.796%
link      6743249   6629913   -113336  -1.681%
nm        4047529   3991777   -55752   -1.377%
objdump   4456151   4388151   -68000   -1.526%
pack      2435040   2398072   -36968   -1.518%
pprof     13804080  13565808  -238272  -1.726%
test2json 2690043   2645987   -44056   -1.638%
trace     10418492  10232716  -185776  -1.783%
vet       7258259   7121259   -137000  -1.888%
total     113145867 111204202 -1941665 -1.716%
The situation on linux/arm64 is as follow:
name                      old time/op       new time/op       delta
Template                        280ms ± 1%        282ms ± 1%  +0.75%  (p=0.000 n=46+48)
Unicode                         124ms ± 2%        124ms ± 2%  +0.37%  (p=0.045 n=50+50)
GoTypes                         1.69s ± 1%        1.70s ± 1%  +0.56%  (p=0.000 n=49+50)
Compiler                        122ms ± 1%        123ms ± 1%  +0.93%  (p=0.000 n=50+50)
SSA                             12.6s ± 1%        12.7s ± 0%  +0.72%  (p=0.000 n=50+50)
Flate                           170ms ± 1%        172ms ± 1%  +0.97%  (p=0.000 n=49+49)
GoParser                        262ms ± 1%        263ms ± 1%  +0.39%  (p=0.000 n=49+48)
Reflect                         639ms ± 1%        650ms ± 1%  +1.63%  (p=0.000 n=49+49)
Tar                             243ms ± 1%        245ms ± 1%  +0.82%  (p=0.000 n=50+50)
XML                             324ms ± 1%        327ms ± 1%  +0.72%  (p=0.000 n=50+49)
LinkCompiler                    597ms ± 1%        596ms ± 1%  -0.27%  (p=0.001 n=48+47)
ExternalLinkCompiler            1.90s ± 1%        1.88s ± 1%  -1.00%  (p=0.000 n=50+50)
LinkWithoutDebugCompiler        364ms ± 1%        363ms ± 1%    ~     (p=0.220 n=49+50)
[Geo mean]                      485ms             488ms       +0.49%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 0%       38.8MB ± 1%    ~     (p=0.093 n=43+49)
Unicode                        28.4MB ± 0%       28.4MB ± 0%  +0.03%  (p=0.000 n=49+45)
GoTypes                         169MB ± 1%        169MB ± 1%  +0.23%  (p=0.010 n=50+50)
Compiler                       23.2MB ± 1%       23.2MB ± 1%  +0.11%  (p=0.000 n=40+44)
SSA                            1.54GB ± 0%       1.55GB ± 0%  +0.45%  (p=0.000 n=47+49)
Flate                          23.8MB ± 2%       23.8MB ± 1%    ~     (p=0.543 n=50+50)
GoParser                       35.3MB ± 1%       35.4MB ± 1%    ~     (p=0.792 n=50+50)
Reflect                        85.2MB ± 1%       85.2MB ± 0%    ~     (p=0.055 n=50+47)
Tar                            34.5MB ± 1%       34.5MB ± 1%  +0.06%  (p=0.015 n=50+50)
XML                            43.8MB ± 2%       43.9MB ± 2%  +0.19%  (p=0.000 n=48+48)
LinkCompiler                    137MB ± 0%        136MB ± 0%  -0.92%  (p=0.000 n=50+50)
ExternalLinkCompiler            127MB ± 0%        127MB ± 0%    ~     (p=0.516 n=50+50)
LinkWithoutDebugCompiler       84.0MB ± 0%       84.0MB ± 0%    ~     (p=0.057 n=50+50)
[Geo mean]                     70.4MB            70.4MB       +0.01%
file      before    after     Δ        %
addr2line 4021557   4002933   -18624   -0.463%
api       5127847   5028503   -99344   -1.937%
asm       5034716   4936836   -97880   -1.944%
buildid   2608118   2594094   -14024   -0.538%
cgo       4488592   4398320   -90272   -2.011%
compile   22501129  22213592  -287537  -1.278%
cover     4742301   4713573   -28728   -0.606%
dist      3388071   3365311   -22760   -0.672%
doc       3802250   3776082   -26168   -0.688%
fix       3306147   3216939   -89208   -2.698%
link      6404483   6363699   -40784   -0.637%
nm        3941026   3921930   -19096   -0.485%
objdump   4383330   4295122   -88208   -2.012%
pack      2404547   2389515   -15032   -0.625%
pprof     12996234  12856818  -139416  -1.073%
test2json 2668500   2586788   -81712   -3.062%
trace     9816276   9609580   -206696  -2.106%
vet       6900682   6787338   -113344  -1.643%
total     108535806 107056973 -1478833 -1.363%
Change-Id: Iaec1cdcaacca8025e9babb0fb8a532fddb70c87d
Reviewed-on: https://go-review.googlesource.com/c/go/+/255239
Reviewed-by: eric fang <eric.fang@arm.com>
Reviewed-by: Keith Randall <khr@golang.org>
Trust: eric fang <eric.fang@arm.com>
											
										 
											2020-07-23 10:24:56 +08:00
										 |  |  | 		d += e | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return d, e | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NoFix16A(divr int16) (int16, int16) { | 
					
						
							|  |  |  | 	var d int16 = 42 | 
					
						
							|  |  |  | 	var e int16 = 84 | 
					
						
							|  |  |  | 	if divr > 5 { | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		d /= divr | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		e %= divr | 
					
						
							| 
									
										
											  
											
												cmd/compile: use depth first topological sort algorithm for layout
The current layout algorithm tries to put consecutive blocks together,
so the priority of the successor block is higher than the priority of
the zero indegree block. This algorithm is beneficial for subsequent
register allocation, but will result in more branch instructions.
The depth-first topological sorting algorithm is a well-known layout
algorithm, which has applications in many languages, and it helps to
reduce branch instructions. This CL applies it to the layout pass.
The test results show that it helps to reduce the code size.
This CL also includes the following changes:
1, Removed the primary predecessor mechanism. The new layout algorithm is
  not very friendly to register allocator in some cases, in order to adapt
  to the new layout algorithm, a new primary predecessor selection strategy
  is introduced.
2, Since the new layout implementation may place non-loop blocks between
  loop blocks, some adaptive modifications have also been made to looprotate
  pass.
3, The layout also affects the results of codegen, so this CL also adjusted
  several codegen tests accordingly.
It is inevitable that this CL will cause the code size or performance of a
few functions to decrease, but the number of cases it improves is much larger
than the number of cases it drops.
Statistical data from compilecmp on linux/amd64 is as follow:
name                      old time/op       new time/op       delta
Template                        382ms ± 4%        382ms ± 4%    ~     (p=0.497 n=49+50)
Unicode                         170ms ± 9%        169ms ± 8%    ~     (p=0.344 n=48+50)
GoTypes                         2.01s ± 4%        2.01s ± 4%    ~     (p=0.628 n=50+48)
Compiler                        190ms ±10%        189ms ± 9%    ~     (p=0.734 n=50+50)
SSA                             11.8s ± 2%        11.8s ± 3%    ~     (p=0.877 n=50+50)
Flate                           241ms ± 9%        241ms ± 8%    ~     (p=0.897 n=50+49)
GoParser                        366ms ± 3%        361ms ± 4%  -1.21%  (p=0.004 n=47+50)
Reflect                         835ms ± 3%        838ms ± 3%    ~     (p=0.275 n=50+49)
Tar                             336ms ± 4%        335ms ± 3%    ~     (p=0.454 n=48+48)
XML                             433ms ± 4%        431ms ± 3%    ~     (p=0.071 n=49+48)
LinkCompiler                    706ms ± 4%        705ms ± 4%    ~     (p=0.608 n=50+49)
ExternalLinkCompiler            1.85s ± 3%        1.83s ± 2%  -1.47%  (p=0.000 n=49+48)
LinkWithoutDebugCompiler        437ms ± 5%        437ms ± 6%    ~     (p=0.953 n=49+50)
[Geo mean]                      615ms             613ms       -0.37%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 1%       38.7MB ± 1%    ~     (p=0.834 n=50+50)
Unicode                        28.1MB ± 0%       28.1MB ± 0%  -0.22%  (p=0.000 n=49+50)
GoTypes                         168MB ± 1%        168MB ± 1%    ~     (p=0.054 n=47+47)
Compiler                       23.0MB ± 1%       23.0MB ± 1%    ~     (p=0.432 n=50+50)
SSA                            1.54GB ± 0%       1.54GB ± 0%  +0.21%  (p=0.000 n=50+50)
Flate                          23.6MB ± 1%       23.6MB ± 1%    ~     (p=0.153 n=43+46)
GoParser                       35.1MB ± 1%       35.1MB ± 2%    ~     (p=0.202 n=50+50)
Reflect                        84.7MB ± 1%       84.7MB ± 1%    ~     (p=0.333 n=48+49)
Tar                            34.5MB ± 1%       34.5MB ± 1%    ~     (p=0.406 n=46+49)
XML                            44.3MB ± 2%       44.2MB ± 3%    ~     (p=0.981 n=50+50)
LinkCompiler                    131MB ± 0%        128MB ± 0%  -2.74%  (p=0.000 n=50+50)
ExternalLinkCompiler            120MB ± 0%        120MB ± 0%  +0.01%  (p=0.007 n=50+50)
LinkWithoutDebugCompiler       77.3MB ± 0%       77.3MB ± 0%  -0.02%  (p=0.000 n=50+50)
[Geo mean]                     69.3MB            69.1MB       -0.22%
file      before    after     Δ        %
addr2line 4104220   4043684   -60536   -1.475%
api       5342502   5249678   -92824   -1.737%
asm       4973785   4858257   -115528  -2.323%
buildid   2667844   2625660   -42184   -1.581%
cgo       4686849   4616313   -70536   -1.505%
compile   23667431  23268406  -399025  -1.686%
cover     4959676   4874108   -85568   -1.725%
dist      3515934   3450422   -65512   -1.863%
doc       3995581   3925469   -70112   -1.755%
fix       3379202   3318522   -60680   -1.796%
link      6743249   6629913   -113336  -1.681%
nm        4047529   3991777   -55752   -1.377%
objdump   4456151   4388151   -68000   -1.526%
pack      2435040   2398072   -36968   -1.518%
pprof     13804080  13565808  -238272  -1.726%
test2json 2690043   2645987   -44056   -1.638%
trace     10418492  10232716  -185776  -1.783%
vet       7258259   7121259   -137000  -1.888%
total     113145867 111204202 -1941665 -1.716%
The situation on linux/arm64 is as follow:
name                      old time/op       new time/op       delta
Template                        280ms ± 1%        282ms ± 1%  +0.75%  (p=0.000 n=46+48)
Unicode                         124ms ± 2%        124ms ± 2%  +0.37%  (p=0.045 n=50+50)
GoTypes                         1.69s ± 1%        1.70s ± 1%  +0.56%  (p=0.000 n=49+50)
Compiler                        122ms ± 1%        123ms ± 1%  +0.93%  (p=0.000 n=50+50)
SSA                             12.6s ± 1%        12.7s ± 0%  +0.72%  (p=0.000 n=50+50)
Flate                           170ms ± 1%        172ms ± 1%  +0.97%  (p=0.000 n=49+49)
GoParser                        262ms ± 1%        263ms ± 1%  +0.39%  (p=0.000 n=49+48)
Reflect                         639ms ± 1%        650ms ± 1%  +1.63%  (p=0.000 n=49+49)
Tar                             243ms ± 1%        245ms ± 1%  +0.82%  (p=0.000 n=50+50)
XML                             324ms ± 1%        327ms ± 1%  +0.72%  (p=0.000 n=50+49)
LinkCompiler                    597ms ± 1%        596ms ± 1%  -0.27%  (p=0.001 n=48+47)
ExternalLinkCompiler            1.90s ± 1%        1.88s ± 1%  -1.00%  (p=0.000 n=50+50)
LinkWithoutDebugCompiler        364ms ± 1%        363ms ± 1%    ~     (p=0.220 n=49+50)
[Geo mean]                      485ms             488ms       +0.49%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 0%       38.8MB ± 1%    ~     (p=0.093 n=43+49)
Unicode                        28.4MB ± 0%       28.4MB ± 0%  +0.03%  (p=0.000 n=49+45)
GoTypes                         169MB ± 1%        169MB ± 1%  +0.23%  (p=0.010 n=50+50)
Compiler                       23.2MB ± 1%       23.2MB ± 1%  +0.11%  (p=0.000 n=40+44)
SSA                            1.54GB ± 0%       1.55GB ± 0%  +0.45%  (p=0.000 n=47+49)
Flate                          23.8MB ± 2%       23.8MB ± 1%    ~     (p=0.543 n=50+50)
GoParser                       35.3MB ± 1%       35.4MB ± 1%    ~     (p=0.792 n=50+50)
Reflect                        85.2MB ± 1%       85.2MB ± 0%    ~     (p=0.055 n=50+47)
Tar                            34.5MB ± 1%       34.5MB ± 1%  +0.06%  (p=0.015 n=50+50)
XML                            43.8MB ± 2%       43.9MB ± 2%  +0.19%  (p=0.000 n=48+48)
LinkCompiler                    137MB ± 0%        136MB ± 0%  -0.92%  (p=0.000 n=50+50)
ExternalLinkCompiler            127MB ± 0%        127MB ± 0%    ~     (p=0.516 n=50+50)
LinkWithoutDebugCompiler       84.0MB ± 0%       84.0MB ± 0%    ~     (p=0.057 n=50+50)
[Geo mean]                     70.4MB            70.4MB       +0.01%
file      before    after     Δ        %
addr2line 4021557   4002933   -18624   -0.463%
api       5127847   5028503   -99344   -1.937%
asm       5034716   4936836   -97880   -1.944%
buildid   2608118   2594094   -14024   -0.538%
cgo       4488592   4398320   -90272   -2.011%
compile   22501129  22213592  -287537  -1.278%
cover     4742301   4713573   -28728   -0.606%
dist      3388071   3365311   -22760   -0.672%
doc       3802250   3776082   -26168   -0.688%
fix       3306147   3216939   -89208   -2.698%
link      6404483   6363699   -40784   -0.637%
nm        3941026   3921930   -19096   -0.485%
objdump   4383330   4295122   -88208   -2.012%
pack      2404547   2389515   -15032   -0.625%
pprof     12996234  12856818  -139416  -1.073%
test2json 2668500   2586788   -81712   -3.062%
trace     9816276   9609580   -206696  -2.106%
vet       6900682   6787338   -113344  -1.643%
total     108535806 107056973 -1478833 -1.363%
Change-Id: Iaec1cdcaacca8025e9babb0fb8a532fddb70c87d
Reviewed-on: https://go-review.googlesource.com/c/go/+/255239
Reviewed-by: eric fang <eric.fang@arm.com>
Reviewed-by: Keith Randall <khr@golang.org>
Trust: eric fang <eric.fang@arm.com>
											
										 
											2020-07-23 10:24:56 +08:00
										 |  |  | 		d += e | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return d, e | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NoFix16B(divd int16) (int16, int16) { | 
					
						
							|  |  |  | 	var d int16 | 
					
						
							|  |  |  | 	var e int16 | 
					
						
							|  |  |  | 	var divr int16 = -1 | 
					
						
							|  |  |  | 	if divd > -32768 { | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		d = divd / divr | 
					
						
							|  |  |  | 		// amd64:-"JMP" | 
					
						
							|  |  |  | 		// 386:-"JMP" | 
					
						
							|  |  |  | 		e = divd % divr | 
					
						
							| 
									
										
											  
											
												cmd/compile: use depth first topological sort algorithm for layout
The current layout algorithm tries to put consecutive blocks together,
so the priority of the successor block is higher than the priority of
the zero indegree block. This algorithm is beneficial for subsequent
register allocation, but will result in more branch instructions.
The depth-first topological sorting algorithm is a well-known layout
algorithm, which has applications in many languages, and it helps to
reduce branch instructions. This CL applies it to the layout pass.
The test results show that it helps to reduce the code size.
This CL also includes the following changes:
1, Removed the primary predecessor mechanism. The new layout algorithm is
  not very friendly to register allocator in some cases, in order to adapt
  to the new layout algorithm, a new primary predecessor selection strategy
  is introduced.
2, Since the new layout implementation may place non-loop blocks between
  loop blocks, some adaptive modifications have also been made to looprotate
  pass.
3, The layout also affects the results of codegen, so this CL also adjusted
  several codegen tests accordingly.
It is inevitable that this CL will cause the code size or performance of a
few functions to decrease, but the number of cases it improves is much larger
than the number of cases it drops.
Statistical data from compilecmp on linux/amd64 is as follow:
name                      old time/op       new time/op       delta
Template                        382ms ± 4%        382ms ± 4%    ~     (p=0.497 n=49+50)
Unicode                         170ms ± 9%        169ms ± 8%    ~     (p=0.344 n=48+50)
GoTypes                         2.01s ± 4%        2.01s ± 4%    ~     (p=0.628 n=50+48)
Compiler                        190ms ±10%        189ms ± 9%    ~     (p=0.734 n=50+50)
SSA                             11.8s ± 2%        11.8s ± 3%    ~     (p=0.877 n=50+50)
Flate                           241ms ± 9%        241ms ± 8%    ~     (p=0.897 n=50+49)
GoParser                        366ms ± 3%        361ms ± 4%  -1.21%  (p=0.004 n=47+50)
Reflect                         835ms ± 3%        838ms ± 3%    ~     (p=0.275 n=50+49)
Tar                             336ms ± 4%        335ms ± 3%    ~     (p=0.454 n=48+48)
XML                             433ms ± 4%        431ms ± 3%    ~     (p=0.071 n=49+48)
LinkCompiler                    706ms ± 4%        705ms ± 4%    ~     (p=0.608 n=50+49)
ExternalLinkCompiler            1.85s ± 3%        1.83s ± 2%  -1.47%  (p=0.000 n=49+48)
LinkWithoutDebugCompiler        437ms ± 5%        437ms ± 6%    ~     (p=0.953 n=49+50)
[Geo mean]                      615ms             613ms       -0.37%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 1%       38.7MB ± 1%    ~     (p=0.834 n=50+50)
Unicode                        28.1MB ± 0%       28.1MB ± 0%  -0.22%  (p=0.000 n=49+50)
GoTypes                         168MB ± 1%        168MB ± 1%    ~     (p=0.054 n=47+47)
Compiler                       23.0MB ± 1%       23.0MB ± 1%    ~     (p=0.432 n=50+50)
SSA                            1.54GB ± 0%       1.54GB ± 0%  +0.21%  (p=0.000 n=50+50)
Flate                          23.6MB ± 1%       23.6MB ± 1%    ~     (p=0.153 n=43+46)
GoParser                       35.1MB ± 1%       35.1MB ± 2%    ~     (p=0.202 n=50+50)
Reflect                        84.7MB ± 1%       84.7MB ± 1%    ~     (p=0.333 n=48+49)
Tar                            34.5MB ± 1%       34.5MB ± 1%    ~     (p=0.406 n=46+49)
XML                            44.3MB ± 2%       44.2MB ± 3%    ~     (p=0.981 n=50+50)
LinkCompiler                    131MB ± 0%        128MB ± 0%  -2.74%  (p=0.000 n=50+50)
ExternalLinkCompiler            120MB ± 0%        120MB ± 0%  +0.01%  (p=0.007 n=50+50)
LinkWithoutDebugCompiler       77.3MB ± 0%       77.3MB ± 0%  -0.02%  (p=0.000 n=50+50)
[Geo mean]                     69.3MB            69.1MB       -0.22%
file      before    after     Δ        %
addr2line 4104220   4043684   -60536   -1.475%
api       5342502   5249678   -92824   -1.737%
asm       4973785   4858257   -115528  -2.323%
buildid   2667844   2625660   -42184   -1.581%
cgo       4686849   4616313   -70536   -1.505%
compile   23667431  23268406  -399025  -1.686%
cover     4959676   4874108   -85568   -1.725%
dist      3515934   3450422   -65512   -1.863%
doc       3995581   3925469   -70112   -1.755%
fix       3379202   3318522   -60680   -1.796%
link      6743249   6629913   -113336  -1.681%
nm        4047529   3991777   -55752   -1.377%
objdump   4456151   4388151   -68000   -1.526%
pack      2435040   2398072   -36968   -1.518%
pprof     13804080  13565808  -238272  -1.726%
test2json 2690043   2645987   -44056   -1.638%
trace     10418492  10232716  -185776  -1.783%
vet       7258259   7121259   -137000  -1.888%
total     113145867 111204202 -1941665 -1.716%
The situation on linux/arm64 is as follow:
name                      old time/op       new time/op       delta
Template                        280ms ± 1%        282ms ± 1%  +0.75%  (p=0.000 n=46+48)
Unicode                         124ms ± 2%        124ms ± 2%  +0.37%  (p=0.045 n=50+50)
GoTypes                         1.69s ± 1%        1.70s ± 1%  +0.56%  (p=0.000 n=49+50)
Compiler                        122ms ± 1%        123ms ± 1%  +0.93%  (p=0.000 n=50+50)
SSA                             12.6s ± 1%        12.7s ± 0%  +0.72%  (p=0.000 n=50+50)
Flate                           170ms ± 1%        172ms ± 1%  +0.97%  (p=0.000 n=49+49)
GoParser                        262ms ± 1%        263ms ± 1%  +0.39%  (p=0.000 n=49+48)
Reflect                         639ms ± 1%        650ms ± 1%  +1.63%  (p=0.000 n=49+49)
Tar                             243ms ± 1%        245ms ± 1%  +0.82%  (p=0.000 n=50+50)
XML                             324ms ± 1%        327ms ± 1%  +0.72%  (p=0.000 n=50+49)
LinkCompiler                    597ms ± 1%        596ms ± 1%  -0.27%  (p=0.001 n=48+47)
ExternalLinkCompiler            1.90s ± 1%        1.88s ± 1%  -1.00%  (p=0.000 n=50+50)
LinkWithoutDebugCompiler        364ms ± 1%        363ms ± 1%    ~     (p=0.220 n=49+50)
[Geo mean]                      485ms             488ms       +0.49%
name                      old alloc/op      new alloc/op      delta
Template                       38.7MB ± 0%       38.8MB ± 1%    ~     (p=0.093 n=43+49)
Unicode                        28.4MB ± 0%       28.4MB ± 0%  +0.03%  (p=0.000 n=49+45)
GoTypes                         169MB ± 1%        169MB ± 1%  +0.23%  (p=0.010 n=50+50)
Compiler                       23.2MB ± 1%       23.2MB ± 1%  +0.11%  (p=0.000 n=40+44)
SSA                            1.54GB ± 0%       1.55GB ± 0%  +0.45%  (p=0.000 n=47+49)
Flate                          23.8MB ± 2%       23.8MB ± 1%    ~     (p=0.543 n=50+50)
GoParser                       35.3MB ± 1%       35.4MB ± 1%    ~     (p=0.792 n=50+50)
Reflect                        85.2MB ± 1%       85.2MB ± 0%    ~     (p=0.055 n=50+47)
Tar                            34.5MB ± 1%       34.5MB ± 1%  +0.06%  (p=0.015 n=50+50)
XML                            43.8MB ± 2%       43.9MB ± 2%  +0.19%  (p=0.000 n=48+48)
LinkCompiler                    137MB ± 0%        136MB ± 0%  -0.92%  (p=0.000 n=50+50)
ExternalLinkCompiler            127MB ± 0%        127MB ± 0%    ~     (p=0.516 n=50+50)
LinkWithoutDebugCompiler       84.0MB ± 0%       84.0MB ± 0%    ~     (p=0.057 n=50+50)
[Geo mean]                     70.4MB            70.4MB       +0.01%
file      before    after     Δ        %
addr2line 4021557   4002933   -18624   -0.463%
api       5127847   5028503   -99344   -1.937%
asm       5034716   4936836   -97880   -1.944%
buildid   2608118   2594094   -14024   -0.538%
cgo       4488592   4398320   -90272   -2.011%
compile   22501129  22213592  -287537  -1.278%
cover     4742301   4713573   -28728   -0.606%
dist      3388071   3365311   -22760   -0.672%
doc       3802250   3776082   -26168   -0.688%
fix       3306147   3216939   -89208   -2.698%
link      6404483   6363699   -40784   -0.637%
nm        3941026   3921930   -19096   -0.485%
objdump   4383330   4295122   -88208   -2.012%
pack      2404547   2389515   -15032   -0.625%
pprof     12996234  12856818  -139416  -1.073%
test2json 2668500   2586788   -81712   -3.062%
trace     9816276   9609580   -206696  -2.106%
vet       6900682   6787338   -113344  -1.643%
total     108535806 107056973 -1478833 -1.363%
Change-Id: Iaec1cdcaacca8025e9babb0fb8a532fddb70c87d
Reviewed-on: https://go-review.googlesource.com/c/go/+/255239
Reviewed-by: eric fang <eric.fang@arm.com>
Reviewed-by: Keith Randall <khr@golang.org>
Trust: eric fang <eric.fang@arm.com>
											
										 
											2020-07-23 10:24:56 +08:00
										 |  |  | 		d += e | 
					
						
							| 
									
										
										
										
											2018-08-06 19:50:38 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return d, e | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | // Check that len() and cap() calls divided by powers of two are | 
					
						
							|  |  |  | // optimized into shifts and ands | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func LenDiv1(a []int) int { | 
					
						
							|  |  |  | 	// 386:"SHRL\t[$]10" | 
					
						
							|  |  |  | 	// amd64:"SHRQ\t[$]10" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm64:"LSR\t[$]10",-"SDIV" | 
					
						
							|  |  |  | 	// arm:"SRL\t[$]10",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SRD"\t[$]10" | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 	return len(a) / 1024 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func LenDiv2(s string) int { | 
					
						
							|  |  |  | 	// 386:"SHRL\t[$]11" | 
					
						
							|  |  |  | 	// amd64:"SHRQ\t[$]11" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm64:"LSR\t[$]11",-"SDIV" | 
					
						
							|  |  |  | 	// arm:"SRL\t[$]11",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SRD\t[$]11" | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 	return len(s) / (4097 >> 1) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func LenMod1(a []int) int { | 
					
						
							|  |  |  | 	// 386:"ANDL\t[$]1023" | 
					
						
							| 
									
										
										
										
											2021-10-10 17:56:16 +02:00
										 |  |  | 	// amd64:"ANDL\t[$]1023" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm64:"AND\t[$]1023",-"SDIV" | 
					
						
							|  |  |  | 	// arm/6:"AND",-".*udiv" | 
					
						
							|  |  |  | 	// arm/7:"BFC",-".*udiv",-"AND" | 
					
						
							| 
									
										
										
										
											2023-09-27 12:15:04 -05:00
										 |  |  | 	// ppc64x:"RLDICL" | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 	return len(a) % 1024 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func LenMod2(s string) int { | 
					
						
							|  |  |  | 	// 386:"ANDL\t[$]2047" | 
					
						
							| 
									
										
										
										
											2021-10-10 17:56:16 +02:00
										 |  |  | 	// amd64:"ANDL\t[$]2047" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm64:"AND\t[$]2047",-"SDIV" | 
					
						
							|  |  |  | 	// arm/6:"AND",-".*udiv" | 
					
						
							|  |  |  | 	// arm/7:"BFC",-".*udiv",-"AND" | 
					
						
							| 
									
										
										
										
											2023-09-27 12:15:04 -05:00
										 |  |  | 	// ppc64x:"RLDICL" | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 	return len(s) % (4097 >> 1) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func CapDiv(a []int) int { | 
					
						
							|  |  |  | 	// 386:"SHRL\t[$]12" | 
					
						
							|  |  |  | 	// amd64:"SHRQ\t[$]12" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm64:"LSR\t[$]12",-"SDIV" | 
					
						
							|  |  |  | 	// arm:"SRL\t[$]12",-".*udiv" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SRD\t[$]12" | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 	return cap(a) / ((1 << 11) + 2048) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func CapMod(a []int) int { | 
					
						
							|  |  |  | 	// 386:"ANDL\t[$]4095" | 
					
						
							| 
									
										
										
										
											2021-10-10 17:56:16 +02:00
										 |  |  | 	// amd64:"ANDL\t[$]4095" | 
					
						
							| 
									
										
										
										
											2018-11-30 09:30:36 +00:00
										 |  |  | 	// arm64:"AND\t[$]4095",-"SDIV" | 
					
						
							|  |  |  | 	// arm/6:"AND",-".*udiv" | 
					
						
							|  |  |  | 	// arm/7:"BFC",-".*udiv",-"AND" | 
					
						
							| 
									
										
										
										
											2023-09-27 12:15:04 -05:00
										 |  |  | 	// ppc64x:"RLDICL" | 
					
						
							| 
									
										
										
										
											2018-03-15 10:06:37 +01:00
										 |  |  | 	return cap(a) % ((1 << 11) + 2048) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-04-23 13:49:51 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | func AddMul(x int) int { | 
					
						
							|  |  |  | 	// amd64:"LEAQ\t1" | 
					
						
							|  |  |  | 	return 2*x + 1 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-13 10:38:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 13:21:03 +00:00
										 |  |  | func MULA(a, b, c uint32) (uint32, uint32, uint32) { | 
					
						
							|  |  |  | 	// arm:`MULA`,-`MUL\s` | 
					
						
							|  |  |  | 	// arm64:`MADDW`,-`MULW` | 
					
						
							|  |  |  | 	r0 := a*b + c | 
					
						
							| 
									
										
										
										
											2018-10-29 09:01:14 +00:00
										 |  |  | 	// arm:`MULA`,-`MUL\s` | 
					
						
							| 
									
										
										
										
											2018-09-27 13:21:03 +00:00
										 |  |  | 	// arm64:`MADDW`,-`MULW` | 
					
						
							|  |  |  | 	r1 := c*79 + a | 
					
						
							| 
									
										
										
										
											2018-10-29 09:01:14 +00:00
										 |  |  | 	// arm:`ADD`,-`MULA`,-`MUL\s` | 
					
						
							| 
									
										
										
										
											2018-09-27 13:21:03 +00:00
										 |  |  | 	// arm64:`ADD`,-`MADD`,-`MULW` | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:`ADD`,-`MULLD` | 
					
						
							| 
									
										
										
										
											2018-09-27 13:21:03 +00:00
										 |  |  | 	r2 := b*64 + c | 
					
						
							|  |  |  | 	return r0, r1, r2 | 
					
						
							| 
									
										
										
										
											2018-08-13 10:38:25 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-15 02:25:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | func MULS(a, b, c uint32) (uint32, uint32, uint32) { | 
					
						
							|  |  |  | 	// arm/7:`MULS`,-`MUL\s` | 
					
						
							| 
									
										
										
										
											2018-10-29 09:01:14 +00:00
										 |  |  | 	// arm/6:`SUB`,`MUL\s`,-`MULS` | 
					
						
							| 
									
										
										
										
											2018-10-15 02:25:10 +00:00
										 |  |  | 	// arm64:`MSUBW`,-`MULW` | 
					
						
							|  |  |  | 	r0 := c - a*b | 
					
						
							| 
									
										
										
										
											2018-10-29 09:01:14 +00:00
										 |  |  | 	// arm/7:`MULS`,-`MUL\s` | 
					
						
							|  |  |  | 	// arm/6:`SUB`,`MUL\s`,-`MULS` | 
					
						
							| 
									
										
										
										
											2018-10-15 02:25:10 +00:00
										 |  |  | 	// arm64:`MSUBW`,-`MULW` | 
					
						
							|  |  |  | 	r1 := a - c*79 | 
					
						
							| 
									
										
										
										
											2018-10-29 09:01:14 +00:00
										 |  |  | 	// arm/7:`SUB`,-`MULS`,-`MUL\s` | 
					
						
							| 
									
										
										
										
											2018-10-15 02:25:10 +00:00
										 |  |  | 	// arm64:`SUB`,-`MSUBW`,-`MULW` | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:`SUB`,-`MULLD` | 
					
						
							| 
									
										
										
										
											2018-10-15 02:25:10 +00:00
										 |  |  | 	r2 := c - b*64 | 
					
						
							|  |  |  | 	return r0, r1, r2 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-03-19 12:26:22 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | func addSpecial(a, b, c uint32) (uint32, uint32, uint32) { | 
					
						
							|  |  |  | 	// amd64:`INCL` | 
					
						
							|  |  |  | 	a++ | 
					
						
							|  |  |  | 	// amd64:`DECL` | 
					
						
							|  |  |  | 	b-- | 
					
						
							|  |  |  | 	// amd64:`SUBL.*-128` | 
					
						
							|  |  |  | 	c += 128 | 
					
						
							|  |  |  | 	return a, b, c | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-07 13:44:51 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Divide -> shift rules usually require fixup for negative inputs. | 
					
						
							|  |  |  | // If the input is non-negative, make sure the fixup is eliminated. | 
					
						
							|  |  |  | func divInt(v int64) int64 { | 
					
						
							|  |  |  | 	if v < 0 { | 
					
						
							|  |  |  | 		return 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// amd64:-`.*SARQ.*63,`, -".*SHRQ", ".*SARQ.*[$]9," | 
					
						
							|  |  |  | 	return v / 512 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // The reassociate rules "x - (z + C) -> (x - z) - C" and | 
					
						
							|  |  |  | // "(z + C) -x -> C + (z - x)" can optimize the following cases. | 
					
						
							|  |  |  | func constantFold1(i0, j0, i1, j1, i2, j2, i3, j3 int) (int, int, int, int) { | 
					
						
							|  |  |  | 	// arm64:"SUB","ADD\t[$]2" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SUB","ADD\t[$]2" | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r0 := (i0 + 3) - (j0 + 1) | 
					
						
							|  |  |  | 	// arm64:"SUB","SUB\t[$]4" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SUB","ADD\t[$]-4" | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r1 := (i1 - 3) - (j1 + 1) | 
					
						
							|  |  |  | 	// arm64:"SUB","ADD\t[$]4" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SUB","ADD\t[$]4" | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r2 := (i2 + 3) - (j2 - 1) | 
					
						
							|  |  |  | 	// arm64:"SUB","SUB\t[$]2" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"SUB","ADD\t[$]-2" | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r3 := (i3 - 3) - (j3 - 1) | 
					
						
							|  |  |  | 	return r0, r1, r2, r3 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The reassociate rules "x - (z + C) -> (x - z) - C" and | 
					
						
							|  |  |  | // "(C - z) - x -> C - (z + x)" can optimize the following cases. | 
					
						
							|  |  |  | func constantFold2(i0, j0, i1, j1 int) (int, int) { | 
					
						
							|  |  |  | 	// arm64:"ADD","MOVD\t[$]2","SUB" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `SUBC\tR[0-9]+,\s[$]2,\sR` | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r0 := (3 - i0) - (j0 + 1) | 
					
						
							|  |  |  | 	// arm64:"ADD","MOVD\t[$]4","SUB" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x: `SUBC\tR[0-9]+,\s[$]4,\sR` | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r1 := (3 - i1) - (j1 - 1) | 
					
						
							|  |  |  | 	return r0, r1 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func constantFold3(i, j int) int { | 
					
						
							| 
									
										
										
										
											2024-11-11 12:21:14 -08:00
										 |  |  | 	// arm64: "LSL\t[$]5,","SUB\tR[0-9]+<<1,",-"ADD" | 
					
						
							| 
									
										
										
										
											2023-01-25 11:53:10 -06:00
										 |  |  | 	// ppc64x:"MULLD\t[$]30","MULLD" | 
					
						
							| 
									
										
										
											
												cmd/compile: add more generic rewrite rules to reassociate (op (op y C) x|C)
With this patch, opt pass can expose more obvious constant-folding
opportunites.
Example:
func test(i int) int {return (i+8)-(i+4)}
The previous version:
  MOVD	"".i(FP), R0
  ADD	$8, R0, R1
  ADD	$4, R0, R0
  SUB	R0, R1, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
The optimized version:
  MOVD	$4, R0
  MOVD	R0, "".~r1+8(FP)
  RET	(R30)
This patch removes some existing reassociation rules, such as "x+(z-C)",
because the current generic rewrite rules will canonicalize "x-const"
to "x+(-const)", making "x+(z-C)" equal to "x+(z+(-C))".
This patch also adds test cases.
Change-Id: I857108ba0b5fcc18a879eeab38e2551bc4277797
Reviewed-on: https://go-review.googlesource.com/c/go/+/237137
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
											
										 
											2020-05-28 18:11:52 +08:00
										 |  |  | 	r := (5 * i) * (6 * j) | 
					
						
							|  |  |  | 	return r | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-08-20 23:41:40 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | // ----------------- // | 
					
						
							|  |  |  | //  Integer Min/Max  // | 
					
						
							|  |  |  | // ----------------- // | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func Int64Min(a, b int64) int64 { | 
					
						
							|  |  |  | 	// amd64: "CMPQ","CMOVQLT" | 
					
						
							|  |  |  | 	// arm64: "CMP","CSEL" | 
					
						
							|  |  |  | 	// riscv64/rva20u64:"BLT\t" | 
					
						
							| 
									
										
										
										
											2025-02-23 22:31:35 +11:00
										 |  |  | 	// riscv64/rva22u64,riscv64/rva23u64:"MIN\t" | 
					
						
							| 
									
										
										
										
											2024-08-20 23:41:40 +10:00
										 |  |  | 	return min(a, b) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func Int64Max(a, b int64) int64 { | 
					
						
							|  |  |  | 	// amd64: "CMPQ","CMOVQGT" | 
					
						
							|  |  |  | 	// arm64: "CMP","CSEL" | 
					
						
							|  |  |  | 	// riscv64/rva20u64:"BLT\t" | 
					
						
							| 
									
										
										
										
											2025-02-23 22:31:35 +11:00
										 |  |  | 	// riscv64/rva22u64,riscv64/rva23u64:"MAX\t" | 
					
						
							| 
									
										
										
										
											2024-08-20 23:41:40 +10:00
										 |  |  | 	return max(a, b) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func Uint64Min(a, b uint64) uint64 { | 
					
						
							|  |  |  | 	// amd64: "CMPQ","CMOVQCS" | 
					
						
							|  |  |  | 	// arm64: "CMP","CSEL" | 
					
						
							|  |  |  | 	// riscv64/rva20u64:"BLTU" | 
					
						
							| 
									
										
										
										
											2025-02-23 22:31:35 +11:00
										 |  |  | 	// riscv64/rva22u64,riscv64/rva23u64:"MINU" | 
					
						
							| 
									
										
										
										
											2024-08-20 23:41:40 +10:00
										 |  |  | 	return min(a, b) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func Uint64Max(a, b uint64) uint64 { | 
					
						
							|  |  |  | 	// amd64: "CMPQ","CMOVQHI" | 
					
						
							|  |  |  | 	// arm64: "CMP","CSEL" | 
					
						
							|  |  |  | 	// riscv64/rva20u64:"BLTU" | 
					
						
							| 
									
										
										
										
											2025-02-23 22:31:35 +11:00
										 |  |  | 	// riscv64/rva22u64,riscv64/rva23u64:"MAXU" | 
					
						
							| 
									
										
										
										
											2024-08-20 23:41:40 +10:00
										 |  |  | 	return max(a, b) | 
					
						
							|  |  |  | } |