2015-12-12 92 views
0

我正在尝试编写通过UDP进行通信的tic-tack-toe游戏。 现在我有一些代码:从缓存中获取缓冲

int recv_txt(int sock, struct sockaddr_in *incoming_addr) 
{ 
    char bud[MAX_LENGTH]; 
    unsigned int incoming_add_len; 
    incoming_add_len = sizeof(*incoming_addr); 
    if(recvfrom(sock, &buf, MAX_LENGTH, 0 (struct sockaddr*)incoming_addr, &incoming_addr_len) < 0) 
    { 
     return 0; 
    } 
    printf("received %s", buf); 
    return 1; 
} 

int main(int argv, char **argc) 
{ 
    /** some code to create socket etc */ 
    struct sockaddr_in incoming_addr; 
    for(;;) 
    { 
     recv_txt(sock, &incoming_addr); 

     //here I would like to create new thread, which will process the buffer from recv_txt and send response; 
    } 
} 

,现在我需要从recv_txt得到缓冲,把它里面的结构是这样的:

struct M_s 
{ 
    struct sockaddr_in address; 
    char[MAX_LENGTH] buffer; 
} 

,并把它传递给新的线程。但我无法从recv_txt获得缓冲区。我很新的C,现在我不能使用指针非常好。

感谢您的任何建议。

编辑

我试图通过Frankie_c提供的解决方案,但我有问题,现在printf。 当我尝试在main

LP_ARGS_STRUCT args = recv_txt(sock) 
printf("data from: %s", inet_ntoa(args->address.sin_adrr)) 

我没有得到任何打印到控制台或遇到错误信息。

EDIT2 - 全码

#include <stdio.h> 
#include <unistd.h> 
#include <stdlib.h> 
#include <string.h> 

#include <sys/socket.h> 
#include <netinet/in.h> 
#include <arpa/inet.h> 

#include <pthread.h> 

#define SRV_PORT   23456 
#define MAX_TXT_LENGTH  1000 
#define MAX_ROOMS   1 

int sock; 

typedef struct args_s 
{ 
    struct sockaddr_in address; 
    char buffer[MAX_TXT_LENGTH]; 
} ARGS_STRUCT, *LP_ARGS_STRUCT; 
//typedef struct args_s args; 


LP_ARGS_STRUCT recv_txt(int sock) 
{ 
    LP_ARGS_STRUCT args = malloc(sizeof(ARGS_STRUCT)); 

    //memory error 
    if(!args) 
    { 
     return 1; 
    } 

    unsigned int incoming_addr_len = sizeof(args->address); 

    //incoming_addr_len = sizeof(*incoming_addr); 
    if (recvfrom(sock, &args->buffer, MAX_TXT_LENGTH, 0, 
        (struct sockaddr*)&args->address, &incoming_addr_len) < 0) 
    { 
     free(args); 
     return NULL; 
    } 
    printf("received: %s %s\n", args->buffer, inet_ntoa(args->address.sin_addr)); 
    return 1; 
} 

int main(int argv, char **argc) 
{ 
    int i = 0; 

    int optval; 
    struct sockaddr_in addr, incoming_addr; 
    char buffer[MAX_TXT_LENGTH]; 

    /* create socket */ 
    sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); 
    if (sock < 0) 
    { 
     err("socket()"); 
    } 

    /* set reusable flag */ 
    optval = 1; 
    setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)); 

    /* prepare inet address */ 
    memset(&addr, 0, sizeof(addr)); 
    addr.sin_family = AF_INET; 
    addr.sin_port = htons(SRV_PORT); 
    addr.sin_addr.s_addr = htonl(INADDR_ANY); /* listen on all interfaces */ 
    if (bind(sock, (struct sockaddr*)&addr, sizeof(addr)) < 0) 
    { 
     err("bind"); 
    } 


    for(;;) 
    { 


     LP_ARGS_STRUCT args = recv_txt(sock); 
     if(!args) continue; 
     printf("%s\n", args->buffer); 

    } 

    return 0; 
} 

和接收数据包后,我得到:

received: Hello 192.168.56.1 
Segmentation fault (code dumped) 

Process returned 139 (0x8B) 
+0

您是否考虑过使用更高级别的语言编写此程序?你想要做的事情是可能的,但是你在极深的一端跳进了鲨鱼坦克。 – zwol

+0

但是如何从'recv_txt'获得缓冲区,它接收到的缓冲区只是该函数的本地区域,您不会将其返回或复制到作为参数传递的缓冲区。 –

+0

@zwol 我知道,但我没有选择。这是从学校项目和客户端这个游戏是用Java编写的,但服务器必须在C. – Sk1X1

回答

1

对于原来的问题试试这个:

typedef struct M_s 
{ 
    struct sockaddr_in address; 
    char[MAX_LENGTH] buffer; 
} M_STRUCT, *LP_M_STRUCT; //Create a typedef for struct 

//Modify your function to return a structure pointer and having just socket as par... 
LP_M_STRUCT recv_txt(int sock) 
{ 
    LP_M_STRUCT pMs = malloc(sizeof(M_STRUCT)); 
    if (!pMs) 
    { 
     //Handle memory error here! 
     return NULL; 
    } 

    unsigned int incoming_add_len = sizeof(pMs->incoming_addr); 

    if (recvfrom(sock, &pMs->buffer, MAX_LENGTH, 0, (struct sockaddr *)&pMs->incoming_addr, &incoming_addr_len) < 0) 
    { 
     free(pMs);  //Release memory 
     return NULL; //Return nothing... 
    } 

    //If we are here we were successfull. 
    //The structure pMs holds all data we need... so give it to caller... 
    printf("From %s received %s", inet_ntoa(pMs->address.sin_addr), pMs->buffer); 
    return pMs; 
} 

int main(int argv, char **argc) 
{ 
    /** some code to create socket etc */ 

    for (;;) 
    { 
     LP_M_STRUCT pMs = recv_txt(sock); 
     if (!pMs) //Check if we get a sign of life on UDP... 
      continue; //No continue to wait for ... 

     //We print values again to check that it is ok... 
     printf("From %s received %s", inet_ntoa(pMs->address.sin_addr), pMs->buffer); 

     //here create new thread to process the buffer from recv_txt and send response; 
     //The new thread code have to release structure memory when done!!! 
    } 
} 

编辑:你的问题是你不返回分配结构成功时,并且在失败时不返回NULL。请参阅:

LP_ARGS_STRUCT recv_txt(int sock) 
{ 
    LP_ARGS_STRUCT args = malloc(sizeof(ARGS_STRUCT)); 

    //memory error 
    if(!args) 
    { 
     //return 1; !!!WRONG!!! 
     return NULL; 
    } 

    unsigned int incoming_addr_len = sizeof(args->address); 

    //incoming_addr_len = sizeof(*incoming_addr); 
    if (recvfrom(sock, &args->buffer, MAX_TXT_LENGTH, 0, 
        (struct sockaddr*)&args->address, &incoming_addr_len) < 0) 
    { 
     free(args); 
     return NULL; 
    } 
    printf("received: %s %s\n", args->buffer, inet_ntoa(args->address.sin_addr)); 
    //return 1; WRONG! You have to return the allocated struct 
    return args; 
} 
+0

感谢您的答复。我试过你的解决方案,现在我遇到了'printf'的问题。我用精确的代码编辑了我的问题。你能告诉我做错了什么吗? – Sk1X1

+0

我误解了你的评论。抱歉。无论如何,我更新了我的答案在接收例程printf中的错误。让我看看更好看。 –

+0

感谢您的帮助,'recv_txt'中的'printf'现在运行良好,但我不知道如何修复'main'中的'printf'。 另外我编辑我的问题,以提供我的'printf'错误的更具体位置。 – Sk1X1