184 lines
5.7 KiB
C
184 lines
5.7 KiB
C
|
/*
|
||
|
* ip6.h
|
||
|
*
|
||
|
* Internet Protocol, Version 6 (RFC 2460).
|
||
|
*
|
||
|
* Copyright (c) 2002 Dug Song <dugsong@monkey.org>
|
||
|
*
|
||
|
* $Id: ip6.h,v 1.6 2004/02/23 10:01:15 dugsong Exp $
|
||
|
*/
|
||
|
|
||
|
#ifndef DNET_IP6_H
|
||
|
#define DNET_IP6_H
|
||
|
|
||
|
#define IP6_ADDR_LEN 16
|
||
|
#define IP6_ADDR_BITS 128
|
||
|
|
||
|
#define IP6_HDR_LEN 40 /* IPv6 header length */
|
||
|
#define IP6_LEN_MIN IP6_HDR_LEN
|
||
|
#define IP6_LEN_MAX 65535 /* non-jumbo payload */
|
||
|
|
||
|
#define IP6_MTU_MIN 1280 /* minimum MTU (1024 + 256) */
|
||
|
|
||
|
typedef struct ip6_addr {
|
||
|
uint8_t data[IP6_ADDR_LEN];
|
||
|
} ip6_addr_t;
|
||
|
|
||
|
#ifndef __GNUC__
|
||
|
# define __attribute__(x)
|
||
|
# pragma pack(1)
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
* IPv6 header
|
||
|
*/
|
||
|
struct ip6_hdr {
|
||
|
union {
|
||
|
struct ip6_hdr_ctl {
|
||
|
uint32_t ip6_un1_flow; /* 20 bits of flow ID */
|
||
|
uint16_t ip6_un1_plen; /* payload length */
|
||
|
uint8_t ip6_un1_nxt; /* next header */
|
||
|
uint8_t ip6_un1_hlim; /* hop limit */
|
||
|
} ip6_un1;
|
||
|
uint8_t ip6_un2_vfc; /* 4 bits version, top 4 bits class */
|
||
|
} ip6_ctlun;
|
||
|
ip6_addr_t ip6_src;
|
||
|
ip6_addr_t ip6_dst;
|
||
|
} __attribute__((__packed__));
|
||
|
|
||
|
#define ip6_vfc ip6_ctlun.ip6_un2_vfc
|
||
|
#define ip6_flow ip6_ctlun.ip6_un1.ip6_un1_flow
|
||
|
#define ip6_plen ip6_ctlun.ip6_un1.ip6_un1_plen
|
||
|
#define ip6_nxt ip6_ctlun.ip6_un1.ip6_un1_nxt /* IP_PROTO_* */
|
||
|
#define ip6_hlim ip6_ctlun.ip6_un1.ip6_un1_hlim
|
||
|
|
||
|
#define IP6_VERSION 0x60
|
||
|
#define IP6_VERSION_MASK 0xf0 /* ip6_vfc version */
|
||
|
|
||
|
#if DNET_BYTESEX == DNET_BIG_ENDIAN
|
||
|
#define IP6_FLOWINFO_MASK 0x0fffffff /* ip6_flow info (28 bits) */
|
||
|
#define IP6_FLOWLABEL_MASK 0x000fffff /* ip6_flow label (20 bits) */
|
||
|
#elif DNET_BYTESEX == DNET_LIL_ENDIAN
|
||
|
#define IP6_FLOWINFO_MASK 0xffffff0f /* ip6_flow info (28 bits) */
|
||
|
#define IP6_FLOWLABEL_MASK 0xffff0f00 /* ip6_flow label (20 bits) */
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
* Hop limit (ip6_hlim)
|
||
|
*/
|
||
|
#define IP6_HLIM_DEFAULT 64
|
||
|
#define IP6_HLIM_MAX 255
|
||
|
|
||
|
/*
|
||
|
* Preferred extension header order from RFC 2460, 4.1:
|
||
|
*
|
||
|
* IP_PROTO_IPV6, IP_PROTO_HOPOPTS, IP_PROTO_DSTOPTS, IP_PROTO_ROUTING,
|
||
|
* IP_PROTO_FRAGMENT, IP_PROTO_AH, IP_PROTO_ESP, IP_PROTO_DSTOPTS, IP_PROTO_*
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* Routing header data (IP_PROTO_ROUTING)
|
||
|
*/
|
||
|
struct ip6_ext_data_routing {
|
||
|
uint8_t type; /* routing type */
|
||
|
uint8_t segleft; /* segments left */
|
||
|
/* followed by routing type specific data */
|
||
|
} __attribute__((__packed__));
|
||
|
|
||
|
struct ip6_ext_data_routing0 {
|
||
|
uint8_t type; /* always zero */
|
||
|
uint8_t segleft; /* segments left */
|
||
|
uint8_t reserved; /* reserved field */
|
||
|
uint8_t slmap[3]; /* strict/loose bit map */
|
||
|
ip6_addr_t addr[1]; /* up to 23 addresses */
|
||
|
} __attribute__((__packed__));
|
||
|
|
||
|
/*
|
||
|
* Fragment header data (IP_PROTO_FRAGMENT)
|
||
|
*/
|
||
|
struct ip6_ext_data_fragment {
|
||
|
uint16_t offlg; /* offset, reserved, and flag */
|
||
|
uint32_t ident; /* identification */
|
||
|
} __attribute__((__packed__));
|
||
|
|
||
|
/*
|
||
|
* Fragmentation offset, reserved, and flags (offlg)
|
||
|
*/
|
||
|
#if DNET_BYTESEX == DNET_BIG_ENDIAN
|
||
|
#define IP6_OFF_MASK 0xfff8 /* mask out offset from offlg */
|
||
|
#define IP6_RESERVED_MASK 0x0006 /* reserved bits in offlg */
|
||
|
#define IP6_MORE_FRAG 0x0001 /* more-fragments flag */
|
||
|
#elif DNET_BYTESEX == DNET_LIL_ENDIAN
|
||
|
#define IP6_OFF_MASK 0xf8ff /* mask out offset from offlg */
|
||
|
#define IP6_RESERVED_MASK 0x0600 /* reserved bits in offlg */
|
||
|
#define IP6_MORE_FRAG 0x0100 /* more-fragments flag */
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
* Option types, for IP_PROTO_HOPOPTS, IP_PROTO_DSTOPTS headers
|
||
|
*/
|
||
|
#define IP6_OPT_PAD1 0x00 /* 00 0 00000 */
|
||
|
#define IP6_OPT_PADN 0x01 /* 00 0 00001 */
|
||
|
#define IP6_OPT_JUMBO 0xC2 /* 11 0 00010 = 194 */
|
||
|
#define IP6_OPT_JUMBO_LEN 6
|
||
|
#define IP6_OPT_RTALERT 0x05 /* 00 0 00101 */
|
||
|
#define IP6_OPT_RTALERT_LEN 4
|
||
|
#define IP6_OPT_RTALERT_MLD 0 /* Datagram contains an MLD message */
|
||
|
#define IP6_OPT_RTALERT_RSVP 1 /* Datagram contains an RSVP message */
|
||
|
#define IP6_OPT_RTALERT_ACTNET 2 /* contains an Active Networks msg */
|
||
|
#define IP6_OPT_LEN_MIN 2
|
||
|
|
||
|
#define IP6_OPT_TYPE(o) ((o) & 0xC0) /* high 2 bits of opt_type */
|
||
|
#define IP6_OPT_TYPE_SKIP 0x00 /* continue processing on failure */
|
||
|
#define IP6_OPT_TYPE_DISCARD 0x40 /* discard packet on failure */
|
||
|
#define IP6_OPT_TYPE_FORCEICMP 0x80 /* discard and send ICMP on failure */
|
||
|
#define IP6_OPT_TYPE_ICMP 0xC0 /* ...only if non-multicast dst */
|
||
|
|
||
|
#define IP6_OPT_MUTABLE 0x20 /* option data may change en route */
|
||
|
|
||
|
/*
|
||
|
* Extension header (chained via {ip6,ext}_nxt, following IPv6 header)
|
||
|
*/
|
||
|
struct ip6_ext_hdr {
|
||
|
uint8_t ext_nxt; /* next header */
|
||
|
uint8_t ext_len; /* following length in units of 8 octets */
|
||
|
union {
|
||
|
struct ip6_ext_data_routing routing;
|
||
|
struct ip6_ext_data_fragment fragment;
|
||
|
} ext_data;
|
||
|
} __attribute__((__packed__));
|
||
|
|
||
|
#ifndef __GNUC__
|
||
|
# pragma pack()
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
* Reserved addresses
|
||
|
*/
|
||
|
#define IP6_ADDR_UNSPEC \
|
||
|
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
|
||
|
#define IP6_ADDR_LOOPBACK \
|
||
|
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01"
|
||
|
|
||
|
#define ip6_pack_hdr(hdr, fc, fl, plen, nxt, hlim, src, dst) do { \
|
||
|
struct ip6_hdr *ip6 = (struct ip6_hdr *)(hdr); \
|
||
|
ip6->ip6_flow = htonl(((uint32_t)(fc) << 28) & \
|
||
|
(IP6_FLOWLABEL_MASK | (fl))); \
|
||
|
ip6->ip6_vfc = (IP6_VERSION | ((fc) >> 4)); \
|
||
|
ip6->ip6_plen = htons((plen)); \
|
||
|
ip6->ip6_nxt = (nxt); ip6->ip6_hlim = (hlim); \
|
||
|
memmove(&ip6->ip6_src, &(src), IP6_ADDR_LEN); \
|
||
|
memmove(&ip6->ip6_dst, &(dst), IP6_ADDR_LEN); \
|
||
|
} while (0);
|
||
|
|
||
|
__BEGIN_DECLS
|
||
|
char *ip6_ntop(const ip6_addr_t *ip6, char *dst, size_t size);
|
||
|
int ip6_pton(const char *src, ip6_addr_t *dst);
|
||
|
char *ip6_ntoa(const ip6_addr_t *ip6);
|
||
|
#define ip6_aton ip6_pton
|
||
|
|
||
|
void ip6_checksum(void *buf, size_t len);
|
||
|
__END_DECLS
|
||
|
|
||
|
#endif /* DNET_IP6_H */
|