/*############################################################################ # Copyright 2016-2017 Intel Corporation # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. ############################################################################*/ /*! * \file * \brief Pseudo random number generator implementation. */ #include <ippcp.h> #include <stdlib.h> #include <time.h> #include "src/prng.h" EpidStatus PrngCreate(void** prng) { // Security note: // Random number generator used in the samples not claimed to be a // cryptographically secure pseudo-random number generator. EpidStatus sts = kEpidErr; int prng_ctx_size = 0; IppsPRNGState* prng_ctx = NULL; int seed_ctx_size = 0; IppsBigNumState* seed_ctx = NULL; time_t seed_value; if (!prng) return kEpidBadArgErr; if (ippStsNoErr != ippsPRNGGetSize(&prng_ctx_size)) return kEpidErr; if (ippStsNoErr != ippsBigNumGetSize((sizeof(seed_value) + 3) / 4, &seed_ctx_size)) return kEpidErr; do { prng_ctx = (IppsPRNGState*)calloc(1, prng_ctx_size); if (!prng_ctx) { sts = kEpidNoMemErr; break; } if (ippStsNoErr != ippsPRNGInit(sizeof(seed_value) * 8, prng_ctx)) { sts = kEpidErr; break; } // seed PRNG seed_ctx = (IppsBigNumState*)calloc(1, seed_ctx_size); if (!seed_ctx) { sts = kEpidNoMemErr; break; } if (ippStsNoErr != ippsBigNumInit((sizeof(seed_value) + 3) / 4, seed_ctx)) { sts = kEpidErr; break; } time(&seed_value); if (ippStsNoErr != ippsSetOctString_BN((void*)&seed_value, sizeof(seed_value), seed_ctx)) { sts = kEpidErr; break; } if (ippStsNoErr != ippsPRNGSetSeed(seed_ctx, prng_ctx)) { sts = kEpidErr; break; } *prng = prng_ctx; prng_ctx = NULL; sts = kEpidNoErr; } while (0); if (seed_ctx) free(seed_ctx); if (prng_ctx) free(prng_ctx); return sts; } void PrngDelete(void** prng) { if (prng && *prng) { free(*prng); *prng = NULL; } } // simple wrapper to hide IPP implementation. int __STDCALL PrngGen(unsigned int* rand_data, int num_bits, void* user_data) { return ippsPRNGen(rand_data, num_bits, (IppsPRNGState*)user_data); }