I’ve been looking for methods to improve Emacs performance especially with my configuration being over >3k. I’m not particularly interested in startup-time since I never close Emacs. Here’s what I found so far
(setq package-native-compile t
gcmh-high-cons-threshold 100000000
gc-cons-threshold 100000000
scroll-conservatively 101
jit-lock-defer-time 0
large-file-warning-threshold nil)
(add-hook 'after-init-hook #'(lambda () (setq gc-cons-threshold (* 100 1000 1000))))
(defvar gc-timer nil)
(defun salih/maybe-gc ()
(let ((original gc-cons-threshold))
(setq gc-cons-threshold 800000)
(setq gc-cons-threshold original
gc-timer (run-with-timer 2 nil #'salih/schedule-maybe-gc))))
(defun salih/schedule-maybe-gc ()
(setq gc-timer (run-with-idle-timer 2 nil #'salih/maybe-gc)))
(salih/schedule-maybe-gc)
I can tell that I’ve noticed some improvements.
Define fast. Speed is relative. Your fast and my fast may not be the same. But my answer is lazy load everything.
If not in startup, what performance deficits and improvements have you noticed? How do you measure or notice the improvements?
> I can tell that I’ve noticed some improvements.
I can tell this claim is worthless without data.
For me, the main issue was startup speed from my customizations. This I sped up by:
- Making use of
emacsclient
over new emacs sessions. - Combining scattered customization files into a single large
emacs.el
file. - Use
defvar
andautoload
overrequire
,load
,eval-after-load
.
The last one is mostly to allow compiler- and flycheck warnings to work, without prematurely loading dependencies of my customization code.
- Making use of
switch to neovim
Have a look at Emacs client/server mode.
It creates a single hidden instance of Emacs, and lets all other emacs windows connect to it.You can start the server with your computer start up, and then opening clients is instant.
Buy a good cpu
Upgraded to 29.1
(advice-add 'jsonrpc--log-event :override #'ignore)
Can you say what’s going on here?
I’m assuming it’s avoiding doing work, but from which package? Something in core Emacs?
Doom Emacs has great optimizations for this.
You should manage your packages and defer those that are not immediately necessary (using use-package options and consider using benchmark to control initialization time).
If you truly want to achieve blazing speed, consider running Emacs as a daemon (Emacs Daemon).
I’ve focused on what I really need and avoid cluttering my config. I’ve spent considerable time understanding what performs well and what doesn’t. I try to minimize the use of global modes and refrain from installing additional packages if built-in ones do the job. I’m very discerning when it comes to choosing any new package. Essentially, I have to “sell” myself on a new package, if you know what I mean. Yes, I do use byte-compilation for Lisp code and feel that I’ve managed to make things quite efficient and clean. You can take a look and borrow any solutions from my setup here: https://github.com/sergeyklay/.emacs.d
With the new
--init-directory
switch of Emacs 29, you can define a completely separate environment for Emacs that could be used to make more mean, lean and specialized versions of Emacs. For example, instead of having your Emacs do everything, IRC, email, programming, Org mode stuff etc. etc. etc. you could have a shell command invoking each one, e.g. Eirc, Eemail, Eorg, etc.
LSP-mode: By following instructions here: https://emacs-lsp.github.io/lsp-mode/page/performance/
For the rest: M-x profiler-start, Mx profiler-stop, then M-x profiler-report.
If you want to improve the bootup performance, you can use the Client/Server feature that Emacs provides. It can utilize it to significantly speed up your Emacs.
Run the Emacs server by:
emacs --daemon
And Emacs client connects to it automatically:
emacsclient sample.cpp
- Get better hardware if you are still on old hardware 5 or more years ago. A budget $300 - $500 PC with modern CPUs and a decent SSD will be significant faster than whatever 5 years ago.
- If you’re on Windows, compiled your own Emacs, then compile every packages (built-in or 3rd party) with native-comp. Here’s a guide: https://www.reddit.com/r/emacs/comments/131354i/guide_compile_your_own_emacs_to_make_it_really/
- ???
- Enjoy!
I can tell that I’ve noticed some improvements.
I guess you haven’t run your Emacs for a long time, because ~100 megabytes of allocated memory will take its time to GC check. With that amount of RAM, you will probably notice Emacs “stuttering”, like freezing for small periods of time when using it normally. The bigger the gc-cons-threshold value, the longer time it will take for GC to check the memory. Emacs does not have an incremental and multithreaded GC, so your Emacs will appear as frozen to you.
(defun salih/maybe-gc () (let ((original gc-cons-threshold)) (setq gc-cons-threshold 800000) (setq gc-cons-threshold original gc-timer (run-with-timer 2 nil #'salih/schedule-maybe-gc))))
Have you even looked at gc-cons-threshold value; after your idle timer has finished the work? Looking at your code, I believe you will be surprised because it does not seem to be what you think it will be.
What is the point of the first setq there?
In the third line, you are setting the value to hardcoded ~800 kb (I think it is the on 64-bit systems, but it is not so important), just to immediately override it with 100 meg as you defined it in the previous code piece (third line of your code), one after the gcmh. Secondly, what is the reason to use gcmh package if you are going to do it all manually :). GCMH will do exactly the same thing you are doing manually in that example if I am not mistaken; I don’t use that package myself, but someone wrote it for the purpose of automating that little hack you are trying to make there. IMO, either use that package and be happy, or do it all manually.
What is the purpose of setq-ing gc-timer to run your function in that let-body (last line) which appears to just do exactly the same - you are again doing the same thing in your salih/schedule-maybe-gc. You are just telling Emacs in an infinite loop to set up a new timer when it is idle. If you believe you are telling Emacs to actually garbage collect something, you are wrong; you are just setting up another timer.
In other words; your code does not do what you believe it does; it is rather plain wrong, in other words, buggy, to put it mildly.
I can tell that I’ve noticed some improvements.
I can tell you haven’t, you just don’t know about it. When other posters here told you to benchmark they were correct.
we are not in an academic seminar, such anecdotal statements should be authentic enough.
We don’t benchmark because of being academics, but because of ourselves. If we won’t to improve something in whatever terms, cpu execution time, memory usage, number of resources allocated (timers, files, sockets etc.) you have to measure. You can’t know for sure if you don’t measure, there is no way. Your computer can be doing stuff, your application can be doing stuff, and so on. Modern computer systems are not deterministic in terms that hardware usage being exactly the same each time you run an application. Execution time is highly dependent on your OS and CPU scheduler(s), memory usage patterns and so on, some of the things your application usually does not influence explicitly.
Without measuring you are walking with a blindfold. However, you seem to have other problems than just benchmarking your stuff; you should really read the manual about stuff you are trying to improve or change, use built-in help; C-h f/v to see what stuff does and try to understand it. Read whichever blog posts you have found again and reflect carefully on what they say and why. Just blindly copying stuff without understanding it results in stuff like your code above.
Finally, to answer your original question, it all depends on how you wish to use your Emacs. What might be fast for one usage pattern, might not be fast for another one. Again, you will have to know what you are doing and to measure for your particular use-case.