On Sun, 27 Oct 2019 at 13:38, Richard Henderson <richard.henderson@xxxxxxxxxx> wrote: > > On 10/24/19 1:57 PM, Ard Biesheuvel wrote: > >>> + > >>> +#ifdef CONFIG_ARCH_RANDOM > >>> + > >>> +/* > >>> + * Note that these two interfaces, random and random_seed, are strongly > >>> + * tied to the Intel instructions RDRAND and RDSEED. RDSEED is the > >>> + * "enhanced" version and has stronger guarantees. The ARMv8.5-RNG > >>> + * instruction RNDR corresponds to RDSEED, thus we put our implementation > >>> + * into the random_seed set of functions. > >>> + * > > > > Is that accurate? The ARM ARM says that RNDR is backed by a DRBG which > > > > "" > > ...is reseeded after an IMPLEMENTATION DEFINED number of random > > numbers has been generated and read > > using the RNDR register. > > """ > > > > which means that you cannot rely on this reseeding to take place at all. > > > > So the way I read this, RNDR ~= RDRAND and RNDRRS ~= RDSEED, and we > > should wire up the functions below accordingly. > > No, that reading is not correct, and is exactly what I was trying to explain in > that paragraph. > > RNDR ~= RDSEED. > > It's a matter of standards conformance: > > RDRAND: NIST SP800-90A. > > RDSEED: NIST SP800-90B, > NIST SP800-90C. > > RNDR: NIST SP800-90A Rev 1, > NIST SP800-90B, > NIST SP800-22, > FIPS 140-2, > BSI AIS-31, > NIST SP800-90C. > That is not what the ARM ARM says (DDI0487E.a K12.1): The *TRNG* that seeds the DRBG that backs both RNDR and RNDRRS should conform to • The NIST SP800-90B standard. • The NIST SP800-22 standard. • The FIPS 140-2 standard. • The BSI AIS-31 standard. This DRBG itself should conform to NIST SP800-90A Rev 1, and is reseeded at an implementation defined rate when RNDR is used, or every time when RNDRRS is used. So the output of the TRNG itself is not accessible directly, and both RNDR and RNDRRS return output generated by a DRBG. NIST SP800-90A suggests a minimum seed size of 440 bits, so using RNDRRS to generate 64-bit seeds is reasonable, even though it comes from a DRBG. But RNDR is definitely not equivalent to RDSEED. > > >>> + * Note as well that Intel does not have an instruction that corresponds > >>> + * to the RNDRRS instruction, so there's no generic interface for that. > >>> + */ > > ... > > >>> +static inline bool arch_has_random(void) > >>> +{ > >>> + return false; > >>> +} > >>> + > >>> +static inline bool arch_get_random_long(unsigned long *v) > >>> +{ > >>> + return false; > >>> +} > >>> + > >>> +static inline bool arch_get_random_int(unsigned int *v) > >>> +{ > >>> + return false; > >>> +} > >>> + > >>> +static inline bool arch_has_random_seed(void) > >>> +{ > >>> + return cpus_have_const_cap(ARM64_HAS_RNG); > >>> +} > >>> + > >>> +static inline bool arch_get_random_seed_long(unsigned long *v) > >>> +{ > >>> + /* If RNDR fails, attempt to re-seed with RNDRRS. */ > >>> + return arch_has_random_seed() && (arm_rndr(v) || arm_rndrrs(v)); > >>> +} > >> > >> Here we clobber the value at v even if the reads of RNDR and RNDRRS > >> failed. Is that ok? > >> > >> Maybe we want the accessors to only assign to v upon success. > >> > > > > I'd be more comfortable with this if arch_get_random_*() were > > annotated as __must_check, and the assignment only done conditionally. > > We should probably make that change generically rather than make it arm64 specific. > > >>> +static bool can_use_rng(const struct arm64_cpu_capabilities *entry, int scope) > >>> +{ > >>> + unsigned long tmp; > >>> + int i; > >>> + > >>> + if (!has_cpuid_feature(entry, scope)) > >>> + return false; > >>> + > >>> + /* > >>> + * Re-seed from the true random number source. > >>> + * If this fails, disable the feature. > >>> + */ > >>> + for (i = 0; i < 10; ++i) { > >>> + if (arm_rndrrs(&tmp)) > >>> + return true; > >>> + } > >> > >> The ARM ARM (ARM DDI 0487E.a) says: > >> > >> | Reseeded Random Number. Returns a 64-bit random number which is > >> | reseeded from the True Random Number source at an IMPLEMENTATION > >> | DEFINED rate. > >> > >> ... and: > >> > >> | If the instruction cannot return a genuine random number in a > >> | reasonable period of time, PSTATE.NZCV is set to 0b0100 and the data > >> | value returned in UNKNOWN. > >> > >> ... so it's not clear to me if the retry logic makes sense. Naively I'd > >> expect "reasonable period of time" to include 10 attempts. > >> > >> Given we'll have to handle failure elsewhere, I suspect that it might be > >> best to assume that this works until we encounter evidence to the > >> contrary. > >> > > > > The architecture isn't very clear about what a reasonable period of > > time is, but the fact that it can fail transiently suggests that > > trying it only once doesn't make a lot of sense. However, I'm not sure > > whether having a go at probe time is helpful in deciding whether we > > can use it at all, and I'd be inclined to drop this test altogether. > > This is especially true since we cannot stop EL0 or VMs at EL1 using > > the instruction. > > As mentioned in reply to Mark specifically, the other two architectures do the > same thing. From the comment for x86, I read this as "if something goes wrong, > it's likely completely broken and won't work the first time we check". > > Yes, we can't prevent EL0 or VMs from using the insn, but we can avoid having > the kernel itself attempt to use it if the hardware self-check fails. > > OTOH, given that every user of arch_get_random() checks the result, and has a > fallback for failure, I suppose we don't need to do anything special. Just use > the insn and let Z=1 use the fallback failure path. > > > r~