xref: /openbmc/linux/security/selinux/netlink.c (revision d0034a7a4ac7fae708146ac0059b9c47a1543f0d)
1d2912cb1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
21da177e4SLinus Torvalds /*
31da177e4SLinus Torvalds  * Netlink event notifications for SELinux.
41da177e4SLinus Torvalds  *
51da177e4SLinus Torvalds  * Author: James Morris <jmorris@redhat.com>
61da177e4SLinus Torvalds  *
71da177e4SLinus Torvalds  * Copyright (C) 2004 Red Hat, Inc., James Morris <jmorris@redhat.com>
81da177e4SLinus Torvalds  */
91da177e4SLinus Torvalds #include <linux/init.h>
101da177e4SLinus Torvalds #include <linux/types.h>
115a0e3ad6STejun Heo #include <linux/slab.h>
121da177e4SLinus Torvalds #include <linux/stddef.h>
131da177e4SLinus Torvalds #include <linux/kernel.h>
1444fc7ea0SPaul Gortmaker #include <linux/export.h>
151da177e4SLinus Torvalds #include <linux/skbuff.h>
161da177e4SLinus Torvalds #include <linux/selinux_netlink.h>
17b4b51029SEric W. Biederman #include <net/net_namespace.h>
1801f534d0SDavid S. Miller #include <net/netlink.h>
191da177e4SLinus Torvalds 
206a3fbe81SJames Morris #include "security.h"
216a3fbe81SJames Morris 
22*cd2bb4cbSOndrej Mosnacek static struct sock *selnl __ro_after_init;
231da177e4SLinus Torvalds 
selnl_msglen(int msgtype)241da177e4SLinus Torvalds static int selnl_msglen(int msgtype)
251da177e4SLinus Torvalds {
261da177e4SLinus Torvalds 	int ret = 0;
271da177e4SLinus Torvalds 
281da177e4SLinus Torvalds 	switch (msgtype) {
291da177e4SLinus Torvalds 	case SELNL_MSG_SETENFORCE:
301da177e4SLinus Torvalds 		ret = sizeof(struct selnl_msg_setenforce);
311da177e4SLinus Torvalds 		break;
321da177e4SLinus Torvalds 
331da177e4SLinus Torvalds 	case SELNL_MSG_POLICYLOAD:
341da177e4SLinus Torvalds 		ret = sizeof(struct selnl_msg_policyload);
351da177e4SLinus Torvalds 		break;
361da177e4SLinus Torvalds 
371da177e4SLinus Torvalds 	default:
381da177e4SLinus Torvalds 		BUG();
391da177e4SLinus Torvalds 	}
401da177e4SLinus Torvalds 	return ret;
411da177e4SLinus Torvalds }
421da177e4SLinus Torvalds 
selnl_add_payload(struct nlmsghdr * nlh,int len,int msgtype,void * data)431da177e4SLinus Torvalds static void selnl_add_payload(struct nlmsghdr *nlh, int len, int msgtype, void *data)
441da177e4SLinus Torvalds {
451da177e4SLinus Torvalds 	switch (msgtype) {
461da177e4SLinus Torvalds 	case SELNL_MSG_SETENFORCE: {
4701f534d0SDavid S. Miller 		struct selnl_msg_setenforce *msg = nlmsg_data(nlh);
481da177e4SLinus Torvalds 
491da177e4SLinus Torvalds 		memset(msg, 0, len);
501da177e4SLinus Torvalds 		msg->val = *((int *)data);
511da177e4SLinus Torvalds 		break;
521da177e4SLinus Torvalds 	}
531da177e4SLinus Torvalds 
541da177e4SLinus Torvalds 	case SELNL_MSG_POLICYLOAD: {
5501f534d0SDavid S. Miller 		struct selnl_msg_policyload *msg = nlmsg_data(nlh);
561da177e4SLinus Torvalds 
571da177e4SLinus Torvalds 		memset(msg, 0, len);
581da177e4SLinus Torvalds 		msg->seqno = *((u32 *)data);
591da177e4SLinus Torvalds 		break;
601da177e4SLinus Torvalds 	}
611da177e4SLinus Torvalds 
621da177e4SLinus Torvalds 	default:
631da177e4SLinus Torvalds 		BUG();
641da177e4SLinus Torvalds 	}
651da177e4SLinus Torvalds }
661da177e4SLinus Torvalds 
selnl_notify(int msgtype,void * data)671da177e4SLinus Torvalds static void selnl_notify(int msgtype, void *data)
681da177e4SLinus Torvalds {
691da177e4SLinus Torvalds 	int len;
7027a884dcSArnaldo Carvalho de Melo 	sk_buff_data_t tmp;
711da177e4SLinus Torvalds 	struct sk_buff *skb;
721da177e4SLinus Torvalds 	struct nlmsghdr *nlh;
731da177e4SLinus Torvalds 
741da177e4SLinus Torvalds 	len = selnl_msglen(msgtype);
751da177e4SLinus Torvalds 
7677954983SHong zhi guo 	skb = nlmsg_new(len, GFP_USER);
771da177e4SLinus Torvalds 	if (!skb)
781da177e4SLinus Torvalds 		goto oom;
791da177e4SLinus Torvalds 
801da177e4SLinus Torvalds 	tmp = skb->tail;
8101f534d0SDavid S. Miller 	nlh = nlmsg_put(skb, 0, 0, msgtype, len, 0);
8201f534d0SDavid S. Miller 	if (!nlh)
8301f534d0SDavid S. Miller 		goto out_kfree_skb;
841da177e4SLinus Torvalds 	selnl_add_payload(nlh, len, msgtype, data);
851da177e4SLinus Torvalds 	nlh->nlmsg_len = skb->tail - tmp;
86ac6d439dSPatrick McHardy 	NETLINK_CB(skb).dst_group = SELNLGRP_AVC;
87ac6d439dSPatrick McHardy 	netlink_broadcast(selnl, skb, 0, SELNLGRP_AVC, GFP_USER);
881da177e4SLinus Torvalds out:
891da177e4SLinus Torvalds 	return;
901da177e4SLinus Torvalds 
9101f534d0SDavid S. Miller out_kfree_skb:
921da177e4SLinus Torvalds 	kfree_skb(skb);
931da177e4SLinus Torvalds oom:
94d85a7833Speter enderborg 	pr_err("SELinux:  OOM in %s\n", __func__);
951da177e4SLinus Torvalds 	goto out;
961da177e4SLinus Torvalds }
971da177e4SLinus Torvalds 
selnl_notify_setenforce(int val)981da177e4SLinus Torvalds void selnl_notify_setenforce(int val)
991da177e4SLinus Torvalds {
1001da177e4SLinus Torvalds 	selnl_notify(SELNL_MSG_SETENFORCE, &val);
1011da177e4SLinus Torvalds }
1021da177e4SLinus Torvalds 
selnl_notify_policyload(u32 seqno)1031da177e4SLinus Torvalds void selnl_notify_policyload(u32 seqno)
1041da177e4SLinus Torvalds {
1051da177e4SLinus Torvalds 	selnl_notify(SELNL_MSG_POLICYLOAD, &seqno);
1061da177e4SLinus Torvalds }
1071da177e4SLinus Torvalds 
selnl_init(void)1081da177e4SLinus Torvalds static int __init selnl_init(void)
1091da177e4SLinus Torvalds {
110a31f2d17SPablo Neira Ayuso 	struct netlink_kernel_cfg cfg = {
111a31f2d17SPablo Neira Ayuso 		.groups	= SELNLGRP_MAX,
1129785e10aSPablo Neira Ayuso 		.flags	= NL_CFG_F_NONROOT_RECV,
113a31f2d17SPablo Neira Ayuso 	};
114a31f2d17SPablo Neira Ayuso 
1159f00d977SPablo Neira Ayuso 	selnl = netlink_kernel_create(&init_net, NETLINK_SELINUX, &cfg);
1161da177e4SLinus Torvalds 	if (selnl == NULL)
1171da177e4SLinus Torvalds 		panic("SELinux:  Cannot create netlink socket.");
1181da177e4SLinus Torvalds 	return 0;
1191da177e4SLinus Torvalds }
1201da177e4SLinus Torvalds 
1211da177e4SLinus Torvalds __initcall(selnl_init);
122