put decode_pid in shared mem, so if player process dies, the master

can still kill the decode process.

git-svn-id: https://svn.musicpd.org/mpd/trunk@107 09075e82-0dd4-0310-85a5-a0d7c8717e4f
This commit is contained in:
Warren Dukes 2004-02-27 22:25:06 +00:00
parent ad4246d6cb
commit 8b19235b61
4 changed files with 43 additions and 18 deletions

View File

@ -49,31 +49,35 @@
#define FADE_CHUNKS 1024 #define FADE_CHUNKS 1024
int decode_pid = 0; int * decode_pid = NULL;
void decodeSigHandler(int sig) { void decodeSigHandler(int sig) {
if(sig==SIGCHLD) { if(sig==SIGCHLD) {
int status; int status;
if(decode_pid==wait3(&status,WNOHANG,NULL)) { if(decode_pid && *decode_pid==wait3(&status,WNOHANG,NULL)) {
if(WIFSIGNALED(status) && WTERMSIG(status)!=SIGTERM) { if(WIFSIGNALED(status) && WTERMSIG(status)!=SIGTERM) {
ERROR("decode process died from a " ERROR("decode process died from a "
"non-TERM signal: %i\n", "non-TERM signal: %i\n",
WTERMSIG(status)); WTERMSIG(status));
} }
decode_pid = 0; *decode_pid = 0;
} }
} }
else if(sig==SIGTERM) { else if(sig==SIGTERM) {
int pid = decode_pid; if(decode_pid) {
if(pid>0) kill(pid,SIGTERM); int pid = *decode_pid;
if(pid>0) kill(pid,SIGTERM);
}
exit(0); exit(0);
} }
} }
void stopDecode(DecoderControl * dc) { void stopDecode(DecoderControl * dc) {
if(decode_pid>0 && (dc->start || dc->state==DECODE_STATE_DECODE)) { if(decode_pid && *decode_pid>0 &&
(dc->start || dc->state==DECODE_STATE_DECODE))
{
dc->stop = 1; dc->stop = 1;
while(decode_pid>0 && dc->stop) usleep(10); while(decode_pid && *decode_pid>0 && dc->stop) usleep(10);
} }
} }
@ -106,7 +110,7 @@ int calculateCrossFadeChunks(PlayerControl * pc, AudioFormat * af) {
int waitOnDecode(PlayerControl * pc, AudioFormat * af, DecoderControl * dc, int waitOnDecode(PlayerControl * pc, AudioFormat * af, DecoderControl * dc,
Buffer * cb) Buffer * cb)
{ {
while(decode_pid>0 && dc->start) usleep(10); while(decode_pid && *decode_pid>0 && dc->start) usleep(10);
if(dc->start || dc->error!=DECODE_ERROR_NOERROR) { if(dc->start || dc->error!=DECODE_ERROR_NOERROR) {
strncpy(pc->erroredFile,pc->file,MAXPATHLEN); strncpy(pc->erroredFile,pc->file,MAXPATHLEN);
@ -135,7 +139,7 @@ int waitOnDecode(PlayerControl * pc, AudioFormat * af, DecoderControl * dc,
void decodeSeek(PlayerControl * pc, AudioFormat * af, DecoderControl * dc, void decodeSeek(PlayerControl * pc, AudioFormat * af, DecoderControl * dc,
Buffer * cb) Buffer * cb)
{ {
if(decode_pid>0) { if(decode_pid && *decode_pid>0) {
cb->next = -1; cb->next = -1;
if(dc->state!=DECODE_STATE_DECODE || dc->error || if(dc->state!=DECODE_STATE_DECODE || dc->error ||
strcmp(dc->file,pc->file)!=0) strcmp(dc->file,pc->file)!=0)
@ -147,7 +151,7 @@ void decodeSeek(PlayerControl * pc, AudioFormat * af, DecoderControl * dc,
dc->error = 0; dc->error = 0;
waitOnDecode(pc,af,dc,cb); waitOnDecode(pc,af,dc,cb);
} }
if(decode_pid>0 && dc->state==DECODE_STATE_DECODE) { if(*decode_pid>0 && dc->state==DECODE_STATE_DECODE) {
dc->seekWhere = pc->seekWhere > pc->totalTime-1 ? dc->seekWhere = pc->seekWhere > pc->totalTime-1 ?
pc->totalTime-1 : pc->seekWhere; pc->totalTime-1 : pc->seekWhere;
dc->seekWhere = 1 > dc->seekWhere ? 1 : dc->seekWhere; dc->seekWhere = 1 > dc->seekWhere ? 1 : dc->seekWhere;
@ -155,7 +159,7 @@ void decodeSeek(PlayerControl * pc, AudioFormat * af, DecoderControl * dc,
dc->seek = 1; dc->seek = 1;
pc->elapsedTime = dc->seekWhere; pc->elapsedTime = dc->seekWhere;
pc->bitRate = 0; pc->bitRate = 0;
while(decode_pid>0 && dc->seek) usleep(10); while(*decode_pid>0 && dc->seek) usleep(10);
} }
} }
pc->seek = 0; pc->seek = 0;
@ -193,9 +197,10 @@ void decodeSeek(PlayerControl * pc, AudioFormat * af, DecoderControl * dc,
int decoderInit(PlayerControl * pc, Buffer * cb, AudioFormat *af, int decoderInit(PlayerControl * pc, Buffer * cb, AudioFormat *af,
DecoderControl * dc) { DecoderControl * dc) {
decode_pid = fork(); decode_pid = &(pc->decode_pid);
*decode_pid = fork();
if(decode_pid==0) { if(*decode_pid==0) {
/* CHILD */ /* CHILD */
while(1) { while(1) {
@ -242,7 +247,7 @@ int decoderInit(PlayerControl * pc, Buffer * cb, AudioFormat *af,
exit(0); exit(0);
/* END OF CHILD */ /* END OF CHILD */
} }
else if(decode_pid<0) { else if(*decode_pid<0) {
strncpy(pc->erroredFile,pc->file,MAXPATHLEN); strncpy(pc->erroredFile,pc->file,MAXPATHLEN);
pc->error = PLAYER_ERROR_SYSTEM; pc->error = PLAYER_ERROR_SYSTEM;
return -1; return -1;
@ -274,7 +279,7 @@ void decode() {
dc->start = 1; dc->start = 1;
cb->next = -1; cb->next = -1;
if(decode_pid<=0) { if(decode_pid==NULL || *decode_pid<=0) {
if(decoderInit(pc,cb,af,dc)<0) return; if(decoderInit(pc,cb,af,dc)<0) return;
} }
@ -299,7 +304,7 @@ void decode() {
pc->beginTime = pc->elapsedTime; pc->beginTime = pc->elapsedTime;
kill(getppid(),SIGUSR1); kill(getppid(),SIGUSR1);
while(decode_pid>0 && !cb->wrap && cb->end-cb->begin<bbp && while(*decode_pid>0 && !cb->wrap && cb->end-cb->begin<bbp &&
dc->state==DECODE_STATE_DECODE) dc->state==DECODE_STATE_DECODE)
{ {
processDecodeInput(); processDecodeInput();
@ -443,7 +448,7 @@ void decode() {
} }
pc->beginTime = cb->times[cb->begin]; pc->beginTime = cb->times[cb->begin];
} }
else if(decode_pid<=0 || else if(*decode_pid<=0 ||
(dc->state==DECODE_STATE_STOP && !dc->start)) (dc->state==DECODE_STATE_STOP && !dc->start))
{ {
quit = 1; quit = 1;

View File

@ -48,6 +48,8 @@ int player_pid = 0;
int player_termSent = 0; int player_termSent = 0;
void resetPlayer() { void resetPlayer() {
int pid;
player_pid = 0; player_pid = 0;
player_termSent = 0; player_termSent = 0;
getPlayerData()->playerControl.stop = 0; getPlayerData()->playerControl.stop = 0;
@ -58,12 +60,17 @@ void resetPlayer() {
getPlayerData()->playerControl.state = PLAYER_STATE_STOP; getPlayerData()->playerControl.state = PLAYER_STATE_STOP;
getPlayerData()->playerControl.queueState = PLAYER_QUEUE_UNLOCKED; getPlayerData()->playerControl.queueState = PLAYER_QUEUE_UNLOCKED;
getPlayerData()->playerControl.seek = 0; getPlayerData()->playerControl.seek = 0;
/* kill decode process if it got left running */
pid = getPlayerData()->playerControl.decode_pid;
if(pid>0) kill(pid,SIGTERM);
getPlayerData()->playerControl.decode_pid = 0;
} }
void player_sigHandler(int signal) { void player_sigHandler(int signal) {
if(signal==SIGCHLD) { if(signal==SIGCHLD) {
int status; int status;
if(player_pid==wait3(&status,WNOHANG,NULL)) { int pid = wait3(&status,WNOHANG,NULL);
if(player_pid==pid) {
if(WIFSIGNALED(status) && WTERMSIG(status)!=SIGTERM) { if(WIFSIGNALED(status) && WTERMSIG(status)!=SIGTERM) {
ERROR("player process died from a " ERROR("player process died from a "
"non-TERM signal: %i\n", "non-TERM signal: %i\n",
@ -71,6 +78,17 @@ void player_sigHandler(int signal) {
} }
resetPlayer(); resetPlayer();
} }
else if(pid==getPlayerData()->playerControl.decode_pid &&
player_pid<=0)
{
if(WIFSIGNALED(status) && WTERMSIG(status)!=SIGTERM) {
ERROR("(caught by master parent) "
"decode process died from a "
"non-TERM signal: %i\n",
WTERMSIG(status));
}
getPlayerData()->playerControl.decode_pid = 0;
}
} }
} }

View File

@ -74,6 +74,7 @@ typedef struct _PlayerControl {
float crossFade; float crossFade;
mpd_sint8 softwareVolume; mpd_sint8 softwareVolume;
double totalPlayTime; double totalPlayTime;
int decode_pid;
} PlayerControl; } PlayerControl;
void player_sigHandler(int signal); void player_sigHandler(int signal);

View File

@ -107,6 +107,7 @@ void initPlayerData() {
playerData_pd->playerControl.crossFade = crossfade; playerData_pd->playerControl.crossFade = crossfade;
playerData_pd->playerControl.softwareVolume = 100; playerData_pd->playerControl.softwareVolume = 100;
playerData_pd->playerControl.totalPlayTime = 0; playerData_pd->playerControl.totalPlayTime = 0;
playerData_pd->playerControl.decode_pid = 0;
playerData_pd->decoderControl.stop = 0; playerData_pd->decoderControl.stop = 0;
playerData_pd->decoderControl.start = 0; playerData_pd->decoderControl.start = 0;