Problém už jsem vyřešil, bylo potřeba po select ověřit socket na chybu pomocí getsockopt(), nyní již vše funguje.
Příspěvky odeslané z IP adresy 213.192.14.–
Zdravím,
mám problém s následujícím zdrojákem, který má za úkol navázet spojení za pomoci neblokovacího režimu (v IPv6). Problém nastává u select, kdy je vždy vrácena hodnota větší než 0, i když daná adresa v síti vůbec není nebo na ní nic nenaslouchá na připojení.
Díky za rady.
for(ptr=result; ptr != NULL; ptr=ptr->ai_next)
{
sock_client = socket(ptr->ai_family, ptr->ai_socktype, ptr->ai_protocol);
if(sock_client == -1)
{
chyba();
}
nastaveni_soc = fcntl(sock_client, F_GETFL, 0);
if(fcntl(sock_client, F_SETFL, nastaveni_soc | O_NONBLOCK) == -1)
{
chyba();
}
FD_ZERO(&sada);
FD_SET(sock_client, &sada);
timeout.tv_sec = 5;
timeout.tv_usec = 0;
i = connect(sock_client, ptr->ai_addr, (int)ptr->ai_addrlen);
if(i == 0)
{
if(fcntl(sock_client, F_SETFL, nastaveni_soc) == -1)
{
chyba();
}
continue;
}
if(i < 0)
{
if(errno != EINPROGRESS)
{
if(fcntl(sock_client, F_SETFL, nastaveni_soc) == -1)
{
chyba();
}
break;
}
}
i = select(sock_client + 1, NULL, &sada, NULL, &timeout);
if(i < 0)
{
if(fcntl(sock_client, F_SETFL, nastaveni_soc) == -1)
{
chyba();
}
continue;
}
if(i == 0)
{
cout << "Trying again";
}
if(i > 0)
{
vysl = itoa(errno, vysl, 10);
cout....
vysl = itoa(i, vysl, 10);
cout...
vysl = itoa(sock_client, vysl, 10);
cout...
}
if(fcntl(sock_client, F_SETFL, nastaveni_soc) == -1)
{
chyba();
}
break;
}
Díky,
tak tím by mělo být již vše hotové :-)
Tak jsem se nakonec pohnul o něco vpřed:
rozlišení IP jsem udělal za pomoci getaddrinfo() a funguje bez problémů.
Podařilo se mi i úspěšně vyzkoušet multicast na IPv4 a taktéž vypadá, že funguje.
U IPv6 ovšem nevím, jak mám nastavit onen multicast. Konfigurace setsockopt(..., SO_MULTICAST, ..) se sice na internetu objevuje, ovšem u mě není podle kompilátoru deklarována a nepodařilo se mi najít, kde se nachází (která hlavičková knihovna ji obsahuje). Na žádnou jinou chybu jsem zatím nenarazil.
Momenálně mám IPv6 ve stavu, kdy neobsahuje vůbec setsockopt(), pouze konstrukce:
sap6.sin6_family = AF_INET6;
inet_pton( AF_INET6, multicast, (void *)&sap6.sin6_addr.s6_addr);
sap6.sin6_port = htons(port);
a následně hned sendto()
Jak doplním ono nastavení multicastu, aby vše správně fungovalo?
Díky
Zdravím,
snažím se vytvořit aplikaci pro Wake on LAN, tato aplikace bere jako vstup MAC adresu a masku podsítě (nebo broadcast).
V IPv4 funguje bez problémů, ale rád bych ji upravil i pro IPv6.
Abych podrobně popsal můj problém:
Pro odeslání magického packetu je potřeba vytvořit socket, aby bylo možné odeslat tento packet pomocí sendto() a k tomu je potřeba jej nastavit buď na AF_INET pro IPv4 nebo pro AF_INET6 pro IPv6 - to není problém.
Ten nastává u nastavení socketu setsockopt(), ve kterém se u IPv4 socketu nastavuje SO_BROADCAST (nenašel jsem jakýkoli jiný příklad nastavení a nemám prozatím možnost ověřit, zda takto lze opravdu poslat Wake on LAN na multicast či unicast).
V IPv6 ovšem nelze použít SO_BROADCAST, protokol toto nastavení neobsahuje a nepodařilo se mi najít, co použít místo něj. Nějaké tipy? Materiálů kolem této problematiky je jako šafránu.
Dále bych potřeboval nějak rozlišit, zda byla vložena maska IPv6 či IPv4 sítě, abych mohl vytvořit ten správný socket - jak na to v c++?
Díky za tipy