]> git.proxmox.com Git - mirror_frr.git/blame - pimd/pim_util.c
zebra: Allow ns delete to happen after under/over flow checks
[mirror_frr.git] / pimd / pim_util.c
CommitLineData
12e41d03 1/*
896014f4
DL
2 * PIM for Quagga
3 * Copyright (C) 2008 Everton da Silva Marques
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; see the file COPYING; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18 */
12e41d03
DL
19
20#include <zebra.h>
21
22#include "log.h"
b815998a 23#include "prefix.h"
b0f525a8 24#include "plist.h"
12e41d03
DL
25
26#include "pim_util.h"
27
28/*
29 RFC 3376: 4.1.7. QQIC (Querier's Query Interval Code)
d62a17ae 30
12e41d03
DL
31 If QQIC < 128, QQI = QQIC
32 If QQIC >= 128, QQI = (mant | 0x10) << (exp + 3)
d62a17ae 33
12e41d03
DL
34 0 1 2 3 4 5 6 7
35 +-+-+-+-+-+-+-+-+
36 |1| exp | mant |
37 +-+-+-+-+-+-+-+-+
d62a17ae 38
12e41d03
DL
39 Since exp=0..7 then (exp+3)=3..10, then QQI has
40 one of the following bit patterns:
d62a17ae 41
12e41d03
DL
42 exp=0: QQI = 0000.0000.1MMM.M000
43 exp=1: QQI = 0000.0001.MMMM.0000
44 ...
45 exp=6: QQI = 001M.MMM0.0000.0000
46 exp=7: QQI = 01MM.MM00.0000.0000
47 --------- ---------
48 0x4 0x0 0x0 0x0
49*/
50uint8_t igmp_msg_encode16to8(uint16_t value)
51{
d62a17ae 52 uint8_t code;
53
54 if (value < 128) {
55 code = value;
56 } else {
57 uint16_t mask = 0x4000;
58 uint8_t exp;
59 uint16_t mant;
60 for (exp = 7; exp > 0; --exp) {
61 if (mask & value)
62 break;
63 mask >>= 1;
64 }
65 mant = 0x000F & (value >> (exp + 3));
66 code = ((uint8_t)1 << 7) | ((uint8_t)exp << 4) | (uint8_t)mant;
67 }
68
69 return code;
12e41d03
DL
70}
71
72/*
73 RFC 3376: 4.1.7. QQIC (Querier's Query Interval Code)
d62a17ae 74
12e41d03
DL
75 If QQIC < 128, QQI = QQIC
76 If QQIC >= 128, QQI = (mant | 0x10) << (exp + 3)
d62a17ae 77
12e41d03
DL
78 0 1 2 3 4 5 6 7
79 +-+-+-+-+-+-+-+-+
80 |1| exp | mant |
81 +-+-+-+-+-+-+-+-+
82*/
83uint16_t igmp_msg_decode8to16(uint8_t code)
84{
d62a17ae 85 uint16_t value;
86
87 if (code < 128) {
88 value = code;
89 } else {
90 uint16_t mant = (code & 0x0F);
91 uint8_t exp = (code & 0x70) >> 4;
92 value = (mant | 0x10) << (exp + 3);
93 }
94
95 return value;
12e41d03
DL
96}
97
98void pim_pkt_dump(const char *label, const uint8_t *buf, int size)
99{
d62a17ae 100 zlog_debug("%s: pkt dump size=%d", label, size);
101 zlog_hexdump(buf, size);
12e41d03 102}
b815998a 103
d62a17ae 104int pim_is_group_224_0_0_0_24(struct in_addr group_addr)
06d1c450 105{
d62a17ae 106 static int first = 1;
107 static struct prefix group_224;
108 struct prefix group;
06d1c450 109
d62a17ae 110 if (first) {
2e8345c1
DS
111 if (!str2prefix("224.0.0.0/24", &group_224))
112 return 0;
d62a17ae 113 first = 0;
114 }
06d1c450 115
d62a17ae 116 group.family = AF_INET;
117 group.u.prefix4 = group_addr;
b0f525a8 118 group.prefixlen = IPV4_MAX_PREFIXLEN;
06d1c450 119
d62a17ae 120 return prefix_match(&group_224, &group);
06d1c450
DS
121}
122
d62a17ae 123int pim_is_group_224_4(struct in_addr group_addr)
b815998a 124{
d62a17ae 125 static int first = 1;
126 static struct prefix group_all;
127 struct prefix group;
b815998a 128
d62a17ae 129 if (first) {
2e8345c1
DS
130 if (!str2prefix("224.0.0.0/4", &group_all))
131 return 0;
d62a17ae 132 first = 0;
133 }
b815998a 134
d62a17ae 135 group.family = AF_INET;
136 group.u.prefix4 = group_addr;
137 group.prefixlen = 32;
b815998a 138
d62a17ae 139 return prefix_match(&group_all, &group);
b815998a 140}
b0f525a8
QY
141
142bool pim_is_group_filtered(struct pim_interface *pim_ifp, struct in_addr *grp)
143{
144 struct prefix grp_pfx;
145 struct prefix_list *pl;
146
147 if (!pim_ifp->boundary_oil_plist)
148 return false;
149
150 grp_pfx.family = AF_INET;
151 grp_pfx.prefixlen = 32;
152 grp_pfx.u.prefix4 = *grp;
153
154 pl = prefix_list_lookup(AFI_IP, pim_ifp->boundary_oil_plist);
155 return pl ? prefix_list_apply(pl, &grp_pfx) == PREFIX_DENY : false;
156}