2 * OSPF network related functions
3 * Copyright (C) 1999 Toshiaki Takada
5 * This file is part of GNU Zebra.
7 * GNU Zebra is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the
9 * Free Software Foundation; either version 2, or (at your option) any
12 * GNU Zebra is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with GNU Zebra; see the file COPYING. If not, write to the Free
19 * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
29 #include "sockunion.h"
34 extern struct zebra_privs_t ospfd_privs
;
36 #include "ospfd/ospfd.h"
37 #include "ospfd/ospf_network.h"
38 #include "ospfd/ospf_interface.h"
39 #include "ospfd/ospf_asbr.h"
40 #include "ospfd/ospf_lsa.h"
41 #include "ospfd/ospf_lsdb.h"
42 #include "ospfd/ospf_neighbor.h"
43 #include "ospfd/ospf_packet.h"
44 #include "ospfd/ospf_dump.h"
48 /* Join to the OSPF ALL SPF ROUTERS multicast group. */
50 ospf_if_add_allspfrouters (struct ospf
*top
, struct prefix
*p
,
55 ret
= setsockopt_multicast_ipv4 (top
->fd
, IP_ADD_MEMBERSHIP
,
56 p
->u
.prefix4
, htonl (OSPF_ALLSPFROUTERS
),
59 zlog_warn ("can't setsockopt IP_ADD_MEMBERSHIP (fd %d, addr %s, "
60 "ifindex %u, AllSPFRouters): %s; perhaps a kernel limit "
61 "on # of multicast group memberships has been exceeded?",
62 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
64 zlog_info ("interface %s [%u] join AllSPFRouters Multicast group.",
65 inet_ntoa (p
->u
.prefix4
), ifindex
);
71 ospf_if_drop_allspfrouters (struct ospf
*top
, struct prefix
*p
,
76 ret
= setsockopt_multicast_ipv4 (top
->fd
, IP_DROP_MEMBERSHIP
,
77 p
->u
.prefix4
, htonl (OSPF_ALLSPFROUTERS
),
80 zlog_warn ("can't setsockopt IP_DROP_MEMBERSHIP (fd %d, addr %s, "
81 "ifindex %u, AllSPFRouters): %s",
82 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
84 zlog_info ("interface %s [%u] leave AllSPFRouters Multicast group.",
85 inet_ntoa (p
->u
.prefix4
), ifindex
);
90 /* Join to the OSPF ALL Designated ROUTERS multicast group. */
92 ospf_if_add_alldrouters (struct ospf
*top
, struct prefix
*p
, unsigned int
97 ret
= setsockopt_multicast_ipv4 (top
->fd
, IP_ADD_MEMBERSHIP
,
98 p
->u
.prefix4
, htonl (OSPF_ALLDROUTERS
),
101 zlog_warn ("can't setsockopt IP_ADD_MEMBERSHIP (fd %d, addr %s, "
102 "ifindex %u, AllDRouters): %s; perhaps a kernel limit "
103 "on # of multicast group memberships has been exceeded?",
104 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
106 zlog_info ("interface %s [%u] join AllDRouters Multicast group.",
107 inet_ntoa (p
->u
.prefix4
), ifindex
);
113 ospf_if_drop_alldrouters (struct ospf
*top
, struct prefix
*p
, unsigned int
118 ret
= setsockopt_multicast_ipv4 (top
->fd
, IP_DROP_MEMBERSHIP
,
119 p
->u
.prefix4
, htonl (OSPF_ALLDROUTERS
),
122 zlog_warn ("can't setsockopt IP_DROP_MEMBERSHIP (fd %d, addr %s, "
123 "ifindex %u, AllDRouters): %s",
124 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
126 zlog_info ("interface %s [%u] leave AllDRouters Multicast group.",
127 inet_ntoa (p
->u
.prefix4
), ifindex
);
133 ospf_if_ipmulticast (struct ospf
*top
, struct prefix
*p
, unsigned int ifindex
)
141 /* Prevent receiving self-origined multicast packets. */
142 ret
= setsockopt (top
->fd
, IPPROTO_IP
, IP_MULTICAST_LOOP
, (void *)&val
, len
);
144 zlog_warn ("can't setsockopt IP_MULTICAST_LOOP(0) for fd %d: %s",
145 top
->fd
, safe_strerror(errno
));
147 /* Explicitly set multicast ttl to 1 -- endo. */
149 ret
= setsockopt (top
->fd
, IPPROTO_IP
, IP_MULTICAST_TTL
, (void *)&val
, len
);
151 zlog_warn ("can't setsockopt IP_MULTICAST_TTL(1) for fd %d: %s",
152 top
->fd
, safe_strerror (errno
));
154 ret
= setsockopt_multicast_ipv4 (top
->fd
, IP_MULTICAST_IF
,
155 p
->u
.prefix4
, 0, ifindex
);
157 zlog_warn("can't setsockopt IP_MULTICAST_IF(fd %d, addr %s, "
159 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
165 ospf_sock_init (void)
170 if ( ospfd_privs
.change (ZPRIVS_RAISE
) )
171 zlog_err ("ospf_sock_init: could not raise privs, %s",
172 safe_strerror (errno
) );
174 ospf_sock
= socket (AF_INET
, SOCK_RAW
, IPPROTO_OSPFIGP
);
177 int save_errno
= errno
;
178 if ( ospfd_privs
.change (ZPRIVS_LOWER
) )
179 zlog_err ("ospf_sock_init: could not lower privs, %s",
180 safe_strerror (errno
) );
181 zlog_err ("ospf_read_sock_init: socket: %s", safe_strerror (save_errno
));
186 /* we will include IP header with packet */
187 ret
= setsockopt (ospf_sock
, IPPROTO_IP
, IP_HDRINCL
, &hincl
, sizeof (hincl
));
190 int save_errno
= errno
;
191 if ( ospfd_privs
.change (ZPRIVS_LOWER
) )
192 zlog_err ("ospf_sock_init: could not lower privs, %s",
193 safe_strerror (errno
) );
194 zlog_warn ("Can't set IP_HDRINCL option for fd %d: %s",
195 ospf_sock
, safe_strerror(save_errno
));
197 #elif defined (IPTOS_PREC_INTERNETCONTROL)
198 #warning "IP_HDRINCL not available on this system"
199 #warning "using IPTOS_PREC_INTERNETCONTROL"
200 ret
= setsockopt_ipv4_tos(ospf_sock
, IPTOS_PREC_INTERNETCONTROL
);
203 int save_errno
= errno
;
204 if ( ospfd_privs
.change (ZPRIVS_LOWER
) )
205 zlog_err ("ospf_sock_init: could not lower privs, %s",
206 safe_strerror (errno
) );
207 zlog_warn ("can't set sockopt IP_TOS %d to socket %d: %s",
208 tos
, ospf_sock
, safe_strerror(save_errno
));
209 close (ospf_sock
); /* Prevent sd leak. */
212 #else /* !IPTOS_PREC_INTERNETCONTROL */
213 #warning "IP_HDRINCL not available, nor is IPTOS_PREC_INTERNETCONTROL"
214 zlog_warn ("IP_HDRINCL option not available");
215 #endif /* IP_HDRINCL */
217 ret
= setsockopt_ifindex (AF_INET
, ospf_sock
, 1);
220 zlog_warn ("Can't set pktinfo option for fd %d", ospf_sock
);
222 if (ospfd_privs
.change (ZPRIVS_LOWER
))
224 zlog_err ("ospf_sock_init: could not lower privs, %s",
225 safe_strerror (errno
) );
232 ospf_adjust_sndbuflen (struct ospf
* ospf
, int buflen
)
235 /* Check if any work has to be done at all. */
236 if (ospf
->maxsndbuflen
>= buflen
)
238 if (IS_DEBUG_OSPF (zebra
, ZEBRA_INTERFACE
))
239 zlog_debug ("%s: adjusting OSPF send buffer size to %d",
241 if (ospfd_privs
.change (ZPRIVS_RAISE
))
242 zlog_err ("%s: could not raise privs, %s", __func__
,
243 safe_strerror (errno
));
244 /* Now we try to set SO_SNDBUF to what our caller has requested
245 * (the MTU of a newly added interface). However, if the OS has
246 * truncated the actual buffer size to somewhat less size, try
247 * to detect it and update our records appropriately. The OS
248 * may allocate more buffer space, than requested, this isn't
251 ret
= setsockopt_so_sendbuf (ospf
->fd
, buflen
);
252 newbuflen
= getsockopt_so_sendbuf (ospf
->fd
);
253 if (ret
< 0 || newbuflen
< buflen
)
254 zlog_warn ("%s: tried to set SO_SNDBUF to %d, but got %d",
255 __func__
, buflen
, newbuflen
);
257 ospf
->maxsndbuflen
= newbuflen
;
259 zlog_warn ("%s: failed to get SO_SNDBUF", __func__
);
260 if (ospfd_privs
.change (ZPRIVS_LOWER
))
261 zlog_err ("%s: could not lower privs, %s", __func__
,
262 safe_strerror (errno
));