* Updated to sedoctool to read bool files and tunable

files separately.
* Changed the xml tag of <boolean> to <bool> to be consistent
  with gen_bool().
* Modified the implementation of segenxml to use regular
  expressions.
This commit is contained in:
Chris PeBenito 2005-10-07 18:08:50 +00:00
parent 473ea7166f
commit 4f9f30c8df
9 changed files with 172 additions and 208 deletions

View File

@ -1,3 +1,9 @@
- Updated to sedoctool to read bool files and tunable
files separately.
- Changed the xml tag of <boolean> to <bool> to be consistent
with gen_bool().
- Modified the implementation of segenxml to use regular
expressions.
- Rename context_template() to gen_context() to clarify - Rename context_template() to gen_context() to clarify
that its not a Reference Policy template, but a support that its not a Reference Policy template, but a support
macro. macro.

View File

@ -106,8 +106,10 @@ DOCTEMPLATE = $(DOCS)/templates
# config file paths # config file paths
GLOBALTUN := $(POLDIR)/global_tunables GLOBALTUN := $(POLDIR)/global_tunables
GLOBALBOOL := $(POLDIR)/global_booleans
MOD_CONF := $(POLDIR)/modules.conf MOD_CONF := $(POLDIR)/modules.conf
TUNABLES := $(POLDIR)/tunables.conf TUNABLES := $(POLDIR)/tunables.conf
BOOLEANS := $(POLDIR)/booleans.conf
# install paths # install paths
TOPDIR = $(DESTDIR)/etc/selinux TOPDIR = $(DESTDIR)/etc/selinux
@ -232,11 +234,11 @@ $(MODDIR)/kernel/corenetwork.te: $(MODDIR)/kernel/corenetwork.te.m4 $(MODDIR)/ke
# #
# Create config files # Create config files
# #
conf: $(MOD_CONF) $(TUNABLES) $(GENERATED_TE) $(GENERATED_IF) $(GENERATED_FC) conf: $(MOD_CONF) $(BOOLEANS) $(GENERATED_TE) $(GENERATED_IF) $(GENERATED_FC)
$(MOD_CONF) $(TUNABLES): $(POLXML) $(MOD_CONF) $(BOOLEANS): $(POLXML)
@echo "Updating $(MOD_CONF) and $(TUNABLES)" @echo "Updating $(MOD_CONF) and $(BOOLEANS)"
$(QUIET) cd $(DOCS) && ../$(GENDOC) -t ../$(TUNABLES) -m ../$(MOD_CONF) -x ../$(POLXML) $(QUIET) cd $(DOCS) && ../$(GENDOC) -t ../$(BOOLEANS) -m ../$(MOD_CONF) -x ../$(POLXML)
######################################## ########################################
# #
@ -252,7 +254,7 @@ $(POLXML): $(DETECTED_MODS:.te=.if) $(foreach dir,$(ALL_LAYERS),$(dir)/$(LAYERXM
@mkdir -p tmp @mkdir -p tmp
$(QUIET) echo '<?xml version="1.0" encoding="ISO-8859-1" standalone="no"?>' > $@ $(QUIET) echo '<?xml version="1.0" encoding="ISO-8859-1" standalone="no"?>' > $@
$(QUIET) echo '<!DOCTYPE policy SYSTEM "$(notdir $(XMLDTD))">' >> $@ $(QUIET) echo '<!DOCTYPE policy SYSTEM "$(notdir $(XMLDTD))">' >> $@
$(QUIET) $(GENXML) -w -m $(LAYERXML) -t $(GLOBALTUN) $(ALL_LAYERS) >> $@ $(QUIET) $(GENXML) -w -m $(LAYERXML) -t $(GLOBALTUN) -b $(GLOBALBOOL) $(ALL_LAYERS) >> $@
$(QUIET) if test -x $(XMLLINT) && test -f $(XMLDTD); then \ $(QUIET) if test -x $(XMLLINT) && test -f $(XMLDTD); then \
$(XMLLINT) --noout --dtdvalid $(XMLDTD) $@ ;\ $(XMLLINT) --noout --dtdvalid $(XMLDTD) $@ ;\
fi fi
@ -290,9 +292,9 @@ $(USERPATH)/local.users: config/local.users
# #
install-appconfig: $(APPFILES) install-appconfig: $(APPFILES)
$(INSTALLDIR)/booleans: $(TUNABLES) $(INSTALLDIR)/booleans: $(BOOLEANS)
@mkdir -p $(INSTALLDIR) @mkdir -p $(INSTALLDIR)
$(QUIET) egrep '^[[:blank:]]*[[:alpha:]]' $(TUNABLES) \ $(QUIET) egrep '^[[:blank:]]*[[:alpha:]]' $(BOOLEANS) \
| sed -e 's/false/0/g' -e 's/true/1/g' > tmp/booleans | sed -e 's/false/0/g' -e 's/true/1/g' > tmp/booleans
$(QUIET) install -m 644 tmp/booleans $@ $(QUIET) install -m 644 tmp/booleans $@
@ -356,7 +358,7 @@ bare: clean
rm -f $(SUPPORT)/*.pyc rm -f $(SUPPORT)/*.pyc
rm -f $(FCSORT) rm -f $(FCSORT)
rm -f $(MOD_CONF) rm -f $(MOD_CONF)
rm -f $(TUNABLES) rm -f $(BOOLEANS)
rm -fR $(HTMLDIR) rm -fR $(HTMLDIR)
ifneq ($(GENERATED_TE),) ifneq ($(GENERATED_TE),)
rm -f $(GENERATED_TE) rm -f $(GENERATED_TE)

View File

@ -9,7 +9,7 @@ ALL_INTERFACES := $(ALL_MODULES:.te=.if)
BASE_PKG := base.pp BASE_PKG := base.pp
BASE_FC := base.fc BASE_FC := base.fc
BASE_SECTIONS := tmp/pre_te_files.conf tmp/generated_definitions.conf tmp/all_interfaces.conf tmp/all_attrs_types.conf $(GLOBALTUN) tmp/only_te_rules.conf tmp/all_post.conf BASE_SECTIONS := tmp/pre_te_files.conf tmp/generated_definitions.conf tmp/all_interfaces.conf tmp/all_attrs_types.conf $(GLOBALBOOL) $(GLOBALTUN) tmp/only_te_rules.conf tmp/all_post.conf
BASE_PRE_TE_FILES := $(SECCLASS) $(ISIDS) $(AVS) $(M4SUPPORT) $(POLDIR)/mls $(POLDIR)/mcs BASE_PRE_TE_FILES := $(SECCLASS) $(ISIDS) $(AVS) $(M4SUPPORT) $(POLDIR)/mls $(POLDIR)/mcs
BASE_TE_FILES := $(BASE_MODS) BASE_TE_FILES := $(BASE_MODS)
@ -87,7 +87,7 @@ tmp/generated_definitions.conf: $(ALL_LAYERS) $(BASE_TE_FILES)
$(QUIET) for i in $(notdir $(BASE_TE_FILES)); do \ $(QUIET) for i in $(notdir $(BASE_TE_FILES)); do \
echo "define(\`$$i')" >> $@ ;\ echo "define(\`$$i')" >> $@ ;\
done done
$(QUIET) $(SETTUN) $(TUNABLES) >> $@ $(QUIET) $(SETTUN) $(BOOLEANS) >> $@
tmp/all_interfaces.conf: $(M4SUPPORT) $(ALL_INTERFACES) tmp/all_interfaces.conf: $(M4SUPPORT) $(ALL_INTERFACES)
ifeq ($(ALL_INTERFACES),) ifeq ($(ALL_INTERFACES),)

View File

@ -21,7 +21,7 @@ ALL_FC_FILES := $(ALL_MODULES:.te=.fc)
PRE_TE_FILES := $(SECCLASS) $(ISIDS) $(AVS) $(M4SUPPORT) $(POLDIR)/mls $(POLDIR)/mcs PRE_TE_FILES := $(SECCLASS) $(ISIDS) $(AVS) $(M4SUPPORT) $(POLDIR)/mls $(POLDIR)/mcs
POST_TE_FILES := $(POLDIR)/systemuser $(POLDIR)/users $(POLDIR)/constraints POST_TE_FILES := $(POLDIR)/systemuser $(POLDIR)/users $(POLDIR)/constraints
POLICY_SECTIONS := tmp/pre_te_files.conf tmp/generated_definitions.conf tmp/all_interfaces.conf tmp/all_attrs_types.conf $(GLOBALTUN) tmp/only_te_rules.conf tmp/all_post.conf POLICY_SECTIONS := tmp/pre_te_files.conf tmp/generated_definitions.conf tmp/all_interfaces.conf tmp/all_attrs_types.conf $(GLOBALBOOL) $(GLOBALTUN) tmp/only_te_rules.conf tmp/all_post.conf
######################################## ########################################
# #
@ -102,7 +102,7 @@ tmp/generated_definitions.conf: $(ALL_LAYERS) $(ALL_TE_FILES)
$(QUIET) for i in $(notdir $(ALL_MODULES)); do \ $(QUIET) for i in $(notdir $(ALL_MODULES)); do \
echo "define(\`$$i')" >> $@ ;\ echo "define(\`$$i')" >> $@ ;\
done done
$(QUIET) $(SETTUN) $(TUNABLES) >> $@ $(QUIET) $(SETTUN) $(BOOLEANS) >> $@
tmp/all_interfaces.conf: $(M4SUPPORT) $(ALL_INTERFACES) tmp/all_interfaces.conf: $(M4SUPPORT) $(ALL_INTERFACES)
ifeq ($(ALL_INTERFACES),) ifeq ($(ALL_INTERFACES),)

View File

@ -1,6 +1,6 @@
<!ENTITY % inline.class "pre|p|ul|ol|li"> <!ENTITY % inline.class "pre|p|ul|ol|li">
<!ELEMENT policy (layer+,(tunable|boolean)*)> <!ELEMENT policy (layer+,(tunable|bool)*)>
<!ELEMENT layer (summary,module+)> <!ELEMENT layer (summary,module+)>
<!ATTLIST layer <!ATTLIST layer
name CDATA #REQUIRED> name CDATA #REQUIRED>
@ -14,8 +14,8 @@
<!ATTLIST tunable <!ATTLIST tunable
name CDATA #REQUIRED name CDATA #REQUIRED
dftval CDATA #REQUIRED> dftval CDATA #REQUIRED>
<!ELEMENT boolean (#PCDATA)> <!ELEMENT bool (#PCDATA)>
<!ATTLIST boolean <!ATTLIST bool
name CDATA #REQUIRED name CDATA #REQUIRED
dftval CDATA #REQUIRED> dftval CDATA #REQUIRED>
<!ELEMENT summary (#PCDATA)> <!ELEMENT summary (#PCDATA)>

View File

@ -0,0 +1,10 @@
#
# This file is for the declaration of global booleans.
# To change the default value at build time, the booleans.conf
# file should be used.
#
## Enabling secure mode disallows programs, such as
## newrole, from transitioning to administrative
## user domains.
gen_bool(secure_mode,false)

View File

@ -1,6 +1,6 @@
# #
# This file is for the declaration of global booleans and tunables. # This file is for the declaration of global tunables.
# To change the default value at build time, the tunables.conf # To change the default value at build time, the booleans.conf
# file should be used. # file should be used.
# #
@ -92,11 +92,6 @@ gen_tunable(read_untrusted_content,false)
## Allow ssh to run from inetd instead of as a daemon. ## Allow ssh to run from inetd instead of as a daemon.
gen_tunable(run_ssh_inetd,false) gen_tunable(run_ssh_inetd,false)
## Enabling secure mode disallows programs, such as
## newrole, from transitioning to administrative
## user domains.
gen_bool(secure_mode,false)
## Allow squid to connect to all ports, not just ## Allow squid to connect to all ports, not just
## HTTP, FTP, and Gopher ports. ## HTTP, FTP, and Gopher ports.
gen_tunable(squid_connect_any,false) gen_tunable(squid_connect_any,false)

View File

@ -584,7 +584,7 @@ def gen_docs(doc, working_dir, templatedir):
#build the global boolean index #build the global boolean index
global_bool_buf = [] global_bool_buf = []
for boolean in doc.getElementsByTagName("boolean"): for boolean in doc.getElementsByTagName("bool"):
if boolean.parentNode.nodeName == "policy": if boolean.parentNode.nodeName == "policy":
bool_name = boolean.getAttribute("name") bool_name = boolean.getAttribute("name")
default_value = boolean.getAttribute("dftval") default_value = boolean.getAttribute("dftval")

View File

@ -15,104 +15,51 @@
import sys import sys
import os import os
import glob import glob
import re
# GLOBALS # GLOBALS
class dec_style:
'''
"Declaration Style"
Specifies the syntax of a declaration. Intended to be used with
getParams().
'''
# Example of a line: foo(bar,one,two); # Default values of command line arguments:
# A style that would fit this: dec_style("foo(",3,",",");")
# "foo(" - the opening of it, ends at the begining of the first param.
# 3 - the number of parameters.
# "," - the delimeter to parse apart parameters.
# ");" - the end of the declaration statement.
def __init__(self,open_str,params,delim,close_str):
self.open_str = open_str
self.params = params
self.delim = delim
self.close_str = close_str
INTERFACE = dec_style("interface(`",1,None,"'")
TEMPLATE = dec_style("template(`",1,None,"'")
TUNABLE = dec_style("gen_tunable(",2,",",")")
# boolean FIXME: may have to change in the future.
BOOLEAN = dec_style("gen_bool(",2,",",")")
# Default values of command line arguments.
directory = "./" directory = "./"
warn = False warn = False
meta = "metadata" meta = "metadata"
layers = [] layers = []
tunable_files = [] tunable_files = []
bool_files = []
# Pre compiled regular expressions:
# Matches either an interface or a template declaration. Will give the tuple:
# ("interface" or "template", name)
# Some examples:
# "interface(`kernel_read_system_state',`"
# -> ("interface", "kernel_read_system_state")
# "template(`base_user_template',`"
# -> ("template", "base_user_template")
INTERFACE = re.compile("^\s*(interface|template)\(`([A-Za-z0-9_]*)'")
# Matches either a gen_bool or a gen_tunable statement. Will give the tuple:
# ("tunable" or "bool", name, "true" or "false")
# Some examples:
# "gen_bool(secure_mode, false)"
# -> ("bool", "secure_mode", "false")
# "gen_tunable(allow_kerberos, false)"
# -> ("tunable", "allow_kerberos", "false")
BOOLEAN = re.compile("^\s*gen_(tunable|bool)\(\s*([A-Za-z0-9_]*)\s*,\s*(true|false)\s*\)")
# Matches a XML comment in the policy, which is defined as any line starting
# with two # and at least one character of white space. Will give the single
# valued tuple:
# ("comment")
# Some Examples:
# "## <summary>"
# -> ("<summary>")
# "## The domain allowed access. "
# -> ("The domain allowed access.")
XML_COMMENT = re.compile("^##\s+(.*?)\s*$")
# FUNCTIONS # FUNCTIONS
def getXMLComment(line):
'''
Returns the XML comment, (removes "## " from the front of the line).
Returns False if the line is not an XML comment.
'''
for i in range(0,len(line)-1):
# Check if the first 3 characters are "## "
if line[i:i+3] in ("## ", "##\t"):
# The chars before '#' in the line must be whitespace.
if i > 0 and not line[0:i-1].isspace():
return False
else:
return line[i+3:]
# No XML comment.
return False
def getParams(line, style):
'''
Returns a list of items, containing the values of each parameter.
'''
# Clean out whitespace.
temp_line = line.strip()
# Check to see if the string begins with the specified opening
# string specified by style.
if temp_line[0:len(style.open_str)] == style.open_str:
temp_line = temp_line[len(style.open_str):].strip()
else:
return False
# If there is a delimeter.
if style.delim:
temp_line = temp_line.split(style.delim)
else:
temp_line = [temp_line]
# Only interested in a sertain number of tokens, specified by style.
temp_line = temp_line[:style.params]
# Remove the end of the declaration, specified by style.
end = temp_line[-1].find(style.close_str)
if end == -1:
warning("line \"%s\" may be syntactically incorrect"\
% line.strip())
return False
temp_line[-1] = temp_line[-1][:end]
# Remove whitespace
for i in range(0,len(temp_line)-1):
temp_line[i] = temp_line[i].strip()
return temp_line
def getModuleXML(file_name): def getModuleXML(file_name):
''' '''
Returns the XML data for a module in a list, one line per list item. Returns the XML data for a module in a list, one line per list item.
@ -134,62 +81,65 @@ def getModuleXML(file_name):
% os.path.splitext(os.path.split(file_name)[-1])[0]) % os.path.splitext(os.path.split(file_name)[-1])[0])
temp_buf = [] temp_buf = []
interface = None
# Phases: find header - looking for the header of the file. # finding_header is a flag to denote whether we are still looking
# get header - get the header comments and stop when first # for the XML documentation at the head of the file.
# whitespace is encountered. finding_header = True
# find interface - looking for interfaces to get info for.
phase = "find header" # Get rid of whitespace at top of file
while(module_code and module_code[0].isspace()):
module_code = module_code[1:]
# Go line by line and figure out what to do with it. # Go line by line and figure out what to do with it.
for line in module_code: for line in module_code:
# In this phase, whitespace and stray code is ignored at the if finding_header:
# top of the file. # If there is a XML comment, add it to the temp buffer.
if phase == "find header": comment = XML_COMMENT.match(line)
if comment:
temp_buf.append(comment.group(1) + "\n")
continue
# Once a line that is not an XML comment is reached,
# either put the XML out to module buffer as the
# module's documentation, or attribute it to an
# interface/template.
elif temp_buf:
finding_header = False
interface = INTERFACE.match(line)
if not interface:
module_buf += temp_buf
temp_buf = []
continue
# Skip over empty lines
if line.isspace(): if line.isspace():
continue continue
# Once a comment is encountered, start trying to get the
# header documentation.
elif getXMLComment(line):
phase = "get header"
# If an interface is found, there is no header, and no
# documentation for the interface.
elif getParams(line,INTERFACE)\
or getParams(line,TEMPLATE):
phase = "find interface"
# In this phase, XML comments are being retrieved for the file. # Grab a comment and add it to the temprorary buffer, if it
if phase == "get header": # is there.
if getXMLComment(line): comment = XML_COMMENT.match(line)
temp_buf.append(getXMLComment(line)) if comment:
temp_buf.append(comment.group(1) + "\n")
continue continue
# If the line is whitespace, the file header is over,
# continue on to find interfaces. # Grab the interface information. This is only not true when
elif line.isspace(): # the interface is at the top of the file and there is no
# documentation for the module.
if not interface:
interface = INTERFACE.match(line)
if interface:
# Add the opening tag for the interface/template
module_buf.append("<%s name=\"%s\">\n" % interface.groups())
# Add all the comments attributed to this interface to
# the module buffer.
if temp_buf:
module_buf += temp_buf module_buf += temp_buf
temp_buf = [] temp_buf = []
phase = "find interface"
continue
# Oops! The comments we have been getting weren't part
# of the header so attribute them to an interface
# instead.
elif getParams(line,INTERFACE)\
or getParams(line,TEMPLATE):
phase = "find interface"
# In this phase, XML comments are being attributed # Add default summaries and parameters so that the
if phase == "find interface": # DTD is happy.
if getXMLComment(line):
temp_buf.append(getXMLComment(line))
continue
# If the line is the declaration of a interface,
# infer the interface name and add all the comments
# to the main buffer.
elif getParams(line,INTERFACE):
module_buf.append("<interface name=\"%s\">\n"\
% getParams(line,INTERFACE)[0])
if len(temp_buf):
module_buf += temp_buf
else: else:
module_buf.append("<summary>\n") module_buf.append("<summary>\n")
module_buf.append("Summary is missing!\n") module_buf.append("Summary is missing!\n")
@ -197,31 +147,21 @@ def getModuleXML(file_name):
module_buf.append("<param name=\"?\">\n") module_buf.append("<param name=\"?\">\n")
module_buf.append("Parameter descriptions are missing!\n") module_buf.append("Parameter descriptions are missing!\n")
module_buf.append("</param>\n") module_buf.append("</param>\n")
temp_buf = []
module_buf.append("</interface>\n") # Close the interface/template tag.
module_buf.append("</%s>\n" % interface.group(1))
interface = None
continue continue
elif getParams(line,TEMPLATE):
module_buf.append("<template name =\"%s\">\n"\
% getParams(line,TEMPLATE)[0])
if len(temp_buf):
module_buf += temp_buf
else:
module_buf.append("<summary>\n")
module_buf.append("Summary is missing!\n")
module_buf.append("</summary>\n")
module_buf.append("<param name=\"?\">\n")
module_buf.append("Parameter descriptions are missing!\n")
module_buf.append("</param>\n")
temp_buf = []
module_buf.append("</template>\n")
# The file had no interfaces, just a header.
if phase == "get header":
module_buf += temp_buf
# If there are XML comments at the end of the file, they arn't
# attributed to anything. These are ignored. # If the file just had a header, add the comments to the module buffer.
elif len(temp_buf): if finding_header:
module_buf += temp_buf
# Otherwise there are some lingering XML comments at the bottom, warn
# the user.
elif temp_buf:
warning("orphan XML comments at bottom of file %s" % file_name) warning("orphan XML comments at bottom of file %s" % file_name)
module_buf.append("</module>\n") module_buf.append("</module>\n")
@ -256,9 +196,9 @@ def getLayerXML(directory):
return layer_buf return layer_buf
def getTunableXML(file_name): def getTunableXML(file_name, kind):
''' '''
Return all the XML for the tunables in the file specified. Return all the XML for the tunables/bools in the file specified.
''' '''
# Try to open the file, if it cant, just ignore it. # Try to open the file, if it cant, just ignore it.
@ -277,28 +217,26 @@ def getTunableXML(file_name):
# them. # them.
for line in tunable_code: for line in tunable_code:
# If it is an XML comment, add it to the buffer and go on. # If it is an XML comment, add it to the buffer and go on.
if getXMLComment(line): comment = XML_COMMENT.match(line)
temp_buf.append(getXMLComment(line)) if comment:
temp_buf.append(comment.group(1))
continue continue
# Get the parameters of a TUNABLE style line. # Get the boolean/tunable data.
params = getParams(line,TUNABLE) boolean = BOOLEAN.match(line)
tag = "tunable"
# If the line is not a TUNABLE style declaration, try BOOLEAN. # If we reach a boolean/tunable declaration, attribute all XML
if not params: # in the temp buffer to it and add XML to the tunable buffer.
params = getParams(line,BOOLEAN) if boolean:
tag = "boolean" # If there is a gen_bool in a tunable file or a
# gen_tunable in a boolean file, error and exit.
if boolean.group(1) != kind:
error("%s in a %s file." % (boolean.group(1), kind))
# If the line is one of the two styles above, add a tunable tag tunable_buf.append("<%s name=\"%s\" dftval=\"%s\">\n" % boolean.groups())
# and give it the data from the temprorary buffer.
if params:
tunable_buf.append\
("<%s name=\"%s\" dftval=\"%s\">\n"
% (tag, params[0], params[1]))
tunable_buf += temp_buf tunable_buf += temp_buf
temp_buf = [] temp_buf = []
tunable_buf.append("</%s>\n" % tag) tunable_buf.append("</%s>\n" % boolean.group(1))
# If there are XML comments at the end of the file, they arn't # If there are XML comments at the end of the file, they arn't
# attributed to anything. These are ignored. # attributed to anything. These are ignored.
@ -332,9 +270,13 @@ def getPolicyXML(directory):
for layer in layers: for layer in layers:
policy_buf += getLayerXML(layer) policy_buf += getLayerXML(layer)
# Add to the XML each tunable specified by the user. # Add to the XML each tunable file specified by the user.
for tunable_file in tunable_files: for tunable_file in tunable_files:
policy_buf += getTunableXML(tunable_file) policy_buf += getTunableXML(tunable_file, "tunable")
# Add to the XML each bool file specified by the user.
for bool_file in bool_files:
policy_buf += getTunableXML(bool_file, "bool")
policy_buf.append("</policy>\n") policy_buf.append("</policy>\n")
@ -371,6 +313,9 @@ def usage():
sys.stdout.write("-t --tunable <file> -- "+\ sys.stdout.write("-t --tunable <file> -- "+\
"A file containing tunable declarations\n") "A file containing tunable declarations\n")
sys.stdout.write("-b --bool <file> -- "+\
"A file containing bool declarations\n")
def warning(description): def warning(description):
''' '''
Warns the user of a non-critical error. Warns the user of a non-critical error.
@ -402,7 +347,7 @@ if len(sys.argv) <= 1:
# Parse the command line arguments # Parse the command line arguments
for i in range(1, len(sys.argv)): for i in range(1, len(sys.argv)):
if sys.argv[i-1] in ("-d", "--directory", "-m", "--meta",\ if sys.argv[i-1] in ("-d", "--directory", "-m", "--meta",\
"-t", "--tunable"): "-t", "--tunable", "-b", "--bool"):
continue continue
elif sys.argv[i] in ("-w", "--warn"): elif sys.argv[i] in ("-w", "--warn"):
warn = True warn = True
@ -421,6 +366,12 @@ for i in range(1, len(sys.argv)):
tunable_files.append(sys.argv[i+1]) tunable_files.append(sys.argv[i+1])
else: else:
usage() usage()
elif sys.argv[i] in ("-b", "--bool"):
if i < len(sys.argv)-1:
bool_files.append(sys.argv[i+1])
else:
usage()
else: else:
layers.append(sys.argv[i]) layers.append(sys.argv[i])