Close Menu

    Subscribe to Updates

    Get the latest creative news from FooBar about art, design and business.

    What's Hot

    XRP hits two-month highs as breakout points to $3 target

    May 13, 2025

    Coinbase becomes first crypto company to join the S&P 500

    May 13, 2025

    Coinbase stock jumps 8% as S&P 500 inclusion promises broader exposure

    May 13, 2025
    Facebook X (Twitter) Instagram YouTube
    X (Twitter) Instagram YouTube LinkedIn
    Block Hub News
    • Lithosphere News Releases
    • Altcoins
      • Bitcoin
      • Coinbase
      • Litecoin
    • Crypto
    • Ethereum
    • Blockchain
    Block Hub News
    You are at:Home » Serpent upgrades: More Fun Stuff
    Ethereum

    Serpent upgrades: More Fun Stuff

    Olivia MartinezBy Olivia MartinezApril 25, 2025No Comments7 Mins Read
    Facebook Twitter Pinterest LinkedIn Tumblr Email
    Share
    Facebook Twitter LinkedIn Pinterest Email


    Over the past two weeks our primary focus has been getting all of the clients updated to PoC5 compatibility, and it definitely has been a long road. Among the changes to the VM include:

    • The new init/code mechanism: basically, when you create a contract, the code provided will execute immediately, and then the return value of that code will be what becomes the contract’s code. This allows us to have contract initialization code, but still keep to the same format of [nonce, price, gas, to, value, data] for both transactions and contract creation, also making it easier to create new contracts via forwarding contracts
    • Reordering transaction and contract data: the order is now [nonce, price, gas, to, value, data] in transactions and [gas, to, value, datain, datainsz, dataout, dataoutsz] in messages. Note that Serpent retains the send(to, value, gas), o = msg(to, value, gas, datain, datainsz) and o = msg(to, value, gas, datain, datainsz, dataoutsz) parameters.
    • Fee adjustments: transaction creation now has a fee of 500 gas, and several other fees were updated.
    • The CODECOPY and CALLDATACOPY opcodes: CODECOPY takes code_index, mem_index, len as arguments, and copies the code from code_index … code_index+len-1 to memory mem_index … mem_index+len-1. These are very useful when combined with init/code. There is also now CODESIZE.

    The largest changes, however, have been to the architecture surrounding the protocol. On the GUI side, the C++ and Go clients are evolving rapidly, and we will see more updates from that side coming very shortly. If you have been following Ethereum closely, you have likely seen Denny’s Lotto, a full implementation of a lottery, plus GUI, written and executed inside the C++ client. From here on, the C++ client will shift toward being a more developer-oriented tool, whereas the Go client will start to focus on being a user-facing application (or rather, meta-application). On the compiler side, Serpent has undergone a number of substantial improvements.

    First, the code. You can peek into the Serpent compiler under the hood and you will be able to see all of the functionsavailable, together with their precise translations into EVM code. For example, we have:

    72:     [‘access’, 2, 1,
    73:         [”, ”, 32, ‘MUL’, ‘ADD’, ‘MLOAD’]],

    This means that what access(x,y) is actually doing under the hood is it’s recursively compiling whatever x and y actually are, and then loading the memory at index x + y * 32; hence, x is the pointer to the start of the array and y is the index. This code structure has been around since PoC4, but now I have upgraded the meta-language used to describe translations even further, so as to include even if, while and init/code in this construction (before they were special cases); now, only set and seq remain as special cases, and if I wanted to I could even remove seq by reimplementing it as a rewrite rule.

    The largest changes so far have been for PoC5 compatibility. For example, if you run serpent compile_to_assembly ‘return(msg.data[0]*2)’, you will see:

    [“begincode0.endcode0“,“DUP“,“MSIZE“,“SWAP“,“MSIZE“,“begincode_0.endcode_0″, “DUP”, “MSIZE”, “SWAP”, “MSIZE”, “begincode0​.endcode0​“,“DUP“,“MSIZE“,“SWAP“,“MSIZE“,“begincode_0″, “CALLDATACOPY”, “RETURN”, “~begincode_0”, “#CODE_BEGIN”, 2, 0, “CALLDATALOAD”, “MUL”, “MSIZE”, “SWAP”, “MSIZE”, “MSTORE”, 32, “SWAP”, “RETURN”, “#CODE_END”, “~endcode_0”]

    The actual code there is just:

    [2, 0, “CALLDATALOAD”, “MUL”, “MSIZE”, “SWAP”, “MSIZE”, “MSTORE”, 32, “SWAP”, “RETURN”]

    If you want to see what’s going on here, suppose that a message is coming in with its first datum being 5. We thus have:

    2 -> Stack: [2]
    0 -> Stack: [2, 0]
    CALLDATALOAD -> Stack: [2,5]
    MUL -> Stack: [10]
    MSIZE -> Stack: [10, 0]
    SWAP -> Stack: [0, 10]
    MSIZE -> Stack: [0, 10, 0]
    MSTORE -> Stack: [0], Memory: [0, 0, 0 … 10]
    32 -> Stack: [0, 32], Memory: [0, 0, 0 … 10]
    SWAP -> Stack: [32, 0], Memory: [0, 0, 0 … 10]
    RETURN

    The last RETURN returns the 32 memory bytes starting from 0, or [0, 0, 0 … 10], or the number 10.

    Now, let’s analyze the wrapper code.

    [“begincode0.endcode0“,“DUP“,“MSIZE“,“SWAP“,“MSIZE“,“begincode_0.endcode_0″, “DUP”, “MSIZE”, “SWAP”, “MSIZE”, “begincode0​.endcode0​“,“DUP“,“MSIZE“,“SWAP“,“MSIZE“,“begincode_0″, “CALLDATACOPY”, “RETURN”, “~begincode_0”, “#CODE_BEGIN”, ….. , “#CODE_END”, “~endcode_0”]

    I elided the inner code explained above to make things clearer. The first thing we see are two labels, begincode_0 andendcode_0, and the #CODE_BEGIN and #CODE_END guards. The labels mark the beginning and end of the inner code, and the guards are there for the later stages of the compiler, which understands that everything between the guards should be compiled as if it is a separate program. Now, let’s look at the first parts of the code. In this case, we have ~begincode_0 at position 10 and ~endcode_0 at position 24 in the final code. begincode0andbegincode_0 and begincode0​andendcode_0 are used to refer to these positions, and $begincode_0.endcode_0 refers to the length of the interval between them, 14. Now, remember that during contract initialization the call data is the code that you’re feeding in. Thus, we have:

    14 -> Stack: [14]
    DUP -> Stack: [14, 14]
    MSIZE -> Stack: [14, 14, 0]
    SWAP -> Stack: [14, 0, 14]
    MSIZE -> Stack: [14, 0, 14, 0]
    10 -> Stack: [14, 0, 14, 0, 10]
    CALLDATACOPY -> Stack: [14, 0] Memory: [ … ]
    RETURN

    Notice how the first half of the code cleverly set up the stack so that it would push the inner code into memory indices 0…13, and then immediately return that chunk of memory. In the final compiled code,600e515b525b600a37f26002600035025b525b54602052f2, the inner code sits nicely to the right of the initializer code that simply returns it. In more complex contracts, initializers can also serve functions like setting certain storage slots to values, or even calling or creating other contracts.

    Now, let us introduce the latest and most fun feature of Serpent: imports. One common use case in contract land is that you want to give a contract the ability to spawn off new contracts. Problem is, how to you put the code for the spawned contracts into the spawner contracts? Before, the only solution was the uncomfortable approach of compiling the newer contracts first, and then putting the compiled code into an array. Now, we have a better solution: import.

    Put the following into returnten.se:

    x = create(tx.gas – 100, 0, import(mul2.se))
    return(msg(x,0,tx.gas-100,[5],1))

    Now, put the following into mul2.se:

    return(msg.data[0]*2)

    Now, if you serpent compile returnten.se and run the contract, you notice that, voila, it returns ten. The reason why is obvious. The returnten.se contract creates an instance of the mul2.se contract, and then calls it with the value 5. mul2.se, as the name suggests, is a doubler, and so it returns 5*2 = 10. Note that import is not a function in the standard sense; x = import(‘123.se’) will fail, and import only works in the very specific context of create.

    Now, suppose you are creating a 1000-line monster contract and want to split it up into files. To do that, we use inset. Intoouter.se, put:

    if msg.data[0] == 1:
    inset(inner.se)

    And into inner.se, put:

    return(3)

    Running serpent compile outer.se gives you a nice piece of compiled code that returns 3 if the msg.data[0] argument is equal to one. And that’s all there is to it.

    Upcoming updates to Serpent include:

    • An improvement of this mechanism so it doesn’t load the inner code twice if you try to use import twice with the same filename
    • String literals
    • Space and code-efficiency improvements for array literals
    • A debugging decorator (ie. a compiling function which tells you what lines of Serpent correspond to what bytes of compiled code)

    In the short term, though, my own effort will focus on bugfixes, a cross-client test suite, and continued work on ethereumjs-lib.



    Source link

    Share. Facebook Twitter Pinterest LinkedIn Tumblr Email
    Previous ArticleAdam Kinzinger slams Trump’s crypto ventures as corruption risk
    Next Article Denmark’s ‘historic’ crypto tax change is far from a done deal
    Olivia Martinez

    Related Posts

    Ethereum (ETH) sees major uptick as Pectra upgrade goes live

    May 8, 2025

    Allocation Update – Q1 2025

    May 8, 2025

    CVE-2025-30147 – The curious case of subgroup check on Besu

    May 7, 2025
    Leave A Reply Cancel Reply

    Demo
    Latest Posts

    XRP hits two-month highs as breakout points to $3 target

    May 13, 20250 Views

    Coinbase becomes first crypto company to join the S&P 500

    May 13, 20250 Views

    Coinbase stock jumps 8% as S&P 500 inclusion promises broader exposure

    May 13, 20250 Views

    Top 5 Crypto to Buy in October 2021

    May 13, 20250 Views
    Don't Miss

    TAO and Arbitrum investors rush to buy into 1Fuel presale before anticipated rally

    By Isabella TaylorApril 22, 2025

    For years, Arbitrum and TAO have dominated the market as two of the best altcoins…

    Slasher Ghost, and Other Developments in Proof of Stake

    April 24, 2025

    Bitcoin breaks $93,000 amid rumors that Saudi, UAE, or Qatar is buying

    April 23, 2025
    Stay In Touch
    • Facebook
    • Twitter
    • Pinterest
    • Instagram
    • YouTube
    • Vimeo

    Subscribe to Updates

    Get the latest creative news from SmartMag about art & design.

    Demo
    X (Twitter) Instagram YouTube LinkedIn
    Our Picks

    XRP hits two-month highs as breakout points to $3 target

    May 13, 2025

    Coinbase becomes first crypto company to join the S&P 500

    May 13, 2025

    Coinbase stock jumps 8% as S&P 500 inclusion promises broader exposure

    May 13, 2025
    Most Popular

    TAO and Arbitrum investors rush to buy into 1Fuel presale before anticipated rally

    April 22, 202528 Views

    Slasher Ghost, and Other Developments in Proof of Stake

    April 24, 202519 Views

    Bitcoin breaks $93,000 amid rumors that Saudi, UAE, or Qatar is buying

    April 23, 202510 Views
    © 2025 - 2026

    Type above and press Enter to search. Press Esc to cancel.