// SPDX-License-Identifier: GPL-2.0-only
|
/*
|
* Simplified MAC Kernel (smack) security module
|
*
|
* This file contains the Smack netfilter implementation
|
*
|
* Author:
|
* Casey Schaufler <casey@schaufler-ca.com>
|
*
|
* Copyright (C) 2014 Casey Schaufler <casey@schaufler-ca.com>
|
* Copyright (C) 2014 Intel Corporation.
|
*/
|
|
#include <linux/netfilter_ipv4.h>
|
#include <linux/netfilter_ipv6.h>
|
#include <linux/netdevice.h>
|
#include <net/inet_sock.h>
|
#include <net/net_namespace.h>
|
#include "smack.h"
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
static unsigned int smack_ipv6_output(void *priv,
|
struct sk_buff *skb,
|
const struct nf_hook_state *state)
|
{
|
struct sock *sk = skb_to_full_sk(skb);
|
struct socket_smack *ssp;
|
struct smack_known *skp;
|
|
if (sk && sk->sk_security) {
|
ssp = sk->sk_security;
|
skp = ssp->smk_out;
|
skb->secmark = skp->smk_secid;
|
}
|
|
return NF_ACCEPT;
|
}
|
#endif /* IPV6 */
|
|
static unsigned int smack_ipv4_output(void *priv,
|
struct sk_buff *skb,
|
const struct nf_hook_state *state)
|
{
|
struct sock *sk = skb_to_full_sk(skb);
|
struct socket_smack *ssp;
|
struct smack_known *skp;
|
|
if (sk && sk->sk_security) {
|
ssp = sk->sk_security;
|
skp = ssp->smk_out;
|
skb->secmark = skp->smk_secid;
|
}
|
|
return NF_ACCEPT;
|
}
|
|
static const struct nf_hook_ops smack_nf_ops[] = {
|
{
|
.hook = smack_ipv4_output,
|
.pf = NFPROTO_IPV4,
|
.hooknum = NF_INET_LOCAL_OUT,
|
.priority = NF_IP_PRI_SELINUX_FIRST,
|
},
|
#if IS_ENABLED(CONFIG_IPV6)
|
{
|
.hook = smack_ipv6_output,
|
.pf = NFPROTO_IPV6,
|
.hooknum = NF_INET_LOCAL_OUT,
|
.priority = NF_IP6_PRI_SELINUX_FIRST,
|
},
|
#endif /* IPV6 */
|
};
|
|
static int __net_init smack_nf_register(struct net *net)
|
{
|
return nf_register_net_hooks(net, smack_nf_ops,
|
ARRAY_SIZE(smack_nf_ops));
|
}
|
|
static void __net_exit smack_nf_unregister(struct net *net)
|
{
|
nf_unregister_net_hooks(net, smack_nf_ops, ARRAY_SIZE(smack_nf_ops));
|
}
|
|
static struct pernet_operations smack_net_ops = {
|
.init = smack_nf_register,
|
.exit = smack_nf_unregister,
|
};
|
|
static int __init smack_nf_ip_init(void)
|
{
|
if (smack_enabled == 0)
|
return 0;
|
|
printk(KERN_DEBUG "Smack: Registering netfilter hooks\n");
|
return register_pernet_subsys(&smack_net_ops);
|
}
|
|
__initcall(smack_nf_ip_init);
|