-
Notifications
You must be signed in to change notification settings - Fork 44
/
Copy pathLayerzeroImplementation.sol
408 lines (342 loc) · 14.8 KB
/
LayerzeroImplementation.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
// SPDX-License-Identifier: BUSL-1.1
pragma solidity ^0.8.23;
import { IAmbImplementationV2 as IAmbImplementation } from "src/interfaces/IAmbImplementationV2.sol";
import { IBaseStateRegistry } from "src/interfaces/IBaseStateRegistry.sol";
import { ISuperRBAC } from "src/interfaces/ISuperRBAC.sol";
import { ISuperRegistry } from "src/interfaces/ISuperRegistry.sol";
import { DataLib } from "src/libraries/DataLib.sol";
import { Error } from "src/libraries/Error.sol";
import { ProofLib } from "src/libraries/ProofLib.sol";
import { AMBMessage } from "src/types/DataTypes.sol";
import { ILayerZeroEndpoint } from "src/vendor/layerzero/ILayerZeroEndpoint.sol";
import { ILayerZeroReceiver } from "src/vendor/layerzero/ILayerZeroReceiver.sol";
import { ILayerZeroUserApplicationConfig } from "src/vendor/layerzero/ILayerZeroUserApplicationConfig.sol";
/// @title LayerzeroImplementation
/// @dev Allows state registries to use Layerzero for crosschain communication
/// @author Zeropoint Labs
contract LayerzeroImplementation is IAmbImplementation, ILayerZeroUserApplicationConfig, ILayerZeroReceiver {
using DataLib for uint256;
using ProofLib for AMBMessage;
//////////////////////////////////////////////////////////////
// CONSTANTS //
//////////////////////////////////////////////////////////////
ISuperRegistry public immutable superRegistry;
//////////////////////////////////////////////////////////////
// STATE VARIABLES //
//////////////////////////////////////////////////////////////
ILayerZeroEndpoint public lzEndpoint;
mapping(bytes32 => bool) public ambProtect;
mapping(uint64 => uint16) public ambChainId;
mapping(uint16 => uint64) public superChainId;
mapping(uint16 => bytes) public trustedRemoteLookup;
mapping(uint16 => mapping(bytes => mapping(uint64 => bytes32))) public failedMessages;
//////////////////////////////////////////////////////////////
// EVENTS //
//////////////////////////////////////////////////////////////
event EndpointUpdated(address indexed oldEndpoint_, address indexed newEndpoint_);
event MessageFailed(uint16 indexed srcChainId_, bytes srcAddress_, uint64 indexed nonce_, bytes payload_);
event SetTrustedRemote(uint16 indexed srcChainId_, bytes srcAddress_);
//////////////////////////////////////////////////////////////
// MODIFIERS //
//////////////////////////////////////////////////////////////
modifier onlyProtocolAdmin() {
if (!ISuperRBAC(superRegistry.getAddress(keccak256("SUPER_RBAC"))).hasProtocolAdminRole(msg.sender)) {
revert Error.NOT_PROTOCOL_ADMIN();
}
_;
}
modifier onlyEmergencyAdmin() {
if (!ISuperRBAC(superRegistry.getAddress(keccak256("SUPER_RBAC"))).hasEmergencyAdminRole(msg.sender)) {
revert Error.NOT_EMERGENCY_ADMIN();
}
_;
}
modifier onlyValidStateRegistry() {
if (!superRegistry.isValidStateRegistry(msg.sender)) {
revert Error.NOT_STATE_REGISTRY();
}
_;
}
modifier onlyLzEndpoint() {
// lzReceive must be called by the endpoint for security
if (msg.sender != address(lzEndpoint)) {
revert Error.CALLER_NOT_ENDPOINT();
}
_;
}
//////////////////////////////////////////////////////////////
// CONSTRUCTOR //
//////////////////////////////////////////////////////////////
/// @param superRegistry_ is the super registry address
constructor(ISuperRegistry superRegistry_) {
superRegistry = superRegistry_;
}
//////////////////////////////////////////////////////////////
// CONFIG //
//////////////////////////////////////////////////////////////
/// @dev allows protocol admin to configure layerzero endpoint
/// @param endpoint_ is the layerzero endpoint on the deployed network
function setLzEndpoint(address endpoint_) external onlyProtocolAdmin {
if (endpoint_ == address(0)) revert Error.ZERO_ADDRESS();
if (address(lzEndpoint) == address(0)) {
lzEndpoint = ILayerZeroEndpoint(endpoint_);
emit EndpointUpdated(address(0), endpoint_);
}
}
/// @dev gets the configuration of the current LayerZero implementation supported
/// @param version_ is the messaging library version
/// @param chainId_ is the layerzero chainId for the pending config change
/// @param configType_ is the type of configuration
function getConfig(
uint16 version_,
uint16 chainId_,
address,
uint256 configType_
)
external
view
returns (bytes memory)
{
return lzEndpoint.getConfig(version_, chainId_, address(this), configType_);
}
/// @inheritdoc ILayerZeroUserApplicationConfig
function setConfig(
uint16 version_,
uint16 chainId_,
uint256 configType_,
bytes calldata config_
)
external
override
onlyProtocolAdmin
{
lzEndpoint.setConfig(version_, chainId_, configType_, config_);
}
/// @inheritdoc ILayerZeroUserApplicationConfig
function setSendVersion(uint16 version_) external override onlyProtocolAdmin {
lzEndpoint.setSendVersion(version_);
}
/// @inheritdoc ILayerZeroUserApplicationConfig
function setReceiveVersion(uint16 version_) external override onlyProtocolAdmin {
lzEndpoint.setReceiveVersion(version_);
}
/// @inheritdoc ILayerZeroUserApplicationConfig
function forceResumeReceive(uint16 srcChainId_, bytes calldata srcAddress_) external override onlyEmergencyAdmin {
lzEndpoint.forceResumeReceive(srcChainId_, srcAddress_);
}
/// @dev allows protocol admin to set contract which can receive messages
/// @param srcChainId_ is the layerzero source chain id
/// @param srcAddress_ is the address to set as the trusted remote on the source chain
function setTrustedRemote(uint16 srcChainId_, bytes calldata srcAddress_) external onlyProtocolAdmin {
trustedRemoteLookup[srcChainId_] = srcAddress_;
emit SetTrustedRemote(srcChainId_, srcAddress_);
}
//////////////////////////////////////////////////////////////
// EXTERNAL VIEW FUNCTIONS //
//////////////////////////////////////////////////////////////
/// @dev checks if another contract on another chain is a trusted remote
/// @param srcChainId_ is the layerzero source chain id
/// @param srcAddress_ is the address to check
function isTrustedRemote(uint16 srcChainId_, bytes calldata srcAddress_) external view returns (bool) {
if (srcChainId_ == 0) {
revert Error.INVALID_CHAIN_ID();
}
return keccak256(trustedRemoteLookup[srcChainId_]) == keccak256(srcAddress_);
}
/// @inheritdoc IAmbImplementation
function estimateFees(
uint64 dstChainId_,
bytes memory message_,
bytes memory extraData_
)
external
view
override
returns (uint256 fees)
{
uint16 chainId = ambChainId[dstChainId_];
if (chainId == 0) {
revert Error.INVALID_CHAIN_ID();
}
(fees,) = lzEndpoint.estimateFees(chainId, address(this), message_, false, extraData_);
}
/// @inheritdoc IAmbImplementation
function generateExtraData(uint256 gasLimit) external pure override returns (bytes memory extraData) {
/// @notice encoded layerzero adapter params (version 2). Other values are not used atm.
extraData = abi.encodePacked(uint16(2), gasLimit, uint256(0), address(0));
}
//////////////////////////////////////////////////////////////
// EXTERNAL WRITE FUNCTIONS //
//////////////////////////////////////////////////////////////
/// @inheritdoc IAmbImplementation
function dispatchPayload(
address srcSender_,
uint64 dstChainId_,
bytes memory message_,
bytes memory extraData_
)
external
payable
override
onlyValidStateRegistry
{
uint16 domain = ambChainId[dstChainId_];
if (domain == 0) {
revert Error.INVALID_CHAIN_ID();
}
_lzSend(domain, message_, payable(srcSender_), address(0x0), extraData_, msg.value);
}
/// @inheritdoc IAmbImplementation
function retryPayload(bytes memory data_) external payable override {
(uint16 srcChainId, bytes memory srcAddress, bytes memory payload) = abi.decode(data_, (uint16, bytes, bytes));
lzEndpoint.retryPayload(srcChainId, srcAddress, payload);
}
/// @dev allows protocol admin to add new chain ids in future
/// @param superChainId_ is the identifier of the chain within superform protocol
/// @param ambChainId_ is the identifier of the chain given by the AMB
/// NOTE: cannot be defined in an interface as types vary for each message bridge (amb)
function setChainId(uint64 superChainId_, uint16 ambChainId_) external onlyProtocolAdmin {
if (superChainId_ == 0 || ambChainId_ == 0) {
revert Error.INVALID_CHAIN_ID();
}
/// @dev reset old mappings
uint64 oldSuperChainId = superChainId[ambChainId_];
uint16 oldAmbChainId = ambChainId[superChainId_];
if (oldSuperChainId != 0) {
delete ambChainId[oldSuperChainId];
}
if (oldAmbChainId != 0) {
delete superChainId[oldAmbChainId];
}
ambChainId[superChainId_] = ambChainId_;
superChainId[ambChainId_] = superChainId_;
emit ChainAdded(superChainId_);
}
/// @inheritdoc ILayerZeroReceiver
function lzReceive(
uint16 srcChainId_,
bytes memory srcAddress_,
uint64 nonce_,
bytes memory payload_
)
public
override
onlyLzEndpoint
{
/// @dev decodes payload received and check payload
AMBMessage memory decoded = abi.decode(payload_, (AMBMessage));
_ambProtect(decoded);
bytes memory trustedRemote = trustedRemoteLookup[srcChainId_];
if (
!(
srcAddress_.length == trustedRemote.length && keccak256(srcAddress_) == keccak256(trustedRemote)
&& trustedRemote.length != 0
)
) {
revert Error.INVALID_SRC_SENDER();
}
_blockingLzReceive(srcChainId_, srcAddress_, nonce_, payload_);
}
/// @dev receive failed messages on this Endpoint destination
/// @param srcChainId_ is the layerzero source chain id
/// @param srcAddress_ is the address on the source chain
/// @param payload_ is the payload to store
function nonblockingLzReceive(uint16 srcChainId_, bytes memory srcAddress_, bytes memory payload_) public {
// only internal transaction
if (msg.sender != address(this)) {
revert Error.INVALID_INTERNAL_CALL();
}
_nonblockingLzReceive(srcChainId_, srcAddress_, payload_);
}
/// @dev retry failed messages on this Endpoint destination
/// @param srcChainId_ is the layerzero source chain id
/// @param srcAddress_ is the address on the source chain
/// @param nonce_ is the sequential location of the stuck payload
/// @param payload_ is the payload to retry
function retryMessage(
uint16 srcChainId_,
bytes memory srcAddress_,
uint64 nonce_,
bytes memory payload_
)
public
payable
{
// assert there is message to retry
bytes32 payloadHash = failedMessages[srcChainId_][srcAddress_][nonce_];
if (payloadHash == bytes32(0)) {
revert Error.ZERO_PAYLOAD_HASH();
}
if (keccak256(payload_) != payloadHash) {
revert Error.INVALID_PAYLOAD_HASH();
}
// clear the stored message
failedMessages[srcChainId_][srcAddress_][nonce_] = bytes32(0);
// execute the message. revert if it fails again
_nonblockingLzReceive(srcChainId_, srcAddress_, payload_);
}
//////////////////////////////////////////////////////////////
// INTERNAL FUNCTIONS //
//////////////////////////////////////////////////////////////
function _nonblockingLzReceive(uint16 _srcChainId, bytes memory, bytes memory _payload) internal {
/// @dev decodes payload received
AMBMessage memory decoded = abi.decode(_payload, (AMBMessage));
/// NOTE: experimental split of registry contracts
(,,, uint8 registryId,,) = decoded.txInfo.decodeTxInfo();
IBaseStateRegistry targetRegistry = IBaseStateRegistry(superRegistry.getStateRegistry(registryId));
uint64 srcChainId = superChainId[_srcChainId];
if (srcChainId == 0) {
revert Error.INVALID_CHAIN_ID();
}
targetRegistry.receivePayload(srcChainId, _payload);
}
function _lzSend(
uint16 dstChainId_,
bytes memory payload_,
address payable receiverAddress_,
address zroPaymentAddress_,
bytes memory adapterParams_,
uint256 msgValue_
)
internal
{
bytes memory trustedRemote = trustedRemoteLookup[dstChainId_];
if (trustedRemote.length == 0) {
revert Error.INVALID_CHAIN_ID();
}
lzEndpoint.send{ value: msgValue_ }(
dstChainId_, trustedRemote, payload_, receiverAddress_, zroPaymentAddress_, adapterParams_
);
}
function _blockingLzReceive(
uint16 srcChainId_,
bytes memory srcAddress_,
uint64 nonce_,
bytes memory payload_
)
internal
{
// try-catch all errors/exceptions
try this.nonblockingLzReceive(srcChainId_, srcAddress_, payload_) {
// do nothing
} catch {
// error / exception
failedMessages[srcChainId_][srcAddress_][nonce_] = keccak256(payload_);
emit MessageFailed(srcChainId_, srcAddress_, nonce_, payload_);
}
}
/// @dev prevents the same AMB from delivery a payload and its proof
/// @dev is an additional protection against malicious ambs
function _ambProtect(AMBMessage memory _message) internal {
bytes32 proof;
/// @dev amb protect
if (_message.params.length != 32) {
(, bytes memory payloadBody) = abi.decode(_message.params, (uint8[], bytes));
proof = AMBMessage(_message.txInfo, payloadBody).computeProof();
} else {
proof = abi.decode(_message.params, (bytes32));
}
if (ambProtect[proof]) revert MALICIOUS_DELIVERY();
ambProtect[proof] = true;
}
}