|  | /* | 
|  | * Copyright (c) 2017 JingPiao Chen <chenjingpiao@gmail.com> | 
|  | * 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 <linux/cryptouser.h> | 
|  |  | 
|  | # 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 */ |