@@ -7,6 +7,7 @@ import "wormhole-solidity-sdk/interfaces/IWormhole.sol";
7
7
8
8
import "./libraries/EndpointHelpers.sol " ;
9
9
import "./interfaces/IWormholeEndpoint.sol " ;
10
+ import "./interfaces/ISpecialRelayer.sol " ;
10
11
import "./Endpoint.sol " ;
11
12
12
13
abstract contract WormholeEndpoint is Endpoint , IWormholeEndpoint , IWormholeReceiver {
@@ -24,6 +25,7 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
24
25
25
26
IWormhole public immutable wormhole;
26
27
IWormholeRelayer public immutable wormholeRelayer;
28
+ ISpecialRelayer public immutable specialRelayer;
27
29
uint256 public immutable wormholeEndpoint_evmChainId;
28
30
29
31
struct WormholeEndpointInstruction {
@@ -41,6 +43,9 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
41
43
bytes32 public constant WORMHOLE_RELAYING_ENABLED_CHAINS_SLOT =
42
44
bytes32 (uint256 (keccak256 ("whEndpoint.relayingEnabledChains " )) - 1 );
43
45
46
+ bytes32 public constant SPECIAL_RELAYING_ENABLED_CHAINS_SLOT =
47
+ bytes32 (uint256 (keccak256 ("whEndpoint.specialRelayingEnabledChains " )) - 1 );
48
+
44
49
bytes32 public constant WORMHOLE_EVM_CHAIN_IDS =
45
50
bytes32 (uint256 (keccak256 ("whEndpoint.evmChainIds " )) - 1 );
46
51
@@ -71,18 +76,29 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
71
76
function _getWormholeRelayingEnabledChainsStorage ()
72
77
internal
73
78
pure
74
- returns (mapping (uint16 => bool ) storage $)
79
+ returns (mapping (uint16 => uint256 ) storage $)
75
80
{
76
81
uint256 slot = uint256 (WORMHOLE_RELAYING_ENABLED_CHAINS_SLOT);
77
82
assembly ("memory-safe" ) {
78
83
$.slot := slot
79
84
}
80
85
}
81
86
87
+ function _getSpecialRelayingEnabledChainsStorage ()
88
+ internal
89
+ pure
90
+ returns (mapping (uint16 => uint256 ) storage $)
91
+ {
92
+ uint256 slot = uint256 (SPECIAL_RELAYING_ENABLED_CHAINS_SLOT);
93
+ assembly ("memory-safe" ) {
94
+ $.slot := slot
95
+ }
96
+ }
97
+
82
98
function _getWormholeEvmChainIdsStorage ()
83
99
internal
84
100
pure
85
- returns (mapping (uint16 => bool ) storage $)
101
+ returns (mapping (uint16 => uint256 ) storage $)
86
102
{
87
103
uint256 slot = uint256 (WORMHOLE_EVM_CHAIN_IDS);
88
104
assembly ("memory-safe" ) {
@@ -97,17 +113,22 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
97
113
_;
98
114
}
99
115
100
- constructor (address wormholeCoreBridge , address wormholeRelayerAddr ) {
116
+ constructor (
117
+ address wormholeCoreBridge ,
118
+ address wormholeRelayerAddr ,
119
+ address specialRelayerAddr
120
+ ) {
101
121
wormhole = IWormhole (wormholeCoreBridge);
102
122
wormholeRelayer = IWormholeRelayer (wormholeRelayerAddr);
123
+ specialRelayer = ISpecialRelayer (specialRelayerAddr);
103
124
wormholeEndpoint_evmChainId = block .chainid ;
104
125
}
105
126
106
- function checkInvalidRelayingConfig (uint16 chainId ) internal view returns (bool ) {
127
+ function _checkInvalidRelayingConfig (uint16 chainId ) internal view returns (bool ) {
107
128
return isWormholeRelayingEnabled (chainId) && ! isWormholeEvmChain (chainId);
108
129
}
109
130
110
- function shouldRelayViaStandardRelaying (uint16 chainId ) internal view returns (bool ) {
131
+ function _shouldRelayViaStandardRelaying (uint16 chainId ) internal view returns (bool ) {
111
132
return isWormholeRelayingEnabled (chainId) && isWormholeEvmChain (chainId);
112
133
}
113
134
@@ -122,13 +143,16 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
122
143
return 0 ;
123
144
}
124
145
125
- if (checkInvalidRelayingConfig (targetChain)) {
146
+ if (_checkInvalidRelayingConfig (targetChain)) {
126
147
revert InvalidRelayingConfig (targetChain);
127
148
}
128
149
129
- if (shouldRelayViaStandardRelaying (targetChain)) {
150
+ if (_shouldRelayViaStandardRelaying (targetChain)) {
130
151
(uint256 cost ,) = wormholeRelayer.quoteEVMDeliveryPrice (targetChain, 0 , GAS_LIMIT);
131
152
return cost;
153
+ } else if (isSpecialRelayingEnabled (targetChain)) {
154
+ uint256 cost = specialRelayer.quoteDeliveryPrice (getManagerToken (), targetChain, 0 );
155
+ return cost;
132
156
} else {
133
157
return 0 ;
134
158
}
@@ -151,14 +175,19 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
151
175
WormholeEndpointInstruction memory weIns =
152
176
parseWormholeEndpointInstruction (instruction.payload);
153
177
154
- if (! weIns.shouldSkipRelayerSend && shouldRelayViaStandardRelaying (recipientChain)) {
178
+ if (! weIns.shouldSkipRelayerSend && _shouldRelayViaStandardRelaying (recipientChain)) {
155
179
wormholeRelayer.sendPayloadToEvm {value: deliveryPayment}(
156
180
recipientChain,
157
181
fromWormholeFormat (getWormholeSibling (recipientChain)),
158
182
encodedEndpointPayload,
159
183
0 ,
160
184
GAS_LIMIT
161
185
);
186
+ } else if (! weIns.shouldSkipRelayerSend && isSpecialRelayingEnabled (recipientChain)) {
187
+ uint64 sequence = wormhole.publishMessage (0 , encodedEndpointPayload, CONSISTENCY_LEVEL);
188
+ specialRelayer.requestDelivery {value: deliveryPayment}(
189
+ getManagerToken (), recipientChain, 0 , sequence
190
+ );
162
191
} else {
163
192
wormhole.publishMessage (0 , encodedEndpointPayload, CONSISTENCY_LEVEL);
164
193
}
@@ -285,27 +314,40 @@ abstract contract WormholeEndpoint is Endpoint, IWormholeEndpoint, IWormholeRece
285
314
}
286
315
287
316
function isWormholeRelayingEnabled (uint16 chainId ) public view returns (bool ) {
288
- return _getWormholeRelayingEnabledChainsStorage ()[chainId];
317
+ return toBool ( _getWormholeRelayingEnabledChainsStorage ()[chainId]) ;
289
318
}
290
319
291
320
function _setIsWormholeRelayingEnabled (uint16 chainId , bool isEnabled ) internal {
292
321
if (chainId == 0 ) {
293
322
revert InvalidWormholeChainIdZero ();
294
323
}
295
- _getWormholeRelayingEnabledChainsStorage ()[chainId] = isEnabled;
324
+ _getWormholeRelayingEnabledChainsStorage ()[chainId] = toWord ( isEnabled) ;
296
325
297
326
emit SetIsWormholeRelayingEnabled (chainId, isEnabled);
298
327
}
299
328
329
+ function isSpecialRelayingEnabled (uint16 chainId ) public view returns (bool ) {
330
+ return toBool (_getSpecialRelayingEnabledChainsStorage ()[chainId]);
331
+ }
332
+
333
+ function _setIsSpecialRelayingEnabled (uint16 chainId , bool isEnabled ) internal {
334
+ if (chainId == 0 ) {
335
+ revert InvalidWormholeChainIdZero ();
336
+ }
337
+ _getSpecialRelayingEnabledChainsStorage ()[chainId] = toWord (isEnabled);
338
+
339
+ emit SetIsSpecialRelayingEnabled (chainId, isEnabled);
340
+ }
341
+
300
342
function isWormholeEvmChain (uint16 chainId ) public view returns (bool ) {
301
- return _getWormholeEvmChainIdsStorage ()[chainId];
343
+ return toBool ( _getWormholeEvmChainIdsStorage ()[chainId]) ;
302
344
}
303
345
304
346
function _setIsWormholeEvmChain (uint16 chainId ) internal {
305
347
if (chainId == 0 ) {
306
348
revert InvalidWormholeChainIdZero ();
307
349
}
308
- _getWormholeEvmChainIdsStorage ()[chainId] = true ;
350
+ _getWormholeEvmChainIdsStorage ()[chainId] = TRUE ;
309
351
310
352
emit SetIsWormholeEvmChain (chainId);
311
353
}
0 commit comments