mirror of
https://github.com/rust-lang/rust-analyzer
synced 2024-12-29 06:23:25 +00:00
271 lines
8.4 KiB
Rust
271 lines
8.4 KiB
Rust
//! Display and pretty printing routines.
|
|
|
|
use std::fmt::{self, Write};
|
|
|
|
use hir_expand::mod_path::PathKind;
|
|
use intern::Interned;
|
|
use itertools::Itertools;
|
|
|
|
use crate::{
|
|
db::DefDatabase,
|
|
lang_item::LangItemTarget,
|
|
path::{GenericArg, GenericArgs, Path},
|
|
type_ref::{Mutability, TraitBoundModifier, TypeBound, TypeRef},
|
|
};
|
|
|
|
pub(crate) fn print_path(db: &dyn DefDatabase, path: &Path, buf: &mut dyn Write) -> fmt::Result {
|
|
if let Path::LangItem(it, s) = path {
|
|
write!(buf, "builtin#lang(")?;
|
|
match *it {
|
|
LangItemTarget::ImplDef(it) => write!(buf, "{it:?}")?,
|
|
LangItemTarget::EnumId(it) => {
|
|
write!(buf, "{}", db.enum_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::Function(it) => {
|
|
write!(buf, "{}", db.function_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::Static(it) => {
|
|
write!(buf, "{}", db.static_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::Struct(it) => {
|
|
write!(buf, "{}", db.struct_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::Union(it) => {
|
|
write!(buf, "{}", db.union_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::TypeAlias(it) => {
|
|
write!(buf, "{}", db.type_alias_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::Trait(it) => {
|
|
write!(buf, "{}", db.trait_data(it).name.display(db.upcast()))?
|
|
}
|
|
LangItemTarget::EnumVariant(it) => {
|
|
write!(buf, "{}", db.enum_variant_data(it).name.display(db.upcast()))?
|
|
}
|
|
}
|
|
|
|
if let Some(s) = s {
|
|
write!(buf, "::{}", s.display(db.upcast()))?;
|
|
}
|
|
return write!(buf, ")");
|
|
}
|
|
match path.type_anchor() {
|
|
Some(anchor) => {
|
|
write!(buf, "<")?;
|
|
print_type_ref(db, anchor, buf)?;
|
|
write!(buf, ">::")?;
|
|
}
|
|
None => match path.kind() {
|
|
PathKind::Plain => {}
|
|
PathKind::Super(0) => write!(buf, "self")?,
|
|
PathKind::Super(n) => {
|
|
for i in 0..*n {
|
|
if i == 0 {
|
|
buf.write_str("super")?;
|
|
} else {
|
|
buf.write_str("::super")?;
|
|
}
|
|
}
|
|
}
|
|
PathKind::Crate => write!(buf, "crate")?,
|
|
PathKind::Abs => {}
|
|
PathKind::DollarCrate(_) => write!(buf, "$crate")?,
|
|
},
|
|
}
|
|
|
|
for (i, segment) in path.segments().iter().enumerate() {
|
|
if i != 0 || !matches!(path.kind(), PathKind::Plain) {
|
|
write!(buf, "::")?;
|
|
}
|
|
|
|
write!(buf, "{}", segment.name.display(db.upcast()))?;
|
|
if let Some(generics) = segment.args_and_bindings {
|
|
write!(buf, "::<")?;
|
|
print_generic_args(db, generics, buf)?;
|
|
|
|
write!(buf, ">")?;
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub(crate) fn print_generic_args(
|
|
db: &dyn DefDatabase,
|
|
generics: &GenericArgs,
|
|
buf: &mut dyn Write,
|
|
) -> fmt::Result {
|
|
let mut first = true;
|
|
let args = if generics.has_self_type {
|
|
let (self_ty, args) = generics.args.split_first().unwrap();
|
|
write!(buf, "Self=")?;
|
|
print_generic_arg(db, self_ty, buf)?;
|
|
first = false;
|
|
args
|
|
} else {
|
|
&generics.args
|
|
};
|
|
for arg in args {
|
|
if !first {
|
|
write!(buf, ", ")?;
|
|
}
|
|
first = false;
|
|
print_generic_arg(db, arg, buf)?;
|
|
}
|
|
for binding in generics.bindings.iter() {
|
|
if !first {
|
|
write!(buf, ", ")?;
|
|
}
|
|
first = false;
|
|
write!(buf, "{}", binding.name.display(db.upcast()))?;
|
|
if !binding.bounds.is_empty() {
|
|
write!(buf, ": ")?;
|
|
print_type_bounds(db, &binding.bounds, buf)?;
|
|
}
|
|
if let Some(ty) = &binding.type_ref {
|
|
write!(buf, " = ")?;
|
|
print_type_ref(db, ty, buf)?;
|
|
}
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
pub(crate) fn print_generic_arg(
|
|
db: &dyn DefDatabase,
|
|
arg: &GenericArg,
|
|
buf: &mut dyn Write,
|
|
) -> fmt::Result {
|
|
match arg {
|
|
GenericArg::Type(ty) => print_type_ref(db, ty, buf),
|
|
GenericArg::Const(c) => write!(buf, "{}", c.display(db.upcast())),
|
|
GenericArg::Lifetime(lt) => write!(buf, "{}", lt.name.display(db.upcast())),
|
|
}
|
|
}
|
|
|
|
pub(crate) fn print_type_ref(
|
|
db: &dyn DefDatabase,
|
|
type_ref: &TypeRef,
|
|
buf: &mut dyn Write,
|
|
) -> fmt::Result {
|
|
// FIXME: deduplicate with `HirDisplay` impl
|
|
match type_ref {
|
|
TypeRef::Never => write!(buf, "!")?,
|
|
TypeRef::Placeholder => write!(buf, "_")?,
|
|
TypeRef::Tuple(fields) => {
|
|
write!(buf, "(")?;
|
|
for (i, field) in fields.iter().enumerate() {
|
|
if i != 0 {
|
|
write!(buf, ", ")?;
|
|
}
|
|
print_type_ref(db, field, buf)?;
|
|
}
|
|
write!(buf, ")")?;
|
|
}
|
|
TypeRef::Path(path) => print_path(db, path, buf)?,
|
|
TypeRef::RawPtr(pointee, mtbl) => {
|
|
let mtbl = match mtbl {
|
|
Mutability::Shared => "*const",
|
|
Mutability::Mut => "*mut",
|
|
};
|
|
write!(buf, "{mtbl} ")?;
|
|
print_type_ref(db, pointee, buf)?;
|
|
}
|
|
TypeRef::Reference(pointee, lt, mtbl) => {
|
|
let mtbl = match mtbl {
|
|
Mutability::Shared => "",
|
|
Mutability::Mut => "mut ",
|
|
};
|
|
write!(buf, "&")?;
|
|
if let Some(lt) = lt {
|
|
write!(buf, "{} ", lt.name.display(db.upcast()))?;
|
|
}
|
|
write!(buf, "{mtbl}")?;
|
|
print_type_ref(db, pointee, buf)?;
|
|
}
|
|
TypeRef::Array(elem, len) => {
|
|
write!(buf, "[")?;
|
|
print_type_ref(db, elem, buf)?;
|
|
write!(buf, "; {}]", len.display(db.upcast()))?;
|
|
}
|
|
TypeRef::Slice(elem) => {
|
|
write!(buf, "[")?;
|
|
print_type_ref(db, elem, buf)?;
|
|
write!(buf, "]")?;
|
|
}
|
|
TypeRef::Fn(args_and_ret, varargs, is_unsafe, abi) => {
|
|
let ((_, return_type), args) =
|
|
args_and_ret.split_last().expect("TypeRef::Fn is missing return type");
|
|
if *is_unsafe {
|
|
write!(buf, "unsafe ")?;
|
|
}
|
|
if let Some(abi) = abi {
|
|
buf.write_str("extern ")?;
|
|
buf.write_str(abi)?;
|
|
buf.write_char(' ')?;
|
|
}
|
|
write!(buf, "fn(")?;
|
|
for (i, (_, typeref)) in args.iter().enumerate() {
|
|
if i != 0 {
|
|
write!(buf, ", ")?;
|
|
}
|
|
print_type_ref(db, typeref, buf)?;
|
|
}
|
|
if *varargs {
|
|
if !args.is_empty() {
|
|
write!(buf, ", ")?;
|
|
}
|
|
write!(buf, "...")?;
|
|
}
|
|
write!(buf, ") -> ")?;
|
|
print_type_ref(db, return_type, buf)?;
|
|
}
|
|
TypeRef::Macro(_ast_id) => {
|
|
write!(buf, "<macro>")?;
|
|
}
|
|
TypeRef::Error => write!(buf, "{{unknown}}")?,
|
|
TypeRef::ImplTrait(bounds) => {
|
|
write!(buf, "impl ")?;
|
|
print_type_bounds(db, bounds, buf)?;
|
|
}
|
|
TypeRef::DynTrait(bounds) => {
|
|
write!(buf, "dyn ")?;
|
|
print_type_bounds(db, bounds, buf)?;
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub(crate) fn print_type_bounds(
|
|
db: &dyn DefDatabase,
|
|
bounds: &[Interned<TypeBound>],
|
|
buf: &mut dyn Write,
|
|
) -> fmt::Result {
|
|
for (i, bound) in bounds.iter().enumerate() {
|
|
if i != 0 {
|
|
write!(buf, " + ")?;
|
|
}
|
|
|
|
match bound.as_ref() {
|
|
TypeBound::Path(path, modifier) => {
|
|
match modifier {
|
|
TraitBoundModifier::None => (),
|
|
TraitBoundModifier::Maybe => write!(buf, "?")?,
|
|
}
|
|
print_path(db, path, buf)?;
|
|
}
|
|
TypeBound::ForLifetime(lifetimes, path) => {
|
|
write!(
|
|
buf,
|
|
"for<{}> ",
|
|
lifetimes.iter().map(|it| it.display(db.upcast())).format(", ")
|
|
)?;
|
|
print_path(db, path, buf)?;
|
|
}
|
|
TypeBound::Lifetime(lt) => write!(buf, "{}", lt.name.display(db.upcast()))?,
|
|
TypeBound::Error => write!(buf, "{{unknown}}")?,
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|