编译器
0.8.19+commit.7dd6d404
文件 1 的 4:IDiamondCut.sol
pragma solidity ^0.8.19;
interface IDiamondCut {
enum FacetCutAction {
Add,
Replace,
Remove
}
struct FacetCut {
address facetAddress;
FacetCutAction action;
bytes4[] functionSelectors;
}
function diamondCut(FacetCut[] calldata _diamondCut, address _init, bytes calldata _calldata) external;
event DiamondCut(FacetCut[] _diamondCut, address _init, bytes _calldata);
}
文件 2 的 4:LGDiamond.sol
pragma solidity ^0.8.19;
import '../interfaces/IDiamondCut.sol';
import '../libraries/LibContractOwner.sol';
import '../libraries/LibDiamond.sol';
contract Diamond {
error FunctionDoesNotExist(bytes4 methodSelector);
error DiamondAlreadyInitialized();
constructor(address diamondCutFacet) payable {
initializeDiamond(diamondCutFacet);
}
fallback() external payable {
address facet = LibDiamond.diamondStorage().selectorToFacetAndPosition[msg.sig].facetAddress;
if (facet == address(0)) revert FunctionDoesNotExist(msg.sig);
assembly {
calldatacopy(0, 0, calldatasize())
let result := delegatecall(gas(), facet, 0, calldatasize(), 0, 0)
returndatacopy(0, 0, returndatasize())
switch result
case 0 {
revert(0, returndatasize())
}
default {
return(0, returndatasize())
}
}
}
function initializeDiamond(address diamondCutFacet) public {
LibDiamond.DiamondStorage storage ds = LibDiamond.diamondStorage();
if (ds.initialized) revert DiamondAlreadyInitialized();
IDiamondCut.FacetCut[] memory cut = new IDiamondCut.FacetCut[](1);
cut[0] = IDiamondCut.FacetCut({
facetAddress: diamondCutFacet,
action: IDiamondCut.FacetCutAction.Add,
functionSelectors: new bytes4[](1)
});
cut[0].functionSelectors[0] = IDiamondCut.diamondCut.selector;
LibDiamond.diamondCut(cut);
LibContractOwner.setContractOwner(msg.sender);
ds.initialized = true;
}
receive() external payable {}
}
文件 3 的 4:LibContractOwner.sol
pragma solidity ^0.8.19;
library LibContractOwner {
error CallerIsNotContractOwner();
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
event AdminChanged(address previousAdmin, address newAdmin);
bytes32 private constant ADMIN_SLOT_POSITION = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
struct LibOwnerStorage {
address contractOwner;
}
function ownerStorage() internal pure returns (LibOwnerStorage storage storageSlot) {
bytes32 position = ADMIN_SLOT_POSITION;
assembly {
storageSlot.slot := position
}
}
function setContractOwner(address newOwner) internal {
LibOwnerStorage storage ls = ownerStorage();
address previousOwner = ls.contractOwner;
ls.contractOwner = newOwner;
emit OwnershipTransferred(previousOwner, newOwner);
emit AdminChanged(previousOwner, newOwner);
}
function contractOwner() internal view returns (address owner) {
owner = ownerStorage().contractOwner;
}
function enforceIsContractOwner() internal view {
if (msg.sender != ownerStorage().contractOwner) revert CallerIsNotContractOwner();
}
}
文件 4 的 4:LibDiamond.sol
pragma solidity ^0.8.19;
import '../interfaces/IDiamondCut.sol';
import './LibContractOwner.sol';
library LibDiamond {
error InvalidFacetCutAction(IDiamondCut.FacetCutAction action);
event DiamondCut(IDiamondCut.FacetCut[] _diamondCut, address _init, bytes _calldata);
bytes32 internal constant DIAMOND_STORAGE_POSITION =
0xc8fcad8db84d3cc18b4c41d551ea0ee66dd599cde068d998e57d5e09332c131c;
struct FacetAddressAndPosition {
address facetAddress;
uint96 functionSelectorPosition;
}
struct FacetFunctionSelectors {
bytes4[] functionSelectors;
uint256 facetAddressPosition;
}
struct DiamondStorage {
mapping(bytes4 => FacetAddressAndPosition) selectorToFacetAndPosition;
mapping(address => FacetFunctionSelectors) facetFunctionSelectors;
address[] facetAddresses;
bool initialized;
}
function diamondStorage() internal pure returns (DiamondStorage storage ds) {
bytes32 position = DIAMOND_STORAGE_POSITION;
assembly {
ds.slot := position
}
}
function enforceIsContractOwner() internal view {
LibContractOwner.enforceIsContractOwner();
}
function diamondCut(IDiamondCut.FacetCut[] memory _diamondCut) internal {
for (uint256 facetIndex; facetIndex < _diamondCut.length; ++facetIndex) {
IDiamondCut.FacetCutAction action = _diamondCut[facetIndex].action;
if (action == IDiamondCut.FacetCutAction.Add) {
addFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
} else if (action == IDiamondCut.FacetCutAction.Replace) {
replaceFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
} else if (action == IDiamondCut.FacetCutAction.Remove) {
removeFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
} else {
revert InvalidFacetCutAction(action);
}
}
emit DiamondCut(_diamondCut, address(0), '');
}
function addFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
require(_functionSelectors.length > 0, 'LibDiamondCut: No selectors in facet to cut');
DiamondStorage storage ds = diamondStorage();
require(_facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)");
uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
if (selectorPosition == 0) {
addFacet(ds, _facetAddress);
}
for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; ++selectorIndex) {
bytes4 selector = _functionSelectors[selectorIndex];
address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
require(oldFacetAddress == address(0), "LibDiamondCut: Can't add function that already exists");
addFunction(ds, selector, selectorPosition, _facetAddress);
++selectorPosition;
}
}
function replaceFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
require(_functionSelectors.length > 0, 'LibDiamondCut: No selectors in facet to cut');
DiamondStorage storage ds = diamondStorage();
require(_facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)");
uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
if (selectorPosition == 0) {
addFacet(ds, _facetAddress);
}
for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; ++selectorIndex) {
bytes4 selector = _functionSelectors[selectorIndex];
address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
require(oldFacetAddress != _facetAddress, "LibDiamondCut: Can't replace function with same function");
removeFunction(ds, oldFacetAddress, selector);
addFunction(ds, selector, selectorPosition, _facetAddress);
++selectorPosition;
}
}
function removeFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
require(_functionSelectors.length > 0, 'LibDiamondCut: No selectors in facet to cut');
DiamondStorage storage ds = diamondStorage();
require(_facetAddress == address(0), 'LibDiamondCut: Remove facet address must be address(0)');
for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; ++selectorIndex) {
bytes4 selector = _functionSelectors[selectorIndex];
address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
removeFunction(ds, oldFacetAddress, selector);
}
}
function addFacet(DiamondStorage storage ds, address _facetAddress) internal {
enforceHasContractCode(_facetAddress, 'LibDiamondCut: New facet has no code');
ds.facetFunctionSelectors[_facetAddress].facetAddressPosition = ds.facetAddresses.length;
ds.facetAddresses.push(_facetAddress);
}
function addFunction(
DiamondStorage storage ds,
bytes4 _selector,
uint96 _selectorPosition,
address _facetAddress
) internal {
ds.selectorToFacetAndPosition[_selector].functionSelectorPosition = _selectorPosition;
ds.facetFunctionSelectors[_facetAddress].functionSelectors.push(_selector);
ds.selectorToFacetAndPosition[_selector].facetAddress = _facetAddress;
}
function removeFunction(DiamondStorage storage ds, address _facetAddress, bytes4 _selector) internal {
require(_facetAddress != address(0), "LibDiamondCut: Can't remove function that doesn't exist");
require(_facetAddress != address(this), "LibDiamondCut: Can't remove immutable function");
uint256 selectorPosition = ds.selectorToFacetAndPosition[_selector].functionSelectorPosition;
uint256 lastSelectorPosition = ds.facetFunctionSelectors[_facetAddress].functionSelectors.length - 1;
if (selectorPosition != lastSelectorPosition) {
bytes4 lastSelector = ds.facetFunctionSelectors[_facetAddress].functionSelectors[lastSelectorPosition];
ds.facetFunctionSelectors[_facetAddress].functionSelectors[selectorPosition] = lastSelector;
ds.selectorToFacetAndPosition[lastSelector].functionSelectorPosition = uint96(selectorPosition);
}
ds.facetFunctionSelectors[_facetAddress].functionSelectors.pop();
delete ds.selectorToFacetAndPosition[_selector];
if (lastSelectorPosition == 0) {
uint256 lastFacetAddressPosition = ds.facetAddresses.length - 1;
uint256 facetAddressPosition = ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
if (facetAddressPosition != lastFacetAddressPosition) {
address lastFacetAddress = ds.facetAddresses[lastFacetAddressPosition];
ds.facetAddresses[facetAddressPosition] = lastFacetAddress;
ds.facetFunctionSelectors[lastFacetAddress].facetAddressPosition = facetAddressPosition;
}
ds.facetAddresses.pop();
delete ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
}
}
function enforceHasContractCode(address _contract, string memory _errorMessage) internal view {
uint256 contractSize;
assembly {
contractSize := extcodesize(_contract)
}
require(contractSize > 0, _errorMessage);
}
}
{
"compilationTarget": {
"lib/@cu-tokens/lib/@lagunagames/lg-diamond-template/src/diamond/LGDiamond.sol": "Diamond"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"details": {
"constantOptimizer": true,
"cse": true,
"deduplicate": true,
"inliner": true,
"jumpdestRemover": true,
"orderLiterals": true,
"peephole": true,
"yul": true,
"yulDetails": {
"optimizerSteps": "dhfoDgvulfnTUtnIf:fDnTOc",
"stackAllocation": true
}
},
"runs": 999999
},
"remappings": [
":@cu-common/=lib/@cu-tokens/lib/@lagunagames/cu-common/",
":@cu-tokens/=lib/@cu-tokens/",
":@lagunagames/=lib/@lagunagames/",
":@lagunagames/lg-diamond-template/=lib/@cu-tokens/lib/@lagunagames/lg-diamond-template/",
":@layerzerolabs/=lib/@layerzerolabs/",
":@lg-arb-bridge/=lib/@lg-arb-bridge/src/",
":@lg-diamond-template/=lib/@cu-tokens/lib/@lagunagames/lg-diamond-template/",
":@lg-layerzero/=lib/@lg-layerzero/src/",
":@openzeppelin-contracts/=lib/@cu-tokens/lib/openzeppelin-contracts/",
":@openzeppelin/=lib/@openzeppelin/",
":ds-test/=lib/@cu-tokens/lib/forge-std/lib/ds-test/src/",
":erc4626-tests/=lib/@openzeppelin/lib/erc4626-tests/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/@cu-tokens/lib/openzeppelin-contracts/",
":openzeppelin/=lib/@openzeppelin/contracts/",
":solidity-stringutils/=lib/@lagunagames/lg-diamond-template/lib/solidity-stringutils/"
],
"viaIR": true
}
[{"inputs":[{"internalType":"address","name":"diamondCutFacet","type":"address"}],"stateMutability":"payable","type":"constructor"},{"inputs":[],"name":"DiamondAlreadyInitialized","type":"error"},{"inputs":[{"internalType":"bytes4","name":"methodSelector","type":"bytes4"}],"name":"FunctionDoesNotExist","type":"error"},{"inputs":[{"internalType":"enum IDiamondCut.FacetCutAction","name":"action","type":"uint8"}],"name":"InvalidFacetCutAction","type":"error"},{"stateMutability":"payable","type":"fallback"},{"inputs":[{"internalType":"address","name":"diamondCutFacet","type":"address"}],"name":"initializeDiamond","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]