KaiGai Kohei wrote: > Sorry for the late response due to my extreme busy days on the last week. > No problem, and thanks for your reply. I should have sent a patch as a proposal. I was too busy to make (and test) a patch. > Jacques Thomas wrote: > >> Hi All, >> >> I am trying to document the semantics of bounded types, and I am puzzled >> by two things: >> >> >> 1/ The rationale for type boundaries on *target* types >> I understand type boundaries on source types: the idea is that, if typeA >> is bounded by typeB, then typeA can not exert more permissions than >> typeB. This would be declared with the following statement in the source >> policy: >> TYPEBOUND typeB typeA >> >> What is a good use case for type boundaries on target types ? >> > > If a target type (parent) has a few bounded types (children), we can use > the parent as a shared resource, and children as private types for individual > source types. > > For example, please assume httpd_user_content_t and httpd_guest_content_t > are bounded by httpd_sys_content_t. In this example, a domain which tries > to access to the httpd_user_content_t must have a set of equal or wider > permissions on the httpd_sys_content_t. Ditto for httpd_guest_content_t. > If here are two domains ( user_webapp_t and guest_webapp_t ) with its own > private types individually, we can use the boundaries to restrict permissions > to be allowed on the child types. > > However, we have to admit it is a bit hard to image actual use cases. > In fact, we can apply same restriction using regular types. > Almost everything can be expressed just in terms of types. The TYPEBOUND extension, however, makes it much easier to express the containment relationship that you want to establish between the http server *domain* and the *domains* of its worker threads (for Apache/SELinux plus ). Going with your example, what bothers me is that, if I wanted to give guest_webapp_t read/write/create access to httpd_guest_content_t, I would then have to give guest_webapp_t read/write/create access on httpd_sys_content_t. I think the containment relationship should be in the other direction: guest_webapp_t can not have more permissions on httpd_sys_content_t than it has on httpd_guest_content_t. This reminds me of what is called contravariance in type theory. I think it would be less confusing if boundaries were expressed using different keywords for boundaries on domains and boundaries on "data" types, even if the boundaries can be enforced using the same infrastructure code (which is clearly demonstrated by the current code). Keeping only the code that enforces boundaries on domains (subjects) is the simplest fix, and is adequate if there is no compelling example for why we need boundaries on subjects. > Joshua, do you have any idea for use cases of boundaries in targets? > > > >> 2/ The logic of type_attribute_bounds_av in ss/services.c (kernel side) >> This method is invoked to filter/limit an access vector according to the >> type boundaries. Its logic goes as follows: >> >> if(source type has a bound){ // SOURCE >> ../.. >> if(the bound is not violated) >> return >> ../.. >> } >> >> if(target type has a bound){ // TARGET >> ../.. >> if(the bound is not violated) >> return >> ../.. >> } >> >> if(target and source types have a bound){ // BOTH >> ../.. >> if(these bounds are not violated) >> return >> ../.. >> } >> >> if(there was a violation){ >> send audit message to user space >> } >> >> The early return statements seem to violate the intent of the code. For >> instance, suppose that the source type is bounded and no access bits >> violates these bounds, then the code will never evaluate whether bounds >> on the target type are violated. >> >> It seems to me that the logic should be: >> >> // BOTH >> else // SOURCE >> else // TARGET >> >> Instead of: >> >> // SOURCE >> // TARGET >> // BOTH >> > > Assume P(s,t) means a set of permissions allowed between 's' and 't'. > Assume 'Sc' is bounded by 'Sp', and 'Tc' is bounded by 'Tp'. > > In this case, > - P(Sc,Tp) should be equal or less than P(Sp,Tp) due to the SOURCE check. > - P(Sp,Tc) should be equal or less than P(Sp,Tp) due to the TARGET check. > - P(Sc,Tc) should be equal or less than P(Sp,Tp) due to the BOTH check, > and should P(Sc,Tc) equal to (P(Sc,Tp) AND P(Sp,Tc)). > > However, the current implementation immediately returns when no violations > between P(Sc,Tp) and P(Sp,Tp) on the SOURCE check, even if both of types > have its boundaries. The P(Sc,Tc) should be also masked by P(Sc,Tp), but > it seems to me being bypassed. > This is how I read this code too: if the first mask (SOURCE) is empty, the other masks are not evaluated. > If my logic is not incorrect, the implementation should be fixed as follows: > > type_attribute_bounds_av(Sc,Tc, ...) > { > masked = 0; > > if (Sc has its bounds) > masked |= P(Sc,Tc) & ~P(Sp,Tc); > > if (Tc has its bounds) > masked |= P(Sc,Tc) & ~P(Sc,Tp); > > avd->allowed &= ~masked; > } > > Any comments? > It makes sense. In the current code, there is also a check that (following your style) goes as follows: if ((Sc has its bounds) and (Tc has its bounds)) masked |= P(Sc,Tc) & ~P(Sp,Tp); I am not sure of the meaning of this check. > I also think we have one other a rough option. > It simply applies type boundaries on only sources to restrict its privileges, > and it does not apply any restrictions on target types. > Unless there is a clear use for bounds on targets, I would favor this option. (The "rough" one :-) ) I see mostly room for confusion with the bounds on target types, because of the contravariance issue. Thanks a lot for your time, Jacques -- This message was distributed to subscribers of the selinux mailing list. If you no longer wish to subscribe, send mail to majordomo@xxxxxxxxxxxxx with the words "unsubscribe selinux" without quotes as the message.