我正在开发一个应用程序,其中包含多个服务器套接字,每个服务器套接字都在唯一的线程中运行。
外部实用程序(脚本)由线程之一调用。该脚本调用一个实用程序(客户端),该实用程序将消息发送到服务器套接字之一。
最初,我使用的是system()
来执行这个外部脚本,但我们不能使用它,因为我们必须确保在分叉来执行外部脚本的子进程中关闭服务器套接字。
我现在打电话fork()
and execvp()
我。我fork()
然后在孩子中我关闭所有服务器套接字,然后调用execvp()
执行脚本。
现在,所有这些都运行良好。问题是脚本有时会向服务器应用程序报告错误。该脚本通过调用另一个打开 TCP 套接字并发送适当数据的应用程序(客户端)来发送这些错误。我的问题是客户端应用程序的值是0
由返回socket()
系统调用。
注意:仅当使用我的 forkExec() 函数调用脚本/客户端应用程序时才会发生这种情况。如果手动调用脚本/客户端应用程序socket()
呼叫执行正常,一切正常。
根据这些信息,我怀疑这是我下面的 fork() execvp() 代码中的内容......有什么想法吗?
void forkExec()
{
int stat;
stat = fork();
if (stat < 0)
{
printf("Error forking child: %s", strerror(errno));
}
else if (stat == 0)
{
char *progArgs[3];
/*
* First, close the file descriptors that the child
* shouldn't keep open
*/
close(ServerFd);
close(XMLSocket);
close(ClientFd);
close(EventSocket);
close(monitorSocket);
/* build the arguments for script */
progArgs[0] = calloc(1, strlen("/path_to_script")+1);
strcpy(progArgs[0], "/path_to_script");
progArgs[1] = calloc(1, strlen(arg)+1);
strcpy(progArgs[1], arg);
progArgs[2] = NULL; /* Array of args must be NULL terminated for execvp() */
/* launch the script */
stat = execvp(progArgs[0], progArgs);
if (stat != 0)
{
printf("Error executing script: '%s' '%s' : %s", progArgs[0], progArgs[1], strerror(errno));
}
free(progArgs[0]);
free(progArgs[1]);
exit(0);
}
return;
}
客户端应用程序代码:
static int connectToServer(void)
{
int socketFD = 0;
int status;
struct sockaddr_in address;
struct hostent* hostAddr = gethostbyname("localhost");
socketFD = socket(PF_INET, SOCK_STREAM, 0);
上面的调用返回0。
if (socketFD < 0)
{
fprintf(stderr, "%s-%d: Failed to create socket: %s",
__func__, __LINE__, strerror(errno));
return (-1);
}
memset(&address, 0, sizeof(struct sockaddr));
address.sin_family = AF_INET;
memcpy(&(address.sin_addr.s_addr), hostAddr->h_addr, hostAddr->h_length);
address.sin_port = htons(POLLING_SERVER_PORT);
status = connect(socketFD, (struct sockaddr *)&address, sizeof(address));
if (status < 0)
{
if (errno != ECONNREFUSED)
{
fprintf(stderr, "%s-%d: Failed to connect to server socket: %s",
__func__, __LINE__, strerror(errno));
}
else
{
fprintf(stderr, "%s-%d: Server not yet available...%s",
__func__, __LINE__, strerror(errno));
close(socketFD);
socketFD = 0;
}
}
return socketFD;
}
FYI
操作系统:Linux
架构:ARM32
内核:2.6.26