Isso está acontecendo porque, por padrão, os sockets AF_INET6 funcionarão para IPv4 e IPv6. Consulte a seção 3.7 - Compatibilidade com os nós IPv4 da RFC 3493 - Extensões da interface do soquete básico para IPv6
Aqui está um pequeno exemplo de código que pode gerar esse tipo de situação:
#include <stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define TEST_PORT 5555
#define xstr(s) str(s)
#define str(x) #x
int main (int argc, char **argv)
{
int v6server;
int v4client;
int rc;
struct sockaddr_in6 s6addr = {
.sin6_family = AF_INET6,
.sin6_flowinfo = 0,
.sin6_port = htons(TEST_PORT),
.sin6_addr = in6addr_any
};
struct sockaddr_in c4addr = {
.sin_family = AF_INET,
.sin_port = htons(TEST_PORT),
.sin_addr = inet_addr("127.0.0.1")
};
// Open an IPv6 listener
v6server = socket(AF_INET6, SOCK_STREAM, 0);
if (v6server < 0) perror("socket()");
rc = bind(v6server, (struct sockaddr *)&s6addr, sizeof(s6addr));
if (rc != 0) perror("bind()");
rc = listen(v6server, 0);
if (rc != 0) perror("listen()");
// Connect to the listener with an IPv4 socket
v4client = socket(AF_INET, SOCK_STREAM, 0);
if (v4client < 0) perror("socket()");
rc = connect(v4client, (struct sockaddr *)&c4addr, sizeof(c4addr));
if (rc != 0) perror("connect()");
// inspect open sockets
system("netstat -tan | grep " xstr(TEST_PORT));
close(v4client);
close(v6server);
}
A saída na minha máquina Ubuntu é:
$ make v4v6
cc v4v6.c -o v4v6
$ ./v4v6
tcp 0 0 127.0.0.1:46518 127.0.0.1:5555 ESTABLISHED
tcp6 0 0 :::5555 :::* LISTEN
tcp6 0 0 127.0.0.1:5555 127.0.0.1:46518 ESTABLISHED
$
- A entrada
tcp6 LISTEN
é para o soquete que escuta na porta 5555. Observe que é um soquete AF_INET6, portanto, aceitará conexões de entrada IPv4 e IPv6. - A entrada
tcp ESTABLISHED
é o resultado da conexão de um soquete AF_INET4 ao ouvinte (conexão ativa). - A entrada
tcp6 ESTABLISHED
é para a conexão passiva gerada a partir do soquete do listener. Ele aparece comotcp6
, já que é gerado a partir de umtcp6
listener; no entanto, representa uma conexão de um IPv4.
Vale a pena notar o seguinte:
- Esse comportamento é especial para soquetes AF_INET6. Os soquetes AF_INET (IPv4) simplesmente não podem e não lidam com nada IPv6.
- Esse comportamento pode ser substituído pela opção de soquete IPV6_V6ONLY . Definir essa opção fará com que o soquete somente manipule o IPv6 e não permita nada IPv4.