现有的路由协议都是通过分布式协议逐个配置协商运行的,协议协议,一个就不需要协议咯,至少2个才能够协议着做事情嘛,不过呢,这样就出现网元过多配置困难的问题,对网管软件要求也越来越高,

SDN或许可能改变这个局面,但不管软件如何定义,报文还是在一个一个网络节点传输的,所以我们看到quagga这样多的传统网络配置命令不要觉得繁琐,不配置的话,协议不会工作哦。

嗯,这章的主题是创建对等体,就是通过配置命令来完成的。

“neighbor peer remote-as asn”

router bgp
neighbor 10.0.0.1 remote-as

neighbor peer remote-as 命令就是配置一个对等体,peer是指对等体的地址(ipv4,ipv6地址)。

接下来我们看代码是如何创建对等体的,入口函数:

 DEFUN(neighbor_remote_as,
neighbor_remote_as_cmd,
NEIGHBOR_CMD2 "remote-as " CMD_AS_RANGE,
NEIGHBOR_STR
NEIGHBOR_ADDR_STR2
"Specify a BGP neighbor\n"
AS_STR) {
return peer_remote_as_vty(vty, argv[], argv[], AFI_IP, SAFI_UNICAST);
}

可以看到,bgp对等体之间是单播通信。

 /* If peer does not exist, create new one.  If peer already exists,
set AS number to the peer. */
int
peer_remote_as(struct bgp *bgp, union sockunion *su, as_t *as,
afi_t afi, safi_t safi)

peer_remote_as函数来完成创建新对等体或者为对等体设置新的as号。

为避免一些书籍的作者或者网络作者的一贯的懒惰做法(只授人以鱼),我得补充一下,为什么要创建对等体。

对于路由协议,不管是基于3层的还是2层的,都需要建立自己的寻路数据库,也就是通过邻居找到下一跳,你要走的远,你就得认识更多邻居,以及邻居的邻居,好比一句老话,在家靠父母,出门靠朋友,朋友多路好走,就这么一个道理。

那么话又说回来了,创建对等体呢,就是给自己找邻居,找朋友,不过呢,BGP这个人呢,更像一个干中介的,比如卖房的中介,他自己不建房子,只把建好的房源介绍给要买房的人,同时还维护这个房源信息库,及时更新已经卖掉的房子。

话有点多,我们还是接着看代码吧:

 /* "router bgp" commands. */
DEFUN(router_bgp,
router_bgp_cmd,
"router bgp " CMD_AS_RANGE,
ROUTER_STR
BGP_STR
AS_STR) {
...... ret = bgp_get(&bgp, &as, name); ...... vty->node = BGP_NODE;
vty->index = bgp; return CMD_SUCCESS;
}

router bgp 1

这条命令就是去查找一个as为1的bgp对象(struct bgp),如果没有找到就创建了一个struct bgp的指针,并保存在vty->index里,vty是每个DEFUN宏函数声明里都带的一个参数,在command里是极其重要的角色。

如果传给peer的值不是一个合法的地址,那么会被当做是一个peer group名称来处理

 /* If peer is peer group, call proper function.  */
ret = str2sockunion(peer_str, &su);
if (ret < ) {
ret = peer_group_remote_as(bgp, peer_str, &as);
if (ret < ) {
vty_out(vty, "%% Create the peer-group first%s", VTY_NEWLINE);
return CMD_WARNING;
}
return CMD_SUCCESS;
}

然后呢,是国际惯例,查找一下是不是已经为这个peer地址创建了peer,如果peer地址相同,as值不一样,就修改一下peer的as值。

嗯,如果这个peer已经是某个group的成员,那么就不能成功创建对等体关系了。

如果上面的事情都没有发生,那么就可以创建一个新的对等体了。

 static struct peer*
peer_new(struct bgp *bgp)
 /* Get service port number.  */
sp = getservbyname("bgp", "tcp");
peer->port = (sp == NULL) ? BGP_PORT_DEFAULT : ntohs(sp->s_port);

peer_new函数里设置默认的端口号 BGP_PORT_DEFAULT = 179,并将peer->status = Idle.

接下来会指定是IBGP peer还是EBGP peer。

最后,将这个peer加入到定时器任务中:

 /* Set up peer's events and timers. */
if (!active && peer_active(peer)) bgp_timer_set(peer);

然后我们就要进入对等体的状态机模式了,嗨,下一章,设计模式了喔!

05-11 15:09