Add render configs for memory layout hovers

This commit is contained in:
Lukas Wirth 2023-05-30 16:20:01 +02:00
parent 76d86502f7
commit 3c862507b9
12 changed files with 400 additions and 163 deletions

View file

@ -45,7 +45,7 @@ use hir_def::{
hir::{BindingAnnotation, BindingId, ExprOrPatId, LabelId, Pat}, hir::{BindingAnnotation, BindingId, ExprOrPatId, LabelId, Pat},
item_tree::ItemTreeNode, item_tree::ItemTreeNode,
lang_item::LangItemTarget, lang_item::LangItemTarget,
layout::{self, ReprOptions}, layout::{self, ReprOptions, TargetDataLayout},
macro_id_to_def_id, macro_id_to_def_id,
nameres::{self, diagnostics::DefDiagnostic, ModuleOrigin}, nameres::{self, diagnostics::DefDiagnostic, ModuleOrigin},
per_ns::PerNs, per_ns::PerNs,
@ -62,7 +62,7 @@ use hir_ty::{
consteval::{try_const_usize, unknown_const_as_generic, ConstEvalError, ConstExt}, consteval::{try_const_usize, unknown_const_as_generic, ConstEvalError, ConstExt},
diagnostics::BodyValidationDiagnostic, diagnostics::BodyValidationDiagnostic,
display::HexifiedConst, display::HexifiedConst,
layout::{Layout as TyLayout, LayoutError, RustcEnumVariantIdx, TagEncoding}, layout::{Layout as TyLayout, RustcEnumVariantIdx, TagEncoding},
method_resolution::{self, TyFingerprint}, method_resolution::{self, TyFingerprint},
mir::{self, interpret_mir}, mir::{self, interpret_mir},
primitive::UintTy, primitive::UintTy,
@ -133,6 +133,7 @@ pub use {
}, },
hir_ty::{ hir_ty::{
display::{ClosureStyle, HirDisplay, HirDisplayError, HirWrite}, display::{ClosureStyle, HirDisplay, HirDisplayError, HirWrite},
layout::LayoutError,
mir::MirEvalError, mir::MirEvalError,
PointerCast, Safety, PointerCast, Safety,
}, },
@ -962,7 +963,8 @@ impl Field {
} }
pub fn layout(&self, db: &dyn HirDatabase) -> Result<Layout, LayoutError> { pub fn layout(&self, db: &dyn HirDatabase) -> Result<Layout, LayoutError> {
db.layout_of_ty(self.ty(db).ty.clone(), self.parent.module(db).krate().into()).map(Layout) db.layout_of_ty(self.ty(db).ty.clone(), self.parent.module(db).krate().into())
.map(|layout| Layout(layout, db.target_data_layout(self.krate(db).into()).unwrap()))
} }
pub fn parent_def(&self, _db: &dyn HirDatabase) -> VariantDef { pub fn parent_def(&self, _db: &dyn HirDatabase) -> VariantDef {
@ -1135,23 +1137,8 @@ impl Enum {
self.variants(db).iter().any(|v| !matches!(v.kind(db), StructKind::Unit)) self.variants(db).iter().any(|v| !matches!(v.kind(db), StructKind::Unit))
} }
pub fn layout(self, db: &dyn HirDatabase) -> Result<(Layout, usize), LayoutError> { pub fn layout(self, db: &dyn HirDatabase) -> Result<Layout, LayoutError> {
let layout = Adt::from(self).layout(db)?; Adt::from(self).layout(db)
let tag_size =
if let layout::Variants::Multiple { tag, tag_encoding, .. } = &layout.0.variants {
match tag_encoding {
TagEncoding::Direct => {
let target_data_layout = db
.target_data_layout(self.module(db).krate().id)
.ok_or(LayoutError::TargetLayoutNotAvailable)?;
tag.size(&*target_data_layout).bytes_usize()
}
TagEncoding::Niche { .. } => 0,
}
} else {
0
};
Ok((layout, tag_size))
} }
} }
@ -1214,19 +1201,16 @@ impl Variant {
db.const_eval_discriminant(self.into()) db.const_eval_discriminant(self.into())
} }
/// Return layout of the variant and tag size of the parent enum. pub fn layout(&self, db: &dyn HirDatabase) -> Result<Layout, LayoutError> {
pub fn layout(&self, db: &dyn HirDatabase) -> Result<(Layout, usize), LayoutError> {
let parent_enum = self.parent_enum(db); let parent_enum = self.parent_enum(db);
let (parent_layout, tag_size) = parent_enum.layout(db)?; let parent_layout = parent_enum.layout(db)?;
Ok(( Ok(match &parent_layout.0.variants {
match &parent_layout.0.variants { layout::Variants::Multiple { variants, .. } => Layout(
layout::Variants::Multiple { variants, .. } => { Arc::new(variants[RustcEnumVariantIdx(self.id)].clone()),
Layout(Arc::new(variants[RustcEnumVariantIdx(self.id)].clone())) db.target_data_layout(parent_enum.krate(db).into()).unwrap(),
} ),
_ => parent_layout, _ => parent_layout,
}, })
tag_size,
))
} }
} }
@ -1259,7 +1243,9 @@ impl Adt {
if db.generic_params(self.into()).iter().count() != 0 { if db.generic_params(self.into()).iter().count() != 0 {
return Err(LayoutError::HasPlaceholder); return Err(LayoutError::HasPlaceholder);
} }
db.layout_of_adt(self.into(), Substitution::empty(Interner), self.krate(db).id).map(Layout) let krate = self.krate(db).id;
db.layout_of_adt(self.into(), Substitution::empty(Interner), krate)
.map(|layout| Layout(layout, db.target_data_layout(krate).unwrap()))
} }
/// Turns this ADT into a type. Any type parameters of the ADT will be /// Turns this ADT into a type. Any type parameters of the ADT will be
@ -4244,7 +4230,8 @@ impl Type {
} }
pub fn layout(&self, db: &dyn HirDatabase) -> Result<Layout, LayoutError> { pub fn layout(&self, db: &dyn HirDatabase) -> Result<Layout, LayoutError> {
db.layout_of_ty(self.ty.clone(), self.env.krate).map(Layout) db.layout_of_ty(self.ty.clone(), self.env.krate)
.map(|layout| Layout(layout, db.target_data_layout(self.env.krate).unwrap()))
} }
} }
@ -4356,7 +4343,7 @@ fn closure_source(db: &dyn HirDatabase, closure: ClosureId) -> Option<ast::Closu
} }
#[derive(Clone, Debug, Eq, PartialEq)] #[derive(Clone, Debug, Eq, PartialEq)]
pub struct Layout(Arc<TyLayout>); pub struct Layout(Arc<TyLayout>, Arc<TargetDataLayout>);
impl Layout { impl Layout {
pub fn size(&self) -> u64 { pub fn size(&self) -> u64 {
@ -4367,8 +4354,8 @@ impl Layout {
self.0.align.abi.bytes() self.0.align.abi.bytes()
} }
pub fn niches(&self, db: &dyn HirDatabase, krate: Crate) -> Option<u128> { pub fn niches(&self) -> Option<u128> {
Some(self.0.largest_niche?.available(&*db.target_data_layout(krate.id)?)) Some(self.0.largest_niche?.available(&*self.1))
} }
pub fn field_offset(&self, idx: usize) -> Option<u64> { pub fn field_offset(&self, idx: usize) -> Option<u64> {
@ -4382,6 +4369,19 @@ impl Layout {
layout::FieldsShape::Arbitrary { ref offsets, .. } => Some(offsets.get(idx)?.bytes()), layout::FieldsShape::Arbitrary { ref offsets, .. } => Some(offsets.get(idx)?.bytes()),
} }
} }
pub fn enum_tag_size(&self) -> Option<usize> {
let tag_size =
if let layout::Variants::Multiple { tag, tag_encoding, .. } = &self.0.variants {
match tag_encoding {
TagEncoding::Direct => tag.size(&*self.1).bytes_usize(),
TagEncoding::Niche { .. } => 0,
}
} else {
return None;
};
Some(tag_size)
}
} }
#[derive(Copy, Clone, Debug, Eq, PartialEq)] #[derive(Copy, Clone, Debug, Eq, PartialEq)]

View file

@ -55,7 +55,8 @@ pub(crate) fn convert_named_struct_to_tuple_struct(
// XXX: We don't currently provide this assist for struct definitions inside macros, but if we // XXX: We don't currently provide this assist for struct definitions inside macros, but if we
// are to lift this limitation, don't forget to make `edit_struct_def()` consider macro files // are to lift this limitation, don't forget to make `edit_struct_def()` consider macro files
// too. // too.
let strukt = ctx.find_node_at_offset::<Either<ast::Struct, ast::Variant>>()?; let name = ctx.find_node_at_offset::<ast::Name>()?;
let strukt = name.syntax().parent().and_then(<Either<ast::Struct, ast::Variant>>::cast)?;
let field_list = strukt.as_ref().either(|s| s.field_list(), |v| v.field_list())?; let field_list = strukt.as_ref().either(|s| s.field_list(), |v| v.field_list())?;
let record_fields = match field_list { let record_fields = match field_list {
ast::FieldList::RecordFieldList(it) => it, ast::FieldList::RecordFieldList(it) => it,

View file

@ -50,7 +50,8 @@ pub(crate) fn convert_tuple_struct_to_named_struct(
acc: &mut Assists, acc: &mut Assists,
ctx: &AssistContext<'_>, ctx: &AssistContext<'_>,
) -> Option<()> { ) -> Option<()> {
let strukt = ctx.find_node_at_offset::<Either<ast::Struct, ast::Variant>>()?; let name = ctx.find_node_at_offset::<ast::Name>()?;
let strukt = name.syntax().parent().and_then(<Either<ast::Struct, ast::Variant>>::cast)?;
let field_list = strukt.as_ref().either(|s| s.field_list(), |v| v.field_list())?; let field_list = strukt.as_ref().either(|s| s.field_list(), |v| v.field_list())?;
let tuple_fields = match field_list { let tuple_fields = match field_list {
ast::FieldList::TupleFieldList(it) => it, ast::FieldList::TupleFieldList(it) => it,

View file

@ -273,8 +273,9 @@ fn assist_order_field_struct() {
assert_eq!(assists.next().expect("expected assist").label, "Generate a getter method"); assert_eq!(assists.next().expect("expected assist").label, "Generate a getter method");
assert_eq!(assists.next().expect("expected assist").label, "Generate a mut getter method"); assert_eq!(assists.next().expect("expected assist").label, "Generate a mut getter method");
assert_eq!(assists.next().expect("expected assist").label, "Generate a setter method"); assert_eq!(assists.next().expect("expected assist").label, "Generate a setter method");
assert_eq!(assists.next().expect("expected assist").label, "Convert to tuple struct");
assert_eq!(assists.next().expect("expected assist").label, "Add `#[derive]`"); assert_eq!(assists.next().expect("expected assist").label, "Add `#[derive]`");
assert_eq!(assists.next().expect("expected assist").label, "Generate `new`");
assert_eq!(assists.next().map(|it| it.label.to_string()), None);
} }
#[test] #[test]

View file

@ -27,12 +27,27 @@ use crate::{
#[derive(Clone, Debug, PartialEq, Eq)] #[derive(Clone, Debug, PartialEq, Eq)]
pub struct HoverConfig { pub struct HoverConfig {
pub links_in_hover: bool, pub links_in_hover: bool,
pub memory_layout: bool, pub memory_layout: Option<MemoryLayoutHoverConfig>,
pub documentation: bool, pub documentation: bool,
pub keywords: bool, pub keywords: bool,
pub format: HoverDocFormat, pub format: HoverDocFormat,
} }
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub struct MemoryLayoutHoverConfig {
pub size: Option<MemoryLayoutHoverRenderKind>,
pub offset: Option<MemoryLayoutHoverRenderKind>,
pub alignment: Option<MemoryLayoutHoverRenderKind>,
pub niches: bool,
}
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub enum MemoryLayoutHoverRenderKind {
Decimal,
Hexadecimal,
Both,
}
#[derive(Clone, Debug, PartialEq, Eq)] #[derive(Clone, Debug, PartialEq, Eq)]
pub enum HoverDocFormat { pub enum HoverDocFormat {
Markdown, Markdown,

View file

@ -3,8 +3,8 @@ use std::fmt::Display;
use either::Either; use either::Either;
use hir::{ use hir::{
Adt, AsAssocItem, AttributeTemplate, CaptureKind, HasAttrs, HasCrate, HasSource, HirDisplay, Adt, AsAssocItem, AttributeTemplate, CaptureKind, HasAttrs, HasSource, HirDisplay, Layout,
Layout, Semantics, TypeInfo, LayoutError, Semantics, TypeInfo,
}; };
use ide_db::{ use ide_db::{
base_db::SourceDatabase, base_db::SourceDatabase,
@ -27,7 +27,8 @@ use syntax::{
use crate::{ use crate::{
doc_links::{remove_links, rewrite_links}, doc_links::{remove_links, rewrite_links},
hover::walk_and_push_ty, hover::walk_and_push_ty,
HoverAction, HoverConfig, HoverResult, Markup, HoverAction, HoverConfig, HoverResult, Markup, MemoryLayoutHoverConfig,
MemoryLayoutHoverRenderKind,
}; };
pub(super) fn type_info_of( pub(super) fn type_info_of(
@ -393,32 +394,27 @@ pub(super) fn definition(
let mod_path = definition_mod_path(db, &def); let mod_path = definition_mod_path(db, &def);
let (label, docs) = match def { let (label, docs) = match def {
Definition::Macro(it) => label_and_docs(db, it), Definition::Macro(it) => label_and_docs(db, it),
Definition::Field(it) => label_and_layout_info_and_docs(db, it, config, |&it| { Definition::Field(it) => label_and_layout_info_and_docs(
db,
it,
config,
|&it| it.layout(db),
|_| {
let var_def = it.parent_def(db); let var_def = it.parent_def(db);
let id = it.index(); let id = it.index();
let layout = it.layout(db).ok()?; match var_def {
let offset = match var_def { hir::VariantDef::Struct(s) => {
hir::VariantDef::Struct(s) => Adt::from(s) Adt::from(s).layout(db).ok().and_then(|layout| layout.field_offset(id))
.layout(db) }
.ok()
.and_then(|layout| Some(format!(", offset = {:#X}", layout.field_offset(id)?))),
_ => None, _ => None,
}; }
let niches = niches(db, it, &layout).unwrap_or_default(); },
Some(format!( ),
"size = {:#X}, align = {:#X}{}{niches}",
layout.size(),
layout.align(),
offset.as_deref().unwrap_or_default()
))
}),
Definition::Module(it) => label_and_docs(db, it), Definition::Module(it) => label_and_docs(db, it),
Definition::Function(it) => label_and_docs(db, it), Definition::Function(it) => label_and_docs(db, it),
Definition::Adt(it) => label_and_layout_info_and_docs(db, it, config, |&it| { Definition::Adt(it) => {
let layout = it.layout(db).ok()?; label_and_layout_info_and_docs(db, it, config, |&it| it.layout(db), |_| None)
let niches = niches(db, it, &layout).unwrap_or_default(); }
Some(format!("size = {:#X}, align = {:#X}{niches}", layout.size(), layout.align()))
}),
Definition::Variant(it) => label_value_and_layout_info_and_docs( Definition::Variant(it) => label_value_and_layout_info_and_docs(
db, db,
it, it,
@ -435,16 +431,8 @@ pub(super) fn definition(
None None
} }
}, },
|&it| { |it| it.layout(db),
let (layout, tag_size) = it.layout(db).ok()?; |layout| layout.enum_tag_size(),
let size = layout.size() as usize - tag_size;
if size == 0 {
// There is no value in showing layout info for fieldless variants
return None;
}
let niches = niches(db, it, &layout).unwrap_or_default();
Some(format!("size = {:#X}{niches}", layout.size()))
},
), ),
Definition::Const(it) => label_value_and_docs(db, it, |it| { Definition::Const(it) => label_value_and_docs(db, it, |it| {
let body = it.render_eval(db); let body = it.render_eval(db);
@ -470,11 +458,9 @@ pub(super) fn definition(
}), }),
Definition::Trait(it) => label_and_docs(db, it), Definition::Trait(it) => label_and_docs(db, it),
Definition::TraitAlias(it) => label_and_docs(db, it), Definition::TraitAlias(it) => label_and_docs(db, it),
Definition::TypeAlias(it) => label_and_layout_info_and_docs(db, it, config, |&it| { Definition::TypeAlias(it) => {
let layout = it.ty(db).layout(db).ok()?; label_and_layout_info_and_docs(db, it, config, |&it| it.ty(db).layout(db), |_| None)
let niches = niches(db, it, &layout).unwrap_or_default(); }
Some(format!("size = {:#X}, align = {:#X}{niches}", layout.size(), layout.align(),))
}),
Definition::BuiltinType(it) => { Definition::BuiltinType(it) => {
return famous_defs return famous_defs
.and_then(|fd| builtin(fd, it)) .and_then(|fd| builtin(fd, it))
@ -509,10 +495,6 @@ pub(super) fn definition(
markup(docs, label, mod_path) markup(docs, label, mod_path)
} }
fn niches(db: &RootDatabase, it: impl HasCrate, layout: &Layout) -> Option<String> {
Some(format!(", niches = {}", layout.niches(db, it.krate(db).into())?))
}
fn type_info( fn type_info(
sema: &Semantics<'_, RootDatabase>, sema: &Semantics<'_, RootDatabase>,
config: &HoverConfig, config: &HoverConfig,
@ -557,14 +539,6 @@ fn closure_ty(
TypeInfo { original, adjusted }: &TypeInfo, TypeInfo { original, adjusted }: &TypeInfo,
) -> Option<HoverResult> { ) -> Option<HoverResult> {
let c = original.as_closure()?; let c = original.as_closure()?;
let layout = if config.memory_layout {
original
.layout(sema.db)
.map(|x| format!(" // size = {}, align = {}", x.size(), x.align()))
.unwrap_or_default()
} else {
String::default()
};
let mut captures_rendered = c.captured_items(sema.db) let mut captures_rendered = c.captured_items(sema.db)
.into_iter() .into_iter()
.map(|it| { .map(|it| {
@ -600,17 +574,23 @@ fn closure_ty(
} else { } else {
String::new() String::new()
}; };
let mut markup = format!("```rust\n{}", c.display_with_id(sema.db),);
if let Some(layout) =
render_memory_layout(config.memory_layout, || original.layout(sema.db), |_| None, |_| None)
{
format_to!(markup, "{layout}");
}
format_to!(
markup,
"\n{}\n```{adjusted}\n\n## Captures\n{}",
c.display_with_impl(sema.db),
captures_rendered,
);
let mut res = HoverResult::default(); let mut res = HoverResult::default();
res.actions.push(HoverAction::goto_type_from_targets(sema.db, targets)); res.actions.push(HoverAction::goto_type_from_targets(sema.db, targets));
res.markup = format!( res.markup = markup.into();
"```rust\n{}{}\n{}\n```{adjusted}\n\n## Captures\n{}",
c.display_with_id(sema.db),
layout,
c.display_with_impl(sema.db),
captures_rendered,
)
.into();
Some(res) Some(res)
} }
@ -644,48 +624,59 @@ where
(label, docs) (label, docs)
} }
fn label_and_layout_info_and_docs<D, E, V>( fn label_and_layout_info_and_docs<D, E, E2>(
db: &RootDatabase, db: &RootDatabase,
def: D, def: D,
config: &HoverConfig, config: &HoverConfig,
layout_extractor: E, layout_extractor: E,
layout_offset_extractor: E2,
) -> (String, Option<hir::Documentation>) ) -> (String, Option<hir::Documentation>)
where where
D: HasAttrs + HirDisplay, D: HasAttrs + HirDisplay,
E: Fn(&D) -> Option<V>, E: Fn(&D) -> Result<Layout, LayoutError>,
V: Display, E2: Fn(&Layout) -> Option<u64>,
{ {
let label = match config.memory_layout.then(|| layout_extractor(&def)).flatten() { let mut label = def.display(db).to_string();
Some(layout) => format!("{} // {layout}", def.display(db)), if let Some(layout) = render_memory_layout(
_ => def.display(db).to_string(), config.memory_layout,
}; || layout_extractor(&def),
layout_offset_extractor,
|_| None,
) {
format_to!(label, "{layout}");
}
let docs = def.attrs(db).docs(); let docs = def.attrs(db).docs();
(label, docs) (label, docs)
} }
fn label_value_and_layout_info_and_docs<D, E, E2, V, L>( fn label_value_and_layout_info_and_docs<D, E, E2, E3, V>(
db: &RootDatabase, db: &RootDatabase,
def: D, def: D,
config: &HoverConfig, config: &HoverConfig,
value_extractor: E, value_extractor: E,
layout_extractor: E2, layout_extractor: E2,
layout_tag_extractor: E3,
) -> (String, Option<hir::Documentation>) ) -> (String, Option<hir::Documentation>)
where where
D: HasAttrs + HirDisplay, D: HasAttrs + HirDisplay,
E: Fn(&D) -> Option<V>, E: Fn(&D) -> Option<V>,
E2: Fn(&D) -> Option<L>, E2: Fn(&D) -> Result<Layout, LayoutError>,
E3: Fn(&Layout) -> Option<usize>,
V: Display, V: Display,
L: Display,
{ {
let value = value_extractor(&def); let value = value_extractor(&def);
let label = match value { let mut label = match value {
Some(value) => format!("{} = {value}", def.display(db)), Some(value) => format!("{} = {value}", def.display(db)),
None => def.display(db).to_string(), None => def.display(db).to_string(),
}; };
let label = match config.memory_layout.then(|| layout_extractor(&def)).flatten() { if let Some(layout) = render_memory_layout(
Some(layout) => format!("{} // {layout}", label), config.memory_layout,
_ => label, || layout_extractor(&def),
}; |_| None,
layout_tag_extractor,
) {
format_to!(label, "{layout}");
}
let docs = def.attrs(db).docs(); let docs = def.attrs(db).docs();
(label, docs) (label, docs)
} }
@ -769,14 +760,87 @@ fn local(db: &RootDatabase, it: hir::Local, config: &HoverConfig) -> Option<Mark
} }
None => format!("{is_mut}self: {ty}"), None => format!("{is_mut}self: {ty}"),
}; };
if config.memory_layout { if let Some(layout) =
if let Ok(layout) = it.ty(db).layout(db) { render_memory_layout(config.memory_layout, || it.ty(db).layout(db), |_| None, |_| None)
format_to!(desc, " // size = {}, align = {}", layout.size(), layout.align()); {
} format_to!(desc, "{layout}");
} }
markup(None, desc, None) markup(None, desc, None)
} }
fn render_memory_layout(
config: Option<MemoryLayoutHoverConfig>,
layout: impl FnOnce() -> Result<Layout, LayoutError>,
offset: impl FnOnce(&Layout) -> Option<u64>,
tag: impl FnOnce(&Layout) -> Option<usize>,
) -> Option<String> {
// field
let config = config?;
let layout = layout().ok()?;
let mut label = String::from(" // ");
if let Some(render) = config.size {
let size = match tag(&layout) {
Some(tag) => layout.size() as usize - tag,
None => layout.size() as usize,
};
format_to!(label, "size = ");
match render {
MemoryLayoutHoverRenderKind::Decimal => format_to!(label, "{size}"),
MemoryLayoutHoverRenderKind::Hexadecimal => format_to!(label, "{size:#X}"),
MemoryLayoutHoverRenderKind::Both if size >= 10 => {
format_to!(label, "{size} ({size:#X})")
}
MemoryLayoutHoverRenderKind::Both => format_to!(label, "{size}"),
}
format_to!(label, ", ");
}
if let Some(render) = config.alignment {
let align = layout.align();
format_to!(label, "align = ");
match render {
MemoryLayoutHoverRenderKind::Decimal => format_to!(label, "{align}",),
MemoryLayoutHoverRenderKind::Hexadecimal => format_to!(label, "{align:#X}",),
MemoryLayoutHoverRenderKind::Both if align >= 10 => {
format_to!(label, "{align} ({align:#X})")
}
MemoryLayoutHoverRenderKind::Both => {
format_to!(label, "{align}")
}
}
format_to!(label, ", ");
}
if let Some(render) = config.offset {
if let Some(offset) = offset(&layout) {
format_to!(label, "offset = ");
match render {
MemoryLayoutHoverRenderKind::Decimal => format_to!(label, "{offset}"),
MemoryLayoutHoverRenderKind::Hexadecimal => format_to!(label, "{offset:#X}"),
MemoryLayoutHoverRenderKind::Both if offset >= 10 => {
format_to!(label, "{offset} ({offset:#X})")
}
MemoryLayoutHoverRenderKind::Both => {
format_to!(label, "{offset}")
}
}
format_to!(label, ", ");
}
}
if config.niches {
if let Some(niches) = layout.niches() {
format_to!(label, "niches = {niches}, ");
}
}
label.pop(); // ' '
label.pop(); // ','
Some(label)
}
struct KeywordHint { struct KeywordHint {
description: String, description: String,
keyword_mod: String, keyword_mod: String,

View file

@ -2,11 +2,18 @@ use expect_test::{expect, Expect};
use ide_db::base_db::{FileLoader, FileRange}; use ide_db::base_db::{FileLoader, FileRange};
use syntax::TextRange; use syntax::TextRange;
use crate::{fixture, HoverConfig, HoverDocFormat}; use crate::{
fixture, HoverConfig, HoverDocFormat, MemoryLayoutHoverConfig, MemoryLayoutHoverRenderKind,
};
const HOVER_BASE_CONFIG: HoverConfig = HoverConfig { const HOVER_BASE_CONFIG: HoverConfig = HoverConfig {
links_in_hover: false, links_in_hover: false,
memory_layout: true, memory_layout: Some(MemoryLayoutHoverConfig {
size: Some(MemoryLayoutHoverRenderKind::Both),
offset: Some(MemoryLayoutHoverRenderKind::Both),
alignment: Some(MemoryLayoutHoverRenderKind::Both),
niches: true,
}),
documentation: true, documentation: true,
format: HoverDocFormat::Markdown, format: HoverDocFormat::Markdown,
keywords: true, keywords: true,
@ -62,7 +69,7 @@ fn check_hover_no_memory_layout(ra_fixture: &str, expect: Expect) {
let (analysis, position) = fixture::position(ra_fixture); let (analysis, position) = fixture::position(ra_fixture);
let hover = analysis let hover = analysis
.hover( .hover(
&HoverConfig { memory_layout: false, ..HOVER_BASE_CONFIG }, &HoverConfig { memory_layout: None, ..HOVER_BASE_CONFIG },
FileRange { file_id: position.file_id, range: TextRange::empty(position.offset) }, FileRange { file_id: position.file_id, range: TextRange::empty(position.offset) },
) )
.unwrap() .unwrap()
@ -237,7 +244,7 @@ fn main() {
expect![[r#" expect![[r#"
*|* *|*
```rust ```rust
{closure#0} // size = 8, align = 8 {closure#0} // size = 8, align = 8, niches = 1
impl Fn(i32) -> i32 impl Fn(i32) -> i32
``` ```
@ -292,7 +299,7 @@ fn main() {
expect![[r#" expect![[r#"
*|* *|*
```rust ```rust
{closure#0} // size = 16, align = 8 {closure#0} // size = 16 (0x10), align = 8, niches = 1
impl FnOnce() impl FnOnce()
``` ```
@ -320,7 +327,7 @@ fn main() {
expect![[r#" expect![[r#"
*|* *|*
```rust ```rust
{closure#0} // size = 8, align = 8 {closure#0} // size = 8, align = 8, niches = 1
impl FnMut() impl FnMut()
``` ```
@ -344,7 +351,7 @@ fn main() {
"#, "#,
expect![[r#" expect![[r#"
```rust ```rust
{closure#0} // size = 8, align = 8 {closure#0} // size = 8, align = 8, niches = 1
impl FnOnce() -> S2 impl FnOnce() -> S2
``` ```
Coerced to: &impl FnOnce() -> S2 Coerced to: &impl FnOnce() -> S2
@ -667,7 +674,7 @@ struct Foo { fiel$0d_a: u8, field_b: i32, field_c: i16 }
``` ```
```rust ```rust
field_a: u8 // size = 0x1, align = 0x1, offset = 0x4 field_a: u8 // size = 1, align = 1, offset = 4
``` ```
"#]], "#]],
); );
@ -692,7 +699,7 @@ fn main() {
``` ```
```rust ```rust
field_a: u32 // size = 0x4, align = 0x4, offset = 0x0 field_a: u32 // size = 4, align = 4, offset = 0
``` ```
"#]], "#]],
); );
@ -714,7 +721,7 @@ fn main() {
``` ```
```rust ```rust
field_a: u32 // size = 0x4, align = 0x4, offset = 0x0 field_a: u32 // size = 4, align = 4, offset = 0
``` ```
"#]], "#]],
); );
@ -1528,7 +1535,7 @@ fn test_hover_function_pointer_show_identifiers() {
``` ```
```rust ```rust
type foo = fn(a: i32, b: i32) -> i32 // size = 0x8, align = 0x8, niches = 1 type foo = fn(a: i32, b: i32) -> i32 // size = 8, align = 8, niches = 1
``` ```
"#]], "#]],
); );
@ -1546,7 +1553,7 @@ fn test_hover_function_pointer_no_identifier() {
``` ```
```rust ```rust
type foo = fn(i32, i32) -> i32 // size = 0x8, align = 0x8, niches = 1 type foo = fn(i32, i32) -> i32 // size = 8, align = 8, niches = 1
``` ```
"#]], "#]],
); );
@ -1674,7 +1681,7 @@ fn foo() { let bar = Ba$0r; }
``` ```
```rust ```rust
struct Bar // size = 0x0, align = 0x1 struct Bar // size = 0, align = 1
``` ```
--- ---
@ -1710,7 +1717,7 @@ fn foo() { let bar = Ba$0r; }
``` ```
```rust ```rust
struct Bar // size = 0x0, align = 0x1 struct Bar // size = 0, align = 1
``` ```
--- ---
@ -1739,7 +1746,7 @@ fn foo() { let bar = Ba$0r; }
``` ```
```rust ```rust
struct Bar // size = 0x0, align = 0x1 struct Bar // size = 0, align = 1
``` ```
--- ---
@ -1767,7 +1774,7 @@ pub struct B$0ar
``` ```
```rust ```rust
pub struct Bar // size = 0x0, align = 0x1 pub struct Bar // size = 0, align = 1
``` ```
--- ---
@ -1794,7 +1801,7 @@ pub struct B$0ar
``` ```
```rust ```rust
pub struct Bar // size = 0x0, align = 0x1 pub struct Bar // size = 0, align = 1
``` ```
--- ---
@ -1883,7 +1890,7 @@ fn test_hover_layout_of_variant() {
``` ```
```rust ```rust
Variant1(u8, u16) // size = 0x4 Variant1(u8, u16) // size = 4, align = 2
``` ```
"#]], "#]],
); );
@ -1904,7 +1911,7 @@ fn test_hover_layout_of_enum() {
``` ```
```rust ```rust
enum Foo // size = 0x10, align = 0x8, niches = 254 enum Foo // size = 16 (0x10), align = 8, niches = 254
``` ```
"#]], "#]],
); );
@ -3204,7 +3211,7 @@ fn main() {
*f* *f*
```rust ```rust
f: &i32 // size = 8, align = 8 f: &i32 // size = 8, align = 8, niches = 1
``` ```
--- ---
@ -3213,7 +3220,7 @@ fn main() {
``` ```
```rust ```rust
f: i32 // size = 0x4, align = 0x4, offset = 0x0 f: i32 // size = 4, align = 4, offset = 0
``` ```
"#]], "#]],
); );
@ -3353,7 +3360,7 @@ impl Foo {
*self* *self*
```rust ```rust
self: &Foo // size = 8, align = 8 self: &Foo // size = 8, align = 8, niches = 1
``` ```
"#]], "#]],
); );
@ -3758,7 +3765,7 @@ type Fo$0o2 = Foo<2>;
``` ```
```rust ```rust
type Foo2 = Foo<2> // size = 0x0, align = 0x1 type Foo2 = Foo<2> // size = 0, align = 1
``` ```
"#]], "#]],
); );
@ -3800,7 +3807,7 @@ enum E {
``` ```
```rust ```rust
A = 8 A = 8 // size = 1, align = 1
``` ```
--- ---
@ -3825,7 +3832,7 @@ enum E {
``` ```
```rust ```rust
A = 12 (0xC) A = 12 (0xC) // size = 1, align = 1
``` ```
--- ---
@ -3851,7 +3858,7 @@ enum E {
``` ```
```rust ```rust
B = 2 B = 2 // size = 1, align = 1
``` ```
--- ---
@ -3877,7 +3884,7 @@ enum E {
``` ```
```rust ```rust
B = 5 B = 5 // size = 1, align = 1
``` ```
--- ---
@ -4411,7 +4418,7 @@ fn foo(e: E) {
``` ```
```rust ```rust
A = 3 A = 3 // size = 0, align = 1
``` ```
--- ---
@ -4433,7 +4440,7 @@ fn main() {
*tile4* *tile4*
```rust ```rust
let tile4: [u32; 8] // size = 32, align = 4 let tile4: [u32; 8] // size = 32 (0x20), align = 4
``` ```
"#]], "#]],
); );
@ -4669,7 +4676,7 @@ pub fn gimme() -> theitem::TheItem {
``` ```
```rust ```rust
pub struct TheItem // size = 0x0, align = 0x1 pub struct TheItem // size = 0, align = 1
``` ```
--- ---
@ -4817,7 +4824,7 @@ mod string {
``` ```
```rust ```rust
struct String // size = 0x0, align = 0x1 struct String // size = 0, align = 1
``` ```
--- ---
@ -5486,7 +5493,7 @@ foo_macro!(
``` ```
```rust ```rust
pub struct Foo // size = 0x0, align = 0x1 pub struct Foo // size = 0, align = 1
``` ```
--- ---
@ -5511,7 +5518,7 @@ pub struct Foo(i32);
``` ```
```rust ```rust
pub struct Foo // size = 0x4, align = 0x4 pub struct Foo // size = 4, align = 4
``` ```
--- ---
@ -5610,7 +5617,7 @@ enum Enum {
``` ```
```rust ```rust
RecordV { field: u32 } // size = 0x4 RecordV { field: u32 } // size = 4, align = 4
``` ```
"#]], "#]],
); );
@ -5632,7 +5639,7 @@ enum Enum {
``` ```
```rust ```rust
field: u32 // size = 0x4, align = 0x4 field: u32 // size = 4, align = 4
``` ```
"#]], "#]],
); );
@ -6134,7 +6141,7 @@ fn test() {
``` ```
```rust ```rust
f: u32 // size = 0x4, align = 0x4, offset = 0x0 f: u32 // size = 4, align = 4, offset = 0
``` ```
"#]], "#]],
); );

View file

@ -84,7 +84,10 @@ pub use crate::{
file_structure::{StructureNode, StructureNodeKind}, file_structure::{StructureNode, StructureNodeKind},
folding_ranges::{Fold, FoldKind}, folding_ranges::{Fold, FoldKind},
highlight_related::{HighlightRelatedConfig, HighlightedRange}, highlight_related::{HighlightRelatedConfig, HighlightedRange},
hover::{HoverAction, HoverConfig, HoverDocFormat, HoverGotoTypeData, HoverResult}, hover::{
HoverAction, HoverConfig, HoverDocFormat, HoverGotoTypeData, HoverResult,
MemoryLayoutHoverConfig, MemoryLayoutHoverRenderKind,
},
inlay_hints::{ inlay_hints::{
AdjustmentHints, AdjustmentHintsMode, ClosureReturnTypeHints, DiscriminantHints, InlayHint, AdjustmentHints, AdjustmentHintsMode, ClosureReturnTypeHints, DiscriminantHints, InlayHint,
InlayHintLabel, InlayHintLabelPart, InlayHintPosition, InlayHintsConfig, InlayKind, InlayHintLabel, InlayHintLabelPart, InlayHintPosition, InlayHintsConfig, InlayKind,

View file

@ -138,7 +138,7 @@ impl StaticIndex<'_> {
}); });
let hover_config = HoverConfig { let hover_config = HoverConfig {
links_in_hover: true, links_in_hover: true,
memory_layout: true, memory_layout: None,
documentation: true, documentation: true,
keywords: true, keywords: true,
format: crate::HoverDocFormat::Markdown, format: crate::HoverDocFormat::Markdown,

View file

@ -14,7 +14,7 @@ use flycheck::FlycheckConfig;
use ide::{ use ide::{
AssistConfig, CallableSnippets, CompletionConfig, DiagnosticsConfig, ExprFillDefaultMode, AssistConfig, CallableSnippets, CompletionConfig, DiagnosticsConfig, ExprFillDefaultMode,
HighlightConfig, HighlightRelatedConfig, HoverConfig, HoverDocFormat, InlayHintsConfig, HighlightConfig, HighlightRelatedConfig, HoverConfig, HoverDocFormat, InlayHintsConfig,
JoinLinesConfig, Snippet, SnippetScope, JoinLinesConfig, MemoryLayoutHoverConfig, MemoryLayoutHoverRenderKind, Snippet, SnippetScope,
}; };
use ide_db::{ use ide_db::{
imports::insert_use::{ImportGranularity, InsertUseConfig, PrefixKind}, imports::insert_use::{ImportGranularity, InsertUseConfig, PrefixKind},
@ -317,8 +317,16 @@ config_data! {
hover_documentation_keywords_enable: bool = "true", hover_documentation_keywords_enable: bool = "true",
/// Use markdown syntax for links on hover. /// Use markdown syntax for links on hover.
hover_links_enable: bool = "true", hover_links_enable: bool = "true",
/// How to render the align information in a memory layout hover.
hover_memoryLayout_alignment: Option<MemoryLayoutHoverRenderKindDef> = "\"hexadecimal\"",
/// Whether to show memory layout data on hover. /// Whether to show memory layout data on hover.
hover_memoryLayout_enable: bool = "true", hover_memoryLayout_enable: bool = "true",
/// How to render the niche information in a memory layout hover.
hover_memoryLayout_niches: Option<bool> = "false",
/// How to render the offset information in a memory layout hover.
hover_memoryLayout_offset: Option<MemoryLayoutHoverRenderKindDef> = "\"hexadecimal\"",
/// How to render the size information in a memory layout hover.
hover_memoryLayout_size: Option<MemoryLayoutHoverRenderKindDef> = "\"both\"",
/// Whether to enforce the import granularity setting for all files. If set to false rust-analyzer will try to keep import styles consistent per file. /// Whether to enforce the import granularity setting for all files. If set to false rust-analyzer will try to keep import styles consistent per file.
imports_granularity_enforce: bool = "false", imports_granularity_enforce: bool = "false",
@ -1514,9 +1522,19 @@ impl Config {
} }
pub fn hover(&self) -> HoverConfig { pub fn hover(&self) -> HoverConfig {
let mem_kind = |kind| match kind {
MemoryLayoutHoverRenderKindDef::Both => MemoryLayoutHoverRenderKind::Both,
MemoryLayoutHoverRenderKindDef::Decimal => MemoryLayoutHoverRenderKind::Decimal,
MemoryLayoutHoverRenderKindDef::Hexadecimal => MemoryLayoutHoverRenderKind::Hexadecimal,
};
HoverConfig { HoverConfig {
links_in_hover: self.data.hover_links_enable, links_in_hover: self.data.hover_links_enable,
memory_layout: self.data.hover_memoryLayout_enable, memory_layout: self.data.hover_memoryLayout_enable.then_some(MemoryLayoutHoverConfig {
size: self.data.hover_memoryLayout_size.map(mem_kind),
offset: self.data.hover_memoryLayout_offset.map(mem_kind),
alignment: self.data.hover_memoryLayout_alignment.map(mem_kind),
niches: self.data.hover_memoryLayout_niches.unwrap_or_default(),
}),
documentation: self.data.hover_documentation_enable, documentation: self.data.hover_documentation_enable,
format: { format: {
let is_markdown = try_or_def!(self let is_markdown = try_or_def!(self
@ -1726,6 +1744,9 @@ mod de_unit_v {
named_unit_variant!(reborrow); named_unit_variant!(reborrow);
named_unit_variant!(fieldless); named_unit_variant!(fieldless);
named_unit_variant!(with_block); named_unit_variant!(with_block);
named_unit_variant!(decimal);
named_unit_variant!(hexadecimal);
named_unit_variant!(both);
} }
#[derive(Deserialize, Debug, Clone, Copy)] #[derive(Deserialize, Debug, Clone, Copy)]
@ -1956,6 +1977,18 @@ enum WorkspaceSymbolSearchKindDef {
AllSymbols, AllSymbols,
} }
#[derive(Deserialize, Debug, Copy, Clone)]
#[serde(rename_all = "snake_case")]
#[serde(untagged)]
pub enum MemoryLayoutHoverRenderKindDef {
#[serde(deserialize_with = "de_unit_v::decimal")]
Decimal,
#[serde(deserialize_with = "de_unit_v::hexadecimal")]
Hexadecimal,
#[serde(deserialize_with = "de_unit_v::both")]
Both,
}
macro_rules! _config_data { macro_rules! _config_data {
(struct $name:ident { (struct $name:ident {
$( $(
@ -2038,7 +2071,9 @@ fn get_field<T: DeserializeOwned>(
None None
} }
}) })
.unwrap_or_else(|| serde_json::from_str(default).unwrap()) .unwrap_or_else(|| {
serde_json::from_str(default).unwrap_or_else(|e| panic!("{e} on: `{default}`"))
})
} }
fn schema(fields: &[(&'static str, &'static str, &[&str], &str)]) -> serde_json::Value { fn schema(fields: &[(&'static str, &'static str, &[&str], &str)]) -> serde_json::Value {
@ -2366,6 +2401,22 @@ fn field_props(field: &str, ty: &str, doc: &[&str], default: &str) -> serde_json
"`hide`: Shows `...` for every closure type", "`hide`: Shows `...` for every closure type",
], ],
}, },
"Option<MemoryLayoutHoverRenderKindDef>" => set! {
"anyOf": [
{
"type": "null"
},
{
"type": "string",
"enum": ["both", "decimal", "hexadecimal", ],
"enumDescriptions": [
"Render as 12 (0xC)",
"Render as 12",
"Render as 0xC"
],
},
],
},
_ => panic!("missing entry for {ty}: {default}"), _ => panic!("missing entry for {ty}: {default}"),
} }

View file

@ -428,11 +428,31 @@ Whether to show keyword hover popups. Only applies when
-- --
Use markdown syntax for links on hover. Use markdown syntax for links on hover.
-- --
[[rust-analyzer.hover.memoryLayout.alignment]]rust-analyzer.hover.memoryLayout.alignment (default: `"hexadecimal"`)::
+
--
How to render the align information in a memory layout hover.
--
[[rust-analyzer.hover.memoryLayout.enable]]rust-analyzer.hover.memoryLayout.enable (default: `true`):: [[rust-analyzer.hover.memoryLayout.enable]]rust-analyzer.hover.memoryLayout.enable (default: `true`)::
+ +
-- --
Whether to show memory layout data on hover. Whether to show memory layout data on hover.
-- --
[[rust-analyzer.hover.memoryLayout.niches]]rust-analyzer.hover.memoryLayout.niches (default: `false`)::
+
--
How to render the niche information in a memory layout hover.
--
[[rust-analyzer.hover.memoryLayout.offset]]rust-analyzer.hover.memoryLayout.offset (default: `"hexadecimal"`)::
+
--
How to render the offset information in a memory layout hover.
--
[[rust-analyzer.hover.memoryLayout.size]]rust-analyzer.hover.memoryLayout.size (default: `"both"`)::
+
--
How to render the size information in a memory layout hover.
--
[[rust-analyzer.imports.granularity.enforce]]rust-analyzer.imports.granularity.enforce (default: `false`):: [[rust-analyzer.imports.granularity.enforce]]rust-analyzer.imports.granularity.enforce (default: `false`)::
+ +
-- --

View file

@ -966,11 +966,85 @@
"default": true, "default": true,
"type": "boolean" "type": "boolean"
}, },
"rust-analyzer.hover.memoryLayout.alignment": {
"markdownDescription": "How to render the align information in a memory layout hover.",
"default": "hexadecimal",
"anyOf": [
{
"type": "null"
},
{
"type": "string",
"enum": [
"both",
"decimal",
"hexadecimal"
],
"enumDescriptions": [
"Render as 12 (0xC)",
"Render as 12",
"Render as 0xC"
]
}
]
},
"rust-analyzer.hover.memoryLayout.enable": { "rust-analyzer.hover.memoryLayout.enable": {
"markdownDescription": "Whether to show memory layout data on hover.", "markdownDescription": "Whether to show memory layout data on hover.",
"default": true, "default": true,
"type": "boolean" "type": "boolean"
}, },
"rust-analyzer.hover.memoryLayout.niches": {
"markdownDescription": "How to render the niche information in a memory layout hover.",
"default": false,
"type": [
"null",
"boolean"
]
},
"rust-analyzer.hover.memoryLayout.offset": {
"markdownDescription": "How to render the offset information in a memory layout hover.",
"default": "hexadecimal",
"anyOf": [
{
"type": "null"
},
{
"type": "string",
"enum": [
"both",
"decimal",
"hexadecimal"
],
"enumDescriptions": [
"Render as 12 (0xC)",
"Render as 12",
"Render as 0xC"
]
}
]
},
"rust-analyzer.hover.memoryLayout.size": {
"markdownDescription": "How to render the size information in a memory layout hover.",
"default": "both",
"anyOf": [
{
"type": "null"
},
{
"type": "string",
"enum": [
"both",
"decimal",
"hexadecimal"
],
"enumDescriptions": [
"Render as 12 (0xC)",
"Render as 12",
"Render as 0xC"
]
}
]
},
"rust-analyzer.imports.granularity.enforce": { "rust-analyzer.imports.granularity.enforce": {
"markdownDescription": "Whether to enforce the import granularity setting for all files. If set to false rust-analyzer will try to keep import styles consistent per file.", "markdownDescription": "Whether to enforce the import granularity setting for all files. If set to false rust-analyzer will try to keep import styles consistent per file.",
"default": false, "default": false,