[pipe] Fix crash if pipe is deleted after being written to (caused by double free)

This commit is contained in:
ejurgensen 2017-07-08 01:41:12 +02:00
parent a23bb0b7e2
commit 1549d0d4b5

View File

@ -494,15 +494,17 @@ static enum command_state
pipe_watch_update(void *arg, int *retval) pipe_watch_update(void *arg, int *retval)
{ {
union pipe_arg *cmdarg = arg; union pipe_arg *cmdarg = arg;
struct pipe *pipelist = NULL; struct pipe *pipelist;
struct pipe *pipe; struct pipe *pipe;
struct pipe *next; struct pipe *next;
int count; int count;
if (cmdarg) if (cmdarg)
pipelist = cmdarg->pipelist; pipelist = cmdarg->pipelist;
else
pipelist = NULL;
count = 0; // Removes pipes that are gone from the watchlist
for (pipe = pipe_watch_list; pipe; pipe = next) for (pipe = pipe_watch_list; pipe; pipe = next)
{ {
next = pipe->next; next = pipe->next;
@ -512,29 +514,33 @@ pipe_watch_update(void *arg, int *retval)
DPRINTF(E_DBG, L_PLAYER, "Pipe watch deleted: '%s'\n", pipe->path); DPRINTF(E_DBG, L_PLAYER, "Pipe watch deleted: '%s'\n", pipe->path);
watch_del(pipe); watch_del(pipe);
pipelist_remove(&pipe_watch_list, pipe); // Will free pipe pipelist_remove(&pipe_watch_list, pipe); // Will free pipe
continue; }
} }
count++; // Looks for new pipes and adds them to the watchlist
pipelist_remove(&pipelist, pipe); for (pipe = pipelist, count = 0; pipe; pipe = next, count++)
}
for (pipe = pipelist; pipe; pipe = next)
{ {
next = pipe->next; next = pipe->next;
count++;
if (count > PIPE_MAX_WATCH) if (count > PIPE_MAX_WATCH)
{ {
DPRINTF(E_LOG, L_PLAYER, "Max open pipes reached (%d), will not watch '%s'\n", PIPE_MAX_WATCH, pipe->path); DPRINTF(E_LOG, L_PLAYER, "Max open pipes reached (%d), will not watch '%s'\n", PIPE_MAX_WATCH, pipe->path);
pipelist_remove(&pipelist, pipe); pipe_free(pipe);
continue; continue;
} }
if (!pipelist_find(pipe_watch_list, pipe->id))
{
DPRINTF(E_DBG, L_PLAYER, "Pipe watch added: '%s'\n", pipe->path); DPRINTF(E_DBG, L_PLAYER, "Pipe watch added: '%s'\n", pipe->path);
watch_add(pipe); watch_add(pipe);
pipelist_add(&pipe_watch_list, pipe); // Changes pipe->next pipelist_add(&pipe_watch_list, pipe); // Changes pipe->next
} }
else
{
DPRINTF(E_DBG, L_PLAYER, "Pipe watch exists: '%s'\n", pipe->path);
pipe_free(pipe);
}
}
*retval = 0; *retval = 0;
return COMMAND_END; return COMMAND_END;