All of lore.kernel.org
 help / color / mirror / Atom feed
* bind() and PACKET_MULTICAST
@ 2021-04-06 19:13 Tom Cook
  2021-04-08 16:31 ` Tom Cook
  0 siblings, 1 reply; 2+ messages in thread
From: Tom Cook @ 2021-04-06 19:13 UTC (permalink / raw)
  To: Network Development

Can someone please suggest why the code below doesn't do as expected?
I expect it to bind an AF_PACKET socket to an interface and receive
packets with ethertype 0x5eeb that arrive at multicast MAC address
77:68:76:68:76:69 on that interface.  In practice, nothing arrives.

If I comment out the call to bind(), it receives packets with
ethertype 0x5eeb that are addressed to 77:68:76:68:76:69 and are
received on any interface on the system, not just eth0.  (There are no
packets with ethertype 0x5eeb sent to any other address, so this may
be coincidence.)

If I change either use of ether_type to be ETH_P_ALL instead (and
re-instate the bind() call), then it receives all ethernet frames
received on eth0.

Is this a bug?  Or is it as expected and I have to use some other
mechanism (BPF?) to filter the frames?

Thanks for any assistance,
Tom

Code:

#include <arpa/inet.h>
#include <linux/if_packet.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <net/ethernet.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>

const unsigned short eth_type = 0x5eeb;

int main() {
    int fd = socket(AF_PACKET, SOCK_RAW, htons(eth_type));
    if (fd < 0) {
        perror("socket");
        exit(1);
    }

    struct ifreq ifr;
    const char * if_name = "eth0";
    size_t if_name_len = strlen (if_name);
    memcpy(ifr.ifr_name, if_name, if_name_len);
    ioctl(fd, SIOCGIFINDEX, &ifr);
    printf("Interface has index %d\n", ifr.ifr_ifindex);

    struct sockaddr_ll addr = {0};
    addr.sll_family = AF_PACKET;
    addr.sll_ifindex = ifr.ifr_ifindex;
    addr.sll_protocol = htons(eth_type);
    addr.sll_pkttype = PACKET_MULTICAST;
    if (bind(fd, (struct sockaddr*)&addr, sizeof(addr)) < 0) {
        perror("bind");
        exit(1);
    }

    unsigned char mcast[ETH_ALEN] = {0x77, 0x68, 0x76, 0x68, 0x76, 0x69};
    struct packet_mreq mreq = {0};
    mreq.mr_ifindex = ifr.ifr_ifindex;
    mreq.mr_type = PACKET_MR_MULTICAST;
    memcpy(mreq.mr_address, mcast, ETH_ALEN);
    mreq.mr_alen = ETH_ALEN;
    if(setsockopt(fd, SOL_SOCKET, PACKET_ADD_MEMBERSHIP, &mreq,
sizeof(mreq)) < 0) {
        perror("setsockopt");
        exit(1);
    }

    char buf [2048];
    struct sockaddr_ll src_addr;
    socklen_t src_addr_len = sizeof(src_addr);
    ssize_t count = recvfrom(fd, buf, sizeof(buf), 0, (struct
sockaddr*)&src_addr, &src_addr_len);
    if (count == -1) {
        perror("recvfrom");
        exit(1);
    } else {
        printf("Received frame.\n");
        printf("Dest MAC: ");
        for (int ii = 0; ii < 5; ii++) {
            printf("%02hhx:", buf[ii]);
        }
        printf("%02hhx\n", buf[5]);
        printf("Src MAC: ");
        for (int ii = 6; ii < 11; ii++) {
            printf("%02hhx:", buf[ii]);
        }
        printf("%02hhx\n", buf[11]);
    }
}

And here is a short Python3 programme to generate such frames (install
pyroute2 package and run as `sudo python3 test.py eth0`):

import socket
from pyroute2 import IPDB
import sys
import struct
import binascii
import time

ip = IPDB()

SMAC=bytes.fromhex(ip.interfaces[sys.argv[1]]['address'].replace(':', ''))
DMAC=bytes.fromhex('776876687669')

s = socket.socket(socket.AF_PACKET, socket.SOCK_RAW)
s.bind((sys.argv[1], 0x5eeb))
#s.bind((sys.argv[1], 0))

dgram = struct.pack("!6s6sHH", DMAC, SMAC, 0x5eeb, 0x7668)
print(' '.join('{:02x}'.format(x) for x in dgram))

while True:
    s.send(dgram)
    time.sleep(0.1)

^ permalink raw reply	[flat|nested] 2+ messages in thread

* Re: bind() and PACKET_MULTICAST
  2021-04-06 19:13 bind() and PACKET_MULTICAST Tom Cook
@ 2021-04-08 16:31 ` Tom Cook
  0 siblings, 0 replies; 2+ messages in thread
From: Tom Cook @ 2021-04-08 16:31 UTC (permalink / raw)
  To: Network Development

Never mind.  In case anyone was wondering, PACKET_ADD_MEMBERSHIP is a
SOL_PACKET option, not SOL_SOCKET.  Only took me two days to spot!

On Tue, Apr 6, 2021 at 8:13 PM Tom Cook <tom.k.cook@gmail.com> wrote:
>
> Can someone please suggest why the code below doesn't do as expected?
> I expect it to bind an AF_PACKET socket to an interface and receive
> packets with ethertype 0x5eeb that arrive at multicast MAC address
> 77:68:76:68:76:69 on that interface.  In practice, nothing arrives.
>
> If I comment out the call to bind(), it receives packets with
> ethertype 0x5eeb that are addressed to 77:68:76:68:76:69 and are
> received on any interface on the system, not just eth0.  (There are no
> packets with ethertype 0x5eeb sent to any other address, so this may
> be coincidence.)
>
> If I change either use of ether_type to be ETH_P_ALL instead (and
> re-instate the bind() call), then it receives all ethernet frames
> received on eth0.
>
> Is this a bug?  Or is it as expected and I have to use some other
> mechanism (BPF?) to filter the frames?
>
> Thanks for any assistance,
> Tom
>
> Code:
>
> #include <arpa/inet.h>
> #include <linux/if_packet.h>
> #include <sys/socket.h>
> #include <sys/ioctl.h>
> #include <net/if.h>
> #include <net/ethernet.h>
> #include <string.h>
> #include <stdio.h>
> #include <stdlib.h>
>
> const unsigned short eth_type = 0x5eeb;
>
> int main() {
>     int fd = socket(AF_PACKET, SOCK_RAW, htons(eth_type));
>     if (fd < 0) {
>         perror("socket");
>         exit(1);
>     }
>
>     struct ifreq ifr;
>     const char * if_name = "eth0";
>     size_t if_name_len = strlen (if_name);
>     memcpy(ifr.ifr_name, if_name, if_name_len);
>     ioctl(fd, SIOCGIFINDEX, &ifr);
>     printf("Interface has index %d\n", ifr.ifr_ifindex);
>
>     struct sockaddr_ll addr = {0};
>     addr.sll_family = AF_PACKET;
>     addr.sll_ifindex = ifr.ifr_ifindex;
>     addr.sll_protocol = htons(eth_type);
>     addr.sll_pkttype = PACKET_MULTICAST;
>     if (bind(fd, (struct sockaddr*)&addr, sizeof(addr)) < 0) {
>         perror("bind");
>         exit(1);
>     }
>
>     unsigned char mcast[ETH_ALEN] = {0x77, 0x68, 0x76, 0x68, 0x76, 0x69};
>     struct packet_mreq mreq = {0};
>     mreq.mr_ifindex = ifr.ifr_ifindex;
>     mreq.mr_type = PACKET_MR_MULTICAST;
>     memcpy(mreq.mr_address, mcast, ETH_ALEN);
>     mreq.mr_alen = ETH_ALEN;
>     if(setsockopt(fd, SOL_SOCKET, PACKET_ADD_MEMBERSHIP, &mreq,
> sizeof(mreq)) < 0) {
>         perror("setsockopt");
>         exit(1);
>     }
>
>     char buf [2048];
>     struct sockaddr_ll src_addr;
>     socklen_t src_addr_len = sizeof(src_addr);
>     ssize_t count = recvfrom(fd, buf, sizeof(buf), 0, (struct
> sockaddr*)&src_addr, &src_addr_len);
>     if (count == -1) {
>         perror("recvfrom");
>         exit(1);
>     } else {
>         printf("Received frame.\n");
>         printf("Dest MAC: ");
>         for (int ii = 0; ii < 5; ii++) {
>             printf("%02hhx:", buf[ii]);
>         }
>         printf("%02hhx\n", buf[5]);
>         printf("Src MAC: ");
>         for (int ii = 6; ii < 11; ii++) {
>             printf("%02hhx:", buf[ii]);
>         }
>         printf("%02hhx\n", buf[11]);
>     }
> }
>
> And here is a short Python3 programme to generate such frames (install
> pyroute2 package and run as `sudo python3 test.py eth0`):
>
> import socket
> from pyroute2 import IPDB
> import sys
> import struct
> import binascii
> import time
>
> ip = IPDB()
>
> SMAC=bytes.fromhex(ip.interfaces[sys.argv[1]]['address'].replace(':', ''))
> DMAC=bytes.fromhex('776876687669')
>
> s = socket.socket(socket.AF_PACKET, socket.SOCK_RAW)
> s.bind((sys.argv[1], 0x5eeb))
> #s.bind((sys.argv[1], 0))
>
> dgram = struct.pack("!6s6sHH", DMAC, SMAC, 0x5eeb, 0x7668)
> print(' '.join('{:02x}'.format(x) for x in dgram))
>
> while True:
>     s.send(dgram)
>     time.sleep(0.1)

^ permalink raw reply	[flat|nested] 2+ messages in thread

end of thread, other threads:[~2021-04-08 16:32 UTC | newest]

Thread overview: 2+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2021-04-06 19:13 bind() and PACKET_MULTICAST Tom Cook
2021-04-08 16:31 ` Tom Cook

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.