|
|
56d343 |
2016-01-22 Torvald Riegel <triegel@redhat.com>
|
|
|
56d343 |
|
|
|
56d343 |
* beginend.cc (GTM::gtm_thread::serial_lock): Put on cacheline
|
|
|
56d343 |
boundary.
|
|
|
56d343 |
(htm_fastpath): Remove.
|
|
|
56d343 |
(gtm_thread::begin_transaction): Fix HTM fastpath.
|
|
|
56d343 |
(_ITM_commitTransaction): Adapt.
|
|
|
56d343 |
(_ITM_commitTransactionEH): Adapt.
|
|
|
56d343 |
* libitm/config/linux/rwlock.h (gtm_rwlock): Add htm_fastpath member
|
|
|
56d343 |
and accessors.
|
|
|
56d343 |
* libitm/config/posix/rwlock.h (gtm_rwlock): Likewise.
|
|
|
56d343 |
* libitm/config/posix/rwlock.cc (gtm_rwlock::gtm_rwlock): Adapt.
|
|
|
56d343 |
* libitm/libitm_i.h (htm_fastpath): Remove declaration.
|
|
|
56d343 |
* libitm/method-serial.cc (htm_mg): Adapt.
|
|
|
56d343 |
(gtm_thread::serialirr_mode): Adapt.
|
|
|
56d343 |
* libitm/query.cc (_ITM_inTransaction, _ITM_getTransactionId): Adapt.
|
|
|
56d343 |
|
|
|
56d343 |
--- libitm/beginend.cc
|
|
|
56d343 |
+++ libitm/beginend.cc
|
|
|
56d343 |
@@ -32,7 +32,11 @@ using namespace GTM;
|
|
|
56d343 |
extern __thread gtm_thread_tls _gtm_thr_tls;
|
|
|
56d343 |
#endif
|
|
|
56d343 |
|
|
|
56d343 |
-gtm_rwlock GTM::gtm_thread::serial_lock;
|
|
|
56d343 |
+// Put this at the start of a cacheline so that serial_lock's writers and
|
|
|
56d343 |
+// htm_fastpath fields are on the same cacheline, so that HW transactions
|
|
|
56d343 |
+// only have to pay one cacheline capacity to monitor both.
|
|
|
56d343 |
+gtm_rwlock GTM::gtm_thread::serial_lock
|
|
|
56d343 |
+ __attribute__((aligned(HW_CACHELINE_SIZE)));
|
|
|
56d343 |
gtm_thread *GTM::gtm_thread::list_of_threads = 0;
|
|
|
56d343 |
unsigned GTM::gtm_thread::number_of_threads = 0;
|
|
|
56d343 |
|
|
|
56d343 |
@@ -54,9 +58,6 @@ static pthread_mutex_t global_tid_lock = PTHREAD_MUTEX_INITIALIZER;
|
|
|
56d343 |
static pthread_key_t thr_release_key;
|
|
|
56d343 |
static pthread_once_t thr_release_once = PTHREAD_ONCE_INIT;
|
|
|
56d343 |
|
|
|
56d343 |
-// See gtm_thread::begin_transaction.
|
|
|
56d343 |
-uint32_t GTM::htm_fastpath = 0;
|
|
|
56d343 |
-
|
|
|
56d343 |
/* Allocate a transaction structure. */
|
|
|
56d343 |
void *
|
|
|
56d343 |
GTM::gtm_thread::operator new (size_t s)
|
|
|
56d343 |
@@ -174,9 +175,11 @@ GTM::gtm_thread::begin_transaction (uint32_t prop, const gtm_jmpbuf *jb)
|
|
|
56d343 |
// lock's writer flag and thus abort if another thread is or becomes a
|
|
|
56d343 |
// serial transaction. Therefore, if the fastpath is enabled, then a
|
|
|
56d343 |
// transaction is not executing as a HW transaction iff the serial lock is
|
|
|
56d343 |
- // write-locked. This allows us to use htm_fastpath and the serial lock's
|
|
|
56d343 |
- // writer flag to reliable determine whether the current thread runs a HW
|
|
|
56d343 |
- // transaction, and thus we do not need to maintain this information in
|
|
|
56d343 |
+ // write-locked. Also, HW transactions monitor the fastpath control
|
|
|
56d343 |
+ // variable, so that they will only execute if dispatch_htm is still the
|
|
|
56d343 |
+ // current method group. This allows us to use htm_fastpath and the serial
|
|
|
56d343 |
+ // lock's writers flag to reliable determine whether the current thread runs
|
|
|
56d343 |
+ // a HW transaction, and thus we do not need to maintain this information in
|
|
|
56d343 |
// per-thread state.
|
|
|
56d343 |
// If an uninstrumented code path is not available, we can still run
|
|
|
56d343 |
// instrumented code from a HW transaction because the HTM fastpath kicks
|
|
|
56d343 |
@@ -187,9 +190,14 @@ GTM::gtm_thread::begin_transaction (uint32_t prop, const gtm_jmpbuf *jb)
|
|
|
56d343 |
// indeed in serial mode, and HW transactions should never need serial mode
|
|
|
56d343 |
// for any internal changes (e.g., they never abort visibly to the STM code
|
|
|
56d343 |
// and thus do not trigger the standard retry handling).
|
|
|
56d343 |
- if (likely(htm_fastpath && (prop & pr_hasNoAbort)))
|
|
|
56d343 |
+ if (likely(serial_lock.get_htm_fastpath() && (prop & pr_hasNoAbort)))
|
|
|
56d343 |
{
|
|
|
56d343 |
- for (uint32_t t = htm_fastpath; t; t--)
|
|
|
56d343 |
+ // Note that the snapshot of htm_fastpath that we take here could be
|
|
|
56d343 |
+ // outdated, and a different method group than dispatch_htm may have
|
|
|
56d343 |
+ // been chosen in the meantime. Therefore, take care not not touch
|
|
|
56d343 |
+ // anything besides the serial lock, which is independent of method
|
|
|
56d343 |
+ // groups.
|
|
|
56d343 |
+ for (uint32_t t = serial_lock.get_htm_fastpath(); t; t--)
|
|
|
56d343 |
{
|
|
|
56d343 |
uint32_t ret = htm_begin();
|
|
|
56d343 |
if (htm_begin_success(ret))
|
|
|
56d343 |
@@ -197,9 +205,11 @@ GTM::gtm_thread::begin_transaction (uint32_t prop, const gtm_jmpbuf *jb)
|
|
|
56d343 |
// We are executing a transaction now.
|
|
|
56d343 |
// Monitor the writer flag in the serial-mode lock, and abort
|
|
|
56d343 |
// if there is an active or waiting serial-mode transaction.
|
|
|
56d343 |
+ // Also checks that htm_fastpath is still nonzero and thus
|
|
|
56d343 |
+ // HW transactions are allowed to run.
|
|
|
56d343 |
// Note that this can also happen due to an enclosing
|
|
|
56d343 |
// serial-mode transaction; we handle this case below.
|
|
|
56d343 |
- if (unlikely(serial_lock.is_write_locked()))
|
|
|
56d343 |
+ if (unlikely(serial_lock.htm_fastpath_disabled()))
|
|
|
56d343 |
htm_abort();
|
|
|
56d343 |
else
|
|
|
56d343 |
// We do not need to set a_saveLiveVariables because of HTM.
|
|
|
56d343 |
@@ -210,9 +220,12 @@ GTM::gtm_thread::begin_transaction (uint32_t prop, const gtm_jmpbuf *jb)
|
|
|
56d343 |
// retrying the transaction will be successful.
|
|
|
56d343 |
if (!htm_abort_should_retry(ret))
|
|
|
56d343 |
break;
|
|
|
56d343 |
+ // Check whether the HTM fastpath has been disabled.
|
|
|
56d343 |
+ if (!serial_lock.get_htm_fastpath())
|
|
|
56d343 |
+ break;
|
|
|
56d343 |
// Wait until any concurrent serial-mode transactions have finished.
|
|
|
56d343 |
// This is an empty critical section, but won't be elided.
|
|
|
56d343 |
- if (serial_lock.is_write_locked())
|
|
|
56d343 |
+ if (serial_lock.htm_fastpath_disabled())
|
|
|
56d343 |
{
|
|
|
56d343 |
tx = gtm_thr();
|
|
|
56d343 |
if (unlikely(tx == NULL))
|
|
|
56d343 |
@@ -618,7 +631,7 @@ _ITM_commitTransaction(void)
|
|
|
56d343 |
// a serial-mode transaction. If we are, then there will be no other
|
|
|
56d343 |
// concurrent serial-mode transaction.
|
|
|
56d343 |
// See gtm_thread::begin_transaction.
|
|
|
56d343 |
- if (likely(htm_fastpath && !gtm_thread::serial_lock.is_write_locked()))
|
|
|
56d343 |
+ if (likely(!gtm_thread::serial_lock.htm_fastpath_disabled()))
|
|
|
56d343 |
{
|
|
|
56d343 |
htm_commit();
|
|
|
56d343 |
return;
|
|
|
56d343 |
@@ -634,7 +647,7 @@ _ITM_commitTransactionEH(void *exc_ptr)
|
|
|
56d343 |
{
|
|
|
56d343 |
#if defined(USE_HTM_FASTPATH)
|
|
|
56d343 |
// See _ITM_commitTransaction.
|
|
|
56d343 |
- if (likely(htm_fastpath && !gtm_thread::serial_lock.is_write_locked()))
|
|
|
56d343 |
+ if (likely(!gtm_thread::serial_lock.htm_fastpath_disabled()))
|
|
|
56d343 |
{
|
|
|
56d343 |
htm_commit();
|
|
|
56d343 |
return;
|
|
|
56d343 |
--- libitm/config/linux/rwlock.h
|
|
|
56d343 |
+++ libitm/config/linux/rwlock.h
|
|
|
56d343 |
@@ -39,16 +39,29 @@ struct gtm_thread;
|
|
|
56d343 |
//
|
|
|
56d343 |
// In this implementation, writers are given highest priority access but
|
|
|
56d343 |
// read-to-write upgrades do not have a higher priority than writers.
|
|
|
56d343 |
+//
|
|
|
56d343 |
+// Do not change the layout of this class; it must remain a POD type with
|
|
|
56d343 |
+// standard layout, and the writers field must be first (i.e., so the
|
|
|
56d343 |
+// assembler code can assume that its address is equal to the address of the
|
|
|
56d343 |
+// respective instance of the class), and htm_fastpath must be second.
|
|
|
56d343 |
|
|
|
56d343 |
class gtm_rwlock
|
|
|
56d343 |
{
|
|
|
56d343 |
- // TODO Put futexes on different cachelines?
|
|
|
56d343 |
std::atomic<int> writers; // Writers' futex.
|
|
|
56d343 |
+ // We put the HTM fastpath control variable here so that HTM fastpath
|
|
|
56d343 |
+ // transactions can check efficiently whether they are allowed to run.
|
|
|
56d343 |
+ // This must be accessed atomically because threads can load this value
|
|
|
56d343 |
+ // when they are neither a registered reader nor writer (i.e., when they
|
|
|
56d343 |
+ // attempt to execute the HTM fastpath).
|
|
|
56d343 |
+ std::atomic<uint32_t> htm_fastpath;
|
|
|
56d343 |
+ // TODO Put these futexes on different cachelines? (writers and htm_fastpath
|
|
|
56d343 |
+ // should remain on the same cacheline.
|
|
|
56d343 |
std::atomic<int> writer_readers;// A confirmed writer waits here for readers.
|
|
|
56d343 |
std::atomic<int> readers; // Readers wait here for writers (iff true).
|
|
|
56d343 |
|
|
|
56d343 |
public:
|
|
|
56d343 |
- gtm_rwlock() : writers(0), writer_readers(0), readers(0) {};
|
|
|
56d343 |
+ gtm_rwlock() : writers(0), htm_fastpath(0), writer_readers(0), readers(0)
|
|
|
56d343 |
+ { }
|
|
|
56d343 |
|
|
|
56d343 |
void read_lock (gtm_thread *tx);
|
|
|
56d343 |
void read_unlock (gtm_thread *tx);
|
|
|
56d343 |
@@ -59,12 +72,28 @@ class gtm_rwlock
|
|
|
56d343 |
bool write_upgrade (gtm_thread *tx);
|
|
|
56d343 |
void write_upgrade_finish (gtm_thread *tx);
|
|
|
56d343 |
|
|
|
56d343 |
- // Returns true iff there is a concurrent active or waiting writer.
|
|
|
56d343 |
- // This is primarily useful for simple HyTM approaches, and the value being
|
|
|
56d343 |
- // checked is loaded with memory_order_relaxed.
|
|
|
56d343 |
- bool is_write_locked()
|
|
|
56d343 |
+ // Returns true iff there is a concurrent active or waiting writer, or
|
|
|
56d343 |
+ // htm_fastpath is zero. This is primarily useful for simple HyTM
|
|
|
56d343 |
+ // approaches, and the values being checked are loaded with
|
|
|
56d343 |
+ // memory_order_relaxed.
|
|
|
56d343 |
+ bool htm_fastpath_disabled ()
|
|
|
56d343 |
+ {
|
|
|
56d343 |
+ return writers.load (memory_order_relaxed) != 0
|
|
|
56d343 |
+ || htm_fastpath.load (memory_order_relaxed) == 0;
|
|
|
56d343 |
+ }
|
|
|
56d343 |
+
|
|
|
56d343 |
+ // This does not need to return an exact value, hence relaxed MO is
|
|
|
56d343 |
+ // sufficient.
|
|
|
56d343 |
+ uint32_t get_htm_fastpath ()
|
|
|
56d343 |
+ {
|
|
|
56d343 |
+ return htm_fastpath.load (memory_order_relaxed);
|
|
|
56d343 |
+ }
|
|
|
56d343 |
+ // This must only be called while having acquired the write lock, and other
|
|
|
56d343 |
+ // threads do not need to load an exact value; hence relaxed MO is
|
|
|
56d343 |
+ // sufficient.
|
|
|
56d343 |
+ void set_htm_fastpath (uint32_t val)
|
|
|
56d343 |
{
|
|
|
56d343 |
- return writers.load (memory_order_relaxed) != 0;
|
|
|
56d343 |
+ htm_fastpath.store (val, memory_order_relaxed);
|
|
|
56d343 |
}
|
|
|
56d343 |
|
|
|
56d343 |
protected:
|
|
|
56d343 |
--- libitm/config/posix/rwlock.h
|
|
|
56d343 |
+++ libitm/config/posix/rwlock.h
|
|
|
56d343 |
@@ -44,19 +44,32 @@ struct gtm_thread;
|
|
|
56d343 |
//
|
|
|
56d343 |
// In this implementation, writers are given highest priority access but
|
|
|
56d343 |
// read-to-write upgrades do not have a higher priority than writers.
|
|
|
56d343 |
+//
|
|
|
56d343 |
+// Do not change the layout of this class; it must remain a POD type with
|
|
|
56d343 |
+// standard layout, and the summary field must be first (i.e., so the
|
|
|
56d343 |
+// assembler code can assume that its address is equal to the address of the
|
|
|
56d343 |
+// respective instance of the class), and htm_fastpath must be second.
|
|
|
56d343 |
|
|
|
56d343 |
class gtm_rwlock
|
|
|
56d343 |
{
|
|
|
56d343 |
- pthread_mutex_t mutex; // Held if manipulating any field.
|
|
|
56d343 |
- pthread_cond_t c_readers; // Readers wait here
|
|
|
56d343 |
- pthread_cond_t c_writers; // Writers wait here for writers
|
|
|
56d343 |
- pthread_cond_t c_confirmed_writers; // Writers wait here for readers
|
|
|
56d343 |
-
|
|
|
56d343 |
static const unsigned a_writer = 1; // An active writer.
|
|
|
56d343 |
static const unsigned w_writer = 2; // The w_writers field != 0
|
|
|
56d343 |
static const unsigned w_reader = 4; // The w_readers field != 0
|
|
|
56d343 |
|
|
|
56d343 |
std::atomic<unsigned int> summary; // Bitmask of the above.
|
|
|
56d343 |
+
|
|
|
56d343 |
+ // We put the HTM fastpath control variable here so that HTM fastpath
|
|
|
56d343 |
+ // transactions can check efficiently whether they are allowed to run.
|
|
|
56d343 |
+ // This must be accessed atomically because threads can load this value
|
|
|
56d343 |
+ // when they are neither a registered reader nor writer (i.e., when they
|
|
|
56d343 |
+ // attempt to execute the HTM fastpath).
|
|
|
56d343 |
+ std::atomic<uint32_t> htm_fastpath;
|
|
|
56d343 |
+
|
|
|
56d343 |
+ pthread_mutex_t mutex; // Held if manipulating any field.
|
|
|
56d343 |
+ pthread_cond_t c_readers; // Readers wait here
|
|
|
56d343 |
+ pthread_cond_t c_writers; // Writers wait here for writers
|
|
|
56d343 |
+ pthread_cond_t c_confirmed_writers; // Writers wait here for readers
|
|
|
56d343 |
+
|
|
|
56d343 |
unsigned int a_readers; // Nr active readers as observed by a writer
|
|
|
56d343 |
unsigned int w_readers; // Nr waiting readers
|
|
|
56d343 |
unsigned int w_writers; // Nr waiting writers
|
|
|
56d343 |
@@ -74,12 +87,28 @@ class gtm_rwlock
|
|
|
56d343 |
bool write_upgrade (gtm_thread *tx);
|
|
|
56d343 |
void write_upgrade_finish (gtm_thread *tx);
|
|
|
56d343 |
|
|
|
56d343 |
- // Returns true iff there is a concurrent active or waiting writer.
|
|
|
56d343 |
- // This is primarily useful for simple HyTM approaches, and the value being
|
|
|
56d343 |
- // checked is loaded with memory_order_relaxed.
|
|
|
56d343 |
- bool is_write_locked()
|
|
|
56d343 |
+ // Returns true iff there is a concurrent active or waiting writer, or
|
|
|
56d343 |
+ // htm_fastpath is zero. This is primarily useful for simple HyTM
|
|
|
56d343 |
+ // approaches, and the values being checked are loaded with
|
|
|
56d343 |
+ // memory_order_relaxed.
|
|
|
56d343 |
+ bool htm_fastpath_disabled ()
|
|
|
56d343 |
+ {
|
|
|
56d343 |
+ return (summary.load (memory_order_relaxed) & (a_writer | w_writer))
|
|
|
56d343 |
+ || htm_fastpath.load (memory_order_relaxed) == 0;
|
|
|
56d343 |
+ }
|
|
|
56d343 |
+
|
|
|
56d343 |
+ // This does not need to return an exact value, hence relaxed MO is
|
|
|
56d343 |
+ // sufficient.
|
|
|
56d343 |
+ uint32_t get_htm_fastpath ()
|
|
|
56d343 |
+ {
|
|
|
56d343 |
+ return htm_fastpath.load (memory_order_relaxed);
|
|
|
56d343 |
+ }
|
|
|
56d343 |
+ // This must only be called while having acquired the write lock, and other
|
|
|
56d343 |
+ // threads do not need to load an exact value; hence relaxed MO is
|
|
|
56d343 |
+ // sufficient.
|
|
|
56d343 |
+ void set_htm_fastpath (uint32_t val)
|
|
|
56d343 |
{
|
|
|
56d343 |
- return summary.load (memory_order_relaxed) & (a_writer | w_writer);
|
|
|
56d343 |
+ htm_fastpath.store (val, memory_order_relaxed);
|
|
|
56d343 |
}
|
|
|
56d343 |
|
|
|
56d343 |
protected:
|
|
|
56d343 |
--- libitm/config/posix/rwlock.cc
|
|
|
56d343 |
+++ libitm/config/posix/rwlock.cc
|
|
|
56d343 |
@@ -30,11 +30,12 @@ namespace GTM HIDDEN {
|
|
|
56d343 |
// ??? Move this back to the header file when constexpr is implemented.
|
|
|
56d343 |
|
|
|
56d343 |
gtm_rwlock::gtm_rwlock()
|
|
|
56d343 |
- : mutex (PTHREAD_MUTEX_INITIALIZER),
|
|
|
56d343 |
+ : summary (0),
|
|
|
56d343 |
+ htm_fastpath (0),
|
|
|
56d343 |
+ mutex (PTHREAD_MUTEX_INITIALIZER),
|
|
|
56d343 |
c_readers (PTHREAD_COND_INITIALIZER),
|
|
|
56d343 |
c_writers (PTHREAD_COND_INITIALIZER),
|
|
|
56d343 |
c_confirmed_writers (PTHREAD_COND_INITIALIZER),
|
|
|
56d343 |
- summary (0),
|
|
|
56d343 |
a_readers (0),
|
|
|
56d343 |
w_readers (0),
|
|
|
56d343 |
w_writers (0)
|
|
|
56d343 |
--- libitm/libitm_i.h
|
|
|
56d343 |
+++ libitm/libitm_i.h
|
|
|
56d343 |
@@ -336,10 +336,6 @@ extern abi_dispatch *dispatch_htm();
|
|
|
56d343 |
|
|
|
56d343 |
extern gtm_cacheline_mask gtm_mask_stack(gtm_cacheline *, gtm_cacheline_mask);
|
|
|
56d343 |
|
|
|
56d343 |
-// Control variable for the HTM fastpath that uses serial mode as fallback.
|
|
|
56d343 |
-// Non-zero if the HTM fastpath is enabled. See gtm_thread::begin_transaction.
|
|
|
56d343 |
-extern uint32_t htm_fastpath;
|
|
|
56d343 |
-
|
|
|
56d343 |
} // namespace GTM
|
|
|
56d343 |
|
|
|
56d343 |
#endif // LIBITM_I_H
|
|
|
56d343 |
--- libitm/method-serial.cc
|
|
|
56d343 |
+++ libitm/method-serial.cc
|
|
|
56d343 |
@@ -222,13 +222,13 @@ struct htm_mg : public method_group
|
|
|
56d343 |
// Enable the HTM fastpath if the HW is available. The fastpath is
|
|
|
56d343 |
// initially disabled.
|
|
|
56d343 |
#ifdef USE_HTM_FASTPATH
|
|
|
56d343 |
- htm_fastpath = htm_init();
|
|
|
56d343 |
+ gtm_thread::serial_lock.set_htm_fastpath(htm_init());
|
|
|
56d343 |
#endif
|
|
|
56d343 |
}
|
|
|
56d343 |
virtual void fini()
|
|
|
56d343 |
{
|
|
|
56d343 |
// Disable the HTM fastpath.
|
|
|
56d343 |
- htm_fastpath = 0;
|
|
|
56d343 |
+ gtm_thread::serial_lock.set_htm_fastpath(0);
|
|
|
56d343 |
}
|
|
|
56d343 |
};
|
|
|
56d343 |
|
|
|
56d343 |
@@ -288,7 +288,7 @@ GTM::gtm_thread::serialirr_mode ()
|
|
|
56d343 |
#if defined(USE_HTM_FASTPATH)
|
|
|
56d343 |
// HTM fastpath. If we are executing a HW transaction, don't go serial but
|
|
|
56d343 |
// continue. See gtm_thread::begin_transaction.
|
|
|
56d343 |
- if (likely(htm_fastpath && !gtm_thread::serial_lock.is_write_locked()))
|
|
|
56d343 |
+ if (likely(!gtm_thread::serial_lock.htm_fastpath_disabled()))
|
|
|
56d343 |
return;
|
|
|
56d343 |
#endif
|
|
|
56d343 |
|
|
|
56d343 |
--- libitm/query.cc
|
|
|
56d343 |
+++ libitm/query.cc
|
|
|
56d343 |
@@ -49,7 +49,7 @@ _ITM_inTransaction (void)
|
|
|
56d343 |
// a transaction and thus we can't deduce this by looking at just the serial
|
|
|
56d343 |
// lock. This function isn't used in practice currently, so the easiest
|
|
|
56d343 |
// way to handle it is to just abort.
|
|
|
56d343 |
- if (htm_fastpath && htm_transaction_active())
|
|
|
56d343 |
+ if (gtm_thread::serial_lock.get_htm_fastpath() && htm_transaction_active())
|
|
|
56d343 |
htm_abort();
|
|
|
56d343 |
#endif
|
|
|
56d343 |
struct gtm_thread *tx = gtm_thr();
|
|
|
56d343 |
@@ -69,7 +69,7 @@ _ITM_getTransactionId (void)
|
|
|
56d343 |
{
|
|
|
56d343 |
#if defined(USE_HTM_FASTPATH)
|
|
|
56d343 |
// See ITM_inTransaction.
|
|
|
56d343 |
- if (htm_fastpath && htm_transaction_active())
|
|
|
56d343 |
+ if (gtm_thread::serial_lock.get_htm_fastpath() && htm_transaction_active())
|
|
|
56d343 |
htm_abort();
|
|
|
56d343 |
#endif
|
|
|
56d343 |
struct gtm_thread *tx = gtm_thr();
|