/* * Copyright (c) 2021, ARM Limited and Contributors. All rights reserved. * * SPDX-License-Identifier: BSD-3-Clause */ #include #include #include #include #include #include #include #include #include "trng_entropy_pool.h" static const uuid_t uuid_null; /* handle the RND call in SMC 32 bit mode */ static uintptr_t trng_rnd32(uint32_t nbits, void *handle) { uint32_t mask = ~0U; uint64_t ent[2]; if (nbits == 0U || nbits > 96U) { SMC_RET1(handle, TRNG_E_INVALID_PARAMS); } if (!trng_pack_entropy(nbits, &ent[0])) { SMC_RET1(handle, TRNG_E_NO_ENTROPY); } if ((nbits % 32U) != 0U) { mask >>= 32U - (nbits % 32U); } switch ((nbits - 1U) / 32U) { case 0: SMC_RET4(handle, TRNG_E_SUCCESS, 0, 0, ent[0] & mask); break; /* unreachable */ case 1: SMC_RET4(handle, TRNG_E_SUCCESS, 0, (ent[0] >> 32) & mask, ent[0] & 0xFFFFFFFF); break; /* unreachable */ case 2: SMC_RET4(handle, TRNG_E_SUCCESS, ent[1] & mask, (ent[0] >> 32) & 0xFFFFFFFF, ent[0] & 0xFFFFFFFF); break; /* unreachable */ default: SMC_RET1(handle, TRNG_E_INVALID_PARAMS); break; /* unreachable */ } } /* handle the RND call in SMC 64 bit mode */ static uintptr_t trng_rnd64(uint32_t nbits, void *handle) { uint64_t mask = ~0ULL; uint64_t ent[3]; if (nbits == 0U || nbits > 192U) { SMC_RET1(handle, TRNG_E_INVALID_PARAMS); } if (!trng_pack_entropy(nbits, &ent[0])) { SMC_RET1(handle, TRNG_E_NO_ENTROPY); } /* Mask off higher bits if only part of register requested */ if ((nbits % 64U) != 0U) { mask >>= 64U - (nbits % 64U); } switch ((nbits - 1U) / 64U) { case 0: SMC_RET4(handle, TRNG_E_SUCCESS, 0, 0, ent[0] & mask); break; /* unreachable */ case 1: SMC_RET4(handle, TRNG_E_SUCCESS, 0, ent[1] & mask, ent[0]); break; /* unreachable */ case 2: SMC_RET4(handle, TRNG_E_SUCCESS, ent[2] & mask, ent[1], ent[0]); break; /* unreachable */ default: SMC_RET1(handle, TRNG_E_INVALID_PARAMS); break; /* unreachable */ } } void trng_setup(void) { trng_entropy_pool_setup(); plat_entropy_setup(); } /* Predicate indicating that a function id is part of TRNG */ bool is_trng_fid(uint32_t smc_fid) { return ((smc_fid == ARM_TRNG_VERSION) || (smc_fid == ARM_TRNG_FEATURES) || (smc_fid == ARM_TRNG_GET_UUID) || (smc_fid == ARM_TRNG_RND32) || (smc_fid == ARM_TRNG_RND64)); } uintptr_t trng_smc_handler(uint32_t smc_fid, u_register_t x1, u_register_t x2, u_register_t x3, u_register_t x4, void *cookie, void *handle, u_register_t flags) { if (!memcmp(&plat_trng_uuid, &uuid_null, sizeof(uuid_t))) { SMC_RET1(handle, TRNG_E_NOT_IMPLEMENTED); } switch (smc_fid) { case ARM_TRNG_VERSION: SMC_RET1(handle, MAKE_SMCCC_VERSION( TRNG_VERSION_MAJOR, TRNG_VERSION_MINOR )); break; /* unreachable */ case ARM_TRNG_FEATURES: if (is_trng_fid((uint32_t)x1)) { SMC_RET1(handle, TRNG_E_SUCCESS); } else { SMC_RET1(handle, TRNG_E_NOT_SUPPORTED); } break; /* unreachable */ case ARM_TRNG_GET_UUID: SMC_UUID_RET(handle, plat_trng_uuid); break; /* unreachable */ case ARM_TRNG_RND32: return trng_rnd32((uint32_t)x1, handle); case ARM_TRNG_RND64: return trng_rnd64((uint32_t)x1, handle); default: WARN("Unimplemented TRNG Service Call: 0x%x\n", smc_fid); SMC_RET1(handle, TRNG_E_NOT_IMPLEMENTED); break; /* unreachable */ } }