Commit d166c802 authored by Casey Schaufler's avatar Casey Schaufler

Smack: Bring-up access mode

People keep asking me for permissive mode, and I keep saying "no".

Permissive mode is wrong for more reasons than I can enumerate,
but the compelling one is that it's once on, never off.

Nonetheless, there is an argument to be made for running a
process with lots of permissions, logging which are required,
and then locking the process down. There wasn't a way to do
that with Smack, but this provides it.

The notion is that you start out by giving the process an
appropriate Smack label, such as "ATBirds". You create rules
with a wide range of access and the "b" mode. On Tizen it
might be:

	ATBirds	System	rwxalb
	ATBirds	User	rwxalb
	ATBirds	_	rwxalb
	User	ATBirds	wb
	System	ATBirds	wb

Accesses that fail will generate audit records. Accesses
that succeed because of rules marked with a "b" generate
log messages identifying the rule, the program and as much
object information as is convenient.

When the system is properly configured and the programs
brought in line with the labeling scheme the "b" mode can
be removed from the rules. When the system is ready for
production the facility can be configured out.

This provides the developer the convenience of permissive
mode without creating a system that looks like it is
enforcing a policy while it is not.
Signed-off-by: default avatarCasey Schaufler <casey@schaufler-ca.com>
parent d83d2c26
...@@ -12,3 +12,19 @@ config SECURITY_SMACK ...@@ -12,3 +12,19 @@ config SECURITY_SMACK
of other mandatory security schemes. of other mandatory security schemes.
If you are unsure how to answer this question, answer N. If you are unsure how to answer this question, answer N.
config SECURITY_SMACK_BRINGUP
bool "Reporting on access granted by Smack rules"
depends on SECURITY_SMACK
default n
help
Enable the bring-up ("b") access mode in Smack rules.
When access is granted by a rule with the "b" mode a
message about the access requested is generated. The
intention is that a process can be granted a wide set
of access initially with the bringup mode set on the
rules. The developer can use the information to
identify which rules are necessary and what accesses
may be inappropriate. The developer can reduce the
access rule set once the behavior is well understood.
This is a superior mechanism to the oft abused
"permissive" mode of other systems.
...@@ -191,6 +191,7 @@ struct smk_port_label { ...@@ -191,6 +191,7 @@ struct smk_port_label {
*/ */
#define MAY_TRANSMUTE 0x00001000 /* Controls directory labeling */ #define MAY_TRANSMUTE 0x00001000 /* Controls directory labeling */
#define MAY_LOCK 0x00002000 /* Locks should be writes, but ... */ #define MAY_LOCK 0x00002000 /* Locks should be writes, but ... */
#define MAY_BRINGUP 0x00004000 /* Report use of this rule */
/* /*
* Just to make the common cases easier to deal with * Just to make the common cases easier to deal with
...@@ -200,9 +201,9 @@ struct smk_port_label { ...@@ -200,9 +201,9 @@ struct smk_port_label {
#define MAY_NOT 0 #define MAY_NOT 0
/* /*
* Number of access types used by Smack (rwxatl) * Number of access types used by Smack (rwxatlb)
*/ */
#define SMK_NUM_ACCESS_TYPE 6 #define SMK_NUM_ACCESS_TYPE 7
/* SMACK data */ /* SMACK data */
struct smack_audit_data { struct smack_audit_data {
......
...@@ -178,16 +178,27 @@ int smk_access(struct smack_known *subject_known, char *object_label, ...@@ -178,16 +178,27 @@ int smk_access(struct smack_known *subject_known, char *object_label,
&subject_known->smk_rules); &subject_known->smk_rules);
rcu_read_unlock(); rcu_read_unlock();
if (may > 0 && (request & may) == request) if (may <= 0 || (request & may) != request) {
rc = -EACCES;
goto out_audit; goto out_audit;
}
#ifdef CONFIG_SECURITY_SMACK_BRINGUP
/*
* Return a positive value if using bringup mode.
* This allows the hooks to identify checks that
* succeed because of "b" rules.
*/
if (may & MAY_BRINGUP)
rc = MAY_BRINGUP;
#endif
rc = -EACCES;
out_audit: out_audit:
#ifdef CONFIG_AUDIT #ifdef CONFIG_AUDIT
if (a) if (a)
smack_log(subject_known->smk_known, object_label, request, smack_log(subject_known->smk_known, object_label, request,
rc, a); rc, a);
#endif #endif
return rc; return rc;
} }
...@@ -214,7 +225,7 @@ int smk_tskacc(struct task_smack *subject, char *obj_label, ...@@ -214,7 +225,7 @@ int smk_tskacc(struct task_smack *subject, char *obj_label,
* Check the global rule list * Check the global rule list
*/ */
rc = smk_access(skp, obj_label, mode, NULL); rc = smk_access(skp, obj_label, mode, NULL);
if (rc == 0) { if (rc >= 0) {
/* /*
* If there is an entry in the task's rule list * If there is an entry in the task's rule list
* it can further restrict access. * it can further restrict access.
...@@ -328,6 +339,13 @@ void smack_log(char *subject_label, char *object_label, int request, ...@@ -328,6 +339,13 @@ void smack_log(char *subject_label, char *object_label, int request,
struct smack_audit_data *sad; struct smack_audit_data *sad;
struct common_audit_data *a = &ad->a; struct common_audit_data *a = &ad->a;
#ifdef CONFIG_SECURITY_SMACK_BRINGUP
/*
* The result may be positive in bringup mode.
*/
if (result > 0)
result = 0;
#endif
/* check if we have to log the current event */ /* check if we have to log the current event */
if (result != 0 && (log_policy & SMACK_AUDIT_DENIED) == 0) if (result != 0 && (log_policy & SMACK_AUDIT_DENIED) == 0)
return; return;
......
This diff is collapsed.
...@@ -304,6 +304,10 @@ static int smk_perm_from_str(const char *string) ...@@ -304,6 +304,10 @@ static int smk_perm_from_str(const char *string)
case 'L': case 'L':
perm |= MAY_LOCK; perm |= MAY_LOCK;
break; break;
case 'b':
case 'B':
perm |= MAY_BRINGUP;
break;
default: default:
return perm; return perm;
} }
...@@ -616,6 +620,8 @@ static void smk_rule_show(struct seq_file *s, struct smack_rule *srp, int max) ...@@ -616,6 +620,8 @@ static void smk_rule_show(struct seq_file *s, struct smack_rule *srp, int max)
seq_putc(s, 't'); seq_putc(s, 't');
if (srp->smk_access & MAY_LOCK) if (srp->smk_access & MAY_LOCK)
seq_putc(s, 'l'); seq_putc(s, 'l');
if (srp->smk_access & MAY_BRINGUP)
seq_putc(s, 'b');
seq_putc(s, '\n'); seq_putc(s, '\n');
} }
...@@ -1880,7 +1886,10 @@ static ssize_t smk_user_access(struct file *file, const char __user *buf, ...@@ -1880,7 +1886,10 @@ static ssize_t smk_user_access(struct file *file, const char __user *buf,
else if (res != -ENOENT) else if (res != -ENOENT)
return -EINVAL; return -EINVAL;
data[0] = res == 0 ? '1' : '0'; /*
* smk_access() can return a value > 0 in the "bringup" case.
*/
data[0] = res >= 0 ? '1' : '0';
data[1] = '\0'; data[1] = '\0';
simple_transaction_set(file, 2); simple_transaction_set(file, 2);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment