• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Author : Joshua Brindle <jbrindle@tresys.com>
2  *	    Karl MacMillan <kmacmillan@tresys.com>
3  *          Jason Tang     <jtang@tresys.com>
4  *	Added support for binary policy modules
5  *
6  * Copyright (C) 2004 - 2005 Tresys Technology, LLC
7  *	This program is free software; you can redistribute it and/or modify
8  *  	it under the terms of the GNU General Public License as published by
9  *	the Free Software Foundation, version 2.
10  */
11 
12 #include <assert.h>
13 #include <stdarg.h>
14 #include <stdlib.h>
15 #include <string.h>
16 
17 #include <sepol/policydb/policydb.h>
18 #include <sepol/policydb/avrule_block.h>
19 #include <sepol/policydb/conditional.h>
20 
21 #include "queue.h"
22 #include "module_compiler.h"
23 
24 union stack_item_u {
25 	avrule_block_t *avrule;
26 	cond_list_t *cond_list;
27 };
28 
29 typedef struct scope_stack {
30 	union stack_item_u u;
31 	int type;		/* for above union: 1 = avrule block, 2 = conditional */
32 	avrule_decl_t *decl;	/* if in an avrule block, which
33 				 * declaration is current */
34 	avrule_t *last_avrule;
35 	int in_else;		/* if in an avrule block, within ELSE branch */
36 	int require_given;	/* 1 if this block had at least one require */
37 	struct scope_stack *parent, *child;
38 } scope_stack_t;
39 
40 extern policydb_t *policydbp;
41 extern queue_t id_queue;
42 extern int yyerror(const char *msg);
43 __attribute__ ((format(printf, 1, 2)))
44 extern void yyerror2(const char *fmt, ...);
45 
46 static int push_stack(int stack_type, ...);
47 static void pop_stack(void);
48 
49 /* keep track of the last item added to the stack */
50 static scope_stack_t *stack_top = NULL;
51 static avrule_block_t *last_block;
52 static uint32_t next_decl_id = 1;
53 
54 static const char * const flavor_str[SYM_NUM] = {
55 	[SYM_COMMONS] = "common",
56 	[SYM_CLASSES] = "class",
57 	[SYM_ROLES] = "role",
58 	[SYM_TYPES] = "type",
59 	[SYM_USERS] = "user",
60 	[SYM_BOOLS] = "bool",
61 	[SYM_LEVELS] = "level",
62 	[SYM_CATS] = "cat"
63 };
64 
print_error_msg(int ret,uint32_t symbol_type)65 static void print_error_msg(int ret, uint32_t symbol_type)
66 {
67 	switch (ret) {
68 	case -3:
69 		yyerror("Out of memory!");
70 		break;
71 	case -2:
72 		yyerror2("Duplicate declaration of %s", flavor_str[symbol_type]);
73 		break;
74 	case -1:
75 		yyerror2("Could not declare %s here", flavor_str[symbol_type]);
76 		break;
77 	default:
78 		yyerror("Unknown error");
79 	}
80 }
81 
define_policy(int pass,int module_header_given)82 int define_policy(int pass, int module_header_given)
83 {
84 	char *id;
85 
86 	if (module_header_given) {
87 		if (policydbp->policy_type != POLICY_MOD) {
88 			yyerror
89 			    ("Module specification found while not building a policy module.\n");
90 			return -1;
91 		}
92 
93 		if (pass == 2) {
94 			while ((id = queue_remove(id_queue)) != NULL)
95 				free(id);
96 		} else {
97 			id = (char *)queue_remove(id_queue);
98 			if (!id) {
99 				yyerror("no module name");
100 				return -1;
101 			}
102 			free(policydbp->name);
103 			policydbp->name = id;
104 			if ((policydbp->version =
105 			     queue_remove(id_queue)) == NULL) {
106 				yyerror
107 				    ("Expected a module version but none was found.");
108 				return -1;
109 			}
110 		}
111 	} else {
112 		if (policydbp->policy_type == POLICY_MOD) {
113 			yyerror
114 			    ("Building a policy module, but no module specification found.\n");
115 			return -1;
116 		}
117 	}
118 	/* the first declaration within the global avrule
119 	   block will always have an id of 1 */
120 	next_decl_id = 2;
121 
122 	/* reset the scoping stack */
123 	while (stack_top != NULL) {
124 		pop_stack();
125 	}
126 	if (push_stack(1, policydbp->global, policydbp->global->branch_list) ==
127 	    -1) {
128 		return -1;
129 	}
130 	last_block = policydbp->global;
131 	return 0;
132 }
133 
134 /* Given the current parse stack, returns 1 if a declaration or require would
135  * be allowed here or 0 if not.  For example, declarations and requirements are
136  * not allowed in conditionals, so if there are any conditionals in the
137  * current scope stack then this would return a 0.
138  */
is_creation_allowed(void)139 static int is_creation_allowed(void)
140 {
141 	if (stack_top->type != 1 || stack_top->in_else) {
142 		return 0;
143 	}
144 	return 1;
145 }
146 
147 /* Attempt to declare or require a symbol within the current scope.
148  * Returns:
149  *  0: Success - Symbol had not been previously created.
150  *  1: Success - Symbol had already been created and caller must free datum.
151  * -1: Failure - Symbol cannot be created here
152  * -2: Failure - Duplicate declaration or type/attribute mismatch
153  * -3: Failure - Out of memory or some other error
154  */
create_symbol(uint32_t symbol_type,hashtab_key_t key,hashtab_datum_t datum,uint32_t * dest_value,uint32_t scope)155 static int create_symbol(uint32_t symbol_type, hashtab_key_t key, hashtab_datum_t datum,
156 			 uint32_t * dest_value, uint32_t scope)
157 {
158 	avrule_decl_t *decl = stack_top->decl;
159 	int ret;
160 
161 	if (!is_creation_allowed()) {
162 		return -1;
163 	}
164 
165 	ret = symtab_insert(policydbp, symbol_type, key, datum, scope,
166 			    decl->decl_id, dest_value);
167 
168 	if (ret == 1 && dest_value) {
169 		hashtab_datum_t s =
170 			hashtab_search(policydbp->symtab[symbol_type].table,
171 				       key);
172 		assert(s != NULL);
173 
174 		if (symbol_type == SYM_LEVELS) {
175 			*dest_value = ((level_datum_t *)s)->level->sens;
176 		} else {
177 			*dest_value = ((symtab_datum_t *)s)->value;
178 		}
179 	} else if (ret == -2) {
180 		return -2;
181 	} else if (ret < 0) {
182 		return -3;
183 	}
184 
185 	return ret;
186 }
187 
188 /* Attempt to declare a symbol within the current declaration.  If
189  * currently within a non-conditional and in a non-else branch then
190  * insert the symbol, return 0 on success if symbol was undeclared.
191  * For roles and users, it is legal to have multiple declarations; as
192  * such return 1 to indicate that caller must free() the datum because
193  * it was not added.  If symbols may not be declared here return -1.
194  * For duplicate declarations return -2.  For all else, including out
195  * of memory, return -3.  Note that dest_value and datum_value might
196  * not be restricted pointers. */
declare_symbol(uint32_t symbol_type,hashtab_key_t key,hashtab_datum_t datum,uint32_t * dest_value,uint32_t * datum_value)197 int declare_symbol(uint32_t symbol_type,
198 		   hashtab_key_t key, hashtab_datum_t datum,
199 		   uint32_t * dest_value, uint32_t * datum_value)
200 {
201 	avrule_decl_t *decl = stack_top->decl;
202 	int ret = create_symbol(symbol_type, key, datum, dest_value, SCOPE_DECL);
203 
204 	if (ret < 0) {
205 		return ret;
206 	}
207 
208 	if (ebitmap_set_bit(decl->declared.scope + symbol_type,
209 			    *datum_value - 1, 1)) {
210 		return -3;
211 	}
212 
213 	return ret;
214 }
215 
role_implicit_bounds(hashtab_t roles_tab,char * role_id,role_datum_t * role)216 static int role_implicit_bounds(hashtab_t roles_tab,
217 				char *role_id, role_datum_t *role)
218 {
219 	role_datum_t *bounds;
220 	char *bounds_id, *delim;
221 
222 	delim = strrchr(role_id, '.');
223 	if (!delim)
224 		return 0;	/* no implicit boundary */
225 
226 	bounds_id = strdup(role_id);
227 	if (!bounds_id) {
228 		yyerror("out of memory");
229 		return -1;
230 	}
231 	bounds_id[(size_t)(delim - role_id)] = '\0';
232 
233 	bounds = hashtab_search(roles_tab, bounds_id);
234 	if (!bounds) {
235 		yyerror2("role %s doesn't exist, is implicit bounds of %s",
236 			 bounds_id, role_id);
237 		return -1;
238 	}
239 
240 	if (!role->bounds)
241 		role->bounds = bounds->s.value;
242 	else if (role->bounds != bounds->s.value) {
243 		yyerror2("role %s has inconsistent bounds %s/%s",
244 			 role_id, bounds_id,
245 			 policydbp->p_role_val_to_name[role->bounds - 1]);
246 		return -1;
247 	}
248 	free(bounds_id);
249 
250 	return 0;
251 }
252 
create_role(uint32_t scope,unsigned char isattr,role_datum_t ** role,char ** key)253 static int create_role(uint32_t scope, unsigned char isattr, role_datum_t **role, char **key)
254 {
255 	char *id = queue_remove(id_queue);
256 	role_datum_t *datum = NULL;
257 	int ret;
258 	uint32_t value;
259 
260 	*role = NULL;
261 	*key = NULL;
262 	isattr = isattr ? ROLE_ATTRIB : ROLE_ROLE;
263 
264 	if (id == NULL) {
265 		yyerror("no role name");
266 		return -1;
267 	}
268 
269 	datum = malloc(sizeof(*datum));
270 	if (datum == NULL) {
271 		yyerror("Out of memory!");
272 		free(id);
273 		return -1;
274 	}
275 
276 	role_datum_init(datum);
277 	datum->flavor = isattr;
278 
279 	if (scope == SCOPE_DECL) {
280 		ret = declare_symbol(SYM_ROLES, id, datum, &value, &value);
281 	} else {
282 		ret = require_symbol(SYM_ROLES, id, datum, &value, &value);
283 	}
284 
285 	datum->s.value = value;
286 
287 	if (ret == 0) {
288 		*role = datum;
289 		*key = strdup(id);
290 		if (*key == NULL) {
291 			yyerror("Out of memory!");
292 			return -1;
293 		}
294 	} else if (ret == 1) {
295 		*role = hashtab_search(policydbp->symtab[SYM_ROLES].table, id);
296 		if (*role && (isattr != (*role)->flavor)) {
297 			yyerror2("Identifier %s used as both an attribute and a role",
298 				 id);
299 			free(id);
300 			role_datum_destroy(datum);
301 			free(datum);
302 			return -1;
303 		}
304 		*role = datum;
305 		*key = id;
306 	} else {
307 		print_error_msg(ret, SYM_ROLES);
308 		free(id);
309 		role_datum_destroy(datum);
310 		free(datum);
311 	}
312 
313 	return ret;
314 }
315 
declare_role(unsigned char isattr)316 role_datum_t *declare_role(unsigned char isattr)
317 {
318 	char *key = NULL;
319 	role_datum_t *role = NULL;
320 	role_datum_t *dest_role = NULL;
321 	hashtab_t roles_tab;
322 	int ret, ret2;
323 
324 	ret = create_role(SCOPE_DECL, isattr, &role, &key);
325 	if (ret < 0) {
326 		return NULL;
327 	}
328 
329 	/* create a new role_datum_t for this decl, if necessary */
330 	assert(stack_top->type == 1);
331 
332 	if (stack_top->parent == NULL) {
333 		/* in parent, so use global symbol table */
334 		roles_tab = policydbp->p_roles.table;
335 	} else {
336 		roles_tab = stack_top->decl->p_roles.table;
337 	}
338 
339 	dest_role = hashtab_search(roles_tab, key);
340 	if (dest_role == NULL) {
341 		if (ret == 0) {
342 			dest_role = malloc(sizeof(*dest_role));
343 			if (dest_role == NULL) {
344 				yyerror("Out of memory!");
345 				free(key);
346 				return NULL;
347 			}
348 			role_datum_init(dest_role);
349 			dest_role->s.value = role->s.value;
350 			dest_role->flavor = role->flavor;
351 		} else {
352 			dest_role = role;
353 		}
354 		ret2 = role_implicit_bounds(roles_tab, key, dest_role);
355 		if (ret2 != 0) {
356 			free(key);
357 			role_datum_destroy(dest_role);
358 			free(dest_role);
359 			return NULL;
360 		}
361 		ret2 = hashtab_insert(roles_tab, key, dest_role);
362 		if (ret2 != 0) {
363 			yyerror("Out of memory!");
364 			free(key);
365 			role_datum_destroy(dest_role);
366 			free(dest_role);
367 			return NULL;
368 		}
369 	} else {
370 		free(key);
371 		if (ret == 1) {
372 			role_datum_destroy(role);
373 			free(role);
374 		}
375 	}
376 
377 	if (ret == 0) {
378 		ret2 = ebitmap_set_bit(&dest_role->dominates, dest_role->s.value - 1, 1);
379 		if (ret2 != 0) {
380 			yyerror("out of memory");
381 			return NULL;
382 		}
383 	}
384 
385 	return dest_role;
386 }
387 
create_type(uint32_t scope,unsigned char isattr,type_datum_t ** type)388 static int create_type(uint32_t scope, unsigned char isattr, type_datum_t **type)
389 {
390 	char *id;
391 	type_datum_t *datum;
392 	int ret;
393 	uint32_t value = 0;
394 
395 	*type = NULL;
396 	isattr = isattr ? TYPE_ATTRIB : TYPE_TYPE;
397 
398 	id = (char *)queue_remove(id_queue);
399 	if (!id) {
400 		yyerror("no type/attribute name?");
401 		return -1;
402 	}
403 	if (strcmp(id, "self") == 0) {
404 		yyerror("\"self\" is a reserved type name.");
405 		free(id);
406 		return -1;
407 	}
408 
409 	datum = malloc(sizeof(*datum));
410 	if (!datum) {
411 		yyerror("Out of memory!");
412 		free(id);
413 		return -1;
414 	}
415 	type_datum_init(datum);
416 	datum->primary = 1;
417 	datum->flavor = isattr;
418 
419 	if (scope == SCOPE_DECL) {
420 		ret = declare_symbol(SYM_TYPES, id, datum, &value, &value);
421 	} else {
422 		ret = require_symbol(SYM_TYPES, id, datum, &value, &value);
423 	}
424 
425 	if (ret == 0) {
426 		datum->s.value = value;
427 		*type = datum;
428 	} else if (ret == 1) {
429 		type_datum_destroy(datum);
430 		free(datum);
431 		*type = hashtab_search(policydbp->symtab[SYM_TYPES].table, id);
432 		if (*type && (isattr != (*type)->flavor)) {
433 			yyerror2("Identifier %s used as both an attribute and a type",
434 				 id);
435 			free(id);
436 			return -1;
437 		}
438 		free(id);
439 	} else {
440 		print_error_msg(ret, SYM_TYPES);
441 		free(id);
442 		type_datum_destroy(datum);
443 		free(datum);
444 	}
445 
446 	return ret;
447 }
448 
declare_type(unsigned char primary,unsigned char isattr)449 type_datum_t *declare_type(unsigned char primary, unsigned char isattr)
450 {
451 	type_datum_t *type = NULL;
452 	int ret = create_type(SCOPE_DECL, isattr, &type);
453 
454 	if (ret == 0) {
455 		type->primary = primary;
456 	}
457 
458 	return type;
459 }
460 
user_implicit_bounds(hashtab_t users_tab,char * user_id,user_datum_t * user)461 static int user_implicit_bounds(hashtab_t users_tab,
462 				char *user_id, user_datum_t *user)
463 {
464 	user_datum_t *bounds;
465 	char *bounds_id, *delim;
466 
467 	delim = strrchr(user_id, '.');
468 	if (!delim)
469 		return 0;	/* no implicit boundary */
470 
471 	bounds_id = strdup(user_id);
472 	if (!bounds_id) {
473 		yyerror("out of memory");
474 		return -1;
475 	}
476 	bounds_id[(size_t)(delim - user_id)] = '\0';
477 
478 	bounds = hashtab_search(users_tab, bounds_id);
479 	if (!bounds) {
480 		yyerror2("user %s doesn't exist, is implicit bounds of %s",
481 			 bounds_id, user_id);
482 		return -1;
483 	}
484 
485 	if (!user->bounds)
486 		user->bounds = bounds->s.value;
487 	else if (user->bounds != bounds->s.value) {
488 		yyerror2("user %s has inconsistent bounds %s/%s",
489 			 user_id, bounds_id,
490 			 policydbp->p_role_val_to_name[user->bounds - 1]);
491 		return -1;
492 	}
493 	free(bounds_id);
494 
495 	return 0;
496 }
497 
create_user(uint32_t scope,user_datum_t ** user,char ** key)498 static int create_user(uint32_t scope, user_datum_t **user, char **key)
499 {
500 	char *id = queue_remove(id_queue);
501 	user_datum_t *datum = NULL;
502 	int ret;
503 	uint32_t value;
504 
505 	*user = NULL;
506 	*key = NULL;
507 
508 	if (id == NULL) {
509 		yyerror("no user name");
510 		return -1;
511 	}
512 
513 	datum = malloc(sizeof(*datum));
514 	if (datum == NULL) {
515 		yyerror("Out of memory!");
516 		free(id);
517 		return -1;
518 	}
519 
520 	user_datum_init(datum);
521 
522 	if (scope == SCOPE_DECL) {
523 		ret = declare_symbol(SYM_USERS, id, datum, &value, &value);
524 	} else {
525 		ret = require_symbol(SYM_USERS, id, datum, &value, &value);
526 	}
527 
528 	datum->s.value = value;
529 
530 	if (ret == 0) {
531 		*user = datum;
532 		*key = strdup(id);
533 		if (*key == NULL) {
534 			yyerror("Out of memory!");
535 			return -1;
536 		}
537 	} else if (ret == 1) {
538 		*user = datum;
539 		*key = id;
540 	} else {
541 		print_error_msg(ret, SYM_USERS);
542 		free(id);
543 		user_datum_destroy(datum);
544 		free(datum);
545 	}
546 
547 	return ret;
548 }
549 
declare_user(void)550 user_datum_t *declare_user(void)
551 {
552 	char *key = NULL;
553 	user_datum_t *user = NULL;
554 	user_datum_t *dest_user = NULL;
555 	hashtab_t users_tab;
556 	int ret, ret2;
557 
558 	ret = create_user(SCOPE_DECL, &user, &key);
559 	if (ret < 0) {
560 		return NULL;
561 	}
562 
563 	/* create a new user_datum_t for this decl, if necessary */
564 	assert(stack_top->type == 1);
565 
566 	if (stack_top->parent == NULL) {
567 		/* in parent, so use global symbol table */
568 		users_tab = policydbp->p_users.table;
569 	} else {
570 		users_tab = stack_top->decl->p_users.table;
571 	}
572 
573 	dest_user = hashtab_search(users_tab, key);
574 	if (dest_user == NULL) {
575 		if (ret == 0) {
576 			dest_user = malloc(sizeof(*dest_user));
577 			if (dest_user == NULL) {
578 				yyerror("Out of memory!");
579 				free(key);
580 				return NULL;
581 			}
582 			user_datum_init(dest_user);
583 			dest_user->s.value = user->s.value;
584 		} else {
585 			dest_user = user;
586 		}
587 		ret2 = user_implicit_bounds(users_tab, key, dest_user);
588 		if (ret2 != 0) {
589 			free(key);
590 			user_datum_destroy(dest_user);
591 			free(dest_user);
592 			return NULL;
593 		}
594 		ret2 = hashtab_insert(users_tab, key, dest_user);
595 		if (ret2 != 0) {
596 			yyerror("Out of memory!");
597 			free(key);
598 			user_datum_destroy(dest_user);
599 			free(dest_user);
600 			return NULL;
601 		}
602 	} else {
603 		free(key);
604 		if (ret == 1) {
605 			user_datum_destroy(user);
606 			free(user);
607 		}
608 	}
609 
610 	return dest_user;
611 }
612 
613 /* Return a type_datum_t for the local avrule_decl with the given ID.
614  * If it does not exist, create one with the same value as 'value'.
615  * This function assumes that the ID is within scope.  c.f.,
616  * is_id_in_scope().
617  *
618  * NOTE: this function usurps ownership of id afterwards.  The caller
619  * shall not reference it nor free() it afterwards.
620  */
get_local_type(char * id,uint32_t value,unsigned char isattr)621 type_datum_t *get_local_type(char *id, uint32_t value, unsigned char isattr)
622 {
623 	type_datum_t *dest_typdatum;
624 	hashtab_t types_tab;
625 	assert(stack_top->type == 1);
626 	if (stack_top->parent == NULL) {
627 		/* in global, so use global symbol table */
628 		types_tab = policydbp->p_types.table;
629 	} else {
630 		types_tab = stack_top->decl->p_types.table;
631 	}
632 	dest_typdatum = hashtab_search(types_tab, id);
633 	if (!dest_typdatum) {
634 		dest_typdatum = (type_datum_t *) malloc(sizeof(type_datum_t));
635 		if (dest_typdatum == NULL) {
636 			free(id);
637 			return NULL;
638 		}
639 		type_datum_init(dest_typdatum);
640 		dest_typdatum->s.value = value;
641 		dest_typdatum->flavor = isattr ? TYPE_ATTRIB : TYPE_TYPE;
642 		dest_typdatum->primary = 1;
643 		if (hashtab_insert(types_tab, id, dest_typdatum)) {
644 			free(id);
645 			type_datum_destroy(dest_typdatum);
646 			free(dest_typdatum);
647 			return NULL;
648 		}
649 
650 	} else {
651 		free(id);
652 		if (dest_typdatum->flavor != isattr ? TYPE_ATTRIB : TYPE_TYPE) {
653 			return NULL;
654 		}
655 	}
656 	return dest_typdatum;
657 }
658 
659 /* Return a role_datum_t for the local avrule_decl with the given ID.
660  * If it does not exist, create one with the same value as 'value'.
661  * This function assumes that the ID is within scope.  c.f.,
662  * is_id_in_scope().
663  *
664  * NOTE: this function usurps ownership of id afterwards.  The caller
665  * shall not reference it nor free() it afterwards.
666  */
get_local_role(char * id,uint32_t value,unsigned char isattr)667 role_datum_t *get_local_role(char *id, uint32_t value, unsigned char isattr)
668 {
669 	role_datum_t *dest_roledatum;
670 	hashtab_t roles_tab;
671 
672 	assert(stack_top->type == 1);
673 
674 	if (stack_top->parent == NULL) {
675 		/* in global, so use global symbol table */
676 		roles_tab = policydbp->p_roles.table;
677 	} else {
678 		roles_tab = stack_top->decl->p_roles.table;
679 	}
680 
681 	dest_roledatum = hashtab_search(roles_tab, id);
682 	if (!dest_roledatum) {
683 		dest_roledatum = (role_datum_t *)malloc(sizeof(role_datum_t));
684 		if (dest_roledatum == NULL) {
685 			free(id);
686 			return NULL;
687 		}
688 
689 		role_datum_init(dest_roledatum);
690 		dest_roledatum->s.value = value;
691 		dest_roledatum->flavor = isattr ? ROLE_ATTRIB : ROLE_ROLE;
692 
693 		if (hashtab_insert(roles_tab, id, dest_roledatum)) {
694 			free(id);
695 			role_datum_destroy(dest_roledatum);
696 			free(dest_roledatum);
697 			return NULL;
698 		}
699 	} else {
700 		free(id);
701 		if (dest_roledatum->flavor != isattr ? ROLE_ATTRIB : ROLE_ROLE)
702 			return NULL;
703 	}
704 
705 	return dest_roledatum;
706 }
707 
708 /* Attempt to require a symbol within the current scope.  If currently
709  * within an optional (and not its else branch), add the symbol to the
710  * required list.  Return 0 on success, 1 if caller needs to free()
711  * datum.  If symbols may not be declared here return -1.  For duplicate
712  * declarations return -2.  For all else, including out of memory,
713  * return -3..  Note that dest_value and datum_value might not be
714  * restricted pointers.
715  */
require_symbol(uint32_t symbol_type,hashtab_key_t key,hashtab_datum_t datum,uint32_t * dest_value,uint32_t * datum_value)716 int require_symbol(uint32_t symbol_type,
717 		   hashtab_key_t key, hashtab_datum_t datum,
718 		   uint32_t * dest_value, uint32_t * datum_value)
719 {
720 	avrule_decl_t *decl = stack_top->decl;
721 	int ret = create_symbol(symbol_type, key, datum, dest_value, SCOPE_REQ);
722 
723 	if (ret < 0) {
724 		return ret;
725 	}
726 
727 	if (ebitmap_set_bit(decl->required.scope + symbol_type,
728 			    *datum_value - 1, 1)) {
729 		return -3;
730 	}
731 
732 	stack_top->require_given = 1;
733 	return ret;
734 }
735 
add_perm_to_class(uint32_t perm_value,uint32_t class_value)736 int add_perm_to_class(uint32_t perm_value, uint32_t class_value)
737 {
738 	avrule_decl_t *decl = stack_top->decl;
739 	scope_index_t *scope;
740 
741 	assert(perm_value >= 1);
742 	assert(class_value >= 1);
743 	scope = &decl->required;
744 	if (class_value > scope->class_perms_len) {
745 		uint32_t i;
746 		ebitmap_t *new_map = realloc(scope->class_perms_map,
747 					     class_value * sizeof(*new_map));
748 		if (new_map == NULL) {
749 			return -1;
750 		}
751 		scope->class_perms_map = new_map;
752 		for (i = scope->class_perms_len; i < class_value; i++) {
753 			ebitmap_init(scope->class_perms_map + i);
754 		}
755 		scope->class_perms_len = class_value;
756 	}
757 	if (ebitmap_set_bit(scope->class_perms_map + class_value - 1,
758 			    perm_value - 1, 1)) {
759 		return -1;
760 	}
761 	return 0;
762 }
763 
perm_destroy(hashtab_key_t key,hashtab_datum_t datum,void * p)764 static int perm_destroy(hashtab_key_t key, hashtab_datum_t datum, void *p
765 			__attribute__ ((unused)))
766 {
767 	if (key)
768 		free(key);
769 	free(datum);
770 	return 0;
771 }
772 
class_datum_destroy(class_datum_t * cladatum)773 static void class_datum_destroy(class_datum_t * cladatum)
774 {
775 	if (cladatum != NULL) {
776 		hashtab_map(cladatum->permissions.table, perm_destroy, NULL);
777 		hashtab_destroy(cladatum->permissions.table);
778 		free(cladatum);
779 	}
780 }
781 
require_class(int pass)782 int require_class(int pass)
783 {
784 	char *class_id = queue_remove(id_queue);
785 	char *perm_id = NULL;
786 	class_datum_t *datum = NULL;
787 	perm_datum_t *perm = NULL;
788 	int ret;
789 
790 	if (pass == 2) {
791 		free(class_id);
792 		while ((perm_id = queue_remove(id_queue)) != NULL)
793 			free(perm_id);
794 		return 0;
795 	}
796 
797 	/* first add the class if it is not already there */
798 	if (class_id == NULL) {
799 		yyerror("no class name for class definition?");
800 		return -1;
801 	}
802 
803 	if ((datum = calloc(1, sizeof(*datum))) == NULL ||
804 	    symtab_init(&datum->permissions, PERM_SYMTAB_SIZE)) {
805 		yyerror("Out of memory!");
806 		class_datum_destroy(datum);
807 		return -1;
808 	}
809 	ret =
810 	    require_symbol(SYM_CLASSES, class_id, datum, &datum->s.value,
811 			   &datum->s.value);
812 	if (ret < 0) {
813 		print_error_msg(ret, SYM_CLASSES);
814 		free(class_id);
815 		class_datum_destroy(datum);
816 		return -1;
817 	}
818 
819 	if (ret == 0) {
820 		/* a new class was added; reindex everything */
821 		if (policydb_index_classes(policydbp)) {
822 			yyerror("Out of memory!");
823 			return -1;
824 		}
825 	} else {
826 		class_datum_destroy(datum);
827 		datum = hashtab_search(policydbp->p_classes.table, class_id);
828 		assert(datum);	/* the class datum should have existed */
829 		free(class_id);
830 	}
831 
832 	/* now add each of the permissions to this class's requirements */
833 	while ((perm_id = queue_remove(id_queue)) != NULL) {
834 		int allocated = 0;
835 
836 		/* Is the permission already in the table? */
837 		perm = hashtab_search(datum->permissions.table, perm_id);
838 		if (!perm && datum->comdatum)
839 			perm =
840 			    hashtab_search(datum->comdatum->permissions.table,
841 					   perm_id);
842 		if (perm) {
843 			/* Yes, drop the name. */
844 			free(perm_id);
845 		} else {
846 			/* No - allocate and insert an entry for it. */
847 			if (policydbp->policy_type == POLICY_BASE) {
848 				yyerror2
849 				    ("Base policy - require of permission %s without prior declaration.",
850 				     perm_id);
851 				free(perm_id);
852 				return -1;
853 			}
854 			if (datum->permissions.nprim >= PERM_SYMTAB_SIZE) {
855 				yyerror2("Class %s would have too many permissions "
856 					 "to fit in an access vector with permission %s",
857 					 policydbp->p_class_val_to_name[datum->s.value - 1],
858 					 perm_id);
859 				free(perm_id);
860 				return -1;
861 			}
862 			allocated = 1;
863 			if ((perm = malloc(sizeof(*perm))) == NULL) {
864 				yyerror("Out of memory!");
865 				free(perm_id);
866 				return -1;
867 			}
868 			memset(perm, 0, sizeof(*perm));
869 			ret =
870 			    hashtab_insert(datum->permissions.table, perm_id,
871 					   perm);
872 			if (ret) {
873 				yyerror("Out of memory!");
874 				free(perm_id);
875 				free(perm);
876 				return -1;
877 			}
878 			perm->s.value = datum->permissions.nprim + 1;
879 		}
880 
881 		if (add_perm_to_class(perm->s.value, datum->s.value) == -1) {
882 			yyerror("Out of memory!");
883 			return -1;
884 		}
885 
886 		/* Update number of primitives if we allocated one. */
887 		if (allocated)
888 			datum->permissions.nprim++;
889 	}
890 	return 0;
891 }
892 
require_role_or_attribute(int pass,unsigned char isattr)893 static int require_role_or_attribute(int pass, unsigned char isattr)
894 {
895 	char *key = NULL;
896 	role_datum_t *role = NULL;
897 	int ret;
898 
899 	if (pass == 2) {
900 		free(queue_remove(id_queue));
901 		return 0;
902 	}
903 
904 	ret = create_role(SCOPE_REQ, isattr, &role, &key);
905 	if (ret < 0) {
906 		return -1;
907 	}
908 
909 	free(key);
910 
911 	if (ret == 0) {
912 		ret = ebitmap_set_bit(&role->dominates, role->s.value - 1, 1);
913 		if (ret != 0) {
914 			yyerror("Out of memory");
915 			return -1;
916 		}
917 	} else {
918 		role_datum_destroy(role);
919 		free(role);
920 	}
921 
922 	return 0;
923 }
924 
require_role(int pass)925 int require_role(int pass)
926 {
927 	return require_role_or_attribute(pass, 0);
928 }
929 
require_attribute_role(int pass)930 int require_attribute_role(int pass)
931 {
932 	return require_role_or_attribute(pass, 1);
933 }
934 
require_type_or_attribute(int pass,unsigned char isattr)935 static int require_type_or_attribute(int pass, unsigned char isattr)
936 {
937 	type_datum_t *type = NULL;
938 	int ret;
939 
940 	if (pass == 2) {
941 		free(queue_remove(id_queue));
942 		return 0;
943 	}
944 
945 	ret = create_type(SCOPE_REQ, isattr, &type);
946 
947 	if (ret < 0) {
948 		return -1;
949 	}
950 
951 	return 0;
952 }
953 
require_type(int pass)954 int require_type(int pass)
955 {
956 	return require_type_or_attribute(pass, 0);
957 }
958 
require_attribute(int pass)959 int require_attribute(int pass)
960 {
961 	return require_type_or_attribute(pass, 1);
962 }
963 
require_user(int pass)964 int require_user(int pass)
965 {
966 	char *key = NULL;
967 	user_datum_t *user = NULL;
968 	int ret;
969 
970 	if (pass == 1) {
971 		free(queue_remove(id_queue));
972 		return 0;
973 	}
974 
975 	ret = create_user(SCOPE_REQ, &user, &key);
976 	if (ret < 0) {
977 		return -1;
978 	}
979 
980 	free(key);
981 
982 	if (ret == 1) {
983 		user_datum_destroy(user);
984 		free(user);
985 	}
986 
987 	return 0;
988 }
989 
require_bool_tunable(int pass,int is_tunable)990 static int require_bool_tunable(int pass, int is_tunable)
991 {
992 	char *id = queue_remove(id_queue);
993 	cond_bool_datum_t *booldatum = NULL;
994 	int retval;
995 	if (pass == 2) {
996 		free(id);
997 		return 0;
998 	}
999 	if (id == NULL) {
1000 		yyerror("no boolean name");
1001 		return -1;
1002 	}
1003 	if ((booldatum = calloc(1, sizeof(*booldatum))) == NULL) {
1004 		cond_destroy_bool(id, booldatum, NULL);
1005 		yyerror("Out of memory!");
1006 		return -1;
1007 	}
1008 	if (is_tunable)
1009 		booldatum->flags |= COND_BOOL_FLAGS_TUNABLE;
1010 	retval =
1011 	    require_symbol(SYM_BOOLS, id, booldatum,
1012 			   &booldatum->s.value, &booldatum->s.value);
1013 	if (retval != 0) {
1014 		cond_destroy_bool(id, booldatum, NULL);
1015 		if (retval < 0) {
1016 			print_error_msg(retval, SYM_BOOLS);
1017 			return -1;
1018 		}
1019 	}
1020 
1021 	return 0;
1022 }
1023 
require_bool(int pass)1024 int require_bool(int pass)
1025 {
1026 	return require_bool_tunable(pass, 0);
1027 }
1028 
require_tunable(int pass)1029 int require_tunable(int pass)
1030 {
1031 	return require_bool_tunable(pass, 1);
1032 }
1033 
require_sens(int pass)1034 int require_sens(int pass)
1035 {
1036 	char *id = queue_remove(id_queue);
1037 	level_datum_t *level = NULL;
1038 	int retval;
1039 	if (pass == 2) {
1040 		free(id);
1041 		return 0;
1042 	}
1043 	if (!id) {
1044 		yyerror("no sensitivity name");
1045 		return -1;
1046 	}
1047 	level = malloc(sizeof(level_datum_t));
1048 	if (!level) {
1049 		free(id);
1050 		yyerror("Out of memory!");
1051 		return -1;
1052 	}
1053 	level_datum_init(level);
1054 	level->level = malloc(sizeof(mls_level_t));
1055 	if (!level->level) {
1056 		free(id);
1057 		level_datum_destroy(level);
1058 		free(level);
1059 		yyerror("Out of memory!");
1060 		return -1;
1061 	}
1062 	mls_level_init(level->level);
1063 	retval = require_symbol(SYM_LEVELS, id, level,
1064 				&level->level->sens, &level->level->sens);
1065 	if (retval != 0) {
1066 		free(id);
1067 		mls_level_destroy(level->level);
1068 		free(level->level);
1069 		level_datum_destroy(level);
1070 		free(level);
1071 		if (retval < 0) {
1072 			print_error_msg(retval, SYM_LEVELS);
1073 			return -1;
1074 		}
1075 	}
1076 
1077 	return 0;
1078 }
1079 
require_cat(int pass)1080 int require_cat(int pass)
1081 {
1082 	char *id = queue_remove(id_queue);
1083 	cat_datum_t *cat = NULL;
1084 	int retval;
1085 	if (pass == 2) {
1086 		free(id);
1087 		return 0;
1088 	}
1089 	if (!id) {
1090 		yyerror("no category name");
1091 		return -1;
1092 	}
1093 	cat = malloc(sizeof(cat_datum_t));
1094 	if (!cat) {
1095 		free(id);
1096 		yyerror("Out of memory!");
1097 		return -1;
1098 	}
1099 	cat_datum_init(cat);
1100 
1101 	retval = require_symbol(SYM_CATS, id, cat,
1102 				&cat->s.value, &cat->s.value);
1103 	if (retval != 0) {
1104 		free(id);
1105 		cat_datum_destroy(cat);
1106 		free(cat);
1107 		if (retval < 0) {
1108 			print_error_msg(retval, SYM_CATS);
1109 			return -1;
1110 		}
1111 	}
1112 
1113 	return 0;
1114 }
1115 
is_scope_in_stack(const scope_datum_t * scope,const scope_stack_t * stack)1116 static int is_scope_in_stack(const scope_datum_t * scope, const scope_stack_t * stack)
1117 {
1118 	uint32_t i;
1119 	if (stack == NULL) {
1120 		return 0;	/* no matching scope found */
1121 	}
1122 	if (stack->type == 1) {
1123 		const avrule_decl_t *decl = stack->decl;
1124 		for (i = 0; i < scope->decl_ids_len; i++) {
1125 			if (scope->decl_ids[i] == decl->decl_id) {
1126 				return 1;
1127 			}
1128 		}
1129 	} else {
1130 		/* note that conditionals can't declare or require
1131 		 * symbols, so skip this level */
1132 	}
1133 
1134 	/* not within scope of this stack, so try its parent */
1135 	return is_scope_in_stack(scope, stack->parent);
1136 }
1137 
is_id_in_scope(uint32_t symbol_type,const_hashtab_key_t id)1138 int is_id_in_scope(uint32_t symbol_type, const_hashtab_key_t id)
1139 {
1140 	const scope_datum_t *scope =
1141 	    (scope_datum_t *) hashtab_search(policydbp->scope[symbol_type].
1142 					     table, id);
1143 	if (scope == NULL) {
1144 		return 1;	/* id is not known, so return success */
1145 	}
1146 	return is_scope_in_stack(scope, stack_top);
1147 }
1148 
is_perm_in_scope_index(uint32_t perm_value,uint32_t class_value,const scope_index_t * scope)1149 static int is_perm_in_scope_index(uint32_t perm_value, uint32_t class_value,
1150 				  const scope_index_t * scope)
1151 {
1152 	if (class_value > scope->class_perms_len) {
1153 		return 1;
1154 	}
1155 	if (ebitmap_get_bit(scope->class_perms_map + class_value - 1,
1156 			    perm_value - 1)) {
1157 		return 1;
1158 	}
1159 	return 0;
1160 }
1161 
is_perm_in_stack(uint32_t perm_value,uint32_t class_value,const scope_stack_t * stack)1162 static int is_perm_in_stack(uint32_t perm_value, uint32_t class_value,
1163 			    const scope_stack_t * stack)
1164 {
1165 	if (stack == NULL) {
1166 		return 0;	/* no matching scope found */
1167 	}
1168 	if (stack->type == 1) {
1169 		avrule_decl_t *decl = stack->decl;
1170 		if (is_perm_in_scope_index
1171 		    (perm_value, class_value, &decl->required)
1172 		    || is_perm_in_scope_index(perm_value, class_value,
1173 					      &decl->declared)) {
1174 			return 1;
1175 		}
1176 	} else {
1177 		/* note that conditionals can't declare or require
1178 		 * symbols, so skip this level */
1179 	}
1180 
1181 	/* not within scope of this stack, so try its parent */
1182 	return is_perm_in_stack(perm_value, class_value, stack->parent);
1183 }
1184 
is_perm_in_scope(const_hashtab_key_t perm_id,const_hashtab_key_t class_id)1185 int is_perm_in_scope(const_hashtab_key_t perm_id, const_hashtab_key_t class_id)
1186 {
1187 	const class_datum_t *cladatum =
1188 	    (class_datum_t *) hashtab_search(policydbp->p_classes.table,
1189 					     class_id);
1190 	const perm_datum_t *perdatum;
1191 	if (cladatum == NULL) {
1192 		return 1;
1193 	}
1194 	perdatum = (perm_datum_t *) hashtab_search(cladatum->permissions.table,
1195 						   perm_id);
1196 	if (perdatum == NULL) {
1197 		return 1;
1198 	}
1199 	return is_perm_in_stack(perdatum->s.value, cladatum->s.value,
1200 				stack_top);
1201 }
1202 
get_current_cond_list(cond_list_t * cond)1203 cond_list_t *get_current_cond_list(cond_list_t * cond)
1204 {
1205 	/* FIX ME: do something different here if in a nested
1206 	 * conditional? */
1207 	avrule_decl_t *decl = stack_top->decl;
1208 	return get_decl_cond_list(policydbp, decl, cond);
1209 }
1210 
1211 /* Append the new conditional node to the existing ones.  During
1212  * expansion the list will be reversed -- i.e., the last AV rule will
1213  * be the first one listed in the policy.  This matches the behavior
1214  * of the upstream compiler. */
append_cond_list(cond_list_t * cond)1215 void append_cond_list(cond_list_t * cond)
1216 {
1217 	cond_list_t *old_cond = get_current_cond_list(cond);
1218 	avrule_t *tmp;
1219 	assert(old_cond != NULL);	/* probably out of memory */
1220 	if (old_cond->avtrue_list == NULL) {
1221 		old_cond->avtrue_list = cond->avtrue_list;
1222 	} else {
1223 		for (tmp = old_cond->avtrue_list; tmp->next != NULL;
1224 		     tmp = tmp->next) ;
1225 		tmp->next = cond->avtrue_list;
1226 	}
1227 	if (old_cond->avfalse_list == NULL) {
1228 		old_cond->avfalse_list = cond->avfalse_list;
1229 	} else {
1230 		for (tmp = old_cond->avfalse_list; tmp->next != NULL;
1231 		     tmp = tmp->next) ;
1232 		tmp->next = cond->avfalse_list;
1233 	}
1234 
1235 	old_cond->flags |= cond->flags;
1236 }
1237 
append_avrule(avrule_t * avrule)1238 void append_avrule(avrule_t * avrule)
1239 {
1240 	avrule_decl_t *decl = stack_top->decl;
1241 
1242 	/* currently avrules follow a completely different code path
1243 	 * for handling avrules and compute types
1244 	 * (define_cond_avrule_te_avtab, define_cond_compute_type);
1245 	 * therefore there ought never be a conditional on top of the
1246 	 * scope stack */
1247 	assert(stack_top->type == 1);
1248 
1249 	if (stack_top->last_avrule == NULL) {
1250 		decl->avrules = avrule;
1251 	} else {
1252 		stack_top->last_avrule->next = avrule;
1253 	}
1254 	stack_top->last_avrule = avrule;
1255 }
1256 
1257 /* this doesn't actually append, but really prepends it */
append_role_trans(role_trans_rule_t * role_tr_rules)1258 void append_role_trans(role_trans_rule_t * role_tr_rules)
1259 {
1260 	avrule_decl_t *decl = stack_top->decl;
1261 
1262 	/* role transitions are not allowed within conditionals */
1263 	assert(stack_top->type == 1);
1264 
1265 	role_tr_rules->next = decl->role_tr_rules;
1266 	decl->role_tr_rules = role_tr_rules;
1267 }
1268 
1269 /* this doesn't actually append, but really prepends it */
append_role_allow(role_allow_rule_t * role_allow_rules)1270 void append_role_allow(role_allow_rule_t * role_allow_rules)
1271 {
1272 	avrule_decl_t *decl = stack_top->decl;
1273 
1274 	/* role allows are not allowed within conditionals */
1275 	assert(stack_top->type == 1);
1276 
1277 	role_allow_rules->next = decl->role_allow_rules;
1278 	decl->role_allow_rules = role_allow_rules;
1279 }
1280 
1281 /* this doesn't actually append, but really prepends it */
append_filename_trans(filename_trans_rule_t * filename_trans_rules)1282 void append_filename_trans(filename_trans_rule_t * filename_trans_rules)
1283 {
1284 	avrule_decl_t *decl = stack_top->decl;
1285 
1286 	/* filename transitions are not allowed within conditionals */
1287 	assert(stack_top->type == 1);
1288 
1289 	filename_trans_rules->next = decl->filename_trans_rules;
1290 	decl->filename_trans_rules = filename_trans_rules;
1291 }
1292 
1293 /* this doesn't actually append, but really prepends it */
append_range_trans(range_trans_rule_t * range_tr_rules)1294 void append_range_trans(range_trans_rule_t * range_tr_rules)
1295 {
1296 	avrule_decl_t *decl = stack_top->decl;
1297 
1298 	/* range transitions are not allowed within conditionals */
1299 	assert(stack_top->type == 1);
1300 
1301 	range_tr_rules->next = decl->range_tr_rules;
1302 	decl->range_tr_rules = range_tr_rules;
1303 }
1304 
begin_optional(int pass)1305 int begin_optional(int pass)
1306 {
1307 	avrule_block_t *block = NULL;
1308 	avrule_decl_t *decl;
1309 	if (pass == 1) {
1310 		/* allocate a new avrule block for this optional block */
1311 		if ((block = avrule_block_create()) == NULL ||
1312 		    (decl = avrule_decl_create(next_decl_id)) == NULL) {
1313 			goto cleanup;
1314 		}
1315 		block->flags |= AVRULE_OPTIONAL;
1316 		block->branch_list = decl;
1317 		last_block->next = block;
1318 	} else {
1319 		/* select the next block from the chain built during pass 1 */
1320 		block = last_block->next;
1321 		assert(block != NULL &&
1322 		       block->branch_list != NULL &&
1323 		       block->branch_list->decl_id == next_decl_id);
1324 		decl = block->branch_list;
1325 	}
1326 	if (push_stack(1, block, decl) == -1) {
1327 		goto cleanup;
1328 	}
1329 	stack_top->last_avrule = NULL;
1330 	last_block = block;
1331 	next_decl_id++;
1332 	return 0;
1333       cleanup:
1334 	yyerror("Out of memory!");
1335 	avrule_block_destroy(block);
1336 	return -1;
1337 }
1338 
end_optional(int pass)1339 int end_optional(int pass __attribute__ ((unused)))
1340 {
1341 	/* once nested conditionals are allowed, do the stack unfolding here */
1342 	pop_stack();
1343 	return 0;
1344 }
1345 
begin_optional_else(int pass)1346 int begin_optional_else(int pass)
1347 {
1348 	avrule_decl_t *decl;
1349 	assert(stack_top->type == 1 && stack_top->in_else == 0);
1350 	if (pass == 1) {
1351 		/* allocate a new declaration and add it to the
1352 		 * current chain */
1353 		if ((decl = avrule_decl_create(next_decl_id)) == NULL) {
1354 			yyerror("Out of memory!");
1355 			return -1;
1356 		}
1357 		stack_top->decl->next = decl;
1358 	} else {
1359 		/* pick the (hopefully last) declaration of this
1360 		   avrule block, built from pass 1 */
1361 		decl = stack_top->decl->next;
1362 		assert(decl != NULL &&
1363 		       decl->next == NULL && decl->decl_id == next_decl_id);
1364 	}
1365 	stack_top->in_else = 1;
1366 	stack_top->decl = decl;
1367 	stack_top->last_avrule = NULL;
1368 	stack_top->require_given = 0;
1369 	next_decl_id++;
1370 	return 0;
1371 }
1372 
copy_requirements(avrule_decl_t * dest,const scope_stack_t * stack)1373 static int copy_requirements(avrule_decl_t * dest, const scope_stack_t * stack)
1374 {
1375 	uint32_t i;
1376 	if (stack == NULL) {
1377 		return 0;
1378 	}
1379 	if (stack->type == 1) {
1380 		const scope_index_t *src_scope = &stack->decl->required;
1381 		scope_index_t *dest_scope = &dest->required;
1382 		for (i = 0; i < SYM_NUM; i++) {
1383 			const ebitmap_t *src_bitmap = &src_scope->scope[i];
1384 			ebitmap_t *dest_bitmap = &dest_scope->scope[i];
1385 			if (ebitmap_union(dest_bitmap, src_bitmap)) {
1386 				yyerror("Out of memory!");
1387 				return -1;
1388 			}
1389 		}
1390 		/* now copy class permissions */
1391 		if (src_scope->class_perms_len > dest_scope->class_perms_len) {
1392 			ebitmap_t *new_map =
1393 			    realloc(dest_scope->class_perms_map,
1394 				    src_scope->class_perms_len *
1395 				    sizeof(*new_map));
1396 			if (new_map == NULL) {
1397 				yyerror("Out of memory!");
1398 				return -1;
1399 			}
1400 			dest_scope->class_perms_map = new_map;
1401 			for (i = dest_scope->class_perms_len;
1402 			     i < src_scope->class_perms_len; i++) {
1403 				ebitmap_init(dest_scope->class_perms_map + i);
1404 			}
1405 			dest_scope->class_perms_len =
1406 			    src_scope->class_perms_len;
1407 		}
1408 		for (i = 0; i < src_scope->class_perms_len; i++) {
1409 			const ebitmap_t *src_bitmap = &src_scope->class_perms_map[i];
1410 			ebitmap_t *dest_bitmap =
1411 			    &dest_scope->class_perms_map[i];
1412 			if (ebitmap_union(dest_bitmap, src_bitmap)) {
1413 				yyerror("Out of memory!");
1414 				return -1;
1415 			}
1416 		}
1417 	}
1418 	return copy_requirements(dest, stack->parent);
1419 }
1420 
1421 /* During pass 1, check that at least one thing was required within
1422  * this block, for those places where a REQUIRED is necessary.  During
1423  * pass 2, have this block inherit its parents' requirements.  Return
1424  * 0 on success, -1 on failure. */
end_avrule_block(int pass)1425 int end_avrule_block(int pass)
1426 {
1427 	avrule_decl_t *decl = stack_top->decl;
1428 	assert(stack_top->type == 1);
1429 	if (pass == 2) {
1430 		/* this avrule_decl inherits all of its parents'
1431 		 * requirements */
1432 		if (copy_requirements(decl, stack_top->parent) == -1) {
1433 			return -1;
1434 		}
1435 		return 0;
1436 	}
1437 	if (!stack_top->in_else && !stack_top->require_given) {
1438 		if (policydbp->policy_type == POLICY_BASE
1439 		    && stack_top->parent != NULL) {
1440 			/* if this is base no require should be in the global block */
1441 			return 0;
1442 		} else {
1443 			/* non-ELSE branches must have at least one thing required */
1444 			yyerror("This block has no require section.");
1445 			return -1;
1446 		}
1447 	}
1448 	return 0;
1449 }
1450 
1451 /* Push a new scope on to the stack and update the 'last' pointer.
1452  * Return 0 on success, -1 if out * of memory. */
push_stack(int stack_type,...)1453 static int push_stack(int stack_type, ...)
1454 {
1455 	scope_stack_t *s = calloc(1, sizeof(*s));
1456 	va_list ap;
1457 	if (s == NULL) {
1458 		return -1;
1459 	}
1460 	va_start(ap, stack_type);
1461 	switch (s->type = stack_type) {
1462 	case 1:{
1463 			s->u.avrule = va_arg(ap, avrule_block_t *);
1464 			s->decl = va_arg(ap, avrule_decl_t *);
1465 			break;
1466 		}
1467 	case 2:{
1468 			s->u.cond_list = va_arg(ap, cond_list_t *);
1469 			break;
1470 		}
1471 	default:
1472 		/* invalid stack type given */
1473 		assert(0);
1474 	}
1475 	va_end(ap);
1476 	s->parent = stack_top;
1477 	s->child = NULL;
1478 	stack_top = s;
1479 	return 0;
1480 }
1481 
1482 /* Pop off the most recently added from the stack.  Update the 'last'
1483  * pointer. */
pop_stack(void)1484 static void pop_stack(void)
1485 {
1486 	scope_stack_t *parent;
1487 	assert(stack_top != NULL);
1488 	parent = stack_top->parent;
1489 	if (parent != NULL) {
1490 		parent->child = NULL;
1491 	}
1492 	free(stack_top);
1493 	stack_top = parent;
1494 }
1495