tx · C5h8Grb2u1EhzvHBwWWChXAXPxMCGWd1XQip5tJNx73p 3Mw7mAf2p8drjeSSDEbq6yyFhjqTgDcwGFB: -0.04000000 Waves 2022.05.05 19:33 [2038641] smart account 3Mw7mAf2p8drjeSSDEbq6yyFhjqTgDcwGFB > SELF 0.00000000 Waves
{ "type": 13, "id": "C5h8Grb2u1EhzvHBwWWChXAXPxMCGWd1XQip5tJNx73p", "fee": 4000000, "feeAssetId": null, "timestamp": 1651768427912, "version": 2, "chainId": 84, "sender": "3Mw7mAf2p8drjeSSDEbq6yyFhjqTgDcwGFB", "senderPublicKey": "Ti5K4A5REvfgdfj3SWhhb2wJXRKenVb3WPQ6tE1TaKe", "proofs": [ "XP6btsFXXhZQ5hbvf734k6tC9UAj6f61B2vLwuGiqhNDaQ989oye1ebWhpC1tQESVow2LAUJAmbSWoG6Yq71kg4" ], "script": "base64:AAIFAAAAAAAAABYIAhIDCgEIEgsKCQEBAQEBCAgIARIAAAAADgAAAAACcHMCAAAADVBSRVNBTEVfU1RBUlQAAAAAAnBkAgAAABBQUkVTQUxFX0RVUkFUSU9OAAAAAAJjcwIAAAALQ0xBSU1fU1RBUlQAAAAAAmNkAgAAAA5DTEFJTV9EVVJBVElPTgAAAAABcAIAAAAJUFJJQ0VfSURPAAAAAARwMUlEAgAAABBQUklDRV8xX0FTU0VUX0lEAAAAAARwMklEAgAAABBQUklDRV8yX0FTU0VUX0lEAAAAAARwM0lEAgAAABBQUklDRV8zX0FTU0VUX0lEAAAAAANtaWECAAAAEU1JTl9BTU9VTlRfSU5WRVNUAAAAAAVtYWluYQIAAAAKTUFJTl9BU1NFVAAAAAAHZm9yU2FsZQIAAAAOVE9UQUxfRk9SX1NBTEUAAAAACXRvdGFsU2FsZQIAAAAKVE9UQUxfU0FMRQAAAAAHdGludmVzdAIAAAANVE9UQUxfSU5WRVNUXwEAAAAPZ2V0QXNzZXRJZlZhbGlkAAAAAQAAAAdhc3NldElkCQACWAAAAAEJAQAAABN2YWx1ZU9yRXJyb3JNZXNzYWdlAAAAAggJAQAAAAV2YWx1ZQAAAAEJAAPsAAAAAQkAAlkAAAABBQAAAAdhc3NldElkAAAAAmlkCQABLAAAAAIJAAEsAAAAAgIAAAAKQXNzZXQgSWQ6IAUAAAAHYXNzZXRJZAIAAAAMIGlzIGludmFsaWQuAAAAAwAAAAFpAQAAAAhhZGRBZG1pbgAAAAEAAAAHYWRkcmVzcwMJAAAAAAAAAggFAAAAAWkAAAAGY2FsbGVyBQAAAAR0aGlzCQAETAAAAAIJAQAAAAtCaW5hcnlFbnRyeQAAAAICAAAABWFkbWluCQACWQAAAAEFAAAAB2FkZHJlc3MFAAAAA25pbAkAAAIAAAABAgAAAC5Pbmx5IHRoZSBBZG1pbiBpdHNlbGYgY2FuIGludm9rZSB0aGlzIGZ1bmN0aW9uAAAAAWkBAAAAC2NvbnN0cnVjdG9yAAAACQAAAAxwcmVzYWxlU3RhcnQAAAAPcHJlc2FsZUR1cmF0aW9uAAAACmNsYWltU3RhcnQAAAANY2xhaW1EdXJhdGlvbgAAAAVwcmljZQAAAA9wcmljZTFBc3NldElkNTgAAAAPcHJpY2UyQXNzZXRJZDU4AAAAD3ByaWNlM0Fzc2V0SWQ1OAAAAA9taW5JbnZlc3RBbW91bnQEAAAABmFzc2V0MQkBAAAAD2dldEFzc2V0SWZWYWxpZAAAAAEFAAAAD3ByaWNlMUFzc2V0SWQ1OAQAAAAGYXNzZXQyCQEAAAAPZ2V0QXNzZXRJZlZhbGlkAAAAAQUAAAAPcHJpY2UyQXNzZXRJZDU4BAAAAAZhc3NldDMJAQAAAA9nZXRBc3NldElmVmFsaWQAAAABBQAAAA9wcmljZTNBc3NldElkNTgDAwkAAAAAAAACCAUAAAABaQAAAAZjYWxsZXIFAAAABHRoaXMGCQAAAAAAAAIIBQAAAAFpAAAABmNhbGxlcgkBAAAAB0FkZHJlc3MAAAABCQEAAAARQGV4dHJOYXRpdmUoMTA1MikAAAACBQAAAAR0aGlzAgAAAAVhZG1pbgMJAABnAAAAAgUAAAAPcHJlc2FsZUR1cmF0aW9uBQAAAAxwcmVzYWxlU3RhcnQJAAACAAAAAQIAAAAyUHJlc2FsZUR1cmF0aW9uIGNhbid0IGJlIGJpZ2dlciB0aGFuICBQcmVzYWxlU3RhcnQDCQAAZwAAAAIFAAAAD3ByZXNhbGVEdXJhdGlvbgUAAAAKY2xhaW1TdGFydAkAAAIAAAABAgAAAC9wcmVzYWxlRHVyYXRpb24gY2FuJ3QgYmUgYmlnZ2VyIHRoYW4gY2xhaW1TdGFydAMJAABnAAAAAgUAAAANY2xhaW1EdXJhdGlvbgUAAAAKY2xhaW1TdGFydAkAAAIAAAABAgAAAC5jbGFpbUR1cmF0aW9uIGNhbid0IGJlIGJpZ2dlciB0aGFuICBjbGFpbVN0YXJ0AwkAAGcAAAACBQAAAA9taW5JbnZlc3RBbW91bnQFAAAABXByaWNlCQAAAgAAAAECAAAAKm1pbkludmVzdEFtb3VudCBjYW4ndCBiZSBiaWdnZXIgdGhhbiBwcmljZQMDAwkAAAAAAAACBQAAAAZhc3NldDEFAAAABmFzc2V0MgYJAAAAAAAAAgUAAAAGYXNzZXQxBQAAAAZhc3NldDMGCQAAAAAAAAIFAAAABmFzc2V0MgUAAAAGYXNzZXQzCQAAAgAAAAECAAAALEVhY2ggQXNzZXQgbXVzdCBiZSBkaWZmZXJlbnQgZnJvbSB0aGUgb3RoZXIuAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAQkAAAIAAAABAgAAACJleGFjdGx5IDEgcGF5bWVudCBtdXN0IGJlIGF0dGFjaGVkCQAETAAAAAIJAQAAAAtCaW5hcnlFbnRyeQAAAAIFAAAABW1haW5hCQEAAAAFdmFsdWUAAAABCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAdhc3NldElkCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIFAAAABHAxSUQFAAAAD3ByaWNlMUFzc2V0SWQ1OAkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAARwMklEBQAAAA9wcmljZTJBc3NldElkNTgJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAEcDNJRAUAAAAPcHJpY2UzQXNzZXRJZDU4CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAJwcwUAAAAMcHJlc2FsZVN0YXJ0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAJwZAUAAAAPcHJlc2FsZUR1cmF0aW9uCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAJjcwUAAAAKY2xhaW1TdGFydAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAACY2QFAAAADWNsYWltRHVyYXRpb24JAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAAXAFAAAABXByaWNlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAANtaWEFAAAAD21pbkludmVzdEFtb3VudAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAHZm9yU2FsZQgJAQAAAAV2YWx1ZQAAAAEJAAGRAAAAAggFAAAAAWkAAAAIcGF5bWVudHMAAAAAAAAAAAAAAAAGYW1vdW50CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAl0b3RhbFNhbGUAAAAAAAAAAAAFAAAAA25pbAkAAAIAAAABAgAAAC5Pbmx5IHRoZSBBZG1pbiBpdHNlbGYgY2FuIGludm9rZSB0aGlzIGZ1bmN0aW9uAAAAAWkBAAAABmludmVzdAAAAAAEAAAACW1haW5Bc3NldAkABBwAAAACBQAAAAR0aGlzBQAAAAVtYWluYQQAAAAMcHJlc2FsZVN0YXJ0CQEAAAARQGV4dHJOYXRpdmUoMTA1MCkAAAACBQAAAAR0aGlzBQAAAAJwcwQAAAAPcHJlc2FsZUR1cmF0aW9uCQEAAAARQGV4dHJOYXRpdmUoMTA1MCkAAAACBQAAAAR0aGlzBQAAAAJwZAQAAAAKcHJlc2FsZUVuZAkAAGQAAAACBQAAAAxwcmVzYWxlU3RhcnQFAAAAD3ByZXNhbGVEdXJhdGlvbgQAAAAFcHJpY2UJAQAAABFAZXh0ck5hdGl2ZSgxMDUwKQAAAAIFAAAABHRoaXMFAAAAAXAEAAAAAWgFAAAABmhlaWdodAQAAAADcG10CQEAAAAFdmFsdWUAAAABCQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAABAAAAApwbXRBc3NldElkCQACWAAAAAEJAQAAAAV2YWx1ZQAAAAEIBQAAAANwbXQAAAAHYXNzZXRJZAQAAAAJcG10QW1vdW50CAUAAAADcG10AAAABmFtb3VudAQAAAAGYXNzZXQxCQEAAAARQGV4dHJOYXRpdmUoMTA1MykAAAACBQAAAAR0aGlzBQAAAARwMUlEBAAAAAZhc3NldDIJAQAAABFAZXh0ck5hdGl2ZSgxMDUzKQAAAAIFAAAABHRoaXMFAAAABHAySUQEAAAABmFzc2V0MwkBAAAAEUBleHRyTmF0aXZlKDEwNTMpAAAAAgUAAAAEdGhpcwUAAAAEcDNJRAQAAAAKdG90YWxTYWxlZQkBAAAAEUBleHRyTmF0aXZlKDEwNTApAAAAAgUAAAAEdGhpcwUAAAAJdG90YWxTYWxlBAAAAAhmb3JTYWxlZQkBAAAAEUBleHRyTmF0aXZlKDEwNTApAAAAAgUAAAAEdGhpcwUAAAAHZm9yU2FsZQQAAAAJbWluSW52ZXN0CQEAAAARQGV4dHJOYXRpdmUoMTA1MCkAAAACBQAAAAR0aGlzBQAAAANtaWEDCQAAZwAAAAIFAAAADHByZXNhbGVTdGFydAUAAAABaAkAAAIAAAABAgAAACBwcmVzYWxlIGhhcyBub3QgYmVlbiBzdGFydGVkIHlldAMJAABnAAAAAgUAAAABaAUAAAAKcHJlc2FsZUVuZAkAAAIAAAABAgAAAB5wcmVzYWxlIGhhcyBiZWVuIGFscmVhZHkgZW5kZWQDCQEAAAACIT0AAAACCQABkAAAAAEIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAABCQAAAgAAAAECAAAAHWV4YWN0bHkgMSBwYXltZW50IGlzIGV4cGVjdGVkAwMDCQEAAAACIT0AAAACBQAAAApwbXRBc3NldElkBQAAAAZhc3NldDEJAQAAAAIhPQAAAAIFAAAACnBtdEFzc2V0SWQFAAAABmFzc2V0MgcJAQAAAAIhPQAAAAIFAAAACnBtdEFzc2V0SWQFAAAABmFzc2V0MwcJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAZaW52YWxpZCBwYXltZW50IGFzc2V0IGlkOgUAAAAKcG10QXNzZXRJZAIAAAAKIEV4cGVjdGVkOgUAAAAGYXNzZXQxAwkAAGcAAAACBQAAAAp0b3RhbFNhbGVlBQAAAAhmb3JTYWxlZQkAAAIAAAABAgAAAD5QcmUtc2FsZSBhc3NldCBoYXMgYmVlbiAtIHNvbGQgY29uc2lkZXIgdG8gdXNlIHNtYWxsZXIgcGF5bWVudAMJAABmAAAAAgUAAAAJbWluSW52ZXN0BQAAAAlwbXRBbW91bnQJAAACAAAAAQIAAAAsQW1vdW50IGxlc3MgdGhhbiB0aGUgbWluaW11bSB0byBiZSBpbnZlc3RlZC4EAAAACmdldFRJbnZlc3QJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAQAAABFAZXh0ck5hdGl2ZSgxMDUwKQAAAAIFAAAABHRoaXMJAAEsAAAAAgUAAAAHdGludmVzdAkAAlgAAAABCAgFAAAAAWkAAAAGY2FsbGVyAAAABWJ5dGVzAAAAAAAAAAAABAAAAARjYWxjCQAAaQAAAAIFAAAACXBtdEFtb3VudAUAAAAFcHJpY2UEAAAAC3RvdGFsQW1vdW50CQAAZAAAAAIFAAAACXBtdEFtb3VudAUAAAAKZ2V0VEludmVzdAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAAd0aW52ZXN0CQACWAAAAAEICAUAAAABaQAAAAZjYWxsZXIAAAAFYnl0ZXMFAAAAC3RvdGFsQW1vdW50CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAl0b3RhbFNhbGUJAABkAAAAAgUAAAAKdG90YWxTYWxlZQUAAAAEY2FsYwkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAABGNhbGMFAAAACW1haW5Bc3NldAkABEwAAAACCQEAAAAMQm9vbGVhbkVudHJ5AAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgUAAAAHdGludmVzdAkAAlgAAAABCAgFAAAAAWkAAAAGY2FsbGVyAAAABWJ5dGVzAgAAAAJfXwUAAAAKcG10QXNzZXRJZAYFAAAAA25pbAAAAAEAAAACdHgBAAAABnZlcmlmeQAAAAAJAAH0AAAAAwgFAAAAAnR4AAAACWJvZHlCeXRlcwkAAZEAAAACCAUAAAACdHgAAAAGcHJvb2ZzAAAAAAAAAAAACAUAAAACdHgAAAAPc2VuZGVyUHVibGljS2V52/4pVQ==", "height": 2038641, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: r3PrWhbYEtmByewYPc9EQ54C9ZcnEzjJRHbJpPd5ojg Next: DRcowy2N1fHFAcv72zf7nAZoFUqzVtX6SsPBRpcHhFFr Diff:
Old | New | Differences | |
---|---|---|---|
43 | 43 | let asset2 = getAssetIfValid(price2AssetId58) | |
44 | 44 | let asset3 = getAssetIfValid(price3AssetId58) | |
45 | 45 | if (if ((i.caller == this)) | |
46 | - | then | |
47 | - | else | |
46 | + | then true | |
47 | + | else (i.caller == Address(getBinaryValue(this, "admin")))) | |
48 | 48 | then if ((presaleDuration >= presaleStart)) | |
49 | 49 | then throw("PresaleDuration can't be bigger than PresaleStart") | |
50 | 50 | else if ((presaleDuration >= claimStart)) |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let ps = "PRESALE_START" | |
5 | 5 | ||
6 | 6 | let pd = "PRESALE_DURATION" | |
7 | 7 | ||
8 | 8 | let cs = "CLAIM_START" | |
9 | 9 | ||
10 | 10 | let cd = "CLAIM_DURATION" | |
11 | 11 | ||
12 | 12 | let p = "PRICE_IDO" | |
13 | 13 | ||
14 | 14 | let p1ID = "PRICE_1_ASSET_ID" | |
15 | 15 | ||
16 | 16 | let p2ID = "PRICE_2_ASSET_ID" | |
17 | 17 | ||
18 | 18 | let p3ID = "PRICE_3_ASSET_ID" | |
19 | 19 | ||
20 | 20 | let mia = "MIN_AMOUNT_INVEST" | |
21 | 21 | ||
22 | 22 | let maina = "MAIN_ASSET" | |
23 | 23 | ||
24 | 24 | let forSale = "TOTAL_FOR_SALE" | |
25 | 25 | ||
26 | 26 | let totalSale = "TOTAL_SALE" | |
27 | 27 | ||
28 | 28 | let tinvest = "TOTAL_INVEST_" | |
29 | 29 | ||
30 | 30 | func getAssetIfValid (assetId) = toBase58String(valueOrErrorMessage(value(assetInfo(fromBase58String(assetId))).id, (("Asset Id: " + assetId) + " is invalid."))) | |
31 | 31 | ||
32 | 32 | ||
33 | 33 | @Callable(i) | |
34 | 34 | func addAdmin (address) = if ((i.caller == this)) | |
35 | 35 | then [BinaryEntry("admin", fromBase58String(address))] | |
36 | 36 | else throw("Only the Admin itself can invoke this function") | |
37 | 37 | ||
38 | 38 | ||
39 | 39 | ||
40 | 40 | @Callable(i) | |
41 | 41 | func constructor (presaleStart,presaleDuration,claimStart,claimDuration,price,price1AssetId58,price2AssetId58,price3AssetId58,minInvestAmount) = { | |
42 | 42 | let asset1 = getAssetIfValid(price1AssetId58) | |
43 | 43 | let asset2 = getAssetIfValid(price2AssetId58) | |
44 | 44 | let asset3 = getAssetIfValid(price3AssetId58) | |
45 | 45 | if (if ((i.caller == this)) | |
46 | - | then | |
47 | - | else | |
46 | + | then true | |
47 | + | else (i.caller == Address(getBinaryValue(this, "admin")))) | |
48 | 48 | then if ((presaleDuration >= presaleStart)) | |
49 | 49 | then throw("PresaleDuration can't be bigger than PresaleStart") | |
50 | 50 | else if ((presaleDuration >= claimStart)) | |
51 | 51 | then throw("presaleDuration can't be bigger than claimStart") | |
52 | 52 | else if ((claimDuration >= claimStart)) | |
53 | 53 | then throw("claimDuration can't be bigger than claimStart") | |
54 | 54 | else if ((minInvestAmount >= price)) | |
55 | 55 | then throw("minInvestAmount can't be bigger than price") | |
56 | 56 | else if (if (if ((asset1 == asset2)) | |
57 | 57 | then true | |
58 | 58 | else (asset1 == asset3)) | |
59 | 59 | then true | |
60 | 60 | else (asset2 == asset3)) | |
61 | 61 | then throw("Each Asset must be different from the other.") | |
62 | 62 | else if ((size(i.payments) != 1)) | |
63 | 63 | then throw("exactly 1 payment must be attached") | |
64 | 64 | else [BinaryEntry(maina, value(i.payments[0].assetId)), StringEntry(p1ID, price1AssetId58), StringEntry(p2ID, price2AssetId58), StringEntry(p3ID, price3AssetId58), IntegerEntry(ps, presaleStart), IntegerEntry(pd, presaleDuration), IntegerEntry(cs, claimStart), IntegerEntry(cd, claimDuration), IntegerEntry(p, price), IntegerEntry(mia, minInvestAmount), IntegerEntry(forSale, value(i.payments[0]).amount), IntegerEntry(totalSale, 0)] | |
65 | 65 | else throw("Only the Admin itself can invoke this function") | |
66 | 66 | } | |
67 | 67 | ||
68 | 68 | ||
69 | 69 | ||
70 | 70 | @Callable(i) | |
71 | 71 | func invest () = { | |
72 | 72 | let mainAsset = getBinary(this, maina) | |
73 | 73 | let presaleStart = getIntegerValue(this, ps) | |
74 | 74 | let presaleDuration = getIntegerValue(this, pd) | |
75 | 75 | let presaleEnd = (presaleStart + presaleDuration) | |
76 | 76 | let price = getIntegerValue(this, p) | |
77 | 77 | let h = height | |
78 | 78 | let pmt = value(i.payments[0]) | |
79 | 79 | let pmtAssetId = toBase58String(value(pmt.assetId)) | |
80 | 80 | let pmtAmount = pmt.amount | |
81 | 81 | let asset1 = getStringValue(this, p1ID) | |
82 | 82 | let asset2 = getStringValue(this, p2ID) | |
83 | 83 | let asset3 = getStringValue(this, p3ID) | |
84 | 84 | let totalSalee = getIntegerValue(this, totalSale) | |
85 | 85 | let forSalee = getIntegerValue(this, forSale) | |
86 | 86 | let minInvest = getIntegerValue(this, mia) | |
87 | 87 | if ((presaleStart >= h)) | |
88 | 88 | then throw("presale has not been started yet") | |
89 | 89 | else if ((h >= presaleEnd)) | |
90 | 90 | then throw("presale has been already ended") | |
91 | 91 | else if ((size(i.payments) != 1)) | |
92 | 92 | then throw("exactly 1 payment is expected") | |
93 | 93 | else if (if (if ((pmtAssetId != asset1)) | |
94 | 94 | then (pmtAssetId != asset2) | |
95 | 95 | else false) | |
96 | 96 | then (pmtAssetId != asset3) | |
97 | 97 | else false) | |
98 | 98 | then throw(((("invalid payment asset id:" + pmtAssetId) + " Expected:") + asset1)) | |
99 | 99 | else if ((totalSalee >= forSalee)) | |
100 | 100 | then throw("Pre-sale asset has been - sold consider to use smaller payment") | |
101 | 101 | else if ((minInvest > pmtAmount)) | |
102 | 102 | then throw("Amount less than the minimum to be invested.") | |
103 | 103 | else { | |
104 | 104 | let getTInvest = valueOrElse(getIntegerValue(this, (tinvest + toBase58String(i.caller.bytes))), 0) | |
105 | 105 | let calc = (pmtAmount / price) | |
106 | 106 | let totalAmount = (pmtAmount + getTInvest) | |
107 | 107 | [IntegerEntry((tinvest + toBase58String(i.caller.bytes)), totalAmount), IntegerEntry(totalSale, (totalSalee + calc)), ScriptTransfer(i.caller, calc, mainAsset), BooleanEntry((((tinvest + toBase58String(i.caller.bytes)) + "__") + pmtAssetId), true)] | |
108 | 108 | } | |
109 | 109 | } | |
110 | 110 | ||
111 | 111 | ||
112 | 112 | @Verifier(tx) | |
113 | 113 | func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
114 | 114 |
github/deemru/w8io/169f3d6 45.97 ms ◑