165 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * Copyright (C) 2014  Tresys Technology, LLC
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software
 | |
|  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | |
|  * 02110-1301, USA.
 | |
|  */
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <getopt.h>
 | |
| #include <libgen.h>
 | |
| #include <signal.h>
 | |
| #include <stdarg.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #include <sepol/module.h>
 | |
| #include <sepol/module_to_cil.h>
 | |
| #include <sepol/policydb/module.h>
 | |
| 
 | |
| char *progname;
 | |
| 
 | |
| __attribute__ ((format(printf, 1, 2)))
 | |
| static void log_err(const char *fmt, ...)
 | |
| {
 | |
| 	va_list argptr;
 | |
| 	va_start(argptr, fmt);
 | |
| 	if (vfprintf(stderr, fmt, argptr) < 0) {
 | |
| 		_exit(EXIT_FAILURE);
 | |
| 	}
 | |
| 	va_end(argptr);
 | |
| 	if (fprintf(stderr, "\n") < 0) {
 | |
| 		_exit(EXIT_FAILURE);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static __attribute__((__noreturn__)) void usage(int err)
 | |
| {
 | |
| 	fprintf(stderr, "Usage: %s [OPTIONS] [IN_FILE [OUT_FILE]]\n", progname);
 | |
| 	fprintf(stderr, "\n");
 | |
| 	fprintf(stderr, "Read an SELinux policy package (.pp) and output the equivalent CIL.\n");
 | |
| 	fprintf(stderr, "If IN_FILE is not provided or is -, read SELinux policy package from\n");
 | |
| 	fprintf(stderr, "standard input. If OUT_FILE is not provided or is -, output CIL to\n");
 | |
| 	fprintf(stderr, "standard output.\n");
 | |
| 	fprintf(stderr, "\n");
 | |
| 	fprintf(stderr, "Options:\n");
 | |
| 	fprintf(stderr, "  -h, --help      print this message and exit\n");
 | |
| 	exit(err);
 | |
| }
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
| 	int rc = -1;
 | |
| 	int opt;
 | |
| 	static struct option long_opts[] = {
 | |
| 		{ "help", 0, NULL, 'h' },
 | |
| 		{ NULL, 0, NULL, 0 }
 | |
| 	};
 | |
| 	struct sepol_module_package *mod_pkg = NULL;
 | |
| 	const char *ifile = NULL;
 | |
| 	const char *ofile = NULL;
 | |
| 	FILE *in = NULL;
 | |
| 	FILE *out = NULL;
 | |
| 
 | |
| 	// ignore sigpipe so we can check the return code of write, and potentially
 | |
| 	// return a more helpful error message
 | |
| 	signal(SIGPIPE, SIG_IGN);
 | |
| 
 | |
| 	progname = basename(argv[0]);
 | |
| 
 | |
| 	while ((opt = getopt_long(argc, argv, "h", long_opts, NULL)) != -1) {
 | |
| 		switch (opt) {
 | |
| 		case 'h':
 | |
| 			usage(0);
 | |
| 		case '?':
 | |
| 		default:
 | |
| 			usage(1);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (argc >= optind + 1 && strcmp(argv[1], "-") != 0) {
 | |
| 		ifile = argv[1];
 | |
| 		in = fopen(ifile, "rb");
 | |
| 		if (in == NULL) {
 | |
| 			log_err("Failed to open %s: %s", ifile, strerror(errno));
 | |
| 			rc = -1;
 | |
| 			goto exit;
 | |
| 		}
 | |
| 	} else {
 | |
| 		ifile = "stdin";
 | |
| 		in = stdin;
 | |
| 	}
 | |
| 
 | |
| 	if (argc >= optind + 2 && strcmp(argv[2], "-") != 0) {
 | |
| 		ofile = argv[2];
 | |
| 		out = fopen(ofile, "w");
 | |
| 		if (out == NULL) {
 | |
| 			log_err("Failed to open %s: %s", ofile, strerror(errno));
 | |
| 			rc = -1;
 | |
| 			goto exit;
 | |
| 		}
 | |
| 	} else {
 | |
| 		out = stdout;
 | |
| 	}
 | |
| 
 | |
| 	if (argc >= optind + 3) {
 | |
| 		log_err("Too many arguments");
 | |
| 		usage(1);
 | |
| 	}
 | |
| 
 | |
| 	rc = sepol_ppfile_to_module_package(in, &mod_pkg);
 | |
| 	if (rc != 0) {
 | |
| 		goto exit;
 | |
| 	}
 | |
| 	fclose(in);
 | |
| 	in = NULL;
 | |
| 
 | |
| 	if (ofile) {
 | |
| 		char *mod_name = mod_pkg->policy->p.name;
 | |
| 		char *cil_path = strdup(ofile);
 | |
| 		if (cil_path == NULL) {
 | |
| 			log_err("No memory available for strdup\n");
 | |
| 			rc = -1;
 | |
| 			goto exit;
 | |
| 		}
 | |
| 		char *cil_name = basename(cil_path);
 | |
| 		char *separator = strrchr(cil_name, '.');
 | |
| 		if (separator) {
 | |
| 			*separator = '\0';
 | |
| 		}
 | |
| 		if (mod_name && strcmp(mod_name, cil_name) != 0) {
 | |
| 			fprintf(stderr,	"Warning: SELinux userspace will refer to the module from %s as %s rather than %s\n", ifile, mod_name, cil_name);
 | |
| 		}
 | |
| 		free(cil_path);
 | |
| 	}
 | |
| 
 | |
| 	rc = sepol_module_package_to_cil(out, mod_pkg);
 | |
| 	if (rc != 0) {
 | |
| 		goto exit;
 | |
| 	}
 | |
| 
 | |
| exit:
 | |
| 	if (in != NULL) {
 | |
| 		fclose(in);
 | |
| 	}
 | |
| 	if (out != NULL) {
 | |
| 		fclose(out);
 | |
| 	}
 | |
| 	sepol_module_package_free(mod_pkg);
 | |
| 
 | |
| 	return rc;
 | |
| }
 |