preface
C #include
It’s hard to say what // Go: is at first, but let’s take a look at something very simple that almost everyone who writes code knows: #include in C. I’m guessing most people’s first line of code is #include. The complete one is #include
. The meaning is simple, introduce a stdio.h. Who introduction? The answer is the compiler. The # character, then, gives the compiler an indication of what to do next.
pragma
In computer programming, pragma is a language structure that instructs a compiler how it should handle its input. Directives are not part of the programming language syntax and vary from compiler to compiler.
here
WikiIt’s described in detail. It’s worth taking a look.
Go language compilation instructions
The official documentation
Golang.org/cmd/compil….
Such as //go: is the implementation of the go language compiler instructions. Those of you who have seen the Go SDK are familiar with this, as you often see it on the top line of code function declarations. // This is not a comment. Indeed, it exists in the form of comments.
Compiler source codeYou can see the full instructions here, but be careful,
//go:
It’s continuous,
//
和
go
There is no space between them.
Common instructions explained in detail
//go:noinline
noinline
As the name suggests, don’t inline.
The Inline Inline
Inline
, which occurs at compile time, is a compiler optimization that replaces the point at which a function call is called with the body of the called function. The Wiki:
The Inline definition
useInline
There are some advantages, but there are also some problems.
Advantage:
- Reduce the overhead of function calls and improve execution speed.
- Larger function bodies after replication open up the possibility of other compilation optimizations, such as interprocedural optimizations
- Eliminating branches and improving spatial locality and instruction ordering also improves performance.
Question:
- Space growth from code replication.
- If there is a lot of duplicate code, it will reduce the cache hit ratio, which is especially fatal to CPU cache.
Therefore, in practical use, the use of inlining should be carefully considered and balanced to make the most of it. In short, it is profitable to use inline for short and less demanding functions.
The example of inline
func appendStr(word string) string {
return "new " + word
}Copy the code
GOOS= Linux GOARCH=386 go tool compile -s main.go > main.S
0x0015 00021 (main.go:4) LEAL "".. autotmp_3+28(SP), AX 0x0019 00025 (main.go:4) PCDATA $2, $0 0x0019 00025 (main.go:4) MOVL AX, (SP) 0x001c 00028 (main.go:4) PCDATA $2, $1 0x001c 00028 (main.go:4) LEAL go.string."new "(SB), AX 0x0022 00034 (main.go:4) PCDATA $2, $0 0x0022 00034 (main.go:4) MOVL AX, 4(SP) 0x0026 00038 (main.go:4) MOVL $4, 8(SP) 0x002e 00046 (main.go:4) PCDATA $2, $1 0x002e 00046 (main.go:4) LEAL go.string."hello"(SB), AX 0x0034 00052 (main.go:4) PCDATA $2, $0 0x0034 00052 (main.go:4) MOVL AX, 12(SP) 0x0038 00056 (main.go:4) MOVL $5, 16(SP) 0x0040 00064 (main.go:4) CALL runtime.concatstring2(SB)Copy the code
As you can see, it does not call appendStr, but inline the functions of the body directly.
So what if you don’t want to be inlined? It is time to use go//:noinline, as follows:
//go:noinline
func appendStr(word string) string {
return "new " + word
}Copy the code
After compilation:
0x0015 00021 (main.go:4) LEAL go.string."hello"(SB), AX
0x001b 00027 (main.go:4) PCDATA $2, $0
0x001b 00027 (main.go:4) MOVL AX, (SP)
0x001e 00030 (main.go:4) MOVL $5, 4(SP)
0x0026 00038 (main.go:4) CALL "".appendStr(SB)Copy the code
Instead of inlining, the compiler calls appendStr directly.
//go:nosplit
Nosplit skips stack overflow detection.
What is stack overflow?
Because the starting stack size of a Goroutine is limited and relatively small, it is possible to support many goroutines concurrently and efficiently schedule them. Stack. go: _StackMin is 2048 bytes, or 2k bytes. It is not static, but grows dynamically when not enough. Then, there must be a detection mechanism to ensure that the stack is running out of time before it grows. Back to the topic, nosplit is just going to skip this mechanism.
Pros and cons
Obviously, performance can be improved by not performing stack overflow checking, but there is also the possibility of stack overflow causing compilation failures.
//go:noescape
Noescape is used to forbid escape, and it must indicate a function that has only a declaration and no body.
What is escape?
Go is a more memory-safe language than C or C++. One of the main points is that it can automatically move variables beyond their life cycle from the function stack to the heap. Escape refers to this behavior.
Please refer to my previous article,
Escape analysis.
Pros and cons
The most obvious benefit is that the GC pressure is reduced. Since it has told the compiler that the following function will not escape anyway, when the function returns, its resources will be destroyed as well. Doing so, however, means bypassing the compiler’s escape check, which can lead to serious errors and consequences once it is run.
//go:norace
We know that in multithreaded programs, data contention is inevitable. Normally, when the compiler detects a data contention, it will alert you. Such as:
var sum int
func main() {
go add()
go add()
}
func add() {
sum++
}Copy the code
The go run -race main.go command uses -race to cause the compiler to report data race problems. You will see:
==================
WARNING: DATA RACE
Read at 0x00000112f470 by goroutine 6:
main.add()
/Users/sxs/Documents/go/src/test/main.go:15 +0x3a
Previous write at 0x00000112f470 by goroutine 5:
main.add()
/Users/sxs/Documents/go/src/test/main.go:15 +0x56
Goroutine 6 (running) created at:
main.main()
/Users/sxs/Documents/go/src/test/main.go:11 +0x5a
Goroutine 5 (finished) created at:
main.main()
/Users/sxs/Documents/go/src/test/main.go:10 +0x42
==================
Found 1 data race(s)Copy the code
Indicates that two Goroutine add() implementations are competing.
Pros and cons
I can’t think of any other benefit to using Norace other than reducing compilation time. However, the disadvantage is obvious, that is, data competition can lead to program uncertainty.
conclusion
I think for the most part, you don’t need to use it in programming//go:
The Go compiler dictates that unless you are sure your program’s performance bottleneck is in the compiler, you should first worry about other things that are more likely to cause bottlenecks.
reference
- dave.cheney.net/2018/…