1#[cfg(unix)]
4use std::os::unix::io::{AsFd, AsRawFd};
5use std::{
6 boxed::Box as Box_,
7 sync::{Arc, Mutex, OnceLock},
8};
9
10#[cfg(feature = "v2_80")]
11use crate::StrVRef;
12use crate::{GStr, GString, LogWriterOutput, ffi, translate::*};
13
14#[derive(Debug)]
15pub struct LogHandlerId(u32);
16
17#[doc(hidden)]
18impl FromGlib<u32> for LogHandlerId {
19 #[inline]
20 unsafe fn from_glib(value: u32) -> Self {
21 Self(value)
22 }
23}
24
25#[doc(hidden)]
26impl IntoGlib for LogHandlerId {
27 type GlibType = u32;
28
29 #[inline]
30 fn into_glib(self) -> u32 {
31 self.0
32 }
33}
34
35#[derive(Copy, Clone, Debug, PartialEq, Eq)]
36pub enum LogLevel {
37 #[doc(alias = "G_LOG_LEVEL_ERROR")]
38 Error,
39 #[doc(alias = "G_LOG_LEVEL_CRITICAL")]
40 Critical,
41 #[doc(alias = "G_LOG_LEVEL_WARNING")]
42 Warning,
43 #[doc(alias = "G_LOG_LEVEL_MESSAGE")]
44 Message,
45 #[doc(alias = "G_LOG_LEVEL_INFO")]
46 Info,
47 #[doc(alias = "G_LOG_LEVEL_DEBUG")]
48 Debug,
49}
50
51#[doc(hidden)]
52impl IntoGlib for LogLevel {
53 type GlibType = u32;
54
55 #[inline]
56 fn into_glib(self) -> u32 {
57 match self {
58 Self::Error => ffi::G_LOG_LEVEL_ERROR,
59 Self::Critical => ffi::G_LOG_LEVEL_CRITICAL,
60 Self::Warning => ffi::G_LOG_LEVEL_WARNING,
61 Self::Message => ffi::G_LOG_LEVEL_MESSAGE,
62 Self::Info => ffi::G_LOG_LEVEL_INFO,
63 Self::Debug => ffi::G_LOG_LEVEL_DEBUG,
64 }
65 }
66}
67
68#[doc(hidden)]
69impl FromGlib<u32> for LogLevel {
70 #[inline]
71 unsafe fn from_glib(value: u32) -> Self {
72 if value & ffi::G_LOG_LEVEL_ERROR != 0 {
73 Self::Error
74 } else if value & ffi::G_LOG_LEVEL_CRITICAL != 0 {
75 Self::Critical
76 } else if value & ffi::G_LOG_LEVEL_WARNING != 0 {
77 Self::Warning
78 } else if value & ffi::G_LOG_LEVEL_MESSAGE != 0 {
79 Self::Message
80 } else if value & ffi::G_LOG_LEVEL_INFO != 0 {
81 Self::Info
82 } else if value & ffi::G_LOG_LEVEL_DEBUG != 0 {
83 Self::Debug
84 } else {
85 panic!("Unknown log level: {value}")
86 }
87 }
88}
89
90impl LogLevel {
91 #[doc(hidden)]
92 pub fn priority(&self) -> &'static str {
93 match self {
94 Self::Error => "3",
95 Self::Critical => "4",
96 Self::Warning => "4",
97 Self::Message => "5",
98 Self::Info => "6",
99 Self::Debug => "7",
100 }
101 }
102}
103
104bitflags::bitflags! {
105 #[doc(alias = "GLogLevelFlags")]
106 #[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
107 pub struct LogLevels: u32 {
108 #[doc(alias = "G_LOG_LEVEL_ERROR")]
109 const LEVEL_ERROR = ffi::G_LOG_LEVEL_ERROR;
110 #[doc(alias = "G_LOG_LEVEL_CRITICAL")]
111 const LEVEL_CRITICAL = ffi::G_LOG_LEVEL_CRITICAL;
112 #[doc(alias = "G_LOG_LEVEL_WARNING")]
113 const LEVEL_WARNING = ffi::G_LOG_LEVEL_WARNING;
114 #[doc(alias = "G_LOG_LEVEL_MESSAGE")]
115 const LEVEL_MESSAGE = ffi::G_LOG_LEVEL_MESSAGE;
116 #[doc(alias = "G_LOG_LEVEL_INFO")]
117 const LEVEL_INFO = ffi::G_LOG_LEVEL_INFO;
118 #[doc(alias = "G_LOG_LEVEL_DEBUG")]
119 const LEVEL_DEBUG = ffi::G_LOG_LEVEL_DEBUG;
120 }
121}
122
123#[doc(hidden)]
124impl IntoGlib for LogLevels {
125 type GlibType = ffi::GLogLevelFlags;
126
127 #[inline]
128 fn into_glib(self) -> ffi::GLogLevelFlags {
129 self.bits()
130 }
131}
132
133#[doc(hidden)]
134impl FromGlib<ffi::GLogLevelFlags> for LogLevels {
135 #[inline]
136 unsafe fn from_glib(value: ffi::GLogLevelFlags) -> Self {
137 Self::from_bits_truncate(value)
138 }
139}
140
141fn to_log_flags(fatal: bool, recursion: bool) -> u32 {
142 (if fatal { ffi::G_LOG_FLAG_FATAL } else { 0 })
143 | if recursion {
144 ffi::G_LOG_FLAG_RECURSION
145 } else {
146 0
147 }
148}
149
150#[doc(alias = "g_log_set_handler_full")]
151pub fn log_set_handler<P: Fn(Option<&str>, LogLevel, &str) + Send + Sync + 'static>(
152 log_domain: Option<&str>,
153 log_levels: LogLevels,
154 fatal: bool,
155 recursion: bool,
156 log_func: P,
157) -> LogHandlerId {
158 let log_func_data: Box_<P> = Box_::new(log_func);
159 unsafe extern "C" fn log_func_func<
160 P: Fn(Option<&str>, LogLevel, &str) + Send + Sync + 'static,
161 >(
162 log_domain: *const libc::c_char,
163 log_level: ffi::GLogLevelFlags,
164 message: *const libc::c_char,
165 user_data: ffi::gpointer,
166 ) {
167 unsafe {
168 let log_domain: Borrowed<Option<GString>> = from_glib_borrow(log_domain);
169 let message: Borrowed<GString> = from_glib_borrow(message);
170 let callback: &P = &*(user_data as *mut _);
171 (*callback)(
172 (*log_domain).as_ref().map(|s| s.as_str()),
173 from_glib(log_level),
174 message.as_str(),
175 );
176 }
177 }
178 let log_func = Some(log_func_func::<P> as _);
179 unsafe extern "C" fn destroy_func<
180 P: Fn(Option<&str>, LogLevel, &str) + Send + Sync + 'static,
181 >(
182 data: ffi::gpointer,
183 ) {
184 unsafe {
185 let _callback: Box_<P> = Box_::from_raw(data as *mut _);
186 }
187 }
188 let destroy_call4 = Some(destroy_func::<P> as _);
189 let super_callback0: Box_<P> = log_func_data;
190 unsafe {
191 from_glib(ffi::g_log_set_handler_full(
192 log_domain.to_glib_none().0,
193 log_levels.into_glib() | to_log_flags(fatal, recursion),
194 log_func,
195 Box_::into_raw(super_callback0) as *mut _,
196 destroy_call4,
197 ))
198 }
199}
200
201#[doc(alias = "g_log_remove_handler")]
202pub fn log_remove_handler(log_domain: Option<&str>, handler_id: LogHandlerId) {
203 unsafe {
204 ffi::g_log_remove_handler(log_domain.to_glib_none().0, handler_id.into_glib());
205 }
206}
207
208#[doc(alias = "g_log_set_always_fatal")]
209pub fn log_set_always_fatal(fatal_levels: LogLevels) -> LogLevels {
210 unsafe { from_glib(ffi::g_log_set_always_fatal(fatal_levels.into_glib())) }
211}
212
213#[doc(alias = "g_log_set_fatal_mask")]
214pub fn log_set_fatal_mask(log_domain: Option<&str>, fatal_levels: LogLevels) -> LogLevels {
215 unsafe {
216 from_glib(ffi::g_log_set_fatal_mask(
217 log_domain.to_glib_none().0,
218 fatal_levels.into_glib(),
219 ))
220 }
221}
222
223type PrintCallback = dyn Fn(&str) + Send + Sync + 'static;
224
225fn print_handler() -> &'static Mutex<Option<Arc<PrintCallback>>> {
226 static MUTEX: OnceLock<Mutex<Option<Arc<PrintCallback>>>> = OnceLock::new();
227 MUTEX.get_or_init(|| Mutex::new(None))
228}
229
230#[doc(alias = "g_set_print_handler")]
233pub fn set_print_handler<P: Fn(&str) + Send + Sync + 'static>(func: P) {
234 unsafe extern "C" fn func_func(string: *const libc::c_char) {
235 unsafe {
236 if let Some(callback) = print_handler()
237 .lock()
238 .expect("Failed to lock PRINT_HANDLER")
239 .as_ref()
240 .map(Arc::clone)
241 {
242 let string: Borrowed<GString> = from_glib_borrow(string);
243 (*callback)(string.as_str())
244 }
245 }
246 }
247 *print_handler()
248 .lock()
249 .expect("Failed to lock PRINT_HANDLER to change callback") = Some(Arc::new(func));
250 unsafe { ffi::g_set_print_handler(Some(func_func as _)) };
251}
252
253pub fn unset_print_handler() {
256 *print_handler()
257 .lock()
258 .expect("Failed to lock PRINT_HANDLER to remove callback") = None;
259 unsafe { ffi::g_set_print_handler(None) };
260}
261
262fn printerr_handler() -> &'static Mutex<Option<Arc<PrintCallback>>> {
263 static MUTEX: OnceLock<Mutex<Option<Arc<PrintCallback>>>> = OnceLock::new();
264 MUTEX.get_or_init(|| Mutex::new(None))
265}
266
267#[doc(alias = "g_set_printerr_handler")]
270pub fn set_printerr_handler<P: Fn(&str) + Send + Sync + 'static>(func: P) {
271 unsafe extern "C" fn func_func(string: *const libc::c_char) {
272 unsafe {
273 if let Some(callback) = printerr_handler()
274 .lock()
275 .expect("Failed to lock PRINTERR_HANDLER")
276 .as_ref()
277 .map(Arc::clone)
278 {
279 let string: Borrowed<GString> = from_glib_borrow(string);
280 (*callback)(string.as_str())
281 }
282 }
283 }
284 *printerr_handler()
285 .lock()
286 .expect("Failed to lock PRINTERR_HANDLER to change callback") = Some(Arc::new(func));
287 unsafe { ffi::g_set_printerr_handler(Some(func_func as _)) };
288}
289
290pub fn unset_printerr_handler() {
293 *printerr_handler()
294 .lock()
295 .expect("Failed to lock PRINTERR_HANDLER to remove callback") = None;
296 unsafe { ffi::g_set_printerr_handler(None) };
297}
298
299type LogCallback = dyn Fn(Option<&str>, LogLevel, &str) + Send + Sync + 'static;
300
301fn default_handler() -> &'static Mutex<Option<Arc<LogCallback>>> {
302 static MUTEX: OnceLock<Mutex<Option<Arc<LogCallback>>>> = OnceLock::new();
303 MUTEX.get_or_init(|| Mutex::new(None))
304}
305
306#[doc(alias = "g_log_set_default_handler")]
309pub fn log_set_default_handler<P: Fn(Option<&str>, LogLevel, &str) + Send + Sync + 'static>(
310 log_func: P,
311) {
312 unsafe extern "C" fn func_func(
313 log_domain: *const libc::c_char,
314 log_levels: ffi::GLogLevelFlags,
315 message: *const libc::c_char,
316 _user_data: ffi::gpointer,
317 ) {
318 unsafe {
319 if let Some(callback) = default_handler()
320 .lock()
321 .expect("Failed to lock DEFAULT_HANDLER")
322 .as_ref()
323 .map(Arc::clone)
324 {
325 let log_domain: Borrowed<Option<GString>> = from_glib_borrow(log_domain);
326 let message: Borrowed<GString> = from_glib_borrow(message);
327 (*callback)(
328 (*log_domain).as_ref().map(|s| s.as_str()),
329 from_glib(log_levels),
330 message.as_str(),
331 );
332 }
333 }
334 }
335 *default_handler()
336 .lock()
337 .expect("Failed to lock DEFAULT_HANDLER to change callback") = Some(Arc::new(log_func));
338 unsafe { ffi::g_log_set_default_handler(Some(func_func as _), std::ptr::null_mut()) };
339}
340
341#[doc(alias = "g_log_set_default_handler")]
344pub fn log_unset_default_handler() {
345 *default_handler()
346 .lock()
347 .expect("Failed to lock DEFAULT_HANDLER to remove callback") = None;
348 unsafe {
349 ffi::g_log_set_default_handler(Some(ffi::g_log_default_handler), std::ptr::null_mut())
350 };
351}
352
353#[doc(alias = "g_log_default_handler")]
354pub fn log_default_handler(log_domain: Option<&str>, log_level: LogLevel, message: Option<&str>) {
355 unsafe {
356 ffi::g_log_default_handler(
357 log_domain.to_glib_none().0,
358 log_level.into_glib(),
359 message.to_glib_none().0,
360 std::ptr::null_mut(),
361 )
362 }
363}
364
365#[repr(transparent)]
373#[derive(Debug)]
374#[doc(alias = "GLogField")]
375pub struct LogField<'a>(ffi::GLogField, std::marker::PhantomData<&'a GStr>);
376
377impl<'a> LogField<'a> {
378 pub fn new(key: &'a GStr, value: &[u8]) -> Self {
381 let (value, length) = if value.is_empty() {
382 (&[0u8] as &[u8], -1isize)
385 } else {
386 (value, value.len().try_into().unwrap())
387 };
388 Self(
389 ffi::GLogField {
390 key: key.as_ptr(),
391 value: value.as_ptr() as *const _,
392 length,
393 },
394 Default::default(),
395 )
396 }
397 pub fn new_user_data(key: &'a GStr, data: usize) -> Self {
407 Self(
408 ffi::GLogField {
409 key: key.as_ptr(),
410 value: std::ptr::without_provenance(data),
411 length: 0,
412 },
413 Default::default(),
414 )
415 }
416 pub fn key(&self) -> &str {
419 unsafe { std::ffi::CStr::from_ptr(self.0.key as *const _) }
420 .to_str()
421 .unwrap()
422 }
423 pub fn value_bytes(&self) -> Option<&[u8]> {
427 match self.0.length {
428 0 => None,
429 n if n < 0 => {
430 Some(unsafe { std::ffi::CStr::from_ptr(self.0.value as *const _) }.to_bytes())
431 }
432 _ => Some(unsafe {
433 std::slice::from_raw_parts(self.0.value as *const u8, self.0.length as usize)
434 }),
435 }
436 }
437 pub fn value_str(&self) -> Option<&str> {
441 std::str::from_utf8(self.value_bytes()?).ok()
442 }
443 pub fn user_data(&self) -> Option<usize> {
447 (self.0.length == 0).then_some(self.0.value.addr())
448 }
449}
450
451type WriterCallback = dyn Fn(LogLevel, &[LogField<'_>]) -> LogWriterOutput + Send + Sync + 'static;
452
453static WRITER_FUNC: OnceLock<Box<WriterCallback>> = OnceLock::new();
454
455#[doc(alias = "g_log_set_writer_func")]
456pub fn log_set_writer_func<
457 P: Fn(LogLevel, &[LogField<'_>]) -> LogWriterOutput + Send + Sync + 'static,
458>(
459 writer_func: P,
460) {
461 if WRITER_FUNC.set(Box::new(writer_func)).is_err() {
462 panic!("Writer func can only be set once");
463 }
464 unsafe extern "C" fn writer_trampoline(
465 log_level: ffi::GLogLevelFlags,
466 fields: *const ffi::GLogField,
467 n_fields: libc::size_t,
468 _user_data: ffi::gpointer,
469 ) -> ffi::GLogWriterOutput {
470 unsafe {
471 let writer_func = WRITER_FUNC.get().unwrap();
472 let fields = std::slice::from_raw_parts(fields as *const LogField<'_>, n_fields);
473 writer_func(from_glib(log_level), fields).into_glib()
474 }
475 }
476 unsafe {
477 ffi::g_log_set_writer_func(Some(writer_trampoline), std::ptr::null_mut(), None);
478 }
479}
480
481#[macro_export]
482#[doc(hidden)]
483macro_rules! g_log_inner {
484 ($log_domain:expr, $log_level:expr, $format:literal $(,$arg:expr)* $(,)?) => {{
485 let mut w = $crate::GStringBuilder::default();
486
487 if !std::fmt::Write::write_fmt(&mut w, std::format_args!($format, $($arg),*)).is_err() {
489 unsafe {
490 $crate::ffi::g_log(
491 $crate::translate::ToGlibPtr::to_glib_none(&$log_domain).0,
492 <$crate::LogLevel as $crate::translate::IntoGlib>::into_glib(
493 $log_level
494 ),
495 b"%s\0".as_ptr() as *const _,
496 $crate::translate::ToGlibPtr::<*const std::os::raw::c_char>::to_glib_none(
497 &w.into_string()
498 ).0,
499 );
500 }
501 }
502 }};
503}
504
505#[macro_export]
546macro_rules! g_log {
547 ($log_level:expr, $format:literal $(,$arg:expr)* $(,)?) => {{
548 $crate::g_log_inner!(None::<&str>, $log_level, $format, $($arg),*);
549 }};
550 ($log_domain:expr, $log_level:expr, $format:literal $(,$arg:expr)* $(,)?) => {{
551 let log_domain = <Option<&str> as std::convert::From<_>>::from($log_domain);
552 $crate::g_log_inner!(log_domain, $log_level, $format, $($arg),*);
553 }};
554}
555
556#[macro_export]
584macro_rules! g_error {
585 ($log_domain:expr, $format:literal, $($arg:expr),* $(,)?) => {{
586 $crate::g_log!($log_domain, $crate::LogLevel::Error, $format, $($arg),*);
587 }};
588 ($log_domain:expr, $format:literal $(,)?) => {{
589 $crate::g_log!($log_domain, $crate::LogLevel::Error, $format);
590 }};
591}
592
593#[macro_export]
621macro_rules! g_critical {
622 ($log_domain:expr, $format:literal, $($arg:expr),* $(,)?) => {{
623 $crate::g_log!($log_domain, $crate::LogLevel::Critical, $format, $($arg),*);
624 }};
625 ($log_domain:expr, $format:literal $(,)?) => {{
626 $crate::g_log!($log_domain, $crate::LogLevel::Critical, $format);
627 }};
628}
629
630#[macro_export]
658macro_rules! g_warning {
659 ($log_domain:expr, $format:literal, $($arg:expr),* $(,)?) => {{
660 $crate::g_log!($log_domain, $crate::LogLevel::Warning, $format, $($arg),*);
661 }};
662 ($log_domain:expr, $format:literal $(,)?) => {{
663 $crate::g_log!($log_domain, $crate::LogLevel::Warning, $format);
664 }};
665}
666
667#[macro_export]
695macro_rules! g_message {
696 ($log_domain:expr, $format:literal, $($arg:expr),* $(,)?) => {{
697 $crate::g_log!($log_domain, $crate::LogLevel::Message, $format, $($arg),*);
698 }};
699 ($log_domain:expr, $format:literal $(,)?) => {{
700 $crate::g_log!($log_domain, $crate::LogLevel::Message, $format);
701 }};
702}
703
704#[macro_export]
732macro_rules! g_info {
733 ($log_domain:expr, $format:literal, $($arg:expr),* $(,)?) => {{
734 $crate::g_log!($log_domain, $crate::LogLevel::Info, $format, $($arg),*);
735 }};
736 ($log_domain:expr, $format:literal $(,)?) => {{
737 $crate::g_log!($log_domain, $crate::LogLevel::Info, $format);
738 }};
739}
740
741#[macro_export]
769macro_rules! g_debug {
770 ($log_domain:expr, $format:literal, $($arg:expr),* $(,)?) => {{
771 $crate::g_log!($log_domain, $crate::LogLevel::Debug, $format, $($arg),*);
772 }};
773 ($log_domain:expr, $format:literal $(,)?) => {{
774 $crate::g_log!($log_domain, $crate::LogLevel::Debug, $format);
775 }};
776}
777
778#[doc(hidden)]
779#[macro_export]
780macro_rules! g_print_inner {
781 ($func:ident, $format:expr $(, $arg:expr)* $(,)?) => {{
782 let mut w = $crate::GStringBuilder::default();
783
784 if !std::fmt::Write::write_fmt(&mut w, std::format_args!($format, $($arg),*)).is_err() {
786 unsafe {
787 $crate::ffi::$func(
788 b"%s\0".as_ptr() as *const _,
789 $crate::translate::ToGlibPtr::<*const std::os::raw::c_char>::to_glib_none(
790 &w.into_string()
791 ).0,
792 );
793 }
794 }
795 }};
796}
797
798#[macro_export]
819macro_rules! g_print {
820 ($format:expr $(,$arg:expr)* $(,)?) => {{
821 $crate::g_print_inner!(g_print, $format, $($arg),*);
822 }};
823}
824
825#[macro_export]
846macro_rules! g_printerr {
847 ($format:expr $(, $arg:expr)* $(,)?) => {{
848 $crate::g_print_inner!(g_printerr, $format, $($arg),*);
849 }};
850}
851
852#[macro_export]
889macro_rules! log_structured {
890 ($log_domain:expr, $log_level:expr, {$($key:expr => $format:expr $(,$arg:expr)* $(,)?);+ $(;)?} $(,)?) => {{
891 let log_domain = <Option<&str> as std::convert::From<_>>::from($log_domain);
892 let log_domain_str = log_domain.unwrap_or_default();
893 let level: $crate::LogLevel = $log_level;
894 let field_count =
895 <[()]>::len(&[$($crate::log_structured_inner!(@clear $key)),+])
896 + log_domain.map(|_| 2usize).unwrap_or(1usize)
897 + 3;
898
899 let mut line = [0u8; 32]; let line = {
901 use std::io::Write;
902
903 let mut cursor = std::io::Cursor::new(&mut line[..]);
904 std::write!(&mut cursor, "{}", line!()).unwrap();
905 let pos = cursor.position() as usize;
906 &line[..pos]
907 };
908
909 $crate::log_structured_array(
910 level,
911 &[
912 $crate::LogField::new(
913 $crate::gstr!("PRIORITY"),
914 level.priority().as_bytes(),
915 ),
916 $crate::LogField::new(
917 $crate::gstr!("CODE_FILE"),
918 file!().as_bytes(),
919 ),
920 $crate::LogField::new(
921 $crate::gstr!("CODE_LINE"),
922 line,
923 ),
924 $crate::LogField::new(
925 $crate::gstr!("CODE_FUNC"),
926 $crate::function_name!().as_bytes(),
927 ),
928 $(
929 $crate::LogField::new(
930 $crate::log_structured_inner!(@key $key),
931 $crate::log_structured_inner!(@value $format $(,$arg)*),
932 ),
933 )+
934 $crate::LogField::new(
935 $crate::gstr!("GLIB_DOMAIN"),
936 log_domain_str.as_bytes(),
937 ),
938 ][0..field_count],
939 )
940 }};
941}
942
943#[doc(hidden)]
944#[macro_export]
945macro_rules! log_structured_inner {
946 (@clear $($_:tt)*) => { () };
947 (@key $key:literal) => { $crate::gstr!($key) };
948 (@key $key:expr) => { std::convert::AsRef::<$crate::GStr>::as_ref(&$key) };
949 (@value $value:expr) => { std::convert::AsRef::<[u8]>::as_ref(&$value) };
950 (@value $format:expr $(,$arg:expr)+) => {
951 {
952 let mut builder = $crate::GStringBuilder::default();
953 if std::fmt::Write::write_fmt(&mut builder, format_args!($format, $($arg),+)).is_err() {
954 return;
955 }
956 builder.into_string()
957 }.as_str().as_bytes()
958 };
959}
960
961#[doc(alias = "g_log_structured_array")]
962#[inline]
963pub fn log_structured_array(log_level: LogLevel, fields: &[LogField<'_>]) {
964 unsafe {
965 ffi::g_log_structured_array(
966 log_level.into_glib(),
967 fields.as_ptr() as *const ffi::GLogField,
968 fields.len(),
969 )
970 }
971}
972
973#[doc(alias = "g_log_variant")]
974#[inline]
975pub fn log_variant(log_domain: Option<&str>, log_level: LogLevel, fields: &crate::Variant) {
976 unsafe {
977 ffi::g_log_variant(
978 log_domain.to_glib_none().0,
979 log_level.into_glib(),
980 fields.to_glib_none().0,
981 );
982 }
983}
984
985#[cfg(unix)]
986#[cfg_attr(docsrs, doc(cfg(unix)))]
987#[doc(alias = "g_log_writer_supports_color")]
988#[inline]
989pub fn log_writer_supports_color(output_fd: impl AsFd) -> bool {
990 unsafe {
991 from_glib(ffi::g_log_writer_supports_color(
992 output_fd.as_fd().as_raw_fd(),
993 ))
994 }
995}
996
997#[cfg(unix)]
998#[cfg_attr(docsrs, doc(cfg(unix)))]
999#[doc(alias = "g_log_writer_is_journald")]
1000#[inline]
1001pub fn log_writer_is_journald(output_fd: impl AsFd) -> bool {
1002 unsafe { from_glib(ffi::g_log_writer_is_journald(output_fd.as_fd().as_raw_fd())) }
1003}
1004
1005#[doc(alias = "g_log_writer_format_fields")]
1006#[inline]
1007pub fn log_writer_format_fields(
1008 log_level: LogLevel,
1009 fields: &[LogField<'_>],
1010 use_color: bool,
1011) -> GString {
1012 unsafe {
1013 from_glib_full(ffi::g_log_writer_format_fields(
1014 log_level.into_glib(),
1015 fields.as_ptr() as *const ffi::GLogField,
1016 fields.len(),
1017 use_color.into_glib(),
1018 ))
1019 }
1020}
1021
1022#[doc(alias = "g_log_writer_journald")]
1023#[inline]
1024pub fn log_writer_journald(log_level: LogLevel, fields: &[LogField<'_>]) -> LogWriterOutput {
1025 unsafe {
1026 from_glib(ffi::g_log_writer_journald(
1027 log_level.into_glib(),
1028 fields.as_ptr() as *const ffi::GLogField,
1029 fields.len(),
1030 std::ptr::null_mut(),
1031 ))
1032 }
1033}
1034
1035#[doc(alias = "g_log_writer_standard_streams")]
1036#[inline]
1037pub fn log_writer_standard_streams(
1038 log_level: LogLevel,
1039 fields: &[LogField<'_>],
1040) -> LogWriterOutput {
1041 unsafe {
1042 from_glib(ffi::g_log_writer_standard_streams(
1043 log_level.into_glib(),
1044 fields.as_ptr() as *const ffi::GLogField,
1045 fields.len(),
1046 std::ptr::null_mut(),
1047 ))
1048 }
1049}
1050
1051#[doc(alias = "g_log_writer_default")]
1052#[inline]
1053pub fn log_writer_default(log_level: LogLevel, fields: &[LogField<'_>]) -> LogWriterOutput {
1054 unsafe {
1055 from_glib(ffi::g_log_writer_default(
1056 log_level.into_glib(),
1057 fields.as_ptr() as *const ffi::GLogField,
1058 fields.len(),
1059 std::ptr::null_mut(),
1060 ))
1061 }
1062}
1063
1064#[cfg(feature = "v2_68")]
1075#[cfg_attr(docsrs, doc(cfg(feature = "v2_68")))]
1076#[doc(alias = "g_log_writer_default_set_use_stderr")]
1077#[inline]
1078pub unsafe fn log_writer_default_set_use_stderr(use_stderr: bool) {
1079 unsafe {
1080 ffi::g_log_writer_default_set_use_stderr(use_stderr.into_glib());
1081 }
1082}
1083
1084#[cfg(feature = "v2_68")]
1085#[cfg_attr(docsrs, doc(cfg(feature = "v2_68")))]
1086#[doc(alias = "g_log_writer_default_would_drop")]
1087#[inline]
1088pub fn log_writer_default_would_drop(log_level: LogLevel, log_domain: Option<&str>) -> bool {
1089 unsafe {
1090 from_glib(ffi::g_log_writer_default_would_drop(
1091 log_level.into_glib(),
1092 log_domain.to_glib_none().0,
1093 ))
1094 }
1095}
1096
1097#[cfg(feature = "v2_80")]
1098#[cfg_attr(docsrs, doc(cfg(feature = "v2_80")))]
1099#[doc(alias = "g_log_writer_default_set_debug_domains")]
1100#[inline]
1101pub fn log_writer_default_set_debug_domains(domains: &StrVRef) {
1102 unsafe { ffi::g_log_writer_default_set_debug_domains(domains.to_glib_none().0) }
1103}