Changes in 4.19.323 staging: iio: frequency: ad9833: Get frequency value statically staging: iio: frequency: ad9833: Load clock using clock framework staging: iio: frequency: ad9834: Validate frequency parameter value usbnet: ipheth: fix carrier detection in modes 1 and 4 net: ethernet: use ip_hdrlen() instead of bit shift net: phy: vitesse: repair vsc73xx autonegotiation scripts: kconfig: merge_config: config files: add a trailing newline arm64: dts: rockchip: override BIOS_DISABLE signal via GPIO hog on RK3399 Puma net/mlx5: Update the list of the PCI supported devices net: ftgmac100: Enable TX interrupt to avoid TX timeout net: dpaa: Pad packets to ETH_ZLEN soundwire: stream: Revert "soundwire: stream: fix programming slave ports for non-continous port maps" selftests/vm: remove call to ksft_set_plan() selftests/kcmp: remove call to ksft_set_plan() ASoC: allow module autoloading for table db1200_pids pinctrl: at91: make it work with current gpiolib microblaze: don't treat zero reserved memory regions as error net: ftgmac100: Ensure tx descriptor updates are visible wifi: iwlwifi: mvm: fix iwl_mvm_max_scan_ie_fw_cmd_room() wifi: iwlwifi: mvm: don't wait for tx queues if firmware is dead ASoC: tda7419: fix module autoloading spi: bcm63xx: Enable module autoloading x86/hyperv: Set X86_FEATURE_TSC_KNOWN_FREQ when Hyper-V provides frequency ocfs2: add bounds checking to ocfs2_xattr_find_entry() ocfs2: strict bound check before memcmp in ocfs2_xattr_find_entry() gpio: prevent potential speculation leaks in gpio_device_get_desc() USB: serial: pl2303: add device id for Macrosilicon MS3020 ACPI: PMIC: Remove unneeded check in tps68470_pmic_opregion_probe() wifi: ath9k: fix parameter check in ath9k_init_debug() wifi: ath9k: Remove error checks when creating debugfs entries netfilter: nf_tables: elements with timeout below CONFIG_HZ never expire wifi: cfg80211: fix UBSAN noise in cfg80211_wext_siwscan() wifi: cfg80211: fix two more possible UBSAN-detected off-by-one errors wifi: mac80211: use two-phase skb reclamation in ieee80211_do_stop() can: bcm: Clear bo->bcm_proc_read after remove_proc_entry(). Bluetooth: btusb: Fix not handling ZPL/short-transfer block, bfq: fix possible UAF for bfqq->bic with merge chain block, bfq: choose the last bfqq from merge chain in bfq_setup_cooperator() block, bfq: don't break merge chain in bfq_split_bfqq() spi: ppc4xx: handle irq_of_parse_and_map() errors spi: ppc4xx: Avoid returning 0 when failed to parse and map IRQ ARM: versatile: fix OF node leak in CPUs prepare reset: berlin: fix OF node leak in probe() error path clocksource/drivers/qcom: Add missing iounmap() on errors in msm_dt_timer_init() hwmon: (max16065) Fix overflows seen when writing limits mtd: slram: insert break after errors in parsing the map hwmon: (ntc_thermistor) fix module autoloading power: supply: max17042_battery: Fix SOC threshold calc w/ no current sense fbdev: hpfb: Fix an error handling path in hpfb_dio_probe() drm/stm: Fix an error handling path in stm_drm_platform_probe() drm/amd: fix typo drm/amdgpu: Replace one-element array with flexible-array member drm/amdgpu: properly handle vbios fake edid sizing drm/radeon: Replace one-element array with flexible-array member drm/radeon: properly handle vbios fake edid sizing drm/rockchip: vop: Allow 4096px width scaling drm/radeon/evergreen_cs: fix int overflow errors in cs track offsets jfs: fix out-of-bounds in dbNextAG() and diAlloc() drm/msm/a5xx: properly clear preemption records on resume drm/msm/a5xx: fix races in preemption evaluation stage ipmi: docs: don't advertise deprecated sysfs entries drm/msm: fix %s null argument error xen: use correct end address of kernel for conflict checking xen/swiotlb: simplify range_straddles_page_boundary() xen/swiotlb: add alignment check for dma buffers selftests/bpf: Fix error compiling test_lru_map.c xz: cleanup CRC32 edits from 2018 kthread: add kthread_work tracepoints kthread: fix task state in kthread worker if being frozen jbd2: introduce/export functions jbd2_journal_submit|finish_inode_data_buffers() ext4: clear EXT4_GROUP_INFO_WAS_TRIMMED_BIT even mount with discard smackfs: Use rcu_assign_pointer() to ensure safe assignment in smk_set_cipso ext4: avoid negative min_clusters in find_group_orlov() ext4: return error on ext4_find_inline_entry ext4: avoid OOB when system.data xattr changes underneath the filesystem nilfs2: fix potential null-ptr-deref in nilfs_btree_insert() nilfs2: determine empty node blocks as corrupted nilfs2: fix potential oob read in nilfs_btree_check_delete() perf sched timehist: Fix missing free of session in perf_sched__timehist() perf sched timehist: Fixed timestamp error when unable to confirm event sched_in time perf time-utils: Fix 32-bit nsec parsing clk: rockchip: Set parent rate for DCLK_VOP clock on RK3228 drivers: media: dvb-frontends/rtl2832: fix an out-of-bounds write error drivers: media: dvb-frontends/rtl2830: fix an out-of-bounds write error PCI: xilinx-nwl: Fix register misspelling RDMA/iwcm: Fix WARNING:at_kernel/workqueue.c:#check_flush_dependency pinctrl: single: fix missing error code in pcs_probe() clk: ti: dra7-atl: Fix leak of of_nodes pinctrl: mvebu: Fix devinit_dove_pinctrl_probe function RDMA/cxgb4: Added NULL check for lookup_atid ntb: intel: Fix the NULL vs IS_ERR() bug for debugfs_create_dir() nfsd: call cache_put if xdr_reserve_space returns NULL f2fs: enhance to update i_mode and acl atomically in f2fs_setattr() f2fs: fix typo f2fs: fix to update i_ctime in __f2fs_setxattr() f2fs: remove unneeded check condition in __f2fs_setxattr() f2fs: reduce expensive checkpoint trigger frequency coresight: tmc: sg: Do not leak sg_table netfilter: nf_reject_ipv6: fix nf_reject_ip6_tcphdr_put() net: seeq: Fix use after free vulnerability in ether3 Driver Due to Race Condition tcp: introduce tcp_skb_timestamp_us() helper tcp: check skb is non-NULL in tcp_rto_delta_us() net: qrtr: Update packets cloning when broadcasting netfilter: ctnetlink: compile ctnetlink_label_size with CONFIG_NF_CONNTRACK_EVENTS crypto: aead,cipher - zeroize key buffer after use Remove *.orig pattern from .gitignore soc: versatile: integrator: fix OF node leak in probe() error path USB: appledisplay: close race between probe and completion handler USB: misc: cypress_cy7c63: check for short transfer firmware_loader: Block path traversal tty: rp2: Fix reset with non forgiving PCIe host bridges drbd: Fix atomicity violation in drbd_uuid_set_bm() drbd: Add NULL check for net_conf to prevent dereference in state validation ACPI: sysfs: validate return type of _STR method f2fs: prevent possible int overflow in dir_block_index() f2fs: avoid potential int overflow in sanity_check_area_boundary() vfs: fix race between evice_inodes() and find_inode()&iput() fs: Fix file_set_fowner LSM hook inconsistencies nfs: fix memory leak in error path of nfs4_do_reclaim PCI: xilinx-nwl: Use irq_data_get_irq_chip_data() PCI: xilinx-nwl: Fix off-by-one in INTx IRQ handler soc: versatile: realview: fix memory leak during device remove soc: versatile: realview: fix soc_dev leak during device remove usb: yurex: Replace snprintf() with the safer scnprintf() variant USB: misc: yurex: fix race between read and write pps: remove usage of the deprecated ida_simple_xx() API pps: add an error check in parport_attach i2c: aspeed: Update the stop sw state when the bus recovery occurs i2c: isch: Add missed 'else' usb: yurex: Fix inconsistent locking bug in yurex_read() mailbox: rockchip: fix a typo in module autoloading mailbox: bcm2835: Fix timeout during suspend mode ceph: remove the incorrect Fw reference check when dirtying pages netfilter: uapi: NFTA_FLOWTABLE_HOOK is NLA_NESTED netfilter: nf_tables: prevent nf_skb_duplicated corruption r8152: Factor out OOB link list waits net: ethernet: lantiq_etop: fix memory disclosure net: avoid potential underflow in qdisc_pkt_len_init() with UFO net: add more sanity checks to qdisc_pkt_len_init() ipv4: ip_gre: Fix drops of small packets in ipgre_xmit sctp: set sk_state back to CLOSED if autobind fails in sctp_listen_start ALSA: hda/generic: Unconditionally prefer preferred_dacs pairs ALSA: hda/conexant: Fix conflicting quirk for System76 Pangolin f2fs: Require FMODE_WRITE for atomic write ioctls wifi: ath9k: fix possible integer overflow in ath9k_get_et_stats() wifi: ath9k_htc: Use __skb_set_length() for resetting urb before resubmit net: hisilicon: hip04: fix OF node leak in probe() net: hisilicon: hns_dsaf_mac: fix OF node leak in hns_mac_get_info() net: hisilicon: hns_mdio: fix OF node leak in probe() ACPICA: Fix memory leak if acpi_ps_get_next_namepath() fails ACPICA: Fix memory leak if acpi_ps_get_next_field() fails ACPI: EC: Do not release locks during operation region accesses ACPICA: check null return of ACPI_ALLOCATE_ZEROED() in acpi_db_convert_to_package() tipc: guard against string buffer overrun net: mvpp2: Increase size of queue_name buffer ipv4: Check !in_dev earlier for ioctl(SIOCSIFADDR). ipv4: Mask upper DSCP bits and ECN bits in NETLINK_FIB_LOOKUP family tcp: avoid reusing FIN_WAIT2 when trying to find port in connect() process ACPICA: iasl: handle empty connection_node wifi: mwifiex: Fix memcpy() field-spanning write warning in mwifiex_cmd_802_11_scan_ext() signal: Replace BUG_ON()s ALSA: asihpi: Fix potential OOB array access ALSA: hdsp: Break infinite MIDI input flush loop fbdev: pxafb: Fix possible use after free in pxafb_task() power: reset: brcmstb: Do not go into infinite loop if reset fails ata: sata_sil: Rename sil_blacklist to sil_quirks jfs: UBSAN: shift-out-of-bounds in dbFindBits jfs: Fix uaf in dbFreeBits jfs: check if leafidx greater than num leaves per dmap tree jfs: Fix uninit-value access of new_ea in ea_buffer drm/amd/display: Check stream before comparing them drm/amd/display: Fix index out of bounds in degamma hardware format translation drm/printer: Allow NULL data in devcoredump printer scsi: aacraid: Rearrange order of struct aac_srb_unit drm/radeon/r100: Handle unknown family in r100_cp_init_microcode() of/irq: Refer to actual buffer size in of_irq_parse_one() ext4: ext4_search_dir should return a proper error ext4: fix i_data_sem unlock order in ext4_ind_migrate() spi: s3c64xx: fix timeout counters in flush_fifo selftests: breakpoints: use remaining time to check if suspend succeed selftests: vDSO: fix vDSO symbols lookup for powerpc64 i2c: xiic: Wait for TX empty to avoid missed TX NAKs spi: bcm63xx: Fix module autoloading perf/core: Fix small negative period being ignored parisc: Fix itlb miss handler for 64-bit programs ALSA: core: add isascii() check to card ID generator ext4: no need to continue when the number of entries is 1 ext4: propagate errors from ext4_find_extent() in ext4_insert_range() ext4: fix incorrect tid assumption in __jbd2_log_wait_for_space() ext4: aovid use-after-free in ext4_ext_insert_extent() ext4: fix double brelse() the buffer of the extents path ext4: fix incorrect tid assumption in ext4_wait_for_tail_page_commit() parisc: Fix 64-bit userspace syscall path of/irq: Support #msi-cells=<0> in of_msi_get_domain jbd2: stop waiting for space when jbd2_cleanup_journal_tail() returns error ocfs2: fix the la space leak when unmounting an ocfs2 volume ocfs2: fix uninit-value in ocfs2_get_block() ocfs2: reserve space for inline xattr before attaching reflink tree ocfs2: cancel dqi_sync_work before freeing oinfo ocfs2: remove unreasonable unlock in ocfs2_read_blocks ocfs2: fix null-ptr-deref when journal load failed. ocfs2: fix possible null-ptr-deref in ocfs2_set_buffer_uptodate riscv: define ILLEGAL_POINTER_VALUE for 64bit aoe: fix the potential use-after-free problem in more places clk: rockchip: fix error for unknown clocks media: uapi/linux/cec.h: cec_msg_set_reply_to: zero flags media: venus: fix use after free bug in venus_remove due to race condition iio: magnetometer: ak8975: Fix reading for ak099xx sensors tomoyo: fallback to realpath if symlink's pathname does not exist Input: adp5589-keys - fix adp5589_gpio_get_value() btrfs: wait for fixup workers before stopping cleaner kthread during umount gpio: davinci: fix lazy disable ext4: avoid ext4_error()'s caused by ENOMEM in the truncate path ext4: fix slab-use-after-free in ext4_split_extent_at() ext4: update orig_path in ext4_find_extent() arm64: Add Cortex-715 CPU part definition arm64: cputype: Add Neoverse-N3 definitions arm64: errata: Expand speculative SSBS workaround once more uprobes: fix kernel info leak via "[uprobes]" vma nfsd: use ktime_get_seconds() for timestamps nfsd: fix delegation_blocked() to block correctly for at least 30 seconds rtc: at91sam9: drop platform_data support rtc: at91sam9: fix OF node leak in probe() error path ACPI: battery: Simplify battery hook locking ACPI: battery: Fix possible crash when unregistering a battery hook ext4: fix inode tree inconsistency caused by ENOMEM net: ethernet: cortina: Drop TSO support tracing: Remove precision vsnprintf() check from print event drm: Move drm_mode_setcrtc() local re-init to failure path drm/crtc: fix uninitialized variable use even harder virtio_console: fix misc probe bugs Input: synaptics-rmi4 - fix UAF of IRQ domain on driver removal bpf: Check percpu map value size first s390/facility: Disable compile time optimization for decompressor code s390/mm: Add cond_resched() to cmm_alloc/free_pages() ext4: nested locking for xattr inode s390/cpum_sf: Remove WARN_ON_ONCE statements ktest.pl: Avoid false positives with grub2 skip regex clk: bcm: bcm53573: fix OF node leak in init i2c: i801: Use a different adapter-name for IDF adapters PCI: Mark Creative Labs EMU20k2 INTx masking as broken media: videobuf2-core: clear memory related fields in __vb2_plane_dmabuf_put() usb: chipidea: udc: enable suspend interrupt after usb reset tools/iio: Add memory allocation failure check for trigger_name driver core: bus: Return -EIO instead of 0 when show/store invalid bus attribute fbdev: sisfb: Fix strbuf array overflow NFS: Remove print_overflow_msg() SUNRPC: Fix integer overflow in decode_rc_list() tcp: fix tcp_enter_recovery() to zero retrans_stamp when it's safe netfilter: br_netfilter: fix panic with metadata_dst skb Bluetooth: RFCOMM: FIX possible deadlock in rfcomm_sk_state_change gpio: aspeed: Add the flush write to ensure the write complete. clk: Add (devm_)clk_get_optional() functions clk: generalize devm_clk_get() a bit clk: Provide new devm_clk helpers for prepared and enabled clocks gpio: aspeed: Use devm_clk api to manage clock source igb: Do not bring the device up after non-fatal error net: ibm: emac: mal: fix wrong goto ppp: fix ppp_async_encode() illegal access net: ipv6: ensure we call ipv6_mc_down() at most once CDC-NCM: avoid overflow in sanity checking HID: plantronics: Workaround for an unexcepted opposite volume key Revert "usb: yurex: Replace snprintf() with the safer scnprintf() variant" usb: xhci: Fix problem with xhci resume from suspend usb: storage: ignore bogus device raised by JieLi BR21 USB sound chip net: Fix an unsafe loop on the list posix-clock: Fix missing timespec64 check in pc_clock_settime() arm64: probes: Remove broken LDR (literal) uprobe support arm64: probes: Fix simulate_ldr*_literal() PCI: Add function 0 DMA alias quirk for Glenfly Arise chip fat: fix uninitialized variable KVM: Fix a data race on last_boosted_vcpu in kvm_vcpu_on_spin() net: dsa: mv88e6xxx: Fix out-of-bound access s390/sclp_vt220: Convert newlines to CRLF instead of LFCR KVM: s390: Change virtual to physical address access in diag 0x258 handler x86/cpufeatures: Define X86_FEATURE_AMD_IBPB_RET drm/vmwgfx: Handle surface check failure correctly iio: dac: stm32-dac-core: add missing select REGMAP_MMIO in Kconfig iio: adc: ti-ads8688: add missing select IIO_(TRIGGERED_)BUFFER in Kconfig iio: hid-sensors: Fix an error handling path in _hid_sensor_set_report_latency() iio: light: opt3001: add missing full-scale range value Bluetooth: Remove debugfs directory on module init failure Bluetooth: btusb: Fix regression with fake CSR controllers 0a12:0001 xhci: Fix incorrect stream context type macro USB: serial: option: add support for Quectel EG916Q-GL USB: serial: option: add Telit FN920C04 MBIM compositions parport: Proper fix for array out-of-bounds access x86/apic: Always explicitly disarm TSC-deadline timer nilfs2: propagate directory read errors from nilfs_find_entry() clk: Fix pointer casting to prevent oops in devm_clk_release() clk: Fix slab-out-of-bounds error in devm_clk_release() RDMA/bnxt_re: Fix incorrect AVID type in WQE structure RDMA/cxgb4: Fix RDMA_CM_EVENT_UNREACHABLE error for iWARP RDMA/bnxt_re: Return more meaningful error drm/msm/dsi: fix 32-bit signed integer extension in pclk_rate calculation macsec: don't increment counters for an unrelated SA net: ethernet: aeroflex: fix potential memory leak in greth_start_xmit_gbit() net: systemport: fix potential memory leak in bcm_sysport_xmit() usb: typec: altmode should keep reference to parent Bluetooth: bnep: fix wild-memory-access in proto_unregister arm64:uprobe fix the uprobe SWBP_INSN in big-endian arm64: probes: Fix uprobes for big-endian kernels KVM: s390: gaccess: Refactor gpa and length calculation KVM: s390: gaccess: Refactor access address range check KVM: s390: gaccess: Cleanup access to guest pages KVM: s390: gaccess: Check if guest address is in memslot udf: fix uninit-value use in udf_get_fileshortad jfs: Fix sanity check in dbMount net/sun3_82586: fix potential memory leak in sun3_82586_send_packet() be2net: fix potential memory leak in be_xmit() net: usb: usbnet: fix name regression posix-clock: posix-clock: Fix unbalanced locking in pc_clock_settime() ALSA: hda/realtek: Update default depop procedure drm/amd: Guard against bad data for ATIF ACPI method ACPI: button: Add DMI quirk for Samsung Galaxy Book2 to fix initial lid detection issue nilfs2: fix kernel bug due to missing clearing of buffer delay flag hv_netvsc: Fix VF namespace also in synthetic NIC NETDEV_REGISTER event selinux: improve error checking in sel_write_load() arm64/uprobes: change the uprobe_opcode_t typedef to fix the sparse warning xfrm: validate new SA's prefixlen using SA family when sel.family is unset usb: dwc3: remove generic PHY calibrate() calls usb: dwc3: Add splitdisable quirk for Hisilicon Kirin Soc usb: dwc3: core: Stop processing of pending events if controller is halted cgroup: Fix potential overflow issue when checking max_depth wifi: mac80211: skip non-uploaded keys in ieee80211_iter_keys gtp: simplify error handling code in 'gtp_encap_enable()' gtp: allow -1 to be specified as file description from userspace net/sched: stop qdisc_tree_reduce_backlog on TC_H_ROOT bpf: Fix out-of-bounds write in trie_get_next_key() net: support ip generic csum processing in skb_csum_hwoffload_help net: skip offload for NETIF_F_IPV6_CSUM if ipv6 header contains extension netfilter: nft_payload: sanitize offset and length before calling skb_checksum() firmware: arm_sdei: Fix the input parameter of cpuhp_remove_state() net: amd: mvme147: Fix probe banner message misc: sgi-gru: Don't disable preemption in GRU driver usbip: tools: Fix detach_port() invalid port error path usb: phy: Fix API devm_usb_put_phy() can not release the phy xhci: Fix Link TRB DMA in command ring stopped completion event Revert "driver core: Fix uevent_show() vs driver detach race" wifi: mac80211: do not pass a stopped vif to the driver in .get_txpower wifi: ath10k: Fix memory leak in management tx wifi: iwlegacy: Clear stale interrupts before resuming device nilfs2: fix potential deadlock with newly created symlinks ocfs2: pass u64 to ocfs2_truncate_inline maybe overflow nilfs2: fix kernel bug due to missing clearing of checked flag mm: shmem: fix data-race in shmem_getattr() vt: prevent kernel-infoleak in con_font_get() Linux 4.19.323 Change-Id: I2348f834187153067ab46b3b48b8fe7da9cee1f1 Signed-off-by: Greg Kroah-Hartman <gregkh@google.com>
2151 lines
50 KiB
C
2151 lines
50 KiB
C
/* Updated: Karl MacMillan <kmacmillan@tresys.com>
|
|
*
|
|
* Added conditional policy language extensions
|
|
*
|
|
* Updated: Hewlett-Packard <paul@paul-moore.com>
|
|
*
|
|
* Added support for the policy capability bitmap
|
|
*
|
|
* Copyright (C) 2007 Hewlett-Packard Development Company, L.P.
|
|
* Copyright (C) 2003 - 2004 Tresys Technology, LLC
|
|
* Copyright (C) 2004 Red Hat, Inc., James Morris <jmorris@redhat.com>
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, version 2.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/pagemap.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/vmalloc.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/mount.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/init.h>
|
|
#include <linux/string.h>
|
|
#include <linux/security.h>
|
|
#include <linux/major.h>
|
|
#include <linux/seq_file.h>
|
|
#include <linux/percpu.h>
|
|
#include <linux/audit.h>
|
|
#include <linux/uaccess.h>
|
|
#include <linux/kobject.h>
|
|
#include <linux/ctype.h>
|
|
|
|
/* selinuxfs pseudo filesystem for exporting the security policy API.
|
|
Based on the proc code and the fs/nfsd/nfsctl.c code. */
|
|
|
|
#include "flask.h"
|
|
#include "avc.h"
|
|
#include "avc_ss.h"
|
|
#include "security.h"
|
|
#include "objsec.h"
|
|
#include "conditional.h"
|
|
|
|
enum sel_inos {
|
|
SEL_ROOT_INO = 2,
|
|
SEL_LOAD, /* load policy */
|
|
SEL_ENFORCE, /* get or set enforcing status */
|
|
SEL_CONTEXT, /* validate context */
|
|
SEL_ACCESS, /* compute access decision */
|
|
SEL_CREATE, /* compute create labeling decision */
|
|
SEL_RELABEL, /* compute relabeling decision */
|
|
SEL_USER, /* compute reachable user contexts */
|
|
SEL_POLICYVERS, /* return policy version for this kernel */
|
|
SEL_COMMIT_BOOLS, /* commit new boolean values */
|
|
SEL_MLS, /* return if MLS policy is enabled */
|
|
SEL_DISABLE, /* disable SELinux until next reboot */
|
|
SEL_MEMBER, /* compute polyinstantiation membership decision */
|
|
SEL_CHECKREQPROT, /* check requested protection, not kernel-applied one */
|
|
SEL_COMPAT_NET, /* whether to use old compat network packet controls */
|
|
SEL_REJECT_UNKNOWN, /* export unknown reject handling to userspace */
|
|
SEL_DENY_UNKNOWN, /* export unknown deny handling to userspace */
|
|
SEL_STATUS, /* export current status using mmap() */
|
|
SEL_POLICY, /* allow userspace to read the in kernel policy */
|
|
SEL_VALIDATE_TRANS, /* compute validatetrans decision */
|
|
SEL_INO_NEXT, /* The next inode number to use */
|
|
};
|
|
|
|
struct selinux_fs_info {
|
|
struct dentry *bool_dir;
|
|
unsigned int bool_num;
|
|
char **bool_pending_names;
|
|
unsigned int *bool_pending_values;
|
|
struct dentry *class_dir;
|
|
unsigned long last_class_ino;
|
|
bool policy_opened;
|
|
struct dentry *policycap_dir;
|
|
struct mutex mutex;
|
|
unsigned long last_ino;
|
|
struct selinux_state *state;
|
|
struct super_block *sb;
|
|
};
|
|
|
|
static int selinux_fs_info_create(struct super_block *sb)
|
|
{
|
|
struct selinux_fs_info *fsi;
|
|
|
|
fsi = kzalloc(sizeof(*fsi), GFP_KERNEL);
|
|
if (!fsi)
|
|
return -ENOMEM;
|
|
|
|
mutex_init(&fsi->mutex);
|
|
fsi->last_ino = SEL_INO_NEXT - 1;
|
|
fsi->state = &selinux_state;
|
|
fsi->sb = sb;
|
|
sb->s_fs_info = fsi;
|
|
return 0;
|
|
}
|
|
|
|
static void selinux_fs_info_free(struct super_block *sb)
|
|
{
|
|
struct selinux_fs_info *fsi = sb->s_fs_info;
|
|
int i;
|
|
|
|
if (fsi) {
|
|
for (i = 0; i < fsi->bool_num; i++)
|
|
kfree(fsi->bool_pending_names[i]);
|
|
kfree(fsi->bool_pending_names);
|
|
kfree(fsi->bool_pending_values);
|
|
}
|
|
kfree(sb->s_fs_info);
|
|
sb->s_fs_info = NULL;
|
|
}
|
|
|
|
#define SEL_INITCON_INO_OFFSET 0x01000000
|
|
#define SEL_BOOL_INO_OFFSET 0x02000000
|
|
#define SEL_CLASS_INO_OFFSET 0x04000000
|
|
#define SEL_POLICYCAP_INO_OFFSET 0x08000000
|
|
#define SEL_INO_MASK 0x00ffffff
|
|
|
|
#define TMPBUFLEN 12
|
|
static ssize_t sel_read_enforce(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%d",
|
|
enforcing_enabled(fsi->state));
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
#ifdef CONFIG_SECURITY_SELINUX_DEVELOP
|
|
static ssize_t sel_write_enforce(struct file *file, const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *page = NULL;
|
|
ssize_t length;
|
|
int old_value, new_value;
|
|
|
|
if (count >= PAGE_SIZE)
|
|
return -ENOMEM;
|
|
|
|
/* No partial writes. */
|
|
if (*ppos != 0)
|
|
return -EINVAL;
|
|
|
|
page = memdup_user_nul(buf, count);
|
|
if (IS_ERR(page))
|
|
return PTR_ERR(page);
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(page, "%d", &new_value) != 1)
|
|
goto out;
|
|
|
|
new_value = !!new_value;
|
|
|
|
old_value = enforcing_enabled(state);
|
|
if (new_value != old_value) {
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__SETENFORCE,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
audit_log(audit_context(), GFP_KERNEL, AUDIT_MAC_STATUS,
|
|
"enforcing=%d old_enforcing=%d auid=%u ses=%u"
|
|
" enabled=%d old-enabled=%d lsm=selinux res=1",
|
|
new_value, old_value,
|
|
from_kuid(&init_user_ns, audit_get_loginuid(current)),
|
|
audit_get_sessionid(current),
|
|
selinux_enabled, selinux_enabled);
|
|
enforcing_set(state, new_value);
|
|
if (new_value)
|
|
avc_ss_reset(state->avc, 0);
|
|
selnl_notify_setenforce(new_value);
|
|
selinux_status_update_setenforce(state, new_value);
|
|
if (!new_value)
|
|
call_lsm_notifier(LSM_POLICY_CHANGE, NULL);
|
|
}
|
|
length = count;
|
|
out:
|
|
kfree(page);
|
|
return length;
|
|
}
|
|
#else
|
|
#define sel_write_enforce NULL
|
|
#endif
|
|
|
|
static const struct file_operations sel_enforce_ops = {
|
|
.read = sel_read_enforce,
|
|
.write = sel_write_enforce,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_read_handle_unknown(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
ino_t ino = file_inode(filp)->i_ino;
|
|
int handle_unknown = (ino == SEL_REJECT_UNKNOWN) ?
|
|
security_get_reject_unknown(state) :
|
|
!security_get_allow_unknown(state);
|
|
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%d", handle_unknown);
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
static const struct file_operations sel_handle_unknown_ops = {
|
|
.read = sel_read_handle_unknown,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static int sel_open_handle_status(struct inode *inode, struct file *filp)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
struct page *status = selinux_kernel_status_page(fsi->state);
|
|
|
|
if (!status)
|
|
return -ENOMEM;
|
|
|
|
filp->private_data = status;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t sel_read_handle_status(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct page *status = filp->private_data;
|
|
|
|
BUG_ON(!status);
|
|
|
|
return simple_read_from_buffer(buf, count, ppos,
|
|
page_address(status),
|
|
sizeof(struct selinux_kernel_status));
|
|
}
|
|
|
|
static int sel_mmap_handle_status(struct file *filp,
|
|
struct vm_area_struct *vma)
|
|
{
|
|
struct page *status = filp->private_data;
|
|
unsigned long size = vma->vm_end - vma->vm_start;
|
|
|
|
BUG_ON(!status);
|
|
|
|
/* only allows one page from the head */
|
|
if (vma->vm_pgoff > 0 || size != PAGE_SIZE)
|
|
return -EIO;
|
|
/* disallow writable mapping */
|
|
if (vma->vm_flags & VM_WRITE)
|
|
return -EPERM;
|
|
/* disallow mprotect() turns it into writable */
|
|
vma->vm_flags &= ~VM_MAYWRITE;
|
|
|
|
return remap_pfn_range(vma, vma->vm_start,
|
|
page_to_pfn(status),
|
|
size, vma->vm_page_prot);
|
|
}
|
|
|
|
static const struct file_operations sel_handle_status_ops = {
|
|
.open = sel_open_handle_status,
|
|
.read = sel_read_handle_status,
|
|
.mmap = sel_mmap_handle_status,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
#ifdef CONFIG_SECURITY_SELINUX_DISABLE
|
|
static ssize_t sel_write_disable(struct file *file, const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
char *page;
|
|
ssize_t length;
|
|
int new_value;
|
|
int enforcing;
|
|
|
|
if (count >= PAGE_SIZE)
|
|
return -ENOMEM;
|
|
|
|
/* No partial writes. */
|
|
if (*ppos != 0)
|
|
return -EINVAL;
|
|
|
|
page = memdup_user_nul(buf, count);
|
|
if (IS_ERR(page))
|
|
return PTR_ERR(page);
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(page, "%d", &new_value) != 1)
|
|
goto out;
|
|
|
|
if (new_value) {
|
|
enforcing = enforcing_enabled(fsi->state);
|
|
length = selinux_disable(fsi->state);
|
|
if (length)
|
|
goto out;
|
|
audit_log(audit_context(), GFP_KERNEL, AUDIT_MAC_STATUS,
|
|
"enforcing=%d old_enforcing=%d auid=%u ses=%u"
|
|
" enabled=%d old-enabled=%d lsm=selinux res=1",
|
|
enforcing, enforcing,
|
|
from_kuid(&init_user_ns, audit_get_loginuid(current)),
|
|
audit_get_sessionid(current), 0, 1);
|
|
}
|
|
|
|
length = count;
|
|
out:
|
|
kfree(page);
|
|
return length;
|
|
}
|
|
#else
|
|
#define sel_write_disable NULL
|
|
#endif
|
|
|
|
static const struct file_operations sel_disable_ops = {
|
|
.write = sel_write_disable,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_read_policyvers(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%u", POLICYDB_VERSION_MAX);
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
static const struct file_operations sel_policyvers_ops = {
|
|
.read = sel_read_policyvers,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
/* declaration for sel_write_load */
|
|
static int sel_make_bools(struct selinux_fs_info *fsi);
|
|
static int sel_make_classes(struct selinux_fs_info *fsi);
|
|
static int sel_make_policycap(struct selinux_fs_info *fsi);
|
|
|
|
/* declaration for sel_make_class_dirs */
|
|
static struct dentry *sel_make_dir(struct dentry *dir, const char *name,
|
|
unsigned long *ino);
|
|
|
|
static ssize_t sel_read_mls(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%d",
|
|
security_mls_enabled(fsi->state));
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
static const struct file_operations sel_mls_ops = {
|
|
.read = sel_read_mls,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
struct policy_load_memory {
|
|
size_t len;
|
|
void *data;
|
|
};
|
|
|
|
static int sel_open_policy(struct inode *inode, struct file *filp)
|
|
{
|
|
struct selinux_fs_info *fsi = inode->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
struct policy_load_memory *plm = NULL;
|
|
int rc;
|
|
|
|
BUG_ON(filp->private_data);
|
|
|
|
mutex_lock(&fsi->mutex);
|
|
|
|
rc = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__READ_POLICY, NULL);
|
|
if (rc)
|
|
goto err;
|
|
|
|
rc = -EBUSY;
|
|
if (fsi->policy_opened)
|
|
goto err;
|
|
|
|
rc = -ENOMEM;
|
|
plm = kzalloc(sizeof(*plm), GFP_KERNEL);
|
|
if (!plm)
|
|
goto err;
|
|
|
|
if (i_size_read(inode) != security_policydb_len(state)) {
|
|
inode_lock(inode);
|
|
i_size_write(inode, security_policydb_len(state));
|
|
inode_unlock(inode);
|
|
}
|
|
|
|
rc = security_read_policy(state, &plm->data, &plm->len);
|
|
if (rc)
|
|
goto err;
|
|
|
|
fsi->policy_opened = 1;
|
|
|
|
filp->private_data = plm;
|
|
|
|
mutex_unlock(&fsi->mutex);
|
|
|
|
return 0;
|
|
err:
|
|
mutex_unlock(&fsi->mutex);
|
|
|
|
if (plm)
|
|
vfree(plm->data);
|
|
kfree(plm);
|
|
return rc;
|
|
}
|
|
|
|
static int sel_release_policy(struct inode *inode, struct file *filp)
|
|
{
|
|
struct selinux_fs_info *fsi = inode->i_sb->s_fs_info;
|
|
struct policy_load_memory *plm = filp->private_data;
|
|
|
|
BUG_ON(!plm);
|
|
|
|
fsi->policy_opened = 0;
|
|
|
|
vfree(plm->data);
|
|
kfree(plm);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t sel_read_policy(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct policy_load_memory *plm = filp->private_data;
|
|
int ret;
|
|
|
|
ret = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__READ_POLICY, NULL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return simple_read_from_buffer(buf, count, ppos, plm->data, plm->len);
|
|
}
|
|
|
|
static vm_fault_t sel_mmap_policy_fault(struct vm_fault *vmf)
|
|
{
|
|
struct policy_load_memory *plm = vmf->vma->vm_file->private_data;
|
|
unsigned long offset;
|
|
struct page *page;
|
|
|
|
if (vmf->flags & (FAULT_FLAG_MKWRITE | FAULT_FLAG_WRITE))
|
|
return VM_FAULT_SIGBUS;
|
|
|
|
offset = vmf->pgoff << PAGE_SHIFT;
|
|
if (offset >= roundup(plm->len, PAGE_SIZE))
|
|
return VM_FAULT_SIGBUS;
|
|
|
|
page = vmalloc_to_page(plm->data + offset);
|
|
get_page(page);
|
|
|
|
vmf->page = page;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct vm_operations_struct sel_mmap_policy_ops = {
|
|
.fault = sel_mmap_policy_fault,
|
|
.page_mkwrite = sel_mmap_policy_fault,
|
|
};
|
|
|
|
static int sel_mmap_policy(struct file *filp, struct vm_area_struct *vma)
|
|
{
|
|
if (vma->vm_flags & VM_SHARED) {
|
|
/* do not allow mprotect to make mapping writable */
|
|
vma->vm_flags &= ~VM_MAYWRITE;
|
|
|
|
if (vma->vm_flags & VM_WRITE)
|
|
return -EACCES;
|
|
}
|
|
|
|
vma->vm_flags |= VM_DONTEXPAND | VM_DONTDUMP;
|
|
vma->vm_ops = &sel_mmap_policy_ops;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct file_operations sel_policy_ops = {
|
|
.open = sel_open_policy,
|
|
.read = sel_read_policy,
|
|
.mmap = sel_mmap_policy,
|
|
.release = sel_release_policy,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static int sel_make_policy_nodes(struct selinux_fs_info *fsi)
|
|
{
|
|
int ret;
|
|
|
|
ret = sel_make_bools(fsi);
|
|
if (ret) {
|
|
pr_err("SELinux: failed to load policy booleans\n");
|
|
return ret;
|
|
}
|
|
|
|
ret = sel_make_classes(fsi);
|
|
if (ret) {
|
|
pr_err("SELinux: failed to load policy classes\n");
|
|
return ret;
|
|
}
|
|
|
|
ret = sel_make_policycap(fsi);
|
|
if (ret) {
|
|
pr_err("SELinux: failed to load policy capabilities\n");
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t sel_write_load(struct file *file, const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
ssize_t length;
|
|
void *data = NULL;
|
|
|
|
/* no partial writes */
|
|
if (*ppos)
|
|
return -EINVAL;
|
|
/* no empty policies */
|
|
if (!count)
|
|
return -EINVAL;
|
|
|
|
if (count > 64 * 1024 * 1024)
|
|
return -EFBIG;
|
|
|
|
mutex_lock(&fsi->mutex);
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__LOAD_POLICY, NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
data = vmalloc(count);
|
|
if (!data) {
|
|
length = -ENOMEM;
|
|
goto out;
|
|
}
|
|
if (copy_from_user(data, buf, count) != 0) {
|
|
length = -EFAULT;
|
|
goto out;
|
|
}
|
|
|
|
length = security_load_policy(fsi->state, data, count);
|
|
if (length) {
|
|
pr_warn_ratelimited("SELinux: failed to load policy\n");
|
|
goto out;
|
|
}
|
|
|
|
length = sel_make_policy_nodes(fsi);
|
|
if (length)
|
|
goto out1;
|
|
|
|
length = count;
|
|
|
|
out1:
|
|
audit_log(audit_context(), GFP_KERNEL, AUDIT_MAC_POLICY_LOAD,
|
|
"auid=%u ses=%u lsm=selinux res=1",
|
|
from_kuid(&init_user_ns, audit_get_loginuid(current)),
|
|
audit_get_sessionid(current));
|
|
|
|
out:
|
|
mutex_unlock(&fsi->mutex);
|
|
vfree(data);
|
|
return length;
|
|
}
|
|
|
|
static const struct file_operations sel_load_ops = {
|
|
.write = sel_write_load,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_write_context(struct file *file, char *buf, size_t size)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *canon = NULL;
|
|
u32 sid, len;
|
|
ssize_t length;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__CHECK_CONTEXT, NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_context_to_sid(state, buf, size, &sid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_sid_to_context(state, sid, &canon, &len);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ERANGE;
|
|
if (len > SIMPLE_TRANSACTION_LIMIT) {
|
|
pr_err("SELinux: %s: context size (%u) exceeds "
|
|
"payload max\n", __func__, len);
|
|
goto out;
|
|
}
|
|
|
|
memcpy(buf, canon, len);
|
|
length = len;
|
|
out:
|
|
kfree(canon);
|
|
return length;
|
|
}
|
|
|
|
static ssize_t sel_read_checkreqprot(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%u", fsi->state->checkreqprot);
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
static ssize_t sel_write_checkreqprot(struct file *file, const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
char *page;
|
|
ssize_t length;
|
|
unsigned int new_value;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__SETCHECKREQPROT,
|
|
NULL);
|
|
if (length)
|
|
return length;
|
|
|
|
if (count >= PAGE_SIZE)
|
|
return -ENOMEM;
|
|
|
|
/* No partial writes. */
|
|
if (*ppos != 0)
|
|
return -EINVAL;
|
|
|
|
page = memdup_user_nul(buf, count);
|
|
if (IS_ERR(page))
|
|
return PTR_ERR(page);
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(page, "%u", &new_value) != 1)
|
|
goto out;
|
|
|
|
fsi->state->checkreqprot = new_value ? 1 : 0;
|
|
length = count;
|
|
out:
|
|
kfree(page);
|
|
return length;
|
|
}
|
|
static const struct file_operations sel_checkreqprot_ops = {
|
|
.read = sel_read_checkreqprot,
|
|
.write = sel_write_checkreqprot,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_write_validatetrans(struct file *file,
|
|
const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *oldcon = NULL, *newcon = NULL, *taskcon = NULL;
|
|
char *req = NULL;
|
|
u32 osid, nsid, tsid;
|
|
u16 tclass;
|
|
int rc;
|
|
|
|
rc = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__VALIDATE_TRANS, NULL);
|
|
if (rc)
|
|
goto out;
|
|
|
|
rc = -ENOMEM;
|
|
if (count >= PAGE_SIZE)
|
|
goto out;
|
|
|
|
/* No partial writes. */
|
|
rc = -EINVAL;
|
|
if (*ppos != 0)
|
|
goto out;
|
|
|
|
req = memdup_user_nul(buf, count);
|
|
if (IS_ERR(req)) {
|
|
rc = PTR_ERR(req);
|
|
req = NULL;
|
|
goto out;
|
|
}
|
|
|
|
rc = -ENOMEM;
|
|
oldcon = kzalloc(count + 1, GFP_KERNEL);
|
|
if (!oldcon)
|
|
goto out;
|
|
|
|
newcon = kzalloc(count + 1, GFP_KERNEL);
|
|
if (!newcon)
|
|
goto out;
|
|
|
|
taskcon = kzalloc(count + 1, GFP_KERNEL);
|
|
if (!taskcon)
|
|
goto out;
|
|
|
|
rc = -EINVAL;
|
|
if (sscanf(req, "%s %s %hu %s", oldcon, newcon, &tclass, taskcon) != 4)
|
|
goto out;
|
|
|
|
rc = security_context_str_to_sid(state, oldcon, &osid, GFP_KERNEL);
|
|
if (rc)
|
|
goto out;
|
|
|
|
rc = security_context_str_to_sid(state, newcon, &nsid, GFP_KERNEL);
|
|
if (rc)
|
|
goto out;
|
|
|
|
rc = security_context_str_to_sid(state, taskcon, &tsid, GFP_KERNEL);
|
|
if (rc)
|
|
goto out;
|
|
|
|
rc = security_validate_transition_user(state, osid, nsid, tsid, tclass);
|
|
if (!rc)
|
|
rc = count;
|
|
out:
|
|
kfree(req);
|
|
kfree(oldcon);
|
|
kfree(newcon);
|
|
kfree(taskcon);
|
|
return rc;
|
|
}
|
|
|
|
static const struct file_operations sel_transition_ops = {
|
|
.write = sel_write_validatetrans,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
/*
|
|
* Remaining nodes use transaction based IO methods like nfsd/nfsctl.c
|
|
*/
|
|
static ssize_t sel_write_access(struct file *file, char *buf, size_t size);
|
|
static ssize_t sel_write_create(struct file *file, char *buf, size_t size);
|
|
static ssize_t sel_write_relabel(struct file *file, char *buf, size_t size);
|
|
static ssize_t sel_write_user(struct file *file, char *buf, size_t size);
|
|
static ssize_t sel_write_member(struct file *file, char *buf, size_t size);
|
|
|
|
static ssize_t (*const write_op[])(struct file *, char *, size_t) = {
|
|
[SEL_ACCESS] = sel_write_access,
|
|
[SEL_CREATE] = sel_write_create,
|
|
[SEL_RELABEL] = sel_write_relabel,
|
|
[SEL_USER] = sel_write_user,
|
|
[SEL_MEMBER] = sel_write_member,
|
|
[SEL_CONTEXT] = sel_write_context,
|
|
};
|
|
|
|
static ssize_t selinux_transaction_write(struct file *file, const char __user *buf, size_t size, loff_t *pos)
|
|
{
|
|
ino_t ino = file_inode(file)->i_ino;
|
|
char *data;
|
|
ssize_t rv;
|
|
|
|
if (ino >= ARRAY_SIZE(write_op) || !write_op[ino])
|
|
return -EINVAL;
|
|
|
|
data = simple_transaction_get(file, buf, size);
|
|
if (IS_ERR(data))
|
|
return PTR_ERR(data);
|
|
|
|
rv = write_op[ino](file, data, size);
|
|
if (rv > 0) {
|
|
simple_transaction_set(file, rv);
|
|
rv = size;
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
static const struct file_operations transaction_ops = {
|
|
.write = selinux_transaction_write,
|
|
.read = simple_transaction_read,
|
|
.release = simple_transaction_release,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
/*
|
|
* payload - write methods
|
|
* If the method has a response, the response should be put in buf,
|
|
* and the length returned. Otherwise return 0 or and -error.
|
|
*/
|
|
|
|
static ssize_t sel_write_access(struct file *file, char *buf, size_t size)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *scon = NULL, *tcon = NULL;
|
|
u32 ssid, tsid;
|
|
u16 tclass;
|
|
struct av_decision avd;
|
|
ssize_t length;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__COMPUTE_AV, NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
scon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!scon)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
tcon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!tcon)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(buf, "%s %s %hu", scon, tcon, &tclass) != 3)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, scon, &ssid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, tcon, &tsid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
security_compute_av_user(state, ssid, tsid, tclass, &avd);
|
|
|
|
length = scnprintf(buf, SIMPLE_TRANSACTION_LIMIT,
|
|
"%x %x %x %x %u %x",
|
|
avd.allowed, 0xffffffff,
|
|
avd.auditallow, avd.auditdeny,
|
|
avd.seqno, avd.flags);
|
|
out:
|
|
kfree(tcon);
|
|
kfree(scon);
|
|
return length;
|
|
}
|
|
|
|
static ssize_t sel_write_create(struct file *file, char *buf, size_t size)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *scon = NULL, *tcon = NULL;
|
|
char *namebuf = NULL, *objname = NULL;
|
|
u32 ssid, tsid, newsid;
|
|
u16 tclass;
|
|
ssize_t length;
|
|
char *newcon = NULL;
|
|
u32 len;
|
|
int nargs;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__COMPUTE_CREATE,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
scon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!scon)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
tcon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!tcon)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
namebuf = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!namebuf)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
nargs = sscanf(buf, "%s %s %hu %s", scon, tcon, &tclass, namebuf);
|
|
if (nargs < 3 || nargs > 4)
|
|
goto out;
|
|
if (nargs == 4) {
|
|
/*
|
|
* If and when the name of new object to be queried contains
|
|
* either whitespace or multibyte characters, they shall be
|
|
* encoded based on the percentage-encoding rule.
|
|
* If not encoded, the sscanf logic picks up only left-half
|
|
* of the supplied name; splitted by a whitespace unexpectedly.
|
|
*/
|
|
char *r, *w;
|
|
int c1, c2;
|
|
|
|
r = w = namebuf;
|
|
do {
|
|
c1 = *r++;
|
|
if (c1 == '+')
|
|
c1 = ' ';
|
|
else if (c1 == '%') {
|
|
c1 = hex_to_bin(*r++);
|
|
if (c1 < 0)
|
|
goto out;
|
|
c2 = hex_to_bin(*r++);
|
|
if (c2 < 0)
|
|
goto out;
|
|
c1 = (c1 << 4) | c2;
|
|
}
|
|
*w++ = c1;
|
|
} while (c1 != '\0');
|
|
|
|
objname = namebuf;
|
|
}
|
|
|
|
length = security_context_str_to_sid(state, scon, &ssid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, tcon, &tsid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_transition_sid_user(state, ssid, tsid, tclass,
|
|
objname, &newsid);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_sid_to_context(state, newsid, &newcon, &len);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ERANGE;
|
|
if (len > SIMPLE_TRANSACTION_LIMIT) {
|
|
pr_err("SELinux: %s: context size (%u) exceeds "
|
|
"payload max\n", __func__, len);
|
|
goto out;
|
|
}
|
|
|
|
memcpy(buf, newcon, len);
|
|
length = len;
|
|
out:
|
|
kfree(newcon);
|
|
kfree(namebuf);
|
|
kfree(tcon);
|
|
kfree(scon);
|
|
return length;
|
|
}
|
|
|
|
static ssize_t sel_write_relabel(struct file *file, char *buf, size_t size)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *scon = NULL, *tcon = NULL;
|
|
u32 ssid, tsid, newsid;
|
|
u16 tclass;
|
|
ssize_t length;
|
|
char *newcon = NULL;
|
|
u32 len;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__COMPUTE_RELABEL,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
scon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!scon)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
tcon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!tcon)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(buf, "%s %s %hu", scon, tcon, &tclass) != 3)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, scon, &ssid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, tcon, &tsid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_change_sid(state, ssid, tsid, tclass, &newsid);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_sid_to_context(state, newsid, &newcon, &len);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ERANGE;
|
|
if (len > SIMPLE_TRANSACTION_LIMIT)
|
|
goto out;
|
|
|
|
memcpy(buf, newcon, len);
|
|
length = len;
|
|
out:
|
|
kfree(newcon);
|
|
kfree(tcon);
|
|
kfree(scon);
|
|
return length;
|
|
}
|
|
|
|
static ssize_t sel_write_user(struct file *file, char *buf, size_t size)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *con = NULL, *user = NULL, *ptr;
|
|
u32 sid, *sids = NULL;
|
|
ssize_t length;
|
|
char *newcon;
|
|
int i, rc;
|
|
u32 len, nsids;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__COMPUTE_USER,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
con = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!con)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
user = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!user)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(buf, "%s %s", con, user) != 2)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, con, &sid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_get_user_sids(state, sid, user, &sids, &nsids);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = sprintf(buf, "%u", nsids) + 1;
|
|
ptr = buf + length;
|
|
for (i = 0; i < nsids; i++) {
|
|
rc = security_sid_to_context(state, sids[i], &newcon, &len);
|
|
if (rc) {
|
|
length = rc;
|
|
goto out;
|
|
}
|
|
if ((length + len) >= SIMPLE_TRANSACTION_LIMIT) {
|
|
kfree(newcon);
|
|
length = -ERANGE;
|
|
goto out;
|
|
}
|
|
memcpy(ptr, newcon, len);
|
|
kfree(newcon);
|
|
ptr += len;
|
|
length += len;
|
|
}
|
|
out:
|
|
kfree(sids);
|
|
kfree(user);
|
|
kfree(con);
|
|
return length;
|
|
}
|
|
|
|
static ssize_t sel_write_member(struct file *file, char *buf, size_t size)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *scon = NULL, *tcon = NULL;
|
|
u32 ssid, tsid, newsid;
|
|
u16 tclass;
|
|
ssize_t length;
|
|
char *newcon = NULL;
|
|
u32 len;
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__COMPUTE_MEMBER,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
scon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!scon)
|
|
goto out;
|
|
|
|
length = -ENOMEM;
|
|
tcon = kzalloc(size + 1, GFP_KERNEL);
|
|
if (!tcon)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(buf, "%s %s %hu", scon, tcon, &tclass) != 3)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, scon, &ssid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_context_str_to_sid(state, tcon, &tsid, GFP_KERNEL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_member_sid(state, ssid, tsid, tclass, &newsid);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = security_sid_to_context(state, newsid, &newcon, &len);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -ERANGE;
|
|
if (len > SIMPLE_TRANSACTION_LIMIT) {
|
|
pr_err("SELinux: %s: context size (%u) exceeds "
|
|
"payload max\n", __func__, len);
|
|
goto out;
|
|
}
|
|
|
|
memcpy(buf, newcon, len);
|
|
length = len;
|
|
out:
|
|
kfree(newcon);
|
|
kfree(tcon);
|
|
kfree(scon);
|
|
return length;
|
|
}
|
|
|
|
static struct inode *sel_make_inode(struct super_block *sb, int mode)
|
|
{
|
|
struct inode *ret = new_inode(sb);
|
|
|
|
if (ret) {
|
|
ret->i_mode = mode;
|
|
ret->i_atime = ret->i_mtime = ret->i_ctime = current_time(ret);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t sel_read_bool(struct file *filep, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filep)->i_sb->s_fs_info;
|
|
char *page = NULL;
|
|
ssize_t length;
|
|
ssize_t ret;
|
|
int cur_enforcing;
|
|
unsigned index = file_inode(filep)->i_ino & SEL_INO_MASK;
|
|
const char *name = filep->f_path.dentry->d_name.name;
|
|
|
|
mutex_lock(&fsi->mutex);
|
|
|
|
ret = -EINVAL;
|
|
if (index >= fsi->bool_num || strcmp(name,
|
|
fsi->bool_pending_names[index]))
|
|
goto out_unlock;
|
|
|
|
ret = -ENOMEM;
|
|
page = (char *)get_zeroed_page(GFP_KERNEL);
|
|
if (!page)
|
|
goto out_unlock;
|
|
|
|
cur_enforcing = security_get_bool_value(fsi->state, index);
|
|
if (cur_enforcing < 0) {
|
|
ret = cur_enforcing;
|
|
goto out_unlock;
|
|
}
|
|
length = scnprintf(page, PAGE_SIZE, "%d %d", cur_enforcing,
|
|
fsi->bool_pending_values[index]);
|
|
mutex_unlock(&fsi->mutex);
|
|
ret = simple_read_from_buffer(buf, count, ppos, page, length);
|
|
out_free:
|
|
free_page((unsigned long)page);
|
|
return ret;
|
|
|
|
out_unlock:
|
|
mutex_unlock(&fsi->mutex);
|
|
goto out_free;
|
|
}
|
|
|
|
static ssize_t sel_write_bool(struct file *filep, const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filep)->i_sb->s_fs_info;
|
|
char *page = NULL;
|
|
ssize_t length;
|
|
int new_value;
|
|
unsigned index = file_inode(filep)->i_ino & SEL_INO_MASK;
|
|
const char *name = filep->f_path.dentry->d_name.name;
|
|
|
|
if (count >= PAGE_SIZE)
|
|
return -ENOMEM;
|
|
|
|
/* No partial writes. */
|
|
if (*ppos != 0)
|
|
return -EINVAL;
|
|
|
|
page = memdup_user_nul(buf, count);
|
|
if (IS_ERR(page))
|
|
return PTR_ERR(page);
|
|
|
|
mutex_lock(&fsi->mutex);
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__SETBOOL,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (index >= fsi->bool_num || strcmp(name,
|
|
fsi->bool_pending_names[index]))
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(page, "%d", &new_value) != 1)
|
|
goto out;
|
|
|
|
if (new_value)
|
|
new_value = 1;
|
|
|
|
fsi->bool_pending_values[index] = new_value;
|
|
length = count;
|
|
|
|
out:
|
|
mutex_unlock(&fsi->mutex);
|
|
kfree(page);
|
|
return length;
|
|
}
|
|
|
|
static const struct file_operations sel_bool_ops = {
|
|
.read = sel_read_bool,
|
|
.write = sel_write_bool,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_commit_bools_write(struct file *filep,
|
|
const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filep)->i_sb->s_fs_info;
|
|
char *page = NULL;
|
|
ssize_t length;
|
|
int new_value;
|
|
|
|
if (count >= PAGE_SIZE)
|
|
return -ENOMEM;
|
|
|
|
/* No partial writes. */
|
|
if (*ppos != 0)
|
|
return -EINVAL;
|
|
|
|
page = memdup_user_nul(buf, count);
|
|
if (IS_ERR(page))
|
|
return PTR_ERR(page);
|
|
|
|
mutex_lock(&fsi->mutex);
|
|
|
|
length = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__SETBOOL,
|
|
NULL);
|
|
if (length)
|
|
goto out;
|
|
|
|
length = -EINVAL;
|
|
if (sscanf(page, "%d", &new_value) != 1)
|
|
goto out;
|
|
|
|
length = 0;
|
|
if (new_value && fsi->bool_pending_values)
|
|
length = security_set_bools(fsi->state, fsi->bool_num,
|
|
fsi->bool_pending_values);
|
|
|
|
if (!length)
|
|
length = count;
|
|
|
|
out:
|
|
mutex_unlock(&fsi->mutex);
|
|
kfree(page);
|
|
return length;
|
|
}
|
|
|
|
static const struct file_operations sel_commit_bools_ops = {
|
|
.write = sel_commit_bools_write,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static void sel_remove_entries(struct dentry *de)
|
|
{
|
|
d_genocide(de);
|
|
shrink_dcache_parent(de);
|
|
}
|
|
|
|
#define BOOL_DIR_NAME "booleans"
|
|
|
|
static int sel_make_bools(struct selinux_fs_info *fsi)
|
|
{
|
|
int i, ret;
|
|
ssize_t len;
|
|
struct dentry *dentry = NULL;
|
|
struct dentry *dir = fsi->bool_dir;
|
|
struct inode *inode = NULL;
|
|
struct inode_security_struct *isec;
|
|
char **names = NULL, *page;
|
|
int num;
|
|
int *values = NULL;
|
|
u32 sid;
|
|
|
|
/* remove any existing files */
|
|
for (i = 0; i < fsi->bool_num; i++)
|
|
kfree(fsi->bool_pending_names[i]);
|
|
kfree(fsi->bool_pending_names);
|
|
kfree(fsi->bool_pending_values);
|
|
fsi->bool_num = 0;
|
|
fsi->bool_pending_names = NULL;
|
|
fsi->bool_pending_values = NULL;
|
|
|
|
sel_remove_entries(dir);
|
|
|
|
ret = -ENOMEM;
|
|
page = (char *)get_zeroed_page(GFP_KERNEL);
|
|
if (!page)
|
|
goto out;
|
|
|
|
ret = security_get_bools(fsi->state, &num, &names, &values);
|
|
if (ret)
|
|
goto out;
|
|
|
|
for (i = 0; i < num; i++) {
|
|
ret = -ENOMEM;
|
|
dentry = d_alloc_name(dir, names[i]);
|
|
if (!dentry)
|
|
goto out;
|
|
|
|
ret = -ENOMEM;
|
|
inode = sel_make_inode(dir->d_sb, S_IFREG | S_IRUGO | S_IWUSR);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
goto out;
|
|
}
|
|
|
|
ret = -ENAMETOOLONG;
|
|
len = snprintf(page, PAGE_SIZE, "/%s/%s", BOOL_DIR_NAME, names[i]);
|
|
if (len >= PAGE_SIZE) {
|
|
dput(dentry);
|
|
iput(inode);
|
|
goto out;
|
|
}
|
|
|
|
isec = (struct inode_security_struct *)inode->i_security;
|
|
ret = security_genfs_sid(fsi->state, "selinuxfs", page,
|
|
SECCLASS_FILE, &sid);
|
|
if (ret) {
|
|
pr_warn_ratelimited("SELinux: no sid found, defaulting to security isid for %s\n",
|
|
page);
|
|
sid = SECINITSID_SECURITY;
|
|
}
|
|
|
|
isec->sid = sid;
|
|
isec->initialized = LABEL_INITIALIZED;
|
|
inode->i_fop = &sel_bool_ops;
|
|
inode->i_ino = i|SEL_BOOL_INO_OFFSET;
|
|
d_add(dentry, inode);
|
|
}
|
|
fsi->bool_num = num;
|
|
fsi->bool_pending_names = names;
|
|
fsi->bool_pending_values = values;
|
|
|
|
free_page((unsigned long)page);
|
|
return 0;
|
|
out:
|
|
free_page((unsigned long)page);
|
|
|
|
if (names) {
|
|
for (i = 0; i < num; i++)
|
|
kfree(names[i]);
|
|
kfree(names);
|
|
}
|
|
kfree(values);
|
|
sel_remove_entries(dir);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t sel_read_avc_cache_threshold(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%u",
|
|
avc_get_cache_threshold(state->avc));
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
static ssize_t sel_write_avc_cache_threshold(struct file *file,
|
|
const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *page;
|
|
ssize_t ret;
|
|
unsigned int new_value;
|
|
|
|
ret = avc_has_perm(&selinux_state,
|
|
current_sid(), SECINITSID_SECURITY,
|
|
SECCLASS_SECURITY, SECURITY__SETSECPARAM,
|
|
NULL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (count >= PAGE_SIZE)
|
|
return -ENOMEM;
|
|
|
|
/* No partial writes. */
|
|
if (*ppos != 0)
|
|
return -EINVAL;
|
|
|
|
page = memdup_user_nul(buf, count);
|
|
if (IS_ERR(page))
|
|
return PTR_ERR(page);
|
|
|
|
ret = -EINVAL;
|
|
if (sscanf(page, "%u", &new_value) != 1)
|
|
goto out;
|
|
|
|
avc_set_cache_threshold(state->avc, new_value);
|
|
|
|
ret = count;
|
|
out:
|
|
kfree(page);
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t sel_read_avc_hash_stats(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *page;
|
|
ssize_t length;
|
|
|
|
page = (char *)__get_free_page(GFP_KERNEL);
|
|
if (!page)
|
|
return -ENOMEM;
|
|
|
|
length = avc_get_hash_stats(state->avc, page);
|
|
if (length >= 0)
|
|
length = simple_read_from_buffer(buf, count, ppos, page, length);
|
|
free_page((unsigned long)page);
|
|
|
|
return length;
|
|
}
|
|
|
|
static ssize_t sel_read_sidtab_hash_stats(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(filp)->i_sb->s_fs_info;
|
|
struct selinux_state *state = fsi->state;
|
|
char *page;
|
|
ssize_t length;
|
|
|
|
page = (char *)__get_free_page(GFP_KERNEL);
|
|
if (!page)
|
|
return -ENOMEM;
|
|
|
|
length = security_sidtab_hash_stats(state, page);
|
|
if (length >= 0)
|
|
length = simple_read_from_buffer(buf, count, ppos, page,
|
|
length);
|
|
free_page((unsigned long)page);
|
|
|
|
return length;
|
|
}
|
|
|
|
static const struct file_operations sel_sidtab_hash_stats_ops = {
|
|
.read = sel_read_sidtab_hash_stats,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static const struct file_operations sel_avc_cache_threshold_ops = {
|
|
.read = sel_read_avc_cache_threshold,
|
|
.write = sel_write_avc_cache_threshold,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static const struct file_operations sel_avc_hash_stats_ops = {
|
|
.read = sel_read_avc_hash_stats,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
#ifdef CONFIG_SECURITY_SELINUX_AVC_STATS
|
|
static struct avc_cache_stats *sel_avc_get_stat_idx(loff_t *idx)
|
|
{
|
|
int cpu;
|
|
|
|
for (cpu = *idx; cpu < nr_cpu_ids; ++cpu) {
|
|
if (!cpu_possible(cpu))
|
|
continue;
|
|
*idx = cpu + 1;
|
|
return &per_cpu(avc_cache_stats, cpu);
|
|
}
|
|
(*idx)++;
|
|
return NULL;
|
|
}
|
|
|
|
static void *sel_avc_stats_seq_start(struct seq_file *seq, loff_t *pos)
|
|
{
|
|
loff_t n = *pos - 1;
|
|
|
|
if (*pos == 0)
|
|
return SEQ_START_TOKEN;
|
|
|
|
return sel_avc_get_stat_idx(&n);
|
|
}
|
|
|
|
static void *sel_avc_stats_seq_next(struct seq_file *seq, void *v, loff_t *pos)
|
|
{
|
|
return sel_avc_get_stat_idx(pos);
|
|
}
|
|
|
|
static int sel_avc_stats_seq_show(struct seq_file *seq, void *v)
|
|
{
|
|
struct avc_cache_stats *st = v;
|
|
|
|
if (v == SEQ_START_TOKEN) {
|
|
seq_puts(seq,
|
|
"lookups hits misses allocations reclaims frees\n");
|
|
} else {
|
|
unsigned int lookups = st->lookups;
|
|
unsigned int misses = st->misses;
|
|
unsigned int hits = lookups - misses;
|
|
seq_printf(seq, "%u %u %u %u %u %u\n", lookups,
|
|
hits, misses, st->allocations,
|
|
st->reclaims, st->frees);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void sel_avc_stats_seq_stop(struct seq_file *seq, void *v)
|
|
{ }
|
|
|
|
static const struct seq_operations sel_avc_cache_stats_seq_ops = {
|
|
.start = sel_avc_stats_seq_start,
|
|
.next = sel_avc_stats_seq_next,
|
|
.show = sel_avc_stats_seq_show,
|
|
.stop = sel_avc_stats_seq_stop,
|
|
};
|
|
|
|
static int sel_open_avc_cache_stats(struct inode *inode, struct file *file)
|
|
{
|
|
return seq_open(file, &sel_avc_cache_stats_seq_ops);
|
|
}
|
|
|
|
static const struct file_operations sel_avc_cache_stats_ops = {
|
|
.open = sel_open_avc_cache_stats,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = seq_release,
|
|
};
|
|
#endif
|
|
|
|
static int sel_make_avc_files(struct dentry *dir)
|
|
{
|
|
struct super_block *sb = dir->d_sb;
|
|
struct selinux_fs_info *fsi = sb->s_fs_info;
|
|
int i;
|
|
static const struct tree_descr files[] = {
|
|
{ "cache_threshold",
|
|
&sel_avc_cache_threshold_ops, S_IRUGO|S_IWUSR },
|
|
{ "hash_stats", &sel_avc_hash_stats_ops, S_IRUGO },
|
|
#ifdef CONFIG_SECURITY_SELINUX_AVC_STATS
|
|
{ "cache_stats", &sel_avc_cache_stats_ops, S_IRUGO },
|
|
#endif
|
|
};
|
|
|
|
for (i = 0; i < ARRAY_SIZE(files); i++) {
|
|
struct inode *inode;
|
|
struct dentry *dentry;
|
|
|
|
dentry = d_alloc_name(dir, files[i].name);
|
|
if (!dentry)
|
|
return -ENOMEM;
|
|
|
|
inode = sel_make_inode(dir->d_sb, S_IFREG|files[i].mode);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
inode->i_fop = files[i].ops;
|
|
inode->i_ino = ++fsi->last_ino;
|
|
d_add(dentry, inode);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int sel_make_ss_files(struct dentry *dir)
|
|
{
|
|
struct super_block *sb = dir->d_sb;
|
|
struct selinux_fs_info *fsi = sb->s_fs_info;
|
|
int i;
|
|
static struct tree_descr files[] = {
|
|
{ "sidtab_hash_stats", &sel_sidtab_hash_stats_ops, S_IRUGO },
|
|
};
|
|
|
|
for (i = 0; i < ARRAY_SIZE(files); i++) {
|
|
struct inode *inode;
|
|
struct dentry *dentry;
|
|
|
|
dentry = d_alloc_name(dir, files[i].name);
|
|
if (!dentry)
|
|
return -ENOMEM;
|
|
|
|
inode = sel_make_inode(dir->d_sb, S_IFREG|files[i].mode);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
inode->i_fop = files[i].ops;
|
|
inode->i_ino = ++fsi->last_ino;
|
|
d_add(dentry, inode);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t sel_read_initcon(struct file *file, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
char *con;
|
|
u32 sid, len;
|
|
ssize_t ret;
|
|
|
|
sid = file_inode(file)->i_ino&SEL_INO_MASK;
|
|
ret = security_sid_to_context(fsi->state, sid, &con, &len);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = simple_read_from_buffer(buf, count, ppos, con, len);
|
|
kfree(con);
|
|
return ret;
|
|
}
|
|
|
|
static const struct file_operations sel_initcon_ops = {
|
|
.read = sel_read_initcon,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static int sel_make_initcon_files(struct dentry *dir)
|
|
{
|
|
int i;
|
|
|
|
for (i = 1; i <= SECINITSID_NUM; i++) {
|
|
struct inode *inode;
|
|
struct dentry *dentry;
|
|
dentry = d_alloc_name(dir, security_get_initial_sid_context(i));
|
|
if (!dentry)
|
|
return -ENOMEM;
|
|
|
|
inode = sel_make_inode(dir->d_sb, S_IFREG|S_IRUGO);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
inode->i_fop = &sel_initcon_ops;
|
|
inode->i_ino = i|SEL_INITCON_INO_OFFSET;
|
|
d_add(dentry, inode);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline unsigned long sel_class_to_ino(u16 class)
|
|
{
|
|
return (class * (SEL_VEC_MAX + 1)) | SEL_CLASS_INO_OFFSET;
|
|
}
|
|
|
|
static inline u16 sel_ino_to_class(unsigned long ino)
|
|
{
|
|
return (ino & SEL_INO_MASK) / (SEL_VEC_MAX + 1);
|
|
}
|
|
|
|
static inline unsigned long sel_perm_to_ino(u16 class, u32 perm)
|
|
{
|
|
return (class * (SEL_VEC_MAX + 1) + perm) | SEL_CLASS_INO_OFFSET;
|
|
}
|
|
|
|
static inline u32 sel_ino_to_perm(unsigned long ino)
|
|
{
|
|
return (ino & SEL_INO_MASK) % (SEL_VEC_MAX + 1);
|
|
}
|
|
|
|
static ssize_t sel_read_class(struct file *file, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
unsigned long ino = file_inode(file)->i_ino;
|
|
char res[TMPBUFLEN];
|
|
ssize_t len = snprintf(res, sizeof(res), "%d", sel_ino_to_class(ino));
|
|
return simple_read_from_buffer(buf, count, ppos, res, len);
|
|
}
|
|
|
|
static const struct file_operations sel_class_ops = {
|
|
.read = sel_read_class,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_read_perm(struct file *file, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
unsigned long ino = file_inode(file)->i_ino;
|
|
char res[TMPBUFLEN];
|
|
ssize_t len = snprintf(res, sizeof(res), "%d", sel_ino_to_perm(ino));
|
|
return simple_read_from_buffer(buf, count, ppos, res, len);
|
|
}
|
|
|
|
static const struct file_operations sel_perm_ops = {
|
|
.read = sel_read_perm,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static ssize_t sel_read_policycap(struct file *file, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info;
|
|
int value;
|
|
char tmpbuf[TMPBUFLEN];
|
|
ssize_t length;
|
|
unsigned long i_ino = file_inode(file)->i_ino;
|
|
|
|
value = security_policycap_supported(fsi->state, i_ino & SEL_INO_MASK);
|
|
length = scnprintf(tmpbuf, TMPBUFLEN, "%d", value);
|
|
|
|
return simple_read_from_buffer(buf, count, ppos, tmpbuf, length);
|
|
}
|
|
|
|
static const struct file_operations sel_policycap_ops = {
|
|
.read = sel_read_policycap,
|
|
.llseek = generic_file_llseek,
|
|
};
|
|
|
|
static int sel_make_perm_files(char *objclass, int classvalue,
|
|
struct dentry *dir)
|
|
{
|
|
struct selinux_fs_info *fsi = dir->d_sb->s_fs_info;
|
|
int i, rc, nperms;
|
|
char **perms;
|
|
|
|
rc = security_get_permissions(fsi->state, objclass, &perms, &nperms);
|
|
if (rc)
|
|
return rc;
|
|
|
|
for (i = 0; i < nperms; i++) {
|
|
struct inode *inode;
|
|
struct dentry *dentry;
|
|
|
|
rc = -ENOMEM;
|
|
dentry = d_alloc_name(dir, perms[i]);
|
|
if (!dentry)
|
|
goto out;
|
|
|
|
rc = -ENOMEM;
|
|
inode = sel_make_inode(dir->d_sb, S_IFREG|S_IRUGO);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
goto out;
|
|
}
|
|
|
|
inode->i_fop = &sel_perm_ops;
|
|
/* i+1 since perm values are 1-indexed */
|
|
inode->i_ino = sel_perm_to_ino(classvalue, i + 1);
|
|
d_add(dentry, inode);
|
|
}
|
|
rc = 0;
|
|
out:
|
|
for (i = 0; i < nperms; i++)
|
|
kfree(perms[i]);
|
|
kfree(perms);
|
|
return rc;
|
|
}
|
|
|
|
static int sel_make_class_dir_entries(char *classname, int index,
|
|
struct dentry *dir)
|
|
{
|
|
struct super_block *sb = dir->d_sb;
|
|
struct selinux_fs_info *fsi = sb->s_fs_info;
|
|
struct dentry *dentry = NULL;
|
|
struct inode *inode = NULL;
|
|
int rc;
|
|
|
|
dentry = d_alloc_name(dir, "index");
|
|
if (!dentry)
|
|
return -ENOMEM;
|
|
|
|
inode = sel_make_inode(dir->d_sb, S_IFREG|S_IRUGO);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
inode->i_fop = &sel_class_ops;
|
|
inode->i_ino = sel_class_to_ino(index);
|
|
d_add(dentry, inode);
|
|
|
|
dentry = sel_make_dir(dir, "perms", &fsi->last_class_ino);
|
|
if (IS_ERR(dentry))
|
|
return PTR_ERR(dentry);
|
|
|
|
rc = sel_make_perm_files(classname, index, dentry);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int sel_make_classes(struct selinux_fs_info *fsi)
|
|
{
|
|
|
|
int rc, nclasses, i;
|
|
char **classes;
|
|
|
|
/* delete any existing entries */
|
|
sel_remove_entries(fsi->class_dir);
|
|
|
|
rc = security_get_classes(fsi->state, &classes, &nclasses);
|
|
if (rc)
|
|
return rc;
|
|
|
|
/* +2 since classes are 1-indexed */
|
|
fsi->last_class_ino = sel_class_to_ino(nclasses + 2);
|
|
|
|
for (i = 0; i < nclasses; i++) {
|
|
struct dentry *class_name_dir;
|
|
|
|
class_name_dir = sel_make_dir(fsi->class_dir, classes[i],
|
|
&fsi->last_class_ino);
|
|
if (IS_ERR(class_name_dir)) {
|
|
rc = PTR_ERR(class_name_dir);
|
|
goto out;
|
|
}
|
|
|
|
/* i+1 since class values are 1-indexed */
|
|
rc = sel_make_class_dir_entries(classes[i], i + 1,
|
|
class_name_dir);
|
|
if (rc)
|
|
goto out;
|
|
}
|
|
rc = 0;
|
|
out:
|
|
for (i = 0; i < nclasses; i++)
|
|
kfree(classes[i]);
|
|
kfree(classes);
|
|
return rc;
|
|
}
|
|
|
|
static int sel_make_policycap(struct selinux_fs_info *fsi)
|
|
{
|
|
unsigned int iter;
|
|
struct dentry *dentry = NULL;
|
|
struct inode *inode = NULL;
|
|
|
|
sel_remove_entries(fsi->policycap_dir);
|
|
|
|
for (iter = 0; iter <= POLICYDB_CAPABILITY_MAX; iter++) {
|
|
if (iter < ARRAY_SIZE(selinux_policycap_names))
|
|
dentry = d_alloc_name(fsi->policycap_dir,
|
|
selinux_policycap_names[iter]);
|
|
else
|
|
dentry = d_alloc_name(fsi->policycap_dir, "unknown");
|
|
|
|
if (dentry == NULL)
|
|
return -ENOMEM;
|
|
|
|
inode = sel_make_inode(fsi->sb, S_IFREG | 0444);
|
|
if (inode == NULL) {
|
|
dput(dentry);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
inode->i_fop = &sel_policycap_ops;
|
|
inode->i_ino = iter | SEL_POLICYCAP_INO_OFFSET;
|
|
d_add(dentry, inode);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct dentry *sel_make_dir(struct dentry *dir, const char *name,
|
|
unsigned long *ino)
|
|
{
|
|
struct dentry *dentry = d_alloc_name(dir, name);
|
|
struct inode *inode;
|
|
|
|
if (!dentry)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
inode = sel_make_inode(dir->d_sb, S_IFDIR | S_IRUGO | S_IXUGO);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
return ERR_PTR(-ENOMEM);
|
|
}
|
|
|
|
inode->i_op = &simple_dir_inode_operations;
|
|
inode->i_fop = &simple_dir_operations;
|
|
inode->i_ino = ++(*ino);
|
|
/* directory inodes start off with i_nlink == 2 (for "." entry) */
|
|
inc_nlink(inode);
|
|
d_add(dentry, inode);
|
|
/* bump link count on parent directory, too */
|
|
inc_nlink(d_inode(dir));
|
|
|
|
return dentry;
|
|
}
|
|
|
|
#define NULL_FILE_NAME "null"
|
|
|
|
static int sel_fill_super(struct super_block *sb, void *data, int silent)
|
|
{
|
|
struct selinux_fs_info *fsi;
|
|
int ret;
|
|
struct dentry *dentry;
|
|
struct inode *inode;
|
|
struct inode_security_struct *isec;
|
|
|
|
static const struct tree_descr selinux_files[] = {
|
|
[SEL_LOAD] = {"load", &sel_load_ops, S_IRUSR|S_IWUSR},
|
|
[SEL_ENFORCE] = {"enforce", &sel_enforce_ops, S_IRUGO|S_IWUSR},
|
|
[SEL_CONTEXT] = {"context", &transaction_ops, S_IRUGO|S_IWUGO},
|
|
[SEL_ACCESS] = {"access", &transaction_ops, S_IRUGO|S_IWUGO},
|
|
[SEL_CREATE] = {"create", &transaction_ops, S_IRUGO|S_IWUGO},
|
|
[SEL_RELABEL] = {"relabel", &transaction_ops, S_IRUGO|S_IWUGO},
|
|
[SEL_USER] = {"user", &transaction_ops, S_IRUGO|S_IWUGO},
|
|
[SEL_POLICYVERS] = {"policyvers", &sel_policyvers_ops, S_IRUGO},
|
|
[SEL_COMMIT_BOOLS] = {"commit_pending_bools", &sel_commit_bools_ops, S_IWUSR},
|
|
[SEL_MLS] = {"mls", &sel_mls_ops, S_IRUGO},
|
|
[SEL_DISABLE] = {"disable", &sel_disable_ops, S_IWUSR},
|
|
[SEL_MEMBER] = {"member", &transaction_ops, S_IRUGO|S_IWUGO},
|
|
[SEL_CHECKREQPROT] = {"checkreqprot", &sel_checkreqprot_ops, S_IRUGO|S_IWUSR},
|
|
[SEL_REJECT_UNKNOWN] = {"reject_unknown", &sel_handle_unknown_ops, S_IRUGO},
|
|
[SEL_DENY_UNKNOWN] = {"deny_unknown", &sel_handle_unknown_ops, S_IRUGO},
|
|
[SEL_STATUS] = {"status", &sel_handle_status_ops, S_IRUGO},
|
|
[SEL_POLICY] = {"policy", &sel_policy_ops, S_IRUGO},
|
|
[SEL_VALIDATE_TRANS] = {"validatetrans", &sel_transition_ops,
|
|
S_IWUGO},
|
|
/* last one */ {""}
|
|
};
|
|
|
|
ret = selinux_fs_info_create(sb);
|
|
if (ret)
|
|
goto err;
|
|
|
|
ret = simple_fill_super(sb, SELINUX_MAGIC, selinux_files);
|
|
if (ret)
|
|
goto err;
|
|
|
|
fsi = sb->s_fs_info;
|
|
fsi->bool_dir = sel_make_dir(sb->s_root, BOOL_DIR_NAME, &fsi->last_ino);
|
|
if (IS_ERR(fsi->bool_dir)) {
|
|
ret = PTR_ERR(fsi->bool_dir);
|
|
fsi->bool_dir = NULL;
|
|
goto err;
|
|
}
|
|
|
|
ret = -ENOMEM;
|
|
dentry = d_alloc_name(sb->s_root, NULL_FILE_NAME);
|
|
if (!dentry)
|
|
goto err;
|
|
|
|
ret = -ENOMEM;
|
|
inode = sel_make_inode(sb, S_IFCHR | S_IRUGO | S_IWUGO);
|
|
if (!inode) {
|
|
dput(dentry);
|
|
goto err;
|
|
}
|
|
|
|
inode->i_ino = ++fsi->last_ino;
|
|
isec = (struct inode_security_struct *)inode->i_security;
|
|
isec->sid = SECINITSID_DEVNULL;
|
|
isec->sclass = SECCLASS_CHR_FILE;
|
|
isec->initialized = LABEL_INITIALIZED;
|
|
|
|
init_special_inode(inode, S_IFCHR | S_IRUGO | S_IWUGO, MKDEV(MEM_MAJOR, 3));
|
|
d_add(dentry, inode);
|
|
|
|
dentry = sel_make_dir(sb->s_root, "avc", &fsi->last_ino);
|
|
if (IS_ERR(dentry)) {
|
|
ret = PTR_ERR(dentry);
|
|
goto err;
|
|
}
|
|
|
|
ret = sel_make_avc_files(dentry);
|
|
if (ret)
|
|
goto err;
|
|
|
|
dentry = sel_make_dir(sb->s_root, "ss", &fsi->last_ino);
|
|
if (IS_ERR(dentry)) {
|
|
ret = PTR_ERR(dentry);
|
|
goto err;
|
|
}
|
|
|
|
ret = sel_make_ss_files(dentry);
|
|
if (ret)
|
|
goto err;
|
|
|
|
dentry = sel_make_dir(sb->s_root, "initial_contexts", &fsi->last_ino);
|
|
if (IS_ERR(dentry)) {
|
|
ret = PTR_ERR(dentry);
|
|
goto err;
|
|
}
|
|
|
|
ret = sel_make_initcon_files(dentry);
|
|
if (ret)
|
|
goto err;
|
|
|
|
fsi->class_dir = sel_make_dir(sb->s_root, "class", &fsi->last_ino);
|
|
if (IS_ERR(fsi->class_dir)) {
|
|
ret = PTR_ERR(fsi->class_dir);
|
|
fsi->class_dir = NULL;
|
|
goto err;
|
|
}
|
|
|
|
fsi->policycap_dir = sel_make_dir(sb->s_root, "policy_capabilities",
|
|
&fsi->last_ino);
|
|
if (IS_ERR(fsi->policycap_dir)) {
|
|
ret = PTR_ERR(fsi->policycap_dir);
|
|
fsi->policycap_dir = NULL;
|
|
goto err;
|
|
}
|
|
|
|
ret = sel_make_policy_nodes(fsi);
|
|
if (ret)
|
|
goto err;
|
|
return 0;
|
|
err:
|
|
pr_err("SELinux: %s: failed while creating inodes\n",
|
|
__func__);
|
|
|
|
selinux_fs_info_free(sb);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static struct dentry *sel_mount(struct file_system_type *fs_type,
|
|
int flags, const char *dev_name, void *data)
|
|
{
|
|
return mount_single(fs_type, flags, data, sel_fill_super);
|
|
}
|
|
|
|
static void sel_kill_sb(struct super_block *sb)
|
|
{
|
|
selinux_fs_info_free(sb);
|
|
kill_litter_super(sb);
|
|
}
|
|
|
|
static struct file_system_type sel_fs_type = {
|
|
.name = "selinuxfs",
|
|
.mount = sel_mount,
|
|
.kill_sb = sel_kill_sb,
|
|
};
|
|
|
|
struct vfsmount *selinuxfs_mount;
|
|
struct path selinux_null;
|
|
|
|
static int __init init_sel_fs(void)
|
|
{
|
|
struct qstr null_name = QSTR_INIT(NULL_FILE_NAME,
|
|
sizeof(NULL_FILE_NAME)-1);
|
|
int err;
|
|
|
|
if (!selinux_enabled)
|
|
return 0;
|
|
|
|
err = sysfs_create_mount_point(fs_kobj, "selinux");
|
|
if (err)
|
|
return err;
|
|
|
|
err = register_filesystem(&sel_fs_type);
|
|
if (err) {
|
|
sysfs_remove_mount_point(fs_kobj, "selinux");
|
|
return err;
|
|
}
|
|
|
|
selinux_null.mnt = selinuxfs_mount = kern_mount(&sel_fs_type);
|
|
if (IS_ERR(selinuxfs_mount)) {
|
|
pr_err("selinuxfs: could not mount!\n");
|
|
err = PTR_ERR(selinuxfs_mount);
|
|
selinuxfs_mount = NULL;
|
|
}
|
|
selinux_null.dentry = d_hash_and_lookup(selinux_null.mnt->mnt_root,
|
|
&null_name);
|
|
if (IS_ERR(selinux_null.dentry)) {
|
|
pr_err("selinuxfs: could not lookup null!\n");
|
|
err = PTR_ERR(selinux_null.dentry);
|
|
selinux_null.dentry = NULL;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
__initcall(init_sel_fs);
|
|
|
|
#ifdef CONFIG_SECURITY_SELINUX_DISABLE
|
|
void exit_sel_fs(void)
|
|
{
|
|
sysfs_remove_mount_point(fs_kobj, "selinux");
|
|
dput(selinux_null.dentry);
|
|
kern_unmount(selinuxfs_mount);
|
|
unregister_filesystem(&sel_fs_type);
|
|
}
|
|
#endif
|