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