2015-12-17 20:56:48 +03:00
/*
2016-01-31 02:50:54 +03:00
* This file is part of net - yy - unix strace test .
*
2016-01-06 18:55:12 +03:00
* Copyright ( c ) 2014 - 2016 Dmitry V . Levin < ldv @ altlinux . org >
2015-12-17 20:56:48 +03:00
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*/
2016-01-06 18:55:12 +03:00
# include "tests.h"
2014-12-25 03:11:40 +03:00
# include <assert.h>
2016-01-31 02:50:54 +03:00
# include <errno.h>
2014-12-25 03:11:40 +03:00
# include <stddef.h>
2015-01-09 01:29:12 +03:00
# include <stdint.h>
2014-12-25 03:11:40 +03:00
# include <string.h>
# include <unistd.h>
# include <sys/socket.h>
# include <sys/un.h>
# include <linux/netlink.h>
# include <linux/sock_diag.h>
# include <linux/unix_diag.h>
2015-01-09 06:03:39 +03:00
# if !defined NETLINK_SOCK_DIAG && defined NETLINK_INET_DIAG
# define NETLINK_SOCK_DIAG NETLINK_INET_DIAG
# endif
2016-01-06 18:55:12 +03:00
static void
2016-02-19 04:30:34 +03:00
send_query ( const int fd )
2014-12-25 03:11:40 +03:00
{
struct sockaddr_nl nladdr = {
. nl_family = AF_NETLINK
} ;
struct {
struct nlmsghdr nlh ;
struct unix_diag_req udr ;
} req = {
. nlh = {
. nlmsg_len = sizeof ( req ) ,
. nlmsg_type = SOCK_DIAG_BY_FAMILY ,
2016-02-19 04:30:34 +03:00
. nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP
2014-12-25 03:11:40 +03:00
} ,
. udr = {
2016-01-31 03:35:29 +03:00
. sdiag_family = AF_UNIX ,
2014-12-25 03:11:40 +03:00
. udiag_states = - 1 ,
2016-02-19 04:30:34 +03:00
. udiag_show = UDIAG_SHOW_NAME | UDIAG_SHOW_PEER
2014-12-25 03:11:40 +03:00
}
} ;
struct iovec iov = {
. iov_base = & req ,
. iov_len = sizeof ( req )
} ;
struct msghdr msg = {
2016-01-31 02:50:54 +03:00
. msg_name = ( void * ) & nladdr ,
2014-12-25 03:11:40 +03:00
. msg_namelen = sizeof ( nladdr ) ,
. msg_iov = & iov ,
. msg_iovlen = 1
} ;
2016-01-06 18:55:12 +03:00
if ( sendmsg ( fd , & msg , 0 ) < = 0 )
perror_msg_and_skip ( " sendmsg " ) ;
2014-12-25 03:11:40 +03:00
}
2016-01-06 18:55:12 +03:00
static void
2016-02-19 04:30:34 +03:00
check_responses ( const int fd )
2014-12-25 03:11:40 +03:00
{
2016-02-24 05:51:09 +03:00
static long buf [ 8192 / sizeof ( long ) ] ;
2014-12-25 03:11:40 +03:00
struct sockaddr_nl nladdr = {
. nl_family = AF_NETLINK
} ;
struct iovec iov = {
. iov_base = buf ,
. iov_len = sizeof ( buf )
} ;
struct msghdr msg = {
2016-01-31 02:50:54 +03:00
. msg_name = ( void * ) & nladdr ,
2014-12-25 03:11:40 +03:00
. msg_namelen = sizeof ( nladdr ) ,
. msg_iov = & iov ,
. msg_iovlen = 1
} ;
ssize_t ret = recvmsg ( fd , & msg , 0 ) ;
if ( ret < = 0 )
2016-01-06 18:55:12 +03:00
perror_msg_and_skip ( " recvmsg " ) ;
2014-12-25 03:11:40 +03:00
2016-01-31 02:50:54 +03:00
struct nlmsghdr * h = ( struct nlmsghdr * ) buf ;
2016-01-06 18:55:12 +03:00
if ( ! NLMSG_OK ( h , ret ) )
error_msg_and_skip ( " !NLMSG_OK " ) ;
2016-01-31 02:50:54 +03:00
if ( h - > nlmsg_type = = NLMSG_ERROR ) {
const struct nlmsgerr * err = NLMSG_DATA ( h ) ;
if ( h - > nlmsg_len < NLMSG_LENGTH ( sizeof ( * err ) ) )
error_msg_and_skip ( " NLMSG_ERROR " ) ;
errno = - err - > error ;
perror_msg_and_skip ( " NLMSG_ERROR " ) ;
}
if ( h - > nlmsg_type ! = SOCK_DIAG_BY_FAMILY )
error_msg_and_skip ( " unexpected nlmsg_type %u " ,
( unsigned ) h - > nlmsg_type ) ;
const struct unix_diag_msg * diag = NLMSG_DATA ( h ) ;
if ( h - > nlmsg_len < NLMSG_LENGTH ( sizeof ( * diag ) ) )
error_msg_and_skip ( " short response " ) ;
2014-12-25 03:11:40 +03:00
}
# define SUN_PATH "netlink_unix_diag_socket"
int main ( void )
{
struct sockaddr_un addr = {
. sun_family = AF_UNIX ,
. sun_path = SUN_PATH
} ;
socklen_t len = offsetof ( struct sockaddr_un , sun_path ) + sizeof ( SUN_PATH ) ;
close ( 0 ) ;
close ( 1 ) ;
( void ) unlink ( SUN_PATH ) ;
2016-04-04 04:35:28 +03:00
if ( socket ( AF_LOCAL , SOCK_STREAM , 0 ) )
perror_msg_and_skip ( " socket AF_LOCAL " ) ;
2016-01-06 18:55:12 +03:00
if ( bind ( 0 , ( struct sockaddr * ) & addr , len ) )
perror_msg_and_skip ( " bind " ) ;
if ( listen ( 0 , 5 ) )
perror_msg_and_skip ( " listen " ) ;
2015-03-18 23:18:27 +03:00
2014-12-25 03:11:40 +03:00
assert ( unlink ( SUN_PATH ) = = 0 ) ;
if ( socket ( AF_NETLINK , SOCK_RAW , NETLINK_SOCK_DIAG ) ! = 1 )
2016-01-06 18:55:12 +03:00
perror_msg_and_skip ( " socket AF_NETLINK " ) ;
2014-12-25 03:11:40 +03:00
2016-02-19 04:30:34 +03:00
send_query ( 1 ) ;
check_responses ( 1 ) ;
2016-01-06 18:55:12 +03:00
return 0 ;
2014-12-25 03:11:40 +03:00
}