Compare commits

...

3 Commits

Author SHA1 Message Date
Luke Street 4ddceaec0e Version 0.1.5 2023-02-02 16:00:00 -05:00
Luke Street 53a5797881 Update MSRV to 1.58 2023-02-02 15:59:55 -05:00
Luke Street c595c9dc86 Fix argument-less function pointer arguments with return type 2023-02-02 15:56:09 -05:00
5 changed files with 33 additions and 25 deletions

View File

@ -1,6 +1,6 @@
[package]
name = "cwdemangle"
version = "0.1.4"
version = "0.1.5"
edition = "2018"
authors = ["Luke Street <luke@street.dev>"]
license = "MIT OR Apache-2.0"
@ -10,6 +10,7 @@ description = """
CodeWarrior C++ symbol demangler
"""
categories = ["command-line-utilities"]
rust-version = "1.58"
[dependencies]
argh = "0.1.8"

View File

@ -6,7 +6,7 @@
[crates.io]: https://crates.io/crates/cwdemangle
[Api Rustdoc]: https://img.shields.io/badge/api-rustdoc-blue.svg
[rustdoc]: https://docs.rs/cwdemangle
[Rust Version]: https://img.shields.io/badge/rust-1.51+-blue.svg?maxAge=3600
[Rust Version]: https://img.shields.io/badge/rust-1.58+-blue.svg?maxAge=3600
A CodeWarrior C++ symbol demangler.

View File

@ -42,8 +42,7 @@ where T: FromArgs
};
Err(argh::EarlyExit {
output: format!(
"{} -V, --version print version information and exit",
help
"{help} -V, --version print version information and exit"
),
status: Ok(()),
})

View File

@ -86,14 +86,14 @@ fn demangle_name<'a>(str: &'a str, options: &DemangleOptions) -> Option<(String,
let (size, rest) = parse_digits(str)?;
// hack for template argument constants
if rest.is_empty() || rest.starts_with(',') {
let out = format!("{}", size);
let out = format!("{size}");
return Some((out.clone(), out, rest));
}
if rest.len() < size {
return None;
}
let (name, args) = demangle_template_args(&rest[..size], options)?;
Some((name.to_string(), format!("{}{}", name, args), &rest[size..]))
Some((name.to_string(), format!("{name}{args}"), &rest[size..]))
}
fn demangle_qualified_name<'a>(
@ -142,7 +142,7 @@ fn demangle_arg<'a>(
if str.starts_with('M') {
is_member = true;
let (_, member, rest) = demangle_qualified_name(&str[1..], options)?;
pre = format!("{}::*{}", member, pre);
pre = format!("{member}::*{pre}");
if !rest.starts_with('F') {
return None;
}
@ -162,7 +162,7 @@ fn demangle_arg<'a>(
}
} else if post.starts_with('*') {
post = post[1..].trim_start().to_string();
pre = format!("*{}", pre);
pre = format!("*{pre}");
} else {
return None;
}
@ -172,8 +172,8 @@ fn demangle_arg<'a>(
}
let (ret_pre, ret_post, rest) = demangle_arg(&rest[1..], options)?;
let const_str = if const_member { " const" } else { "" };
let res_pre = format!("{} ({}{}", ret_pre, pre, post);
let res_post = format!(")({}){}{}", args, const_str, ret_post);
let res_pre = format!("{ret_pre} ({pre}{post}");
let res_post = format!(")({args}){const_str}{ret_post}");
return Some((res_pre, res_post, rest));
}
if let Some(rest) = str.strip_prefix('A') {
@ -183,10 +183,10 @@ fn demangle_arg<'a>(
}
let (arg_pre, arg_post, rest) = demangle_arg(&rest[1..], options)?;
if !post.is_empty() {
post = format!("({})", post);
post = format!("({post})");
}
result = format!("{}{}{}", pre, arg_pre, post);
let ret_post = format!("[{}]{}", count, arg_post);
result = format!("{pre}{arg_pre}{post}");
let ret_post = format!("[{count}]{arg_post}");
return Some((result, ret_post, rest));
}
result.push_str(match str.chars().next()? {
@ -201,6 +201,7 @@ fn demangle_arg<'a>(
'w' => "wchar_t",
'v' => "void",
'e' => "...",
'_' => return Some((result, String::new(), rest)),
_ => return None,
});
result += post.as_str();
@ -234,13 +235,13 @@ fn demangle_special_function(
) -> Option<String> {
if let Some(rest) = str.strip_prefix("op") {
let (arg_pre, arg_post, _) = demangle_arg(rest, options)?;
return Some(format!("operator {}{}", arg_pre, arg_post));
return Some(format!("operator {arg_pre}{arg_post}"));
}
let (op, args) = demangle_template_args(str, options)?;
Some(format!(
"{}{}",
match op {
"dt" => return Some(format!("~{}{}", class_name, args)),
"dt" => return Some(format!("~{class_name}{args}")),
"ct" => class_name,
"nw" => "operator new",
"nwa" => "operator new[]",
@ -285,7 +286,7 @@ fn demangle_special_function(
"cl" => "operator()",
"vc" => "operator[]",
"vt" => "__vtable",
_ => return Some(format!("__{}{}", op, args)),
_ => return Some(format!("__{op}{args}")),
},
args
))
@ -336,7 +337,7 @@ pub fn demangle(mut str: &str, options: &DemangleOptions) -> Option<String> {
}
} else {
let (name, args) = demangle_template_args(fn_name_out, options)?;
fn_name = format!("{}{}", name, args);
fn_name = format!("{name}{args}");
}
// Handle old static function variables (GC CW)
@ -349,7 +350,7 @@ pub fn demangle(mut str: &str, options: &DemangleOptions) -> Option<String> {
}
if var == "init" {
// Sadly, $localstatic doesn't provide the variable name in guard/init
static_var = format!("{} guard", var_type);
static_var = format!("{var_type} guard");
} else {
static_var = var.to_string();
}
@ -376,9 +377,9 @@ pub fn demangle(mut str: &str, options: &DemangleOptions) -> Option<String> {
str = &str[1..];
let (args, rest) = demangle_function_args(str, options)?;
if options.omit_empty_parameters && args == "void" {
fn_name = format!("{}()", fn_name);
fn_name = format!("{fn_name}()");
} else {
fn_name = format!("{}({})", fn_name, args);
fn_name = format!("{fn_name}({args})");
}
str = rest;
}
@ -393,16 +394,16 @@ pub fn demangle(mut str: &str, options: &DemangleOptions) -> Option<String> {
return None;
}
if cnst {
fn_name = format!("{} const", fn_name);
fn_name = format!("{fn_name} const");
}
if !qualified.is_empty() {
fn_name = format!("{}::{}", qualified, fn_name);
fn_name = format!("{qualified}::{fn_name}");
}
if !return_type_pre.is_empty() {
fn_name = format!("{} {}{}", return_type_pre, fn_name, return_type_post);
fn_name = format!("{return_type_pre} {fn_name}{return_type_post}");
}
if !static_var.is_empty() {
fn_name = format!("{}::{}", fn_name, static_var);
fn_name = format!("{fn_name}::{static_var}");
}
Some(fn_name)
}
@ -678,6 +679,13 @@ mod tests {
demangle("test__FRCPCPCi", &options),
Some("test(const int* const* const&)".to_string()),
);
assert_eq!(
demangle(
"__ct__Q34nw4r2ut14CharStrmReaderFMQ34nw4r2ut14CharStrmReaderFPCvPv_Us",
&options
),
Some("nw4r::ut::CharStrmReader::CharStrmReader(unsigned short (nw4r::ut::CharStrmReader::*)())".to_string())
);
}
#[test]

View File

@ -21,7 +21,7 @@ fn main() -> Result<(), &'static str> {
return if let Some(symbol) =
demangle(args.symbol.as_str(), &DemangleOptions { omit_empty_parameters: !args.keep_void })
{
println!("{}", symbol);
println!("{symbol}");
Ok(())
} else {
Err("Failed to demangle symbol")