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