If you have any questions or suggestions, please contact us in time. My official account is “Brain fried Fish”, GitHub address: github.com/eddycjy.
Hello everyone, I am a fried fish who is learning how to steam fish.
As mentioned earlier in the Go1.16 feature introduction, as of V1.16, the default memory management policy for Go under Linux will be changed from MADV_FREE to MADV_DONTNEED.
At this time there may be at least two groups of friends, respectively:
- I knew what it was, had been tortured by the question, and suddenly I saw the light.
- I don’t know what it is, there’s all kinds of confusion, what it’s all about.
The soul of torture
Do you have the following questions, or are you clear:
- What the text says
MADV_FREE
What is? - What the text says
MADV_DONTNEED
What is? - Why the Go language Linux environment in particular?
- Why do you say from
MADV_FREE
Changed backMADV_DONTNEED
?
In today’s article we will expand and explain, let us understand the changing memory mechanism is exactly what.
Madvise love and hate
On Linux, the system calls the madvise(addr, length, advise) method in the Go Runtime to tell the kernel how to handle the length bytes starting with addr.
One of the key points is “how to handle”. In Linux, Go currently supports two strategies:
- MADV_FREE: The kernel will only mark these process pages as recyclable in the page table and recycle them as needed.
- MADV_DONTNEED: The kernel marks these pages as “unallocated” in the process’s page table, making the process’s RSS smaller. The OS can then assign the corresponding physical pages to other processes.
The impact it has had
Go officials just happened to make the following changes in 2019’s Go1.12.
- Go1.12 before.
- Go. 12 – Go1.15.
Go1.12 before
Go Runtime uses the MADV_DONTNEED policy by default on Linux.
// Madvise (v, n, _MADV_DONTNEED)Copy the code
Overall, process RSS can drop faster, but not as fast in terms of performance efficiency.
Go1.12 – Go1.15
If the current Linux kernel version is >=4.5, Go Runtime uses the more efficient MADV_FREE policy by default on Linux.
var advise uint32 if debug.madvdontneed ! = 0 { advise = _MADV_DONTNEED } else { advise = atomic.Load(&adviseUnused) } if errno := madvise(v, n, int32(advise)); advise == _MADV_FREE && errno ! = 0 {// MADV_FREE was added in Linux 4.5. Fall back to MADV_DONTNEED if it is // not supported. atomic.Store(&adviseUnused, _MADV_DONTNEED) madvise(v, n, _MADV_DONTNEED) }Copy the code
Overall, the process RSS doesn’t drop immediately, it doesn’t drop until the system is under memory pressure.
Side effects
The story isn’t always pretty, and it’s clear that the tweaks to madvise’s MADV_FREE policy from Go1.12 were “one-sided.”
As we can see from the cases in the community, this adjustment has brought many problems:
- Problems causing user experience: There are always cases in Go Issues where memory leaks are considered, but the conditions are not met and the memory is not released immediately.
- Mix-up of statistics and monitoring tools: On Grafana and other monitoring tools, we found that the container process memory was high and was released slowly, causing alarm and panic.
- This leads to poor integration of individual management systems associated with memory usage: patterns such as Kubernetes HPA, or custom scaling policies that are difficult to evaluate.
- Squeezing other applications on the same host: Not all Go applications necessarily run independently on a single host, which naturally leads to squeezing other applications on the same host, which is difficult to evaluate.
From the community feedback is a lot of problems, more harm than good.
The performance was supposed to be better, but there were some new issues in the real world, including incompatibility with Android flow management.
It’s like penny wise and pound foolish.
Go1.16: A turnaround
Since there are so many questions from the community, has anyone asked? Yes, a lot.
It only took 1-2 days to complete the discussion of issues proposed to change back to MADV_DONTNEED.
Conclusions were quickly drawn and merge CL closed issues.
Go1.16 Modified as follows:
func parsedebugvars() { // defaults debug.cgocheck = 1 debug.invalidptr = 1 if GOOS == "linux" { debug.madvdontneed = 1 }... }Copy the code
Debug. madvdontneed = 1
conclusion
In this article, we introduced and explained the strategy adjustment of the Madvise method of Go language in Linux, and introduced the side effects and countermeasures brought by the adjustment one by one.
This change is a good confirmation, pull a whole body. And you’re going to have to pay attention to that as well.
My official account
Share Go language, micro service architecture and strange system design, welcome to pay attention to my public number and I exchange and communication.
The best relationship is mutual achievement. Your praise is the biggest motivation for the creation of fried fish. Thank you for your support.