|
@@ -28,6 +28,9 @@ union pkthdr {
|
|
u8 type;
|
|
u8 type;
|
|
u8 code;
|
|
u8 code;
|
|
} icmphdr;
|
|
} icmphdr;
|
|
|
|
+ struct {
|
|
|
|
+ u8 type;
|
|
|
|
+ } igmphdr;
|
|
};
|
|
};
|
|
|
|
|
|
static bool
|
|
static bool
|
|
@@ -57,12 +60,12 @@ ebt_ip_mt(const struct sk_buff *skb, struct xt_action_param *par)
|
|
if (NF_INVF(info, EBT_IP_PROTO, info->protocol != ih->protocol))
|
|
if (NF_INVF(info, EBT_IP_PROTO, info->protocol != ih->protocol))
|
|
return false;
|
|
return false;
|
|
if (!(info->bitmask & (EBT_IP_DPORT | EBT_IP_SPORT |
|
|
if (!(info->bitmask & (EBT_IP_DPORT | EBT_IP_SPORT |
|
|
- EBT_IP_ICMP)))
|
|
|
|
|
|
+ EBT_IP_ICMP | EBT_IP_IGMP)))
|
|
return true;
|
|
return true;
|
|
if (ntohs(ih->frag_off) & IP_OFFSET)
|
|
if (ntohs(ih->frag_off) & IP_OFFSET)
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- /* min icmp headersize is 4, so sizeof(_pkthdr) is ok. */
|
|
|
|
|
|
+ /* min icmp/igmp headersize is 4, so sizeof(_pkthdr) is ok. */
|
|
pptr = skb_header_pointer(skb, ih->ihl*4,
|
|
pptr = skb_header_pointer(skb, ih->ihl*4,
|
|
sizeof(_pkthdr), &_pkthdr);
|
|
sizeof(_pkthdr), &_pkthdr);
|
|
if (pptr == NULL)
|
|
if (pptr == NULL)
|
|
@@ -88,6 +91,11 @@ ebt_ip_mt(const struct sk_buff *skb, struct xt_action_param *par)
|
|
pptr->icmphdr.code < info->icmp_code[0] ||
|
|
pptr->icmphdr.code < info->icmp_code[0] ||
|
|
pptr->icmphdr.code > info->icmp_code[1]))
|
|
pptr->icmphdr.code > info->icmp_code[1]))
|
|
return false;
|
|
return false;
|
|
|
|
+ if ((info->bitmask & EBT_IP_IGMP) &&
|
|
|
|
+ NF_INVF(info, EBT_IP_IGMP,
|
|
|
|
+ pptr->igmphdr.type < info->igmp_type[0] ||
|
|
|
|
+ pptr->igmphdr.type > info->igmp_type[1]))
|
|
|
|
+ return false;
|
|
}
|
|
}
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
@@ -124,6 +132,13 @@ static int ebt_ip_mt_check(const struct xt_mtchk_param *par)
|
|
info->icmp_code[0] > info->icmp_code[1])
|
|
info->icmp_code[0] > info->icmp_code[1])
|
|
return -EINVAL;
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
+ if (info->bitmask & EBT_IP_IGMP) {
|
|
|
|
+ if ((info->invflags & EBT_IP_PROTO) ||
|
|
|
|
+ info->protocol != IPPROTO_IGMP)
|
|
|
|
+ return -EINVAL;
|
|
|
|
+ if (info->igmp_type[0] > info->igmp_type[1])
|
|
|
|
+ return -EINVAL;
|
|
|
|
+ }
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|