security/apparmor: apply apparmor patchs from

https://gitlab.com/apparmor/apparmor/-/tree/master/kernel-patches/v4.8
This commit is contained in:
Mauro (mdrjr) Ribeiro
2022-05-04 14:36:19 -03:00
parent 5d0b4dbdfc
commit 713d2ee982
16 changed files with 1182 additions and 5 deletions

View File

@@ -1,5 +1,6 @@
#
# Generated include files
#
net_names.h
capability_names.h
rlim_names.h

View File

@@ -4,10 +4,10 @@ obj-$(CONFIG_SECURITY_APPARMOR) += apparmor.o
apparmor-y := apparmorfs.o audit.o capability.o context.o ipc.o lib.o match.o \
path.o domain.o policy.o policy_unpack.o procattr.o lsm.o \
resource.o sid.o file.o
resource.o sid.o file.o net.o mount.o
apparmor-$(CONFIG_SECURITY_APPARMOR_HASH) += crypto.o
clean-files := capability_names.h rlim_names.h
clean-files := capability_names.h rlim_names.h net_names.h
# Build a lower case string table of capability names
@@ -25,6 +25,38 @@ cmd_make-caps = echo "static const char *const capability_names[] = {" > $@ ;\
-e 's/^\#define[ \t]+CAP_([A-Z0-9_]+)[ \t]+([0-9]+)/\L\1/p' | \
tr '\n' ' ' | sed -e 's/ $$/"\n/' >> $@
# Build a lower case string table of address family names
# Transform lines from
# define AF_LOCAL 1 /* POSIX name for AF_UNIX */
# #define AF_INET 2 /* Internet IP Protocol */
# to
# [1] = "local",
# [2] = "inet",
#
# and build the securityfs entries for the mapping.
# Transforms lines from
# #define AF_INET 2 /* Internet IP Protocol */
# to
# #define AA_FS_AF_MASK "local inet"
quiet_cmd_make-af = GEN $@
cmd_make-af = echo "static const char *address_family_names[] = {" > $@ ;\
sed $< >>$@ -r -n -e "/AF_MAX/d" -e "/AF_LOCAL/d" -e \
's/^\#define[ \t]+AF_([A-Z0-9_]+)[ \t]+([0-9]+)(.*)/[\2] = "\L\1",/p';\
echo "};" >> $@ ;\
echo -n '\#define AA_FS_AF_MASK "' >> $@ ;\
sed -r -n 's/^\#define[ \t]+AF_([A-Z0-9_]+)[ \t]+([0-9]+)(.*)/\L\1/p'\
$< | tr '\n' ' ' | sed -e 's/ $$/"\n/' >> $@
# Build a lower case string table of sock type names
# Transform lines from
# SOCK_STREAM = 1,
# to
# [1] = "stream",
quiet_cmd_make-sock = GEN $@
cmd_make-sock = echo "static const char *sock_type_names[] = {" >> $@ ;\
sed $^ >>$@ -r -n \
-e 's/^\tSOCK_([A-Z0-9_]+)[\t]+=[ \t]+([0-9]+)(.*)/[\2] = "\L\1",/p';\
echo "};" >> $@
# Build a lower case string table of rlimit names.
# Transforms lines from
@@ -61,6 +93,7 @@ cmd_make-rlim = echo "static const char *const rlim_names[RLIM_NLIMITS] = {" \
tr '\n' ' ' | sed -e 's/ $$/"\n/' >> $@
$(obj)/capability.o : $(obj)/capability_names.h
$(obj)/net.o : $(obj)/net_names.h
$(obj)/resource.o : $(obj)/rlim_names.h
$(obj)/capability_names.h : $(srctree)/include/uapi/linux/capability.h \
$(src)/Makefile
@@ -68,3 +101,8 @@ $(obj)/capability_names.h : $(srctree)/include/uapi/linux/capability.h \
$(obj)/rlim_names.h : $(srctree)/include/uapi/asm-generic/resource.h \
$(src)/Makefile
$(call cmd,make-rlim)
$(obj)/net_names.h : $(srctree)/include/linux/socket.h \
$(srctree)/include/linux/net.h \
$(src)/Makefile
$(call cmd,make-af)
$(call cmd,make-sock)

View File

@@ -800,13 +800,27 @@ static struct aa_fs_entry aa_fs_entry_domain[] = {
static struct aa_fs_entry aa_fs_entry_policy[] = {
AA_FS_FILE_BOOLEAN("set_load", 1),
{}
{ }
};
static struct aa_fs_entry aa_fs_entry_mount[] = {
AA_FS_FILE_STRING("mask", "mount umount"),
{ }
};
static struct aa_fs_entry aa_fs_entry_namespaces[] = {
AA_FS_FILE_BOOLEAN("profile", 1),
AA_FS_FILE_BOOLEAN("pivot_root", 1),
{ }
};
static struct aa_fs_entry aa_fs_entry_features[] = {
AA_FS_DIR("policy", aa_fs_entry_policy),
AA_FS_DIR("domain", aa_fs_entry_domain),
AA_FS_DIR("file", aa_fs_entry_file),
AA_FS_DIR("network", aa_fs_entry_network),
AA_FS_DIR("mount", aa_fs_entry_mount),
AA_FS_DIR("namespaces", aa_fs_entry_namespaces),
AA_FS_FILE_U64("capability", VFS_CAP_FLAGS_MASK),
AA_FS_DIR("rlimit", aa_fs_entry_rlimit),
AA_FS_DIR("caps", aa_fs_entry_caps),

View File

@@ -44,6 +44,10 @@ const char *const op_table[] = {
"file_mmap",
"file_mprotect",
"pivotroot",
"mount",
"umount",
"create",
"post_create",
"bind",

View File

@@ -236,7 +236,7 @@ static const char *next_name(int xtype, const char *name)
*
* Returns: refcounted profile, or NULL on failure (MAYBE NULL)
*/
static struct aa_profile *x_table_lookup(struct aa_profile *profile, u32 xindex)
struct aa_profile *x_table_lookup(struct aa_profile *profile, u32 xindex)
{
struct aa_profile *new_profile = NULL;
struct aa_namespace *ns = profile->ns;

View File

@@ -30,8 +30,9 @@
#define AA_CLASS_NET 4
#define AA_CLASS_RLIMITS 5
#define AA_CLASS_DOMAIN 6
#define AA_CLASS_MOUNT 7
#define AA_CLASS_LAST AA_CLASS_DOMAIN
#define AA_CLASS_LAST AA_CLASS_MOUNT
/* Control parameters settable through module/boot flags */
extern enum audit_mode aa_g_audit;

View File

@@ -72,6 +72,10 @@ enum aa_ops {
OP_FMMAP,
OP_FMPROT,
OP_PIVOTROOT,
OP_MOUNT,
OP_UMOUNT,
OP_CREATE,
OP_POST_CREATE,
OP_BIND,
@@ -119,12 +123,23 @@ struct apparmor_audit_data {
int rlim;
unsigned long max;
} rlim;
struct {
const char *src_name;
const char *type;
const char *trans;
const char *data;
unsigned long flags;
} mnt;
struct {
const char *target;
u32 request;
u32 denied;
kuid_t ouid;
} fs;
struct {
int type, protocol;
struct sock *sk;
} net;
};
};

View File

@@ -23,6 +23,8 @@ struct aa_domain {
char **table;
};
struct aa_profile *x_table_lookup(struct aa_profile *profile, u32 xindex);
int apparmor_bprm_set_creds(struct linux_binprm *bprm);
int apparmor_bprm_secureexec(struct linux_binprm *bprm);
void apparmor_bprm_committing_creds(struct linux_binprm *bprm);

View File

@@ -0,0 +1,54 @@
/*
* AppArmor security module
*
* This file contains AppArmor file mediation function definitions.
*
* Copyright 2012 Canonical Ltd.
*
* 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 of the
* License.
*/
#ifndef __AA_MOUNT_H
#define __AA_MOUNT_H
#include <linux/fs.h>
#include <linux/path.h>
#include "domain.h"
#include "policy.h"
/* mount perms */
#define AA_MAY_PIVOTROOT 0x01
#define AA_MAY_MOUNT 0x02
#define AA_MAY_UMOUNT 0x04
#define AA_AUDIT_DATA 0x40
#define AA_CONT_MATCH 0x40
#define AA_MS_IGNORE_MASK (MS_KERNMOUNT | MS_NOSEC | MS_ACTIVE | MS_BORN)
int aa_remount(struct aa_profile *profile, const struct path *path,
unsigned long flags, void *data);
int aa_bind_mount(struct aa_profile *profile, const struct path *path,
const char *old_name, unsigned long flags);
int aa_mount_change_type(struct aa_profile *profile, const struct path *path,
unsigned long flags);
int aa_move_mount(struct aa_profile *profile, const struct path *path,
const char *old_name);
int aa_new_mount(struct aa_profile *profile, const char *dev_name,
const struct path *path, const char *type, unsigned long flags,
void *data);
int aa_umount(struct aa_profile *profile, struct vfsmount *mnt, int flags);
int aa_pivotroot(struct aa_profile *profile, const struct path *old_path,
const struct path *new_path);
#endif /* __AA_MOUNT_H */

View File

@@ -0,0 +1,44 @@
/*
* AppArmor security module
*
* This file contains AppArmor network mediation definitions.
*
* Copyright (C) 1998-2008 Novell/SUSE
* Copyright 2009-2012 Canonical Ltd.
*
* 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 of the
* License.
*/
#ifndef __AA_NET_H
#define __AA_NET_H
#include <net/sock.h>
#include "apparmorfs.h"
/* struct aa_net - network confinement data
* @allowed: basic network families permissions
* @audit_network: which network permissions to force audit
* @quiet_network: which network permissions to quiet rejects
*/
struct aa_net {
u16 allow[AF_MAX];
u16 audit[AF_MAX];
u16 quiet[AF_MAX];
};
extern struct aa_fs_entry aa_fs_entry_network[];
extern int aa_net_perm(int op, struct aa_profile *profile, u16 family,
int type, int protocol, struct sock *sk);
extern int aa_revalidate_sk(int op, struct sock *sk);
static inline void aa_free_net_rules(struct aa_net *new)
{
/* NOP */
}
#endif /* __AA_NET_H */

View File

@@ -27,6 +27,7 @@
#include "capability.h"
#include "domain.h"
#include "file.h"
#include "net.h"
#include "resource.h"
extern const char *const aa_profile_mode_names[];
@@ -176,6 +177,7 @@ struct aa_replacedby {
* @policy: general match rules governing policy
* @file: The set of rules governing basic file access and domain transitions
* @caps: capabilities for the profile
* @net: network controls for the profile
* @rlimits: rlimits for the profile
*
* @dents: dentries for the profiles file entries in apparmorfs
@@ -217,6 +219,7 @@ struct aa_profile {
struct aa_policydb policy;
struct aa_file_rules file;
struct aa_caps caps;
struct aa_net net;
struct aa_rlimit rlimits;
unsigned char *hash;

View File

@@ -32,9 +32,11 @@
#include "include/context.h"
#include "include/file.h"
#include "include/ipc.h"
#include "include/net.h"
#include "include/path.h"
#include "include/policy.h"
#include "include/procattr.h"
#include "include/mount.h"
/* Flag indicating whether initialization completed */
int apparmor_initialized __initdata;
@@ -468,6 +470,61 @@ static int apparmor_file_mprotect(struct vm_area_struct *vma,
!(vma->vm_flags & VM_SHARED) ? MAP_PRIVATE : 0);
}
static int apparmor_sb_mount(const char *dev_name, const struct path *path,
const char *type, unsigned long flags, void *data)
{
struct aa_profile *profile;
int error = 0;
/* Discard magic */
if ((flags & MS_MGC_MSK) == MS_MGC_VAL)
flags &= ~MS_MGC_MSK;
flags &= ~AA_MS_IGNORE_MASK;
profile = __aa_current_profile();
if (!unconfined(profile)) {
if (flags & MS_REMOUNT)
error = aa_remount(profile, path, flags, data);
else if (flags & MS_BIND)
error = aa_bind_mount(profile, path, dev_name, flags);
else if (flags & (MS_SHARED | MS_PRIVATE | MS_SLAVE |
MS_UNBINDABLE))
error = aa_mount_change_type(profile, path, flags);
else if (flags & MS_MOVE)
error = aa_move_mount(profile, path, dev_name);
else
error = aa_new_mount(profile, dev_name, path, type,
flags, data);
}
return error;
}
static int apparmor_sb_umount(struct vfsmount *mnt, int flags)
{
struct aa_profile *profile;
int error = 0;
profile = __aa_current_profile();
if (!unconfined(profile))
error = aa_umount(profile, mnt, flags);
return error;
}
static int apparmor_sb_pivotroot(const struct path *old_path,
const struct path *new_path)
{
struct aa_profile *profile;
int error = 0;
profile = __aa_current_profile();
if (!unconfined(profile))
error = aa_pivotroot(profile, old_path, new_path);
return error;
}
static int apparmor_getprocattr(struct task_struct *task, char *name,
char **value)
{
@@ -584,12 +641,114 @@ static int apparmor_task_setrlimit(struct task_struct *task,
return error;
}
static int apparmor_socket_create(int family, int type, int protocol, int kern)
{
struct aa_profile *profile;
int error = 0;
if (kern)
return 0;
profile = __aa_current_profile();
if (!unconfined(profile))
error = aa_net_perm(OP_CREATE, profile, family, type, protocol,
NULL);
return error;
}
static int apparmor_socket_bind(struct socket *sock,
struct sockaddr *address, int addrlen)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_BIND, sk);
}
static int apparmor_socket_connect(struct socket *sock,
struct sockaddr *address, int addrlen)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_CONNECT, sk);
}
static int apparmor_socket_listen(struct socket *sock, int backlog)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_LISTEN, sk);
}
static int apparmor_socket_accept(struct socket *sock, struct socket *newsock)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_ACCEPT, sk);
}
static int apparmor_socket_sendmsg(struct socket *sock,
struct msghdr *msg, int size)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_SENDMSG, sk);
}
static int apparmor_socket_recvmsg(struct socket *sock,
struct msghdr *msg, int size, int flags)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_RECVMSG, sk);
}
static int apparmor_socket_getsockname(struct socket *sock)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_GETSOCKNAME, sk);
}
static int apparmor_socket_getpeername(struct socket *sock)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_GETPEERNAME, sk);
}
static int apparmor_socket_getsockopt(struct socket *sock, int level,
int optname)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_GETSOCKOPT, sk);
}
static int apparmor_socket_setsockopt(struct socket *sock, int level,
int optname)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_SETSOCKOPT, sk);
}
static int apparmor_socket_shutdown(struct socket *sock, int how)
{
struct sock *sk = sock->sk;
return aa_revalidate_sk(OP_SOCK_SHUTDOWN, sk);
}
static struct security_hook_list apparmor_hooks[] = {
LSM_HOOK_INIT(ptrace_access_check, apparmor_ptrace_access_check),
LSM_HOOK_INIT(ptrace_traceme, apparmor_ptrace_traceme),
LSM_HOOK_INIT(capget, apparmor_capget),
LSM_HOOK_INIT(capable, apparmor_capable),
LSM_HOOK_INIT(sb_mount, apparmor_sb_mount),
LSM_HOOK_INIT(sb_umount, apparmor_sb_umount),
LSM_HOOK_INIT(sb_pivotroot, apparmor_sb_pivotroot),
LSM_HOOK_INIT(path_link, apparmor_path_link),
LSM_HOOK_INIT(path_unlink, apparmor_path_unlink),
LSM_HOOK_INIT(path_symlink, apparmor_path_symlink),
@@ -613,6 +772,19 @@ static struct security_hook_list apparmor_hooks[] = {
LSM_HOOK_INIT(getprocattr, apparmor_getprocattr),
LSM_HOOK_INIT(setprocattr, apparmor_setprocattr),
LSM_HOOK_INIT(socket_create, apparmor_socket_create),
LSM_HOOK_INIT(socket_bind, apparmor_socket_bind),
LSM_HOOK_INIT(socket_connect, apparmor_socket_connect),
LSM_HOOK_INIT(socket_listen, apparmor_socket_listen),
LSM_HOOK_INIT(socket_accept, apparmor_socket_accept),
LSM_HOOK_INIT(socket_sendmsg, apparmor_socket_sendmsg),
LSM_HOOK_INIT(socket_recvmsg, apparmor_socket_recvmsg),
LSM_HOOK_INIT(socket_getsockname, apparmor_socket_getsockname),
LSM_HOOK_INIT(socket_getpeername, apparmor_socket_getpeername),
LSM_HOOK_INIT(socket_getsockopt, apparmor_socket_getsockopt),
LSM_HOOK_INIT(socket_setsockopt, apparmor_socket_setsockopt),
LSM_HOOK_INIT(socket_shutdown, apparmor_socket_shutdown),
LSM_HOOK_INIT(cred_alloc_blank, apparmor_cred_alloc_blank),
LSM_HOOK_INIT(cred_free, apparmor_cred_free),
LSM_HOOK_INIT(cred_prepare, apparmor_cred_prepare),

620
security/apparmor/mount.c Normal file
View File

@@ -0,0 +1,620 @@
/*
* AppArmor security module
*
* This file contains AppArmor mediation of files
*
* Copyright (C) 1998-2008 Novell/SUSE
* Copyright 2009-2012 Canonical Ltd.
*
* 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 of the
* License.
*/
#include <linux/fs.h>
#include <linux/mount.h>
#include <linux/namei.h>
#include "include/apparmor.h"
#include "include/audit.h"
#include "include/context.h"
#include "include/domain.h"
#include "include/file.h"
#include "include/match.h"
#include "include/mount.h"
#include "include/path.h"
#include "include/policy.h"
static void audit_mnt_flags(struct audit_buffer *ab, unsigned long flags)
{
if (flags & MS_RDONLY)
audit_log_format(ab, "ro");
else
audit_log_format(ab, "rw");
if (flags & MS_NOSUID)
audit_log_format(ab, ", nosuid");
if (flags & MS_NODEV)
audit_log_format(ab, ", nodev");
if (flags & MS_NOEXEC)
audit_log_format(ab, ", noexec");
if (flags & MS_SYNCHRONOUS)
audit_log_format(ab, ", sync");
if (flags & MS_REMOUNT)
audit_log_format(ab, ", remount");
if (flags & MS_MANDLOCK)
audit_log_format(ab, ", mand");
if (flags & MS_DIRSYNC)
audit_log_format(ab, ", dirsync");
if (flags & MS_NOATIME)
audit_log_format(ab, ", noatime");
if (flags & MS_NODIRATIME)
audit_log_format(ab, ", nodiratime");
if (flags & MS_BIND)
audit_log_format(ab, flags & MS_REC ? ", rbind" : ", bind");
if (flags & MS_MOVE)
audit_log_format(ab, ", move");
if (flags & MS_SILENT)
audit_log_format(ab, ", silent");
if (flags & MS_POSIXACL)
audit_log_format(ab, ", acl");
if (flags & MS_UNBINDABLE)
audit_log_format(ab, flags & MS_REC ? ", runbindable" :
", unbindable");
if (flags & MS_PRIVATE)
audit_log_format(ab, flags & MS_REC ? ", rprivate" :
", private");
if (flags & MS_SLAVE)
audit_log_format(ab, flags & MS_REC ? ", rslave" :
", slave");
if (flags & MS_SHARED)
audit_log_format(ab, flags & MS_REC ? ", rshared" :
", shared");
if (flags & MS_RELATIME)
audit_log_format(ab, ", relatime");
if (flags & MS_I_VERSION)
audit_log_format(ab, ", iversion");
if (flags & MS_STRICTATIME)
audit_log_format(ab, ", strictatime");
if (flags & MS_NOUSER)
audit_log_format(ab, ", nouser");
}
/**
* audit_cb - call back for mount specific audit fields
* @ab: audit_buffer (NOT NULL)
* @va: audit struct to audit values of (NOT NULL)
*/
static void audit_cb(struct audit_buffer *ab, void *va)
{
struct common_audit_data *sa = va;
if (sa->aad->mnt.type) {
audit_log_format(ab, " fstype=");
audit_log_untrustedstring(ab, sa->aad->mnt.type);
}
if (sa->aad->mnt.src_name) {
audit_log_format(ab, " srcname=");
audit_log_untrustedstring(ab, sa->aad->mnt.src_name);
}
if (sa->aad->mnt.trans) {
audit_log_format(ab, " trans=");
audit_log_untrustedstring(ab, sa->aad->mnt.trans);
}
if (sa->aad->mnt.flags || sa->aad->op == OP_MOUNT) {
audit_log_format(ab, " flags=\"");
audit_mnt_flags(ab, sa->aad->mnt.flags);
audit_log_format(ab, "\"");
}
if (sa->aad->mnt.data) {
audit_log_format(ab, " options=");
audit_log_untrustedstring(ab, sa->aad->mnt.data);
}
}
/**
* audit_mount - handle the auditing of mount operations
* @profile: the profile being enforced (NOT NULL)
* @gfp: allocation flags
* @op: operation being mediated (NOT NULL)
* @name: name of object being mediated (MAYBE NULL)
* @src_name: src_name of object being mediated (MAYBE_NULL)
* @type: type of filesystem (MAYBE_NULL)
* @trans: name of trans (MAYBE NULL)
* @flags: filesystem idependent mount flags
* @data: filesystem mount flags
* @request: permissions requested
* @perms: the permissions computed for the request (NOT NULL)
* @info: extra information message (MAYBE NULL)
* @error: 0 if operation allowed else failure error code
*
* Returns: %0 or error on failure
*/
static int audit_mount(struct aa_profile *profile, gfp_t gfp, int op,
const char *name, const char *src_name,
const char *type, const char *trans,
unsigned long flags, const void *data, u32 request,
struct file_perms *perms, const char *info, int error)
{
int audit_type = AUDIT_APPARMOR_AUTO;
struct common_audit_data sa = { };
struct apparmor_audit_data aad = { };
if (likely(!error)) {
u32 mask = perms->audit;
if (unlikely(AUDIT_MODE(profile) == AUDIT_ALL))
mask = 0xffff;
/* mask off perms that are not being force audited */
request &= mask;
if (likely(!request))
return 0;
audit_type = AUDIT_APPARMOR_AUDIT;
} else {
/* only report permissions that were denied */
request = request & ~perms->allow;
if (request & perms->kill)
audit_type = AUDIT_APPARMOR_KILL;
/* quiet known rejects, assumes quiet and kill do not overlap */
if ((request & perms->quiet) &&
AUDIT_MODE(profile) != AUDIT_NOQUIET &&
AUDIT_MODE(profile) != AUDIT_ALL)
request &= ~perms->quiet;
if (!request)
return COMPLAIN_MODE(profile) ?
complain_error(error) : error;
}
sa.type = LSM_AUDIT_DATA_NONE;
sa.aad = &aad;
sa.aad->op = op;
sa.aad->name = name;
sa.aad->mnt.src_name = src_name;
sa.aad->mnt.type = type;
sa.aad->mnt.trans = trans;
sa.aad->mnt.flags = flags;
if (data && (perms->audit & AA_AUDIT_DATA))
sa.aad->mnt.data = data;
sa.aad->info = info;
sa.aad->error = error;
return aa_audit(audit_type, profile, gfp, &sa, audit_cb);
}
/**
* match_mnt_flags - Do an ordered match on mount flags
* @dfa: dfa to match against
* @state: state to start in
* @flags: mount flags to match against
*
* Mount flags are encoded as an ordered match. This is done instead of
* checking against a simple bitmask, to allow for logical operations
* on the flags.
*
* Returns: next state after flags match
*/
static unsigned int match_mnt_flags(struct aa_dfa *dfa, unsigned int state,
unsigned long flags)
{
unsigned int i;
for (i = 0; i <= 31 ; ++i) {
if ((1 << i) & flags)
state = aa_dfa_next(dfa, state, i + 1);
}
return state;
}
/**
* compute_mnt_perms - compute mount permission associated with @state
* @dfa: dfa to match against (NOT NULL)
* @state: state match finished in
*
* Returns: mount permissions
*/
static struct file_perms compute_mnt_perms(struct aa_dfa *dfa,
unsigned int state)
{
struct file_perms perms;
perms.kill = 0;
perms.allow = dfa_user_allow(dfa, state);
perms.audit = dfa_user_audit(dfa, state);
perms.quiet = dfa_user_quiet(dfa, state);
perms.xindex = dfa_user_xindex(dfa, state);
return perms;
}
static const char const *mnt_info_table[] = {
"match succeeded",
"failed mntpnt match",
"failed srcname match",
"failed type match",
"failed flags match",
"failed data match"
};
/*
* Returns 0 on success else element that match failed in, this is the
* index into the mnt_info_table above
*/
static int do_match_mnt(struct aa_dfa *dfa, unsigned int start,
const char *mntpnt, const char *devname,
const char *type, unsigned long flags,
void *data, bool binary, struct file_perms *perms)
{
unsigned int state;
state = aa_dfa_match(dfa, start, mntpnt);
state = aa_dfa_null_transition(dfa, state);
if (!state)
return 1;
if (devname)
state = aa_dfa_match(dfa, state, devname);
state = aa_dfa_null_transition(dfa, state);
if (!state)
return 2;
if (type)
state = aa_dfa_match(dfa, state, type);
state = aa_dfa_null_transition(dfa, state);
if (!state)
return 3;
state = match_mnt_flags(dfa, state, flags);
if (!state)
return 4;
*perms = compute_mnt_perms(dfa, state);
if (perms->allow & AA_MAY_MOUNT)
return 0;
/* only match data if not binary and the DFA flags data is expected */
if (data && !binary && (perms->allow & AA_CONT_MATCH)) {
state = aa_dfa_null_transition(dfa, state);
if (!state)
return 4;
state = aa_dfa_match(dfa, state, data);
if (!state)
return 5;
*perms = compute_mnt_perms(dfa, state);
if (perms->allow & AA_MAY_MOUNT)
return 0;
}
/* failed at end of flags match */
return 4;
}
/**
* match_mnt - handle path matching for mount
* @profile: the confining profile
* @mntpnt: string for the mntpnt (NOT NULL)
* @devname: string for the devname/src_name (MAYBE NULL)
* @type: string for the dev type (MAYBE NULL)
* @flags: mount flags to match
* @data: fs mount data (MAYBE NULL)
* @binary: whether @data is binary
* @perms: Returns: permission found by the match
* @info: Returns: infomation string about the match for logging
*
* Returns: 0 on success else error
*/
static int match_mnt(struct aa_profile *profile, const char *mntpnt,
const char *devname, const char *type,
unsigned long flags, void *data, bool binary,
struct file_perms *perms, const char **info)
{
int pos;
if (!profile->policy.dfa)
return -EACCES;
pos = do_match_mnt(profile->policy.dfa,
profile->policy.start[AA_CLASS_MOUNT],
mntpnt, devname, type, flags, data, binary, perms);
if (pos) {
*info = mnt_info_table[pos];
return -EACCES;
}
return 0;
}
static int path_flags(struct aa_profile *profile, const struct path *path)
{
return profile->path_flags |
S_ISDIR(path->dentry->d_inode->i_mode) ? PATH_IS_DIR : 0;
}
int aa_remount(struct aa_profile *profile, const struct path *path,
unsigned long flags, void *data)
{
struct file_perms perms = { };
const char *name, *info = NULL;
char *buffer = NULL;
int binary, error;
binary = path->dentry->d_sb->s_type->fs_flags & FS_BINARY_MOUNTDATA;
error = aa_path_name(path, path_flags(profile, path), &buffer, &name,
&info);
if (error)
goto audit;
error = match_mnt(profile, name, NULL, NULL, flags, data, binary,
&perms, &info);
audit:
error = audit_mount(profile, GFP_KERNEL, OP_MOUNT, name, NULL, NULL,
NULL, flags, data, AA_MAY_MOUNT, &perms, info,
error);
kfree(buffer);
return error;
}
int aa_bind_mount(struct aa_profile *profile, const struct path *path,
const char *dev_name, unsigned long flags)
{
struct file_perms perms = { };
char *buffer = NULL, *old_buffer = NULL;
const char *name, *old_name = NULL, *info = NULL;
struct path old_path;
int error;
if (!dev_name || !*dev_name)
return -EINVAL;
flags &= MS_REC | MS_BIND;
error = aa_path_name(path, path_flags(profile, path), &buffer, &name,
&info);
if (error)
goto audit;
error = kern_path(dev_name, LOOKUP_FOLLOW|LOOKUP_AUTOMOUNT, &old_path);
if (error)
goto audit;
error = aa_path_name(&old_path, path_flags(profile, &old_path),
&old_buffer, &old_name, &info);
path_put(&old_path);
if (error)
goto audit;
error = match_mnt(profile, name, old_name, NULL, flags, NULL, 0,
&perms, &info);
audit:
error = audit_mount(profile, GFP_KERNEL, OP_MOUNT, name, old_name,
NULL, NULL, flags, NULL, AA_MAY_MOUNT, &perms,
info, error);
kfree(buffer);
kfree(old_buffer);
return error;
}
int aa_mount_change_type(struct aa_profile *profile, const struct path *path,
unsigned long flags)
{
struct file_perms perms = { };
char *buffer = NULL;
const char *name, *info = NULL;
int error;
/* These are the flags allowed by do_change_type() */
flags &= (MS_REC | MS_SILENT | MS_SHARED | MS_PRIVATE | MS_SLAVE |
MS_UNBINDABLE);
error = aa_path_name(path, path_flags(profile, path), &buffer, &name,
&info);
if (error)
goto audit;
error = match_mnt(profile, name, NULL, NULL, flags, NULL, 0, &perms,
&info);
audit:
error = audit_mount(profile, GFP_KERNEL, OP_MOUNT, name, NULL, NULL,
NULL, flags, NULL, AA_MAY_MOUNT, &perms, info,
error);
kfree(buffer);
return error;
}
int aa_move_mount(struct aa_profile *profile, const struct path *path,
const char *orig_name)
{
struct file_perms perms = { };
char *buffer = NULL, *old_buffer = NULL;
const char *name, *old_name = NULL, *info = NULL;
struct path old_path;
int error;
if (!orig_name || !*orig_name)
return -EINVAL;
error = aa_path_name(path, path_flags(profile, path), &buffer, &name,
&info);
if (error)
goto audit;
error = kern_path(orig_name, LOOKUP_FOLLOW, &old_path);
if (error)
goto audit;
error = aa_path_name(&old_path, path_flags(profile, &old_path),
&old_buffer, &old_name, &info);
path_put(&old_path);
if (error)
goto audit;
error = match_mnt(profile, name, old_name, NULL, MS_MOVE, NULL, 0,
&perms, &info);
audit:
error = audit_mount(profile, GFP_KERNEL, OP_MOUNT, name, old_name,
NULL, NULL, MS_MOVE, NULL, AA_MAY_MOUNT, &perms,
info, error);
kfree(buffer);
kfree(old_buffer);
return error;
}
int aa_new_mount(struct aa_profile *profile, const char *orig_dev_name,
const struct path *path, const char *type, unsigned long flags,
void *data)
{
struct file_perms perms = { };
char *buffer = NULL, *dev_buffer = NULL;
const char *name = NULL, *dev_name = NULL, *info = NULL;
int binary = 1;
int error;
dev_name = orig_dev_name;
if (type) {
int requires_dev;
struct file_system_type *fstype = get_fs_type(type);
if (!fstype)
return -ENODEV;
binary = fstype->fs_flags & FS_BINARY_MOUNTDATA;
requires_dev = fstype->fs_flags & FS_REQUIRES_DEV;
put_filesystem(fstype);
if (requires_dev) {
struct path dev_path;
if (!dev_name || !*dev_name) {
error = -ENOENT;
goto out;
}
error = kern_path(dev_name, LOOKUP_FOLLOW, &dev_path);
if (error)
goto audit;
error = aa_path_name(&dev_path,
path_flags(profile, &dev_path),
&dev_buffer, &dev_name, &info);
path_put(&dev_path);
if (error)
goto audit;
}
}
error = aa_path_name(path, path_flags(profile, path), &buffer, &name,
&info);
if (error)
goto audit;
error = match_mnt(profile, name, dev_name, type, flags, data, binary,
&perms, &info);
audit:
error = audit_mount(profile, GFP_KERNEL, OP_MOUNT, name, dev_name,
type, NULL, flags, data, AA_MAY_MOUNT, &perms, info,
error);
kfree(buffer);
kfree(dev_buffer);
out:
return error;
}
int aa_umount(struct aa_profile *profile, struct vfsmount *mnt, int flags)
{
struct file_perms perms = { };
char *buffer = NULL;
const char *name, *info = NULL;
int error;
struct path path = { mnt, mnt->mnt_root };
error = aa_path_name(&path, path_flags(profile, &path), &buffer, &name,
&info);
if (error)
goto audit;
if (!error && profile->policy.dfa) {
unsigned int state;
state = aa_dfa_match(profile->policy.dfa,
profile->policy.start[AA_CLASS_MOUNT],
name);
perms = compute_mnt_perms(profile->policy.dfa, state);
}
if (AA_MAY_UMOUNT & ~perms.allow)
error = -EACCES;
audit:
error = audit_mount(profile, GFP_KERNEL, OP_UMOUNT, name, NULL, NULL,
NULL, 0, NULL, AA_MAY_UMOUNT, &perms, info, error);
kfree(buffer);
return error;
}
int aa_pivotroot(struct aa_profile *profile, const struct path *old_path,
const struct path *new_path)
{
struct file_perms perms = { };
struct aa_profile *target = NULL;
char *old_buffer = NULL, *new_buffer = NULL;
const char *old_name, *new_name = NULL, *info = NULL;
int error;
error = aa_path_name(old_path, path_flags(profile, old_path),
&old_buffer, &old_name, &info);
if (error)
goto audit;
error = aa_path_name(new_path, path_flags(profile, new_path),
&new_buffer, &new_name, &info);
if (error)
goto audit;
if (profile->policy.dfa) {
unsigned int state;
state = aa_dfa_match(profile->policy.dfa,
profile->policy.start[AA_CLASS_MOUNT],
new_name);
state = aa_dfa_null_transition(profile->policy.dfa, state);
state = aa_dfa_match(profile->policy.dfa, state, old_name);
perms = compute_mnt_perms(profile->policy.dfa, state);
}
if (AA_MAY_PIVOTROOT & perms.allow) {
if ((perms.xindex & AA_X_TYPE_MASK) == AA_X_TABLE) {
target = x_table_lookup(profile, perms.xindex);
if (!target)
error = -ENOENT;
else
error = aa_replace_current_profile(target);
}
} else
error = -EACCES;
audit:
error = audit_mount(profile, GFP_KERNEL, OP_PIVOTROOT, new_name,
old_name, NULL, target ? target->base.name : NULL,
0, NULL, AA_MAY_PIVOTROOT, &perms, info, error);
aa_put_profile(target);
kfree(old_buffer);
kfree(new_buffer);
return error;
}

162
security/apparmor/net.c Normal file
View File

@@ -0,0 +1,162 @@
/*
* AppArmor security module
*
* This file contains AppArmor network mediation
*
* Copyright (C) 1998-2008 Novell/SUSE
* Copyright 2009-2012 Canonical Ltd.
*
* 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 of the
* License.
*/
#include "include/apparmor.h"
#include "include/audit.h"
#include "include/context.h"
#include "include/net.h"
#include "include/policy.h"
#include "net_names.h"
struct aa_fs_entry aa_fs_entry_network[] = {
AA_FS_FILE_STRING("af_mask", AA_FS_AF_MASK),
{ }
};
/* audit callback for net specific fields */
static void audit_cb(struct audit_buffer *ab, void *va)
{
struct common_audit_data *sa = va;
audit_log_format(ab, " family=");
if (address_family_names[sa->u.net->family]) {
audit_log_string(ab, address_family_names[sa->u.net->family]);
} else {
audit_log_format(ab, "\"unknown(%d)\"", sa->u.net->family);
}
audit_log_format(ab, " sock_type=");
if (sock_type_names[sa->aad->net.type]) {
audit_log_string(ab, sock_type_names[sa->aad->net.type]);
} else {
audit_log_format(ab, "\"unknown(%d)\"", sa->aad->net.type);
}
audit_log_format(ab, " protocol=%d", sa->aad->net.protocol);
}
/**
* audit_net - audit network access
* @profile: profile being enforced (NOT NULL)
* @op: operation being checked
* @family: network family
* @type: network type
* @protocol: network protocol
* @sk: socket auditing is being applied to
* @error: error code for failure else 0
*
* Returns: %0 or sa->error else other errorcode on failure
*/
static int audit_net(struct aa_profile *profile, int op, u16 family, int type,
int protocol, struct sock *sk, int error)
{
int audit_type = AUDIT_APPARMOR_AUTO;
struct common_audit_data sa;
struct apparmor_audit_data aad = { };
struct lsm_network_audit net = { };
if (sk) {
sa.type = LSM_AUDIT_DATA_NET;
} else {
sa.type = LSM_AUDIT_DATA_NONE;
}
/* todo fill in socket addr info */
sa.aad = &aad;
sa.u.net = &net;
sa.aad->op = op,
sa.u.net->family = family;
sa.u.net->sk = sk;
sa.aad->net.type = type;
sa.aad->net.protocol = protocol;
sa.aad->error = error;
if (likely(!sa.aad->error)) {
u16 audit_mask = profile->net.audit[sa.u.net->family];
if (likely((AUDIT_MODE(profile) != AUDIT_ALL) &&
!(1 << sa.aad->net.type & audit_mask)))
return 0;
audit_type = AUDIT_APPARMOR_AUDIT;
} else {
u16 quiet_mask = profile->net.quiet[sa.u.net->family];
u16 kill_mask = 0;
u16 denied = (1 << sa.aad->net.type);
if (denied & kill_mask)
audit_type = AUDIT_APPARMOR_KILL;
if ((denied & quiet_mask) &&
AUDIT_MODE(profile) != AUDIT_NOQUIET &&
AUDIT_MODE(profile) != AUDIT_ALL)
return COMPLAIN_MODE(profile) ? 0 : sa.aad->error;
}
return aa_audit(audit_type, profile, GFP_KERNEL, &sa, audit_cb);
}
/**
* aa_net_perm - very course network access check
* @op: operation being checked
* @profile: profile being enforced (NOT NULL)
* @family: network family
* @type: network type
* @protocol: network protocol
*
* Returns: %0 else error if permission denied
*/
int aa_net_perm(int op, struct aa_profile *profile, u16 family, int type,
int protocol, struct sock *sk)
{
u16 family_mask;
int error;
if ((family < 0) || (family >= AF_MAX))
return -EINVAL;
if ((type < 0) || (type >= SOCK_MAX))
return -EINVAL;
/* unix domain and netlink sockets are handled by ipc */
if (family == AF_UNIX || family == AF_NETLINK)
return 0;
family_mask = profile->net.allow[family];
error = (family_mask & (1 << type)) ? 0 : -EACCES;
return audit_net(profile, op, family, type, protocol, sk, error);
}
/**
* aa_revalidate_sk - Revalidate access to a sock
* @op: operation being checked
* @sk: sock being revalidated (NOT NULL)
*
* Returns: %0 else error if permission denied
*/
int aa_revalidate_sk(int op, struct sock *sk)
{
struct aa_profile *profile;
int error = 0;
/* aa_revalidate_sk should not be called from interrupt context
* don't mediate these calls as they are not task related
*/
if (in_interrupt())
return 0;
profile = __aa_current_profile();
if (!unconfined(profile))
error = aa_net_perm(op, profile, sk->sk_family, sk->sk_type,
sk->sk_protocol, sk);
return error;
}

View File

@@ -603,6 +603,7 @@ void aa_free_profile(struct aa_profile *profile)
aa_free_file_rules(&profile->file);
aa_free_cap_rules(&profile->caps);
aa_free_net_rules(&profile->net);
aa_free_rlimit_rules(&profile->rlimits);
kzfree(profile->dirname);

View File

@@ -193,6 +193,19 @@ fail:
return 0;
}
static bool unpack_u16(struct aa_ext *e, u16 *data, const char *name)
{
if (unpack_nameX(e, AA_U16, name)) {
if (!inbounds(e, sizeof(u16)))
return 0;
if (data)
*data = le16_to_cpu(get_unaligned((u16 *) e->pos));
e->pos += sizeof(u16);
return 1;
}
return 0;
}
static bool unpack_u32(struct aa_ext *e, u32 *data, const char *name)
{
if (unpack_nameX(e, AA_U32, name)) {
@@ -476,6 +489,7 @@ static struct aa_profile *unpack_profile(struct aa_ext *e)
{
struct aa_profile *profile = NULL;
const char *name = NULL;
size_t size = 0;
int i, error = -EPROTO;
kernel_cap_t tmpcap;
u32 tmp;
@@ -576,6 +590,38 @@ static struct aa_profile *unpack_profile(struct aa_ext *e)
if (!unpack_rlimits(e, profile))
goto fail;
size = unpack_array(e, "net_allowed_af");
if (size) {
for (i = 0; i < size; i++) {
/* discard extraneous rules that this kernel will
* never request
*/
if (i >= AF_MAX) {
u16 tmp;
if (!unpack_u16(e, &tmp, NULL) ||
!unpack_u16(e, &tmp, NULL) ||
!unpack_u16(e, &tmp, NULL))
goto fail;
continue;
}
if (!unpack_u16(e, &profile->net.allow[i], NULL))
goto fail;
if (!unpack_u16(e, &profile->net.audit[i], NULL))
goto fail;
if (!unpack_u16(e, &profile->net.quiet[i], NULL))
goto fail;
}
if (!unpack_nameX(e, AA_ARRAYEND, NULL))
goto fail;
}
/*
* allow unix domain and netlink sockets they are handled
* by IPC
*/
profile->net.allow[AF_UNIX] = 0xffff;
profile->net.allow[AF_NETLINK] = 0xffff;
if (unpack_nameX(e, AA_STRUCT, "policydb")) {
/* generic policy dfa - optional and may be NULL */
profile->policy.dfa = unpack_dfa(e);