diff mbox series

libsepol,checkpolicy: support omitting unused initial sid contexts

Message ID 20200128172600.12896-1-sds@tycho.nsa.gov (mailing list archive)
State Superseded
Headers show
Series libsepol,checkpolicy: support omitting unused initial sid contexts | expand

Commit Message

Stephen Smalley Jan. 28, 2020, 5:26 p.m. UTC
Remove restrictions in libsepol and checkpolicy that required all
declared initial SIDs to be assigned a context.  With this patch,
it is possible to build and load a policy that drops the sid <sidname>
<context> declarations for the unused initial SIDs.  It is still
required to retain the sid <sidname> declarations (in the flask
definitions) in order to preserve the initial SID ordering/values.
The unused initial SIDs can be renamed, e.g. to add an unused_
prefix or similar, if desired, since the names used in the policy
are not stored in the kernel binary policy.

In CIL policies, the (sid ...) and (sidorder (...)) statements
must be left intact for compatibility but the (sidcontext ...)
statements for the unused initial SIDs can be omitted after this change.

With current kernels, if one removes an unused initial SID context
from policy, builds policy with this change applied and loads the
policy into the kernel, cat /sys/fs/selinux/initial_contexts/<sidname>
will show the unlabeled context.  With the kernel patch to remove unused
initial SIDs, the /sys/fs/selinux/initial_contexts/<sidname>
file will not be created for unused initial SIDs in the first place.

NB If an unused initial SID was assigned a context different from
the unlabeled context in existing policy, then it is not safe to
remove that initial SID context from policy and reload policy on
the running kernel that was booted with the original policy.  This
is because that kernel may have assigned that SID to various kernel
objects already and those objects will then be treated as having
the unlabeled context after the removal.  In refpolicy, examples
of such initial SIDs are the "fs" SID and the "sysctl" SID.  Even
though these initial SIDs are not directly used (in code) by the current
kernel, their contexts are being applied to filesystems and sysctl files by
policy and therefore the SIDs are being assigned to objects.

NB The "sysctl" SID was in use by the kernel up until
commit 8e6c96935fcc1ed3dbebc96fddfef3f2f2395afc ("security/selinux:
fix /proc/sys/ labeling) circa v2.6.39.  Removing its context from
policy will break kernels < 2.6.39.

Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
---
 checkpolicy/test/dismod.c     |  6 +++--
 libsepol/cil/src/cil_binary.c |  4 ++-
 libsepol/cil/src/cil_verify.c | 10 -------
 libsepol/src/expand.c         |  8 ++----
 libsepol/src/policydb.c       | 51 ++++++++++++++++++++++++-----------
 5 files changed, 44 insertions(+), 35 deletions(-)

Comments

Dominick Grift Jan. 28, 2020, 6:04 p.m. UTC | #1
On Tue, Jan 28, 2020 at 12:26:00PM -0500, Stephen Smalley wrote:
> Remove restrictions in libsepol and checkpolicy that required all
> declared initial SIDs to be assigned a context.  With this patch,
> it is possible to build and load a policy that drops the sid <sidname>
> <context> declarations for the unused initial SIDs.  It is still
> required to retain the sid <sidname> declarations (in the flask
> definitions) in order to preserve the initial SID ordering/values.
> The unused initial SIDs can be renamed, e.g. to add an unused_
> prefix or similar, if desired, since the names used in the policy
> are not stored in the kernel binary policy.
> 
> In CIL policies, the (sid ...) and (sidorder (...)) statements
> must be left intact for compatibility but the (sidcontext ...)
> statements for the unused initial SIDs can be omitted after this change.
> 
> With current kernels, if one removes an unused initial SID context
> from policy, builds policy with this change applied and loads the
> policy into the kernel, cat /sys/fs/selinux/initial_contexts/<sidname>
> will show the unlabeled context.  With the kernel patch to remove unused
> initial SIDs, the /sys/fs/selinux/initial_contexts/<sidname>
> file will not be created for unused initial SIDs in the first place.
> 
> NB If an unused initial SID was assigned a context different from
> the unlabeled context in existing policy, then it is not safe to
> remove that initial SID context from policy and reload policy on
> the running kernel that was booted with the original policy.  This
> is because that kernel may have assigned that SID to various kernel
> objects already and those objects will then be treated as having
> the unlabeled context after the removal.  In refpolicy, examples
> of such initial SIDs are the "fs" SID and the "sysctl" SID.  Even
> though these initial SIDs are not directly used (in code) by the current
> kernel, their contexts are being applied to filesystems and sysctl files by
> policy and therefore the SIDs are being assigned to objects.
> 
> NB The "sysctl" SID was in use by the kernel up until
> commit 8e6c96935fcc1ed3dbebc96fddfef3f2f2395afc ("security/selinux:
> fix /proc/sys/ labeling) circa v2.6.39.  Removing its context from
> policy will break kernels < 2.6.39.
> 
> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>

Thanks

> ---
>  checkpolicy/test/dismod.c     |  6 +++--
>  libsepol/cil/src/cil_binary.c |  4 ++-
>  libsepol/cil/src/cil_verify.c | 10 -------
>  libsepol/src/expand.c         |  8 ++----
>  libsepol/src/policydb.c       | 51 ++++++++++++++++++++++++-----------
>  5 files changed, 44 insertions(+), 35 deletions(-)
> 
> diff --git a/checkpolicy/test/dismod.c b/checkpolicy/test/dismod.c
> index 8d6be2ff9522..49d610d94b00 100644
> --- a/checkpolicy/test/dismod.c
> +++ b/checkpolicy/test/dismod.c
> @@ -441,11 +441,13 @@ void display_initial_sids(policydb_t * p, FILE * fp)
>  
>  	fprintf(fp, "Initial SIDs:\n");
>  	for (cur = p->ocontexts[OCON_ISID]; cur != NULL; cur = cur->next) {
> +		if (!cur->context[0].user)
> +			continue;
>  		user = p->p_user_val_to_name[cur->context[0].user - 1];
>  		role = p->p_role_val_to_name[cur->context[0].role - 1];
>  		type = p->p_type_val_to_name[cur->context[0].type - 1];
> -		fprintf(fp, "\t%s: sid %d, context %s:%s:%s\n",
> -			cur->u.name, cur->sid[0], user, role, type);
> +		fprintf(fp, "\tsid %d, context %s:%s:%s\n",
> +			cur->sid[0], user, role, type);
>  	}
>  #if 0
>  	fprintf(fp, "Policy Initial SIDs:\n");
> diff --git a/libsepol/cil/src/cil_binary.c b/libsepol/cil/src/cil_binary.c
> index f8e20d32f9f1..e52b6679c289 100644
> --- a/libsepol/cil/src/cil_binary.c
> +++ b/libsepol/cil/src/cil_binary.c
> @@ -3067,9 +3067,11 @@ int cil_sidorder_to_policydb(policydb_t *pdb, const struct cil_db *db)
>  		struct cil_sid *cil_sid = (struct cil_sid*)curr->data;
>  		struct cil_context *cil_context = cil_sid->context;
>  
> +		/* even if no context, we must preserve initial SID values */
> +		count++;
> +
>  		if (cil_context != NULL) {
>  			ocontext_t *new_ocon = cil_add_ocontext(&pdb->ocontexts[OCON_ISID], &tail);
> -			count++;
>  			new_ocon->sid[0] = count;
>  			new_ocon->u.name = cil_strdup(cil_sid->datum.fqn);
>  			rc = __cil_context_to_sepol_context(pdb, cil_context, &new_ocon->context[0]);
> diff --git a/libsepol/cil/src/cil_verify.c b/libsepol/cil/src/cil_verify.c
> index 018514dc1bb9..0367a8e58273 100644
> --- a/libsepol/cil/src/cil_verify.c
> +++ b/libsepol/cil/src/cil_verify.c
> @@ -428,22 +428,12 @@ int __cil_verify_ordered(struct cil_tree_node *current, enum cil_flavor flavor)
>  int __cil_verify_initsids(struct cil_list *sids)
>  {
>  	int rc = SEPOL_OK;
> -	struct cil_list_item *i;
>  
>  	if (sids->head == NULL) {
>  		cil_log(CIL_ERR, "At least one initial sid must be defined in the policy\n");
>  		return SEPOL_ERR;
>  	}
>  
> -	cil_list_for_each(i, sids) {
> -		struct cil_sid *sid = i->data;
> -		if (sid->context == NULL) {
> -			struct cil_tree_node *node = sid->datum.nodes->head->data;
> -			cil_tree_log(node, CIL_ERR, "No context assigned to SID %s declared",sid->datum.name);
> -			rc = SEPOL_ERR;
> -		}
> -	}
> -
>  	return rc;
>  }
>  
> diff --git a/libsepol/src/expand.c b/libsepol/src/expand.c
> index 796121cf5ad8..5f8e75c38632 100644
> --- a/libsepol/src/expand.c
> +++ b/libsepol/src/expand.c
> @@ -2159,6 +2159,8 @@ static int ocontext_copy_selinux(expand_state_t *state)
>  	for (i = 0; i < OCON_NUM; i++) {
>  		l = NULL;
>  		for (c = state->base->ocontexts[i]; c; c = c->next) {
> +			if (i == OCON_ISID && c->context[0].user == 0)
> +				continue;
>  			n = malloc(sizeof(ocontext_t));
>  			if (!n) {
>  				ERR(state->handle, "Out of memory!");
> @@ -2172,12 +2174,6 @@ static int ocontext_copy_selinux(expand_state_t *state)
>  			l = n;
>  			switch (i) {
>  			case OCON_ISID:
> -				if (c->context[0].user == 0) {
> -					ERR(state->handle,
> -					    "Missing context for %s initial sid",
> -					    c->u.name);
> -					return -1;
> -				}
>  				n->sid[0] = c->sid[0];
>  				break;
>  			case OCON_FS:	/* FALLTHROUGH */
> diff --git a/libsepol/src/policydb.c b/libsepol/src/policydb.c
> index 745e546baa3a..03360c88e7d3 100644
> --- a/libsepol/src/policydb.c
> +++ b/libsepol/src/policydb.c
> @@ -1610,10 +1610,8 @@ int policydb_load_isids(policydb_t * p, sidtab_t * s)
>  
>  	head = p->ocontexts[OCON_ISID];
>  	for (c = head; c; c = c->next) {
> -		if (!c->context[0].user) {
> -			ERR(NULL, "SID %s was never defined", c->u.name);
> -			return -1;
> -		}
> +		if (!c->context[0].user)
> +			continue;
>  		if (sepol_sidtab_insert(s, c->sid[0], &c->context[0])) {
>  			ERR(NULL, "unable to load initial SID %s", c->u.name);
>  			return -1;
> @@ -2020,12 +2018,8 @@ static int mls_range_to_semantic(mls_range_t * r, mls_semantic_range_t * sr)
>  	return 0;
>  }
>  
> -/*
> - * Read and validate a security context structure
> - * from a policydb binary representation file.
> - */
> -static int context_read_and_validate(context_struct_t * c,
> -				     policydb_t * p, struct policy_file *fp)
> +static int context_read_no_validate(context_struct_t * c,
> +				    policydb_t * p, struct policy_file *fp)
>  {
>  	uint32_t buf[3];
>  	int rc;
> @@ -2049,6 +2043,19 @@ static int context_read_and_validate(context_struct_t * c,
>  		}
>  	}
>  
> +	return 0;
> +}
> +
> +/*
> + * Read and validate a security context structure
> + * from a policydb binary representation file.
> + */
> +static int context_read_and_validate(context_struct_t * c,
> +				     policydb_t * p, struct policy_file *fp)
> +{
> +	if (context_read_no_validate(c, p, fp))
> +		return -1;
> +
>  	if (!policydb_context_isvalid(p, c)) {
>  		ERR(fp->handle, "invalid security context");
>  		context_destroy(c);
> @@ -2797,6 +2804,7 @@ static int ocontext_read_selinux(struct policydb_compat_info *info,
>  	unsigned int i, j;
>  	size_t nel, len;
>  	ocontext_t *l, *c;
> +	context_struct_t tmpc;
>  	uint32_t buf[8];
>  	int rc;
>  
> @@ -2807,6 +2815,22 @@ static int ocontext_read_selinux(struct policydb_compat_info *info,
>  		nel = le32_to_cpu(buf[0]);
>  		l = NULL;
>  		for (j = 0; j < nel; j++) {
> +			if (i == OCON_ISID) {
> +				rc = next_entry(buf, fp, sizeof(uint32_t));
> +				if (rc < 0)
> +					return -1;
> +				context_init(&tmpc);
> +				if (context_read_no_validate
> +				    (&tmpc, p, fp))
> +					return -1;
> +				if (!tmpc.user)
> +					continue;
> +				if (!policydb_context_isvalid(p, &tmpc)) {
> +				  ERR(fp->handle, "invalid security context");
> +				  context_destroy(&tmpc);
> +				  return -1;
> +				}
> +			}
>  			c = calloc(1, sizeof(ocontext_t));
>  			if (!c) {
>  				return -1;
> @@ -2819,13 +2843,8 @@ static int ocontext_read_selinux(struct policydb_compat_info *info,
>  			l = c;
>  			switch (i) {
>  			case OCON_ISID:
> -				rc = next_entry(buf, fp, sizeof(uint32_t));
> -				if (rc < 0)
> -					return -1;
>  				c->sid[0] = le32_to_cpu(buf[0]);
> -				if (context_read_and_validate
> -				    (&c->context[0], p, fp))
> -					return -1;
> +				memcpy(&c->context[0], &tmpc, sizeof tmpc);
>  				break;
>  			case OCON_FS:
>  			case OCON_NETIF:
> -- 
> 2.24.1
>
James Carter Jan. 28, 2020, 7:45 p.m. UTC | #2
On 1/28/20 12:26 PM, Stephen Smalley wrote:
> Remove restrictions in libsepol and checkpolicy that required all
> declared initial SIDs to be assigned a context.  With this patch,
> it is possible to build and load a policy that drops the sid <sidname>
> <context> declarations for the unused initial SIDs.  It is still
> required to retain the sid <sidname> declarations (in the flask
> definitions) in order to preserve the initial SID ordering/values.
> The unused initial SIDs can be renamed, e.g. to add an unused_
> prefix or similar, if desired, since the names used in the policy
> are not stored in the kernel binary policy.
> 
> In CIL policies, the (sid ...) and (sidorder (...)) statements
> must be left intact for compatibility but the (sidcontext ...)
> statements for the unused initial SIDs can be omitted after this change.
> 
> With current kernels, if one removes an unused initial SID context
> from policy, builds policy with this change applied and loads the
> policy into the kernel, cat /sys/fs/selinux/initial_contexts/<sidname>
> will show the unlabeled context.  With the kernel patch to remove unused
> initial SIDs, the /sys/fs/selinux/initial_contexts/<sidname>
> file will not be created for unused initial SIDs in the first place.
> 
> NB If an unused initial SID was assigned a context different from
> the unlabeled context in existing policy, then it is not safe to
> remove that initial SID context from policy and reload policy on
> the running kernel that was booted with the original policy.  This
> is because that kernel may have assigned that SID to various kernel
> objects already and those objects will then be treated as having
> the unlabeled context after the removal.  In refpolicy, examples
> of such initial SIDs are the "fs" SID and the "sysctl" SID.  Even
> though these initial SIDs are not directly used (in code) by the current
> kernel, their contexts are being applied to filesystems and sysctl files by
> policy and therefore the SIDs are being assigned to objects.
> 
> NB The "sysctl" SID was in use by the kernel up until
> commit 8e6c96935fcc1ed3dbebc96fddfef3f2f2395afc ("security/selinux:
> fix /proc/sys/ labeling) circa v2.6.39.  Removing its context from
> policy will break kernels < 2.6.39.
> 
> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
> ---
>   checkpolicy/test/dismod.c     |  6 +++--
>   libsepol/cil/src/cil_binary.c |  4 ++-
>   libsepol/cil/src/cil_verify.c | 10 -------
>   libsepol/src/expand.c         |  8 ++----
>   libsepol/src/policydb.c       | 51 ++++++++++++++++++++++++-----------
>   5 files changed, 44 insertions(+), 35 deletions(-)
> 
> diff --git a/checkpolicy/test/dismod.c b/checkpolicy/test/dismod.c
> index 8d6be2ff9522..49d610d94b00 100644
> --- a/checkpolicy/test/dismod.c
> +++ b/checkpolicy/test/dismod.c
> @@ -441,11 +441,13 @@ void display_initial_sids(policydb_t * p, FILE * fp)
>   
>   	fprintf(fp, "Initial SIDs:\n");
>   	for (cur = p->ocontexts[OCON_ISID]; cur != NULL; cur = cur->next) {
> +		if (!cur->context[0].user)
> +			continue;
>   		user = p->p_user_val_to_name[cur->context[0].user - 1];
>   		role = p->p_role_val_to_name[cur->context[0].role - 1];
>   		type = p->p_type_val_to_name[cur->context[0].type - 1];
> -		fprintf(fp, "\t%s: sid %d, context %s:%s:%s\n",
> -			cur->u.name, cur->sid[0], user, role, type);
> +		fprintf(fp, "\tsid %d, context %s:%s:%s\n",
> +			cur->sid[0], user, role, type);
>   	}
>   #if 0
>   	fprintf(fp, "Policy Initial SIDs:\n");
> diff --git a/libsepol/cil/src/cil_binary.c b/libsepol/cil/src/cil_binary.c
> index f8e20d32f9f1..e52b6679c289 100644
> --- a/libsepol/cil/src/cil_binary.c
> +++ b/libsepol/cil/src/cil_binary.c
> @@ -3067,9 +3067,11 @@ int cil_sidorder_to_policydb(policydb_t *pdb, const struct cil_db *db)
>   		struct cil_sid *cil_sid = (struct cil_sid*)curr->data;
>   		struct cil_context *cil_context = cil_sid->context;
>   
> +		/* even if no context, we must preserve initial SID values */
> +		count++;
> +
>   		if (cil_context != NULL) {
>   			ocontext_t *new_ocon = cil_add_ocontext(&pdb->ocontexts[OCON_ISID], &tail);
> -			count++;
>   			new_ocon->sid[0] = count;
>   			new_ocon->u.name = cil_strdup(cil_sid->datum.fqn);
>   			rc = __cil_context_to_sepol_context(pdb, cil_context, &new_ocon->context[0]);
> diff --git a/libsepol/cil/src/cil_verify.c b/libsepol/cil/src/cil_verify.c
> index 018514dc1bb9..0367a8e58273 100644
> --- a/libsepol/cil/src/cil_verify.c
> +++ b/libsepol/cil/src/cil_verify.c
> @@ -428,22 +428,12 @@ int __cil_verify_ordered(struct cil_tree_node *current, enum cil_flavor flavor)
>   int __cil_verify_initsids(struct cil_list *sids)
>   {
>   	int rc = SEPOL_OK;
> -	struct cil_list_item *i;
>   
>   	if (sids->head == NULL) {
>   		cil_log(CIL_ERR, "At least one initial sid must be defined in the policy\n");
>   		return SEPOL_ERR;
>   	}
>   
> -	cil_list_for_each(i, sids) {
> -		struct cil_sid *sid = i->data;
> -		if (sid->context == NULL) {
> -			struct cil_tree_node *node = sid->datum.nodes->head->data;
> -			cil_tree_log(node, CIL_ERR, "No context assigned to SID %s declared",sid->datum.name);
> -			rc = SEPOL_ERR;
> -		}
> -	}
> -
>   	return rc;

Can just return SEPOL_OK here and get rid of the variable rc.

>   }
>   
> diff --git a/libsepol/src/expand.c b/libsepol/src/expand.c
> index 796121cf5ad8..5f8e75c38632 100644
> --- a/libsepol/src/expand.c
> +++ b/libsepol/src/expand.c
> @@ -2159,6 +2159,8 @@ static int ocontext_copy_selinux(expand_state_t *state)
>   	for (i = 0; i < OCON_NUM; i++) {
>   		l = NULL;
>   		for (c = state->base->ocontexts[i]; c; c = c->next) {
> +			if (i == OCON_ISID && c->context[0].user == 0)
> +				continue;
>   			n = malloc(sizeof(ocontext_t));
>   			if (!n) {
>   				ERR(state->handle, "Out of memory!");
> @@ -2172,12 +2174,6 @@ static int ocontext_copy_selinux(expand_state_t *state)
>   			l = n;
>   			switch (i) {
>   			case OCON_ISID:
> -				if (c->context[0].user == 0) {
> -					ERR(state->handle,
> -					    "Missing context for %s initial sid",
> -					    c->u.name);
> -					return -1;
> -				}
>   				n->sid[0] = c->sid[0];
>   				break;
>   			case OCON_FS:	/* FALLTHROUGH */
> diff --git a/libsepol/src/policydb.c b/libsepol/src/policydb.c
> index 745e546baa3a..03360c88e7d3 100644
> --- a/libsepol/src/policydb.c
> +++ b/libsepol/src/policydb.c
> @@ -1610,10 +1610,8 @@ int policydb_load_isids(policydb_t * p, sidtab_t * s)
>   
>   	head = p->ocontexts[OCON_ISID];
>   	for (c = head; c; c = c->next) {
> -		if (!c->context[0].user) {
> -			ERR(NULL, "SID %s was never defined", c->u.name);
> -			return -1;
> -		}
> +		if (!c->context[0].user)
> +			continue;
>   		if (sepol_sidtab_insert(s, c->sid[0], &c->context[0])) {
>   			ERR(NULL, "unable to load initial SID %s", c->u.name);
>   			return -1;
> @@ -2020,12 +2018,8 @@ static int mls_range_to_semantic(mls_range_t * r, mls_semantic_range_t * sr)
>   	return 0;
>   }
>   
> -/*
> - * Read and validate a security context structure
> - * from a policydb binary representation file.
> - */
> -static int context_read_and_validate(context_struct_t * c,
> -				     policydb_t * p, struct policy_file *fp)
> +static int context_read_no_validate(context_struct_t * c,
> +				    policydb_t * p, struct policy_file *fp)
>   {
>   	uint32_t buf[3];
>   	int rc;
> @@ -2049,6 +2043,19 @@ static int context_read_and_validate(context_struct_t * c,
>   		}
>   	}
>   
> +	return 0;
> +}
> +
> +/*
> + * Read and validate a security context structure
> + * from a policydb binary representation file.
> + */
> +static int context_read_and_validate(context_struct_t * c,
> +				     policydb_t * p, struct policy_file *fp)
> +{
> +	if (context_read_no_validate(c, p, fp))
> +		return -1;
> +
>   	if (!policydb_context_isvalid(p, c)) {
>   		ERR(fp->handle, "invalid security context");
>   		context_destroy(c);
> @@ -2797,6 +2804,7 @@ static int ocontext_read_selinux(struct policydb_compat_info *info,
>   	unsigned int i, j;
>   	size_t nel, len;
>   	ocontext_t *l, *c;
> +	context_struct_t tmpc;
>   	uint32_t buf[8];
>   	int rc;
>   
> @@ -2807,6 +2815,22 @@ static int ocontext_read_selinux(struct policydb_compat_info *info,
>   		nel = le32_to_cpu(buf[0]);
>   		l = NULL;
>   		for (j = 0; j < nel; j++) {
> +			if (i == OCON_ISID) {
> +				rc = next_entry(buf, fp, sizeof(uint32_t));
> +				if (rc < 0)
> +					return -1;
> +				context_init(&tmpc);
> +				if (context_read_no_validate
> +				    (&tmpc, p, fp))
> +					return -1;
> +				if (!tmpc.user)
> +					continue;
> +				if (!policydb_context_isvalid(p, &tmpc)) {
> +				  ERR(fp->handle, "invalid security context");
> +				  context_destroy(&tmpc);
> +				  return -1;
> +				}
> +			}
>   			c = calloc(1, sizeof(ocontext_t));
>   			if (!c) {
>   				return -1;
> @@ -2819,13 +2843,8 @@ static int ocontext_read_selinux(struct policydb_compat_info *info,
>   			l = c;
>   			switch (i) {
>   			case OCON_ISID:
> -				rc = next_entry(buf, fp, sizeof(uint32_t));
> -				if (rc < 0)
> -					return -1;
>   				c->sid[0] = le32_to_cpu(buf[0]);
> -				if (context_read_and_validate
> -				    (&c->context[0], p, fp))
> -					return -1;
> +				memcpy(&c->context[0], &tmpc, sizeof tmpc);
>   				break;
>   			case OCON_FS:
>   			case OCON_NETIF:
> 

I see that you don't do anything with ocontext_read_xen(), so Xen policies still 
require initial sid contexts? If so, doesn't that mean that CIL behavior needs 
to depend on whether or not a Xen policy is being created?

Jim
Stephen Smalley Jan. 28, 2020, 8:42 p.m. UTC | #3
On 1/28/20 2:45 PM, jwcart2 wrote:
> On 1/28/20 12:26 PM, Stephen Smalley wrote:
>> Remove restrictions in libsepol and checkpolicy that required all
>> declared initial SIDs to be assigned a context.  With this patch,
>> it is possible to build and load a policy that drops the sid <sidname>
>> <context> declarations for the unused initial SIDs.  It is still
>> required to retain the sid <sidname> declarations (in the flask
>> definitions) in order to preserve the initial SID ordering/values.
>> The unused initial SIDs can be renamed, e.g. to add an unused_
>> prefix or similar, if desired, since the names used in the policy
>> are not stored in the kernel binary policy.
>>
>> In CIL policies, the (sid ...) and (sidorder (...)) statements
>> must be left intact for compatibility but the (sidcontext ...)
>> statements for the unused initial SIDs can be omitted after this change.
>>
>> With current kernels, if one removes an unused initial SID context
>> from policy, builds policy with this change applied and loads the
>> policy into the kernel, cat /sys/fs/selinux/initial_contexts/<sidname>
>> will show the unlabeled context.  With the kernel patch to remove unused
>> initial SIDs, the /sys/fs/selinux/initial_contexts/<sidname>
>> file will not be created for unused initial SIDs in the first place.
>>
>> NB If an unused initial SID was assigned a context different from
>> the unlabeled context in existing policy, then it is not safe to
>> remove that initial SID context from policy and reload policy on
>> the running kernel that was booted with the original policy.  This
>> is because that kernel may have assigned that SID to various kernel
>> objects already and those objects will then be treated as having
>> the unlabeled context after the removal.  In refpolicy, examples
>> of such initial SIDs are the "fs" SID and the "sysctl" SID.  Even
>> though these initial SIDs are not directly used (in code) by the current
>> kernel, their contexts are being applied to filesystems and sysctl 
>> files by
>> policy and therefore the SIDs are being assigned to objects.
>>
>> NB The "sysctl" SID was in use by the kernel up until
>> commit 8e6c96935fcc1ed3dbebc96fddfef3f2f2395afc ("security/selinux:
>> fix /proc/sys/ labeling) circa v2.6.39.  Removing its context from
>> policy will break kernels < 2.6.39.
>>
>> Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
>> ---

[...]

> I see that you don't do anything with ocontext_read_xen(), so Xen 
> policies still require initial sid contexts? If so, doesn't that mean 
> that CIL behavior needs to depend on whether or not a Xen policy is 
> being created?

On second thought, I don't think I need to change 
ocontext_read_selinux() or the kernel to accept/ignore these entries.  I 
just need to make sure that they are not included in the generated 
policy file in the first place.  It was already possible that the 
OCON_ISID list might not include every initial SID.  So I don't think it 
will need to be target-specific.
diff mbox series

Patch

diff --git a/checkpolicy/test/dismod.c b/checkpolicy/test/dismod.c
index 8d6be2ff9522..49d610d94b00 100644
--- a/checkpolicy/test/dismod.c
+++ b/checkpolicy/test/dismod.c
@@ -441,11 +441,13 @@  void display_initial_sids(policydb_t * p, FILE * fp)
 
 	fprintf(fp, "Initial SIDs:\n");
 	for (cur = p->ocontexts[OCON_ISID]; cur != NULL; cur = cur->next) {
+		if (!cur->context[0].user)
+			continue;
 		user = p->p_user_val_to_name[cur->context[0].user - 1];
 		role = p->p_role_val_to_name[cur->context[0].role - 1];
 		type = p->p_type_val_to_name[cur->context[0].type - 1];
-		fprintf(fp, "\t%s: sid %d, context %s:%s:%s\n",
-			cur->u.name, cur->sid[0], user, role, type);
+		fprintf(fp, "\tsid %d, context %s:%s:%s\n",
+			cur->sid[0], user, role, type);
 	}
 #if 0
 	fprintf(fp, "Policy Initial SIDs:\n");
diff --git a/libsepol/cil/src/cil_binary.c b/libsepol/cil/src/cil_binary.c
index f8e20d32f9f1..e52b6679c289 100644
--- a/libsepol/cil/src/cil_binary.c
+++ b/libsepol/cil/src/cil_binary.c
@@ -3067,9 +3067,11 @@  int cil_sidorder_to_policydb(policydb_t *pdb, const struct cil_db *db)
 		struct cil_sid *cil_sid = (struct cil_sid*)curr->data;
 		struct cil_context *cil_context = cil_sid->context;
 
+		/* even if no context, we must preserve initial SID values */
+		count++;
+
 		if (cil_context != NULL) {
 			ocontext_t *new_ocon = cil_add_ocontext(&pdb->ocontexts[OCON_ISID], &tail);
-			count++;
 			new_ocon->sid[0] = count;
 			new_ocon->u.name = cil_strdup(cil_sid->datum.fqn);
 			rc = __cil_context_to_sepol_context(pdb, cil_context, &new_ocon->context[0]);
diff --git a/libsepol/cil/src/cil_verify.c b/libsepol/cil/src/cil_verify.c
index 018514dc1bb9..0367a8e58273 100644
--- a/libsepol/cil/src/cil_verify.c
+++ b/libsepol/cil/src/cil_verify.c
@@ -428,22 +428,12 @@  int __cil_verify_ordered(struct cil_tree_node *current, enum cil_flavor flavor)
 int __cil_verify_initsids(struct cil_list *sids)
 {
 	int rc = SEPOL_OK;
-	struct cil_list_item *i;
 
 	if (sids->head == NULL) {
 		cil_log(CIL_ERR, "At least one initial sid must be defined in the policy\n");
 		return SEPOL_ERR;
 	}
 
-	cil_list_for_each(i, sids) {
-		struct cil_sid *sid = i->data;
-		if (sid->context == NULL) {
-			struct cil_tree_node *node = sid->datum.nodes->head->data;
-			cil_tree_log(node, CIL_ERR, "No context assigned to SID %s declared",sid->datum.name);
-			rc = SEPOL_ERR;
-		}
-	}
-
 	return rc;
 }
 
diff --git a/libsepol/src/expand.c b/libsepol/src/expand.c
index 796121cf5ad8..5f8e75c38632 100644
--- a/libsepol/src/expand.c
+++ b/libsepol/src/expand.c
@@ -2159,6 +2159,8 @@  static int ocontext_copy_selinux(expand_state_t *state)
 	for (i = 0; i < OCON_NUM; i++) {
 		l = NULL;
 		for (c = state->base->ocontexts[i]; c; c = c->next) {
+			if (i == OCON_ISID && c->context[0].user == 0)
+				continue;
 			n = malloc(sizeof(ocontext_t));
 			if (!n) {
 				ERR(state->handle, "Out of memory!");
@@ -2172,12 +2174,6 @@  static int ocontext_copy_selinux(expand_state_t *state)
 			l = n;
 			switch (i) {
 			case OCON_ISID:
-				if (c->context[0].user == 0) {
-					ERR(state->handle,
-					    "Missing context for %s initial sid",
-					    c->u.name);
-					return -1;
-				}
 				n->sid[0] = c->sid[0];
 				break;
 			case OCON_FS:	/* FALLTHROUGH */
diff --git a/libsepol/src/policydb.c b/libsepol/src/policydb.c
index 745e546baa3a..03360c88e7d3 100644
--- a/libsepol/src/policydb.c
+++ b/libsepol/src/policydb.c
@@ -1610,10 +1610,8 @@  int policydb_load_isids(policydb_t * p, sidtab_t * s)
 
 	head = p->ocontexts[OCON_ISID];
 	for (c = head; c; c = c->next) {
-		if (!c->context[0].user) {
-			ERR(NULL, "SID %s was never defined", c->u.name);
-			return -1;
-		}
+		if (!c->context[0].user)
+			continue;
 		if (sepol_sidtab_insert(s, c->sid[0], &c->context[0])) {
 			ERR(NULL, "unable to load initial SID %s", c->u.name);
 			return -1;
@@ -2020,12 +2018,8 @@  static int mls_range_to_semantic(mls_range_t * r, mls_semantic_range_t * sr)
 	return 0;
 }
 
-/*
- * Read and validate a security context structure
- * from a policydb binary representation file.
- */
-static int context_read_and_validate(context_struct_t * c,
-				     policydb_t * p, struct policy_file *fp)
+static int context_read_no_validate(context_struct_t * c,
+				    policydb_t * p, struct policy_file *fp)
 {
 	uint32_t buf[3];
 	int rc;
@@ -2049,6 +2043,19 @@  static int context_read_and_validate(context_struct_t * c,
 		}
 	}
 
+	return 0;
+}
+
+/*
+ * Read and validate a security context structure
+ * from a policydb binary representation file.
+ */
+static int context_read_and_validate(context_struct_t * c,
+				     policydb_t * p, struct policy_file *fp)
+{
+	if (context_read_no_validate(c, p, fp))
+		return -1;
+
 	if (!policydb_context_isvalid(p, c)) {
 		ERR(fp->handle, "invalid security context");
 		context_destroy(c);
@@ -2797,6 +2804,7 @@  static int ocontext_read_selinux(struct policydb_compat_info *info,
 	unsigned int i, j;
 	size_t nel, len;
 	ocontext_t *l, *c;
+	context_struct_t tmpc;
 	uint32_t buf[8];
 	int rc;
 
@@ -2807,6 +2815,22 @@  static int ocontext_read_selinux(struct policydb_compat_info *info,
 		nel = le32_to_cpu(buf[0]);
 		l = NULL;
 		for (j = 0; j < nel; j++) {
+			if (i == OCON_ISID) {
+				rc = next_entry(buf, fp, sizeof(uint32_t));
+				if (rc < 0)
+					return -1;
+				context_init(&tmpc);
+				if (context_read_no_validate
+				    (&tmpc, p, fp))
+					return -1;
+				if (!tmpc.user)
+					continue;
+				if (!policydb_context_isvalid(p, &tmpc)) {
+				  ERR(fp->handle, "invalid security context");
+				  context_destroy(&tmpc);
+				  return -1;
+				}
+			}
 			c = calloc(1, sizeof(ocontext_t));
 			if (!c) {
 				return -1;
@@ -2819,13 +2843,8 @@  static int ocontext_read_selinux(struct policydb_compat_info *info,
 			l = c;
 			switch (i) {
 			case OCON_ISID:
-				rc = next_entry(buf, fp, sizeof(uint32_t));
-				if (rc < 0)
-					return -1;
 				c->sid[0] = le32_to_cpu(buf[0]);
-				if (context_read_and_validate
-				    (&c->context[0], p, fp))
-					return -1;
+				memcpy(&c->context[0], &tmpc, sizeof tmpc);
 				break;
 			case OCON_FS:
 			case OCON_NETIF: