2021-04-09 07:39:07 -05: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 03:58:53 -05:00
2021-06-29 10:35:37 -05:00
use std ::fmt ::{ self , Debug } ;
2019-03-14 16:03:39 -05:00
2021-06-15 13:28:37 -05:00
use base_db ::CrateId ;
2021-04-06 04:45:41 -05:00
use chalk_ir ::BoundVar ;
2020-04-25 09:57:59 -05:00
use hir_def ::{
2021-04-12 09:24:48 -05:00
body ,
2021-03-14 07:03:39 -05:00
db ::DefDatabase ,
find_path ,
generics ::TypeParamProvenance ,
2021-05-24 08:13:23 -05:00
intern ::{ Internable , Interned } ,
2021-03-14 07:03:39 -05:00
item_scope ::ItemInNs ,
2021-04-01 14:04:02 -05:00
path ::{ Path , PathKind } ,
2021-06-06 13:41:15 -05:00
type_ref ::{ TraitBoundModifier , TypeBound , TypeRef } ,
2021-03-14 07:03:39 -05:00
visibility ::Visibility ,
2021-06-15 13:28:37 -05:00
AssocContainerId , HasModule , Lookup , ModuleId , TraitId ,
2020-04-25 09:57:59 -05:00
} ;
2021-04-12 09:24:48 -05:00
use hir_expand ::{ hygiene ::Hygiene , name ::Name } ;
2021-06-29 18:34:51 -05:00
use itertools ::Itertools ;
2019-03-14 16:03:39 -05:00
2021-03-01 14:57:39 -06:00
use crate ::{
2021-06-29 10:35:37 -05:00
const_from_placeholder_idx ,
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 } ,
AdtId , AliasEq , AliasTy , CallableDefId , CallableSig , Const , ConstValue , DomainGoal , GenericArg ,
ImplTraitId , Interner , Lifetime , LifetimeData , LifetimeOutlives , Mutability , OpaqueTy ,
ProjectionTy , ProjectionTyExt , QuantifiedWhereClause , Scalar , TraitRef , TraitRefExt , Ty , TyExt ,
TyKind , WhereClause ,
2021-03-01 14:57:39 -06:00
} ;
2020-04-25 09:57:59 -05:00
pub struct HirFormatter < ' a > {
2020-03-13 10:05:46 -05:00
pub db : & ' a dyn HirDatabase ,
2020-04-25 09:57:59 -05:00
fmt : & ' a mut dyn fmt ::Write ,
2019-11-18 11:02:28 -06:00
buf : String ,
curr_size : usize ,
2020-01-22 08:44:05 -06:00
pub ( crate ) max_size : Option < usize > ,
2019-12-23 09:53:35 -06:00
omit_verbose_types : bool ,
2020-04-25 09:57:59 -05:00
display_target : DisplayTarget ,
2019-03-14 16:03:39 -05:00
}
pub trait HirDisplay {
2020-04-25 09:57:59 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > ;
2019-11-18 11:02:28 -06:00
2020-10-28 11:58:16 -05: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 07:42:34 -05:00
assert! (
! matches! ( display_target , DisplayTarget ::SourceCode { .. } ) ,
" HirDisplayWrapper cannot fail with DisplaySourceCodeError, use HirDisplay::hir_fmt directly instead "
) ;
2020-10-28 11:58:16 -05:00
HirDisplayWrapper { db , t : self , max_size , omit_verbose_types , display_target }
}
2020-04-25 09:57:59 -05:00
/// Returns a `Display`able type that is human-readable.
/// Use this for showing types to the user (e.g. diagnostics)
2020-03-13 10:05:46 -05:00
fn display < ' a > ( & ' a self , db : & ' a dyn HirDatabase ) -> HirDisplayWrapper < ' a , Self >
2019-03-14 16:03:39 -05:00
where
Self : Sized ,
{
2020-04-25 09:57:59 -05:00
HirDisplayWrapper {
db ,
t : self ,
max_size : None ,
omit_verbose_types : false ,
display_target : DisplayTarget ::Diagnostics ,
}
2019-11-18 11:02:28 -06:00
}
2020-04-25 09:57:59 -05: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 10:05:46 -05:00
fn display_truncated < ' a > (
2019-11-18 11:02:28 -06:00
& ' a self ,
2020-03-13 10:05:46 -05:00
db : & ' a dyn HirDatabase ,
2019-12-19 08:43:41 -06:00
max_size : Option < usize > ,
2020-03-13 10:05:46 -05:00
) -> HirDisplayWrapper < ' a , Self >
2019-11-18 11:02:28 -06:00
where
Self : Sized ,
{
2020-04-25 09:57:59 -05: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 06:29:42 -05: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 09:09:47 -05:00
fn display_test < ' a > ( & ' a self , db : & ' a dyn HirDatabase ) -> HirDisplayWrapper < ' a , Self >
where
Self : Sized ,
{
HirDisplayWrapper {
2020-10-28 06:29:42 -05:00
db ,
2020-10-28 09:09:47 -05:00
t : self ,
2020-10-28 06:29:42 -05:00
max_size : None ,
omit_verbose_types : false ,
2020-10-28 09:09:47 -05:00
display_target : DisplayTarget ::Test ,
}
2019-03-14 16:03:39 -05:00
}
}
2020-04-25 09:57:59 -05:00
impl < ' a > HirFormatter < ' a > {
2019-03-14 16:03:39 -05:00
pub fn write_joined < T : HirDisplay > (
& mut self ,
iter : impl IntoIterator < Item = T > ,
sep : & str ,
2020-04-25 09:57:59 -05:00
) -> Result < ( ) , HirDisplayError > {
2019-03-14 16:03:39 -05:00
let mut first = true ;
for e in iter {
if ! first {
write! ( self , " {} " , sep ) ? ;
}
first = false ;
e . hir_fmt ( self ) ? ;
}
Ok ( ( ) )
}
/// This allows using the `write!` macro directly with a `HirFormatter`.
2020-04-25 09:57:59 -05:00
pub fn write_fmt ( & mut self , args : fmt ::Arguments ) -> Result < ( ) , HirDisplayError > {
2019-11-18 11:02:28 -06: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 09:57:59 -05:00
self . fmt . write_str ( & self . buf ) . map_err ( HirDisplayError ::from )
2019-11-18 11:02:28 -06:00
}
pub fn should_truncate ( & self ) -> bool {
2019-12-19 08:43:41 -06:00
if let Some ( max_size ) = self . max_size {
2019-11-18 11:02:28 -06:00
self . curr_size > = max_size
} else {
false
}
2019-03-14 16:03:39 -05:00
}
2019-12-07 16:54:18 -06:00
2019-12-23 09:53:35 -06:00
pub fn omit_verbose_types ( & self ) -> bool {
self . omit_verbose_types
2019-12-07 16:54:18 -06:00
}
}
2020-04-25 09:57:59 -05:00
#[ derive(Clone, Copy) ]
2020-10-28 11:58:16 -05:00
pub enum DisplayTarget {
2020-04-25 09:57:59 -05: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 05:20:05 -05:00
/// Only for test purpose to keep real types
2020-10-28 09:09:47 -05:00
Test ,
2020-04-25 09:57:59 -05:00
}
2020-05-10 11:09:22 -05:00
impl DisplayTarget {
fn is_source_code ( & self ) -> bool {
2021-01-05 06:45:46 -06:00
matches! ( self , Self ::SourceCode { .. } )
2020-05-10 11:09:22 -05:00
}
2020-10-28 05:20:05 -05:00
fn is_test ( & self ) -> bool {
2020-10-28 09:09:47 -05:00
matches! ( self , Self ::Test )
2020-10-28 05:20:05 -05:00
}
2020-05-10 11:09:22 -05:00
}
2020-04-25 09:57:59 -05:00
#[ derive(Debug) ]
pub enum DisplaySourceCodeError {
PathNotFound ,
2020-12-12 11:18:19 -06:00
UnknownType ,
2021-03-22 02:17:01 -05:00
Closure ,
2020-04-25 09:57:59 -05: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 16:03:39 -05:00
2020-03-13 10:05:46 -05:00
impl < ' a , T > fmt ::Display for HirDisplayWrapper < ' a , T >
2019-03-14 16:03:39 -05:00
where
T : HirDisplay ,
{
fn fmt ( & self , f : & mut fmt ::Formatter ) -> fmt ::Result {
2020-04-25 09:57:59 -05:00
match self . t . hir_fmt ( & mut HirFormatter {
db : self . db ,
2019-11-18 11:02:28 -06:00
fmt : f ,
buf : String ::with_capacity ( 20 ) ,
curr_size : 0 ,
2020-04-25 09:57:59 -05: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 07:42:34 -05:00
panic! ( " HirDisplay::hir_fmt failed with DisplaySourceCodeError when calling Display::fmt! " )
2020-04-25 09:57:59 -05:00
}
}
2019-03-14 16:03:39 -05:00
}
}
2020-02-14 08:01:25 -06:00
const TYPE_HINT_TRUNCATION : & str = " … " ;
2021-03-14 07:03:39 -05:00
impl < T : HirDisplay > HirDisplay for & '_ T {
2020-04-25 09:57:59 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-02-14 08:01:25 -06:00
HirDisplay ::hir_fmt ( * self , f )
}
}
2021-05-24 08:13:23 -05: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 12:13:37 -06:00
impl HirDisplay for ProjectionTy {
2020-04-25 09:57:59 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-02-14 08:01:25 -06:00
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2021-02-28 12:13:37 -06:00
let trait_ = f . db . trait_data ( self . trait_ ( f . db ) ) ;
2021-04-06 07:16:07 -05:00
write! ( f , " < " ) ? ;
self . self_type_parameter ( & Interner ) . hir_fmt ( f ) ? ;
write! ( f , " as {} " , trait_ . name ) ? ;
2021-04-01 14:04:02 -05:00
if self . substitution . len ( & Interner ) > 1 {
2021-02-28 12:13:37 -06:00
write! ( f , " < " ) ? ;
2021-04-08 11:25:18 -05:00
f . write_joined ( & self . substitution . as_slice ( & Interner ) [ 1 .. ] , " , " ) ? ;
2021-02-28 12:13:37 -06:00
write! ( f , " > " ) ? ;
}
2021-03-14 10:26:12 -05:00
write! ( f , " >::{} " , f . db . type_alias_data ( from_assoc_type_id ( self . associated_ty_id ) ) . name ) ? ;
2021-02-28 12:13:37 -06:00
Ok ( ( ) )
}
}
2021-03-18 20:07:15 -05: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-04-01 14:04:02 -05:00
self . substitution . at ( & Interner , 0 ) . hir_fmt ( f )
}
}
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 06:51:04 -05:00
crate ::GenericArgData ::Lifetime ( lt ) = > lt . hir_fmt ( f ) ,
crate ::GenericArgData ::Const ( c ) = > c . hir_fmt ( f ) ,
2021-04-01 14:04:02 -05:00
}
2021-03-18 20:07:15 -05:00
}
}
2021-04-06 04:45:41 -05: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 ) ,
ConstValue ::InferenceVar ( .. ) = > write! ( f , " _ " ) ,
ConstValue ::Placeholder ( idx ) = > {
let id = const_from_placeholder_idx ( f . db , idx ) ;
let generics = generics ( f . db . upcast ( ) , id . parent ) ;
let param_data = & generics . params . consts [ id . local_id ] ;
write! ( f , " {} " , param_data . name )
}
2021-05-11 07:06:33 -05:00
ConstValue ::Concrete ( c ) = > write! ( f , " {} " , c . interned ) ,
2021-04-06 04:45:41 -05:00
}
}
}
impl HirDisplay for BoundVar {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
write! ( f , " ?{}.{} " , self . debruijn . depth ( ) , self . index )
}
}
2021-02-28 12:13:37 -06:00
impl HirDisplay for Ty {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
2021-04-03 06:08:29 -05:00
match self . kind ( & Interner ) {
2021-03-13 07:44:51 -06: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 11:40:55 -05:00
TyKind ::Slice ( t ) = > {
2020-10-28 11:58:16 -05:00
write! ( f , " [ " ) ? ;
t . hir_fmt ( f ) ? ;
write! ( f , " ] " ) ? ;
2020-02-14 08:01:25 -06:00
}
2021-04-06 04:45:41 -05:00
TyKind ::Array ( t , c ) = > {
2020-10-28 11:58:16 -05:00
write! ( f , " [ " ) ? ;
t . hir_fmt ( f ) ? ;
2021-04-06 04:45:41 -05:00
write! ( f , " ; " ) ? ;
c . hir_fmt ( f ) ? ;
write! ( f , " ] " ) ? ;
2020-02-14 08:01:25 -06:00
}
2021-04-05 15:08:16 -05:00
TyKind ::Raw ( m , t ) | TyKind ::Ref ( m , _ , t ) = > {
2021-04-03 06:08:29 -05:00
if matches! ( self . kind ( & Interner ) , TyKind ::Raw ( .. ) ) {
2021-03-01 12:30:34 -06:00
write! (
f ,
" *{} " ,
match m {
Mutability ::Not = > " const " ,
Mutability ::Mut = > " mut " ,
}
) ? ;
2020-10-06 07:40:27 -05:00
} else {
2021-03-01 12:30:34 -06:00
write! (
f ,
" &{} " ,
match m {
Mutability ::Not = > " " ,
Mutability ::Mut = > " mut " ,
}
) ? ;
2021-01-12 13:19:13 -06:00
}
2021-03-13 13:05:47 -06:00
// FIXME: all this just to decide whether to use parentheses...
2021-08-12 09:31:00 -05:00
let contains_impl_fn = | bounds : & [ QuantifiedWhereClause ] | {
2021-08-03 07: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
}
} )
} ;
let ( preds_to_print , has_impl_fn_pred ) = match t . kind ( & Interner ) {
2021-03-21 07:22:22 -05:00
TyKind ::Dyn ( dyn_ty ) if dyn_ty . bounds . skip_binders ( ) . interned ( ) . len ( ) > 1 = > {
2021-08-03 07:01:00 -05:00
let bounds = dyn_ty . bounds . skip_binders ( ) . interned ( ) ;
2021-08-12 09:31:00 -05:00
( bounds . len ( ) , contains_impl_fn ( bounds ) )
2021-01-12 13:19:13 -06:00
}
2021-08-02 07:59:28 -05:00
TyKind ::Alias ( AliasTy ::Opaque ( OpaqueTy {
2021-03-14 10:33:27 -05:00
opaque_ty_id ,
2021-08-02 07:59:28 -05:00
substitution : parameters ,
2021-07-08 14:19:53 -05:00
} ) )
2021-08-02 07:59:28 -05: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 13:05:47 -06:00
if let ImplTraitId ::ReturnTypeImplTrait ( func , idx ) = impl_trait_id {
2021-08-03 07:01:00 -05:00
let datas =
2021-03-13 13:05:47 -06: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-04-05 11:49:26 -05:00
let bounds = data . substitute ( & Interner , parameters ) ;
2021-08-03 07: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 09:31:00 -05:00
( len , contains_impl_fn ( bounds . skip_binders ( ) ) )
2021-03-13 13:05:47 -06:00
} else {
2021-08-03 07:01:00 -05:00
( 0 , false )
2021-03-13 13:05:47 -06:00
}
2021-01-12 13:19:13 -06:00
}
2021-08-03 07:01:00 -05:00
_ = > ( 0 , false ) ,
2021-01-12 13:19:13 -06:00
} ;
2021-08-03 07:01:00 -05:00
if has_impl_fn_pred & & preds_to_print < = 2 {
return t . hir_fmt ( f ) ;
2020-10-06 07:40:27 -05:00
}
2021-08-03 07:01:00 -05:00
if preds_to_print > 1 {
2020-10-06 07:40:27 -05:00
write! ( f , " ( " ) ? ;
2021-04-06 07:16:07 -05:00
t . hir_fmt ( f ) ? ;
2020-10-06 07:40:27 -05:00
write! ( f , " ) " ) ? ;
} else {
2021-04-06 07:16:07 -05:00
t . hir_fmt ( f ) ? ;
2020-10-06 07:40:27 -05:00
}
2020-02-14 08:01:25 -06:00
}
2021-03-13 07:44:51 -06:00
TyKind ::Tuple ( _ , substs ) = > {
2021-04-01 14:04:02 -05:00
if substs . len ( & Interner ) = = 1 {
2020-10-28 11:58:16 -05:00
write! ( f , " ( " ) ? ;
2021-04-01 14:04:02 -05:00
substs . at ( & Interner , 0 ) . hir_fmt ( f ) ? ;
2020-10-28 11:58:16 -05:00
write! ( f , " ,) " ) ? ;
2020-02-14 08:01:25 -06:00
} else {
write! ( f , " ( " ) ? ;
2021-04-08 11:25:18 -05:00
f . write_joined ( & * substs . as_slice ( & Interner ) , " , " ) ? ;
2020-02-14 08:01:25 -06:00
write! ( f , " ) " ) ? ;
}
}
2021-03-13 07:44:51 -06:00
TyKind ::Function ( fn_ptr ) = > {
2021-02-28 15:12:07 -06:00
let sig = CallableSig ::from_fn_ptr ( fn_ptr ) ;
2020-12-09 11:06:56 -06:00
sig . hir_fmt ( f ) ? ;
2020-02-14 08:01:25 -06:00
}
2021-03-13 07:44:51 -06:00
TyKind ::FnDef ( def , parameters ) = > {
2021-03-13 10:55:50 -06:00
let def = from_chalk ( f . db , * def ) ;
2021-04-05 11:49:26 -05:00
let sig = f . db . callable_item_signature ( def ) . substitute ( & Interner , parameters ) ;
2020-03-28 16:25:29 -05:00
match def {
2020-07-16 06:15:00 -05: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 16:25:29 -05:00
write! ( f , " {} " , f . db . enum_data ( e . parent ) . variants [ e . local_id ] . name ) ?
2020-02-14 08:01:25 -06:00
}
} ;
2021-04-01 14:04:02 -05:00
if parameters . len ( & Interner ) > 0 {
2020-03-13 10:05:46 -05:00
let generics = generics ( f . db . upcast ( ) , def . into ( ) ) ;
2020-02-14 08:01:25 -06:00
let ( parent_params , self_param , type_params , _impl_trait_params ) =
generics . provenance_split ( ) ;
let total_len = parent_params + self_param + type_params ;
// 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-04-08 11:25:18 -05:00
f . write_joined ( & parameters . as_slice ( & Interner ) [ .. total_len ] , " , " ) ? ;
2020-02-14 08:01:25 -06:00
write! ( f , " > " ) ? ;
}
}
write! ( f , " ( " ) ? ;
f . write_joined ( sig . params ( ) , " , " ) ? ;
2020-04-05 10:25:47 -05:00
write! ( f , " ) " ) ? ;
let ret = sig . ret ( ) ;
2021-04-03 13:22:59 -05:00
if ! ret . is_unit ( ) {
2021-04-06 07:16:07 -05:00
write! ( f , " -> " ) ? ;
ret . hir_fmt ( f ) ? ;
2020-04-05 10:25:47 -05:00
}
2020-02-14 08:01:25 -06:00
}
2021-03-13 07:44:51 -06:00
TyKind ::Adt ( AdtId ( def_id ) , parameters ) = > {
2020-04-25 09:57:59 -05:00
match f . display_target {
2020-10-28 09:09:47 -05:00
DisplayTarget ::Diagnostics | DisplayTarget ::Test = > {
2021-02-28 12:46:59 -06:00
let name = match * def_id {
2021-03-01 14:57:39 -06: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 09:57:59 -05:00
} ;
write! ( f , " {} " , name ) ? ;
}
2020-10-28 09:09:47 -05:00
DisplayTarget ::SourceCode { module_id } = > {
2020-10-28 05:20:05 -05:00
if let Some ( path ) = find_path ::find_path (
f . db . upcast ( ) ,
2021-02-28 12:46:59 -06:00
ItemInNs ::Types ( ( * def_id ) . into ( ) ) ,
2020-10-28 05:20:05 -05:00
module_id ,
) {
write! ( f , " {} " , path ) ? ;
} else {
return Err ( HirDisplayError ::DisplaySourceCodeError (
DisplaySourceCodeError ::PathNotFound ,
) ) ;
}
}
2020-04-25 09:57:59 -05:00
}
2021-04-01 14:04:02 -05:00
if parameters . len ( & Interner ) > 0 {
2021-03-13 07:44:51 -06:00
let parameters_to_write = if f . display_target . is_source_code ( )
| | f . omit_verbose_types ( )
{
match self
2021-03-13 10:55:50 -06:00
. as_generic_def ( f . db )
2021-03-13 07:44:51 -06:00
. map ( | generic_def_id | f . db . generic_defaults ( generic_def_id ) )
. filter ( | defaults | ! defaults . is_empty ( ) )
{
2021-04-08 11:25:18 -05:00
None = > parameters . as_slice ( & Interner ) ,
2021-03-13 07:44:51 -06:00
Some ( default_parameters ) = > {
let mut default_from = 0 ;
2021-04-01 14:04:02 -05:00
for ( i , parameter ) in parameters . iter ( & Interner ) . enumerate ( ) {
match (
2021-04-03 06:08:29 -05:00
parameter . assert_ty_ref ( & Interner ) . kind ( & Interner ) ,
2021-04-01 14:04:02 -05:00
default_parameters . get ( i ) ,
) {
2021-04-05 07:37:11 -05:00
( & TyKind ::Error , _ ) | ( _ , None ) = > {
2021-03-13 07:44:51 -06:00
default_from = i + 1 ;
}
( _ , Some ( default_parameter ) ) = > {
2021-04-05 14:56:40 -05:00
let actual_default =
default_parameter . clone ( ) . substitute (
& Interner ,
& subst_prefix ( parameters , i ) ,
) ;
2021-04-01 14:04:02 -05:00
if parameter . assert_ty_ref ( & Interner ) ! = & actual_default
{
2020-06-26 09:36:59 -05:00
default_from = i + 1 ;
2020-05-10 11:09:22 -05:00
}
2020-02-14 08:01:25 -06:00
}
}
}
2021-04-08 11:25:18 -05:00
& parameters . as_slice ( & Interner ) [ 0 .. default_from ]
2020-02-14 08:01:25 -06:00
}
2021-03-13 07:44:51 -06:00
}
} else {
2021-04-08 11:25:18 -05:00
parameters . as_slice ( & Interner )
2021-03-13 07:44:51 -06:00
} ;
2020-05-10 11:09:22 -05:00
if ! parameters_to_write . is_empty ( ) {
write! ( f , " < " ) ? ;
f . write_joined ( parameters_to_write , " , " ) ? ;
write! ( f , " > " ) ? ;
}
2020-02-14 08:01:25 -06:00
}
}
2021-03-13 10:36:07 -06:00
TyKind ::AssociatedType ( assoc_type_id , parameters ) = > {
let type_alias = from_assoc_type_id ( * assoc_type_id ) ;
2020-03-13 10:05:46 -05:00
let trait_ = match type_alias . lookup ( f . db . upcast ( ) ) . container {
2020-02-14 08:01:25 -06:00
AssocContainerId ::TraitId ( it ) = > it ,
_ = > panic! ( " not an associated type " ) ,
} ;
2020-03-31 11:00:23 -05:00
let trait_ = f . db . trait_data ( trait_ ) ;
2021-03-13 10:36:07 -06:00
let type_alias_data = f . db . type_alias_data ( type_alias ) ;
2020-10-27 14:23:09 -05:00
2020-10-28 05:20:05 -05: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 09:09:47 -05:00
if f . display_target . is_test ( ) {
write! ( f , " {}::{} " , trait_ . name , type_alias_data . name ) ? ;
2021-04-01 14:04:02 -05:00
if parameters . len ( & Interner ) > 0 {
2020-10-27 14:23:09 -05:00
write! ( f , " < " ) ? ;
2021-04-08 11:25:18 -05:00
f . write_joined ( & * parameters . as_slice ( & Interner ) , " , " ) ? ;
2020-10-27 14:23:09 -05:00
write! ( f , " > " ) ? ;
}
} else {
2021-03-13 10:36:07 -06:00
let projection_ty = ProjectionTy {
2021-03-14 10:26:12 -05:00
associated_ty_id : to_assoc_type_id ( type_alias ) ,
substitution : parameters . clone ( ) ,
2021-03-13 10:36:07 -06:00
} ;
2020-10-28 09:09:47 -05:00
projection_ty . hir_fmt ( f ) ? ;
2020-02-14 08:01:25 -06:00
}
}
2021-04-05 07:38:28 -05:00
TyKind ::Foreign ( type_alias ) = > {
2021-03-13 10:23:19 -06:00
let type_alias = f . db . type_alias_data ( from_foreign_def_id ( * type_alias ) ) ;
2020-09-12 21:24:19 -05:00
write! ( f , " {} " , type_alias . name ) ? ;
}
2021-03-13 07:44:51 -06:00
TyKind ::OpaqueType ( opaque_ty_id , parameters ) = > {
2021-03-13 13:05:47 -06: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 16:00:44 -06: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-04-05 11:49:26 -05:00
let bounds = data . substitute ( & Interner , & parameters ) ;
2021-06-15 13:28:37 -05:00
let krate = func . lookup ( f . db . upcast ( ) ) . module ( f . db . upcast ( ) ) . krate ( ) ;
2021-06-15 03:53:20 -05:00
write_bounds_like_dyn_trait_with_prefix (
" impl " ,
bounds . skip_binders ( ) ,
2021-06-15 13:28:37 -05:00
SizedByDefault ::Sized { anchor : krate } ,
2021-06-15 03:53:20 -05:00
f ,
) ? ;
2020-09-10 07:01:23 -05: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 16:00:44 -06:00
}
2021-03-13 13:05:47 -06:00
ImplTraitId ::AsyncBlockTypeImplTrait ( .. ) = > {
2020-09-10 07:01:23 -05:00
write! ( f , " impl Future<Output = " ) ? ;
2021-04-01 14:04:02 -05:00
parameters . at ( & Interner , 0 ) . hir_fmt ( f ) ? ;
2020-09-10 07:01:23 -05:00
write! ( f , " > " ) ? ;
}
}
2020-03-04 16:00:44 -06:00
}
2021-03-13 07:44:51 -06:00
TyKind ::Closure ( .. , substs ) = > {
2021-03-22 02:17:01 -05:00
if f . display_target . is_source_code ( ) {
return Err ( HirDisplayError ::DisplaySourceCodeError (
DisplaySourceCodeError ::Closure ,
) ) ;
}
2021-04-01 14:04:02 -05:00
let sig = substs . at ( & Interner , 0 ) . assert_ty_ref ( & Interner ) . callable_sig ( f . db ) ;
2020-03-27 12:56:18 -05:00
if let Some ( sig ) = sig {
if sig . params ( ) . is_empty ( ) {
write! ( f , " || " ) ? ;
} else if f . omit_verbose_types ( ) {
write! ( f , " |{}| " , TYPE_HINT_TRUNCATION ) ? ;
} else {
write! ( f , " | " ) ? ;
f . write_joined ( sig . params ( ) , " , " ) ? ;
write! ( f , " | " ) ? ;
} ;
2020-07-20 15:50:41 -05:00
2021-04-06 07:16:07 -05:00
write! ( f , " -> " ) ? ;
sig . ret ( ) . hir_fmt ( f ) ? ;
2020-02-14 08:01:25 -06:00
} else {
2020-03-27 12:56:18 -05:00
write! ( f , " {{closure}} " ) ? ;
}
2020-02-14 08:01:25 -06:00
}
2021-03-13 12:47:34 -06:00
TyKind ::Placeholder ( idx ) = > {
let id = from_placeholder_idx ( f . db , * idx ) ;
2020-03-13 10:05:46 -05:00
let generics = generics ( f . db . upcast ( ) , id . parent ) ;
2020-02-14 08:01:25 -06:00
let param_data = & generics . params . types [ id . local_id ] ;
match param_data . provenance {
TypeParamProvenance ::TypeParamList | TypeParamProvenance ::TraitSelf = > {
write! ( f , " {} " , param_data . name . clone ( ) . unwrap_or_else ( Name ::missing ) ) ?
}
TypeParamProvenance ::ArgumentImplTrait = > {
2021-04-04 06:07:06 -05:00
let substs = generics . type_params_subst ( f . db ) ;
2021-04-05 09:38:37 -05:00
let bounds =
f . db . generic_predicates ( id . parent )
. into_iter ( )
2021-04-05 11:49:26 -05:00
. map ( | pred | pred . clone ( ) . substitute ( & Interner , & substs ) )
2021-04-05 09:38:37 -05:00
. filter ( | wc | match & wc . skip_binders ( ) {
WhereClause ::Implemented ( tr ) = > {
2021-04-07 13:41:52 -05:00
& tr . self_type_parameter ( & Interner ) = = self
2021-04-05 09:38:37 -05:00
}
WhereClause ::AliasEq ( AliasEq {
alias : AliasTy ::Projection ( proj ) ,
ty : _ ,
2021-04-07 13:41:52 -05:00
} ) = > & proj . self_type_parameter ( & Interner ) = = self ,
2021-04-05 09:38:37 -05:00
_ = > false ,
} )
. collect ::< Vec < _ > > ( ) ;
2021-06-15 13:28:37 -05:00
let krate = id . parent . module ( f . db . upcast ( ) ) . krate ( ) ;
2021-06-15 03:53:20 -05:00
write_bounds_like_dyn_trait_with_prefix (
" impl " ,
& bounds ,
2021-06-15 13:28:37 -05:00
SizedByDefault ::Sized { anchor : krate } ,
2021-06-15 03:53:20 -05:00
f ,
) ? ;
2020-02-14 08:01:25 -06:00
}
}
}
2021-04-06 04:45:41 -05:00
TyKind ::BoundVar ( idx ) = > idx . hir_fmt ( f ) ? ,
2021-03-21 07:22:22 -05:00
TyKind ::Dyn ( dyn_ty ) = > {
write_bounds_like_dyn_trait_with_prefix (
" dyn " ,
dyn_ty . bounds . skip_binders ( ) . interned ( ) ,
2021-06-15 03:53:20 -05:00
SizedByDefault ::NotSized ,
2021-03-21 07:22:22 -05:00
f ,
) ? ;
2020-02-14 08:01:25 -06:00
}
2021-03-13 07:44:51 -06:00
TyKind ::Alias ( AliasTy ::Projection ( p_ty ) ) = > p_ty . hir_fmt ( f ) ? ,
TyKind ::Alias ( AliasTy ::Opaque ( opaque_ty ) ) = > {
2021-03-13 13:05:47 -06: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 16:00:44 -06: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-04-05 11:49:26 -05:00
let bounds = data . substitute ( & Interner , & opaque_ty . substitution ) ;
2021-06-15 13:28:37 -05:00
let krate = func . lookup ( f . db . upcast ( ) ) . module ( f . db . upcast ( ) ) . krate ( ) ;
2021-06-15 03:53:20 -05:00
write_bounds_like_dyn_trait_with_prefix (
" impl " ,
bounds . skip_binders ( ) ,
2021-06-15 13:28:37 -05:00
SizedByDefault ::Sized { anchor : krate } ,
2021-06-15 03:53:20 -05:00
f ,
) ? ;
2020-09-10 07:01:23 -05:00
}
2021-03-13 13:05:47 -06:00
ImplTraitId ::AsyncBlockTypeImplTrait ( .. ) = > {
2020-09-10 07:01:23 -05:00
write! ( f , " {{async block}} " ) ? ;
2020-03-04 16:00:44 -06:00
}
} ;
}
2021-04-05 07:37:11 -05:00
TyKind ::Error = > {
2020-12-12 11:18:19 -06:00
if f . display_target . is_source_code ( ) {
return Err ( HirDisplayError ::DisplaySourceCodeError (
DisplaySourceCodeError ::UnknownType ,
) ) ;
}
write! ( f , " {{unknown}} " ) ? ;
}
2021-03-13 07:44:51 -06:00
TyKind ::InferenceVar ( .. ) = > write! ( f , " _ " ) ? ,
2021-04-08 06:51:04 -05:00
TyKind ::Generator ( .. ) = > write! ( f , " {{generator}} " ) ? ,
TyKind ::GeneratorWitness ( .. ) = > write! ( f , " {{generator witness}} " ) ? ,
2020-02-14 08:01:25 -06:00
}
Ok ( ( ) )
}
}
2021-02-28 15:12:07 -06:00
impl HirDisplay for CallableSig {
2020-12-09 11:06:56 -06: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 13:22:59 -05:00
if ! ret . is_unit ( ) {
2021-04-06 07:16:07 -05:00
write! ( f , " -> " ) ? ;
ret . hir_fmt ( f ) ? ;
2020-12-09 11:06:56 -06:00
}
Ok ( ( ) )
}
}
2021-01-12 13:19:13 -06:00
fn fn_traits ( db : & dyn DefDatabase , trait_ : TraitId ) -> impl Iterator < Item = TraitId > {
2021-03-09 12:09:02 -06:00
let krate = trait_ . lookup ( db ) . container . krate ( ) ;
2021-06-29 10:35:37 -05:00
utils ::fn_traits ( db , krate )
2021-01-12 13:19:13 -06:00
}
2021-06-15 03:53:20 -05:00
#[ derive(Clone, Copy, PartialEq, Eq) ]
pub enum SizedByDefault {
NotSized ,
2021-06-15 13: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 } = > {
let sized_trait =
db . lang_item ( anchor , " sized " . into ( ) ) . and_then ( | lang_item | lang_item . as_trait ( ) ) ;
Some ( trait_ ) = = sized_trait
}
}
}
2021-06-15 03:53:20 -05:00
}
2021-02-20 13:43:04 -06:00
pub fn write_bounds_like_dyn_trait_with_prefix (
prefix : & str ,
2021-03-21 07:22:22 -05:00
predicates : & [ QuantifiedWhereClause ] ,
2021-06-15 03:53:20 -05:00
default_sized : SizedByDefault ,
2021-02-20 13:43:04 -06:00
f : & mut HirFormatter ,
) -> Result < ( ) , HirDisplayError > {
write! ( f , " {} " , prefix ) ? ;
2021-06-15 13:28:37 -05:00
if ! predicates . is_empty ( )
| | predicates . is_empty ( ) & & matches! ( default_sized , SizedByDefault ::Sized { .. } )
{
2021-02-20 13:43:04 -06:00
write! ( f , " " ) ? ;
2021-06-15 03:53:20 -05:00
write_bounds_like_dyn_trait ( predicates , default_sized , f )
2021-02-20 13:43:04 -06:00
} else {
Ok ( ( ) )
}
}
fn write_bounds_like_dyn_trait (
2021-03-21 07:22:22 -05:00
predicates : & [ QuantifiedWhereClause ] ,
2021-06-15 03:53:20 -05:00
default_sized : SizedByDefault ,
2020-03-13 10:05:46 -05:00
f : & mut HirFormatter ,
2020-04-25 09:57:59 -05:00
) -> Result < ( ) , HirDisplayError > {
2020-02-14 08:01:25 -06: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 13:19:13 -06:00
let mut is_fn_trait = false ;
2021-06-15 13:28:37 -05:00
let mut is_sized = false ;
2020-02-14 08:01:25 -06:00
for p in predicates . iter ( ) {
2021-03-21 07:22:22 -05:00
match p . skip_binders ( ) {
2021-03-20 04:46:36 -05:00
WhereClause ::Implemented ( trait_ref ) = > {
2021-03-18 15:53:19 -05:00
let trait_ = trait_ref . hir_trait_id ( ) ;
2021-06-15 13: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-15 17:23:04 -05:00
// Don't print +Sized, but rather +?Sized if absent.
continue ;
2021-06-15 03:53:20 -05:00
}
2021-06-15 17:23:04 -05:00
}
2021-01-12 13:19:13 -06: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 08:01:25 -06:00
write! ( f , " > " ) ? ;
2020-08-13 15:13:34 -05:00
angle_open = false ;
2020-02-14 08:01:25 -06:00
}
if ! first {
write! ( f , " + " ) ? ;
}
2021-04-08 06:51:04 -05:00
// We assume that the self type is ^0.0 (i.e. the
2020-02-14 08:01:25 -06:00
// existential) here, which is the only thing that's
// possible in actual Rust, and hence don't print it
2021-01-12 13:19:13 -06:00
write! ( f , " {} " , f . db . trait_data ( trait_ ) . name ) ? ;
2021-04-08 11:25:18 -05:00
if let [ _ , params @ .. ] = & * trait_ref . substitution . as_slice ( & Interner ) {
2021-01-12 13:19:13 -06:00
if is_fn_trait {
2021-04-01 14:04:02 -05:00
if let Some ( args ) =
params . first ( ) . and_then ( | it | it . assert_ty_ref ( & Interner ) . as_tuple ( ) )
{
2021-01-12 13:19:13 -06:00
write! ( f , " ( " ) ? ;
2021-04-08 11:25:18 -05:00
f . write_joined ( args . as_slice ( & Interner ) , " , " ) ? ;
2021-01-12 13:19:13 -06: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 08:01:25 -06:00
}
}
2021-03-20 04:46:36 -05:00
WhereClause ::AliasEq ( alias_eq ) if is_fn_trait = > {
2021-01-12 13:19:13 -06:00
is_fn_trait = false ;
2021-05-25 08:23:52 -05:00
if ! alias_eq . ty . is_unit ( ) {
write! ( f , " -> " ) ? ;
alias_eq . ty . hir_fmt ( f ) ? ;
}
2021-01-12 13:19:13 -06:00
}
2021-03-20 04:46:36 -05:00
WhereClause ::AliasEq ( AliasEq { ty , alias } ) = > {
2020-02-14 08:01:25 -06: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-18 20:07:15 -05: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 08:01:25 -06:00
}
2021-04-08 06:51:04 -05:00
// FIXME implement these
WhereClause ::LifetimeOutlives ( _ ) = > { }
WhereClause ::TypeOutlives ( _ ) = > { }
2020-02-14 08:01:25 -06:00
}
first = false ;
}
if angle_open {
write! ( f , " > " ) ? ;
}
2021-06-15 13:28:37 -05:00
if matches! ( default_sized , SizedByDefault ::Sized { .. } ) {
if ! is_sized {
2021-06-15 03:53:20 -05:00
write! ( f , " {}?Sized " , if first { " " } else { " + " } ) ? ;
} else if first {
write! ( f , " Sized " ) ? ;
}
}
2020-02-14 08:01:25 -06:00
Ok ( ( ) )
}
2021-04-07 13:26:27 -05: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 08:01:25 -06:00
2021-04-07 13:26:27 -05:00
tr . self_type_parameter ( & Interner ) . hir_fmt ( f ) ? ;
if use_as {
write! ( f , " as " ) ? ;
} else {
write! ( f , " : " ) ? ;
2020-02-14 08:01:25 -06:00
}
2021-04-07 13:26:27 -05:00
write! ( f , " {} " , f . db . trait_data ( tr . hir_trait_id ( ) ) . name ) ? ;
if tr . substitution . len ( & Interner ) > 1 {
write! ( f , " < " ) ? ;
2021-04-08 11:25:18 -05:00
f . write_joined ( & tr . substitution . as_slice ( & Interner ) [ 1 .. ] , " , " ) ? ;
2021-04-07 13:26:27 -05:00
write! ( f , " > " ) ? ;
}
Ok ( ( ) )
2020-02-14 08:01:25 -06:00
}
impl HirDisplay for TraitRef {
2020-04-25 09:57:59 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2021-04-07 13:26:27 -05:00
fmt_trait_ref ( self , f , false )
2020-02-14 08:01:25 -06:00
}
}
2021-03-20 04:46:36 -05:00
impl HirDisplay for WhereClause {
2020-04-25 09:57:59 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-02-14 08:01:25 -06:00
if f . should_truncate ( ) {
return write! ( f , " {} " , TYPE_HINT_TRUNCATION ) ;
}
match self {
2021-03-20 04:46:36 -05:00
WhereClause ::Implemented ( trait_ref ) = > trait_ref . hir_fmt ( f ) ? ,
WhereClause ::AliasEq ( AliasEq { alias : AliasTy ::Projection ( projection_ty ) , ty } ) = > {
2020-02-14 08:01:25 -06:00
write! ( f , " < " ) ? ;
2021-04-07 13:26:27 -05:00
fmt_trait_ref ( & projection_ty . trait_ref ( f . db ) , f , true ) ? ;
2020-02-14 08:01:25 -06:00
write! (
f ,
2020-10-28 11:58:16 -05:00
" >::{} = " ,
2021-03-18 20:07:15 -05:00
f . db . type_alias_data ( from_assoc_type_id ( projection_ty . associated_ty_id ) ) . name ,
2020-02-14 08:01:25 -06:00
) ? ;
2021-03-18 20:07:15 -05:00
ty . hir_fmt ( f ) ? ;
2020-02-14 08:01:25 -06:00
}
2021-03-20 04:51:00 -05:00
WhereClause ::AliasEq ( _ ) = > write! ( f , " {{error}} " ) ? ,
2021-04-08 06:51:04 -05:00
// FIXME implement these
WhereClause ::TypeOutlives ( .. ) = > { }
WhereClause ::LifetimeOutlives ( .. ) = > { }
2020-02-14 08:01:25 -06:00
}
Ok ( ( ) )
}
}
2021-04-05 13:46:15 -05: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 06:49:32 -06:00
impl HirDisplay for Lifetime {
2021-04-05 13:46:15 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
self . interned ( ) . hir_fmt ( f )
}
}
impl HirDisplay for LifetimeData {
2020-12-11 06:49:32 -06:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
2021-04-06 04:45:41 -05:00
LifetimeData ::BoundVar ( idx ) = > idx . hir_fmt ( f ) ,
2021-04-05 13:46:15 -05:00
LifetimeData ::InferenceVar ( _ ) = > write! ( f , " _ " ) ,
LifetimeData ::Placeholder ( idx ) = > {
let id = lt_from_placeholder_idx ( f . db , * idx ) ;
2020-12-11 06:49:32 -06:00
let generics = generics ( f . db . upcast ( ) , id . parent ) ;
let param_data = & generics . params . lifetimes [ id . local_id ] ;
2021-04-05 13:46:15 -05:00
write! ( f , " {} " , param_data . name )
2020-12-11 06:49:32 -06:00
}
2021-04-05 13:46:15 -05:00
LifetimeData ::Static = > write! ( f , " 'static " ) ,
LifetimeData ::Empty ( _ ) = > Ok ( ( ) ) ,
LifetimeData ::Erased = > Ok ( ( ) ) ,
LifetimeData ::Phantom ( _ , _ ) = > Ok ( ( ) ) ,
2020-12-11 06:49:32 -06:00
}
}
}
2021-03-20 05:23:59 -05:00
impl HirDisplay for DomainGoal {
2020-04-25 09:57:59 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
2020-10-06 07:40:27 -05:00
match self {
2021-03-20 05:23:59 -05:00
DomainGoal ::Holds ( wc ) = > {
write! ( f , " Holds( " ) ? ;
wc . hir_fmt ( f ) ? ;
2021-04-08 06:51:04 -05:00
write! ( f , " ) " ) ? ;
2020-10-28 11:58:16 -05:00
}
2021-04-08 06:51:04 -05:00
_ = > write! ( f , " ? " ) ? ,
2020-10-06 07:40:27 -05:00
}
2021-04-08 06:51:04 -05:00
Ok ( ( ) )
2020-02-14 08:01:25 -06:00
}
}
2021-03-14 07:03:39 -05: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 06:39:48 -05:00
TypeRef ::Array ( inner , len ) = > {
2021-03-14 07:03:39 -05:00
write! ( f , " [ " ) ? ;
inner . hir_fmt ( f ) ? ;
2021-05-12 06:39:48 -05:00
write! ( f , " ; {}] " , len ) ? ;
2021-03-14 07:03:39 -05:00
}
TypeRef ::Slice ( inner ) = > {
write! ( f , " [ " ) ? ;
inner . hir_fmt ( f ) ? ;
write! ( f , " ] " ) ? ;
}
TypeRef ::Fn ( tys , is_varargs ) = > {
// FIXME: Function pointer qualifiers.
write! ( f , " fn( " ) ? ;
f . write_joined ( & tys [ .. tys . len ( ) - 1 ] , " , " ) ? ;
if * is_varargs {
write! ( f , " {}... " , if tys . len ( ) = = 1 { " " } else { " , " } ) ? ;
}
write! ( f , " ) " ) ? ;
let ret_ty = tys . last ( ) . unwrap ( ) ;
match ret_ty {
TypeRef ::Tuple ( tup ) if tup . is_empty ( ) = > { }
_ = > {
write! ( f , " -> " ) ? ;
ret_ty . hir_fmt ( f ) ? ;
}
}
}
TypeRef ::ImplTrait ( bounds ) = > {
write! ( f , " impl " ) ? ;
f . write_joined ( bounds , " + " ) ? ;
}
TypeRef ::DynTrait ( bounds ) = > {
write! ( f , " dyn " ) ? ;
f . write_joined ( bounds , " + " ) ? ;
}
2021-04-12 09:24:48 -05:00
TypeRef ::Macro ( macro_call ) = > {
let macro_call = macro_call . to_node ( f . db . upcast ( ) ) ;
2021-05-06 12:59:54 -05:00
let ctx = body ::LowerCtx ::with_hygiene ( f . db . upcast ( ) , & Hygiene ::new_unhygienic ( ) ) ;
2021-04-12 09:24:48 -05:00
match macro_call . path ( ) {
2021-05-06 16:23:50 -05:00
Some ( path ) = > match Path ::from_src ( path , & ctx ) {
2021-04-12 09:24:48 -05: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 07:03:39 -05:00
}
Ok ( ( ) )
}
}
impl HirDisplay for TypeBound {
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
2021-06-06 13:41:15 -05:00
TypeBound ::Path ( path , modifier ) = > {
match modifier {
TraitBoundModifier ::None = > ( ) ,
TraitBoundModifier ::Maybe = > write! ( f , " ? " ) ? ,
}
path . hir_fmt ( f )
}
2021-03-14 07:03:39 -05:00
TypeBound ::Lifetime ( lifetime ) = > write! ( f , " {} " , lifetime . name ) ,
2021-06-29 18:34:51 -05:00
TypeBound ::ForLifetime ( lifetimes , path ) = > {
write! ( f , " for<{}> " , lifetimes . iter ( ) . format ( " , " ) ) ? ;
path . hir_fmt ( f )
}
2021-03-14 07:03:39 -05: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 ) = > { }
( _ , PathKind ::Abs ) = > write! ( f , " :: " ) ? ,
( _ , PathKind ::Crate ) = > write! ( f , " crate " ) ? ,
( _ , PathKind ::Super ( 0 ) ) = > write! ( f , " self " ) ? ,
( _ , PathKind ::Super ( n ) ) = > {
write! ( f , " super " ) ? ;
for _ in 0 .. * n {
write! ( f , " ::super " ) ? ;
}
}
( _ , PathKind ::DollarCrate ( _ ) ) = > write! ( f , " {{extern_crate}} " ) ? ,
}
for ( seg_idx , segment ) in self . segments ( ) . iter ( ) . enumerate ( ) {
if seg_idx ! = 0 {
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?
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 14:04:02 -05:00
impl HirDisplay for hir_def ::path ::GenericArg {
2021-03-14 07:03:39 -05:00
fn hir_fmt ( & self , f : & mut HirFormatter ) -> Result < ( ) , HirDisplayError > {
match self {
2021-04-01 14:04:02 -05:00
hir_def ::path ::GenericArg ::Type ( ty ) = > ty . hir_fmt ( f ) ,
hir_def ::path ::GenericArg ::Lifetime ( lifetime ) = > write! ( f , " {} " , lifetime . name ) ,
2021-03-14 07:03:39 -05:00
}
}
}