Thread (6 messages) 6 messages, 3 authors, 2013-02-27

Re: [PATCH] net: af_packet: Validate parameter size for PACKET_HDRLEN control message

From: Daniel Borkmann <hidden>
Date: 2013-02-27 21:18:11

On 02/27/2013 09:33 PM, Guenter Roeck wrote:
On Wed, Feb 27, 2013 at 03:26:30PM -0500, David Miller wrote:
quoted
From: Daniel Borkmann <redacted>
Date: Wed, 27 Feb 2013 21:22:17 +0100
quoted
On 02/27/2013 08:46 PM, Guenter Roeck wrote:
quoted
Building af_packet may fail with

In function ‘copy_from_user’,
      inlined from ‘packet_getsockopt’ at
      net/packet/af_packet.c:3215:21:
arch/x86/include/asm/uaccess_32.h:211:26: error: call to
      ‘copy_from_user_overflow’ declared with attribute error:
      copy_from_user()
      buffer size is not provably correct

if built with W=1 due to a missing parameter size validation.

Signed-off-by: Guenter Roeck <linux@roeck-us.net>
---
   net/packet/af_packet.c |    2 ++
   1 file changed, 2 insertions(+)
diff --git a/net/packet/af_packet.c b/net/packet/af_packet.c
index c7bfeff..1976b23 100644
--- a/net/packet/af_packet.c
+++ b/net/packet/af_packet.c
@@ -3210,6 +3210,8 @@ static int packet_getsockopt(struct socket
*sock, int level, int optname,
   		val = po->tp_version;
   		break;
   	case PACKET_HDRLEN:
+		if (len < sizeof(int))
+			return -EINVAL;
I think this could break some user space applications here, those who
e.g. only pass
an uint16_t to packet_getsockopt with PACKET_HDRLEN.
Well, their shit is broken on big endian then.
There must be something else going on anyway ... yes, my patch fixes the
warning/error, but copy_from_user should only bail out if the copy size
can be larger than the provided buffer (unless I misunderstand the code
in copy_from_user). And the second check should take care of that.
Fair enough, from what I read the implementation on x86_64 uses gcc's
__builtin_object_size(<X>, 0) [1]. Since the <to> (<X>) argument is known
at compile time (val:int), __builtin_object_size() will return sizeof(int)-1,
the number of bytes from val start to the end of the object val pointer
points to. Since our length that we pass can be [0, sizeof(int)] the
compiler cannot prove it, if the copy_from_user() buffer size is correct.
Thus, "buffer size is not provably correct". Applications not passing int
to this getsockopt(2) are screwed up then anyway.

   [1] http://gcc.gnu.org/onlinedocs/gcc/Object-Size-Checking.html
Keyboard shortcuts
hback out one level
jnext message in thread
kprevious message in thread
ldrill in
Escclose help / fold thread tree
?toggle this help