Skip to content

Commit 3b8377d

Browse files
committed
Merge branch 'variable-length-ll-headers'
Willem de Bruijn says: ==================== net: validate variable length ll headers Allow device-specific validation of link layer headers. Existing checks drop all packets shorter than hard_header_len. For variable length protocols, such packets can be valid. patch 1 adds header_ops.validate and dev_validate_header patch 2 implements the protocol specific callback for AX25 patch 3 replaces ll_header_truncated with dev_validate_header ==================== Signed-off-by: David S. Miller <[email protected]>
2 parents 9531ab6 + 9ed988c commit 3b8377d

File tree

3 files changed

+53
-27
lines changed

3 files changed

+53
-27
lines changed

include/linux/netdevice.h

Lines changed: 20 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -268,6 +268,7 @@ struct header_ops {
268268
void (*cache_update)(struct hh_cache *hh,
269269
const struct net_device *dev,
270270
const unsigned char *haddr);
271+
bool (*validate)(const char *ll_header, unsigned int len);
271272
};
272273

273274
/* These flag bits are private to the generic network queueing
@@ -1459,8 +1460,7 @@ enum netdev_priv_flags {
14591460
* @dma: DMA channel
14601461
* @mtu: Interface MTU value
14611462
* @type: Interface hardware type
1462-
* @hard_header_len: Hardware header length, which means that this is the
1463-
* minimum size of a packet.
1463+
* @hard_header_len: Maximum hardware header length.
14641464
*
14651465
* @needed_headroom: Extra headroom the hardware may need, but not in all
14661466
* cases can this be guaranteed
@@ -2687,6 +2687,24 @@ static inline int dev_parse_header(const struct sk_buff *skb,
26872687
return dev->header_ops->parse(skb, haddr);
26882688
}
26892689

2690+
/* ll_header must have at least hard_header_len allocated */
2691+
static inline bool dev_validate_header(const struct net_device *dev,
2692+
char *ll_header, int len)
2693+
{
2694+
if (likely(len >= dev->hard_header_len))
2695+
return true;
2696+
2697+
if (capable(CAP_SYS_RAWIO)) {
2698+
memset(ll_header + len, 0, dev->hard_header_len - len);
2699+
return true;
2700+
}
2701+
2702+
if (dev->header_ops && dev->header_ops->validate)
2703+
return dev->header_ops->validate(ll_header, len);
2704+
2705+
return false;
2706+
}
2707+
26902708
typedef int gifconf_func_t(struct net_device * dev, char __user * bufptr, int len);
26912709
int register_gifconf(unsigned int family, gifconf_func_t *gifconf);
26922710
static inline int unregister_gifconf(unsigned int family)

net/ax25/ax25_ip.c

Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -228,8 +228,23 @@ netdev_tx_t ax25_ip_xmit(struct sk_buff *skb)
228228
}
229229
#endif
230230

231+
static bool ax25_validate_header(const char *header, unsigned int len)
232+
{
233+
ax25_digi digi;
234+
235+
if (!len)
236+
return false;
237+
238+
if (header[0])
239+
return true;
240+
241+
return ax25_addr_parse(header + 1, len - 1, NULL, NULL, &digi, NULL,
242+
NULL);
243+
}
244+
231245
const struct header_ops ax25_header_ops = {
232246
.create = ax25_hard_header,
247+
.validate = ax25_validate_header,
233248
};
234249

235250
EXPORT_SYMBOL(ax25_header_ops);

net/packet/af_packet.c

Lines changed: 18 additions & 25 deletions
Original file line numberDiff line numberDiff line change
@@ -1915,6 +1915,10 @@ static int packet_sendmsg_spkt(struct socket *sock, struct msghdr *msg,
19151915
goto retry;
19161916
}
19171917

1918+
if (!dev_validate_header(dev, skb->data, len)) {
1919+
err = -EINVAL;
1920+
goto out_unlock;
1921+
}
19181922
if (len > (dev->mtu + dev->hard_header_len + extra_len) &&
19191923
!packet_extra_vlan_len_allowed(dev, skb)) {
19201924
err = -EMSGSIZE;
@@ -2393,18 +2397,6 @@ static void tpacket_destruct_skb(struct sk_buff *skb)
23932397
sock_wfree(skb);
23942398
}
23952399

2396-
static bool ll_header_truncated(const struct net_device *dev, int len)
2397-
{
2398-
/* net device doesn't like empty head */
2399-
if (unlikely(len < dev->hard_header_len)) {
2400-
net_warn_ratelimited("%s: packet size is too short (%d < %d)\n",
2401-
current->comm, len, dev->hard_header_len);
2402-
return true;
2403-
}
2404-
2405-
return false;
2406-
}
2407-
24082400
static void tpacket_set_protocol(const struct net_device *dev,
24092401
struct sk_buff *skb)
24102402
{
@@ -2522,16 +2514,20 @@ static int tpacket_fill_skb(struct packet_sock *po, struct sk_buff *skb,
25222514
if (unlikely(err < 0))
25232515
return -EINVAL;
25242516
} else if (copylen) {
2517+
int hdrlen = min_t(int, copylen, tp_len);
2518+
25252519
skb_push(skb, dev->hard_header_len);
25262520
skb_put(skb, copylen - dev->hard_header_len);
2527-
err = skb_store_bits(skb, 0, data, copylen);
2521+
err = skb_store_bits(skb, 0, data, hdrlen);
25282522
if (unlikely(err))
25292523
return err;
2524+
if (!dev_validate_header(dev, skb->data, hdrlen))
2525+
return -EINVAL;
25302526
if (!skb->protocol)
25312527
tpacket_set_protocol(dev, skb);
25322528

2533-
data += copylen;
2534-
to_write -= copylen;
2529+
data += hdrlen;
2530+
to_write -= hdrlen;
25352531
}
25362532

25372533
offset = offset_in_page(data);
@@ -2703,13 +2699,7 @@ static int tpacket_snd(struct packet_sock *po, struct msghdr *msg)
27032699
copylen = __virtio16_to_cpu(vio_le(),
27042700
vnet_hdr->hdr_len);
27052701
}
2706-
if (dev->hard_header_len) {
2707-
if (ll_header_truncated(dev, tp_len)) {
2708-
tp_len = -EINVAL;
2709-
goto tpacket_error;
2710-
}
2711-
copylen = max_t(int, copylen, dev->hard_header_len);
2712-
}
2702+
copylen = max_t(int, copylen, dev->hard_header_len);
27132703
skb = sock_alloc_send_skb(&po->sk,
27142704
hlen + tlen + sizeof(struct sockaddr_ll) +
27152705
(copylen - dev->hard_header_len),
@@ -2905,16 +2895,19 @@ static int packet_snd(struct socket *sock, struct msghdr *msg, size_t len)
29052895
offset = dev_hard_header(skb, dev, ntohs(proto), addr, NULL, len);
29062896
if (unlikely(offset < 0))
29072897
goto out_free;
2908-
} else {
2909-
if (ll_header_truncated(dev, len))
2910-
goto out_free;
29112898
}
29122899

29132900
/* Returns -EFAULT on error */
29142901
err = skb_copy_datagram_from_iter(skb, offset, &msg->msg_iter, len);
29152902
if (err)
29162903
goto out_free;
29172904

2905+
if (sock->type == SOCK_RAW &&
2906+
!dev_validate_header(dev, skb->data, len)) {
2907+
err = -EINVAL;
2908+
goto out_free;
2909+
}
2910+
29182911
sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
29192912

29202913
if (!vnet_hdr.gso_type && (len > dev->mtu + reserve + extra_len) &&

0 commit comments

Comments
 (0)