llc: fix info leak via getsockname()
The LLC code wrongly returns 0, i.e. "success", when the socket is zapped. Together with the uninitialized uaddrlen pointer argument from sys_getsockname this leads to an arbitrary memory leak of up to 128 bytes kernel stack via the getsockname() syscall. Return an error instead when the socket is zapped to prevent the info leak. Also remove the unnecessary memset(0). We don't directly write to the memory pointed by uaddr but memcpy() a local structure at the end of the function that is properly initialized. Signed-off-by: Mathias Krause <minipli@googlemail.com> Cc: Arnaldo Carvalho de Melo <acme@ghostprotocols.net> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
04d4fbca10
commit
3592aaeb80
@ -969,14 +969,13 @@ static int llc_ui_getname(struct socket *sock, struct sockaddr *uaddr,
|
|||||||
struct sockaddr_llc sllc;
|
struct sockaddr_llc sllc;
|
||||||
struct sock *sk = sock->sk;
|
struct sock *sk = sock->sk;
|
||||||
struct llc_sock *llc = llc_sk(sk);
|
struct llc_sock *llc = llc_sk(sk);
|
||||||
int rc = 0;
|
int rc = -EBADF;
|
||||||
|
|
||||||
memset(&sllc, 0, sizeof(sllc));
|
memset(&sllc, 0, sizeof(sllc));
|
||||||
lock_sock(sk);
|
lock_sock(sk);
|
||||||
if (sock_flag(sk, SOCK_ZAPPED))
|
if (sock_flag(sk, SOCK_ZAPPED))
|
||||||
goto out;
|
goto out;
|
||||||
*uaddrlen = sizeof(sllc);
|
*uaddrlen = sizeof(sllc);
|
||||||
memset(uaddr, 0, *uaddrlen);
|
|
||||||
if (peer) {
|
if (peer) {
|
||||||
rc = -ENOTCONN;
|
rc = -ENOTCONN;
|
||||||
if (sk->sk_state != TCP_ESTABLISHED)
|
if (sk->sk_state != TCP_ESTABLISHED)
|
||||||
|
Loading…
Reference in New Issue
Block a user