solana.rst 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  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. ``block.slot`` and ``tx.accounts``.
  38. - When calling an external function or invoking a contract's constructor, one
  39. :ref:`needs to provide <solana_cpi_accounts>` the necessary accounts for the transaction.
  40. - The keyword ``this`` returns the contract's program account, also know as program id.
  41. - Contracts :ref:`cannot be types <contracts_not_types>` on Solana and :ref:`calls to contracts <solana_contract_call>`
  42. follow a different syntax.
  43. - Accounts can be declared on functions using :ref:`annotations <account_management>`.
  44. Compute budget
  45. ++++++++++++++
  46. On Ethereum, when calling a smart contract function, one needs to specify the amount of gas the operation is allowed
  47. to use. Gas serves to pay for a contract execution on chain and can be a way for giving a contract priority execution
  48. when extra gas is offered in a transaction. Each EVM instruction has an associated gas value, which translates to real
  49. ETH cost. Provided that one can afford all the gas expenses, there is no upper boundary for the amount of gas limit
  50. one can provide in a transaction, so Solidity for Ethereum has gas builtins, like ``gasleft``, ``block.gaslimit``,
  51. ``tx.gasprice`` or the Yul ``gas()`` builtin, which returns the amount of gas left for execution.
  52. On the other hand, Solana is optimized for low latency and high transaction throughput and has an equivalent concept to
  53. gas: compute unit. Every smart contract function is allowed the same quantity of compute units (currently that
  54. value is 200k), and every instruction of a contract consumes exactly one compute unit. There is no need to provide
  55. an amount of compute units for a transaction and they are not charged, except when one wants priority execution on
  56. chain, in which case one would pay per compute unit consumed. Therefore, functions for gas are not available on
  57. Solidity for Solana.
  58. Solidity for Solana incompatibilities with Solidity for Ethereum
  59. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  60. - ``msg.sender`` is :ref:`not available on Solana <msg_sender_solana>`.
  61. - There is no ``ecrecover()`` builtin function because Solana does not use the ECDSA algorithm, but there
  62. is a ``signatureVerify()`` function, which can check ed25519 signatures. As a consequence, it is not possible to
  63. recover a signer from a signature.
  64. - Try-catch statements do not work on Solana. If any external call or contract creation fails, the runtime will
  65. halt execution and revert the entire transaction.
  66. - Error definitions and reverts with error messages are not yet working for Solana.
  67. - Value transfer with function call :ref:`does not work <value_transfer>`.
  68. - Many Yul builtins are not available, as specified in the :ref:`availability table <yul_builtins>`.
  69. - External calls on Solana require that accounts be specified, as in :ref:`this example <solana_external_call>`.
  70. - The ERC-20 interface is not compatible with Solana at the moment.
  71. Getting started on Solana
  72. _________________________
  73. Please follow the `Solang Getting Started Guide <https://solana.com/developers/guides/solang/getting-started>`_.
  74. For more examples, see the
  75. `solang's integration tests <https://github.com/hyperledger/solang/tree/main/integration/solana>`_.
  76. Using the Anchor client library
  77. _______________________________
  78. Some notes on using the anchor javascript npm library.
  79. * Solidity function names are converted to camelCase. This means that if in Solidity a function is called ``foo_bar()``,
  80. you must write ``fooBar()`` in your javascript.
  81. * Anchor only allows you to call ``.view()`` on Solidity functions which are declared ``view`` or ``pure``.
  82. * Named return values in Solidity are also converted to camelCase. Unnamed returned are given the name ``return0``, ``return1``, etc,
  83. depending on the position in the returns values.
  84. * Only return values from ``view`` and ``pure`` functions can be decoded. Return values from other functions and are not accessible.
  85. This is a limitation in the Anchor library. Possibly this can be fixed.
  86. * In the case of an error, no return data is decoded. This means that the reason provided in ``revert('reason');`` is not
  87. available as a return value.
  88. * Number arguments for functions are expressed as ``BN`` values and not plain javascript ``Number`` or ``BigInt``.
  89. .. _call_anchor:
  90. Calling Anchor Programs from Solidity
  91. _____________________________________
  92. It is possible to call `Anchor Programs <https://github.com/coral-xyz/anchor>`_
  93. from Solidity. You first have to generate a Solidity interface file from the IDL file using
  94. the :ref:`idl_command`. Then, import the Solidity file in your Solidity using the
  95. ``import "...";`` syntax. Say you have an anchor program called ``bobcat`` with a
  96. function ``pounce``, you can call it like so:
  97. .. include:: ../examples/solana/call_anchor.sol
  98. :code: solidity
  99. Setting the program_id for a contract
  100. _____________________________________
  101. When developing contracts for Solana, programs are usually deployed to a well
  102. known account. The account can be specified in the source code using an annotation
  103. ``@program_id`` if it is known beforehand. If you want to call a contract via an external call,
  104. either the contract must have a ``@program_id`` annotation or the ``{program_id: ..}`` call argument must be present.
  105. .. include:: ../examples/solana/program_id.sol
  106. :code: solidity
  107. .. note::
  108. The program_id ``Foo5mMfYo5RhRcWa4NZ2bwFn4Kdhe8rNK5jchxsKrivA`` was generated using
  109. the command line:
  110. .. code-block:: bash
  111. solana-keygen grind --starts-with Foo:1
  112. .. _payer_seeds_bump:
  113. Setting the payer, seeds, bump, and space for a contract
  114. _________________________________________________________
  115. When a contract is instantiated, there are two accounts required: the program account to hold
  116. the executable code and the data account to save the state variables of the contract. The
  117. program account is deployed once and can be reused for updating the contract. When each
  118. Solidity contract is instantiated (also known as deployed), the data account has to be
  119. created. This can be done by the client-side code, and then the created blank account
  120. is passed to the transaction that runs the constructor code.
  121. Alternatively, the data account can be created by the constructor, on chain. When
  122. this method is used, some parameters must be specified for the account
  123. using annotations. Annotations placed above a constructor can only contain literals or
  124. constant expressions, as is the case for first ``@seed`` and ``@space`` in the following example.
  125. Annotations can also refer to constructor arguments when placed next to them, as the second ``@seed`` and
  126. the ``@bump`` examples below. The ``@payer`` annotation is a special annotation that
  127. :ref:`declares an account <account_management>`.
  128. If the contract has no constructor, annotations can be paired with an empty constructor.
  129. .. include:: ../examples/solana/constructor_annotations.sol
  130. :code: solidity
  131. Creating an account needs a payer, so at a minimum the ``@payer`` annotation must be
  132. specified. If it is missing, then the data account must be created client-side.
  133. The ``@payer`` annotation :ref:`declares a Solana account <account_management>` that must be passed in the transaction.
  134. The size of the data account can be specified with ``@space``. This is a
  135. ``uint64`` expression which can either be a constant or use one of the constructor
  136. arguments. The ``@space`` should at least be the size given when you run ``solang -v``:
  137. .. code-block:: bash
  138. $ solang compile --target solana -v examples/solana/flipper.sol
  139. ...
  140. info: contract flipper uses at least 17 bytes account data
  141. ...
  142. If the data account is going to be a
  143. `program derived address <https://docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses>`_,
  144. then the seeds and bump have to be provided. There can be multiple seeds, and an optional
  145. single bump. If the bump is not provided, then the seeds must not create an
  146. account that falls on the curve. When placed above the constructor, the ``@seed`` can be a string literal,
  147. or a hex string with the format ``hex"4142"``. If before an argument, the seed annotation must refer to an argument
  148. of type ``bytes``, ``address``, or fixed length byte array of ``bytesN``. The ``@bump`` must a single byte of type ``bytes1``.
  149. .. _value_transfer:
  150. Transferring native value with a function call
  151. ______________________________________________
  152. The Solidity language on Ethereum allows value transfers with an external call
  153. or constructor, using the ``auction.bid{value: 501}()`` syntax.
  154. Solana Cross Program Invocation (CPI) does not support this, which means that:
  155. - Specifying ``value:`` on an external call or constructor is not permitted
  156. - The ``payable`` keyword has no effect
  157. - ``msg.value`` is not supported
  158. .. note::
  159. A naive way to implement this is to let the caller transfer
  160. native balance and then inform the callee about the amount transferred by
  161. specifying this in the instruction data. However, it would be trivial to
  162. forge such an operation.
  163. Receive function
  164. ________________
  165. In Solidity the ``receive()`` function, when defined, is called whenever the native
  166. balance for an account gets credited, for example through a contract calling
  167. ``account.transfer(value);``. On Solana, there is no method that implements
  168. this. The balance of an account can be credited without any code being executed.
  169. ``receive()`` functions are not permitted on the Solana target.
  170. .. _msg_sender_solana:
  171. ``msg.sender`` not available on Solana
  172. ______________________________________
  173. On Ethereum, ``msg.sender`` is used to identify either the account that submitted
  174. the transaction, or the caller when one contract calls another. On Ethereum, each
  175. contract execution can only use a single account, which provides the code and data.
  176. On Solana, each contract execution uses many accounts. Consider a rust contract which
  177. calls a Solidity contract: the rust contract can access a few data accounts, and which
  178. of those would be considered the caller? So in many cases there is not a single account
  179. which can be identified as a caller. In addition to that, the Solana VM has no
  180. mechanism for fetching the caller accounts. This means there is no way to implement
  181. ``msg.sender``.
  182. The way to implement this on Solana is to have an authority account for the contract
  183. that must be a signer for the transaction (note that on Solana there
  184. can be many signers too). This is a common construct on Solana contracts.
  185. .. include:: ../examples/solana/use_authority.sol
  186. :code: solidity
  187. Builtin Imports
  188. ________________
  189. Some builtin functionality is only available after importing. The following structs
  190. can be imported via the special builtin import file ``solana``.
  191. .. code-block:: solidity
  192. import {AccountMeta, AccountInfo} from 'solana';
  193. Note that ``{AccountMeta, AccountInfo}`` can be omitted, renamed or imported via
  194. import object.
  195. .. code-block:: solidity
  196. // Now AccountMeta will be known as AM
  197. import {AccountMeta as AM} from 'solana';
  198. // Now AccountMeta will be available as solana.AccountMeta
  199. import 'solana' as solana;
  200. .. note::
  201. The import file ``solana`` is only available when compiling for the Solana
  202. target.
  203. .. _account_info:
  204. Builtin AccountInfo
  205. +++++++++++++++++++
  206. The account info of all the accounts passed into the transaction. ``AccountInfo`` is a builtin
  207. structure with the following fields:
  208. address ``key``
  209. The address (or public key) of the account
  210. uint64 ``lamports``
  211. The lamports of the accounts. This field can be modified, however the lamports need to be
  212. balanced for all accounts by the end of the transaction.
  213. bytes ``data``
  214. The account data. This field can be modified, but use with caution.
  215. address ``owner``
  216. The program that owns this account
  217. uint64 ``rent_epoch``
  218. The next epoch when rent is due.
  219. bool ``is_signer``
  220. Did this account sign the transaction
  221. bool ``is_writable``
  222. Is this account writable in this transaction
  223. bool ``executable``
  224. Is this account a program
  225. .. _account_meta:
  226. Builtin AccountMeta
  227. +++++++++++++++++++
  228. When doing an external call (aka CPI), ``AccountMeta`` specifies which accounts
  229. should be passed to the callee.
  230. address ``pubkey``
  231. The address (or public key) of the account
  232. bool ``is_writable``
  233. Can the callee write to this account
  234. bool ``is_signer``
  235. Can the callee assume this account signed the transaction
  236. Builtin create_program_address
  237. ++++++++++++++++++++++++++++++
  238. This function returns the program derived address for a program address and
  239. the provided seeds. See the Solana documentation on
  240. `program derived addresses <https://edge.docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses>`_.
  241. .. include:: ../examples/solana/builtin_create_program_address.sol
  242. :code: solidity
  243. Builtin try_find_program_address
  244. ++++++++++++++++++++++++++++++++
  245. This function returns the program derived address for a program address and
  246. the provided seeds, along with a seed bump. See the Solana documentation on
  247. `program derived addresses <https://edge.docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses>`_.
  248. .. include:: ../examples/solana/builtin_try_find_program_address.sol
  249. :code: solidity
  250. Solana Library
  251. ______________
  252. In Solang's Github repository, there is a directory called ``solana-library``. It contains libraries for Solidity contracts
  253. to interact with Solana specific instructions. We provide two libraries: one for SPL tokens and another
  254. for Solana's system instructions. In order to use those functionalities, copy the correspondent library
  255. file to your project and import it.
  256. SPL-token
  257. +++++++++
  258. `spl-token <https://spl.solana.com/token>`_ is the Solana native way of creating tokens, minting, burning and
  259. transferring token. This is the Solana equivalent of
  260. `ERC-20 <https://ethereum.org/en/developers/docs/standards/tokens/erc-20/>`_ and
  261. `ERC-721 <https://ethereum.org/en/developers/docs/standards/tokens/erc-721/>`_. Solang's repository contains
  262. a library ``SplToken`` to use spl-token from Solidity. The file
  263. `spl_token.sol <https://github.com/hyperledger/solang/blob/main/solana-library/spl_token.sol>`_ should be copied into
  264. your source tree, and then imported in your solidity files where it is required. The ``SplToken`` library has doc
  265. comments explaining how it should be used.
  266. There is an example in our integration tests of how this should be used. See
  267. `token.sol <https://github.com/hyperledger/solang/blob/main/integration/solana/token.sol>`_ and
  268. `token.spec.ts <https://github.com/hyperledger/solang/blob/main/integration/solana/token.spec.ts>`_.
  269. .. _system_instruction_library:
  270. System Instructions
  271. +++++++++++++++++++
  272. Solana's system instructions enable developers to interact with Solana's System Program. There are functions to
  273. create new accounts, allocate account data, assign accounts to owning programs, transfer lamports from System Program
  274. owned accounts and pay transaction fees. More information about the functions offered can be found both on
  275. `Solana documentation <https://docs.rs/solana-program/1.11.10/solana_program/system_instruction/enum.SystemInstruction.html>`_
  276. and on Solang's `system_instruction.sol <https://github.com/hyperledger/solang/blob/main/solana-library/system_instruction.sol>`_ file.
  277. The usage of system instructions needs the correct setting of writable and signer accounts when interacting with Solidity
  278. contracts on chain. Examples are available on Solang's integration tests.
  279. See `system_instruction_example.sol <https://github.com/hyperledger/solang/blob/main/integration/solana/system_instruction_example.sol>`_
  280. and `system_instruction.spec.ts <https://github.com/hyperledger/solang/blob/main/integration/solana/system_instruction.spec.ts>`_
  281. Minimum balance
  282. +++++++++++++++
  283. In order to instantiate a contract, you need the minimum balance required for a Solana account of a given
  284. size. There is a function ``minimum_balance(uint64 space)`` defined in
  285. `minimum_balance.sol <https://github.com/hyperledger/solang/blob/main/solana-library/minimum_balance.sol>`_
  286. to calculate this.
  287. .. _account_management:
  288. Solana Account Management
  289. _________________________
  290. In a contract constructor, one can optionally write the ``@payer`` annotation, which receives a character sequence as
  291. an argument. This annotation defines a Solana account that is going to pay for the initialization of the contract's data
  292. account. The syntax ``@payer(my_account)`` declares an account named ``my_account``, which will be
  293. required for every call to the constructor.
  294. Similarly, for external functions in a contract, one can declare the necessary accounts, using function annotations.
  295. ``@account(myAcc)`` declares a read only account ``myAcc``, while ``@mutableAccount(otherAcc)`` declares a mutable
  296. account ``otherAcc``. For signer accounts, the annotations follow the syntax ``@signer(mySigner)`` and
  297. ``@mutableSigner(myOtherSigner)``.
  298. Accessing accounts' data
  299. ++++++++++++++++++++++++
  300. Accounts declared on a constructor using the ``@payer`` annotation are available for access inside it. Likewise,
  301. accounts declared on external functions with any of the aforementioned annotations are also available in the
  302. ``tx.accounts`` vector for easy access. For an account declared as ``@account(funder)``, the access follows the
  303. syntax ``tx.accounts.funder``, which returns the :ref:`AccountInfo builtin struct <account_info>`.
  304. .. include:: ../examples/solana/account_access.sol
  305. :code: solidity
  306. .. _solana_cpi_accounts:
  307. External calls with accounts
  308. ++++++++++++++++++++++++++++
  309. In any Solana cross program invocation, including constructor calls, all the accounts a transaction needs must be
  310. informed. If the ``{accounts: ...}`` call argument is missing from an external call, the compiler will automatically
  311. generate the ``AccountMeta`` array that satisfies such a requirement. Currently, that only works if the call is done
  312. in an function declared external, as shown in the example below. In any other case, the ``AccountMeta`` array must
  313. be manually created, following the account ordering the IDL file specifies. If a certain call does not need any accounts,
  314. an empty vector must be passed ``{accounts: []}``.
  315. The following example shows two correct ways of calling a contract. Note that the IDL for the ``BeingBuilt`` contract
  316. has an instruction called ``new``, representing the contract's constructor, whose accounts are specified in the
  317. following order: ``dataAccount``, ``payer_account``, ``systemAccount``. That is the order one must follow when invoking
  318. such a constructor.
  319. .. include:: ../examples/solana/payer_annotation.sol
  320. :code: solidity