| //===- subzero/crosstest/test_sync_atomic_main.cpp - Driver for tests -----===// | 
 | // | 
 | //                        The Subzero Code Generator | 
 | // | 
 | // This file is distributed under the University of Illinois Open Source | 
 | // License. See LICENSE.TXT for details. | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 | // | 
 | // Driver for cross testing atomic intrinsics, via the sync builtins. | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | /* crosstest.py --test=test_sync_atomic.cpp --crosstest-bitcode=0 \ | 
 |    --driver=test_sync_atomic_main.cpp --prefix=Subzero_ \ | 
 |    --output=test_sync_atomic */ | 
 |  | 
 | #include <pthread.h> | 
 | #include <stdint.h> | 
 |  | 
 | #include <cerrno> | 
 | #include <climits> | 
 | #include <cstdlib> | 
 | #include <cstring> | 
 | #include <iostream> | 
 |  | 
 | // Include test_sync_atomic.h twice - once normally, and once within the | 
 | // Subzero_ namespace, corresponding to the llc and Subzero translated | 
 | // object files, respectively. | 
 | #include "test_sync_atomic.h" | 
 | #include "xdefs.h" | 
 | namespace Subzero_ { | 
 | #include "test_sync_atomic.h" | 
 | } | 
 |  | 
 | volatile uint64 Values[] = {0, | 
 |                             1, | 
 |                             0x7e, | 
 |                             0x7f, | 
 |                             0x80, | 
 |                             0x81, | 
 |                             0xfe, | 
 |                             0xff, | 
 |                             0x7ffe, | 
 |                             0x7fff, | 
 |                             0x8000, | 
 |                             0x8001, | 
 |                             0xfffe, | 
 |                             0xffff, | 
 |                             0x007fffff /*Max subnormal + */, | 
 |                             0x00800000 /*Min+ */, | 
 |                             0x7f7fffff /*Max+ */, | 
 |                             0x7f800000 /*+Inf*/, | 
 |                             0xff800000 /*-Inf*/, | 
 |                             0x7fa00000 /*SNaN*/, | 
 |                             0x7fc00000 /*QNaN*/, | 
 |                             0x7ffffffe, | 
 |                             0x7fffffff, | 
 |                             0x80000000, | 
 |                             0x80000001, | 
 |                             0xfffffffe, | 
 |                             0xffffffff, | 
 |                             0x100000000ll, | 
 |                             0x100000001ll, | 
 |                             0x000fffffffffffffll /*Max subnormal + */, | 
 |                             0x0010000000000000ll /*Min+ */, | 
 |                             0x7fefffffffffffffll /*Max+ */, | 
 |                             0x7ff0000000000000ll /*+Inf*/, | 
 |                             0xfff0000000000000ll /*-Inf*/, | 
 |                             0x7ff0000000000001ll /*SNaN*/, | 
 |                             0x7ff8000000000000ll /*QNaN*/, | 
 |                             0x7ffffffffffffffell, | 
 |                             0x7fffffffffffffffll, | 
 |                             0x8000000000000000ll, | 
 |                             0x8000000000000001ll, | 
 |                             0xfffffffffffffffell, | 
 |                             0xffffffffffffffffll}; | 
 |  | 
 | const static size_t NumValues = sizeof(Values) / sizeof(*Values); | 
 |  | 
 | struct { | 
 |   volatile uint8_t l8; | 
 |   volatile uint16_t l16; | 
 |   volatile uint32_t l32; | 
 |   volatile uint64 l64; | 
 | } AtomicLocs; | 
 |  | 
 | template <typename Type> | 
 | void testAtomicRMW(volatile Type *AtomicLoc, size_t &TotalTests, size_t &Passes, | 
 |                    size_t &Failures) { | 
 |   typedef Type (*FuncType)(bool, volatile Type *, Type); | 
 |   static struct { | 
 |     const char *Name; | 
 |     FuncType FuncLlc; | 
 |     FuncType FuncSz; | 
 |   } Funcs[] = { | 
 | #define X(inst)                                                                \ | 
 |   {STR(inst), test_##inst, Subzero_::test_##inst},                             \ | 
 |       {STR(inst) "_alloca", test_alloca_##inst, Subzero_::test_alloca_##inst}, \ | 
 |       {STR(inst) "_const", test_const_##inst, Subzero_::test_const_##inst}, | 
 |       RMWOP_TABLE | 
 | #undef X | 
 |   }; | 
 |   const static size_t NumFuncs = sizeof(Funcs) / sizeof(*Funcs); | 
 |  | 
 |   for (size_t f = 0; f < NumFuncs; ++f) { | 
 |     for (size_t i = 0; i < NumValues; ++i) { | 
 |       Type Value1 = static_cast<Type>(Values[i]); | 
 |       for (size_t j = 0; j < NumValues; ++j) { | 
 |         Type Value2 = static_cast<Type>(Values[j]); | 
 |         for (size_t k = 0; k < 2; ++k) { | 
 |           bool fetch_first = k; | 
 |           ++TotalTests; | 
 |           *AtomicLoc = Value1; | 
 |           Type ResultSz1 = Funcs[f].FuncSz(fetch_first, AtomicLoc, Value2); | 
 |           Type ResultSz2 = *AtomicLoc; | 
 |           *AtomicLoc = Value1; | 
 |           Type ResultLlc1 = Funcs[f].FuncLlc(fetch_first, AtomicLoc, Value2); | 
 |           Type ResultLlc2 = *AtomicLoc; | 
 |           if (ResultSz1 == ResultLlc1 && ResultSz2 == ResultLlc2) { | 
 |             ++Passes; | 
 |           } else { | 
 |             ++Failures; | 
 |             std::cout << "test_" << Funcs[f].Name << (CHAR_BIT * sizeof(Type)) | 
 |                       << "(" << fetch_first << ", " | 
 |                       << static_cast<uint64>(Value1) << ", " | 
 |                       << static_cast<uint64>(Value2) | 
 |                       << "): sz1=" << static_cast<uint64>(ResultSz1) | 
 |                       << " llc1=" << static_cast<uint64>(ResultLlc1) | 
 |                       << " sz2=" << static_cast<uint64>(ResultSz2) | 
 |                       << " llc2=" << static_cast<uint64>(ResultLlc2) << "\n"; | 
 |           } | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | template <typename Type> | 
 | void testValCompareAndSwap(volatile Type *AtomicLoc, size_t &TotalTests, | 
 |                            size_t &Passes, size_t &Failures) { | 
 |   typedef Type (*FuncType)(volatile Type *, Type, Type); | 
 |   static struct { | 
 |     const char *Name; | 
 |     FuncType FuncLlc; | 
 |     FuncType FuncSz; | 
 |   } Funcs[] = {{"val_cmp_swap", test_val_cmp_swap, Subzero_::test_val_cmp_swap}, | 
 |                {"val_cmp_swap_loop", test_val_cmp_swap_loop, | 
 |                 Subzero_::test_val_cmp_swap_loop}}; | 
 |   const static size_t NumFuncs = sizeof(Funcs) / sizeof(*Funcs); | 
 |   for (size_t f = 0; f < NumFuncs; ++f) { | 
 |     for (size_t i = 0; i < NumValues; ++i) { | 
 |       Type Value1 = static_cast<Type>(Values[i]); | 
 |       for (size_t j = 0; j < NumValues; ++j) { | 
 |         Type Value2 = static_cast<Type>(Values[j]); | 
 |         for (size_t f = 0; f < 2; ++f) { | 
 |           bool flip = f; | 
 |           ++TotalTests; | 
 |           *AtomicLoc = Value1; | 
 |           Type ResultSz1 = | 
 |               Funcs[f].FuncSz(AtomicLoc, flip ? Value2 : Value1, Value2); | 
 |           Type ResultSz2 = *AtomicLoc; | 
 |           *AtomicLoc = Value1; | 
 |           Type ResultLlc1 = | 
 |               Funcs[f].FuncLlc(AtomicLoc, flip ? Value2 : Value1, Value2); | 
 |           Type ResultLlc2 = *AtomicLoc; | 
 |           if (ResultSz1 == ResultLlc1 && ResultSz2 == ResultLlc2) { | 
 |             ++Passes; | 
 |           } else { | 
 |             ++Failures; | 
 |             std::cout << "test_" << Funcs[f].Name << (CHAR_BIT * sizeof(Type)) | 
 |                       << "(" << static_cast<uint64>(Value1) << ", " | 
 |                       << static_cast<uint64>(Value2) << ", flip=" << flip | 
 |                       << "): sz1=" << static_cast<uint64>(ResultSz1) | 
 |                       << " llc1=" << static_cast<uint64>(ResultLlc1) | 
 |                       << " sz2=" << static_cast<uint64>(ResultSz2) | 
 |                       << " llc2=" << static_cast<uint64>(ResultLlc2) << "\n"; | 
 |           } | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | template <typename Type> struct ThreadData { | 
 |   Type (*FuncPtr)(bool, volatile Type *, Type); | 
 |   bool Fetch; | 
 |   volatile Type *Ptr; | 
 |   Type Adjustment; | 
 | }; | 
 |  | 
 | template <typename Type> void *threadWrapper(void *Data) { | 
 | #if defined(ARM32) || defined(MIPS32) | 
 |   // Given that most of times these crosstests for ARM are run under qemu, we | 
 |   // set a lower NumReps to allow crosstests to complete within a reasonable | 
 |   // amount of time. | 
 |   static const size_t NumReps = 1000; | 
 | #else  // ARM32 || MIPS32 | 
 |   static const size_t NumReps = 8000; | 
 | #endif // ARM32 || MIPS32 | 
 |  | 
 |   ThreadData<Type> *TData = reinterpret_cast<ThreadData<Type> *>(Data); | 
 |   for (size_t i = 0; i < NumReps; ++i) { | 
 |     (void)TData->FuncPtr(TData->Fetch, TData->Ptr, TData->Adjustment); | 
 |   } | 
 |   return NULL; | 
 | } | 
 |  | 
 | template <typename Type> | 
 | void testAtomicRMWThreads(volatile Type *AtomicLoc, size_t &TotalTests, | 
 |                           size_t &Passes, size_t &Failures) { | 
 |   typedef Type (*FuncType)(bool, volatile Type *, Type); | 
 |   static struct { | 
 |     const char *Name; | 
 |     FuncType FuncLlc; | 
 |     FuncType FuncSz; | 
 |   } Funcs[] = { | 
 | #define X(inst)                                                                \ | 
 |   {STR(inst), test_##inst, Subzero_::test_##inst},                             \ | 
 |       {STR(inst) "_alloca", test_alloca_##inst, Subzero_::test_alloca_##inst}, | 
 |       RMWOP_TABLE | 
 | #undef X | 
 |   }; | 
 |   const static size_t NumFuncs = sizeof(Funcs) / sizeof(*Funcs); | 
 |  | 
 |   // Just test a few values, otherwise it takes a *really* long time. | 
 |   volatile uint64 ValuesSubset[] = {1, 0x7e, 0x000fffffffffffffffll}; | 
 |   const size_t NumValuesSubset = sizeof(ValuesSubset) / sizeof(*ValuesSubset); | 
 |  | 
 |   for (size_t f = 0; f < NumFuncs; ++f) { | 
 |     for (size_t i = 0; i < NumValuesSubset; ++i) { | 
 |       Type Value1 = static_cast<Type>(ValuesSubset[i]); | 
 |       for (size_t j = 0; j < NumValuesSubset; ++j) { | 
 |         Type Value2 = static_cast<Type>(ValuesSubset[j]); | 
 |         bool fetch_first = true; | 
 |         ThreadData<Type> TDataSz = {Funcs[f].FuncSz, fetch_first, AtomicLoc, | 
 |                                     Value2}; | 
 |         ThreadData<Type> TDataLlc = {Funcs[f].FuncLlc, fetch_first, AtomicLoc, | 
 |                                      Value2}; | 
 |         ++TotalTests; | 
 |         const size_t NumThreads = 4; | 
 |         pthread_t t[NumThreads]; | 
 |         pthread_attr_t attr[NumThreads]; | 
 |  | 
 |         // Try N threads w/ just Llc. | 
 |         *AtomicLoc = Value1; | 
 |         for (size_t m = 0; m < NumThreads; ++m) { | 
 |           pthread_attr_init(&attr[m]); | 
 |           if (pthread_create(&t[m], &attr[m], &threadWrapper<Type>, | 
 |                              reinterpret_cast<void *>(&TDataLlc)) != 0) { | 
 |             std::cout << "pthread_create failed w/ " << strerror(errno) << "\n"; | 
 |             abort(); | 
 |           } | 
 |         } | 
 |         for (size_t m = 0; m < NumThreads; ++m) { | 
 |           pthread_join(t[m], NULL); | 
 |         } | 
 |         Type ResultLlc = *AtomicLoc; | 
 |  | 
 |         // Try N threads w/ both Sz and Llc. | 
 |         *AtomicLoc = Value1; | 
 |         for (size_t m = 0; m < NumThreads; ++m) { | 
 |           pthread_attr_init(&attr[m]); | 
 |           if (pthread_create(&t[m], &attr[m], &threadWrapper<Type>, | 
 |                              m % 2 == 0 | 
 |                                  ? reinterpret_cast<void *>(&TDataLlc) | 
 |                                  : reinterpret_cast<void *>(&TDataSz)) != 0) { | 
 |             ++Failures; | 
 |             std::cout << "pthread_create failed w/ " << strerror(errno) << "\n"; | 
 |             abort(); | 
 |           } | 
 |         } | 
 |         for (size_t m = 0; m < NumThreads; ++m) { | 
 |           if (pthread_join(t[m], NULL) != 0) { | 
 |             ++Failures; | 
 |             std::cout << "pthread_join failed w/ " << strerror(errno) << "\n"; | 
 |             abort(); | 
 |           } | 
 |         } | 
 |         Type ResultMixed = *AtomicLoc; | 
 |  | 
 |         if (ResultLlc == ResultMixed) { | 
 |           ++Passes; | 
 |         } else { | 
 |           ++Failures; | 
 |           std::cout << "test_with_threads_" << Funcs[f].Name | 
 |                     << (8 * sizeof(Type)) << "(" << static_cast<uint64>(Value1) | 
 |                     << ", " << static_cast<uint64>(Value2) | 
 |                     << "): llc=" << static_cast<uint64>(ResultLlc) | 
 |                     << " mixed=" << static_cast<uint64>(ResultMixed) << "\n"; | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | int main(int argc, char *argv[]) { | 
 |   size_t TotalTests = 0; | 
 |   size_t Passes = 0; | 
 |   size_t Failures = 0; | 
 |  | 
 |   testAtomicRMW<uint8_t>(&AtomicLocs.l8, TotalTests, Passes, Failures); | 
 |   testAtomicRMW<uint16_t>(&AtomicLocs.l16, TotalTests, Passes, Failures); | 
 |   testAtomicRMW<uint32_t>(&AtomicLocs.l32, TotalTests, Passes, Failures); | 
 |   testAtomicRMW<uint64>(&AtomicLocs.l64, TotalTests, Passes, Failures); | 
 |   testValCompareAndSwap<uint8_t>(&AtomicLocs.l8, TotalTests, Passes, Failures); | 
 |   testValCompareAndSwap<uint16_t>(&AtomicLocs.l16, TotalTests, Passes, | 
 |                                   Failures); | 
 |   testValCompareAndSwap<uint32_t>(&AtomicLocs.l32, TotalTests, Passes, | 
 |                                   Failures); | 
 |   testValCompareAndSwap<uint64>(&AtomicLocs.l64, TotalTests, Passes, Failures); | 
 |   testAtomicRMWThreads<uint8_t>(&AtomicLocs.l8, TotalTests, Passes, Failures); | 
 |   testAtomicRMWThreads<uint16_t>(&AtomicLocs.l16, TotalTests, Passes, Failures); | 
 |   testAtomicRMWThreads<uint32_t>(&AtomicLocs.l32, TotalTests, Passes, Failures); | 
 |   testAtomicRMWThreads<uint64>(&AtomicLocs.l64, TotalTests, Passes, Failures); | 
 |  | 
 |   std::cout << "TotalTests=" << TotalTests << " Passes=" << Passes | 
 |             << " Failures=" << Failures << "\n"; | 
 |   return Failures; | 
 | } |