xref: /llvm-project/lldb/test/Shell/Register/Inputs/x86-multithread-read.cpp (revision caedbc317a5a4f046c345c136387e3ca199d8503)
1*caedbc31SMichał Górny #include <cstdint>
2*caedbc31SMichał Górny #include <mutex>
3*caedbc31SMichał Górny #include <thread>
4*caedbc31SMichał Górny 
5*caedbc31SMichał Górny std::mutex t1_mutex, t2_mutex;
6*caedbc31SMichał Górny 
7*caedbc31SMichał Górny struct test_data {
8*caedbc31SMichał Górny   uint32_t eax;
9*caedbc31SMichał Górny   uint32_t ebx;
10*caedbc31SMichał Górny 
11*caedbc31SMichał Górny   struct alignas(16) {
12*caedbc31SMichał Górny     uint64_t mantissa;
13*caedbc31SMichał Górny     uint16_t sign_exp;
14*caedbc31SMichał Górny   } st0;
15*caedbc31SMichał Górny };
16*caedbc31SMichał Górny 
t_func(std::mutex & t_mutex,const test_data & t_data)17*caedbc31SMichał Górny void t_func(std::mutex &t_mutex, const test_data &t_data) {
18*caedbc31SMichał Górny   std::lock_guard<std::mutex> t_lock(t_mutex);
19*caedbc31SMichał Górny 
20*caedbc31SMichał Górny   asm volatile(
21*caedbc31SMichał Górny     "finit\t\n"
22*caedbc31SMichał Górny     "fldt %2\t\n"
23*caedbc31SMichał Górny     "int3\n\t"
24*caedbc31SMichał Górny     :
25*caedbc31SMichał Górny     : "a"(t_data.eax), "b"(t_data.ebx), "m"(t_data.st0)
26*caedbc31SMichał Górny     : "st"
27*caedbc31SMichał Górny   );
28*caedbc31SMichał Górny }
29*caedbc31SMichał Górny 
main()30*caedbc31SMichał Górny int main() {
31*caedbc31SMichał Górny   test_data t1_data = {
32*caedbc31SMichał Górny     .eax = 0x05060708,
33*caedbc31SMichał Górny     .ebx = 0x15161718,
34*caedbc31SMichał Górny     .st0 = {0x8070605040302010, 0x4000},
35*caedbc31SMichał Górny   };
36*caedbc31SMichał Górny   test_data t2_data = {
37*caedbc31SMichał Górny     .eax = 0x25262728,
38*caedbc31SMichał Górny     .ebx = 0x35363738,
39*caedbc31SMichał Górny     .st0 = {0x8171615141312111, 0xc000},
40*caedbc31SMichał Górny   };
41*caedbc31SMichał Górny 
42*caedbc31SMichał Górny   // block both threads from proceeding
43*caedbc31SMichał Górny   std::unique_lock<std::mutex> m1_lock(t1_mutex);
44*caedbc31SMichał Górny   std::unique_lock<std::mutex> m2_lock(t2_mutex);
45*caedbc31SMichał Górny 
46*caedbc31SMichał Górny   // start both threads
47*caedbc31SMichał Górny   std::thread t1(t_func, std::ref(t1_mutex), std::ref(t1_data));
48*caedbc31SMichał Górny   std::thread t2(t_func, std::ref(t2_mutex), std::ref(t2_data));
49*caedbc31SMichał Górny 
50*caedbc31SMichał Górny   // release lock on thread 1 to make it interrupt the program
51*caedbc31SMichał Górny   m1_lock.unlock();
52*caedbc31SMichał Górny   // wait for thread 1 to finish
53*caedbc31SMichał Górny   t1.join();
54*caedbc31SMichał Górny 
55*caedbc31SMichał Górny   // release lock on thread 2
56*caedbc31SMichał Górny   m2_lock.unlock();
57*caedbc31SMichał Górny   // wait for thread 2 to finish
58*caedbc31SMichał Górny   t2.join();
59*caedbc31SMichał Górny 
60*caedbc31SMichał Górny   return 0;
61*caedbc31SMichał Górny }
62