1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
// Copyright 2019-2020 Parity Technologies (UK) Ltd. // This file is part of Substrate. // Substrate is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // Substrate is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License // along with Substrate. If not, see <http://www.gnu.org/licenses/>. //! Runtime debugging and logging utilities. //! //! This module contains macros and functions that will allow //! you to print logs out of the runtime code. //! //! First and foremost be aware that adding regular logging code to //! your runtime will have a negative effect on the performance //! and size of the blob. Luckily there are some ways to mitigate //! this that are described below. //! //! First component to utilize debug-printing and loggin is actually //! located in `primitives` crate: `sp_core::RuntimeDebug`. //! This custom-derive generates `core::fmt::Debug` implementation, //! just like regular `derive(Debug)`, however it does not generate //! any code when the code is compiled to WASM. This means that //! you can safely sprinkle `RuntimeDebug` in your runtime codebase, //! without affecting the size. This also allows you to print/log //! both when the code is running natively or in WASM, but note //! that WASM debug formatting of structs will be empty. //! //! ```rust,no_run //! use frame_support::debug; //! //! #[derive(sp_core::RuntimeDebug)] //! struct MyStruct { //! a: u64, //! } //! //! // First initialize the logger. //! // //! // This is only required when you want the logs to be printed //! // also during non-native run. //! // Note that enabling the logger has performance impact on //! // WASM runtime execution and should be used sparingly. //! debug::RuntimeLogger::init(); //! //! let x = MyStruct { a: 5 }; //! // will log an info line `"My struct: MyStruct{a:5}"` when running //! // natively, but will only print `"My struct: "` when running WASM. //! debug::info!("My struct: {:?}", x); //! //! // same output here, although this will print to stdout //! // (and without log format) //! debug::print!("My struct: {:?}", x); //! ``` //! //! If you want to avoid extra overhead in WASM, but still be able //! to print / log when the code is executed natively you can use //! macros coming from `native` sub-module. This module enables //! logs conditionally and strips out logs in WASM. //! //! ```rust,no_run //! use frame_support::debug::native; //! //! #[derive(sp_core::RuntimeDebug)] //! struct MyStruct { //! a: u64, //! } //! //! // We don't initialize the logger, since //! // we are not printing anything out in WASM. //! // debug::RuntimeLogger::init(); //! //! let x = MyStruct { a: 5 }; //! //! // Displays an info log when running natively, nothing when WASM. //! native::info!("My struct: {:?}", x); //! //! // same output to stdout, no overhead on WASM. //! native::print!("My struct: {:?}", x); //! ``` use sp_std::vec::Vec; use sp_std::fmt::{self, Debug}; pub use log::{info, debug, error, trace, warn}; pub use crate::runtime_print as print; /// Native-only logging. /// /// Using any functions from this module will have any effect /// only if the runtime is running natively (i.e. not via WASM) #[cfg(feature = "std")] pub mod native { pub use super::{info, debug, error, trace, warn, print}; } /// Native-only logging. /// /// Using any functions from this module will have any effect /// only if the runtime is running natively (i.e. not via WASM) #[cfg(not(feature = "std"))] pub mod native { #[macro_export] macro_rules! noop { ($($arg:tt)+) => {} } pub use noop as info; pub use noop as debug; pub use noop as error; pub use noop as trace; pub use noop as warn; pub use noop as print; } /// Print out a formatted message. /// /// # Example /// /// ``` /// frame_support::runtime_print!("my value is {}", 3); /// ``` #[macro_export] macro_rules! runtime_print { ($($arg:tt)+) => { use core::fmt::Write; let mut w = $crate::debug::Writer::default(); let _ = core::write!(&mut w, $($arg)+); w.print(); } } /// Print out the debuggable type. pub fn debug(data: &impl Debug) { runtime_print!("{:?}", data); } /// A target for `core::write!` macro - constructs a string in memory. #[derive(Default)] pub struct Writer(Vec<u8>); impl fmt::Write for Writer { fn write_str(&mut self, s: &str) -> fmt::Result { self.0.extend(s.as_bytes()); Ok(()) } } impl Writer { /// Print the content of this `Writer` out. pub fn print(&self) { sp_io::misc::print_utf8(&self.0) } } /// Runtime logger implementation - `log` crate backend. /// /// The logger should be initialized if you want to display /// logs inside the runtime that is not necessarily running natively. /// /// When runtime is executed natively any log statements are displayed /// even if this logger is NOT initialized. /// /// Note that even though the logs are not displayed in WASM, they /// may still affect the size and performance of the generated runtime. /// To lower the footprint make sure to only use macros from `native` /// sub-module. pub struct RuntimeLogger; impl RuntimeLogger { /// Initialize the logger. /// /// This is a no-op when running natively (`std`). #[cfg(feature = "std")] pub fn init() {} /// Initialize the logger. /// /// This is a no-op when running natively (`std`). #[cfg(not(feature = "std"))] pub fn init() { static LOGGER: RuntimeLogger = RuntimeLogger;; let _ = log::set_logger(&LOGGER); } } impl log::Log for RuntimeLogger { fn enabled(&self, _metadata: &log::Metadata) -> bool { // to avoid calling to host twice, we pass everything // and let the host decide what to print. // If someone is initializing the logger they should // know what they are doing. true } fn log(&self, record: &log::Record) { use fmt::Write; let mut w = Writer::default(); let _ = core::write!(&mut w, "{}", record.args()); sp_io::logging::log( record.level().into(), record.target(), &w.0, ); } fn flush(&self) {} }