miraland_program/
lib.rs

1//! The base library for all Miraland on-chain Rust programs.
2//!
3//! All Miraland Rust programs that run on-chain will link to this crate, which
4//! acts as a standard library for Miraland programs. Miraland programs also link to
5//! the [Rust standard library][std], though it is [modified][sstd] for the
6//! Miraland runtime environment. While off-chain programs that interact with the
7//! Miraland network _can_ link to this crate, they typically instead use the
8//! [`miraland-sdk`] crate, which reexports all modules from `miraland-program`.
9//!
10//! [std]: https://doc.rust-lang.org/stable/std/
11//! [sstd]: https://docs.solana.com/developing/on-chain-programs/developing-rust#restrictions
12//! [`miraland-sdk`]: https://docs.rs/miraland-sdk/latest/miraland_sdk/
13//!
14//! This library defines
15//!
16//! - macros for declaring the [program entrypoint][pe],
17//! - [core data types][cdt],
18//! - [logging] macros,
19//! - [serialization] methods,
20//! - methods for [cross-program instruction execution][cpi],
21//! - program IDs and instruction constructors for the system program and other
22//!   [native programs][np],
23//! - [sysvar] accessors.
24//!
25//! [pe]: #defining-a-miraland-program
26//! [cdt]: #core-data-types
27//! [logging]: crate::log
28//! [serialization]: #serialization
29//! [np]: #native-programs
30//! [cpi]: #cross-program-instruction-execution
31//! [sysvar]: crate::sysvar
32//!
33//! Idiomatic examples of `miraland-program` usage can be found in
34//! [the Miraland Program Library][spl].
35//!
36//! [spl]: https://github.com/solana-labs/solarti-program-library
37//!
38//! # Defining a solana program
39//!
40//! Miraland program crates have some unique properties compared to typical Rust
41//! programs:
42//!
43//! - They are often compiled for both on-chain use and off-chain use. This is
44//!   primarily because off-chain clients may need access to data types
45//!   defined by the on-chain program.
46//! - They do not define a `main` function, but instead define their entrypoint
47//!   with the [`entrypoint!`] macro.
48//! - They are compiled as the ["cdylib"] crate type for dynamic loading
49//!   by the Miraland runtime.
50//! - They run in a constrained VM environment, and while they do have access to
51//!   the [Rust standard library][std], many features of the standard library,
52//!   particularly related to OS services, will fail at runtime, will silently
53//!   do nothing, or are not defined. See the [restrictions to the Rust standard
54//!   library][sstd] in the Miraland documentation for more.
55//!
56//! [std]: https://doc.rust-lang.org/std/index.html
57//! ["cdylib"]: https://doc.rust-lang.org/reference/linkage.html
58//!
59//! Because multiple crates that are linked together cannot all define
60//! program entrypoints (see the [`entrypoint!`] documentation) a common
61//! convention is to use a [Cargo feature] called `no-entrypoint` to allow
62//! the program entrypoint to be disabled.
63//!
64//! [Cargo feature]: https://doc.rust-lang.org/cargo/reference/features.html
65//!
66//! The skeleton of a Miraland program typically looks like:
67//!
68//! ```
69//! #[cfg(not(feature = "no-entrypoint"))]
70//! pub mod entrypoint {
71//!     use miraland_program::{
72//!         account_info::AccountInfo,
73//!         entrypoint,
74//!         entrypoint::ProgramResult,
75//!         pubkey::Pubkey,
76//!     };
77//!
78//!     entrypoint!(process_instruction);
79//!
80//!     pub fn process_instruction(
81//!         program_id: &Pubkey,
82//!         accounts: &[AccountInfo],
83//!         instruction_data: &[u8],
84//!     ) -> ProgramResult {
85//!         // Decode and dispatch instructions here.
86//!         todo!()
87//!     }
88//! }
89//!
90//! // Additional code goes here.
91//! ```
92//!
93//! With a `Cargo.toml` file that contains
94//!
95//! ```toml
96//! [lib]
97//! crate-type = ["cdylib", "rlib"]
98//!
99//! [features]
100//! no-entrypoint = []
101//! ```
102//!
103//! Note that a Miraland program must specify its crate-type as "cdylib", and
104//! "cdylib" crates will automatically be discovered and built by the `cargo
105//! build-bpf` command. Miraland programs also often have crate-type "rlib" so
106//! they can be linked to other Rust crates.
107//!
108//! # On-chain vs. off-chain compilation targets
109//!
110//! Miraland programs run on the [rbpf] VM, which implements a variant of the
111//! [eBPF] instruction set. Because this crate can be compiled for both on-chain
112//! and off-chain execution, the environments of which are significantly
113//! different, it extensively uses [conditional compilation][cc] to tailor its
114//! implementation to the environment. The `cfg` predicate used for identifying
115//! compilation for on-chain programs is `target_os = "solana"`, as in this
116//! example from the `miraland-program` codebase that logs a message via a
117//! syscall when run on-chain, and via a library call when offchain:
118//!
119//! [rbpf]: https://github.com/solana-labs/rbpf
120//! [eBPF]: https://ebpf.io/
121//! [cc]: https://doc.rust-lang.org/reference/conditional-compilation.html
122//!
123//! ```
124//! pub fn sol_log(message: &str) {
125//!     #[cfg(target_os = "solana")]
126//!     unsafe {
127//!         sol_log_(message.as_ptr(), message.len() as u64);
128//!     }
129//!
130//!     #[cfg(not(target_os = "solana"))]
131//!     program_stubs::sol_log(message);
132//! }
133//! # mod program_stubs {
134//! #     pub(crate) fn sol_log(message: &str) { }
135//! # }
136//! ```
137//!
138//! This `cfg` pattern is suitable as well for user code that needs to work both
139//! on-chain and off-chain.
140//!
141//! `miraland-program` and `miraland-sdk` were previously a single crate. Because of
142//! this history, and because of the dual-usage of `miraland-program` for two
143//! different environments, it contains some features that are not available to
144//! on-chain programs at compile-time. It also contains some on-chain features
145//! that will fail in off-chain scenarios at runtime. This distinction is not
146//! well-reflected in the documentation.
147//!
148//! For a more complete description of Miraland's implementation of eBPF and its
149//! limitations, see the main Miraland documentation for [on-chain programs][ocp].
150//!
151//! [ocp]: https://docs.solana.com/developing/on-chain-programs/overview
152//!
153//! # Core data types
154//!
155//! - [`Pubkey`] — The address of a [Miraland account][acc]. Some account
156//!   addresses are [ed25519] public keys, with corresponding secret keys that
157//!   are managed off-chain. Often, though, account addresses do not have
158//!   corresponding secret keys — as with [_program derived
159//!   addresses_][pdas] — or the secret key is not relevant to the
160//!   operation of a program, and may have even been disposed of. As running
161//!   Miraland programs can not safely create or manage secret keys, the full
162//!   [`Keypair`] is not defined in `miraland-program` but in `miraland-sdk`.
163//! - [`Hash`] — A cryptographic hash. Used to uniquely identify blocks,
164//!   and also for general purpose hashing.
165//! - [`AccountInfo`] — A description of a single Miraland account. All accounts
166//!   that might be accessed by a program invocation are provided to the program
167//!   entrypoint as `AccountInfo`.
168//! - [`Instruction`] — A directive telling the runtime to execute a program,
169//!   passing it a set of accounts and program-specific data.
170//! - [`ProgramError`] and [`ProgramResult`] — The error type that all programs
171//!   must return, reported to the runtime as a `u64`.
172//! - [`Mln`] — The Miraland native token type, with conversions to and from
173//!   [_lamports_], the smallest fractional unit of MLN, in the [`native_token`]
174//!   module.
175//!
176//! [acc]: https://docs.solana.com/developing/programming-model/accounts
177//! [`Pubkey`]: pubkey::Pubkey
178//! [`Hash`]: hash::Hash
179//! [`Instruction`]: instruction::Instruction
180//! [`AccountInfo`]: account_info::AccountInfo
181//! [`ProgramError`]: program_error::ProgramError
182//! [`ProgramResult`]: entrypoint::ProgramResult
183//! [ed25519]: https://ed25519.cr.yp.to/
184//! [`Keypair`]: https://docs.rs/miraland-sdk/latest/miraland_sdk/signer/keypair/struct.Keypair.html
185//! [SHA-256]: https://en.wikipedia.org/wiki/SHA-2
186//! [`Mln`]: native_token::Mln
187//! [_lamports_]: https://docs.solana.com/introduction#what-are-sols
188//!
189//! # Serialization
190//!
191//! Within the Miraland runtime, programs, and network, at least three different
192//! serialization formats are used, and `miraland-program` provides access to
193//! those needed by programs.
194//!
195//! In user-written Miraland program code, serialization is primarily used for
196//! accessing [`AccountInfo`] data and [`Instruction`] data, both of which are
197//! program-specific binary data. Every program is free to decide their own
198//! serialization format, but data received from other sources —
199//! [sysvars][sysvar] for example — must be deserialized using the methods
200//! indicated by the documentation for that data or data type.
201//!
202//! [`AccountInfo`]: account_info::AccountInfo
203//! [`Instruction`]: instruction::Instruction
204//!
205//! The three serialization formats in use in Miraland are:
206//!
207//! - __[Borsh]__, a compact and well-specified format developed by the [NEAR]
208//!   project, suitable for use in protocol definitions and for archival storage.
209//!   It has a [Rust implementation][brust] and a [JavaScript implementation][bjs]
210//!   and is recommended for all purposes.
211//!
212//!   Users need to import the [`borsh`] crate themselves — it is not
213//!   re-exported by `miraland-program`, though this crate provides several useful
214//!   utilities in its [`borsh` module][borshmod] that are not available in the
215//!   `borsh` library.
216//!
217//!   The [`Instruction::new_with_borsh`] function creates an `Instruction` by
218//!   serializing a value with borsh.
219//!
220//!   [Borsh]: https://borsh.io/
221//!   [NEAR]: https://near.org/
222//!   [brust]: https://docs.rs/borsh
223//!   [bjs]: https://github.com/near/borsh-js
224//!   [`borsh`]: https://docs.rs/borsh
225//!   [borshmod]: crate::borsh
226//!   [`Instruction::new_with_borsh`]: instruction::Instruction::new_with_borsh
227//!
228//! - __[Bincode]__, a compact serialization format that implements the [Serde]
229//!   Rust APIs. As it does not have a specification nor a JavaScript
230//!   implementation, and uses more CPU than borsh, it is not recommend for new
231//!   code.
232//!
233//!   Many system program and native program instructions are serialized with
234//!   bincode, and it is used for other purposes in the runtime. In these cases
235//!   Rust programmers are generally not directly exposed to the encoding format
236//!   as it is hidden behind APIs.
237//!
238//!   The [`Instruction::new_with_bincode`] function creates an `Instruction` by
239//!   serializing a value with bincode.
240//!
241//!   [Bincode]: https://docs.rs/bincode
242//!   [Serde]: https://serde.rs/
243//!   [`Instruction::new_with_bincode`]: instruction::Instruction::new_with_bincode
244//!
245//! - __[`Pack`]__, a Miraland-specific serialization API that is used by many
246//!   older programs in the [Miraland Program Library][spl] to define their
247//!   account format. It is difficult to implement and does not define a
248//!   language-independent serialization format. It is not generally recommended
249//!   for new code.
250//!
251//!   [`Pack`]: program_pack::Pack
252//!
253//! Developers should carefully consider the CPU cost of serialization, balanced
254//! against the need for correctness and ease of use: off-the-shelf
255//! serialization formats tend to be more expensive than carefully hand-written
256//! application-specific formats; but application-specific formats are more
257//! difficult to ensure the correctness of, and to provide multi-language
258//! implementations for. It is not uncommon for programs to pack and unpack
259//! their data with hand-written code.
260//!
261//! # Cross-program instruction execution
262//!
263//! Miraland programs may call other programs, termed [_cross-program
264//! invocation_][cpi] (CPI), with the [`invoke`] and [`invoke_signed`]
265//! functions. When calling another program the caller must provide the
266//! [`Instruction`] to be invoked, as well as the [`AccountInfo`] for every
267//! account required by the instruction. Because the only way for a program to
268//! acquire `AccountInfo` values is by receiving them from the runtime at the
269//! [program entrypoint][entrypoint!], any account required by the callee
270//! program must transitively be required by the caller program, and provided by
271//! _its_ caller.
272//!
273//! [`invoke`]: program::invoke
274//! [`invoke_signed`]: program::invoke_signed
275//! [cpi]: https://docs.solana.com/developing/programming-model/calling-between-programs
276//!
277//! A simple example of transferring lamports via CPI:
278//!
279//! ```
280//! use miraland_program::{
281//!     account_info::{next_account_info, AccountInfo},
282//!     entrypoint,
283//!     entrypoint::ProgramResult,
284//!     program::invoke,
285//!     pubkey::Pubkey,
286//!     system_instruction,
287//!     system_program,
288//! };
289//!
290//! entrypoint!(process_instruction);
291//!
292//! fn process_instruction(
293//!     program_id: &Pubkey,
294//!     accounts: &[AccountInfo],
295//!     instruction_data: &[u8],
296//! ) -> ProgramResult {
297//!     let account_info_iter = &mut accounts.iter();
298//!
299//!     let payer = next_account_info(account_info_iter)?;
300//!     let recipient = next_account_info(account_info_iter)?;
301//!
302//!     assert!(payer.is_writable);
303//!     assert!(payer.is_signer);
304//!     assert!(recipient.is_writable);
305//!
306//!     let lamports = 1000000;
307//!
308//!     invoke(
309//!         &system_instruction::transfer(payer.key, recipient.key, lamports),
310//!         &[payer.clone(), recipient.clone()],
311//!     )
312//! }
313//! ```
314//!
315//! Miraland also includes a mechanism to let programs control and sign for
316//! accounts without needing to protect a corresponding secret key, called
317//! [_program derived addresses_][pdas]. PDAs are derived with the
318//! [`Pubkey::find_program_address`] function. With a PDA, a program can call
319//! `invoke_signed` to call another program while virtually "signing" for the
320//! PDA.
321//!
322//! [pdas]: https://docs.solana.com/developing/programming-model/calling-between-programs#program-derived-addresses
323//! [`Pubkey::find_program_address`]: pubkey::Pubkey::find_program_address
324//!
325//! A simple example of creating an account for a PDA:
326//!
327//! ```
328//! use miraland_program::{
329//!     account_info::{next_account_info, AccountInfo},
330//!     entrypoint,
331//!     entrypoint::ProgramResult,
332//!     program::invoke_signed,
333//!     pubkey::Pubkey,
334//!     system_instruction,
335//!     system_program,
336//! };
337//!
338//! entrypoint!(process_instruction);
339//!
340//! fn process_instruction(
341//!     program_id: &Pubkey,
342//!     accounts: &[AccountInfo],
343//!     instruction_data: &[u8],
344//! ) -> ProgramResult {
345//!     let account_info_iter = &mut accounts.iter();
346//!     let payer = next_account_info(account_info_iter)?;
347//!     let vault_pda = next_account_info(account_info_iter)?;
348//!     let system_program = next_account_info(account_info_iter)?;
349//!
350//!     assert!(payer.is_writable);
351//!     assert!(payer.is_signer);
352//!     assert!(vault_pda.is_writable);
353//!     assert_eq!(vault_pda.owner, &system_program::ID);
354//!     assert!(system_program::check_id(system_program.key));
355//!
356//!     let vault_bump_seed = instruction_data[0];
357//!     let vault_seeds = &[b"vault", payer.key.as_ref(), &[vault_bump_seed]];
358//!     let expected_vault_pda = Pubkey::create_program_address(vault_seeds, program_id)?;
359//!
360//!     assert_eq!(vault_pda.key, &expected_vault_pda);
361//!
362//!     let lamports = 10000000;
363//!     let vault_size = 16;
364//!
365//!     invoke_signed(
366//!         &system_instruction::create_account(
367//!             &payer.key,
368//!             &vault_pda.key,
369//!             lamports,
370//!             vault_size,
371//!             &program_id,
372//!         ),
373//!         &[
374//!             payer.clone(),
375//!             vault_pda.clone(),
376//!         ],
377//!         &[
378//!             &[
379//!                 b"vault",
380//!                 payer.key.as_ref(),
381//!                 &[vault_bump_seed],
382//!             ],
383//!         ]
384//!     )?;
385//!     Ok(())
386//! }
387//! ```
388//!
389//! # Native programs
390//!
391//! Some solana programs are [_native programs_][np2], running native machine
392//! code that is distributed with the runtime, with well-known program IDs.
393//!
394//! [np2]: https://docs.solana.com/developing/runtime-facilities/programs
395//!
396//! Some native programs can be [invoked][cpi] by other programs, but some can
397//! only be executed as "top-level" instructions included by off-chain clients
398//! in a [`Transaction`].
399//!
400//! [`Transaction`]: https://docs.rs/miraland-sdk/latest/miraland_sdk/transaction/struct.Transaction.html
401//!
402//! This crate defines the program IDs for most native programs. Even though
403//! some native programs cannot be invoked by other programs, a Miraland program
404//! may need access to their program IDs. For example, a program may need to
405//! verify that an ed25519 signature verification instruction was included in
406//! the same transaction as its own instruction. For many native programs, this
407//! crate also defines enums that represent the instructions they process, and
408//! constructors for building the instructions.
409//!
410//! Locations of program IDs and instruction constructors are noted in the list
411//! below, as well as whether they are invokable by other programs.
412//!
413//! While some native programs have been active since the genesis block, others
414//! are activated dynamically after a specific [slot], and some are not yet
415//! active. This documentation does not distinguish which native programs are
416//! active on any particular network. The `solana feature status` CLI command
417//! can help in determining active features.
418//!
419//! [slot]: https://docs.solana.com/terminology#slot
420//!
421//! Native programs important to Miraland program authors include:
422//!
423//! - __System Program__: Creates new accounts, allocates account data, assigns
424//!   accounts to owning programs, transfers lamports from System Program owned
425//!   accounts and pays transaction fees.
426//!   - ID: [`miraland_program::system_program`]
427//!   - Instruction: [`miraland_program::system_instruction`]
428//!   - Invokable by programs? yes
429//!
430//! - __Compute Budget Program__: Requests additional CPU or memory resources
431//!   for a transaction. This program does nothing when called from another
432//!   program.
433//!   - ID: [`miraland_sdk::compute_budget`](https://docs.rs/miraland-sdk/latest/miraland_sdk/compute_budget/index.html)
434//!   - Instruction: [`miraland_sdk::compute_budget`](https://docs.rs/miraland-sdk/latest/miraland_sdk/compute_budget/index.html)
435//!   - Invokable by programs? no
436//!
437//! - __ed25519 Program__: Verifies an ed25519 signature.
438//!   - ID: [`miraland_program::ed25519_program`]
439//!   - Instruction: [`miraland_sdk::ed25519_instruction`](https://docs.rs/miraland-sdk/latest/miraland_sdk/ed25519_instruction/index.html)
440//!   - Invokable by programs? no
441//!
442//! - __secp256k1 Program__: Verifies secp256k1 public key recovery operations.
443//!   - ID: [`miraland_program::secp256k1_program`]
444//!   - Instruction: [`miraland_sdk::secp256k1_instruction`](https://docs.rs/miraland-sdk/latest/miraland_sdk/secp256k1_instruction/index.html)
445//!   - Invokable by programs? no
446//!
447//! - __BPF Loader__: Deploys, and executes immutable programs on the chain.
448//!   - ID: [`miraland_program::bpf_loader`]
449//!   - Instruction: [`miraland_program::loader_instruction`]
450//!   - Invokable by programs? yes
451//!
452//! - __Upgradable BPF Loader__: Deploys, upgrades, and executes upgradable
453//!   programs on the chain.
454//!   - ID: [`miraland_program::bpf_loader_upgradeable`]
455//!   - Instruction: [`miraland_program::loader_upgradeable_instruction`]
456//!   - Invokable by programs? yes
457//!
458//! - __Deprecated BPF Loader__: Deploys, and executes immutable programs on the
459//!   chain.
460//!   - ID: [`miraland_program::bpf_loader_deprecated`]
461//!   - Instruction: [`miraland_program::loader_instruction`]
462//!   - Invokable by programs? yes
463//!
464//! [lut]: https://docs.solana.com/proposals/versioned-transactions
465
466#![allow(incomplete_features)]
467#![cfg_attr(RUSTC_WITH_SPECIALIZATION, feature(specialization))]
468#![cfg_attr(RUSTC_NEEDS_PROC_MACRO_HYGIENE, feature(proc_macro_hygiene))]
469
470// Allows macro expansion of `use ::miraland_program::*` to work within this crate
471extern crate self as miraland_program;
472
473pub mod account_info;
474pub mod address_lookup_table;
475pub mod alt_bn128;
476pub(crate) mod atomic_u64;
477pub mod big_mod_exp;
478pub mod blake3;
479pub mod borsh;
480pub mod borsh0_10;
481pub mod borsh0_9;
482pub mod borsh1;
483pub mod bpf_loader;
484pub mod bpf_loader_deprecated;
485pub mod bpf_loader_upgradeable;
486pub mod clock;
487pub mod compute_units;
488pub mod debug_account_data;
489pub mod decode_error;
490pub mod ed25519_program;
491pub mod entrypoint;
492pub mod entrypoint_deprecated;
493pub mod epoch_rewards;
494pub mod epoch_schedule;
495pub mod feature;
496pub mod fee_calculator;
497pub mod hash;
498pub mod incinerator;
499pub mod instruction;
500pub mod keccak;
501pub mod lamports;
502pub mod last_restart_slot;
503pub mod loader_instruction;
504pub mod loader_upgradeable_instruction;
505pub mod loader_v4;
506pub mod loader_v4_instruction;
507pub mod log;
508pub mod message;
509pub mod native_token;
510pub mod nonce;
511pub mod poseidon;
512pub mod program;
513pub mod program_error;
514pub mod program_memory;
515pub mod program_option;
516pub mod program_pack;
517pub mod program_stubs;
518pub mod program_utils;
519pub mod pubkey;
520pub mod rent;
521pub mod sanitize;
522pub mod secp256k1_program;
523pub mod secp256k1_recover;
524pub mod serde_varint;
525pub mod serialize_utils;
526pub mod short_vec;
527pub mod slot_hashes;
528pub mod slot_history;
529pub mod stable_layout;
530pub mod stake;
531pub mod stake_history;
532pub mod syscalls;
533pub mod system_instruction;
534pub mod system_program;
535pub mod sysvar;
536pub mod vote;
537pub mod wasm;
538
539#[deprecated(
540    since = "1.17.0",
541    note = "Please use `miraland_sdk::address_lookup_table::AddressLookupTableAccount` instead"
542)]
543pub mod address_lookup_table_account {
544    pub use crate::address_lookup_table::AddressLookupTableAccount;
545}
546
547#[cfg(target_os = "solana")]
548pub use miraland_sdk_macro::wasm_bindgen_stub as wasm_bindgen;
549/// Re-export of [wasm-bindgen].
550///
551/// [wasm-bindgen]: https://rustwasm.github.io/docs/wasm-bindgen/
552#[cfg(not(target_os = "solana"))]
553pub use wasm_bindgen::prelude::wasm_bindgen;
554
555/// The [config native program][np].
556///
557/// [np]: https://docs.miraland.top/runtime/programs#config-program
558pub mod config {
559    pub mod program {
560        crate::declare_id!("Config1111111111111111111111111111111111111");
561    }
562}
563
564/// A vector of Miraland SDK IDs.
565pub mod sdk_ids {
566    use {
567        crate::{
568            address_lookup_table, bpf_loader, bpf_loader_deprecated, bpf_loader_upgradeable,
569            config, ed25519_program, feature, incinerator, loader_v4, secp256k1_program,
570            miraland_program::pubkey::Pubkey, stake, system_program, sysvar, vote,
571        },
572        lazy_static::lazy_static,
573    };
574
575    lazy_static! {
576        pub static ref SDK_IDS: Vec<Pubkey> = {
577            let mut sdk_ids = vec![
578                ed25519_program::id(),
579                secp256k1_program::id(),
580                system_program::id(),
581                sysvar::id(),
582                bpf_loader::id(),
583                bpf_loader_upgradeable::id(),
584                incinerator::id(),
585                config::program::id(),
586                vote::program::id(),
587                feature::id(),
588                bpf_loader_deprecated::id(),
589                address_lookup_table::program::id(),
590                loader_v4::id(),
591                stake::program::id(),
592                #[allow(deprecated)]
593                stake::config::id(),
594            ];
595            sdk_ids.extend(sysvar::ALL_IDS.iter());
596            sdk_ids
597        };
598    }
599}
600
601/// Same as [`declare_id`] except that it reports that this ID has been deprecated.
602pub use miraland_sdk_macro::program_declare_deprecated_id as declare_deprecated_id;
603/// Convenience macro to declare a static public key and functions to interact with it.
604///
605/// Input: a single literal base58 string representation of a program's ID.
606///
607/// # Example
608///
609/// ```
610/// # // wrapper is used so that the macro invocation occurs in the item position
611/// # // rather than in the statement position which isn't allowed.
612/// use std::str::FromStr;
613/// use miraland_program::{declare_id, pubkey::Pubkey};
614///
615/// # mod item_wrapper {
616/// #   use miraland_program::declare_id;
617/// declare_id!("My11111111111111111111111111111111111111111");
618/// # }
619/// # use item_wrapper::id;
620///
621/// let my_id = Pubkey::from_str("My11111111111111111111111111111111111111111").unwrap();
622/// assert_eq!(id(), my_id);
623/// ```
624pub use miraland_sdk_macro::program_declare_id as declare_id;
625/// Convenience macro to define a static public key.
626///
627/// Input: a single literal base58 string representation of a Pubkey.
628///
629/// # Example
630///
631/// ```
632/// use std::str::FromStr;
633/// use miraland_program::{pubkey, pubkey::Pubkey};
634///
635/// static ID: Pubkey = pubkey!("My11111111111111111111111111111111111111111");
636///
637/// let my_id = Pubkey::from_str("My11111111111111111111111111111111111111111").unwrap();
638/// assert_eq!(ID, my_id);
639/// ```
640pub use miraland_sdk_macro::program_pubkey as pubkey;
641
642#[macro_use]
643extern crate serde_derive;
644
645#[macro_use]
646extern crate miraland_frozen_abi_macro;
647
648/// Convenience macro for doing integer division where the operation's safety
649/// can be checked at compile-time.
650///
651/// Since `unchecked_div_by_const!()` is supposed to fail at compile-time, abuse
652/// doctests to cover failure modes
653///
654/// # Examples
655///
656/// Literal denominator div-by-zero fails:
657///
658/// ```compile_fail
659/// # use miraland_program::unchecked_div_by_const;
660/// # fn main() {
661/// let _ = unchecked_div_by_const!(10, 0);
662/// # }
663/// ```
664///
665/// Const denominator div-by-zero fails:
666///
667/// ```compile_fail
668/// # use miraland_program::unchecked_div_by_const;
669/// # fn main() {
670/// const D: u64 = 0;
671/// let _ = unchecked_div_by_const!(10, D);
672/// # }
673/// ```
674///
675/// Non-const denominator fails:
676///
677/// ```compile_fail
678/// # use miraland_program::unchecked_div_by_const;
679/// # fn main() {
680/// let d = 0;
681/// let _ = unchecked_div_by_const!(10, d);
682/// # }
683/// ```
684///
685/// Literal denominator div-by-zero fails:
686///
687/// ```compile_fail
688/// # use miraland_program::unchecked_div_by_const;
689/// # fn main() {
690/// const N: u64 = 10;
691/// let _ = unchecked_div_by_const!(N, 0);
692/// # }
693/// ```
694///
695/// Const denominator div-by-zero fails:
696///
697/// ```compile_fail
698/// # use miraland_program::unchecked_div_by_const;
699/// # fn main() {
700/// const N: u64 = 10;
701/// const D: u64 = 0;
702/// let _ = unchecked_div_by_const!(N, D);
703/// # }
704/// ```
705///
706/// Non-const denominator fails:
707///
708/// ```compile_fail
709/// # use miraland_program::unchecked_div_by_const;
710/// # fn main() {
711/// # const N: u64 = 10;
712/// let d = 0;
713/// let _ = unchecked_div_by_const!(N, d);
714/// # }
715/// ```
716///
717/// Literal denominator div-by-zero fails:
718///
719/// ```compile_fail
720/// # use miraland_program::unchecked_div_by_const;
721/// # fn main() {
722/// let n = 10;
723/// let _ = unchecked_div_by_const!(n, 0);
724/// # }
725/// ```
726///
727/// Const denominator div-by-zero fails:
728///
729/// ```compile_fail
730/// # use miraland_program::unchecked_div_by_const;
731/// # fn main() {
732/// let n = 10;
733/// const D: u64 = 0;
734/// let _ = unchecked_div_by_const!(n, D);
735/// # }
736/// ```
737///
738/// Non-const denominator fails:
739///
740/// ```compile_fail
741/// # use miraland_program::unchecked_div_by_const;
742/// # fn main() {
743/// let n = 10;
744/// let d = 0;
745/// let _ = unchecked_div_by_const!(n, d);
746/// # }
747/// ```
748#[macro_export]
749macro_rules! unchecked_div_by_const {
750    ($num:expr, $den:expr) => {{
751        // Ensure the denominator is compile-time constant
752        let _ = [(); ($den - $den) as usize];
753        // Compile-time constant integer div-by-zero passes for some reason
754        // when invoked from a compilation unit other than that where this
755        // macro is defined. Do an explicit zero-check for now. Sorry about the
756        // ugly error messages!
757        // https://users.rust-lang.org/t/unexpected-behavior-of-compile-time-integer-div-by-zero-check-in-declarative-macro/56718
758        let _ = [(); ($den as usize) - 1];
759        #[allow(clippy::arithmetic_side_effects)]
760        let quotient = $num / $den;
761        quotient
762    }};
763}
764
765// This module is purposefully listed after all other exports: because of an
766// interaction within rustdoc between the reexports inside this module of
767// `miraland_program`'s top-level modules, and `miraland_sdk`'s glob re-export of
768// `miraland_program`'s top-level modules, if this module is not lexically last
769// rustdoc fails to generate documentation for the re-exports within
770// `miraland_sdk`.
771#[cfg(not(target_os = "solana"))]
772pub mod example_mocks;
773
774#[cfg(test)]
775mod tests {
776    use super::unchecked_div_by_const;
777
778    #[test]
779    fn test_unchecked_div_by_const() {
780        const D: u64 = 2;
781        const N: u64 = 10;
782        let n = 10;
783        assert_eq!(unchecked_div_by_const!(10, 2), 5);
784        assert_eq!(unchecked_div_by_const!(N, 2), 5);
785        assert_eq!(unchecked_div_by_const!(n, 2), 5);
786        assert_eq!(unchecked_div_by_const!(10, D), 5);
787        assert_eq!(unchecked_div_by_const!(N, D), 5);
788        assert_eq!(unchecked_div_by_const!(n, D), 5);
789    }
790}