| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
 100
 101
 102
 103
 104
 105
 106
 
 | /*
Copyright 2010-2011, D. E. Shaw Research.
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
* Redistributions of source code must retain the above copyright
  notice, this list of conditions, and the following disclaimer.
* 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.
* Neither the name of D. E. Shaw Research nor the names of its
  contributors may be used to endorse or promote products derived from
  this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"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 COPYRIGHT
OWNER OR CONTRIBUTORS 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.
*/
// Check our AES implementation against the example in FIPS-197
#include <Random123/aes.h>
#include <Random123/ReinterpretCtr.hpp>
#if R123_USE_AES_OPENSSL
#include <openssl/aes.h>
#endif
#include <string>
#include <cstdio>
#include <iostream>
#include <cstring>
#include <cassert>
using namespace std;
using namespace r123;
#if !R123_USE_SSE
int main(int, char **){
    std::cout << "No SSE support.  This test is not compiled\n";
    return 0;
}
#else
#include "util_m128.h"
int main(int, char **){
    r123array1xm128i IN, K;
    K.v[0].m =  m128i_from_charbuf("0001020304050607 08090a0b0c0d0e0f");
    IN.v[0].m = m128i_from_charbuf("0011223344556677 8899aabbccddeeff");
    // From FIPS-197, this is the official "right answer"
    r123array1xm128i right_answer;
    right_answer[0] = m128i_from_charbuf("69c4 e0d8 6a7b 0430 d8cd b780 70b4 c55a");
    (void)right_answer;  /* don't complain about an unused variable if neither NI nor OPENSSL are enabled. */
#if R123_USE_AES_NI
    if( haveAESNI() ){
        AESNI1xm128i::key_type xk(K);
        AESNI1xm128i bx;
        AESNI1xm128i::ctr_type x = bx(IN, xk);
        assert( x==right_answer );
        cout << "IN: " << m128i_to_string(IN[0]) <<  "\n";
        cout << "K : " << m128i_to_string(K[0])  << "\n";
        cout << "AES:" << m128i_to_string(x[0])  << "\n";
        cout << "Hooray!  AESNI1xm128i(IN, K) matches the published test vector!\n";
    }else{
        cout << "The AES-NI instructions are not available on this hardware.  Skipping AES-NI tests\n";
    }
#else
    cout << "The AES-NI Bijections are not compiled into this binary.  Skipping AES-NI tests\n";
#endif
    
    // And let's do it with AESOpenSSL.  But since AESOpenSSL has its own
    // format for keys and counters we make a union for the key types and
    // use ReinterpretCtr to wrap a union around the counter types.
#if R123_USE_AES_OPENSSL
#if R123_USE_AES_NI
    typedef AESNI1xm128i::ctr_type nictype;
#else
    typedef r123array1xm128i nictype;
#endif
    AESOpenSSL16x8::ukey_type ouk;
    _mm_storeu_si128((__m128i*)&ouk.v[0], K.v[0].m);
    AESOpenSSL16x8::key_type okey(ouk);
    ReinterpretCtr<nictype, AESOpenSSL16x8> osslb;
    assert( osslb(IN, okey) == right_answer );
    cout << "Hooray!  AESOpenSSL16x8(IN, K) matches the published test vector!\n";
#else
    cout << "The OpenSSL AES implementation is not linked with this binary.  Skipping the AESOpenSSL16x8\n";
#endif // R123_USE_AES_OPENSSL
    return 0;
}
#endif
 |