Is always 0x4
Multiples of 4 Byte, minimum 0x5, greater than 0x5 signs that IPv4 Header Options are present
Arbitrary identifier mostly used by QoS. Can mark specific types of payload.
Congestion control
Length of IPv4 packet including header, size in Byte
Only used for fragmented packets. Ignore if not fragmented, set to 0x0000 or something. All packets with the same value belong to the same fragmented packet.
Always 0x0
If set to 0x1, don't fragment this packet (further). It might already be fragmented.
0x0, this packet is not fragmented.
If set to 0x0 this is the last fragment of the packet.
If set to 0x1, this is not the last fragment of the packet.
Is a multiple of 8 Bytes. This field specifies how much of the original packet has to be skipped, after that fragment must be placed. For the first fragment of a packet the value is 0x0.
Is decremented by each routing hop. If value is 0x0 packet is dropped.
Type of data in the payload.
Checksum of the IPv4 header.
Happy rabbit holing, if you encounter these x.x
0000 xx xx xx xx xx xx xx xx xx xx xx xx 86 dd 60 xx 0010 xx xx xx xx xx xx XX XX XX XX XX XX XX XX XX XX 0020 XX XX XX XX XX XX XX XX XX XX XX XX XX XX XX XX 0030 XX XX XX XX XX XX ...
For IPv6 you only have the Ethernet Type 0x86dd for IPv6 as a distinguisher and the protocol constant 0x60 if the QoS fields aren't used.
XX marks where the addresses are placed.
0000 xx xx xx xx xx xx xx xx xx xx xx xx 08 00 45 00 0010 xx xx 00 00 40 00 xx xx xx xx XX XX XX XX XX XX 0020 XX XX ...
IPv4 packets can be recognized by 0x0800 in the 13th and 14th Byte which is the Ethernet Type for IPv4. Also Byte 15 and 16 is usually 0x4500, as IPv4 Options and traffic control features are often not used.
Byte 21 and 22 are nearly always 0x4000 because fragmentation isn't really used these days anymore. Because of this Byte 19 and 20 can be 0x0000, but some implementations still set this to some other value despite no fragmentation is done.
XX marks where the addresses are placed.
For IPv6 it is very easy. It is just the IPv6 address, as one would write it without the double colons and the zeros filled in. So if you see 0x2a01 or 0x2001 or something like that, this is probably the beginning of an IPv6 address. If you encounter a lot of zeros six Bytes later, for example 0x0000000000000001, then this is an abbreviated IPv6 address, that is just the host ::1 in a /64.
For IPv4 it is not that easy. But you can watch out for widely used network prefixes like 0xc0a8, which is 192.168…
If More Fragments field is 0x0 and Fragment Offset is 0x0, then this packet is not fragmented.
A packet that is fragmented, must have the Don't Fragment field set to 0x1, as the packet can't be fragmented even further.