On Wed, 31 May 2023 at 13:32, Juraj Marcin <juraj@xxxxxxxxxxxxxxx> wrote: > > Currently, filename transitions are stored separately from other type > enforcement rules. This leads to possibly sub-optimal performance and > makes further improvements cumbersome. > > This patch adds a symbol table of filename transitions to the transition > structure added to avtab in the previous patch. It also implements > functions required for reading and writing filename transitions in > kernel policy format and updates computation of new type to use filename > transitions embedded in avtab. Last but not least, it updates the > conflict check in the conditional avtab to account for empty transitions > in the non-conditional avtab. > > These changes are expected to cause higher memory usage, as now there > needs to be a filename transition structure for every stype. This patch > effectively undoes most of the commit c3a276111ea2 ("selinux: optimize > storage of filename transitions"), but this will be mitigated by > providing support for matching prefix/suffix of the filename for > filename transitions in future patches which will reduce to need to have > so many of them. > > On the other hand, the changes do not significantly slow down the > creation of new files. > > Kernel | Mem | Create test_tty | Create test_tty | osbench [1] > | usage | (real time) | (kernel time) | create_files > -----------+-------+-----------------+-----------------+-------------- > reference | 155MB | 1.3440 ms/file | 1.0071 ms/file | 10.6507 us/file > this patch | 198MB | 1.3912 ms/file | 1.0172 ms/file | 10.5567 us/file > > Create test_tty benchmark: > > mknod /dev/test_tty c 4 1 > time for i in `seq 1 10000`; do > mknod /dev/test_tty$i c 4 1 > done > > This benchmark should simulate the worst case scenario as many filename > transitions affect files created in the /dev directory. > > [1] https://github.com/mbitsnbites/osbench > > Reviewed-by: Ondrej Mosnacek <omosnace@xxxxxxxxxx> > Signed-off-by: Juraj Marcin <juraj@xxxxxxxxxxxxxxx> > --- > security/selinux/ss/avtab.c | 516 ++++++++++++++++++++++++++++++ > security/selinux/ss/avtab.h | 7 + > security/selinux/ss/conditional.c | 6 +- > security/selinux/ss/hashtab.h | 6 + > security/selinux/ss/policydb.c | 399 +---------------------- > security/selinux/ss/policydb.h | 25 +- > security/selinux/ss/services.c | 54 +--- > 7 files changed, 555 insertions(+), 458 deletions(-) > > diff --git a/security/selinux/ss/avtab.c b/security/selinux/ss/avtab.c > index a7f348e4509d..fe921e1586e5 100644 > --- a/security/selinux/ss/avtab.c > +++ b/security/selinux/ss/avtab.c > @@ -22,6 +22,7 @@ > #include <linux/errno.h> > #include "avtab.h" > #include "policydb.h" > +#include "hashtab.h" > > static struct kmem_cache *avtab_node_cachep __ro_after_init; > static struct kmem_cache *avtab_trans_cachep __ro_after_init; > @@ -286,6 +287,19 @@ avtab_search_node_next(struct avtab_node *node, int specified) > return NULL; > } > > +static int avtab_trans_destroy_helper(void *k, void *d, void *args) > +{ > + kfree(k); > + kfree(d); > + return 0; > +} > + > +static void avtab_trans_destroy(struct avtab_trans *trans) > +{ > + hashtab_map(&trans->name_trans.table, avtab_trans_destroy_helper, NULL); > + hashtab_destroy(&trans->name_trans.table); > +} > + > void avtab_destroy(struct avtab *h) > { > int i; > @@ -303,6 +317,7 @@ void avtab_destroy(struct avtab *h) > kmem_cache_free(avtab_xperms_cachep, > temp->datum.u.xperms); > } else if (temp->key.specified & AVTAB_TRANSITION) { > + avtab_trans_destroy(temp->datum.u.trans); > kmem_cache_free(avtab_trans_cachep, > temp->datum.u.trans); > } > @@ -587,6 +602,7 @@ int avtab_read_item(struct avtab *a, void *fp, struct policydb *pol, > if (key.specified & AVTAB_TRANSITION) { > if (!policydb_type_isvalid(pol, datum.u.trans->otype)) { > pr_err("SELinux: avtab: invalid transition type\n"); > + avtab_trans_destroy(&trans); > return -EINVAL; > } > } else if (key.specified & AVTAB_TYPE) { > @@ -596,6 +612,8 @@ int avtab_read_item(struct avtab *a, void *fp, struct policydb *pol, > } > } > rc = insertf(a, &key, &datum, p); > + if (rc && key.specified & AVTAB_TRANSITION) > + avtab_trans_destroy(&trans); > return rc; > } > > @@ -656,6 +674,10 @@ int avtab_write_item(struct policydb *p, const struct avtab_node *cur, void *fp) > int rc; > unsigned int i; > > + if (cur->key.specified & AVTAB_TRANSITION && > + !cur->datum.u.trans->otype) > + return 0; > + > buf16[0] = cpu_to_le16(cur->key.source_type); > buf16[1] = cpu_to_le16(cur->key.target_type); > buf16[2] = cpu_to_le16(cur->key.target_class); > @@ -723,3 +745,497 @@ void __init avtab_cache_init(void) > sizeof(struct avtab_extended_perms), > 0, SLAB_PANIC, NULL); > } > + > +/* policydb filename transitions compatibility */ > + > +static int avtab_insert_filename_trans(struct avtab *a, > + const struct avtab_key *key, > + char *name, u32 otype) > +{ > + int rc; > + struct avtab_node *node; > + struct avtab_trans new_trans = {0}; > + struct avtab_datum new_datum = {.u.trans = &new_trans}; > + struct avtab_datum *datum; > + u32 *otype_datum = NULL; > + > + datum = avtab_search(a, key); > + if (!datum) { > + /* > + * insert is acctually unique, but with this function we can get > + * the inserted node and therefore the datum > + */ > + node = avtab_insert_nonunique(a, key, &new_datum); > + if (!node) > + return -ENOMEM; > + datum = &node->datum; > + } > + > + if (hashtab_is_empty(&datum->u.trans->name_trans.table)) { > + rc = symtab_init(&datum->u.trans->name_trans, 1 << 8); > + if (rc) > + return rc; > + } > + > + otype_datum = kmalloc(sizeof(u32), GFP_KERNEL); > + if (!otype_datum) > + return -ENOMEM; > + *otype_datum = otype; > + > + rc = symtab_insert(&datum->u.trans->name_trans, name, otype_datum); > + if (rc) > + kfree(otype_datum); > + > + return rc; > +} > + > +static int filename_trans_read_item(struct avtab *a, void *fp) > +{ > + int rc; > + __le32 buf32[4]; > + u32 len, otype; > + char *name = NULL; > + struct avtab_key key; > + > + /* read length of the name */ > + rc = next_entry(buf32, fp, sizeof(u32)); > + if (rc) > + return rc; > + len = le32_to_cpu(buf32[0]); > + > + /* read the name */ > + rc = str_read(&name, GFP_KERNEL, fp, len); > + if (rc) > + return rc; > + > + /* read stype, ttype, tclass and otype */ > + rc = next_entry(buf32, fp, sizeof(u32) * 4); > + if (rc) > + goto bad; > + > + key.source_type = le32_to_cpu(buf32[0]); > + key.target_type = le32_to_cpu(buf32[1]); > + key.target_class = le32_to_cpu(buf32[2]); > + key.specified = AVTAB_TRANSITION; > + > + otype = le32_to_cpu(buf32[3]); > + > + rc = avtab_insert_filename_trans(a, &key, name, otype); > + if (rc) > + goto bad; > + > + return rc; > + > +bad: > + kfree(name); > + return rc; > +} > + > +static int filename_trans_comp_read_item(struct avtab *a, void *fp) > +{ > + int rc; > + __le32 buf32[3]; > + u32 len, ndatum, i, bit, otype; > + char *name = NULL, *name_copy = NULL; > + struct avtab_key key; > + struct ebitmap stypes; > + struct ebitmap_node *node; > + > + /* read length of the name */ > + rc = next_entry(buf32, fp, sizeof(u32)); > + if (rc) > + return rc; > + len = le32_to_cpu(*buf32); > + > + /* read the name */ > + rc = str_read(&name, GFP_KERNEL, fp, len); > + if (rc) > + goto out; > + > + /* read target type, target class and number of elements for key */ > + rc = next_entry(buf32, fp, sizeof(u32) * 3); > + if (rc) > + goto out; > + > + key.specified = AVTAB_TRANSITION; > + key.target_type = le32_to_cpu(buf32[0]); > + key.target_class = le32_to_cpu(buf32[1]); > + > + ndatum = le32_to_cpu(buf32[2]); > + if (ndatum == 0) { > + pr_err("SELinux: Filename transition key with no datum\n"); > + rc = -ENOENT; > + goto out; > + } > + > + for (i = 0; i < ndatum; i++) { > + rc = ebitmap_read(&stypes, fp); > + if (rc) > + goto out; > + > + rc = next_entry(buf32, fp, sizeof(u32)); > + if (rc) { > + ebitmap_destroy(&stypes); > + goto out; > + } > + otype = le32_to_cpu(*buf32); > + > + ebitmap_for_each_positive_bit(&stypes, node, bit) { > + key.source_type = bit + 1; > + > + name_copy = kmemdup(name, len + 1, GFP_KERNEL); > + if (!name_copy) { > + ebitmap_destroy(&stypes); > + goto out; > + } > + > + rc = avtab_insert_filename_trans(a, &key, name_copy, > + otype); > + if (rc) { > + ebitmap_destroy(&stypes); > + kfree(name_copy); > + goto out; > + } > + } > + > + ebitmap_destroy(&stypes); > + } > + rc = 0; > + > +out: > + kfree(name); > + return rc; > +} > + > +int avtab_filename_trans_read(struct avtab *a, void *fp, struct policydb *p) > +{ > + int rc; > + __le32 buf[1]; > + u32 nel, i; > + > + if (p->policyvers < POLICYDB_VERSION_FILENAME_TRANS) > + return 0; > + > + rc = next_entry(buf, fp, sizeof(u32)); > + if (rc) > + return rc; > + nel = le32_to_cpu(buf[0]); > + > + if (p->policyvers < POLICYDB_VERSION_COMP_FTRANS) { > + for (i = 0; i < nel; i++) { > + rc = filename_trans_read_item(a, fp); > + if (rc) > + return rc; > + } > + } else { > + for (i = 0; i < nel; i++) { > + rc = filename_trans_comp_read_item(a, fp); > + if (rc) > + return rc; > + } > + } > + > + return 0; > +} > + > + > +struct filenametr_write_args { > + void *fp; > + struct avtab_key *key; > +}; > + > +static int filenametr_write_helper(void *k, void *d, void *a) > +{ > + char *name = k; > + u32 *otype = d; > + struct filenametr_write_args *args = a; > + int rc; > + u32 len; > + __le32 buf32[4]; > + > + len = strlen(name); > + buf32[0] = cpu_to_le32(len); > + rc = put_entry(buf32, sizeof(u32), 1, args->fp); > + if (rc) > + return rc; > + > + rc = put_entry(name, sizeof(char), len, args->fp); > + if (rc) > + return rc; > + > + buf32[0] = cpu_to_le32(args->key->source_type); > + buf32[1] = cpu_to_le32(args->key->target_type); > + buf32[2] = cpu_to_le32(args->key->target_class); > + buf32[3] = cpu_to_le32(*otype); > + > + rc = put_entry(buf32, sizeof(u32), 4, args->fp); > + if (rc) > + return rc; > + > + return 0; > +} > + > +struct filenametr_key { > + u32 ttype; /* parent dir context */ > + u16 tclass; /* class of new object */ > + const char *name; /* last path component */ > +}; > + > +struct filenametr_datum { > + struct ebitmap stypes; /* bitmap of source types for this otype */ > + u32 otype; /* resulting type of new object */ > + struct filenametr_datum *next; /* record for next otype*/ > +}; > + > +static int filenametr_comp_write_helper(void *k, void *d, void *fp) > +{ > + struct filenametr_key *key = k; > + struct filenametr_datum *datum = d; > + __le32 buf[3]; > + int rc; > + u32 ndatum, len = strlen(key->name); > + struct filenametr_datum *cur; > + > + buf[0] = cpu_to_le32(len); > + rc = put_entry(buf, sizeof(u32), 1, fp); > + if (rc) > + return rc; > + > + rc = put_entry(key->name, sizeof(char), len, fp); > + if (rc) > + return rc; > + > + ndatum = 0; > + cur = datum; > + do { > + ndatum++; > + cur = cur->next; > + } while (unlikely(cur)); > + > + buf[0] = cpu_to_le32(key->ttype); > + buf[1] = cpu_to_le32(key->tclass); > + buf[2] = cpu_to_le32(ndatum); > + rc = put_entry(buf, sizeof(u32), 3, fp); > + if (rc) > + return rc; > + > + cur = datum; > + do { > + rc = ebitmap_write(&cur->stypes, fp); > + if (rc) > + return rc; > + > + buf[0] = cpu_to_le32(cur->otype); > + rc = put_entry(buf, sizeof(u32), 1, fp); > + if (rc) > + return rc; > + > + cur = cur->next; > + } while (unlikely(cur)); > + > + return 0; > +} > + > +static int filenametr_destroy(void *k, void *d, void *args) > +{ > + struct filenametr_key *key = k; > + struct filenametr_datum *datum = d; > + struct filenametr_datum *next; > + > + kfree(key); The member `name` should be free'd as well. I am seeing a lot of the following reports: unreferenced object 0xffff888135042130 (size 16): comm "audit2allow", pid 522, jiffies 4294901631 (age 736.956s) hex dump (first 16 bytes): 67 72 6f 75 70 2e 65 64 69 74 00 6b 6b 6b 6b a5 group.edit.kkkk. backtrace: [<0000000000000000>] __kmalloc_node_track_caller+0x52/0x1a0 [<0000000000000000>] kmemdup+0x1e/0x40 [<0000000000000000>] filenametr_tab_insert+0x6c5/0x11e0 [<0000000000000000>] hashtab_map+0xe4/0x160 [<0000000000000000>] avtab_filename_trans_write+0x4ab/0x760 [<0000000000000000>] policydb_write+0x895/0xc20 [<0000000000000000>] security_read_policy+0x150/0x290 [<0000000000000000>] sel_open_policy+0x24b/0x510 [<0000000000000000>] do_dentry_open+0x5cf/0xfb0 [<0000000000000000>] do_open+0x418/0xf30 [<0000000000000000>] path_openat+0x23e/0x660 [<0000000000000000>] do_filp_open+0x1f2/0x430 [<0000000000000000>] do_sys_openat2+0x143/0x410 [<0000000000000000>] __x64_sys_openat+0x11f/0x1d0 [<0000000000000000>] do_syscall_64+0x35/0x80 [<0000000000000000>] entry_SYSCALL_64_after_hwframe+0x63/0xcd > + do { > + ebitmap_destroy(&datum->stypes); > + next = datum->next; > + kfree(datum); > + datum = next; > + } while (unlikely(datum)); > + cond_resched(); > + return 0; > +} > + > +static u32 filenametr_hash(const void *k) > +{ > + const struct filenametr_key *ft = k; > + unsigned long hash; > + unsigned int byte_num; > + unsigned char focus; > + > + hash = ft->ttype ^ ft->tclass; > + > + byte_num = 0; > + while ((focus = ft->name[byte_num++])) > + hash = partial_name_hash(focus, hash); > + return hash; > +} > + > +static int filenametr_cmp(const void *k1, const void *k2) > +{ > + const struct filenametr_key *ft1 = k1; > + const struct filenametr_key *ft2 = k2; > + int v; > + > + v = ft1->ttype - ft2->ttype; > + if (v) > + return v; > + > + v = ft1->tclass - ft2->tclass; > + if (v) > + return v; > + > + return strcmp(ft1->name, ft2->name); > +} > + > +static const struct hashtab_key_params filenametr_key_params = { > + .hash = filenametr_hash, > + .cmp = filenametr_cmp, > +}; > + > +struct filenametr_tab_insert_args { > + struct avtab_key *key; > + struct hashtab *tab; > +}; > + > +static int filenametr_tab_insert(void *k, void *d, void *a) > +{ > + char *name = k; > + u32 *otype = d; > + struct filenametr_tab_insert_args *args = a; > + struct filenametr_key key, *ft = NULL; > + struct filenametr_datum *last, *datum = NULL; > + int rc; > + > + key.ttype = args->key->target_type; > + key.tclass = args->key->target_class; > + key.name = name; > + > + last = NULL; > + datum = hashtab_search(args->tab, &key, filenametr_key_params); > + while (datum) { > + if (unlikely(ebitmap_get_bit(&datum->stypes, > + args->key->source_type - 1))) { > + /* conflicting/duplicate rules are ignored */ > + datum = NULL; > + goto bad; > + } > + if (likely(datum->otype == *otype)) > + break; > + last = datum; > + datum = datum->next; > + } > + if (!datum) { > + rc = -ENOMEM; > + datum = kmalloc(sizeof(*datum), GFP_KERNEL); > + if (!datum) > + goto bad; > + > + ebitmap_init(&datum->stypes); > + datum->otype = *otype; > + datum->next = NULL; > + > + if (unlikely(last)) { > + last->next = datum; > + } else { > + rc = -ENOMEM; > + ft = kmemdup(&key, sizeof(key), GFP_KERNEL); > + if (!ft) > + goto bad; > + > + ft->name = kmemdup(key.name, strlen(key.name) + 1, > + GFP_KERNEL); > + if (!ft->name) > + goto bad; > + > + rc = hashtab_insert(args->tab, ft, datum, > + filenametr_key_params); > + if (rc) > + goto bad; > + } > + } > + > + return ebitmap_set_bit(&datum->stypes, args->key->source_type - 1, 1); > + > +bad: > + if (ft) > + kfree(ft->name); > + kfree(ft); > + kfree(datum); > + return rc; > +} > + > +int avtab_filename_trans_write(struct policydb *p, struct avtab *a, void *fp) > +{ > + int rc; > + __le32 buf32[1]; > + u32 i, nel = 0; > + struct avtab_node *cur; > + struct hashtab fnts_tab; > + struct filenametr_tab_insert_args tab_insert_args = {.tab = &fnts_tab}; > + struct filenametr_write_args write_args = {.fp = fp}; > + > + if (p->policyvers < POLICYDB_VERSION_FILENAME_TRANS) > + return 0; > + > + /* count number of filename transitions */ > + for (i = 0; i < a->nslot; i++) { > + for (cur = a->htable[i]; cur; cur = cur->next) { > + if (cur->key.specified & AVTAB_TRANSITION) > + nel += cur->datum.u.trans->name_trans.table.nel; > + } > + } > + > + if (p->policyvers < POLICYDB_VERSION_COMP_FTRANS) { > + buf32[0] = cpu_to_le32(nel); > + rc = put_entry(buf32, sizeof(u32), 1, fp); > + if (rc) > + return rc; > + > + /* write filename transitions */ > + for (i = 0; i < a->nslot; i++) { > + for (cur = a->htable[i]; cur; cur = cur->next) { > + if (cur->key.specified & AVTAB_TRANSITION) { > + write_args.key = &cur->key; > + rc = hashtab_map(&cur->datum.u.trans->name_trans.table, > + filenametr_write_helper, > + &write_args); > + if (rc) > + return rc; > + } > + } > + } > + > + return 0; > + } > + > + /* init temp filename transition table */ > + rc = hashtab_init(&fnts_tab, nel); > + if (rc) > + return rc; > + > + for (i = 0; i < a->nslot; i++) { > + for (cur = a->htable[i]; cur; cur = cur->next) { > + if (cur->key.specified & AVTAB_TRANSITION) { > + tab_insert_args.key = &cur->key; > + rc = hashtab_map(&cur->datum.u.trans->name_trans.table, > + filenametr_tab_insert, > + &tab_insert_args); > + if (rc) > + goto out; > + } > + } > + } > + > + /* write compressed filename transitions */ > + buf32[0] = cpu_to_le32(fnts_tab.nel); > + rc = put_entry(buf32, sizeof(u32), 1, fp); > + if (rc) > + goto out; > + > + rc = hashtab_map(&fnts_tab, filenametr_comp_write_helper, fp); > + > +out: > + /* destroy temp filename transitions table */ > + hashtab_map(&fnts_tab, filenametr_destroy, NULL); > + hashtab_destroy(&fnts_tab); > + > + return rc; > +} > diff --git a/security/selinux/ss/avtab.h b/security/selinux/ss/avtab.h > index 6c8eb7c379cf..162ef1be85e7 100644 > --- a/security/selinux/ss/avtab.h > +++ b/security/selinux/ss/avtab.h > @@ -22,6 +22,7 @@ > #define _SS_AVTAB_H_ > > #include "security.h" > +#include "symtab.h" > > struct avtab_key { > u16 source_type; /* source type */ > @@ -49,6 +50,7 @@ struct avtab_key { > > struct avtab_trans { > u32 otype; /* default resulting type of the new object */ > + struct symtab name_trans; /* filename transitions */ > }; > > /* > @@ -121,5 +123,10 @@ struct avtab_node *avtab_search_node_next(struct avtab_node *node, int specified > #define MAX_AVTAB_HASH_BITS 16 > #define MAX_AVTAB_HASH_BUCKETS (1 << MAX_AVTAB_HASH_BITS) > > +/* policydb filename transitions compatibility */ > + > +int avtab_filename_trans_read(struct avtab *a, void *fp, struct policydb *p); > +int avtab_filename_trans_write(struct policydb *p, struct avtab *a, void *fp); > + > #endif /* _SS_AVTAB_H_ */ > > diff --git a/security/selinux/ss/conditional.c b/security/selinux/ss/conditional.c > index e11219fdf9f7..91392d65563e 100644 > --- a/security/selinux/ss/conditional.c > +++ b/security/selinux/ss/conditional.c > @@ -263,6 +263,7 @@ static int cond_insertf(struct avtab *a, const struct avtab_key *k, > struct policydb *p = data->p; > struct cond_av_list *other = data->other; > struct avtab_node *node_ptr; > + struct avtab_datum *existing; > u32 i; > bool found; > > @@ -272,7 +273,10 @@ static int cond_insertf(struct avtab *a, const struct avtab_key *k, > * cond_te_avtab. > */ > if (k->specified & AVTAB_TYPE) { > - if (avtab_search(&p->te_avtab, k)) { > + existing = avtab_search(&p->te_avtab, k); > + /* empty transition rule is not a conflict */ > + if (existing && !(k->specified & AVTAB_TRANSITION && > + !existing->u.trans->otype)) { > pr_err("SELinux: type rule already exists outside of a conditional.\n"); > return -EINVAL; > } > diff --git a/security/selinux/ss/hashtab.h b/security/selinux/ss/hashtab.h > index 043a773bf0b7..4d04acf4d5af 100644 > --- a/security/selinux/ss/hashtab.h > +++ b/security/selinux/ss/hashtab.h > @@ -145,4 +145,10 @@ int hashtab_duplicate(struct hashtab *new, struct hashtab *orig, > /* Fill info with some hash table statistics */ > void hashtab_stat(struct hashtab *h, struct hashtab_info *info); > > +/* Checks if the hashtab is empty (its size is zero) */ > +static inline int hashtab_is_empty(struct hashtab *h) > +{ > + return !h->size; > +} > + > #endif /* _SS_HASHTAB_H */ > diff --git a/security/selinux/ss/policydb.c b/security/selinux/ss/policydb.c > index 97c0074f9312..928a08835db8 100644 > --- a/security/selinux/ss/policydb.c > +++ b/security/selinux/ss/policydb.c > @@ -323,23 +323,6 @@ static int (*const destroy_f[SYM_NUM]) (void *key, void *datum, void *datap) = { > cat_destroy, > }; > > -static int filenametr_destroy(void *key, void *datum, void *p) > -{ > - struct filename_trans_key *ft = key; > - struct filename_trans_datum *next, *d = datum; > - > - kfree(ft->name); > - kfree(key); > - do { > - ebitmap_destroy(&d->stypes); > - next = d->next; > - kfree(d); > - d = next; > - } while (unlikely(d)); > - cond_resched(); > - return 0; > -} > - > static int range_tr_destroy(void *key, void *datum, void *p) > { > struct mls_range *rt = datum; > @@ -406,50 +389,6 @@ static int roles_init(struct policydb *p) > return rc; > } > > -static u32 filenametr_hash(const void *k) > -{ > - const struct filename_trans_key *ft = k; > - unsigned long hash; > - unsigned int byte_num; > - unsigned char focus; > - > - hash = ft->ttype ^ ft->tclass; > - > - byte_num = 0; > - while ((focus = ft->name[byte_num++])) > - hash = partial_name_hash(focus, hash); > - return hash; > -} > - > -static int filenametr_cmp(const void *k1, const void *k2) > -{ > - const struct filename_trans_key *ft1 = k1; > - const struct filename_trans_key *ft2 = k2; > - int v; > - > - v = ft1->ttype - ft2->ttype; > - if (v) > - return v; > - > - v = ft1->tclass - ft2->tclass; > - if (v) > - return v; > - > - return strcmp(ft1->name, ft2->name); > - > -} > - > -static const struct hashtab_key_params filenametr_key_params = { > - .hash = filenametr_hash, > - .cmp = filenametr_cmp, > -}; > - > -struct filename_trans_datum *policydb_filenametr_search( > - struct policydb *p, struct filename_trans_key *key) > -{ > - return hashtab_search(&p->filename_trans, key, filenametr_key_params); > -} > - > static u32 rangetr_hash(const void *k) > { > const struct range_trans *key = k; > @@ -531,7 +470,6 @@ static void policydb_init(struct policydb *p) > avtab_init(&p->te_avtab); > cond_policydb_init(p); > > - ebitmap_init(&p->filename_trans_ttypes); > ebitmap_init(&p->policycaps); > ebitmap_init(&p->permissive_map); > } > @@ -839,9 +777,6 @@ void policydb_destroy(struct policydb *p) > } > kfree(lra); > > - hashtab_map(&p->filename_trans, filenametr_destroy, NULL); > - hashtab_destroy(&p->filename_trans); > - > hashtab_map(&p->range_tr, range_tr_destroy, NULL); > hashtab_destroy(&p->range_tr); > > @@ -851,7 +786,6 @@ void policydb_destroy(struct policydb *p) > kvfree(p->type_attr_map_array); > } > > - ebitmap_destroy(&p->filename_trans_ttypes); > ebitmap_destroy(&p->policycaps); > ebitmap_destroy(&p->permissive_map); > } > @@ -1066,7 +1000,7 @@ static int context_read_and_validate(struct context *c, > * binary representation file. > */ > > -static int str_read(char **strp, gfp_t flags, void *fp, u32 len) > +int str_read(char **strp, gfp_t flags, void *fp, u32 len) > { > int rc; > char *str; > @@ -1880,220 +1814,6 @@ static int range_read(struct policydb *p, void *fp) > return rc; > } > > -static int filename_trans_read_helper_compat(struct policydb *p, void *fp) > -{ > - struct filename_trans_key key, *ft = NULL; > - struct filename_trans_datum *last, *datum = NULL; > - char *name = NULL; > - u32 len, stype, otype; > - __le32 buf[4]; > - int rc; > - > - /* length of the path component string */ > - rc = next_entry(buf, fp, sizeof(u32)); > - if (rc) > - return rc; > - len = le32_to_cpu(buf[0]); > - > - /* path component string */ > - rc = str_read(&name, GFP_KERNEL, fp, len); > - if (rc) > - return rc; > - > - rc = next_entry(buf, fp, sizeof(u32) * 4); > - if (rc) > - goto out; > - > - stype = le32_to_cpu(buf[0]); > - key.ttype = le32_to_cpu(buf[1]); > - key.tclass = le32_to_cpu(buf[2]); > - key.name = name; > - > - otype = le32_to_cpu(buf[3]); > - > - last = NULL; > - datum = policydb_filenametr_search(p, &key); > - while (datum) { > - if (unlikely(ebitmap_get_bit(&datum->stypes, stype - 1))) { > - /* conflicting/duplicate rules are ignored */ > - datum = NULL; > - goto out; > - } > - if (likely(datum->otype == otype)) > - break; > - last = datum; > - datum = datum->next; > - } > - if (!datum) { > - rc = -ENOMEM; > - datum = kmalloc(sizeof(*datum), GFP_KERNEL); > - if (!datum) > - goto out; > - > - ebitmap_init(&datum->stypes); > - datum->otype = otype; > - datum->next = NULL; > - > - if (unlikely(last)) { > - last->next = datum; > - } else { > - rc = -ENOMEM; > - ft = kmemdup(&key, sizeof(key), GFP_KERNEL); > - if (!ft) > - goto out; > - > - rc = hashtab_insert(&p->filename_trans, ft, datum, > - filenametr_key_params); > - if (rc) > - goto out; > - name = NULL; > - > - rc = ebitmap_set_bit(&p->filename_trans_ttypes, > - key.ttype, 1); > - if (rc) > - return rc; > - } > - } > - kfree(name); > - return ebitmap_set_bit(&datum->stypes, stype - 1, 1); > - > -out: > - kfree(ft); > - kfree(name); > - kfree(datum); > - return rc; > -} > - > -static int filename_trans_read_helper(struct policydb *p, void *fp) > -{ > - struct filename_trans_key *ft = NULL; > - struct filename_trans_datum **dst, *datum, *first = NULL; > - char *name = NULL; > - u32 len, ttype, tclass, ndatum, i; > - __le32 buf[3]; > - int rc; > - > - /* length of the path component string */ > - rc = next_entry(buf, fp, sizeof(u32)); > - if (rc) > - return rc; > - len = le32_to_cpu(buf[0]); > - > - /* path component string */ > - rc = str_read(&name, GFP_KERNEL, fp, len); > - if (rc) > - return rc; > - > - rc = next_entry(buf, fp, sizeof(u32) * 3); > - if (rc) > - goto out; > - > - ttype = le32_to_cpu(buf[0]); > - tclass = le32_to_cpu(buf[1]); > - > - ndatum = le32_to_cpu(buf[2]); > - if (ndatum == 0) { > - pr_err("SELinux: Filename transition key with no datum\n"); > - rc = -ENOENT; > - goto out; > - } > - > - dst = &first; > - for (i = 0; i < ndatum; i++) { > - rc = -ENOMEM; > - datum = kmalloc(sizeof(*datum), GFP_KERNEL); > - if (!datum) > - goto out; > - > - *dst = datum; > - > - /* ebitmap_read() will at least init the bitmap */ > - rc = ebitmap_read(&datum->stypes, fp); > - if (rc) > - goto out; > - > - rc = next_entry(buf, fp, sizeof(u32)); > - if (rc) > - goto out; > - > - datum->otype = le32_to_cpu(buf[0]); > - datum->next = NULL; > - > - dst = &datum->next; > - } > - > - rc = -ENOMEM; > - ft = kmalloc(sizeof(*ft), GFP_KERNEL); > - if (!ft) > - goto out; > - > - ft->ttype = ttype; > - ft->tclass = tclass; > - ft->name = name; > - > - rc = hashtab_insert(&p->filename_trans, ft, first, > - filenametr_key_params); > - if (rc == -EEXIST) > - pr_err("SELinux: Duplicate filename transition key\n"); > - if (rc) > - goto out; > - > - return ebitmap_set_bit(&p->filename_trans_ttypes, ttype, 1); > - > -out: > - kfree(ft); > - kfree(name); > - while (first) { > - datum = first; > - first = first->next; > - > - ebitmap_destroy(&datum->stypes); > - kfree(datum); > - } > - return rc; > -} > - > -static int filename_trans_read(struct policydb *p, void *fp) > -{ > - u32 nel; > - __le32 buf[1]; > - int rc, i; > - > - if (p->policyvers < POLICYDB_VERSION_FILENAME_TRANS) > - return 0; > - > - rc = next_entry(buf, fp, sizeof(u32)); > - if (rc) > - return rc; > - nel = le32_to_cpu(buf[0]); > - > - if (p->policyvers < POLICYDB_VERSION_COMP_FTRANS) { > - p->compat_filename_trans_count = nel; > - > - rc = hashtab_init(&p->filename_trans, (1 << 11)); > - if (rc) > - return rc; > - > - for (i = 0; i < nel; i++) { > - rc = filename_trans_read_helper_compat(p, fp); > - if (rc) > - return rc; > - } > - } else { > - rc = hashtab_init(&p->filename_trans, nel); > - if (rc) > - return rc; > - > - for (i = 0; i < nel; i++) { > - rc = filename_trans_read_helper(p, fp); > - if (rc) > - return rc; > - } > - } > - hash_eval(&p->filename_trans, "filenametr"); > - return 0; > -} > - > static int genfs_read(struct policydb *p, void *fp) > { > int i, j, rc; > @@ -2634,7 +2354,7 @@ int policydb_read(struct policydb *p, void *fp) > lra = ra; > } > > - rc = filename_trans_read(p, fp); > + rc = avtab_filename_trans_read(&p->te_avtab, fp, p); > if (rc) > goto bad; > > @@ -3480,119 +3200,6 @@ static int range_write(struct policydb *p, void *fp) > return 0; > } > > -static int filename_write_helper_compat(void *key, void *data, void *ptr) > -{ > - struct filename_trans_key *ft = key; > - struct filename_trans_datum *datum = data; > - struct ebitmap_node *node; > - void *fp = ptr; > - __le32 buf[4]; > - int rc; > - u32 bit, len = strlen(ft->name); > - > - do { > - ebitmap_for_each_positive_bit(&datum->stypes, node, bit) { > - buf[0] = cpu_to_le32(len); > - rc = put_entry(buf, sizeof(u32), 1, fp); > - if (rc) > - return rc; > - > - rc = put_entry(ft->name, sizeof(char), len, fp); > - if (rc) > - return rc; > - > - buf[0] = cpu_to_le32(bit + 1); > - buf[1] = cpu_to_le32(ft->ttype); > - buf[2] = cpu_to_le32(ft->tclass); > - buf[3] = cpu_to_le32(datum->otype); > - > - rc = put_entry(buf, sizeof(u32), 4, fp); > - if (rc) > - return rc; > - } > - > - datum = datum->next; > - } while (unlikely(datum)); > - > - return 0; > -} > - > -static int filename_write_helper(void *key, void *data, void *ptr) > -{ > - struct filename_trans_key *ft = key; > - struct filename_trans_datum *datum; > - void *fp = ptr; > - __le32 buf[3]; > - int rc; > - u32 ndatum, len = strlen(ft->name); > - > - buf[0] = cpu_to_le32(len); > - rc = put_entry(buf, sizeof(u32), 1, fp); > - if (rc) > - return rc; > - > - rc = put_entry(ft->name, sizeof(char), len, fp); > - if (rc) > - return rc; > - > - ndatum = 0; > - datum = data; > - do { > - ndatum++; > - datum = datum->next; > - } while (unlikely(datum)); > - > - buf[0] = cpu_to_le32(ft->ttype); > - buf[1] = cpu_to_le32(ft->tclass); > - buf[2] = cpu_to_le32(ndatum); > - rc = put_entry(buf, sizeof(u32), 3, fp); > - if (rc) > - return rc; > - > - datum = data; > - do { > - rc = ebitmap_write(&datum->stypes, fp); > - if (rc) > - return rc; > - > - buf[0] = cpu_to_le32(datum->otype); > - rc = put_entry(buf, sizeof(u32), 1, fp); > - if (rc) > - return rc; > - > - datum = datum->next; > - } while (unlikely(datum)); > - > - return 0; > -} > - > -static int filename_trans_write(struct policydb *p, void *fp) > -{ > - __le32 buf[1]; > - int rc; > - > - if (p->policyvers < POLICYDB_VERSION_FILENAME_TRANS) > - return 0; > - > - if (p->policyvers < POLICYDB_VERSION_COMP_FTRANS) { > - buf[0] = cpu_to_le32(p->compat_filename_trans_count); > - rc = put_entry(buf, sizeof(u32), 1, fp); > - if (rc) > - return rc; > - > - rc = hashtab_map(&p->filename_trans, > - filename_write_helper_compat, fp); > - } else { > - buf[0] = cpu_to_le32(p->filename_trans.nel); > - rc = put_entry(buf, sizeof(u32), 1, fp); > - if (rc) > - return rc; > - > - rc = hashtab_map(&p->filename_trans, filename_write_helper, fp); > - } > - return rc; > -} > - > /* > * Write the configuration data in a policy database > * structure to a policy database binary representation > @@ -3703,7 +3310,7 @@ int policydb_write(struct policydb *p, void *fp) > if (rc) > return rc; > > - rc = filename_trans_write(p, fp); > + rc = avtab_filename_trans_write(p, &p->te_avtab, fp); > if (rc) > return rc; > > diff --git a/security/selinux/ss/policydb.h b/security/selinux/ss/policydb.h > index ffc4e7bad205..2ecb24a1611a 100644 > --- a/security/selinux/ss/policydb.h > +++ b/security/selinux/ss/policydb.h > @@ -91,18 +91,6 @@ struct role_trans_datum { > u32 new_role; /* new role */ > }; > > -struct filename_trans_key { > - u32 ttype; /* parent dir context */ > - u16 tclass; /* class of new object */ > - const char *name; /* last path component */ > -}; > - > -struct filename_trans_datum { > - struct ebitmap stypes; /* bitmap of source types for this otype */ > - u32 otype; /* resulting type of new object */ > - struct filename_trans_datum *next; /* record for next otype*/ > -}; > - > struct role_allow { > u32 role; /* current role */ > u32 new_role; /* new role */ > @@ -265,14 +253,6 @@ struct policydb { > /* role transitions */ > struct hashtab role_tr; > > - /* file transitions with the last path component */ > - /* quickly exclude lookups when parent ttype has no rules */ > - struct ebitmap filename_trans_ttypes; > - /* actual set of filename_trans rules */ > - struct hashtab filename_trans; > - /* only used if policyvers < POLICYDB_VERSION_COMP_FTRANS */ > - u32 compat_filename_trans_count; > - > /* bools indexed by (value - 1) */ > struct cond_bool_datum **bool_val_to_struct; > /* type enforcement conditional access vectors and transitions */ > @@ -324,9 +304,6 @@ extern int policydb_role_isvalid(struct policydb *p, unsigned int role); > extern int policydb_read(struct policydb *p, void *fp); > extern int policydb_write(struct policydb *p, void *fp); > > -extern struct filename_trans_datum *policydb_filenametr_search( > - struct policydb *p, struct filename_trans_key *key); > - > extern struct mls_range *policydb_rangetr_search( > struct policydb *p, struct range_trans *key); > > @@ -379,6 +356,8 @@ static inline int put_entry(const void *buf, size_t bytes, int num, struct polic > return 0; > } > > +extern int str_read(char **strp, gfp_t flags, void *fp, u32 len); > + > static inline char *sym_name(struct policydb *p, unsigned int sym_num, unsigned int element_nr) > { > return p->sym_val_to_name[sym_num][element_nr]; > diff --git a/security/selinux/ss/services.c b/security/selinux/ss/services.c > index 8ed12406acba..131647e7ec68 100644 > --- a/security/selinux/ss/services.c > +++ b/security/selinux/ss/services.c > @@ -1661,36 +1661,6 @@ static int compute_sid_handle_invalid_context( > return -EACCES; > } > > -static void filename_compute_type(struct policydb *policydb, > - struct context *newcontext, > - u32 stype, u32 ttype, u16 tclass, > - const char *objname) > -{ > - struct filename_trans_key ft; > - struct filename_trans_datum *datum; > - > - /* > - * Most filename trans rules are going to live in specific directories > - * like /dev or /var/run. This bitmap will quickly skip rule searches > - * if the ttype does not contain any rules. > - */ > - if (!ebitmap_get_bit(&policydb->filename_trans_ttypes, ttype)) > - return; > - > - ft.ttype = ttype; > - ft.tclass = tclass; > - ft.name = objname; > - > - datum = policydb_filenametr_search(policydb, &ft); > - while (datum) { > - if (ebitmap_get_bit(&datum->stypes, stype - 1)) { > - newcontext->type = datum->otype; > - return; > - } > - datum = datum->next; > - } > -} > - > static int security_compute_sid(u32 ssid, > u32 tsid, > u16 orig_tclass, > @@ -1711,6 +1681,7 @@ static int security_compute_sid(u32 ssid, > u16 tclass; > int rc = 0; > bool sock; > + u32 *otype; > > if (!selinux_initialized()) { > switch (orig_tclass) { > @@ -1830,17 +1801,24 @@ static int security_compute_sid(u32 ssid, > > if (avdatum) { > /* Use the type from the type transition/member/change rule. */ > - if (avkey.specified & AVTAB_TRANSITION) > - newcontext.type = avdatum->u.trans->otype; > - else > + if (avkey.specified & AVTAB_TRANSITION) { > + /* > + * use default otype if not empty and then to try to > + * find more specific rule using objname > + */ > + if (avdatum->u.trans->otype) > + newcontext.type = avdatum->u.trans->otype; > + if (objname) { > + otype = symtab_search(&avdatum->u.trans->name_trans, > + objname); > + if (otype) > + newcontext.type = *otype; > + } > + } else { > newcontext.type = avdatum->u.data; > + } > } > > - /* if we have a objname this is a file trans check so check those rules */ > - if (objname) > - filename_compute_type(policydb, &newcontext, scontext->type, > - tcontext->type, tclass, objname); > - > /* Check for class-specific changes. */ > if (specified & AVTAB_TRANSITION) { > /* Look for a role transition rule. */ > -- > 2.40.0 >