C系统调用msgsnd():某些编译器版本上的参数错误无效

时间:2014-12-03 21:24:15

标签: c gcc version system-calls

好的,所以我正在做一个大学项目,很抱歉这个愚蠢的问题。

我的任务是将字符串推送到消息队列,派生子进程并接收消息,然后通过相同的消息队列将响应发送回父进程,并在循环中执行所有操作。

我在家里编写了代码,它工作正常,但是部分工作是让它在大学服务器上运行,这就是我遇到的问题。

以下是我的代码的相关部分:

家长流程:

void fel2()
{   
    int messq, status;
    key_t key = ftok("key", 1);
    messq = msgget( key, 0600 | IPC_CREAT ); 

    OpBike * ob_first = NULL;
    ob_first = ob_read_in(ob_first);
    char confirm_string[64];

    while (exists_busy_bike(ob_first))
    {
        sleep(5);
        ob_first = NULL;
        ob_first = ob_read_in(ob_first);

        int busy_bikes = count_busy_bikes(ob_first);
        ob_first = NULL;

        srand(time(NULL));
        int bret = (rand() % busy_bikes);

        ob_first = ob_read_in(ob_first);
        ob_first = ob_read_in(ob_first);

        int ret_bike_id = get_bikeid_by_jumps(ob_first, bret);

        char ret_bike_id_char[64];
        snprintf(ret_bike_id_char, 64, "%d", ret_bike_id);

        send(messq, ret_bike_id_char);
        fel2_child(messq, key);
        ob_first = NULL;

        wait(NULL);

        strcpy(confirm_string, recv(messq));

        ob_first = NULL;
        ob_first = ob_read_in(ob_first);
        wait(NULL);
    }


    status = msgctl( messq, IPC_RMID, NULL ); 
    wait(NULL);
    return;


}

子进程:

int fel2_child(int messq, key_t key)
{
    pid_t child = fork();

    if (child > 0) // parent process does nothing here
    {
    } 
    else {          // child
        char rbikeid[64];
        strcpy(rbikeid, recv(messq));

        OpBike * ob_first = NULL;
        ob_first = ob_read_in(ob_first);
        OpRent * or_first = NULL;
        or_first = or_read_in(or_first);

        OpBike * bike_ret = get_bike_by_bikeid(ob_first, rbikeid);
        char check[64];
        strcpy(check, bike_ret->bike_id);

        ob_first = NULL;
        ob_first = ob_read_in(ob_first);

        or_add_return(or_first, ob_first, rbikeid);

        ob_first = NULL;
        ob_first = ob_read_in(ob_first);

        OpBike * bike_check = get_bike_by_bikeid(ob_first, rbikeid);
        strcpy(check, bike_check->bike_status);


        send(messq, check);
        exit(0);

    }

    return 0;

}

发送者和接收者功能:

int send( int messq, char * text ) 
{ 
     struct message me = { 119, "asd" }; 
     int status; 

     strcpy(me.mtext, text);

     // THIS IS WHERE EVERYTHING GOES WRONG
     status = msgsnd( messq, &me, strlen ( me.mtext ) + 1 , IPC_NOWAIT ); 

     if ( status < 0 ) 
          perror("msgsnd"); 
     return 0; 
} 

char * recv( int messq) 
{ 
     struct message me; 
     int status; 
     status = msgrcv(messq, &me, 1024, 119, 0 ); 

     if ( status < 0 ) 
          perror("msgsnd"); 

     char * ret;
     ret = malloc(sizeof( char ) * strlen (me.mtext));
     strcpy(ret, me.mtext);

     return ret;
} 

在我的家用电脑上运行上述代码可以完美运行。运行Ubuntu 14.10,我的代码用gcc 4.9.1编译。但是,它在运行SLES 11 SP 1的学校服务器上不起作用,使用gcc 4.3.4编译了代码。

问题是我正在尝试发送的字符串无法正常发送。它到达函数,被复制到消息结构中,然后在某个地方消失,并且唯一回读的是未定义的数据。

我现在有点想法了。我已经尝试将消息队列的声明放在别处(正如我的教授所建议的那样),但这并没有解决问题。我无法在我学校的服务器上更新编译器,所以这就是我必须处理的内容。有没有人通过不同的编译器版本经历过奇怪的系统调用行为?有人知道如何解决这个问题吗?

感谢您提前回复!

编辑2:结构消息的定义。

struct message { 
     long mtype;
     char mtext [ 1024 ]; 
}; 

1 个答案:

答案 0 :(得分:0)

好的,事实证明我的gcc版本(或者我正在使用的linux内核或任何软件组件)会自动在ftok创建密钥文件(注意我的系统上不存在“密钥”)。然而,大学服务器却没有。所以

key_t key = ftok("argv[0]", 1);

key_t key = ftok("filename", 1);

诀窍。