⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 bgp_clist.c

📁 大名鼎鼎的路由器源码。程序分ZEBRA、OSPFRIP等3个包。程序框架采用一个路由协议一个进程的方式
💻 C
📖 第 1 页 / 共 2 页
字号:
/* BGP community-list and extcommunity-list.   Copyright (C) 1999 Kunihiro IshiguroThis file is part of GNU Zebra.GNU Zebra is free software; you can redistribute it and/or modify itunder the terms of the GNU General Public License as published by theFree Software Foundation; either version 2, or (at your option) anylater version.GNU Zebra is distributed in the hope that it will be useful, butWITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNUGeneral Public License for more details.You should have received a copy of the GNU General Public Licensealong with GNU Zebra; see the file COPYING.  If not, write to the FreeSoftware Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA02111-1307, USA.  */#include <zebra.h>#include "command.h"#include "prefix.h"#include "memory.h"#include "bgpd/bgpd.h"#include "bgpd/bgp_community.h"#include "bgpd/bgp_ecommunity.h"#include "bgpd/bgp_aspath.h"#include "bgpd/bgp_regex.h"#include "bgpd/bgp_clist.h"/* Lookup master structure for community-list or   extcommunity-list.  */struct community_list_master *community_list_master_lookup (struct community_list_handler *ch, int style){  if (ch)    switch (style)      {      case COMMUNITY_LIST_STANDARD:      case COMMUNITY_LIST_EXPANDED:      case COMMUNITY_LIST_AUTO:	return &ch->community_list;	break;      case EXTCOMMUNITY_LIST_STANDARD:      case EXTCOMMUNITY_LIST_EXPANDED:      case EXTCOMMUNITY_LIST_AUTO:	return &ch->extcommunity_list;      }  return NULL;}/* Allocate a new community list entry.  */struct community_entry *community_entry_new (){  struct community_entry *new;  new = XMALLOC (MTYPE_COMMUNITY_LIST_ENTRY, sizeof (struct community_entry));  memset (new, 0, sizeof (struct community_entry));  return new;}/* Free community list entry.  */voidcommunity_entry_free (struct community_entry *entry){  switch (entry->style)    {    case COMMUNITY_LIST_STANDARD:      if (entry->u.com)	community_free (entry->u.com);      break;    case EXTCOMMUNITY_LIST_STANDARD:      /* In case of standard extcommunity-list, configuration string	 is made by ecommunity_ecom2str().  */      if (entry->config)	XFREE (MTYPE_ECOMMUNITY_STR, entry->config);      if (entry->u.ecom)	ecommunity_free (entry->u.ecom);      break;    case COMMUNITY_LIST_EXPANDED:    case EXTCOMMUNITY_LIST_EXPANDED:      if (entry->config)	XFREE (MTYPE_COMMUNITY_LIST_CONFIG, entry->config);      if (entry->reg)	bgp_regex_free (entry->reg);    default:      break;    }  XFREE (MTYPE_COMMUNITY_LIST_ENTRY, entry);}/* Allocate a new community-list.  */struct community_list *community_list_new (){  struct community_list *new;  new = XMALLOC (MTYPE_COMMUNITY_LIST, sizeof (struct community_list));  memset (new, 0, sizeof (struct community_list));  return new;}/* Free community-list.  */voidcommunity_list_free (struct community_list *list){  if (list->name)    XFREE (MTYPE_COMMUNITY_LIST_NAME, list->name);  XFREE (MTYPE_COMMUNITY_LIST, list);}struct community_list *community_list_insert (struct community_list_handler *ch,		       char *name, int style){  int i;  long number;  struct community_list *new;  struct community_list *point;  struct community_list_list *list;  struct community_list_master *cm;  /* Lookup community-list master.  */  cm = community_list_master_lookup (ch, style);  if (! cm)    return NULL;  /* Allocate new community_list and copy given name. */  new = community_list_new ();  new->name = XSTRDUP (MTYPE_COMMUNITY_LIST_NAME, name);  /* If name is made by all digit character.  We treat it as     number. */  for (number = 0, i = 0; i < strlen (name); i++)    {      if (isdigit ((int) name[i]))	number = (number * 10) + (name[i] - '0');      else	break;    }  /* In case of name is all digit character */  if (i == strlen (name))    {      new->sort = COMMUNITY_LIST_NUMBER;      /* Set access_list to number list. */      list = &cm->num;      for (point = list->head; point; point = point->next)	if (atol (point->name) >= number)	  break;    }  else    {      new->sort = COMMUNITY_LIST_STRING;      /* Set access_list to string list. */      list = &cm->str;        /* Set point to insertion point. */      for (point = list->head; point; point = point->next)	if (strcmp (point->name, name) >= 0)	  break;    }  /* Link to upper list.  */  new->parent = list;  /* In case of this is the first element of master. */  if (list->head == NULL)    {      list->head = list->tail = new;      return new;    }  /* In case of insertion is made at the tail of access_list. */  if (point == NULL)    {      new->prev = list->tail;      list->tail->next = new;      list->tail = new;      return new;    }  /* In case of insertion is made at the head of access_list. */  if (point == list->head)    {      new->next = list->head;      list->head->prev = new;      list->head = new;      return new;    }  /* Insertion is made at middle of the access_list. */  new->next = point;  new->prev = point->prev;  if (point->prev)    point->prev->next = new;  point->prev = new;  return new;}struct community_list *community_list_lookup (struct community_list_handler *ch,		       char *name, int style){  struct community_list *list;  struct community_list_master *cm;  if (! name)    return NULL;  cm = community_list_master_lookup (ch, style);  if (! cm)    return NULL;  for (list = cm->num.head; list; list = list->next)    if (strcmp (list->name, name) == 0)      return list;  for (list = cm->str.head; list; list = list->next)    if (strcmp (list->name, name) == 0)      return list;  return NULL;}struct community_list *community_list_get (struct community_list_handler *ch, char *name, int style){  struct community_list *list;  list = community_list_lookup (ch, name, style);  if (! list)    list = community_list_insert (ch, name, style);  return list;}voidcommunity_list_delete (struct community_list *list){  struct community_list_list *clist;  struct community_entry *entry, *next;  for (entry = list->head; entry; entry = next)    {      next = entry->next;      community_entry_free (entry);    }  clist = list->parent;  if (list->next)    list->next->prev = list->prev;  else    clist->tail = list->prev;  if (list->prev)    list->prev->next = list->next;  else    clist->head = list->next;  community_list_free (list);}intcommunity_list_empty_p (struct community_list *list){  return (list->head == NULL && list->tail == NULL) ? 1 : 0;}/* Add community-list entry to the list.  */static voidcommunity_list_entry_add (struct community_list *list, 			  struct community_entry *entry){  entry->next = NULL;  entry->prev = list->tail;  if (list->tail)    list->tail->next = entry;  else    list->head = entry;  list->tail = entry;}/* Delete community-list entry from the list.  */static voidcommunity_list_entry_delete (struct community_list *list,			     struct community_entry *entry, int style){  if (entry->next)    entry->next->prev = entry->prev;  else    list->tail = entry->prev;  if (entry->prev)    entry->prev->next = entry->next;  else    list->head = entry->next;  community_entry_free (entry);  if (community_list_empty_p (list))    community_list_delete (list);}/* Lookup community-list entry from the list.  */static struct community_entry *community_list_entry_lookup (struct community_list *list, void *arg,			     int direct){  struct community_entry *entry;  for (entry = list->head; entry; entry = entry->next)    {      switch (entry->style)	{	case COMMUNITY_LIST_STANDARD:	  if (community_cmp (entry->u.com, arg))	    return entry;	  break;	case EXTCOMMUNITY_LIST_STANDARD:	  if (ecommunity_cmp (entry->u.ecom, arg))	    return entry;	  break;	case COMMUNITY_LIST_EXPANDED:	case EXTCOMMUNITY_LIST_EXPANDED:	  if (strcmp (entry->config, arg) == 0)	    return entry;	  break;	default:	  break;	}    }  return NULL;}/* Internal function to perform regular expression match for community   attribute.  */static intcommunity_regexp_match (struct community *com, regex_t *reg){  char *str;  /* When there is no communities attribute it is treated as empty     string.  */  if (com == NULL || com->size == 0)    str = "";  else    str = community_str (com);  /* Regular expression match.  */  if (regexec (reg, str, 0, NULL, 0) == 0)    return 1;  /* No match.  */  return 0;}static intecommunity_regexp_match (struct ecommunity *ecom, regex_t *reg){  char *str;  /* When there is no communities attribute it is treated as empty     string.  */  if (ecom == NULL || ecom->size == 0)    str = "";  else    str = ecommunity_str (ecom);  /* Regular expression match.  */  if (regexec (reg, str, 0, NULL, 0) == 0)    return 1;  /* No match.  */  return 0;}/* When given community attribute matches to the community-list return   1 else return 0.  */intcommunity_list_match (struct community *com, struct community_list *list){  struct community_entry *entry;  for (entry = list->head; entry; entry = entry->next)    {      if (entry->any)	return entry->direct == COMMUNITY_PERMIT ? 1 : 0;      if (entry->style == COMMUNITY_LIST_STANDARD)	{	  if (community_include (entry->u.com, COMMUNITY_INTERNET))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	  if (community_match (com, entry->u.com))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	}      else if (entry->style == COMMUNITY_LIST_EXPANDED)	{	  if (community_regexp_match (com, entry->reg))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	}    }  return 0;}intecommunity_list_match (struct ecommunity *ecom, struct community_list *list){  struct community_entry *entry;  for (entry = list->head; entry; entry = entry->next)    {      if (entry->any)	return entry->direct == COMMUNITY_PERMIT ? 1 : 0;      if (entry->style == EXTCOMMUNITY_LIST_STANDARD)	{	  if (ecommunity_match (ecom, entry->u.ecom))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	}      else if (entry->style == EXTCOMMUNITY_LIST_EXPANDED)	{	  if (ecommunity_regexp_match (ecom, entry->reg))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	}    }  return 0;}/* Perform exact matching.  In case of expanded community-list, do   same thing as community_list_match().  */intcommunity_list_exact_match (struct community *com, struct community_list *list){  struct community_entry *entry;  for (entry = list->head; entry; entry = entry->next)    {      if (entry->any)	return entry->direct == COMMUNITY_PERMIT ? 1 : 0;      if (entry->style == COMMUNITY_LIST_STANDARD)	{	  if (community_include (entry->u.com, COMMUNITY_INTERNET))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	  if (community_cmp (com, entry->u.com))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	}      else if (entry->style == COMMUNITY_LIST_EXPANDED)	{	  if (community_regexp_match (com, entry->reg))	    return entry->direct == COMMUNITY_PERMIT ? 1 : 0;	}    }  return 0;}/* Do regular expression matching with single community val.  */static intcomval_regexp_match (u_int32_t comval, regex_t *reg){  /* Maximum is "65535:65535" + '\0'. */  char c[12];  char *str;  switch (comval)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -