linux-fsdevel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Ingo Molnar <mingo@elte.hu>
To: Kees Cook <keescook@chromium.org>
Cc: Andrew Morton <akpm@linux-foundation.org>,
	linux-kernel@vger.kernel.org, Randy Dunlap <rdunlap@xenotime.net>,
	Alexander Viro <viro@zeniv.linux.org.uk>,
	linux-doc@vger.kernel.org, linux-fsdevel@vger.kernel.org,
	kernel-hardening@lists.openwall.com
Subject: Re: [PATCH] fs: hardlink creation restrictions
Date: Sun, 19 Feb 2012 13:49:16 +0100	[thread overview]
Message-ID: <20120219124916.GB25900@elte.hu> (raw)
In-Reply-To: <20120218193857.GA30985@www.outflux.net>


* Kees Cook <keescook@chromium.org> wrote:

> This is the other half of link restrictions, now that symlink 
> restriction has landed in -mm.

Nice features!

> On systems that have user-writable directories on the same 
> partition as system files, a long-standing class of security 
> issues is the hardlink-based time-of-check-time-of-use race, 
> most commonly seen in world-writable directories like /tmp. 
> The common method of exploitation of this flaw is to cross 
> privilege boundaries when following a given hardlink (i.e. a 
> root process follows a hardlink created by another user). 
> Additionally, an issue exists where users can "pin" a 
> potentially vulnerable setuid/setgid file so that an 
> administrator will not actually upgrade a system fully.
> 
> The solution is to permit hardlinks to only be created when 
> the user is already the existing file's owner, or if they 
> already have read/write access to the existing file.
> 
> Many Linux users are surprised when they learn they can link 
> to files they have no access to, so this change appears to 
> follow the doctrine of "least surprise". Additionally, this 
> change does not violate POSIX, which states "the 
> implementation may require that the calling process has 
> permission to access the existing file"[1].
> 
> This change is known to break some implementations of the "at" 
> daemon, though the version used by Fedora and Ubuntu has been 
> fixed[2] for a while. Otherwise, the change has been 
> undisruptive while in use in Ubuntu for the last 1.5 years.
> 
> This patch is based on the patch in Openwall and grsecurity.  
> I have added a sysctl to enable the protected behavior, 
> documentation, and an audit notification.
> 
> [1] http://pubs.opengroup.org/onlinepubs/9699919799/functions/linkat.html
> [2] http://anonscm.debian.org/gitweb/?p=collab-maint/at.git;a=commitdiff;h=f4114656c3a6c6f6070e315ffdf940a49eda3279
> 
> Signed-off-by: Kees Cook <keescook@chromium.org>
> ---
>  Documentation/sysctl/fs.txt |   21 ++++++++++++
>  fs/Kconfig                  |   46 ++++++++++++++++++++-------
>  fs/namei.c                  |   72 ++++++++++++++++++++++++++++++++++++++++++-
>  include/linux/fs.h          |    1 +
>  kernel/sysctl.c             |   11 ++++++-
>  5 files changed, 137 insertions(+), 14 deletions(-)
> 
> diff --git a/Documentation/sysctl/fs.txt b/Documentation/sysctl/fs.txt
> index 4b47cd5..08458be 100644
> --- a/Documentation/sysctl/fs.txt
> +++ b/Documentation/sysctl/fs.txt
> @@ -32,6 +32,7 @@ Currently, these files are in /proc/sys/fs:
>  - nr_open
>  - overflowuid
>  - overflowgid
> +- protected_nonaccess_hardlinks
>  - protected_sticky_symlinks
>  - suid_dumpable
>  - super-max
> @@ -158,6 +159,26 @@ The default is 65534.
>  
>  ==============================================================
>  
> +protected_nonaccess_hardlinks:
> +
> +A long-standing class of security issues is the hardlink-based
> +time-of-check-time-of-use race, most commonly seen in world-writable
> +directories like /tmp. The common method of exploitation of this flaw
> +is to cross privilege boundaries when following a given hardlink (i.e. a
> +root process follows a hardlink created by another user). Additionally,
> +on systems without separated partitions, this stops unauthorized users
> +from "pinning" vulnerable setuid/setgid files against being upgraded by
> +the administrator.
> +
> +When set to "0", hardlink creation behavior is unrestricted.
> +
> +When set to "1" hardlinks cannot be created by users if they do not
> +already own the source file, or do not have read/write access to it.
> +
> +This protection is based on the restrictions in Openwall and grsecurity.
> +
> +==============================================================
> +
>  protected_sticky_symlinks:
>  
>  A long-standing class of security issues is the symlink-based
> diff --git a/fs/Kconfig b/fs/Kconfig
> index ca279b7..66f9334 100644
> --- a/fs/Kconfig
> +++ b/fs/Kconfig
> @@ -271,27 +271,29 @@ endif # NETWORK_FILESYSTEMS
>  source "fs/nls/Kconfig"
>  source "fs/dlm/Kconfig"
>  
> -config PROTECTED_STICKY_SYMLINKS
> -	bool "Evaluate vulnerable symlink conditions"
> -	default y
> +config PROTECTED_LINKS
> +	bool "Evaluate vulnerable link conditions"
> +	default n
>  	help
> -	  A long-standing class of security issues is the symlink-based
> +	  A long-standing class of security issues is the link-based
>  	  time-of-check-time-of-use race, most commonly seen in
>  	  world-writable directories like /tmp. The common method of
>  	  exploitation of this flaw is to cross privilege boundaries
> -	  when following a given symlink (i.e. a root process follows
> -	  a malicious symlink belonging to another user).
> +	  when following a given link (i.e. a root process follows
> +	  a malicious symlink belonging to another user, or a hardlink
> +	  created to a root-owned file).
>  
> -	  Enabling this adds the logic to examine these dangerous symlink
> -	  conditions. Whether or not the dangerous symlink situations are
> -	  allowed is controlled by PROTECTED_STICKY_SYMLINKS_ENABLED.
> +	  Enabling this adds the logic to examine these dangerous link
> +	  conditions. Whether or not the dangerous link situations are
> +	  allowed is controlled by PROTECTED_NONACCESS_HARDLINKS_ENABLED and
> +	  PROTECTED_STICKY_SYMLINKS_ENABLED.
>  
>  config PROTECTED_STICKY_SYMLINKS_ENABLED
> -	depends on PROTECTED_STICKY_SYMLINKS
> +	depends on PROTECTED_LINKS
>  	bool "Disallow symlink following in sticky world-writable dirs"
>  	default y
>  	help
> -	  Solve ToCToU symlink race vulnerablities by permitting symlinks
> +	  Solve ToCToU symlink race vulnerabilities by permitting symlinks
>  	  to be followed only when outside a sticky world-writable directory,
>  	  or when the uid of the symlink and follower match, or when the
>  	  directory and symlink owners match.
> @@ -300,9 +302,29 @@ config PROTECTED_STICKY_SYMLINKS_ENABLED
>  	  via /proc/sys/kernel/protected_sticky_symlinks.
>  
>  config PROTECTED_STICKY_SYMLINKS_ENABLED_SYSCTL
> -	depends on PROTECTED_STICKY_SYMLINKS
> +	depends on PROTECTED_LINKS
>  	int
>  	default "1" if PROTECTED_STICKY_SYMLINKS_ENABLED
>  	default "0"
>  
> +config PROTECTED_NONACCESS_HARDLINKS_ENABLED
> +	depends on PROTECTED_LINKS
> +	bool "Disallow hardlink creation to non-accessible files"
> +	default y
> +	help
> +	  Solve ToCToU hardlink race vulnerabilities by permitting hardlinks
> +	  to be created only when to a regular file that is owned by the user,
> +	  or is readable and writable by the user. Also blocks users from
> +	  "pinning" vulnerable setuid/setgid programs from being upgraded by
> +	  the administrator.
> +
> +	  When PROC_SYSCTL is enabled, this setting can also be controlled
> +	  via /proc/sys/kernel/protected_nonaccess_hardlinks.

I'd add a:

	See Documentation/sysctl/fs.txt for details.

line as well.

> +config PROTECTED_NONACCESS_HARDLINKS_ENABLED_SYSCTL
> +	depends on PROTECTED_LINKS
> +	int
> +	default "1" if PROTECTED_NONACCESS_HARDLINKS_ENABLED
> +	default "0"

Naming nit: how about dropping the _NONACCESS/_nonaccess names 
complete and turn it into protected_hardlinks? The longer 
variant is not any better descriptive, and needlessly longer.

The PROTECTED_SYMLINKS/PROTECTED_HARDLINKS naming is thus also 
nicely symmetric.

> +#ifdef CONFIG_AUDIT
> +	if (error) {
> +		struct audit_buffer *ab;
> +
> +		ab = audit_log_start(current->audit_context,
> +				     GFP_KERNEL, AUDIT_AVC);
> +		audit_log_format(ab, "op=linkat action=denied");
> +		audit_log_format(ab, " pid=%d comm=", current->pid);
> +		audit_log_untrustedstring(ab, current->comm);
> +		audit_log_d_path(ab, " path=", old_path);
> +		audit_log_format(ab, " dev=");
> +		audit_log_untrustedstring(ab, inode->i_sb->s_id);
> +		audit_log_format(ab, " ino=%lu", inode->i_ino);
> +		audit_log_end(ab);
> +	}
> +#endif

Small detail: don't these audit methods map to nothing on 
!CONFIG_AUDIT, allowing the #ifdef to be dropped? (if not then 
it should really be so.)

Otherwise:

Acked-by: Ingo Molnar <mingo@elte.hu>

Thanks,

	Ingo

  parent reply	other threads:[~2012-02-19 12:49 UTC|newest]

Thread overview: 6+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2012-02-18 19:38 [PATCH] fs: hardlink creation restrictions Kees Cook
2012-02-18 20:56 ` Marcin Slusarz
2012-02-19  1:36   ` Kees Cook
2012-02-19 12:49 ` Ingo Molnar [this message]
2012-02-19 17:45   ` Kees Cook
2012-02-20  8:22     ` Ingo Molnar

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20120219124916.GB25900@elte.hu \
    --to=mingo@elte.hu \
    --cc=akpm@linux-foundation.org \
    --cc=keescook@chromium.org \
    --cc=kernel-hardening@lists.openwall.com \
    --cc=linux-doc@vger.kernel.org \
    --cc=linux-fsdevel@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=rdunlap@xenotime.net \
    --cc=viro@zeniv.linux.org.uk \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).