Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Redrawing room list buffer can be slow when in many rooms #298

Closed
Rutherther opened this issue Sep 13, 2024 · 56 comments
Closed

Redrawing room list buffer can be slow when in many rooms #298

Rutherther opened this issue Sep 13, 2024 · 56 comments
Assignees
Labels
bug Something isn't working priority:A

Comments

@Rutherther
Copy link

OS/platform

Guix System or NixOS

Emacs version and provenance

Tried from both Nix and Guix, both times with jit native compilation enabled.
GNU Emacs 29.4 (build 1, x86_64-pc-linux-gnu, GTK+ Version 3.24.41, cairo version 1.18.0)

Emacs command

./with-emacs.sh --dir /tmp/test

Emacs frame type

GUI

Ement package version and provenance

I've used method in README to obtain from latest commit. That is 3f87a95 when trying this.

Actions taken

  1. execute ement-connect
  2. log in
  3. wait for a bit after the ement room list opens

Observed results

After few seconds Emacs lags, for a few seconds without being able to do anything. This repeats every few seconds between the lags.

Expected results

I should be able to use emacs normally when having ement room list open.

Backtrace

No response

Etc.

       51922  54% + timer-event-handler
       40607  42% + command-execute
        3010   3% + ...
         274   0% + redisplay_internal (C function)
           3   0%   mouse--click-1-maybe-follows-link

I have not used profiler at all till now and I am not sure how to get relevant information. I can run for specific functions, but I don't know what to target.

I observe this behavior both in my regular Emacs installation and on a fresh new emacs version obtained with with-emacs.sh script.

I am on Conduit server. I also have sliding sync running on the server. I am in more than 200 rooms. Few of the rooms are bigger groups. But most of them are bridged DM rooms. I use Mautrix Discord (no guild bridged, only DMs), Mautrix Telegram, Mautrix Whatsapp, Mautrix Meta, Heisenbridge (bridging a few irc groups).

I have tried registering a separate account and join a few rooms (I am in those on my main account) that have more members. I did not observe this behavior on that account.

I have already reached alphapapa through Ement.el room, where it was suggested to open an issue. This was like two weeks ago (sorry for taking so long)

@Rutherther Rutherther added the bug Something isn't working label Sep 13, 2024
@Rutherther
Copy link
Author

Additionally since I named all the bridges I use, I've tried stopping all of those so the bot accounts don't get a chance to update anything temselves. This didn't solve anything

@phil-s
Copy link

phil-s commented Sep 13, 2024

You can fully-expand a node in the profile report with C-uRET

You want to read C-hig (elisp)Profiling

@Rutherther
Copy link
Author

Right, that makes sense, should've read the manual thoroughly...

       17169  94% - timer-event-handler
       17169  94%  - apply
       17169  94%   - plz--respond
       17169  94%    - #<compiled 0x8d5096906d58e94>
       17165  94%     - #<compiled -0xade022a6f6d61b0>
       17165  94%      - apply
       17165  94%       - ement--sync-callback
       17165  94%        - run-hook-with-args
       17157  94%         - ement-room-list-auto-update
       17157  94%          - revert-buffer
       17157  94%           - ement-room-list-revert
       17157  94%            - ement-room-list
       17049  93%             - taxy-magit-section-format-items
       17049  93%              - #<compiled -0xa7c2710a7245255>
       17049  93%               - #<compiled -0xa7c2710a7245255>
       17029  93%                - #<compiled -0xa7c2710a7245255>
       15961  87%                 - #<compiled 0x1f264c6d0a102517>
       15961  87%                  - #<compiled 0x1ccd7f36e7b5ed28>
       15439  84%                   - ement-room-list-column-format-name
       15362  84%                    - ement--room-display-name
       15354  84%                     - #<compiled -0xa2f04426fcf8645>
       15326  84%                      - mapcar
       15318  84%                       - #<compiled -0xc0db1ca136f5bc5>
        6335  34%                        - ement--user-displayname-in
        1387   7%                           #<compiled 0x53b2892b14fda62>
          12   0%                      - cl-remove-if-not
           4   0%                       - cl-remove
           4   0%                        - cl-delete
           4   0%                         - #<compiled -0xd07981da80d5c6f>
           4   0%                            apply
           4   0%                       #<compiled -0x66e54a1270e5219>
          72   0%                      truncate-string-to-width
           5   0%                    - ement--room-direct-p
           5   0%                     - #<compiled 0x1c375647c4cf9cae>
           5   0%                        seq-contains-p
         503   2%                     #<compiled 0x2689ac9dafa9eef>
           8   0%                   - ement-room-list-column-format-latest
           4   0%                      ement--human-format-duration
           4   0%                     map-put!
           3   0%                   - ement-room-list-column-format-notifications
           3   0%                    - ement-room-notification-state
           3   0%                     - map-nested-elt
           3   0%                      - seq-reduce
           3   0%                       - seq-do
           3   0%                        - mapc
           3   0%                         - #<compiled 0x1a0c8572a1b1c6fe>
           3   0%                          - #<subr F616e6f6e796d6f75732d6c616d626461_anonymous_lambda_14>
           3   0%                             map-elt
        1068   5%                 - #<compiled -0xa7c2710a7245255>
        1064   5%                  - #<compiled 0x1f264c6d0a102517>
        1064   5%                   - #<compiled 0x1ccd7f36e7b5ed28>
         972   5%                      #<compiled 0x2689ac9dafa9eef>
          48   0%                    - ement-room-list-column-format-name
          28   0%                     - ement--room-direct-p
          28   0%                      - #<compiled 0x1c375647c4cf9cae>
          24   0%                       - seq-contains-p
          12   0%                        - seq-do
           8   0%                         - mapc
           4   0%                            #<compiled 0xa1508dc680577d0>
           8   0%                       ement--room-display-name
           4   0%                       truncate-string-to-width
           4   0%                     - ement--room-unread-p
           4   0%                      - map-nested-elt
           4   0%                       - seq-reduce
           4   0%                        - seq-do
           4   0%                         - mapc
           4   0%                          - #<compiled 0x1a0cbb08a1b1c6fe>
           4   0%                           - #<subr F616e6f6e796d6f75732d6c616d626461_anonymous_lambda_14>
           4   0%                            - map-elt
           4   0%                               apply
          16   0%                    - map-put!
          16   0%                     - apply
          12   0%                      - #<compiled 0x1fe906d3872180>
           4   0%                         assoc
          12   0%                    - ement-room-list-column-format-notifications
          12   0%                     - ement-room-notification-state
           8   0%                      - #<compiled -0x171045b66f07ec2e>
           8   0%                       - map-nested-elt
           8   0%                        - seq-reduce
           8   0%                         - seq-do
           8   0%                          - mapc
           8   0%                           - #<compiled 0x1a0c833aa1b1c6fe>
           8   0%                            - #<subr F616e6f6e796d6f75732d6c616d626461_anonymous_lambda_14>
           8   0%                             - map-elt
           4   0%                                apply
           4   0%                      - cl-find-if
           4   0%                       - cl-find
           4   0%                          cl-position
           4   0%                    - ement-room-list-column-format-topic
           4   0%                       replace-regexp-in-string
           4   0%                      map-elt
          20   0%                - #<compiled 0x1f264c6d0a102517>
          20   0%                 - #<compiled 0x1ccd7f36e7b5ed28>
          12   0%                    #<compiled 0x2689ac9dafa9eef>
           4   0%                  - ement-room-list-column-format-name
           4   0%                     ement--room-direct-p
           4   0%                  - ement-room-list-column-format-🐱
           4   0%                     map-elt
          60   0%             - taxy-fill
          60   0%              - #<compiled 0xd9ce7609f704410>
          24   0%               - #<lambda 0x5b7290e5d7>
          24   0%                - taxy-take-keyed
          24   0%                 - taxy-take-keyed
          24   0%                  - taxy-take-keyed
          24   0%                   - taxy-take-keyed
          24   0%                    - taxy-take-keyed
          24   0%                     - taxy-take-keyed
          24   0%                      - taxy-take-keyed
          20   0%                       - taxy-take-keyed
          20   0%                        - taxy-take-keyed
          16   0%                         - #<lambda 0x9884c6d4680d254>
          16   0%                          - when
          16   0%                           - if
          12   0%                            - cl-loop
           8   0%                             - cl-block
           4   0%                              - cl--block-wrapper
           4   0%                               - catch
           4   0%                                  let*
           4   0%                                #<compiled -0x118465efb9c3cb7a>
           4   0%                         - taxy-take-keyed
           4   0%                          - cl-find-if
           4   0%                           - cl-find
           4   0%                            - cl-position
           4   0%                               cl--position
           4   0%                       - #<lambda -0x1667dbc0235e70fe>
           4   0%                        - when
           4   0%                         - if
           4   0%                          - cl-loop
           4   0%                             #<compiled -0x18ca13412de6202b>
          20   0%               - #<lambda 0x5b7290e5d7>
          20   0%                - taxy-take-keyed
          20   0%                 - taxy-take-keyed
          20   0%                  - taxy-take-keyed
          20   0%                   - taxy-take-keyed
          20   0%                    - taxy-take-keyed
          20   0%                     - taxy-take-keyed
          20   0%                      - taxy-take-keyed
          20   0%                       - taxy-take-keyed
          16   0%                        - taxy-take-keyed
          16   0%                         - taxy-take-keyed
          12   0%                          - taxy-take-keyed
           8   0%                           - taxy-take-keyed
           8   0%                            - ement-room-list-key-space
           8   0%                             - #<compiled -0xb12d938b9c3b04a>
           8   0%                              - cl-find
           8   0%                               - cl-position
           4   0%                                - cl--position
           4   0%                                   ement-room-id
           4   0%                           - ement-room-list-key-people
           4   0%                            - ement--room-direct-p
           4   0%                             - #<compiled 0x1c375647c4cf9cae>
           4   0%                                seq-contains-p
           4   0%                          - cl-find-if
           4   0%                           - cl-find
           4   0%                              cl-position
           4   0%                        - #<lambda -0x1667dbc0235e6d66>
           4   0%                         - when
           4   0%                          - if
           4   0%                             cl-loop
          16   0%               - #<lambda 0x5b7290e5d7>
          16   0%                - taxy-take-keyed
          16   0%                 - taxy-take-keyed
          16   0%                  - taxy-take-keyed
          16   0%                   - taxy-take-keyed
          16   0%                    - taxy-take-keyed
          16   0%                     - taxy-take-keyed
          16   0%                      - taxy-take-keyed
          16   0%                       - taxy-take-keyed
          16   0%                        - taxy-take-keyed
          12   0%                         - taxy-take-keyed
           8   0%                          - ement-room-list-key-people
           8   0%                           - ement--room-direct-p
           8   0%                            - #<compiled 0x1c375647c4cf9cae>
           8   0%                               seq-contains-p
           4   0%                          - taxy-take-keyed
           4   0%                           - ement-room-list-key-people
           4   0%                            - ement--room-direct-p
           4   0%                             - #<compiled 0x1c375647c4cf9cae>
           4   0%                                seq-contains-p
           4   0%                         - #<lambda -0x1667dbc0235e6d66>
           4   0%                          - when
           4   0%                             if
          36   0%             - taxy-magit-section-insert
          36   0%              - #<compiled -0x1e8784fb555b0cfa>
          36   0%               - #<compiled -0x1aa0cdce6d225f87>
          36   0%                - #<compiled -0x1e8784fb555b0cfa>
          36   0%                 - #<compiled -0x1fa4b3834370d217>
          32   0%                  - #<compiled -0x1e8784fb555b0cfa>
          32   0%                   - #<compiled 0x7c103c4459c9ae>
          28   0%                    - #<compiled -0x1e8784fb555b0cfa>
          24   0%                     - #<compiled -0x3bd58746dd0a204>
          24   0%                      - #<compiled 0x10d6bdcfa45d7357>
          12   0%                       - magit-insert-section--create
           8   0%                        - run-hook-with-args-until-success
           4   0%                         - magit-section-cached-visibility
           4   0%                          - magit-section-ident
           4   0%                             eieio-oref
           4   0%                           taxy-magit-section-visibility
           4   0%                        - apply
           4   0%                         - ement-room-list-section
           4   0%                            apply
           4   0%                     - magit-insert-section--create
           4   0%                        eieio-oset
           4   0%                    - #<compiled 0x10d6bdcfa45d7357>
           4   0%                       #<compiled 0x351f1af2b08050d>
           4   0%                    #<compiled 0x10d6bdcfa45d7357>
           8   0%             - taxy-sort
           8   0%              - taxy-mapc*
           8   0%               - taxy-mapc-taxys
           4   0%                - #<compiled 0x485719bda9c4691>
           4   0%                 - cl-sort
           4   0%                  - #<compiled -0xd2b365d0d882dc3>
           4   0%                   - #<compiled 0x1587f0a2f36a7ac6>
           4   0%                    - ement--room-unread-p
           4   0%                       map-nested-elt
           4   0%                - taxy-mapc-taxys
           4   0%                 - #<compiled 0x1f478977502fc02>
           4   0%                    cl-sort
           4   0%             - taxy-sort*
           4   0%              - taxy-mapc*
           4   0%               - taxy-mapc-taxys
           4   0%                - taxy-mapc-taxys
           4   0%                   #<compiled 0xde8c33a96b565f4>
           8   0%         - ement--auto-sync
           8   0%          - ement--sync
           8   0%           - ement-api
           8   0%            - apply
           8   0%               plz
           4   0%     - plz--coding-system
           4   0%        coding-system-from-name
         624   3% - command-execute
         535   2%  - funcall-interactively
         334   1%   - next-line
         242   1%    - line-move
          12   0%     - line-move-partial
           9   0%      - window-screen-lines
           5   0%       - window-inside-pixel-edges
           2   0%          window-edges
           4   0%         default-line-height
           3   0%      - default-line-height
           1   0%         default-font-height
           8   0%       default-line-height
         201   1%   - previous-line
         178   0%    - line-move
           4   0%     - line-move-partial
           4   0%        default-line-height
          89   0%  - byte-code
          89   0%   - read-extended-command
          89   0%    - read-extended-command-1
          35   0%     - completing-read-default
           7   0%        command-execute
         377   2% + ...
          26   0% + magit-section-post-command-hook
          17   0% + redisplay_internal (C function)
          12   0%   internal-timer-start-idle
           2   0% + magit-section-pre-command-hook

@phil-s
Copy link

phil-s commented Sep 13, 2024

I think you'll want to test 2212b38

@Rutherther
Copy link
Author

Maybe I should've been more specific. Yes, I know about it, I've tested it. alphapapa made it when I told them about the lags. It makes it a bit better, since it doesn't lag when doing operations with emacs quickly, but after I am idle for only a second, it usually lags again, if not right after the second, then after few more seconds.

@phil-s
Copy link

phil-s commented Sep 14, 2024

You can, of course, (setq ement-room-list-update-idle-delay N) to some N which is more than the 1 second value currently specified.

If you have ement-room-list-auto-update enabled at all then it's going to Do Things periodically, but the above lets you decide how often that might happen.

@alphapapa alphapapa changed the title Ement.el lags Emacs a lot Redrawing room list buffer can be slow when in many rooms Sep 14, 2024
@alphapapa
Copy link
Owner

alphapapa commented Sep 14, 2024

There are a few fundamental issues here:

  1. You're in many rooms. If they are active, that means the server is constantly sending new events as they arrive in the rooms.
  2. Every time new events are received, the room list buffer is updated, if it exists.
  3. Rendering a large number of rooms can be somewhat slow, depending on various factors. AFAIK the chief issue is that all of the values for all of the columns for all of the rooms must be realized so that their width can be calculated, so that the columns' widths can be calculated.
  4. It's not clear whether the relevant Elisp code is being natively compiled. You said, "Tried from both Nix and Guix, both times with jit native compilation enabled," but you may not be aware that, e.g. Guix's Emacs packaging does not native-compile Elisp packages which are installed outside of Guix (an unfortunate decision IMO, which they have declined to change). So the relevant Elisp code here may be only byte-compiled, which may be reducing its performance compared to what I experience on my system.

So there are various possible ways to address these issues:

  1. Improve the performance of taxy-magit-section. There might be some caching that could be implemented. That probably won't happen anytime soon, as I have other priorities for the near future. Another possibility would be to see if individual rooms could be updated, but AFAIK magit-section doesn't really support that, so it's unlikely to work; and it might take a few hours of work just to find out.
  2. Increase the idle delay so the room buffer won't be updated unless you're really not using Emacs for a little while. Of course, that means it won't get updated as often, so it will be outdated sometimes.
  3. Stop updating the room list automatically altogether. Either kill the buffer, or disable automatic updates.
  4. Reduce the number of rooms you're in. Obviously that has side effects, so it would probably be a last resort.
  5. The ement-tabulated-room-list library is still present. It may still work. It may work more quickly than the taxy-magit-section-based one, but it does no grouping. Maybe it could be an alternative for you.
  6. Maybe someone could implement a vtable-based room list. It would also have no grouping features, but it might render more quickly.
  7. You could potentially re-evaluate all of the relevant code so as to make all of the functions interpreted, so that the profiler would give a bit more information (i.e. not show functions as compiled, which obscures their purpose). Maybe there are some hot spots that could be optimized.

None of these are great options, but they're all I know of now.

@alphapapa alphapapa added enhancement New feature or request help wanted Extra attention is needed discussion priority:C and removed bug Something isn't working labels Sep 14, 2024
@alphapapa alphapapa added this to the Future milestone Sep 14, 2024
@phil-s
Copy link

phil-s commented Sep 14, 2024

@Rutherther, you should look into all the above suggestions first; but if you've tried everything you can and still can't get a good result, another option could be #294 (comment).

That (or rather a previous version of that code) was my workaround back when I was afflicted by a Matrix server bug which was having a somewhat similar effect to your issue.

@Rutherther
Copy link
Author

Rutherther commented Sep 14, 2024

  1. So if any event from any of the rooms is received, the whole buffer gets redrawn now? Still it's strange that for me this redraw is slow compared to yours since you also are in many rooms.
  2. Yes, Guix decided to disable jit native compilation by default - by making a patch that sets the default value of native-jit-compilation in low level C code - https://git.savannah.gnu.org/cgit/guix.git/tree/gnu/packages/patches/emacs-disable-jit-compilation.patch. That made stuff outside of Guix not native compile (apart from that also many emacs packages from Guix are made to not native compiled by default and one has to actively do something like --with-inputs=emacs-minimal=emacs to get everything native compiled). I set it to true and verified that stuff gets jit native compiled.
    ie.
ement-room-list is an autoloaded interactive native-compiled Lisp
function in ‘ement-room-list.el’.

Thanks for the suggestions, I will try to look into some of them, and probably as the most used workaround for now I will just close this buffer, as I do not need it very often.

@alphapapa
Copy link
Owner

  1. So if any event from any of the rooms is received, the whole buffer gets redrawn now? Still it's strange that for me this redraw is slow compared to yours since you also are in many rooms.

When a sync response is received from the server, the room list buffer is updated, and that re-renders the whole buffer, because anything that was previously rendered could have changed.

3. Yes, Guix decided to disable jit native compilation by default - by making a patch that sets the default value of native-jit-compilation in low level C code - https://git.savannah.gnu.org/cgit/guix.git/tree/gnu/packages/patches/emacs-disable-jit-compilation.patch. That made stuff outside of Guix not native compile (apart from that also many emacs packages from Guix are made to not native compiled by default and one has to actively do something like `--with-inputs=emacs-minimal=emacs` to get everything native compiled).

Yes, see https://debbugs.gnu.org/cgi/bugreport.cgi?bug=71725

I set it to true and verified that stuff gets jit native compiled.
ie.

ement-room-list is an autoloaded interactive native-compiled Lisp
function in ‘ement-room-list.el’.

Ok, but that function is not what renders the room buffer, it just calls the functions that do. So you would need to verify that all of the relevant functions are native-compiled.

Thanks for the suggestions, I will try to look into some of them, and probably as the most used workaround for now I will just close this buffer, as I do not need it very often.

Ok.

@phil-s
Copy link

phil-s commented Sep 19, 2024

@alphapapa, I started using wip/debounce-room-list-update yesterday, and noticed that the idle timer was only going to fire once at most in any idle period, and I wasn't sure that was the intention (although it might be).

I started writing some more complicated idle-time code, and then thought "why not just check how long it's been since the last refresh vs the duration set by the user, and not refresh if it hasn't been long enough. So I did that, and then realised that didn't quite work either (as a long period between syncs could prevent the refresh from happening in any timely fashion).

Then I wrote myself this comment, and turned off my laptop for the day :)

    ;; What we ACTUALLY want (I think):
    ;;
    ;; - If time-since-refresh > threshold, update the buffer.
    ;; - Otherwise, `run-with-timer' or `timer-set-time'
    ;;   as appropriate for a REGULAR (not-idle) timer,
    ;;   to re-try the refresh in (threshold - time-since-refresh)
    ;;   seconds from now.
    ;; - Regardless... once the refresh is actually triggered
    ;;   (whether directly or by timer), run THAT using an
    ;;   idle timer with a short delay (1s).
    ;; - Two new user options, for the two durations.
    ;;   If either is nil, that part doesn't use a timer.
    ;;
    ;; This way:
    ;; - We don't refresh more frequently than every N seconds.
    ;; - We don't refresh while the user is actively doing things,
    ;;   but rather wait until they've paused.

And re-reading that, I'd be inclined to:

  • just create a single repeating non-idle timer on the specified delay (rather than continually creating and discarding timers)
  • keep track of the time of the last actual refresh and whether a refresh is 'pending' (i.e. there has been a sync since the last time we actually performed a refresh), and the timer callback just checks for a pending refresh and runs that (on the idle timer) if necessary.

Does that sound like a good approach?

@alphapapa
Copy link
Owner

alphapapa commented Sep 19, 2024

I don't know. We seem to be making it more complicated, which I'd like to avoid.

But another factor to consider is whether the room list buffer is visible. If it's not in a visible window, then it doesn't seem as urgent to update it, so an idle timer would seem to make sense. If the buffer is visible, I feel like it ought to be updated in real time, otherwise we'd be showing incorrect information to the user. So we might also need to use a function on the window configuration hook to update the buffer immediately if it becomes visible.

I still don't fully understand what the performance problem is. I am in 127 rooms--which isn't as many as 200, but if 200 were causing a significant delay, I'd expect 127 to cause a noticeable one too--and I notice no pauses from updating the buffer. So I also wonder whether users who are reporting problems are running the relevant code as interpreted, byte-compiled, or native-compiled, or some combination. I don't feel like we have enough information to understand what's really going on. There could also be something unusual going on, e.g. if a room's avatar image were very large, and it were having to be resized down to icon size on every redraw (which shouldn't happen, since that image does get cached, but who knows), then only users in such a room would have the problem. IOW this doesn't seem like a universal problem, which suggests there's more to it.

The ultimate solution might be to refactor taxy-magit-section to more effectively cache values between redraws. But that would be a lot of work, so I don't intend to do that soon.

@phil-s
Copy link

phil-s commented Sep 19, 2024

I'd intermittently experienced some performance issues from it in the past week or so, which is why I initially added your patch the other day, but I don't really have any insights -- most of the time it was fine, so it just seemed like the server was deciding to temporarily be super chatty for some reason. I don't recall having similar issues anytime recently, but maybe the server was updated and introduced something relatively noisy.

we might also need to use a function on the window configuration hook to update the buffer immediately if it becomes visible.

Yep, good idea. The buffer-local hook value works well for that sort of thing.

@alphapapa
Copy link
Owner

For future reference, I pushed v0.14.2 of taxy-magit-section today, which includes a minor performance optimization: https://github.com/alphapapa/taxy.el/tree/package/taxy-magit-section?tab=readme-ov-file#0142 I don't know if it will make a noticeable difference, but it may help a little.

@Rutherther
Copy link
Author

So I also wonder whether users who are reporting problems are running the relevant code as interpreted, byte-compiled, or native-compiled, or some combination. I don't feel like we have enough information to understand what's really going on.

@alphapapa Okay, so how should I convince you it's native-compiled? is there an Emacs function that will say if the whole tree is native compiled, or should I send each function description output as I did for the one I already sent as an example?

@Rutherther
Copy link
Author

For future reference, I pushed v0.14.2 of taxy-magit-section today, which includes a minor performance optimization: https://github.com/alphapapa/taxy.el/tree/package/taxy-magit-section?tab=readme-ov-file#0142 I don't know if it will make a noticeable difference, but it may help a little.

Switched to v0.14.2 and no observable difference for me

@phil-s
Copy link

phil-s commented Sep 20, 2024

@Rutherther would you like to give phil-s@6e696d5 a try?

(setq ement-room-list--show-debug-messages nil) will silence the *Messages* saying what it's doing, but I'd encourage you to watch them while testing to get a feel for what it's doing.

The user options ement-room-list-update-interval and ement-room-list-update-idle-delay control the delays. You should set them to values which work for you. If you set them both to nil you get the behaviour of the current release version, and if you only set the idle delay you essentially get the behaviour of the previous patch. With both values non-nil you get the new behaviour.

I'm keen to know whether / to what extent this alleviates things for you.

@Rutherther
Copy link
Author

Hello @phil-s, thanks for that. Yes, debouncing is solving part of the problem. But the underlying problem of rendering being slow is not solved. So Emacs still lags when it syncs, but not only that. Thanks to it being delayed I can actually start doing something in the ement room list, and notice that even toggling sections is causing this slowness for me. I suppose that also does the redraw of the whole buffer.

@Rutherther
Copy link
Author

       46373  96% - ...
       28177  58%  - #<lambda 0x189e4f6bbe25258a>
       28177  58%   - progn
       28177  58%    - let
       28177  58%     - while
       28177  58%      - let
       28177  58%       - funcall
       28177  58%        - #<lambda 0xef1f897f362716f>
       28177  58%         - puthash
       28177  58%          - let*
       28177  58%           - while
       28177  58%            - setq
       28177  58%             - cons
       28177  58%              - funcall
       28177  58%               - #<lambda -0x1002b59e799cf870>
       28177  58%                - let*
       28165  58%                 - funcall
       28087  58%                  - ement-room-list-column-format-name
       27994  58%                   - ement--room-display-name
       27994  58%                    - let*
       27986  58%                     - let
       27970  58%                      - or
       27970  58%                       - funcall
       27906  57%                        - #<lambda 0x121001c773bbe1e2>
       27906  57%                         - let*
       27894  57%                          - if
       27894  57%                           - string-join
       27894  57%                            - delete-dups
       27890  57%                             - mapcar
       27886  57%                              - #<lambda 0x121001dcaa361c72>
       27886  57%                               - ement--user-displayname-in
       27882  57%                                - let*
       27878  57%                                 - if
       27878  57%                                  - let*
       27854  57%                                   - let*
       27850  57%                                    - and
       27850  57%                                     - or
       27850  57%                                      - let*
       27387  56%                                       - while
       23417  48%                                        - and
       19897  41%                                         - progn
       11793  24%                                          - if
        8278  17%                                           - funcall
        3255   6%                                            - #<lambda -0x1e870bd4ffb71f72>
        2154   4%                                               and
        2877   5%                                            setq
        2142   4%                                          setq
           4   0%                                       + progn
           4   0%                                    + if
          20   0%                                   + function
           4   0%                                   and
           4   0%                                  progn
           4   0%                             + let
          12   0%                          + and
          64   0%                        + #<lambda 0x120e306f085321fe>
          12   0%                      + setq
           8   0%                     + function
          57   0%                   + ement--room-unread-p
          20   0%                     truncate-string-to-width
          16   0%                   + ement--room-direct-p
          66   0%                  + ement-room-list-column-format-notifications
           8   0%                  + #<lambda 0x19b1fd14e15fe4>
           4   0%                  + ement-room-list-column-format-latest
          12   0%                 + let*
       12352  25%  - ement--push-joined-room-events
       12352  25%   - ement-notify
       12352  25%    - run-hook-with-args-until-success
       12352  25%     - ement-notify--event-mentions-session-user-p
       12352  25%      - ement-room--event-mentions-user-p
       12352  25%       - ement--user-displayname-in
       12352  25%        - let*
       12352  25%         - if
       12352  25%          - let*
       12352  25%           - let*
       12352  25%            - and
       12352  25%             - or
       12352  25%              - let*
       12352  25%               - while
       12352  25%                - and
       12352  25%                 - progn
       12348  25%                  - if
       12348  25%                   - funcall
       12344  25%                    - #<lambda -0x1e870bd4ffb71f72>
       12344  25%                     - and
       12316  25%                      - eq
       10897  22%                       - progn
        7459  15%                        - or
        6355  13%                         - progn
        4369   9%                          - and
        1600   3%                             memq
          24   0%                      + equal
        3831   7%  - #<lambda 0x189e4f6bbe25258a>
        3831   7%   - progn
        3831   7%    - let
        3831   7%     - while
        3831   7%      - let
        3831   7%       - funcall
        3831   7%        - #<lambda 0xef1f9e7f9b2c16f>
        3831   7%         - puthash
        3831   7%          - let*
        3831   7%           - while
        3831   7%            - setq
        3831   7%             - cons
        3831   7%              - funcall
        3831   7%               - #<lambda -0x1002b59e799cf870>
        3831   7%                - let*
        3819   7%                 - funcall
        3255   6%                  - ement-room-list-column-format-name
        3239   6%                   - ement--room-display-name
        3239   6%                    - let*
        3222   6%                     - let
        3218   6%                      - or
        3218   6%                       - funcall
        3206   6%                        - #<lambda 0x121001c773bbe1e2>
        3206   6%                         - let*
        3198   6%                          - if
        3198   6%                           - string-join
        3198   6%                            - delete-dups
        3186   6%                             - mapcar
        3186   6%                              - #<lambda 0x121001dcaa361c72>
        3182   6%                               - ement--user-displayname-in
        3178   6%                                - let*
        3174   6%                                 - if
        3174   6%                                  - let*
        2388   4%                                   - let*
        2384   4%                                    - and
        2384   4%                                     - or
        2380   4%                                      - let*
        2007   4%                                         while
           4   0%                                    + if
         774   1%                                   + function
           4   0%                                   and
           8   0%                             + let
           8   0%                          + and
          12   0%                        + #<lambda 0x120e306f085321fe>
           4   0%                      + setq
          17   0%                     + function
          16   0%                     truncate-string-to-width
         512   1%                  + #<lambda 0x19b1fd14e15fe4>
          44   0%                  + ement-room-list-column-format-notifications
           4   0%                  + ement-room-list-column-format-latest
           4   0%                  + ement-room-list-column-format-topic
           8   0%                 + alist-get
           4   0%                 + let*
        2009   4%    Automatic GC
           4   0%  + magit-section-ident
        1664   3% + command-execute
          64   0% + timer-event-handler
          13   0% + magit-section-post-command-hook
           7   0%   evil-normal-post-command
           4   0% + redisplay_internal (C function)
           4   0%   plz--stderr-sentinel
           1   0% + which-key--hide-popup

@alphapapa here as per your suggestion of re-evaluating everything, and obtaining profiler report for that. I hope that with interpretation the bottleneck as the same place as with compilation. This also isn't obtained from the timer sync, but from tabbing the sections. There the lag also occurs.

Based on this I've tried redefining ement--user-displayname-in as (defun ement--user-displayname-in (room user) "Steve") and the issue seems to be away for now. Of course that means there is couple of rooms with name Steve instead of the real one. :) But I am wondering... maybe this function could cache the displaynames or be optimized? Why is it taking so much time to get a display name?

@Rutherther
Copy link
Author

The defun I presented is obviously not a good alternative to the function, but I am now trying

(defun ement--user-displayname-in (room user)
  (or (ement-user-displayname user)
      (ement-user-id user)))

and although it still not 100% smooth with this and probably that debounce is good to have, it lowers the delay considerably to something much more pleasant for me.

@phil-s
Copy link

phil-s commented Sep 21, 2024

Well that's weird -- in the regular function there's a hash table caching the results of the expensive look-ups, so that should be fast.

Ah -- the hash table is not updated if it uses the fallback case. So that must be what's happening -- every time it tries and fails at the expensive look-up.

@Rutherther
Copy link
Author

(by changed function I mean the one doing logging, then for the second test I used elpaca-build after reverting the commit to build the whole file, and restarted emacs)

@alphapapa
Copy link
Owner

What I mean is, the changes in b20fda0 need to be native-compiled (or at least, you must be careful to load the same kind of code when conducting the before/after tests). The function that's changed by the diff I shared, which times the execution and prints the message, needn't be native-compiled.

@Icy-Thought
Copy link

Icy-Thought commented Sep 22, 2024

Experiencing the same issue but a lot less rooms, so I thought I should evaluate/compile the code in #298 (comment). What more logs do y'all need for me to gather?

Emacs info:

  • NixOS (unstable branch)
  • GNU Emacs 31.0.50 (can switch to latest stable release for testing)
  • Ement version: 49253c0
Ement: Updated room list buffer at 01:05:12 (elapsed: 0.02).
Mark saved where search started [2 times]
Invalid image size (see ‘max-image-size’) [3 times]
Ement: Updated room list buffer at 01:05:16 (elapsed: 1.78).
Mark saved where search started [2 times]
Invalid image size (see ‘max-image-size’) [3 times]
Ement: Updated room list buffer at 01:05:22 (elapsed: 1.84).
Invalid image size (see ‘max-image-size’) [5 times]
Ement: Updated room list buffer at 01:05:23 (elapsed: 1.37).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:05:25 (elapsed: 1.30).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:05:26 (elapsed: 1.35).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:05:29 (elapsed: 1.31).
evil-line-move: End of buffer [2 times]
evil-line-move: End of bufferInvalid image size (see ‘max-image-size’)
Invalid image size (see ‘max-image-size’) [5 times]
Ement: Updated room list buffer at 01:05:36 (elapsed: 1.28).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:05:40 (elapsed: 1.31).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:06:11 (elapsed: 1.32).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:06:12 (elapsed: 1.29).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:06:27 (elapsed: 1.31).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:06:29 (elapsed: 1.29).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:06:30 (elapsed: 1.31).
Invalid image size (see ‘max-image-size’) [6 times]
Ement: Updated room list buffer at 01:06:33 (elapsed: 1.31).

@phil-s
Copy link

phil-s commented Sep 23, 2024

Invalid image size (see ‘max-image-size’) [6 times]

I guess the first question is where is this coming from, and is it a factor?

Edit: Answers are (a) image.c:

static void
image_size_error (void)
{
  image_error ("Invalid image size (see `max-image-size')");
}

and (b): I'm guessing it's not a factor, but I might be wrong -- and it does kinda look like the same aborted processing is happening repeatedly, which could be having a similar effect to the uncached values we saw earlier in this discussion.

You probably want to figure out which images are causing that, and see if you can test in their absence.

@alphapapa
Copy link
Owner

Experiencing the same issue but a lot less rooms, so I thought I should evaluate/compile the code in #298 (comment). What more logs do y'all need for me to gather?

Thanks. I'd suggest disabling ement-auto-sync and then profiling just revert-buffer in the room list buffer.

@alphapapa
Copy link
Owner

and (b): I'm guessing it's not a factor, but I might be wrong -- and it does kinda look like the same aborted processing is happening repeatedly, which could be having a similar effect to the uncached values we saw earlier in this discussion.

I've been seeing this myself sometimes, and I haven't mentioned it because it's so mysterious. My impression is that it started happening with this commit: alphapapa/taxy.el@9e76b7f

Which reminds me: @Icy-Thought What version of taxy-magit-section do you currently have loaded?

Anyway, I added some debugging code to show what value image-size is returning and for what image, and I saw that, when that image size error is shown, it's for a room avatar image (e.g. 2 out of all the ones I'm in), and the image width being computed was 3, which is, according to the relevant code in image.c, a valid size that should not trigger the error. At the same time, an image width of 5 and 2 were not triggering the error. And then, a few minutes later, the error stopped being triggered for the same images that were triggering it before, and AFAIK I hadn't done anything that could make a difference.

To be more specific, the computed image width I'm referring to is from (floor (car (image-size ...))); the actual image size cons-cell values were usually floats with repeating decimal places, but according to the docs and code, they should have been valid. I can't find any reason in the code for them to be triggering an error.

Also, if that commit did trigger the error, I don't understand why...although I suddenly have an idea: all of my visible frames are maximized, but maybe there's an invisible one that's tiny, and maybe it sometimes gets selected by the loop first...? ...Yeah, looks like it might be something like that: even though I only have one visible Emacs frame at the moment, look what this code returns:

(cl-loop for frame in (frame-list)
         when (member (framep frame) '(x w32 ns pgtk))
         collect (list frame (frame-width frame) (frame-height frame)))
;; ((#<frame  0x110250a0> 28 10) (#<frame *scratch* - GNU Emacs 0x277b0b0> 211 52))

Now those numbers still look like they shouldn't cause an error, but if that first frame is some invisible one, who knows what its sizes might be at various times.

So I can probably fix that in taxy-magit-section soon by making sure the frame actually exists as a tangible (but not necessarily visible) one or something like that.

But IME that image-size error wasn't causing a performance problem, so I don't know if it's causing one for @Icy-Thought here. We'll have to wait for the profiler report.

@alphapapa alphapapa reopened this Sep 23, 2024
@alphapapa alphapapa removed this from the v0.16 milestone Sep 23, 2024
@Icy-Thought
Copy link

I don't know how to profile individual functions tbh, so I just attached the whole profiling data to this message.
profile.txt

What version of taxy-magit-section do you currently have loaded?

installed version: 0.14.2 b7b60a4

@phil-s
Copy link

phil-s commented Sep 23, 2024

even though I only have one visible Emacs frame at the moment, look what this code returns

Yep, there's always one frame at minimum, which means that if you run Emacs as as daemon there's always that invisible initial frame behind the scenes.

(This is a notorious reason for people's frame-based tweaks in their init file suddenly 'breaking' when they start using a daemon, as things start getting applied to the invisible frame instead.)


Edit: Oh, no your case is something different? Or, at least, I see t from framep for an initial frame if I run a daemon... but maybe if you run a daemon from a desktop session or something other than a terminal, it winds up not being t? I don't know. I haven't the foggiest what going on there, at any rate.

@alphapapa
Copy link
Owner

even though I only have one visible Emacs frame at the moment, look what this code returns

Yep, there's always one frame at minimum, which means that if you run Emacs as as daemon there's always that invisible initial frame behind the scenes.

(This is a notorious reason for people's frame-based tweaks in their init file suddenly 'breaking' when they start using a daemon, as things start getting applied to the invisible frame instead.)

@phil-s Yep, I think that explains it.

Edit: Oh, no your case is something different!

It is? Why do you say that?

@phil-s
Copy link

phil-s commented Sep 23, 2024

Too quick for me :) I just added this:

Edit: Oh, no your case is something different? Or, at least, I see t from framep for an initial frame if I run a daemon... but maybe if you run a daemon from a desktop session or something other than a terminal, it winds up not being t? I don't know. I haven't the foggiest what going on there, at any rate.

@alphapapa
Copy link
Owner

I don't know how to profile individual functions tbh, so I just attached the whole profiling data to this message. profile.txt

What version of taxy-magit-section do you currently have loaded?

installed version: 0.14.2 b7b60a4

I still see automatic sync responses in that profiler report, so it would seem that ement-auto-sync wasn't disabled.

Also, there are many byte-compiled functions involved. It would help to M-x find-library on taxy-magit-section and ement-room-list and eval-buffer on each of them before running the profiler, that way the functions won't be obscured by byte-compilation.

@alphapapa
Copy link
Owner

Edit: Oh, no your case is something different? Or, at least, I see t from framep for an initial frame if I run a daemon... but maybe if you run a daemon from a desktop session or something other than a terminal, it winds up not being t? I don't know. I haven't the foggiest what going on there, at any rate.

Yeah, now I'm a bit confused too: server-mode is t, and I can run emacsclient -c and get a new frame of the same Emacs process...but I can't find where in my config server-mode is being enabled. Which is quite strange, since I haven't touched that part of my config for a long time... shrug

@alphapapa
Copy link
Owner

alphapapa commented Sep 23, 2024

Anyway, it seems like the key is frame-visible-p:

(cl-loop for frame in (frame-list)
         when (member (framep frame) '(x w32 ns pgtk))
         collect (list frame (frame-visible-p frame) (frame-width frame) (frame-height frame)))
;; ((#<frame  0x18cde470> nil 25 10)
;;  (#<frame *scratch* 0x97bbe98> t 211 52)
;;  (#<frame *Ement Notifications* 0x277b0b0> t 211 52))

So I think the code in taxy-magit-section needs to be:

(cl-loop for frame in (frame-list)
         when (and (frame-visible-p frame)
                   (memq (framep frame)
                         '(x w32 ns pgtk)))
         return frame)

I'm guessing that will fix the image-size error.

I'm not sure if that will have any effect on performance, but we'll see.

alphapapa added a commit to alphapapa/taxy.el that referenced this issue Sep 23, 2024
When server-mode is enabled, an invisible frame is present, which
still returns X from FRAMEP, but which should not be used for
calculating image sizes.

This should fix an "Invalid image size" error that was happening (when
such an invisible frame happened to be sorted first in FRAME-LIST).

See <alphapapa/ement.el#298 (comment)>.
@alphapapa
Copy link
Owner

Just pushed v0.14.3 of taxy-magit-section, which I hope will fix the "invalid image size" error (which might also help with performance, but I don't know).

@Icy-Thought

This comment was marked as resolved.

@alphapapa
Copy link
Owner

The update you pushed to taxy-magit-section seems to have reduced the lag for me to such an extent that it has become unnoticeable?

That's good to hear. If you have time to use the code in #298 (comment) to time the redrawing now, it would be good to know what the hard numbers are.

As for the profiler-report:

Unfortunately, I don't see anything in that report about reverting the room list buffer. What's needed is:

  1. (setq ement-auto-syncs nil).
  2. Wait for any outstanding syncs to finish (30-40 seconds at most).
  3. Switch to the room list buffer.
  4. M-x profiler-start RET cpu RET.
  5. Press g in the room list buffer.
  6. M-x profiler-stop RET.
  7. M-x profiler-report RET.

That way the only thing that's being profiled is the reverting of the room list buffer (plus minor UI like completion, but that can't be avoided).

@Icy-Thought
Copy link

Profiler report: (only relevant bits. If you still want the whole report, please tell me!)

          34   3%  - funcall-interactively
          34   3%   - revert-buffer
          34   3%    - ement-room-list-revert
          34   3%     - ement-room-list
          11   1%      - taxy-fill
          11   1%       - #<byte-code-function 1F2>
          11   1%        - #<lambda AEF>
          11   1%         - taxy-take-keyed
          11   1%          - taxy-take-keyed
          11   1%           - taxy-take-keyed
          11   1%            - taxy-take-keyed
          10   1%             - taxy-take-keyed
          10   1%              - taxy-take-keyed
          10   1%               - taxy-take-keyed

@alphapapa
Copy link
Owner

@Icy-Thought Please post the whole report, otherwise we don't have a full picture of what Emacs is doing.

@Icy-Thought
Copy link

Apologies. I had assumed that you were not interested in the remaining profiling data. I have attached the profiler.txt file as usual within this comment. If you want me to perform more profiles, please ping me! :)
profile.txt

@alphapapa
Copy link
Owner

@Icy-Thought No problem, thanks. From that report it seems that it must be not taking long at all. How many rooms are you in? And how long is it taking to revert the buffer?

@Icy-Thought
Copy link

Glad to provide the logs! :)
I am part of < 30 rooms (28 if I counted right) and manually executing g r was very quick because it did not have to update many rooms.

@alphapapa
Copy link
Owner

Yeah, with that few rooms, updating the room buffer should be imperceptibly fast. Thanks.

AFAIK this is solved now (other than the idea to debounce the automatic reversion, which is tracked elsewhere), so closing now.

@Icy-Thought
Copy link

Thank you so much for taking the time to fix this issue! Have an awesome day! 🎉

@alphapapa
Copy link
Owner

@Icy-Thought Thanks for the kind words, and for your help in debugging it.

@Icy-Thought
Copy link

Anytime! Just ping and I'll try to help out where I can! 😊

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working priority:A
Projects
None yet
Development

No branches or pull requests

4 participants