From b48db43b5be917b534f40f7d37ec8c1ce6e34891 Mon Sep 17 00:00:00 2001 From: Adam Chidlow Date: Fri, 8 Mar 2024 10:38:28 +0800 Subject: [PATCH] feat: allow TEAL optimizer to handle dup/dupn ops when removing stack shuffling of constants, and collapse any repeated elements with a dup/dupn --- .../amm/out/ConstantProductAMM.approval.teal | 2 +- .../amm/out/ConstantProductAMM.arc32.json | 2 +- .../out_O2/ConstantProductAMM.approval.teal | 2 +- .../calculator/out/MyContract.approval.teal | 2 +- .../out_O2/MyContract.approval.teal | 2 +- examples/sizes.txt | 16 ++++---- .../voting/out/VotingRoundApp.approval.teal | 4 +- examples/voting/out/VotingRoundApp.arc32.json | 2 +- .../out_O2/VotingRoundApp.approval.teal | 4 +- .../teal/optimize/constant_stack_shuffling.py | 41 ++++++++++++------- src/puya/teal/optimize/main.py | 6 ++- .../out/Arc4ArraysContract.approval.teal | 2 +- .../out/Arc4BoolEvalContract.approval.teal | 2 +- .../Arc4DynamicBytesContract.approval.teal | 2 +- .../out/Arc4RefTypesContract.approval.teal | 16 +++++--- .../out/Arc4StructsTypeContract.approval.teal | 2 +- test_cases/arc4_types/out/array.O1.log | 2 +- test_cases/arc4_types/out/array.O2.log | 2 +- test_cases/arc4_types/out/structs.O1.log | 6 +-- test_cases/arc4_types/out/structs.O2.log | 6 +-- .../out_O2/Arc4ArraysContract.approval.teal | 2 +- .../out_O2/Arc4BoolEvalContract.approval.teal | 2 +- .../Arc4DynamicBytesContract.approval.teal | 2 +- .../out_O2/Arc4RefTypesContract.approval.teal | 5 +-- .../Arc4StructsTypeContract.approval.teal | 2 +- .../out/BooleanBinaryOps.approval.teal | 8 ++-- .../out_O2/BooleanBinaryOps.approval.teal | 8 ++-- .../contains/out/MyContract.approval.teal | 6 +-- .../contains/out_O2/MyContract.approval.teal | 6 +-- .../everything/out/MyContract.approval.teal | 4 +- .../everything/out/MyContract.arc32.json | 2 +- .../out_O2/MyContract.approval.teal | 4 +- .../out/MyContract.approval.teal | 2 +- .../out/itxn_loop.approval.teal | 2 +- .../out_O2/MyContract.approval.teal | 2 +- .../out_O2/itxn_loop.approval.teal | 2 +- .../out/ImmediateVariants.approval.teal | 2 +- .../out_O2/ImmediateVariants.approval.teal | 2 +- .../less_simple/out/MyContract.approval.teal | 2 +- .../out_O2/MyContract.approval.teal | 2 +- test_cases/log/out/MyContract.approval.teal | 3 +- .../log/out_O2/MyContract.approval.teal | 3 +- .../nested_loops/out/Nested.approval.teal | 4 +- .../nested_loops/out_O2/Nested.approval.teal | 4 +- .../out/MyContract.approval.teal | 2 +- .../reversed_iteration/out/trace.O1.log | 2 +- .../reversed_iteration/out/trace.O2.log | 2 +- .../out_O2/MyContract.approval.teal | 2 +- .../out/MyContract.approval.teal | 11 ++--- .../out_O2/MyContract.approval.teal | 7 +--- .../out/TupleSupport.approval.teal | 2 +- .../out_O2/TupleSupport.approval.teal | 2 +- .../out/Baddie.approval.teal | 12 +++--- .../out_O2/Baddie.approval.teal | 12 +++--- 54 files changed, 135 insertions(+), 123 deletions(-) diff --git a/examples/amm/out/ConstantProductAMM.approval.teal b/examples/amm/out/ConstantProductAMM.approval.teal index ee2f767797..4097c6654a 100644 --- a/examples/amm/out/ConstantProductAMM.approval.teal +++ b/examples/amm/out/ConstantProductAMM.approval.teal @@ -390,7 +390,7 @@ _create_pool_token: global CurrentApplicationAddress // amm/contract.py:279 // reserve=Global.current_application_address, - global CurrentApplicationAddress + dup // amm/contract.py:280 // fee=0, int 0 diff --git a/examples/amm/out/ConstantProductAMM.arc32.json b/examples/amm/out/ConstantProductAMM.arc32.json index a2c910efa4..d9275cf50b 100644 --- a/examples/amm/out/ConstantProductAMM.arc32.json +++ b/examples/amm/out/ConstantProductAMM.arc32.json @@ -65,7 +65,7 @@ } }, "source": { - "approval": "#pragma version 10

examples.amm.contract.ConstantProductAMM.approval_program:
    txn ApplicationID
    bnz main_entrypoint@2
    callsub __init__

main_entrypoint@2:
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn NumAppArgs
    bz main_bare_routing@11
    method "set_governor(account)void"
    method "bootstrap(pay,asset,asset)uint64"
    method "mint(axfer,axfer,asset,asset,asset)void"
    method "burn(axfer,asset,asset,asset)void"
    method "swap(axfer,asset,asset)void"
    txna ApplicationArgs 0
    match main_set_governor_route@4 main_bootstrap_route@5 main_mint_route@6 main_burn_route@7 main_swap_route@8
    err // reject transaction

main_set_governor_route@4:
    // amm/contract.py:43
    // @arc4.abimethod()
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txna ApplicationArgs 1
    btoi
    txnas Accounts
    // amm/contract.py:43
    // @arc4.abimethod()
    callsub set_governor
    int 1
    return

main_bootstrap_route@5:
    // amm/contract.py:49
    // @arc4.abimethod()
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int pay
    ==
    assert // transaction type is pay
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    // amm/contract.py:49
    // @arc4.abimethod()
    callsub bootstrap
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_mint_route@6:
    // amm/contract.py:83-89
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 2
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    txna ApplicationArgs 3
    btoi
    txnas Assets
    // amm/contract.py:83-89
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    callsub mint
    int 1
    return

main_burn_route@7:
    // amm/contract.py:149-155
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    txna ApplicationArgs 3
    btoi
    txnas Assets
    // amm/contract.py:149-155
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    callsub burn
    int 1
    return

main_swap_route@8:
    // amm/contract.py:206-211
    // @arc4.abimethod(
    //     default_args={
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    // amm/contract.py:206-211
    // @arc4.abimethod(
    //     default_args={
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    callsub swap
    int 1
    return

main_bare_routing@11:
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn OnCompletion
    !
    assert // reject transaction
    txn ApplicationID
    !
    assert // is creating
    int 1
    return


// examples.amm.contract.ConstantProductAMM.set_governor(new_governor: bytes) -> void:
set_governor:
    // amm/contract.py:43-44
    // @arc4.abimethod()
    // def set_governor(self, new_governor: Account) -> None:
    proto 1 0
    // amm/contract.py:46
    // self._check_is_governor()
    callsub _check_is_governor
    // amm/contract.py:47
    // self.governor = new_governor
    byte "governor"
    frame_dig -1
    app_global_put
    retsub


// examples.amm.contract.ConstantProductAMM._check_is_governor() -> void:
_check_is_governor:
    // amm/contract.py:264-265
    // @subroutine
    // def _check_is_governor(self) -> None:
    proto 0 0
    // amm/contract.py:267
    // Txn.sender == self.governor
    txn Sender
    int 0
    byte "governor"
    app_global_get_ex
    assert // check governor exists
    ==
    // amm/contract.py:266-268
    // assert (
    //     Txn.sender == self.governor
    // ), "Only the account set in global_state.governor may call this method"
    assert // Only the account set in global_state.governor may call this method
    retsub


// examples.amm.contract.ConstantProductAMM.bootstrap(seed: uint64, a_asset: uint64, b_asset: uint64) -> bytes:
bootstrap:
    // amm/contract.py:49-52
    // @arc4.abimethod()
    // def bootstrap(
    //     self, seed: gtxn.PaymentTransaction, a_asset: Asset, b_asset: Asset
    // ) -> arc4.UInt64:
    proto 3 1
    // amm/contract.py:68
    // assert not self.pool_token, "application has already been bootstrapped"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    !
    assert // application has already been bootstrapped
    // amm/contract.py:69
    // self._check_is_governor()
    callsub _check_is_governor
    // amm/contract.py:70
    // assert Global.group_size == 2, "group size not 2"
    global GroupSize
    int 2
    ==
    assert // group size not 2
    // amm/contract.py:71
    // assert seed.receiver == Global.current_application_address, "receiver not app address"
    frame_dig -3
    gtxns Receiver
    global CurrentApplicationAddress
    ==
    assert // receiver not app address
    // amm/contract.py:73
    // assert seed.amount >= 300_000, "amount minimum not met"  # 0.3 Algos
    frame_dig -3
    gtxns Amount
    int 300000
    >=
    assert // amount minimum not met
    // amm/contract.py:74
    // assert a_asset.id < b_asset.id, "asset a must be less than asset b"
    frame_dig -2
    frame_dig -1
    <
    assert // asset a must be less than asset b
    // amm/contract.py:75
    // self.asset_a = a_asset
    byte "asset_a"
    frame_dig -2
    app_global_put
    // amm/contract.py:76
    // self.asset_b = b_asset
    byte "asset_b"
    frame_dig -1
    app_global_put
    // amm/contract.py:77
    // self.pool_token = self._create_pool_token()
    callsub _create_pool_token
    byte "pool_token"
    swap
    app_global_put
    // amm/contract.py:79
    // self._do_opt_in(self.asset_a)
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    callsub _do_opt_in
    // amm/contract.py:80
    // self._do_opt_in(self.asset_b)
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    callsub _do_opt_in
    // amm/contract.py:81
    // return arc4.UInt64(self.pool_token.id)
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    itob
    retsub


// examples.amm.contract.ConstantProductAMM._create_pool_token() -> uint64:
_create_pool_token:
    // amm/contract.py:270-271
    // @subroutine
    // def _create_pool_token(self) -> Asset:
    proto 0 1
    // amm/contract.py:273-282
    // itxn.AssetConfig(
    //     asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    //     unit_name=b"dbt",
    //     total=TOTAL_SUPPLY,
    //     decimals=3,
    //     manager=Global.current_application_address,
    //     reserve=Global.current_application_address,
    //     fee=0,
    // )
    // .submit()
    itxn_begin
    // amm/contract.py:274
    // asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    asset_params_get AssetUnitName
    assert // asset exists
    byte "DPT-"
    swap
    concat
    byte "-"
    concat
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    asset_params_get AssetUnitName
    assert // asset exists
    concat
    // amm/contract.py:278
    // manager=Global.current_application_address,
    global CurrentApplicationAddress
    // amm/contract.py:279
    // reserve=Global.current_application_address,
    global CurrentApplicationAddress
    // amm/contract.py:280
    // fee=0,
    int 0
    itxn_field Fee
    itxn_field ConfigAssetReserve
    itxn_field ConfigAssetManager
    // amm/contract.py:277
    // decimals=3,
    int 3
    itxn_field ConfigAssetDecimals
    // amm/contract.py:276
    // total=TOTAL_SUPPLY,
    int 10000000000
    itxn_field ConfigAssetTotal
    // amm/contract.py:275
    // unit_name=b"dbt",
    byte "dbt"
    itxn_field ConfigAssetUnitName
    itxn_field ConfigAssetName
    // amm/contract.py:273
    // itxn.AssetConfig(
    int acfg
    itxn_field TypeEnum
    // amm/contract.py:273-282
    // itxn.AssetConfig(
    //     asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    //     unit_name=b"dbt",
    //     total=TOTAL_SUPPLY,
    //     decimals=3,
    //     manager=Global.current_application_address,
    //     reserve=Global.current_application_address,
    //     fee=0,
    // )
    // .submit()
    itxn_submit
    itxn CreatedAssetID
    // amm/contract.py:272-284
    // return (
    //     itxn.AssetConfig(
    //         asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    //         unit_name=b"dbt",
    //         total=TOTAL_SUPPLY,
    //         decimals=3,
    //         manager=Global.current_application_address,
    //         reserve=Global.current_application_address,
    //         fee=0,
    //     )
    //     .submit()
    //     .created_asset
    // )
    retsub


// examples.amm.contract.ConstantProductAMM._do_opt_in(asset: uint64) -> void:
_do_opt_in:
    // amm/contract.py:286-287
    // @subroutine
    // def _do_opt_in(self, asset: Asset) -> None:
    proto 1 0
    // amm/contract.py:289
    // receiver=Global.current_application_address,
    global CurrentApplicationAddress
    // amm/contract.py:288-292
    // do_asset_transfer(
    //     receiver=Global.current_application_address,
    //     asset=asset,
    //     amount=UInt64(0),
    // )
    frame_dig -1
    // amm/contract.py:291
    // amount=UInt64(0),
    int 0
    // amm/contract.py:288-292
    // do_asset_transfer(
    //     receiver=Global.current_application_address,
    //     asset=asset,
    //     amount=UInt64(0),
    // )
    callsub do_asset_transfer
    retsub


// examples.amm.contract.do_asset_transfer(receiver: bytes, asset: uint64, amount: uint64) -> void:
do_asset_transfer:
    // amm/contract.py:359-360
    // @subroutine
    // def do_asset_transfer(*, receiver: Account, asset: Asset, amount: UInt64) -> None:
    proto 3 0
    // amm/contract.py:361-366
    // itxn.AssetTransfer(
    //     xfer_asset=asset,
    //     asset_amount=amount,
    //     asset_receiver=receiver,
    //     fee=0,
    // ).submit()
    itxn_begin
    // amm/contract.py:365
    // fee=0,
    int 0
    itxn_field Fee
    frame_dig -3
    itxn_field AssetReceiver
    frame_dig -1
    itxn_field AssetAmount
    frame_dig -2
    itxn_field XferAsset
    // amm/contract.py:361
    // itxn.AssetTransfer(
    int axfer
    itxn_field TypeEnum
    // amm/contract.py:361-366
    // itxn.AssetTransfer(
    //     xfer_asset=asset,
    //     asset_amount=amount,
    //     asset_receiver=receiver,
    //     fee=0,
    // ).submit()
    itxn_submit
    retsub


// examples.amm.contract.ConstantProductAMM.mint(a_xfer: uint64, b_xfer: uint64, pool_asset: uint64, a_asset: uint64, b_asset: uint64) -> void:
mint:
    // amm/contract.py:83-97
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    // def mint(
    //     self,
    //     a_xfer: gtxn.AssetTransferTransaction,
    //     b_xfer: gtxn.AssetTransferTransaction,
    //     pool_asset: Asset,
    //     a_asset: Asset,
    //     b_asset: Asset,
    // ) -> None:
    proto 5 0
    // amm/contract.py:113
    // self._check_bootstrapped()
    callsub _check_bootstrapped
    // amm/contract.py:115-116
    // # well-formed mint
    // assert pool_asset == self.pool_token, "asset pool incorrect"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    frame_dig -3
    ==
    assert // asset pool incorrect
    // amm/contract.py:117
    // assert a_asset == self.asset_a, "asset a incorrect"
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    frame_dig -2
    ==
    assert // asset a incorrect
    // amm/contract.py:118
    // assert b_asset == self.asset_b, "asset b incorrect"
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    frame_dig -1
    ==
    assert // asset b incorrect
    // amm/contract.py:119
    // assert a_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -5
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:120
    // assert b_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -4
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:124
    // a_xfer.asset_receiver == Global.current_application_address
    frame_dig -5
    gtxns AssetReceiver
    global CurrentApplicationAddress
    ==
    // amm/contract.py:122-125
    // # valid asset a xfer
    // assert (
    //     a_xfer.asset_receiver == Global.current_application_address
    // ), "receiver not app address"
    assert // receiver not app address
    // amm/contract.py:126
    // assert a_xfer.xfer_asset == self.asset_a, "asset a incorrect"
    frame_dig -5
    gtxns XferAsset
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    ==
    assert // asset a incorrect
    // amm/contract.py:127
    // assert a_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -5
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:131
    // b_xfer.asset_receiver == Global.current_application_address
    frame_dig -4
    gtxns AssetReceiver
    global CurrentApplicationAddress
    ==
    // amm/contract.py:129-132
    // # valid asset b xfer
    // assert (
    //     b_xfer.asset_receiver == Global.current_application_address
    // ), "receiver not app address"
    assert // receiver not app address
    // amm/contract.py:133
    // assert b_xfer.xfer_asset == self.asset_b, "asset b incorrect"
    frame_dig -4
    gtxns XferAsset
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    ==
    assert // asset b incorrect
    // amm/contract.py:134
    // assert b_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -4
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:137
    // pool_balance=self._current_pool_balance(),
    callsub _current_pool_balance
    // amm/contract.py:138
    // a_balance=self._current_a_balance(),
    callsub _current_a_balance
    // amm/contract.py:139
    // b_balance=self._current_b_balance(),
    callsub _current_b_balance
    // amm/contract.py:136-142
    // to_mint = tokens_to_mint(
    //     pool_balance=self._current_pool_balance(),
    //     a_balance=self._current_a_balance(),
    //     b_balance=self._current_b_balance(),
    //     a_amount=a_xfer.asset_amount,
    //     b_amount=b_xfer.asset_amount,
    // )
    uncover 4
    uncover 4
    callsub tokens_to_mint
    // amm/contract.py:143
    // assert to_mint > 0, "send amount too low"
    dup
    int 0
    >
    assert // send amount too low
    // amm/contract.py:145-146
    // # mint tokens
    // do_asset_transfer(receiver=Txn.sender, asset=self.pool_token, amount=to_mint)
    txn Sender
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:147
    // self._update_ratio()
    callsub _update_ratio
    retsub


// examples.amm.contract.ConstantProductAMM._check_bootstrapped() -> void:
_check_bootstrapped:
    // amm/contract.py:253-254
    // @subroutine
    // def _check_bootstrapped(self) -> None:
    proto 0 0
    // amm/contract.py:255
    // assert self.pool_token, "bootstrap method needs to be called first"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    assert // bootstrap method needs to be called first
    retsub


// examples.amm.contract.ConstantProductAMM._current_pool_balance() -> uint64:
_current_pool_balance:
    // amm/contract.py:294-295
    // @subroutine
    // def _current_pool_balance(self) -> UInt64:
    proto 0 1
    // amm/contract.py:296
    // return self.pool_token.balance(Global.current_application_address)
    global CurrentApplicationAddress
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    asset_holding_get AssetBalance
    assert // account opted into asset
    retsub


// examples.amm.contract.ConstantProductAMM._current_a_balance() -> uint64:
_current_a_balance:
    // amm/contract.py:298-299
    // @subroutine
    // def _current_a_balance(self) -> UInt64:
    proto 0 1
    // amm/contract.py:300
    // return self.asset_a.balance(Global.current_application_address)
    global CurrentApplicationAddress
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    asset_holding_get AssetBalance
    assert // account opted into asset
    retsub


// examples.amm.contract.ConstantProductAMM._current_b_balance() -> uint64:
_current_b_balance:
    // amm/contract.py:302-303
    // @subroutine
    // def _current_b_balance(self) -> UInt64:
    proto 0 1
    // amm/contract.py:304
    // return self.asset_b.balance(Global.current_application_address)
    global CurrentApplicationAddress
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    asset_holding_get AssetBalance
    assert // account opted into asset
    retsub


// examples.amm.contract.tokens_to_mint(pool_balance: uint64, a_balance: uint64, b_balance: uint64, a_amount: uint64, b_amount: uint64) -> uint64:
tokens_to_mint:
    // amm/contract.py:325-333
    // @subroutine
    // def tokens_to_mint(
    //     *,
    //     pool_balance: UInt64,
    //     a_balance: UInt64,
    //     b_balance: UInt64,
    //     a_amount: UInt64,
    //     b_amount: UInt64,
    // ) -> UInt64:
    proto 5 1
    byte ""
    dup
    // amm/contract.py:334
    // is_initial_mint = a_balance == a_amount and b_balance == b_amount
    frame_dig -4
    frame_dig -2
    ==
    bz tokens_to_mint_bool_false@3
    frame_dig -3
    frame_dig -1
    ==
    bz tokens_to_mint_bool_false@3
    int 1
    b tokens_to_mint_bool_merge@4

tokens_to_mint_bool_false@3:
    int 0

tokens_to_mint_bool_merge@4:
    // amm/contract.py:335
    // if is_initial_mint:
    bz tokens_to_mint_after_if_else@6
    // amm/contract.py:336
    // return op.sqrt(a_amount * b_amount) - SCALE
    frame_dig -2
    frame_dig -1
    *
    sqrt
    int 1000
    -
    frame_bury 0
    retsub

tokens_to_mint_after_if_else@6:
    // amm/contract.py:337
    // issued = TOTAL_SUPPLY - pool_balance
    int 10000000000
    frame_dig -5
    -
    // amm/contract.py:338
    // a_ratio = SCALE * a_amount // (a_balance - a_amount)
    int 1000
    frame_dig -2
    *
    frame_dig -4
    frame_dig -2
    -
    /
    dup
    frame_bury 0
    // amm/contract.py:339
    // b_ratio = SCALE * b_amount // (b_balance - b_amount)
    int 1000
    frame_dig -1
    *
    frame_dig -3
    frame_dig -1
    -
    /
    dup
    frame_bury 1
    // amm/contract.py:340
    // if a_ratio < b_ratio:
    <
    bz tokens_to_mint_else_body@8
    // amm/contract.py:341
    // return a_ratio * issued // SCALE
    frame_dig 0
    *
    int 1000
    /
    frame_bury 0
    retsub

tokens_to_mint_else_body@8:
    // amm/contract.py:343
    // return b_ratio * issued // SCALE
    frame_dig 1
    *
    int 1000
    /
    frame_bury 0
    retsub


// examples.amm.contract.ConstantProductAMM._update_ratio() -> void:
_update_ratio:
    // amm/contract.py:257-258
    // @subroutine
    // def _update_ratio(self) -> None:
    proto 0 0
    // amm/contract.py:259
    // a_balance = self._current_a_balance()
    callsub _current_a_balance
    // amm/contract.py:260
    // b_balance = self._current_b_balance()
    callsub _current_b_balance
    swap
    // amm/contract.py:262
    // self.ratio = a_balance * SCALE // b_balance
    int 1000
    *
    swap
    /
    byte "ratio"
    swap
    app_global_put
    retsub


// examples.amm.contract.ConstantProductAMM.burn(pool_xfer: uint64, pool_asset: uint64, a_asset: uint64, b_asset: uint64) -> void:
burn:
    // amm/contract.py:149-162
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    // def burn(
    //     self,
    //     pool_xfer: gtxn.AssetTransferTransaction,
    //     pool_asset: Asset,
    //     a_asset: Asset,
    //     b_asset: Asset,
    // ) -> None:
    proto 4 0
    // amm/contract.py:172
    // self._check_bootstrapped()
    callsub _check_bootstrapped
    // amm/contract.py:174
    // assert pool_asset == self.pool_token, "asset pool incorrect"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    frame_dig -3
    ==
    assert // asset pool incorrect
    // amm/contract.py:175
    // assert a_asset == self.asset_a, "asset a incorrect"
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    frame_dig -2
    ==
    assert // asset a incorrect
    // amm/contract.py:176
    // assert b_asset == self.asset_b, "asset b incorrect"
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    frame_dig -1
    ==
    assert // asset b incorrect
    // amm/contract.py:179
    // pool_xfer.asset_receiver == Global.current_application_address
    frame_dig -4
    gtxns AssetReceiver
    global CurrentApplicationAddress
    ==
    // amm/contract.py:178-180
    // assert (
    //     pool_xfer.asset_receiver == Global.current_application_address
    // ), "receiver not app address"
    assert // receiver not app address
    // amm/contract.py:181
    // assert pool_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -4
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:182
    // assert pool_xfer.xfer_asset == self.pool_token, "asset pool incorrect"
    frame_dig -4
    gtxns XferAsset
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    ==
    assert // asset pool incorrect
    // amm/contract.py:183
    // assert pool_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -4
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:185-187
    // # Get the total number of tokens issued
    // # !important: this happens prior to receiving the current axfer of pool tokens
    // pool_balance = self._current_pool_balance()
    callsub _current_pool_balance
    // amm/contract.py:190
    // supply=self._current_a_balance(),
    callsub _current_a_balance
    // amm/contract.py:188-192
    // a_amt = tokens_to_burn(
    //     pool_balance=pool_balance,
    //     supply=self._current_a_balance(),
    //     amount=pool_xfer.asset_amount,
    // )
    dig 1
    swap
    dig 3
    callsub tokens_to_burn
    cover 2
    // amm/contract.py:195
    // supply=self._current_b_balance(),
    callsub _current_b_balance
    // amm/contract.py:193-197
    // b_amt = tokens_to_burn(
    //     pool_balance=pool_balance,
    //     supply=self._current_b_balance(),
    //     amount=pool_xfer.asset_amount,
    // )
    uncover 2
    callsub tokens_to_burn
    swap
    // amm/contract.py:199-200
    // # Send back commensurate amt of a
    // do_asset_transfer(receiver=Txn.sender, asset=self.asset_a, amount=a_amt)
    txn Sender
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:202-203
    // # Send back commensurate amt of b
    // do_asset_transfer(receiver=Txn.sender, asset=self.asset_b, amount=b_amt)
    txn Sender
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:204
    // self._update_ratio()
    callsub _update_ratio
    retsub


// examples.amm.contract.tokens_to_burn(pool_balance: uint64, supply: uint64, amount: uint64) -> uint64:
tokens_to_burn:
    // amm/contract.py:346-347
    // @subroutine
    // def tokens_to_burn(*, pool_balance: UInt64, supply: UInt64, amount: UInt64) -> UInt64:
    proto 3 1
    // amm/contract.py:348
    // issued = TOTAL_SUPPLY - pool_balance - amount
    int 10000000000
    frame_dig -3
    -
    frame_dig -1
    -
    // amm/contract.py:349
    // return supply * amount // issued
    frame_dig -2
    frame_dig -1
    *
    swap
    /
    retsub


// examples.amm.contract.ConstantProductAMM.swap(swap_xfer: uint64, a_asset: uint64, b_asset: uint64) -> void:
swap:
    // amm/contract.py:206-217
    // @arc4.abimethod(
    //     default_args={
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    // def swap(
    //     self,
    //     swap_xfer: gtxn.AssetTransferTransaction,
    //     a_asset: Asset,
    //     b_asset: Asset,
    // ) -> None:
    proto 3 0
    byte ""
    dup
    // amm/contract.py:225
    // self._check_bootstrapped()
    callsub _check_bootstrapped
    // amm/contract.py:227
    // assert a_asset == self.asset_a, "asset a incorrect"
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    frame_dig -2
    ==
    assert // asset a incorrect
    // amm/contract.py:228
    // assert b_asset == self.asset_b, "asset b incorrect"
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    frame_dig -1
    ==
    assert // asset b incorrect
    // amm/contract.py:230
    // assert swap_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -3
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:231
    // assert swap_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -3
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:234
    // case self.asset_a:
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    // amm/contract.py:238
    // case self.asset_b:
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    // amm/contract.py:233
    // match swap_xfer.xfer_asset:
    frame_dig -3
    gtxns XferAsset
    // amm/contract.py:233-243
    // match swap_xfer.xfer_asset:
    //     case self.asset_a:
    //         in_supply = self._current_b_balance()
    //         out_supply = self._current_a_balance()
    //         out_asset = self.asset_a
    //     case self.asset_b:
    //         in_supply = self._current_a_balance()
    //         out_supply = self._current_b_balance()
    //         out_asset = self.asset_b
    //     case _:
    //         assert False, "asset id incorrect"
    match swap_switch_case_0@1 swap_switch_case_1@2
    // amm/contract.py:243
    // assert False, "asset id incorrect"
    err // asset id incorrect

swap_switch_case_0@1:
    // amm/contract.py:235
    // in_supply = self._current_b_balance()
    callsub _current_b_balance
    frame_bury 0
    // amm/contract.py:236
    // out_supply = self._current_a_balance()
    callsub _current_a_balance
    // amm/contract.py:237
    // out_asset = self.asset_a
    int 0
    byte "asset_a"
    app_global_get_ex
    swap
    frame_bury 1
    assert // check asset_a exists
    b swap_switch_case_next@4

swap_switch_case_1@2:
    // amm/contract.py:239
    // in_supply = self._current_a_balance()
    callsub _current_a_balance
    frame_bury 0
    // amm/contract.py:240
    // out_supply = self._current_b_balance()
    callsub _current_b_balance
    // amm/contract.py:241
    // out_asset = self.asset_b
    int 0
    byte "asset_b"
    app_global_get_ex
    swap
    frame_bury 1
    assert // check asset_b exists

swap_switch_case_next@4:
    // amm/contract.py:245-247
    // to_swap = tokens_to_swap(
    //     in_amount=swap_xfer.asset_amount, in_supply=in_supply, out_supply=out_supply
    // )
    frame_dig 2
    frame_dig 0
    uncover 2
    callsub tokens_to_swap
    // amm/contract.py:248
    // assert to_swap > 0, "send amount too low"
    dup
    int 0
    >
    assert // send amount too low
    // amm/contract.py:250
    // do_asset_transfer(receiver=Txn.sender, asset=out_asset, amount=to_swap)
    txn Sender
    frame_dig 1
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:251
    // self._update_ratio()
    callsub _update_ratio
    retsub


// examples.amm.contract.tokens_to_swap(in_amount: uint64, in_supply: uint64, out_supply: uint64) -> uint64:
tokens_to_swap:
    // amm/contract.py:352-353
    // @subroutine
    // def tokens_to_swap(*, in_amount: UInt64, in_supply: UInt64, out_supply: UInt64) -> UInt64:
    proto 3 1
    // amm/contract.py:354
    // in_total = SCALE * (in_supply - in_amount) + (in_amount * FACTOR)
    frame_dig -2
    frame_dig -3
    -
    int 1000
    *
    frame_dig -3
    int 995
    *
    dup
    cover 2
    +
    swap
    // amm/contract.py:355
    // out_total = in_amount * FACTOR * out_supply
    frame_dig -1
    *
    // amm/contract.py:356
    // return out_total // in_total
    swap
    /
    retsub


// examples.amm.contract.ConstantProductAMM.__init__() -> void:
__init__:
    // amm/contract.py:28
    // def __init__(self) -> None:
    proto 0 0
    // amm/contract.py:32-33
    // # The asset id of asset A
    // self.asset_a = Asset(0)
    byte "asset_a"
    int 0
    app_global_put
    // amm/contract.py:34-35
    // # The asset id of asset B
    // self.asset_b = Asset(0)
    byte "asset_b"
    int 0
    app_global_put
    // amm/contract.py:36-37
    // # The current governor of this contract, allowed to do admin type actions
    // self.governor = Txn.sender
    byte "governor"
    txn Sender
    app_global_put
    // amm/contract.py:38-39
    // # The asset id of the Pool Token, used to track share of pool the holder may recover
    // self.pool_token = Asset(0)
    byte "pool_token"
    int 0
    app_global_put
    // amm/contract.py:40-41
    // # The ratio between assets (A*Scale/B)
    // self.ratio = UInt64(0)
    byte "ratio"
    int 0
    app_global_put
    retsub
", + "approval": "#pragma version 10

examples.amm.contract.ConstantProductAMM.approval_program:
    txn ApplicationID
    bnz main_entrypoint@2
    callsub __init__

main_entrypoint@2:
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn NumAppArgs
    bz main_bare_routing@11
    method "set_governor(account)void"
    method "bootstrap(pay,asset,asset)uint64"
    method "mint(axfer,axfer,asset,asset,asset)void"
    method "burn(axfer,asset,asset,asset)void"
    method "swap(axfer,asset,asset)void"
    txna ApplicationArgs 0
    match main_set_governor_route@4 main_bootstrap_route@5 main_mint_route@6 main_burn_route@7 main_swap_route@8
    err // reject transaction

main_set_governor_route@4:
    // amm/contract.py:43
    // @arc4.abimethod()
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txna ApplicationArgs 1
    btoi
    txnas Accounts
    // amm/contract.py:43
    // @arc4.abimethod()
    callsub set_governor
    int 1
    return

main_bootstrap_route@5:
    // amm/contract.py:49
    // @arc4.abimethod()
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int pay
    ==
    assert // transaction type is pay
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    // amm/contract.py:49
    // @arc4.abimethod()
    callsub bootstrap
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_mint_route@6:
    // amm/contract.py:83-89
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 2
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    txna ApplicationArgs 3
    btoi
    txnas Assets
    // amm/contract.py:83-89
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    callsub mint
    int 1
    return

main_burn_route@7:
    // amm/contract.py:149-155
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    txna ApplicationArgs 3
    btoi
    txnas Assets
    // amm/contract.py:149-155
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    callsub burn
    int 1
    return

main_swap_route@8:
    // amm/contract.py:206-211
    // @arc4.abimethod(
    //     default_args={
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int axfer
    ==
    assert // transaction type is axfer
    txna ApplicationArgs 1
    btoi
    txnas Assets
    txna ApplicationArgs 2
    btoi
    txnas Assets
    // amm/contract.py:206-211
    // @arc4.abimethod(
    //     default_args={
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    callsub swap
    int 1
    return

main_bare_routing@11:
    // amm/contract.py:27
    // class ConstantProductAMM(ARC4Contract):
    txn OnCompletion
    !
    assert // reject transaction
    txn ApplicationID
    !
    assert // is creating
    int 1
    return


// examples.amm.contract.ConstantProductAMM.set_governor(new_governor: bytes) -> void:
set_governor:
    // amm/contract.py:43-44
    // @arc4.abimethod()
    // def set_governor(self, new_governor: Account) -> None:
    proto 1 0
    // amm/contract.py:46
    // self._check_is_governor()
    callsub _check_is_governor
    // amm/contract.py:47
    // self.governor = new_governor
    byte "governor"
    frame_dig -1
    app_global_put
    retsub


// examples.amm.contract.ConstantProductAMM._check_is_governor() -> void:
_check_is_governor:
    // amm/contract.py:264-265
    // @subroutine
    // def _check_is_governor(self) -> None:
    proto 0 0
    // amm/contract.py:267
    // Txn.sender == self.governor
    txn Sender
    int 0
    byte "governor"
    app_global_get_ex
    assert // check governor exists
    ==
    // amm/contract.py:266-268
    // assert (
    //     Txn.sender == self.governor
    // ), "Only the account set in global_state.governor may call this method"
    assert // Only the account set in global_state.governor may call this method
    retsub


// examples.amm.contract.ConstantProductAMM.bootstrap(seed: uint64, a_asset: uint64, b_asset: uint64) -> bytes:
bootstrap:
    // amm/contract.py:49-52
    // @arc4.abimethod()
    // def bootstrap(
    //     self, seed: gtxn.PaymentTransaction, a_asset: Asset, b_asset: Asset
    // ) -> arc4.UInt64:
    proto 3 1
    // amm/contract.py:68
    // assert not self.pool_token, "application has already been bootstrapped"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    !
    assert // application has already been bootstrapped
    // amm/contract.py:69
    // self._check_is_governor()
    callsub _check_is_governor
    // amm/contract.py:70
    // assert Global.group_size == 2, "group size not 2"
    global GroupSize
    int 2
    ==
    assert // group size not 2
    // amm/contract.py:71
    // assert seed.receiver == Global.current_application_address, "receiver not app address"
    frame_dig -3
    gtxns Receiver
    global CurrentApplicationAddress
    ==
    assert // receiver not app address
    // amm/contract.py:73
    // assert seed.amount >= 300_000, "amount minimum not met"  # 0.3 Algos
    frame_dig -3
    gtxns Amount
    int 300000
    >=
    assert // amount minimum not met
    // amm/contract.py:74
    // assert a_asset.id < b_asset.id, "asset a must be less than asset b"
    frame_dig -2
    frame_dig -1
    <
    assert // asset a must be less than asset b
    // amm/contract.py:75
    // self.asset_a = a_asset
    byte "asset_a"
    frame_dig -2
    app_global_put
    // amm/contract.py:76
    // self.asset_b = b_asset
    byte "asset_b"
    frame_dig -1
    app_global_put
    // amm/contract.py:77
    // self.pool_token = self._create_pool_token()
    callsub _create_pool_token
    byte "pool_token"
    swap
    app_global_put
    // amm/contract.py:79
    // self._do_opt_in(self.asset_a)
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    callsub _do_opt_in
    // amm/contract.py:80
    // self._do_opt_in(self.asset_b)
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    callsub _do_opt_in
    // amm/contract.py:81
    // return arc4.UInt64(self.pool_token.id)
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    itob
    retsub


// examples.amm.contract.ConstantProductAMM._create_pool_token() -> uint64:
_create_pool_token:
    // amm/contract.py:270-271
    // @subroutine
    // def _create_pool_token(self) -> Asset:
    proto 0 1
    // amm/contract.py:273-282
    // itxn.AssetConfig(
    //     asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    //     unit_name=b"dbt",
    //     total=TOTAL_SUPPLY,
    //     decimals=3,
    //     manager=Global.current_application_address,
    //     reserve=Global.current_application_address,
    //     fee=0,
    // )
    // .submit()
    itxn_begin
    // amm/contract.py:274
    // asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    asset_params_get AssetUnitName
    assert // asset exists
    byte "DPT-"
    swap
    concat
    byte "-"
    concat
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    asset_params_get AssetUnitName
    assert // asset exists
    concat
    // amm/contract.py:278
    // manager=Global.current_application_address,
    global CurrentApplicationAddress
    // amm/contract.py:279
    // reserve=Global.current_application_address,
    dup
    // amm/contract.py:280
    // fee=0,
    int 0
    itxn_field Fee
    itxn_field ConfigAssetReserve
    itxn_field ConfigAssetManager
    // amm/contract.py:277
    // decimals=3,
    int 3
    itxn_field ConfigAssetDecimals
    // amm/contract.py:276
    // total=TOTAL_SUPPLY,
    int 10000000000
    itxn_field ConfigAssetTotal
    // amm/contract.py:275
    // unit_name=b"dbt",
    byte "dbt"
    itxn_field ConfigAssetUnitName
    itxn_field ConfigAssetName
    // amm/contract.py:273
    // itxn.AssetConfig(
    int acfg
    itxn_field TypeEnum
    // amm/contract.py:273-282
    // itxn.AssetConfig(
    //     asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    //     unit_name=b"dbt",
    //     total=TOTAL_SUPPLY,
    //     decimals=3,
    //     manager=Global.current_application_address,
    //     reserve=Global.current_application_address,
    //     fee=0,
    // )
    // .submit()
    itxn_submit
    itxn CreatedAssetID
    // amm/contract.py:272-284
    // return (
    //     itxn.AssetConfig(
    //         asset_name=b"DPT-" + self.asset_a.unit_name + b"-" + self.asset_b.unit_name,
    //         unit_name=b"dbt",
    //         total=TOTAL_SUPPLY,
    //         decimals=3,
    //         manager=Global.current_application_address,
    //         reserve=Global.current_application_address,
    //         fee=0,
    //     )
    //     .submit()
    //     .created_asset
    // )
    retsub


// examples.amm.contract.ConstantProductAMM._do_opt_in(asset: uint64) -> void:
_do_opt_in:
    // amm/contract.py:286-287
    // @subroutine
    // def _do_opt_in(self, asset: Asset) -> None:
    proto 1 0
    // amm/contract.py:289
    // receiver=Global.current_application_address,
    global CurrentApplicationAddress
    // amm/contract.py:288-292
    // do_asset_transfer(
    //     receiver=Global.current_application_address,
    //     asset=asset,
    //     amount=UInt64(0),
    // )
    frame_dig -1
    // amm/contract.py:291
    // amount=UInt64(0),
    int 0
    // amm/contract.py:288-292
    // do_asset_transfer(
    //     receiver=Global.current_application_address,
    //     asset=asset,
    //     amount=UInt64(0),
    // )
    callsub do_asset_transfer
    retsub


// examples.amm.contract.do_asset_transfer(receiver: bytes, asset: uint64, amount: uint64) -> void:
do_asset_transfer:
    // amm/contract.py:359-360
    // @subroutine
    // def do_asset_transfer(*, receiver: Account, asset: Asset, amount: UInt64) -> None:
    proto 3 0
    // amm/contract.py:361-366
    // itxn.AssetTransfer(
    //     xfer_asset=asset,
    //     asset_amount=amount,
    //     asset_receiver=receiver,
    //     fee=0,
    // ).submit()
    itxn_begin
    // amm/contract.py:365
    // fee=0,
    int 0
    itxn_field Fee
    frame_dig -3
    itxn_field AssetReceiver
    frame_dig -1
    itxn_field AssetAmount
    frame_dig -2
    itxn_field XferAsset
    // amm/contract.py:361
    // itxn.AssetTransfer(
    int axfer
    itxn_field TypeEnum
    // amm/contract.py:361-366
    // itxn.AssetTransfer(
    //     xfer_asset=asset,
    //     asset_amount=amount,
    //     asset_receiver=receiver,
    //     fee=0,
    // ).submit()
    itxn_submit
    retsub


// examples.amm.contract.ConstantProductAMM.mint(a_xfer: uint64, b_xfer: uint64, pool_asset: uint64, a_asset: uint64, b_asset: uint64) -> void:
mint:
    // amm/contract.py:83-97
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    // def mint(
    //     self,
    //     a_xfer: gtxn.AssetTransferTransaction,
    //     b_xfer: gtxn.AssetTransferTransaction,
    //     pool_asset: Asset,
    //     a_asset: Asset,
    //     b_asset: Asset,
    // ) -> None:
    proto 5 0
    // amm/contract.py:113
    // self._check_bootstrapped()
    callsub _check_bootstrapped
    // amm/contract.py:115-116
    // # well-formed mint
    // assert pool_asset == self.pool_token, "asset pool incorrect"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    frame_dig -3
    ==
    assert // asset pool incorrect
    // amm/contract.py:117
    // assert a_asset == self.asset_a, "asset a incorrect"
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    frame_dig -2
    ==
    assert // asset a incorrect
    // amm/contract.py:118
    // assert b_asset == self.asset_b, "asset b incorrect"
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    frame_dig -1
    ==
    assert // asset b incorrect
    // amm/contract.py:119
    // assert a_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -5
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:120
    // assert b_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -4
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:124
    // a_xfer.asset_receiver == Global.current_application_address
    frame_dig -5
    gtxns AssetReceiver
    global CurrentApplicationAddress
    ==
    // amm/contract.py:122-125
    // # valid asset a xfer
    // assert (
    //     a_xfer.asset_receiver == Global.current_application_address
    // ), "receiver not app address"
    assert // receiver not app address
    // amm/contract.py:126
    // assert a_xfer.xfer_asset == self.asset_a, "asset a incorrect"
    frame_dig -5
    gtxns XferAsset
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    ==
    assert // asset a incorrect
    // amm/contract.py:127
    // assert a_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -5
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:131
    // b_xfer.asset_receiver == Global.current_application_address
    frame_dig -4
    gtxns AssetReceiver
    global CurrentApplicationAddress
    ==
    // amm/contract.py:129-132
    // # valid asset b xfer
    // assert (
    //     b_xfer.asset_receiver == Global.current_application_address
    // ), "receiver not app address"
    assert // receiver not app address
    // amm/contract.py:133
    // assert b_xfer.xfer_asset == self.asset_b, "asset b incorrect"
    frame_dig -4
    gtxns XferAsset
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    ==
    assert // asset b incorrect
    // amm/contract.py:134
    // assert b_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -4
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:137
    // pool_balance=self._current_pool_balance(),
    callsub _current_pool_balance
    // amm/contract.py:138
    // a_balance=self._current_a_balance(),
    callsub _current_a_balance
    // amm/contract.py:139
    // b_balance=self._current_b_balance(),
    callsub _current_b_balance
    // amm/contract.py:136-142
    // to_mint = tokens_to_mint(
    //     pool_balance=self._current_pool_balance(),
    //     a_balance=self._current_a_balance(),
    //     b_balance=self._current_b_balance(),
    //     a_amount=a_xfer.asset_amount,
    //     b_amount=b_xfer.asset_amount,
    // )
    uncover 4
    uncover 4
    callsub tokens_to_mint
    // amm/contract.py:143
    // assert to_mint > 0, "send amount too low"
    dup
    int 0
    >
    assert // send amount too low
    // amm/contract.py:145-146
    // # mint tokens
    // do_asset_transfer(receiver=Txn.sender, asset=self.pool_token, amount=to_mint)
    txn Sender
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:147
    // self._update_ratio()
    callsub _update_ratio
    retsub


// examples.amm.contract.ConstantProductAMM._check_bootstrapped() -> void:
_check_bootstrapped:
    // amm/contract.py:253-254
    // @subroutine
    // def _check_bootstrapped(self) -> None:
    proto 0 0
    // amm/contract.py:255
    // assert self.pool_token, "bootstrap method needs to be called first"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    assert // bootstrap method needs to be called first
    retsub


// examples.amm.contract.ConstantProductAMM._current_pool_balance() -> uint64:
_current_pool_balance:
    // amm/contract.py:294-295
    // @subroutine
    // def _current_pool_balance(self) -> UInt64:
    proto 0 1
    // amm/contract.py:296
    // return self.pool_token.balance(Global.current_application_address)
    global CurrentApplicationAddress
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    asset_holding_get AssetBalance
    assert // account opted into asset
    retsub


// examples.amm.contract.ConstantProductAMM._current_a_balance() -> uint64:
_current_a_balance:
    // amm/contract.py:298-299
    // @subroutine
    // def _current_a_balance(self) -> UInt64:
    proto 0 1
    // amm/contract.py:300
    // return self.asset_a.balance(Global.current_application_address)
    global CurrentApplicationAddress
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    asset_holding_get AssetBalance
    assert // account opted into asset
    retsub


// examples.amm.contract.ConstantProductAMM._current_b_balance() -> uint64:
_current_b_balance:
    // amm/contract.py:302-303
    // @subroutine
    // def _current_b_balance(self) -> UInt64:
    proto 0 1
    // amm/contract.py:304
    // return self.asset_b.balance(Global.current_application_address)
    global CurrentApplicationAddress
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    asset_holding_get AssetBalance
    assert // account opted into asset
    retsub


// examples.amm.contract.tokens_to_mint(pool_balance: uint64, a_balance: uint64, b_balance: uint64, a_amount: uint64, b_amount: uint64) -> uint64:
tokens_to_mint:
    // amm/contract.py:325-333
    // @subroutine
    // def tokens_to_mint(
    //     *,
    //     pool_balance: UInt64,
    //     a_balance: UInt64,
    //     b_balance: UInt64,
    //     a_amount: UInt64,
    //     b_amount: UInt64,
    // ) -> UInt64:
    proto 5 1
    byte ""
    dup
    // amm/contract.py:334
    // is_initial_mint = a_balance == a_amount and b_balance == b_amount
    frame_dig -4
    frame_dig -2
    ==
    bz tokens_to_mint_bool_false@3
    frame_dig -3
    frame_dig -1
    ==
    bz tokens_to_mint_bool_false@3
    int 1
    b tokens_to_mint_bool_merge@4

tokens_to_mint_bool_false@3:
    int 0

tokens_to_mint_bool_merge@4:
    // amm/contract.py:335
    // if is_initial_mint:
    bz tokens_to_mint_after_if_else@6
    // amm/contract.py:336
    // return op.sqrt(a_amount * b_amount) - SCALE
    frame_dig -2
    frame_dig -1
    *
    sqrt
    int 1000
    -
    frame_bury 0
    retsub

tokens_to_mint_after_if_else@6:
    // amm/contract.py:337
    // issued = TOTAL_SUPPLY - pool_balance
    int 10000000000
    frame_dig -5
    -
    // amm/contract.py:338
    // a_ratio = SCALE * a_amount // (a_balance - a_amount)
    int 1000
    frame_dig -2
    *
    frame_dig -4
    frame_dig -2
    -
    /
    dup
    frame_bury 0
    // amm/contract.py:339
    // b_ratio = SCALE * b_amount // (b_balance - b_amount)
    int 1000
    frame_dig -1
    *
    frame_dig -3
    frame_dig -1
    -
    /
    dup
    frame_bury 1
    // amm/contract.py:340
    // if a_ratio < b_ratio:
    <
    bz tokens_to_mint_else_body@8
    // amm/contract.py:341
    // return a_ratio * issued // SCALE
    frame_dig 0
    *
    int 1000
    /
    frame_bury 0
    retsub

tokens_to_mint_else_body@8:
    // amm/contract.py:343
    // return b_ratio * issued // SCALE
    frame_dig 1
    *
    int 1000
    /
    frame_bury 0
    retsub


// examples.amm.contract.ConstantProductAMM._update_ratio() -> void:
_update_ratio:
    // amm/contract.py:257-258
    // @subroutine
    // def _update_ratio(self) -> None:
    proto 0 0
    // amm/contract.py:259
    // a_balance = self._current_a_balance()
    callsub _current_a_balance
    // amm/contract.py:260
    // b_balance = self._current_b_balance()
    callsub _current_b_balance
    swap
    // amm/contract.py:262
    // self.ratio = a_balance * SCALE // b_balance
    int 1000
    *
    swap
    /
    byte "ratio"
    swap
    app_global_put
    retsub


// examples.amm.contract.ConstantProductAMM.burn(pool_xfer: uint64, pool_asset: uint64, a_asset: uint64, b_asset: uint64) -> void:
burn:
    // amm/contract.py:149-162
    // @arc4.abimethod(
    //     default_args={
    //         "pool_asset": "pool_token",
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    // def burn(
    //     self,
    //     pool_xfer: gtxn.AssetTransferTransaction,
    //     pool_asset: Asset,
    //     a_asset: Asset,
    //     b_asset: Asset,
    // ) -> None:
    proto 4 0
    // amm/contract.py:172
    // self._check_bootstrapped()
    callsub _check_bootstrapped
    // amm/contract.py:174
    // assert pool_asset == self.pool_token, "asset pool incorrect"
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    frame_dig -3
    ==
    assert // asset pool incorrect
    // amm/contract.py:175
    // assert a_asset == self.asset_a, "asset a incorrect"
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    frame_dig -2
    ==
    assert // asset a incorrect
    // amm/contract.py:176
    // assert b_asset == self.asset_b, "asset b incorrect"
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    frame_dig -1
    ==
    assert // asset b incorrect
    // amm/contract.py:179
    // pool_xfer.asset_receiver == Global.current_application_address
    frame_dig -4
    gtxns AssetReceiver
    global CurrentApplicationAddress
    ==
    // amm/contract.py:178-180
    // assert (
    //     pool_xfer.asset_receiver == Global.current_application_address
    // ), "receiver not app address"
    assert // receiver not app address
    // amm/contract.py:181
    // assert pool_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -4
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:182
    // assert pool_xfer.xfer_asset == self.pool_token, "asset pool incorrect"
    frame_dig -4
    gtxns XferAsset
    int 0
    byte "pool_token"
    app_global_get_ex
    assert // check pool_token exists
    ==
    assert // asset pool incorrect
    // amm/contract.py:183
    // assert pool_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -4
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:185-187
    // # Get the total number of tokens issued
    // # !important: this happens prior to receiving the current axfer of pool tokens
    // pool_balance = self._current_pool_balance()
    callsub _current_pool_balance
    // amm/contract.py:190
    // supply=self._current_a_balance(),
    callsub _current_a_balance
    // amm/contract.py:188-192
    // a_amt = tokens_to_burn(
    //     pool_balance=pool_balance,
    //     supply=self._current_a_balance(),
    //     amount=pool_xfer.asset_amount,
    // )
    dig 1
    swap
    dig 3
    callsub tokens_to_burn
    cover 2
    // amm/contract.py:195
    // supply=self._current_b_balance(),
    callsub _current_b_balance
    // amm/contract.py:193-197
    // b_amt = tokens_to_burn(
    //     pool_balance=pool_balance,
    //     supply=self._current_b_balance(),
    //     amount=pool_xfer.asset_amount,
    // )
    uncover 2
    callsub tokens_to_burn
    swap
    // amm/contract.py:199-200
    // # Send back commensurate amt of a
    // do_asset_transfer(receiver=Txn.sender, asset=self.asset_a, amount=a_amt)
    txn Sender
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:202-203
    // # Send back commensurate amt of b
    // do_asset_transfer(receiver=Txn.sender, asset=self.asset_b, amount=b_amt)
    txn Sender
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:204
    // self._update_ratio()
    callsub _update_ratio
    retsub


// examples.amm.contract.tokens_to_burn(pool_balance: uint64, supply: uint64, amount: uint64) -> uint64:
tokens_to_burn:
    // amm/contract.py:346-347
    // @subroutine
    // def tokens_to_burn(*, pool_balance: UInt64, supply: UInt64, amount: UInt64) -> UInt64:
    proto 3 1
    // amm/contract.py:348
    // issued = TOTAL_SUPPLY - pool_balance - amount
    int 10000000000
    frame_dig -3
    -
    frame_dig -1
    -
    // amm/contract.py:349
    // return supply * amount // issued
    frame_dig -2
    frame_dig -1
    *
    swap
    /
    retsub


// examples.amm.contract.ConstantProductAMM.swap(swap_xfer: uint64, a_asset: uint64, b_asset: uint64) -> void:
swap:
    // amm/contract.py:206-217
    // @arc4.abimethod(
    //     default_args={
    //         "a_asset": "asset_a",
    //         "b_asset": "asset_b",
    //     },
    // )
    // def swap(
    //     self,
    //     swap_xfer: gtxn.AssetTransferTransaction,
    //     a_asset: Asset,
    //     b_asset: Asset,
    // ) -> None:
    proto 3 0
    byte ""
    dup
    // amm/contract.py:225
    // self._check_bootstrapped()
    callsub _check_bootstrapped
    // amm/contract.py:227
    // assert a_asset == self.asset_a, "asset a incorrect"
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    frame_dig -2
    ==
    assert // asset a incorrect
    // amm/contract.py:228
    // assert b_asset == self.asset_b, "asset b incorrect"
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    frame_dig -1
    ==
    assert // asset b incorrect
    // amm/contract.py:230
    // assert swap_xfer.asset_amount > 0, "amount minimum not met"
    frame_dig -3
    gtxns AssetAmount
    dup
    int 0
    >
    assert // amount minimum not met
    // amm/contract.py:231
    // assert swap_xfer.sender == Txn.sender, "sender invalid"
    frame_dig -3
    gtxns Sender
    txn Sender
    ==
    assert // sender invalid
    // amm/contract.py:234
    // case self.asset_a:
    int 0
    byte "asset_a"
    app_global_get_ex
    assert // check asset_a exists
    // amm/contract.py:238
    // case self.asset_b:
    int 0
    byte "asset_b"
    app_global_get_ex
    assert // check asset_b exists
    // amm/contract.py:233
    // match swap_xfer.xfer_asset:
    frame_dig -3
    gtxns XferAsset
    // amm/contract.py:233-243
    // match swap_xfer.xfer_asset:
    //     case self.asset_a:
    //         in_supply = self._current_b_balance()
    //         out_supply = self._current_a_balance()
    //         out_asset = self.asset_a
    //     case self.asset_b:
    //         in_supply = self._current_a_balance()
    //         out_supply = self._current_b_balance()
    //         out_asset = self.asset_b
    //     case _:
    //         assert False, "asset id incorrect"
    match swap_switch_case_0@1 swap_switch_case_1@2
    // amm/contract.py:243
    // assert False, "asset id incorrect"
    err // asset id incorrect

swap_switch_case_0@1:
    // amm/contract.py:235
    // in_supply = self._current_b_balance()
    callsub _current_b_balance
    frame_bury 0
    // amm/contract.py:236
    // out_supply = self._current_a_balance()
    callsub _current_a_balance
    // amm/contract.py:237
    // out_asset = self.asset_a
    int 0
    byte "asset_a"
    app_global_get_ex
    swap
    frame_bury 1
    assert // check asset_a exists
    b swap_switch_case_next@4

swap_switch_case_1@2:
    // amm/contract.py:239
    // in_supply = self._current_a_balance()
    callsub _current_a_balance
    frame_bury 0
    // amm/contract.py:240
    // out_supply = self._current_b_balance()
    callsub _current_b_balance
    // amm/contract.py:241
    // out_asset = self.asset_b
    int 0
    byte "asset_b"
    app_global_get_ex
    swap
    frame_bury 1
    assert // check asset_b exists

swap_switch_case_next@4:
    // amm/contract.py:245-247
    // to_swap = tokens_to_swap(
    //     in_amount=swap_xfer.asset_amount, in_supply=in_supply, out_supply=out_supply
    // )
    frame_dig 2
    frame_dig 0
    uncover 2
    callsub tokens_to_swap
    // amm/contract.py:248
    // assert to_swap > 0, "send amount too low"
    dup
    int 0
    >
    assert // send amount too low
    // amm/contract.py:250
    // do_asset_transfer(receiver=Txn.sender, asset=out_asset, amount=to_swap)
    txn Sender
    frame_dig 1
    uncover 2
    callsub do_asset_transfer
    // amm/contract.py:251
    // self._update_ratio()
    callsub _update_ratio
    retsub


// examples.amm.contract.tokens_to_swap(in_amount: uint64, in_supply: uint64, out_supply: uint64) -> uint64:
tokens_to_swap:
    // amm/contract.py:352-353
    // @subroutine
    // def tokens_to_swap(*, in_amount: UInt64, in_supply: UInt64, out_supply: UInt64) -> UInt64:
    proto 3 1
    // amm/contract.py:354
    // in_total = SCALE * (in_supply - in_amount) + (in_amount * FACTOR)
    frame_dig -2
    frame_dig -3
    -
    int 1000
    *
    frame_dig -3
    int 995
    *
    dup
    cover 2
    +
    swap
    // amm/contract.py:355
    // out_total = in_amount * FACTOR * out_supply
    frame_dig -1
    *
    // amm/contract.py:356
    // return out_total // in_total
    swap
    /
    retsub


// examples.amm.contract.ConstantProductAMM.__init__() -> void:
__init__:
    // amm/contract.py:28
    // def __init__(self) -> None:
    proto 0 0
    // amm/contract.py:32-33
    // # The asset id of asset A
    // self.asset_a = Asset(0)
    byte "asset_a"
    int 0
    app_global_put
    // amm/contract.py:34-35
    // # The asset id of asset B
    // self.asset_b = Asset(0)
    byte "asset_b"
    int 0
    app_global_put
    // amm/contract.py:36-37
    // # The current governor of this contract, allowed to do admin type actions
    // self.governor = Txn.sender
    byte "governor"
    txn Sender
    app_global_put
    // amm/contract.py:38-39
    // # The asset id of the Pool Token, used to track share of pool the holder may recover
    // self.pool_token = Asset(0)
    byte "pool_token"
    int 0
    app_global_put
    // amm/contract.py:40-41
    // # The ratio between assets (A*Scale/B)
    // self.ratio = UInt64(0)
    byte "ratio"
    int 0
    app_global_put
    retsub
", "clear": "I3ByYWdtYSB2ZXJzaW9uIDEwCgpleGFtcGxlcy5hbW0uY29udHJhY3QuQ29uc3RhbnRQcm9kdWN0QU1NLmNsZWFyX3N0YXRlX3Byb2dyYW06CiAgICAvLyBhbW0vY29udHJhY3QucHk6MjcKICAgIC8vIGNsYXNzIENvbnN0YW50UHJvZHVjdEFNTShBUkM0Q29udHJhY3QpOgogICAgaW50IDEKICAgIHJldHVybgo=" }, "state": { diff --git a/examples/amm/out_O2/ConstantProductAMM.approval.teal b/examples/amm/out_O2/ConstantProductAMM.approval.teal index 1de180f63c..c560cd68d5 100644 --- a/examples/amm/out_O2/ConstantProductAMM.approval.teal +++ b/examples/amm/out_O2/ConstantProductAMM.approval.teal @@ -257,7 +257,7 @@ _create_pool_token: assert // asset exists concat global CurrentApplicationAddress - global CurrentApplicationAddress + dup int 0 itxn_field Fee itxn_field ConfigAssetReserve diff --git a/examples/calculator/out/MyContract.approval.teal b/examples/calculator/out/MyContract.approval.teal index 99c6b851bf..da739b617d 100644 --- a/examples/calculator/out/MyContract.approval.teal +++ b/examples/calculator/out/MyContract.approval.teal @@ -19,7 +19,7 @@ examples.calculator.contract.MyContract.approval_program: int 0 // calculator/contract.py:24 // action = UInt64(0) - int 0 + dup bury 3 // calculator/contract.py:22 // a = UInt64(0) diff --git a/examples/calculator/out_O2/MyContract.approval.teal b/examples/calculator/out_O2/MyContract.approval.teal index 005bc881fa..bccb395df3 100644 --- a/examples/calculator/out_O2/MyContract.approval.teal +++ b/examples/calculator/out_O2/MyContract.approval.teal @@ -9,7 +9,7 @@ examples.calculator.contract.MyContract.approval_program: int 0 bury 3 int 0 - int 0 + dup bury 3 int 0 itob diff --git a/examples/sizes.txt b/examples/sizes.txt index 89dc1d57b0..125d7acb91 100644 --- a/examples/sizes.txt +++ b/examples/sizes.txt @@ -1,17 +1,17 @@ Name O0 size O1 size O1 ⏷ O2 size O2 ⏷ abi_routing/Reference 1179 1019 160 1019 0 - amm/ConstantProductAMM 1213 1114 99 1114 0 + amm/ConstantProductAMM 1213 1113 100 1113 0 application/Reference 175 168 7 168 0 arc4_numeric_comparisons/UIntNOrdering 1220 908 312 908 0 arc4_types/Arc4Arrays 588 376 212 376 0 - arc4_types/Arc4BoolEval 731 20 711 20 0 + arc4_types/Arc4BoolEval 731 19 712 19 0 arc4_types/Arc4BoolType 329 57 272 57 0 arc4_types/Arc4DynamicBytes 247 128 119 128 0 arc4_types/Arc4DynamicStringArray 230 112 118 112 0 arc4_types/Arc4MutableParams 362 222 140 220 2 arc4_types/Arc4Mutation 2803 1452 1351 1451 1 arc4_types/Arc4NumericTypes 538 8 530 8 0 - arc4_types/Arc4RefTypes 47 43 4 43 0 + arc4_types/Arc4RefTypes 47 39 8 39 0 arc4_types/Arc4StringTypes 309 8 301 8 0 arc4_types/Arc4StructsFromAnotherModule 67 12 55 12 0 arc4_types/Arc4StructsType 296 237 59 237 0 @@ -21,7 +21,7 @@ augmented_assignment/Augmented 157 156 1 156 0 avm_types_in_abi/Test 226 173 53 173 0 biguint_binary_ops/BiguintBinaryOps 280 216 64 216 0 - boolean_binary_ops/BooleanBinaryOps 308 302 6 302 0 + boolean_binary_ops/BooleanBinaryOps 308 298 10 298 0 bytes_ops/BiguintBinaryOps 143 139 4 139 0 calculator 346 317 29 315 2 callsub 32 32 0 32 0 @@ -38,7 +38,7 @@ global_state/AppState 305 301 4 301 0 hello_world/HelloWorld 23 22 1 22 0 hello_world_arc4/HelloWorld 110 89 21 89 0 - inner_transactions 1847 1174 673 1174 0 + inner_transactions 1847 1173 674 1173 0 inner_transactions/Greeter 325 302 23 302 0 inner_transactions/itxn_loop 203 184 19 184 0 intrinsics/ImmediateVariants 164 162 2 162 0 @@ -46,7 +46,7 @@ less_simple 171 148 23 148 0 local_state/LocalState 305 298 7 286 12 local_state/LocalStateWithOffsets 318 309 9 297 12 - log 172 168 4 168 0 + log 172 167 5 167 0 match 490 455 35 455 0 merkle/MerkleTree 217 210 7 210 0 nested_loops/Nested 243 201 42 201 0 @@ -65,9 +65,9 @@ stubs/Bytes 1769 258 1511 258 0 stubs/Uint64 371 8 363 8 0 template_variables/TemplateVariables 168 155 13 155 0 - too_many_permutations 108 107 1 107 0 + too_many_permutations 108 106 2 106 0 transaction/Transaction 893 849 44 849 0 - tuple_support/TupleSupport 442 294 148 294 0 + tuple_support/TupleSupport 442 292 150 292 0 typed_abi_call/Greeter 1285 1168 117 1168 0 typed_abi_call/Logger 569 478 91 478 0 unary/Unary 134 96 38 96 0 diff --git a/examples/voting/out/VotingRoundApp.approval.teal b/examples/voting/out/VotingRoundApp.approval.teal index 18dd2ae77d..48ca4e8374 100644 --- a/examples/voting/out/VotingRoundApp.approval.teal +++ b/examples/voting/out/VotingRoundApp.approval.teal @@ -552,7 +552,7 @@ close: int 0 // voting/voting.py:129 // for question_index, question_options_arc in uenumerate(self.option_counts): - int 0 + dup byte "option_counts" app_global_get_ex assert // check option_counts exists @@ -1187,7 +1187,7 @@ vote: int 0 // voting/voting.py:192 // for question_index in urange(questions_count): - int 0 + dup vote_for_header@1: // voting/voting.py:192 diff --git a/examples/voting/out/VotingRoundApp.arc32.json b/examples/voting/out/VotingRoundApp.arc32.json index d924a39fb3..b3e543642b 100644 --- a/examples/voting/out/VotingRoundApp.arc32.json +++ b/examples/voting/out/VotingRoundApp.arc32.json @@ -51,7 +51,7 @@ } }, "source": { - "approval": "#pragma version 10

examples.voting.voting.VotingRoundApp.approval_program:
    txn ApplicationID
    bnz main_entrypoint@2
    callsub __init__

main_entrypoint@2:
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    method "create(string,byte[],string,uint64,uint64,uint8[],uint64,string)void"
    method "bootstrap(pay)void"
    method "close()void"
    method "get_preconditions(byte[])(uint64,uint64,uint64,uint64)"
    method "vote(pay,byte[],uint8[])void"
    txna ApplicationArgs 0
    match main_create_route@3 main_bootstrap_route@4 main_close_route@5 main_get_preconditions_route@6 main_vote_route@7
    err // reject transaction

main_create_route@3:
    // voting/voting.py:55
    // @arc4.abimethod(create=True)
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    !
    assert // is creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txna ApplicationArgs 1
    txna ApplicationArgs 2
    extract 2 0
    txna ApplicationArgs 3
    txna ApplicationArgs 4
    btoi
    txna ApplicationArgs 5
    btoi
    txna ApplicationArgs 6
    txna ApplicationArgs 7
    btoi
    txna ApplicationArgs 8
    // voting/voting.py:55
    // @arc4.abimethod(create=True)
    callsub create
    int 1
    return

main_bootstrap_route@4:
    // voting/voting.py:79
    // @arc4.abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int pay
    ==
    assert // transaction type is pay
    // voting/voting.py:79
    // @arc4.abimethod
    callsub bootstrap
    int 1
    return

main_close_route@5:
    // voting/voting.py:107
    // @arc4.abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    callsub close
    int 1
    return

main_get_preconditions_route@6:
    // voting/voting.py:157
    // @arc4.abimethod(readonly=True)
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txna ApplicationArgs 1
    // voting/voting.py:157
    // @arc4.abimethod(readonly=True)
    callsub get_preconditions
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_vote_route@7:
    // voting/voting.py:166
    // @arc4.abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int pay
    ==
    assert // transaction type is pay
    txna ApplicationArgs 1
    extract 2 0
    txna ApplicationArgs 2
    // voting/voting.py:166
    // @arc4.abimethod
    callsub vote
    int 1
    return


// examples.voting.voting.VotingRoundApp.create(vote_id: bytes, snapshot_public_key: bytes, metadata_ipfs_cid: bytes, start_time: uint64, end_time: uint64, option_counts: bytes, quorum: uint64, nft_image_url: bytes) -> void:
create:
    // voting/voting.py:55-66
    // @arc4.abimethod(create=True)
    // def create(
    //     self,
    //     vote_id: arc4.String,
    //     snapshot_public_key: Bytes,
    //     metadata_ipfs_cid: arc4.String,
    //     start_time: UInt64,
    //     end_time: UInt64,
    //     option_counts: VoteIndexArray,
    //     quorum: UInt64,
    //     nft_image_url: arc4.String,
    // ) -> None:
    proto 8 0
    // voting/voting.py:67
    // assert start_time < end_time, "End time should be after start time"
    frame_dig -5
    frame_dig -4
    <
    assert // End time should be after start time
    // voting/voting.py:68
    // assert end_time >= Global.latest_timestamp, "End time should be in the future"
    frame_dig -4
    global LatestTimestamp
    >=
    assert // End time should be in the future
    // voting/voting.py:70
    // self.vote_id = vote_id.decode()
    frame_dig -8
    extract 2 0
    byte "vote_id"
    swap
    app_global_put
    // voting/voting.py:71
    // self.snapshot_public_key = snapshot_public_key
    byte "snapshot_public_key"
    frame_dig -7
    app_global_put
    // voting/voting.py:72
    // self.metadata_ipfs_cid = metadata_ipfs_cid.decode()
    frame_dig -6
    extract 2 0
    byte "metadata_ipfs_cid"
    swap
    app_global_put
    // voting/voting.py:73
    // self.start_time = start_time
    byte "start_time"
    frame_dig -5
    app_global_put
    // voting/voting.py:74
    // self.end_time = end_time
    byte "end_time"
    frame_dig -4
    app_global_put
    // voting/voting.py:75
    // self.quorum = quorum
    byte "quorum"
    frame_dig -2
    app_global_put
    // voting/voting.py:76
    // self.nft_image_url = nft_image_url.decode()
    frame_dig -1
    extract 2 0
    byte "nft_image_url"
    swap
    app_global_put
    // voting/voting.py:77
    // self.store_option_counts(option_counts.copy())
    frame_dig -3
    callsub store_option_counts
    pop
    retsub


// examples.voting.voting.VotingRoundApp.store_option_counts(option_counts: bytes) -> bytes:
store_option_counts:
    // voting/voting.py:215-216
    // @subroutine
    // def store_option_counts(self, option_counts: VoteIndexArray) -> None:
    proto 1 1
    // voting/voting.py:217
    // assert option_counts.length, "option_counts should be non-empty"
    frame_dig -1
    int 0
    extract_uint16
    dupn 2
    assert // option_counts should be non-empty
    // voting/voting.py:218
    // assert option_counts.length <= 112, "Can't have more than 112 questions"
    int 112
    <=
    assert // Can't have more than 112 questions
    // voting/voting.py:220
    // total_options = UInt64(0)
    int 0
    // voting/voting.py:221
    // for item in option_counts:
    frame_dig -1
    extract 2 0
    int 0

store_option_counts_for_header@1:
    // voting/voting.py:221
    // for item in option_counts:
    frame_dig 3
    frame_dig 0
    <
    bz store_option_counts_after_for@4
    frame_dig 2
    frame_dig 3
    dup
    cover 2
    int 1
    extract3
    // voting/voting.py:222
    // total_options += item.decode()
    btoi
    frame_dig 1
    +
    frame_bury 1
    int 1
    +
    frame_bury 3
    b store_option_counts_for_header@1

store_option_counts_after_for@4:
    // voting/voting.py:223
    // assert total_options <= 128, "Can't have more than 128 vote options"
    frame_dig 1
    dup
    int 128
    <=
    assert // Can't have more than 128 vote options
    // voting/voting.py:225
    // self.option_counts = option_counts.copy()
    byte "option_counts"
    frame_dig -1
    app_global_put
    // voting/voting.py:226
    // self.total_options = total_options
    byte "total_options"
    swap
    app_global_put
    frame_dig -1
    frame_bury 0
    retsub


// examples.voting.voting.VotingRoundApp.bootstrap(fund_min_bal_req: uint64) -> void:
bootstrap:
    // voting/voting.py:79-80
    // @arc4.abimethod
    // def bootstrap(self, fund_min_bal_req: gtxn.PaymentTransaction) -> None:
    proto 1 0
    // voting/voting.py:81
    // assert not self.is_bootstrapped, "Must not be already bootstrapped"
    int 0
    byte "is_bootstrapped"
    app_global_get_ex
    assert // check is_bootstrapped exists
    !
    assert // Must not be already bootstrapped
    // voting/voting.py:82
    // self.is_bootstrapped = True
    byte "is_bootstrapped"
    int 1
    app_global_put
    // voting/voting.py:85
    // fund_min_bal_req.receiver == Global.current_application_address
    frame_dig -1
    gtxns Receiver
    global CurrentApplicationAddress
    ==
    // voting/voting.py:84-86
    // assert (
    //     fund_min_bal_req.receiver == Global.current_application_address
    // ), "Payment must be to app address"
    assert // Payment must be to app address
    // voting/voting.py:88
    // tally_box_size = self.total_options * VOTE_COUNT_BYTES
    int 0
    byte "total_options"
    app_global_get_ex
    assert // check total_options exists
    int 8
    *
    // voting/voting.py:98-99
    // # tally box value
    // + (tally_box_size * BOX_BYTE_MIN_BALANCE)
    dup
    int 400
    *
    // voting/voting.py:90-97
    // # minimum balance req for: ALGOs + Vote result NFT asset
    // ASSET_MIN_BALANCE * 2
    // # create NFT fee
    // + 1000
    // # tally box
    // + BOX_FLAT_MIN_BALANCE
    // # tally box key "V"
    // + BOX_BYTE_MIN_BALANCE
    int 203900
    // voting/voting.py:90-99
    // # minimum balance req for: ALGOs + Vote result NFT asset
    // ASSET_MIN_BALANCE * 2
    // # create NFT fee
    // + 1000
    // # tally box
    // + BOX_FLAT_MIN_BALANCE
    // # tally box key "V"
    // + BOX_BYTE_MIN_BALANCE
    // # tally box value
    // + (tally_box_size * BOX_BYTE_MIN_BALANCE)
    +
    // voting/voting.py:101
    // log(min_balance_req)
    dup
    itob
    log
    // voting/voting.py:103
    // fund_min_bal_req.amount == min_balance_req
    frame_dig -1
    gtxns Amount
    ==
    // voting/voting.py:102-104
    // assert (
    //     fund_min_bal_req.amount == min_balance_req
    // ), "Payment must be for the exact min balance requirement"
    assert // Payment must be for the exact min balance requirement
    // voting/voting.py:105
    // assert op.Box.create(TALLY_BOX_KEY, tally_box_size)
    byte "V"
    swap
    box_create
    assert
    retsub


// examples.voting.voting.VotingRoundApp.close() -> void:
close:
    // voting/voting.py:107-108
    // @arc4.abimethod
    // def close(self) -> None:
    proto 0 0
    int 0
    dup
    byte ""
    dupn 2
    // voting/voting.py:109
    // ensure_budget(20000, fee_source=OpUpFeeSource.GroupCredit)
    int 20000
    int 0
    callsub ensure_budget
    // voting/voting.py:110
    // assert not self.close_time, "Already closed"
    int 0
    byte "close_time"
    app_global_get_ex
    bury 1
    !
    assert // Already closed
    // voting/voting.py:111
    // self.close_time.value = Global.latest_timestamp
    byte "close_time"
    global LatestTimestamp
    app_global_put
    // voting/voting.py:116
    // + self.vote_id
    int 0
    byte "vote_id"
    app_global_get_ex
    assert // check vote_id exists
    // voting/voting.py:114-115
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    byte "{\"standard\":\"arc69\",\"description\":\"This is a voting result NFT for voting round with ID "
    // voting/voting.py:114-116
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    swap
    concat
    // voting/voting.py:117
    // + b'.","properties":{"metadata":"ipfs://'
    byte ".\",\"properties\":{\"metadata\":\"ipfs://"
    // voting/voting.py:114-117
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    concat
    // voting/voting.py:118
    // + self.metadata_ipfs_cid
    int 0
    byte "metadata_ipfs_cid"
    app_global_get_ex
    assert // check metadata_ipfs_cid exists
    // voting/voting.py:114-118
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    concat
    // voting/voting.py:119
    // + b'","id":"'
    byte "\",\"id\":\""
    // voting/voting.py:114-119
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    concat
    // voting/voting.py:120
    // + self.vote_id
    int 0
    byte "vote_id"
    app_global_get_ex
    assert // check vote_id exists
    // voting/voting.py:114-120
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    concat
    // voting/voting.py:121
    // + b'","quorum":'
    byte "\",\"quorum\":"
    // voting/voting.py:114-121
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    concat
    // voting/voting.py:122
    // + itoa(self.quorum)
    int 0
    byte "quorum"
    app_global_get_ex
    assert // check quorum exists
    callsub itoa
    // voting/voting.py:114-122
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    concat
    // voting/voting.py:123
    // + b',"voterCount":'
    byte ",\"voterCount\":"
    // voting/voting.py:114-123
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    // + b',"voterCount":'
    concat
    // voting/voting.py:124
    // + itoa(self.voter_count)
    int 0
    byte "voter_count"
    app_global_get_ex
    assert // check voter_count exists
    callsub itoa
    // voting/voting.py:114-124
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    // + b',"voterCount":'
    // + itoa(self.voter_count)
    concat
    // voting/voting.py:125
    // + b',"tallies":['
    byte ",\"tallies\":["
    // voting/voting.py:114-125
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    // + b',"voterCount":'
    // + itoa(self.voter_count)
    // + b',"tallies":['
    concat
    // voting/voting.py:128
    // current_index = UInt64(0)
    int 0
    // voting/voting.py:129
    // for question_index, question_options_arc in uenumerate(self.option_counts):
    int 0
    byte "option_counts"
    app_global_get_ex
    assert // check option_counts exists
    dup
    int 0
    extract_uint16
    swap
    extract 2 0
    int 0

close_for_header@1:
    // voting/voting.py:129
    // for question_index, question_options_arc in uenumerate(self.option_counts):
    frame_dig 9
    frame_dig 7
    <
    bz close_after_for@15
    frame_dig 8
    frame_dig 9
    dup
    cover 2
    int 1
    extract3
    frame_bury 1
    // voting/voting.py:130
    // if question_index > 0:
    int 0
    >
    frame_dig 5
    frame_bury 0
    bz close_after_if_else@4
    // voting/voting.py:131
    // note += b","
    frame_dig 5
    byte ","
    concat
    frame_bury 0

close_after_if_else@4:
    frame_dig 0
    dup
    frame_bury 5
    // voting/voting.py:132
    // question_options = question_options_arc.decode()
    frame_dig 1
    btoi
    dup
    frame_bury 4
    // voting/voting.py:133
    // if question_options > 0:
    int 0
    >
    frame_dig 6
    frame_bury 2
    swap
    frame_bury 0
    bz close_after_if_else@13
    // voting/voting.py:134
    // note += b"["
    frame_dig 5
    byte "["
    concat
    frame_bury 5
    // voting/voting.py:135
    // for option_index in urange(question_options):
    int 0
    frame_bury 3

close_for_header@6:
    // voting/voting.py:135
    // for option_index in urange(question_options):
    frame_dig 3
    frame_dig 4
    <
    bz close_after_for@12
    // voting/voting.py:136
    // if option_index > 0:
    frame_dig 3
    int 0
    >
    frame_dig 5
    frame_bury 0
    bz close_after_if_else@9
    // voting/voting.py:137
    // note += b","
    frame_dig 5
    byte ","
    concat
    frame_bury 0

close_after_if_else@9:
    frame_dig 0
    // voting/voting.py:138
    // votes_for_option = get_vote_from_box(current_index)
    frame_dig 6
    dup
    cover 2
    callsub get_vote_from_box
    // voting/voting.py:139
    // note += itoa(votes_for_option)
    callsub itoa
    concat
    frame_bury 5
    // voting/voting.py:140
    // current_index += 1
    int 1
    +
    frame_bury 6
    // voting/voting.py:135
    // for option_index in urange(question_options):
    frame_dig 3
    int 1
    +
    frame_bury 3
    b close_for_header@6

close_after_for@12:
    // voting/voting.py:141
    // note += b"]"
    frame_dig 5
    byte "]"
    concat
    frame_dig 6
    frame_bury 2
    frame_bury 0

close_after_if_else@13:
    frame_dig 2
    frame_bury 6
    frame_dig 0
    frame_bury 5
    frame_dig 9
    int 1
    +
    frame_bury 9
    b close_for_header@1

close_after_for@15:
    // voting/voting.py:142
    // note += b"]}}"
    frame_dig 5
    byte "]}}"
    concat
    // voting/voting.py:144-153
    // itxn.AssetConfig(
    //     total=1,
    //     decimals=0,
    //     default_frozen=False,
    //     asset_name=b"[VOTE RESULT] " + self.vote_id,
    //     unit_name=b"VOTERSLT",
    //     url=self.nft_image_url,
    //     note=note,
    // )
    // .submit()
    itxn_begin
    // voting/voting.py:148
    // asset_name=b"[VOTE RESULT] " + self.vote_id,
    int 0
    byte "vote_id"
    app_global_get_ex
    assert // check vote_id exists
    byte "[VOTE RESULT] "
    swap
    concat
    // voting/voting.py:150
    // url=self.nft_image_url,
    int 0
    byte "nft_image_url"
    app_global_get_ex
    assert // check nft_image_url exists
    uncover 2
    itxn_field Note
    itxn_field ConfigAssetURL
    // voting/voting.py:149
    // unit_name=b"VOTERSLT",
    byte "VOTERSLT"
    itxn_field ConfigAssetUnitName
    itxn_field ConfigAssetName
    // voting/voting.py:147
    // default_frozen=False,
    int 0
    itxn_field ConfigAssetDefaultFrozen
    // voting/voting.py:146
    // decimals=0,
    int 0
    itxn_field ConfigAssetDecimals
    // voting/voting.py:145
    // total=1,
    int 1
    itxn_field ConfigAssetTotal
    // voting/voting.py:144
    // itxn.AssetConfig(
    int acfg
    itxn_field TypeEnum
    // voting/voting.py:144-153
    // itxn.AssetConfig(
    //     total=1,
    //     decimals=0,
    //     default_frozen=False,
    //     asset_name=b"[VOTE RESULT] " + self.vote_id,
    //     unit_name=b"VOTERSLT",
    //     url=self.nft_image_url,
    //     note=note,
    // )
    // .submit()
    itxn_submit
    // voting/voting.py:143
    // self.nft_asset_id = (
    byte "nft_asset_id"
    // voting/voting.py:144-153
    // itxn.AssetConfig(
    //     total=1,
    //     decimals=0,
    //     default_frozen=False,
    //     asset_name=b"[VOTE RESULT] " + self.vote_id,
    //     unit_name=b"VOTERSLT",
    //     url=self.nft_image_url,
    //     note=note,
    // )
    // .submit()
    itxn CreatedAssetID
    // voting/voting.py:143-155
    // self.nft_asset_id = (
    //     itxn.AssetConfig(
    //         total=1,
    //         decimals=0,
    //         default_frozen=False,
    //         asset_name=b"[VOTE RESULT] " + self.vote_id,
    //         unit_name=b"VOTERSLT",
    //         url=self.nft_image_url,
    //         note=note,
    //     )
    //     .submit()
    //     .created_asset.id
    // )
    app_global_put
    retsub


// puyapy.ensure_budget(required_budget: uint64, fee_source: uint64) -> void:
ensure_budget:
    // <puya>/puyapy.py:11-17
    proto 2 0
    // <puya>/puyapy.py:18
    frame_dig -2
    int 10
    +

ensure_budget_while_top@1:
    // <puya>/puyapy.py:19
    frame_dig 0
    global OpcodeBudget
    >
    bz ensure_budget_after_while@7
    // <puya>/puyapy.py:20
    itxn_begin
    // <puya>/puyapy.py:21
    int appl
    itxn_field TypeEnum
    // <puya>/puyapy.py:22
    int DeleteApplication
    itxn_field OnCompletion
    // <puya>/puyapy.py:23
    byte 0x068101
    itxn_field ApprovalProgram
    // <puya>/puyapy.py:24
    byte 0x068101
    itxn_field ClearStateProgram
    // <puya>/puyapy.py:25-29
    frame_dig -1
    switch ensure_budget_switch_case_0@3 ensure_budget_switch_case_1@4
    b ensure_budget_switch_case_next@6

ensure_budget_switch_case_0@3:
    // <puya>/puyapy.py:27
    int 0
    itxn_field Fee
    b ensure_budget_switch_case_next@6

ensure_budget_switch_case_1@4:
    // <puya>/puyapy.py:29
    global MinTxnFee
    itxn_field Fee

ensure_budget_switch_case_next@6:
    // <puya>/puyapy.py:30
    itxn_submit
    b ensure_budget_while_top@1

ensure_budget_after_while@7:
    retsub


// examples.voting.voting.itoa(i: uint64) -> bytes:
itoa:
    // voting/voting.py:253-254
    // @subroutine
    // def itoa(i: UInt64) -> Bytes:
    proto 1 1
    // voting/voting.py:257
    // if i < radix:
    frame_dig -1
    // voting/voting.py:256
    // radix = digits.length
    int 10
    // voting/voting.py:257
    // if i < radix:
    <
    bz itoa_after_if_else@2
    // voting/voting.py:258
    // return digits[i]
    frame_dig -1
    int 1
    +
    // voting/voting.py:255
    // digits = Bytes(b"0123456789")
    byte "0123456789"
    // voting/voting.py:258
    // return digits[i]
    frame_dig -1
    uncover 2
    substring3
    retsub

itoa_after_if_else@2:
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    frame_dig -1
    // voting/voting.py:256
    // radix = digits.length
    int 10
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    /
    callsub itoa
    frame_dig -1
    // voting/voting.py:256
    // radix = digits.length
    int 10
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    %
    dup
    int 1
    +
    // voting/voting.py:255
    // digits = Bytes(b"0123456789")
    byte "0123456789"
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    cover 2
    substring3
    concat
    retsub


// examples.voting.voting.get_vote_from_box(index: uint64) -> uint64:
get_vote_from_box:
    // voting/voting.py:238-239
    // @subroutine
    // def get_vote_from_box(index: UInt64) -> UInt64:
    proto 1 1
    // voting/voting.py:240
    // box_data, exists = op.Box.get(TALLY_BOX_KEY)
    byte "V"
    box_get
    // voting/voting.py:241
    // assert exists, "Box not created"
    assert // Box not created
    // voting/voting.py:242
    // return op.btoi(op.extract(box_data, index, VOTE_COUNT_BYTES))
    frame_dig -1
    int 8
    extract3
    btoi
    retsub


// examples.voting.voting.VotingRoundApp.get_preconditions(signature: bytes) -> bytes:
get_preconditions:
    // voting/voting.py:157-158
    // @arc4.abimethod(readonly=True)
    // def get_preconditions(self, signature: arc4.DynamicBytes) -> VotingPreconditions:
    proto 1 1
    // voting/voting.py:160
    // is_voting_open=arc4.UInt64(self.voting_open()),
    callsub voting_open
    itob
    // voting/voting.py:161
    // is_allowed_to_vote=arc4.UInt64(self.allowed_to_vote(signature.bytes[2:])),
    frame_dig -1
    len
    int 2
    dig 1
    <
    swap
    dup
    int 2
    uncover 3
    select
    frame_dig -1
    swap
    uncover 2
    substring3
    callsub allowed_to_vote
    itob
    // voting/voting.py:162
    // has_already_voted=arc4.UInt64(self.already_voted()),
    callsub already_voted
    itob
    // voting/voting.py:163
    // current_time=arc4.UInt64(Global.latest_timestamp),
    global LatestTimestamp
    itob
    // voting/voting.py:159-164
    // return VotingPreconditions(
    //     is_voting_open=arc4.UInt64(self.voting_open()),
    //     is_allowed_to_vote=arc4.UInt64(self.allowed_to_vote(signature.bytes[2:])),
    //     has_already_voted=arc4.UInt64(self.already_voted()),
    //     current_time=arc4.UInt64(Global.latest_timestamp),
    // )
    uncover 3
    uncover 3
    concat
    uncover 2
    concat
    swap
    concat
    retsub


// examples.voting.voting.VotingRoundApp.voting_open() -> uint64:
voting_open:
    // voting/voting.py:202-203
    // @subroutine
    // def voting_open(self) -> bool:
    proto 0 1
    byte ""
    // voting/voting.py:205
    // self.is_bootstrapped
    int 0
    byte "is_bootstrapped"
    app_global_get_ex
    assert // check is_bootstrapped exists
    // voting/voting.py:205-207
    // self.is_bootstrapped
    // and not self.close_time
    // and self.start_time <= Global.latest_timestamp <= self.end_time
    bz voting_open_bool_false@5
    // voting/voting.py:206
    // and not self.close_time
    int 0
    byte "close_time"
    app_global_get_ex
    bury 1
    bnz voting_open_bool_false@5
    // voting/voting.py:207
    // and self.start_time <= Global.latest_timestamp <= self.end_time
    int 0
    byte "start_time"
    app_global_get_ex
    assert // check start_time exists
    global LatestTimestamp
    dup
    frame_bury 0
    <=
    bz voting_open_bool_false@5
    int 0
    byte "end_time"
    app_global_get_ex
    assert // check end_time exists
    frame_dig 0
    >=
    bz voting_open_bool_false@5
    int 1
    b voting_open_bool_merge@6

voting_open_bool_false@5:
    int 0

voting_open_bool_merge@6:
    // voting/voting.py:204-208
    // return (
    //     self.is_bootstrapped
    //     and not self.close_time
    //     and self.start_time <= Global.latest_timestamp <= self.end_time
    // )
    swap
    retsub


// examples.voting.voting.VotingRoundApp.allowed_to_vote(signature: bytes) -> uint64:
allowed_to_vote:
    // voting/voting.py:228-229
    // @subroutine
    // def allowed_to_vote(self, signature: Bytes) -> bool:
    proto 1 1
    // voting/voting.py:230
    // ensure_budget(2000)
    int 2000
    int 2
    callsub ensure_budget
    // voting/voting.py:232
    // Txn.sender.bytes,
    txn Sender
    // voting/voting.py:234
    // self.snapshot_public_key,
    int 0
    byte "snapshot_public_key"
    app_global_get_ex
    assert // check snapshot_public_key exists
    // voting/voting.py:231-235
    // return op.ed25519verify_bare(
    //     Txn.sender.bytes,
    //     signature,
    //     self.snapshot_public_key,
    // )
    frame_dig -1
    swap
    ed25519verify_bare
    retsub


// examples.voting.voting.VotingRoundApp.already_voted() -> uint64:
already_voted:
    // voting/voting.py:210-211
    // @subroutine
    // def already_voted(self) -> bool:
    proto 0 1
    // voting/voting.py:212
    // (votes, exists) = op.Box.get(Txn.sender.bytes)
    txn Sender
    box_get
    bury 1
    // voting/voting.py:213
    // return exists
    retsub


// examples.voting.voting.VotingRoundApp.vote(fund_min_bal_req: uint64, signature: bytes, answer_ids: bytes) -> void:
vote:
    // voting/voting.py:166-172
    // @arc4.abimethod
    // def vote(
    //     self,
    //     fund_min_bal_req: gtxn.PaymentTransaction,
    //     signature: Bytes,
    //     answer_ids: VoteIndexArray,
    // ) -> None:
    proto 3 0
    // voting/voting.py:173
    // ensure_budget(7700, fee_source=OpUpFeeSource.GroupCredit)
    int 7700
    int 0
    callsub ensure_budget
    // voting/voting.py:174-175
    // # Check voting preconditions
    // assert self.allowed_to_vote(signature), "Not allowed to vote"
    frame_dig -2
    callsub allowed_to_vote
    assert // Not allowed to vote
    // voting/voting.py:176
    // assert self.voting_open(), "Voting not open"
    callsub voting_open
    assert // Voting not open
    // voting/voting.py:177
    // assert not self.already_voted(), "Already voted"
    callsub already_voted
    !
    assert // Already voted
    // voting/voting.py:178
    // questions_count = self.option_counts.length
    int 0
    byte "option_counts"
    app_global_get_ex
    assert // check option_counts exists
    int 0
    extract_uint16
    dup
    // voting/voting.py:179
    // assert answer_ids.length == questions_count, "Number of answers incorrect"
    frame_dig -1
    int 0
    extract_uint16
    dup
    cover 2
    dup
    uncover 2
    ==
    assert // Number of answers incorrect
    // voting/voting.py:182
    // (32 + 2 + VOTE_INDEX_BYTES * answer_ids.length) * BOX_BYTE_MIN_BALANCE
    int 34
    +
    int 400
    *
    // voting/voting.py:180-181
    // # Check voter box is funded
    // min_bal_req = BOX_FLAT_MIN_BALANCE + (
    int 2500
    // voting/voting.py:180-183
    // # Check voter box is funded
    // min_bal_req = BOX_FLAT_MIN_BALANCE + (
    //     (32 + 2 + VOTE_INDEX_BYTES * answer_ids.length) * BOX_BYTE_MIN_BALANCE
    // )
    +
    // voting/voting.py:185
    // fund_min_bal_req.receiver == Global.current_application_address
    frame_dig -3
    gtxns Receiver
    global CurrentApplicationAddress
    ==
    // voting/voting.py:184-186
    // assert (
    //     fund_min_bal_req.receiver == Global.current_application_address
    // ), "Payment must be to app address"
    assert // Payment must be to app address
    // voting/voting.py:188
    // log(min_bal_req)
    dup
    itob
    log
    // voting/voting.py:189
    // assert fund_min_bal_req.amount == min_bal_req, "Payment must be the exact min balance"
    frame_dig -3
    gtxns Amount
    ==
    assert // Payment must be the exact min balance
    // voting/voting.py:190-191
    // # Record the vote for each question
    // cumulative_offset = UInt64(0)
    int 0
    // voting/voting.py:192
    // for question_index in urange(questions_count):
    int 0

vote_for_header@1:
    // voting/voting.py:192
    // for question_index in urange(questions_count):
    frame_dig 3
    frame_dig 0
    <
    bz vote_after_for@5
    // voting/voting.py:193-194
    // # Load the user's vote for this question
    // answer_option_index = answer_ids[question_index].decode()
    frame_dig 3
    dup
    frame_dig 1
    <
    assert // Index access is out of bounds
    frame_dig -1
    extract 2 0
    dig 1
    int 1
    extract3
    btoi
    swap
    // voting/voting.py:195
    // options_count = self.option_counts[question_index].decode()
    int 0
    byte "option_counts"
    app_global_get_ex
    assert // check option_counts exists
    dup
    int 0
    extract_uint16
    dig 2
    >
    assert // Index access is out of bounds
    extract 2 0
    dig 1
    int 1
    extract3
    btoi
    // voting/voting.py:196
    // assert answer_option_index < options_count, "Answer option index invalid"
    dig 2
    dig 1
    <
    assert // Answer option index invalid
    // voting/voting.py:197
    // increment_vote_in_box(cumulative_offset + answer_option_index)
    frame_dig 2
    dup
    uncover 4
    +
    callsub increment_vote_in_box
    // voting/voting.py:198
    // cumulative_offset += options_count
    +
    frame_bury 2
    // voting/voting.py:199
    // op.Box.put(Txn.sender.bytes, answer_ids.bytes)
    txn Sender
    frame_dig -1
    box_put
    // voting/voting.py:200
    // self.voter_count += 1
    int 0
    byte "voter_count"
    app_global_get_ex
    assert // check voter_count exists
    int 1
    +
    byte "voter_count"
    swap
    app_global_put
    // voting/voting.py:192
    // for question_index in urange(questions_count):
    int 1
    +
    frame_bury 3
    b vote_for_header@1

vote_after_for@5:
    retsub


// examples.voting.voting.increment_vote_in_box(index: uint64) -> void:
increment_vote_in_box:
    // voting/voting.py:245-246
    // @subroutine
    // def increment_vote_in_box(index: UInt64) -> None:
    proto 1 0
    // voting/voting.py:247
    // box_data, exists = op.Box.get(TALLY_BOX_KEY)
    byte "V"
    box_get
    // voting/voting.py:248
    // assert exists, "Box not created"
    assert // Box not created
    // voting/voting.py:249
    // current_vote = op.btoi(op.extract(box_data, index, VOTE_COUNT_BYTES))
    frame_dig -1
    int 8
    extract3
    btoi
    // voting/voting.py:250
    // op.Box.replace(TALLY_BOX_KEY, index, op.itob(current_vote + 1))
    int 1
    +
    itob
    byte "V"
    frame_dig -1
    uncover 2
    box_replace
    retsub


// examples.voting.voting.VotingRoundApp.__init__() -> void:
__init__:
    // voting/voting.py:49
    // def __init__(self) -> None:
    proto 0 0
    // voting/voting.py:50
    // self.is_bootstrapped = False
    byte "is_bootstrapped"
    int 0
    app_global_put
    // voting/voting.py:51-52
    // # The minimum number of voters who have voted
    // self.voter_count = UInt64(0)
    byte "voter_count"
    int 0
    app_global_put
    retsub
", + "approval": "#pragma version 10

examples.voting.voting.VotingRoundApp.approval_program:
    txn ApplicationID
    bnz main_entrypoint@2
    callsub __init__

main_entrypoint@2:
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    method "create(string,byte[],string,uint64,uint64,uint8[],uint64,string)void"
    method "bootstrap(pay)void"
    method "close()void"
    method "get_preconditions(byte[])(uint64,uint64,uint64,uint64)"
    method "vote(pay,byte[],uint8[])void"
    txna ApplicationArgs 0
    match main_create_route@3 main_bootstrap_route@4 main_close_route@5 main_get_preconditions_route@6 main_vote_route@7
    err // reject transaction

main_create_route@3:
    // voting/voting.py:55
    // @arc4.abimethod(create=True)
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    !
    assert // is creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txna ApplicationArgs 1
    txna ApplicationArgs 2
    extract 2 0
    txna ApplicationArgs 3
    txna ApplicationArgs 4
    btoi
    txna ApplicationArgs 5
    btoi
    txna ApplicationArgs 6
    txna ApplicationArgs 7
    btoi
    txna ApplicationArgs 8
    // voting/voting.py:55
    // @arc4.abimethod(create=True)
    callsub create
    int 1
    return

main_bootstrap_route@4:
    // voting/voting.py:79
    // @arc4.abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int pay
    ==
    assert // transaction type is pay
    // voting/voting.py:79
    // @arc4.abimethod
    callsub bootstrap
    int 1
    return

main_close_route@5:
    // voting/voting.py:107
    // @arc4.abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    callsub close
    int 1
    return

main_get_preconditions_route@6:
    // voting/voting.py:157
    // @arc4.abimethod(readonly=True)
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txna ApplicationArgs 1
    // voting/voting.py:157
    // @arc4.abimethod(readonly=True)
    callsub get_preconditions
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_vote_route@7:
    // voting/voting.py:166
    // @arc4.abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // voting/voting.py:48
    // class VotingRoundApp(ARC4Contract):
    txn GroupIndex
    int 1
    -
    dup
    gtxns TypeEnum
    int pay
    ==
    assert // transaction type is pay
    txna ApplicationArgs 1
    extract 2 0
    txna ApplicationArgs 2
    // voting/voting.py:166
    // @arc4.abimethod
    callsub vote
    int 1
    return


// examples.voting.voting.VotingRoundApp.create(vote_id: bytes, snapshot_public_key: bytes, metadata_ipfs_cid: bytes, start_time: uint64, end_time: uint64, option_counts: bytes, quorum: uint64, nft_image_url: bytes) -> void:
create:
    // voting/voting.py:55-66
    // @arc4.abimethod(create=True)
    // def create(
    //     self,
    //     vote_id: arc4.String,
    //     snapshot_public_key: Bytes,
    //     metadata_ipfs_cid: arc4.String,
    //     start_time: UInt64,
    //     end_time: UInt64,
    //     option_counts: VoteIndexArray,
    //     quorum: UInt64,
    //     nft_image_url: arc4.String,
    // ) -> None:
    proto 8 0
    // voting/voting.py:67
    // assert start_time < end_time, "End time should be after start time"
    frame_dig -5
    frame_dig -4
    <
    assert // End time should be after start time
    // voting/voting.py:68
    // assert end_time >= Global.latest_timestamp, "End time should be in the future"
    frame_dig -4
    global LatestTimestamp
    >=
    assert // End time should be in the future
    // voting/voting.py:70
    // self.vote_id = vote_id.decode()
    frame_dig -8
    extract 2 0
    byte "vote_id"
    swap
    app_global_put
    // voting/voting.py:71
    // self.snapshot_public_key = snapshot_public_key
    byte "snapshot_public_key"
    frame_dig -7
    app_global_put
    // voting/voting.py:72
    // self.metadata_ipfs_cid = metadata_ipfs_cid.decode()
    frame_dig -6
    extract 2 0
    byte "metadata_ipfs_cid"
    swap
    app_global_put
    // voting/voting.py:73
    // self.start_time = start_time
    byte "start_time"
    frame_dig -5
    app_global_put
    // voting/voting.py:74
    // self.end_time = end_time
    byte "end_time"
    frame_dig -4
    app_global_put
    // voting/voting.py:75
    // self.quorum = quorum
    byte "quorum"
    frame_dig -2
    app_global_put
    // voting/voting.py:76
    // self.nft_image_url = nft_image_url.decode()
    frame_dig -1
    extract 2 0
    byte "nft_image_url"
    swap
    app_global_put
    // voting/voting.py:77
    // self.store_option_counts(option_counts.copy())
    frame_dig -3
    callsub store_option_counts
    pop
    retsub


// examples.voting.voting.VotingRoundApp.store_option_counts(option_counts: bytes) -> bytes:
store_option_counts:
    // voting/voting.py:215-216
    // @subroutine
    // def store_option_counts(self, option_counts: VoteIndexArray) -> None:
    proto 1 1
    // voting/voting.py:217
    // assert option_counts.length, "option_counts should be non-empty"
    frame_dig -1
    int 0
    extract_uint16
    dupn 2
    assert // option_counts should be non-empty
    // voting/voting.py:218
    // assert option_counts.length <= 112, "Can't have more than 112 questions"
    int 112
    <=
    assert // Can't have more than 112 questions
    // voting/voting.py:220
    // total_options = UInt64(0)
    int 0
    // voting/voting.py:221
    // for item in option_counts:
    frame_dig -1
    extract 2 0
    int 0

store_option_counts_for_header@1:
    // voting/voting.py:221
    // for item in option_counts:
    frame_dig 3
    frame_dig 0
    <
    bz store_option_counts_after_for@4
    frame_dig 2
    frame_dig 3
    dup
    cover 2
    int 1
    extract3
    // voting/voting.py:222
    // total_options += item.decode()
    btoi
    frame_dig 1
    +
    frame_bury 1
    int 1
    +
    frame_bury 3
    b store_option_counts_for_header@1

store_option_counts_after_for@4:
    // voting/voting.py:223
    // assert total_options <= 128, "Can't have more than 128 vote options"
    frame_dig 1
    dup
    int 128
    <=
    assert // Can't have more than 128 vote options
    // voting/voting.py:225
    // self.option_counts = option_counts.copy()
    byte "option_counts"
    frame_dig -1
    app_global_put
    // voting/voting.py:226
    // self.total_options = total_options
    byte "total_options"
    swap
    app_global_put
    frame_dig -1
    frame_bury 0
    retsub


// examples.voting.voting.VotingRoundApp.bootstrap(fund_min_bal_req: uint64) -> void:
bootstrap:
    // voting/voting.py:79-80
    // @arc4.abimethod
    // def bootstrap(self, fund_min_bal_req: gtxn.PaymentTransaction) -> None:
    proto 1 0
    // voting/voting.py:81
    // assert not self.is_bootstrapped, "Must not be already bootstrapped"
    int 0
    byte "is_bootstrapped"
    app_global_get_ex
    assert // check is_bootstrapped exists
    !
    assert // Must not be already bootstrapped
    // voting/voting.py:82
    // self.is_bootstrapped = True
    byte "is_bootstrapped"
    int 1
    app_global_put
    // voting/voting.py:85
    // fund_min_bal_req.receiver == Global.current_application_address
    frame_dig -1
    gtxns Receiver
    global CurrentApplicationAddress
    ==
    // voting/voting.py:84-86
    // assert (
    //     fund_min_bal_req.receiver == Global.current_application_address
    // ), "Payment must be to app address"
    assert // Payment must be to app address
    // voting/voting.py:88
    // tally_box_size = self.total_options * VOTE_COUNT_BYTES
    int 0
    byte "total_options"
    app_global_get_ex
    assert // check total_options exists
    int 8
    *
    // voting/voting.py:98-99
    // # tally box value
    // + (tally_box_size * BOX_BYTE_MIN_BALANCE)
    dup
    int 400
    *
    // voting/voting.py:90-97
    // # minimum balance req for: ALGOs + Vote result NFT asset
    // ASSET_MIN_BALANCE * 2
    // # create NFT fee
    // + 1000
    // # tally box
    // + BOX_FLAT_MIN_BALANCE
    // # tally box key "V"
    // + BOX_BYTE_MIN_BALANCE
    int 203900
    // voting/voting.py:90-99
    // # minimum balance req for: ALGOs + Vote result NFT asset
    // ASSET_MIN_BALANCE * 2
    // # create NFT fee
    // + 1000
    // # tally box
    // + BOX_FLAT_MIN_BALANCE
    // # tally box key "V"
    // + BOX_BYTE_MIN_BALANCE
    // # tally box value
    // + (tally_box_size * BOX_BYTE_MIN_BALANCE)
    +
    // voting/voting.py:101
    // log(min_balance_req)
    dup
    itob
    log
    // voting/voting.py:103
    // fund_min_bal_req.amount == min_balance_req
    frame_dig -1
    gtxns Amount
    ==
    // voting/voting.py:102-104
    // assert (
    //     fund_min_bal_req.amount == min_balance_req
    // ), "Payment must be for the exact min balance requirement"
    assert // Payment must be for the exact min balance requirement
    // voting/voting.py:105
    // assert op.Box.create(TALLY_BOX_KEY, tally_box_size)
    byte "V"
    swap
    box_create
    assert
    retsub


// examples.voting.voting.VotingRoundApp.close() -> void:
close:
    // voting/voting.py:107-108
    // @arc4.abimethod
    // def close(self) -> None:
    proto 0 0
    int 0
    dup
    byte ""
    dupn 2
    // voting/voting.py:109
    // ensure_budget(20000, fee_source=OpUpFeeSource.GroupCredit)
    int 20000
    int 0
    callsub ensure_budget
    // voting/voting.py:110
    // assert not self.close_time, "Already closed"
    int 0
    byte "close_time"
    app_global_get_ex
    bury 1
    !
    assert // Already closed
    // voting/voting.py:111
    // self.close_time.value = Global.latest_timestamp
    byte "close_time"
    global LatestTimestamp
    app_global_put
    // voting/voting.py:116
    // + self.vote_id
    int 0
    byte "vote_id"
    app_global_get_ex
    assert // check vote_id exists
    // voting/voting.py:114-115
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    byte "{\"standard\":\"arc69\",\"description\":\"This is a voting result NFT for voting round with ID "
    // voting/voting.py:114-116
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    swap
    concat
    // voting/voting.py:117
    // + b'.","properties":{"metadata":"ipfs://'
    byte ".\",\"properties\":{\"metadata\":\"ipfs://"
    // voting/voting.py:114-117
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    concat
    // voting/voting.py:118
    // + self.metadata_ipfs_cid
    int 0
    byte "metadata_ipfs_cid"
    app_global_get_ex
    assert // check metadata_ipfs_cid exists
    // voting/voting.py:114-118
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    concat
    // voting/voting.py:119
    // + b'","id":"'
    byte "\",\"id\":\""
    // voting/voting.py:114-119
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    concat
    // voting/voting.py:120
    // + self.vote_id
    int 0
    byte "vote_id"
    app_global_get_ex
    assert // check vote_id exists
    // voting/voting.py:114-120
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    concat
    // voting/voting.py:121
    // + b'","quorum":'
    byte "\",\"quorum\":"
    // voting/voting.py:114-121
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    concat
    // voting/voting.py:122
    // + itoa(self.quorum)
    int 0
    byte "quorum"
    app_global_get_ex
    assert // check quorum exists
    callsub itoa
    // voting/voting.py:114-122
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    concat
    // voting/voting.py:123
    // + b',"voterCount":'
    byte ",\"voterCount\":"
    // voting/voting.py:114-123
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    // + b',"voterCount":'
    concat
    // voting/voting.py:124
    // + itoa(self.voter_count)
    int 0
    byte "voter_count"
    app_global_get_ex
    assert // check voter_count exists
    callsub itoa
    // voting/voting.py:114-124
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    // + b',"voterCount":'
    // + itoa(self.voter_count)
    concat
    // voting/voting.py:125
    // + b',"tallies":['
    byte ",\"tallies\":["
    // voting/voting.py:114-125
    // b'{"standard":"arc69",'
    // b'"description":"This is a voting result NFT for voting round with ID '
    // + self.vote_id
    // + b'.","properties":{"metadata":"ipfs://'
    // + self.metadata_ipfs_cid
    // + b'","id":"'
    // + self.vote_id
    // + b'","quorum":'
    // + itoa(self.quorum)
    // + b',"voterCount":'
    // + itoa(self.voter_count)
    // + b',"tallies":['
    concat
    // voting/voting.py:128
    // current_index = UInt64(0)
    int 0
    // voting/voting.py:129
    // for question_index, question_options_arc in uenumerate(self.option_counts):
    dup
    byte "option_counts"
    app_global_get_ex
    assert // check option_counts exists
    dup
    int 0
    extract_uint16
    swap
    extract 2 0
    int 0

close_for_header@1:
    // voting/voting.py:129
    // for question_index, question_options_arc in uenumerate(self.option_counts):
    frame_dig 9
    frame_dig 7
    <
    bz close_after_for@15
    frame_dig 8
    frame_dig 9
    dup
    cover 2
    int 1
    extract3
    frame_bury 1
    // voting/voting.py:130
    // if question_index > 0:
    int 0
    >
    frame_dig 5
    frame_bury 0
    bz close_after_if_else@4
    // voting/voting.py:131
    // note += b","
    frame_dig 5
    byte ","
    concat
    frame_bury 0

close_after_if_else@4:
    frame_dig 0
    dup
    frame_bury 5
    // voting/voting.py:132
    // question_options = question_options_arc.decode()
    frame_dig 1
    btoi
    dup
    frame_bury 4
    // voting/voting.py:133
    // if question_options > 0:
    int 0
    >
    frame_dig 6
    frame_bury 2
    swap
    frame_bury 0
    bz close_after_if_else@13
    // voting/voting.py:134
    // note += b"["
    frame_dig 5
    byte "["
    concat
    frame_bury 5
    // voting/voting.py:135
    // for option_index in urange(question_options):
    int 0
    frame_bury 3

close_for_header@6:
    // voting/voting.py:135
    // for option_index in urange(question_options):
    frame_dig 3
    frame_dig 4
    <
    bz close_after_for@12
    // voting/voting.py:136
    // if option_index > 0:
    frame_dig 3
    int 0
    >
    frame_dig 5
    frame_bury 0
    bz close_after_if_else@9
    // voting/voting.py:137
    // note += b","
    frame_dig 5
    byte ","
    concat
    frame_bury 0

close_after_if_else@9:
    frame_dig 0
    // voting/voting.py:138
    // votes_for_option = get_vote_from_box(current_index)
    frame_dig 6
    dup
    cover 2
    callsub get_vote_from_box
    // voting/voting.py:139
    // note += itoa(votes_for_option)
    callsub itoa
    concat
    frame_bury 5
    // voting/voting.py:140
    // current_index += 1
    int 1
    +
    frame_bury 6
    // voting/voting.py:135
    // for option_index in urange(question_options):
    frame_dig 3
    int 1
    +
    frame_bury 3
    b close_for_header@6

close_after_for@12:
    // voting/voting.py:141
    // note += b"]"
    frame_dig 5
    byte "]"
    concat
    frame_dig 6
    frame_bury 2
    frame_bury 0

close_after_if_else@13:
    frame_dig 2
    frame_bury 6
    frame_dig 0
    frame_bury 5
    frame_dig 9
    int 1
    +
    frame_bury 9
    b close_for_header@1

close_after_for@15:
    // voting/voting.py:142
    // note += b"]}}"
    frame_dig 5
    byte "]}}"
    concat
    // voting/voting.py:144-153
    // itxn.AssetConfig(
    //     total=1,
    //     decimals=0,
    //     default_frozen=False,
    //     asset_name=b"[VOTE RESULT] " + self.vote_id,
    //     unit_name=b"VOTERSLT",
    //     url=self.nft_image_url,
    //     note=note,
    // )
    // .submit()
    itxn_begin
    // voting/voting.py:148
    // asset_name=b"[VOTE RESULT] " + self.vote_id,
    int 0
    byte "vote_id"
    app_global_get_ex
    assert // check vote_id exists
    byte "[VOTE RESULT] "
    swap
    concat
    // voting/voting.py:150
    // url=self.nft_image_url,
    int 0
    byte "nft_image_url"
    app_global_get_ex
    assert // check nft_image_url exists
    uncover 2
    itxn_field Note
    itxn_field ConfigAssetURL
    // voting/voting.py:149
    // unit_name=b"VOTERSLT",
    byte "VOTERSLT"
    itxn_field ConfigAssetUnitName
    itxn_field ConfigAssetName
    // voting/voting.py:147
    // default_frozen=False,
    int 0
    itxn_field ConfigAssetDefaultFrozen
    // voting/voting.py:146
    // decimals=0,
    int 0
    itxn_field ConfigAssetDecimals
    // voting/voting.py:145
    // total=1,
    int 1
    itxn_field ConfigAssetTotal
    // voting/voting.py:144
    // itxn.AssetConfig(
    int acfg
    itxn_field TypeEnum
    // voting/voting.py:144-153
    // itxn.AssetConfig(
    //     total=1,
    //     decimals=0,
    //     default_frozen=False,
    //     asset_name=b"[VOTE RESULT] " + self.vote_id,
    //     unit_name=b"VOTERSLT",
    //     url=self.nft_image_url,
    //     note=note,
    // )
    // .submit()
    itxn_submit
    // voting/voting.py:143
    // self.nft_asset_id = (
    byte "nft_asset_id"
    // voting/voting.py:144-153
    // itxn.AssetConfig(
    //     total=1,
    //     decimals=0,
    //     default_frozen=False,
    //     asset_name=b"[VOTE RESULT] " + self.vote_id,
    //     unit_name=b"VOTERSLT",
    //     url=self.nft_image_url,
    //     note=note,
    // )
    // .submit()
    itxn CreatedAssetID
    // voting/voting.py:143-155
    // self.nft_asset_id = (
    //     itxn.AssetConfig(
    //         total=1,
    //         decimals=0,
    //         default_frozen=False,
    //         asset_name=b"[VOTE RESULT] " + self.vote_id,
    //         unit_name=b"VOTERSLT",
    //         url=self.nft_image_url,
    //         note=note,
    //     )
    //     .submit()
    //     .created_asset.id
    // )
    app_global_put
    retsub


// puyapy.ensure_budget(required_budget: uint64, fee_source: uint64) -> void:
ensure_budget:
    // <puya>/puyapy.py:11-17
    proto 2 0
    // <puya>/puyapy.py:18
    frame_dig -2
    int 10
    +

ensure_budget_while_top@1:
    // <puya>/puyapy.py:19
    frame_dig 0
    global OpcodeBudget
    >
    bz ensure_budget_after_while@7
    // <puya>/puyapy.py:20
    itxn_begin
    // <puya>/puyapy.py:21
    int appl
    itxn_field TypeEnum
    // <puya>/puyapy.py:22
    int DeleteApplication
    itxn_field OnCompletion
    // <puya>/puyapy.py:23
    byte 0x068101
    itxn_field ApprovalProgram
    // <puya>/puyapy.py:24
    byte 0x068101
    itxn_field ClearStateProgram
    // <puya>/puyapy.py:25-29
    frame_dig -1
    switch ensure_budget_switch_case_0@3 ensure_budget_switch_case_1@4
    b ensure_budget_switch_case_next@6

ensure_budget_switch_case_0@3:
    // <puya>/puyapy.py:27
    int 0
    itxn_field Fee
    b ensure_budget_switch_case_next@6

ensure_budget_switch_case_1@4:
    // <puya>/puyapy.py:29
    global MinTxnFee
    itxn_field Fee

ensure_budget_switch_case_next@6:
    // <puya>/puyapy.py:30
    itxn_submit
    b ensure_budget_while_top@1

ensure_budget_after_while@7:
    retsub


// examples.voting.voting.itoa(i: uint64) -> bytes:
itoa:
    // voting/voting.py:253-254
    // @subroutine
    // def itoa(i: UInt64) -> Bytes:
    proto 1 1
    // voting/voting.py:257
    // if i < radix:
    frame_dig -1
    // voting/voting.py:256
    // radix = digits.length
    int 10
    // voting/voting.py:257
    // if i < radix:
    <
    bz itoa_after_if_else@2
    // voting/voting.py:258
    // return digits[i]
    frame_dig -1
    int 1
    +
    // voting/voting.py:255
    // digits = Bytes(b"0123456789")
    byte "0123456789"
    // voting/voting.py:258
    // return digits[i]
    frame_dig -1
    uncover 2
    substring3
    retsub

itoa_after_if_else@2:
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    frame_dig -1
    // voting/voting.py:256
    // radix = digits.length
    int 10
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    /
    callsub itoa
    frame_dig -1
    // voting/voting.py:256
    // radix = digits.length
    int 10
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    %
    dup
    int 1
    +
    // voting/voting.py:255
    // digits = Bytes(b"0123456789")
    byte "0123456789"
    // voting/voting.py:259
    // return itoa(i // radix) + digits[i % radix]
    cover 2
    substring3
    concat
    retsub


// examples.voting.voting.get_vote_from_box(index: uint64) -> uint64:
get_vote_from_box:
    // voting/voting.py:238-239
    // @subroutine
    // def get_vote_from_box(index: UInt64) -> UInt64:
    proto 1 1
    // voting/voting.py:240
    // box_data, exists = op.Box.get(TALLY_BOX_KEY)
    byte "V"
    box_get
    // voting/voting.py:241
    // assert exists, "Box not created"
    assert // Box not created
    // voting/voting.py:242
    // return op.btoi(op.extract(box_data, index, VOTE_COUNT_BYTES))
    frame_dig -1
    int 8
    extract3
    btoi
    retsub


// examples.voting.voting.VotingRoundApp.get_preconditions(signature: bytes) -> bytes:
get_preconditions:
    // voting/voting.py:157-158
    // @arc4.abimethod(readonly=True)
    // def get_preconditions(self, signature: arc4.DynamicBytes) -> VotingPreconditions:
    proto 1 1
    // voting/voting.py:160
    // is_voting_open=arc4.UInt64(self.voting_open()),
    callsub voting_open
    itob
    // voting/voting.py:161
    // is_allowed_to_vote=arc4.UInt64(self.allowed_to_vote(signature.bytes[2:])),
    frame_dig -1
    len
    int 2
    dig 1
    <
    swap
    dup
    int 2
    uncover 3
    select
    frame_dig -1
    swap
    uncover 2
    substring3
    callsub allowed_to_vote
    itob
    // voting/voting.py:162
    // has_already_voted=arc4.UInt64(self.already_voted()),
    callsub already_voted
    itob
    // voting/voting.py:163
    // current_time=arc4.UInt64(Global.latest_timestamp),
    global LatestTimestamp
    itob
    // voting/voting.py:159-164
    // return VotingPreconditions(
    //     is_voting_open=arc4.UInt64(self.voting_open()),
    //     is_allowed_to_vote=arc4.UInt64(self.allowed_to_vote(signature.bytes[2:])),
    //     has_already_voted=arc4.UInt64(self.already_voted()),
    //     current_time=arc4.UInt64(Global.latest_timestamp),
    // )
    uncover 3
    uncover 3
    concat
    uncover 2
    concat
    swap
    concat
    retsub


// examples.voting.voting.VotingRoundApp.voting_open() -> uint64:
voting_open:
    // voting/voting.py:202-203
    // @subroutine
    // def voting_open(self) -> bool:
    proto 0 1
    byte ""
    // voting/voting.py:205
    // self.is_bootstrapped
    int 0
    byte "is_bootstrapped"
    app_global_get_ex
    assert // check is_bootstrapped exists
    // voting/voting.py:205-207
    // self.is_bootstrapped
    // and not self.close_time
    // and self.start_time <= Global.latest_timestamp <= self.end_time
    bz voting_open_bool_false@5
    // voting/voting.py:206
    // and not self.close_time
    int 0
    byte "close_time"
    app_global_get_ex
    bury 1
    bnz voting_open_bool_false@5
    // voting/voting.py:207
    // and self.start_time <= Global.latest_timestamp <= self.end_time
    int 0
    byte "start_time"
    app_global_get_ex
    assert // check start_time exists
    global LatestTimestamp
    dup
    frame_bury 0
    <=
    bz voting_open_bool_false@5
    int 0
    byte "end_time"
    app_global_get_ex
    assert // check end_time exists
    frame_dig 0
    >=
    bz voting_open_bool_false@5
    int 1
    b voting_open_bool_merge@6

voting_open_bool_false@5:
    int 0

voting_open_bool_merge@6:
    // voting/voting.py:204-208
    // return (
    //     self.is_bootstrapped
    //     and not self.close_time
    //     and self.start_time <= Global.latest_timestamp <= self.end_time
    // )
    swap
    retsub


// examples.voting.voting.VotingRoundApp.allowed_to_vote(signature: bytes) -> uint64:
allowed_to_vote:
    // voting/voting.py:228-229
    // @subroutine
    // def allowed_to_vote(self, signature: Bytes) -> bool:
    proto 1 1
    // voting/voting.py:230
    // ensure_budget(2000)
    int 2000
    int 2
    callsub ensure_budget
    // voting/voting.py:232
    // Txn.sender.bytes,
    txn Sender
    // voting/voting.py:234
    // self.snapshot_public_key,
    int 0
    byte "snapshot_public_key"
    app_global_get_ex
    assert // check snapshot_public_key exists
    // voting/voting.py:231-235
    // return op.ed25519verify_bare(
    //     Txn.sender.bytes,
    //     signature,
    //     self.snapshot_public_key,
    // )
    frame_dig -1
    swap
    ed25519verify_bare
    retsub


// examples.voting.voting.VotingRoundApp.already_voted() -> uint64:
already_voted:
    // voting/voting.py:210-211
    // @subroutine
    // def already_voted(self) -> bool:
    proto 0 1
    // voting/voting.py:212
    // (votes, exists) = op.Box.get(Txn.sender.bytes)
    txn Sender
    box_get
    bury 1
    // voting/voting.py:213
    // return exists
    retsub


// examples.voting.voting.VotingRoundApp.vote(fund_min_bal_req: uint64, signature: bytes, answer_ids: bytes) -> void:
vote:
    // voting/voting.py:166-172
    // @arc4.abimethod
    // def vote(
    //     self,
    //     fund_min_bal_req: gtxn.PaymentTransaction,
    //     signature: Bytes,
    //     answer_ids: VoteIndexArray,
    // ) -> None:
    proto 3 0
    // voting/voting.py:173
    // ensure_budget(7700, fee_source=OpUpFeeSource.GroupCredit)
    int 7700
    int 0
    callsub ensure_budget
    // voting/voting.py:174-175
    // # Check voting preconditions
    // assert self.allowed_to_vote(signature), "Not allowed to vote"
    frame_dig -2
    callsub allowed_to_vote
    assert // Not allowed to vote
    // voting/voting.py:176
    // assert self.voting_open(), "Voting not open"
    callsub voting_open
    assert // Voting not open
    // voting/voting.py:177
    // assert not self.already_voted(), "Already voted"
    callsub already_voted
    !
    assert // Already voted
    // voting/voting.py:178
    // questions_count = self.option_counts.length
    int 0
    byte "option_counts"
    app_global_get_ex
    assert // check option_counts exists
    int 0
    extract_uint16
    dup
    // voting/voting.py:179
    // assert answer_ids.length == questions_count, "Number of answers incorrect"
    frame_dig -1
    int 0
    extract_uint16
    dup
    cover 2
    dup
    uncover 2
    ==
    assert // Number of answers incorrect
    // voting/voting.py:182
    // (32 + 2 + VOTE_INDEX_BYTES * answer_ids.length) * BOX_BYTE_MIN_BALANCE
    int 34
    +
    int 400
    *
    // voting/voting.py:180-181
    // # Check voter box is funded
    // min_bal_req = BOX_FLAT_MIN_BALANCE + (
    int 2500
    // voting/voting.py:180-183
    // # Check voter box is funded
    // min_bal_req = BOX_FLAT_MIN_BALANCE + (
    //     (32 + 2 + VOTE_INDEX_BYTES * answer_ids.length) * BOX_BYTE_MIN_BALANCE
    // )
    +
    // voting/voting.py:185
    // fund_min_bal_req.receiver == Global.current_application_address
    frame_dig -3
    gtxns Receiver
    global CurrentApplicationAddress
    ==
    // voting/voting.py:184-186
    // assert (
    //     fund_min_bal_req.receiver == Global.current_application_address
    // ), "Payment must be to app address"
    assert // Payment must be to app address
    // voting/voting.py:188
    // log(min_bal_req)
    dup
    itob
    log
    // voting/voting.py:189
    // assert fund_min_bal_req.amount == min_bal_req, "Payment must be the exact min balance"
    frame_dig -3
    gtxns Amount
    ==
    assert // Payment must be the exact min balance
    // voting/voting.py:190-191
    // # Record the vote for each question
    // cumulative_offset = UInt64(0)
    int 0
    // voting/voting.py:192
    // for question_index in urange(questions_count):
    dup

vote_for_header@1:
    // voting/voting.py:192
    // for question_index in urange(questions_count):
    frame_dig 3
    frame_dig 0
    <
    bz vote_after_for@5
    // voting/voting.py:193-194
    // # Load the user's vote for this question
    // answer_option_index = answer_ids[question_index].decode()
    frame_dig 3
    dup
    frame_dig 1
    <
    assert // Index access is out of bounds
    frame_dig -1
    extract 2 0
    dig 1
    int 1
    extract3
    btoi
    swap
    // voting/voting.py:195
    // options_count = self.option_counts[question_index].decode()
    int 0
    byte "option_counts"
    app_global_get_ex
    assert // check option_counts exists
    dup
    int 0
    extract_uint16
    dig 2
    >
    assert // Index access is out of bounds
    extract 2 0
    dig 1
    int 1
    extract3
    btoi
    // voting/voting.py:196
    // assert answer_option_index < options_count, "Answer option index invalid"
    dig 2
    dig 1
    <
    assert // Answer option index invalid
    // voting/voting.py:197
    // increment_vote_in_box(cumulative_offset + answer_option_index)
    frame_dig 2
    dup
    uncover 4
    +
    callsub increment_vote_in_box
    // voting/voting.py:198
    // cumulative_offset += options_count
    +
    frame_bury 2
    // voting/voting.py:199
    // op.Box.put(Txn.sender.bytes, answer_ids.bytes)
    txn Sender
    frame_dig -1
    box_put
    // voting/voting.py:200
    // self.voter_count += 1
    int 0
    byte "voter_count"
    app_global_get_ex
    assert // check voter_count exists
    int 1
    +
    byte "voter_count"
    swap
    app_global_put
    // voting/voting.py:192
    // for question_index in urange(questions_count):
    int 1
    +
    frame_bury 3
    b vote_for_header@1

vote_after_for@5:
    retsub


// examples.voting.voting.increment_vote_in_box(index: uint64) -> void:
increment_vote_in_box:
    // voting/voting.py:245-246
    // @subroutine
    // def increment_vote_in_box(index: UInt64) -> None:
    proto 1 0
    // voting/voting.py:247
    // box_data, exists = op.Box.get(TALLY_BOX_KEY)
    byte "V"
    box_get
    // voting/voting.py:248
    // assert exists, "Box not created"
    assert // Box not created
    // voting/voting.py:249
    // current_vote = op.btoi(op.extract(box_data, index, VOTE_COUNT_BYTES))
    frame_dig -1
    int 8
    extract3
    btoi
    // voting/voting.py:250
    // op.Box.replace(TALLY_BOX_KEY, index, op.itob(current_vote + 1))
    int 1
    +
    itob
    byte "V"
    frame_dig -1
    uncover 2
    box_replace
    retsub


// examples.voting.voting.VotingRoundApp.__init__() -> void:
__init__:
    // voting/voting.py:49
    // def __init__(self) -> None:
    proto 0 0
    // voting/voting.py:50
    // self.is_bootstrapped = False
    byte "is_bootstrapped"
    int 0
    app_global_put
    // voting/voting.py:51-52
    // # The minimum number of voters who have voted
    // self.voter_count = UInt64(0)
    byte "voter_count"
    int 0
    app_global_put
    retsub
", "clear": "I3ByYWdtYSB2ZXJzaW9uIDEwCgpleGFtcGxlcy52b3Rpbmcudm90aW5nLlZvdGluZ1JvdW5kQXBwLmNsZWFyX3N0YXRlX3Byb2dyYW06CiAgICAvLyB2b3Rpbmcvdm90aW5nLnB5OjQ4CiAgICAvLyBjbGFzcyBWb3RpbmdSb3VuZEFwcChBUkM0Q29udHJhY3QpOgogICAgaW50IDEKICAgIHJldHVybgo=" }, "state": { diff --git a/examples/voting/out_O2/VotingRoundApp.approval.teal b/examples/voting/out_O2/VotingRoundApp.approval.teal index 30a958cf91..06aebec8f6 100644 --- a/examples/voting/out_O2/VotingRoundApp.approval.teal +++ b/examples/voting/out_O2/VotingRoundApp.approval.teal @@ -301,7 +301,7 @@ close: byte ",\"tallies\":[" concat int 0 - int 0 + dup byte "option_counts" app_global_get_ex assert // check option_counts exists @@ -686,7 +686,7 @@ vote: == assert // Payment must be the exact min balance int 0 - int 0 + dup vote_for_header@1: frame_dig 3 diff --git a/src/puya/teal/optimize/constant_stack_shuffling.py b/src/puya/teal/optimize/constant_stack_shuffling.py index 84772a9a22..5bee4445cb 100644 --- a/src/puya/teal/optimize/constant_stack_shuffling.py +++ b/src/puya/teal/optimize/constant_stack_shuffling.py @@ -1,5 +1,4 @@ -import functools -import operator +import attrs from puya.teal import models from puya.teal.optimize._data import LOAD_OP_CODES @@ -10,10 +9,13 @@ def perform_constant_stack_shuffling(block: models.TealBlock) -> bool: loads = list[models.TealOp]() modified = False for op in block.ops: - if op.op_code in LOAD_OP_CODES: # noqa: SIM114 - loads.append(op) - elif isinstance(op, models.FrameDig) and op.n < 0: + if _is_constant_load(op): loads.append(op) + elif loads and op.op_code.startswith("dup"): + (n,) = op.immediates or (1,) + assert isinstance(n, int) + loads.extend([attrs.evolve(loads[-1], source_location=op.source_location)] * n) + modified = True elif loads: match op: case models.Uncover(n=n) if n < len(loads): @@ -48,9 +50,7 @@ def constant_dupn_insertion(block: models.TealBlock) -> bool: modified = _collapse_loads(loads) or modified result.extend(loads) loads = [] - if op.op_code in LOAD_OP_CODES or ( - op.op_code == "frame_dig" and int(op.immediates[0]) < 0 - ): + if _is_constant_load(op): loads.append(op) else: result.append(op) @@ -63,10 +63,21 @@ def constant_dupn_insertion(block: models.TealBlock) -> bool: def _collapse_loads(loads: list[models.TealOp]) -> bool: n = len(loads) - 1 - if n >= 2: - dupn_source_location = functools.reduce( - operator.add, (op.source_location for op in loads[1:]) - ) - loads[1:] = [models.DupN(n=n, source_location=dupn_source_location)] - return True - return False + if n < 1: + return False + + if n == 1: + dup_op: models.TealOp = models.Dup(source_location=loads[1].source_location) + else: + dupn_source_location = None + for op in loads[1:]: + if op.source_location is not None: + # TODO: it'd be better to only merge these if they're adjacent + dupn_source_location = op.source_location + dupn_source_location + dup_op = models.DupN(n=n, source_location=dupn_source_location) + loads[1:] = [dup_op] + return True + + +def _is_constant_load(op: models.TealOp) -> bool: + return op.op_code in LOAD_OP_CODES or (isinstance(op, models.FrameDig) and op.n < 0) diff --git a/src/puya/teal/optimize/main.py b/src/puya/teal/optimize/main.py index 45788c2f97..343ec2e61c 100644 --- a/src/puya/teal/optimize/main.py +++ b/src/puya/teal/optimize/main.py @@ -18,10 +18,14 @@ def optimize_block(block: models.TealBlock, *, level: int) -> None: modified = simplify_repeated_rotation_ops(block) or modified modified = peephole(block) or modified - # don't do this one in a loop, only after everything else + # we don't do dup/dupn collapse in the above loop, but after it. + # it's easier to deal with expanded dup/dupn instructions above when looking at + # stack shuffling etc, but once it's done we save ops / program size by collapsing them constant_dupn_insertion(block) if level >= 2: + # this is a brute-force search which can be slow at times, + # so it's only done once and only at higher optimisation levels block.ops = repeated_rotation_ops_search(block.ops) diff --git a/test_cases/arc4_types/out/Arc4ArraysContract.approval.teal b/test_cases/arc4_types/out/Arc4ArraysContract.approval.teal index 5bed38499a..8ca09fb01a 100644 --- a/test_cases/arc4_types/out/Arc4ArraysContract.approval.teal +++ b/test_cases/arc4_types/out/Arc4ArraysContract.approval.teal @@ -7,7 +7,7 @@ test_cases.arc4_types.array.Arc4ArraysContract.approval_program: // arc4_types/array.py:26 // total = UInt64(0) int 0 - int 0 + dup main_for_header@1: // arc4_types/array.py:27 diff --git a/test_cases/arc4_types/out/Arc4BoolEvalContract.approval.teal b/test_cases/arc4_types/out/Arc4BoolEvalContract.approval.teal index e699297718..2ffff11432 100644 --- a/test_cases/arc4_types/out/Arc4BoolEvalContract.approval.teal +++ b/test_cases/arc4_types/out/Arc4BoolEvalContract.approval.teal @@ -4,7 +4,7 @@ test_cases.arc4_types.bool_eval.Arc4BoolEvalContract.approval_program: // arc4_types/bool_eval.py:19 // assert not arc4.Address(Global.zero_address) global ZeroAddress - global ZeroAddress + dup == assert // arc4_types/bool_eval.py:20 diff --git a/test_cases/arc4_types/out/Arc4DynamicBytesContract.approval.teal b/test_cases/arc4_types/out/Arc4DynamicBytesContract.approval.teal index 7708cdb392..e787e64b17 100644 --- a/test_cases/arc4_types/out/Arc4DynamicBytesContract.approval.teal +++ b/test_cases/arc4_types/out/Arc4DynamicBytesContract.approval.teal @@ -6,7 +6,7 @@ test_cases.arc4_types.dynamic_bytes.Arc4DynamicBytesContract.approval_program: // arc4_types/dynamic_bytes.py:11 // total = UInt64(0) int 0 - int 0 + dup main_for_header@1: // arc4_types/dynamic_bytes.py:16 diff --git a/test_cases/arc4_types/out/Arc4RefTypesContract.approval.teal b/test_cases/arc4_types/out/Arc4RefTypesContract.approval.teal index b0917f93e3..09caa4a3e8 100644 --- a/test_cases/arc4_types/out/Arc4RefTypesContract.approval.teal +++ b/test_cases/arc4_types/out/Arc4RefTypesContract.approval.teal @@ -1,19 +1,23 @@ #pragma version 10 test_cases.arc4_types.reference_types.Arc4RefTypesContract.approval_program: - // arc4_types/reference_types.py:6-8 + // arc4_types/reference_types.py:9-11 + // # When creating an address from bytes, we check the length is 32 as we don't know the + // # source of the bytes + // checked_address = arc4.Address(op.Txn.sender.bytes) + txn Sender + // arc4_types/reference_types.py:6-11 // # When creating an address from an account no need to check the length as we assume the // # Account is valid // sender_address = arc4.Address(op.Txn.sender) - txn Sender + // # When creating an address from bytes, we check the length is 32 as we don't know the + // # source of the bytes + // checked_address = arc4.Address(op.Txn.sender.bytes) + dupn 3 // arc4_types/reference_types.py:9-11 // # When creating an address from bytes, we check the length is 32 as we don't know the // # source of the bytes // checked_address = arc4.Address(op.Txn.sender.bytes) - txn Sender - dup - cover 2 - dup len int 32 == diff --git a/test_cases/arc4_types/out/Arc4StructsTypeContract.approval.teal b/test_cases/arc4_types/out/Arc4StructsTypeContract.approval.teal index f5f6f18387..033763c393 100644 --- a/test_cases/arc4_types/out/Arc4StructsTypeContract.approval.teal +++ b/test_cases/arc4_types/out/Arc4StructsTypeContract.approval.teal @@ -7,7 +7,7 @@ test_cases.arc4_types.structs.Arc4StructsTypeContract.approval_program: byte 0x000000083cfbf217000000230384b842 // arc4_types/structs.py:28 // coord_2 = Vector(x=Decimal("35.382882839"), y=Decimal("150.382884930")) - byte 0x000000083cfbf217000000230384b842 + dup // arc4_types/structs.py:29 // coord_3 = add(coord_1.copy(), coord_2.copy()) callsub add diff --git a/test_cases/arc4_types/out/array.O1.log b/test_cases/arc4_types/out/array.O1.log index 308a6173f1..2225d8ccb7 100644 --- a/test_cases/arc4_types/out/array.O1.log +++ b/test_cases/arc4_types/out/array.O1.log @@ -5,7 +5,7 @@ PC Teal Stack 49 byte "" 0, 0x 50 dupn 2 0, 0x, 0x, 0x 52 int 0 0, 0x, 0x, 0x, 0 -53 int 0 0, 0x, 0x, 0x, 0, 0 +53 dup 0, 0x, 0x, 0x, 0, 0 54 dup 0, 0x, 0x, 0x, 0, 0, 0 55 int 2 0, 0x, 0x, 0x, 0, 0, 0, 2 56 < 0, 0x, 0x, 0x, 0, 0, 1 diff --git a/test_cases/arc4_types/out/array.O2.log b/test_cases/arc4_types/out/array.O2.log index 308a6173f1..2225d8ccb7 100644 --- a/test_cases/arc4_types/out/array.O2.log +++ b/test_cases/arc4_types/out/array.O2.log @@ -5,7 +5,7 @@ PC Teal Stack 49 byte "" 0, 0x 50 dupn 2 0, 0x, 0x, 0x 52 int 0 0, 0x, 0x, 0x, 0 -53 int 0 0, 0x, 0x, 0x, 0, 0 +53 dup 0, 0x, 0x, 0x, 0, 0 54 dup 0, 0x, 0x, 0x, 0, 0, 0 55 int 2 0, 0x, 0x, 0x, 0, 0, 0, 2 56 < 0, 0x, 0x, 0x, 0, 0, 1 diff --git a/test_cases/arc4_types/out/structs.O1.log b/test_cases/arc4_types/out/structs.O1.log index 80123ca85b..3c3224f447 100644 --- a/test_cases/arc4_types/out/structs.O1.log +++ b/test_cases/arc4_types/out/structs.O1.log @@ -1,9 +1,9 @@ PC Teal Stack 1 6 -29 byte "" 0x -31 byte 0x000000083cfbf217000000230384b842 0x, 0x000000083CFBF217000000230384B842 -32 byte 0x000000083cfbf217000000230384b842 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 +12 byte "" 0x +14 byte 0x000000083cfbf217000000230384b842 0x, 0x000000083CFBF217000000230384B842 +32 dup 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 33 callsub add 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 100 proto 2 3 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 103 frame_dig -2 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 diff --git a/test_cases/arc4_types/out/structs.O2.log b/test_cases/arc4_types/out/structs.O2.log index 80123ca85b..3c3224f447 100644 --- a/test_cases/arc4_types/out/structs.O2.log +++ b/test_cases/arc4_types/out/structs.O2.log @@ -1,9 +1,9 @@ PC Teal Stack 1 6 -29 byte "" 0x -31 byte 0x000000083cfbf217000000230384b842 0x, 0x000000083CFBF217000000230384B842 -32 byte 0x000000083cfbf217000000230384b842 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 +12 byte "" 0x +14 byte 0x000000083cfbf217000000230384b842 0x, 0x000000083CFBF217000000230384B842 +32 dup 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 33 callsub add 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 100 proto 2 3 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 103 frame_dig -2 0x, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842, 0x000000083CFBF217000000230384B842 diff --git a/test_cases/arc4_types/out_O2/Arc4ArraysContract.approval.teal b/test_cases/arc4_types/out_O2/Arc4ArraysContract.approval.teal index 935e4651e1..35a61053ba 100644 --- a/test_cases/arc4_types/out_O2/Arc4ArraysContract.approval.teal +++ b/test_cases/arc4_types/out_O2/Arc4ArraysContract.approval.teal @@ -5,7 +5,7 @@ test_cases.arc4_types.array.Arc4ArraysContract.approval_program: byte "" dupn 2 int 0 - int 0 + dup main_for_header@1: dup diff --git a/test_cases/arc4_types/out_O2/Arc4BoolEvalContract.approval.teal b/test_cases/arc4_types/out_O2/Arc4BoolEvalContract.approval.teal index 0899a16ca0..b8ebfff45b 100644 --- a/test_cases/arc4_types/out_O2/Arc4BoolEvalContract.approval.teal +++ b/test_cases/arc4_types/out_O2/Arc4BoolEvalContract.approval.teal @@ -2,7 +2,7 @@ test_cases.arc4_types.bool_eval.Arc4BoolEvalContract.approval_program: global ZeroAddress - global ZeroAddress + dup == assert txn Sender diff --git a/test_cases/arc4_types/out_O2/Arc4DynamicBytesContract.approval.teal b/test_cases/arc4_types/out_O2/Arc4DynamicBytesContract.approval.teal index 61b65a697e..bc232301c5 100644 --- a/test_cases/arc4_types/out_O2/Arc4DynamicBytesContract.approval.teal +++ b/test_cases/arc4_types/out_O2/Arc4DynamicBytesContract.approval.teal @@ -4,7 +4,7 @@ test_cases.arc4_types.dynamic_bytes.Arc4DynamicBytesContract.approval_program: byte "" dup int 0 - int 0 + dup main_for_header@1: dup diff --git a/test_cases/arc4_types/out_O2/Arc4RefTypesContract.approval.teal b/test_cases/arc4_types/out_O2/Arc4RefTypesContract.approval.teal index e0b93f06d1..ba86b89090 100644 --- a/test_cases/arc4_types/out_O2/Arc4RefTypesContract.approval.teal +++ b/test_cases/arc4_types/out_O2/Arc4RefTypesContract.approval.teal @@ -2,10 +2,7 @@ test_cases.arc4_types.reference_types.Arc4RefTypesContract.approval_program: txn Sender - txn Sender - dup - cover 2 - dup + dupn 3 len int 32 == diff --git a/test_cases/arc4_types/out_O2/Arc4StructsTypeContract.approval.teal b/test_cases/arc4_types/out_O2/Arc4StructsTypeContract.approval.teal index 3d524d183a..5d82a2dfcd 100644 --- a/test_cases/arc4_types/out_O2/Arc4StructsTypeContract.approval.teal +++ b/test_cases/arc4_types/out_O2/Arc4StructsTypeContract.approval.teal @@ -3,7 +3,7 @@ test_cases.arc4_types.structs.Arc4StructsTypeContract.approval_program: byte "" byte 0x000000083cfbf217000000230384b842 - byte 0x000000083cfbf217000000230384b842 + dup callsub add popn 2 dup diff --git a/test_cases/boolean_binary_ops/out/BooleanBinaryOps.approval.teal b/test_cases/boolean_binary_ops/out/BooleanBinaryOps.approval.teal index 1df1613e9b..64420aea63 100644 --- a/test_cases/boolean_binary_ops/out/BooleanBinaryOps.approval.teal +++ b/test_cases/boolean_binary_ops/out/BooleanBinaryOps.approval.teal @@ -38,13 +38,13 @@ test_boolean_binary_ops: // boolean_binary_ops/contract.py:18 // assert true and true frame_dig -2 - frame_dig -2 + dup && assert // boolean_binary_ops/contract.py:19 // assert not (false and false) frame_dig -1 - frame_dig -1 + dup && ! assert @@ -63,13 +63,13 @@ test_boolean_binary_ops: // boolean_binary_ops/contract.py:23 // assert true or true frame_dig -2 - frame_dig -2 + dup || assert // boolean_binary_ops/contract.py:24 // assert not (false or false) frame_dig -1 - frame_dig -1 + dup || ! assert diff --git a/test_cases/boolean_binary_ops/out_O2/BooleanBinaryOps.approval.teal b/test_cases/boolean_binary_ops/out_O2/BooleanBinaryOps.approval.teal index 3bed323ef8..c960c33a06 100644 --- a/test_cases/boolean_binary_ops/out_O2/BooleanBinaryOps.approval.teal +++ b/test_cases/boolean_binary_ops/out_O2/BooleanBinaryOps.approval.teal @@ -23,11 +23,11 @@ test_boolean_binary_ops: ! assert frame_dig -2 - frame_dig -2 + dup && assert frame_dig -1 - frame_dig -1 + dup && ! assert @@ -40,11 +40,11 @@ test_boolean_binary_ops: || assert frame_dig -2 - frame_dig -2 + dup || assert frame_dig -1 - frame_dig -1 + dup || ! assert diff --git a/test_cases/contains/out/MyContract.approval.teal b/test_cases/contains/out/MyContract.approval.teal index a4086a8cef..ae9eefbb97 100644 --- a/test_cases/contains/out/MyContract.approval.teal +++ b/test_cases/contains/out/MyContract.approval.teal @@ -4,7 +4,7 @@ test_cases.contains.contract.MyContract.approval_program: // contains/contract.py:10 // one_true = self.is_in_tuple_1(UInt64(10), (UInt64(10), not_ten, Bytes(b"five"))) int 10 - int 10 + dup // contains/contract.py:8 // not_ten = UInt64(15) int 15 @@ -34,7 +34,7 @@ test_cases.contains.contract.MyContract.approval_program: // contains/contract.py:16 // Bytes(b"hello"), (Bytes(b"hello"), UInt64(0), Bytes(b"bonjour")) byte "hello" - byte "hello" + dup int 0 byte "bonjour" // contains/contract.py:15-17 @@ -64,7 +64,7 @@ test_cases.contains.contract.MyContract.approval_program: // contains/contract.py:25 // BigUInt(32323423423423), (BigUInt(32323423423423), BigUInt(8439439483934)) byte 0x1d65e22bcbbf - byte 0x1d65e22bcbbf + dup byte 0x07acf5cae41e // contains/contract.py:24-26 // three_true = self.is_in_tuple_3( diff --git a/test_cases/contains/out_O2/MyContract.approval.teal b/test_cases/contains/out_O2/MyContract.approval.teal index b8a20946eb..9c28408a46 100644 --- a/test_cases/contains/out_O2/MyContract.approval.teal +++ b/test_cases/contains/out_O2/MyContract.approval.teal @@ -2,7 +2,7 @@ test_cases.contains.contract.MyContract.approval_program: int 10 - int 10 + dup int 15 byte "five" callsub is_in_tuple_1 @@ -16,7 +16,7 @@ test_cases.contains.contract.MyContract.approval_program: ! assert // Should be false byte "hello" - byte "hello" + dup int 0 byte "bonjour" callsub is_in_tuple_2 @@ -30,7 +30,7 @@ test_cases.contains.contract.MyContract.approval_program: ! assert // Should be false byte 0x1d65e22bcbbf - byte 0x1d65e22bcbbf + dup byte 0x07acf5cae41e callsub is_in_tuple_3 byte 0x3acbc457977e diff --git a/test_cases/everything/out/MyContract.approval.teal b/test_cases/everything/out/MyContract.approval.teal index 56e18a04a4..14aee6919a 100644 --- a/test_cases/everything/out/MyContract.approval.teal +++ b/test_cases/everything/out/MyContract.approval.teal @@ -178,7 +178,7 @@ register: // everything/contract.py:51 // sender_name, sender_name_existed = self.name.maybe(account=0) int 0 - int 0 + dup byte "name" app_local_get_ex bury 1 @@ -231,7 +231,7 @@ say_hello: // everything/contract.py:59 // name, exists = self.name.maybe(account=0) int 0 - int 0 + dup byte "name" app_local_get_ex // everything/contract.py:60 diff --git a/test_cases/everything/out/MyContract.arc32.json b/test_cases/everything/out/MyContract.arc32.json index a426773249..3bfb1bd5a9 100644 --- a/test_cases/everything/out/MyContract.arc32.json +++ b/test_cases/everything/out/MyContract.arc32.json @@ -28,7 +28,7 @@ } }, "source": { - "approval": "#pragma version 10

test_cases.everything.contract.Everything.approval_program:
    // everything/contract.py:37
    // class Everything(ARC4Contract, MyMiddleBase, name="MyContract"):
    method "create()void"
    method "register(string)void"
    method "say_hello()string"
    method "calculate(uint64,uint64)uint64"
    method "close_out()void"
    txna ApplicationArgs 0
    match main_create_route@1 main_register_route@2 main_say_hello_route@3 main_calculate_route@4 main_close_out_route@5
    err // reject transaction

main_create_route@1:
    // everything/contract.py:41
    // @abimethod(create=True)
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    !
    assert // is creating
    callsub create
    int 1
    return

main_register_route@2:
    // everything/contract.py:47
    // @abimethod(allow_actions=["NoOp", "OptIn"])
    int 1
    txn OnCompletion
    shl
    int 3
    &
    assert // OnCompletion is one of NoOp, OptIn
    txn ApplicationID
    assert // is not creating
    // everything/contract.py:37
    // class Everything(ARC4Contract, MyMiddleBase, name="MyContract"):
    txna ApplicationArgs 1
    // everything/contract.py:47
    // @abimethod(allow_actions=["NoOp", "OptIn"])
    callsub register
    int 1
    return

main_say_hello_route@3:
    // everything/contract.py:56
    // @abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    callsub say_hello
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_calculate_route@4:
    // everything/contract.py:64
    // @abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // everything/contract.py:37
    // class Everything(ARC4Contract, MyMiddleBase, name="MyContract"):
    txna ApplicationArgs 1
    txna ApplicationArgs 2
    // everything/contract.py:64
    // @abimethod
    callsub calculate
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_close_out_route@5:
    // everything/contract.py:69
    // @abimethod(allow_actions=["CloseOut"])
    txn OnCompletion
    int CloseOut
    ==
    assert // OnCompletion is CloseOut
    txn ApplicationID
    assert // is not creating
    callsub close_out
    int 1
    return


// test_cases.everything.contract.Everything.create() -> void:
create:
    // everything/contract.py:41-42
    // @abimethod(create=True)
    // def create(self) -> None:
    proto 0 0
    // everything/contract.py:43
    // self._check_ban_list()
    callsub _check_ban_list
    // everything/contract.py:44
    // self.remember_creator()
    callsub remember_creator
    // everything/contract.py:45
    // self.counter = UInt64(ZERO)
    byte "counter"
    int 0
    app_global_put
    retsub


// test_cases.everything.contract.Everything._check_ban_list() -> void:
_check_ban_list:
    // everything/contract.py:77-78
    // @subroutine
    // def _check_ban_list(self) -> None:
    proto 0 0
    // everything/contract.py:79
    // assert op.Txn.sender != get_banned(), "You are banned, goodbye"
    txn Sender
    callsub get_banned
    !=
    assert // You are banned, goodbye
    retsub


// test_cases.everything.contract.get_banned() -> bytes:
get_banned:
    // everything/contract.py:23-24
    // @subroutine
    // def get_banned() -> Account:
    proto 0 1
    // everything/contract.py:25
    // addr = Account(BANNED)
    addr VCMJKWOY5P5P7SKMZFFOCEROPJCZOTIJMNIYNUCKH7LRO45JMJP6UYBIJA
    // everything/contract.py:26
    // return addr
    retsub


// test_cases.everything.my_base.MyBase.remember_creator() -> void:
remember_creator:
    // everything/my_base.py:7-8
    // @subroutine
    // def remember_creator(self) -> None:
    proto 0 0
    // everything/my_base.py:9
    // self.creator = op.Txn.sender
    byte "creator"
    txn Sender
    app_global_put
    retsub


// test_cases.everything.contract.Everything.register(name: bytes) -> void:
register:
    // everything/contract.py:47-48
    // @abimethod(allow_actions=["NoOp", "OptIn"])
    // def register(self, name: String) -> None:
    proto 1 0
    // everything/contract.py:49
    // self._check_ban_list()
    callsub _check_ban_list
    // everything/contract.py:50
    // if op.Txn.on_completion == OnCompleteAction.OptIn:
    txn OnCompletion
    int OptIn
    ==
    bz register_after_if_else@4
    // everything/contract.py:51
    // sender_name, sender_name_existed = self.name.maybe(account=0)
    int 0
    int 0
    byte "name"
    app_local_get_ex
    bury 1
    // everything/contract.py:52
    // if not sender_name_existed:
    bnz register_after_if_else@4
    // everything/contract.py:53
    // self.counter += multiplicative_identity()  # has full FuncDef
    int 0
    byte "counter"
    app_global_get_ex
    assert // check counter exists
    callsub multiplicative_identity
    +
    byte "counter"
    swap
    app_global_put

register_after_if_else@4:
    // everything/contract.py:54
    // self.name[0] = name
    int 0
    byte "name"
    frame_dig -1
    app_local_put
    retsub


// test_cases.everything.my_base.multiplicative_identity() -> uint64:
multiplicative_identity:
    // everything/my_base.py:18-19
    // @subroutine
    // def multiplicative_identity() -> UInt64:
    proto 0 1
    // everything/my_base.py:20
    // return UInt64(1)
    int 1
    retsub


// test_cases.everything.contract.Everything.say_hello() -> bytes:
say_hello:
    // everything/contract.py:56-57
    // @abimethod
    // def say_hello(self) -> String:
    proto 0 1
    // everything/contract.py:58
    // self._check_ban_list()
    callsub _check_ban_list
    // everything/contract.py:59
    // name, exists = self.name.maybe(account=0)
    int 0
    int 0
    byte "name"
    app_local_get_ex
    // everything/contract.py:60
    // if not exists:
    bnz say_hello_after_if_else@2
    // everything/contract.py:61
    // return String("Howdy stranger!")
    byte "\x00\x0fHowdy stranger!"
    swap
    retsub

say_hello_after_if_else@2:
    // everything/contract.py:62
    // return "Hello, " + name + "!"
    frame_dig 0
    extract 2 0
    byte "Hello, "
    swap
    concat
    dup
    len
    itob
    extract 6 0
    swap
    concat
    extract 2 0
    byte "!"
    concat
    dup
    len
    itob
    extract 6 0
    swap
    concat
    swap
    retsub


// test_cases.everything.contract.Everything.calculate(a: bytes, b: bytes) -> bytes:
calculate:
    // everything/contract.py:64-65
    // @abimethod
    // def calculate(self, a: arc4_UInt64, b: arc4_UInt64) -> arc4_UInt64:
    proto 2 1
    // everything/contract.py:66
    // c = super().calculate(a, b)
    frame_dig -2
    frame_dig -1
    callsub MyMiddleBase.calculate
    // everything/contract.py:67
    // return arc4_UInt64(c.decode() * b.decode())
    btoi
    frame_dig -1
    btoi
    *
    itob
    retsub


// test_cases.everything.my_base.MyMiddleBase.calculate(a: bytes, b: bytes) -> bytes:
MyMiddleBase.calculate:
    // everything/my_base.py:13-14
    // @subroutine
    // def calculate(self, a: arc4.UInt64, b: arc4.UInt64) -> arc4.UInt64:
    proto 2 1
    // everything/my_base.py:15
    // return arc4.UInt64(a.decode() + b.decode())
    frame_dig -2
    btoi
    frame_dig -1
    btoi
    +
    itob
    retsub


// test_cases.everything.contract.Everything.close_out() -> void:
close_out:
    // everything/contract.py:69-70
    // @abimethod(allow_actions=["CloseOut"])
    // def close_out(self) -> None:
    proto 0 0
    // everything/contract.py:71
    // self._remove_sender()
    callsub _remove_sender
    retsub


// test_cases.everything.contract.Everything._remove_sender() -> void:
_remove_sender:
    // everything/contract.py:81-82
    // @subroutine
    // def _remove_sender(self) -> None:
    proto 0 0
    // everything/contract.py:83
    // self.counter -= positive_one()
    int 0
    byte "counter"
    app_global_get_ex
    assert // check counter exists
    callsub positive_one
    -
    byte "counter"
    swap
    app_global_put
    retsub


// test_cases.everything.contract.positive_one() -> uint64:
positive_one:
    // everything/contract.py:86-87
    // @subroutine
    // def positive_one() -> UInt64:
    proto 0 1
    // everything/contract.py:88
    // return UInt64(1)
    int 1
    retsub
", + "approval": "#pragma version 10

test_cases.everything.contract.Everything.approval_program:
    // everything/contract.py:37
    // class Everything(ARC4Contract, MyMiddleBase, name="MyContract"):
    method "create()void"
    method "register(string)void"
    method "say_hello()string"
    method "calculate(uint64,uint64)uint64"
    method "close_out()void"
    txna ApplicationArgs 0
    match main_create_route@1 main_register_route@2 main_say_hello_route@3 main_calculate_route@4 main_close_out_route@5
    err // reject transaction

main_create_route@1:
    // everything/contract.py:41
    // @abimethod(create=True)
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    !
    assert // is creating
    callsub create
    int 1
    return

main_register_route@2:
    // everything/contract.py:47
    // @abimethod(allow_actions=["NoOp", "OptIn"])
    int 1
    txn OnCompletion
    shl
    int 3
    &
    assert // OnCompletion is one of NoOp, OptIn
    txn ApplicationID
    assert // is not creating
    // everything/contract.py:37
    // class Everything(ARC4Contract, MyMiddleBase, name="MyContract"):
    txna ApplicationArgs 1
    // everything/contract.py:47
    // @abimethod(allow_actions=["NoOp", "OptIn"])
    callsub register
    int 1
    return

main_say_hello_route@3:
    // everything/contract.py:56
    // @abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    callsub say_hello
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_calculate_route@4:
    // everything/contract.py:64
    // @abimethod
    txn OnCompletion
    !
    assert // OnCompletion is NoOp
    txn ApplicationID
    assert // is not creating
    // everything/contract.py:37
    // class Everything(ARC4Contract, MyMiddleBase, name="MyContract"):
    txna ApplicationArgs 1
    txna ApplicationArgs 2
    // everything/contract.py:64
    // @abimethod
    callsub calculate
    byte 0x151f7c75
    swap
    concat
    log
    int 1
    return

main_close_out_route@5:
    // everything/contract.py:69
    // @abimethod(allow_actions=["CloseOut"])
    txn OnCompletion
    int CloseOut
    ==
    assert // OnCompletion is CloseOut
    txn ApplicationID
    assert // is not creating
    callsub close_out
    int 1
    return


// test_cases.everything.contract.Everything.create() -> void:
create:
    // everything/contract.py:41-42
    // @abimethod(create=True)
    // def create(self) -> None:
    proto 0 0
    // everything/contract.py:43
    // self._check_ban_list()
    callsub _check_ban_list
    // everything/contract.py:44
    // self.remember_creator()
    callsub remember_creator
    // everything/contract.py:45
    // self.counter = UInt64(ZERO)
    byte "counter"
    int 0
    app_global_put
    retsub


// test_cases.everything.contract.Everything._check_ban_list() -> void:
_check_ban_list:
    // everything/contract.py:77-78
    // @subroutine
    // def _check_ban_list(self) -> None:
    proto 0 0
    // everything/contract.py:79
    // assert op.Txn.sender != get_banned(), "You are banned, goodbye"
    txn Sender
    callsub get_banned
    !=
    assert // You are banned, goodbye
    retsub


// test_cases.everything.contract.get_banned() -> bytes:
get_banned:
    // everything/contract.py:23-24
    // @subroutine
    // def get_banned() -> Account:
    proto 0 1
    // everything/contract.py:25
    // addr = Account(BANNED)
    addr VCMJKWOY5P5P7SKMZFFOCEROPJCZOTIJMNIYNUCKH7LRO45JMJP6UYBIJA
    // everything/contract.py:26
    // return addr
    retsub


// test_cases.everything.my_base.MyBase.remember_creator() -> void:
remember_creator:
    // everything/my_base.py:7-8
    // @subroutine
    // def remember_creator(self) -> None:
    proto 0 0
    // everything/my_base.py:9
    // self.creator = op.Txn.sender
    byte "creator"
    txn Sender
    app_global_put
    retsub


// test_cases.everything.contract.Everything.register(name: bytes) -> void:
register:
    // everything/contract.py:47-48
    // @abimethod(allow_actions=["NoOp", "OptIn"])
    // def register(self, name: String) -> None:
    proto 1 0
    // everything/contract.py:49
    // self._check_ban_list()
    callsub _check_ban_list
    // everything/contract.py:50
    // if op.Txn.on_completion == OnCompleteAction.OptIn:
    txn OnCompletion
    int OptIn
    ==
    bz register_after_if_else@4
    // everything/contract.py:51
    // sender_name, sender_name_existed = self.name.maybe(account=0)
    int 0
    dup
    byte "name"
    app_local_get_ex
    bury 1
    // everything/contract.py:52
    // if not sender_name_existed:
    bnz register_after_if_else@4
    // everything/contract.py:53
    // self.counter += multiplicative_identity()  # has full FuncDef
    int 0
    byte "counter"
    app_global_get_ex
    assert // check counter exists
    callsub multiplicative_identity
    +
    byte "counter"
    swap
    app_global_put

register_after_if_else@4:
    // everything/contract.py:54
    // self.name[0] = name
    int 0
    byte "name"
    frame_dig -1
    app_local_put
    retsub


// test_cases.everything.my_base.multiplicative_identity() -> uint64:
multiplicative_identity:
    // everything/my_base.py:18-19
    // @subroutine
    // def multiplicative_identity() -> UInt64:
    proto 0 1
    // everything/my_base.py:20
    // return UInt64(1)
    int 1
    retsub


// test_cases.everything.contract.Everything.say_hello() -> bytes:
say_hello:
    // everything/contract.py:56-57
    // @abimethod
    // def say_hello(self) -> String:
    proto 0 1
    // everything/contract.py:58
    // self._check_ban_list()
    callsub _check_ban_list
    // everything/contract.py:59
    // name, exists = self.name.maybe(account=0)
    int 0
    dup
    byte "name"
    app_local_get_ex
    // everything/contract.py:60
    // if not exists:
    bnz say_hello_after_if_else@2
    // everything/contract.py:61
    // return String("Howdy stranger!")
    byte "\x00\x0fHowdy stranger!"
    swap
    retsub

say_hello_after_if_else@2:
    // everything/contract.py:62
    // return "Hello, " + name + "!"
    frame_dig 0
    extract 2 0
    byte "Hello, "
    swap
    concat
    dup
    len
    itob
    extract 6 0
    swap
    concat
    extract 2 0
    byte "!"
    concat
    dup
    len
    itob
    extract 6 0
    swap
    concat
    swap
    retsub


// test_cases.everything.contract.Everything.calculate(a: bytes, b: bytes) -> bytes:
calculate:
    // everything/contract.py:64-65
    // @abimethod
    // def calculate(self, a: arc4_UInt64, b: arc4_UInt64) -> arc4_UInt64:
    proto 2 1
    // everything/contract.py:66
    // c = super().calculate(a, b)
    frame_dig -2
    frame_dig -1
    callsub MyMiddleBase.calculate
    // everything/contract.py:67
    // return arc4_UInt64(c.decode() * b.decode())
    btoi
    frame_dig -1
    btoi
    *
    itob
    retsub


// test_cases.everything.my_base.MyMiddleBase.calculate(a: bytes, b: bytes) -> bytes:
MyMiddleBase.calculate:
    // everything/my_base.py:13-14
    // @subroutine
    // def calculate(self, a: arc4.UInt64, b: arc4.UInt64) -> arc4.UInt64:
    proto 2 1
    // everything/my_base.py:15
    // return arc4.UInt64(a.decode() + b.decode())
    frame_dig -2
    btoi
    frame_dig -1
    btoi
    +
    itob
    retsub


// test_cases.everything.contract.Everything.close_out() -> void:
close_out:
    // everything/contract.py:69-70
    // @abimethod(allow_actions=["CloseOut"])
    // def close_out(self) -> None:
    proto 0 0
    // everything/contract.py:71
    // self._remove_sender()
    callsub _remove_sender
    retsub


// test_cases.everything.contract.Everything._remove_sender() -> void:
_remove_sender:
    // everything/contract.py:81-82
    // @subroutine
    // def _remove_sender(self) -> None:
    proto 0 0
    // everything/contract.py:83
    // self.counter -= positive_one()
    int 0
    byte "counter"
    app_global_get_ex
    assert // check counter exists
    callsub positive_one
    -
    byte "counter"
    swap
    app_global_put
    retsub


// test_cases.everything.contract.positive_one() -> uint64:
positive_one:
    // everything/contract.py:86-87
    // @subroutine
    // def positive_one() -> UInt64:
    proto 0 1
    // everything/contract.py:88
    // return UInt64(1)
    int 1
    retsub
", "clear": "I3ByYWdtYSB2ZXJzaW9uIDEwCgp0ZXN0X2Nhc2VzLmV2ZXJ5dGhpbmcuY29udHJhY3QuRXZlcnl0aGluZy5jbGVhcl9zdGF0ZV9wcm9ncmFtOgogICAgLy8gZXZlcnl0aGluZy9jb250cmFjdC5weTo3NAogICAgLy8gc2VsZi5fcmVtb3ZlX3NlbmRlcigpCiAgICBjYWxsc3ViIF9yZW1vdmVfc2VuZGVyCiAgICAvLyBldmVyeXRoaW5nL2NvbnRyYWN0LnB5Ojc1CiAgICAvLyByZXR1cm4gVHJ1ZQogICAgaW50IDEKICAgIHJldHVybgoKCi8vIHRlc3RfY2FzZXMuZXZlcnl0aGluZy5jb250cmFjdC5FdmVyeXRoaW5nLl9yZW1vdmVfc2VuZGVyKCkgLT4gdm9pZDoKX3JlbW92ZV9zZW5kZXI6CiAgICAvLyBldmVyeXRoaW5nL2NvbnRyYWN0LnB5OjgxLTgyCiAgICAvLyBAc3Vicm91dGluZQogICAgLy8gZGVmIF9yZW1vdmVfc2VuZGVyKHNlbGYpIC0+IE5vbmU6CiAgICBwcm90byAwIDAKICAgIC8vIGV2ZXJ5dGhpbmcvY29udHJhY3QucHk6ODMKICAgIC8vIHNlbGYuY291bnRlciAtPSBwb3NpdGl2ZV9vbmUoKQogICAgaW50IDAKICAgIGJ5dGUgImNvdW50ZXIiCiAgICBhcHBfZ2xvYmFsX2dldF9leAogICAgYXNzZXJ0IC8vIGNoZWNrIGNvdW50ZXIgZXhpc3RzCiAgICBjYWxsc3ViIHBvc2l0aXZlX29uZQogICAgLQogICAgYnl0ZSAiY291bnRlciIKICAgIHN3YXAKICAgIGFwcF9nbG9iYWxfcHV0CiAgICByZXRzdWIKCgovLyB0ZXN0X2Nhc2VzLmV2ZXJ5dGhpbmcuY29udHJhY3QucG9zaXRpdmVfb25lKCkgLT4gdWludDY0Ogpwb3NpdGl2ZV9vbmU6CiAgICAvLyBldmVyeXRoaW5nL2NvbnRyYWN0LnB5Ojg2LTg3CiAgICAvLyBAc3Vicm91dGluZQogICAgLy8gZGVmIHBvc2l0aXZlX29uZSgpIC0+IFVJbnQ2NDoKICAgIHByb3RvIDAgMQogICAgLy8gZXZlcnl0aGluZy9jb250cmFjdC5weTo4OAogICAgLy8gcmV0dXJuIFVJbnQ2NCgxKQogICAgaW50IDEKICAgIHJldHN1Ygo=" }, "state": { diff --git a/test_cases/everything/out_O2/MyContract.approval.teal b/test_cases/everything/out_O2/MyContract.approval.teal index bd5f8f6e3d..5ef78eecc5 100644 --- a/test_cases/everything/out_O2/MyContract.approval.teal +++ b/test_cases/everything/out_O2/MyContract.approval.teal @@ -123,7 +123,7 @@ register: == bz register_after_if_else@4 int 0 - int 0 + dup byte "name" app_local_get_ex bury 1 @@ -158,7 +158,7 @@ say_hello: proto 0 1 callsub _check_ban_list int 0 - int 0 + dup byte "name" app_local_get_ex bnz say_hello_after_if_else@2 diff --git a/test_cases/inner_transactions/out/MyContract.approval.teal b/test_cases/inner_transactions/out/MyContract.approval.teal index d2f13907fc..e4814b1fa9 100644 --- a/test_cases/inner_transactions/out/MyContract.approval.teal +++ b/test_cases/inner_transactions/out/MyContract.approval.teal @@ -93,7 +93,7 @@ test1: global CurrentApplicationAddress // inner_transactions/contract.py:51 // reserve=op.Global.current_application_address, - global CurrentApplicationAddress + dup // inner_transactions/contract.py:54 // self.name = Bytes(b"AST2") byte "name" diff --git a/test_cases/inner_transactions/out/itxn_loop.approval.teal b/test_cases/inner_transactions/out/itxn_loop.approval.teal index 7d57d2833f..586dca1577 100644 --- a/test_cases/inner_transactions/out/itxn_loop.approval.teal +++ b/test_cases/inner_transactions/out/itxn_loop.approval.teal @@ -11,7 +11,7 @@ test_cases.inner_transactions.itxn_loop.MyContract.approval_program: int 0 // inner_transactions/itxn_loop.py:28 // for i in urange(4): - int 0 + dup main_for_header@1: // inner_transactions/itxn_loop.py:28 diff --git a/test_cases/inner_transactions/out_O2/MyContract.approval.teal b/test_cases/inner_transactions/out_O2/MyContract.approval.teal index 161c0eca27..d6a4a21472 100644 --- a/test_cases/inner_transactions/out_O2/MyContract.approval.teal +++ b/test_cases/inner_transactions/out_O2/MyContract.approval.teal @@ -48,7 +48,7 @@ test1: app_global_get_ex assert // check name exists global CurrentApplicationAddress - global CurrentApplicationAddress + dup byte "name" byte "AST2" app_global_put diff --git a/test_cases/inner_transactions/out_O2/itxn_loop.approval.teal b/test_cases/inner_transactions/out_O2/itxn_loop.approval.teal index 0392faa8c4..2a3e52cb4c 100644 --- a/test_cases/inner_transactions/out_O2/itxn_loop.approval.teal +++ b/test_cases/inner_transactions/out_O2/itxn_loop.approval.teal @@ -5,7 +5,7 @@ test_cases.inner_transactions.itxn_loop.MyContract.approval_program: dupn 3 byte "" int 0 - int 0 + dup main_for_header@1: dup diff --git a/test_cases/intrinsics/out/ImmediateVariants.approval.teal b/test_cases/intrinsics/out/ImmediateVariants.approval.teal index 1e84c96ba5..67e9ca4b77 100644 --- a/test_cases/intrinsics/out/ImmediateVariants.approval.teal +++ b/test_cases/intrinsics/out/ImmediateVariants.approval.teal @@ -49,7 +49,7 @@ test_cases.intrinsics.immediate_variants.ImmediateVariants.approval_program: // intrinsics/immediate_variants.py:27 // assert GTxn.application_args(UInt64(0), UInt64(0)) == first_arg int 0 - int 0 + dup gtxnsas ApplicationArgs dig 1 == diff --git a/test_cases/intrinsics/out_O2/ImmediateVariants.approval.teal b/test_cases/intrinsics/out_O2/ImmediateVariants.approval.teal index a877bf2a22..7434484116 100644 --- a/test_cases/intrinsics/out_O2/ImmediateVariants.approval.teal +++ b/test_cases/intrinsics/out_O2/ImmediateVariants.approval.teal @@ -31,7 +31,7 @@ test_cases.intrinsics.immediate_variants.ImmediateVariants.approval_program: == assert int 0 - int 0 + dup gtxnsas ApplicationArgs dig 1 == diff --git a/test_cases/less_simple/out/MyContract.approval.teal b/test_cases/less_simple/out/MyContract.approval.teal index 09c881e5b0..47c3014fbf 100644 --- a/test_cases/less_simple/out/MyContract.approval.teal +++ b/test_cases/less_simple/out/MyContract.approval.teal @@ -9,7 +9,7 @@ test_cases.less_simple.contract.MyContract.approval_program: int 0 // less_simple/contract.py:11 // product_of_odds = UInt64(0) - int 0 + dup main_while_top@1: // less_simple/contract.py:12 diff --git a/test_cases/less_simple/out_O2/MyContract.approval.teal b/test_cases/less_simple/out_O2/MyContract.approval.teal index 021ef5f2db..dc6bdbf315 100644 --- a/test_cases/less_simple/out_O2/MyContract.approval.teal +++ b/test_cases/less_simple/out_O2/MyContract.approval.teal @@ -3,7 +3,7 @@ test_cases.less_simple.contract.MyContract.approval_program: int 1 int 0 - int 0 + dup main_while_top@1: dig 2 diff --git a/test_cases/log/out/MyContract.approval.teal b/test_cases/log/out/MyContract.approval.teal index ba3ac845e4..c6d4f11bfd 100644 --- a/test_cases/log/out/MyContract.approval.teal +++ b/test_cases/log/out/MyContract.approval.teal @@ -2,10 +2,9 @@ test_cases.log.contract.MyContract.approval_program: int 0 - dupn 2 // log/contract.py:6 // log(0) - int 0 + dupn 3 itob log // log/contract.py:7 diff --git a/test_cases/log/out_O2/MyContract.approval.teal b/test_cases/log/out_O2/MyContract.approval.teal index 02a1fbca04..972175f377 100644 --- a/test_cases/log/out_O2/MyContract.approval.teal +++ b/test_cases/log/out_O2/MyContract.approval.teal @@ -2,8 +2,7 @@ test_cases.log.contract.MyContract.approval_program: int 0 - dupn 2 - int 0 + dupn 3 itob log byte "1" diff --git a/test_cases/nested_loops/out/Nested.approval.teal b/test_cases/nested_loops/out/Nested.approval.teal index 679dcdc480..f7f6a2ad24 100644 --- a/test_cases/nested_loops/out/Nested.approval.teal +++ b/test_cases/nested_loops/out/Nested.approval.teal @@ -8,7 +8,7 @@ test_cases.nested_loops.contract.Nested.approval_program: int 0 // nested_loops/contract.py:11 // for a in urange(n): - int 0 + dup main_for_header@1: // nested_loops/contract.py:11 @@ -184,7 +184,7 @@ main_after_for@30: // nested_loops/contract.py:25 // for index, item in uenumerate(urange(UInt64(10))): int 0 - int 0 + dup bury 6 bury 4 diff --git a/test_cases/nested_loops/out_O2/Nested.approval.teal b/test_cases/nested_loops/out_O2/Nested.approval.teal index c13f79a93f..81dec8f51c 100644 --- a/test_cases/nested_loops/out_O2/Nested.approval.teal +++ b/test_cases/nested_loops/out_O2/Nested.approval.teal @@ -4,7 +4,7 @@ test_cases.nested_loops.contract.Nested.approval_program: byte "" dupn 7 int 0 - int 0 + dup main_for_header@1: dup @@ -114,7 +114,7 @@ main_after_for@30: int 0 bury 3 int 0 - int 0 + dup bury 6 bury 4 diff --git a/test_cases/reversed_iteration/out/MyContract.approval.teal b/test_cases/reversed_iteration/out/MyContract.approval.teal index de49303e23..027cb87499 100644 --- a/test_cases/reversed_iteration/out/MyContract.approval.teal +++ b/test_cases/reversed_iteration/out/MyContract.approval.teal @@ -8,7 +8,7 @@ test_cases.reversed_iteration.contract.MyContract.approval_program: int 0 // reversed_iteration/contract.py:11 // for x in reversed(arc4.StaticArray[arc4.UInt8, typing.Literal[0]]()): - int 0 + dup main_for_header@7: // reversed_iteration/contract.py:11 diff --git a/test_cases/reversed_iteration/out/trace.O1.log b/test_cases/reversed_iteration/out/trace.O1.log index 00f2fcb517..e1b04307d7 100644 --- a/test_cases/reversed_iteration/out/trace.O1.log +++ b/test_cases/reversed_iteration/out/trace.O1.log @@ -6,7 +6,7 @@ PC Teal Stack 32 byte "" 0, 0, 0, 0, 0x 33 dupn 11 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x 35 int 0 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0 -36 int 0 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0 +36 dup 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0 37 dig 1 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0, 0 39 int 0 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0, 0, 0 40 < 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0, 0 diff --git a/test_cases/reversed_iteration/out/trace.O2.log b/test_cases/reversed_iteration/out/trace.O2.log index 00f2fcb517..e1b04307d7 100644 --- a/test_cases/reversed_iteration/out/trace.O2.log +++ b/test_cases/reversed_iteration/out/trace.O2.log @@ -6,7 +6,7 @@ PC Teal Stack 32 byte "" 0, 0, 0, 0, 0x 33 dupn 11 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x 35 int 0 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0 -36 int 0 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0 +36 dup 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0 37 dig 1 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0, 0 39 int 0 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0, 0, 0 40 < 0, 0, 0, 0, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0x, 0, 0, 0 diff --git a/test_cases/reversed_iteration/out_O2/MyContract.approval.teal b/test_cases/reversed_iteration/out_O2/MyContract.approval.teal index a29c985a3d..ed01a918a9 100644 --- a/test_cases/reversed_iteration/out_O2/MyContract.approval.teal +++ b/test_cases/reversed_iteration/out_O2/MyContract.approval.teal @@ -6,7 +6,7 @@ test_cases.reversed_iteration.contract.MyContract.approval_program: byte "" dupn 11 int 0 - int 0 + dup main_for_header@7: dig 1 diff --git a/test_cases/too_many_permutations/out/MyContract.approval.teal b/test_cases/too_many_permutations/out/MyContract.approval.teal index 2e9de4a447..13b4b5cfa3 100644 --- a/test_cases/too_many_permutations/out/MyContract.approval.teal +++ b/test_cases/too_many_permutations/out/MyContract.approval.teal @@ -4,20 +4,21 @@ test_cases.too_many_permutations.contract.MyContract.approval_program: // too_many_permutations/contract.py:6 // a = op.Txn.application_args(0) txna ApplicationArgs 0 - dup // too_many_permutations/contract.py:7 // b = op.Txn.application_args(1) txna ApplicationArgs 1 - swap // too_many_permutations/contract.py:8 // c = op.Txn.application_args(2) txna ApplicationArgs 2 - dup - cover 2 // too_many_permutations/contract.py:9 // d = op.Txn.application_args(3) txna ApplicationArgs 3 - cover 2 + // too_many_permutations/contract.py:6 + // a = op.Txn.application_args(0) + txna ApplicationArgs 0 + // too_many_permutations/contract.py:8 + // c = op.Txn.application_args(2) + txna ApplicationArgs 2 // too_many_permutations/contract.py:11 // assert (a != c) or (b != d) != diff --git a/test_cases/too_many_permutations/out_O2/MyContract.approval.teal b/test_cases/too_many_permutations/out_O2/MyContract.approval.teal index f14e5e12b3..e87c5040bf 100644 --- a/test_cases/too_many_permutations/out_O2/MyContract.approval.teal +++ b/test_cases/too_many_permutations/out_O2/MyContract.approval.teal @@ -2,14 +2,11 @@ test_cases.too_many_permutations.contract.MyContract.approval_program: txna ApplicationArgs 0 - dup txna ApplicationArgs 1 - swap txna ApplicationArgs 2 - dup - cover 2 txna ApplicationArgs 3 - cover 2 + txna ApplicationArgs 0 + txna ApplicationArgs 2 != bnz main_bool_true@2 dig 2 diff --git a/test_cases/tuple_support/out/TupleSupport.approval.teal b/test_cases/tuple_support/out/TupleSupport.approval.teal index f7cde5e3c9..a1cdccbb37 100644 --- a/test_cases/tuple_support/out/TupleSupport.approval.teal +++ b/test_cases/tuple_support/out/TupleSupport.approval.teal @@ -42,7 +42,7 @@ main_entrypoint@2: // tuple_support/tuple_support.py:34 // max_uint64 = UInt64(2**64 - 1) int 18446744073709551615 - int 18446744073709551615 + dup // tuple_support/tuple_support.py:35 // hi, mid, lo = addw2(op.addw(max_uint64, max_uint64), op.addw(a, b)) addw diff --git a/test_cases/tuple_support/out_O2/TupleSupport.approval.teal b/test_cases/tuple_support/out_O2/TupleSupport.approval.teal index c00b9f71ce..689fa3c871 100644 --- a/test_cases/tuple_support/out_O2/TupleSupport.approval.teal +++ b/test_cases/tuple_support/out_O2/TupleSupport.approval.teal @@ -26,7 +26,7 @@ main_entrypoint@2: callsub bytes_combine log int 18446744073709551615 - int 18446744073709551615 + dup addw uncover 3 uncover 3 diff --git a/test_cases/undefined_phi_args/out/Baddie.approval.teal b/test_cases/undefined_phi_args/out/Baddie.approval.teal index e042bdf575..407a45536a 100644 --- a/test_cases/undefined_phi_args/out/Baddie.approval.teal +++ b/test_cases/undefined_phi_args/out/Baddie.approval.teal @@ -103,7 +103,7 @@ main_else_body@12: // undefined_phi_args/baddie.py:23 // assert test_uint_undefined(True, True) == 8 int 1 - int 1 + dup callsub test_uint_undefined int 8 == @@ -111,7 +111,7 @@ main_else_body@12: // undefined_phi_args/baddie.py:24 // assert test_uint_undefined(False, False) == 10 int 0 - int 0 + dup callsub test_uint_undefined int 10 == @@ -128,7 +128,7 @@ main_else_body@14: // undefined_phi_args/baddie.py:26 // assert test_bytes_undefined(True, True) == 8 int 1 - int 1 + dup callsub test_bytes_undefined byte 0x08 b== @@ -136,7 +136,7 @@ main_else_body@14: // undefined_phi_args/baddie.py:27 // assert test_bytes_undefined(False, False) == 10 int 0 - int 0 + dup callsub test_bytes_undefined byte 0x0a b== @@ -147,7 +147,7 @@ main_else_body@16: // undefined_phi_args/baddie.py:29 // assert test_mixed_undefined(True, True) == 8 int 1 - int 1 + dup callsub test_mixed_undefined byte 0x08 b== @@ -155,7 +155,7 @@ main_else_body@16: // undefined_phi_args/baddie.py:30 // assert test_mixed_undefined(False, False) == 10 int 0 - int 0 + dup callsub test_mixed_undefined byte 0x0a b== diff --git a/test_cases/undefined_phi_args/out_O2/Baddie.approval.teal b/test_cases/undefined_phi_args/out_O2/Baddie.approval.teal index 89fab29738..3d3deb7869 100644 --- a/test_cases/undefined_phi_args/out_O2/Baddie.approval.teal +++ b/test_cases/undefined_phi_args/out_O2/Baddie.approval.teal @@ -77,13 +77,13 @@ main_else_body@12: == bz main_else_body@14 int 1 - int 1 + dup callsub test_uint_undefined int 8 == assert int 0 - int 0 + dup callsub test_uint_undefined int 10 == @@ -96,13 +96,13 @@ main_else_body@14: == bz main_else_body@16 int 1 - int 1 + dup callsub test_bytes_undefined byte 0x08 b== assert int 0 - int 0 + dup callsub test_bytes_undefined byte 0x0a b== @@ -111,13 +111,13 @@ main_else_body@14: main_else_body@16: int 1 - int 1 + dup callsub test_mixed_undefined byte 0x08 b== assert int 0 - int 0 + dup callsub test_mixed_undefined byte 0x0a b==