SPU: fix possible livelock

The bug affects TSX path
This commit is contained in:
Nekotekina 2018-04-07 02:16:32 +03:00
parent 3681507136
commit 0797164fac
4 changed files with 33 additions and 19 deletions

View File

@ -39,11 +39,11 @@ namespace utils
bool has_xop(); bool has_xop();
inline bool transaction_enter() FORCE_INLINE bool transaction_enter(uint* out = nullptr)
{ {
while (true) while (true)
{ {
const auto status = _xbegin(); const uint status = _xbegin();
if (status == _XBEGIN_STARTED) if (status == _XBEGIN_STARTED)
{ {
@ -52,6 +52,11 @@ namespace utils
if (!(status & _XABORT_RETRY)) if (!(status & _XABORT_RETRY))
{ {
if (out)
{
*out = status;
}
return false; return false;
} }
} }

View File

@ -964,7 +964,7 @@ extern bool ppu_stwcx(ppu_thread& ppu, u32 addr, u32 reg_value)
if (s_use_rtm && utils::transaction_enter()) if (s_use_rtm && utils::transaction_enter())
{ {
if (!vm::g_mutex.is_lockable()) if (!vm::g_mutex.is_lockable() || vm::g_mutex.is_reading())
{ {
_xabort(0); _xabort(0);
} }
@ -1008,7 +1008,7 @@ extern bool ppu_stdcx(ppu_thread& ppu, u32 addr, u64 reg_value)
if (s_use_rtm && utils::transaction_enter()) if (s_use_rtm && utils::transaction_enter())
{ {
if (!vm::g_mutex.is_lockable()) if (!vm::g_mutex.is_lockable() || vm::g_mutex.is_reading())
{ {
_xabort(0); _xabort(0);
} }

View File

@ -313,7 +313,7 @@ std::string SPUThread::dump() const
std::string ret = cpu_thread::dump(); std::string ret = cpu_thread::dump();
// Print some transaction statistics // Print some transaction statistics
fmt::append(ret, "\nTX: %u; Fail: %u", tx_success, tx_failure); fmt::append(ret, "\nTX: %u; Fail: %u (0x%x)", tx_success, tx_failure, tx_status);
fmt::append(ret, "\nRaddr: 0x%08x; R: 0x%x", raddr, raddr ? +vm::reservation_acquire(raddr, 128) : 0); fmt::append(ret, "\nRaddr: 0x%08x; R: 0x%x", raddr, raddr ? +vm::reservation_acquire(raddr, 128) : 0);
fmt::append(ret, "\nTag Mask: 0x%08x", ch_tag_mask); fmt::append(ret, "\nTag Mask: 0x%08x", ch_tag_mask);
fmt::append(ret, "\nMFC Stall: 0x%08x", ch_stall_mask); fmt::append(ret, "\nMFC Stall: 0x%08x", ch_stall_mask);
@ -855,7 +855,7 @@ void SPUThread::do_putlluc(const spu_mfc_cmd& args)
if (s_use_rtm && utils::transaction_enter()) if (s_use_rtm && utils::transaction_enter())
{ {
// First transaction attempt // First transaction attempt
if (!vm::g_mutex.is_lockable()) if (!vm::g_mutex.is_lockable() || vm::g_mutex.is_reading())
{ {
_xabort(0); _xabort(0);
} }
@ -864,12 +864,11 @@ void SPUThread::do_putlluc(const spu_mfc_cmd& args)
vm::reservation_update(addr, 128); vm::reservation_update(addr, 128);
vm::notify(addr, 128); vm::notify(addr, 128);
_xend(); _xend();
tx_success++;
return; return;
} }
else if (s_use_rtm) else if (s_use_rtm)
{ {
vm::reader_lock lock; vm::writer_lock lock(0);
if (utils::transaction_enter()) if (utils::transaction_enter())
{ {
@ -877,15 +876,18 @@ void SPUThread::do_putlluc(const spu_mfc_cmd& args)
data = to_write; data = to_write;
vm::reservation_update(addr, 128); vm::reservation_update(addr, 128);
_xend(); _xend();
tx_success++;
vm::notify(addr, 128);
return;
} }
else else
{ {
tx_failure++; vm::reservation_update(addr, 128, true);
_mm_sfence();
data = to_write;
_mm_sfence();
vm::reservation_update(addr, 128);
} }
vm::notify(addr, 128);
return;
} }
vm::writer_lock lock(0); vm::writer_lock lock(0);
@ -1102,7 +1104,6 @@ bool SPUThread::process_mfc_cmd(spu_mfc_cmd args)
if (LIKELY(vm::reservation_acquire(raddr, 128) == rtime)) if (LIKELY(vm::reservation_acquire(raddr, 128) == rtime))
{ {
// Copy to LS // Copy to LS
tx_success++;
_ref<decltype(rdata)>(args.lsa & 0x3ffff) = rdata; _ref<decltype(rdata)>(args.lsa & 0x3ffff) = rdata;
ch_atomic_stat.set_value(MFC_GETLLAR_SUCCESS); ch_atomic_stat.set_value(MFC_GETLLAR_SUCCESS);
return true; return true;
@ -1123,11 +1124,9 @@ bool SPUThread::process_mfc_cmd(spu_mfc_cmd args)
rdata = data; rdata = data;
_xend(); _xend();
tx_success++;
} }
else else
{ {
tx_failure++;
vm::reader_lock lock; vm::reader_lock lock;
rtime = vm::reservation_acquire(raddr, 128); rtime = vm::reservation_acquire(raddr, 128);
rdata = data; rdata = data;
@ -1153,7 +1152,7 @@ bool SPUThread::process_mfc_cmd(spu_mfc_cmd args)
if (s_use_rtm && utils::transaction_enter()) if (s_use_rtm && utils::transaction_enter())
{ {
// First transaction attempt // First transaction attempt
if (!vm::g_mutex.is_lockable()) if (!vm::g_mutex.is_lockable() || vm::g_mutex.is_reading())
{ {
_xabort(0); _xabort(0);
} }
@ -1173,12 +1172,15 @@ bool SPUThread::process_mfc_cmd(spu_mfc_cmd args)
else if (s_use_rtm) else if (s_use_rtm)
{ {
// Second transaction attempt // Second transaction attempt
vm::reader_lock lock; vm::writer_lock lock(0);
// Touch memory without modifying the value
vm::_ref<atomic_t<u32>>(args.eal) += 0;
// Touch reservation memory area as well // Touch reservation memory area as well
vm::reservation_acquire(raddr, 128) += 0; vm::reservation_acquire(raddr, 128) += 0;
if (utils::transaction_enter()) if (utils::transaction_enter(&tx_status))
{ {
if (rtime == vm::reservation_acquire(raddr, 128) && rdata == data) if (rtime == vm::reservation_acquire(raddr, 128) && rdata == data)
{ {
@ -1199,6 +1201,12 @@ bool SPUThread::process_mfc_cmd(spu_mfc_cmd args)
} }
else else
{ {
// Workaround MSVC
if (rtime == vm::reservation_acquire(raddr, 128) && rdata == data)
{
vm::reservation_update(raddr, 128);
}
// Don't fallback to heavyweight lock, just give up // Don't fallback to heavyweight lock, just give up
tx_failure++; tx_failure++;
} }

View File

@ -593,6 +593,7 @@ public:
u64 tx_success = 0; u64 tx_success = 0;
u64 tx_failure = 0; u64 tx_failure = 0;
uint tx_status = 0;
void push_snr(u32 number, u32 value); void push_snr(u32 number, u32 value);
void do_dma_transfer(const spu_mfc_cmd& args); void do_dma_transfer(const spu_mfc_cmd& args);