solana.rst 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. Solana
  2. ======
  3. Solana Overview
  4. _______________
  5. As the underlying Solana environment is different than that of Ethereum, Solidity inner workings
  6. have been modified to function properly. For example, A Solidity contract on Solana utilizes two accounts: a
  7. data account and a program account. The program account stores the contract's executable binary and owns
  8. the data account, which holds all the storage variables. On Ethereum a single account can store executable
  9. code and data.
  10. Contract upgrades
  11. +++++++++++++++++
  12. Provided that the data layout from a new contract is compatible with that of an old one, it is possible
  13. to update the binary in the program account and retain the same data, rendering
  14. contract upgrades implemented in Solidity unnecessary. Solana's CLI tool provides a command to both do
  15. an initial deploy of a program, and redeploy it later.:
  16. .. code-block:: bash
  17. solana program deploy --program-id <KEYPAIR_FILEPATH> <PROGRAM_FILEPATH>
  18. where ``<KEYPAIR_FILEPATH>`` is the program's keypair json file and ``<PROGRAM_FILEPATH>``
  19. is the program binary ``.so`` file. For more information about redeploying a program,
  20. check `Solana's documentation <https://docs.solana.com/cli/deploy-a-program#redeploy-a-program>`_.
  21. Data types
  22. ++++++++++
  23. - An account address consists of a 32-bytes key, which is represented by the ``address`` type. This data model
  24. differs from Ethereum 20-bytes addresses.
  25. - Solana's virtual machine registers are 64-bit wide, so 64-bit integers ``uint64`` and ``int64`` are preferable over
  26. ``uint256`` and ``int256``. An operation with types wider than 64-bits is split into multiple operations, making
  27. it slower and consuming more compute units. This is the case, for instance, with multiplication, division and modulo
  28. using `uint256`.
  29. - Likewise, all balances and values on Solana are 64-bit wide, so the builtin functions for
  30. *address* ``.balance``, ``.transfer()`` and ``.send()`` use 64-bit integers.
  31. - An address literal has to be specified using the ``address"36VtvSbE6jVGGQytYWSaDPG7uZphaxEjpJHUUpuUbq4D"`` syntax.
  32. - Ethereum syntax for addresses ``0xE0f5206BBD039e7b0592d8918820024e2a7437b9`` is not supported.
  33. Runtime
  34. +++++++
  35. - The Solana target requires `Solana <https://www.solana.com/>`_ v1.8.1.
  36. - Function selectors are eight bytes wide and known as *discriminators*.
  37. - Solana provides different builtins, e.g. ``tx.program_id`` and ``tx.accounts``.
  38. Compute budget
  39. ++++++++++++++
  40. On Ethereum, when calling a smart contract function, one needs to specify the amount of gas the operation is allowed
  41. to use. Gas serves to pay for a contract execution on chain and can be a way for giving a contract priority execution
  42. when extra gas is offered in a transaction. Each EVM instruction has an associated gas value, which translates to real
  43. ETH cost. Provided that one can afford all the gas expenses, there is no upper boundary for the amount of gas limit
  44. one can provide in a transaction, so Solidity for Ethereum has gas builtins, like ``gasleft``, ``block.gaslimit``,
  45. ``tx.gasprice`` or the Yul ``gas()`` builtin, which returns the amount of gas left for execution.
  46. On the other hand, Solana is optimized for low latency and high transaction throughput and has an equivalent concept to
  47. gas: compute unit. Every smart contract function is allowed the same quantity of compute units (currently that
  48. value is 200k), and every instruction of a contract consumes exactly one compute unit. There is no need to provide
  49. an amount of compute units for a transaction and they are not charged, except when one wants priority execution on
  50. chain, in which case one would pay per compute unit consumed. Therefore, functions for gas are not available on
  51. Solidity for Solana.
  52. Solidity for Solana incompatibilities with Solidity for Ethereum
  53. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  54. - ``msg.sender`` is :ref:`not available on Solana <msg_sender_solana>`.
  55. - There is no ``ecrecover()`` builtin function because Solana does not use the ECDSA algorithm, but there
  56. is a ``signatureVerify()`` function, which can check ed25519 signatures. As a consequence, it is not possible to
  57. recover a signer from a signature.
  58. - Try-catch statements do not work on Solana. If any external call or contract creation fails, the runtime will
  59. halt execution and revert the entire transaction.
  60. - Error definitions and reverts with error messages are not yet working for Solana.
  61. - Value transfer with function call :ref:`does not work <value_transfer>`.
  62. - Many Yul builtins are not available, as specified in the :ref:`availability table <yul_builtins>`.
  63. - External calls on Solana require that accounts be specified, as in :ref:`this example <solana_external_call>`.
  64. - The ERC-20 interface is not compatible with Solana at the moment.
  65. Build your Solidity for Solana
  66. ______________________________
  67. .. code-block:: bash
  68. solang compile --target solana flipper.sol -v
  69. This will produce two files called ``flipper.json`` and ``flipper.so``. The former is an Anchor style IDL file and the latter is
  70. the ELF BPF shared object containing the program. For each contract in the source code, Solang will create both an IDL file
  71. and a binary file.
  72. Each program will need to be deployed to a program_id. Usually, the program_id is a well-known account which is specified
  73. in the Solidity source code using the ``@program_id("F1ipperKF9EfD821ZbbYjS319LXYiBmjhzkkf5a26rC")`` annotation on the contract.
  74. A private key for the account is needed to deploy. You can generate your own private key using the command line tool
  75. ``solana-keygen``.
  76. .. code-block:: bash
  77. echo "[4,10,246,143,43,1,234,17,159,249,41,16,230,9,198,162,107,221,233,124,34,15,16,57,205,53,237,217,149,17,229,195,3,150,242,90,91,222,117,26,196,224,214,105,82,62,237,137,92,67,213,23,14,206,230,155,43,36,85,254,247,11,226,145]" > flipper-keypair.json
  78. solana program deploy flipper.so
  79. After deploying the program, you can start on the client side, which needs the anchor npm library:
  80. .. code-block:: bash
  81. npm install @project-serum/anchor
  82. Write the following javascript to a file called ``flipper.js``.
  83. .. code-block:: javascript
  84. const { readFileSync } = require('fs');
  85. const anchor = require('@project-serum/anchor');
  86. const IDL = JSON.parse(readFileSync('./flipper.json', 'utf8'));
  87. const PROGRAM_SO = readFileSync('./flipper.so');
  88. (async function () {
  89. const provider = anchor.AnchorProvider.env();
  90. const dataAccount = anchor.web3.Keypair.generate();
  91. const programId = new anchor.web3.PublicKey(IDL.metadata.address);
  92. const wallet = provider.wallet.publicKey;
  93. const program = new anchor.Program(IDL, programId, provider);
  94. await program.methods.new(wallet, true)
  95. .accounts({ dataAccount: dataAccount.publicKey })
  96. .signers([dataAccount]).rpc();
  97. const val1 = await program.methods.get()
  98. .accounts({ dataAccount: dataAccount.publicKey })
  99. .view();
  100. console.log(`state: ${val1}`);
  101. await program.methods.flip()
  102. .accounts({ dataAccount: dataAccount.publicKey })
  103. .rpc();
  104. const val2 = await program.methods.get()
  105. .accounts({ dataAccount: dataAccount.publicKey })
  106. .view();
  107. console.log(`state: ${val2}`);
  108. })();
  109. Now you'll have to set the ``ANCHOR_WALLET`` and ``ANCHOR_PROVIDER_URL`` environment variables to the correct values in order to run the example.
  110. .. code-block:: bash
  111. export ANCHOR_WALLET=$HOME/.config/solana/id.json
  112. export ANCHOR_PROVIDER_URL=http://127.0.0.1:8899
  113. node flipper.js
  114. For more examples, see the
  115. `solang's integration tests <https://github.com/hyperledger/solang/tree/main/integration/solana>`_.
  116. Using the Anchor client library
  117. _______________________________
  118. Some notes on using the anchor javascript npm library.
  119. * Solidity function names are converted to camelCase. This means that if in Solidity a function is called ``foo_bar()``,
  120. you must write ``fooBar()`` in your javascript.
  121. * Anchor only allows you to call ``.view()`` on Solidity functions which are declared ``view`` or ``pure``.
  122. * Named return values in Solidity are also converted to camelCase. Unnamed returned are given the name ``return0``, ``return1``, etc,
  123. depending on the position in the returns values.
  124. * Only return values from ``view`` and ``pure`` functions can be decoded. Return values from other functions and are not accessible.
  125. This is a limitation in the Anchor library. Possibly this can be fixed.
  126. * In the case of an error, no return data is decoded. This means that the reason provided in ``revert('reason');`` is not
  127. available as a return value.
  128. * Number arguments for functions are expressed as ``BN`` values and not plain javascript ``Number`` or ``BigInt``.
  129. .. _call_anchor:
  130. Calling Anchor Programs from Solidity
  131. _____________________________________
  132. It is possible to call `Anchor Programs <https://github.com/coral-xyz/anchor>`_
  133. from Solidity. You first have to generate a Solidity interface file from the IDL file using
  134. the :ref:`idl_command`. Then, import the Solidity file in your Solidity using the
  135. ``import "...";`` syntax. Say you have an anchor program called ``bobcat`` with a
  136. function ``pounce``, you can call it like so:
  137. .. include:: ../examples/solana/call_anchor.sol
  138. :code: solidity
  139. Setting the program_id for a contract
  140. _____________________________________
  141. When developing contracts for Solana, programs are usually deployed to a well
  142. known account. The account can be specified in the source code using an annotation
  143. ``@program_id``. If you want to instantiate a contract using the
  144. ``new ContractName()`` syntax, then the contract must have a program_id annotation.
  145. .. include:: ../examples/solana/program_id.sol
  146. :code: solidity
  147. .. note::
  148. The program_id ``Foo5mMfYo5RhRcWa4NZ2bwFn4Kdhe8rNK5jchxsKrivA`` was generated using
  149. the command line:
  150. .. code-block:: bash
  151. solana-keygen grind --starts-with Foo:1
  152. Setting the payer, seeds, bump, and space for a contract
  153. _________________________________________________________
  154. When a contract is instantiated, there are two accounts required: the program account to hold
  155. the executable code and the data account to save the state variables of the contract. The
  156. program account is deployed once and can be reused for updating the contract. When each
  157. Solidity contract is instantiated (also known as deployed), the data account has to be
  158. created. This can be done by the client-side code, and then the created blank account
  159. is passed to the transaction that runs the constructor code.
  160. Alternatively, the data account can be created by the constructor, on chain. When
  161. this method is used, some parameters must be specified for the account
  162. using annotations. Those are placed before the constructor. If there is no
  163. constructor present, then an empty constructor can be added. The constructor
  164. arguments can be used in the annotations.
  165. .. include:: ../examples/solana/constructor_annotations.sol
  166. :code: solidity
  167. Creating an account needs a payer, so at a minimum the ``@payer`` annotation must be
  168. specified. If it is missing, then the data account must be created client-side.
  169. The ``@payer`` requires an address. This can be a constructor argument or
  170. an address literal.
  171. The size of the data account can be specified with ``@space``. This is a
  172. ``uint64`` expression which can either be a constant or use one of the constructor
  173. arguments. The ``@space`` should at least be the size given when you run ``solang -v``:
  174. .. code-block:: bash
  175. $ solang compile --target solana -v examples/solana/flipper.sol
  176. ...
  177. info: contract flipper uses at least 17 bytes account data
  178. ...
  179. If the data account is going to be a
  180. `program derived address <https://docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses>`_,
  181. then the seeds and bump have to be provided. There can be multiple seeds, and an optional
  182. single bump. If the bump is not provided, then the seeds must not create an
  183. account that falls on the curve. The ``@seed`` can be a string literal,
  184. or a hex string with the format ``hex"4142"``, or a constructor argument of type
  185. ``bytes``. The ``@bump`` must a single byte of type ``bytes1``.
  186. .. _value_transfer:
  187. Transferring native value with a function call
  188. ______________________________________________
  189. The Solidity language on Ethereum allows value transfers with an external call
  190. or constructor, using the ``auction.bid{value: 501}()`` syntax.
  191. Solana Cross Program Invocation (CPI) does not support this, which means that:
  192. - Specifying ``value:`` on an external call or constructor is not permitted
  193. - The ``payable`` keyword has no effect
  194. - ``msg.value`` is not supported
  195. .. note::
  196. A naive way to implement this is to let the caller transfer
  197. native balance and then inform the callee about the amount transferred by
  198. specifying this in the instruction data. However, it would be trivial to
  199. forge such an operation.
  200. Receive function
  201. ________________
  202. In Solidity the ``receive()`` function, when defined, is called whenever the native
  203. balance for an account gets credited, for example through a contract calling
  204. ``account.transfer(value);``. On Solana, there is no method that implements
  205. this. The balance of an account can be credited without any code being executed.
  206. ``receive()`` functions are not permitted on the Solana target.
  207. .. _msg_sender_solana:
  208. ``msg.sender`` not available on Solana
  209. ______________________________________
  210. On Ethereum, ``msg.sender`` is used to identify either the account that submitted
  211. the transaction, or the caller when one contract calls another. On Ethereum, each
  212. contract execution can only use a single account, which provides the code and data.
  213. On Solana, each contract execution uses many accounts. Consider a rust contract which
  214. calls a Solidity contract: the rust contract can access a few data accounts, and which
  215. of those would be considered the caller? So in many cases there is not a single account
  216. which can be identified as a caller. In addition to that, the Solana VM has no
  217. mechanism for fetching the caller accounts. This means there is no way to implement
  218. ``msg.sender``.
  219. The way to implement this on Solana is to have an authority account for the contract
  220. that must be a signer for the transaction (note that on Solana there
  221. can be many signers too). This is a common construct on Solana contracts.
  222. .. include:: ../examples/solana/use_authority.sol
  223. :code: solidity
  224. Builtin Imports
  225. ________________
  226. Some builtin functionality is only available after importing. The following structs
  227. can be imported via the special builtin import file ``solana``.
  228. .. code-block:: solidity
  229. import {AccountMeta, AccountInfo} from 'solana';
  230. Note that ``{AccountMeta, AccountInfo}`` can be omitted, renamed or imported via
  231. import object.
  232. .. code-block:: solidity
  233. // Now AccountMeta will be known as AM
  234. import {AccountMeta as AM} from 'solana';
  235. // Now AccountMeta will be available as solana.AccountMeta
  236. import 'solana' as solana;
  237. .. note::
  238. The import file ``solana`` is only available when compiling for the Solana
  239. target.
  240. .. _account_info:
  241. Builtin AccountInfo
  242. +++++++++++++++++++
  243. The account info of all the accounts passed into the transaction. ``AccountInfo`` is a builtin
  244. structure with the following fields:
  245. address ``key``
  246. The address (or public key) of the account
  247. uint64 ``lamports``
  248. The lamports of the accounts. This field can be modified, however the lamports need to be
  249. balanced for all accounts by the end of the transaction.
  250. bytes ``data```
  251. The account data. This field can be modified, but use with caution.
  252. address ``owner``
  253. The program that owns this account
  254. uint64 ``rent_epoch``
  255. The next epoch when rent is due.
  256. bool ``is_signer``
  257. Did this account sign the transaction
  258. bool ``is_writable``
  259. Is this account writable in this transaction
  260. bool ``executable``
  261. Is this account a program
  262. .. _account_meta:
  263. Builtin AccountMeta
  264. +++++++++++++++++++
  265. When doing an external call (aka CPI), ``AccountMeta`` specifies which accounts
  266. should be passed to the callee.
  267. address ``pubkey``
  268. The address (or public key) of the account
  269. bool ``is_writable``
  270. Can the callee write to this account
  271. bool ``is_signer``
  272. Can the callee assume this account signed the transaction
  273. Builtin create_program_address
  274. ++++++++++++++++++++++++++++++
  275. This function returns the program derived address for a program address and
  276. the provided seeds. See the Solana documentation on
  277. `program derived addresses <https://edge.docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses>`_.
  278. .. include:: ../examples/solana/builtin_create_program_address.sol
  279. :code: solidity
  280. Builtin try_find_program_address
  281. ++++++++++++++++++++++++++++++++
  282. This function returns the program derived address for a program address and
  283. the provided seeds, along with a seed bump. See the Solana documentation on
  284. `program derived addresses <https://edge.docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses>`_.
  285. .. include:: ../examples/solana/builtin_try_find_program_address.sol
  286. :code: solidity
  287. Solana Library
  288. ______________
  289. In Solang's Github repository, there is a directory called ``solana-library``. It contains libraries for Solidity contracts
  290. to interact with Solana specific instructions. We provide two libraries: one for SPL tokens and another
  291. for Solana's system instructions. In order to use those functionalities, copy the correspondent library
  292. file to your project and import it.
  293. SPL-token
  294. +++++++++
  295. `spl-token <https://spl.solana.com/token>`_ is the Solana native way of creating tokens, minting, burning and
  296. transferring token. This is the Solana equivalent of
  297. `ERC-20 <https://ethereum.org/en/developers/docs/standards/tokens/erc-20/>`_ and
  298. `ERC-721 <https://ethereum.org/en/developers/docs/standards/tokens/erc-721/>`_. Solang's repository contains
  299. a library ``SplToken`` to use spl-token from Solidity. The file
  300. `spl_token.sol <https://github.com/hyperledger/solang/blob/main/solana-library/spl_token.sol>`_ should be copied into
  301. your source tree, and then imported in your solidity files where it is required. The ``SplToken`` library has doc
  302. comments explaining how it should be used.
  303. There is an example in our integration tests of how this should be used. See
  304. `token.sol <https://github.com/hyperledger/solang/blob/main/integration/solana/token.sol>`_ and
  305. `token.spec.ts <https://github.com/hyperledger/solang/blob/main/integration/solana/token.spec.ts>`_.
  306. .. _system_instruction_library:
  307. System Instructions
  308. +++++++++++++++++++
  309. Solana's system instructions enable developers to interact with Solana's System Program. There are functions to
  310. create new accounts, allocate account data, assign accounts to owning programs, transfer lamports from System Program
  311. owned accounts and pay transaction fees. More information about the functions offered can be found both on
  312. `Solana documentation <https://docs.rs/solana-program/1.11.10/solana_program/system_instruction/enum.SystemInstruction.html>`_
  313. and on Solang's `system_instruction.sol <https://github.com/hyperledger/solang/blob/main/solana-library/system_instruction.sol>`_ file.
  314. The usage of system instructions needs the correct setting of writable and signer accounts when interacting with Solidity
  315. contracts on chain. Examples are available on Solang's integration tests.
  316. See `system_instruction_example.sol <https://github.com/hyperledger/solang/blob/main/integration/solana/system_instruction_example.sol>`_
  317. and `system_instruction.spec.ts <https://github.com/hyperledger/solang/blob/main/integration/solana/system_instruction.spec.ts>`_