aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRichard Haines <richard_c_haines@btinternet.com>2015-06-01 15:33:26 +0100
committerStephen Smalley <sds@tycho.nsa.gov>2015-06-03 10:43:33 -0400
commitaf41e2b5ff197f5e66262868fdc3a03963cd104e (patch)
treed502d36f3d05feac444d5d92be679294cd9c4556
parenta24fc044f4ddd7d8679f2219454d103036d312ef (diff)
downloadandroid_external_selinux-af41e2b5ff197f5e66262868fdc3a03963cd104e.tar.gz
android_external_selinux-af41e2b5ff197f5e66262868fdc3a03963cd104e.tar.bz2
android_external_selinux-af41e2b5ff197f5e66262868fdc3a03963cd104e.zip
libselinux: Add read_spec_entries function to replace sscanf
Currently sscanf is used with %ms parameters that are not supported on all platforms. The new read_spec_entries function may be used to replace these where required. This patch updates sefcontext_compile, label_file and label_android_property services to use the new function. The file and property services have been tested on Android emulator and the file service on Fedora 21. Signed-off-by: Richard Haines <richard_c_haines@btinternet.com>
-rw-r--r--libselinux/src/label_android_property.c18
-rw-r--r--libselinux/src/label_file.c16
-rw-r--r--libselinux/src/label_internal.h6
-rw-r--r--libselinux/src/label_support.c88
-rw-r--r--libselinux/utils/sefcontext_compile.c35
5 files changed, 119 insertions, 44 deletions
diff --git a/libselinux/src/label_android_property.c b/libselinux/src/label_android_property.c
index 1a293569..dadb1602 100644
--- a/libselinux/src/label_android_property.c
+++ b/libselinux/src/label_android_property.c
@@ -82,23 +82,15 @@ static int process_line(struct selabel_handle *rec,
const char *path, char *line_buf,
int pass, unsigned lineno)
{
- int items, len;
- char buf1[BUFSIZ], buf2[BUFSIZ];
- char *buf_p, *prop = buf1, *context = buf2;
+ int items;
+ char *prop = NULL, *context = NULL;
struct saved_data *data = (struct saved_data *)rec->data;
spec_t *spec_arr = data->spec_arr;
unsigned int nspec = data->nspec;
- len = strlen(line_buf);
- if (line_buf[len - 1] == '\n')
- line_buf[len - 1] = 0;
- buf_p = line_buf;
- while (isspace(*buf_p))
- buf_p++;
- /* Skip comment lines and empty lines. */
- if (*buf_p == '#' || *buf_p == 0)
- return 0;
- items = sscanf(line_buf, "%255s %255s", prop, context);
+ items = read_spec_entries(line_buf, 2, &prop, &context);
+ if (items <= 0)
+ return items;
if (items != 2) {
selinux_log(SELINUX_WARNING,
"%s: line %u is missing fields, skipping\n", path,
diff --git a/libselinux/src/label_file.c b/libselinux/src/label_file.c
index bfb64af4..b8061f4c 100644
--- a/libselinux/src/label_file.c
+++ b/libselinux/src/label_file.c
@@ -154,22 +154,16 @@ static int process_line(struct selabel_handle *rec,
char *line_buf, unsigned lineno)
{
int items, len, rc;
- char *buf_p, *regex, *type, *context;
+ char *regex = NULL, *type = NULL, *context = NULL;
struct saved_data *data = (struct saved_data *)rec->data;
struct spec *spec_arr;
unsigned int nspec = data->nspec;
const char *errbuf = NULL;
- len = strlen(line_buf);
- if (line_buf[len - 1] == '\n')
- line_buf[len - 1] = 0;
- buf_p = line_buf;
- while (isspace(*buf_p))
- buf_p++;
- /* Skip comment lines and empty lines. */
- if (*buf_p == '#' || *buf_p == 0)
- return 0;
- items = sscanf(line_buf, "%ms %ms %ms", &regex, &type, &context);
+ items = read_spec_entries(line_buf, 3, &regex, &type, &context);
+ if (items <= 0)
+ return items;
+
if (items < 2) {
COMPAT_LOG(SELINUX_WARNING,
"%s: line %u is missing fields, skipping\n", path,
diff --git a/libselinux/src/label_internal.h b/libselinux/src/label_internal.h
index ef6ad260..0e582b62 100644
--- a/libselinux/src/label_internal.h
+++ b/libselinux/src/label_internal.h
@@ -102,4 +102,10 @@ compat_validate(struct selabel_handle *rec,
struct selabel_lookup_rec *contexts,
const char *path, unsigned lineno) hidden;
+/*
+ * The read_spec_entries function may be used to
+ * replace sscanf to read entries from spec files.
+ */
+extern int read_spec_entries(char *line_buf, int num_args, ...);
+
#endif /* _SELABEL_INTERNAL_H_ */
diff --git a/libselinux/src/label_support.c b/libselinux/src/label_support.c
new file mode 100644
index 00000000..7464f430
--- /dev/null
+++ b/libselinux/src/label_support.c
@@ -0,0 +1,88 @@
+/*
+ * This file contains helper functions for labeling support.
+ *
+ * Author : Richard Haines <richard_c_haines@btinternet.com>
+ */
+
+#include <stdlib.h>
+#include <stdarg.h>
+#include <ctype.h>
+#include <string.h>
+#include "label_internal.h"
+
+/*
+ * The read_spec_entries and read_spec_entry functions may be used to
+ * replace sscanf to read entries from spec files. The file and
+ * property services now use these.
+ */
+
+/* Read an entry from a spec file (e.g. file_contexts) */
+static inline int read_spec_entry(char **entry, char **ptr)
+{
+ int entry_len = 0;
+ *entry = NULL;
+ char *tmp_buf = NULL;
+
+ while (isspace(**ptr) && **ptr != '\0')
+ (*ptr)++;
+
+ tmp_buf = *ptr;
+
+ while (!isspace(**ptr) && **ptr != '\0') {
+ (*ptr)++;
+ entry_len++;
+ }
+
+ *entry = strndup(tmp_buf, entry_len);
+ if (!*entry)
+ return -1;
+
+ return 0;
+}
+
+/*
+ * line_buf - Buffer containing the spec entries .
+ * num_args - The number of spec parameter entries to process.
+ * ... - A 'char **spec_entry' for each parameter.
+ * returns - The number of items processed.
+ *
+ * This function calls read_spec_entry() to do the actual string processing.
+ */
+int read_spec_entries(char *line_buf, int num_args, ...)
+{
+ char **spec_entry, *buf_p;
+ int len, rc, items;
+ va_list ap;
+
+ len = strlen(line_buf);
+ if (line_buf[len - 1] == '\n')
+ line_buf[len - 1] = '\0';
+
+ buf_p = line_buf;
+ while (isspace(*buf_p))
+ buf_p++;
+
+ /* Skip comment lines and empty lines. */
+ if (*buf_p == '#' || *buf_p == '\0')
+ return 0;
+
+ /* Process the spec file entries */
+ va_start(ap, num_args);
+
+ for (items = 0; items < num_args; items++) {
+ spec_entry = va_arg(ap, char **);
+
+ if (len - 1 == buf_p - line_buf) {
+ va_end(ap);
+ return items;
+ }
+
+ rc = read_spec_entry(spec_entry, &buf_p);
+ if (rc < 0) {
+ va_end(ap);
+ return rc;
+ }
+ }
+ va_end(ap);
+ return items;
+}
diff --git a/libselinux/utils/sefcontext_compile.c b/libselinux/utils/sefcontext_compile.c
index 33a4a424..03bc0a7e 100644
--- a/libselinux/utils/sefcontext_compile.c
+++ b/libselinux/utils/sefcontext_compile.c
@@ -29,11 +29,10 @@ static int process_file(struct saved_data *data, const char *filename)
line_num = 0;
while ((len = getline(&line_buf, &line_len, context_file)) != -1) {
- char *context;
- char *mode;
- char *regex;
+ char *context = NULL;
+ char *mode = NULL;
+ char *regex = NULL;
char *cp, *anchored_regex;
- char *buf_p;
pcre *re;
pcre_extra *sd;
const char *err;
@@ -41,23 +40,20 @@ static int process_file(struct saved_data *data, const char *filename)
size_t regex_len;
int32_t stem_id;
- len = strlen(line_buf);
- if (line_buf[len - 1] == '\n')
- line_buf[len - 1] = 0;
- buf_p = line_buf;
- while (isspace(*buf_p))
- buf_p++;
- /* Skip comment lines and empty lines. */
- if (*buf_p == '#' || *buf_p == 0)
- continue;
+ line_num++;
- items = sscanf(line_buf, "%ms %ms %ms", &regex, &mode, &context);
- if (items < 2 || items > 3) {
- fprintf(stderr, "invalid entry, skipping:%s", line_buf);
- continue;
- }
+ items = read_spec_entries(line_buf, 3, &regex, &mode, &context);
+ if (items < 0)
+ return -1;
- if (items == 2) {
+ if (items == 0)
+ continue;
+ else if (items == 1) {
+ fprintf(stderr,
+ "line: %u has invalid entry - skipping: %s\n",
+ line_num, line_buf);
+ continue;
+ } else if (items == 2) {
context = mode;
mode = NULL;
}
@@ -115,7 +111,6 @@ static int process_file(struct saved_data *data, const char *filename)
free(anchored_regex);
spec->sd = sd;
- line_num++;
data->nspec++;
}