|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2024 Google LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file or at
|
|
|
|
// https://developers.google.com/open-source/licenses/bsd
|
|
|
|
|
|
|
|
use super::{upb_MiniTable, RawMessage};
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
/// Returns the minimum needed length (excluding NULL) that `buf` has to be
|
|
|
|
/// to hold the `msg`s debug string.
|
|
|
|
///
|
|
|
|
/// SAFETY:
|
|
|
|
/// - `msg` is pointing at a valid upb_Message with associated minitable
|
|
|
|
/// `mt`
|
|
|
|
/// - `buf` is legally writable for `size` bytes (`buf` may be nullptr if
|
|
|
|
/// `size` is 0)
|
|
|
|
fn upb_DebugString(
|
|
|
|
msg: RawMessage,
|
|
|
|
mt: *const upb_MiniTable,
|
|
|
|
options: i32,
|
|
|
|
buf: *mut u8,
|
|
|
|
size: usize,
|
|
|
|
) -> usize;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
#[repr(i32)]
|
|
|
|
enum Options {
|
|
|
|
// When set, prints everything on a single line.
|
|
|
|
SingleLine = 1,
|
|
|
|
|
|
|
|
// When set, unknown fields are not printed.
|
|
|
|
SkipUnknown = 2,
|
|
|
|
|
|
|
|
// When set, maps are *not* sorted (this avoids allocating tmp mem).
|
|
|
|
NoSortMaps = 4,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a string of field number to value entries of a message.
|
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// - `mt` must correspond to the `msg`s minitable.
|
|
|
|
pub unsafe fn debug_string(msg: RawMessage, mt: *const upb_MiniTable) -> String {
|
|
|
|
// Only find out the length first to then allocate a buffer of the minimum size
|
|
|
|
// needed.
|
|
|
|
// SAFETY:
|
|
|
|
// - `msg` is a legally dereferencable upb_Message whose associated minitable is
|
|
|
|
// `mt`
|
|
|
|
// - `buf` is nullptr and `buf_len` is 0
|
|
|
|
let len =
|
|
|
|
unsafe { upb_DebugString(msg, mt, Options::NoSortMaps as i32, core::ptr::null_mut(), 0) };
|
|
|
|
assert!(len < isize::MAX as usize);
|
|
|
|
// +1 for the trailing NULL
|
|
|
|
let mut buf = vec![0u8; len + 1];
|
|
|
|
// SAFETY:
|
|
|
|
// - `msg` is a legally dereferencable upb_Message whose associated minitable is
|
|
|
|
// `mt`
|
|
|
|
// - `buf` is legally writable for 'buf_len' bytes
|
|
|
|
let written_len = unsafe {
|
|
|
|
upb_DebugString(msg, mt, Options::NoSortMaps as i32, buf.as_mut_ptr(), buf.len())
|
|
|
|
};
|
|
|
|
assert_eq!(len, written_len);
|
|
|
|
String::from_utf8_lossy(buf.as_slice()).to_string()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
use googletest::gtest;
|
|
|
|
|
|
|
|
#[gtest]
|
|
|
|
fn assert_text_linked() {
|
|
|
|
use crate::assert_linked;
|
|
|
|
assert_linked!(upb_DebugString);
|
|
|
|
}
|
|
|
|
}
|