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