This forum has been archived. All content is frozen. Please use KDE Discuss instead.

[SVN 597763] DHT UDP Port not binded (bound?)

Tags: None
(comma "," separated)
Moshroum
Registered Member
Posts
63
Karma
0
I checked "netstat -l" because nothing happened in dht and I noticed that the tcp was shown but not the udp port for dht action. Startet and stopped ktorrent but nothing happened. After setting the port to another one everything goes in the right direction, but when I stop ktorrent again it isnt bind/binded/bounded (whats the right word?).

I havent checked the code but will do it in some minutes (need to push my guests out of the room :twisted: ).

OK, the socks->bind in RPCServer::start definitly says that it was everything ok (but it isn't).

EDIT: Yes, I tested if the port doesnt get any data with "nc -u PORT". When I change the post -> LISTEN in "netstat -l" i can connect to the udp port and create some parse errors (ok, next time i write correct data to it ;) ). I really dont know why it doesnt listen normally to it.

PS: I am using qt 3.3.6 and kdelibs 3.5.5 on linux 2.6.18.


EDIT: A fix (?) is to enable blocking instead of disable it. This is not my idea but the idea of the kdecore api documentation of KDatagramSocket: "If the socket is blocking, the socket will be bound when this function returns."
It doesnt they how to really reaaalllyy REAALLLLYYY (sry, but it makes me angry) bind it to a port with non blocking sockets.
Maybe it is possible to set blocking after the bind attempt? It works here but I dont know why it works when I do it a second time and it was set to non-blocking before the bind.

To change to blocking mode also fixes the problem that it crashes when you change the port relative fast 3 (or 2?) times in a row.

Working test:
Code: Select all
Index: libktorrent/kademlia/rpcserver.cpp
===================================================================
--- libktorrent/kademlia/rpcserver.cpp  (revision 597775)
+++ libktorrent/kademlia/rpcserver.cpp  (working copy)
@@ -42,7 +42,6 @@
        RPCServer::RPCServer(DHT* dh_table,Uint16 port,QObject *parent) : QObject(parent),dh_table(dh_table),next_mtid(0),port(port)
        {
                sock = new KDatagramSocket(this);
-               sock->setBlocking(false);
                sock->setAddressReuseable(true);
        }

@@ -58,8 +57,9 @@

        void RPCServer::start()
        {
+               sock->bind(QString::null,QString::number(port));
+               sock->setBlocking(false);
                connect(sock,SIGNAL(readyRead()),this,SLOT(readPacket()));
-               sock->bind(QString::null,QString::number(port));
        }

        void RPCServer::stop()
George
Moderator
Posts
5421
Karma
1

Sun Oct 22, 2006 10:50 am
OK, I have more or less done what you suggested. I guess it can't hurt that bind is blocking.
jdong
Registered Member
Posts
358
Karma
0

Mon Oct 23, 2006 2:40 pm
Does this also apply to 2.0.x, or just SVN?
Moshroum
Registered Member
Posts
63
Karma
0

Mon Oct 23, 2006 4:11 pm
Could be....or not. I never run a real release of ktorrent, just svn. Just use "netstat -l|grep udp" and look if your dht port is shown. But the patch itself should work with ktorrent 2.0.3 (please use the patch from svn - my patch was just a quick test).
jdong
Registered Member
Posts
358
Karma
0

Mon Oct 23, 2006 4:35 pm
Yes, I realize the patch applies to 2.0.3 cleanly, but I don't know if that's a bug with KTorrent 2.0.3 or just something that's shown up in SVN.
George
Moderator
Posts
5421
Karma
1

Mon Oct 23, 2006 5:04 pm
jdong wrote:Yes, I realize the patch applies to 2.0.3 cleanly, but I don't know if that's a bug with KTorrent 2.0.3 or just something that's shown up in SVN.


This bug is also in 2.0.3. I don't actually understand this bug. So what if it isn't bound immediatly ? We can wait, binding does not take long normally, as it is an entirely local operation.
Moshroum
Registered Member
Posts
63
Karma
0

Mon Oct 23, 2006 5:10 pm
I also dont understand it a small low level test with enable non-blocking before binding works perfect:
Code: Select all
/*
** listener.c -- a datagram sockets "server" demo
*/

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

#define MYPORT 4950    // the port users will be connecting to

#define MAXBUFLEN 100

int main(void)
{
    int sockfd;
    struct sockaddr_in my_addr;    // my address information
    struct sockaddr_in their_addr; // connector's address information
    socklen_t addr_len;
    int numbytes;
    char buf[MAXBUFLEN];
   int flags;

    if ((sockfd = socket(PF_INET, SOCK_DGRAM, 0)) == -1) {
        perror("socket");
        exit(1);
    }

    my_addr.sin_family = AF_INET;         // host byte order
    my_addr.sin_port = htons(MYPORT);     // short, network byte order
    my_addr.sin_addr.s_addr = INADDR_ANY; // automatically fill with my IP
    memset(&(my_addr.sin_zero), '\0', 8); // zero the rest of the struct

if (-1 != (flags = fcntl(sockfd, F_GETFL, 0)))
   fcntl(sockfd, F_SETFL, flags | O_NONBLOCK);

    if (bind(sockfd, (struct sockaddr *)&my_addr,
        sizeof(struct sockaddr)) == -1) {
        perror("bind");
        exit(1);
    }

while (1){
    addr_len = sizeof(struct sockaddr);
    if ((numbytes=recvfrom(sockfd, buf, MAXBUFLEN-1 , 0,
        (struct sockaddr *)&their_addr, &addr_len)) == -1) {
       continue;
    }

    printf("got packet from %s\n",inet_ntoa(their_addr.sin_addr));
    printf("packet is %d bytes long\n",numbytes);
    buf[numbytes] = '\0';
    printf("packet contains \"%s\"\n",buf);
}
    close(sockfd);

    return 0;
}
George
Moderator
Posts
5421
Karma
1

Tue Oct 24, 2006 5:18 pm
Maybe some obscure bug in KDatagramSocket ? Anyway, bug fixed.


Bookmarks



Who is online

Registered users: bartoloni, Bing [Bot], Evergrowing, Google [Bot], q.ignora, watchstar