Forum Home
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Popular

    [Dev] Documenting Feathercoin Specific Software settings - Part 9

    Technical Development
    1
    37
    6244
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • wrapper
      wrapper Moderators last edited by

      Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

      Add Multisgin Page : - commit

      https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

      Include a multi-signature address Page

      src/qt/sendcoinsentry.h

       +    void setFieldEnable(bool enable);
      

      Code added

       +    void setRemoveEnabled(bool enabled);
      

      Code added

      1 Reply Last reply Reply Quote 0
      • wrapper
        wrapper Moderators last edited by

        Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

        Add Multisgin Page : - commit

        https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

        Include a multi-signature address Page

        src/qt/walletframe.cpp

         +void WalletFrame::gotoMultiSigPage()
         +{
         +    QMap<QString, WalletView*>::const_iterator i;
         +    for (i = mapWalletViews.constBegin(); i != mapWalletViews.constEnd();  ++i)
         +        i.value()->gotoMultiSigPage();
         +}
         +
        

        Code added

        1 Reply Last reply Reply Quote 0
        • wrapper
          wrapper Moderators last edited by

          Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

          Add Multisgin Page : - commit

          https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

          Include a multi-signature address Page

          src/qt/walletframe.cpp

           +qint64 WalletModel::getSharedBalance(const CCoinControl *coinControl) const
           +{
           +    if (coinControl)
           +    {
           +        int64 nBalance = 0;
           +        std::vector<COutput> vCoins;
           +        wallet->AvailableSharedCoins(vCoins, true, coinControl);
           +        BOOST_FOREACH(const COutput& out, vCoins)
           +            nBalance += out.tx->vout[out.i].nValue;   
           +        
           +        return nBalance;
           +    }
           +
           +    return wallet->GetSharedBalance();
           +}
           +
           +qint64 WalletModel::getSharedUnconfirmedBalance() const
           +{
           +    return wallet->GetSharedUnconfirmedBalance();
           +}
           +
           +qint64 WalletModel::getSharedImmatureBalance() const
           +{
           +    return wallet->GetSharedImmatureBalance();
           +}
           +
          

          Code added

           +WalletModel::SendCoinsReturn WalletModel::createRawTransaction(
           +    const QList<SendCoinsRecipient> &recipients, CTransaction& txNew, const CCoinControl *coinControl, bool isMultiSig)
           +{
           +    qint64 total = 0;
           +    QSet<QString> setAddress;
           +    QString hex;
           +
           +    if(recipients.empty())
           +    {
           +        return OK;
           +    }
           +
           +    // Pre-check input data for validity
           +    foreach(const SendCoinsRecipient &rcp, recipients)
           +    {
           +        if(!validateAddress(rcp.address))
           +        {
           +            return InvalidAddress;
           +        }
           +        setAddress.insert(rcp.address);
           +
           +        if(rcp.amount <= 0)
           +        {
           +            return InvalidAmount;
           +        }
           +        total      += rcp.amount;
           +    }
           +
           +    if(recipients.size() > setAddress.size())
           +    {
           +        return DuplicateAddress;
           +    }
           +
           +    int64 nBalance;
           +    if ( isMultiSig )
           +        nBalance = getSharedBalance(coinControl);
           +    else
           +        nBalance = getBalance(coinControl);
           +
           +    if(total > nBalance)
           +    {
           +        return AmountExceedsBalance;
           +    }
           +
           +    if((total + nTransactionFee) > nBalance)
           +    {
           +        return SendCoinsReturn(AmountWithFeeExceedsBalance, nTransactionFee);
           +    }
           +
           +    {
           +        LOCK2(cs_main, wallet->cs_wallet);
           +
           +        // Sendmany
           +        std::vector<std::pair<CScript, int64> > vecSend;
           +        foreach(const SendCoinsRecipient &rcp, recipients)
           +        {
           +            CScript scriptPubKey;
           +            scriptPubKey.SetDestination(CBitcoinAddress(rcp.address.toStdString()).Get());
           +            vecSend.push_back(make_pair(scriptPubKey, rcp.amount));
           +        }
           +
           +        int64 nFeeRequired = 0;
           +        std::string strFailReason;
           +        CReserveKey reservekey(wallet);
           +        bool fCreated = wallet->CreateRawTransaction(vecSend, txNew, nFeeRequired, strFailReason, isMultiSig, reservekey, coinControl);
           +
           +        if(!fCreated)
           +        {
           +            if((total  + nFeeRequired) > nBalance)
           +            {
           +                return SendCoinsReturn(AmountWithFeeExceedsBalance, nFeeRequired);
           +            }
           +            emit message(tr("Send Coins"), QString::fromStdString(strFailReason),
           +                         CClientUIInterface::MSG_ERROR);
           +            return TransactionCreationFailed;
           +        }
           +        /*if(!uiInterface.ThreadSafeAskFee(nFeeRequired))
           +        {
           +            return Aborted;
           +        }*/
           +        hex = QString::fromStdString(txNew.GetHash().GetHex());
           +    }
           +
           +    // Add addresses / update labels that we've sent to to the address book
           +    foreach(const SendCoinsRecipient &rcp, recipients)
           +    {
           +        std::string strAddress = rcp.address.toStdString();
           +        CTxDestination dest = CBitcoinAddress(strAddress).Get();
           +        std::string strLabel = rcp.label.toStdString();
           +        {
           +            LOCK(wallet->cs_wallet);
           +
           +            //std::map<CTxDestination, std::string>::iterator mi = wallet->mapAddressBook.find(dest);
           +            std::map<CTxDestination, CAddressBookData>::iterator mi = wallet->mapAddressBook.find(dest);
           +
           +            // Check if we have a new address or an updated label
           +            if (mi == wallet->mapAddressBook.end() || mi->second.name != strLabel)
           +            {
           +            		std::string purpose;
           +                //wallet->SetAddressBookName(dest, strLabel);
           +                wallet->SetAddressBook(dest, strLabel,purpose);
           +            }
           +        }
           +    }
           +
           +    return SendCoinsReturn(OK, 0, hex);
           +}
           +
          

          Code added

          1 Reply Last reply Reply Quote 0
          • wrapper
            wrapper Moderators last edited by

            Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

            Add Multisgin Page : - commit

            https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

            Include a multi-signature address Page

            src/qt/walletmodel.h

             +    
             +    qint64 getSharedBalance(const CCoinControl *coinControl=NULL) const;
             +    qint64 getSharedUnconfirmedBalance() const;
             +    qint64 getSharedImmatureBalance() const;
            

            Code added

             +        /*SendCoinsReturn(StatusCode status = OK):
             +            status(status) {}*/
             +        SendCoinsReturn(StatusCode status=Aborted,
             +                         qint64 fee=0,
             +                         QString hex=QString()):
             +            status(status), fee(fee), hex(hex) {}
            

            Code replaced

             +        qint64 fee; // is used in case status is "AmountWithFeeExceedsBalance"
             +        QString hex; // is filled with the transaction hash if status is "OK"
            

            Code added

             +    SendCoinsReturn createRawTransaction(const QList<SendCoinsRecipient> &recipients, CTransaction& txNew, const CCoinControl *coinControl, bool isMultiSig);
            
             +		bool isMultiSig;
             +    bool was_locked;
            
             +		CWallet *getWallet(){ return wallet; }
            

            Code added

            1 Reply Last reply Reply Quote 0
            • wrapper
              wrapper Moderators last edited by wrapper

              Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

              Add Multisgin Page : - commit

              https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

              Include a multi-signature address Page

              src/qt/walletview.cpp

               +#include "addressbookpage.h"
              
               +#include "multisigdialog.h"
              
               +    multiSigPage = new MultiSigDialog();
              
               +    addWidget(multiSigPage);
              

              Code added

               +    connect(transactionView, SIGNAL(doubleClicked(QModelIndex)), transactionView, SLOT(showDetails()));    
              

              Code replaced

               +    connect(sendCoinsPage, SIGNAL(sendCoins(QString)), this, SLOT(gotoSendCoinsPage(QString))); 
               +    
               +    // Clicking on "Send Coins" in the address book sends you to the send coins tab
               +    connect(transactionView, SIGNAL(sendCoins(QString)), this, SLOT(gotoSendCoinsPage(QString))); 
               +    // Clicking on "Verify Message" in the address book opens the verify message tab in the Sign/Verify Message dialog
               +    connect(transactionView, SIGNAL(verifyMessage(QString)), this, SLOT(gotoVerifyMessageTab(QString)));
               +    // Clicking on "Sign Message" in the receive coins page opens the sign message tab in the Sign/Verify Message dialog
               +    connect(transactionView, SIGNAL(signMessage(QString)), this, SLOT(gotoSignMessageTab(QString)));
              

              Code replaced

               +    multiSigPage->setModel(walletModel);
              
               +void WalletView::gotoMultiSigPage()
               +{
               +    setCurrentWidget(multiSigPage);
               +}
               +
              

              Code added

               +    AddressBookPage *dlg = new AddressBookPage(AddressBookPage::ForEditing, AddressBookPage::SendingTab, this);    
              

              Code replaced

              1 Reply Last reply Reply Quote 0
              • wrapper
                wrapper Moderators last edited by wrapper

                Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

                Add Multisgin Page : - commit

                https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

                Include a multi-signature address Page

                src/qt/walletview.h

                 +class AddressBookPage;
                
                 +class MultiSigDialog;
                
                 +    AddressBookPage *addressBookPage;
                
                 +    MultiSigDialog *multiSigPage;
                

                Code added

                 +		ReportView *reportView;		
                

                Code replaced

                 +    /** Switch to multisig page */
                 +    void gotoMultiSigPage();
                

                Code added

                1 Reply Last reply Reply Quote 0
                • wrapper
                  wrapper Moderators last edited by

                  Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

                  Add Multisgin Page : - commit

                  https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

                  Include a multi-signature address Page

                  src/script.cpp

                   +static const size_t nMaxNumSize = 4;
                  
                   +CBigNum CastToBigNum(const valtype& vch)
                   +{
                   +    if (vch.size() > nMaxNumSize)
                   +        throw runtime_error("CastToBigNum() : overflow");
                   +    // Get rid of extra leading zeros
                   +    return CBigNum(CBigNum(vch).getvch());
                   +}
                   +
                  

                  Code added

                   +class CKeyStoreIsMyShareVisitor : public boost::static_visitor<bool>
                   +{
                   +private:
                   +    const CKeyStore *keystore;
                   +public:
                   +    CKeyStoreIsMyShareVisitor(const CKeyStore *keystoreIn) : keystore(keystoreIn) { }
                   +    bool operator()(const CNoDestination &dest) const { return false; }
                   +    bool operator()(const CKeyID &keyID) const { return false; }
                   +    bool operator()(const CScriptID &scriptID) const { 
                   +        CScript scriptPubKey;
                   +        bool haveCScript = keystore->GetCScript(scriptID, scriptPubKey);
                   +        if ( !haveCScript )
                   +            return false;
                   +
                   +        return IsMyShare(*keystore, scriptPubKey);
                   +    }
                   +    bool operator()(const CStealthAddress &stxAddr) const
                   +    {
                   +        return false;
                   +    }
                   +};
                   +
                   +bool IsMyShare(const CKeyStore& keystore, const CTxDestination &dest)
                   +{
                   +    return boost::apply_visitor(CKeyStoreIsMyShareVisitor(&keystore), dest);
                   +}
                   +
                  

                  Code added

                   +bool IsMyShare(const CKeyStore& keystore, const CScript& scriptPubKey)
                   +{
                   +    vector<valtype> vSolutions;
                   +    txnouttype whichType;
                   +    if (!Solver(scriptPubKey, whichType, vSolutions))
                   +        return false;
                   +
                   +    CKeyID keyID;
                   +    switch (whichType)
                   +    {
                   +    case TX_NONSTANDARD:
                   +    case TX_NULL_DATA:
                   +    case TX_PUBKEY:
                   +    case TX_PUBKEYHASH:
                   +        return false;
                   +    case TX_SCRIPTHASH:
                   +    {
                   +        CScript subscript;
                   +        if (!keystore.GetCScript(CScriptID(uint160(vSolutions[0])), subscript))
                   +            return false;
                   +
                   +        return IsMyShare(keystore, subscript);
                   +    }
                   +    case TX_MULTISIG:
                   +    {
                   +        vector<valtype> keys(vSolutions.begin()+1, vSolutions.begin() +vSolutions.size()-1);
                   +        int nkeys = HaveKeys(keys, keystore) ;
                   +        return nkeys > 0 && nkeys < keys.size();
                   +    }
                   +    default:
                   +        break;
                   +    }
                   +    return false;
                   +}
                   +
                  

                  Code added

                   +bool EvalMultiSigScript(vector<vector<unsigned char> >& stack, const CScript& script, const CTransaction& txTo, unsigned int nIn, unsigned int flags, int nHashType, bool *bIsSign)
                   +{
                   +    CAutoBN_CTX pctx;
                   +    CScript::const_iterator pc = script.begin();
                   +    CScript::const_iterator pend = script.end();
                   +    CScript::const_iterator pbegincodehash = script.begin();
                   +    opcodetype opcode;
                   +    valtype vchPushValue;
                   +    vector<valtype> altstack;
                   +    if (script.size() > 10000)
                   +        return false;
                   +    int nOpCount = 0;
                   +    bool fStrictEncodings = flags & SCRIPT_VERIFY_STRICTENC;
                   +
                   +    try
                   +    {
                   +        while (pc < pend)
                   +        {
                   +            bool fExec = true;
                   +
                   +            //
                   +            // Read instruction
                   +            //
                   +            if (!script.GetOp(pc, opcode, vchPushValue))
                   +                return false;
                   +            if (vchPushValue.size() > MAX_SCRIPT_ELEMENT_SIZE)
                   +                return false;
                   +            if (opcode > OP_16 &&  ++nOpCount > 201)
                   +                return false;
                   +
                   +            if (opcode == OP_CAT ||
                   +                opcode == OP_SUBSTR ||
                   +                opcode == OP_LEFT ||
                   +                opcode == OP_RIGHT ||
                   +                opcode == OP_INVERT ||
                   +                opcode == OP_AND ||
                   +                opcode == OP_OR ||
                   +                opcode == OP_XOR ||
                   +                opcode == OP_2MUL ||
                   +                opcode == OP_2DIV ||
                   +                opcode == OP_MUL ||
                   +                opcode == OP_DIV ||
                   +                opcode == OP_MOD ||
                   +                opcode == OP_LSHIFT ||
                   +                opcode == OP_RSHIFT)
                   +                return false; // Disabled opcodes.
                   +
                   +            if (fExec && 0 <= opcode && opcode <= OP_PUSHDATA4)
                   +                stack.push_back(vchPushValue);
                   +            else if (fExec || (OP_IF <= opcode && opcode <= OP_ENDIF))
                   +            switch (opcode)
                   +            {
                   +                //
                   +                // Push value
                   +                //
                   +                case OP_1NEGATE:
                   +                case OP_1:
                   +                case OP_2:
                   +                case OP_3:
                   +                case OP_4:
                   +                case OP_5:
                   +                case OP_6:
                   +                case OP_7:
                   +                case OP_8:
                   +                case OP_9:
                   +                case OP_10:
                   +                case OP_11:
                   +                case OP_12:
                   +                case OP_13:
                   +                case OP_14:
                   +                case OP_15:
                   +                case OP_16:
                   +                {
                   +                    // ( -- value)
                   +                    //CBigNum bn((int)opcode - (int)(OP_1 - 1));
                   +                    CScriptNum bn((int)opcode - (int)(OP_1 - 1));
                   +                    stack.push_back(bn.getvch());
                   +                }
                   +                break;
                   +
                   +
                   +                //
                   +                // Control
                   +                //
                   +                case OP_NOP:
                   +                case OP_NOP1: case OP_NOP2: case OP_NOP3: case OP_NOP4: case OP_NOP5:
                   +                case OP_NOP6: case OP_NOP7: case OP_NOP8: case OP_NOP9: case OP_NOP10:
                   +                break;
                   +
                   +                case OP_IF:
                   +                case OP_NOTIF:
                   +                case OP_ELSE:
                   +                case OP_ENDIF:
                   +                    return false;
                   +
                   +                case OP_VERIFY:
                   +                {
                   +                    // (true -- ) or
                   +                    // (false -- false) and return
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    bool fValue = CastToBool(stacktop(-1));
                   +                    if (fValue)
                   +                        popstack(stack);
                   +                    else
                   +                        return false;
                   +                }
                   +                break;
                   +
                   +                case OP_RETURN:
                   +                {
                   +                    return false;
                   +                }
                   +                break;
                   +
                   +
                   +                //
                   +                // Stack ops
                   +                //
                   +                case OP_TOALTSTACK:
                   +                {
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    altstack.push_back(stacktop(-1));
                   +                    popstack(stack);
                   +                }
                   +                break;
                   +
                   +                case OP_FROMALTSTACK:
                   +                {
                   +                    if (altstack.size() < 1)
                   +                        return false;
                   +                    stack.push_back(altstacktop(-1));
                   +                    popstack(altstack);
                   +                }
                   +                break;
                   +
                   +                case OP_2DROP:
                   +                {
                   +                    // (x1 x2 -- )
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    popstack(stack);
                   +                    popstack(stack);
                   +                }
                   +                break;
                   +
                   +                case OP_2DUP:
                   +                {
                   +                    // (x1 x2 -- x1 x2 x1 x2)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    valtype vch1 = stacktop(-2);
                   +                    valtype vch2 = stacktop(-1);
                   +                    stack.push_back(vch1);
                   +                    stack.push_back(vch2);
                   +                }
                   +                break;
                   +
                   +                case OP_3DUP:
                   +                {
                   +                    // (x1 x2 x3 -- x1 x2 x3 x1 x2 x3)
                   +                    if (stack.size() < 3)
                   +                        return false;
                   +                    valtype vch1 = stacktop(-3);
                   +                    valtype vch2 = stacktop(-2);
                   +                    valtype vch3 = stacktop(-1);
                   +                    stack.push_back(vch1);
                   +                    stack.push_back(vch2);
                   +                    stack.push_back(vch3);
                   +                }
                   +                break;
                   +
                   +                case OP_2OVER:
                   +                {
                   +                    // (x1 x2 x3 x4 -- x1 x2 x3 x4 x1 x2)
                   +                    if (stack.size() < 4)
                   +                        return false;
                   +                    valtype vch1 = stacktop(-4);
                   +                    valtype vch2 = stacktop(-3);
                   +                    stack.push_back(vch1);
                   +                    stack.push_back(vch2);
                   +                }
                   +                break;
                   +
                   +                case OP_2ROT:
                   +                {
                   +                    // (x1 x2 x3 x4 x5 x6 -- x3 x4 x5 x6 x1 x2)
                   +                    if (stack.size() < 6)
                   +                        return false;
                   +                    valtype vch1 = stacktop(-6);
                   +                    valtype vch2 = stacktop(-5);
                   +                    stack.erase(stack.end()-6, stack.end()-4);
                   +                    stack.push_back(vch1);
                   +                    stack.push_back(vch2);
                   +                }
                   +                break;
                   +
                   +                case OP_2SWAP:
                   +                {
                   +                    // (x1 x2 x3 x4 -- x3 x4 x1 x2)
                   +                    if (stack.size() < 4)
                   +                        return false;
                   +                    swap(stacktop(-4), stacktop(-2));
                   +                    swap(stacktop(-3), stacktop(-1));
                   +                }
                   +                break;
                   +
                   +                case OP_IFDUP:
                   +                {
                   +                    // (x - 0 | x x)
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    valtype vch = stacktop(-1);
                   +                    if (CastToBool(vch))
                   +                        stack.push_back(vch);
                   +                }
                   +                break;
                   +
                   +                case OP_DEPTH:
                   +                {
                   +                    // -- stacksize
                   +                    //CBigNum bn(stack.size());
                   +                    CScriptNum bn(stack.size());
                   +                    stack.push_back(bn.getvch());
                   +                }
                   +                break;
                   +
                   +                case OP_DROP:
                   +                {
                   +                    // (x -- )
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    popstack(stack);
                   +                }
                   +                break;
                   +
                   +                case OP_DUP:
                   +                {
                   +                    // (x -- x x)
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    valtype vch = stacktop(-1);
                   +                    stack.push_back(vch);
                   +                }
                   +                break;
                   +
                   +                case OP_NIP:
                   +                {
                   +                    // (x1 x2 -- x2)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    stack.erase(stack.end() - 2);
                   +                }
                   +                break;
                   +
                   +                case OP_OVER:
                   +                {
                   +                    // (x1 x2 -- x1 x2 x1)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    valtype vch = stacktop(-2);
                   +                    stack.push_back(vch);
                   +                }
                   +                break;
                   +
                   +                case OP_PICK:
                   +                case OP_ROLL:
                   +                {
                   +                    // (xn ... x2 x1 x0 n - xn ... x2 x1 x0 xn)
                   +                    // (xn ... x2 x1 x0 n - ... x2 x1 x0 xn)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    int n = CastToBigNum(stacktop(-1)).getint();
                   +                    popstack(stack);
                   +                    if (n < 0 || n >= (int)stack.size())
                   +                        return false;
                   +                    valtype vch = stacktop(-n-1);
                   +                    if (opcode == OP_ROLL)
                   +                        stack.erase(stack.end()-n-1);
                   +                    stack.push_back(vch);
                   +                }
                   +                break;
                   +
                   +                case OP_ROT:
                   +                {
                   +                    // (x1 x2 x3 -- x2 x3 x1)
                   +                    //  x2 x1 x3  after first swap
                   +                    //  x2 x3 x1  after second swap
                   +                    if (stack.size() < 3)
                   +                        return false;
                   +                    swap(stacktop(-3), stacktop(-2));
                   +                    swap(stacktop(-2), stacktop(-1));
                   +                }
                   +                break;
                   +
                   +                case OP_SWAP:
                   +                {
                   +                    // (x1 x2 -- x2 x1)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    swap(stacktop(-2), stacktop(-1));
                   +                }
                   +                break;
                   +
                   +                case OP_TUCK:
                   +                {
                   +                    // (x1 x2 -- x2 x1 x2)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    valtype vch = stacktop(-1);
                   +                    stack.insert(stack.end()-2, vch);
                   +                }
                   +                break;
                   +
                   +
                   +                case OP_SIZE:
                   +                {
                   +                    // (in -- in size)
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    //CBigNum bn(stacktop(-1).size());
                   +                    CScriptNum bn(stacktop(-1).size());
                   +                    stack.push_back(bn.getvch());
                   +                }
                   +                break;
                   +
                   +
                   +                //
                   +                // Bitwise logic
                   +                //
                   +                case OP_EQUAL:
                   +                case OP_EQUALVERIFY:
                   +                //case OP_NOTEQUAL: // use OP_NUMNOTEQUAL
                   +                {
                   +                    // (x1 x2 - bool)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    valtype& vch1 = stacktop(-2);
                   +                    valtype& vch2 = stacktop(-1);
                   +                    bool fEqual = (vch1 == vch2);
                   +                    // OP_NOTEQUAL is disabled because it would be too easy to say
                   +                    // something like n != 1 and have some wiseguy pass in 1 with extra
                   +                    // zero bytes after it (numerically, 0x01 == 0x0001 == 0x000001)
                   +                    //if (opcode == OP_NOTEQUAL)
                   +                    //    fEqual = !fEqual;
                   +                    popstack(stack);
                   +                    popstack(stack);
                   +                    stack.push_back(fEqual ? vchTrue : vchFalse);
                   +                    if (opcode == OP_EQUALVERIFY)
                   +                    {
                   +                        if (fEqual)
                   +                            popstack(stack);
                   +                        else
                   +                            return false;
                   +                    }
                   +                }
                   +                break;
                   +
                   +
                   +                //
                   +                // Numeric
                   +                //
                   +                case OP_1ADD:
                   +                case OP_1SUB:
                   +                case OP_NEGATE:
                   +                case OP_ABS:
                   +                case OP_NOT:
                   +                case OP_0NOTEQUAL:
                   +                {
                   +                    // (in -- out)
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    //CBigNum bn = CastToBigNum(stacktop(-1));
                   +                    CScriptNum bn(stacktop(-1));
                   +                    switch (opcode)
                   +                    {
                   +                    case OP_1ADD:       bn += bnOne; break;
                   +                    case OP_1SUB:       bn -= bnOne; break;
                   +                    case OP_NEGATE:     bn = -bn; break;
                   +                    case OP_ABS:        if (bn < bnZero) bn = -bn; break;
                   +                    case OP_NOT:        bn = (bn == bnZero); break;
                   +                    case OP_0NOTEQUAL:  bn = (bn != bnZero); break;
                   +                    default:            assert(!"invalid opcode"); break;
                   +                    }
                   +                    popstack(stack);
                   +                    stack.push_back(bn.getvch());
                   +                }
                   +                break;
                   +
                   +                case OP_ADD:
                   +                case OP_SUB:
                   +                case OP_BOOLAND:
                   +                case OP_BOOLOR:
                   +                case OP_NUMEQUAL:
                   +                case OP_NUMEQUALVERIFY:
                   +                case OP_NUMNOTEQUAL:
                   +                case OP_LESSTHAN:
                   +                case OP_GREATERTHAN:
                   +                case OP_LESSTHANOREQUAL:
                   +                case OP_GREATERTHANOREQUAL:
                   +                case OP_MIN:
                   +                case OP_MAX:
                   +                {
                   +                    // (x1 x2 -- out)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +                    /*CBigNum bn1 = CastToBigNum(stacktop(-2));
                   +                    CBigNum bn2 = CastToBigNum(stacktop(-1));
                   +                    CBigNum bn;*/
                   +                    CScriptNum bn1(stacktop(-2));
                   +                    CScriptNum bn2(stacktop(-1));
                   +                    CScriptNum bn(0);
                   +                    switch (opcode)
                   +                    {
                   +                    case OP_ADD:
                   +                        bn = bn1 + bn2;
                   +                        break;
                   +
                   +                    case OP_SUB:
                   +                        bn = bn1 - bn2;
                   +                        break;
                   +
                   +                    case OP_BOOLAND:             bn = (bn1 != bnZero && bn2 != bnZero); break;
                   +                    case OP_BOOLOR:              bn = (bn1 != bnZero || bn2 != bnZero); break;
                   +                    case OP_NUMEQUAL:            bn = (bn1 == bn2); break;
                   +                    case OP_NUMEQUALVERIFY:      bn = (bn1 == bn2); break;
                   +                    case OP_NUMNOTEQUAL:         bn = (bn1 != bn2); break;
                   +                    case OP_LESSTHAN:            bn = (bn1 < bn2); break;
                   +                    case OP_GREATERTHAN:         bn = (bn1 > bn2); break;
                   +                    case OP_LESSTHANOREQUAL:     bn = (bn1 <= bn2); break;
                   +                    case OP_GREATERTHANOREQUAL:  bn = (bn1 >= bn2); break;
                   +                    case OP_MIN:                 bn = (bn1 < bn2 ? bn1 : bn2); break;
                   +                    case OP_MAX:                 bn = (bn1 > bn2 ? bn1 : bn2); break;
                   +                    default:                     assert(!"invalid opcode"); break;
                   +                    }
                   +                    popstack(stack);
                   +                    popstack(stack);
                   +                    stack.push_back(bn.getvch());
                   +
                   +                    if (opcode == OP_NUMEQUALVERIFY)
                   +                    {
                   +                        if (CastToBool(stacktop(-1)))
                   +                            popstack(stack);
                   +                        else
                   +                            return false;
                   +                    }
                   +                }
                   +                break;
                   +
                   +                case OP_WITHIN:
                   +                {
                   +                    // (x min max -- out)
                   +                    if (stack.size() < 3)
                   +                        return false;
                   +                    /*CBigNum bn1 = CastToBigNum(stacktop(-3));
                   +                    CBigNum bn2 = CastToBigNum(stacktop(-2));
                   +                    CBigNum bn3 = CastToBigNum(stacktop(-1));*/
                   +                    CScriptNum bn1(stacktop(-3));
                   +                    CScriptNum bn2(stacktop(-2));
                   +                    CScriptNum bn3(stacktop(-1));
                   +                    bool fValue = (bn2 <= bn1 && bn1 < bn3);
                   +                    popstack(stack);
                   +                    popstack(stack);
                   +                    popstack(stack);
                   +                    stack.push_back(fValue ? vchTrue : vchFalse);
                   +                }
                   +                break;
                   +
                   +
                   +                //
                   +                // Crypto
                   +                //
                   +                case OP_RIPEMD160:
                   +                case OP_SHA1:
                   +                case OP_SHA256:
                   +                case OP_HASH160:
                   +                case OP_HASH256:
                   +                {
                   +                    // (in -- hash)
                   +                    if (stack.size() < 1)
                   +                        return false;
                   +                    valtype& vch = stacktop(-1);
                   +                    valtype vchHash((opcode == OP_RIPEMD160 || opcode == OP_SHA1 || opcode == OP_HASH160) ? 20 : 32);
                   +                    if (opcode == OP_RIPEMD160)
                   +                        RIPEMD160(&vch[0], vch.size(), &vchHash[0]);
                   +                    else if (opcode == OP_SHA1)
                   +                        SHA1(&vch[0], vch.size(), &vchHash[0]);
                   +                    else if (opcode == OP_SHA256)
                   +                        SHA256(&vch[0], vch.size(), &vchHash[0]);
                   +                    else if (opcode == OP_HASH160)
                   +                    {
                   +                        uint160 hash160 = Hash160(vch);
                   +                        memcpy(&vchHash[0], &hash160, sizeof(hash160));
                   +                    }
                   +                    else if (opcode == OP_HASH256)
                   +                    {
                   +                        uint256 hash = Hash(vch.begin(), vch.end());
                   +                        memcpy(&vchHash[0], &hash, sizeof(hash));
                   +                    }
                   +                    popstack(stack);
                   +                    stack.push_back(vchHash);
                   +                }
                   +                break;
                   +
                   +                case OP_CODESEPARATOR:
                   +                {
                   +                    // Hash starts after the code separator
                   +                    pbegincodehash = pc;
                   +                }
                   +                break;
                   +
                   +                case OP_CHECKSIG:
                   +                case OP_CHECKSIGVERIFY:
                   +                {
                   +                    // (sig pubkey -- bool)
                   +                    if (stack.size() < 2)
                   +                        return false;
                   +
                   +                    valtype& vchSig    = stacktop(-2);
                   +                    valtype& vchPubKey = stacktop(-1);
                   +
                   +                    ////// debug print
                   +                    //PrintHex(vchSig.begin(), vchSig.end(), "sig: %s\n");
                   +                    //PrintHex(vchPubKey.begin(), vchPubKey.end(), "pubkey: %s\n");
                   +
                   +                    // Subset of script starting at the most recent codeseparator
                   +                    CScript scriptCode(pbegincodehash, pend);
                   +
                   +                    // Drop the signature, since there's no way for a signature to sign itself
                   +                    scriptCode.FindAndDelete(CScript(vchSig));
                   +
                   +                    bool fSuccess = (!fStrictEncodings || (IsCanonicalSignature(vchSig,flags) && IsCanonicalPubKey(vchPubKey,flags)));
                   +                    if (fSuccess)
                   +                        fSuccess = CheckSig(vchSig, vchPubKey, scriptCode, txTo, nIn, nHashType, flags);
                   +
                   +                    popstack(stack);
                   +                    popstack(stack);
                   +                    stack.push_back(fSuccess ? vchTrue : vchFalse);
                   +                    if (opcode == OP_CHECKSIGVERIFY)
                   +                    {
                   +                        if (fSuccess)
                   +                            popstack(stack);
                   +                        else
                   +                            return false;
                   +                    }
                   +                }
                   +                break;
                   +
                   +                case OP_CHECKMULTISIG:
                   +                case OP_CHECKMULTISIGVERIFY:
                   +                {
                   +                    // ([sig ...] num_of_signatures [pubkey ...] num_of_pubkeys -- bool)
                   +                    int i = 1;
                   +                    if ((int)stack.size() < i)
                   +                        return false;
                   +
                   +                    int nKeysCount = CastToBigNum(stacktop(-i)).getint();
                   +                    if (nKeysCount < 0 || nKeysCount > 20)
                   +                        return false;
                   +                    nOpCount += nKeysCount;
                   +                    if (nOpCount > 201)
                   +                        return false;
                   +                    int ikey = ++i;
                   +                    i += nKeysCount;
                   +                    if ((int)stack.size() < i)
                   +                        return false;
                   +
                   +                    int nSigsCount = CastToBigNum(stacktop(-i)).getint();
                   +                    if (nSigsCount < 0 || nSigsCount > nKeysCount)
                   +                        return false;
                   +                    int isig = ++i;
                   +                    i += nSigsCount;
                   +                    //if ((int)stack.size() < i)
                   +                        //return false;
                   +                    int curSigsCount = nSigsCount + stack.size()- i;
                   +                    if ( curSigsCount > nSigsCount )
                   +                        curSigsCount = nSigsCount;
                   +
                   +                    // Subset of script starting at the most recent codeseparator
                   +                    CScript scriptCode(pbegincodehash, pend);
                   +
                   +                    // Drop the signatures, since there's no way for a signature to sign itself
                   +                    for (int k = 0; k < curSigsCount; k++)
                   +                    {
                   +                        valtype& vchSig = stacktop(-isig-k);
                   +                        scriptCode.FindAndDelete(CScript(vchSig));
                   +                    }
                   +
                   +                    bool fSuccess = true;
                   +                    int okCount = 0;
                   +                    for ( int n = 0; n < nKeysCount; ++ n )
                   +                    {
                   +                        for ( int m = 0; m < curSigsCount; ++ m )
                   +                        {
                   +                            valtype& vchSig    = stacktop(-isig - m);
                   +                            valtype& vchPubKey = stacktop(-ikey - n);
                   +
                   +                            // Check signature
                   +                            bool fOk = (!fStrictEncodings || (IsCanonicalSignature(vchSig,flags) && IsCanonicalPubKey(vchPubKey,flags)));
                   +                            if (fOk)
                   +                                fOk = CheckSig(vchSig, vchPubKey, scriptCode, txTo, nIn, nHashType, flags);
                   +
                   +                            if (fOk) {
                   +                                bIsSign[nKeysCount - 1 - n] = true;
                   +                                okCount += 1;
                   +                            }
                   +                        }
                   +                    }
                   +
                   +                    if ( okCount < nSigsCount )
                   +                        fSuccess = false;
                   +
                   +                    return fSuccess;
                   +                }
                   +                break;
                   +
                   +                default:
                   +                    return false;
                   +            }
                   +        }
                   +    }
                   +    catch (...)
                   +    {
                   +        return false;
                   +    }
                   +
                   +
                   +    return true;
                   +}
                   +
                   +bool VerifyMultiSigScript(const CScript& scriptSig, const CScript& scriptPubKey, const CTransaction& txTo, unsigned int nIn, int nHashType, bool *bIsSign)
                   +{
                   +    vector<vector<unsigned char> > stack, stackCopy;
                   +    if (!EvalScript(stack, scriptSig, txTo, nIn, SCRIPT_VERIFY_P2SH | SCRIPT_VERIFY_STRICTENC, nHashType))
                   +        return false;
                   +
                   +    stackCopy = stack;
                   +    if (!EvalScript(stack, scriptPubKey, txTo, nIn, SCRIPT_VERIFY_P2SH | SCRIPT_VERIFY_STRICTENC, nHashType))
                   +        return false;
                   +
                   +    if (stack.empty())
                   +        return false;
                   +
                   +    if (CastToBool(stack.back()) == false)
                   +        return false;
                   +
                   +    if (scriptPubKey.IsPayToScriptHash())
                   +    {
                   +        if (!scriptSig.IsPushOnly())
                   +            return false;
                   +
                   +        assert(!stackCopy.empty());
                   +
                   +        const valtype& pubKeySerialized = stackCopy.back();
                   +        CScript pubKey2(pubKeySerialized.begin(), pubKeySerialized.end());
                   +        popstack(stackCopy);
                   +
                   +        if (!EvalMultiSigScript(stackCopy, pubKey2, txTo, nIn, SCRIPT_VERIFY_P2SH, nHashType, bIsSign))
                   +            return false;
                   +
                   +        if (stackCopy.empty())
                   +            return false;
                   +
                   +        return CastToBool(stackCopy.back());
                   +    }
                   +
                   +    return false;
                   +}
                   +
                  

                  Code added

                  1 Reply Last reply Reply Quote 0
                  • wrapper
                    wrapper Moderators last edited by

                    Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

                    Add Multisgin Page : - commit

                    https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

                    Include a multi-signature address Page

                    src/script.h

                     +bool IsMyShare(const CKeyStore& keystore, const CScript& scriptPubKey);
                    
                     +bool IsMyShare(const CKeyStore& keystore, const CTxDestination &dest);
                    
                     +bool VerifyMultiSigScript(const CScript& scriptSig, const CScript& scriptPubKey, const CTransaction& txTo, unsigned int nIn, int nHashType, bool *bIsSign);
                    

                    Code added

                    1 Reply Last reply Reply Quote 0
                    • wrapper
                      wrapper Moderators last edited by

                      Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

                      Add Multisgin Page : - commit

                      https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

                      Include a multi-signature address Page

                      src/wallet.cpp

                       +#include "walletdb.h"
                       +#include "crypter.h"
                       +#include "ui_interface.h"
                      

                      Code added

                       -    NotifyAddressBookChanged(this, address, strName, ::IsMine(*this, address),
                       -                             strPurpose, (fUpdated ? CT_UPDATED : CT_NEW) );
                      
                       +    bool fMine = ::IsMine(*this, address);
                      
                       +    if ( ::IsMyShare(*this, address) )
                       +        fMine = false;
                       +    //NotifyAddressBookChanged(this, address, strName, ::IsMine(*this, address), strPurpose, (fUpdated ? CT_UPDATED : CT_NEW) );
                       +    NotifyAddressBookChanged(this, address, strName, fMine, strPurpose, (fUpdated ? CT_UPDATED : CT_NEW) );
                      

                      Code replaced

                       +}
                       +
                       +
                       +
                       +
                       +
                       +/* 
                       + *  for shared wallet
                       + */
                       +
                       +bool CWallet::SelectSharedCoins(int64 nTargetValue, set<pair<const CWalletTx*,unsigned int> >& setCoinsRet, int64& nValueRet, const CCoinControl* coinControl) const
                       +{
                       +    vector<COutput> vCoins;
                       +    AvailableSharedCoins(vCoins, false, coinControl);
                       +    
                       +    // coin control -> return all selected outputs (we want all selected to go into the transaction for sure)
                       +    if (coinControl && coinControl->HasSelected())
                       +    {
                       +        BOOST_FOREACH(const COutput& out, vCoins)
                       +        {
                       +            nValueRet += out.tx->vout[out.i].nValue;
                       +            setCoinsRet.insert(make_pair(out.tx, out.i));
                       +        }
                       +        return (nValueRet >= nTargetValue);
                       +    }
                       +
                       +    return (SelectCoinsMinConf(nTargetValue, 1, 6, vCoins, setCoinsRet, nValueRet) ||
                       +            SelectCoinsMinConf(nTargetValue, 1, 1, vCoins, setCoinsRet, nValueRet) ||
                       +            SelectCoinsMinConf(nTargetValue, 0, 1, vCoins, setCoinsRet, nValueRet));
                       +}
                       +
                       +bool CWallet::CreateRawTransaction(const std::vector<std::pair<CScript, int64> >& vecSend, CTransaction& txNew, 
                       +    int64& nFeeRet, std::string& strFailReason, bool isMultiSig, CReserveKey& reservekey, const CCoinControl *coinControl)
                       +{
                       +    int64 nValue = 0;
                       +    BOOST_FOREACH (const PAIRTYPE(CScript, int64)& s, vecSend)
                       +    {
                       +        if (nValue < 0)
                       +        {
                       +            strFailReason = _("Transaction amounts must be positive");
                       +            return false;
                       +        }
                       +        nValue += s.second;
                       +    }
                       +    if (vecSend.empty() || nValue < 0)
                       +    {
                       +        strFailReason = _("Transaction amounts must be positive");
                       +        return false;
                       +    }
                       +
                       +    int64 nTotalValue = nValue + nFeeRet;
                       +    CTransaction rawTx;
                       +
                       +    // vouts to the payees
                       +    BOOST_FOREACH (const PAIRTYPE(CScript, int64)& s, vecSend)
                       +    {
                       +        CTxOut txout(s.second, s.first);
                       +        if (txout.IsDust((CTransaction::nMinRelayTxFee)))
                       +        {
                       +            strFailReason = _("Transaction amount too small");
                       +            return false;
                       +        }
                       +        rawTx.vout.push_back(txout);
                       +    }
                       +
                       +    // Choose coins to use
                       +    set<pair<const CWalletTx*,unsigned int> > setCoins;
                       +    int64 nValueIn = 0;
                       +    if ( isMultiSig )
                       +    {
                       +        if (!SelectSharedCoins(nTotalValue, setCoins, nValueIn, coinControl))
                       +        {
                       +            strFailReason = _("Insufficient funds");
                       +            return false;
                       +        }
                       +    }
                       +    else
                       +    {
                       +        if (!SelectCoins(nTotalValue, setCoins, nValueIn, coinControl))
                       +        {
                       +            strFailReason = _("Insufficient funds");
                       +            return false;
                       +        }
                       +    }
                       +
                       +    int64 nChange = nValueIn - nTotalValue;
                       +    if (nChange > 0)
                       +    {
                       +        CScript scriptChange;
                       +        
                       +        // coin control: send change to custom address
                       +        if (coinControl && !boost::get<CNoDestination>(&coinControl->destChange))
                       +            scriptChange.SetDestination(coinControl->destChange);
                       +        else
                       +        {
                       +            // Reserve a new key pair from key pool
                       +            CPubKey vchPubKey;
                       +            assert(reservekey.GetReservedKey(vchPubKey)); // should never fail, as we just unlocked
                       +
                       +            scriptChange.SetDestination(vchPubKey.GetID());
                       +        }
                       +
                       +        CTxOut newTxOut(nChange, scriptChange);
                       +
                       +        if (newTxOut.IsDust((CTransaction::nMinRelayTxFee)))
                       +        {
                       +            nFeeRet += nChange;
                       +        }
                       +        else
                       +        {
                       +            vector<CTxOut>::iterator position = rawTx.vout.begin()+GetRandInt(rawTx.vout.size()+1);
                       +            rawTx.vout.insert(position, newTxOut);
                       +        }
                       +    }
                       +    else
                       +        reservekey.ReturnKey();
                       +
                       +    // Fill vin
                       +    BOOST_FOREACH(const PAIRTYPE(const CWalletTx*,unsigned int)& coin, setCoins)
                       +        rawTx.vin.push_back(CTxIn(coin.first->GetHash(),coin.second));
                       +
                       +    if ( !isMultiSig )
                       +    {
                       +        // Sign
                       +        int nIn = 0;
                       +        BOOST_FOREACH(const PAIRTYPE(const CWalletTx*,unsigned int)& coin, setCoins)
                       +            if (!SignSignature(*this, *coin.first, rawTx, nIn++))
                       +            {
                       +                strFailReason = _("Signing transaction failed");
                       +                return false;
                       +            }
                       +    }
                       +    
                       +    // Limit size
                       +    unsigned int nBytes = ::GetSerializeSize(rawTx, SER_NETWORK, PROTOCOL_VERSION);
                       +    if (nBytes >= MAX_STANDARD_TX_SIZE)
                       +    {
                       +        strFailReason = _("Transaction too large");
                       +        return false;
                       +    }
                       +
                       +    txNew = rawTx;
                       +    return true;
                       +}
                       +
                       +bool CWallet::IsMyShare(const CTxIn& txin) const
                       +{
                       +    {
                       +        LOCK(cs_wallet);
                       +        map<uint256, CWalletTx>::const_iterator mi = mapWallet.find(txin.prevout.hash);
                       +        if (mi != mapWallet.end())
                       +        {
                       +            const CWalletTx& prev = (*mi).second;
                       +            if (txin.prevout.n < prev.vout.size())
                       +                if (IsMyShare(prev.vout[txin.prevout.n]))
                       +                    return true;
                       +        }
                       +    }
                       +    return false;
                       +}
                       +
                       +bool CWallet::IsMyShare(const CTxOut& txout) const
                       +{
                       +    return ::IsMyShare(*this, txout.scriptPubKey);
                       +}
                       +
                       +bool CWallet::IsMyShare(const CTransaction& tx) const
                       +{
                       +    BOOST_FOREACH(const CTxOut& txout, tx.vout)
                       +        if (IsMyShare(txout) && txout.nValue >= DUST_HARD_LIMIT)
                       +            return true;
                       +
                       +    return false;
                       +}
                       +
                       +int64 CWallet::GetSharedBalance() const
                       +{
                       +    int64 nTotal = 0;
                       +    {
                       +        LOCK(cs_wallet);
                       +        for (map<uint256, CWalletTx>::const_iterator it = mapWallet.begin(); it != mapWallet.end(); ++it)
                       +        {
                       +            const CWalletTx* pcoin = &(*it).second;
                       +            if (pcoin->IsConfirmed())
                       +                nTotal += pcoin->GetSharedAvailableCredit();
                       +        }
                       +    }
                       +
                       +    return nTotal;
                       +}
                       +
                       +int64 CWallet::GetSharedUnconfirmedBalance() const
                       +{
                       +    int64 nTotal = 0;
                       +    {
                       +        LOCK(cs_wallet);
                       +        for (map<uint256, CWalletTx>::const_iterator it = mapWallet.begin(); it != mapWallet.end(); ++it)
                       +        {
                       +            const CWalletTx* pcoin = &(*it).second;
                       +            if (!pcoin->IsFinal() || !pcoin->IsConfirmed())
                       +                nTotal += pcoin->GetSharedAvailableCredit();
                       +        }
                       +    }
                       +    return nTotal;
                       +}
                       +
                       +int64 CWallet::GetSharedImmatureBalance() const
                       +{
                       +    return 0;
                       +}
                       +
                       +int64 CWallet::GetShareDebit(const CTxIn &txin) const
                       +{
                       +    {
                       +        LOCK(cs_wallet);
                       +        map<uint256, CWalletTx>::const_iterator mi = mapWallet.find(txin.prevout.hash);
                       +        if (mi != mapWallet.end())
                       +        {
                       +            const CWalletTx& prev = (*mi).second;
                       +            if (txin.prevout.n < prev.vout.size())
                       +                if (IsMyShare(prev.vout[txin.prevout.n]))
                       +                    return prev.vout[txin.prevout.n].nValue;
                       +        }
                       +    }
                       +    return 0;
                       +}
                       +
                       +void CWallet::AvailableSharedCoins(vector<COutput>& vCoins, bool fOnlyConfirmed, const CCoinControl *coinControl) const
                       +{
                       +    vCoins.clear();
                       +
                       +    {
                       +        LOCK(cs_wallet);
                       +        for (map<uint256, CWalletTx>::const_iterator it = mapWallet.begin(); it != mapWallet.end(); ++it)
                       +        {
                       +            const CWalletTx* pcoin = &(*it).second;
                       +
                       +            if (!pcoin->IsFinal())
                       +                continue;
                       +
                       +            if (fOnlyConfirmed && !pcoin->IsConfirmed())
                       +                continue;
                       +
                       +            if (pcoin->IsCoinBase() && pcoin->GetBlocksToMaturity() > 0)
                       +                continue;
                       +
                       +            for (unsigned int i = 0; i < pcoin->vout.size(); i++) {
                       +                if (!(pcoin->IsSpent(i)) && IsMyShare(pcoin->vout[i]) &&
                       +                    !IsLockedCoin((*it).first, i) && pcoin->vout[i].nValue >= DUST_HARD_LIMIT &&
                       +                    (!coinControl || !coinControl->HasSelected() || coinControl->IsSelected((*it).first, i))) 
                       +                        vCoins.push_back(COutput(pcoin, i, pcoin->GetDepthInMainChain()));
                       +            }
                       +        }
                       +    }
                       +}
                       +
                       +
                       +
                      

                      Code added

                      1 Reply Last reply Reply Quote 0
                      • wrapper
                        wrapper Moderators last edited by

                        Feathercoin specific changes made to convert Bitcoin to FTC 0.9.6.*

                        Add Multisgin Page : - commit

                        https://github.com/FeatherCoin/Feathercoin/commit/83737e90c292f18fe0285677cefbd70125492e1a

                        Include a multi-signature address Page

                        src/wallet.h

                         +#include "script.h"
                        

                        Code added

                         +    	
                         +    bool CreateRawTransaction(const std::vector<std::pair<CScript, int64> >& vecSend, CTransaction& txNew, 
                         +        int64& nFeeRet, std::string& strFailReason, bool isMultiSig, CReserveKey& reservekey, const CCoinControl *coinControl=NULL);
                         +        	
                         +    /* 
                         +     *  for shared wallet
                         +     */
                         +    bool IsMyShare(const CTxIn& txin) const;
                         +    bool IsMyShare(const CTxOut& txout) const;
                         +    bool IsMyShare(const CTransaction& tx) const;
                         +    int64 GetSharedBalance() const;
                         +    int64 GetSharedUnconfirmedBalance() const;
                         +    int64 GetSharedImmatureBalance() const;
                         +    int64 GetSharedCredit(const CTxOut& txout) const
                         +    {
                         +        if (!MoneyRange(txout.nValue))
                         +            throw std::runtime_error("CWallet::GetSharedCredit() : value out of range");
                         +        return (IsMyShare(txout) ? txout.nValue : 0);
                         +    }
                         +    bool IsFromMyShare(const CTransaction& tx) const
                         +    {
                         +        return (GetShareDebit(tx) > 0);
                         +    }
                         +    int64 GetShareDebit(const CTransaction& tx) const
                         +    {
                         +        int64 nDebit = 0;
                         +        BOOST_FOREACH(const CTxIn& txin, tx.vin)
                         +        {
                         +            nDebit += GetShareDebit(txin);
                         +            if (!MoneyRange(nDebit))
                         +                throw std::runtime_error("CWallet::GetDebit() : value out of range");
                         +        }
                         +        return nDebit;
                         +    }
                         +    int64 GetShareDebit(const CTxIn& txin) const;
                         +    void AvailableSharedCoins(std::vector<COutput>& vCoins, bool fOnlyConfirmed=true, const CCoinControl *coinControl=NULL) const;
                         +    bool SelectSharedCoins(int64 nTargetValue, std::set<std::pair<const CWalletTx*,unsigned int> >& setCoinsRet, int64& nValueRet, const CCoinControl* coinControl=NULL) const;
                        

                        Code added

                         +    std::vector<CMerkleTx> vtxPrev;
                        
                         +    std::vector<char> vfSpent; // which outputs are already spent
                        
                         +        fSharedAvailableCreditCached = false;
                        

                        Code added

                         +    
                         +    bool IsSpent(unsigned int nOut) const
                         +    {
                         +        if (nOut >= vout.size())
                         +            throw std::runtime_error("CWalletTx::IsSpent() : nOut out of range");
                         +        if (nOut >= vfSpent.size())
                         +            return false;
                         +        return (!!vfSpent[nOut]);
                         +    }
                        

                        Code added

                         +    
                         +    bool IsConfirmed() const
                         +    {
                         +        // Quick answer in most cases
                         +        if (!IsFinal())
                         +            return false;
                         +        if (GetDepthInMainChain() >= 1)
                         +            return true;
                         +        if (!IsFromMe()) // using wtx's cached debit
                         +            return false;
                         +
                         +        // If no confirmations but it's from us, we can still
                         +        // consider it confirmed if all dependencies are confirmed
                         +        std::map<uint256, const CMerkleTx*> mapPrev;
                         +        std::vector<const CMerkleTx*> vWorkQueue;
                         +        vWorkQueue.reserve(vtxPrev.size()+1);
                         +        vWorkQueue.push_back(this);
                         +        for (unsigned int i = 0; i < vWorkQueue.size(); i++)
                         +        {
                         +            const CMerkleTx* ptx = vWorkQueue[i];
                         +
                         +            if (!ptx->IsFinal())
                         +                return false;
                         +            if (ptx->GetDepthInMainChain() >= 1)
                         +                continue;
                         +            if (!pwallet->IsFromMe(*ptx))
                         +                return false;
                         +
                         +            if (mapPrev.empty())
                         +            {
                         +                BOOST_FOREACH(const CMerkleTx& tx, vtxPrev)
                         +                    mapPrev[tx.GetHash()] = &tx;
                         +            }
                         +
                         +            BOOST_FOREACH(const CTxIn& txin, ptx->vin)
                         +            {
                         +                if (!mapPrev.count(txin.prevout.hash))
                         +                    return false;
                         +                vWorkQueue.push_back(mapPrev[txin.prevout.hash]);
                         +            }
                         +        }
                         +        return true;
                         +    }
                        

                        Code added

                         +    	
                         +    /* 
                         +     *  for shared wallet
                         +     */
                         +    mutable bool fSharedAvailableCreditCached;
                         +    mutable int64 nSharedAvailableCreditCached;
                         +    int64 GetSharedAvailableCredit(bool fUseCache=true) const
                         +    {
                         +        // Must wait until coinbase is safely deep enough in the chain before valuing it
                         +        if (IsCoinBase() && GetBlocksToMaturity() > 0)
                         +            return 0;
                         +
                         +        //if (fUseCache && fSharedAvailableCreditCached)
                         +            //return nSharedAvailableCreditCached;
                         +
                         +        int64 nCredit = 0;
                         +        for (unsigned int i = 0; i < vout.size(); i++)
                         +        {
                         +            if (!IsSpent(i))
                         +            {
                         +                const CTxOut &txout = vout[i];
                         +                nCredit += pwallet->GetSharedCredit(txout);
                         +                if (!MoneyRange(nCredit))
                         +                    throw std::runtime_error("CWalletTx::GetSharedAvailableCredit() : value out of range");
                         +            }
                         +        }
                         +
                         +        nSharedAvailableCreditCached = nCredit;
                         +        fSharedAvailableCreditCached = true;
                         +        return nCredit;
                         +    }
                        

                        Code added

                        1 Reply Last reply Reply Quote 0
                        • First post
                          Last post