mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-08 15:04:45 +00:00
[SK_BUFF]: Introduce skb_network_header()
For the places where we need a pointer to the network header, it is still legal to touch skb->nh.raw directly if just adding to, subtracting from or setting it to another layer header. Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
bbe735e424
commit
d56f90a7c9
@ -872,7 +872,8 @@ typedef struct {
|
|||||||
static void
|
static void
|
||||||
isdn_net_log_skb(struct sk_buff * skb, isdn_net_local * lp)
|
isdn_net_log_skb(struct sk_buff * skb, isdn_net_local * lp)
|
||||||
{
|
{
|
||||||
u_char *p = skb->nh.raw; /* hopefully, this was set correctly */
|
/* hopefully, this was set correctly */
|
||||||
|
const u_char *p = skb_network_header(skb);
|
||||||
unsigned short proto = ntohs(skb->protocol);
|
unsigned short proto = ntohs(skb->protocol);
|
||||||
int data_ofs;
|
int data_ofs;
|
||||||
ip_ports *ipp;
|
ip_ports *ipp;
|
||||||
@ -880,7 +881,7 @@ isdn_net_log_skb(struct sk_buff * skb, isdn_net_local * lp)
|
|||||||
|
|
||||||
addinfo[0] = '\0';
|
addinfo[0] = '\0';
|
||||||
/* This check stolen from 2.1.72 dev_queue_xmit_nit() */
|
/* This check stolen from 2.1.72 dev_queue_xmit_nit() */
|
||||||
if (skb->nh.raw < skb->data || skb->nh.raw >= skb->tail) {
|
if (p < skb->data || p >= skb->tail) {
|
||||||
/* fall back to old isdn_net_log_packet method() */
|
/* fall back to old isdn_net_log_packet method() */
|
||||||
char * buf = skb->data;
|
char * buf = skb->data;
|
||||||
|
|
||||||
|
@ -106,7 +106,7 @@ struct arp_pkt {
|
|||||||
|
|
||||||
static inline struct arp_pkt *arp_pkt(const struct sk_buff *skb)
|
static inline struct arp_pkt *arp_pkt(const struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
return (struct arp_pkt *)skb->nh.raw;
|
return (struct arp_pkt *)skb_network_header(skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Forward declaration */
|
/* Forward declaration */
|
||||||
|
@ -76,7 +76,8 @@ static DEFINE_PER_CPU(struct pcpu_lstats, pcpu_lstats);
|
|||||||
static void emulate_large_send_offload(struct sk_buff *skb)
|
static void emulate_large_send_offload(struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
struct iphdr *iph = skb->nh.iph;
|
struct iphdr *iph = skb->nh.iph;
|
||||||
struct tcphdr *th = (struct tcphdr*)(skb->nh.raw + (iph->ihl * 4));
|
struct tcphdr *th = (struct tcphdr *)(skb_network_header(skb) +
|
||||||
|
(iph->ihl * 4));
|
||||||
unsigned int doffset = (iph->ihl + th->doff) * 4;
|
unsigned int doffset = (iph->ihl + th->doff) * 4;
|
||||||
unsigned int mtu = skb_shinfo(skb)->gso_size + doffset;
|
unsigned int mtu = skb_shinfo(skb)->gso_size + doffset;
|
||||||
unsigned int offset = 0;
|
unsigned int offset = 0;
|
||||||
@ -93,7 +94,7 @@ static void emulate_large_send_offload(struct sk_buff *skb)
|
|||||||
skb_set_mac_header(nskb, -ETH_HLEN);
|
skb_set_mac_header(nskb, -ETH_HLEN);
|
||||||
skb_reset_network_header(nskb);
|
skb_reset_network_header(nskb);
|
||||||
iph = nskb->nh.iph;
|
iph = nskb->nh.iph;
|
||||||
memcpy(nskb->data, skb->nh.raw, doffset);
|
memcpy(nskb->data, skb_network_header(skb), doffset);
|
||||||
if (skb_copy_bits(skb,
|
if (skb_copy_bits(skb,
|
||||||
doffset + offset,
|
doffset + offset,
|
||||||
nskb->data + doffset,
|
nskb->data + doffset,
|
||||||
@ -108,7 +109,7 @@ static void emulate_large_send_offload(struct sk_buff *skb)
|
|||||||
memcpy(nskb->cb, skb->cb, sizeof(skb->cb));
|
memcpy(nskb->cb, skb->cb, sizeof(skb->cb));
|
||||||
nskb->pkt_type = skb->pkt_type;
|
nskb->pkt_type = skb->pkt_type;
|
||||||
|
|
||||||
th = (struct tcphdr*)(nskb->nh.raw + iph->ihl*4);
|
th = (struct tcphdr *)(skb_network_header(nskb) + iph->ihl * 4);
|
||||||
iph->tot_len = htons(frag_size + doffset);
|
iph->tot_len = htons(frag_size + doffset);
|
||||||
iph->id = htons(id);
|
iph->id = htons(id);
|
||||||
iph->check = 0;
|
iph->check = 0;
|
||||||
|
@ -729,16 +729,18 @@ static int pasemi_mac_start_tx(struct sk_buff *skb, struct net_device *dev)
|
|||||||
dflags = XCT_MACTX_O | XCT_MACTX_ST | XCT_MACTX_SS | XCT_MACTX_CRC_PAD;
|
dflags = XCT_MACTX_O | XCT_MACTX_ST | XCT_MACTX_SS | XCT_MACTX_CRC_PAD;
|
||||||
|
|
||||||
if (skb->ip_summed == CHECKSUM_PARTIAL) {
|
if (skb->ip_summed == CHECKSUM_PARTIAL) {
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
|
||||||
switch (skb->nh.iph->protocol) {
|
switch (skb->nh.iph->protocol) {
|
||||||
case IPPROTO_TCP:
|
case IPPROTO_TCP:
|
||||||
dflags |= XCT_MACTX_CSUM_TCP;
|
dflags |= XCT_MACTX_CSUM_TCP;
|
||||||
dflags |= XCT_MACTX_IPH((skb->h.raw - skb->nh.raw) >> 2);
|
dflags |= XCT_MACTX_IPH((skb->h.raw - skb->nh.raw) >> 2);
|
||||||
dflags |= XCT_MACTX_IPO(skb->nh.raw - skb->data);
|
dflags |= XCT_MACTX_IPO(nh - skb->data);
|
||||||
break;
|
break;
|
||||||
case IPPROTO_UDP:
|
case IPPROTO_UDP:
|
||||||
dflags |= XCT_MACTX_CSUM_UDP;
|
dflags |= XCT_MACTX_CSUM_UDP;
|
||||||
dflags |= XCT_MACTX_IPH((skb->h.raw - skb->nh.raw) >> 2);
|
dflags |= XCT_MACTX_IPH((skb->h.raw - skb->nh.raw) >> 2);
|
||||||
dflags |= XCT_MACTX_IPO(skb->nh.raw - skb->data);
|
dflags |= XCT_MACTX_IPO(nh - skb->data);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3778,9 +3778,11 @@ qeth_get_cast_type(struct qeth_card *card, struct sk_buff *skb)
|
|||||||
}
|
}
|
||||||
/* try something else */
|
/* try something else */
|
||||||
if (skb->protocol == ETH_P_IPV6)
|
if (skb->protocol == ETH_P_IPV6)
|
||||||
return (skb->nh.raw[24] == 0xff) ? RTN_MULTICAST : 0;
|
return (skb_network_header(skb)[24] == 0xff) ?
|
||||||
|
RTN_MULTICAST : 0;
|
||||||
else if (skb->protocol == ETH_P_IP)
|
else if (skb->protocol == ETH_P_IP)
|
||||||
return ((skb->nh.raw[16] & 0xf0) == 0xe0) ? RTN_MULTICAST : 0;
|
return ((skb_network_header(skb)[16] & 0xf0) == 0xe0) ?
|
||||||
|
RTN_MULTICAST : 0;
|
||||||
/* ... */
|
/* ... */
|
||||||
if (!memcmp(skb->data, skb->dev->broadcast, 6))
|
if (!memcmp(skb->data, skb->dev->broadcast, 6))
|
||||||
return RTN_BROADCAST;
|
return RTN_BROADCAST;
|
||||||
|
@ -116,7 +116,7 @@ struct pppoe_hdr {
|
|||||||
|
|
||||||
static inline struct pppoe_hdr *pppoe_hdr(const struct sk_buff *skb)
|
static inline struct pppoe_hdr *pppoe_hdr(const struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
return (struct pppoe_hdr *)skb->nh.raw;
|
return (struct pppoe_hdr *)skb_network_header(skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct pppoe_opt {
|
struct pppoe_opt {
|
||||||
|
@ -960,6 +960,11 @@ static inline void skb_reserve(struct sk_buff *skb, int len)
|
|||||||
skb->tail += len;
|
skb->tail += len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline unsigned char *skb_network_header(const struct sk_buff *skb)
|
||||||
|
{
|
||||||
|
return skb->nh.raw;
|
||||||
|
}
|
||||||
|
|
||||||
static inline void skb_reset_network_header(struct sk_buff *skb)
|
static inline void skb_reset_network_header(struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
skb->nh.raw = skb->data;
|
skb->nh.raw = skb->data;
|
||||||
|
@ -120,7 +120,7 @@ extern int cipso_v4_rbm_strictvalid;
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#define CIPSO_V4_OPTEXIST(x) (IPCB(x)->opt.cipso != 0)
|
#define CIPSO_V4_OPTEXIST(x) (IPCB(x)->opt.cipso != 0)
|
||||||
#define CIPSO_V4_OPTPTR(x) ((x)->nh.raw + IPCB(x)->opt.cipso)
|
#define CIPSO_V4_OPTPTR(x) (skb_network_header(x) + IPCB(x)->opt.cipso)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* DOI List Functions
|
* DOI List Functions
|
||||||
|
@ -114,12 +114,14 @@ static inline int INET_ECN_set_ce(struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
switch (skb->protocol) {
|
switch (skb->protocol) {
|
||||||
case __constant_htons(ETH_P_IP):
|
case __constant_htons(ETH_P_IP):
|
||||||
if (skb->nh.raw + sizeof(struct iphdr) <= skb->tail)
|
if (skb_network_header(skb) + sizeof(struct iphdr) <=
|
||||||
|
skb->tail)
|
||||||
return IP_ECN_set_ce(skb->nh.iph);
|
return IP_ECN_set_ce(skb->nh.iph);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case __constant_htons(ETH_P_IPV6):
|
case __constant_htons(ETH_P_IPV6):
|
||||||
if (skb->nh.raw + sizeof(struct ipv6hdr) <= skb->tail)
|
if (skb_network_header(skb) + sizeof(struct ipv6hdr) <=
|
||||||
|
skb->tail)
|
||||||
return IP6_ECN_set_ce(skb->nh.ipv6h);
|
return IP6_ECN_set_ce(skb->nh.ipv6h);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -203,7 +203,7 @@ struct llc_pdu_sn {
|
|||||||
|
|
||||||
static inline struct llc_pdu_sn *llc_pdu_sn_hdr(struct sk_buff *skb)
|
static inline struct llc_pdu_sn *llc_pdu_sn_hdr(struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
return (struct llc_pdu_sn *)skb->nh.raw;
|
return (struct llc_pdu_sn *)skb_network_header(skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Un-numbered PDU format (3 bytes in length) */
|
/* Un-numbered PDU format (3 bytes in length) */
|
||||||
@ -215,7 +215,7 @@ struct llc_pdu_un {
|
|||||||
|
|
||||||
static inline struct llc_pdu_un *llc_pdu_un_hdr(struct sk_buff *skb)
|
static inline struct llc_pdu_un *llc_pdu_un_hdr(struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
return (struct llc_pdu_un *)skb->nh.raw;
|
return (struct llc_pdu_un *)skb_network_header(skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -326,7 +326,7 @@ static inline unsigned char * tcf_get_base_ptr(struct sk_buff *skb, int layer)
|
|||||||
case TCF_LAYER_LINK:
|
case TCF_LAYER_LINK:
|
||||||
return skb->data;
|
return skb->data;
|
||||||
case TCF_LAYER_NETWORK:
|
case TCF_LAYER_NETWORK:
|
||||||
return skb->nh.raw;
|
return skb_network_header(skb);
|
||||||
case TCF_LAYER_TRANSPORT:
|
case TCF_LAYER_TRANSPORT:
|
||||||
return skb->h.raw;
|
return skb->h.raw;
|
||||||
}
|
}
|
||||||
|
@ -374,7 +374,8 @@ static int check_hbh_len(struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
unsigned char *raw = (u8 *) (skb->nh.ipv6h + 1);
|
unsigned char *raw = (u8 *) (skb->nh.ipv6h + 1);
|
||||||
u32 pkt_len;
|
u32 pkt_len;
|
||||||
int off = raw - skb->nh.raw;
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
int off = raw - nh;
|
||||||
int len = (raw[1] + 1) << 3;
|
int len = (raw[1] + 1) << 3;
|
||||||
|
|
||||||
if ((raw + len) - skb->data > skb_headlen(skb))
|
if ((raw + len) - skb->data > skb_headlen(skb))
|
||||||
@ -384,9 +385,9 @@ static int check_hbh_len(struct sk_buff *skb)
|
|||||||
len -= 2;
|
len -= 2;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
int optlen = skb->nh.raw[off + 1] + 2;
|
int optlen = nh[off + 1] + 2;
|
||||||
|
|
||||||
switch (skb->nh.raw[off]) {
|
switch (nh[off]) {
|
||||||
case IPV6_TLV_PAD0:
|
case IPV6_TLV_PAD0:
|
||||||
optlen = 1;
|
optlen = 1;
|
||||||
break;
|
break;
|
||||||
@ -395,9 +396,9 @@ static int check_hbh_len(struct sk_buff *skb)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case IPV6_TLV_JUMBO:
|
case IPV6_TLV_JUMBO:
|
||||||
if (skb->nh.raw[off + 1] != 4 || (off & 3) != 2)
|
if (nh[off + 1] != 4 || (off & 3) != 2)
|
||||||
goto bad;
|
goto bad;
|
||||||
pkt_len = ntohl(*(__be32 *) (skb->nh.raw + off + 2));
|
pkt_len = ntohl(*(__be32 *) (nh + off + 2));
|
||||||
if (pkt_len <= IPV6_MAXPLEN ||
|
if (pkt_len <= IPV6_MAXPLEN ||
|
||||||
skb->nh.ipv6h->payload_len)
|
skb->nh.ipv6h->payload_len)
|
||||||
goto bad;
|
goto bad;
|
||||||
@ -406,6 +407,7 @@ static int check_hbh_len(struct sk_buff *skb)
|
|||||||
if (pskb_trim_rcsum(skb,
|
if (pskb_trim_rcsum(skb,
|
||||||
pkt_len + sizeof(struct ipv6hdr)))
|
pkt_len + sizeof(struct ipv6hdr)))
|
||||||
goto bad;
|
goto bad;
|
||||||
|
nh = skb_network_header(skb);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
if (optlen > len)
|
if (optlen > len)
|
||||||
|
@ -1068,8 +1068,8 @@ static void dev_queue_xmit_nit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
*/
|
*/
|
||||||
skb_reset_mac_header(skb2);
|
skb_reset_mac_header(skb2);
|
||||||
|
|
||||||
if (skb2->nh.raw < skb2->data ||
|
if (skb_network_header(skb2) < skb2->data ||
|
||||||
skb2->nh.raw > skb2->tail) {
|
skb_network_header(skb2) > skb2->tail) {
|
||||||
if (net_ratelimit())
|
if (net_ratelimit())
|
||||||
printk(KERN_CRIT "protocol %04x is "
|
printk(KERN_CRIT "protocol %04x is "
|
||||||
"buggy, dev %s\n",
|
"buggy, dev %s\n",
|
||||||
@ -1207,7 +1207,7 @@ struct sk_buff *skb_gso_segment(struct sk_buff *skb, int features)
|
|||||||
BUG_ON(skb_shinfo(skb)->frag_list);
|
BUG_ON(skb_shinfo(skb)->frag_list);
|
||||||
|
|
||||||
skb_reset_mac_header(skb);
|
skb_reset_mac_header(skb);
|
||||||
skb->mac_len = skb->nh.raw - skb->data;
|
skb->mac_len = skb->nh.raw - skb->mac.raw;
|
||||||
__skb_pull(skb, skb->mac_len);
|
__skb_pull(skb, skb->mac_len);
|
||||||
|
|
||||||
if (unlikely(skb->ip_summed != CHECKSUM_PARTIAL)) {
|
if (unlikely(skb->ip_summed != CHECKSUM_PARTIAL)) {
|
||||||
@ -1224,7 +1224,8 @@ struct sk_buff *skb_gso_segment(struct sk_buff *skb, int features)
|
|||||||
segs = ERR_PTR(err);
|
segs = ERR_PTR(err);
|
||||||
if (err || skb_gso_ok(skb, features))
|
if (err || skb_gso_ok(skb, features))
|
||||||
break;
|
break;
|
||||||
__skb_push(skb, skb->data - skb->nh.raw);
|
__skb_push(skb, (skb->data -
|
||||||
|
skb_network_header(skb)));
|
||||||
}
|
}
|
||||||
segs = ptype->gso_segment(skb, features);
|
segs = ptype->gso_segment(skb, features);
|
||||||
break;
|
break;
|
||||||
|
@ -42,7 +42,7 @@ static void *__load_pointer(struct sk_buff *skb, int k)
|
|||||||
u8 *ptr = NULL;
|
u8 *ptr = NULL;
|
||||||
|
|
||||||
if (k >= SKF_NET_OFF)
|
if (k >= SKF_NET_OFF)
|
||||||
ptr = skb->nh.raw + k - SKF_NET_OFF;
|
ptr = skb_network_header(skb) + k - SKF_NET_OFF;
|
||||||
else if (k >= SKF_LL_OFF)
|
else if (k >= SKF_LL_OFF)
|
||||||
ptr = skb_mac_header(skb) + k - SKF_LL_OFF;
|
ptr = skb_mac_header(skb) + k - SKF_LL_OFF;
|
||||||
|
|
||||||
|
@ -261,8 +261,8 @@ static int dccp_v6_send_response(struct sock *sk, struct request_sock *req,
|
|||||||
|
|
||||||
if (rxopt->srcrt)
|
if (rxopt->srcrt)
|
||||||
opt = ipv6_invert_rthdr(sk,
|
opt = ipv6_invert_rthdr(sk,
|
||||||
(struct ipv6_rt_hdr *)(pktopts->nh.raw +
|
(struct ipv6_rt_hdr *)(skb_network_header(pktopts) +
|
||||||
rxopt->srcrt));
|
rxopt->srcrt));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opt != NULL && opt->srcrt != NULL) {
|
if (opt != NULL && opt->srcrt != NULL) {
|
||||||
@ -573,8 +573,8 @@ static struct sock *dccp_v6_request_recv_sock(struct sock *sk,
|
|||||||
|
|
||||||
if (rxopt->srcrt)
|
if (rxopt->srcrt)
|
||||||
opt = ipv6_invert_rthdr(sk,
|
opt = ipv6_invert_rthdr(sk,
|
||||||
(struct ipv6_rt_hdr *)(ireq6->pktopts->nh.raw +
|
(struct ipv6_rt_hdr *)(skb_network_header(ireq6->pktopts) +
|
||||||
rxopt->srcrt));
|
rxopt->srcrt));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dst == NULL) {
|
if (dst == NULL) {
|
||||||
|
@ -386,7 +386,7 @@ static int dn_return_short(struct sk_buff *skb)
|
|||||||
__le16 tmp;
|
__le16 tmp;
|
||||||
|
|
||||||
/* Add back headers */
|
/* Add back headers */
|
||||||
skb_push(skb, skb->data - skb->nh.raw);
|
skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
|
|
||||||
if ((skb = skb_unshare(skb, GFP_ATOMIC)) == NULL)
|
if ((skb = skb_unshare(skb, GFP_ATOMIC)) == NULL)
|
||||||
return NET_RX_DROP;
|
return NET_RX_DROP;
|
||||||
@ -425,7 +425,7 @@ static int dn_return_long(struct sk_buff *skb)
|
|||||||
unsigned char tmp[ETH_ALEN];
|
unsigned char tmp[ETH_ALEN];
|
||||||
|
|
||||||
/* Add back all headers */
|
/* Add back all headers */
|
||||||
skb_push(skb, skb->data - skb->nh.raw);
|
skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
|
|
||||||
if ((skb = skb_unshare(skb, GFP_ATOMIC)) == NULL)
|
if ((skb = skb_unshare(skb, GFP_ATOMIC)) == NULL)
|
||||||
return NET_RX_DROP;
|
return NET_RX_DROP;
|
||||||
|
@ -1184,7 +1184,7 @@ static struct sk_buff *inet_gso_segment(struct sk_buff *skb, int features)
|
|||||||
iph->id = htons(id++);
|
iph->id = htons(id++);
|
||||||
iph->tot_len = htons(skb->len - skb->mac_len);
|
iph->tot_len = htons(skb->len - skb->mac_len);
|
||||||
iph->check = 0;
|
iph->check = 0;
|
||||||
iph->check = ip_fast_csum(skb->nh.raw, iph->ihl);
|
iph->check = ip_fast_csum(skb_network_header(skb), iph->ihl);
|
||||||
} while ((skb = skb->next));
|
} while ((skb = skb->next));
|
||||||
|
|
||||||
out:
|
out:
|
||||||
|
@ -154,7 +154,7 @@ static int ah_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
ah = (struct ip_auth_hdr*)skb->data;
|
ah = (struct ip_auth_hdr*)skb->data;
|
||||||
iph = skb->nh.iph;
|
iph = skb->nh.iph;
|
||||||
|
|
||||||
ihl = skb->data - skb->nh.raw;
|
ihl = skb->data - skb_network_header(skb);
|
||||||
memcpy(work_buf, iph, ihl);
|
memcpy(work_buf, iph, ihl);
|
||||||
|
|
||||||
iph->ttl = 0;
|
iph->ttl = 0;
|
||||||
@ -181,7 +181,8 @@ static int ah_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
((struct iphdr*)work_buf)->protocol = ah->nexthdr;
|
((struct iphdr*)work_buf)->protocol = ah->nexthdr;
|
||||||
skb->h.raw = memcpy(skb->nh.raw += ah_hlen, work_buf, ihl);
|
skb->nh.raw += ah_hlen;
|
||||||
|
skb->h.raw = memcpy(skb_network_header(skb), work_buf, ihl);
|
||||||
__skb_pull(skb, ah_hlen + ihl);
|
__skb_pull(skb, ah_hlen + ihl);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -57,9 +57,10 @@ static int esp_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
*(u8*)(trailer->tail + clen-skb->len - 2) = (clen - skb->len)-2;
|
*(u8*)(trailer->tail + clen-skb->len - 2) = (clen - skb->len)-2;
|
||||||
pskb_put(skb, trailer, clen - skb->len);
|
pskb_put(skb, trailer, clen - skb->len);
|
||||||
|
|
||||||
__skb_push(skb, skb->data - skb->nh.raw);
|
__skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
top_iph = skb->nh.iph;
|
top_iph = skb->nh.iph;
|
||||||
esph = (struct ip_esp_hdr *)(skb->nh.raw + top_iph->ihl*4);
|
esph = (struct ip_esp_hdr *)(skb_network_header(skb) +
|
||||||
|
top_iph->ihl * 4);
|
||||||
top_iph->tot_len = htons(skb->len + alen);
|
top_iph->tot_len = htons(skb->len + alen);
|
||||||
*(u8*)(trailer->tail - 1) = top_iph->protocol;
|
*(u8*)(trailer->tail - 1) = top_iph->protocol;
|
||||||
|
|
||||||
@ -222,7 +223,7 @@ static int esp_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
|
|
||||||
if (x->encap) {
|
if (x->encap) {
|
||||||
struct xfrm_encap_tmpl *encap = x->encap;
|
struct xfrm_encap_tmpl *encap = x->encap;
|
||||||
struct udphdr *uh = (void *)(skb->nh.raw + ihl);
|
struct udphdr *uh = (void *)(skb_network_header(skb) + ihl);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* 1) if the NAT-T peer's IP or port changed then
|
* 1) if the NAT-T peer's IP or port changed then
|
||||||
|
@ -484,7 +484,7 @@ void icmp_send(struct sk_buff *skb_in, int type, int code, __be32 info)
|
|||||||
u8 _inner_type, *itp;
|
u8 _inner_type, *itp;
|
||||||
|
|
||||||
itp = skb_header_pointer(skb_in,
|
itp = skb_header_pointer(skb_in,
|
||||||
skb_in->nh.raw +
|
skb_network_header(skb_in) +
|
||||||
(iph->ihl << 2) +
|
(iph->ihl << 2) +
|
||||||
offsetof(struct icmphdr,
|
offsetof(struct icmphdr,
|
||||||
type) -
|
type) -
|
||||||
@ -536,7 +536,7 @@ void icmp_send(struct sk_buff *skb_in, int type, int code, __be32 info)
|
|||||||
icmp_param.data.icmph.un.gateway = info;
|
icmp_param.data.icmph.un.gateway = info;
|
||||||
icmp_param.data.icmph.checksum = 0;
|
icmp_param.data.icmph.checksum = 0;
|
||||||
icmp_param.skb = skb_in;
|
icmp_param.skb = skb_in;
|
||||||
icmp_param.offset = skb_in->nh.raw - skb_in->data;
|
icmp_param.offset = skb_network_offset(skb_in);
|
||||||
icmp_out_count(icmp_param.data.icmph.type);
|
icmp_out_count(icmp_param.data.icmph.type);
|
||||||
inet_sk(icmp_socket->sk)->tos = tos;
|
inet_sk(icmp_socket->sk)->tos = tos;
|
||||||
ipc.addr = iph->saddr;
|
ipc.addr = iph->saddr;
|
||||||
|
@ -658,7 +658,7 @@ static struct sk_buff *ip_frag_reasm(struct ipq *qp, struct net_device *dev)
|
|||||||
}
|
}
|
||||||
|
|
||||||
skb_shinfo(head)->frag_list = head->next;
|
skb_shinfo(head)->frag_list = head->next;
|
||||||
skb_push(head, head->data - head->nh.raw);
|
skb_push(head, head->data - skb_network_header(head));
|
||||||
atomic_sub(head->truesize, &ip_frag_mem);
|
atomic_sub(head->truesize, &ip_frag_mem);
|
||||||
|
|
||||||
for (fp=head->next; fp; fp = fp->next) {
|
for (fp=head->next; fp; fp = fp->next) {
|
||||||
|
@ -40,7 +40,7 @@
|
|||||||
void ip_options_build(struct sk_buff * skb, struct ip_options * opt,
|
void ip_options_build(struct sk_buff * skb, struct ip_options * opt,
|
||||||
__be32 daddr, struct rtable *rt, int is_frag)
|
__be32 daddr, struct rtable *rt, int is_frag)
|
||||||
{
|
{
|
||||||
unsigned char * iph = skb->nh.raw;
|
unsigned char *iph = skb_network_header(skb);
|
||||||
|
|
||||||
memcpy(&(IPCB(skb)->opt), opt, sizeof(struct ip_options));
|
memcpy(&(IPCB(skb)->opt), opt, sizeof(struct ip_options));
|
||||||
memcpy(iph+sizeof(struct iphdr), opt->__data, opt->optlen);
|
memcpy(iph+sizeof(struct iphdr), opt->__data, opt->optlen);
|
||||||
@ -104,7 +104,7 @@ int ip_options_echo(struct ip_options * dopt, struct sk_buff * skb)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
sptr = skb->nh.raw;
|
sptr = skb_network_header(skb);
|
||||||
dptr = dopt->__data;
|
dptr = dopt->__data;
|
||||||
|
|
||||||
if (skb->dst)
|
if (skb->dst)
|
||||||
@ -217,7 +217,7 @@ int ip_options_echo(struct ip_options * dopt, struct sk_buff * skb)
|
|||||||
|
|
||||||
void ip_options_fragment(struct sk_buff * skb)
|
void ip_options_fragment(struct sk_buff * skb)
|
||||||
{
|
{
|
||||||
unsigned char * optptr = skb->nh.raw + sizeof(struct iphdr);
|
unsigned char *optptr = skb_network_header(skb) + sizeof(struct iphdr);
|
||||||
struct ip_options * opt = &(IPCB(skb)->opt);
|
struct ip_options * opt = &(IPCB(skb)->opt);
|
||||||
int l = opt->optlen;
|
int l = opt->optlen;
|
||||||
int optlen;
|
int optlen;
|
||||||
@ -264,7 +264,7 @@ int ip_options_compile(struct ip_options * opt, struct sk_buff * skb)
|
|||||||
|
|
||||||
if (!opt) {
|
if (!opt) {
|
||||||
opt = &(IPCB(skb)->opt);
|
opt = &(IPCB(skb)->opt);
|
||||||
iph = skb->nh.raw;
|
iph = skb_network_header(skb);
|
||||||
opt->optlen = ((struct iphdr *)iph)->ihl*4 - sizeof(struct iphdr);
|
opt->optlen = ((struct iphdr *)iph)->ihl*4 - sizeof(struct iphdr);
|
||||||
optptr = iph + sizeof(struct iphdr);
|
optptr = iph + sizeof(struct iphdr);
|
||||||
opt->is_data = 0;
|
opt->is_data = 0;
|
||||||
@ -563,7 +563,7 @@ void ip_forward_options(struct sk_buff *skb)
|
|||||||
struct ip_options * opt = &(IPCB(skb)->opt);
|
struct ip_options * opt = &(IPCB(skb)->opt);
|
||||||
unsigned char * optptr;
|
unsigned char * optptr;
|
||||||
struct rtable *rt = (struct rtable*)skb->dst;
|
struct rtable *rt = (struct rtable*)skb->dst;
|
||||||
unsigned char *raw = skb->nh.raw;
|
unsigned char *raw = skb_network_header(skb);
|
||||||
|
|
||||||
if (opt->rr_needaddr) {
|
if (opt->rr_needaddr) {
|
||||||
optptr = (unsigned char *)raw + opt->rr;
|
optptr = (unsigned char *)raw + opt->rr;
|
||||||
@ -609,7 +609,7 @@ int ip_options_rcv_srr(struct sk_buff *skb)
|
|||||||
int srrspace, srrptr;
|
int srrspace, srrptr;
|
||||||
__be32 nexthop;
|
__be32 nexthop;
|
||||||
struct iphdr *iph = skb->nh.iph;
|
struct iphdr *iph = skb->nh.iph;
|
||||||
unsigned char * optptr = skb->nh.raw + opt->srr;
|
unsigned char *optptr = skb_network_header(skb) + opt->srr;
|
||||||
struct rtable *rt = (struct rtable*)skb->dst;
|
struct rtable *rt = (struct rtable*)skb->dst;
|
||||||
struct rtable *rt2;
|
struct rtable *rt2;
|
||||||
int err;
|
int err;
|
||||||
|
@ -503,7 +503,7 @@ int ip_fragment(struct sk_buff *skb, int (*output)(struct sk_buff*))
|
|||||||
frag->h.raw = frag->data;
|
frag->h.raw = frag->data;
|
||||||
__skb_push(frag, hlen);
|
__skb_push(frag, hlen);
|
||||||
skb_reset_network_header(frag);
|
skb_reset_network_header(frag);
|
||||||
memcpy(frag->nh.raw, iph, hlen);
|
memcpy(skb_network_header(frag), iph, hlen);
|
||||||
iph = frag->nh.iph;
|
iph = frag->nh.iph;
|
||||||
iph->tot_len = htons(frag->len);
|
iph->tot_len = htons(frag->len);
|
||||||
ip_copy_metadata(frag, skb);
|
ip_copy_metadata(frag, skb);
|
||||||
@ -607,7 +607,7 @@ int ip_fragment(struct sk_buff *skb, int (*output)(struct sk_buff*))
|
|||||||
* Copy the packet header into the new buffer.
|
* Copy the packet header into the new buffer.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
memcpy(skb2->nh.raw, skb->data, hlen);
|
memcpy(skb_network_header(skb2), skb->data, hlen);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Copy a block of the IP datagram.
|
* Copy a block of the IP datagram.
|
||||||
@ -1198,7 +1198,7 @@ int ip_push_pending_frames(struct sock *sk)
|
|||||||
tail_skb = &(skb_shinfo(skb)->frag_list);
|
tail_skb = &(skb_shinfo(skb)->frag_list);
|
||||||
|
|
||||||
/* move skb->data to ip header from ext header */
|
/* move skb->data to ip header from ext header */
|
||||||
if (skb->data < skb->nh.raw)
|
if (skb->data < skb_network_header(skb))
|
||||||
__skb_pull(skb, skb_network_offset(skb));
|
__skb_pull(skb, skb_network_offset(skb));
|
||||||
while ((tmp_skb = __skb_dequeue(&sk->sk_write_queue)) != NULL) {
|
while ((tmp_skb = __skb_dequeue(&sk->sk_write_queue)) != NULL) {
|
||||||
__skb_pull(tmp_skb, skb->h.raw - skb->nh.raw);
|
__skb_pull(tmp_skb, skb->h.raw - skb->nh.raw);
|
||||||
|
@ -273,7 +273,8 @@ void ip_icmp_error(struct sock *sk, struct sk_buff *skb, int err,
|
|||||||
serr->ee.ee_pad = 0;
|
serr->ee.ee_pad = 0;
|
||||||
serr->ee.ee_info = info;
|
serr->ee.ee_info = info;
|
||||||
serr->ee.ee_data = 0;
|
serr->ee.ee_data = 0;
|
||||||
serr->addr_offset = (u8*)&(((struct iphdr*)(skb->h.icmph+1))->daddr) - skb->nh.raw;
|
serr->addr_offset = (u8 *)&(((struct iphdr *)(skb->h.icmph + 1))->daddr) -
|
||||||
|
skb_network_header(skb);
|
||||||
serr->port = port;
|
serr->port = port;
|
||||||
|
|
||||||
skb->h.raw = payload;
|
skb->h.raw = payload;
|
||||||
@ -309,7 +310,7 @@ void ip_local_error(struct sock *sk, int err, __be32 daddr, __be16 port, u32 inf
|
|||||||
serr->ee.ee_pad = 0;
|
serr->ee.ee_pad = 0;
|
||||||
serr->ee.ee_info = info;
|
serr->ee.ee_info = info;
|
||||||
serr->ee.ee_data = 0;
|
serr->ee.ee_data = 0;
|
||||||
serr->addr_offset = (u8*)&iph->daddr - skb->nh.raw;
|
serr->addr_offset = (u8 *)&iph->daddr - skb_network_header(skb);
|
||||||
serr->port = port;
|
serr->port = port;
|
||||||
|
|
||||||
skb->h.raw = skb->tail;
|
skb->h.raw = skb->tail;
|
||||||
@ -355,7 +356,8 @@ int ip_recv_error(struct sock *sk, struct msghdr *msg, int len)
|
|||||||
sin = (struct sockaddr_in *)msg->msg_name;
|
sin = (struct sockaddr_in *)msg->msg_name;
|
||||||
if (sin) {
|
if (sin) {
|
||||||
sin->sin_family = AF_INET;
|
sin->sin_family = AF_INET;
|
||||||
sin->sin_addr.s_addr = *(__be32*)(skb->nh.raw + serr->addr_offset);
|
sin->sin_addr.s_addr = *(__be32 *)(skb_network_header(skb) +
|
||||||
|
serr->addr_offset);
|
||||||
sin->sin_port = serr->port;
|
sin->sin_port = serr->port;
|
||||||
memset(&sin->sin_zero, 0, sizeof(sin->sin_zero));
|
memset(&sin->sin_zero, 0, sizeof(sin->sin_zero));
|
||||||
}
|
}
|
||||||
|
@ -563,7 +563,7 @@ static int ipmr_cache_report(struct sk_buff *pkt, vifi_t vifi, int assert)
|
|||||||
*/
|
*/
|
||||||
msg = (struct igmpmsg*)skb_push(skb, sizeof(struct iphdr));
|
msg = (struct igmpmsg*)skb_push(skb, sizeof(struct iphdr));
|
||||||
skb->nh.raw = skb->h.raw = (u8*)msg;
|
skb->nh.raw = skb->h.raw = (u8*)msg;
|
||||||
memcpy(msg, pkt->nh.raw, sizeof(struct iphdr));
|
memcpy(msg, skb_network_header(pkt), sizeof(struct iphdr));
|
||||||
msg->im_msgtype = IGMPMSG_WHOLEPKT;
|
msg->im_msgtype = IGMPMSG_WHOLEPKT;
|
||||||
msg->im_mbz = 0;
|
msg->im_mbz = 0;
|
||||||
msg->im_vif = reg_vif_num;
|
msg->im_vif = reg_vif_num;
|
||||||
|
@ -338,7 +338,7 @@ static inline int app_tcp_pkt_out(struct ip_vs_conn *cp, struct sk_buff **pskb,
|
|||||||
if (!ip_vs_make_skb_writable(pskb, tcp_offset + sizeof(*th)))
|
if (!ip_vs_make_skb_writable(pskb, tcp_offset + sizeof(*th)))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
th = (struct tcphdr *)((*pskb)->nh.raw + tcp_offset);
|
th = (struct tcphdr *)(skb_network_header(*pskb) + tcp_offset);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Remember seq number in case this pkt gets resized
|
* Remember seq number in case this pkt gets resized
|
||||||
@ -413,7 +413,7 @@ static inline int app_tcp_pkt_in(struct ip_vs_conn *cp, struct sk_buff **pskb,
|
|||||||
if (!ip_vs_make_skb_writable(pskb, tcp_offset + sizeof(*th)))
|
if (!ip_vs_make_skb_writable(pskb, tcp_offset + sizeof(*th)))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
th = (struct tcphdr *)((*pskb)->nh.raw + tcp_offset);
|
th = (struct tcphdr *)(skb_network_header(*pskb) + tcp_offset);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Remember seq number in case this pkt gets resized
|
* Remember seq number in case this pkt gets resized
|
||||||
|
@ -559,7 +559,8 @@ void ip_vs_nat_icmp(struct sk_buff *skb, struct ip_vs_protocol *pp,
|
|||||||
{
|
{
|
||||||
struct iphdr *iph = skb->nh.iph;
|
struct iphdr *iph = skb->nh.iph;
|
||||||
unsigned int icmp_offset = iph->ihl*4;
|
unsigned int icmp_offset = iph->ihl*4;
|
||||||
struct icmphdr *icmph = (struct icmphdr *)(skb->nh.raw + icmp_offset);
|
struct icmphdr *icmph = (struct icmphdr *)(skb_network_header(skb) +
|
||||||
|
icmp_offset);
|
||||||
struct iphdr *ciph = (struct iphdr *)(icmph + 1);
|
struct iphdr *ciph = (struct iphdr *)(icmph + 1);
|
||||||
|
|
||||||
if (inout) {
|
if (inout) {
|
||||||
|
@ -31,7 +31,7 @@ target(struct sk_buff **pskb,
|
|||||||
}
|
}
|
||||||
|
|
||||||
arp = (*pskb)->nh.arph;
|
arp = (*pskb)->nh.arph;
|
||||||
arpptr = (*pskb)->nh.raw + sizeof(*arp);
|
arpptr = skb_network_header(*pskb) + sizeof(*arp);
|
||||||
pln = arp->ar_pln;
|
pln = arp->ar_pln;
|
||||||
hln = arp->ar_hln;
|
hln = arp->ar_hln;
|
||||||
/* We assume that pln and hln were checked in the match */
|
/* We assume that pln and hln were checked in the match */
|
||||||
|
@ -105,7 +105,7 @@ ipv4_prepare(struct sk_buff **pskb, unsigned int hooknum, unsigned int *dataoff,
|
|||||||
return -NF_DROP;
|
return -NF_DROP;
|
||||||
}
|
}
|
||||||
|
|
||||||
*dataoff = (*pskb)->nh.raw - (*pskb)->data + (*pskb)->nh.iph->ihl*4;
|
*dataoff = skb_network_offset(*pskb) + (*pskb)->nh.iph->ihl * 4;
|
||||||
*protonum = (*pskb)->nh.iph->protocol;
|
*protonum = (*pskb)->nh.iph->protocol;
|
||||||
|
|
||||||
return NF_ACCEPT;
|
return NF_ACCEPT;
|
||||||
@ -151,10 +151,9 @@ static unsigned int ipv4_conntrack_help(unsigned int hooknum,
|
|||||||
if (!help || !help->helper)
|
if (!help || !help->helper)
|
||||||
return NF_ACCEPT;
|
return NF_ACCEPT;
|
||||||
|
|
||||||
return help->helper->help(pskb,
|
return help->helper->help(pskb, (skb_network_offset(*pskb) +
|
||||||
(*pskb)->nh.raw - (*pskb)->data
|
(*pskb)->nh.iph->ihl * 4),
|
||||||
+ (*pskb)->nh.iph->ihl*4,
|
ct, ctinfo);
|
||||||
ct, ctinfo);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned int ipv4_conntrack_defrag(unsigned int hooknum,
|
static unsigned int ipv4_conntrack_defrag(unsigned int hooknum,
|
||||||
|
@ -256,7 +256,7 @@ int raw_rcv(struct sock *sk, struct sk_buff *skb)
|
|||||||
}
|
}
|
||||||
nf_reset(skb);
|
nf_reset(skb);
|
||||||
|
|
||||||
skb_push(skb, skb->data - skb->nh.raw);
|
skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
|
|
||||||
raw_rcv_skb(sk, skb);
|
raw_rcv_skb(sk, skb);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -3634,7 +3634,7 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list,
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
skb_set_mac_header(nskb, skb_mac_header(skb) - skb->head);
|
skb_set_mac_header(nskb, skb_mac_header(skb) - skb->head);
|
||||||
nskb->nh.raw = nskb->data + (skb->nh.raw - skb->head);
|
nskb->nh.raw = nskb->data + (skb_network_header(skb) - skb->head);
|
||||||
nskb->h.raw = nskb->data + (skb->h.raw - skb->head);
|
nskb->h.raw = nskb->data + (skb->h.raw - skb->head);
|
||||||
|
|
||||||
skb_reserve(nskb, header);
|
skb_reserve(nskb, header);
|
||||||
|
@ -146,7 +146,7 @@ int xfrm4_rcv_encap(struct sk_buff *skb, __u16 encap_type)
|
|||||||
return 0;
|
return 0;
|
||||||
} else {
|
} else {
|
||||||
#ifdef CONFIG_NETFILTER
|
#ifdef CONFIG_NETFILTER
|
||||||
__skb_push(skb, skb->data - skb->nh.raw);
|
__skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
skb->nh.iph->tot_len = htons(skb->len);
|
skb->nh.iph->tot_len = htons(skb->len);
|
||||||
ip_send_check(skb->nh.iph);
|
ip_send_check(skb->nh.iph);
|
||||||
|
|
||||||
|
@ -98,7 +98,7 @@ static int xfrm4_beet_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
}
|
}
|
||||||
|
|
||||||
skb->nh.raw = skb->data + (phlen - sizeof(*iph));
|
skb->nh.raw = skb->data + (phlen - sizeof(*iph));
|
||||||
memmove(skb->nh.raw, iph, sizeof(*iph));
|
memmove(skb_network_header(skb), iph, sizeof(*iph));
|
||||||
skb->h.raw = skb->data + (phlen + optlen);
|
skb->h.raw = skb->data + (phlen + optlen);
|
||||||
skb->data = skb->h.raw;
|
skb->data = skb->h.raw;
|
||||||
|
|
||||||
@ -112,7 +112,7 @@ static int xfrm4_beet_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
else
|
else
|
||||||
iph->protocol = protocol;
|
iph->protocol = protocol;
|
||||||
iph->check = 0;
|
iph->check = 0;
|
||||||
iph->check = ip_fast_csum(skb->nh.raw, iph->ihl);
|
iph->check = ip_fast_csum(skb_network_header(skb), iph->ihl);
|
||||||
err = 0;
|
err = 0;
|
||||||
out:
|
out:
|
||||||
return err;
|
return err;
|
||||||
|
@ -34,7 +34,7 @@ static int xfrm4_transport_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
|
|
||||||
skb_push(skb, x->props.header_len);
|
skb_push(skb, x->props.header_len);
|
||||||
skb_reset_network_header(skb);
|
skb_reset_network_header(skb);
|
||||||
memmove(skb->nh.raw, iph, ihl);
|
memmove(skb_network_header(skb), iph, ihl);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -51,7 +51,7 @@ static int xfrm4_transport_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
int ihl = skb->data - skb->h.raw;
|
int ihl = skb->data - skb->h.raw;
|
||||||
|
|
||||||
if (skb->h.raw != skb->nh.raw) {
|
if (skb->h.raw != skb->nh.raw) {
|
||||||
memmove(skb->h.raw, skb->nh.raw, ihl);
|
memmove(skb->h.raw, skb_network_header(skb), ihl);
|
||||||
skb->nh.raw = skb->h.raw;
|
skb->nh.raw = skb->h.raw;
|
||||||
}
|
}
|
||||||
skb->nh.iph->tot_len = htons(skb->len + ihl);
|
skb->nh.iph->tot_len = htons(skb->len + ihl);
|
||||||
|
@ -210,7 +210,7 @@ static void
|
|||||||
_decode_session4(struct sk_buff *skb, struct flowi *fl)
|
_decode_session4(struct sk_buff *skb, struct flowi *fl)
|
||||||
{
|
{
|
||||||
struct iphdr *iph = skb->nh.iph;
|
struct iphdr *iph = skb->nh.iph;
|
||||||
u8 *xprth = skb->nh.raw + iph->ihl*4;
|
u8 *xprth = skb_network_header(skb) + iph->ihl * 4;
|
||||||
|
|
||||||
memset(fl, 0, sizeof(struct flowi));
|
memset(fl, 0, sizeof(struct flowi));
|
||||||
if (!(iph->frag_off & htons(IP_MF | IP_OFFSET))) {
|
if (!(iph->frag_off & htons(IP_MF | IP_OFFSET))) {
|
||||||
|
@ -693,7 +693,8 @@ int ipv6_opt_accepted(struct sock *sk, struct sk_buff *skb)
|
|||||||
if (np->rxopt.all) {
|
if (np->rxopt.all) {
|
||||||
if ((opt->hop && (np->rxopt.bits.hopopts ||
|
if ((opt->hop && (np->rxopt.bits.hopopts ||
|
||||||
np->rxopt.bits.ohopopts)) ||
|
np->rxopt.bits.ohopopts)) ||
|
||||||
((IPV6_FLOWINFO_MASK & *(__be32*)skb->nh.raw) &&
|
((IPV6_FLOWINFO_MASK &
|
||||||
|
*(__be32 *)skb_network_header(skb)) &&
|
||||||
np->rxopt.bits.rxflow) ||
|
np->rxopt.bits.rxflow) ||
|
||||||
(opt->srcrt && (np->rxopt.bits.srcrt ||
|
(opt->srcrt && (np->rxopt.bits.srcrt ||
|
||||||
np->rxopt.bits.osrcrt)) ||
|
np->rxopt.bits.osrcrt)) ||
|
||||||
|
@ -238,8 +238,8 @@ static int ah6_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
top_iph = (struct ipv6hdr *)skb->data;
|
top_iph = (struct ipv6hdr *)skb->data;
|
||||||
top_iph->payload_len = htons(skb->len - sizeof(*top_iph));
|
top_iph->payload_len = htons(skb->len - sizeof(*top_iph));
|
||||||
|
|
||||||
nexthdr = *skb->nh.raw;
|
nexthdr = *skb_network_header(skb);
|
||||||
*skb->nh.raw = IPPROTO_AH;
|
*skb_network_header(skb) = IPPROTO_AH;
|
||||||
|
|
||||||
/* When there are no extension headers, we only need to save the first
|
/* When there are no extension headers, we only need to save the first
|
||||||
* 8 bytes of the base IP header.
|
* 8 bytes of the base IP header.
|
||||||
@ -341,7 +341,7 @@ static int ah6_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
pskb_expand_head(skb, 0, 0, GFP_ATOMIC))
|
pskb_expand_head(skb, 0, 0, GFP_ATOMIC))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
hdr_len = skb->data - skb->nh.raw;
|
hdr_len = skb->data - skb_network_header(skb);
|
||||||
ah = (struct ipv6_auth_hdr*)skb->data;
|
ah = (struct ipv6_auth_hdr*)skb->data;
|
||||||
ahp = x->data;
|
ahp = x->data;
|
||||||
nexthdr = ah->nexthdr;
|
nexthdr = ah->nexthdr;
|
||||||
@ -354,7 +354,7 @@ static int ah6_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
if (!pskb_may_pull(skb, ah_hlen))
|
if (!pskb_may_pull(skb, ah_hlen))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
tmp_hdr = kmemdup(skb->nh.raw, hdr_len, GFP_ATOMIC);
|
tmp_hdr = kmemdup(skb_network_header(skb), hdr_len, GFP_ATOMIC);
|
||||||
if (!tmp_hdr)
|
if (!tmp_hdr)
|
||||||
goto out;
|
goto out;
|
||||||
if (ipv6_clear_mutable_options(skb->nh.ipv6h, hdr_len, XFRM_POLICY_IN))
|
if (ipv6_clear_mutable_options(skb->nh.ipv6h, hdr_len, XFRM_POLICY_IN))
|
||||||
@ -382,7 +382,9 @@ static int ah6_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
skb->h.raw = memcpy(skb->nh.raw += ah_hlen, tmp_hdr, hdr_len);
|
skb->nh.raw += ah_hlen;
|
||||||
|
memcpy(skb_network_header(skb), tmp_hdr, hdr_len);
|
||||||
|
skb->h.raw = skb->nh.raw;
|
||||||
__skb_pull(skb, ah_hlen + hdr_len);
|
__skb_pull(skb, ah_hlen + hdr_len);
|
||||||
|
|
||||||
kfree(tmp_hdr);
|
kfree(tmp_hdr);
|
||||||
|
@ -227,7 +227,8 @@ void ipv6_icmp_error(struct sock *sk, struct sk_buff *skb, int err,
|
|||||||
serr->ee.ee_pad = 0;
|
serr->ee.ee_pad = 0;
|
||||||
serr->ee.ee_info = info;
|
serr->ee.ee_info = info;
|
||||||
serr->ee.ee_data = 0;
|
serr->ee.ee_data = 0;
|
||||||
serr->addr_offset = (u8*)&(((struct ipv6hdr*)(icmph+1))->daddr) - skb->nh.raw;
|
serr->addr_offset = (u8 *)&(((struct ipv6hdr *)(icmph + 1))->daddr) -
|
||||||
|
skb_network_header(skb);
|
||||||
serr->port = port;
|
serr->port = port;
|
||||||
|
|
||||||
skb->h.raw = payload;
|
skb->h.raw = payload;
|
||||||
@ -264,7 +265,7 @@ void ipv6_local_error(struct sock *sk, int err, struct flowi *fl, u32 info)
|
|||||||
serr->ee.ee_pad = 0;
|
serr->ee.ee_pad = 0;
|
||||||
serr->ee.ee_info = info;
|
serr->ee.ee_info = info;
|
||||||
serr->ee.ee_data = 0;
|
serr->ee.ee_data = 0;
|
||||||
serr->addr_offset = (u8*)&iph->daddr - skb->nh.raw;
|
serr->addr_offset = (u8 *)&iph->daddr - skb_network_header(skb);
|
||||||
serr->port = fl->fl_ip_dport;
|
serr->port = fl->fl_ip_dport;
|
||||||
|
|
||||||
skb->h.raw = skb->tail;
|
skb->h.raw = skb->tail;
|
||||||
@ -310,21 +311,24 @@ int ipv6_recv_error(struct sock *sk, struct msghdr *msg, int len)
|
|||||||
|
|
||||||
sin = (struct sockaddr_in6 *)msg->msg_name;
|
sin = (struct sockaddr_in6 *)msg->msg_name;
|
||||||
if (sin) {
|
if (sin) {
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
sin->sin6_family = AF_INET6;
|
sin->sin6_family = AF_INET6;
|
||||||
sin->sin6_flowinfo = 0;
|
sin->sin6_flowinfo = 0;
|
||||||
sin->sin6_port = serr->port;
|
sin->sin6_port = serr->port;
|
||||||
sin->sin6_scope_id = 0;
|
sin->sin6_scope_id = 0;
|
||||||
if (serr->ee.ee_origin == SO_EE_ORIGIN_ICMP6) {
|
if (serr->ee.ee_origin == SO_EE_ORIGIN_ICMP6) {
|
||||||
ipv6_addr_copy(&sin->sin6_addr,
|
ipv6_addr_copy(&sin->sin6_addr,
|
||||||
(struct in6_addr *)(skb->nh.raw + serr->addr_offset));
|
(struct in6_addr *)(nh + serr->addr_offset));
|
||||||
if (np->sndflow)
|
if (np->sndflow)
|
||||||
sin->sin6_flowinfo = *(__be32*)(skb->nh.raw + serr->addr_offset - 24) & IPV6_FLOWINFO_MASK;
|
sin->sin6_flowinfo =
|
||||||
|
(*(__be32 *)(nh + serr->addr_offset - 24) &
|
||||||
|
IPV6_FLOWINFO_MASK);
|
||||||
if (ipv6_addr_type(&sin->sin6_addr) & IPV6_ADDR_LINKLOCAL)
|
if (ipv6_addr_type(&sin->sin6_addr) & IPV6_ADDR_LINKLOCAL)
|
||||||
sin->sin6_scope_id = IP6CB(skb)->iif;
|
sin->sin6_scope_id = IP6CB(skb)->iif;
|
||||||
} else {
|
} else {
|
||||||
ipv6_addr_set(&sin->sin6_addr, 0, 0,
|
ipv6_addr_set(&sin->sin6_addr, 0, 0,
|
||||||
htonl(0xffff),
|
htonl(0xffff),
|
||||||
*(__be32*)(skb->nh.raw + serr->addr_offset));
|
*(__be32 *)(nh + serr->addr_offset));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -382,6 +386,7 @@ int datagram_recv_ctl(struct sock *sk, struct msghdr *msg, struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
struct ipv6_pinfo *np = inet6_sk(sk);
|
struct ipv6_pinfo *np = inet6_sk(sk);
|
||||||
struct inet6_skb_parm *opt = IP6CB(skb);
|
struct inet6_skb_parm *opt = IP6CB(skb);
|
||||||
|
unsigned char *nh = skb_network_header(skb);
|
||||||
|
|
||||||
if (np->rxopt.bits.rxinfo) {
|
if (np->rxopt.bits.rxinfo) {
|
||||||
struct in6_pktinfo src_info;
|
struct in6_pktinfo src_info;
|
||||||
@ -401,14 +406,14 @@ int datagram_recv_ctl(struct sock *sk, struct msghdr *msg, struct sk_buff *skb)
|
|||||||
put_cmsg(msg, SOL_IPV6, IPV6_TCLASS, sizeof(tclass), &tclass);
|
put_cmsg(msg, SOL_IPV6, IPV6_TCLASS, sizeof(tclass), &tclass);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (np->rxopt.bits.rxflow && (*(__be32*)skb->nh.raw & IPV6_FLOWINFO_MASK)) {
|
if (np->rxopt.bits.rxflow && (*(__be32 *)nh & IPV6_FLOWINFO_MASK)) {
|
||||||
__be32 flowinfo = *(__be32*)skb->nh.raw & IPV6_FLOWINFO_MASK;
|
__be32 flowinfo = *(__be32 *)nh & IPV6_FLOWINFO_MASK;
|
||||||
put_cmsg(msg, SOL_IPV6, IPV6_FLOWINFO, sizeof(flowinfo), &flowinfo);
|
put_cmsg(msg, SOL_IPV6, IPV6_FLOWINFO, sizeof(flowinfo), &flowinfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* HbH is allowed only once */
|
/* HbH is allowed only once */
|
||||||
if (np->rxopt.bits.hopopts && opt->hop) {
|
if (np->rxopt.bits.hopopts && opt->hop) {
|
||||||
u8 *ptr = skb->nh.raw + opt->hop;
|
u8 *ptr = nh + opt->hop;
|
||||||
put_cmsg(msg, SOL_IPV6, IPV6_HOPOPTS, (ptr[1]+1)<<3, ptr);
|
put_cmsg(msg, SOL_IPV6, IPV6_HOPOPTS, (ptr[1]+1)<<3, ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -428,7 +433,7 @@ int datagram_recv_ctl(struct sock *sk, struct msghdr *msg, struct sk_buff *skb)
|
|||||||
|
|
||||||
while (off <= opt->lastopt) {
|
while (off <= opt->lastopt) {
|
||||||
unsigned len;
|
unsigned len;
|
||||||
u8 *ptr = skb->nh.raw + off;
|
u8 *ptr = nh + off;
|
||||||
|
|
||||||
switch(nexthdr) {
|
switch(nexthdr) {
|
||||||
case IPPROTO_DSTOPTS:
|
case IPPROTO_DSTOPTS:
|
||||||
@ -470,19 +475,19 @@ int datagram_recv_ctl(struct sock *sk, struct msghdr *msg, struct sk_buff *skb)
|
|||||||
put_cmsg(msg, SOL_IPV6, IPV6_2292HOPLIMIT, sizeof(hlim), &hlim);
|
put_cmsg(msg, SOL_IPV6, IPV6_2292HOPLIMIT, sizeof(hlim), &hlim);
|
||||||
}
|
}
|
||||||
if (np->rxopt.bits.ohopopts && opt->hop) {
|
if (np->rxopt.bits.ohopopts && opt->hop) {
|
||||||
u8 *ptr = skb->nh.raw + opt->hop;
|
u8 *ptr = nh + opt->hop;
|
||||||
put_cmsg(msg, SOL_IPV6, IPV6_2292HOPOPTS, (ptr[1]+1)<<3, ptr);
|
put_cmsg(msg, SOL_IPV6, IPV6_2292HOPOPTS, (ptr[1]+1)<<3, ptr);
|
||||||
}
|
}
|
||||||
if (np->rxopt.bits.odstopts && opt->dst0) {
|
if (np->rxopt.bits.odstopts && opt->dst0) {
|
||||||
u8 *ptr = skb->nh.raw + opt->dst0;
|
u8 *ptr = nh + opt->dst0;
|
||||||
put_cmsg(msg, SOL_IPV6, IPV6_2292DSTOPTS, (ptr[1]+1)<<3, ptr);
|
put_cmsg(msg, SOL_IPV6, IPV6_2292DSTOPTS, (ptr[1]+1)<<3, ptr);
|
||||||
}
|
}
|
||||||
if (np->rxopt.bits.osrcrt && opt->srcrt) {
|
if (np->rxopt.bits.osrcrt && opt->srcrt) {
|
||||||
struct ipv6_rt_hdr *rthdr = (struct ipv6_rt_hdr *)(skb->nh.raw + opt->srcrt);
|
struct ipv6_rt_hdr *rthdr = (struct ipv6_rt_hdr *)(nh + opt->srcrt);
|
||||||
put_cmsg(msg, SOL_IPV6, IPV6_2292RTHDR, (rthdr->hdrlen+1) << 3, rthdr);
|
put_cmsg(msg, SOL_IPV6, IPV6_2292RTHDR, (rthdr->hdrlen+1) << 3, rthdr);
|
||||||
}
|
}
|
||||||
if (np->rxopt.bits.odstopts && opt->dst1) {
|
if (np->rxopt.bits.odstopts && opt->dst1) {
|
||||||
u8 *ptr = skb->nh.raw + opt->dst1;
|
u8 *ptr = nh + opt->dst1;
|
||||||
put_cmsg(msg, SOL_IPV6, IPV6_2292DSTOPTS, (ptr[1]+1)<<3, ptr);
|
put_cmsg(msg, SOL_IPV6, IPV6_2292DSTOPTS, (ptr[1]+1)<<3, ptr);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -92,8 +92,8 @@ static int esp6_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
top_iph = (struct ipv6hdr *)__skb_push(skb, hdr_len);
|
top_iph = (struct ipv6hdr *)__skb_push(skb, hdr_len);
|
||||||
esph = (struct ipv6_esp_hdr *)skb->h.raw;
|
esph = (struct ipv6_esp_hdr *)skb->h.raw;
|
||||||
top_iph->payload_len = htons(skb->len + alen - sizeof(*top_iph));
|
top_iph->payload_len = htons(skb->len + alen - sizeof(*top_iph));
|
||||||
*(u8*)(trailer->tail - 1) = *skb->nh.raw;
|
*(u8 *)(trailer->tail - 1) = *skb_network_header(skb);
|
||||||
*skb->nh.raw = IPPROTO_ESP;
|
*skb_network_header(skb) = IPPROTO_ESP;
|
||||||
|
|
||||||
esph->spi = x->id.spi;
|
esph->spi = x->id.spi;
|
||||||
esph->seq_no = htonl(++x->replay.oseq);
|
esph->seq_no = htonl(++x->replay.oseq);
|
||||||
|
@ -50,13 +50,14 @@
|
|||||||
|
|
||||||
int ipv6_find_tlv(struct sk_buff *skb, int offset, int type)
|
int ipv6_find_tlv(struct sk_buff *skb, int offset, int type)
|
||||||
{
|
{
|
||||||
int packet_len = skb->tail - skb->nh.raw;
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
int packet_len = skb->tail - nh;
|
||||||
struct ipv6_opt_hdr *hdr;
|
struct ipv6_opt_hdr *hdr;
|
||||||
int len;
|
int len;
|
||||||
|
|
||||||
if (offset + 2 > packet_len)
|
if (offset + 2 > packet_len)
|
||||||
goto bad;
|
goto bad;
|
||||||
hdr = (struct ipv6_opt_hdr*)(skb->nh.raw + offset);
|
hdr = (struct ipv6_opt_hdr *)(nh + offset);
|
||||||
len = ((hdr->hdrlen + 1) << 3);
|
len = ((hdr->hdrlen + 1) << 3);
|
||||||
|
|
||||||
if (offset + len > packet_len)
|
if (offset + len > packet_len)
|
||||||
@ -66,7 +67,7 @@ int ipv6_find_tlv(struct sk_buff *skb, int offset, int type)
|
|||||||
len -= 2;
|
len -= 2;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
int opttype = skb->nh.raw[offset];
|
int opttype = nh[offset];
|
||||||
int optlen;
|
int optlen;
|
||||||
|
|
||||||
if (opttype == type)
|
if (opttype == type)
|
||||||
@ -77,7 +78,7 @@ int ipv6_find_tlv(struct sk_buff *skb, int offset, int type)
|
|||||||
optlen = 1;
|
optlen = 1;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
optlen = skb->nh.raw[offset + 1] + 2;
|
optlen = nh[offset + 1] + 2;
|
||||||
if (optlen > len)
|
if (optlen > len)
|
||||||
goto bad;
|
goto bad;
|
||||||
break;
|
break;
|
||||||
@ -113,7 +114,7 @@ static int ip6_tlvopt_unknown(struct sk_buff **skbp, int optoff)
|
|||||||
{
|
{
|
||||||
struct sk_buff *skb = *skbp;
|
struct sk_buff *skb = *skbp;
|
||||||
|
|
||||||
switch ((skb->nh.raw[optoff] & 0xC0) >> 6) {
|
switch ((skb_network_header(skb)[optoff] & 0xC0) >> 6) {
|
||||||
case 0: /* ignore */
|
case 0: /* ignore */
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
@ -141,6 +142,7 @@ static int ip6_parse_tlv(struct tlvtype_proc *procs, struct sk_buff **skbp)
|
|||||||
{
|
{
|
||||||
struct sk_buff *skb = *skbp;
|
struct sk_buff *skb = *skbp;
|
||||||
struct tlvtype_proc *curr;
|
struct tlvtype_proc *curr;
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
int off = skb->h.raw - skb->nh.raw;
|
int off = skb->h.raw - skb->nh.raw;
|
||||||
int len = ((skb->h.raw[1]+1)<<3);
|
int len = ((skb->h.raw[1]+1)<<3);
|
||||||
|
|
||||||
@ -151,9 +153,9 @@ static int ip6_parse_tlv(struct tlvtype_proc *procs, struct sk_buff **skbp)
|
|||||||
len -= 2;
|
len -= 2;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
int optlen = skb->nh.raw[off+1]+2;
|
int optlen = nh[off + 1] + 2;
|
||||||
|
|
||||||
switch (skb->nh.raw[off]) {
|
switch (nh[off]) {
|
||||||
case IPV6_TLV_PAD0:
|
case IPV6_TLV_PAD0:
|
||||||
optlen = 1;
|
optlen = 1;
|
||||||
break;
|
break;
|
||||||
@ -165,7 +167,7 @@ static int ip6_parse_tlv(struct tlvtype_proc *procs, struct sk_buff **skbp)
|
|||||||
if (optlen > len)
|
if (optlen > len)
|
||||||
goto bad;
|
goto bad;
|
||||||
for (curr=procs; curr->type >= 0; curr++) {
|
for (curr=procs; curr->type >= 0; curr++) {
|
||||||
if (curr->type == skb->nh.raw[off]) {
|
if (curr->type == nh[off]) {
|
||||||
/* type specific length/alignment
|
/* type specific length/alignment
|
||||||
checks will be performed in the
|
checks will be performed in the
|
||||||
func(). */
|
func(). */
|
||||||
@ -211,7 +213,7 @@ static int ipv6_dest_hao(struct sk_buff **skbp, int optoff)
|
|||||||
opt->dsthao = opt->dst1;
|
opt->dsthao = opt->dst1;
|
||||||
opt->dst1 = 0;
|
opt->dst1 = 0;
|
||||||
|
|
||||||
hao = (struct ipv6_destopt_hao *)(skb->nh.raw + optoff);
|
hao = (struct ipv6_destopt_hao *)(skb_network_header(skb) + optoff);
|
||||||
|
|
||||||
if (hao->length != 16) {
|
if (hao->length != 16) {
|
||||||
LIMIT_NETDEBUG(
|
LIMIT_NETDEBUG(
|
||||||
@ -244,8 +246,9 @@ static int ipv6_dest_hao(struct sk_buff **skbp, int optoff)
|
|||||||
|
|
||||||
/* update all variable using below by copied skbuff */
|
/* update all variable using below by copied skbuff */
|
||||||
*skbp = skb = skb2;
|
*skbp = skb = skb2;
|
||||||
hao = (struct ipv6_destopt_hao *)(skb2->nh.raw + optoff);
|
hao = (struct ipv6_destopt_hao *)(skb_network_header(skb2) +
|
||||||
ipv6h = (struct ipv6hdr *)skb2->nh.raw;
|
optoff);
|
||||||
|
ipv6h = skb2->nh.ipv6h;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (skb->ip_summed == CHECKSUM_COMPLETE)
|
if (skb->ip_summed == CHECKSUM_COMPLETE)
|
||||||
@ -406,7 +409,8 @@ static int ipv6_rthdr_rcv(struct sk_buff **skbp)
|
|||||||
default:
|
default:
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
||||||
IPSTATS_MIB_INHDRERRORS);
|
IPSTATS_MIB_INHDRERRORS);
|
||||||
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD, (&hdr->type) - skb->nh.raw);
|
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD,
|
||||||
|
(&hdr->type) - skb_network_header(skb));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -443,7 +447,7 @@ static int ipv6_rthdr_rcv(struct sk_buff **skbp)
|
|||||||
skb->h.raw += (hdr->hdrlen + 1) << 3;
|
skb->h.raw += (hdr->hdrlen + 1) << 3;
|
||||||
opt->dst0 = opt->dst1;
|
opt->dst0 = opt->dst1;
|
||||||
opt->dst1 = 0;
|
opt->dst1 = 0;
|
||||||
opt->nhoff = (&hdr->nexthdr) - skb->nh.raw;
|
opt->nhoff = (&hdr->nexthdr) - skb_network_header(skb);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -452,7 +456,9 @@ static int ipv6_rthdr_rcv(struct sk_buff **skbp)
|
|||||||
if (hdr->hdrlen & 0x01) {
|
if (hdr->hdrlen & 0x01) {
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
||||||
IPSTATS_MIB_INHDRERRORS);
|
IPSTATS_MIB_INHDRERRORS);
|
||||||
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD, (&hdr->hdrlen) - skb->nh.raw);
|
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD,
|
||||||
|
((&hdr->hdrlen) -
|
||||||
|
skb_network_header(skb)));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -479,7 +485,9 @@ static int ipv6_rthdr_rcv(struct sk_buff **skbp)
|
|||||||
if (hdr->segments_left > n) {
|
if (hdr->segments_left > n) {
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
||||||
IPSTATS_MIB_INHDRERRORS);
|
IPSTATS_MIB_INHDRERRORS);
|
||||||
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD, (&hdr->segments_left) - skb->nh.raw);
|
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD,
|
||||||
|
((&hdr->segments_left) -
|
||||||
|
skb_network_header(skb)));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -547,7 +555,7 @@ static int ipv6_rthdr_rcv(struct sk_buff **skbp)
|
|||||||
dst_release(xchg(&skb->dst, NULL));
|
dst_release(xchg(&skb->dst, NULL));
|
||||||
ip6_route_input(skb);
|
ip6_route_input(skb);
|
||||||
if (skb->dst->error) {
|
if (skb->dst->error) {
|
||||||
skb_push(skb, skb->data - skb->nh.raw);
|
skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
dst_input(skb);
|
dst_input(skb);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -565,7 +573,7 @@ static int ipv6_rthdr_rcv(struct sk_buff **skbp)
|
|||||||
goto looped_back;
|
goto looped_back;
|
||||||
}
|
}
|
||||||
|
|
||||||
skb_push(skb, skb->data - skb->nh.raw);
|
skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
dst_input(skb);
|
dst_input(skb);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -656,13 +664,14 @@ EXPORT_SYMBOL_GPL(ipv6_invert_rthdr);
|
|||||||
static int ipv6_hop_ra(struct sk_buff **skbp, int optoff)
|
static int ipv6_hop_ra(struct sk_buff **skbp, int optoff)
|
||||||
{
|
{
|
||||||
struct sk_buff *skb = *skbp;
|
struct sk_buff *skb = *skbp;
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
|
||||||
if (skb->nh.raw[optoff+1] == 2) {
|
if (nh[optoff + 1] == 2) {
|
||||||
IP6CB(skb)->ra = optoff;
|
IP6CB(skb)->ra = optoff;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
LIMIT_NETDEBUG(KERN_DEBUG "ipv6_hop_ra: wrong RA length %d\n",
|
LIMIT_NETDEBUG(KERN_DEBUG "ipv6_hop_ra: wrong RA length %d\n",
|
||||||
skb->nh.raw[optoff+1]);
|
nh[optoff + 1]);
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -672,17 +681,18 @@ static int ipv6_hop_ra(struct sk_buff **skbp, int optoff)
|
|||||||
static int ipv6_hop_jumbo(struct sk_buff **skbp, int optoff)
|
static int ipv6_hop_jumbo(struct sk_buff **skbp, int optoff)
|
||||||
{
|
{
|
||||||
struct sk_buff *skb = *skbp;
|
struct sk_buff *skb = *skbp;
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
u32 pkt_len;
|
u32 pkt_len;
|
||||||
|
|
||||||
if (skb->nh.raw[optoff+1] != 4 || (optoff&3) != 2) {
|
if (nh[optoff + 1] != 4 || (optoff & 3) != 2) {
|
||||||
LIMIT_NETDEBUG(KERN_DEBUG "ipv6_hop_jumbo: wrong jumbo opt length/alignment %d\n",
|
LIMIT_NETDEBUG(KERN_DEBUG "ipv6_hop_jumbo: wrong jumbo opt length/alignment %d\n",
|
||||||
skb->nh.raw[optoff+1]);
|
nh[optoff+1]);
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
||||||
IPSTATS_MIB_INHDRERRORS);
|
IPSTATS_MIB_INHDRERRORS);
|
||||||
goto drop;
|
goto drop;
|
||||||
}
|
}
|
||||||
|
|
||||||
pkt_len = ntohl(*(__be32*)(skb->nh.raw+optoff+2));
|
pkt_len = ntohl(*(__be32 *)(nh + optoff + 2));
|
||||||
if (pkt_len <= IPV6_MAXPLEN) {
|
if (pkt_len <= IPV6_MAXPLEN) {
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst), IPSTATS_MIB_INHDRERRORS);
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst), IPSTATS_MIB_INHDRERRORS);
|
||||||
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD, optoff+2);
|
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD, optoff+2);
|
||||||
@ -727,7 +737,7 @@ int ipv6_parse_hopopts(struct sk_buff **skbp)
|
|||||||
struct inet6_skb_parm *opt = IP6CB(skb);
|
struct inet6_skb_parm *opt = IP6CB(skb);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* skb->nh.raw is equal to skb->data, and
|
* skb_network_header(skb) is equal to skb->data, and
|
||||||
* skb->h.raw - skb->nh.raw is always equal to
|
* skb->h.raw - skb->nh.raw is always equal to
|
||||||
* sizeof(struct ipv6hdr) by definition of
|
* sizeof(struct ipv6hdr) by definition of
|
||||||
* hop-by-hop options.
|
* hop-by-hop options.
|
||||||
|
@ -284,7 +284,8 @@ static void mip6_addr_swap(struct sk_buff *skb)
|
|||||||
if (opt->dsthao) {
|
if (opt->dsthao) {
|
||||||
off = ipv6_find_tlv(skb, opt->dsthao, IPV6_TLV_HAO);
|
off = ipv6_find_tlv(skb, opt->dsthao, IPV6_TLV_HAO);
|
||||||
if (likely(off >= 0)) {
|
if (likely(off >= 0)) {
|
||||||
hao = (struct ipv6_destopt_hao *)(skb->nh.raw + off);
|
hao = (struct ipv6_destopt_hao *)
|
||||||
|
(skb_network_header(skb) + off);
|
||||||
ipv6_addr_copy(&tmp, &iph->saddr);
|
ipv6_addr_copy(&tmp, &iph->saddr);
|
||||||
ipv6_addr_copy(&iph->saddr, &hao->addr);
|
ipv6_addr_copy(&iph->saddr, &hao->addr);
|
||||||
ipv6_addr_copy(&hao->addr, &tmp);
|
ipv6_addr_copy(&hao->addr, &tmp);
|
||||||
|
@ -163,7 +163,7 @@ static inline int ip6_input_finish(struct sk_buff *skb)
|
|||||||
if (!pskb_pull(skb, skb->h.raw - skb->data))
|
if (!pskb_pull(skb, skb->h.raw - skb->data))
|
||||||
goto discard;
|
goto discard;
|
||||||
nhoff = IP6CB(skb)->nhoff;
|
nhoff = IP6CB(skb)->nhoff;
|
||||||
nexthdr = skb->nh.raw[nhoff];
|
nexthdr = skb_network_header(skb)[nhoff];
|
||||||
|
|
||||||
raw_sk = sk_head(&raw_v6_htable[nexthdr & (MAX_INET_PROTOS - 1)]);
|
raw_sk = sk_head(&raw_v6_htable[nexthdr & (MAX_INET_PROTOS - 1)]);
|
||||||
if (raw_sk && !ipv6_raw_deliver(skb, nexthdr))
|
if (raw_sk && !ipv6_raw_deliver(skb, nexthdr))
|
||||||
@ -181,7 +181,7 @@ static inline int ip6_input_finish(struct sk_buff *skb)
|
|||||||
indefinitely. */
|
indefinitely. */
|
||||||
nf_reset(skb);
|
nf_reset(skb);
|
||||||
|
|
||||||
skb_postpull_rcsum(skb, skb->nh.raw,
|
skb_postpull_rcsum(skb, skb_network_header(skb),
|
||||||
skb->h.raw - skb->nh.raw);
|
skb->h.raw - skb->nh.raw);
|
||||||
hdr = skb->nh.ipv6h;
|
hdr = skb->nh.ipv6h;
|
||||||
if (ipv6_addr_is_multicast(&hdr->daddr) &&
|
if (ipv6_addr_is_multicast(&hdr->daddr) &&
|
||||||
|
@ -323,10 +323,11 @@ static int ip6_forward_proxy_check(struct sk_buff *skb)
|
|||||||
if (nexthdr == IPPROTO_ICMPV6) {
|
if (nexthdr == IPPROTO_ICMPV6) {
|
||||||
struct icmp6hdr *icmp6;
|
struct icmp6hdr *icmp6;
|
||||||
|
|
||||||
if (!pskb_may_pull(skb, skb->nh.raw + offset + 1 - skb->data))
|
if (!pskb_may_pull(skb, (skb_network_header(skb) +
|
||||||
|
offset + 1 - skb->data)))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
icmp6 = (struct icmp6hdr *)(skb->nh.raw + offset);
|
icmp6 = (struct icmp6hdr *)(skb_network_header(skb) + offset);
|
||||||
|
|
||||||
switch (icmp6->icmp6_type) {
|
switch (icmp6->icmp6_type) {
|
||||||
case NDISC_ROUTER_SOLICITATION:
|
case NDISC_ROUTER_SOLICITATION:
|
||||||
@ -392,7 +393,7 @@ int ip6_forward(struct sk_buff *skb)
|
|||||||
* that different fragments will go along one path. --ANK
|
* that different fragments will go along one path. --ANK
|
||||||
*/
|
*/
|
||||||
if (opt->ra) {
|
if (opt->ra) {
|
||||||
u8 *ptr = skb->nh.raw + opt->ra;
|
u8 *ptr = skb_network_header(skb) + opt->ra;
|
||||||
if (ip6_call_ra_chain(skb, (ptr[2]<<8) + ptr[3]))
|
if (ip6_call_ra_chain(skb, (ptr[2]<<8) + ptr[3]))
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -527,7 +528,7 @@ int ip6_find_1stfragopt(struct sk_buff *skb, u8 **nexthdr)
|
|||||||
{
|
{
|
||||||
u16 offset = sizeof(struct ipv6hdr);
|
u16 offset = sizeof(struct ipv6hdr);
|
||||||
struct ipv6_opt_hdr *exthdr = (struct ipv6_opt_hdr*)(skb->nh.ipv6h + 1);
|
struct ipv6_opt_hdr *exthdr = (struct ipv6_opt_hdr*)(skb->nh.ipv6h + 1);
|
||||||
unsigned int packet_len = skb->tail - skb->nh.raw;
|
unsigned int packet_len = skb->tail - skb_network_header(skb);
|
||||||
int found_rhdr = 0;
|
int found_rhdr = 0;
|
||||||
*nexthdr = &skb->nh.ipv6h->nexthdr;
|
*nexthdr = &skb->nh.ipv6h->nexthdr;
|
||||||
|
|
||||||
@ -554,7 +555,8 @@ int ip6_find_1stfragopt(struct sk_buff *skb, u8 **nexthdr)
|
|||||||
|
|
||||||
offset += ipv6_optlen(exthdr);
|
offset += ipv6_optlen(exthdr);
|
||||||
*nexthdr = &exthdr->nexthdr;
|
*nexthdr = &exthdr->nexthdr;
|
||||||
exthdr = (struct ipv6_opt_hdr*)(skb->nh.raw + offset);
|
exthdr = (struct ipv6_opt_hdr *)(skb_network_header(skb) +
|
||||||
|
offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
return offset;
|
return offset;
|
||||||
@ -620,7 +622,7 @@ static int ip6_fragment(struct sk_buff *skb, int (*output)(struct sk_buff *))
|
|||||||
/* BUILD HEADER */
|
/* BUILD HEADER */
|
||||||
|
|
||||||
*prevhdr = NEXTHDR_FRAGMENT;
|
*prevhdr = NEXTHDR_FRAGMENT;
|
||||||
tmp_hdr = kmemdup(skb->nh.raw, hlen, GFP_ATOMIC);
|
tmp_hdr = kmemdup(skb_network_header(skb), hlen, GFP_ATOMIC);
|
||||||
if (!tmp_hdr) {
|
if (!tmp_hdr) {
|
||||||
IP6_INC_STATS(ip6_dst_idev(skb->dst), IPSTATS_MIB_FRAGFAILS);
|
IP6_INC_STATS(ip6_dst_idev(skb->dst), IPSTATS_MIB_FRAGFAILS);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
@ -630,7 +632,7 @@ static int ip6_fragment(struct sk_buff *skb, int (*output)(struct sk_buff *))
|
|||||||
fh = (struct frag_hdr*)__skb_push(skb, sizeof(struct frag_hdr));
|
fh = (struct frag_hdr*)__skb_push(skb, sizeof(struct frag_hdr));
|
||||||
__skb_push(skb, hlen);
|
__skb_push(skb, hlen);
|
||||||
skb_reset_network_header(skb);
|
skb_reset_network_header(skb);
|
||||||
memcpy(skb->nh.raw, tmp_hdr, hlen);
|
memcpy(skb_network_header(skb), tmp_hdr, hlen);
|
||||||
|
|
||||||
ipv6_select_ident(skb, fh);
|
ipv6_select_ident(skb, fh);
|
||||||
fh->nexthdr = nexthdr;
|
fh->nexthdr = nexthdr;
|
||||||
@ -654,7 +656,8 @@ static int ip6_fragment(struct sk_buff *skb, int (*output)(struct sk_buff *))
|
|||||||
fh = (struct frag_hdr*)__skb_push(frag, sizeof(struct frag_hdr));
|
fh = (struct frag_hdr*)__skb_push(frag, sizeof(struct frag_hdr));
|
||||||
__skb_push(frag, hlen);
|
__skb_push(frag, hlen);
|
||||||
skb_reset_network_header(frag);
|
skb_reset_network_header(frag);
|
||||||
memcpy(frag->nh.raw, tmp_hdr, hlen);
|
memcpy(skb_network_header(frag), tmp_hdr,
|
||||||
|
hlen);
|
||||||
offset += skb->len - hlen - sizeof(struct frag_hdr);
|
offset += skb->len - hlen - sizeof(struct frag_hdr);
|
||||||
fh->nexthdr = nexthdr;
|
fh->nexthdr = nexthdr;
|
||||||
fh->reserved = 0;
|
fh->reserved = 0;
|
||||||
@ -753,7 +756,7 @@ static int ip6_fragment(struct sk_buff *skb, int (*output)(struct sk_buff *))
|
|||||||
/*
|
/*
|
||||||
* Copy the packet header into the new buffer.
|
* Copy the packet header into the new buffer.
|
||||||
*/
|
*/
|
||||||
memcpy(frag->nh.raw, skb->data, hlen);
|
memcpy(skb_network_header(frag), skb->data, hlen);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Build fragment header.
|
* Build fragment header.
|
||||||
@ -1329,7 +1332,7 @@ int ip6_push_pending_frames(struct sock *sk)
|
|||||||
tail_skb = &(skb_shinfo(skb)->frag_list);
|
tail_skb = &(skb_shinfo(skb)->frag_list);
|
||||||
|
|
||||||
/* move skb->data to ip header from ext header */
|
/* move skb->data to ip header from ext header */
|
||||||
if (skb->data < skb->nh.raw)
|
if (skb->data < skb_network_header(skb))
|
||||||
__skb_pull(skb, skb_network_offset(skb));
|
__skb_pull(skb, skb_network_offset(skb));
|
||||||
while ((tmp_skb = __skb_dequeue(&sk->sk_write_queue)) != NULL) {
|
while ((tmp_skb = __skb_dequeue(&sk->sk_write_queue)) != NULL) {
|
||||||
__skb_pull(tmp_skb, skb->h.raw - skb->nh.raw);
|
__skb_pull(tmp_skb, skb->h.raw - skb->nh.raw);
|
||||||
|
@ -995,9 +995,10 @@ ip6ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
!ip6_tnl_xmit_ctl(t) || ip6_tnl_addr_conflict(t, ipv6h))
|
!ip6_tnl_xmit_ctl(t) || ip6_tnl_addr_conflict(t, ipv6h))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if ((offset = parse_tlv_tnl_enc_lim(skb, skb->nh.raw)) > 0) {
|
offset = parse_tlv_tnl_enc_lim(skb, skb_network_header(skb));
|
||||||
|
if (offset > 0) {
|
||||||
struct ipv6_tlv_tnl_enc_lim *tel;
|
struct ipv6_tlv_tnl_enc_lim *tel;
|
||||||
tel = (struct ipv6_tlv_tnl_enc_lim *) &skb->nh.raw[offset];
|
tel = (struct ipv6_tlv_tnl_enc_lim *)&skb_network_header(skb)[offset];
|
||||||
if (tel->encap_limit == 0) {
|
if (tel->encap_limit == 0) {
|
||||||
icmpv6_send(skb, ICMPV6_PARAMPROB,
|
icmpv6_send(skb, ICMPV6_PARAMPROB,
|
||||||
ICMPV6_HDR_FIELD, offset + 2, skb->dev);
|
ICMPV6_HDR_FIELD, offset + 2, skb->dev);
|
||||||
|
@ -166,10 +166,10 @@ static int ipcomp6_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
top_iph->payload_len = htons(skb->len - sizeof(struct ipv6hdr));
|
top_iph->payload_len = htons(skb->len - sizeof(struct ipv6hdr));
|
||||||
|
|
||||||
ipch = (struct ipv6_comp_hdr *)start;
|
ipch = (struct ipv6_comp_hdr *)start;
|
||||||
ipch->nexthdr = *skb->nh.raw;
|
ipch->nexthdr = *skb_network_header(skb);
|
||||||
ipch->flags = 0;
|
ipch->flags = 0;
|
||||||
ipch->cpi = htons((u16 )ntohl(x->id.spi));
|
ipch->cpi = htons((u16 )ntohl(x->id.spi));
|
||||||
*skb->nh.raw = IPPROTO_COMP;
|
*skb_network_header(skb) = IPPROTO_COMP;
|
||||||
|
|
||||||
out_ok:
|
out_ok:
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -99,14 +99,16 @@ int mip6_mh_filter(struct sock *sk, struct sk_buff *skb)
|
|||||||
if (mh->ip6mh_hdrlen < mip6_mh_len(mh->ip6mh_type)) {
|
if (mh->ip6mh_hdrlen < mip6_mh_len(mh->ip6mh_type)) {
|
||||||
LIMIT_NETDEBUG(KERN_DEBUG "mip6: MH message too short: %d vs >=%d\n",
|
LIMIT_NETDEBUG(KERN_DEBUG "mip6: MH message too short: %d vs >=%d\n",
|
||||||
mh->ip6mh_hdrlen, mip6_mh_len(mh->ip6mh_type));
|
mh->ip6mh_hdrlen, mip6_mh_len(mh->ip6mh_type));
|
||||||
mip6_param_prob(skb, 0, (&mh->ip6mh_hdrlen) - skb->nh.raw);
|
mip6_param_prob(skb, 0, ((&mh->ip6mh_hdrlen) -
|
||||||
|
skb_network_header(skb)));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mh->ip6mh_proto != IPPROTO_NONE) {
|
if (mh->ip6mh_proto != IPPROTO_NONE) {
|
||||||
LIMIT_NETDEBUG(KERN_DEBUG "mip6: MH invalid payload proto = %d\n",
|
LIMIT_NETDEBUG(KERN_DEBUG "mip6: MH invalid payload proto = %d\n",
|
||||||
mh->ip6mh_proto);
|
mh->ip6mh_proto);
|
||||||
mip6_param_prob(skb, 0, (&mh->ip6mh_proto) - skb->nh.raw);
|
mip6_param_prob(skb, 0, ((&mh->ip6mh_proto) -
|
||||||
|
skb_network_header(skb)));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -152,8 +154,8 @@ static int mip6_destopt_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
iph = (struct ipv6hdr *)skb->data;
|
iph = (struct ipv6hdr *)skb->data;
|
||||||
iph->payload_len = htons(skb->len - sizeof(*iph));
|
iph->payload_len = htons(skb->len - sizeof(*iph));
|
||||||
|
|
||||||
nexthdr = *skb->nh.raw;
|
nexthdr = *skb_network_header(skb);
|
||||||
*skb->nh.raw = IPPROTO_DSTOPTS;
|
*skb_network_header(skb) = IPPROTO_DSTOPTS;
|
||||||
|
|
||||||
dstopt = (struct ipv6_destopt_hdr *)skb->h.raw;
|
dstopt = (struct ipv6_destopt_hdr *)skb->h.raw;
|
||||||
dstopt->nexthdr = nexthdr;
|
dstopt->nexthdr = nexthdr;
|
||||||
@ -215,7 +217,8 @@ static int mip6_destopt_reject(struct xfrm_state *x, struct sk_buff *skb, struct
|
|||||||
if (likely(opt->dsthao)) {
|
if (likely(opt->dsthao)) {
|
||||||
offset = ipv6_find_tlv(skb, opt->dsthao, IPV6_TLV_HAO);
|
offset = ipv6_find_tlv(skb, opt->dsthao, IPV6_TLV_HAO);
|
||||||
if (likely(offset >= 0))
|
if (likely(offset >= 0))
|
||||||
hao = (struct ipv6_destopt_hao *)(skb->nh.raw + offset);
|
hao = (struct ipv6_destopt_hao *)
|
||||||
|
(skb_network_header(skb) + offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
skb_get_timestamp(skb, &stamp);
|
skb_get_timestamp(skb, &stamp);
|
||||||
@ -254,7 +257,8 @@ static int mip6_destopt_offset(struct xfrm_state *x, struct sk_buff *skb,
|
|||||||
{
|
{
|
||||||
u16 offset = sizeof(struct ipv6hdr);
|
u16 offset = sizeof(struct ipv6hdr);
|
||||||
struct ipv6_opt_hdr *exthdr = (struct ipv6_opt_hdr*)(skb->nh.ipv6h + 1);
|
struct ipv6_opt_hdr *exthdr = (struct ipv6_opt_hdr*)(skb->nh.ipv6h + 1);
|
||||||
unsigned int packet_len = skb->tail - skb->nh.raw;
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
unsigned int packet_len = skb->tail - nh;
|
||||||
int found_rhdr = 0;
|
int found_rhdr = 0;
|
||||||
|
|
||||||
*nexthdr = &skb->nh.ipv6h->nexthdr;
|
*nexthdr = &skb->nh.ipv6h->nexthdr;
|
||||||
@ -288,7 +292,7 @@ static int mip6_destopt_offset(struct xfrm_state *x, struct sk_buff *skb,
|
|||||||
|
|
||||||
offset += ipv6_optlen(exthdr);
|
offset += ipv6_optlen(exthdr);
|
||||||
*nexthdr = &exthdr->nexthdr;
|
*nexthdr = &exthdr->nexthdr;
|
||||||
exthdr = (struct ipv6_opt_hdr*)(skb->nh.raw + offset);
|
exthdr = (struct ipv6_opt_hdr *)(nh + offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
return offset;
|
return offset;
|
||||||
@ -361,8 +365,8 @@ static int mip6_rthdr_output(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
iph = (struct ipv6hdr *)skb->data;
|
iph = (struct ipv6hdr *)skb->data;
|
||||||
iph->payload_len = htons(skb->len - sizeof(*iph));
|
iph->payload_len = htons(skb->len - sizeof(*iph));
|
||||||
|
|
||||||
nexthdr = *skb->nh.raw;
|
nexthdr = *skb_network_header(skb);
|
||||||
*skb->nh.raw = IPPROTO_ROUTING;
|
*skb_network_header(skb) = IPPROTO_ROUTING;
|
||||||
|
|
||||||
rt2 = (struct rt2_hdr *)skb->h.raw;
|
rt2 = (struct rt2_hdr *)skb->h.raw;
|
||||||
rt2->rt_hdr.nexthdr = nexthdr;
|
rt2->rt_hdr.nexthdr = nexthdr;
|
||||||
@ -384,7 +388,8 @@ static int mip6_rthdr_offset(struct xfrm_state *x, struct sk_buff *skb,
|
|||||||
{
|
{
|
||||||
u16 offset = sizeof(struct ipv6hdr);
|
u16 offset = sizeof(struct ipv6hdr);
|
||||||
struct ipv6_opt_hdr *exthdr = (struct ipv6_opt_hdr*)(skb->nh.ipv6h + 1);
|
struct ipv6_opt_hdr *exthdr = (struct ipv6_opt_hdr*)(skb->nh.ipv6h + 1);
|
||||||
unsigned int packet_len = skb->tail - skb->nh.raw;
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
unsigned int packet_len = skb->tail - nh;
|
||||||
int found_rhdr = 0;
|
int found_rhdr = 0;
|
||||||
|
|
||||||
*nexthdr = &skb->nh.ipv6h->nexthdr;
|
*nexthdr = &skb->nh.ipv6h->nexthdr;
|
||||||
@ -397,7 +402,7 @@ static int mip6_rthdr_offset(struct xfrm_state *x, struct sk_buff *skb,
|
|||||||
case NEXTHDR_ROUTING:
|
case NEXTHDR_ROUTING:
|
||||||
if (offset + 3 <= packet_len) {
|
if (offset + 3 <= packet_len) {
|
||||||
struct ipv6_rt_hdr *rt;
|
struct ipv6_rt_hdr *rt;
|
||||||
rt = (struct ipv6_rt_hdr *)(skb->nh.raw + offset);
|
rt = (struct ipv6_rt_hdr *)(nh + offset);
|
||||||
if (rt->type != 0)
|
if (rt->type != 0)
|
||||||
return offset;
|
return offset;
|
||||||
}
|
}
|
||||||
@ -417,7 +422,7 @@ static int mip6_rthdr_offset(struct xfrm_state *x, struct sk_buff *skb,
|
|||||||
|
|
||||||
offset += ipv6_optlen(exthdr);
|
offset += ipv6_optlen(exthdr);
|
||||||
*nexthdr = &exthdr->nexthdr;
|
*nexthdr = &exthdr->nexthdr;
|
||||||
exthdr = (struct ipv6_opt_hdr*)(skb->nh.raw + offset);
|
exthdr = (struct ipv6_opt_hdr *)(nh + offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
return offset;
|
return offset;
|
||||||
|
@ -408,11 +408,12 @@ static int nf_ct_frag6_queue(struct nf_ct_frag6_queue *fq, struct sk_buff *skb,
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (skb->ip_summed == CHECKSUM_COMPLETE)
|
if (skb->ip_summed == CHECKSUM_COMPLETE) {
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
skb->csum = csum_sub(skb->csum,
|
skb->csum = csum_sub(skb->csum,
|
||||||
csum_partial(skb->nh.raw,
|
csum_partial(nh, (u8 *)(fhdr + 1) - nh,
|
||||||
(u8*)(fhdr + 1) - skb->nh.raw,
|
|
||||||
0));
|
0));
|
||||||
|
}
|
||||||
|
|
||||||
/* Is this the final fragment? */
|
/* Is this the final fragment? */
|
||||||
if (!(fhdr->frag_off & htons(IP6_MF))) {
|
if (!(fhdr->frag_off & htons(IP6_MF))) {
|
||||||
@ -583,7 +584,9 @@ nf_ct_frag6_reasm(struct nf_ct_frag6_queue *fq, struct net_device *dev)
|
|||||||
BUG_TRAP(NFCT_FRAG6_CB(head)->offset == 0);
|
BUG_TRAP(NFCT_FRAG6_CB(head)->offset == 0);
|
||||||
|
|
||||||
/* Unfragmented part is taken from the first segment. */
|
/* Unfragmented part is taken from the first segment. */
|
||||||
payload_len = (head->data - head->nh.raw) - sizeof(struct ipv6hdr) + fq->len - sizeof(struct frag_hdr);
|
payload_len = ((head->data - skb_network_header(head)) -
|
||||||
|
sizeof(struct ipv6hdr) + fq->len -
|
||||||
|
sizeof(struct frag_hdr));
|
||||||
if (payload_len > IPV6_MAXPLEN) {
|
if (payload_len > IPV6_MAXPLEN) {
|
||||||
DEBUGP("payload len is too large.\n");
|
DEBUGP("payload len is too large.\n");
|
||||||
goto out_oversize;
|
goto out_oversize;
|
||||||
@ -624,7 +627,7 @@ nf_ct_frag6_reasm(struct nf_ct_frag6_queue *fq, struct net_device *dev)
|
|||||||
|
|
||||||
/* We have to remove fragment header from datagram and to relocate
|
/* We have to remove fragment header from datagram and to relocate
|
||||||
* header in order to calculate ICV correctly. */
|
* header in order to calculate ICV correctly. */
|
||||||
head->nh.raw[fq->nhoffset] = head->h.raw[0];
|
skb_network_header(head)[fq->nhoffset] = head->h.raw[0];
|
||||||
memmove(head->head + sizeof(struct frag_hdr), head->head,
|
memmove(head->head + sizeof(struct frag_hdr), head->head,
|
||||||
(head->data - head->head) - sizeof(struct frag_hdr));
|
(head->data - head->head) - sizeof(struct frag_hdr));
|
||||||
head->mac.raw += sizeof(struct frag_hdr);
|
head->mac.raw += sizeof(struct frag_hdr);
|
||||||
@ -632,7 +635,7 @@ nf_ct_frag6_reasm(struct nf_ct_frag6_queue *fq, struct net_device *dev)
|
|||||||
|
|
||||||
skb_shinfo(head)->frag_list = head->next;
|
skb_shinfo(head)->frag_list = head->next;
|
||||||
head->h.raw = head->data;
|
head->h.raw = head->data;
|
||||||
skb_push(head, head->data - head->nh.raw);
|
skb_push(head, head->data - skb_network_header(head));
|
||||||
atomic_sub(head->truesize, &nf_ct_frag6_mem);
|
atomic_sub(head->truesize, &nf_ct_frag6_mem);
|
||||||
|
|
||||||
for (fp=head->next; fp; fp = fp->next) {
|
for (fp=head->next; fp; fp = fp->next) {
|
||||||
@ -653,7 +656,9 @@ nf_ct_frag6_reasm(struct nf_ct_frag6_queue *fq, struct net_device *dev)
|
|||||||
|
|
||||||
/* Yes, and fold redundant checksum back. 8) */
|
/* Yes, and fold redundant checksum back. 8) */
|
||||||
if (head->ip_summed == CHECKSUM_COMPLETE)
|
if (head->ip_summed == CHECKSUM_COMPLETE)
|
||||||
head->csum = csum_partial(head->nh.raw, head->h.raw-head->nh.raw, head->csum);
|
head->csum = csum_partial(skb_network_header(head),
|
||||||
|
head->h.raw - head->nh.raw,
|
||||||
|
head->csum);
|
||||||
|
|
||||||
fq->fragments = NULL;
|
fq->fragments = NULL;
|
||||||
|
|
||||||
|
@ -361,7 +361,7 @@ int rawv6_rcv(struct sock *sk, struct sk_buff *skb)
|
|||||||
skb->ip_summed = CHECKSUM_UNNECESSARY;
|
skb->ip_summed = CHECKSUM_UNNECESSARY;
|
||||||
|
|
||||||
if (skb->ip_summed == CHECKSUM_COMPLETE) {
|
if (skb->ip_summed == CHECKSUM_COMPLETE) {
|
||||||
skb_postpull_rcsum(skb, skb->nh.raw,
|
skb_postpull_rcsum(skb, skb_network_header(skb),
|
||||||
skb->h.raw - skb->nh.raw);
|
skb->h.raw - skb->nh.raw);
|
||||||
if (!csum_ipv6_magic(&skb->nh.ipv6h->saddr,
|
if (!csum_ipv6_magic(&skb->nh.ipv6h->saddr,
|
||||||
&skb->nh.ipv6h->daddr,
|
&skb->nh.ipv6h->daddr,
|
||||||
@ -488,7 +488,8 @@ static int rawv6_push_pending_frames(struct sock *sk, struct flowi *fl,
|
|||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
offset = rp->offset;
|
offset = rp->offset;
|
||||||
total_len = inet_sk(sk)->cork.length - (skb->nh.raw - skb->data);
|
total_len = inet_sk(sk)->cork.length - (skb_network_header(skb) -
|
||||||
|
skb->data);
|
||||||
if (offset >= total_len - 1) {
|
if (offset >= total_len - 1) {
|
||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
ip6_flush_pending_frames(sk);
|
ip6_flush_pending_frames(sk);
|
||||||
|
@ -436,13 +436,18 @@ static void ip6_frag_queue(struct frag_queue *fq, struct sk_buff *skb,
|
|||||||
if ((unsigned int)end > IPV6_MAXPLEN) {
|
if ((unsigned int)end > IPV6_MAXPLEN) {
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst),
|
||||||
IPSTATS_MIB_INHDRERRORS);
|
IPSTATS_MIB_INHDRERRORS);
|
||||||
icmpv6_param_prob(skb,ICMPV6_HDR_FIELD, (u8*)&fhdr->frag_off - skb->nh.raw);
|
icmpv6_param_prob(skb, ICMPV6_HDR_FIELD,
|
||||||
|
((u8 *)&fhdr->frag_off -
|
||||||
|
skb_network_header(skb)));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (skb->ip_summed == CHECKSUM_COMPLETE)
|
if (skb->ip_summed == CHECKSUM_COMPLETE) {
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
skb->csum = csum_sub(skb->csum,
|
skb->csum = csum_sub(skb->csum,
|
||||||
csum_partial(skb->nh.raw, (u8*)(fhdr+1)-skb->nh.raw, 0));
|
csum_partial(nh, (u8 *)(fhdr + 1) - nh,
|
||||||
|
0));
|
||||||
|
}
|
||||||
|
|
||||||
/* Is this the final fragment? */
|
/* Is this the final fragment? */
|
||||||
if (!(fhdr->frag_off & htons(IP6_MF))) {
|
if (!(fhdr->frag_off & htons(IP6_MF))) {
|
||||||
@ -605,7 +610,9 @@ static int ip6_frag_reasm(struct frag_queue *fq, struct sk_buff **skb_in,
|
|||||||
BUG_TRAP(FRAG6_CB(head)->offset == 0);
|
BUG_TRAP(FRAG6_CB(head)->offset == 0);
|
||||||
|
|
||||||
/* Unfragmented part is taken from the first segment. */
|
/* Unfragmented part is taken from the first segment. */
|
||||||
payload_len = (head->data - head->nh.raw) - sizeof(struct ipv6hdr) + fq->len - sizeof(struct frag_hdr);
|
payload_len = ((head->data - skb_network_header(head)) -
|
||||||
|
sizeof(struct ipv6hdr) + fq->len -
|
||||||
|
sizeof(struct frag_hdr));
|
||||||
if (payload_len > IPV6_MAXPLEN)
|
if (payload_len > IPV6_MAXPLEN)
|
||||||
goto out_oversize;
|
goto out_oversize;
|
||||||
|
|
||||||
@ -639,7 +646,7 @@ static int ip6_frag_reasm(struct frag_queue *fq, struct sk_buff **skb_in,
|
|||||||
/* We have to remove fragment header from datagram and to relocate
|
/* We have to remove fragment header from datagram and to relocate
|
||||||
* header in order to calculate ICV correctly. */
|
* header in order to calculate ICV correctly. */
|
||||||
nhoff = fq->nhoffset;
|
nhoff = fq->nhoffset;
|
||||||
head->nh.raw[nhoff] = head->h.raw[0];
|
skb_network_header(head)[nhoff] = head->h.raw[0];
|
||||||
memmove(head->head + sizeof(struct frag_hdr), head->head,
|
memmove(head->head + sizeof(struct frag_hdr), head->head,
|
||||||
(head->data - head->head) - sizeof(struct frag_hdr));
|
(head->data - head->head) - sizeof(struct frag_hdr));
|
||||||
head->mac.raw += sizeof(struct frag_hdr);
|
head->mac.raw += sizeof(struct frag_hdr);
|
||||||
@ -647,7 +654,7 @@ static int ip6_frag_reasm(struct frag_queue *fq, struct sk_buff **skb_in,
|
|||||||
|
|
||||||
skb_shinfo(head)->frag_list = head->next;
|
skb_shinfo(head)->frag_list = head->next;
|
||||||
head->h.raw = head->data;
|
head->h.raw = head->data;
|
||||||
skb_push(head, head->data - head->nh.raw);
|
skb_push(head, head->data - skb_network_header(head));
|
||||||
atomic_sub(head->truesize, &ip6_frag_mem);
|
atomic_sub(head->truesize, &ip6_frag_mem);
|
||||||
|
|
||||||
for (fp=head->next; fp; fp = fp->next) {
|
for (fp=head->next; fp; fp = fp->next) {
|
||||||
@ -671,7 +678,9 @@ static int ip6_frag_reasm(struct frag_queue *fq, struct sk_buff **skb_in,
|
|||||||
|
|
||||||
/* Yes, and fold redundant checksum back. 8) */
|
/* Yes, and fold redundant checksum back. 8) */
|
||||||
if (head->ip_summed == CHECKSUM_COMPLETE)
|
if (head->ip_summed == CHECKSUM_COMPLETE)
|
||||||
head->csum = csum_partial(head->nh.raw, head->h.raw-head->nh.raw, head->csum);
|
head->csum = csum_partial(skb_network_header(head),
|
||||||
|
head->h.raw - head->nh.raw,
|
||||||
|
head->csum);
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
IP6_INC_STATS_BH(__in6_dev_get(dev), IPSTATS_MIB_REASMOKS);
|
IP6_INC_STATS_BH(__in6_dev_get(dev), IPSTATS_MIB_REASMOKS);
|
||||||
@ -725,7 +734,7 @@ static int ipv6_frag_rcv(struct sk_buff **skbp)
|
|||||||
skb->h.raw += sizeof(struct frag_hdr);
|
skb->h.raw += sizeof(struct frag_hdr);
|
||||||
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst), IPSTATS_MIB_REASMOKS);
|
IP6_INC_STATS_BH(ip6_dst_idev(skb->dst), IPSTATS_MIB_REASMOKS);
|
||||||
|
|
||||||
IP6CB(skb)->nhoff = (u8*)fhdr - skb->nh.raw;
|
IP6CB(skb)->nhoff = (u8 *)fhdr - skb_network_header(skb);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -486,7 +486,9 @@ static int tcp_v6_send_synack(struct sock *sk, struct request_sock *req,
|
|||||||
struct sk_buff *pktopts = treq->pktopts;
|
struct sk_buff *pktopts = treq->pktopts;
|
||||||
struct inet6_skb_parm *rxopt = IP6CB(pktopts);
|
struct inet6_skb_parm *rxopt = IP6CB(pktopts);
|
||||||
if (rxopt->srcrt)
|
if (rxopt->srcrt)
|
||||||
opt = ipv6_invert_rthdr(sk, (struct ipv6_rt_hdr*)(pktopts->nh.raw + rxopt->srcrt));
|
opt = ipv6_invert_rthdr(sk,
|
||||||
|
(struct ipv6_rt_hdr *)(skb_network_header(pktopts) +
|
||||||
|
rxopt->srcrt));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opt && opt->srcrt) {
|
if (opt && opt->srcrt) {
|
||||||
@ -1389,7 +1391,9 @@ static struct sock * tcp_v6_syn_recv_sock(struct sock *sk, struct sk_buff *skb,
|
|||||||
opt == NULL && treq->pktopts) {
|
opt == NULL && treq->pktopts) {
|
||||||
struct inet6_skb_parm *rxopt = IP6CB(treq->pktopts);
|
struct inet6_skb_parm *rxopt = IP6CB(treq->pktopts);
|
||||||
if (rxopt->srcrt)
|
if (rxopt->srcrt)
|
||||||
opt = ipv6_invert_rthdr(sk, (struct ipv6_rt_hdr *)(treq->pktopts->nh.raw + rxopt->srcrt));
|
opt = ipv6_invert_rthdr(sk,
|
||||||
|
(struct ipv6_rt_hdr *)(skb_network_header(treq->pktopts) +
|
||||||
|
rxopt->srcrt));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dst == NULL) {
|
if (dst == NULL) {
|
||||||
|
@ -28,7 +28,7 @@ int xfrm6_rcv_spi(struct sk_buff *skb, __be32 spi)
|
|||||||
unsigned int nhoff;
|
unsigned int nhoff;
|
||||||
|
|
||||||
nhoff = IP6CB(skb)->nhoff;
|
nhoff = IP6CB(skb)->nhoff;
|
||||||
nexthdr = skb->nh.raw[nhoff];
|
nexthdr = skb_network_header(skb)[nhoff];
|
||||||
|
|
||||||
seq = 0;
|
seq = 0;
|
||||||
if (!spi && (err = xfrm_parse_spi(skb, nexthdr, &spi, &seq)) != 0)
|
if (!spi && (err = xfrm_parse_spi(skb, nexthdr, &spi, &seq)) != 0)
|
||||||
@ -58,7 +58,7 @@ int xfrm6_rcv_spi(struct sk_buff *skb, __be32 spi)
|
|||||||
if (nexthdr <= 0)
|
if (nexthdr <= 0)
|
||||||
goto drop_unlock;
|
goto drop_unlock;
|
||||||
|
|
||||||
skb->nh.raw[nhoff] = nexthdr;
|
skb_network_header(skb)[nhoff] = nexthdr;
|
||||||
|
|
||||||
if (x->props.replay_window)
|
if (x->props.replay_window)
|
||||||
xfrm_replay_advance(x, seq);
|
xfrm_replay_advance(x, seq);
|
||||||
@ -113,7 +113,7 @@ int xfrm6_rcv_spi(struct sk_buff *skb, __be32 spi)
|
|||||||
} else {
|
} else {
|
||||||
#ifdef CONFIG_NETFILTER
|
#ifdef CONFIG_NETFILTER
|
||||||
skb->nh.ipv6h->payload_len = htons(skb->len);
|
skb->nh.ipv6h->payload_len = htons(skb->len);
|
||||||
__skb_push(skb, skb->data - skb->nh.raw);
|
__skb_push(skb, skb->data - skb_network_header(skb));
|
||||||
|
|
||||||
NF_HOOK(PF_INET6, NF_IP6_PRE_ROUTING, skb, skb->dev, NULL,
|
NF_HOOK(PF_INET6, NF_IP6_PRE_ROUTING, skb, skb->dev, NULL,
|
||||||
ip6_rcv_finish);
|
ip6_rcv_finish);
|
||||||
|
@ -67,7 +67,7 @@ static int xfrm6_beet_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
skb_push(skb, size);
|
skb_push(skb, size);
|
||||||
memmove(skb->data, skb->nh.raw, size);
|
memmove(skb->data, skb_network_header(skb), size);
|
||||||
skb_reset_network_header(skb);
|
skb_reset_network_header(skb);
|
||||||
|
|
||||||
old_mac = skb_mac_header(skb);
|
old_mac = skb_mac_header(skb);
|
||||||
|
@ -53,8 +53,10 @@ static int xfrm6_transport_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
int ihl = skb->data - skb->h.raw;
|
int ihl = skb->data - skb->h.raw;
|
||||||
|
|
||||||
if (skb->h.raw != skb->nh.raw)
|
if (skb->h.raw != skb->nh.raw) {
|
||||||
skb->nh.raw = memmove(skb->h.raw, skb->nh.raw, ihl);
|
memmove(skb->h.raw, skb_network_header(skb), ihl);
|
||||||
|
skb->nh.raw = skb->h.raw;
|
||||||
|
}
|
||||||
skb->nh.ipv6h->payload_len = htons(skb->len + ihl -
|
skb->nh.ipv6h->payload_len = htons(skb->len + ihl -
|
||||||
sizeof(struct ipv6hdr));
|
sizeof(struct ipv6hdr));
|
||||||
skb->h.raw = skb->data;
|
skb->h.raw = skb->data;
|
||||||
|
@ -87,9 +87,10 @@ static int xfrm6_tunnel_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
int err = -EINVAL;
|
int err = -EINVAL;
|
||||||
const unsigned char *old_mac;
|
const unsigned char *old_mac;
|
||||||
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
|
||||||
if (skb->nh.raw[IP6CB(skb)->nhoff] != IPPROTO_IPV6
|
if (nh[IP6CB(skb)->nhoff] != IPPROTO_IPV6 &&
|
||||||
&& skb->nh.raw[IP6CB(skb)->nhoff] != IPPROTO_IPIP)
|
nh[IP6CB(skb)->nhoff] != IPPROTO_IPIP)
|
||||||
goto out;
|
goto out;
|
||||||
if (!pskb_may_pull(skb, sizeof(struct ipv6hdr)))
|
if (!pskb_may_pull(skb, sizeof(struct ipv6hdr)))
|
||||||
goto out;
|
goto out;
|
||||||
@ -98,7 +99,8 @@ static int xfrm6_tunnel_input(struct xfrm_state *x, struct sk_buff *skb)
|
|||||||
(err = pskb_expand_head(skb, 0, 0, GFP_ATOMIC)))
|
(err = pskb_expand_head(skb, 0, 0, GFP_ATOMIC)))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
if (skb->nh.raw[IP6CB(skb)->nhoff] == IPPROTO_IPV6) {
|
nh = skb_network_header(skb);
|
||||||
|
if (nh[IP6CB(skb)->nhoff] == IPPROTO_IPV6) {
|
||||||
if (x->props.flags & XFRM_STATE_DECAP_DSCP)
|
if (x->props.flags & XFRM_STATE_DECAP_DSCP)
|
||||||
ipv6_copy_dscp(skb->nh.ipv6h, skb->h.ipv6h);
|
ipv6_copy_dscp(skb->nh.ipv6h, skb->h.ipv6h);
|
||||||
if (!(x->props.flags & XFRM_STATE_NOECN))
|
if (!(x->props.flags & XFRM_STATE_NOECN))
|
||||||
|
@ -273,14 +273,16 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl)
|
|||||||
u16 offset = skb->h.raw - skb->nh.raw;
|
u16 offset = skb->h.raw - skb->nh.raw;
|
||||||
struct ipv6hdr *hdr = skb->nh.ipv6h;
|
struct ipv6hdr *hdr = skb->nh.ipv6h;
|
||||||
struct ipv6_opt_hdr *exthdr;
|
struct ipv6_opt_hdr *exthdr;
|
||||||
u8 nexthdr = skb->nh.raw[IP6CB(skb)->nhoff];
|
const unsigned char *nh = skb_network_header(skb);
|
||||||
|
u8 nexthdr = nh[IP6CB(skb)->nhoff];
|
||||||
|
|
||||||
memset(fl, 0, sizeof(struct flowi));
|
memset(fl, 0, sizeof(struct flowi));
|
||||||
ipv6_addr_copy(&fl->fl6_dst, &hdr->daddr);
|
ipv6_addr_copy(&fl->fl6_dst, &hdr->daddr);
|
||||||
ipv6_addr_copy(&fl->fl6_src, &hdr->saddr);
|
ipv6_addr_copy(&fl->fl6_src, &hdr->saddr);
|
||||||
|
|
||||||
while (pskb_may_pull(skb, skb->nh.raw + offset + 1 - skb->data)) {
|
while (pskb_may_pull(skb, nh + offset + 1 - skb->data)) {
|
||||||
exthdr = (struct ipv6_opt_hdr*)(skb->nh.raw + offset);
|
nh = skb_network_header(skb);
|
||||||
|
exthdr = (struct ipv6_opt_hdr *)(nh + offset);
|
||||||
|
|
||||||
switch (nexthdr) {
|
switch (nexthdr) {
|
||||||
case NEXTHDR_ROUTING:
|
case NEXTHDR_ROUTING:
|
||||||
@ -288,7 +290,7 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl)
|
|||||||
case NEXTHDR_DEST:
|
case NEXTHDR_DEST:
|
||||||
offset += ipv6_optlen(exthdr);
|
offset += ipv6_optlen(exthdr);
|
||||||
nexthdr = exthdr->nexthdr;
|
nexthdr = exthdr->nexthdr;
|
||||||
exthdr = (struct ipv6_opt_hdr*)(skb->nh.raw + offset);
|
exthdr = (struct ipv6_opt_hdr *)(nh + offset);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case IPPROTO_UDP:
|
case IPPROTO_UDP:
|
||||||
@ -296,7 +298,7 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl)
|
|||||||
case IPPROTO_TCP:
|
case IPPROTO_TCP:
|
||||||
case IPPROTO_SCTP:
|
case IPPROTO_SCTP:
|
||||||
case IPPROTO_DCCP:
|
case IPPROTO_DCCP:
|
||||||
if (pskb_may_pull(skb, skb->nh.raw + offset + 4 - skb->data)) {
|
if (pskb_may_pull(skb, nh + offset + 4 - skb->data)) {
|
||||||
__be16 *ports = (__be16 *)exthdr;
|
__be16 *ports = (__be16 *)exthdr;
|
||||||
|
|
||||||
fl->fl_ip_sport = ports[0];
|
fl->fl_ip_sport = ports[0];
|
||||||
@ -306,7 +308,7 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
case IPPROTO_ICMPV6:
|
case IPPROTO_ICMPV6:
|
||||||
if (pskb_may_pull(skb, skb->nh.raw + offset + 2 - skb->data)) {
|
if (pskb_may_pull(skb, nh + offset + 2 - skb->data)) {
|
||||||
u8 *icmp = (u8 *)exthdr;
|
u8 *icmp = (u8 *)exthdr;
|
||||||
|
|
||||||
fl->fl_icmp_type = icmp[0];
|
fl->fl_icmp_type = icmp[0];
|
||||||
@ -317,7 +319,7 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl)
|
|||||||
|
|
||||||
#ifdef CONFIG_IPV6_MIP6
|
#ifdef CONFIG_IPV6_MIP6
|
||||||
case IPPROTO_MH:
|
case IPPROTO_MH:
|
||||||
if (pskb_may_pull(skb, skb->nh.raw + offset + 3 - skb->data)) {
|
if (pskb_may_pull(skb, nh + offset + 3 - skb->data)) {
|
||||||
struct ip6_mh *mh;
|
struct ip6_mh *mh;
|
||||||
mh = (struct ip6_mh *)exthdr;
|
mh = (struct ip6_mh *)exthdr;
|
||||||
|
|
||||||
|
@ -54,7 +54,7 @@ tcpmss_mangle_packet(struct sk_buff **pskb,
|
|||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
tcplen = (*pskb)->len - tcphoff;
|
tcplen = (*pskb)->len - tcphoff;
|
||||||
tcph = (struct tcphdr *)((*pskb)->nh.raw + tcphoff);
|
tcph = (struct tcphdr *)(skb_network_header(*pskb) + tcphoff);
|
||||||
|
|
||||||
/* Since it passed flags test in tcp match, we know it is is
|
/* Since it passed flags test in tcp match, we know it is is
|
||||||
not a fragment, and has data >= tcp header length. SYN
|
not a fragment, and has data >= tcp header length. SYN
|
||||||
@ -113,7 +113,7 @@ tcpmss_mangle_packet(struct sk_buff **pskb,
|
|||||||
return -1;
|
return -1;
|
||||||
kfree_skb(*pskb);
|
kfree_skb(*pskb);
|
||||||
*pskb = newskb;
|
*pskb = newskb;
|
||||||
tcph = (struct tcphdr *)((*pskb)->nh.raw + tcphoff);
|
tcph = (struct tcphdr *)(skb_network_header(*pskb) + tcphoff);
|
||||||
}
|
}
|
||||||
|
|
||||||
skb_put((*pskb), TCPOLEN_MSS);
|
skb_put((*pskb), TCPOLEN_MSS);
|
||||||
|
@ -136,7 +136,7 @@ static int tcf_pedit(struct sk_buff *skb, struct tc_action *a,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pptr = skb->nh.raw;
|
pptr = skb_network_header(skb);
|
||||||
|
|
||||||
spin_lock(&p->tcf_lock);
|
spin_lock(&p->tcf_lock);
|
||||||
|
|
||||||
|
@ -119,7 +119,7 @@ static int u32_classify(struct sk_buff *skb, struct tcf_proto *tp, struct tcf_re
|
|||||||
} stack[TC_U32_MAXDEPTH];
|
} stack[TC_U32_MAXDEPTH];
|
||||||
|
|
||||||
struct tc_u_hnode *ht = (struct tc_u_hnode*)tp->root;
|
struct tc_u_hnode *ht = (struct tc_u_hnode*)tp->root;
|
||||||
u8 *ptr = skb->nh.raw;
|
u8 *ptr = skb_network_header(skb);
|
||||||
struct tc_u_knode *n;
|
struct tc_u_knode *n;
|
||||||
int sdepth = 0;
|
int sdepth = 0;
|
||||||
int off2 = 0;
|
int off2 = 0;
|
||||||
|
@ -22,7 +22,7 @@ static int em_u32_match(struct sk_buff *skb, struct tcf_ematch *em,
|
|||||||
struct tcf_pkt_info *info)
|
struct tcf_pkt_info *info)
|
||||||
{
|
{
|
||||||
struct tc_u32_key *key = (struct tc_u32_key *) em->data;
|
struct tc_u32_key *key = (struct tc_u32_key *) em->data;
|
||||||
unsigned char *ptr = skb->nh.raw;
|
const unsigned char *ptr = skb_network_header(skb);
|
||||||
|
|
||||||
if (info) {
|
if (info) {
|
||||||
if (info->ptr)
|
if (info->ptr)
|
||||||
|
Loading…
Reference in New Issue
Block a user