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_ipv4_multicast (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_debug ("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_ipv4_multicast (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_debug ("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
, ifindex_t ifindex
)
96 ret
= setsockopt_ipv4_multicast (top
->fd
, IP_ADD_MEMBERSHIP
,
97 p
->u
.prefix4
, htonl (OSPF_ALLDROUTERS
),
100 zlog_warn ("can't setsockopt IP_ADD_MEMBERSHIP (fd %d, addr %s, "
101 "ifindex %u, AllDRouters): %s; perhaps a kernel limit "
102 "on # of multicast group memberships has been exceeded?",
103 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
105 zlog_debug ("interface %s [%u] join AllDRouters Multicast group.",
106 inet_ntoa (p
->u
.prefix4
), ifindex
);
112 ospf_if_drop_alldrouters (struct ospf
*top
, struct prefix
*p
, ifindex_t ifindex
)
116 ret
= setsockopt_ipv4_multicast (top
->fd
, IP_DROP_MEMBERSHIP
,
117 p
->u
.prefix4
, htonl (OSPF_ALLDROUTERS
),
120 zlog_warn ("can't setsockopt IP_DROP_MEMBERSHIP (fd %d, addr %s, "
121 "ifindex %u, AllDRouters): %s",
122 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
124 zlog_debug ("interface %s [%u] leave AllDRouters Multicast group.",
125 inet_ntoa (p
->u
.prefix4
), ifindex
);
131 ospf_if_ipmulticast (struct ospf
*top
, struct prefix
*p
, ifindex_t ifindex
)
139 /* Prevent receiving self-origined multicast packets. */
140 ret
= setsockopt (top
->fd
, IPPROTO_IP
, IP_MULTICAST_LOOP
, (void *)&val
, len
);
142 zlog_warn ("can't setsockopt IP_MULTICAST_LOOP(0) for fd %d: %s",
143 top
->fd
, safe_strerror(errno
));
145 /* Explicitly set multicast ttl to 1 -- endo. */
147 ret
= setsockopt (top
->fd
, IPPROTO_IP
, IP_MULTICAST_TTL
, (void *)&val
, len
);
149 zlog_warn ("can't setsockopt IP_MULTICAST_TTL(1) for fd %d: %s",
150 top
->fd
, safe_strerror (errno
));
152 ret
= setsockopt_ipv4_multicast_if (top
->fd
, p
->u
.prefix4
, ifindex
);
154 zlog_warn("can't setsockopt IP_MULTICAST_IF(fd %d, addr %s, "
156 top
->fd
, inet_ntoa(p
->u
.prefix4
), ifindex
, safe_strerror(errno
));
162 ospf_sock_init (void)
166 int bufsize
= (8 * 1024 * 1024);
168 socklen_t optlen
= sizeof(optval
);
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
) );
228 if ((ret
= setsockopt (ospf_sock
, SOL_SOCKET
, SO_RCVBUF
,
229 &bufsize
, sizeof (bufsize
))) < 0)
231 zlog_err ("Couldn't increase raw rbuf size: %s\n", safe_strerror(errno
));
234 if ((ret
= getsockopt (ospf_sock
, SOL_SOCKET
, SO_RCVBUF
,
235 &optval
, &optlen
)) < 0)
237 zlog_err("getsockopt of SO_RCVBUF failed with error %s\n", safe_strerror(errno
));
239 if (optval
< bufsize
)
241 zlog_err("Unable to SO_RCVBUF to %d, set to %d\n", bufsize
, optval
);
245 if ((ret
= setsockopt (ospf_sock
, SOL_SOCKET
, SO_SNDBUF
,
246 &bufsize
, sizeof (bufsize
))) < 0)
248 zlog_err ("Couldn't increase raw wbuf size: %s\n", safe_strerror(errno
));
251 if ((ret
= getsockopt (ospf_sock
, SOL_SOCKET
, SO_SNDBUF
,
252 &optval
, &optlen
)) < 0)
254 zlog_err ("getsockopt of SO_SNDBUF failed with error %s\n", safe_strerror(errno
));
256 if (optval
< bufsize
)
258 zlog_err ("Unable to SO_SNDBUF to %d, set to %d\n", bufsize
, optval
);
265 ospf_adjust_sndbuflen (struct ospf
* ospf
, unsigned int buflen
)
268 /* Check if any work has to be done at all. */
269 if (ospf
->maxsndbuflen
>= buflen
)
271 if (IS_DEBUG_OSPF (zebra
, ZEBRA_INTERFACE
))
272 zlog_debug ("%s: adjusting OSPF send buffer size to %d",
274 if (ospfd_privs
.change (ZPRIVS_RAISE
))
275 zlog_err ("%s: could not raise privs, %s", __func__
,
276 safe_strerror (errno
));
277 /* Now we try to set SO_SNDBUF to what our caller has requested
278 * (the MTU of a newly added interface). However, if the OS has
279 * truncated the actual buffer size to somewhat less size, try
280 * to detect it and update our records appropriately. The OS
281 * may allocate more buffer space, than requested, this isn't
284 ret
= setsockopt_so_sendbuf (ospf
->fd
, buflen
);
285 newbuflen
= getsockopt_so_sendbuf (ospf
->fd
);
286 if (ret
< 0 || newbuflen
< 0 || newbuflen
< (int) buflen
)
287 zlog_warn ("%s: tried to set SO_SNDBUF to %u, but got %d",
288 __func__
, buflen
, newbuflen
);
290 ospf
->maxsndbuflen
= (unsigned int)newbuflen
;
292 zlog_warn ("%s: failed to get SO_SNDBUF", __func__
);
293 if (ospfd_privs
.change (ZPRIVS_LOWER
))
294 zlog_err ("%s: could not lower privs, %s", __func__
,
295 safe_strerror (errno
));