日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 物聯(lián)網(wǎng) > 區(qū)塊鏈
[導(dǎo)讀] 委托調(diào)用(delegatecall) 委托調(diào)用是一種特殊的低級函數(shù)調(diào)用,旨在從另一個(通常是庫)合約中調(diào)用函數(shù)。 delegateCall()的優(yōu)點是可以保留當(dāng)前調(diào)用合約環(huán)境的內(nèi)

委托調(diào)用(delegatecall)

委托調(diào)用是一種特殊的低級函數(shù)調(diào)用,旨在從另一個(通常是庫)合約中調(diào)用函數(shù)。

delegateCall()的優(yōu)點是可以保留當(dāng)前調(diào)用合約環(huán)境的內(nèi)容。此環(huán)境包括其storage及其msg.sender,msg.value屬性。

以太坊將數(shù)據(jù)存儲在存儲“插槽”中,即32字節(jié)大小的插槽。每次將變量保存到存儲時,它會自動占用當(dāng)前插槽中的剩余空間,或者按順序占用下一個插槽。

在下圖中,合約A向合約B的saveX()函數(shù)發(fā)出委托調(diào)用,該函數(shù)最終會改變合同A的存儲。

首先,Contract A通過delegatecall調(diào)用saveX函數(shù)。委托調(diào)用覆蓋合約B的存儲與存儲調(diào)用合約akaStorage A.

接下來,執(zhí)行thesaveX函數(shù)。請注意,最初合約B存儲到存儲槽0中。因此,當(dāng)此函數(shù)現(xiàn)在引用變量bar時,再次查看槽0。

但是現(xiàn)在插槽0是引用指針foo,因此foo設(shè)置為x。bar仍然超出范圍,沒有受到影響。

當(dāng)合約A向合約B發(fā)出委托調(diào)用時,它允許合約B自由地改變其存儲A.

顯然,當(dāng)開發(fā)人員在不安全的存儲環(huán)境中使用delegatecall()或從惡意庫繼承時,安全風(fēng)險就會發(fā)生

如果存儲變量是通過低級的delegatecall訪問的,那么兩個合約的存儲布局必須對等,以便被調(diào)用合約能夠按名稱正確訪問調(diào)用合約的存儲變量。當(dāng)然,如果將存儲指針作為函數(shù)參數(shù)傳遞,就像在高級庫中那樣,情況就不是同了。

現(xiàn)在利用您對delegatecall()的理解來獲得此級別合同的所有權(quán)!

細節(jié)演練

1、delegation.sol對庫合約delegaTIon.sol進行delegatecall。

2、注意Delegate.sol有一個名為pwn()的公共函數(shù),它將所有者變量的所有權(quán)更改為調(diào)用該函數(shù)的任何人!

contract Delegate {

address public owner; // Occupies slot 0

。..

funcTIon pwn() public {

owner = msg.sender; // Save msg.sender to slot 0

}

}

3、注意,委托合同的槽0也存儲了owner,確切地說是您想要更改的變量!此外,如果您設(shè)法調(diào)用DelegaTIon.sol中的回退函數(shù)來調(diào)用pwn(),您將成為調(diào)用合約的所有者。

funcTIon() public {

if(delegate.delegatecall(msg.data)) {

this;

}

}

4、在以太坊中,您可以通過在事務(wù)中發(fā)送數(shù)據(jù)來調(diào)用公共函數(shù)。格式如下:

contractInstance.call(bytes4(sha3(“functionName(inputType)”))

5、使用Remix IDE或控制臺,調(diào)用Delegation.sol的回退功能:

// I did so in the console, having already computed

// the bytes4(sha3(“pwn()”))

await sendTransaction({

from: “0x1733d5adaccbe8057dba822ea74806361d181654”,

to: “0xe3895c413b0035512c029878d1ce4d8702d02320”,

data: “0xdd365b8b0000000000000000000000000000000000000000000000000000000000000000”

});

6、wait contract.owner()顯示您現(xiàn)在是所有者!

提示:您可以執(zhí)行Remix調(diào)試器(在Javascript VM模式下)以查看存儲環(huán)境如何更改! 您可以在Remix調(diào)試器的storage fully loaded列表中找到存儲插槽。

關(guān)鍵細節(jié)

· 使用更高級別的call()函數(shù)繼承庫,特別是當(dāng)你i)不需要更改合同存儲和ii)不關(guān)心gas控制時。

· 更改合同存儲的庫繼承時,請確保將存儲插槽與庫的存儲插槽對齊,以避免出現(xiàn)這些邊緣情況。

· 對調(diào)用delegatecalls的函數(shù)進行身份驗證并進行條件檢查。

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除( 郵箱:macysun@21ic.com )。
換一批
延伸閱讀
關(guān)閉