2021-04-09 14:39:07 +02:00
//! The `HirDisplay` trait, which serves two purposes: Turning various bits from
//! HIR back into source code, and just displaying them for debugging/testing
//! purposes.
2019-09-30 11:58:53 +03:00
2021-06-29 17:35:37 +02:00
use std ::fmt ::{ self , Debug } ;
2019-03-14 22:03:39 +01:00
2021-06-15 23:28:37 +05:00
use base_db ::CrateId ;
2021-04-06 11:45:41 +02:00
use chalk_ir ::BoundVar ;
2020-04-25 16:57:59 +02:00
use hir_def ::{
2021-04-12 16:24:48 +02:00
body ,
2021-03-14 20:03:39 +08:00
db ::DefDatabase ,
find_path ,
2021-12-29 17:05:59 +03:30
generics ::{ TypeOrConstParamData , TypeParamProvenance } ,
2021-05-24 15:13:23 +02:00
intern ::{ Internable , Interned } ,
2021-03-14 20:03:39 +08:00
item_scope ::ItemInNs ,
2021-04-01 21:04:02 +02:00
path ::{ Path , PathKind } ,
2022-03-09 22:20:24 +03:30
type_ref ::{ ConstScalar , TraitBoundModifier , TypeBound , TypeRef } ,
2021-03-14 20:03:39 +08:00
visibility ::Visibility ,
2021-12-07 17:31:26 +01:00
HasModule , ItemContainerId , Lookup , ModuleId , TraitId ,
2020-04-25 16:57:59 +02:00
} ;
2021-04-12 16:24:48 +02:00
use hir_expand ::{ hygiene ::Hygiene , name ::Name } ;
2021-06-30 01:34:51 +02:00
use itertools ::Itertools ;
2021-12-10 20:01:24 +01:00
use syntax ::SmolStr ;
2019-03-14 22:03:39 +01:00
2021-03-01 21:57:39 +01:00
use crate ::{
2021-06-29 17:35:37 +02:00
db ::HirDatabase ,
from_assoc_type_id , from_foreign_def_id , from_placeholder_idx , lt_from_placeholder_idx ,
mapping ::from_chalk ,
primitive , subst_prefix , to_assoc_type_id ,
utils ::{ self , generics } ,
2022-03-09 22:20:24 +03:30
AdtId , AliasEq , AliasTy , Binders , CallableDefId , CallableSig , Const , ConstValue , DomainGoal ,
GenericArg , ImplTraitId , Interner , Lifetime , LifetimeData , LifetimeOutlives , Mutability ,
OpaqueTy , ProjectionTy , ProjectionTyExt , QuantifiedWhereClause , Scalar , Substitution , TraitRef ,
TraitRefExt , Ty , TyExt , TyKind , WhereClause ,
2021-03-01 21:57:39 +01:00
} ;
2020-04-25 16:57:59 +02:00
pub struct HirFormatter < ' a > {
2020-03-13 16:05:46 +01:00
pub db : & ' a dyn HirDatabase ,
2020-04-25 16:57:59 +02:00
fmt : & ' a mut dyn fmt ::Write ,
2019-11-18 18:02:28 +01:00
buf : String ,
curr_size : usize ,
2020-01-22 16:44:05 +02:00
pub ( crate ) max_size : Option < usize > ,
2019-12-23 17:53:35 +02:00
omit_verbose_types : bool ,
2020-04-25 16:57:59 +02:00
display_target : DisplayTarget ,
2019-03-14 22:03:39 +01:00
}
pub trait HirDisplay {
2020-04-25 16:57:59 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > ;
2019-11-18 18:02:28 +01:00
2020-10-28 17:58:16 +01:00
/// Returns a `Display`able type that is human-readable.
fn into_displayable < ' a > (
& ' a self ,
db : & ' a dyn HirDatabase ,
max_size : Option < usize > ,
omit_verbose_types : bool ,
display_target : DisplayTarget ,
) -> HirDisplayWrapper < ' a , Self >
where
Self : Sized ,
{
2021-04-06 14:42:34 +02:00
assert! (
! matches! ( display_target , DisplayTarget ::SourceCode { .. } ) ,
" HirDisplayWrapper cannot fail with DisplaySourceCodeError, use HirDisplay::hir_fmt directly instead "
) ;
2020-10-28 17:58:16 +01:00
HirDisplayWrapper { db , t : self , max_size , omit_verbose_types , display_target }
}
2020-04-25 16:57:59 +02:00
/// Returns a `Display`able type that is human-readable.
/// Use this for showing types to the user (e.g. diagnostics)
2020-03-13 16:05:46 +01:00
fn display < ' a > ( & ' a self , db : & ' a dyn HirDatabase ) -> HirDisplayWrapper < ' a , Self >
2019-03-14 22:03:39 +01:00
where
Self : Sized ,
{
2020-04-25 16:57:59 +02:00
HirDisplayWrapper {
db ,
t : self ,
max_size : None ,
omit_verbose_types : false ,
display_target : DisplayTarget ::Diagnostics ,
}
2019-11-18 18:02:28 +01:00
}
2020-04-25 16:57:59 +02:00
/// Returns a `Display`able type that is human-readable and tries to be succinct.
/// Use this for showing types to the user where space is constrained (e.g. doc popups)
2020-03-13 16:05:46 +01:00
fn display_truncated < ' a > (
2019-11-18 18:02:28 +01:00
& ' a self ,
2020-03-13 16:05:46 +01:00
db : & ' a dyn HirDatabase ,
2019-12-19 16:43:41 +02:00
max_size : Option < usize > ,
2020-03-13 16:05:46 +01:00
) -> HirDisplayWrapper < ' a , Self >
2019-11-18 18:02:28 +01:00
where
Self : Sized ,
{
2020-04-25 16:57:59 +02:00
HirDisplayWrapper {
db ,
t : self ,
max_size ,
omit_verbose_types : true ,
display_target : DisplayTarget ::Diagnostics ,
}
}
/// Returns a String representation of `self` that can be inserted into the given module.
/// Use this when generating code (e.g. assists)
fn display_source_code < ' a > (
& ' a self ,
db : & ' a dyn HirDatabase ,
module_id : ModuleId ,
) -> Result < String , DisplaySourceCodeError > {
let mut result = String ::new ( ) ;
match self . hir_fmt ( & mut HirFormatter {
db ,
fmt : & mut result ,
buf : String ::with_capacity ( 20 ) ,
curr_size : 0 ,
max_size : None ,
omit_verbose_types : false ,
display_target : DisplayTarget ::SourceCode { module_id } ,
2020-10-28 12:29:42 +01:00
} ) {
Ok ( ( ) ) = > { }
Err ( HirDisplayError ::FmtError ) = > panic! ( " Writing to String can't fail! " ) ,
Err ( HirDisplayError ::DisplaySourceCodeError ( e ) ) = > return Err ( e ) ,
} ;
Ok ( result )
}
/// Returns a String representation of `self` for test purposes
2020-10-28 15:09:47 +01:00
fn display_test < ' a > ( & ' a self , db : & ' a dyn HirDatabase ) -> HirDisplayWrapper < ' a , Self >
where
Self : Sized ,
{
HirDisplayWrapper {
2020-10-28 12:29:42 +01:00
db ,
2020-10-28 15:09:47 +01:00
t : self ,
2020-10-28 12:29:42 +01:00
max_size : None ,
omit_verbose_types : false ,
2020-10-28 15:09:47 +01:00
display_target : DisplayTarget ::Test ,
}
2019-03-14 22:03:39 +01:00
}
}
2020-04-25 16:57:59 +02:00
impl < ' a > HirFormatter < ' a > {
2019-03-14 22:03:39 +01:00
pub fn write_joined < T : HirDisplay > (
& mut self ,
iter : impl IntoIterator < Item = T > ,
sep : & str ,
2020-04-25 16:57:59 +02:00
) -> Result < ( ) , HirDisplayError > {
2019-03-14 22:03:39 +01:00
let mut first = true ;
for e in iter {
if ! first {
write! ( self , " {} " , sep ) ? ;
}
first = false ;
2022-01-10 16:55:53 +01:00
// Abbreviate multiple omitted types with a single ellipsis.
if self . should_truncate ( ) {
return write! ( self , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2019-03-14 22:03:39 +01:00
e . hir_fmt ( self ) ? ;
}
Ok ( ( ) )
}
/// This allows using the `write!` macro directly with a `HirFormatter`.
2020-04-25 16:57:59 +02:00
pub fn write_fmt ( & mut self , args : fmt ::Arguments ) -> Result < ( ) , HirDisplayError > {
2019-11-18 18:02:28 +01:00
// We write to a buffer first to track output size
self . buf . clear ( ) ;
fmt ::write ( & mut self . buf , args ) ? ;
self . curr_size + = self . buf . len ( ) ;
// Then we write to the internal formatter from the buffer
2020-04-25 16:57:59 +02:00
self . fmt . write_str ( & self . buf ) . map_err ( HirDisplayError ::from )
2019-11-18 18:02:28 +01:00
}
pub fn should_truncate ( & self ) -> bool {
2021-10-03 23:53:01 +11:00
match self . max_size {
Some ( max_size ) = > self . curr_size > = max_size ,
None = > false ,
2019-11-18 18:02:28 +01:00
}
2019-03-14 22:03:39 +01:00
}
2019-12-08 00:54:18 +02:00
2019-12-23 17:53:35 +02:00
pub fn omit_verbose_types ( & self ) -> bool {
self . omit_verbose_types
2019-12-08 00:54:18 +02:00
}
}
2020-04-25 16:57:59 +02:00
#[ derive(Clone, Copy) ]
2020-10-28 17:58:16 +01:00
pub enum DisplayTarget {
2020-04-25 16:57:59 +02:00
/// Display types for inlays, doc popups, autocompletion, etc...
/// Showing `{unknown}` or not qualifying paths is fine here.
/// There's no reason for this to fail.
Diagnostics ,
/// Display types for inserting them in source files.
/// The generated code should compile, so paths need to be qualified.
SourceCode { module_id : ModuleId } ,
2020-10-28 11:20:05 +01:00
/// Only for test purpose to keep real types
2020-10-28 15:09:47 +01:00
Test ,
2020-04-25 16:57:59 +02:00
}
2020-05-10 18:09:22 +02:00
impl DisplayTarget {
fn is_source_code ( & self ) -> bool {
2021-01-05 15:45:46 +03:00
matches! ( self , Self ::SourceCode { .. } )
2020-05-10 18:09:22 +02:00
}
2020-10-28 11:20:05 +01:00
fn is_test ( & self ) -> bool {
2020-10-28 15:09:47 +01:00
matches! ( self , Self ::Test )
2020-10-28 11:20:05 +01:00
}
2020-05-10 18:09:22 +02:00
}
2020-04-25 16:57:59 +02:00
#[ derive(Debug) ]
pub enum DisplaySourceCodeError {
PathNotFound ,
2020-12-12 18:18:19 +01:00
UnknownType ,
2021-03-22 15:17:01 +08:00
Closure ,
2020-04-25 16:57:59 +02:00
}
pub enum HirDisplayError {
/// Errors that can occur when generating source code
DisplaySourceCodeError ( DisplaySourceCodeError ) ,
/// `FmtError` is required to be compatible with std::fmt::Display
FmtError ,
}
impl From < fmt ::Error > for HirDisplayError {
fn from ( _ : fmt ::Error ) -> Self {
Self ::FmtError
}
}
pub struct HirDisplayWrapper < ' a , T > {
db : & ' a dyn HirDatabase ,
t : & ' a T ,
max_size : Option < usize > ,
omit_verbose_types : bool ,
display_target : DisplayTarget ,
}
2019-03-14 22:03:39 +01:00
2020-03-13 16:05:46 +01:00
impl < ' a , T > fmt ::Display for HirDisplayWrapper < ' a , T >
2019-03-14 22:03:39 +01:00
where
T : HirDisplay ,
{
fn fmt ( & self , f : & mut fmt ::Formatter ) -> fmt ::Result {
2020-04-25 16:57:59 +02:00
match self . t . hir_fmt ( & mut HirFormatter {
db : self . db ,
2019-11-18 18:02:28 +01:00
fmt : f ,
buf : String ::with_capacity ( 20 ) ,
curr_size : 0 ,
2020-04-25 16:57:59 +02:00
max_size : self . max_size ,
omit_verbose_types : self . omit_verbose_types ,
display_target : self . display_target ,
} ) {
Ok ( ( ) ) = > Ok ( ( ) ) ,
Err ( HirDisplayError ::FmtError ) = > Err ( fmt ::Error ) ,
Err ( HirDisplayError ::DisplaySourceCodeError ( _ ) ) = > {
// This should never happen
2021-04-06 14:42:34 +02:00
panic! ( " HirDisplay::hir_fmt failed with DisplaySourceCodeError when calling Display::fmt! " )
2020-04-25 16:57:59 +02:00
}
}
2019-03-14 22:03:39 +01:00
}
}
2020-02-14 15:01:25 +01:00
const TYPE_HINT_TRUNCATION : & str = " … " ;
2021-03-14 20:03:39 +08:00
impl < T : HirDisplay > HirDisplay for & '_ T {
2020-04-25 16:57:59 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-02-14 15:01:25 +01:00
HirDisplay ::hir_fmt ( * self , f )
}
}
2021-05-24 15:13:23 +02:00
impl < T : HirDisplay + Internable > HirDisplay for Interned < T > {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
HirDisplay ::hir_fmt ( self . as_ref ( ) , f )
}
}
2021-02-28 19:13:37 +01:00
impl HirDisplay for ProjectionTy {
2020-04-25 16:57:59 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-02-14 15:01:25 +01:00
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2021-02-28 19:13:37 +01:00
let trait_ = f . db . trait_data ( self . trait_ ( f . db ) ) ;
2021-04-06 14:16:07 +02:00
write! ( f , " < " ) ? ;
2021-12-19 18:58:39 +02:00
self . self_type_parameter ( Interner ) . hir_fmt ( f ) ? ;
2021-04-06 14:16:07 +02:00
write! ( f , " as {} " , trait_ . name ) ? ;
2021-12-19 18:58:39 +02:00
if self . substitution . len ( Interner ) > 1 {
2021-02-28 19:13:37 +01:00
write! ( f , " < " ) ? ;
2021-12-19 18:58:39 +02:00
f . write_joined ( & self . substitution . as_slice ( Interner ) [ 1 .. ] , " , " ) ? ;
2021-02-28 19:13:37 +01:00
write! ( f , " > " ) ? ;
}
2021-03-14 16:26:12 +01:00
write! ( f , " >::{} " , f . db . type_alias_data ( from_assoc_type_id ( self . associated_ty_id ) ) . name ) ? ;
2021-02-28 19:13:37 +01:00
Ok ( ( ) )
}
}
2021-03-19 02:07:15 +01:00
impl HirDisplay for OpaqueTy {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2021-12-19 18:58:39 +02:00
self . substitution . at ( Interner , 0 ) . hir_fmt ( f )
2021-04-01 21:04:02 +02:00
}
}
impl HirDisplay for GenericArg {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self . interned ( ) {
crate ::GenericArgData ::Ty ( ty ) = > ty . hir_fmt ( f ) ,
2021-04-08 13:51:04 +02:00
crate ::GenericArgData ::Lifetime ( lt ) = > lt . hir_fmt ( f ) ,
crate ::GenericArgData ::Const ( c ) = > c . hir_fmt ( f ) ,
2021-04-01 21:04:02 +02:00
}
2021-03-19 02:07:15 +01:00
}
}
2021-04-06 11:45:41 +02:00
impl HirDisplay for Const {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
let data = self . interned ( ) ;
match data . value {
ConstValue ::BoundVar ( idx ) = > idx . hir_fmt ( f ) ,
2022-03-09 22:20:24 +03:30
ConstValue ::InferenceVar ( .. ) = > write! ( f , " #c# " ) ,
2021-04-06 11:45:41 +02:00
ConstValue ::Placeholder ( idx ) = > {
2021-12-29 17:05:59 +03:30
let id = from_placeholder_idx ( f . db , idx ) ;
2021-04-06 11:45:41 +02:00
let generics = generics ( f . db . upcast ( ) , id . parent ) ;
2022-03-09 22:20:24 +03:30
let param_data = & generics . params . type_or_consts [ id . local_id ] ;
2021-12-29 17:05:59 +03:30
write! ( f , " {} " , param_data . name ( ) . unwrap ( ) )
2021-04-06 11:45:41 +02:00
}
2021-05-11 05:06:33 -07:00
ConstValue ::Concrete ( c ) = > write! ( f , " {} " , c . interned ) ,
2021-04-06 11:45:41 +02:00
}
}
}
impl HirDisplay for BoundVar {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
write! ( f , " ?{}.{} " , self . debruijn . depth ( ) , self . index )
}
}
2021-02-28 19:13:37 +01:00
impl HirDisplay for Ty {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2022-02-15 14:47:51 +00:00
2021-12-19 18:58:39 +02:00
match self . kind ( Interner ) {
2021-03-13 14:44:51 +01:00
TyKind ::Never = > write! ( f , " ! " ) ? ,
TyKind ::Str = > write! ( f , " str " ) ? ,
TyKind ::Scalar ( Scalar ::Bool ) = > write! ( f , " bool " ) ? ,
TyKind ::Scalar ( Scalar ::Char ) = > write! ( f , " char " ) ? ,
& TyKind ::Scalar ( Scalar ::Float ( t ) ) = > write! ( f , " {} " , primitive ::float_ty_to_string ( t ) ) ? ,
& TyKind ::Scalar ( Scalar ::Int ( t ) ) = > write! ( f , " {} " , primitive ::int_ty_to_string ( t ) ) ? ,
& TyKind ::Scalar ( Scalar ::Uint ( t ) ) = > write! ( f , " {} " , primitive ::uint_ty_to_string ( t ) ) ? ,
2021-03-14 17:40:55 +01:00
TyKind ::Slice ( t ) = > {
2020-10-28 17:58:16 +01:00
write! ( f , " [ " ) ? ;
t . hir_fmt ( f ) ? ;
write! ( f , " ] " ) ? ;
2020-02-14 15:01:25 +01:00
}
2021-04-06 11:45:41 +02:00
TyKind ::Array ( t , c ) = > {
2020-10-28 17:58:16 +01:00
write! ( f , " [ " ) ? ;
t . hir_fmt ( f ) ? ;
2021-04-06 11:45:41 +02:00
write! ( f , " ; " ) ? ;
c . hir_fmt ( f ) ? ;
write! ( f , " ] " ) ? ;
2020-02-14 15:01:25 +01:00
}
2021-04-05 22:08:16 +02:00
TyKind ::Raw ( m , t ) | TyKind ::Ref ( m , _ , t ) = > {
2021-12-19 18:58:39 +02:00
if matches! ( self . kind ( Interner ) , TyKind ::Raw ( .. ) ) {
2021-03-01 19:30:34 +01:00
write! (
f ,
" *{} " ,
match m {
Mutability ::Not = > " const " ,
Mutability ::Mut = > " mut " ,
}
) ? ;
2020-10-06 14:40:27 +02:00
} else {
2021-03-01 19:30:34 +01:00
write! (
f ,
" &{} " ,
match m {
Mutability ::Not = > " " ,
Mutability ::Mut = > " mut " ,
}
) ? ;
2021-01-12 20:19:13 +01:00
}
2021-03-13 20:05:47 +01:00
// FIXME: all this just to decide whether to use parentheses...
2021-08-12 19:31:00 +05:00
let contains_impl_fn = | bounds : & [ QuantifiedWhereClause ] | {
2021-08-03 17:01:00 +05:00
bounds . iter ( ) . any ( | bound | {
if let WhereClause ::Implemented ( trait_ref ) = bound . skip_binders ( ) {
let trait_ = trait_ref . hir_trait_id ( ) ;
fn_traits ( f . db . upcast ( ) , trait_ ) . any ( | it | it = = trait_ )
} else {
false
}
} )
} ;
2021-12-19 18:58:39 +02:00
let ( preds_to_print , has_impl_fn_pred ) = match t . kind ( Interner ) {
2021-03-21 13:22:22 +01:00
TyKind ::Dyn ( dyn_ty ) if dyn_ty . bounds . skip_binders ( ) . interned ( ) . len ( ) > 1 = > {
2021-08-03 17:01:00 +05:00
let bounds = dyn_ty . bounds . skip_binders ( ) . interned ( ) ;
2021-08-12 19:31:00 +05:00
( bounds . len ( ) , contains_impl_fn ( bounds ) )
2021-01-12 20:19:13 +01:00
}
2021-08-02 15:59:28 +03:00
TyKind ::Alias ( AliasTy ::Opaque ( OpaqueTy {
2021-03-14 16:33:27 +01:00
opaque_ty_id ,
2021-08-02 15:59:28 +03:00
substitution : parameters ,
2021-07-08 21:19:53 +02:00
} ) )
2021-08-02 15:59:28 +03:00
| TyKind ::OpaqueType ( opaque_ty_id , parameters ) = > {
let impl_trait_id =
f . db . lookup_intern_impl_trait_id ( ( * opaque_ty_id ) . into ( ) ) ;
2021-03-13 20:05:47 +01:00
if let ImplTraitId ::ReturnTypeImplTrait ( func , idx ) = impl_trait_id {
2021-08-03 17:01:00 +05:00
let datas =
2021-03-13 20:05:47 +01:00
f . db . return_type_impl_traits ( func )
. expect ( " impl trait id without data " ) ;
let data = ( * datas )
. as_ref ( )
. map ( | rpit | rpit . impl_traits [ idx as usize ] . bounds . clone ( ) ) ;
2021-12-19 18:58:39 +02:00
let bounds = data . substitute ( Interner , parameters ) ;
2021-08-03 17:01:00 +05:00
let mut len = bounds . skip_binders ( ) . len ( ) ;
// Don't count Sized but count when it absent
// (i.e. when explicit ?Sized bound is set).
let default_sized = SizedByDefault ::Sized {
anchor : func . lookup ( f . db . upcast ( ) ) . module ( f . db . upcast ( ) ) . krate ( ) ,
} ;
let sized_bounds = bounds
. skip_binders ( )
. iter ( )
. filter ( | b | {
matches! (
b . skip_binders ( ) ,
WhereClause ::Implemented ( trait_ref )
if default_sized . is_sized_trait (
trait_ref . hir_trait_id ( ) ,
f . db . upcast ( ) ,
) ,
)
} )
. count ( ) ;
match sized_bounds {
0 = > len + = 1 ,
_ = > {
len = len . saturating_sub ( sized_bounds ) ;
}
}
2021-08-12 19:31:00 +05:00
( len , contains_impl_fn ( bounds . skip_binders ( ) ) )
2021-03-13 20:05:47 +01:00
} else {
2021-08-03 17:01:00 +05:00
( 0 , false )
2021-03-13 20:05:47 +01:00
}
2021-01-12 20:19:13 +01:00
}
2021-08-03 17:01:00 +05:00
_ = > ( 0 , false ) ,
2021-01-12 20:19:13 +01:00
} ;
2021-08-03 17:01:00 +05:00
if has_impl_fn_pred & & preds_to_print < = 2 {
return t . hir_fmt ( f ) ;
2020-10-06 14:40:27 +02:00
}
2021-08-03 17:01:00 +05:00
if preds_to_print > 1 {
2020-10-06 14:40:27 +02:00
write! ( f , " ( " ) ? ;
2021-04-06 14:16:07 +02:00
t . hir_fmt ( f ) ? ;
2020-10-06 14:40:27 +02:00
write! ( f , " ) " ) ? ;
} else {
2021-04-06 14:16:07 +02:00
t . hir_fmt ( f ) ? ;
2020-10-06 14:40:27 +02:00
}
2020-02-14 15:01:25 +01:00
}
2021-03-13 14:44:51 +01:00
TyKind ::Tuple ( _ , substs ) = > {
2021-12-19 18:58:39 +02:00
if substs . len ( Interner ) = = 1 {
2020-10-28 17:58:16 +01:00
write! ( f , " ( " ) ? ;
2021-12-19 18:58:39 +02:00
substs . at ( Interner , 0 ) . hir_fmt ( f ) ? ;
2020-10-28 17:58:16 +01:00
write! ( f , " ,) " ) ? ;
2020-02-14 15:01:25 +01:00
} else {
write! ( f , " ( " ) ? ;
2021-12-19 18:58:39 +02:00
f . write_joined ( & * substs . as_slice ( Interner ) , " , " ) ? ;
2020-02-14 15:01:25 +01:00
write! ( f , " ) " ) ? ;
}
}
2021-03-13 14:44:51 +01:00
TyKind ::Function ( fn_ptr ) = > {
2021-02-28 22:12:07 +01:00
let sig = CallableSig ::from_fn_ptr ( fn_ptr ) ;
2020-12-09 18:06:56 +01:00
sig . hir_fmt ( f ) ? ;
2020-02-14 15:01:25 +01:00
}
2021-03-13 14:44:51 +01:00
TyKind ::FnDef ( def , parameters ) = > {
2021-03-13 17:55:50 +01:00
let def = from_chalk ( f . db , * def ) ;
2021-12-19 18:58:39 +02:00
let sig = f . db . callable_item_signature ( def ) . substitute ( Interner , parameters ) ;
2020-03-28 23:25:29 +02:00
match def {
2020-07-16 13:15:00 +02:00
CallableDefId ::FunctionId ( ff ) = > {
write! ( f , " fn {} " , f . db . function_data ( ff ) . name ) ?
}
CallableDefId ::StructId ( s ) = > write! ( f , " {} " , f . db . struct_data ( s ) . name ) ? ,
CallableDefId ::EnumVariantId ( e ) = > {
2020-03-28 23:25:29 +02:00
write! ( f , " {} " , f . db . enum_data ( e . parent ) . variants [ e . local_id ] . name ) ?
2020-02-14 15:01:25 +01:00
}
} ;
2021-12-19 18:58:39 +02:00
if parameters . len ( Interner ) > 0 {
2020-03-13 16:05:46 +01:00
let generics = generics ( f . db . upcast ( ) , def . into ( ) ) ;
2022-03-08 19:51:35 +03:30
let ( parent_params , self_param , type_params , const_params , _impl_trait_params ) =
2020-02-14 15:01:25 +01:00
generics . provenance_split ( ) ;
2022-03-08 19:51:35 +03:30
let total_len = parent_params + self_param + type_params + const_params ;
2020-02-14 15:01:25 +01:00
// We print all params except implicit impl Trait params. Still a bit weird; should we leave out parent and self?
if total_len > 0 {
write! ( f , " < " ) ? ;
2021-12-19 18:58:39 +02:00
f . write_joined ( & parameters . as_slice ( Interner ) [ .. total_len ] , " , " ) ? ;
2020-02-14 15:01:25 +01:00
write! ( f , " > " ) ? ;
}
}
write! ( f , " ( " ) ? ;
f . write_joined ( sig . params ( ) , " , " ) ? ;
2020-04-05 18:25:47 +03:00
write! ( f , " ) " ) ? ;
let ret = sig . ret ( ) ;
2021-04-03 20:22:59 +02:00
if ! ret . is_unit ( ) {
2021-04-06 14:16:07 +02:00
write! ( f , " -> " ) ? ;
ret . hir_fmt ( f ) ? ;
2020-04-05 18:25:47 +03:00
}
2020-02-14 15:01:25 +01:00
}
2021-03-13 14:44:51 +01:00
TyKind ::Adt ( AdtId ( def_id ) , parameters ) = > {
2020-04-25 16:57:59 +02:00
match f . display_target {
2020-10-28 15:09:47 +01:00
DisplayTarget ::Diagnostics | DisplayTarget ::Test = > {
2021-02-28 19:46:59 +01:00
let name = match * def_id {
2021-03-01 21:57:39 +01:00
hir_def ::AdtId ::StructId ( it ) = > f . db . struct_data ( it ) . name . clone ( ) ,
hir_def ::AdtId ::UnionId ( it ) = > f . db . union_data ( it ) . name . clone ( ) ,
hir_def ::AdtId ::EnumId ( it ) = > f . db . enum_data ( it ) . name . clone ( ) ,
2020-04-25 16:57:59 +02:00
} ;
write! ( f , " {} " , name ) ? ;
}
2020-10-28 15:09:47 +01:00
DisplayTarget ::SourceCode { module_id } = > {
2020-10-28 11:20:05 +01:00
if let Some ( path ) = find_path ::find_path (
f . db . upcast ( ) ,
2021-02-28 19:46:59 +01:00
ItemInNs ::Types ( ( * def_id ) . into ( ) ) ,
2020-10-28 11:20:05 +01:00
module_id ,
) {
write! ( f , " {} " , path ) ? ;
} else {
return Err ( HirDisplayError ::DisplaySourceCodeError (
DisplaySourceCodeError ::PathNotFound ,
) ) ;
}
}
2020-04-25 16:57:59 +02:00
}
2021-12-19 18:58:39 +02:00
if parameters . len ( Interner ) > 0 {
2021-03-13 14:44:51 +01:00
let parameters_to_write = if f . display_target . is_source_code ( )
| | f . omit_verbose_types ( )
{
match self
2021-03-13 17:55:50 +01:00
. as_generic_def ( f . db )
2021-03-13 14:44:51 +01:00
. map ( | generic_def_id | f . db . generic_defaults ( generic_def_id ) )
. filter ( | defaults | ! defaults . is_empty ( ) )
{
2021-12-19 18:58:39 +02:00
None = > parameters . as_slice ( Interner ) ,
2021-03-13 14:44:51 +01:00
Some ( default_parameters ) = > {
2022-03-09 22:20:24 +03:30
fn should_show (
parameter : & GenericArg ,
default_parameters : & [ Binders < GenericArg > ] ,
i : usize ,
parameters : & Substitution ,
) -> bool {
if parameter . ty ( Interner ) . map ( | x | x . kind ( Interner ) )
= = Some ( & TyKind ::Error )
{
return true ;
}
if let Some ( ConstValue ::Concrete ( c ) ) =
parameter . constant ( Interner ) . map ( | x | x . data ( Interner ) . value )
{
if c . interned = = ConstScalar ::Unknown {
return true ;
}
}
let default_parameter = match default_parameters . get ( i ) {
Some ( x ) = > x ,
None = > return true ,
} ;
let actual_default = default_parameter
. clone ( )
. substitute ( Interner , & subst_prefix ( parameters , i ) ) ;
parameter ! = & actual_default
}
2021-03-13 14:44:51 +01:00
let mut default_from = 0 ;
2021-12-19 18:58:39 +02:00
for ( i , parameter ) in parameters . iter ( Interner ) . enumerate ( ) {
2022-03-09 22:20:24 +03:30
if should_show ( parameter , & default_parameters , i , parameters ) {
default_from = i + 1 ;
2020-02-14 15:01:25 +01:00
}
}
2021-12-19 18:58:39 +02:00
& parameters . as_slice ( Interner ) [ 0 .. default_from ]
2020-02-14 15:01:25 +01:00
}
2021-03-13 14:44:51 +01:00
}
} else {
2021-12-19 18:58:39 +02:00
parameters . as_slice ( Interner )
2021-03-13 14:44:51 +01:00
} ;
2020-05-10 18:09:22 +02:00
if ! parameters_to_write . is_empty ( ) {
write! ( f , " < " ) ? ;
2022-01-03 01:34:33 +02:00
if f . display_target . is_source_code ( ) {
let mut first = true ;
for generic_arg in parameters_to_write {
if ! first {
write! ( f , " , " ) ? ;
}
first = false ;
if generic_arg . ty ( Interner ) . map ( | ty | ty . kind ( Interner ) )
= = Some ( & TyKind ::Error )
{
write! ( f , " _ " ) ? ;
} else {
generic_arg . hir_fmt ( f ) ? ;
}
}
} else {
f . write_joined ( parameters_to_write , " , " ) ? ;
}
2020-05-10 18:09:22 +02:00
write! ( f , " > " ) ? ;
}
2020-02-14 15:01:25 +01:00
}
}
2021-03-13 17:36:07 +01:00
TyKind ::AssociatedType ( assoc_type_id , parameters ) = > {
let type_alias = from_assoc_type_id ( * assoc_type_id ) ;
2020-03-13 16:05:46 +01:00
let trait_ = match type_alias . lookup ( f . db . upcast ( ) ) . container {
2021-12-07 17:31:26 +01:00
ItemContainerId ::TraitId ( it ) = > it ,
2020-02-14 15:01:25 +01:00
_ = > panic! ( " not an associated type " ) ,
} ;
2020-03-31 19:00:23 +03:00
let trait_ = f . db . trait_data ( trait_ ) ;
2021-03-13 17:36:07 +01:00
let type_alias_data = f . db . type_alias_data ( type_alias ) ;
2020-10-27 20:23:09 +01:00
2020-10-28 11:20:05 +01:00
// Use placeholder associated types when the target is test (https://rust-lang.github.io/chalk/book/clauses/type_equality.html#placeholder-associated-types)
2020-10-28 15:09:47 +01:00
if f . display_target . is_test ( ) {
write! ( f , " {}::{} " , trait_ . name , type_alias_data . name ) ? ;
2021-12-19 18:58:39 +02:00
if parameters . len ( Interner ) > 0 {
2020-10-27 20:23:09 +01:00
write! ( f , " < " ) ? ;
2021-12-19 18:58:39 +02:00
f . write_joined ( & * parameters . as_slice ( Interner ) , " , " ) ? ;
2020-10-27 20:23:09 +01:00
write! ( f , " > " ) ? ;
}
} else {
2021-03-13 17:36:07 +01:00
let projection_ty = ProjectionTy {
2021-03-14 16:26:12 +01:00
associated_ty_id : to_assoc_type_id ( type_alias ) ,
substitution : parameters . clone ( ) ,
2021-03-13 17:36:07 +01:00
} ;
2020-10-28 15:09:47 +01:00
projection_ty . hir_fmt ( f ) ? ;
2020-02-14 15:01:25 +01:00
}
}
2021-04-05 15:38:28 +03:00
TyKind ::Foreign ( type_alias ) = > {
2021-03-13 17:23:19 +01:00
let type_alias = f . db . type_alias_data ( from_foreign_def_id ( * type_alias ) ) ;
2020-09-13 10:24:19 +08:00
write! ( f , " {} " , type_alias . name ) ? ;
}
2021-03-13 14:44:51 +01:00
TyKind ::OpaqueType ( opaque_ty_id , parameters ) = > {
2021-03-13 20:05:47 +01:00
let impl_trait_id = f . db . lookup_intern_impl_trait_id ( ( * opaque_ty_id ) . into ( ) ) ;
match impl_trait_id {
ImplTraitId ::ReturnTypeImplTrait ( func , idx ) = > {
2020-03-04 23:00:44 +01:00
let datas =
f . db . return_type_impl_traits ( func ) . expect ( " impl trait id without data " ) ;
let data = ( * datas )
. as_ref ( )
. map ( | rpit | rpit . impl_traits [ idx as usize ] . bounds . clone ( ) ) ;
2021-12-19 18:58:39 +02:00
let bounds = data . substitute ( Interner , & parameters ) ;
2021-06-15 23:28:37 +05:00
let krate = func . lookup ( f . db . upcast ( ) ) . module ( f . db . upcast ( ) ) . krate ( ) ;
2021-06-15 13:53:20 +05:00
write_bounds_like_dyn_trait_with_prefix (
" impl " ,
bounds . skip_binders ( ) ,
2021-06-15 23:28:37 +05:00
SizedByDefault ::Sized { anchor : krate } ,
2021-06-15 13:53:20 +05:00
f ,
) ? ;
2020-09-10 20:01:23 +08:00
// FIXME: it would maybe be good to distinguish this from the alias type (when debug printing), and to show the substitution
2020-03-04 23:00:44 +01:00
}
2021-03-13 20:05:47 +01:00
ImplTraitId ::AsyncBlockTypeImplTrait ( .. ) = > {
2020-09-10 20:01:23 +08:00
write! ( f , " impl Future<Output = " ) ? ;
2021-12-19 18:58:39 +02:00
parameters . at ( Interner , 0 ) . hir_fmt ( f ) ? ;
2020-09-10 20:01:23 +08:00
write! ( f , " > " ) ? ;
}
}
2020-03-04 23:00:44 +01:00
}
2021-03-13 14:44:51 +01:00
TyKind ::Closure ( .. , substs ) = > {
2021-03-22 15:17:01 +08:00
if f . display_target . is_source_code ( ) {
return Err ( HirDisplayError ::DisplaySourceCodeError (
DisplaySourceCodeError ::Closure ,
) ) ;
}
2021-12-19 18:58:39 +02:00
let sig = substs . at ( Interner , 0 ) . assert_ty_ref ( Interner ) . callable_sig ( f . db ) ;
2020-03-27 18:56:18 +01:00
if let Some ( sig ) = sig {
if sig . params ( ) . is_empty ( ) {
write! ( f , " || " ) ? ;
2021-11-22 18:27:03 +01:00
} else if f . should_truncate ( ) {
2020-03-27 18:56:18 +01:00
write! ( f , " |{}| " , TYPE_HINT_TRUNCATION ) ? ;
} else {
write! ( f , " | " ) ? ;
f . write_joined ( sig . params ( ) , " , " ) ? ;
write! ( f , " | " ) ? ;
} ;
2020-07-20 23:50:41 +03:00
2021-04-06 14:16:07 +02:00
write! ( f , " -> " ) ? ;
sig . ret ( ) . hir_fmt ( f ) ? ;
2020-02-14 15:01:25 +01:00
} else {
2020-03-27 18:56:18 +01:00
write! ( f , " {{closure}} " ) ? ;
}
2020-02-14 15:01:25 +01:00
}
2021-03-13 19:47:34 +01:00
TyKind ::Placeholder ( idx ) = > {
let id = from_placeholder_idx ( f . db , * idx ) ;
2020-03-13 16:05:46 +01:00
let generics = generics ( f . db . upcast ( ) , id . parent ) ;
2022-03-09 22:20:24 +03:30
let param_data = & generics . params . type_or_consts [ id . local_id ] ;
2021-12-29 17:05:59 +03:30
match param_data {
TypeOrConstParamData ::TypeParamData ( p ) = > match p . provenance {
TypeParamProvenance ::TypeParamList | TypeParamProvenance ::TraitSelf = > {
write! ( f , " {} " , p . name . clone ( ) . unwrap_or_else ( Name ::missing ) ) ?
}
TypeParamProvenance ::ArgumentImplTrait = > {
2022-03-09 22:20:24 +03:30
let substs = generics . placeholder_subst ( f . db ) ;
2021-12-29 17:05:59 +03:30
let bounds =
f . db . generic_predicates ( id . parent )
. iter ( )
. map ( | pred | pred . clone ( ) . substitute ( Interner , & substs ) )
. filter ( | wc | match & wc . skip_binders ( ) {
WhereClause ::Implemented ( tr ) = > {
& tr . self_type_parameter ( Interner ) = = self
}
WhereClause ::AliasEq ( AliasEq {
alias : AliasTy ::Projection ( proj ) ,
ty : _ ,
} ) = > & proj . self_type_parameter ( Interner ) = = self ,
_ = > false ,
} )
. collect ::< Vec < _ > > ( ) ;
let krate = id . parent . module ( f . db . upcast ( ) ) . krate ( ) ;
write_bounds_like_dyn_trait_with_prefix (
" impl " ,
& bounds ,
SizedByDefault ::Sized { anchor : krate } ,
f ,
) ? ;
}
} ,
TypeOrConstParamData ::ConstParamData ( p ) = > {
write! ( f , " {} " , p . name ) ? ;
2020-02-14 15:01:25 +01:00
}
}
}
2021-04-06 11:45:41 +02:00
TyKind ::BoundVar ( idx ) = > idx . hir_fmt ( f ) ? ,
2021-03-21 13:22:22 +01:00
TyKind ::Dyn ( dyn_ty ) = > {
write_bounds_like_dyn_trait_with_prefix (
" dyn " ,
dyn_ty . bounds . skip_binders ( ) . interned ( ) ,
2021-06-15 13:53:20 +05:00
SizedByDefault ::NotSized ,
2021-03-21 13:22:22 +01:00
f ,
) ? ;
2020-02-14 15:01:25 +01:00
}
2021-03-13 14:44:51 +01:00
TyKind ::Alias ( AliasTy ::Projection ( p_ty ) ) = > p_ty . hir_fmt ( f ) ? ,
TyKind ::Alias ( AliasTy ::Opaque ( opaque_ty ) ) = > {
2021-03-13 20:05:47 +01:00
let impl_trait_id = f . db . lookup_intern_impl_trait_id ( opaque_ty . opaque_ty_id . into ( ) ) ;
match impl_trait_id {
ImplTraitId ::ReturnTypeImplTrait ( func , idx ) = > {
2020-03-04 23:00:44 +01:00
let datas =
f . db . return_type_impl_traits ( func ) . expect ( " impl trait id without data " ) ;
let data = ( * datas )
. as_ref ( )
. map ( | rpit | rpit . impl_traits [ idx as usize ] . bounds . clone ( ) ) ;
2021-12-19 18:58:39 +02:00
let bounds = data . substitute ( Interner , & opaque_ty . substitution ) ;
2021-06-15 23:28:37 +05:00
let krate = func . lookup ( f . db . upcast ( ) ) . module ( f . db . upcast ( ) ) . krate ( ) ;
2021-06-15 13:53:20 +05:00
write_bounds_like_dyn_trait_with_prefix (
" impl " ,
bounds . skip_binders ( ) ,
2021-06-15 23:28:37 +05:00
SizedByDefault ::Sized { anchor : krate } ,
2021-06-15 13:53:20 +05:00
f ,
) ? ;
2020-09-10 20:01:23 +08:00
}
2021-03-13 20:05:47 +01:00
ImplTraitId ::AsyncBlockTypeImplTrait ( .. ) = > {
2020-09-10 20:01:23 +08:00
write! ( f , " {{async block}} " ) ? ;
2020-03-04 23:00:44 +01:00
}
} ;
}
2021-04-05 15:37:11 +03:00
TyKind ::Error = > {
2020-12-12 18:18:19 +01:00
if f . display_target . is_source_code ( ) {
return Err ( HirDisplayError ::DisplaySourceCodeError (
DisplaySourceCodeError ::UnknownType ,
) ) ;
}
write! ( f , " {{unknown}} " ) ? ;
}
2021-03-13 14:44:51 +01:00
TyKind ::InferenceVar ( .. ) = > write! ( f , " _ " ) ? ,
2021-04-08 13:51:04 +02:00
TyKind ::Generator ( .. ) = > write! ( f , " {{generator}} " ) ? ,
TyKind ::GeneratorWitness ( .. ) = > write! ( f , " {{generator witness}} " ) ? ,
2020-02-14 15:01:25 +01:00
}
Ok ( ( ) )
}
}
2021-02-28 22:12:07 +01:00
impl HirDisplay for CallableSig {
2020-12-09 18:06:56 +01:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
write! ( f , " fn( " ) ? ;
f . write_joined ( self . params ( ) , " , " ) ? ;
if self . is_varargs {
if self . params ( ) . is_empty ( ) {
write! ( f , " ... " ) ? ;
} else {
write! ( f , " , ... " ) ? ;
}
}
write! ( f , " ) " ) ? ;
let ret = self . ret ( ) ;
2021-04-03 20:22:59 +02:00
if ! ret . is_unit ( ) {
2021-04-06 14:16:07 +02:00
write! ( f , " -> " ) ? ;
ret . hir_fmt ( f ) ? ;
2020-12-09 18:06:56 +01:00
}
Ok ( ( ) )
}
}
2021-01-12 20:19:13 +01:00
fn fn_traits ( db : & dyn DefDatabase , trait_ : TraitId ) -> impl Iterator < Item = TraitId > {
2021-03-09 19:09:02 +01:00
let krate = trait_ . lookup ( db ) . container . krate ( ) ;
2021-06-29 17:35:37 +02:00
utils ::fn_traits ( db , krate )
2021-01-12 20:19:13 +01:00
}
2021-06-15 13:53:20 +05:00
#[ derive(Clone, Copy, PartialEq, Eq) ]
pub enum SizedByDefault {
NotSized ,
2021-06-15 23:28:37 +05:00
Sized { anchor : CrateId } ,
}
impl SizedByDefault {
fn is_sized_trait ( self , trait_ : TraitId , db : & dyn DefDatabase ) -> bool {
match self {
Self ::NotSized = > false ,
Self ::Sized { anchor } = > {
2021-12-10 20:01:24 +01:00
let sized_trait = db
. lang_item ( anchor , SmolStr ::new_inline ( " sized " ) )
. and_then ( | lang_item | lang_item . as_trait ( ) ) ;
2021-06-15 23:28:37 +05:00
Some ( trait_ ) = = sized_trait
}
}
}
2021-06-15 13:53:20 +05:00
}
2021-02-20 20:43:04 +01:00
pub fn write_bounds_like_dyn_trait_with_prefix (
prefix : & str ,
2021-03-21 13:22:22 +01:00
predicates : & [ QuantifiedWhereClause ] ,
2021-06-15 13:53:20 +05:00
default_sized : SizedByDefault ,
2021-02-20 20:43:04 +01:00
f : & mut HirFormatter ,
) -> Result < ( ) , HirDisplayError > {
write! ( f , " {} " , prefix ) ? ;
2021-06-15 23:28:37 +05:00
if ! predicates . is_empty ( )
| | predicates . is_empty ( ) & & matches! ( default_sized , SizedByDefault ::Sized { .. } )
{
2021-02-20 20:43:04 +01:00
write! ( f , " " ) ? ;
2021-06-15 13:53:20 +05:00
write_bounds_like_dyn_trait ( predicates , default_sized , f )
2021-02-20 20:43:04 +01:00
} else {
Ok ( ( ) )
}
}
fn write_bounds_like_dyn_trait (
2021-03-21 13:22:22 +01:00
predicates : & [ QuantifiedWhereClause ] ,
2021-06-15 13:53:20 +05:00
default_sized : SizedByDefault ,
2020-03-13 16:05:46 +01:00
f : & mut HirFormatter ,
2020-04-25 16:57:59 +02:00
) -> Result < ( ) , HirDisplayError > {
2020-02-14 15:01:25 +01:00
// Note: This code is written to produce nice results (i.e.
// corresponding to surface Rust) for types that can occur in
// actual Rust. It will have weird results if the predicates
// aren't as expected (i.e. self types = $0, projection
// predicates for a certain trait come after the Implemented
// predicate for that trait).
let mut first = true ;
let mut angle_open = false ;
2021-01-12 20:19:13 +01:00
let mut is_fn_trait = false ;
2021-06-15 23:28:37 +05:00
let mut is_sized = false ;
2020-02-14 15:01:25 +01:00
for p in predicates . iter ( ) {
2021-03-21 13:22:22 +01:00
match p . skip_binders ( ) {
2021-03-20 10:46:36 +01:00
WhereClause ::Implemented ( trait_ref ) = > {
2021-03-18 21:53:19 +01:00
let trait_ = trait_ref . hir_trait_id ( ) ;
2021-06-15 23:28:37 +05:00
if default_sized . is_sized_trait ( trait_ , f . db . upcast ( ) ) {
is_sized = true ;
if matches! ( default_sized , SizedByDefault ::Sized { .. } ) {
2021-06-16 03:23:04 +05:00
// Don't print +Sized, but rather +?Sized if absent.
continue ;
2021-06-15 13:53:20 +05:00
}
2021-06-16 03:23:04 +05:00
}
2021-01-12 20:19:13 +01:00
if ! is_fn_trait {
is_fn_trait = fn_traits ( f . db . upcast ( ) , trait_ ) . any ( | it | it = = trait_ ) ;
}
if ! is_fn_trait & & angle_open {
2020-02-14 15:01:25 +01:00
write! ( f , " > " ) ? ;
2020-08-13 22:13:34 +02:00
angle_open = false ;
2020-02-14 15:01:25 +01:00
}
if ! first {
write! ( f , " + " ) ? ;
}
2021-04-08 13:51:04 +02:00
// We assume that the self type is ^0.0 (i.e. the
2020-02-14 15:01:25 +01:00
// existential) here, which is the only thing that's
// possible in actual Rust, and hence don't print it
2021-01-12 20:19:13 +01:00
write! ( f , " {} " , f . db . trait_data ( trait_ ) . name ) ? ;
2021-12-19 18:58:39 +02:00
if let [ _ , params @ .. ] = & * trait_ref . substitution . as_slice ( Interner ) {
2021-01-12 20:19:13 +01:00
if is_fn_trait {
2021-04-01 21:04:02 +02:00
if let Some ( args ) =
2021-12-19 18:58:39 +02:00
params . first ( ) . and_then ( | it | it . assert_ty_ref ( Interner ) . as_tuple ( ) )
2021-04-01 21:04:02 +02:00
{
2021-01-12 20:19:13 +01:00
write! ( f , " ( " ) ? ;
2021-12-19 18:58:39 +02:00
f . write_joined ( args . as_slice ( Interner ) , " , " ) ? ;
2021-01-12 20:19:13 +01:00
write! ( f , " ) " ) ? ;
}
} else if ! params . is_empty ( ) {
write! ( f , " < " ) ? ;
f . write_joined ( params , " , " ) ? ;
// there might be assoc type bindings, so we leave the angle brackets open
angle_open = true ;
}
2020-02-14 15:01:25 +01:00
}
}
2021-03-20 10:46:36 +01:00
WhereClause ::AliasEq ( alias_eq ) if is_fn_trait = > {
2021-01-12 20:19:13 +01:00
is_fn_trait = false ;
2021-05-25 15:23:52 +02:00
if ! alias_eq . ty . is_unit ( ) {
write! ( f , " -> " ) ? ;
alias_eq . ty . hir_fmt ( f ) ? ;
}
2021-01-12 20:19:13 +01:00
}
2021-03-20 10:46:36 +01:00
WhereClause ::AliasEq ( AliasEq { ty , alias } ) = > {
2020-02-14 15:01:25 +01:00
// in types in actual Rust, these will always come
// after the corresponding Implemented predicate
if angle_open {
write! ( f , " , " ) ? ;
} else {
write! ( f , " < " ) ? ;
angle_open = true ;
}
2021-03-19 02:07:15 +01:00
if let AliasTy ::Projection ( proj ) = alias {
let type_alias =
f . db . type_alias_data ( from_assoc_type_id ( proj . associated_ty_id ) ) ;
write! ( f , " {} = " , type_alias . name ) ? ;
}
ty . hir_fmt ( f ) ? ;
2020-02-14 15:01:25 +01:00
}
2021-04-08 13:51:04 +02:00
// FIXME implement these
WhereClause ::LifetimeOutlives ( _ ) = > { }
WhereClause ::TypeOutlives ( _ ) = > { }
2020-02-14 15:01:25 +01:00
}
first = false ;
}
if angle_open {
write! ( f , " > " ) ? ;
}
2021-06-15 23:28:37 +05:00
if matches! ( default_sized , SizedByDefault ::Sized { .. } ) {
if ! is_sized {
2021-06-15 13:53:20 +05:00
write! ( f , " {}?Sized " , if first { " " } else { " + " } ) ? ;
} else if first {
write! ( f , " Sized " ) ? ;
}
}
2020-02-14 15:01:25 +01:00
Ok ( ( ) )
}
2021-04-07 20:26:27 +02:00
fn fmt_trait_ref ( tr : & TraitRef , f : & mut HirFormatter , use_as : bool ) -> Result < ( ) , HirDisplayError > {
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2020-02-14 15:01:25 +01:00
2021-12-19 18:58:39 +02:00
tr . self_type_parameter ( Interner ) . hir_fmt ( f ) ? ;
2021-04-07 20:26:27 +02:00
if use_as {
write! ( f , " as " ) ? ;
} else {
write! ( f , " : " ) ? ;
2020-02-14 15:01:25 +01:00
}
2021-04-07 20:26:27 +02:00
write! ( f , " {} " , f . db . trait_data ( tr . hir_trait_id ( ) ) . name ) ? ;
2021-12-19 18:58:39 +02:00
if tr . substitution . len ( Interner ) > 1 {
2021-04-07 20:26:27 +02:00
write! ( f , " < " ) ? ;
2021-12-19 18:58:39 +02:00
f . write_joined ( & tr . substitution . as_slice ( Interner ) [ 1 .. ] , " , " ) ? ;
2021-04-07 20:26:27 +02:00
write! ( f , " > " ) ? ;
}
Ok ( ( ) )
2020-02-14 15:01:25 +01:00
}
impl HirDisplay for TraitRef {
2020-04-25 16:57:59 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2021-04-07 20:26:27 +02:00
fmt_trait_ref ( self , f , false )
2020-02-14 15:01:25 +01:00
}
}
2021-03-20 10:46:36 +01:00
impl HirDisplay for WhereClause {
2020-04-25 16:57:59 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-02-14 15:01:25 +01:00
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
match self {
2021-03-20 10:46:36 +01:00
WhereClause ::Implemented ( trait_ref ) = > trait_ref . hir_fmt ( f ) ? ,
WhereClause ::AliasEq ( AliasEq { alias : AliasTy ::Projection ( projection_ty ) , ty } ) = > {
2020-02-14 15:01:25 +01:00
write! ( f , " < " ) ? ;
2021-04-07 20:26:27 +02:00
fmt_trait_ref ( & projection_ty . trait_ref ( f . db ) , f , true ) ? ;
2020-02-14 15:01:25 +01:00
write! (
f ,
2020-10-28 17:58:16 +01:00
" >::{} = " ,
2021-03-19 02:07:15 +01:00
f . db . type_alias_data ( from_assoc_type_id ( projection_ty . associated_ty_id ) ) . name ,
2020-02-14 15:01:25 +01:00
) ? ;
2021-03-19 02:07:15 +01:00
ty . hir_fmt ( f ) ? ;
2020-02-14 15:01:25 +01:00
}
2021-03-20 10:51:00 +01:00
WhereClause ::AliasEq ( _ ) = > write! ( f , " {{error}} " ) ? ,
2021-04-08 13:51:04 +02:00
// FIXME implement these
WhereClause ::TypeOutlives ( .. ) = > { }
WhereClause ::LifetimeOutlives ( .. ) = > { }
2020-02-14 15:01:25 +01:00
}
Ok ( ( ) )
}
}
2021-04-05 20:46:15 +02:00
impl HirDisplay for LifetimeOutlives {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
self . a . hir_fmt ( f ) ? ;
write! ( f , " : " ) ? ;
self . b . hir_fmt ( f )
}
}
2020-12-11 13:49:32 +01:00
impl HirDisplay for Lifetime {
2021-04-05 20:46:15 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
self . interned ( ) . hir_fmt ( f )
}
}
impl HirDisplay for LifetimeData {
2020-12-11 13:49:32 +01:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
2021-04-06 11:45:41 +02:00
LifetimeData ::BoundVar ( idx ) = > idx . hir_fmt ( f ) ,
2021-04-05 20:46:15 +02:00
LifetimeData ::InferenceVar ( _ ) = > write! ( f , " _ " ) ,
LifetimeData ::Placeholder ( idx ) = > {
let id = lt_from_placeholder_idx ( f . db , * idx ) ;
2020-12-11 13:49:32 +01:00
let generics = generics ( f . db . upcast ( ) , id . parent ) ;
let param_data = & generics . params . lifetimes [ id . local_id ] ;
2021-04-05 20:46:15 +02:00
write! ( f , " {} " , param_data . name )
2020-12-11 13:49:32 +01:00
}
2021-04-05 20:46:15 +02:00
LifetimeData ::Static = > write! ( f , " 'static " ) ,
LifetimeData ::Empty ( _ ) = > Ok ( ( ) ) ,
LifetimeData ::Erased = > Ok ( ( ) ) ,
LifetimeData ::Phantom ( _ , _ ) = > Ok ( ( ) ) ,
2020-12-11 13:49:32 +01:00
}
}
}
2021-03-20 11:23:59 +01:00
impl HirDisplay for DomainGoal {
2020-04-25 16:57:59 +02:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-10-06 14:40:27 +02:00
match self {
2021-03-20 11:23:59 +01:00
DomainGoal ::Holds ( wc ) = > {
write! ( f , " Holds( " ) ? ;
wc . hir_fmt ( f ) ? ;
2021-04-08 13:51:04 +02:00
write! ( f , " ) " ) ? ;
2020-10-28 17:58:16 +01:00
}
2021-04-08 13:51:04 +02:00
_ = > write! ( f , " ? " ) ? ,
2020-10-06 14:40:27 +02:00
}
2021-04-08 13:51:04 +02:00
Ok ( ( ) )
2020-02-14 15:01:25 +01:00
}
}
2021-03-14 20:03:39 +08:00
pub fn write_visibility (
module_id : ModuleId ,
vis : Visibility ,
f : & mut HirFormatter ,
) -> Result < ( ) , HirDisplayError > {
match vis {
Visibility ::Public = > write! ( f , " pub " ) ,
Visibility ::Module ( vis_id ) = > {
let def_map = module_id . def_map ( f . db . upcast ( ) ) ;
let root_module_id = def_map . module_id ( def_map . root ( ) ) ;
if vis_id = = module_id {
// pub(self) or omitted
Ok ( ( ) )
} else if root_module_id = = vis_id {
write! ( f , " pub(crate) " )
} else if module_id . containing_module ( f . db . upcast ( ) ) = = Some ( vis_id ) {
write! ( f , " pub(super) " )
} else {
write! ( f , " pub(in ...) " )
}
}
}
}
impl HirDisplay for TypeRef {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
TypeRef ::Never = > write! ( f , " ! " ) ? ,
TypeRef ::Placeholder = > write! ( f , " _ " ) ? ,
TypeRef ::Tuple ( elems ) = > {
write! ( f , " ( " ) ? ;
f . write_joined ( elems , " , " ) ? ;
if elems . len ( ) = = 1 {
write! ( f , " , " ) ? ;
}
write! ( f , " ) " ) ? ;
}
TypeRef ::Path ( path ) = > path . hir_fmt ( f ) ? ,
TypeRef ::RawPtr ( inner , mutability ) = > {
let mutability = match mutability {
hir_def ::type_ref ::Mutability ::Shared = > " *const " ,
hir_def ::type_ref ::Mutability ::Mut = > " *mut " ,
} ;
write! ( f , " {} " , mutability ) ? ;
inner . hir_fmt ( f ) ? ;
}
TypeRef ::Reference ( inner , lifetime , mutability ) = > {
let mutability = match mutability {
hir_def ::type_ref ::Mutability ::Shared = > " " ,
hir_def ::type_ref ::Mutability ::Mut = > " mut " ,
} ;
write! ( f , " & " ) ? ;
if let Some ( lifetime ) = lifetime {
write! ( f , " {} " , lifetime . name ) ? ;
}
write! ( f , " {} " , mutability ) ? ;
inner . hir_fmt ( f ) ? ;
}
2021-05-12 04:39:48 -07:00
TypeRef ::Array ( inner , len ) = > {
2021-03-14 20:03:39 +08:00
write! ( f , " [ " ) ? ;
inner . hir_fmt ( f ) ? ;
2021-05-12 04:39:48 -07:00
write! ( f , " ; {}] " , len ) ? ;
2021-03-14 20:03:39 +08:00
}
TypeRef ::Slice ( inner ) = > {
write! ( f , " [ " ) ? ;
inner . hir_fmt ( f ) ? ;
write! ( f , " ] " ) ? ;
}
2022-02-15 14:39:22 +00:00
TypeRef ::Fn ( parameters , is_varargs ) = > {
2022-02-18 09:12:52 +00:00
// FIXME: Function pointer qualifiers.
2021-03-14 20:03:39 +08:00
write! ( f , " fn( " ) ? ;
2022-02-21 10:29:38 +00:00
if let Some ( ( ( _ , return_type ) , function_parameters ) ) = parameters . split_last ( ) {
for index in 0 .. function_parameters . len ( ) {
let ( param_name , param_type ) = & function_parameters [ index ] ;
if let Some ( name ) = param_name {
write! ( f , " {}: " , name ) ? ;
}
2022-02-15 19:27:56 +00:00
2022-02-21 10:29:38 +00:00
param_type . hir_fmt ( f ) ? ;
2022-02-15 14:39:22 +00:00
2022-02-21 10:29:38 +00:00
if index ! = function_parameters . len ( ) - 1 {
write! ( f , " , " ) ? ;
}
2022-02-15 14:39:22 +00:00
}
2022-02-21 10:29:38 +00:00
if * is_varargs {
write! ( f , " {}... " , if parameters . len ( ) = = 1 { " " } else { " , " } ) ? ;
}
write! ( f , " ) " ) ? ;
match & return_type {
2022-02-15 14:58:06 +00:00
TypeRef ::Tuple ( tup ) if tup . is_empty ( ) = > { }
_ = > {
write! ( f , " -> " ) ? ;
2022-02-21 10:29:38 +00:00
return_type . hir_fmt ( f ) ? ;
2022-02-15 14:58:06 +00:00
}
2021-03-14 20:03:39 +08:00
}
}
}
TypeRef ::ImplTrait ( bounds ) = > {
write! ( f , " impl " ) ? ;
f . write_joined ( bounds , " + " ) ? ;
}
TypeRef ::DynTrait ( bounds ) = > {
write! ( f , " dyn " ) ? ;
f . write_joined ( bounds , " + " ) ? ;
}
2021-04-12 16:24:48 +02:00
TypeRef ::Macro ( macro_call ) = > {
let macro_call = macro_call . to_node ( f . db . upcast ( ) ) ;
2021-05-06 19:59:54 +02:00
let ctx = body ::LowerCtx ::with_hygiene ( f . db . upcast ( ) , & Hygiene ::new_unhygienic ( ) ) ;
2021-04-12 16:24:48 +02:00
match macro_call . path ( ) {
2021-05-06 23:23:50 +02:00
Some ( path ) = > match Path ::from_src ( path , & ctx ) {
2021-04-12 16:24:48 +02:00
Some ( path ) = > path . hir_fmt ( f ) ? ,
None = > write! ( f , " {{macro}} " ) ? ,
} ,
None = > write! ( f , " {{macro}} " ) ? ,
}
write! ( f , " !(..) " ) ? ;
}
TypeRef ::Error = > write! ( f , " {{error}} " ) ? ,
2021-03-14 20:03:39 +08:00
}
Ok ( ( ) )
}
}
impl HirDisplay for TypeBound {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
2021-06-06 23:41:15 +05:00
TypeBound ::Path ( path , modifier ) = > {
match modifier {
TraitBoundModifier ::None = > ( ) ,
TraitBoundModifier ::Maybe = > write! ( f , " ? " ) ? ,
}
path . hir_fmt ( f )
}
2021-03-14 20:03:39 +08:00
TypeBound ::Lifetime ( lifetime ) = > write! ( f , " {} " , lifetime . name ) ,
2021-06-30 01:34:51 +02:00
TypeBound ::ForLifetime ( lifetimes , path ) = > {
write! ( f , " for<{}> " , lifetimes . iter ( ) . format ( " , " ) ) ? ;
path . hir_fmt ( f )
}
2021-03-14 20:03:39 +08:00
TypeBound ::Error = > write! ( f , " {{error}} " ) ,
}
}
}
impl HirDisplay for Path {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match ( self . type_anchor ( ) , self . kind ( ) ) {
( Some ( anchor ) , _ ) = > {
write! ( f , " < " ) ? ;
anchor . hir_fmt ( f ) ? ;
write! ( f , " > " ) ? ;
}
( _ , PathKind ::Plain ) = > { }
2021-09-07 17:49:46 +08:00
( _ , PathKind ::Abs ) = > { }
2021-03-14 20:03:39 +08:00
( _ , PathKind ::Crate ) = > write! ( f , " crate " ) ? ,
( _ , PathKind ::Super ( 0 ) ) = > write! ( f , " self " ) ? ,
( _ , PathKind ::Super ( n ) ) = > {
2021-09-07 17:49:46 +08:00
for i in 0 .. * n {
if i > 0 {
write! ( f , " :: " ) ? ;
}
write! ( f , " super " ) ? ;
2021-03-14 20:03:39 +08:00
}
}
2022-02-25 18:38:51 +01:00
( _ , PathKind ::DollarCrate ( id ) ) = > {
// Resolve `$crate` to the crate's display name.
// FIXME: should use the dependency name instead if available, but that depends on
// the crate invoking `HirDisplay`
let crate_graph = f . db . crate_graph ( ) ;
let name = crate_graph [ * id ]
. display_name
. as_ref ( )
. map ( | name | name . canonical_name ( ) )
. unwrap_or ( " $crate " ) ;
write! ( f , " {name} " ) ?
}
2021-03-14 20:03:39 +08:00
}
for ( seg_idx , segment ) in self . segments ( ) . iter ( ) . enumerate ( ) {
2021-09-07 16:54:02 +08:00
if ! matches! ( self . kind ( ) , PathKind ::Plain ) | | seg_idx > 0 {
2021-03-14 20:03:39 +08:00
write! ( f , " :: " ) ? ;
}
write! ( f , " {} " , segment . name ) ? ;
if let Some ( generic_args ) = segment . args_and_bindings {
// We should be in type context, so format as `Foo<Bar>` instead of `Foo::<Bar>`.
// Do we actually format expressions?
2021-11-19 19:58:00 +01:00
if generic_args . desugared_from_fn {
// First argument will be a tuple, which already includes the parentheses.
2021-11-21 02:39:22 +02:00
// If the tuple only contains 1 item, write it manually to avoid the trailing `,`.
if let hir_def ::path ::GenericArg ::Type ( TypeRef ::Tuple ( v ) ) =
& generic_args . args [ 0 ]
{
if v . len ( ) = = 1 {
write! ( f , " ( " ) ? ;
v [ 0 ] . hir_fmt ( f ) ? ;
write! ( f , " ) " ) ? ;
} else {
generic_args . args [ 0 ] . hir_fmt ( f ) ? ;
}
}
2021-11-19 19:58:00 +01:00
if let Some ( ret ) = & generic_args . bindings [ 0 ] . type_ref {
2021-11-21 02:39:22 +02:00
if ! matches! ( ret , TypeRef ::Tuple ( v ) if v . is_empty ( ) ) {
write! ( f , " -> " ) ? ;
ret . hir_fmt ( f ) ? ;
}
2021-11-19 19:58:00 +01:00
}
return Ok ( ( ) ) ;
}
2021-03-14 20:03:39 +08:00
write! ( f , " < " ) ? ;
let mut first = true ;
for arg in & generic_args . args {
if first {
first = false ;
if generic_args . has_self_type {
// FIXME: Convert to `<Ty as Trait>` form.
write! ( f , " Self = " ) ? ;
}
} else {
write! ( f , " , " ) ? ;
}
arg . hir_fmt ( f ) ? ;
}
for binding in & generic_args . bindings {
if first {
first = false ;
} else {
write! ( f , " , " ) ? ;
}
write! ( f , " {} " , binding . name ) ? ;
match & binding . type_ref {
Some ( ty ) = > {
write! ( f , " = " ) ? ;
ty . hir_fmt ( f ) ?
}
None = > {
write! ( f , " : " ) ? ;
f . write_joined ( & binding . bounds , " + " ) ? ;
}
}
}
write! ( f , " > " ) ? ;
}
}
Ok ( ( ) )
}
}
2021-04-01 21:04:02 +02:00
impl HirDisplay for hir_def ::path ::GenericArg {
2021-03-14 20:03:39 +08:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
2021-04-01 21:04:02 +02:00
hir_def ::path ::GenericArg ::Type ( ty ) = > ty . hir_fmt ( f ) ,
2022-03-09 22:20:24 +03:30
hir_def ::path ::GenericArg ::Const ( c ) = > write! ( f , " {} " , c ) ,
2021-04-01 21:04:02 +02:00
hir_def ::path ::GenericArg ::Lifetime ( lifetime ) = > write! ( f , " {} " , lifetime . name ) ,
2021-03-14 20:03:39 +08:00
}
}
}