繁体   English   中英

execl在while(1)循环中在服务器端不起作用; C脚本

[英]execl doesn't work in a while(1) loop, server side; C script

我有一个小C脚本有问题,该C脚本应作为服务器运行,并为到达的每条消息启动一个弹出窗口。 execl语法是正确的,因为如果我尝试使用

main() { execl(...); }

有用。

当我将它放在while(1)循环中时,它不起作用。 其他所有功能都可以正常工作,例如printf或字符串操作,但execl却不能。 即使我叉它也不起作用。 我该如何运作?

我已经尝试过fork() ,但是它也不起作用。

这是完整的服务器C代码。

#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <string.h>

#define BUFLEN 512
#define PORT 9930

void diep(char *s) {
     perror(s);
     exit(1);
}

int main() {
    struct sockaddr_in si_me, si_other;
    int s, i, slen=sizeof(si_other), broadcastPermission;
    char buf[100], zeni[BUFLEN];

    if ((s=socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP))==-1)
       diep("socket");

    broadcastPermission = 1;
    if (setsockopt(s, SOL_SOCKET, SO_BROADCAST, (void *) &broadcastPermission, sizeof(broadcastPermission)) < 0)
        diep("setsockopt() failed");

    memset((char *) &si_me, 0, sizeof(si_me));
    si_me.sin_family = AF_INET;
    si_me.sin_port = htons(PORT);
    si_me.sin_addr.s_addr = htonl(INADDR_ANY);
    if (bind(s, &si_me, sizeof(si_me))==-1)
       diep("bind");

    while (1) {
        if (recvfrom(s, buf, BUFLEN, 0, &si_other, &slen)==-1) diep("recvfrom()");
        //printf("Received packet from %s:%d\nData: %s\n", inet_ntoa(si_other.sin_addr), ntohs(si_other.sin_port), buf);

 strcpy(zeni, "");
 strcat(zeni, "zenity --warning --title Hack!! --text ");
 strcat(zeni, buf);
 printf("cmd: %s\n", zeni);
 //system (zeni);
 execl("/usr/bin/zenity", "/usr/bin/zenity", "--warning", "--title", "Warn!", "--text", buf, (char *) NULL);
    }

    close(s);
    return 0;

}

@jweyrich已经指出了您使用recvfrom一些问题,但是还有一个更根本的问题。 编码

while (1) {
    recvfrom(...);
    execl(...);
}

最多只能执行一次。 这是因为exec系列系统调用(包括execl )用对execl的调用中给定的程序替换了当前正在执行的程序。 实际上, execl从不返回,除非有错误。

要在Unix中创建新的子进程,必须首先调用fork ,它克隆现有进程,然后在子调用execl (或一些相关的系统调用)中用实际要运行的程序替换子进程。 手动正确地执行此操作有些棘手,因此system功能可以为您解决这一问题,但是它有其自身的缺点。

您对recvfrom的调用中有堆栈溢出。

recvfrom(s, buf, BUFLEN, 0, &si_other, &slen)

我相信您混合了两个缓冲区。 您正在使用buf ,大小为100,但告诉它的大小为BUFLEN,即512。每当有人发送100个以上的字节时,程序很可能会崩溃。

除此之外, recvfrom可能不会返回,因为它没有收到任何东西。 您的printf是否正在执行?


更新:@Daniel和@Dale指出,除非发生错误,否则execl不会返回。 从手册页引用:

exec系列功能将当前的过程映像替换为新的过程映像。

或者,您可以使用system

请阅读exec(3)手册页。

在您的情况下,execl系统调用将用/ usr / bin / zenity的映像替换当前进程的映像”。

您有两种解决方案:在此处尝试使用system(3)或在子运行execlp内进行fork。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM