|  | /* | 
|  | * Copyright (c) 2017 JingPiao Chen <chenjingpiao@gmail.com> | 
|  | * 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 <sys/socket.h> | 
|  |  | 
|  | #ifndef AF_SMC | 
|  | # define AF_SMC 43 | 
|  | #endif | 
|  |  | 
|  | #include <stdio.h> | 
|  | #include <string.h> | 
|  | #include <stdint.h> | 
|  | #include <arpa/inet.h> | 
|  | #include "test_nlattr.h" | 
|  | #include <linux/rtnetlink.h> | 
|  | #include <linux/smc_diag.h> | 
|  | #include <linux/sock_diag.h> | 
|  |  | 
|  | #ifndef SMC_CLNT | 
|  | # define SMC_CLNT 0 | 
|  | #endif | 
|  | #ifndef SMC_ACTIVE | 
|  | # define SMC_ACTIVE 1 | 
|  | #endif | 
|  |  | 
|  | static const char address[] = "12.34.56.78"; | 
|  |  | 
|  | static void | 
|  | init_smc_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 smc_diag_msg *const msg = NLMSG_DATA(nlh); | 
|  | SET_STRUCT(struct smc_diag_msg, msg, | 
|  | .diag_family = AF_SMC, | 
|  | .diag_state = SMC_ACTIVE | 
|  | ); | 
|  |  | 
|  | if (!inet_pton(AF_INET, address, msg->id.idiag_src) || | 
|  | !inet_pton(AF_INET, address, msg->id.idiag_dst)) | 
|  | perror_msg_and_skip("inet_pton"); | 
|  | } | 
|  |  | 
|  | static void | 
|  | print_smc_diag_msg(const unsigned int msg_len) | 
|  | { | 
|  | printf("{len=%u, type=SOCK_DIAG_BY_FAMILY" | 
|  | ", flags=NLM_F_DUMP, seq=0, pid=0}" | 
|  | ", {diag_family=AF_SMC, diag_state=SMC_ACTIVE" | 
|  | ", diag_fallback=SMC_DIAG_MODE_SMCR, diag_shutdown=0" | 
|  | ", id={idiag_sport=htons(0), idiag_dport=htons(0)" | 
|  | ", idiag_src=inet_addr(\"%s\")" | 
|  | ", idiag_dst=inet_addr(\"%s\")" | 
|  | ", idiag_if=0, idiag_cookie=[0, 0]}" | 
|  | ", diag_uid=0, diag_inode=0}", | 
|  | msg_len, address, address); | 
|  | } | 
|  |  | 
|  | #define PRINT_FIELD_SMC_DIAG_CURSOR(prefix_, where_, field_)		\ | 
|  | do {								\ | 
|  | printf("%s%s=", (prefix_), #field_);			\ | 
|  | PRINT_FIELD_U("{", (where_).field_, reserved);		\ | 
|  | PRINT_FIELD_U(", ", (where_).field_, wrap);		\ | 
|  | PRINT_FIELD_U(", ", (where_).field_, count);		\ | 
|  | printf("}");						\ | 
|  | } while (0) | 
|  |  | 
|  | int main(void) | 
|  | { | 
|  | skip_if_unavailable("/proc/self/fd/"); | 
|  |  | 
|  | static const struct smc_diag_conninfo cinfo = { | 
|  | .token = 0xabcdefac, | 
|  | .sndbuf_size = 0xbcdaefad, | 
|  | .rmbe_size = 0xcdbaefab, | 
|  | .peer_rmbe_size = 0xdbcdedaf, | 
|  | .rx_prod = { | 
|  | .reserved = 0xabc1, | 
|  | .wrap = 0xbca1, | 
|  | .count = 0xcdedbad1 | 
|  | }, | 
|  | .rx_cons = { | 
|  | .reserved = 0xabc2, | 
|  | .wrap = 0xbca2, | 
|  | .count = 0xcdedbad2 | 
|  | }, | 
|  | .tx_prod = { | 
|  | .reserved = 0xabc3, | 
|  | .wrap = 0xbca3, | 
|  | .count = 0xcdedbad3 | 
|  | }, | 
|  | .tx_cons = { | 
|  | .reserved = 0xabc4, | 
|  | .wrap = 0xbca4, | 
|  | .count = 0xcdedbad4 | 
|  | }, | 
|  | .rx_prod_flags = 0xff, | 
|  | .rx_conn_state_flags = 0xff, | 
|  | .tx_prod_flags = 0xff, | 
|  | .tx_conn_state_flags = 0xff, | 
|  | .tx_prep = { | 
|  | .reserved = 0xabc5, | 
|  | .wrap = 0xbca5, | 
|  | .count = 0xcdedbad5 | 
|  | }, | 
|  | .tx_sent = { | 
|  | .reserved = 0xabc6, | 
|  | .wrap = 0xbca6, | 
|  | .count = 0xcdedbad6 | 
|  | }, | 
|  | .tx_fin = { | 
|  | .reserved = 0xabc7, | 
|  | .wrap = 0xbca7, | 
|  | .count = 0xcdedbad7 | 
|  | } | 
|  | }; | 
|  | static const struct smc_diag_lgrinfo linfo = { | 
|  | .lnk[0] = { | 
|  | .link_id = 0xaf, | 
|  | .ibport = 0xfa, | 
|  | .ibname = "123", | 
|  | .gid = "456", | 
|  | .peer_gid = "789" | 
|  | }, | 
|  | .role = SMC_CLNT | 
|  | }; | 
|  | static const struct smcd_diag_dmbinfo dinfo = { | 
|  | .linkid     = 0xdeadc0de, | 
|  | .peer_gid   = 0xbefeededbadc0dedULL, | 
|  | .my_gid     = 0xdeec0dedfacebeefULL, | 
|  | .token      = 0xcafedecaffeedeedULL, | 
|  | .peer_token = 0xfeedfacebeeff00dULL, | 
|  | }; | 
|  | static const struct smc_diag_fallback fb1 = { | 
|  | .reason         = 0, | 
|  | .peer_diagnosis = 0x03020000, | 
|  | }; | 
|  | static const struct smc_diag_fallback fb2 = { | 
|  | .reason         = 0x03060000, | 
|  | .peer_diagnosis = 0x99999999, | 
|  | }; | 
|  | static uint8_t sd1 = 0x23; | 
|  | static uint8_t sd2 = 0x40; | 
|  |  | 
|  | int fd = create_nl_socket(NETLINK_SOCK_DIAG); | 
|  | const unsigned int hdrlen = sizeof(struct smc_diag_msg); | 
|  | void *const nlh0 = midtail_alloc(NLMSG_SPACE(hdrlen), | 
|  | NLA_HDRLEN + | 
|  | MAX(sizeof(cinfo), sizeof(linfo))); | 
|  |  | 
|  | static char pattern[4096]; | 
|  | fill_memory_ex(pattern, sizeof(pattern), 'a', 'z' - 'a' + 1); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_SHUTDOWN, pattern, sd1, | 
|  | printf("RCV_SHUTDOWN|SEND_SHUTDOWN|0x20")); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_SHUTDOWN, pattern, sd2, | 
|  | printf("0x40 /* ???_SHUTDOWN */")); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_CONNINFO, pattern, cinfo, | 
|  | PRINT_FIELD_U("{", cinfo, token); | 
|  | PRINT_FIELD_U(", ", cinfo, sndbuf_size); | 
|  | PRINT_FIELD_U(", ", cinfo, rmbe_size); | 
|  | PRINT_FIELD_U(", ", cinfo, peer_rmbe_size); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, rx_prod); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, rx_cons); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, tx_prod); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, tx_cons); | 
|  | printf(", rx_prod_flags=0xff"); | 
|  | printf(", rx_conn_state_flags=0xff"); | 
|  | printf(", tx_prod_flags=0xff"); | 
|  | printf(", tx_conn_state_flags=0xff"); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, tx_prep); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, tx_sent); | 
|  | PRINT_FIELD_SMC_DIAG_CURSOR(", ", cinfo, tx_fin); | 
|  | printf("}")); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_LGRINFO, pattern, linfo, | 
|  | PRINT_FIELD_U("{lnk[0]={", linfo.lnk[0], link_id); | 
|  | printf(", ibname=\"%s\"", linfo.lnk[0].ibname); | 
|  | PRINT_FIELD_U(", ", linfo.lnk[0], ibport); | 
|  | printf(", gid=\"%s\"", linfo.lnk[0].gid); | 
|  | printf(", peer_gid=\"%s\"}", linfo.lnk[0].peer_gid); | 
|  | printf(", role=SMC_CLNT}")); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_DMBINFO, pattern, dinfo, | 
|  | PRINT_FIELD_U("{", dinfo, linkid); | 
|  | PRINT_FIELD_X(", ", dinfo, peer_gid); | 
|  | PRINT_FIELD_X(", ", dinfo, my_gid); | 
|  | PRINT_FIELD_X(", ", dinfo, token); | 
|  | PRINT_FIELD_X(", ", dinfo, peer_token); | 
|  | printf("}")); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_FALLBACK, pattern, fb1, | 
|  | printf("{reason=0 /* SMC_CLC_DECL_??? */"); | 
|  | printf(", peer_diagnosis=0x3020000" | 
|  | " /* SMC_CLC_DECL_IPSEC */}")); | 
|  |  | 
|  | TEST_NLATTR_OBJECT(fd, nlh0, hdrlen, | 
|  | init_smc_diag_msg, print_smc_diag_msg, | 
|  | SMC_DIAG_FALLBACK, pattern, fb2, | 
|  | printf("{reason=0x3060000" | 
|  | " /* SMC_CLC_DECL_OPTUNSUPP */"); | 
|  | printf(", peer_diagnosis=0x99999999" | 
|  | " /* SMC_CLC_DECL_??? */}")); | 
|  |  | 
|  | printf("+++ exited with 0 +++\n"); | 
|  | return 0; | 
|  | } |