mirror of
				https://source.denx.de/u-boot/u-boot.git
				synced 2025-11-03 18:01:41 +01:00 
			
		
		
		
	Commands "ping6" and "tftpboot ... -ipv6" did not work on ARM because machine code expects 4-byte alignment and some structures from net6.h are not aligned in memory. Fix by adding __packed, since it is already used in this file. Signed-off-by: Sergei Antonov <saproj@gmail.com> Reviewed-by: Viacheslav Mitrofanov <v.v.mitrofanov@yadro.com>
		
			
				
	
	
		
			433 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			433 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* SPDX-License-Identifier: GPL-2.0+ */
 | 
						|
/*
 | 
						|
 * Copyright (C) 2013 Allied Telesis Labs NZ
 | 
						|
 * Chris Packham, <judge.packham@gmail.com>
 | 
						|
 *
 | 
						|
 * Copyright (C) 2022 YADRO
 | 
						|
 * Viacheslav Mitrofanov <v.v.mitrofanov@yadro.com>
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef __NET6_H__
 | 
						|
#define __NET6_H__
 | 
						|
 | 
						|
#include <net.h>
 | 
						|
#include <linux/ctype.h>
 | 
						|
 | 
						|
/* struct in6_addr - 128 bits long IPv6 address */
 | 
						|
struct in6_addr {
 | 
						|
	union {
 | 
						|
		u8	u6_addr8[16];
 | 
						|
		__be16	u6_addr16[8];
 | 
						|
		__be32	u6_addr32[4];
 | 
						|
	} in6_u;
 | 
						|
 | 
						|
#define s6_addr		in6_u.u6_addr8
 | 
						|
#define s6_addr16	in6_u.u6_addr16
 | 
						|
#define s6_addr32	in6_u.u6_addr32
 | 
						|
} __packed;
 | 
						|
 | 
						|
#define IN6ADDRSZ	sizeof(struct in6_addr)
 | 
						|
#define INETHADDRSZ	sizeof(net_ethaddr)
 | 
						|
 | 
						|
#define PROT_IP6	0x86DD	/* IPv6 protocol */
 | 
						|
#define PROT_ICMPV6	58	/* ICMPv6 protocol*/
 | 
						|
 | 
						|
#define IPV6_ADDRSCOPE_INTF	0x01
 | 
						|
#define IPV6_ADDRSCOPE_LINK	0x02
 | 
						|
#define IPV6_ADDRSCOPE_AMDIN	0x04
 | 
						|
#define IPV6_ADDRSCOPE_SITE	0x05
 | 
						|
#define IPV6_ADDRSCOPE_ORG	0x08
 | 
						|
#define IPV6_ADDRSCOPE_GLOBAL	0x0E
 | 
						|
 | 
						|
#define USE_IP6_CMD_PARAM	"-ipv6"
 | 
						|
 | 
						|
/**
 | 
						|
 * struct ipv6hdr - Internet Protocol V6 (IPv6) header.
 | 
						|
 *
 | 
						|
 * IPv6 packet header as defined in RFC 2460.
 | 
						|
 */
 | 
						|
struct ip6_hdr {
 | 
						|
#if defined(__LITTLE_ENDIAN_BITFIELD)
 | 
						|
	u8	priority:4,
 | 
						|
		version:4;
 | 
						|
#elif defined(__BIG_ENDIAN_BITFIELD)
 | 
						|
	u8	version:4,
 | 
						|
		priority:4;
 | 
						|
#else
 | 
						|
#error  "Please fix <asm/byteorder.h>"
 | 
						|
#endif
 | 
						|
	u8		flow_lbl[3];
 | 
						|
	__be16		payload_len;
 | 
						|
	u8		nexthdr;
 | 
						|
	u8		hop_limit;
 | 
						|
	struct in6_addr	saddr;
 | 
						|
	struct in6_addr	daddr;
 | 
						|
} __packed;
 | 
						|
#define IP6_HDR_SIZE (sizeof(struct ip6_hdr))
 | 
						|
 | 
						|
/* struct udp_hdr - User Datagram Protocol header */
 | 
						|
struct udp_hdr {
 | 
						|
	u16		udp_src;	/* UDP source port		*/
 | 
						|
	u16		udp_dst;	/* UDP destination port		*/
 | 
						|
	u16		udp_len;	/* Length of UDP packet		*/
 | 
						|
	u16		udp_xsum;	/* Checksum			*/
 | 
						|
} __packed;
 | 
						|
 | 
						|
/*
 | 
						|
 * Handy for static initialisations of struct in6_addr, atlhough the
 | 
						|
 * c99 '= { 0 }' idiom might work depending on you compiler.
 | 
						|
 */
 | 
						|
#define ZERO_IPV6_ADDR { { { 0x00, 0x00, 0x00, 0x00, \
 | 
						|
			  0x00, 0x00, 0x00, 0x00, \
 | 
						|
			  0x00, 0x00, 0x00, 0x00, \
 | 
						|
			  0x00, 0x00, 0x00, 0x00 } } }
 | 
						|
 | 
						|
#define IPV6_LINK_LOCAL_PREFIX	0xfe80
 | 
						|
 | 
						|
/* hop limit for neighbour discovery packets */
 | 
						|
#define IPV6_NDISC_HOPLIMIT             255
 | 
						|
#define NDISC_TIMEOUT			5000UL
 | 
						|
#define NDISC_TIMEOUT_COUNT             3
 | 
						|
 | 
						|
/* struct icmp6hdr - Internet Control Message Protocol header for IPV6 */
 | 
						|
struct icmp6hdr {
 | 
						|
	u8	icmp6_type;
 | 
						|
#define IPV6_ICMP_ECHO_REQUEST			128
 | 
						|
#define IPV6_ICMP_ECHO_REPLY			129
 | 
						|
#define IPV6_NDISC_ROUTER_SOLICITATION		133
 | 
						|
#define IPV6_NDISC_ROUTER_ADVERTISEMENT		134
 | 
						|
#define IPV6_NDISC_NEIGHBOUR_SOLICITATION	135
 | 
						|
#define IPV6_NDISC_NEIGHBOUR_ADVERTISEMENT	136
 | 
						|
#define IPV6_NDISC_REDIRECT			137
 | 
						|
	u8	icmp6_code;
 | 
						|
	__be16	icmp6_cksum;
 | 
						|
 | 
						|
	/* ICMPv6 data */
 | 
						|
	union {
 | 
						|
		__be32	un_data32[1];
 | 
						|
		__be16	un_data16[2];
 | 
						|
		u8	un_data8[4];
 | 
						|
 | 
						|
		/* struct icmpv6_echo - echo request/reply message format */
 | 
						|
		struct icmpv6_echo {
 | 
						|
			__be16		identifier;
 | 
						|
			__be16		sequence;
 | 
						|
		} u_echo;
 | 
						|
 | 
						|
		/* struct icmpv6_nd_advt - Neighbor Advertisement format */
 | 
						|
		struct icmpv6_nd_advt {
 | 
						|
#if defined(__LITTLE_ENDIAN_BITFIELD)
 | 
						|
			__be32		reserved:5,
 | 
						|
					override:1,
 | 
						|
					solicited:1,
 | 
						|
					router:1,
 | 
						|
					reserved2:24;
 | 
						|
#elif defined(__BIG_ENDIAN_BITFIELD)
 | 
						|
			__be32		router:1,
 | 
						|
					solicited:1,
 | 
						|
					override:1,
 | 
						|
					reserved:29;
 | 
						|
#else
 | 
						|
#error	"Please fix <asm/byteorder.h>"
 | 
						|
#endif
 | 
						|
		} u_nd_advt;
 | 
						|
 | 
						|
		/* struct icmpv6_nd_ra - Router Advertisement format */
 | 
						|
		struct icmpv6_nd_ra {
 | 
						|
			u8		hop_limit;
 | 
						|
#if defined(__LITTLE_ENDIAN_BITFIELD)
 | 
						|
			u8		reserved:6,
 | 
						|
					other:1,
 | 
						|
					managed:1;
 | 
						|
 | 
						|
#elif defined(__BIG_ENDIAN_BITFIELD)
 | 
						|
			u8		managed:1,
 | 
						|
					other:1,
 | 
						|
					reserved:6;
 | 
						|
#else
 | 
						|
#error	"Please fix <asm/byteorder.h>"
 | 
						|
#endif
 | 
						|
			__be16		rt_lifetime;
 | 
						|
		} u_nd_ra;
 | 
						|
	} icmp6_dataun;
 | 
						|
#define icmp6_identifier	icmp6_dataun.u_echo.identifier
 | 
						|
#define icmp6_sequence		icmp6_dataun.u_echo.sequence
 | 
						|
#define icmp6_pointer		icmp6_dataun.un_data32[0]
 | 
						|
#define icmp6_mtu		icmp6_dataun.un_data32[0]
 | 
						|
#define icmp6_unused		icmp6_dataun.un_data32[0]
 | 
						|
#define icmp6_maxdelay		icmp6_dataun.un_data16[0]
 | 
						|
#define icmp6_router		icmp6_dataun.u_nd_advt.router
 | 
						|
#define icmp6_solicited		icmp6_dataun.u_nd_advt.solicited
 | 
						|
#define icmp6_override		icmp6_dataun.u_nd_advt.override
 | 
						|
#define icmp6_ndiscreserved	icmp6_dataun.u_nd_advt.reserved
 | 
						|
#define icmp6_hop_limit		icmp6_dataun.u_nd_ra.hop_limit
 | 
						|
#define icmp6_addrconf_managed	icmp6_dataun.u_nd_ra.managed
 | 
						|
#define icmp6_addrconf_other	icmp6_dataun.u_nd_ra.other
 | 
						|
#define icmp6_rt_lifetime	icmp6_dataun.u_nd_ra.rt_lifetime
 | 
						|
} __packed;
 | 
						|
 | 
						|
extern struct in6_addr const net_null_addr_ip6;	/* NULL IPv6 address */
 | 
						|
extern struct in6_addr net_gateway6;	/* Our gateways IPv6 address */
 | 
						|
extern struct in6_addr net_ip6;	/* Our IPv6 addr (0 = unknown) */
 | 
						|
extern struct in6_addr net_link_local_ip6;	/* Our link local IPv6 addr */
 | 
						|
extern u32 net_prefix_length;	/* Our prefixlength (0 = unknown) */
 | 
						|
extern struct in6_addr net_server_ip6;	/* Server IPv6 addr (0 = unknown) */
 | 
						|
extern struct in6_addr net_ping_ip6; /* the ipv6 address to ping */
 | 
						|
extern bool use_ip6;
 | 
						|
 | 
						|
#if IS_ENABLED(CONFIG_IPV6)
 | 
						|
/**
 | 
						|
 * string_to_ip6() - Convert IPv6 string addr to inner IPV6 addr format
 | 
						|
 *
 | 
						|
 * Examples of valid strings:
 | 
						|
 *	2001:db8::0:1234:1
 | 
						|
 *	2001:0db8:0000:0000:0000:0000:1234:0001
 | 
						|
 *	::1
 | 
						|
 *	::ffff:192.168.1.1
 | 
						|
 *
 | 
						|
 * Examples of invalid strings
 | 
						|
 *	2001:db8::0::0          (:: can only appear once)
 | 
						|
 *	2001:db8:192.168.1.1::1 (v4 part can only appear at the end)
 | 
						|
 *	192.168.1.1             (we don't implicity map v4)
 | 
						|
 *
 | 
						|
 * @s:		IPv6 string addr format
 | 
						|
 * @len:	IPv6 string addr length
 | 
						|
 * @addr:	converted IPv6 addr
 | 
						|
 * Return: 0 if conversion successful, -EINVAL if fail
 | 
						|
 */
 | 
						|
int string_to_ip6(const char *s, size_t len, struct in6_addr *addr);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_is_unspecified_addr() - Check if IPv6 addr is not set i.e. is zero
 | 
						|
 *
 | 
						|
 * @addr:	IPv6 addr
 | 
						|
 * Return:  0 if addr is not set, -1 if is set
 | 
						|
 */
 | 
						|
int ip6_is_unspecified_addr(struct in6_addr *addr);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_is_our_addr() - Check if IPv6 addr belongs to our host addr
 | 
						|
 *
 | 
						|
 * We have 2 addresses that we should respond to. A link local address and a
 | 
						|
 * global address. This returns true if the specified address matches either
 | 
						|
 * of these.
 | 
						|
 *
 | 
						|
 * @addr:	addr to check
 | 
						|
 * Return: 0 if addr is our, -1 otherwise
 | 
						|
 */
 | 
						|
int ip6_is_our_addr(struct in6_addr *addr);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_addr_in_subnet() - Check if two IPv6 addresses are in the same subnet
 | 
						|
 *
 | 
						|
 * @our_addr:		first IPv6 addr
 | 
						|
 * @neigh_addr:		second IPv6 addr
 | 
						|
 * @prefix_length:	network mask length
 | 
						|
 * Return: 0 if two addresses in the same subnet, -1 otherwise
 | 
						|
 */
 | 
						|
int ip6_addr_in_subnet(struct in6_addr *our_addr, struct in6_addr *neigh_addr,
 | 
						|
		       u32 prefix_length);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_make_lladd() - rMake up IPv6 Link Local address
 | 
						|
 *
 | 
						|
 * @lladdr:	formed IPv6 Link Local address
 | 
						|
 * @enetaddr:	MAC addr of a device
 | 
						|
 */
 | 
						|
void ip6_make_lladdr(struct in6_addr *lladr, unsigned char const enetaddr[6]);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_make_snma() - aMake up Solicited Node Multicast Address from IPv6 addr
 | 
						|
 *
 | 
						|
 * @mcast_addr:	formed SNMA addr
 | 
						|
 * @ip6_addr:	base IPv6 addr
 | 
						|
 */
 | 
						|
void ip6_make_snma(struct in6_addr *mcast_addr, struct in6_addr *ip6_addr);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_make_mult_ethdstaddr() - Make up IPv6 multicast addr
 | 
						|
 *
 | 
						|
 * @enetaddr:	MAC addr of a device
 | 
						|
 * @mcast_addr:	formed IPv6 multicast addr
 | 
						|
 */
 | 
						|
void ip6_make_mult_ethdstaddr(unsigned char enetaddr[6],
 | 
						|
			      struct in6_addr *mcast_addr);
 | 
						|
 | 
						|
/**
 | 
						|
 * csum_partial() - Compute an internet checksum
 | 
						|
 *
 | 
						|
 * @buff:	buffer to be checksummed
 | 
						|
 * @len:	length of buffer
 | 
						|
 * @sum:	initial sum to be added in
 | 
						|
 * Return: internet checksum of the buffer
 | 
						|
 */
 | 
						|
unsigned int csum_partial(const unsigned char *buff, int len, unsigned int sum);
 | 
						|
 | 
						|
/**
 | 
						|
 * csum_ipv6_magic() - Compute checksum of IPv6 "psuedo-header" per RFC2460 section 8.1
 | 
						|
 *
 | 
						|
 * @saddr:	source IPv6 addr
 | 
						|
 * @daddr:	destination IPv6 add
 | 
						|
 * @len:	data length to be checksummed
 | 
						|
 * @proto:	IPv6 above protocol code
 | 
						|
 * @csum:	upper layer checksum
 | 
						|
 * Return: computed checksum
 | 
						|
 */
 | 
						|
unsigned short int csum_ipv6_magic(struct in6_addr *saddr,
 | 
						|
				   struct in6_addr *daddr, u16 len,
 | 
						|
				   unsigned short proto, unsigned int csum);
 | 
						|
 | 
						|
/**
 | 
						|
 * ip6_add_hdr() - Make up IPv6 header
 | 
						|
 *
 | 
						|
 * @xip:	pointer to IPv6 header to be formed
 | 
						|
 * @src:	source IPv6 addr
 | 
						|
 * @dest:	destination IPv6 addr
 | 
						|
 * @nextheader:	next header type
 | 
						|
 * @hoplimit:	hop limit
 | 
						|
 * @payload_len: payload length
 | 
						|
 * Return: IPv6 header length
 | 
						|
 */
 | 
						|
int ip6_add_hdr(uchar *xip, struct in6_addr *src, struct in6_addr *dest,
 | 
						|
		int nextheader, int hoplimit, int payload_len);
 | 
						|
 | 
						|
/**
 | 
						|
 * net_send_udp_packet6() - Make up UDP packet and send it
 | 
						|
 *
 | 
						|
 * @ether:	destination MAC addr
 | 
						|
 * @dest:	destination IPv6 addr
 | 
						|
 * @dport:	destination port
 | 
						|
 * @sport:	source port
 | 
						|
 * @len:	UDP packet length
 | 
						|
 * Return: 0 if send successfully, -1 otherwise
 | 
						|
 */
 | 
						|
int net_send_udp_packet6(uchar *ether, struct in6_addr *dest, int dport,
 | 
						|
			 int sport, int len);
 | 
						|
 | 
						|
/**
 | 
						|
 * net_ip6_handler() - Handle IPv6 packet
 | 
						|
 *
 | 
						|
 * @et:		pointer to the beginning of the packet
 | 
						|
 * @ip6:	pointer to the beginning of IPv6 protocol
 | 
						|
 * @len:	incoming packet len
 | 
						|
 * Return: 0 if handle packet successfully, -EINVAL in case of invalid protocol
 | 
						|
 */
 | 
						|
int net_ip6_handler(struct ethernet_hdr *et, struct ip6_hdr *ip6, int len);
 | 
						|
 | 
						|
/**
 | 
						|
 * net_copy_ip6() - Copy IPv6 addr
 | 
						|
 *
 | 
						|
 * @to:		destination IPv6 addr
 | 
						|
 * @from:	source IPv6 addr
 | 
						|
 */
 | 
						|
static inline void net_copy_ip6(void *to, const void *from)
 | 
						|
{
 | 
						|
	memcpy((void *)to, from, sizeof(struct in6_addr));
 | 
						|
}
 | 
						|
#else
 | 
						|
static inline int
 | 
						|
string_to_ip6(const char *s, size_t len, struct in6_addr *addr)
 | 
						|
{
 | 
						|
	return -EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
static inline int ip6_is_unspecified_addr(struct in6_addr *addr)
 | 
						|
{
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
static inline int ip6_is_our_addr(struct in6_addr *addr)
 | 
						|
{
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
static inline int
 | 
						|
ip6_addr_in_subnet(struct in6_addr *our_addr, struct in6_addr *neigh_addr,
 | 
						|
		   u32 prefix_length)
 | 
						|
{
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
static inline void
 | 
						|
ip6_make_lladdr(struct in6_addr *lladdr, unsigned char const enetaddr[6])
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static inline void
 | 
						|
ip6_make_snma(struct in6_addr *mcast_addr, struct in6_addr *ip6_addr)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static inline void
 | 
						|
ip6_make_mult_ethdstaddr(unsigned char enetaddr[6],
 | 
						|
			 struct in6_addr *mcast_addr)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static inline unsigned int
 | 
						|
csum_partial(const unsigned char *buff, int len, unsigned int sum)
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline unsigned short
 | 
						|
csum_ipv6_magic(struct in6_addr *saddr,
 | 
						|
		struct in6_addr *daddr, u16 len,
 | 
						|
		unsigned short proto, unsigned int csum)
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline unsigned int
 | 
						|
ip6_add_hdr(uchar *xip, struct in6_addr *src, struct in6_addr *dest,
 | 
						|
	    int nextheader, int hoplimit, int payload_len)
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline int
 | 
						|
net_send_udp_packet6(uchar *ether, struct in6_addr *dest,
 | 
						|
		     int dport, int sport, int len)
 | 
						|
{
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
static inline int
 | 
						|
net_ip6_handler(struct ethernet_hdr *et, struct ip6_hdr *ip6,
 | 
						|
		int len)
 | 
						|
{
 | 
						|
	return -EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
static inline void net_copy_ip6(void *to, const void *from)
 | 
						|
{
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#if IS_ENABLED(CONFIG_CMD_PING6)
 | 
						|
/* Send ping requset */
 | 
						|
void ping6_start(void);
 | 
						|
 | 
						|
/**
 | 
						|
 * ping6_receive() - Handle reception of ICMPv6 echo request/reply
 | 
						|
 *
 | 
						|
 * @et:		pointer to incoming patcket
 | 
						|
 * @ip6:	pointer to IPv6 protocol
 | 
						|
 * @len:	packet length
 | 
						|
 * Return: 0 if success, -EINVAL in case of failure during reception
 | 
						|
 */
 | 
						|
int ping6_receive(struct ethernet_hdr *et, struct ip6_hdr *ip6, int len);
 | 
						|
#else
 | 
						|
static inline void ping6_start(void)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static inline
 | 
						|
int ping6_receive(struct ethernet_hdr *et, struct ip6_hdr *ip6, int len)
 | 
						|
{
 | 
						|
	return -EINVAL;
 | 
						|
}
 | 
						|
#endif /* CONFIG_CMD_PING6 */
 | 
						|
 | 
						|
#endif /* __NET6_H__ */
 |