@@ -2939,25 +2939,26 @@ fn check_expr_with_expectation_and_lvalue_pref<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
2939
2939
base : & ' tcx hir:: Expr ,
2940
2940
field : & Spanned < ast:: Name > ) {
2941
2941
check_expr_with_lvalue_pref ( fcx, base, lvalue_pref) ;
2942
- let expr_t = structurally_resolved_type ( fcx, expr. span ,
2943
- fcx. expr_ty ( base) ) ;
2944
- // FIXME(eddyb) #12808 Integrate privacy into this auto-deref loop.
2942
+ let expr_t = structurally_resolved_type ( fcx, expr. span , fcx. expr_ty ( base) ) ;
2943
+ let mut private_candidate = None ;
2945
2944
let ( _, autoderefs, field_ty) = autoderef ( fcx,
2946
2945
expr. span ,
2947
2946
expr_t,
2948
2947
|| Some ( base) ,
2949
2948
UnresolvedTypeAction :: Error ,
2950
2949
lvalue_pref,
2951
2950
|base_t, _| {
2952
- match base_t. sty {
2953
- ty:: TyStruct ( base_def, substs) => {
2954
- debug ! ( "struct named {:?}" , base_t) ;
2955
- base_def. struct_variant ( )
2956
- . find_field_named ( field. node )
2957
- . map ( |f| fcx. field_ty ( expr. span , f, substs) )
2951
+ if let ty:: TyStruct ( base_def, substs) = base_t. sty {
2952
+ debug ! ( "struct named {:?}" , base_t) ;
2953
+ if let Some ( field) = base_def. struct_variant ( ) . find_field_named ( field. node ) {
2954
+ let field_ty = fcx. field_ty ( expr. span , field, substs) ;
2955
+ if field. vis == hir:: Public || fcx. private_item_is_visible ( base_def. did ) {
2956
+ return Some ( field_ty) ;
2957
+ }
2958
+ private_candidate = Some ( ( base_def. did , field_ty) ) ;
2958
2959
}
2959
- _ => None
2960
2960
}
2961
+ None
2961
2962
} ) ;
2962
2963
match field_ty {
2963
2964
Some ( field_ty) => {
@@ -2968,12 +2969,14 @@ fn check_expr_with_expectation_and_lvalue_pref<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
2968
2969
None => { }
2969
2970
}
2970
2971
2971
- if field. node == special_idents:: invalid. name {
2972
+ if let Some ( ( did, field_ty) ) = private_candidate {
2973
+ let struct_path = fcx. tcx ( ) . item_path_str ( did) ;
2974
+ let msg = format ! ( "field `{}` of struct `{}` is private" , field. node, struct_path) ;
2975
+ fcx. tcx ( ) . sess . span_err ( expr. span , & msg) ;
2976
+ fcx. write_ty ( expr. id , field_ty) ;
2977
+ } else if field. node == special_idents:: invalid. name {
2972
2978
fcx. write_error ( expr. id ) ;
2973
- return ;
2974
- }
2975
-
2976
- if method:: exists ( fcx, field. span , field. node , expr_t, expr. id ) {
2979
+ } else if method:: exists ( fcx, field. span , field. node , expr_t, expr. id ) {
2977
2980
fcx. type_error_struct ( field. span ,
2978
2981
|actual| {
2979
2982
format ! ( "attempted to take value of method `{}` on type \
@@ -2984,6 +2987,7 @@ fn check_expr_with_expectation_and_lvalue_pref<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
2984
2987
"maybe a `()` to call it is missing? \
2985
2988
If not, try an anonymous function")
2986
2989
. emit ( ) ;
2990
+ fcx. write_error ( expr. id ) ;
2987
2991
} else {
2988
2992
let mut err = fcx. type_error_struct (
2989
2993
expr. span ,
@@ -2999,9 +3003,8 @@ fn check_expr_with_expectation_and_lvalue_pref<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
2999
3003
suggest_field_names ( & mut err, def. struct_variant ( ) , field, vec ! [ ] ) ;
3000
3004
}
3001
3005
err. emit ( ) ;
3006
+ fcx. write_error ( expr. id ) ;
3002
3007
}
3003
-
3004
- fcx. write_error ( expr. id ) ;
3005
3008
}
3006
3009
3007
3010
// displays hints about the closest matches in field names
@@ -3036,36 +3039,37 @@ fn check_expr_with_expectation_and_lvalue_pref<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
3036
3039
base : & ' tcx hir:: Expr ,
3037
3040
idx : codemap:: Spanned < usize > ) {
3038
3041
check_expr_with_lvalue_pref ( fcx, base, lvalue_pref) ;
3039
- let expr_t = structurally_resolved_type ( fcx, expr. span ,
3040
- fcx . expr_ty ( base ) ) ;
3042
+ let expr_t = structurally_resolved_type ( fcx, expr. span , fcx . expr_ty ( base ) ) ;
3043
+ let mut private_candidate = None ;
3041
3044
let mut tuple_like = false ;
3042
- // FIXME(eddyb) #12808 Integrate privacy into this auto-deref loop.
3043
3045
let ( _, autoderefs, field_ty) = autoderef ( fcx,
3044
3046
expr. span ,
3045
3047
expr_t,
3046
3048
|| Some ( base) ,
3047
3049
UnresolvedTypeAction :: Error ,
3048
3050
lvalue_pref,
3049
3051
|base_t, _| {
3050
- match base_t. sty {
3051
- ty:: TyStruct ( base_def, substs) => {
3052
- tuple_like = base_def. struct_variant ( ) . is_tuple_struct ( ) ;
3053
- if tuple_like {
3054
- debug ! ( "tuple struct named {:?}" , base_t) ;
3055
- base_def. struct_variant ( )
3056
- . fields
3057
- . get ( idx. node )
3058
- . map ( |f| fcx. field_ty ( expr. span , f, substs) )
3059
- } else {
3060
- None
3061
- }
3062
- }
3052
+ let ( base_def, substs) = match base_t. sty {
3053
+ ty:: TyStruct ( base_def, substs) => ( base_def, substs) ,
3063
3054
ty:: TyTuple ( ref v) => {
3064
3055
tuple_like = true ;
3065
- if idx. node < v. len ( ) { Some ( v[ idx. node ] ) } else { None }
3056
+ return if idx. node < v. len ( ) { Some ( v[ idx. node ] ) } else { None }
3066
3057
}
3067
- _ => None
3058
+ _ => return None ,
3059
+ } ;
3060
+
3061
+ tuple_like = base_def. struct_variant ( ) . is_tuple_struct ( ) ;
3062
+ if !tuple_like { return None }
3063
+
3064
+ debug ! ( "tuple struct named {:?}" , base_t) ;
3065
+ if let Some ( field) = base_def. struct_variant ( ) . fields . get ( idx. node ) {
3066
+ let field_ty = fcx. field_ty ( expr. span , field, substs) ;
3067
+ if field. vis == hir:: Public || fcx. private_item_is_visible ( base_def. did ) {
3068
+ return Some ( field_ty) ;
3069
+ }
3070
+ private_candidate = Some ( ( base_def. did , field_ty) ) ;
3068
3071
}
3072
+ None
3069
3073
} ) ;
3070
3074
match field_ty {
3071
3075
Some ( field_ty) => {
@@ -3075,6 +3079,15 @@ fn check_expr_with_expectation_and_lvalue_pref<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
3075
3079
}
3076
3080
None => { }
3077
3081
}
3082
+
3083
+ if let Some ( ( did, field_ty) ) = private_candidate {
3084
+ let struct_path = fcx. tcx ( ) . item_path_str ( did) ;
3085
+ let msg = format ! ( "field `{}` of struct `{}` is private" , idx. node, struct_path) ;
3086
+ fcx. tcx ( ) . sess . span_err ( expr. span , & msg) ;
3087
+ fcx. write_ty ( expr. id , field_ty) ;
3088
+ return ;
3089
+ }
3090
+
3078
3091
fcx. type_error_message (
3079
3092
expr. span ,
3080
3093
|actual| {
@@ -3745,23 +3758,30 @@ pub fn resolve_ty_and_def_ufcs<'a, 'b, 'tcx>(fcx: &FnCtxt<'b, 'tcx>,
3745
3758
& ty_segments[ base_ty_end..] ) ;
3746
3759
let item_segment = path. segments . last ( ) . unwrap ( ) ;
3747
3760
let item_name = item_segment. identifier . name ;
3748
- match method:: resolve_ufcs ( fcx, span, item_name, ty, node_id) {
3749
- Ok ( def) => {
3750
- // Write back the new resolution.
3751
- fcx. ccx . tcx . def_map . borrow_mut ( )
3752
- . insert ( node_id, def:: PathResolution {
3753
- base_def : def,
3754
- depth : 0
3755
- } ) ;
3756
- Some ( ( Some ( ty) , slice:: ref_slice ( item_segment) , def) )
3757
- }
3761
+ let def = match method:: resolve_ufcs ( fcx, span, item_name, ty, node_id) {
3762
+ Ok ( def) => Some ( def) ,
3758
3763
Err ( error) => {
3764
+ let def = match error {
3765
+ method:: MethodError :: PrivateMatch ( def) => Some ( def) ,
3766
+ _ => None ,
3767
+ } ;
3759
3768
if item_name != special_idents:: invalid. name {
3760
3769
method:: report_error ( fcx, span, ty, item_name, None , error) ;
3761
3770
}
3762
- fcx. write_error ( node_id) ;
3763
- None
3771
+ def
3764
3772
}
3773
+ } ;
3774
+
3775
+ if let Some ( def) = def {
3776
+ // Write back the new resolution.
3777
+ fcx. ccx . tcx . def_map . borrow_mut ( ) . insert ( node_id, def:: PathResolution {
3778
+ base_def : def,
3779
+ depth : 0 ,
3780
+ } ) ;
3781
+ Some ( ( Some ( ty) , slice:: ref_slice ( item_segment) , def) )
3782
+ } else {
3783
+ fcx. write_error ( node_id) ;
3784
+ None
3765
3785
}
3766
3786
}
3767
3787
}
0 commit comments