xref: /llvm-project/lldb/test/Shell/Register/Inputs/x86-multithread-write.cpp (revision caedbc317a5a4f046c345c136387e3ca199d8503)
1*caedbc31SMichał Górny #include <cinttypes>
2*caedbc31SMichał Górny #include <cstdint>
3*caedbc31SMichał Górny #include <cstdio>
4*caedbc31SMichał Górny #include <mutex>
5*caedbc31SMichał Górny #include <thread>
6*caedbc31SMichał Górny 
7*caedbc31SMichał Górny std::mutex t1_mutex, t2_mutex;
8*caedbc31SMichał Górny 
9*caedbc31SMichał Górny struct test_data {
10*caedbc31SMichał Górny   uint32_t eax;
11*caedbc31SMichał Górny   uint32_t ebx;
12*caedbc31SMichał Górny 
13*caedbc31SMichał Górny   struct alignas(16) {
14*caedbc31SMichał Górny     uint8_t data[10];
15*caedbc31SMichał Górny   } st0;
16*caedbc31SMichał Górny };
17*caedbc31SMichał Górny 
18*caedbc31SMichał Górny constexpr test_data filler = {
19*caedbc31SMichał Górny   .eax = 0xffffffff,
20*caedbc31SMichał Górny   .ebx = 0xffffffff,
21*caedbc31SMichał Górny   .st0 = {{0x1f, 0x2f, 0x3f, 0x4f, 0x5f, 0x6f, 0x7f, 0x8f, 0x80, 0x40}},
22*caedbc31SMichał Górny };
23*caedbc31SMichał Górny 
t_func(std::mutex & t_mutex)24*caedbc31SMichał Górny void t_func(std::mutex &t_mutex) {
25*caedbc31SMichał Górny   std::lock_guard<std::mutex> t_lock(t_mutex);
26*caedbc31SMichał Górny   test_data out = filler;
27*caedbc31SMichał Górny 
28*caedbc31SMichał Górny   asm volatile(
29*caedbc31SMichał Górny     "finit\t\n"
30*caedbc31SMichał Górny     "fldt %2\t\n"
31*caedbc31SMichał Górny     "int3\n\t"
32*caedbc31SMichał Górny     "fstpt %2\t\n"
33*caedbc31SMichał Górny     : "+a"(out.eax), "+b"(out.ebx)
34*caedbc31SMichał Górny     : "m"(out.st0)
35*caedbc31SMichał Górny     : "memory", "st"
36*caedbc31SMichał Górny   );
37*caedbc31SMichał Górny 
38*caedbc31SMichał Górny   printf("eax = 0x%08" PRIx32 "\n", out.eax);
39*caedbc31SMichał Górny   printf("ebx = 0x%08" PRIx32 "\n", out.ebx);
40*caedbc31SMichał Górny   printf("st0 = { ");
41*caedbc31SMichał Górny   for (int i = 0; i < sizeof(out.st0.data); ++i)
42*caedbc31SMichał Górny     printf("0x%02" PRIx8 " ", out.st0.data[i]);
43*caedbc31SMichał Górny   printf("}\n");
44*caedbc31SMichał Górny }
45*caedbc31SMichał Górny 
main()46*caedbc31SMichał Górny int main() {
47*caedbc31SMichał Górny   // block both threads from proceeding
48*caedbc31SMichał Górny   std::unique_lock<std::mutex> m1_lock(t1_mutex);
49*caedbc31SMichał Górny   std::unique_lock<std::mutex> m2_lock(t2_mutex);
50*caedbc31SMichał Górny 
51*caedbc31SMichał Górny   // start both threads
52*caedbc31SMichał Górny   std::thread t1(t_func, std::ref(t1_mutex));
53*caedbc31SMichał Górny   std::thread t2(t_func, std::ref(t2_mutex));
54*caedbc31SMichał Górny 
55*caedbc31SMichał Górny   // release lock on thread 1 to make it interrupt the program
56*caedbc31SMichał Górny   m1_lock.unlock();
57*caedbc31SMichał Górny   // wait for thread 1 to finish
58*caedbc31SMichał Górny   t1.join();
59*caedbc31SMichał Górny 
60*caedbc31SMichał Górny   // release lock on thread 2
61*caedbc31SMichał Górny   m2_lock.unlock();
62*caedbc31SMichał Górny   // wait for thread 2 to finish
63*caedbc31SMichał Górny   t2.join();
64*caedbc31SMichał Górny 
65*caedbc31SMichał Górny   return 0;
66*caedbc31SMichał Górny }
67