Foro de elhacker.net

Programación => Programación General => Mensaje iniciado por: lapras en 4 Enero 2011, 14:18 pm



Título: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 4 Enero 2011, 14:18 pm
He hecho 2 programas de sockets en C++.
Consiste enviar la cadena "hola" desde un ordenador a otro y hacer que lo muestre en la pantalla del otro ordenador.
Lo he conseguido hacer pero el problema es que el 80% de las veces los datos llegan corruptos. Esto me extraña por que según tengo yo entendido los sockets de flujo usan el protocolo TCP y en este protocolo es el mismo el que se encarga de que lleguen los resultados adecuadamente.
¿Por que sucede esto?¿Como se puede evitar?

Datos: un ordenador es linux y el otro es windows. No estoy usando datagramas.


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Garfield07 en 4 Enero 2011, 16:35 pm
Si pero podrias poner el code?


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 4 Enero 2011, 20:07 pm
Si, claro(aunque no creo que sea necesario):
1º:
Código
  1. #include<iostream>
  2. #include<string.h>
  3. #include<stdio.h>
  4. #include<sys/types.h>
  5. #include<winsock.h>
  6. #include<conio.h>
  7.  
  8. int main(){
  9.    {
  10.        WSADATA wsaData;   /* Si esto no funciona */
  11.        //WSAData wsaData; /* prueba esto en su lugar */
  12.  
  13.        if (WSAStartup(MAKEWORD(1, 1), &wsaData) != 0)  
  14.        {
  15.            fprintf(stderr, "WSAStartup failed.\n");
  16.            exit(1);
  17.        }  
  18.    }
  19.    ///////////////////////////////////////////////////////////////////
  20.  
  21.  
  22.  
  23.  
  24.    struct sockaddr_in my_addr, they_addr;
  25.    SOCKET sockfd, newfd;
  26.    sockfd=socket(AF_INET,SOCK_STREAM,0);
  27.    if(sockfd==-1){
  28.           printf("error en socket");
  29.           std::cin.get();
  30.           exit(-1);
  31.           }
  32.  
  33.    my_addr.sin_family = AF_INET;        
  34.    my_addr.sin_port = htons(54332);      
  35.  
  36.    if(my_addr.sin_addr.s_addr = htonl(INADDR_ANY)==-1){
  37.            printf("error en ip\n");
  38.            std::cin.get();
  39.            exit(-1);
  40.            }
  41.    memset(&(my_addr.sin_zero), '\0', 8);
  42.  
  43.    if(bind(sockfd, (struct sockaddr *)&my_addr,  sizeof(struct sockaddr))==-1){
  44.        printf("error en bind()");
  45.        std::cout<<WSAGetLastError()<<std::endl;
  46.        std::cin.get();
  47.        exit(-1);
  48.    }
  49.    if(listen(sockfd, 10)==-1){
  50.        printf("error en listen()");
  51.        getch();
  52.        exit(-1);
  53.    }
  54.    int size_sockaddr=sizeof(sockaddr);
  55.    newfd=accept(sockfd,(struct sockaddr *)&they_addr, &size_sockaddr);
  56.    int bytes_sent;
  57.    //send(newfd, "hola", 5, 0);
  58.    do{bytes_sent=send(newfd, "hola", 5, 0);}
  59.    while (bytes_sent!=5);
  60.  
  61.    WSACleanup();
  62.    printf("todo en orden\n");
  63.    getchar();
  64. }

2º:

Código
  1. #include<stdio.h>
  2. #include<iostream>
  3. #include<string.h>
  4. #include<arpa/inet.h>
  5. #include<sys/types.h>
  6. #include<netinet/in.h>
  7.  
  8. int main(){
  9. int sockfd;
  10. struct sockaddr_in dest_addr;
  11. sockfd=socket(AF_INET, SOCK_STREAM, 0);
  12. if(sockfd==-1){
  13.        printf("error en el socket");
  14.        std::cin.get();
  15.        return -1;
  16.        }
  17. dest_addr.sin_family=AF_INET;
  18. dest_addr.sin_port=htons(54332);
  19. dest_addr.sin_addr.s_addr=inet_addr("192.168.1.112");
  20. if(dest_addr.sin_addr.s_addr==-1){
  21.        printf("error en ip");
  22.        getchar();
  23.        return -1;
  24.        }
  25. memset(&(dest_addr.sin_zero),0, 8);
  26. if(connect(sockfd, (struct sockaddr*)&dest_addr, sizeof(struct sockaddr))==-1){
  27.        printf("error en conect");
  28.        getchar();
  29.        return -1;
  30. }
  31. char cadena[5];
  32. recv(sockfd, &cadena, 5, 0);
  33. cadena[4]=0;
  34. std::cout<<cadena<<std::endl;
  35. printf("todo en orden\n");
  36. }


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Garfield07 en 4 Enero 2011, 20:14 pm
¿Y por que no lo has puesto en el foro de C/C++?
PD: Luego miro el code que ahora voy a jugar un poquito  :P


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 4 Enero 2011, 20:33 pm
Por que a  pesar de estar escrito en C++ la duda no tiene que ver en absoluto con C++. Tiene que ver con la programación de sockets. xD


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Garfield07 en 4 Enero 2011, 22:29 pm
Exactamente  ::) Necesitaria ver el code para ver si tiene algo que lo provoque. Los sockets se acaban al compilar xD no tienen vuelta de hoja...


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Khronos14 en 4 Enero 2011, 22:42 pm
Hola tuket, yo probaría a enviar los datos de esta forma para ver si es este el problema.

Código
  1. char buff[] = "hola";
  2. send(newfd, buff, 5, 0);
  3.  

Si tengo tiempo, luego pruebo los programas.

Saludos.


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Garfield07 en 4 Enero 2011, 22:48 pm
Podrias empezar mirando codes de sockets que hay en el foro de C/C++


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 4 Enero 2011, 22:59 pm
Hola, gracias por responder.
Citar
Hola tuket, yo probaría a enviar los datos de esta forma para ver si es este el problema.

Código
char buff[] = "hola";
send(newfd, buff, 5, 0);
 

Si tengo tiempo, luego pruebo los programas.

Saludos.
Eso ya lo probé, pasa lo mismo: la gran mayoría de las veces los datos llegan corruptos. Yo intuyo que se debe de hacer alguna comprobación pero no se.


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Garfield07 en 4 Enero 2011, 23:04 pm
http://foro.elhacker.net/ejercicios/ejercicios_en_c-t34911.0.html
Sockets


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 4 Enero 2011, 23:28 pm
Bueno. Ya que nadie entiende lo que se pregunta lo voy a repetir de otra manera.
Segun wikipedia:
Citar
TCP
El protocolo garantiza que los datos serán entregados en su destino sin errores y en el mismo orden en que se transmitieron.
Lo que significa que si envió los datos por TCP tienen que llegar seguro correctamente.
Como dice esta pagina(http://www.fismat.umich.mx/mn1/manual/node24.html):
Citar
Socket de flujo
da un flujo de datos de dos vías, confiable, y sin duplicados sin límites de grabación. El flujo opera en forma parecida a una conversación telefónica. El tipo del socket es SOCK_STREAM, el cual en el dominio de Internet usa TCP (Transmission Control Protocol).
SOCK_STREAM usa el protocolo TCP lo que significa que la cadena "hola" tiene que llegar por cojones al ordenador opuesto(de hecho en ocasiones si que llega).
Mi pregunta es: ¿Por que demonios no llega?¿Se debe a la recepción?¿Wikipedia miente?¿Hay interferencias?¿los extraterrestres están jodiéndome manipulando los datos?
XD espero que ahora se entienda mejor

PD: gracias por el link Sagrini pero no se a que viene eso.


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: Khronos14 en 5 Enero 2011, 00:05 am
tuket, el protocolo TCP/IP no falla. Lo que falla son los programas, o el cliente o el servidor.

Saludos.


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 5 Enero 2011, 00:10 am
Es bueno saberlo  :)


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: ukol en 5 Enero 2011, 18:07 pm
Te faltó cerrar el socket(closesocket/shutdown en wsock) quizá sea eso(código 1).
http://msdn.microsoft.com/en-us/library/bb530744%28v=VS.85%29.aspx (http://msdn.microsoft.com/en-us/library/bb530744%28v=VS.85%29.aspx)

Edito: Por cierto, en el segundo código no miras si da error, si devuelve -1 miras el errno con strerror(errno) y te hubiera dado mas información de lo que pasa. Que lo sepas pa la próxima, en windows no se como se hace pero seguro hay forma.


Título: Re: Imperfeccion en la transmisión con sockets de flujo.
Publicado por: lapras en 5 Enero 2011, 19:43 pm
Nada ya esta solucionado dejo el código cambiado por si a alguien le interesa:
Código
  1. #include<windows.h>
  2. #include<iostream>
  3. #include<string.h>
  4. #include<stdio.h>
  5. #include<sys/types.h>
  6. #include<winsock.h>
  7. #include<conio.h>
  8.  
  9. int main(){
  10.    {
  11.        WSADATA wsaData;   /* Si esto no funciona */
  12.        //WSAData wsaData; /* prueba esto en su lugar */
  13.  
  14.        if (WSAStartup(MAKEWORD(1, 1), &wsaData) != 0)  
  15.        {
  16.            fprintf(stderr, "WSAStartup failed.\n");
  17.            exit(1);
  18.        }  
  19.    }
  20.    ///////////////////////////////////////////////////////////////////
  21.  
  22.  
  23.  
  24.  
  25.    struct sockaddr_in my_addr, they_addr;
  26.    SOCKET sockfd, newfd;
  27.    sockfd=socket(AF_INET,SOCK_STREAM,0);
  28.    if(sockfd==-1){
  29.           printf("error en socket");
  30.           std::cin.get();
  31.           exit(-1);
  32.           }
  33.  
  34.    my_addr.sin_family = AF_INET;        
  35.    my_addr.sin_port = htons(54336);      
  36.  
  37.    if(my_addr.sin_addr.s_addr = htonl(INADDR_ANY)==-1){
  38.            printf("error en ip\n");
  39.            std::cin.get();
  40.            exit(-1);
  41.            }
  42.    memset(&(my_addr.sin_zero), '\0', 8);
  43.  
  44.    if(bind(sockfd, (struct sockaddr *)&my_addr,  sizeof(struct sockaddr))==-1){
  45.        printf("error en bind()");
  46.        std::cout<<WSAGetLastError()<<std::endl;
  47.        std::cin.get();
  48.        exit(-1);
  49.    }
  50.    if(listen(sockfd, 10)==-1){
  51.        printf("error en listen()");
  52.        getch();
  53.        exit(-1);
  54.    }
  55.    int bytes_sent;
  56.    int size_sockaddr=sizeof(sockaddr);
  57.    while(1){
  58.    newfd=accept(sockfd,(struct sockaddr *)&they_addr, &size_sockaddr);
  59.  
  60.    //send(newfd, "hola", 5, 0);
  61.    bytes_sent=send(newfd, "mola", 5, 0);
  62.    closesocket(newfd);
  63.    }                        
  64.    WSACleanup();
  65.    printf("todo en orden\n");
  66.    getchar();
  67. }