static int s_dump_debug_info = 0;
static int s_pid_dump_debug_info = 0;
static int s_req_processed = 0;
+static int s_skip_write = 0;
+static int (*pthread_atfork_func)(void (*prepare)(void), void (*parent)(void),
+ void (*child)(void)) = NULL;
static int *s_busy_workers = NULL;
static int *s_accepting_workers = NULL;
static int *s_global_counter = &s_req_processed;
static int s_max_busy_workers = -1;
static char *s_stderr_log_path = NULL;
+static int s_stderr_is_pipe = 0;
static int s_ignore_pid = -1;
LSAPI_Request g_req =
{
char buf[1024];
char *p = buf;
- if (flag & LSAPI_LOG_TIMESTAMP_BITS)
+ if ((flag & LSAPI_LOG_TIMESTAMP_BITS) &&
+ !((flag & LSAPI_LOG_TIMESTAMP_STDERR) && s_stderr_is_pipe))
{
struct timeval tv;
struct tm tm;
#endif
+#define lsapi_log(...) LSAPI_Log(LSAPI_LOG_TIMESTAMP_FULL|LSAPI_LOG_TIMESTAMP_STDERR|LSAPI_LOG_PID, __VA_ARGS__)
+
static int lsapi_parent_dead()
{
int ret;
int left = totalLen;
int n = count;
+
+ if (s_skip_write)
+ return totalLen;
+
while(( left > 0 )&&g_running )
{
ret = writev( fd, *pVec, n );
int fd = open( pSecretFile, O_RDONLY , 0600 );
if ( fd == -1 )
{
- fprintf( stderr, "LSAPI: failed to open secret file: %s!\n", pSecretFile );
+ lsapi_log("LSAPI: failed to open secret file: %s!\n", pSecretFile );
return -1;
}
if ( fstat( fd, &st ) == -1 )
{
- fprintf( stderr, "LSAPI: failed to check state of file: %s!\n", pSecretFile );
+ lsapi_log("LSAPI: failed to check state of file: %s!\n", pSecretFile );
close( fd );
return -1;
}
/*
if ( st.st_uid != s_uid )
{
- fprintf( stderr, "LSAPI: file owner check failure: %s!\n", pSecretFile );
+ lsapi_log("LSAPI: file owner check failure: %s!\n", pSecretFile );
close( fd );
return -1;
}
*/
if ( st.st_mode & 0077 )
{
- fprintf( stderr, "LSAPI: file permission check failure: %s\n", pSecretFile );
+ lsapi_log("LSAPI: file permission check failure: %s\n", pSecretFile );
close( fd );
return -1;
}
if ( read( fd, s_secret, 16 ) < 16 )
{
- fprintf( stderr, "LSAPI: failed to read secret from secret file: %s\n", pSecretFile );
+ lsapi_log("LSAPI: failed to read secret from secret file: %s\n", pSecretFile );
close( fd );
return -1;
}
ret = (*fp_lve_enter)(s_lve, uid, -1, -1, &cookie);
if ( ret < 0 )
{
- fprintf( stderr, "Pid (%d): enter LVE (%d) : ressult: %d !\n", getpid(), uid, ret );
+ lsapi_log("enter LVE (%d) : ressult: %d !\n", uid, ret );
LSAPI_perror_r(pReq, "LSAPI: lve_enter() failure, reached resource limit.", NULL );
lsapi_lve_error( pReq );
return -1;
ret = (*fp_lve_jail)( pw, error_msg );
if ( ret < 0 )
{
- fprintf( stderr, "LSAPI (%d): LVE jail(%d) ressult: %d, error: %s !\n",
- getpid(), uid, ret, error_msg );
+ lsapi_log("LSAPI: LVE jail(%d) ressult: %d, error: %s !\n",
+ uid, ret, error_msg );
LSAPI_perror_r( pReq, "LSAPI: jail() failure.", NULL );
return -1;
}
--pReq->m_pHeader->m_cntSpecialEnv;
uid = *(uint32_t *)pEnv->pValue;
gid = *(((uint32_t *)pEnv->pValue) + 1 );
- //fprintf( stderr, "LSAPI: SUEXEC_UGID set UID: %d, GID: %d\n", uid, gid );
+ //lsapi_log("LSAPI: SUEXEC_UGID set UID: %d, GID: %d\n", uid, gid );
}
else
{
- fprintf( stderr, "LSAPI: missing SUEXEC_UGID env, use default user!\n" );
+ lsapi_log("LSAPI: missing SUEXEC_UGID env, use default user!\n" );
pEnv = NULL;
}
if ( pEnv&& lsapi_suexec_auth( pReq, pAuth->pValue, pAuth->valLen, pEnv->pValue, pEnv->valLen ) == 0 )
else
{
//authentication error
- fprintf( stderr, "LSAPI: SUEXEC_AUTH authentication failed, use default user!\n" );
+ lsapi_log("LSAPI: SUEXEC_AUTH authentication failed, use default user!\n" );
uid = 0;
}
}
else
{
- //fprintf( stderr, "LSAPI: no SUEXEC_AUTH env, use default user!\n" );
+ //lsapi_log("LSAPI: no SUEXEC_AUTH env, use default user!\n" );
}
}
|| pReq->m_pHeader->m_requestMethodOff < 0
|| pReq->m_pHeader->m_requestMethodOff >= totalLen)
{
- fprintf(stderr, "%d: bad request header - ERROR#1\n", getpid());
+ lsapi_log("Bad request header - ERROR#1\n");
return -1;
}
pReq->m_pScriptFile = pReq->m_pReqBuf + pReq->m_pHeader->m_scriptFileOff;
pBegin += pReq->m_pHeader->m_httpHeaderLen;
if ( pBegin != pEnd )
{
- fprintf( stderr, "%d: request header does match total size, total: %d, "
- "real: %ld\n", getpid(), totalLen, pBegin - pReq->m_pReqBuf );
+ lsapi_log("Request header does match total size, total: %d, "
+ "real: %ld\n", totalLen, pBegin - pReq->m_pReqBuf );
return -1;
}
if ( shouldFixEndian )
if (validateHeaders(pReq) == -1)
{
- fprintf(stderr, "%d: bad request header - ERROR#2\n", getpid());
+ lsapi_log("Bad request header - ERROR#2\n");
return -1;
}
packetLen = verifyHeader( &pReq->m_pHeader->m_pktHeader, LSAPI_BEGIN_REQUEST );
if ( packetLen < 0 )
{
- fprintf( stderr, "%d: packetLen < 0\n", getpid() );
+ lsapi_log("packetLen < 0\n");
return -1;
}
if ( packetLen > LSAPI_MAX_HEADER_LEN )
{
- fprintf( stderr, "%d: packetLen > %d\n", getpid(), LSAPI_MAX_HEADER_LEN );
+ lsapi_log("packetLen > %d\n", LSAPI_MAX_HEADER_LEN );
return -1;
}
}
if ( parseRequest( pReq, packetLen ) < 0 )
{
- fprintf( stderr, "%d: parseRequest error\n", getpid() );
+ lsapi_log("ParseRequest error\n");
return -1;
}
return -1;
g_inited = 1;
s_ppid = getppid();
+ void *pthread_lib = dlopen("libpthread.so", RTLD_LAZY);
+ if (pthread_lib)
+ pthread_atfork_func = dlsym(pthread_lib, "pthread_atfork");
+
}
return 0;
}
{
int sig_num = WTERMSIG( status );
int dump = WCOREDUMP( status );
- fprintf( stderr, "Child process with pid: %d was killed by signal: "
+ lsapi_log("Child process with pid: %d was killed by signal: "
"%d, core dump: %d\n", pid, sig_num, dump );
}
if ( pid == s_pid_dump_debug_info )
if ( kill( s_pid_dump_debug_info, 0 ) == 0 )
return;
}
- s_pid_dump_debug_info = fork();
- fprintf( stderr, "[%s] Possible runaway process, PPID: %d, PID: %d, "
+ lsapi_log("Possible runaway process, PPID: %d, PID: %d, "
"reqCount: %d, process time: %ld, checkpoint time: %ld, start "
- "time: %ld\n", ctime(&tmCur), getpid(), pStatus->m_pid,
+ "time: %ld\n", getpid(), pStatus->m_pid,
pStatus->m_iReqCounter, tmCur - pStatus->m_tmReqBegin,
tmCur - pStatus->m_tmLastCheckPoint, tmCur - pStatus->m_tmStart );
- snprintf( achCmd, 1024, "gdb --batch -ex \"attach %d\" -ex \"set height 0\" "
- "-ex \"bt\" >&2;PATH=$PATH:/usr/sbin lsof -p %d >&2",
- pStatus->m_pid, pStatus->m_pid );
- if ( system( achCmd ) == -1 )
- perror( "system()" );
- exit( 0 );
+
+ s_pid_dump_debug_info = fork();
+ if (s_pid_dump_debug_info == 0)
+ {
+ snprintf( achCmd, 1024, "gdb --batch -ex \"attach %d\" -ex \"set height 0\" "
+ "-ex \"bt\" >&2;PATH=$PATH:/usr/sbin lsof -p %d >&2",
+ pStatus->m_pid, pStatus->m_pid );
+ if ( system( achCmd ) == -1 )
+ perror( "system()" );
+ exit( 0 );
+ }
}
if ( pStatus->m_iKillSent > 5 )
{
tobekilled = SIGKILL;
- fprintf( stderr, "Force killing runaway process PID: %d"
+ lsapi_log("Force killing runaway process PID: %d"
" with SIGKILL\n", pStatus->m_pid );
}
else
{
tobekilled = SIGTERM;
- fprintf( stderr, "Killing runaway process PID: %d with "
+ lsapi_log("Killing runaway process PID: %d with "
"SIGTERM\n", pStatus->m_pid );
}
}
}
if ( abs( g_prefork_server->m_iCurChildren - count ) > 1 )
{
- fprintf( stderr, "Children tracking is wrong: PID: %d, Cur Children: %d,"
- " count: %d, idle: %d, dying: %d\n", getpid(),
- g_prefork_server->m_iCurChildren, count, idle, dying );
+ lsapi_log("Children tracking is wrong: Cur Children: %d,"
+ " count: %d, idle: %d, dying: %d\n",
+ g_prefork_server->m_iCurChildren, count, idle, dying );
}
}
//}
+void set_skip_write()
+{ s_skip_write = 1; }
+
+
static int lsapi_prefork_server_accept( lsapi_prefork_server * pServer,
LSAPI_Request * pReq )
{
act.sa_flags = 0;
act.sa_handler = lsapi_sigchild;
+ sigemptyset(&(act.sa_mask));
if( sigaction( SIGCHLD, &act, &old_child ) )
{
perror( "Can't set signal handler for SIGCHILD" );
/* Set up handler to kill children upon exit */
act.sa_flags = 0;
act.sa_handler = lsapi_cleanup;
+ sigemptyset(&(act.sa_mask));
if( sigaction( SIGTERM, &act, &old_term ) ||
sigaction( SIGINT, &act, &old_int ) ||
sigaction( SIGUSR1, &act, &old_usr1 ) ||
if (pServer->m_iCurChildren >=
pServer->m_iMaxChildren + pServer->m_iExtraChildren)
{
- fprintf( stderr, "Reached max children process limit: %d, extra: %d,"
+ lsapi_log("Reached max children process limit: %d, extra: %d,"
" current: %d, busy: %d, please increase LSAPI_CHILDREN.\n",
pServer->m_iMaxChildren, pServer->m_iExtraChildren,
pServer->m_iCurChildren,
pReq->m_fd = lsapi_accept( pServer->m_fd );
if ( pReq->m_fd != -1 )
{
+ wait_secs = 0;
child_status = find_child_status( 0 );
if ( child_status )
memset( child_status, 0, sizeof( *child_status ) );
s_proc_group_timer_cb = NULL;
s_worker_status = child_status;
+ if (pthread_atfork_func)
+ (*pthread_atfork_func)(NULL, NULL, set_skip_write);
+
s_worker_status->m_connected = 1;
if (s_busy_workers)
__sync_add_and_fetch(s_busy_workers, 1);
}
-void lsapi_error( const char * pMessage, int err_no )
+void lsapi_perror( const char * pMessage, int err_no )
{
- fprintf( stderr, "%d: %s, errno: %d (%s)\n", getpid(), pMessage, err_no,
- strerror( err_no ) );
+ lsapi_log("%s, errno: %d (%s)\n", pMessage, err_no,
+ strerror( err_no ) );
}
fd_set readfds;
struct timeval timeout;
+ if (s_skip_write)
+ return -1;
+
LSAPI_Finish_r( pReq );
if ( g_prefork_server )
{
if (( errno == EINTR )||( errno == EAGAIN))
continue;
- lsapi_error( "lsapi_accept() error", errno );
+ lsapi_perror( "lsapi_accept() error", errno );
return -1;
}
}
int newfd = open(full_path, O_WRONLY | O_CREAT | O_APPEND, 0644);
if (newfd == -1)
{
- LSAPI_perror_r(NULL, "failed to open custom stderr log", full_path);
+ LSAPI_perror_r(NULL, "Failed to open custom stderr log", full_path);
return -1;
}
if (newfd != 2)
return -1;
if (lsapi_check_path(p, full_path, PATH_MAX) == -1)
{
- LSAPI_perror_r(NULL, "invalid custom stderr log path", p);
+ LSAPI_perror_r(NULL, "Invalid custom stderr log path", p);
return -1;
}
return lsapi_reopen_stderr2(full_path);
{
lsapi_reopen_stderr(p);
}
+ if (!s_stderr_log_path)
+ s_stderr_is_pipe = isPipe(STDERR_FILENO);
p = getenv( "PHP_LSAPI_MAX_REQUESTS" );
if ( !p )