2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2011-12-15 02:44:35 +00:00
# include <linux/types.h>
# include <linux/spinlock.h>
# include <linux/sock_diag.h>
# include <linux/unix_diag.h>
# include <linux/skbuff.h>
2011-12-20 04:33:03 +00:00
# include <linux/module.h>
2019-05-20 19:43:51 -05:00
# include <linux/uidgid.h>
2011-12-15 02:44:35 +00:00
# include <net/netlink.h>
# include <net/af_unix.h>
# include <net/tcp_states.h>
2019-05-20 19:43:51 -05:00
# include <net/sock.h>
2011-12-15 02:44:35 +00:00
2011-12-15 02:45:24 +00:00
static int sk_diag_dump_name ( struct sock * sk , struct sk_buff * nlskb )
{
2022-06-21 10:19:13 -07:00
/* might or might not have a hash table lock */
2019-02-15 20:09:35 +00:00
struct unix_address * addr = smp_load_acquire ( & unix_sk ( sk ) - > addr ) ;
2011-12-15 02:45:24 +00:00
2012-06-26 23:36:10 +00:00
if ( ! addr )
return 0 ;
2011-12-15 02:45:24 +00:00
2021-11-24 11:14:19 +09:00
return nla_put ( nlskb , UNIX_DIAG_NAME ,
addr - > len - offsetof ( struct sockaddr_un , sun_path ) ,
2012-06-26 23:36:10 +00:00
addr - > name - > sun_path ) ;
2011-12-15 02:45:24 +00:00
}
2011-12-15 02:45:43 +00:00
static int sk_diag_dump_vfs ( struct sock * sk , struct sk_buff * nlskb )
{
2012-03-14 21:54:32 -04:00
struct dentry * dentry = unix_sk ( sk ) - > path . dentry ;
2011-12-15 02:45:43 +00:00
if ( dentry ) {
2012-06-26 23:36:10 +00:00
struct unix_diag_vfs uv = {
2015-03-17 22:26:21 +00:00
. udiag_vfs_ino = d_backing_inode ( dentry ) - > i_ino ,
2012-06-26 23:36:10 +00:00
. udiag_vfs_dev = dentry - > d_sb - > s_dev ,
} ;
return nla_put ( nlskb , UNIX_DIAG_VFS , sizeof ( uv ) , & uv ) ;
2011-12-15 02:45:43 +00:00
}
return 0 ;
}
2011-12-15 02:45:58 +00:00
static int sk_diag_dump_peer ( struct sock * sk , struct sk_buff * nlskb )
{
struct sock * peer ;
int ino ;
peer = unix_peer_get ( sk ) ;
if ( peer ) {
unix_state_lock ( peer ) ;
ino = sock_i_ino ( peer ) ;
unix_state_unlock ( peer ) ;
sock_put ( peer ) ;
2012-06-26 23:36:10 +00:00
return nla_put_u32 ( nlskb , UNIX_DIAG_PEER , ino ) ;
2011-12-15 02:45:58 +00:00
}
return 0 ;
}
2011-12-15 02:46:14 +00:00
static int sk_diag_dump_icons ( struct sock * sk , struct sk_buff * nlskb )
{
struct sk_buff * skb ;
2012-06-26 23:36:10 +00:00
struct nlattr * attr ;
2011-12-15 02:46:14 +00:00
u32 * buf ;
int i ;
if ( sk - > sk_state = = TCP_LISTEN ) {
spin_lock ( & sk - > sk_receive_queue . lock ) ;
2012-06-26 23:36:10 +00:00
attr = nla_reserve ( nlskb , UNIX_DIAG_ICONS ,
sk - > sk_receive_queue . qlen * sizeof ( u32 ) ) ;
if ( ! attr )
goto errout ;
buf = nla_data ( attr ) ;
2011-12-15 02:46:14 +00:00
i = 0 ;
skb_queue_walk ( & sk - > sk_receive_queue , skb ) {
struct sock * req , * peer ;
req = skb - > sk ;
/*
* The state lock is outer for the same sk ' s
* queue lock . With the other ' s queue locked it ' s
* OK to lock the state .
*/
unix_state_lock_nested ( req ) ;
peer = unix_sk ( req ) - > peer ;
2011-12-26 14:41:55 -05:00
buf [ i + + ] = ( peer ? sock_i_ino ( peer ) : 0 ) ;
2011-12-15 02:46:14 +00:00
unix_state_unlock ( req ) ;
}
spin_unlock ( & sk - > sk_receive_queue . lock ) ;
}
return 0 ;
2012-06-26 23:36:10 +00:00
errout :
2011-12-15 02:46:14 +00:00
spin_unlock ( & sk - > sk_receive_queue . lock ) ;
return - EMSGSIZE ;
}
2011-12-15 02:46:31 +00:00
static int sk_diag_show_rqlen ( struct sock * sk , struct sk_buff * nlskb )
{
2012-06-26 23:36:10 +00:00
struct unix_diag_rqlen rql ;
2011-12-30 00:54:39 +00:00
if ( sk - > sk_state = = TCP_LISTEN ) {
2012-06-26 23:36:10 +00:00
rql . udiag_rqueue = sk - > sk_receive_queue . qlen ;
rql . udiag_wqueue = sk - > sk_max_ack_backlog ;
2011-12-30 00:54:39 +00:00
} else {
2012-06-26 23:36:10 +00:00
rql . udiag_rqueue = ( u32 ) unix_inq_len ( sk ) ;
rql . udiag_wqueue = ( u32 ) unix_outq_len ( sk ) ;
2011-12-30 00:54:39 +00:00
}
2012-06-26 23:36:10 +00:00
return nla_put ( nlskb , UNIX_DIAG_RQLEN , sizeof ( rql ) , & rql ) ;
2011-12-15 02:46:31 +00:00
}
2022-11-27 10:24:11 +09:00
static int sk_diag_dump_uid ( struct sock * sk , struct sk_buff * nlskb ,
struct user_namespace * user_ns )
2019-05-20 19:43:51 -05:00
{
2022-11-27 10:24:11 +09:00
uid_t uid = from_kuid_munged ( user_ns , sock_i_uid ( sk ) ) ;
2019-05-20 19:43:51 -05:00
return nla_put ( nlskb , UNIX_DIAG_UID , sizeof ( uid_t ) , & uid ) ;
}
2011-12-15 02:44:52 +00:00
static int sk_diag_fill ( struct sock * sk , struct sk_buff * skb , struct unix_diag_req * req ,
2022-11-27 10:24:11 +09:00
struct user_namespace * user_ns ,
u32 portid , u32 seq , u32 flags , int sk_ino )
2011-12-15 02:44:52 +00:00
{
struct nlmsghdr * nlh ;
struct unix_diag_msg * rep ;
2012-09-07 20:12:54 +00:00
nlh = nlmsg_put ( skb , portid , seq , SOCK_DIAG_BY_FAMILY , sizeof ( * rep ) ,
2012-06-26 23:36:10 +00:00
flags ) ;
2012-06-26 21:41:00 -07:00
if ( ! nlh )
2012-06-26 23:36:10 +00:00
return - EMSGSIZE ;
2011-12-15 02:44:52 +00:00
2012-06-26 21:41:00 -07:00
rep = nlmsg_data ( nlh ) ;
2011-12-15 02:44:52 +00:00
rep - > udiag_family = AF_UNIX ;
rep - > udiag_type = sk - > sk_type ;
rep - > udiag_state = sk - > sk_state ;
2013-09-30 22:05:40 +02:00
rep - > pad = 0 ;
2011-12-15 02:44:52 +00:00
rep - > udiag_ino = sk_ino ;
sock_diag_save_cookie ( sk , rep - > udiag_cookie ) ;
2011-12-15 02:45:24 +00:00
if ( ( req - > udiag_show & UDIAG_SHOW_NAME ) & &
2011-12-30 09:27:43 +00:00
sk_diag_dump_name ( sk , skb ) )
2012-06-26 21:41:00 -07:00
goto out_nlmsg_trim ;
2011-12-15 02:45:24 +00:00
2011-12-15 02:45:43 +00:00
if ( ( req - > udiag_show & UDIAG_SHOW_VFS ) & &
2011-12-30 09:27:43 +00:00
sk_diag_dump_vfs ( sk , skb ) )
2012-06-26 21:41:00 -07:00
goto out_nlmsg_trim ;
2011-12-15 02:45:43 +00:00
2011-12-15 02:45:58 +00:00
if ( ( req - > udiag_show & UDIAG_SHOW_PEER ) & &
2011-12-30 09:27:43 +00:00
sk_diag_dump_peer ( sk , skb ) )
2012-06-26 21:41:00 -07:00
goto out_nlmsg_trim ;
2011-12-15 02:45:58 +00:00
2011-12-15 02:46:14 +00:00
if ( ( req - > udiag_show & UDIAG_SHOW_ICONS ) & &
2011-12-30 09:27:43 +00:00
sk_diag_dump_icons ( sk , skb ) )
2012-06-26 21:41:00 -07:00
goto out_nlmsg_trim ;
2011-12-15 02:46:14 +00:00
2011-12-15 02:46:31 +00:00
if ( ( req - > udiag_show & UDIAG_SHOW_RQLEN ) & &
2011-12-30 09:27:43 +00:00
sk_diag_show_rqlen ( sk , skb ) )
2012-06-26 21:41:00 -07:00
goto out_nlmsg_trim ;
2011-12-30 09:27:43 +00:00
if ( ( req - > udiag_show & UDIAG_SHOW_MEMINFO ) & &
sock_diag_put_meminfo ( sk , skb , UNIX_DIAG_MEMINFO ) )
2012-06-26 21:41:00 -07:00
goto out_nlmsg_trim ;
2011-12-15 02:46:31 +00:00
2012-10-23 22:29:56 +04:00
if ( nla_put_u8 ( skb , UNIX_DIAG_SHUTDOWN , sk - > sk_shutdown ) )
goto out_nlmsg_trim ;
2019-05-20 19:43:51 -05:00
if ( ( req - > udiag_show & UDIAG_SHOW_UID ) & &
2022-11-27 10:24:11 +09:00
sk_diag_dump_uid ( sk , skb , user_ns ) )
2019-05-20 19:43:51 -05:00
goto out_nlmsg_trim ;
2015-01-16 22:09:00 +01:00
nlmsg_end ( skb , nlh ) ;
return 0 ;
2011-12-15 02:44:52 +00:00
2012-06-26 21:41:00 -07:00
out_nlmsg_trim :
2012-06-26 23:36:10 +00:00
nlmsg_cancel ( skb , nlh ) ;
2011-12-15 02:44:52 +00:00
return - EMSGSIZE ;
}
static int sk_diag_dump ( struct sock * sk , struct sk_buff * skb , struct unix_diag_req * req ,
2022-11-27 10:24:11 +09:00
struct user_namespace * user_ns ,
u32 portid , u32 seq , u32 flags )
2011-12-15 02:44:52 +00:00
{
int sk_ino ;
unix_state_lock ( sk ) ;
sk_ino = sock_i_ino ( sk ) ;
unix_state_unlock ( sk ) ;
if ( ! sk_ino )
return 0 ;
2022-11-27 10:24:11 +09:00
return sk_diag_fill ( sk , skb , req , user_ns , portid , seq , flags , sk_ino ) ;
2011-12-15 02:44:52 +00:00
}
2011-12-15 02:44:35 +00:00
static int unix_diag_dump ( struct sk_buff * skb , struct netlink_callback * cb )
{
2012-07-16 04:28:49 +00:00
struct net * net = sock_net ( skb - > sk ) ;
2022-06-21 10:19:11 -07:00
int num , s_num , slot , s_slot ;
struct unix_diag_req * req ;
2011-12-15 02:44:52 +00:00
2012-06-26 21:41:00 -07:00
req = nlmsg_data ( cb - > nlh ) ;
2011-12-15 02:44:52 +00:00
s_slot = cb - > args [ 0 ] ;
num = s_num = cb - > args [ 1 ] ;
2022-06-21 10:19:09 -07:00
for ( slot = s_slot ; slot < UNIX_HASH_SIZE ; s_num = 0 , slot + + ) {
2011-12-15 02:44:52 +00:00
struct sock * sk ;
num = 0 ;
2022-06-21 10:19:11 -07:00
spin_lock ( & net - > unx . table . locks [ slot ] ) ;
2022-06-21 10:19:12 -07:00
sk_for_each ( sk , & net - > unx . table . buckets [ slot ] ) {
2011-12-15 02:44:52 +00:00
if ( num < s_num )
goto next ;
if ( ! ( req - > udiag_states & ( 1 < < sk - > sk_state ) ) )
goto next ;
2022-11-27 10:24:11 +09:00
if ( sk_diag_dump ( sk , skb , req , sk_user_ns ( skb - > sk ) ,
2012-09-07 20:12:54 +00:00
NETLINK_CB ( cb - > skb ) . portid ,
2011-12-30 09:27:43 +00:00
cb - > nlh - > nlmsg_seq ,
2021-11-24 11:14:30 +09:00
NLM_F_MULTI ) < 0 ) {
2022-06-21 10:19:11 -07:00
spin_unlock ( & net - > unx . table . locks [ slot ] ) ;
2011-12-15 02:44:52 +00:00
goto done ;
2021-11-24 11:14:30 +09:00
}
2011-12-15 02:44:52 +00:00
next :
num + + ;
}
2022-06-21 10:19:11 -07:00
spin_unlock ( & net - > unx . table . locks [ slot ] ) ;
2011-12-15 02:44:52 +00:00
}
done :
cb - > args [ 0 ] = slot ;
cb - > args [ 1 ] = num ;
return skb - > len ;
2011-12-15 02:44:35 +00:00
}
2022-06-21 10:19:11 -07:00
static struct sock * unix_lookup_by_ino ( struct net * net , unsigned int ino )
2011-12-15 02:45:07 +00:00
{
struct sock * sk ;
2021-11-24 11:14:30 +09:00
int i ;
2011-12-15 02:45:07 +00:00
2022-06-21 10:19:09 -07:00
for ( i = 0 ; i < UNIX_HASH_SIZE ; i + + ) {
2022-06-21 10:19:11 -07:00
spin_lock ( & net - > unx . table . locks [ i ] ) ;
2022-06-21 10:19:12 -07:00
sk_for_each ( sk , & net - > unx . table . buckets [ i ] ) {
2011-12-15 02:45:07 +00:00
if ( ino = = sock_i_ino ( sk ) ) {
sock_hold ( sk ) ;
2022-06-21 10:19:11 -07:00
spin_unlock ( & net - > unx . table . locks [ i ] ) ;
2011-12-15 02:45:07 +00:00
return sk ;
}
2022-06-21 10:19:12 -07:00
}
2022-06-21 10:19:11 -07:00
spin_unlock ( & net - > unx . table . locks [ i ] ) ;
2011-12-15 02:45:07 +00:00
}
return NULL ;
}
2011-12-15 02:44:35 +00:00
static int unix_diag_get_exact ( struct sk_buff * in_skb ,
const struct nlmsghdr * nlh ,
struct unix_diag_req * req )
{
2012-07-16 04:28:49 +00:00
struct net * net = sock_net ( in_skb - > sk ) ;
2022-06-21 10:19:11 -07:00
unsigned int extra_len ;
struct sk_buff * rep ;
struct sock * sk ;
int err ;
2011-12-15 02:45:07 +00:00
2022-06-21 10:19:11 -07:00
err = - EINVAL ;
2011-12-15 02:45:07 +00:00
if ( req - > udiag_ino = = 0 )
goto out_nosk ;
2022-06-21 10:19:11 -07:00
sk = unix_lookup_by_ino ( net , req - > udiag_ino ) ;
2011-12-15 02:45:07 +00:00
err = - ENOENT ;
if ( sk = = NULL )
goto out_nosk ;
err = sock_diag_check_cookie ( sk , req - > udiag_cookie ) ;
if ( err )
goto out ;
extra_len = 256 ;
again :
err = - ENOMEM ;
2012-06-26 23:36:10 +00:00
rep = nlmsg_new ( sizeof ( struct unix_diag_msg ) + extra_len , GFP_KERNEL ) ;
2011-12-15 02:45:07 +00:00
if ( ! rep )
goto out ;
2022-11-27 10:24:11 +09:00
err = sk_diag_fill ( sk , rep , req , sk_user_ns ( NETLINK_CB ( in_skb ) . sk ) ,
NETLINK_CB ( in_skb ) . portid ,
2011-12-15 02:45:07 +00:00
nlh - > nlmsg_seq , 0 , req - > udiag_ino ) ;
if ( err < 0 ) {
2012-06-26 23:36:10 +00:00
nlmsg_free ( rep ) ;
2011-12-15 02:45:07 +00:00
extra_len + = 256 ;
if ( extra_len > = PAGE_SIZE )
goto out ;
goto again ;
}
2021-07-13 10:48:24 +08:00
err = nlmsg_unicast ( net - > diag_nlsk , rep , NETLINK_CB ( in_skb ) . portid ) ;
2011-12-15 02:45:07 +00:00
out :
if ( sk )
sock_put ( sk ) ;
out_nosk :
return err ;
2011-12-15 02:44:35 +00:00
}
static int unix_diag_handler_dump ( struct sk_buff * skb , struct nlmsghdr * h )
{
int hdrlen = sizeof ( struct unix_diag_req ) ;
if ( nlmsg_len ( h ) < hdrlen )
return - EINVAL ;
2012-02-24 14:30:15 +00:00
if ( h - > nlmsg_flags & NLM_F_DUMP ) {
struct netlink_dump_control c = {
. dump = unix_diag_dump ,
} ;
2022-06-21 10:19:08 -07:00
return netlink_dump_start ( sock_net ( skb - > sk ) - > diag_nlsk , skb , h , & c ) ;
2012-02-24 14:30:15 +00:00
} else
2012-06-26 21:41:00 -07:00
return unix_diag_get_exact ( skb , h , nlmsg_data ( h ) ) ;
2011-12-15 02:44:35 +00:00
}
2012-04-24 18:21:07 +00:00
static const struct sock_diag_handler unix_diag_handler = {
2011-12-15 02:44:35 +00:00
. family = AF_UNIX ,
. dump = unix_diag_handler_dump ,
} ;
static int __init unix_diag_init ( void )
{
return sock_diag_register ( & unix_diag_handler ) ;
}
static void __exit unix_diag_exit ( void )
{
sock_diag_unregister ( & unix_diag_handler ) ;
}
module_init ( unix_diag_init ) ;
module_exit ( unix_diag_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_NET_PF_PROTO_TYPE ( PF_NETLINK , NETLINK_SOCK_DIAG , 1 /* AF_LOCAL */ ) ;