/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2022 Mikolaj Wielgus * Copyright (C) 2022 KiCad Developers, see AUTHORS.txt for contributors. * * 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 3 * 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, you may find one here: * https://www.gnu.org/licenses/gpl-3.0.html * or you may search the http://www.gnu.org website for the version 3 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include #include #include #include namespace SIM_MODEL_SERIALIZER_PARSER { using namespace SIM_MODEL_SERIALIZER_GRAMMAR; template struct fieldParamValuePairsSelector : std::false_type {}; template <> struct fieldParamValuePairsSelector : std::true_type {}; template <> struct fieldParamValuePairsSelector : std::true_type {}; template <> struct fieldParamValuePairsSelector : std::true_type {}; template struct pinSequenceSelector : std::false_type {}; template <> struct pinSequenceSelector : std::true_type {}; template <> struct pinSequenceSelector : std::true_type {}; template <> struct pinSequenceSelector : std::true_type {}; template struct fieldInferValueSelector : std::false_type {}; template <> struct fieldInferValueSelector> : std::true_type {}; } std::string SIM_MODEL_SERIALIZER::GenerateDevice() const { return m_model.GetDeviceInfo().fieldValue; } std::string SIM_MODEL_SERIALIZER::GenerateType() const { return m_model.GetTypeInfo().fieldValue; } std::string SIM_MODEL_SERIALIZER::GenerateValue() const { const SIM_MODEL::PARAM& param = m_model.GetParamOverride( 0 ); std::string result = param.value->ToString(); if( result == "" ) result = m_model.GetDeviceInfo().fieldValue; return result; } std::string SIM_MODEL_SERIALIZER::GenerateParams() const { std::string result; bool isFirst = true; for( int i = 0; i < m_model.GetParamCount(); ++i ) { if( i == 0 && m_model.IsStoredInValue() ) continue; const SIM_MODEL::PARAM& param = m_model.GetParamOverride( i ); if( param.value->ToString() == "" && !( i == 0 && m_model.HasPrimaryValue() && !m_model.IsStoredInValue() ) ) { continue; } // If the parameter is an enum and the value is default, don't write anything. if( param.info.enumValues.size() >= 1 && param.value->ToString() == param.info.defaultValue ) continue; std::string paramValuePair = GenerateParamValuePair( param ); if( paramValuePair == "" ) continue; // Prevent adding empty spaces. if( isFirst ) // Don't add a space at the beginning. isFirst = false; else result.append( " " ); result.append( paramValuePair ); } return result; } std::string SIM_MODEL_SERIALIZER::GeneratePins() const { std::string result; std::vector> pins = m_model.GetPins(); // m_model.GetPins() returns pins in the order they appear in the model, but the keys in the // key=value pairs we create here are symbol pin numbers, so we sort the pins so that they are // ordered by the latter instead. std::sort( pins.begin(), pins.end(), []( const SIM_MODEL::PIN& lhs, const SIM_MODEL::PIN& rhs ) { return StrNumCmp( lhs.symbolPinNumber, rhs.symbolPinNumber, true ) < 0; } ); bool isFirst = true; for( const SIM_MODEL::PIN& pin : pins ) { if( pin.symbolPinNumber != "" ) { if( !isFirst ) result.append( " " ); else isFirst = false; result.append( fmt::format( "{}={}", pin.symbolPinNumber, pin.name ) ); } } return result; } std::string SIM_MODEL_SERIALIZER::GenerateEnable() const { return m_model.IsEnabled() ? "" : "0"; } SIM_MODEL::TYPE SIM_MODEL_SERIALIZER::ParseDeviceAndType( const std::string& aDevice, const std::string& aType ) { for( SIM_MODEL::TYPE type : SIM_MODEL::TYPE_ITERATOR() ) { if( aType == SIM_MODEL::TypeInfo( type ).fieldValue && aDevice == SIM_MODEL::DeviceInfo( SIM_MODEL::TypeInfo( type ).deviceType ).fieldValue ) { return type; } } return SIM_MODEL::TYPE::NONE; } void SIM_MODEL_SERIALIZER::ParseValue( const std::string& aValue ) { try { tao::pegtl::string_input<> in( aValue, SIM_MODEL::VALUE_FIELD ); auto root = tao::pegtl::parse_tree::parse ( in ); for( const auto& node : root->children ) { if( node->is_type>() && node->string() != "" ) { m_model.SetParamValue( 0, node->string() ); } } } catch( const tao::pegtl::parse_error& e ) { wxString msg; msg.Printf( wxT( "ParseValue: parse <%s>, error <%s>" ), aValue.c_str(), e.what() ); THROW_IO_ERROR( msg ); } m_model.SetIsStoredInValue( true ); } bool SIM_MODEL_SERIALIZER::ParseParams( const std::string& aParams ) { tao::pegtl::string_input<> in( aParams, SIM_MODEL::PARAMS_FIELD ); std::unique_ptr root; try { // Using parse tree instead of actions because we don't care about performance that much, // and having a tree greatly simplifies things. root = tao::pegtl::parse_tree::parse ( in ); } catch( const tao::pegtl::parse_error& e ) { wxString msg; msg.Printf( wxT( "ParseParams: parse <%s>, error <%s>" ), aParams.c_str(), e.what() ); THROW_IO_ERROR( msg ); } std::string paramName; bool isPrimaryValueSet = false; for( const auto& node : root->children ) { if( node->is_type() ) paramName = node->string(); // TODO: Do something with number. // It doesn't seem too useful? else if( node->is_type() || node->is_type() ) { wxASSERT( paramName != "" ); // TODO: Shouldn't be named "...fromSpiceCode" here... m_model.SetParamValue( paramName, node->string(), SIM_VALUE_GRAMMAR::NOTATION::SI ); if( paramName == m_model.GetParam( 0 ).info.name ) isPrimaryValueSet = true; } else if( node->is_type() ) { std::string str = node->string(); // Unescape quotes. boost::replace_all( str, "\\\"", "\"" ); m_model.SetParamValue( paramName, str, SIM_VALUE_GRAMMAR::NOTATION::SI ); } else { wxFAIL; } } return !m_model.HasPrimaryValue() || isPrimaryValueSet; } void SIM_MODEL_SERIALIZER::ParsePins( const std::string& aPins ) { if( aPins == "" ) return; tao::pegtl::string_input<> in( aPins, PINS_FIELD ); std::unique_ptr root; try { root = tao::pegtl::parse_tree::parse ( in ); for( const auto& node : root->children ) { std::string symbolPinNumber = node->children.at( 0 )->string(); std::string pinName = node->children.at( 1 )->string(); m_model.SetPinSymbolPinNumber( pinName, symbolPinNumber ); } } catch( const tao::pegtl::parse_error& e ) { wxString msg; msg.Printf( wxT( "ParsePins: parse <%s>, error <%s>" ), aPins.c_str(), e.what() ); THROW_IO_ERROR( msg ); } } void SIM_MODEL_SERIALIZER::ParseEnable( const std::string& aEnable ) { if( aEnable == "" ) return; char c = boost::to_lower_copy( aEnable )[0]; if( c == 'n' || c == 'f' || c == '0' ) m_model.SetIsEnabled( false ); } std::string SIM_MODEL_SERIALIZER::GenerateParamValuePair( const SIM_MODEL::PARAM& aParam ) const { std::string name = aParam.info.name; // Because of collisions with instance parameters, we append some model parameters with "_". if( boost::ends_with( aParam.info.name, "_" ) ) name = aParam.info.name.substr( 0, aParam.info.name.length() - 1 ); std::string value = aParam.value->ToString(); if( value == "" || value.find( ' ' ) != std::string::npos ) value = fmt::format( "\"{}\"", value ); return fmt::format( "{}={}", name, value ); }