You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			94 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			94 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "test.h"
 | |
| #include <atomic>
 | |
| #include <vector>
 | |
| #include <sanitizer/tsan_interface.h>
 | |
| 
 | |
| // A very primitive mutex annotated with tsan annotations.
 | |
| class Mutex {
 | |
|  public:
 | |
|   Mutex(bool prof, unsigned create_flags, unsigned destroy_flags=0)
 | |
|       : prof_(prof)
 | |
|       , locked_(false)
 | |
|       , seq_(0)
 | |
|       , destroy_flags_(destroy_flags) {
 | |
|     __tsan_mutex_create(this, create_flags);
 | |
|   }
 | |
| 
 | |
|   ~Mutex() {
 | |
|     __tsan_mutex_destroy(this, destroy_flags_);
 | |
|   }
 | |
| 
 | |
|   void Lock() {
 | |
|     __tsan_mutex_pre_lock(this, 0);
 | |
|     LockImpl();
 | |
|     __tsan_mutex_post_lock(this, 0, 0);
 | |
|   }
 | |
| 
 | |
|   bool TryLock() {
 | |
|     __tsan_mutex_pre_lock(this, __tsan_mutex_try_lock);
 | |
|     bool ok = TryLockImpl();
 | |
|     __tsan_mutex_post_lock(this, __tsan_mutex_try_lock |
 | |
|         (ok ? 0 : __tsan_mutex_try_lock_failed), 0);
 | |
|     return ok;
 | |
|   }
 | |
| 
 | |
|   void Unlock() {
 | |
|     __tsan_mutex_pre_unlock(this, 0);
 | |
|     UnlockImpl();
 | |
|     __tsan_mutex_post_unlock(this, 0);
 | |
|   }
 | |
| 
 | |
|   void Wait() {
 | |
|     for (int seq = seq_; seq == seq_;) {
 | |
|       Unlock();
 | |
|       usleep(100);
 | |
|       Lock();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void Broadcast() {
 | |
|     __tsan_mutex_pre_signal(this, 0);
 | |
|     LockImpl(false);
 | |
|     seq_++;
 | |
|     UnlockImpl();
 | |
|     __tsan_mutex_post_signal(this, 0);
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   const bool prof_;
 | |
|   std::atomic<bool> locked_;
 | |
|   int seq_;
 | |
|   unsigned destroy_flags_;
 | |
| 
 | |
|   // This models mutex profiling subsystem.
 | |
|   static Mutex prof_mu_;
 | |
|   static int prof_data_;
 | |
| 
 | |
|   void LockImpl(bool prof = true) {
 | |
|     while (!TryLockImpl())
 | |
|       usleep(100);
 | |
|     if (prof && prof_)
 | |
|       Prof();
 | |
|   }
 | |
| 
 | |
|   bool TryLockImpl() {
 | |
|     return !locked_.exchange(true);
 | |
|   }
 | |
| 
 | |
|   void UnlockImpl() {
 | |
|     locked_.store(false);
 | |
|   }
 | |
| 
 | |
|   void Prof() {
 | |
|       // This happens inside of mutex lock annotations.
 | |
|       __tsan_mutex_pre_divert(this, 0);
 | |
|       prof_mu_.Lock();
 | |
|       prof_data_++;
 | |
|       prof_mu_.Unlock();
 | |
|       __tsan_mutex_post_divert(this, 0);
 | |
|   }
 | |
| };
 | |
| 
 | |
| Mutex Mutex::prof_mu_(false, __tsan_mutex_linker_init);
 | |
| int Mutex::prof_data_;
 |