/* * Copyright (c) 2017 JingPiao Chen * Copyright (c) 2017 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 "defs.h" #ifdef HAVE_LINUX_CRYPTOUSER_H # include "netlink.h" # include "nlattr.h" # include "print_fields.h" # include # include "xlat/crypto_nl_attrs.h" static bool decode_crypto_report_generic(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len, const void *const opaque_data) { tprints("{type="); printstr_ex(tcp, addr, len, QUOTE_0_TERMINATED); tprints("}"); return true; } static bool decode_crypto_report_hash(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len, const void *const opaque_data) { # ifdef HAVE_STRUCT_CRYPTO_REPORT_HASH struct crypto_report_hash rhash; if (len < sizeof(rhash)) printstrn(tcp, addr, len); else if (!umove_or_printaddr(tcp, addr, &rhash)) { PRINT_FIELD_CSTRING("{", rhash, type); PRINT_FIELD_U(", ", rhash, blocksize); PRINT_FIELD_U(", ", rhash, digestsize); tprints("}"); } # else printstrn(tcp, addr, len); # endif return true; } static bool decode_crypto_report_blkcipher(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len, const void *const opaque_data) { # ifdef HAVE_STRUCT_CRYPTO_REPORT_BLKCIPHER struct crypto_report_blkcipher rblkcipher; if (len < sizeof(rblkcipher)) printstrn(tcp, addr, len); else if (!umove_or_printaddr(tcp, addr, &rblkcipher)) { PRINT_FIELD_CSTRING("{", rblkcipher, type); PRINT_FIELD_CSTRING(", ", rblkcipher, geniv); PRINT_FIELD_U(", ", rblkcipher, blocksize); PRINT_FIELD_U(", ", rblkcipher, min_keysize); PRINT_FIELD_U(", ", rblkcipher, max_keysize); PRINT_FIELD_U(", ", rblkcipher, ivsize); tprints("}"); } # else printstrn(tcp, addr, len); # endif return true; } static bool decode_crypto_report_aead(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len, const void *const opaque_data) { # ifdef HAVE_STRUCT_CRYPTO_REPORT_AEAD struct crypto_report_aead raead; if (len < sizeof(raead)) printstrn(tcp, addr, len); else if (!umove_or_printaddr(tcp, addr, &raead)) { PRINT_FIELD_CSTRING("{", raead, type); PRINT_FIELD_CSTRING(", ", raead, geniv); PRINT_FIELD_U(", ", raead, blocksize); PRINT_FIELD_U(", ", raead, maxauthsize); PRINT_FIELD_U(", ", raead, ivsize); tprints("}"); } # else printstrn(tcp, addr, len); # endif return true; } static bool decode_crypto_report_rng(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len, const void *const opaque_data) { # ifdef HAVE_STRUCT_CRYPTO_REPORT_RNG struct crypto_report_rng rrng; if (len < sizeof(rrng)) printstrn(tcp, addr, len); else if (!umove_or_printaddr(tcp, addr, &rrng)) { PRINT_FIELD_CSTRING("{", rrng, type); PRINT_FIELD_U(", ", rrng, seedsize); tprints("}"); } # else printstrn(tcp, addr, len); # endif return true; } static bool decode_crypto_report_cipher(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len, const void *const opaque_data) { # ifdef HAVE_STRUCT_CRYPTO_REPORT_CIPHER struct crypto_report_cipher rcipher; if (len < sizeof(rcipher)) printstrn(tcp, addr, len); else if (!umove_or_printaddr(tcp, addr, &rcipher)) { PRINT_FIELD_CSTRING("{", rcipher, type); PRINT_FIELD_U(", ", rcipher, blocksize); PRINT_FIELD_U(", ", rcipher, min_keysize); PRINT_FIELD_U(", ", rcipher, max_keysize); tprints("}"); } # else printstrn(tcp, addr, len); # endif return true; } static const nla_decoder_t crypto_user_alg_nla_decoders[] = { [CRYPTOCFGA_PRIORITY_VAL] = decode_nla_u32, [CRYPTOCFGA_REPORT_LARVAL] = decode_crypto_report_generic, [CRYPTOCFGA_REPORT_HASH] = decode_crypto_report_hash, [CRYPTOCFGA_REPORT_BLKCIPHER] = decode_crypto_report_blkcipher, [CRYPTOCFGA_REPORT_AEAD] = decode_crypto_report_aead, [CRYPTOCFGA_REPORT_COMPRESS] = decode_crypto_report_generic, [CRYPTOCFGA_REPORT_RNG] = decode_crypto_report_rng, [CRYPTOCFGA_REPORT_CIPHER] = decode_crypto_report_cipher, [CRYPTOCFGA_REPORT_AKCIPHER] = decode_crypto_report_generic, [CRYPTOCFGA_REPORT_KPP] = decode_crypto_report_generic, [CRYPTOCFGA_REPORT_ACOMP] = decode_crypto_report_generic }; static void decode_crypto_user_alg(struct tcb *const tcp, const kernel_ulong_t addr, const unsigned int len) { struct crypto_user_alg alg; if (len < sizeof(alg)) printstrn(tcp, addr, len); else if (!umove_or_printaddr(tcp, addr, &alg)) { PRINT_FIELD_CSTRING("{", alg, cru_name); PRINT_FIELD_CSTRING(", ", alg, cru_driver_name); PRINT_FIELD_CSTRING(", ", alg, cru_module_name); PRINT_FIELD_X(", ", alg, cru_type); PRINT_FIELD_X(", ", alg, cru_mask); PRINT_FIELD_U(", ", alg, cru_refcnt); PRINT_FIELD_X(", ", alg, cru_flags); tprints("}"); const size_t offset = NLMSG_ALIGN(sizeof(alg)); if (len > offset) { tprints(", "); decode_nlattr(tcp, addr + offset, len - offset, crypto_nl_attrs, "CRYPTOCFGA_???", crypto_user_alg_nla_decoders, ARRAY_SIZE(crypto_user_alg_nla_decoders), NULL); } } } bool decode_netlink_crypto(struct tcb *const tcp, const struct nlmsghdr *const nlmsghdr, const kernel_ulong_t addr, const unsigned int len) { switch (nlmsghdr->nlmsg_type) { case CRYPTO_MSG_NEWALG: case CRYPTO_MSG_DELALG: case CRYPTO_MSG_UPDATEALG: case CRYPTO_MSG_GETALG: decode_crypto_user_alg(tcp, addr, len); break; default: return false; } return true; } #endif /* HAVE_LINUX_CRYPTOUSER_H */