NETworking с UDP - PullRequest
       29

NETworking с UDP

0 голосов
/ 10 февраля 2012

что происходит, когда UDP-пакет отправляется на хост с помощью sendto ().все биты, которые я отправляю, отправляются (известно из возвращаемого значения). Немедленно я использую recvfrom (), который ничего не выводит, но программа не завершает работу (то есть, не возвращает).программа должна завершиться, если ответ не получен.

что будет отвечать на UDP-пакет из порта.

заблокирован ли этот пакет брандмауэром ??если да, то почему возвращаемое значение sendto неотрицательно.

Ответы [ 2 ]

0 голосов
/ 12 февраля 2012

у вас должна быть какая-то ошибка на вашем клиенте или сервере.попробуйте сначала localhost, чтобы избежать проблем с брандмауэром

это пример неблокирующего клиента / сервера udp, который я использовал для своих тестов, он использует ioctl () для проверки наличия данных для чтения в сокете, однакоесли вы хотите сделать какое-то серьезное приложение, использование epoll будет более эффективным, также вы можете указать время ожидания в микросекундах:

[null@localhost tests]$ cat udpserv.c 
#include <stdlib.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <string.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <errno.h>

#define BUFLEN 512
#define NPACK 15
#define PORT 9930

void diep(char *s)
{
  printf("erno=%d errstr=%s\n",errno,strerror(errno));
  perror(s);
  exit(1);
}

int main(void)
{
  struct sockaddr_in si_me, si_other;
  int s,ret,nbytes, i, slen=sizeof(si_other);
  char buf[BUFLEN];

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

  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,  (struct sockaddr*) &si_me, sizeof(si_me))==-1)
      diep("bind");

  fcntl(s, F_SETFL, fcntl(s, F_GETFL, 0) | O_NONBLOCK);
  sleep(10);  
  for (i=0; i<NPACK; i++) {
    ret=ioctl(s,FIONREAD,&nbytes);
    if (ret==-1) {
        printf("error on FIONREAD\n");
    } else {
        printf("nbytes=%d\n",nbytes);
    }
    if (recvfrom(s, buf, BUFLEN, 0,  (struct sockaddr*) &si_other, &slen)==-1)
      diep("recvfrom()");
    printf("Received first half of packet from %s:%d\nData: %s\n\n", 
           inet_ntoa(si_other.sin_addr), ntohs(si_other.sin_port), buf);
  }

  close(s);
  return 0;

}



[null@localhost tests]$ cat udpclient.c 
#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>

#define SRV_IP "127.0.0.1"
#define BUFLEN 200
#define NPACK 10
#define PORT 9930

/* diep(), #includes and #defines like in the server */
void diep(char *s)
{
  perror(s);
  exit(1);
}

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

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

  memset((char *) &si_other, 0, sizeof(si_other));
  si_other.sin_family = AF_INET;
  si_other.sin_port = htons(PORT);
  if (inet_aton(SRV_IP, &si_other.sin_addr)==0) {
    fprintf(stderr, "inet_aton() failed\n");
    exit(1);
 }

 for (i=0; i<NPACK; i++) {
    printf("Sending packet %d\n", i);
    sprintf(buf, "This is packet %d\n", i);
    if (sendto(s, buf, BUFLEN, 0, (struct sockaddr*) &si_other, slen)==-1)
      diep("sendto()");
 }

 close(s);
 return 0;
}

sendto () не отрицательна, поскольку возвращает количество отправленных байтов.проверьте страницу руководства для sendto

0 голосов
/ 11 февраля 2012

recvfrom() будет блокироваться до получения сообщения, если вы не установите сокет в неблокирующее состояние.

Нужные интерфейсы:

  • ioctl() с FIONBIO или O_NONBLOCK (в зависимости от вашей платформы),
  • select() длядождитесь поступления данных или через некоторое время

Также помните, что адрес и номер порта для sendto() обычно имеют порядок байтов в сети, поэтому посмотрите на ntohl и ntohs.

...