2016-06-09 22:50:03 +00:00
"""
mbed SDK
Copyright ( c ) 2016 ARM Limited
Licensed under the Apache License , Version 2.0 ( the " License " ) ;
you may not use this file except in compliance with the License .
You may obtain a copy of the License at
http : / / www . apache . org / licenses / LICENSE - 2.0
Unless required by applicable law or agreed to in writing , software
distributed under the License is distributed on an " AS IS " BASIS ,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND , either express or implied .
See the License for the specific language governing permissions and
limitations under the License .
"""
# Implementation of mbed configuration mechanism
from copy import deepcopy
from collections import OrderedDict
from tools . utils import json_file_to_dict , ToolException
from tools . targets import Target
import os
# Base class for all configuration exceptions
class ConfigException ( Exception ) :
pass
# This class keeps information about a single configuration parameter
class ConfigParameter :
# name: the name of the configuration parameter
# data: the data associated with the configuration parameter
# unit_name: the unit (target/library/application) that defines this parameter
# unit_ kind: the kind of the unit ("target", "library" or "application")
def __init__ ( self , name , data , unit_name , unit_kind ) :
self . name = self . get_full_name ( name , unit_name , unit_kind , allow_prefix = False )
self . defined_by = self . get_display_name ( unit_name , unit_kind )
2016-06-17 09:56:10 +00:00
self . set_value ( data . get ( " value " , None ) , unit_name , unit_kind )
2016-06-09 22:50:03 +00:00
self . help_text = data . get ( " help " , None )
self . required = data . get ( " required " , False )
self . macro_name = data . get ( " macro_name " , " MBED_CONF_ %s " % self . sanitize ( self . name . upper ( ) ) )
2016-06-15 12:10:32 +00:00
self . config_errors = [ ]
2016-06-09 22:50:03 +00:00
# Return the full (prefixed) name of a parameter.
# If the parameter already has a prefix, check if it is valid
# name: the simple (unqualified) name of the parameter
# unit_name: the unit (target/library/application) that defines this parameter
# unit_kind: the kind of the unit ("target", "library" or "application")
# label: the name of the label in the 'target_config_overrides' section (optional)
# allow_prefix: True to allo the original name to have a prefix, False otherwise
@staticmethod
def get_full_name ( name , unit_name , unit_kind , label = None , allow_prefix = True ) :
if name . find ( ' . ' ) == - 1 : # the name is not prefixed
if unit_kind == " target " :
prefix = " target. "
elif unit_kind == " application " :
prefix = " app. "
else :
prefix = unit_name + ' . '
return prefix + name
# The name has a prefix, so check if it is valid
if not allow_prefix :
raise ConfigException ( " Invalid parameter name ' %s ' in ' %s ' " % ( name , ConfigParameter . get_display_name ( unit_name , unit_kind , label ) ) )
temp = name . split ( " . " )
# Check if the parameter syntax is correct (must be unit_name.parameter_name)
if len ( temp ) != 2 :
raise ConfigException ( " Invalid parameter name ' %s ' in ' %s ' " % ( name , ConfigParameter . get_display_name ( unit_name , unit_kind , label ) ) )
prefix = temp [ 0 ]
# Check if the given parameter prefix matches the expected prefix
if ( unit_kind == " library " and prefix != unit_name ) or ( unit_kind == " target " and prefix != " target " ) :
raise ConfigException ( " Invalid prefix ' %s ' for parameter name ' %s ' in ' %s ' " % ( prefix , name , ConfigParameter . get_display_name ( unit_name , unit_kind , label ) ) )
return name
# Return the name displayed for a unit when interogating the origin
# and the last set place of a parameter
# unit_name: the unit (target/library/application) that defines this parameter
# unit_kind: the kind of the unit ("target", "library" or "application")
# label: the name of the label in the 'target_config_overrides' section (optional)
@staticmethod
def get_display_name ( unit_name , unit_kind , label = None ) :
if unit_kind == " target " :
return " target: " + unit_name
elif unit_kind == " application " :
return " application %s " % ( " [ %s ] " % label if label else " " )
else : # library
return " library: %s %s " % ( unit_name , " [ %s ] " % label if label else " " )
# "Sanitize" a name so that it is a valid C macro name
# Currently it simply replaces '.' and '-' with '_'
# name: the un-sanitized name.
@staticmethod
def sanitize ( name ) :
return name . replace ( ' . ' , ' _ ' ) . replace ( ' - ' , ' _ ' )
2016-06-17 09:56:10 +00:00
# Sets a value for this parameter, remember the place where it was set.
# If the value is a boolean, it is converted to 1 (for True) or to 0 (for False).
2016-06-09 22:50:03 +00:00
# value: the value of the parameter
# unit_name: the unit (target/library/application) that defines this parameter
# unit_ kind: the kind of the unit ("target", "library" or "application")
# label: the name of the label in the 'target_config_overrides' section (optional)
def set_value ( self , value , unit_name , unit_kind , label = None ) :
2016-06-17 09:56:10 +00:00
self . value = int ( value ) if isinstance ( value , bool ) else value
2016-06-09 22:50:03 +00:00
self . set_by = self . get_display_name ( unit_name , unit_kind , label )
# Return the string representation of this configuration parameter
def __str__ ( self ) :
if self . value is not None :
return ' %s = %s (macro name: " %s " ) ' % ( self . name , self . value , self . macro_name )
else :
return ' %s has no value ' % self . name
# Return a verbose description of this configuration paramater as a string
def get_verbose_description ( self ) :
desc = " Name: %s %s \n " % ( self . name , " (required parameter) " if self . required else " " )
if self . help_text :
desc + = " Description: %s \n " % self . help_text
desc + = " Defined by: %s \n " % self . defined_by
if not self . value :
return desc + " No value set "
desc + = " Macro name: %s \n " % self . macro_name
desc + = " Value: %s (set by %s ) " % ( self . value , self . set_by )
return desc
# A representation of a configuration macro. It handles both macros without a value (MACRO)
# and with a value (MACRO=VALUE)
class ConfigMacro :
def __init__ ( self , name , unit_name , unit_kind ) :
self . name = name
self . defined_by = ConfigParameter . get_display_name ( unit_name , unit_kind )
if name . find ( " = " ) != - 1 :
tmp = name . split ( " = " )
if len ( tmp ) != 2 :
raise ValueError ( " Invalid macro definition ' %s ' in ' %s ' " % ( name , self . defined_by ) )
self . macro_name = tmp [ 0 ]
2016-06-15 21:23:04 +00:00
self . macro_value = tmp [ 1 ]
2016-06-09 22:50:03 +00:00
else :
self . macro_name = name
2016-06-15 21:23:04 +00:00
self . macro_value = None
2016-06-09 22:50:03 +00:00
# 'Config' implements the mbed configuration mechanism
class Config :
# Libraries and applications have different names for their configuration files
__mbed_app_config_name = " mbed_app.json "
__mbed_lib_config_name = " mbed_lib.json "
# Allowed keys in configuration dictionaries
# (targets can have any kind of keys, so this validation is not applicable to them)
__allowed_keys = {
2016-06-11 20:30:26 +00:00
" library " : set ( [ " name " , " config " , " target_overrides " , " macros " , " __config_path " ] ) ,
" application " : set ( [ " config " , " custom_targets " , " target_overrides " , " macros " , " __config_path " ] )
2016-06-09 22:50:03 +00:00
}
2016-06-14 17:41:29 +00:00
# Allowed features in configurations
__allowed_features = [
" UVISOR " , " BLE " , " CLIENT " , " IPV4 " , " IPV6 "
]
2016-06-09 22:50:03 +00:00
# The initialization arguments for Config are:
# target: the name of the mbed target used for this configuration instance
# top_level_dirs: a list of top level source directories (where mbed_abb_config.json could be found)
# __init__ will look for the application configuration file in top_level_dirs.
# If found once, it'll parse it and check if it has a custom_targets function.
# If it does, it'll update the list of targets if need.
# If found more than once, an exception is raised
# top_level_dirs can be None (in this case, mbed_app_config.json will not be searched)
def __init__ ( self , target , top_level_dirs = [ ] ) :
app_config_location = None
for s in ( top_level_dirs or [ ] ) :
full_path = os . path . join ( s , self . __mbed_app_config_name )
if os . path . isfile ( full_path ) :
if app_config_location is not None :
raise ConfigException ( " Duplicate ' %s ' file in ' %s ' and ' %s ' " % ( self . __mbed_app_config_name , app_config_location , full_path ) )
else :
app_config_location = full_path
self . app_config_data = json_file_to_dict ( app_config_location ) if app_config_location else { }
# Check the keys in the application configuration data
unknown_keys = set ( self . app_config_data . keys ( ) ) - self . __allowed_keys [ " application " ]
if unknown_keys :
raise ConfigException ( " Unknown key(s) ' %s ' in %s " % ( " , " . join ( unknown_keys ) , self . __mbed_app_config_name ) )
# Update the list of targets with the ones defined in the application config, if applicable
Target . add_py_targets ( self . app_config_data . get ( " custom_targets " , { } ) )
self . lib_config_data = { }
# Make sure that each config is processed only once
self . processed_configs = { }
self . target = target if isinstance ( target , str ) else target . name
self . target_labels = Target . get_target ( self . target ) . get_labels ( )
2016-06-14 17:36:41 +00:00
self . added_features = set ( )
self . removed_features = set ( )
self . removed_unecessary_features = False
2016-06-09 22:50:03 +00:00
# Add one or more configuration files
def add_config_files ( self , flist ) :
for f in flist :
if not f . endswith ( self . __mbed_lib_config_name ) :
continue
full_path = os . path . normpath ( os . path . abspath ( f ) )
# Check that we didn't already process this file
if self . processed_configs . has_key ( full_path ) :
continue
self . processed_configs [ full_path ] = True
# Read the library configuration and add a "__full_config_path" attribute to it
cfg = json_file_to_dict ( f )
cfg [ " __config_path " ] = full_path
# If there's already a configuration for a module with the same name, exit with error
if self . lib_config_data . has_key ( cfg [ " name " ] ) :
raise ConfigException ( " Library name ' %s ' is not unique (defined in ' %s ' and ' %s ' ) " % ( cfg [ " name " ] , full_path , self . lib_config_data [ cfg [ " name " ] ] [ " __config_path " ] ) )
self . lib_config_data [ cfg [ " name " ] ] = cfg
# Helper function: process a "config_parameters" section in either a target, a library or the application
# data: a dictionary with the configuration parameters
# params: storage for the discovered configuration parameters
# unit_name: the unit (target/library/application) that defines this parameter
# unit_kind: the kind of the unit ("target", "library" or "application")
def _process_config_parameters ( self , data , params , unit_name , unit_kind ) :
for name , v in data . items ( ) :
full_name = ConfigParameter . get_full_name ( name , unit_name , unit_kind )
# If the parameter was already defined, raise an error
if full_name in params :
raise ConfigException ( " Parameter name ' %s ' defined in both ' %s ' and ' %s ' " % ( name , ConfigParameter . get_display_name ( unit_name , unit_kind ) , params [ full_name ] . defined_by ) )
# Otherwise add it to the list of known parameters
# If "v" is not a dictionary, this is a shortcut definition, otherwise it is a full definition
params [ full_name ] = ConfigParameter ( name , v if isinstance ( v , dict ) else { " value " : v } , unit_name , unit_kind )
return params
2016-06-14 17:36:41 +00:00
# Add features to the available features
def remove_features ( self , features ) :
for feature in features :
if feature in self . added_features :
raise ConfigException ( " Configuration conflict. Feature %s both added and removed. " % feature )
self . removed_features | = set ( features )
# Remove features from the available features
def add_features ( self , features ) :
for feature in features :
2016-06-17 09:19:11 +00:00
if ( feature in self . removed_features
2016-06-14 17:36:41 +00:00
or ( self . removed_unecessary_features and feature not in self . added_features ) ) :
raise ConfigException ( " Configuration conflict. Feature %s both added and removed. " % feature )
self . added_features | = set ( features )
2016-06-09 22:50:03 +00:00
# Helper function: process "config_parameters" and "target_config_overrides" in a given dictionary
# data: the configuration data of the library/appliation
# params: storage for the discovered configuration parameters
# unit_name: the unit (library/application) that defines this parameter
# unit_kind: the kind of the unit ("library" or "application")
def _process_config_and_overrides ( self , data , params , unit_name , unit_kind ) :
2016-06-15 12:10:32 +00:00
self . config_errors = [ ]
2016-06-09 22:50:03 +00:00
self . _process_config_parameters ( data . get ( " config " , { } ) , params , unit_name , unit_kind )
for label , overrides in data . get ( " target_overrides " , { } ) . items ( ) :
# If the label is defined by the target or it has the special value "*", process the overrides
if ( label == ' * ' ) or ( label in self . target_labels ) :
2016-06-14 17:36:41 +00:00
# Parse out features
2016-06-15 15:25:31 +00:00
if ' target.features ' in overrides :
features = overrides [ ' target.features ' ]
self . remove_features ( self . added_features - set ( features ) )
2016-06-14 17:36:41 +00:00
self . add_features ( features )
self . removed_unecessary_features = True
2016-06-15 15:25:31 +00:00
del overrides [ ' target.features ' ]
2016-06-14 17:36:41 +00:00
2016-06-15 15:25:31 +00:00
if ' target.features_add ' in overrides :
self . add_features ( overrides [ ' target.features_add ' ] )
del overrides [ ' target.features_add ' ]
2016-06-14 17:36:41 +00:00
2016-06-15 15:25:31 +00:00
if ' target.features_remove ' in overrides :
self . remove_features ( overrides [ ' target.features_remove ' ] )
del overrides [ ' target.features_remove ' ]
2016-06-11 20:30:26 +00:00
# Consider the others as overrides
2016-06-09 22:50:03 +00:00
for name , v in overrides . items ( ) :
# Get the full name of the parameter
full_name = ConfigParameter . get_full_name ( name , unit_name , unit_kind , label )
2016-06-15 00:42:31 +00:00
if full_name in params :
params [ full_name ] . set_value ( v , unit_name , unit_kind , label )
2016-06-15 12:10:32 +00:00
else :
2016-06-17 09:19:11 +00:00
self . config_errors . append ( ConfigException ( " Attempt to override undefined parameter ' %s ' in ' %s ' "
2016-06-15 12:10:32 +00:00
% ( full_name , ConfigParameter . get_display_name ( unit_name , unit_kind , label ) ) ) )
2016-06-09 22:50:03 +00:00
return params
# Read and interpret configuration data defined by targets
def get_target_config_data ( self ) :
# We consider the resolution order for our target and sort it by level reversed,
# so that we first look at the top level target (the parent), then its direct children,
# then the children's children and so on, until we reach self.target
# TODO: this might not work so well in some multiple inheritance scenarios
# At each step, look at two keys of the target data:
# - config_parameters: used to define new configuration parameters
# - config_overrides: used to override already defined configuration parameters
params , json_data = { } , Target . get_json_target_data ( )
resolution_order = [ e [ 0 ] for e in sorted ( Target . get_target ( self . target ) . resolution_order , key = lambda e : e [ 1 ] , reverse = True ) ]
for tname in resolution_order :
# Read the target data directly from its description
t = json_data [ tname ]
# Process definitions first
self . _process_config_parameters ( t . get ( " config " , { } ) , params , tname , " target " )
# Then process overrides
for name , v in t . get ( " overrides " , { } ) . items ( ) :
full_name = ConfigParameter . get_full_name ( name , tname , " target " )
# If the parameter name is not defined or if there isn't a path from this target to the target where the
# parameter was defined in the target inheritance tree, raise an error
# We need to use 'defined_by[7:]' to remove the "target:" prefix from defined_by
if ( not full_name in params ) or ( not params [ full_name ] . defined_by [ 7 : ] in Target . get_target ( tname ) . resolution_order_names ) :
raise ConfigException ( " Attempt to override undefined parameter ' %s ' in ' %s ' " % ( name , ConfigParameter . get_display_name ( tname , " target " ) ) )
# Otherwise update the value of the parameter
params [ full_name ] . set_value ( v , tname , " target " )
return params
# Helper function: process a macro definition, checking for incompatible duplicate definitions
# mlist: list of macro names to process
# macros: dictionary with currently discovered macros
# unit_name: the unit (library/application) that defines this macro
# unit_kind: the kind of the unit ("library" or "application")
def _process_macros ( self , mlist , macros , unit_name , unit_kind ) :
for mname in mlist :
m = ConfigMacro ( mname , unit_name , unit_kind )
if ( m . macro_name in macros ) and ( macros [ m . macro_name ] . name != mname ) :
# Found an incompatible definition of the macro in another module, so raise an error
full_unit_name = ConfigParameter . get_display_name ( unit_name , unit_kind )
raise ConfigException ( " Macro ' %s ' defined in both ' %s ' and ' %s ' with incompatible values " % ( m . macro_name , macros [ m . macro_name ] . defined_by , full_unit_name ) )
macros [ m . macro_name ] = m
# Read and interpret configuration data defined by libs
# It is assumed that "add_config_files" above was already called and the library configuration data
# exists in self.lib_config_data
def get_lib_config_data ( self ) :
all_params , macros = { } , { }
for lib_name , lib_data in self . lib_config_data . items ( ) :
unknown_keys = set ( lib_data . keys ( ) ) - self . __allowed_keys [ " library " ]
if unknown_keys :
raise ConfigException ( " Unknown key(s) ' %s ' in %s " % ( " , " . join ( unknown_keys ) , lib_name ) )
all_params . update ( self . _process_config_and_overrides ( lib_data , { } , lib_name , " library " ) )
self . _process_macros ( lib_data . get ( " macros " , [ ] ) , macros , lib_name , " library " )
return all_params , macros
# Read and interpret the configuration data defined by the target
# The target can override any configuration parameter, as well as define its own configuration data
# params: the dictionary with configuration parameters found so far (in the target and in libraries)
# macros: the list of macros defined in the configuration
def get_app_config_data ( self , params , macros ) :
app_cfg = self . app_config_data
# The application can have a "config_parameters" and a "target_config_overrides" section just like a library
self . _process_config_and_overrides ( app_cfg , params , " app " , " application " )
# The application can also defined macros
self . _process_macros ( app_cfg . get ( " macros " , [ ] ) , macros , " app " , " application " )
# Return the configuration data in two parts:
# - params: a dictionary with (name, ConfigParam) entries
2016-06-15 21:23:04 +00:00
# - macros: the list of macros defined with "macros" in libraries and in the application (as ConfigMacro instances)
2016-06-09 22:50:03 +00:00
def get_config_data ( self ) :
all_params = self . get_target_config_data ( )
lib_params , macros = self . get_lib_config_data ( )
all_params . update ( lib_params )
self . get_app_config_data ( all_params , macros )
2016-06-15 21:23:04 +00:00
return all_params , macros
2016-06-09 22:50:03 +00:00
# Helper: verify if there are any required parameters without a value in 'params'
def _check_required_parameters ( self , params ) :
for p in params . values ( ) :
if p . required and ( p . value is None ) :
raise ConfigException ( " Required parameter ' %s ' defined by ' %s ' doesn ' t have a value " % ( p . name , p . defined_by ) )
# Return the macro definitions generated for a dictionary of configuration parameters
# params: a dictionary of (name, ConfigParameters instance) mappings
@staticmethod
def parameters_to_macros ( params ) :
return [ ' %s = %s ' % ( m . macro_name , m . value ) for m in params . values ( ) if m . value is not None ]
2016-06-15 21:23:04 +00:00
# Return the macro definitions generated for a dictionary of ConfigMacros (as returned by get_config_data)
# params: a dictionary of (name, ConfigMacro instance) mappings
@staticmethod
def config_macros_to_macros ( macros ) :
return [ m . name for m in macros . values ( ) ]
2016-06-09 22:50:03 +00:00
# Return the configuration data converted to a list of C macros
def get_config_data_macros ( self ) :
params , macros = self . get_config_data ( )
self . _check_required_parameters ( params )
2016-06-15 21:23:04 +00:00
return self . config_macros_to_macros ( macros ) + self . parameters_to_macros ( params )
2016-06-14 17:36:41 +00:00
# Returns any features in the configuration data
def get_features ( self ) :
params , _ = self . get_config_data ( )
self . _check_required_parameters ( params )
2016-06-14 17:41:29 +00:00
features = ( ( set ( Target . get_target ( self . target ) . features )
2016-06-14 17:36:41 +00:00
| self . added_features ) - self . removed_features )
2016-06-14 17:41:29 +00:00
for feature in features :
if feature not in self . __allowed_features :
raise ConfigException ( " Feature ' %s ' is not a supported features " % feature )
return features
2016-06-15 12:10:32 +00:00
# Validate configuration settings. This either returns True or raises an exception
def validate_config ( self ) :
if self . config_errors :
raise self . config_errors [ 0 ]
return True
2016-06-15 21:23:04 +00:00
2016-06-17 09:19:11 +00:00
2016-06-16 10:52:41 +00:00
# Loads configuration data from resources. Also expands resources based on defined features settings
def load_resources ( self , resources ) :
# Update configuration files until added features creates no changes
prev_features = set ( )
while True :
# Add/update the configuration with any .json files found while scanning
self . add_config_files ( resources . json_files )
# Add features while we find new ones
features = self . get_features ( )
if features == prev_features :
break
for feature in features :
if feature in resources . features :
resources . add ( resources . features [ feature ] )
prev_features = features
self . validate_config ( )
return resources
2016-06-17 09:19:11 +00:00
2016-06-16 10:52:41 +00:00
2016-06-15 21:23:04 +00:00
# Return the configuration data converted to the content of a C header file,
# meant to be included to a C/C++ file. The content is returned as a string.
# If 'fname' is given, the content is also written to the file called "fname".
# WARNING: if 'fname' names an existing file, that file will be overwritten!
def get_config_data_header ( self , fname = None ) :
params , macros = self . get_config_data ( )
self . _check_required_parameters ( params )
header_data = " // Automatically generated configuration file. \n "
header_data + = " // DO NOT EDIT, content will be overwritten. \n \n "
header_data + = " #ifndef __MBED_CONFIG_DATA__ \n "
header_data + = " #define __MBED_CONFIG_DATA__ \n \n "
# Compute maximum length of macro names for proper alignment
max_param_macro_name_len = max ( [ len ( m . macro_name ) for m in params . values ( ) if m . value is not None ] ) if params else 0
max_direct_macro_name_len = max ( [ len ( m . macro_name ) for m in macros . values ( ) ] ) if macros else 0
max_macro_name_len = max ( max_param_macro_name_len , max_direct_macro_name_len )
# Compute maximum length of macro values for proper alignment
max_param_macro_val_len = max ( [ len ( str ( m . value ) ) for m in params . values ( ) if m . value is not None ] ) if params else 0
max_direct_macro_val_len = max ( [ len ( m . macro_value or " " ) for m in macros . values ( ) ] ) if macros else 0
max_macro_val_len = max ( max_param_macro_val_len , max_direct_macro_val_len )
# Generate config parameters first
if params :
header_data + = " // Configuration parameters \n "
for m in params . values ( ) :
if m . value is not None :
2016-06-16 08:41:22 +00:00
header_data + = " #define { 0:< {1} } { 2!s:< {3} } // set by {4} \n " . format ( m . macro_name , max_macro_name_len , m . value , max_macro_val_len , m . set_by )
2016-06-15 21:23:04 +00:00
# Then macros
if macros :
header_data + = " // Macros \n "
for m in macros . values ( ) :
if m . macro_value :
2016-06-16 08:41:22 +00:00
header_data + = " #define { 0:< {1} } { 2!s:< {3} } // defined by {4} \n " . format ( m . macro_name , max_macro_name_len , m . macro_value , max_macro_val_len , m . defined_by )
2016-06-15 21:23:04 +00:00
else :
2016-06-16 08:41:22 +00:00
header_data + = " #define { 0:< {1} } // defined by {2} \n " . format ( m . macro_name , max_macro_name_len + max_macro_val_len + 1 , m . defined_by )
2016-06-15 21:23:04 +00:00
header_data + = " \n #endif \n "
# If fname is given, write "header_data" to it
if fname :
with open ( fname , " wt " ) as f :
f . write ( header_data )
return header_data