diff options
author | wm4 <wm4@nowhere> | 2014-10-19 01:42:28 +0200 |
---|---|---|
committer | wm4 <wm4@nowhere> | 2014-10-19 05:51:37 +0200 |
commit | 987146362e35820cd96d7de4f48b13ec4ee25d7c (patch) | |
tree | 8b8aa214ae108c252e15e603f6f1be47fb694e51 /stream | |
parent | 0328fa225245f0e27278f944ba6651c16e90b8bd (diff) | |
download | mpv-987146362e35820cd96d7de4f48b13ec4ee25d7c.tar.bz2 mpv-987146362e35820cd96d7de4f48b13ec4ee25d7c.tar.xz |
lua: add an utility function for starting processes
Because 1) Lua is terrible, and 2) popen() is terrible. Unfortunately,
since Unix is also terrible, this turned out more complicated than I
hoped. As a consequence and to avoid that this code has to be maintained
forever, add a disclaimer that any function in Lua's utils module can
disappear any time. The complexity seems a bit ridiculous, especially
for a feature so far removed from actual video playback, so if it turns
out that we don't really need this function, it will be dropped again.
The motivation for this commit is the same as with 8e4fa5fc.
Note that there is an "#ifndef __GLIBC__". The GNU people are very
special people and thought it'd be convenient to actually declare
"environ", even though the POSIX people, which are also very special
people, state that no header declares this and that the user has to
declare this manually. Since the GNU people overtook the Unix world with
their very clever "embrace, extend, extinguish" strategy, but not 100%,
and trying to build without _GNU_SOURCE is hopeless; but since there
might be Unix environments which support _GNU_SOURCE features partially,
this means that in practice "environ" will be randomly declared or not
declared by system headers. Also, gcc was written by very clever people
too, and prints a warning if an external variable is declared twice (I
didn't check, but I suppose redeclaring is legal C, and not even the gcc
people are clever enough to only warn against a definitely not legal C
construct, although sometimes they do this), ...and since we at mpv hate
compiler warnings, we seek to silence them all. Adding a configure test
just for a warning seems too radical, so we special-case this against
__GLIBC__, which is hopefully not defined on other libcs, especially not
libcs which don't implement all aspects of _GNU_SOURCE, and redefine
"environ" on systems even if the headers define it already (because they
support _GNU_SOURCE - as I mentioned before, the clever GNU people wrote
software THAT portable that other libcs just gave up and implemented
parts of _GNU_SOURCE, although probably not all), which means that
compiling mpv will print a warning about "environ" being redefined, but
at least this won't happen on my system, so all is fine. However, should
someone complain about this warning, I will force whoever complained
about this warning to read this ENTIRE commit message, and if possible,
will also force them to eat a printed-out copy of the GNU Manifesto, and
if that is not enough, maybe this person could even be forced to
convince the very clever POSIX people of not doing crap like this:
having the user to manually declare somewhat central symbols - but I
doubt it's possible, because the POSIX people are too far gone and only
care about maintaining compatibility with old versions of AIX and HP-UX.
Oh, also, this code contains some subtle and obvious issues, but writing
about this is not fun.
Diffstat (limited to 'stream')
-rw-r--r-- | stream/stream.c | 36 | ||||
-rw-r--r-- | stream/stream.h | 1 |
2 files changed, 30 insertions, 7 deletions
diff --git a/stream/stream.c b/stream/stream.c index fe80028091..8a3d981809 100644 --- a/stream/stream.c +++ b/stream/stream.c @@ -18,20 +18,16 @@ #include <stdio.h> #include <stdlib.h> +#include <sys/types.h> #include <unistd.h> +#include <errno.h> -#include <sys/types.h> -#include <sys/stat.h> -#ifndef __MINGW32__ -#include <sys/ioctl.h> -#include <sys/wait.h> -#endif -#include <fcntl.h> #include <strings.h> #include <assert.h> #include <libavutil/common.h> #include "osdep/atomics.h" +#include "osdep/io.h" #include "talloc.h" @@ -988,12 +984,22 @@ struct bstr stream_read_complete(struct stream *s, void *talloc_ctx, struct mp_cancel { atomic_bool triggered; + int wakeup_pipe[2]; }; +static void cancel_destroy(void *p) +{ + struct mp_cancel *c = p; + close(c->wakeup_pipe[0]); + close(c->wakeup_pipe[1]); +} + struct mp_cancel *mp_cancel_new(void *talloc_ctx) { struct mp_cancel *c = talloc_ptrtype(talloc_ctx, c); + talloc_set_destructor(c, cancel_destroy); *c = (struct mp_cancel){.triggered = ATOMIC_VAR_INIT(false)}; + mp_make_wakeup_pipe(c->wakeup_pipe); return c; } @@ -1001,12 +1007,21 @@ struct mp_cancel *mp_cancel_new(void *talloc_ctx) void mp_cancel_trigger(struct mp_cancel *c) { atomic_store(&c->triggered, true); + write(c->wakeup_pipe[1], &(char){0}, 1); } // Restore original state. (Allows reusing a mp_cancel.) void mp_cancel_reset(struct mp_cancel *c) { atomic_store(&c->triggered, false); + // Flush it fully. + while (1) { + int r = read(c->wakeup_pipe[0], &(char[256]){0}, 256); + if (r < 0 && errno == EINTR) + continue; + if (r <= 0) + break; + } } // Return whether the caller should abort. @@ -1016,6 +1031,13 @@ bool mp_cancel_test(struct mp_cancel *c) return c ? atomic_load(&c->triggered) : false; } +// The FD becomes readable if mp_cancel_test() would return true. +// Don't actually read from it, just use it for poll(). +int mp_cancel_get_fd(struct mp_cancel *c) +{ + return c->wakeup_pipe[0]; +} + void stream_print_proto_list(struct mp_log *log) { int count = 0; diff --git a/stream/stream.h b/stream/stream.h index 56c431ed46..e7a0bb5dca 100644 --- a/stream/stream.h +++ b/stream/stream.h @@ -263,6 +263,7 @@ struct mp_cancel *mp_cancel_new(void *talloc_ctx); void mp_cancel_trigger(struct mp_cancel *c); bool mp_cancel_test(struct mp_cancel *c); void mp_cancel_reset(struct mp_cancel *c); +int mp_cancel_get_fd(struct mp_cancel *c); // stream_file.c char *mp_file_url_to_filename(void *talloc_ctx, bstr url); |