| /* |
| * Copyright (c) 2017 JingPiao Chen <[email protected]> |
| * Copyright (c) 2017-2018 The strace developers. |
| * 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. |
| */ |
| |
| #include "tests.h" |
| |
| #include <stdio.h> |
| #include <string.h> |
| #include <stdint.h> |
| #include <net/if.h> |
| #include "test_nlattr.h" |
| #include <sys/socket.h> |
| #include <linux/filter.h> |
| #include <linux/packet_diag.h> |
| #include <linux/rtnetlink.h> |
| #include <linux/sock_diag.h> |
| |
| static void |
| init_packet_diag_msg(struct nlmsghdr *const nlh, const unsigned int msg_len) |
| { |
| SET_STRUCT(struct nlmsghdr, nlh, |
| .nlmsg_len = msg_len, |
| .nlmsg_type = SOCK_DIAG_BY_FAMILY, |
| .nlmsg_flags = NLM_F_DUMP |
| ); |
| |
| struct packet_diag_msg *const msg = NLMSG_DATA(nlh); |
| SET_STRUCT(struct packet_diag_msg, msg, |
| .pdiag_family = AF_PACKET, |
| .pdiag_type = SOCK_STREAM, |
| .pdiag_num = 3, |
| ); |
| } |
| |
| static void |
| print_packet_diag_msg(const unsigned int msg_len) |
| { |
| printf("{len=%u, type=SOCK_DIAG_BY_FAMILY" |
| ", flags=NLM_F_DUMP, seq=0, pid=0}" |
| ", {pdiag_family=AF_PACKET" |
| ", pdiag_type=SOCK_STREAM, pdiag_num=ETH_P_ALL" |
| ", pdiag_ino=0, pdiag_cookie=[0, 0]}", |
| msg_len); |
| } |
| |
| static void |
| print_packet_diag_mclist(const struct packet_diag_mclist *const dml, size_t i) |
| { |
| printf("{pdmc_index=" IFINDEX_LO_STR); |
| PRINT_FIELD_U(", ", *dml, pdmc_count); |
| PRINT_FIELD_U(", ", *dml, pdmc_type); |
| PRINT_FIELD_U(", ", *dml, pdmc_alen); |
| printf(", pdmc_addr="); |
| print_quoted_hex(dml->pdmc_addr, dml->pdmc_alen); |
| printf("}"); |
| } |
| |
| static const struct sock_filter filter[] = { |
| BPF_STMT(BPF_LD|BPF_B|BPF_ABS, SKF_AD_OFF+SKF_AD_PKTTYPE), |
| BPF_STMT(BPF_RET|BPF_K, 0x2a) |
| }; |
| |
| static void |
| print_sock_filter(const struct sock_filter *const f, size_t i) |
| { |
| if (f == filter) |
| printf("BPF_STMT(BPF_LD|BPF_B|BPF_ABS" |
| ", SKF_AD_OFF+SKF_AD_PKTTYPE)"); |
| else |
| printf("BPF_STMT(BPF_RET|BPF_K, 0x2a)"); |
| } |
| |
| int |
| main(void) |
| { |
| skip_if_unavailable("/proc/self/fd/"); |
| |
| struct packet_diag_info pinfo = { |
| .pdi_index = ifindex_lo(), |
| .pdi_version = 2, |
| .pdi_reserve = 0xcfaacdaf, |
| .pdi_copy_thresh = 0xdabacdaf, |
| .pdi_tstamp = 0xeafbaadf, |
| .pdi_flags = PDI_RUNNING |
| }; |
| const struct packet_diag_mclist dml[] = { |
| { |
| .pdmc_index = ifindex_lo(), |
| .pdmc_count = 0xabcdaefc, |
| .pdmc_type = 0xcdaf, |
| .pdmc_alen = 4, |
| .pdmc_addr = "1234" |
| }, |
| { |
| .pdmc_index = ifindex_lo(), |
| .pdmc_count = 0xdaefeafc, |
| .pdmc_type = 0xadef, |
| .pdmc_alen = 4, |
| .pdmc_addr = "5678" |
| } |
| }; |
| static const struct packet_diag_ring pdr = { |
| .pdr_block_size = 0xabcdafed, |
| .pdr_block_nr = 0xbcadefae, |
| .pdr_frame_size = 0xcabdfeac, |
| .pdr_frame_nr = 0xdeaeadef, |
| .pdr_retire_tmo = 0xedbafeac, |
| .pdr_sizeof_priv = 0xfeadeacd, |
| .pdr_features = 0xadebadea |
| }; |
| |
| int fd = create_nl_socket(NETLINK_SOCK_DIAG); |
| const unsigned int hdrlen = sizeof(struct packet_diag_msg); |
| void *const nlh0 = midtail_alloc(NLMSG_SPACE(hdrlen), |
| NLA_HDRLEN + sizeof(dml)); |
| |
| static char pattern[4096]; |
| fill_memory_ex(pattern, sizeof(pattern), 'a', 'z' - 'a' + 1); |
| |
| TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, |
| init_packet_diag_msg, print_packet_diag_msg, |
| PACKET_DIAG_INFO, pattern, pinfo, |
| printf("{pdi_index=%s", IFINDEX_LO_STR); |
| printf(", pdi_version=TPACKET_V3"); |
| PRINT_FIELD_U(", ", pinfo, pdi_reserve); |
| PRINT_FIELD_U(", ", pinfo, pdi_copy_thresh); |
| PRINT_FIELD_U(", ", pinfo, pdi_tstamp); |
| printf(", pdi_flags=PDI_RUNNING}")); |
| |
| TEST_NLATTR_ARRAY(fd, nlh0, hdrlen, |
| init_packet_diag_msg, print_packet_diag_msg, |
| PACKET_DIAG_MCLIST, pattern, dml, |
| print_packet_diag_mclist); |
| |
| TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, |
| init_packet_diag_msg, print_packet_diag_msg, |
| PACKET_DIAG_RX_RING, pattern, pdr, |
| PRINT_FIELD_U("{", pdr, pdr_block_size); |
| PRINT_FIELD_U(", ", pdr, pdr_block_nr); |
| PRINT_FIELD_U(", ", pdr, pdr_frame_size); |
| PRINT_FIELD_U(", ", pdr, pdr_frame_nr); |
| PRINT_FIELD_U(", ", pdr, pdr_retire_tmo); |
| PRINT_FIELD_U(", ", pdr, pdr_sizeof_priv); |
| PRINT_FIELD_U(", ", pdr, pdr_features); |
| printf("}")); |
| |
| TEST_NLATTR_ARRAY(fd, nlh0, hdrlen, |
| init_packet_diag_msg, print_packet_diag_msg, |
| PACKET_DIAG_FILTER, pattern, filter, |
| print_sock_filter); |
| |
| printf("+++ exited with 0 +++\n"); |
| return 0; |
| } |