diff mbox

[v2] selinux: support distinctions among all network address families

Message ID 1481036419-30057-1-git-send-email-sds@tycho.nsa.gov (mailing list archive)
State Accepted
Headers show

Commit Message

Stephen Smalley Dec. 6, 2016, 3 p.m. UTC
Extend SELinux to support distinctions among all network address families
implemented by the kernel by defining new socket security classes
and mapping to them. Otherwise, many sockets are mapped to the generic
socket class and are indistinguishable in policy.  This has come up
previously with regard to selectively allowing access to bluetooth sockets,
and more recently with regard to selectively allowing access to AF_ALG
sockets.  Guido Trentalancia submitted a patch that took a similar approach
to add only support for distinguishing AF_ALG sockets, but this generalizes
his approach to handle all address families implemented by the kernel.
Socket security classes are also added for ICMP and SCTP sockets.
Socket security classes were not defined for AF_* values that are reserved
but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
AF_ECONET, AF_SNA, AF_WANPIPE.

Backward compatibility is provided by only enabling the finer-grained
socket classes if a new policy capability is set in the policy; older
policies will behave as before.  The legacy redhat1 policy capability
that was only ever used in testing within Fedora for ptrace_child
is reclaimed for this purpose; as far as I can tell, this policy
capability is not enabled in any supported distro policy.

Add a pair of conditional compilation guards to detect when new AF_* values
are added so that we can update SELinux accordingly rather than having to
belatedly update it long after new address families are introduced.

Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
---
v2 reworks the style based on comments from Guido Trentalancia and adds
security classes for SCTP and ICMP sockets.  The security class support
for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
we only duplicate the class definition for rawip_socket for SCTP and ICMP
sockets since that is how they were previously mapped.  The SCTP definition
can be further fleshed out by the SCTP patch set itself.

 security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
 security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
 security/selinux/include/security.h |  3 +-
 security/selinux/selinuxfs.c        |  2 +-
 security/selinux/ss/services.c      |  3 ++
 5 files changed, 147 insertions(+), 2 deletions(-)

Comments

Paul Moore Dec. 7, 2016, 12:13 a.m. UTC | #1
On Tue, Dec 6, 2016 at 10:00 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
> Extend SELinux to support distinctions among all network address families
> implemented by the kernel by defining new socket security classes
> and mapping to them. Otherwise, many sockets are mapped to the generic
> socket class and are indistinguishable in policy.  This has come up
> previously with regard to selectively allowing access to bluetooth sockets,
> and more recently with regard to selectively allowing access to AF_ALG
> sockets.  Guido Trentalancia submitted a patch that took a similar approach
> to add only support for distinguishing AF_ALG sockets, but this generalizes
> his approach to handle all address families implemented by the kernel.
> Socket security classes are also added for ICMP and SCTP sockets.
> Socket security classes were not defined for AF_* values that are reserved
> but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
> AF_ECONET, AF_SNA, AF_WANPIPE.
>
> Backward compatibility is provided by only enabling the finer-grained
> socket classes if a new policy capability is set in the policy; older
> policies will behave as before.  The legacy redhat1 policy capability
> that was only ever used in testing within Fedora for ptrace_child
> is reclaimed for this purpose; as far as I can tell, this policy
> capability is not enabled in any supported distro policy.
>
> Add a pair of conditional compilation guards to detect when new AF_* values
> are added so that we can update SELinux accordingly rather than having to
> belatedly update it long after new address families are introduced.
>
> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
> ---
> v2 reworks the style based on comments from Guido Trentalancia and adds
> security classes for SCTP and ICMP sockets.  The security class support
> for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
> we only duplicate the class definition for rawip_socket for SCTP and ICMP
> sockets since that is how they were previously mapped.  The SCTP definition
> can be further fleshed out by the SCTP patch set itself.
>
>  security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
>  security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
>  security/selinux/include/security.h |  3 +-
>  security/selinux/selinuxfs.c        |  2 +-
>  security/selinux/ss/services.c      |  3 ++
>  5 files changed, 147 insertions(+), 2 deletions(-)

You mentioned IGMP previously, if we have a class for ICMP, it seems
reasonable to have one for IGMP, don't you think?  Although this does
spiral a bit if we consider all the IPPROTO* protocols.

> diff --git a/security/selinux/hooks.c b/security/selinux/hooks.c
> index 98a2e92..b4402b2 100644
> --- a/security/selinux/hooks.c
> +++ b/security/selinux/hooks.c
> @@ -1268,6 +1268,8 @@ static inline int default_protocol_dgram(int protocol)
>
>  static inline u16 socket_type_to_security_class(int family, int type, int protocol)
>  {
> +       int extsockclass = selinux_policycap_extsockclass;
> +
>         switch (family) {
>         case PF_UNIX:
>                 switch (type) {
> @@ -1282,13 +1284,18 @@ static inline u16 socket_type_to_security_class(int family, int type, int protoc
>         case PF_INET6:
>                 switch (type) {
>                 case SOCK_STREAM:
> +               case SOCK_SEQPACKET:
>                         if (default_protocol_stream(protocol))
>                                 return SECCLASS_TCP_SOCKET;
> +                       else if (extsockclass && protocol == IPPROTO_SCTP)
> +                               return SECCLASS_SCTP_SOCKET;
>                         else
>                                 return SECCLASS_RAWIP_SOCKET;
>                 case SOCK_DGRAM:
>                         if (default_protocol_dgram(protocol))
>                                 return SECCLASS_UDP_SOCKET;
> +                       else if (extsockclass && protocol == IPPROTO_ICMP)
> +                               return SECCLASS_ICMP_SOCKET;
>                         else
>                                 return SECCLASS_RAWIP_SOCKET;
>                 case SOCK_DCCP:
> @@ -1342,6 +1349,72 @@ static inline u16 socket_type_to_security_class(int family, int type, int protoc
>                 return SECCLASS_APPLETALK_SOCKET;
>         }
>
> +       if (extsockclass) {
> +               switch (family) {
> +               case PF_AX25:
> +                       return SECCLASS_AX25_SOCKET;
> +               case PF_IPX:
> +                       return SECCLASS_IPX_SOCKET;
> +               case PF_NETROM:
> +                       return SECCLASS_NETROM_SOCKET;
> +               case PF_BRIDGE:
> +                       return SECCLASS_BRIDGE_SOCKET;
> +               case PF_ATMPVC:
> +                       return SECCLASS_ATMPVC_SOCKET;
> +               case PF_X25:
> +                       return SECCLASS_X25_SOCKET;
> +               case PF_ROSE:
> +                       return SECCLASS_ROSE_SOCKET;
> +               case PF_DECnet:
> +                       return SECCLASS_DECNET_SOCKET;
> +               case PF_ATMSVC:
> +                       return SECCLASS_ATMSVC_SOCKET;
> +               case PF_RDS:
> +                       return SECCLASS_RDS_SOCKET;
> +               case PF_IRDA:
> +                       return SECCLASS_IRDA_SOCKET;
> +               case PF_PPPOX:
> +                       return SECCLASS_PPPOX_SOCKET;
> +               case PF_LLC:
> +                       return SECCLASS_LLC_SOCKET;
> +               case PF_IB:
> +                       return SECCLASS_IB_SOCKET;
> +               case PF_MPLS:
> +                       return SECCLASS_MPLS_SOCKET;
> +               case PF_CAN:
> +                       return SECCLASS_CAN_SOCKET;
> +               case PF_TIPC:
> +                       return SECCLASS_TIPC_SOCKET;
> +               case PF_BLUETOOTH:
> +                       return SECCLASS_BLUETOOTH_SOCKET;
> +               case PF_IUCV:
> +                       return SECCLASS_IUCV_SOCKET;
> +               case PF_RXRPC:
> +                       return SECCLASS_RXRPC_SOCKET;
> +               case PF_ISDN:
> +                       return SECCLASS_ISDN_SOCKET;
> +               case PF_PHONET:
> +                       return SECCLASS_PHONET_SOCKET;
> +               case PF_IEEE802154:
> +                       return SECCLASS_IEEE802154_SOCKET;
> +               case PF_CAIF:
> +                       return SECCLASS_CAIF_SOCKET;
> +               case PF_ALG:
> +                       return SECCLASS_ALG_SOCKET;
> +               case PF_NFC:
> +                       return SECCLASS_NFC_SOCKET;
> +               case PF_VSOCK:
> +                       return SECCLASS_VSOCK_SOCKET;
> +               case PF_KCM:
> +                       return SECCLASS_KCM_SOCKET;
> +               case PF_QIPCRTR:
> +                       return SECCLASS_QIPCRTR_SOCKET;
> +#if PF_MAX > 43
> +#error New address family defined, please update this function.
> +#endif
> +               }
> +       }
> +
>         return SECCLASS_SOCKET;
>  }
>
> diff --git a/security/selinux/include/classmap.h b/security/selinux/include/classmap.h
> index e2d4ad3a..8dc67f4 100644
> --- a/security/selinux/include/classmap.h
> +++ b/security/selinux/include/classmap.h
> @@ -169,5 +169,73 @@ struct security_class_mapping secclass_map[] = {
>           { COMMON_CAP_PERMS, NULL } },
>         { "cap2_userns",
>           { COMMON_CAP2_PERMS, NULL } },
> +       { "sctp_socket",
> +         { COMMON_SOCK_PERMS,
> +           "node_bind", NULL } },
> +       { "icmp_socket",
> +         { COMMON_SOCK_PERMS,
> +           "node_bind", NULL } },
> +       { "ax25_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "ipx_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "netrom_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "bridge_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "atmpvc_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "x25_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "rose_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "decnet_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "atmsvc_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "rds_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "irda_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "pppox_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "llc_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "ib_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "mpls_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "can_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "tipc_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "bluetooth_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "iucv_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "rxrpc_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "isdn_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "phonet_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "ieee802154_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "caif_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "alg_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "nfc_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "vsock_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "kcm_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
> +       { "qipcrtr_socket",
> +         { COMMON_SOCK_PERMS, NULL } },
>         { NULL }
>    };
> +
> +#if PF_MAX > 43
> +#error New address family defined, please update secclass_map.
> +#endif
> diff --git a/security/selinux/include/security.h b/security/selinux/include/security.h
> index 308a286..beaa14b 100644
> --- a/security/selinux/include/security.h
> +++ b/security/selinux/include/security.h
> @@ -69,7 +69,7 @@ extern int selinux_enabled;
>  enum {
>         POLICYDB_CAPABILITY_NETPEER,
>         POLICYDB_CAPABILITY_OPENPERM,
> -       POLICYDB_CAPABILITY_REDHAT1,
> +       POLICYDB_CAPABILITY_EXTSOCKCLASS,
>         POLICYDB_CAPABILITY_ALWAYSNETWORK,
>         __POLICYDB_CAPABILITY_MAX
>  };
> @@ -77,6 +77,7 @@ enum {
>
>  extern int selinux_policycap_netpeer;
>  extern int selinux_policycap_openperm;
> +extern int selinux_policycap_extsockclass;
>  extern int selinux_policycap_alwaysnetwork;
>
>  /*
> diff --git a/security/selinux/selinuxfs.c b/security/selinux/selinuxfs.c
> index cf9293e..0aac402 100644
> --- a/security/selinux/selinuxfs.c
> +++ b/security/selinux/selinuxfs.c
> @@ -45,7 +45,7 @@
>  static char *policycap_names[] = {
>         "network_peer_controls",
>         "open_perms",
> -       "redhat1",
> +       "extended_socket_class",
>         "always_check_network"
>  };
>
> diff --git a/security/selinux/ss/services.c b/security/selinux/ss/services.c
> index 082b20c..a70fcee 100644
> --- a/security/selinux/ss/services.c
> +++ b/security/selinux/ss/services.c
> @@ -72,6 +72,7 @@
>
>  int selinux_policycap_netpeer;
>  int selinux_policycap_openperm;
> +int selinux_policycap_extsockclass;
>  int selinux_policycap_alwaysnetwork;
>
>  static DEFINE_RWLOCK(policy_rwlock);
> @@ -1988,6 +1989,8 @@ static void security_load_policycaps(void)
>                                                   POLICYDB_CAPABILITY_NETPEER);
>         selinux_policycap_openperm = ebitmap_get_bit(&policydb.policycaps,
>                                                   POLICYDB_CAPABILITY_OPENPERM);
> +       selinux_policycap_extsockclass = ebitmap_get_bit(&policydb.policycaps,
> +                                         POLICYDB_CAPABILITY_EXTSOCKCLASS);
>         selinux_policycap_alwaysnetwork = ebitmap_get_bit(&policydb.policycaps,
>                                                   POLICYDB_CAPABILITY_ALWAYSNETWORK);
>  }
> --
> 2.7.4
>
Stephen Smalley Dec. 7, 2016, 1:25 p.m. UTC | #2
On 12/06/2016 07:13 PM, Paul Moore wrote:
> On Tue, Dec 6, 2016 at 10:00 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>> Extend SELinux to support distinctions among all network address families
>> implemented by the kernel by defining new socket security classes
>> and mapping to them. Otherwise, many sockets are mapped to the generic
>> socket class and are indistinguishable in policy.  This has come up
>> previously with regard to selectively allowing access to bluetooth sockets,
>> and more recently with regard to selectively allowing access to AF_ALG
>> sockets.  Guido Trentalancia submitted a patch that took a similar approach
>> to add only support for distinguishing AF_ALG sockets, but this generalizes
>> his approach to handle all address families implemented by the kernel.
>> Socket security classes are also added for ICMP and SCTP sockets.
>> Socket security classes were not defined for AF_* values that are reserved
>> but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
>> AF_ECONET, AF_SNA, AF_WANPIPE.
>>
>> Backward compatibility is provided by only enabling the finer-grained
>> socket classes if a new policy capability is set in the policy; older
>> policies will behave as before.  The legacy redhat1 policy capability
>> that was only ever used in testing within Fedora for ptrace_child
>> is reclaimed for this purpose; as far as I can tell, this policy
>> capability is not enabled in any supported distro policy.
>>
>> Add a pair of conditional compilation guards to detect when new AF_* values
>> are added so that we can update SELinux accordingly rather than having to
>> belatedly update it long after new address families are introduced.
>>
>> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
>> ---
>> v2 reworks the style based on comments from Guido Trentalancia and adds
>> security classes for SCTP and ICMP sockets.  The security class support
>> for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
>> we only duplicate the class definition for rawip_socket for SCTP and ICMP
>> sockets since that is how they were previously mapped.  The SCTP definition
>> can be further fleshed out by the SCTP patch set itself.
>>
>>  security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
>>  security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
>>  security/selinux/include/security.h |  3 +-
>>  security/selinux/selinuxfs.c        |  2 +-
>>  security/selinux/ss/services.c      |  3 ++
>>  5 files changed, 147 insertions(+), 2 deletions(-)
> 
> You mentioned IGMP previously, if we have a class for ICMP, it seems
> reasonable to have one for IGMP, don't you think?  Although this does
> spiral a bit if we consider all the IPPROTO* protocols.

I thought about it, but the kernel does not provide IGMP sockets per se,
unlike ICMP or SCTP sockets (i.e. ipv4/af_inet.c:inetsw_array[] defines
an entry for SOCK_DGRAM, IPPROTO_ICMP and sctp/protocol.c defines and
registers inet_protosw entries for SOCK_STREAM, IPPROTO_SCTP and
SOCK_SEQPACKET, IPPROTO_SCTP; there is no equivalent for IGMP unless I
missed it).  So IGMP sockets are just raw IP sockets with a particular
protocol value; they have no stream, seqpacket, or dgram semantics, and
it is unclear it is worthwhile to distinguish them in policy.
Paul Moore Dec. 7, 2016, 3:14 p.m. UTC | #3
On Wed, Dec 7, 2016 at 8:25 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
> On 12/06/2016 07:13 PM, Paul Moore wrote:
>> On Tue, Dec 6, 2016 at 10:00 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>>> Extend SELinux to support distinctions among all network address families
>>> implemented by the kernel by defining new socket security classes
>>> and mapping to them. Otherwise, many sockets are mapped to the generic
>>> socket class and are indistinguishable in policy.  This has come up
>>> previously with regard to selectively allowing access to bluetooth sockets,
>>> and more recently with regard to selectively allowing access to AF_ALG
>>> sockets.  Guido Trentalancia submitted a patch that took a similar approach
>>> to add only support for distinguishing AF_ALG sockets, but this generalizes
>>> his approach to handle all address families implemented by the kernel.
>>> Socket security classes are also added for ICMP and SCTP sockets.
>>> Socket security classes were not defined for AF_* values that are reserved
>>> but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
>>> AF_ECONET, AF_SNA, AF_WANPIPE.
>>>
>>> Backward compatibility is provided by only enabling the finer-grained
>>> socket classes if a new policy capability is set in the policy; older
>>> policies will behave as before.  The legacy redhat1 policy capability
>>> that was only ever used in testing within Fedora for ptrace_child
>>> is reclaimed for this purpose; as far as I can tell, this policy
>>> capability is not enabled in any supported distro policy.
>>>
>>> Add a pair of conditional compilation guards to detect when new AF_* values
>>> are added so that we can update SELinux accordingly rather than having to
>>> belatedly update it long after new address families are introduced.
>>>
>>> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
>>> ---
>>> v2 reworks the style based on comments from Guido Trentalancia and adds
>>> security classes for SCTP and ICMP sockets.  The security class support
>>> for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
>>> we only duplicate the class definition for rawip_socket for SCTP and ICMP
>>> sockets since that is how they were previously mapped.  The SCTP definition
>>> can be further fleshed out by the SCTP patch set itself.
>>>
>>>  security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
>>>  security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
>>>  security/selinux/include/security.h |  3 +-
>>>  security/selinux/selinuxfs.c        |  2 +-
>>>  security/selinux/ss/services.c      |  3 ++
>>>  5 files changed, 147 insertions(+), 2 deletions(-)
>>
>> You mentioned IGMP previously, if we have a class for ICMP, it seems
>> reasonable to have one for IGMP, don't you think?  Although this does
>> spiral a bit if we consider all the IPPROTO* protocols.
>
> I thought about it, but the kernel does not provide IGMP sockets per se,
> unlike ICMP or SCTP sockets (i.e. ipv4/af_inet.c:inetsw_array[] defines
> an entry for SOCK_DGRAM, IPPROTO_ICMP and sctp/protocol.c defines and
> registers inet_protosw entries for SOCK_STREAM, IPPROTO_SCTP and
> SOCK_SEQPACKET, IPPROTO_SCTP; there is no equivalent for IGMP unless I
> missed it).  So IGMP sockets are just raw IP sockets with a particular
> protocol value; they have no stream, seqpacket, or dgram semantics, and
> it is unclear it is worthwhile to distinguish them in policy.

Yes, sorry about that, it looks like you're right.  I thought there
might be some sort of IGMP routing daemon or something that would need
to create an IGMP socket, but it looks like you just configure the
kernel's multicast routing table and the kernel takes care of the IGMP
packets.
Guido Trentalancia Dec. 7, 2016, 3:15 p.m. UTC | #4
Hello.

On Wed, 07/12/2016 at 08.25 -0500, Stephen Smalley wrote:
> On 12/06/2016 07:13 PM, Paul Moore wrote:

[...]

> > You mentioned IGMP previously, if we have a class for ICMP, it
> > seems
> > reasonable to have one for IGMP, don't you think?  Although this
> > does
> > spiral a bit if we consider all the IPPROTO* protocols.
> 
> I thought about it, but the kernel does not provide IGMP sockets per
> se,
> unlike ICMP or SCTP sockets (i.e. ipv4/af_inet.c:inetsw_array[]
> defines
> an entry for SOCK_DGRAM, IPPROTO_ICMP and sctp/protocol.c defines and
> registers inet_protosw entries for SOCK_STREAM, IPPROTO_SCTP and
> SOCK_SEQPACKET, IPPROTO_SCTP; there is no equivalent for IGMP unless
> I
> missed it).  So IGMP sockets are just raw IP sockets with a
> particular
> protocol value; they have no stream, seqpacket, or dgram semantics,
> and
> it is unclear it is worthwhile to distinguish them in policy.

I suppose distinguishing IGMP packets brings little benefit in terms of
security.

Regards,

Guido
Stephen Smalley Dec. 7, 2016, 3:24 p.m. UTC | #5
On 12/07/2016 10:14 AM, Paul Moore wrote:
> On Wed, Dec 7, 2016 at 8:25 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>> On 12/06/2016 07:13 PM, Paul Moore wrote:
>>> On Tue, Dec 6, 2016 at 10:00 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>>>> Extend SELinux to support distinctions among all network address families
>>>> implemented by the kernel by defining new socket security classes
>>>> and mapping to them. Otherwise, many sockets are mapped to the generic
>>>> socket class and are indistinguishable in policy.  This has come up
>>>> previously with regard to selectively allowing access to bluetooth sockets,
>>>> and more recently with regard to selectively allowing access to AF_ALG
>>>> sockets.  Guido Trentalancia submitted a patch that took a similar approach
>>>> to add only support for distinguishing AF_ALG sockets, but this generalizes
>>>> his approach to handle all address families implemented by the kernel.
>>>> Socket security classes are also added for ICMP and SCTP sockets.
>>>> Socket security classes were not defined for AF_* values that are reserved
>>>> but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
>>>> AF_ECONET, AF_SNA, AF_WANPIPE.
>>>>
>>>> Backward compatibility is provided by only enabling the finer-grained
>>>> socket classes if a new policy capability is set in the policy; older
>>>> policies will behave as before.  The legacy redhat1 policy capability
>>>> that was only ever used in testing within Fedora for ptrace_child
>>>> is reclaimed for this purpose; as far as I can tell, this policy
>>>> capability is not enabled in any supported distro policy.
>>>>
>>>> Add a pair of conditional compilation guards to detect when new AF_* values
>>>> are added so that we can update SELinux accordingly rather than having to
>>>> belatedly update it long after new address families are introduced.
>>>>
>>>> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
>>>> ---
>>>> v2 reworks the style based on comments from Guido Trentalancia and adds
>>>> security classes for SCTP and ICMP sockets.  The security class support
>>>> for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
>>>> we only duplicate the class definition for rawip_socket for SCTP and ICMP
>>>> sockets since that is how they were previously mapped.  The SCTP definition
>>>> can be further fleshed out by the SCTP patch set itself.
>>>>
>>>>  security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
>>>>  security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
>>>>  security/selinux/include/security.h |  3 +-
>>>>  security/selinux/selinuxfs.c        |  2 +-
>>>>  security/selinux/ss/services.c      |  3 ++
>>>>  5 files changed, 147 insertions(+), 2 deletions(-)
>>>
>>> You mentioned IGMP previously, if we have a class for ICMP, it seems
>>> reasonable to have one for IGMP, don't you think?  Although this does
>>> spiral a bit if we consider all the IPPROTO* protocols.
>>
>> I thought about it, but the kernel does not provide IGMP sockets per se,
>> unlike ICMP or SCTP sockets (i.e. ipv4/af_inet.c:inetsw_array[] defines
>> an entry for SOCK_DGRAM, IPPROTO_ICMP and sctp/protocol.c defines and
>> registers inet_protosw entries for SOCK_STREAM, IPPROTO_SCTP and
>> SOCK_SEQPACKET, IPPROTO_SCTP; there is no equivalent for IGMP unless I
>> missed it).  So IGMP sockets are just raw IP sockets with a particular
>> protocol value; they have no stream, seqpacket, or dgram semantics, and
>> it is unclear it is worthwhile to distinguish them in policy.
> 
> Yes, sorry about that, it looks like you're right.  I thought there
> might be some sort of IGMP routing daemon or something that would need
> to create an IGMP socket, but it looks like you just configure the
> kernel's multicast routing table and the kernel takes care of the IGMP
> packets.

Well, I think you can create an IGMP socket ala socket(PF_INET,
SOCK_RAW, IPPROTO_IGMP).  And we could map that to a IGMP socket class
if we really wanted to do so.  But that would be the first time we ever
mapped a (PF_INET, SOCK_RAW, proto) to something other than
SECCLASS_RAWIP_SOCKET.  And at that point we'd probably want to map
every individual protocol value.  Not sure we need/want to go there.
Paul Moore Dec. 7, 2016, 11:32 p.m. UTC | #6
On Wed, Dec 7, 2016 at 10:24 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
> On 12/07/2016 10:14 AM, Paul Moore wrote:
>> On Wed, Dec 7, 2016 at 8:25 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>>> On 12/06/2016 07:13 PM, Paul Moore wrote:
>>>> On Tue, Dec 6, 2016 at 10:00 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>>>>> Extend SELinux to support distinctions among all network address families
>>>>> implemented by the kernel by defining new socket security classes
>>>>> and mapping to them. Otherwise, many sockets are mapped to the generic
>>>>> socket class and are indistinguishable in policy.  This has come up
>>>>> previously with regard to selectively allowing access to bluetooth sockets,
>>>>> and more recently with regard to selectively allowing access to AF_ALG
>>>>> sockets.  Guido Trentalancia submitted a patch that took a similar approach
>>>>> to add only support for distinguishing AF_ALG sockets, but this generalizes
>>>>> his approach to handle all address families implemented by the kernel.
>>>>> Socket security classes are also added for ICMP and SCTP sockets.
>>>>> Socket security classes were not defined for AF_* values that are reserved
>>>>> but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
>>>>> AF_ECONET, AF_SNA, AF_WANPIPE.
>>>>>
>>>>> Backward compatibility is provided by only enabling the finer-grained
>>>>> socket classes if a new policy capability is set in the policy; older
>>>>> policies will behave as before.  The legacy redhat1 policy capability
>>>>> that was only ever used in testing within Fedora for ptrace_child
>>>>> is reclaimed for this purpose; as far as I can tell, this policy
>>>>> capability is not enabled in any supported distro policy.
>>>>>
>>>>> Add a pair of conditional compilation guards to detect when new AF_* values
>>>>> are added so that we can update SELinux accordingly rather than having to
>>>>> belatedly update it long after new address families are introduced.
>>>>>
>>>>> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
>>>>> ---
>>>>> v2 reworks the style based on comments from Guido Trentalancia and adds
>>>>> security classes for SCTP and ICMP sockets.  The security class support
>>>>> for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
>>>>> we only duplicate the class definition for rawip_socket for SCTP and ICMP
>>>>> sockets since that is how they were previously mapped.  The SCTP definition
>>>>> can be further fleshed out by the SCTP patch set itself.
>>>>>
>>>>>  security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
>>>>>  security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
>>>>>  security/selinux/include/security.h |  3 +-
>>>>>  security/selinux/selinuxfs.c        |  2 +-
>>>>>  security/selinux/ss/services.c      |  3 ++
>>>>>  5 files changed, 147 insertions(+), 2 deletions(-)
>>>>
>>>> You mentioned IGMP previously, if we have a class for ICMP, it seems
>>>> reasonable to have one for IGMP, don't you think?  Although this does
>>>> spiral a bit if we consider all the IPPROTO* protocols.
>>>
>>> I thought about it, but the kernel does not provide IGMP sockets per se,
>>> unlike ICMP or SCTP sockets (i.e. ipv4/af_inet.c:inetsw_array[] defines
>>> an entry for SOCK_DGRAM, IPPROTO_ICMP and sctp/protocol.c defines and
>>> registers inet_protosw entries for SOCK_STREAM, IPPROTO_SCTP and
>>> SOCK_SEQPACKET, IPPROTO_SCTP; there is no equivalent for IGMP unless I
>>> missed it).  So IGMP sockets are just raw IP sockets with a particular
>>> protocol value; they have no stream, seqpacket, or dgram semantics, and
>>> it is unclear it is worthwhile to distinguish them in policy.
>>
>> Yes, sorry about that, it looks like you're right.  I thought there
>> might be some sort of IGMP routing daemon or something that would need
>> to create an IGMP socket, but it looks like you just configure the
>> kernel's multicast routing table and the kernel takes care of the IGMP
>> packets.
>
> Well, I think you can create an IGMP socket ala socket(PF_INET,
> SOCK_RAW, IPPROTO_IGMP).  And we could map that to a IGMP socket class
> if we really wanted to do so.  But that would be the first time we ever
> mapped a (PF_INET, SOCK_RAW, proto) to something other than
> SECCLASS_RAWIP_SOCKET.  And at that point we'd probably want to map
> every individual protocol value.  Not sure we need/want to go there.

Yes, creating classes for each IPPROTO value is not something I think
we want to do, although I have no problem doing it for key protocols
as necessary.

I just merged this into selinux#next, the pull request for v4.10 was
already sent up to James so this will go in during the v4.11 merge
window (I think this can wait, it might give us time to get some of
the policy ready).
Stephen Smalley Dec. 8, 2016, 7:42 p.m. UTC | #7
On 12/07/2016 06:32 PM, Paul Moore wrote:
> On Wed, Dec 7, 2016 at 10:24 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>> On 12/07/2016 10:14 AM, Paul Moore wrote:
>>> On Wed, Dec 7, 2016 at 8:25 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>>>> On 12/06/2016 07:13 PM, Paul Moore wrote:
>>>>> On Tue, Dec 6, 2016 at 10:00 AM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
>>>>>> Extend SELinux to support distinctions among all network address families
>>>>>> implemented by the kernel by defining new socket security classes
>>>>>> and mapping to them. Otherwise, many sockets are mapped to the generic
>>>>>> socket class and are indistinguishable in policy.  This has come up
>>>>>> previously with regard to selectively allowing access to bluetooth sockets,
>>>>>> and more recently with regard to selectively allowing access to AF_ALG
>>>>>> sockets.  Guido Trentalancia submitted a patch that took a similar approach
>>>>>> to add only support for distinguishing AF_ALG sockets, but this generalizes
>>>>>> his approach to handle all address families implemented by the kernel.
>>>>>> Socket security classes are also added for ICMP and SCTP sockets.
>>>>>> Socket security classes were not defined for AF_* values that are reserved
>>>>>> but unimplemented in the kernel, e.g. AF_NETBEUI, AF_SECURITY, AF_ASH,
>>>>>> AF_ECONET, AF_SNA, AF_WANPIPE.
>>>>>>
>>>>>> Backward compatibility is provided by only enabling the finer-grained
>>>>>> socket classes if a new policy capability is set in the policy; older
>>>>>> policies will behave as before.  The legacy redhat1 policy capability
>>>>>> that was only ever used in testing within Fedora for ptrace_child
>>>>>> is reclaimed for this purpose; as far as I can tell, this policy
>>>>>> capability is not enabled in any supported distro policy.
>>>>>>
>>>>>> Add a pair of conditional compilation guards to detect when new AF_* values
>>>>>> are added so that we can update SELinux accordingly rather than having to
>>>>>> belatedly update it long after new address families are introduced.
>>>>>>
>>>>>> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
>>>>>> ---
>>>>>> v2 reworks the style based on comments from Guido Trentalancia and adds
>>>>>> security classes for SCTP and ICMP sockets.  The security class support
>>>>>> for SCTP sockets was based on Richard Haines' SCTP patch set.  For now,
>>>>>> we only duplicate the class definition for rawip_socket for SCTP and ICMP
>>>>>> sockets since that is how they were previously mapped.  The SCTP definition
>>>>>> can be further fleshed out by the SCTP patch set itself.
>>>>>>
>>>>>>  security/selinux/hooks.c            | 73 +++++++++++++++++++++++++++++++++++++
>>>>>>  security/selinux/include/classmap.h | 68 ++++++++++++++++++++++++++++++++++
>>>>>>  security/selinux/include/security.h |  3 +-
>>>>>>  security/selinux/selinuxfs.c        |  2 +-
>>>>>>  security/selinux/ss/services.c      |  3 ++
>>>>>>  5 files changed, 147 insertions(+), 2 deletions(-)
>>>>>
>>>>> You mentioned IGMP previously, if we have a class for ICMP, it seems
>>>>> reasonable to have one for IGMP, don't you think?  Although this does
>>>>> spiral a bit if we consider all the IPPROTO* protocols.
>>>>
>>>> I thought about it, but the kernel does not provide IGMP sockets per se,
>>>> unlike ICMP or SCTP sockets (i.e. ipv4/af_inet.c:inetsw_array[] defines
>>>> an entry for SOCK_DGRAM, IPPROTO_ICMP and sctp/protocol.c defines and
>>>> registers inet_protosw entries for SOCK_STREAM, IPPROTO_SCTP and
>>>> SOCK_SEQPACKET, IPPROTO_SCTP; there is no equivalent for IGMP unless I
>>>> missed it).  So IGMP sockets are just raw IP sockets with a particular
>>>> protocol value; they have no stream, seqpacket, or dgram semantics, and
>>>> it is unclear it is worthwhile to distinguish them in policy.
>>>
>>> Yes, sorry about that, it looks like you're right.  I thought there
>>> might be some sort of IGMP routing daemon or something that would need
>>> to create an IGMP socket, but it looks like you just configure the
>>> kernel's multicast routing table and the kernel takes care of the IGMP
>>> packets.
>>
>> Well, I think you can create an IGMP socket ala socket(PF_INET,
>> SOCK_RAW, IPPROTO_IGMP).  And we could map that to a IGMP socket class
>> if we really wanted to do so.  But that would be the first time we ever
>> mapped a (PF_INET, SOCK_RAW, proto) to something other than
>> SECCLASS_RAWIP_SOCKET.  And at that point we'd probably want to map
>> every individual protocol value.  Not sure we need/want to go there.
> 
> Yes, creating classes for each IPPROTO value is not something I think
> we want to do, although I have no problem doing it for key protocols
> as necessary.
> 
> I just merged this into selinux#next, the pull request for v4.10 was
> already sent up to James so this will go in during the v4.11 merge
> window (I think this can wait, it might give us time to get some of
> the policy ready).

Wanted to clarify one point and also mention a bug/deficiency of this
patch that I just realized:

- Clarification: At present, the patch will map PF_INET, SOCK_DGRAM,
IPPROTO_ICMP to icmp_socket, but will continue to map PF_INET, SOCK_RAW,
IPPROTO_ICMP to rawip_socket.  That's intentional, since the SOCK_DGRAM
IPPROTO_ICMP sockets were introduced to provide a more restricted form
of ICMP sockets that could be exposed to unprivileged users (based on a
sysctl), so distinguishing those in policy makes sense.  But it could be
surprising.

- Bug/deficiency:  I hadn't noticed that there is a separate
IPPROTO_ICMPV6, so I'll need to modify this patch or post a follow-on
that ensure that we map SOCK_DGRAM,IPPROTO_ICMPV6 to icmp_socket too.
Paul Moore Dec. 9, 2016, 11:56 p.m. UTC | #8
On Thu, Dec 8, 2016 at 2:42 PM, Stephen Smalley <sds@tycho.nsa.gov> wrote:
> Wanted to clarify one point and also mention a bug/deficiency of this
> patch that I just realized:
>
> - Clarification: At present, the patch will map PF_INET, SOCK_DGRAM,
> IPPROTO_ICMP to icmp_socket, but will continue to map PF_INET, SOCK_RAW,
> IPPROTO_ICMP to rawip_socket.  That's intentional, since the SOCK_DGRAM
> IPPROTO_ICMP sockets were introduced to provide a more restricted form
> of ICMP sockets that could be exposed to unprivileged users (based on a
> sysctl), so distinguishing those in policy makes sense.  But it could be
> surprising.

Yes, I have no problem with this behavior.

> - Bug/deficiency:  I hadn't noticed that there is a separate
> IPPROTO_ICMPV6, so I'll need to modify this patch or post a follow-on
> that ensure that we map SOCK_DGRAM,IPPROTO_ICMPV6 to icmp_socket too.

Merged your follow-up patch.
diff mbox

Patch

diff --git a/security/selinux/hooks.c b/security/selinux/hooks.c
index 98a2e92..b4402b2 100644
--- a/security/selinux/hooks.c
+++ b/security/selinux/hooks.c
@@ -1268,6 +1268,8 @@  static inline int default_protocol_dgram(int protocol)
 
 static inline u16 socket_type_to_security_class(int family, int type, int protocol)
 {
+	int extsockclass = selinux_policycap_extsockclass;
+
 	switch (family) {
 	case PF_UNIX:
 		switch (type) {
@@ -1282,13 +1284,18 @@  static inline u16 socket_type_to_security_class(int family, int type, int protoc
 	case PF_INET6:
 		switch (type) {
 		case SOCK_STREAM:
+		case SOCK_SEQPACKET:
 			if (default_protocol_stream(protocol))
 				return SECCLASS_TCP_SOCKET;
+			else if (extsockclass && protocol == IPPROTO_SCTP)
+				return SECCLASS_SCTP_SOCKET;
 			else
 				return SECCLASS_RAWIP_SOCKET;
 		case SOCK_DGRAM:
 			if (default_protocol_dgram(protocol))
 				return SECCLASS_UDP_SOCKET;
+			else if (extsockclass && protocol == IPPROTO_ICMP)
+				return SECCLASS_ICMP_SOCKET;
 			else
 				return SECCLASS_RAWIP_SOCKET;
 		case SOCK_DCCP:
@@ -1342,6 +1349,72 @@  static inline u16 socket_type_to_security_class(int family, int type, int protoc
 		return SECCLASS_APPLETALK_SOCKET;
 	}
 
+	if (extsockclass) {
+		switch (family) {
+		case PF_AX25:
+			return SECCLASS_AX25_SOCKET;
+		case PF_IPX:
+			return SECCLASS_IPX_SOCKET;
+		case PF_NETROM:
+			return SECCLASS_NETROM_SOCKET;
+		case PF_BRIDGE:
+			return SECCLASS_BRIDGE_SOCKET;
+		case PF_ATMPVC:
+			return SECCLASS_ATMPVC_SOCKET;
+		case PF_X25:
+			return SECCLASS_X25_SOCKET;
+		case PF_ROSE:
+			return SECCLASS_ROSE_SOCKET;
+		case PF_DECnet:
+			return SECCLASS_DECNET_SOCKET;
+		case PF_ATMSVC:
+			return SECCLASS_ATMSVC_SOCKET;
+		case PF_RDS:
+			return SECCLASS_RDS_SOCKET;
+		case PF_IRDA:
+			return SECCLASS_IRDA_SOCKET;
+		case PF_PPPOX:
+			return SECCLASS_PPPOX_SOCKET;
+		case PF_LLC:
+			return SECCLASS_LLC_SOCKET;
+		case PF_IB:
+			return SECCLASS_IB_SOCKET;
+		case PF_MPLS:
+			return SECCLASS_MPLS_SOCKET;
+		case PF_CAN:
+			return SECCLASS_CAN_SOCKET;
+		case PF_TIPC:
+			return SECCLASS_TIPC_SOCKET;
+		case PF_BLUETOOTH:
+			return SECCLASS_BLUETOOTH_SOCKET;
+		case PF_IUCV:
+			return SECCLASS_IUCV_SOCKET;
+		case PF_RXRPC:
+			return SECCLASS_RXRPC_SOCKET;
+		case PF_ISDN:
+			return SECCLASS_ISDN_SOCKET;
+		case PF_PHONET:
+			return SECCLASS_PHONET_SOCKET;
+		case PF_IEEE802154:
+			return SECCLASS_IEEE802154_SOCKET;
+		case PF_CAIF:
+			return SECCLASS_CAIF_SOCKET;
+		case PF_ALG:
+			return SECCLASS_ALG_SOCKET;
+		case PF_NFC:
+			return SECCLASS_NFC_SOCKET;
+		case PF_VSOCK:
+			return SECCLASS_VSOCK_SOCKET;
+		case PF_KCM:
+			return SECCLASS_KCM_SOCKET;
+		case PF_QIPCRTR:
+			return SECCLASS_QIPCRTR_SOCKET;
+#if PF_MAX > 43
+#error New address family defined, please update this function.
+#endif
+		}
+	}
+
 	return SECCLASS_SOCKET;
 }
 
diff --git a/security/selinux/include/classmap.h b/security/selinux/include/classmap.h
index e2d4ad3a..8dc67f4 100644
--- a/security/selinux/include/classmap.h
+++ b/security/selinux/include/classmap.h
@@ -169,5 +169,73 @@  struct security_class_mapping secclass_map[] = {
 	  { COMMON_CAP_PERMS, NULL } },
 	{ "cap2_userns",
 	  { COMMON_CAP2_PERMS, NULL } },
+	{ "sctp_socket",
+	  { COMMON_SOCK_PERMS,
+	    "node_bind", NULL } },
+	{ "icmp_socket",
+	  { COMMON_SOCK_PERMS,
+	    "node_bind", NULL } },
+	{ "ax25_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "ipx_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "netrom_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "bridge_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "atmpvc_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "x25_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "rose_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "decnet_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "atmsvc_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "rds_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "irda_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "pppox_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "llc_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "ib_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "mpls_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "can_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "tipc_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "bluetooth_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "iucv_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "rxrpc_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "isdn_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "phonet_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "ieee802154_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "caif_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "alg_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "nfc_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "vsock_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "kcm_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
+	{ "qipcrtr_socket",
+	  { COMMON_SOCK_PERMS, NULL } },
 	{ NULL }
   };
+
+#if PF_MAX > 43
+#error New address family defined, please update secclass_map.
+#endif
diff --git a/security/selinux/include/security.h b/security/selinux/include/security.h
index 308a286..beaa14b 100644
--- a/security/selinux/include/security.h
+++ b/security/selinux/include/security.h
@@ -69,7 +69,7 @@  extern int selinux_enabled;
 enum {
 	POLICYDB_CAPABILITY_NETPEER,
 	POLICYDB_CAPABILITY_OPENPERM,
-	POLICYDB_CAPABILITY_REDHAT1,
+	POLICYDB_CAPABILITY_EXTSOCKCLASS,
 	POLICYDB_CAPABILITY_ALWAYSNETWORK,
 	__POLICYDB_CAPABILITY_MAX
 };
@@ -77,6 +77,7 @@  enum {
 
 extern int selinux_policycap_netpeer;
 extern int selinux_policycap_openperm;
+extern int selinux_policycap_extsockclass;
 extern int selinux_policycap_alwaysnetwork;
 
 /*
diff --git a/security/selinux/selinuxfs.c b/security/selinux/selinuxfs.c
index cf9293e..0aac402 100644
--- a/security/selinux/selinuxfs.c
+++ b/security/selinux/selinuxfs.c
@@ -45,7 +45,7 @@ 
 static char *policycap_names[] = {
 	"network_peer_controls",
 	"open_perms",
-	"redhat1",
+	"extended_socket_class",
 	"always_check_network"
 };
 
diff --git a/security/selinux/ss/services.c b/security/selinux/ss/services.c
index 082b20c..a70fcee 100644
--- a/security/selinux/ss/services.c
+++ b/security/selinux/ss/services.c
@@ -72,6 +72,7 @@ 
 
 int selinux_policycap_netpeer;
 int selinux_policycap_openperm;
+int selinux_policycap_extsockclass;
 int selinux_policycap_alwaysnetwork;
 
 static DEFINE_RWLOCK(policy_rwlock);
@@ -1988,6 +1989,8 @@  static void security_load_policycaps(void)
 						  POLICYDB_CAPABILITY_NETPEER);
 	selinux_policycap_openperm = ebitmap_get_bit(&policydb.policycaps,
 						  POLICYDB_CAPABILITY_OPENPERM);
+	selinux_policycap_extsockclass = ebitmap_get_bit(&policydb.policycaps,
+					  POLICYDB_CAPABILITY_EXTSOCKCLASS);
 	selinux_policycap_alwaysnetwork = ebitmap_get_bit(&policydb.policycaps,
 						  POLICYDB_CAPABILITY_ALWAYSNETWORK);
 }