Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Null/loopback header is incorrectly interpreted as Ethernet2Header #78

Open
GyulyVGC opened this issue Dec 22, 2023 · 9 comments
Open

Comments

@GyulyVGC
Copy link

Hi @JulianSchmid, as you may know I'm using etherparse in Sniffnet.

Some users complained about the impossibility to see traffic when monitoring their VPN TUN interfaces, and today I tried it myself and found out that the problem resides in the fact that etherparse doesn't correctly identify the traffic as null/loopback, but it still identifies it as ethernet.

This is how a sample packet looks like in Wireshark:
Screenshot 2023-12-22 at 12 18 21

As you can see, it's tagged as Null/loopback.

However, etherparse still categorise this as ethernet. The consequences are multiple:

  • source and destination MAC addresses are assigned to the first bytes of the packet even if it's not actually an ethernet header
  • IP and transport header are not detected even if they are present

In particular I double checked and found out that etherparse says that ip and transport header are None, and as you can see from the screenshot below MAC addresses are assigned to the first bytes of the packet, even if those bytes are not supposed to be MAC address but part of the Null/loopback header and part of the IP header.

Screenshot 2023-12-22 at 12 22 53

I think the solution would just require to correctly parse the loopback header and changing the offset to automatically parse the following ip and transport headers.

@GyulyVGC GyulyVGC changed the title Null/loopback layer is incorrectly interpreted as Ethernet2Header Null/loopback header is incorrectly interpreted as Ethernet2Header Dec 22, 2023
@GyulyVGC
Copy link
Author

Alternatively, if you don't want to add full support for loopback header, it could be worth it to just try parsing ip header with the loopback offset in case parsing it via the Ethernet offset resolves to None (as a fallback strategy).

@JulianSchmid
Copy link
Owner

JulianSchmid commented Dec 22, 2023

Hi @GyulyVGC ,

Thanks making me aware.

So what would be needed are some method to start parsing from the null/loopback? E.g.:

  • SlicedPacket::from_null
  • PacketHeaders::from_null
  • SlicedPacket::from_null_le (for big endian, in case the traffic was captured on a machine with different endianness)
  • SlicedPacket::from_null_be (for little endian, in case the traffic was captured on a machine with different endianness)
  • PacketHeaders::from_null_le
  • PacketHeaders::from_null_be

that would parse the Null/Loopback ( https://wiki.wireshark.org/NullLoopback.md ) header and continue downwards?

Greets
Julian

@GyulyVGC
Copy link
Author

GyulyVGC commented Dec 22, 2023

Thanks for your prompt answer.

That would surely solve the problem, but I'm not sure it's the best solution.
I was thinking to parse it starting from_ethernet_slice (or any other method that allows to start form the very first bytes).

The ideal solution would require etherparse to notice that the link header is None, adjusting the offset to retrieve ip and transport headers.

In this way users wouldn't have the concern of calling a different method for each possible case, without considering that we cannot even know a priori if we should call one method with respect to another in some circumstances, i.e. when to call from_null vs when to call from_ethernet_slice?

@JulianSchmid
Copy link
Owner

JulianSchmid commented Dec 22, 2023

Writing something that tries detects if a given slice contains an ethernet II header or a null/loopback header is possible but not really robust. There is nothing in either headers that easily clearly identifies them. The only thing I could think of is to try to see if the byte position of the "ether type" field of an Ethernet II header

image

contains a known ether type value and if this not the case to check if there is a known "address family"/"protocol family" value present in any of the known null/loopback formats. But that falls apart if a to etherparse unknown ether_type is received.

I also had a look at the Wireshark implementation and they also rely on the "link_type" coming from either the interface or the "data link type" present in a pcap file.

But I could offer you that I write a small PR for Sniffnet to check the link type in
https://github.com/GyulyVGC/sniffnet/blob/5274187832a9859c684742265d60c1e204bc68d3/src/secondary_threads/parse_packets.rs#L48 by checking for the link type of the capture:

Something along the lines of:

pub fn parse_packets(
    current_capture_id: &Arc<Mutex<usize>>,
    device: &MyDevice,
    mut cap: Capture<Active>,
    filters: &Filters,
    info_traffic_mutex: &Arc<Mutex<InfoTraffic>>,
    country_mmdb_reader: &Arc<MmdbReader>,
    asn_mmdb_reader: &Arc<MmdbReader>,
) {
    let capture_id = *current_capture_id.lock().unwrap();
    let link_type = cap.get_datalink();

    loop {
        match cap.next_packet() {
            Err(_) => {
                if *current_capture_id.lock().unwrap() != capture_id {
                    return;
                }
                continue;
            }
            Ok(packet) => {
                if *current_capture_id.lock().unwrap() != capture_id {
                    return;
                }
                let slice = match link_type {
                    Linktype::ETHERNET => PacketHeaders::from_ethernet_slice(&packet),
                    Linktype::NULL | Linktype::LOOP => PacketHeaders::from_null_slice(&packet),
                    Linktype::IPV4 | Linktype::IPV6 => PacketHeaders::from_ip_slice(&packet),
                    _ => {
                        return;
                    }
                };
                match slice {

@GyulyVGC
Copy link
Author

Yeah as you said is not robust at all actually... the "ether type" field could potentially contain a valid value even if it's not an ethernet packet.

Relying on the link_type seems much more meaningful actually.

It'd be amazing if you could do that and I thank you very much for the proposal.

@GyulyVGC
Copy link
Author

GyulyVGC commented Dec 30, 2023

Hey @JulianSchmid

I've crafted a little PR myself, it'd be awesome if you could review it.

The TOFU mechanism I'm using is not so robust, so I'm open to other ideas to improve it.

@GyulyVGC
Copy link
Author

GyulyVGC/sniffnet#421

@JulianSchmid
Copy link
Owner

JulianSchmid commented Dec 31, 2023

Hey @JulianSchmid

I've crafted a little PR myself, it'd be awesome if you could review it.

The TOFU mechanism I'm using is not so robust, so I'm open to other ideas to improve it.

👍 I left you a review in GyulyVGC/sniffnet#421

@GyulyVGC
Copy link
Author

Thank you so much 🙏

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants