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

📄 zclient.c

📁 大名鼎鼎的路由器源码。程序分ZEBRA、OSPFRIP等3个包。程序框架采用一个路由协议一个进程的方式
💻 C
📖 第 1 页 / 共 2 页
字号:
/* Zebra's client library. * Copyright (C) 1999 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. */#include <zebra.h>#include "prefix.h"#include "stream.h"#include "network.h"#include "if.h"#include "log.h"#include "thread.h"#include "zclient.h"#include "memory.h"#include "table.h"#include "zebra/rib.h"#include "zebra/zserv.h"/* Zebra client events. */enum event {ZCLIENT_SCHEDULE, ZCLIENT_READ, ZCLIENT_CONNECT};/* Prototype for event manager. */static void zclient_event (enum event, struct zclient *);/* This file local debug flag. */int zclient_debug = 0;/* Allocate zclient structure. */struct zclient *zclient_new (){  struct zclient *zclient;  zclient = XMALLOC (MTYPE_ZCLIENT, sizeof (struct zclient));  memset (zclient, 0, sizeof (struct zclient));  zclient->ibuf = stream_new (ZEBRA_MAX_PACKET_SIZ);  zclient->obuf = stream_new (ZEBRA_MAX_PACKET_SIZ);  return zclient;}/* Free zclient structure. */voidzclient_free (struct zclient *zclient){  XFREE (MTYPE_ZCLIENT, zclient);}/* Initialize zebra client.  Argument redist_default is unwanted   redistribute route type. */voidzclient_init (struct zclient *zclient, int redist_default){  int i;    /* Enable zebra client connection by default. */  zclient->enable = 1;  /* Set -1 to the default socket value. */  zclient->sock = -1;  /* Clear redistribution flags. */  for (i = 0; i < ZEBRA_ROUTE_MAX; i++)    zclient->redist[i] = 0;  /* Set unwanted redistribute route.  bgpd does not need BGP route     redistribution. */  zclient->redist_default = redist_default;  zclient->redist[redist_default] = 1;  /* Set default-information redistribute to zero. */  zclient->default_information = 0;  /* Schedule first zclient connection. */  if (zclient_debug)    zlog_info ("zclient start scheduled");  zclient_event (ZCLIENT_SCHEDULE, zclient);}/* Stop zebra client services. */voidzclient_stop (struct zclient *zclient){  if (zclient_debug)    zlog_info ("zclient stopped");  /* Stop threads. */  if (zclient->t_read)    {      thread_cancel (zclient->t_read);      zclient->t_read = NULL;   }  if (zclient->t_connect)    {      thread_cancel (zclient->t_connect);      zclient->t_connect = NULL;    }  /* Close socket. */  if (zclient->sock >= 0)    {      close (zclient->sock);      zclient->sock = -1;    }  zclient->fail = 0;}voidzclient_reset (struct zclient *zclient){  zclient_stop (zclient);  zclient_init (zclient, zclient->redist_default);}/* Make socket to zebra daemon. Return zebra socket. */intzclient_socket (){  int sock;  int ret;  struct sockaddr_in serv;  /* We should think about IPv6 connection. */  sock = socket (AF_INET, SOCK_STREAM, 0);  if (sock < 0)    return -1;    /* Make server socket. */   memset (&serv, 0, sizeof (struct sockaddr_in));  serv.sin_family = AF_INET;  serv.sin_port = htons (ZEBRA_PORT);#ifdef HAVE_SIN_LEN  serv.sin_len = sizeof (struct sockaddr_in);#endif /* HAVE_SIN_LEN */  serv.sin_addr.s_addr = htonl (INADDR_LOOPBACK);  /* Connect to zebra. */  ret = connect (sock, (struct sockaddr *) &serv, sizeof (serv));  if (ret < 0)    {      close (sock);      return -1;    }  return sock;}/* For sockaddr_un. */#include <sys/un.h>intzclient_socket_un (char *path){  int ret;  int sock, len;  struct sockaddr_un addr;  sock = socket (AF_UNIX, SOCK_STREAM, 0);  if (sock < 0)    return -1;    /* Make server socket. */   memset (&addr, 0, sizeof (struct sockaddr_un));  addr.sun_family = AF_UNIX;  strncpy (addr.sun_path, path, strlen (path));#ifdef HAVE_SUN_LEN  len = addr.sun_len = SUN_LEN(&addr);#else  len = sizeof (addr.sun_family) + strlen (addr.sun_path);#endif /* HAVE_SUN_LEN */  ret = connect (sock, (struct sockaddr *) &addr, len);  if (ret < 0)    {      close (sock);      return -1;    }  return sock;}/* Send simple Zebra message. */intzebra_message_send (struct zclient *zclient, int command){  struct stream *s;  /* Get zclient output buffer. */  s = zclient->obuf;  stream_reset (s);  /* Send very simple command only Zebra message. */  stream_putw (s, 3);  stream_putc (s, command);  return writen (zclient->sock, s->data, 3);}/* Make connection to zebra daemon. */intzclient_start (struct zclient *zclient){  int i;  if (zclient_debug)    zlog_info ("zclient_start is called");  /* zclient is disabled. */  if (! zclient->enable)    return 0;  /* If already connected to the zebra. */  if (zclient->sock >= 0)    return 0;  /* Check connect thread. */  if (zclient->t_connect)    return 0;  /* Make socket. */#ifdef HAVE_TCP_ZEBRA  zclient->sock = zclient_socket ();#else  zclient->sock = zclient_socket_un (ZEBRA_SERV_PATH);#endif /* HAVE_TCP_ZEBRA */  if (zclient->sock < 0)    {      if (zclient_debug)	zlog_info ("zclient connection fail");      zclient->fail++;      zclient_event (ZCLIENT_CONNECT, zclient);      return -1;    }  /* Clear fail count. */  zclient->fail = 0;  if (zclient_debug)    zlog_info ("zclient connect success with socket [%d]", zclient->sock);        /* Create read thread. */  zclient_event (ZCLIENT_READ, zclient);  /* We need interface information. */  zebra_message_send (zclient, ZEBRA_INTERFACE_ADD);  /* Flush all redistribute request. */  for (i = 0; i < ZEBRA_ROUTE_MAX; i++)    if (i != zclient->redist_default && zclient->redist[i])      zebra_redistribute_send (ZEBRA_REDISTRIBUTE_ADD, zclient->sock, i);  /* If default information is needed. */  if (zclient->default_information)    zebra_message_send (zclient, ZEBRA_REDISTRIBUTE_DEFAULT_ADD);  return 0;}/* This function is a wrapper function for calling zclient_start from   timer or event thread. */intzclient_connect (struct thread *t){  struct zclient *zclient;  zclient = THREAD_ARG (t);  zclient->t_connect = NULL;  if (zclient_debug)    zlog_info ("zclient_connect is called");  return zclient_start (zclient);}intzapi_ipv4_add (struct zclient *zclient, struct prefix_ipv4 *p,	       struct zapi_ipv4 *api){  int i;  int psize;  struct stream *s;  /* Reset stream. */  s = zclient->obuf;  stream_reset (s);  /* Length place holder. */  stream_putw (s, 0);  /* Put command, type and nexthop. */  stream_putc (s, ZEBRA_IPV4_ROUTE_ADD);  stream_putc (s, api->type);  stream_putc (s, api->flags);  stream_putc (s, api->message);    /* Put prefix information. */  psize = PSIZE (p->prefixlen);  stream_putc (s, p->prefixlen);  stream_write (s, (u_char *)&p->prefix, psize);  /* Nexthop, ifindex, distance and metric information. */  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_NEXTHOP))    {      if (CHECK_FLAG (api->flags, ZEBRA_FLAG_BLACKHOLE))	{	  stream_putc (s, 1);	  stream_putc (s, ZEBRA_NEXTHOP_BLACKHOLE);	}      else	stream_putc (s, api->nexthop_num + api->ifindex_num);      for (i = 0; i < api->nexthop_num; i++)	{	  stream_putc (s, ZEBRA_NEXTHOP_IPV4);	  stream_put_in_addr (s, api->nexthop[i]);	}      for (i = 0; i < api->ifindex_num; i++)	{	  stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);	  stream_putl (s, api->ifindex[i]);	}    }  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_DISTANCE))    stream_putc (s, api->distance);  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_METRIC))    stream_putl (s, api->metric);  /* Put length at the first point of the stream. */  stream_putw_at (s, 0, stream_get_endp (s));  return writen (zclient->sock, s->data, stream_get_endp (s));}intzapi_ipv4_delete (struct zclient *zclient, struct prefix_ipv4 *p,		  struct zapi_ipv4 *api){  int i;  int psize;  struct stream *s;  /* Reset stream. */  s = zclient->obuf;  stream_reset (s);  /* Length place holder. */  stream_putw (s, 0);  /* Put command, type and nexthop. */  stream_putc (s, ZEBRA_IPV4_ROUTE_DELETE);  stream_putc (s, api->type);  stream_putc (s, api->flags);  stream_putc (s, api->message);    /* Put prefix information. */  psize = PSIZE (p->prefixlen);  stream_putc (s, p->prefixlen);  stream_write (s, (u_char *)&p->prefix, psize);  /* Nexthop, ifindex, distance and metric information. */  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_NEXTHOP))    {      if (CHECK_FLAG (api->flags, ZEBRA_FLAG_BLACKHOLE))	{	  stream_putc (s, 1);	  stream_putc (s, ZEBRA_NEXTHOP_BLACKHOLE);	}      else	stream_putc (s, api->nexthop_num + api->ifindex_num);      for (i = 0; i < api->nexthop_num; i++)	{	  stream_putc (s, ZEBRA_NEXTHOP_IPV4);	  stream_put_in_addr (s, api->nexthop[i]);	}      for (i = 0; i < api->ifindex_num; i++)	{	  stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);	  stream_putl (s, api->ifindex[i]);	}    }  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_DISTANCE))    stream_putc (s, api->distance);  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_METRIC))    stream_putl (s, api->metric);  /* Put length at the first point of the stream. */  stream_putw_at (s, 0, stream_get_endp (s));  return writen (zclient->sock, s->data, stream_get_endp (s));}#ifdef HAVE_IPV6intzapi_ipv6_add (struct zclient *zclient, struct prefix_ipv6 *p,	       struct zapi_ipv6 *api){  int i;  int psize;  struct stream *s;  /* Reset stream. */  s = zclient->obuf;  stream_reset (s);  /* Length place holder. */  stream_putw (s, 0);  /* Put command, type and nexthop. */  stream_putc (s, ZEBRA_IPV6_ROUTE_ADD);  stream_putc (s, api->type);  stream_putc (s, api->flags);  stream_putc (s, api->message);    /* Put prefix information. */  psize = PSIZE (p->prefixlen);  stream_putc (s, p->prefixlen);  stream_write (s, (u_char *)&p->prefix, psize);  /* Nexthop, ifindex, distance and metric information. */  if (CHECK_FLAG (api->message, ZAPI_MESSAGE_NEXTHOP))    {      stream_putc (s, api->nexthop_num + api->ifindex_num);      for (i = 0; i < api->nexthop_num; i++)	{	  stream_putc (s, ZEBRA_NEXTHOP_IPV6);	  stream_write (s, (u_char *)api->nexthop[i], 16);	}      for (i = 0; i < api->ifindex_num; i++)	{	  stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);	  stream_putl (s, api->ifindex[i]);

⌨️ 快捷键说明

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