mirror of
https://github.com/SELinuxProject/selinux
synced 2025-04-01 23:08:09 +00:00
Clang's undefined behavior sanitizer supports checking for unsigned integer overflow and underflow, and implicit conversions. While those operations are well-defined by the C language they can signal logic mistakes or processing of unchecked user input. Annotate functions deliberately making use of integer overflow and adopt the remaining code sites. Example reports: stringrep.c:348:7: runtime error: left shift of 2147483648 by 1 places cannot be represented in type 'access_vector_t' (aka 'unsigned int') seusers.c:98:14: runtime error: implicit conversion from type 'int' of value -1 (32-bit, signed) to type 'gid_t' (aka 'unsigned int') changed the value to 4294967295 (32-bit, unsigned) Signed-off-by: Christian Göttsche <cgzones@googlemail.com> Acked-by: James Carter <jwcart2@gmail.com>
238 lines
5.0 KiB
C
238 lines
5.0 KiB
C
/*
|
|
* Media contexts backend for labeling system
|
|
*
|
|
* Author : Eamon Walsh <ewalsh@tycho.nsa.gov>
|
|
*/
|
|
|
|
#include <sys/stat.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdio_ext.h>
|
|
#include <ctype.h>
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
#include "callbacks.h"
|
|
#include "label_internal.h"
|
|
|
|
/*
|
|
* Internals
|
|
*/
|
|
|
|
/* A context specification. */
|
|
typedef struct spec {
|
|
struct selabel_lookup_rec lr; /* holds contexts for lookup result */
|
|
char *key; /* key string */
|
|
int matches; /* number of matches made during operation */
|
|
} spec_t;
|
|
|
|
struct saved_data {
|
|
unsigned int nspec;
|
|
spec_t *spec_arr;
|
|
};
|
|
|
|
static int process_line(const char *path, char *line_buf, int pass,
|
|
unsigned lineno, struct selabel_handle *rec)
|
|
{
|
|
struct saved_data *data = (struct saved_data *)rec->data;
|
|
int items;
|
|
char *buf_p;
|
|
char *key, *context;
|
|
|
|
buf_p = line_buf;
|
|
while (isspace((unsigned char)*buf_p))
|
|
buf_p++;
|
|
/* Skip comment lines and empty lines. */
|
|
if (*buf_p == '#' || *buf_p == 0)
|
|
return 0;
|
|
items = sscanf(line_buf, "%ms %ms ", &key, &context);
|
|
if (items < 2) {
|
|
selinux_log(SELINUX_WARNING,
|
|
"%s: line %u is missing fields, skipping\n", path,
|
|
lineno);
|
|
if (items == 1)
|
|
free(key);
|
|
return 0;
|
|
}
|
|
|
|
if (pass == 1) {
|
|
data->spec_arr[data->nspec].key = key;
|
|
data->spec_arr[data->nspec].lr.ctx_raw = context;
|
|
}
|
|
|
|
data->nspec++;
|
|
if (pass == 0) {
|
|
free(key);
|
|
free(context);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int init(struct selabel_handle *rec, const struct selinux_opt *opts,
|
|
unsigned n)
|
|
{
|
|
FILE *fp;
|
|
struct saved_data *data = (struct saved_data *)rec->data;
|
|
const char *path = NULL;
|
|
char *line_buf = NULL;
|
|
size_t line_len = 0;
|
|
int status = -1;
|
|
unsigned int lineno, pass, maxnspec;
|
|
struct stat sb;
|
|
|
|
/* Process arguments */
|
|
while (n) {
|
|
n--;
|
|
switch(opts[n].type) {
|
|
case SELABEL_OPT_PATH:
|
|
path = opts[n].value;
|
|
break;
|
|
case SELABEL_OPT_UNUSED:
|
|
case SELABEL_OPT_VALIDATE:
|
|
case SELABEL_OPT_DIGEST:
|
|
break;
|
|
default:
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/* Open the specification file. */
|
|
if (!path)
|
|
path = selinux_media_context_path();
|
|
if ((fp = fopen(path, "re")) == NULL)
|
|
return -1;
|
|
__fsetlocking(fp, FSETLOCKING_BYCALLER);
|
|
|
|
if (fstat(fileno(fp), &sb) < 0)
|
|
goto finish;
|
|
if (!S_ISREG(sb.st_mode)) {
|
|
errno = EINVAL;
|
|
goto finish;
|
|
}
|
|
rec->spec_file = strdup(path);
|
|
|
|
/*
|
|
* Perform two passes over the specification file.
|
|
* The first pass counts the number of specifications and
|
|
* performs simple validation of the input. At the end
|
|
* of the first pass, the spec array is allocated.
|
|
* The second pass performs detailed validation of the input
|
|
* and fills in the spec array.
|
|
*/
|
|
maxnspec = UINT_MAX / sizeof(spec_t);
|
|
for (pass = 0; pass < 2; pass++) {
|
|
lineno = 0;
|
|
data->nspec = 0;
|
|
while (getline(&line_buf, &line_len, fp) > 0 &&
|
|
data->nspec < maxnspec) {
|
|
if (process_line(path, line_buf, pass, ++lineno, rec))
|
|
goto finish;
|
|
}
|
|
|
|
if (pass == 0) {
|
|
if (data->nspec == 0) {
|
|
status = 0;
|
|
goto finish;
|
|
}
|
|
data->spec_arr = malloc(sizeof(spec_t)*data->nspec);
|
|
if (data->spec_arr == NULL)
|
|
goto finish;
|
|
memset(data->spec_arr, 0, sizeof(spec_t)*data->nspec);
|
|
maxnspec = data->nspec;
|
|
|
|
status = fseek(fp, 0L, SEEK_SET);
|
|
if (status == -1)
|
|
goto finish;
|
|
}
|
|
}
|
|
free(line_buf);
|
|
|
|
status = digest_add_specfile(rec->digest, fp, NULL, sb.st_size, path);
|
|
if (status)
|
|
goto finish;
|
|
|
|
digest_gen_hash(rec->digest);
|
|
|
|
finish:
|
|
fclose(fp);
|
|
return status;
|
|
}
|
|
|
|
/*
|
|
* Backend interface routines
|
|
*/
|
|
static void close(struct selabel_handle *rec)
|
|
{
|
|
struct saved_data *data = (struct saved_data *)rec->data;
|
|
struct spec *spec, *spec_arr = data->spec_arr;
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < data->nspec; i++) {
|
|
spec = &spec_arr[i];
|
|
free(spec->key);
|
|
free(spec->lr.ctx_raw);
|
|
free(spec->lr.ctx_trans);
|
|
}
|
|
|
|
if (spec_arr)
|
|
free(spec_arr);
|
|
|
|
free(data);
|
|
}
|
|
|
|
static struct selabel_lookup_rec *lookup(struct selabel_handle *rec,
|
|
const char *key,
|
|
int type __attribute__((unused)))
|
|
{
|
|
struct saved_data *data = (struct saved_data *)rec->data;
|
|
spec_t *spec_arr = data->spec_arr;
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < data->nspec; i++) {
|
|
if (!strncmp(spec_arr[i].key, key, strlen(key) + 1))
|
|
break;
|
|
if (!strncmp(spec_arr[i].key, "*", 2))
|
|
break;
|
|
}
|
|
|
|
if (i >= data->nspec) {
|
|
/* No matching specification. */
|
|
errno = ENOENT;
|
|
return NULL;
|
|
}
|
|
|
|
spec_arr[i].matches++;
|
|
return &spec_arr[i].lr;
|
|
}
|
|
|
|
static void stats(struct selabel_handle *rec)
|
|
{
|
|
struct saved_data *data = (struct saved_data *)rec->data;
|
|
unsigned int i, total = 0;
|
|
|
|
for (i = 0; i < data->nspec; i++)
|
|
total += data->spec_arr[i].matches;
|
|
|
|
selinux_log(SELINUX_INFO, "%u entries, %u matches made\n",
|
|
data->nspec, total);
|
|
}
|
|
|
|
int selabel_media_init(struct selabel_handle *rec,
|
|
const struct selinux_opt *opts,
|
|
unsigned nopts)
|
|
{
|
|
struct saved_data *data;
|
|
|
|
data = (struct saved_data *)malloc(sizeof(*data));
|
|
if (!data)
|
|
return -1;
|
|
memset(data, 0, sizeof(*data));
|
|
|
|
rec->data = data;
|
|
rec->func_close = &close;
|
|
rec->func_lookup = &lookup;
|
|
rec->func_stats = &stats;
|
|
|
|
return init(rec, opts, nopts);
|
|
}
|