mirror of
				https://git.proxmox.com/git/mirror_frr
				synced 2025-11-04 09:45:13 +00:00 
			
		
		
		
	Introduce a new command "interface IFNAME vrf N" to configure an interface in the non-default VRF. Till now, only zebra uses this command. Other daemons will install the command when they support multiple VRFs. Signed-off-by: Feng Lu <lu.feng@6wind.com> Reviewed-by: Alain Ritoux <alain.ritoux@6wind.com> Signed-off-by: Nicolas Dichtel <nicolas.dichtel@6wind.com> Acked-by: Vincent JARDIN <vincent.jardin@6wind.com> Signed-off-by: David Lamparter <equinox@opensourcerouting.org> Conflicts: zebra/interface.c
		
			
				
	
	
		
			382 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			382 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* Interface related header.
 | 
						|
   Copyright (C) 1997, 98, 99 Kunihiro Ishiguro
 | 
						|
 | 
						|
This file is part of GNU Zebra.
 | 
						|
 | 
						|
GNU Zebra is free software; you can redistribute it and/or modify
 | 
						|
it under the terms of the GNU General Public License as published
 | 
						|
by the Free Software Foundation; either version 2, or (at your
 | 
						|
option) any later version.
 | 
						|
 | 
						|
GNU Zebra is distributed in the hope that it will be useful, but
 | 
						|
WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | 
						|
General Public License for more details.
 | 
						|
 | 
						|
You should have received a copy of the GNU General Public License
 | 
						|
along with GNU Zebra; see the file COPYING.  If not, write to the
 | 
						|
Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 | 
						|
Boston, MA 02111-1307, USA.  */
 | 
						|
 | 
						|
#ifndef _ZEBRA_IF_H
 | 
						|
#define _ZEBRA_IF_H
 | 
						|
 | 
						|
#include "linklist.h"
 | 
						|
 | 
						|
/*
 | 
						|
  Interface name length.
 | 
						|
 | 
						|
   Linux define value in /usr/include/linux/if.h.
 | 
						|
   #define IFNAMSIZ        16
 | 
						|
 | 
						|
   FreeBSD define value in /usr/include/net/if.h.
 | 
						|
   #define IFNAMSIZ        16
 | 
						|
*/
 | 
						|
 | 
						|
#define INTERFACE_NAMSIZ      20
 | 
						|
#define INTERFACE_HWADDR_MAX  20
 | 
						|
 | 
						|
#ifdef HAVE_PROC_NET_DEV
 | 
						|
struct if_stats
 | 
						|
{
 | 
						|
  unsigned long rx_packets;   /* total packets received       */
 | 
						|
  unsigned long tx_packets;   /* total packets transmitted    */
 | 
						|
  unsigned long rx_bytes;     /* total bytes received         */
 | 
						|
  unsigned long tx_bytes;     /* total bytes transmitted      */
 | 
						|
  unsigned long rx_errors;    /* bad packets received         */
 | 
						|
  unsigned long tx_errors;    /* packet transmit problems     */
 | 
						|
  unsigned long rx_dropped;   /* no space in linux buffers    */
 | 
						|
  unsigned long tx_dropped;   /* no space available in linux  */
 | 
						|
  unsigned long rx_multicast; /* multicast packets received   */
 | 
						|
  unsigned long rx_compressed;
 | 
						|
  unsigned long tx_compressed;
 | 
						|
  unsigned long collisions;
 | 
						|
 | 
						|
  /* detailed rx_errors: */
 | 
						|
  unsigned long rx_length_errors;
 | 
						|
  unsigned long rx_over_errors;       /* receiver ring buff overflow  */
 | 
						|
  unsigned long rx_crc_errors;        /* recved pkt with crc error    */
 | 
						|
  unsigned long rx_frame_errors;      /* recv'd frame alignment error */
 | 
						|
  unsigned long rx_fifo_errors;       /* recv'r fifo overrun          */
 | 
						|
  unsigned long rx_missed_errors;     /* receiver missed packet     */
 | 
						|
  /* detailed tx_errors */
 | 
						|
  unsigned long tx_aborted_errors;
 | 
						|
  unsigned long tx_carrier_errors;
 | 
						|
  unsigned long tx_fifo_errors;
 | 
						|
  unsigned long tx_heartbeat_errors;
 | 
						|
  unsigned long tx_window_errors;
 | 
						|
};
 | 
						|
#endif /* HAVE_PROC_NET_DEV */
 | 
						|
 | 
						|
/* Interface structure */
 | 
						|
struct interface 
 | 
						|
{
 | 
						|
  /* Interface name.  This should probably never be changed after the
 | 
						|
     interface is created, because the configuration info for this interface
 | 
						|
     is associated with this structure.  For that reason, the interface
 | 
						|
     should also never be deleted (to avoid losing configuration info).
 | 
						|
     To delete, just set ifindex to IFINDEX_INTERNAL to indicate that the
 | 
						|
     interface does not exist in the kernel.
 | 
						|
   */
 | 
						|
  char name[INTERFACE_NAMSIZ + 1];
 | 
						|
 | 
						|
  /* Interface index (should be IFINDEX_INTERNAL for non-kernel or
 | 
						|
     deleted interfaces). */
 | 
						|
  unsigned int ifindex;
 | 
						|
#define IFINDEX_INTERNAL	0
 | 
						|
 | 
						|
  /* Zebra internal interface status */
 | 
						|
  u_char status;
 | 
						|
#define ZEBRA_INTERFACE_ACTIVE     (1 << 0)
 | 
						|
#define ZEBRA_INTERFACE_SUB        (1 << 1)
 | 
						|
#define ZEBRA_INTERFACE_LINKDETECTION (1 << 2)
 | 
						|
 | 
						|
  /* Interface flags. */
 | 
						|
  uint64_t flags;
 | 
						|
 | 
						|
  /* Interface metric */
 | 
						|
  int metric;
 | 
						|
 | 
						|
  /* Interface MTU. */
 | 
						|
  unsigned int mtu;    /* IPv4 MTU */
 | 
						|
  unsigned int mtu6;   /* IPv6 MTU - probably, but not neccessarily same as mtu */
 | 
						|
 | 
						|
  /* Hardware address. */
 | 
						|
#ifdef HAVE_STRUCT_SOCKADDR_DL
 | 
						|
  union {
 | 
						|
    /* note that sdl_storage is never accessed, it only exists to make space.
 | 
						|
     * all actual uses refer to sdl - but use sizeof(sdl_storage)!  this fits
 | 
						|
     * best with C aliasing rules. */
 | 
						|
    struct sockaddr_dl sdl;
 | 
						|
    struct sockaddr_storage sdl_storage;
 | 
						|
  };
 | 
						|
#else
 | 
						|
  unsigned short hw_type;
 | 
						|
  u_char hw_addr[INTERFACE_HWADDR_MAX];
 | 
						|
  int hw_addr_len;
 | 
						|
#endif /* HAVE_STRUCT_SOCKADDR_DL */
 | 
						|
 | 
						|
  /* interface bandwidth, kbits */
 | 
						|
  unsigned int bandwidth;
 | 
						|
  
 | 
						|
  /* description of the interface. */
 | 
						|
  char *desc;			
 | 
						|
 | 
						|
  /* Distribute list. */
 | 
						|
  void *distribute_in;
 | 
						|
  void *distribute_out;
 | 
						|
 | 
						|
  /* Connected address list. */
 | 
						|
  struct list *connected;
 | 
						|
 | 
						|
  /* Neighbor connected address list. */
 | 
						|
  struct list *nbr_connected;
 | 
						|
 | 
						|
  /* Daemon specific interface data pointer. */
 | 
						|
  void *info;
 | 
						|
 | 
						|
  char ptm_enable;             /* Should we look at ptm_status ? */
 | 
						|
  char ptm_status;
 | 
						|
 | 
						|
  /* Statistics fileds. */
 | 
						|
#ifdef HAVE_PROC_NET_DEV
 | 
						|
  struct if_stats stats;
 | 
						|
#endif /* HAVE_PROC_NET_DEV */  
 | 
						|
#ifdef HAVE_NET_RT_IFLIST
 | 
						|
  struct if_data stats;
 | 
						|
#endif /* HAVE_NET_RT_IFLIST */
 | 
						|
 | 
						|
  vrf_id_t vrf_id;
 | 
						|
};
 | 
						|
 | 
						|
/* Connected address structure. */
 | 
						|
struct connected
 | 
						|
{
 | 
						|
  /* Attached interface. */
 | 
						|
  struct interface *ifp;
 | 
						|
 | 
						|
  /* Flags for configuration. */
 | 
						|
  u_char conf;
 | 
						|
#define ZEBRA_IFC_REAL         (1 << 0)
 | 
						|
#define ZEBRA_IFC_CONFIGURED   (1 << 1)
 | 
						|
#define ZEBRA_IFC_QUEUED       (1 << 2)
 | 
						|
  /*
 | 
						|
     The ZEBRA_IFC_REAL flag should be set if and only if this address
 | 
						|
     exists in the kernel and is actually usable. (A case where it exists but
 | 
						|
     is not yet usable would be IPv6 with DAD)
 | 
						|
     The ZEBRA_IFC_CONFIGURED flag should be set if and only if this address
 | 
						|
     was configured by the user from inside quagga.
 | 
						|
     The ZEBRA_IFC_QUEUED flag should be set if and only if the address exists
 | 
						|
     in the kernel. It may and should be set although the address might not be
 | 
						|
     usable yet. (compare with ZEBRA_IFC_REAL)
 | 
						|
   */
 | 
						|
 | 
						|
  /* Flags for connected address. */
 | 
						|
  u_char flags;
 | 
						|
#define ZEBRA_IFA_SECONDARY    (1 << 0)
 | 
						|
#define ZEBRA_IFA_PEER         (1 << 1)
 | 
						|
#define ZEBRA_IFA_UNNUMBERED   (1 << 2)
 | 
						|
  /* N.B. the ZEBRA_IFA_PEER flag should be set if and only if
 | 
						|
     a peer address has been configured.  If this flag is set,
 | 
						|
     the destination field must contain the peer address.  
 | 
						|
     Otherwise, if this flag is not set, the destination address
 | 
						|
     will either contain a broadcast address or be NULL.
 | 
						|
   */
 | 
						|
 | 
						|
  /* Address of connected network. */
 | 
						|
  struct prefix *address;
 | 
						|
 | 
						|
  /* Peer or Broadcast address, depending on whether ZEBRA_IFA_PEER is set.
 | 
						|
     Note: destination may be NULL if ZEBRA_IFA_PEER is not set. */
 | 
						|
  struct prefix *destination;
 | 
						|
 | 
						|
  /* A list of unnumbered IFCs borrowing the address from me */
 | 
						|
  struct list *unnumbered;
 | 
						|
 | 
						|
  /* Pointer to the anchor IFC if I'm unnumbered */
 | 
						|
  struct connected *anchor;
 | 
						|
 | 
						|
  /* Label for Linux 2.2.X and upper. */
 | 
						|
  char *label;
 | 
						|
};
 | 
						|
 | 
						|
/* Nbr Connected address structure. */
 | 
						|
struct nbr_connected
 | 
						|
{
 | 
						|
  /* Attached interface. */
 | 
						|
  struct interface *ifp;
 | 
						|
 | 
						|
  /* Address of connected network. */
 | 
						|
  struct prefix *address;
 | 
						|
};
 | 
						|
 | 
						|
/* Does the destination field contain a peer address? */
 | 
						|
#define CONNECTED_PEER(C) CHECK_FLAG((C)->flags, ZEBRA_IFA_PEER)
 | 
						|
 | 
						|
/* Prefix to insert into the RIB */
 | 
						|
#define CONNECTED_PREFIX(C) \
 | 
						|
	(CONNECTED_PEER(C) ? (C)->destination : (C)->address)
 | 
						|
 | 
						|
/* Identifying address.  We guess that if there's a peer address, but the
 | 
						|
   local address is in the same prefix, then the local address may be unique. */
 | 
						|
#define CONNECTED_ID(C)	\
 | 
						|
	((CONNECTED_PEER(C) && !prefix_match((C)->destination, (C)->address)) ?\
 | 
						|
	 (C)->destination : (C)->address)
 | 
						|
 | 
						|
/* Interface hook sort. */
 | 
						|
#define IF_NEW_HOOK   0
 | 
						|
#define IF_DELETE_HOOK 1
 | 
						|
 | 
						|
/* There are some interface flags which are only supported by some
 | 
						|
   operating system. */
 | 
						|
 | 
						|
#ifndef IFF_NOTRAILERS
 | 
						|
#define IFF_NOTRAILERS 0x0
 | 
						|
#endif /* IFF_NOTRAILERS */
 | 
						|
#ifndef IFF_OACTIVE
 | 
						|
#define IFF_OACTIVE 0x0
 | 
						|
#endif /* IFF_OACTIVE */
 | 
						|
#ifndef IFF_SIMPLEX
 | 
						|
#define IFF_SIMPLEX 0x0
 | 
						|
#endif /* IFF_SIMPLEX */
 | 
						|
#ifndef IFF_LINK0
 | 
						|
#define IFF_LINK0 0x0
 | 
						|
#endif /* IFF_LINK0 */
 | 
						|
#ifndef IFF_LINK1
 | 
						|
#define IFF_LINK1 0x0
 | 
						|
#endif /* IFF_LINK1 */
 | 
						|
#ifndef IFF_LINK2
 | 
						|
#define IFF_LINK2 0x0
 | 
						|
#endif /* IFF_LINK2 */
 | 
						|
#ifndef IFF_NOXMIT
 | 
						|
#define IFF_NOXMIT 0x0
 | 
						|
#endif /* IFF_NOXMIT */
 | 
						|
#ifndef IFF_NORTEXCH
 | 
						|
#define IFF_NORTEXCH 0x0
 | 
						|
#endif /* IFF_NORTEXCH */
 | 
						|
#ifndef IFF_IPV4
 | 
						|
#define IFF_IPV4 0x0
 | 
						|
#endif /* IFF_IPV4 */
 | 
						|
#ifndef IFF_IPV6
 | 
						|
#define IFF_IPV6 0x0
 | 
						|
#endif /* IFF_IPV6 */
 | 
						|
#ifndef IFF_VIRTUAL
 | 
						|
#define IFF_VIRTUAL 0x0
 | 
						|
#endif /* IFF_VIRTUAL */
 | 
						|
 | 
						|
/* Prototypes. */
 | 
						|
extern int if_cmp_func (struct interface *, struct interface *);
 | 
						|
extern struct interface *if_create (const char *name, int namelen);
 | 
						|
extern struct interface *if_lookup_by_index (unsigned int);
 | 
						|
extern struct interface *if_lookup_exact_address (void *matchaddr, int family);
 | 
						|
extern struct interface *if_lookup_address (void *matchaddr, int family);
 | 
						|
extern struct interface *if_lookup_prefix (struct prefix *prefix);
 | 
						|
extern struct connected *if_anchor_lookup_by_address (struct in_addr src);
 | 
						|
 | 
						|
extern struct interface *if_create_vrf (const char *name, int namelen,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
extern struct interface *if_lookup_by_index_vrf (unsigned int,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
extern struct interface *if_lookup_exact_address_vrf (void *matchaddr, int family,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
extern struct interface *if_lookup_address_vrf (void *matchaddr, int family,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
extern struct interface *if_lookup_prefix_vrf (struct prefix *prefix,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
 | 
						|
/* These 2 functions are to be used when the ifname argument is terminated
 | 
						|
   by a '\0' character: */
 | 
						|
extern struct interface *if_lookup_by_name (const char *ifname);
 | 
						|
extern struct interface *if_get_by_name (const char *ifname);
 | 
						|
 | 
						|
extern struct interface *if_lookup_by_name_vrf (const char *ifname,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
extern struct interface *if_get_by_name_vrf (const char *ifname,
 | 
						|
                                vrf_id_t vrf_id);
 | 
						|
 | 
						|
/* For these 2 functions, the namelen argument should be the precise length
 | 
						|
   of the ifname string (not counting any optional trailing '\0' character).
 | 
						|
   In most cases, strnlen should be used to calculate the namelen value. */
 | 
						|
extern struct interface *if_lookup_by_name_len(const char *ifname,
 | 
						|
                                              size_t namelen);
 | 
						|
extern struct interface *if_get_by_name_len(const char *ifname,size_t namelen);
 | 
						|
 | 
						|
extern struct interface *if_lookup_by_name_len_vrf(const char *ifname,
 | 
						|
                                size_t namelen, vrf_id_t vrf_id);
 | 
						|
extern struct interface *if_get_by_name_len_vrf(const char *ifname,
 | 
						|
                                size_t namelen, vrf_id_t vrf_id);
 | 
						|
 | 
						|
 | 
						|
/* Delete the interface, but do not free the structure, and leave it in the
 | 
						|
   interface list.  It is often advisable to leave the pseudo interface 
 | 
						|
   structure because there may be configuration information attached. */
 | 
						|
extern void if_delete_retain (struct interface *);
 | 
						|
 | 
						|
/* Delete and free the interface structure: calls if_delete_retain and then
 | 
						|
   deletes it from the interface list and frees the structure. */
 | 
						|
extern void if_delete (struct interface *);
 | 
						|
 | 
						|
extern int if_is_up (struct interface *);
 | 
						|
extern int if_is_running (struct interface *);
 | 
						|
extern int if_is_operative (struct interface *);
 | 
						|
extern int if_is_no_ptm_operative (struct interface *);
 | 
						|
extern int if_is_loopback (struct interface *);
 | 
						|
extern int if_is_broadcast (struct interface *);
 | 
						|
extern int if_is_pointopoint (struct interface *);
 | 
						|
extern int if_is_multicast (struct interface *);
 | 
						|
extern void if_add_hook (int, int (*)(struct interface *));
 | 
						|
extern void if_init (vrf_id_t, struct list **);
 | 
						|
extern void if_terminate (vrf_id_t, struct list **);
 | 
						|
extern void if_dump_all (void);
 | 
						|
extern const char *if_flag_dump(unsigned long);
 | 
						|
 | 
						|
/* Please use ifindex2ifname instead of if_indextoname where possible;
 | 
						|
   ifindex2ifname uses internal interface info, whereas if_indextoname must
 | 
						|
   make a system call. */
 | 
						|
extern const char *ifindex2ifname (unsigned int);
 | 
						|
extern const char *ifindex2ifname_vrf (unsigned int, vrf_id_t vrf_id);
 | 
						|
 | 
						|
/* Please use ifname2ifindex instead of if_nametoindex where possible;
 | 
						|
   ifname2ifindex uses internal interface info, whereas if_nametoindex must
 | 
						|
   make a system call. */
 | 
						|
extern unsigned int ifname2ifindex(const char *ifname);
 | 
						|
extern unsigned int ifname2ifindex_vrf(const char *ifname, vrf_id_t vrf_id);
 | 
						|
 | 
						|
/* Connected address functions. */
 | 
						|
extern struct connected *connected_new (void);
 | 
						|
extern void connected_free (struct connected *);
 | 
						|
extern void connected_add (struct interface *, struct connected *);
 | 
						|
extern struct connected  *connected_add_by_prefix (struct interface *,
 | 
						|
                                            struct prefix *,
 | 
						|
                                            struct prefix *);
 | 
						|
extern struct connected  *connected_delete_by_prefix (struct interface *, 
 | 
						|
                                               struct prefix *);
 | 
						|
extern struct connected  *connected_lookup_address (struct interface *, 
 | 
						|
                                             struct in_addr);
 | 
						|
extern struct nbr_connected *nbr_connected_new (void);
 | 
						|
extern void nbr_connected_free (struct nbr_connected *);
 | 
						|
struct nbr_connected *nbr_connected_check (struct interface *, struct prefix *);
 | 
						|
 | 
						|
#ifndef HAVE_IF_NAMETOINDEX
 | 
						|
extern unsigned int if_nametoindex (const char *);
 | 
						|
#endif
 | 
						|
#ifndef HAVE_IF_INDEXTONAME
 | 
						|
extern char *if_indextoname (unsigned int, char *);
 | 
						|
#endif
 | 
						|
 | 
						|
/* Exported variables. */
 | 
						|
extern struct list *iflist;
 | 
						|
extern struct cmd_element interface_desc_cmd;
 | 
						|
extern struct cmd_element no_interface_desc_cmd;
 | 
						|
extern struct cmd_element interface_cmd;
 | 
						|
extern struct cmd_element no_interface_cmd;
 | 
						|
extern struct cmd_element interface_vrf_cmd;
 | 
						|
extern struct cmd_element no_interface_vrf_cmd;
 | 
						|
extern struct cmd_element interface_pseudo_cmd;
 | 
						|
extern struct cmd_element no_interface_pseudo_cmd;
 | 
						|
extern struct cmd_element show_address_cmd;
 | 
						|
extern struct cmd_element show_address_vrf_cmd;
 | 
						|
extern struct cmd_element show_address_vrf_all_cmd;
 | 
						|
 | 
						|
#endif /* _ZEBRA_IF_H */
 |