mirror of
https://github.com/kvazar-network/kevacoin.git
synced 2025-01-11 15:48:05 +00:00
Fixed potential deadlocks in GUI code.
Also changed semantics of CWalletTx::GetTxTime(); now always returns the time the transaction was received by this node, not the average block time. And added information about -DDEBUG_LOCKORDER to coding.txt.
This commit is contained in:
parent
6cc4a62c0e
commit
471426fb3b
@ -39,3 +39,47 @@ v vector or similar list objects
|
|||||||
map map or multimap
|
map map or multimap
|
||||||
set set or multiset
|
set set or multiset
|
||||||
bn CBigNum
|
bn CBigNum
|
||||||
|
|
||||||
|
-------------------------
|
||||||
|
Locking/mutex usage notes
|
||||||
|
|
||||||
|
The code is multi-threaded, and uses mutexes and the CRITICAL_BLOCK/TRY_CRITICAL_BLOCK macros to protect data structures.
|
||||||
|
|
||||||
|
Deadlocks due to inconsistent lock ordering (thread 1 locks cs_main and then cs_wallet, while thread 2 locks them in the opposite order: result, deadlock as each waits for the other to release its lock) are a problem. Compile with -DDEBUG_LOCKORDER to get lock order inconsistencies reported in the debug.log file.
|
||||||
|
|
||||||
|
Re-architecting the core code so there are better-defined interfaces between the various components is a goal, with any necessary locking done by the components (e.g. see the self-contained CKeyStore class and its cs_KeyStore lock for example).
|
||||||
|
|
||||||
|
-------
|
||||||
|
Threads
|
||||||
|
|
||||||
|
StartNode : Starts other threads.
|
||||||
|
|
||||||
|
ThreadGetMyExternalIP : Determines outside-the-firewall IP address, sends addr message to connected peers when it determines it.
|
||||||
|
|
||||||
|
ThreadIRCSeed : Joins IRC bootstrapping channel, watching for new peers and advertising this node's IP address.
|
||||||
|
|
||||||
|
ThreadSocketHandler : Sends/Receives data from peers on port 8333.
|
||||||
|
|
||||||
|
ThreadMessageHandler : Higher-level message handling (sending and receiving).
|
||||||
|
|
||||||
|
ThreadOpenConnections : Initiates new connections to peers.
|
||||||
|
|
||||||
|
ThreadTopUpKeyPool : replenishes the keystore's keypool.
|
||||||
|
|
||||||
|
ThreadCleanWalletPassphrase : re-locks an encrypted wallet after user has unlocked it for a period of time.
|
||||||
|
|
||||||
|
SendingDialogStartTransfer : used by pay-via-ip-address code (obsolete)
|
||||||
|
|
||||||
|
ThreadDelayedRepaint : repaint the gui
|
||||||
|
|
||||||
|
ThreadFlushWalletDB : Close the wallet.dat file if it hasn't been used in 500ms.
|
||||||
|
|
||||||
|
ThreadRPCServer : Remote procedure call handler, listens on port 8332 for connections and services them.
|
||||||
|
|
||||||
|
ThreadBitcoinMiner : Generates bitcoins
|
||||||
|
|
||||||
|
ThreadMapPort : Universal plug-and-play startup/shutdown
|
||||||
|
|
||||||
|
Shutdown : Does an orderly shutdown of everything
|
||||||
|
|
||||||
|
ExitTimeout : Windows-only, sleeps 5 seconds then exits application
|
||||||
|
366
src/ui.cpp
366
src/ui.cpp
@ -708,7 +708,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex)
|
|||||||
CBitcoinAddress address;
|
CBitcoinAddress address;
|
||||||
if (ExtractAddress(txout.scriptPubKey, pwalletMain, address))
|
if (ExtractAddress(txout.scriptPubKey, pwalletMain, address))
|
||||||
{
|
{
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
{
|
{
|
||||||
//strDescription += _("Received payment to ");
|
//strDescription += _("Received payment to ");
|
||||||
//strDescription += _("Received with address ");
|
//strDescription += _("Received with address ");
|
||||||
@ -792,7 +792,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex)
|
|||||||
}
|
}
|
||||||
|
|
||||||
string strDescription = _("To: ");
|
string strDescription = _("To: ");
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
if (pwalletMain->mapAddressBook.count(address) && !pwalletMain->mapAddressBook[address].empty())
|
if (pwalletMain->mapAddressBook.count(address) && !pwalletMain->mapAddressBook[address].empty())
|
||||||
strDescription += pwalletMain->mapAddressBook[address] + " ";
|
strDescription += pwalletMain->mapAddressBook[address] + " ";
|
||||||
strDescription += strAddress;
|
strDescription += strAddress;
|
||||||
@ -1032,6 +1032,7 @@ void MainFrameRepaint()
|
|||||||
printf("MainFrameRepaint\n");
|
printf("MainFrameRepaint\n");
|
||||||
wxPaintEvent event;
|
wxPaintEvent event;
|
||||||
pframeMain->fRefresh = true;
|
pframeMain->fRefresh = true;
|
||||||
|
pframeMain->fRefreshListCtrl = true;
|
||||||
pframeMain->GetEventHandler()->AddPendingEvent(event);
|
pframeMain->GetEventHandler()->AddPendingEvent(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1247,83 +1248,80 @@ void CMainFrame::OnMenuOptionsChangeWalletPassphrase(wxCommandEvent& event)
|
|||||||
strOldWalletPass = wxGetPasswordFromUser(_("Enter the current passphrase to the wallet."),
|
strOldWalletPass = wxGetPasswordFromUser(_("Enter the current passphrase to the wallet."),
|
||||||
_("Passphrase")).ToStdString();
|
_("Passphrase")).ToStdString();
|
||||||
|
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_vMasterKey)
|
bool fWasLocked = pwalletMain->IsLocked();
|
||||||
|
pwalletMain->Lock();
|
||||||
|
|
||||||
|
if (!strOldWalletPass.size() || !pwalletMain->Unlock(strOldWalletPass))
|
||||||
{
|
{
|
||||||
bool fWasLocked = pwalletMain->IsLocked();
|
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
||||||
|
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
||||||
|
wxMessageBox(_("The passphrase entered for the wallet decryption was incorrect."), "Bitcoin", wxOK | wxICON_ERROR);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fWasLocked)
|
||||||
pwalletMain->Lock();
|
pwalletMain->Lock();
|
||||||
|
|
||||||
if (!strOldWalletPass.size() || !pwalletMain->Unlock(strOldWalletPass))
|
string strNewWalletPass;
|
||||||
{
|
strNewWalletPass.reserve(100);
|
||||||
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
mlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
||||||
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
|
||||||
wxMessageBox(_("The passphrase entered for the wallet decryption was incorrect."), "Bitcoin", wxOK | wxICON_ERROR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fWasLocked)
|
// obtain new wallet encrypt/decrypt key, from passphrase
|
||||||
pwalletMain->Lock();
|
// Note that the passphrase is not mlock()d during this entry and could potentially
|
||||||
|
// be obtained from disk long after bitcoin has run.
|
||||||
|
strNewWalletPass = wxGetPasswordFromUser(_("Enter the new passphrase for the wallet."),
|
||||||
|
_("Passphrase")).ToStdString();
|
||||||
|
|
||||||
string strNewWalletPass;
|
if (!strNewWalletPass.size())
|
||||||
strNewWalletPass.reserve(100);
|
{
|
||||||
mlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
||||||
|
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
||||||
|
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
||||||
|
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
||||||
|
wxMessageBox(_("Error: The supplied passphrase was too short."), "Bitcoin", wxOK | wxICON_ERROR);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// obtain new wallet encrypt/decrypt key, from passphrase
|
string strNewWalletPassTest;
|
||||||
// Note that the passphrase is not mlock()d during this entry and could potentially
|
strNewWalletPassTest.reserve(100);
|
||||||
// be obtained from disk long after bitcoin has run.
|
mlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
||||||
strNewWalletPass = wxGetPasswordFromUser(_("Enter the new passphrase for the wallet."),
|
|
||||||
|
// obtain new wallet encrypt/decrypt key, from passphrase
|
||||||
|
// Note that the passphrase is not mlock()d during this entry and could potentially
|
||||||
|
// be obtained from disk long after bitcoin has run.
|
||||||
|
strNewWalletPassTest = wxGetPasswordFromUser(_("Re-enter the new passphrase for the wallet."),
|
||||||
_("Passphrase")).ToStdString();
|
_("Passphrase")).ToStdString();
|
||||||
|
|
||||||
if (!strNewWalletPass.size())
|
if (strNewWalletPassTest != strNewWalletPass)
|
||||||
{
|
{
|
||||||
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
|
||||||
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
|
||||||
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
|
||||||
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
|
||||||
wxMessageBox(_("Error: The supplied passphrase was too short."), "Bitcoin", wxOK | wxICON_ERROR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
string strNewWalletPassTest;
|
|
||||||
strNewWalletPassTest.reserve(100);
|
|
||||||
mlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
|
||||||
|
|
||||||
// obtain new wallet encrypt/decrypt key, from passphrase
|
|
||||||
// Note that the passphrase is not mlock()d during this entry and could potentially
|
|
||||||
// be obtained from disk long after bitcoin has run.
|
|
||||||
strNewWalletPassTest = wxGetPasswordFromUser(_("Re-enter the new passphrase for the wallet."),
|
|
||||||
_("Passphrase")).ToStdString();
|
|
||||||
|
|
||||||
if (strNewWalletPassTest != strNewWalletPass)
|
|
||||||
{
|
|
||||||
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
|
||||||
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
|
||||||
fill(strNewWalletPassTest.begin(), strNewWalletPassTest.end(), '\0');
|
|
||||||
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
|
||||||
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
|
||||||
munlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
|
||||||
wxMessageBox(_("Error: the supplied passphrases didn't match."), "Bitcoin", wxOK | wxICON_ERROR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pwalletMain->ChangeWalletPassphrase(strOldWalletPass, strNewWalletPass))
|
|
||||||
{
|
|
||||||
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
|
||||||
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
|
||||||
fill(strNewWalletPassTest.begin(), strNewWalletPassTest.end(), '\0');
|
|
||||||
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
|
||||||
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
|
||||||
munlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
|
||||||
wxMessageBox(_("The passphrase entered for the wallet decryption was incorrect."), "Bitcoin", wxOK | wxICON_ERROR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
||||||
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
||||||
fill(strNewWalletPassTest.begin(), strNewWalletPassTest.end(), '\0');
|
fill(strNewWalletPassTest.begin(), strNewWalletPassTest.end(), '\0');
|
||||||
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
||||||
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
||||||
munlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
munlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
||||||
wxMessageBox(_("Wallet Passphrase Changed."), "Bitcoin");
|
wxMessageBox(_("Error: the supplied passphrases didn't match."), "Bitcoin", wxOK | wxICON_ERROR);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!pwalletMain->ChangeWalletPassphrase(strOldWalletPass, strNewWalletPass))
|
||||||
|
{
|
||||||
|
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
||||||
|
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
||||||
|
fill(strNewWalletPassTest.begin(), strNewWalletPassTest.end(), '\0');
|
||||||
|
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
||||||
|
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
||||||
|
munlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
||||||
|
wxMessageBox(_("The passphrase entered for the wallet decryption was incorrect."), "Bitcoin", wxOK | wxICON_ERROR);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
fill(strOldWalletPass.begin(), strOldWalletPass.end(), '\0');
|
||||||
|
fill(strNewWalletPass.begin(), strNewWalletPass.end(), '\0');
|
||||||
|
fill(strNewWalletPassTest.begin(), strNewWalletPassTest.end(), '\0');
|
||||||
|
munlock(&strOldWalletPass[0], strOldWalletPass.capacity());
|
||||||
|
munlock(&strNewWalletPass[0], strNewWalletPass.capacity());
|
||||||
|
munlock(&strNewWalletPassTest[0], strNewWalletPassTest.capacity());
|
||||||
|
wxMessageBox(_("Wallet Passphrase Changed."), "Bitcoin");
|
||||||
}
|
}
|
||||||
|
|
||||||
void CMainFrame::OnMenuOptionsOptions(wxCommandEvent& event)
|
void CMainFrame::OnMenuOptionsOptions(wxCommandEvent& event)
|
||||||
@ -1387,21 +1385,19 @@ void CMainFrame::OnButtonNew(wxCommandEvent& event)
|
|||||||
string strName = dialog.GetValue();
|
string strName = dialog.GetValue();
|
||||||
|
|
||||||
string strAddress;
|
string strAddress;
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_vMasterKey)
|
|
||||||
{
|
|
||||||
bool fWasLocked = pwalletMain->IsLocked();
|
|
||||||
if (!GetWalletPassphrase())
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Generate new key
|
bool fWasLocked = pwalletMain->IsLocked();
|
||||||
strAddress = CBitcoinAddress(pwalletMain->GetOrReuseKeyFromPool()).ToString();
|
if (!GetWalletPassphrase())
|
||||||
|
return;
|
||||||
|
|
||||||
if (fWasLocked)
|
// Generate new key
|
||||||
pwalletMain->Lock();
|
strAddress = CBitcoinAddress(pwalletMain->GetOrReuseKeyFromPool()).ToString();
|
||||||
}
|
|
||||||
|
if (fWasLocked)
|
||||||
|
pwalletMain->Lock();
|
||||||
|
|
||||||
// Save
|
// Save
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
pwalletMain->SetAddressBookName(strAddress, strName);
|
pwalletMain->SetAddressBookName(strAddress, strName);
|
||||||
SetDefaultReceivingAddress(strAddress);
|
SetDefaultReceivingAddress(strAddress);
|
||||||
}
|
}
|
||||||
@ -1451,7 +1447,7 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails
|
|||||||
#ifdef __WXMSW__
|
#ifdef __WXMSW__
|
||||||
SetSize(nScaleX * GetSize().GetWidth(), nScaleY * GetSize().GetHeight());
|
SetSize(nScaleX * GetSize().GetWidth(), nScaleY * GetSize().GetHeight());
|
||||||
#endif
|
#endif
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
{
|
{
|
||||||
string strHTML;
|
string strHTML;
|
||||||
strHTML.reserve(4000);
|
strHTML.reserve(4000);
|
||||||
@ -2160,38 +2156,39 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event)
|
|||||||
|
|
||||||
if (fBitcoinAddress)
|
if (fBitcoinAddress)
|
||||||
{
|
{
|
||||||
CRITICAL_BLOCK(cs_main)
|
bool fWasLocked = pwalletMain->IsLocked();
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_vMasterKey)
|
if (!GetWalletPassphrase())
|
||||||
{
|
return;
|
||||||
bool fWasLocked = pwalletMain->IsLocked();
|
|
||||||
if (!GetWalletPassphrase())
|
|
||||||
return;
|
|
||||||
|
|
||||||
|
string strError;
|
||||||
|
CRITICAL_BLOCK(cs_main)
|
||||||
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
|
{
|
||||||
// Send to bitcoin address
|
// Send to bitcoin address
|
||||||
CScript scriptPubKey;
|
CScript scriptPubKey;
|
||||||
scriptPubKey.SetBitcoinAddress(address);
|
scriptPubKey.SetBitcoinAddress(address);
|
||||||
|
|
||||||
string strError = pwalletMain->SendMoney(scriptPubKey, nValue, wtx, true);
|
strError = pwalletMain->SendMoney(scriptPubKey, nValue, wtx, true);
|
||||||
if (strError == "")
|
}
|
||||||
wxMessageBox(_("Payment sent "), _("Sending..."));
|
if (strError == "")
|
||||||
else if (strError == "ABORTED")
|
wxMessageBox(_("Payment sent "), _("Sending..."));
|
||||||
{
|
else if (strError == "ABORTED")
|
||||||
if (fWasLocked)
|
{
|
||||||
pwalletMain->Lock();
|
|
||||||
return; // leave send dialog open
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
wxMessageBox(strError + " ", _("Sending..."));
|
|
||||||
EndModal(false);
|
|
||||||
if (fWasLocked)
|
|
||||||
pwalletMain->Lock();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fWasLocked)
|
if (fWasLocked)
|
||||||
pwalletMain->Lock();
|
pwalletMain->Lock();
|
||||||
}
|
return; // leave send dialog open
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
wxMessageBox(strError + " ", _("Sending..."));
|
||||||
|
EndModal(false);
|
||||||
|
if (fWasLocked)
|
||||||
|
pwalletMain->Lock();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fWasLocked)
|
||||||
|
pwalletMain->Lock();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -2212,7 +2209,7 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
if (!pwalletMain->mapAddressBook.count(address))
|
if (!pwalletMain->mapAddressBook.count(address))
|
||||||
pwalletMain->SetAddressBookName(strAddress, "");
|
pwalletMain->SetAddressBookName(strAddress, "");
|
||||||
|
|
||||||
@ -2464,83 +2461,89 @@ void CSendingDialog::OnReply2(CDataStream& vRecv)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
CRITICAL_BLOCK(cs_main)
|
// Pay
|
||||||
|
if (!Status(_("Creating transaction...")))
|
||||||
|
return;
|
||||||
|
if (nPrice + nTransactionFee > pwalletMain->GetBalance())
|
||||||
{
|
{
|
||||||
// Pay
|
Error(_("Insufficient funds"));
|
||||||
if (!Status(_("Creating transaction...")))
|
return;
|
||||||
return;
|
}
|
||||||
if (nPrice + nTransactionFee > pwalletMain->GetBalance())
|
|
||||||
{
|
|
||||||
Error(_("Insufficient funds"));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
CReserveKey reservekey(pwalletMain);
|
CReserveKey reservekey(pwalletMain);
|
||||||
int64 nFeeRequired;
|
int64 nFeeRequired;
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_vMasterKey)
|
bool fWasLocked = pwalletMain->IsLocked();
|
||||||
{
|
if (!GetWalletPassphrase())
|
||||||
bool fWasLocked = pwalletMain->IsLocked();
|
return;
|
||||||
if (!GetWalletPassphrase())
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (!pwalletMain->CreateTransaction(scriptPubKey, nPrice, wtx, reservekey, nFeeRequired))
|
bool fTxCreated = false;
|
||||||
{
|
CRITICAL_BLOCK(cs_main)
|
||||||
if (nPrice + nFeeRequired > pwalletMain->GetBalance())
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
Error(strprintf(_("This transaction requires a transaction fee of at least %s because of its amount, complexity, or use of recently received funds"), FormatMoney(nFeeRequired).c_str()));
|
{
|
||||||
else
|
fTxCreated = pwalletMain->CreateTransaction(scriptPubKey, nPrice, wtx, reservekey, nFeeRequired);
|
||||||
Error(_("Transaction creation failed"));
|
}
|
||||||
return;
|
if (!fTxCreated)
|
||||||
}
|
{
|
||||||
|
if (nPrice + nFeeRequired > pwalletMain->GetBalance())
|
||||||
|
Error(strprintf(_("This transaction requires a transaction fee of at least %s because of its amount, complexity, or use of recently received funds"), FormatMoney(nFeeRequired).c_str()));
|
||||||
|
else
|
||||||
|
Error(_("Transaction creation failed"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (fWasLocked)
|
if (fWasLocked)
|
||||||
pwalletMain->Lock();
|
pwalletMain->Lock();
|
||||||
}
|
|
||||||
|
|
||||||
// Transaction fee
|
// Transaction fee
|
||||||
if (!ThreadSafeAskFee(nFeeRequired, _("Sending..."), this))
|
if (!ThreadSafeAskFee(nFeeRequired, _("Sending..."), this))
|
||||||
{
|
{
|
||||||
Error(_("Transaction aborted"));
|
Error(_("Transaction aborted"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure we're still connected
|
// Make sure we're still connected
|
||||||
CNode* pnode = ConnectNode(addr, 2 * 60 * 60);
|
CNode* pnode = ConnectNode(addr, 2 * 60 * 60);
|
||||||
if (!pnode)
|
if (!pnode)
|
||||||
{
|
{
|
||||||
Error(_("Lost connection, transaction cancelled"));
|
Error(_("Lost connection, transaction cancelled"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Last chance to cancel
|
// Last chance to cancel
|
||||||
Sleep(50);
|
Sleep(50);
|
||||||
|
if (!Status())
|
||||||
|
return;
|
||||||
|
fCanCancel = false;
|
||||||
|
if (fAbort)
|
||||||
|
{
|
||||||
|
fCanCancel = true;
|
||||||
if (!Status())
|
if (!Status())
|
||||||
return;
|
return;
|
||||||
fCanCancel = false;
|
fCanCancel = false;
|
||||||
if (fAbort)
|
|
||||||
{
|
|
||||||
fCanCancel = true;
|
|
||||||
if (!Status())
|
|
||||||
return;
|
|
||||||
fCanCancel = false;
|
|
||||||
}
|
|
||||||
if (!Status(_("Sending payment...")))
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Commit
|
|
||||||
if (!pwalletMain->CommitTransaction(wtx, reservekey))
|
|
||||||
{
|
|
||||||
Error(_("The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here."));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Send payment tx to seller, with response going to OnReply3 via event handler
|
|
||||||
CWalletTx wtxSend = wtx;
|
|
||||||
wtxSend.fFromMe = false;
|
|
||||||
pnode->PushRequest("submitorder", wtxSend, SendingDialogOnReply3, this);
|
|
||||||
|
|
||||||
Status(_("Waiting for confirmation..."));
|
|
||||||
MainFrameRepaint();
|
|
||||||
}
|
}
|
||||||
|
if (!Status(_("Sending payment...")))
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Commit
|
||||||
|
bool fTxCommitted = false;
|
||||||
|
CRITICAL_BLOCK(cs_main)
|
||||||
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
|
{
|
||||||
|
fTxCommitted = pwalletMain->CommitTransaction(wtx, reservekey);
|
||||||
|
}
|
||||||
|
if (!fTxCommitted)
|
||||||
|
{
|
||||||
|
Error(_("The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send payment tx to seller, with response going to OnReply3 via event handler
|
||||||
|
CWalletTx wtxSend = wtx;
|
||||||
|
wtxSend.fFromMe = false;
|
||||||
|
pnode->PushRequest("submitorder", wtxSend, SendingDialogOnReply3, this);
|
||||||
|
|
||||||
|
Status(_("Waiting for confirmation..."));
|
||||||
|
MainFrameRepaint();
|
||||||
}
|
}
|
||||||
|
|
||||||
void SendingDialogOnReply3(void* parg, CDataStream& vRecv)
|
void SendingDialogOnReply3(void* parg, CDataStream& vRecv)
|
||||||
@ -2621,7 +2624,7 @@ CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInit
|
|||||||
m_listCtrlReceiving->SetFocus();
|
m_listCtrlReceiving->SetFocus();
|
||||||
|
|
||||||
// Fill listctrl with address book data
|
// Fill listctrl with address book data
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
{
|
{
|
||||||
string strDefaultReceiving = (string)pframeMain->m_textCtrlAddress->GetValue();
|
string strDefaultReceiving = (string)pframeMain->m_textCtrlAddress->GetValue();
|
||||||
BOOST_FOREACH(const PAIRTYPE(CBitcoinAddress, string)& item, pwalletMain->mapAddressBook)
|
BOOST_FOREACH(const PAIRTYPE(CBitcoinAddress, string)& item, pwalletMain->mapAddressBook)
|
||||||
@ -2682,7 +2685,7 @@ void CAddressBookDialog::OnListEndLabelEdit(wxListEvent& event)
|
|||||||
if (event.IsEditCancelled())
|
if (event.IsEditCancelled())
|
||||||
return;
|
return;
|
||||||
string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1);
|
string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1);
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
pwalletMain->SetAddressBookName(strAddress, string(event.GetText()));
|
pwalletMain->SetAddressBookName(strAddress, string(event.GetText()));
|
||||||
pframeMain->RefreshListCtrl();
|
pframeMain->RefreshListCtrl();
|
||||||
}
|
}
|
||||||
@ -2718,7 +2721,7 @@ void CAddressBookDialog::OnButtonDelete(wxCommandEvent& event)
|
|||||||
if (m_listCtrl->GetItemState(nIndex, wxLIST_STATE_SELECTED))
|
if (m_listCtrl->GetItemState(nIndex, wxLIST_STATE_SELECTED))
|
||||||
{
|
{
|
||||||
string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1);
|
string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1);
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
pwalletMain->DelAddressBookName(strAddress);
|
pwalletMain->DelAddressBookName(strAddress);
|
||||||
m_listCtrl->DeleteItem(nIndex);
|
m_listCtrl->DeleteItem(nIndex);
|
||||||
}
|
}
|
||||||
@ -2778,7 +2781,7 @@ void CAddressBookDialog::OnButtonEdit(wxCommandEvent& event)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Write back
|
// Write back
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
{
|
{
|
||||||
if (strAddress != strAddressOrg)
|
if (strAddress != strAddressOrg)
|
||||||
pwalletMain->DelAddressBookName(strAddressOrg);
|
pwalletMain->DelAddressBookName(strAddressOrg);
|
||||||
@ -2818,22 +2821,19 @@ void CAddressBookDialog::OnButtonNew(wxCommandEvent& event)
|
|||||||
return;
|
return;
|
||||||
strName = dialog.GetValue();
|
strName = dialog.GetValue();
|
||||||
|
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_vMasterKey)
|
bool fWasLocked = pwalletMain->IsLocked();
|
||||||
{
|
if (!GetWalletPassphrase())
|
||||||
bool fWasLocked = pwalletMain->IsLocked();
|
return;
|
||||||
if (!GetWalletPassphrase())
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Generate new key
|
// Generate new key
|
||||||
strAddress = CBitcoinAddress(pwalletMain->GetOrReuseKeyFromPool()).ToString();
|
strAddress = CBitcoinAddress(pwalletMain->GetOrReuseKeyFromPool()).ToString();
|
||||||
|
|
||||||
if (fWasLocked)
|
if (fWasLocked)
|
||||||
pwalletMain->Lock();
|
pwalletMain->Lock();
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add to list and select it
|
// Add to list and select it
|
||||||
CRITICAL_BLOCK(pwalletMain->cs_mapAddressBook)
|
CRITICAL_BLOCK(pwalletMain->cs_wallet)
|
||||||
pwalletMain->SetAddressBookName(strAddress, strName);
|
pwalletMain->SetAddressBookName(strAddress, strName);
|
||||||
int nIndex = InsertLine(m_listCtrl, strName, strAddress);
|
int nIndex = InsertLine(m_listCtrl, strName, strAddress);
|
||||||
SetSelection(m_listCtrl, nIndex);
|
SetSelection(m_listCtrl, nIndex);
|
||||||
|
@ -353,22 +353,6 @@ int64 CWallet::GetDebit(const CTxIn &txin) const
|
|||||||
|
|
||||||
int64 CWalletTx::GetTxTime() const
|
int64 CWalletTx::GetTxTime() const
|
||||||
{
|
{
|
||||||
CRITICAL_BLOCK(cs_main)
|
|
||||||
{
|
|
||||||
if (!fTimeReceivedIsTxTime && hashBlock != 0)
|
|
||||||
{
|
|
||||||
// If we did not receive the transaction directly, we rely on the block's
|
|
||||||
// time to figure out when it happened. We use the median over a range
|
|
||||||
// of blocks to try to filter out inaccurate block times.
|
|
||||||
map<uint256, CBlockIndex*>::iterator mi = mapBlockIndex.find(hashBlock);
|
|
||||||
if (mi != mapBlockIndex.end())
|
|
||||||
{
|
|
||||||
CBlockIndex* pindex = (*mi).second;
|
|
||||||
if (pindex)
|
|
||||||
return pindex->GetMedianTime();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nTimeReceived;
|
return nTimeReceived;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user