From: Tom Tromey <tom@tromey.com>
To: emacs-devel@gnu.org
Subject: Re: call-process blocks the active thread
Date: Fri, 08 Sep 2017 10:12:17 -0600 [thread overview]
Message-ID: <87lglpdva6.fsf@bapiya> (raw)
In-Reply-To: <83k219xu58.fsf@gnu.org> (Eli Zaretskii's message of "Fri, 08 Sep 2017 15:16:35 +0300")
Eli> Of course, it's possible I don't see some clever way to make
Eli> call-process thread-aware, which is why I said that if someone has
Eli> concrete ideas for how to do that, they are encouraged to speak up.
I looked at it briefly. I have two ideas.
Maybe the simplest idea is to rewrite call-process in Lisp, using
make-process and accept-process-output. This would automatically allow
other threads to run. I didn't really investigate this in depth (so for
instance I don't know if this can be done without adding extensions to
make-process), but it seems to me that it would be nice to migrate code
out of C when possible.
Otherwise, I think call_process can be made to yield. A bit of
background (I know Eli doesn't need this but maybe others do):
Only one thread can run Lisp at a time. A global lock enforces this.
However, threads can run non-Lisp code while not holding the global lock
-- this is why Emacs can wait for input while Lisp runs in another
thread.
The way this is done is that a thread calls flush_stack_call_func to
save registers on the stack (so that GC will do the right thing); this
calls a provided callback, and the callback handles releasing the global
lock, doing whatever non-Lisp thing it wants to do, and then
re-acquiring the lock before returning.
A good example of this is thread.c:thread_select, which winds up in
thread.c:really_call_select.
So, the idea for call-process would be to modify the inner loop of
call_process to do this. I think the best spot would be to call
flush_stack_call_func around the call to emacs_read_quit.
However, the tricky thing here is that call-process relies on some
global state. For example, it assumes that current-buffer will not
change while it is working. So, after emacs_read_quit returns, this
code would have to be careful to restore the state it needs. The hard
part of the project is figuring out which bits of state must be
preserved.
It might also be reasonable to wrap the wait_for_termination call in a
lock-releasing function.
Tom
next prev parent reply other threads:[~2017-09-08 16:12 UTC|newest]
Thread overview: 10+ messages / expand[flat|nested] mbox.gz Atom feed top
2017-09-08 3:00 call-process blocks the active thread Elias Mårtenson
2017-09-08 3:56 ` Eric Abrahamsen
2017-09-08 7:01 ` Eli Zaretskii
2017-09-08 7:45 ` Stefan Monnier
2017-09-08 8:06 ` Eli Zaretskii
2017-09-08 10:36 ` Elias Mårtenson
2017-09-08 11:27 ` Philipp Stephani
2017-09-08 12:16 ` Eli Zaretskii
2017-09-08 16:12 ` Tom Tromey [this message]
2017-09-08 18:43 ` Tom Tromey
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=87lglpdva6.fsf@bapiya \
--to=tom@tromey.com \
--cc=emacs-devel@gnu.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
Code repositories for project(s) associated with this external index
https://git.savannah.gnu.org/cgit/emacs.git
https://git.savannah.gnu.org/cgit/emacs/org-mode.git
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.