Bulletproof Simulation settings dialog

This commit is contained in:
Maciej Suminski 2016-08-11 14:41:54 +02:00
parent 16fec4d73e
commit 65c7520544
4 changed files with 141 additions and 32 deletions

View File

@ -24,36 +24,35 @@
#include "dialog_sim_settings.h" #include "dialog_sim_settings.h"
#include <sim/netlist_exporter_pspice_sim.h> #include <sim/netlist_exporter_pspice_sim.h>
#include <confirm.h>
/// @todo ngspice offers more types of analysis, /// @todo ngspice offers more types of analysis,
//so there are a few tabs missing (e.g. pole-zero, distortion, sensitivity) //so there are a few tabs missing (e.g. pole-zero, distortion, sensitivity)
DIALOG_SIM_SETTINGS::DIALOG_SIM_SETTINGS( wxWindow* aParent ) DIALOG_SIM_SETTINGS::DIALOG_SIM_SETTINGS( wxWindow* aParent )
: DIALOG_SIM_SETTINGS_BASE( aParent ), m_exporter( nullptr ) : DIALOG_SIM_SETTINGS_BASE( aParent ), m_exporter( nullptr ), m_spiceEmptyValidator( true )
{ {
m_posFloatValidator.SetMin( 0 );
m_posFloatValidator.SetPrecision( 6 );
m_posIntValidator.SetMin( 1 ); m_posIntValidator.SetMin( 1 );
m_acPointsNumber->SetValidator( m_posIntValidator ); m_acPointsNumber->SetValidator( m_posIntValidator );
m_acFreqStart->SetValidator( m_posFloatValidator ); m_acFreqStart->SetValidator( m_spiceValidator );
m_acFreqStop->SetValidator( m_posFloatValidator ); m_acFreqStop->SetValidator( m_spiceValidator );
m_dcStart1->SetValidator( m_posFloatValidator ); m_dcStart1->SetValidator( m_spiceValidator );
m_dcStop1->SetValidator( m_posFloatValidator ); m_dcStop1->SetValidator( m_spiceValidator );
m_dcIncr1->SetValidator( m_posFloatValidator ); m_dcIncr1->SetValidator( m_spiceValidator );
m_dcStart2->SetValidator( m_posFloatValidator ); m_dcStart2->SetValidator( m_spiceValidator );
m_dcStop2->SetValidator( m_posFloatValidator ); m_dcStop2->SetValidator( m_spiceValidator );
m_dcIncr2->SetValidator( m_posFloatValidator ); m_dcIncr2->SetValidator( m_spiceValidator );
m_noisePointsNumber->SetValidator( m_posIntValidator ); m_noisePointsNumber->SetValidator( m_posIntValidator );
m_noiseFreqStart->SetValidator( m_posFloatValidator ); m_noiseFreqStart->SetValidator( m_spiceValidator );
m_noiseFreqStop->SetValidator( m_posFloatValidator ); m_noiseFreqStop->SetValidator( m_spiceValidator );
m_transStep->SetValidator( m_posFloatValidator ); m_transStep->SetValidator( m_spiceValidator );
m_transFinal->SetValidator( m_posFloatValidator ); m_transFinal->SetValidator( m_spiceValidator );
m_transInitial->SetValidator( m_posFloatValidator ); m_transInitial->SetValidator( m_spiceEmptyValidator );
m_sdbSizerOK->SetDefault(); m_sdbSizerOK->SetDefault();
updateNetlistOpts(); updateNetlistOpts();
@ -70,12 +69,14 @@ bool DIALOG_SIM_SETTINGS::TransferDataFromWindow()
// AC analysis // AC analysis
if( page == m_pgAC ) if( page == m_pgAC )
{ {
if( m_acPointsNumber->IsEmpty() || m_acFreqStart->IsEmpty() || m_acFreqStop->IsEmpty() ) if( !m_pgAC->Validate() )
return false; return false;
m_simCommand = wxString::Format( ".ac %s %s %s %s", m_simCommand = wxString::Format( ".ac %s %s %s %s",
scaleToString( m_acScale->GetSelection() ), scaleToString( m_acScale->GetSelection() ),
m_acPointsNumber->GetValue(), m_acFreqStart->GetValue(), m_acFreqStop->GetValue() ); m_acPointsNumber->GetValue(),
SPICE_VALUE( m_acFreqStart->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_acFreqStop->GetValue() ).ToSpiceString() );
} }
@ -84,30 +85,67 @@ bool DIALOG_SIM_SETTINGS::TransferDataFromWindow()
{ {
// At least one source has to be enabled // At least one source has to be enabled
if( !m_dcEnable1->IsChecked() && !m_dcEnable1->IsChecked() ) if( !m_dcEnable1->IsChecked() && !m_dcEnable1->IsChecked() )
{
DisplayError( this, wxT( "You need to enable at least one source" ) );
return false; return false;
}
wxString simCmd = wxString( ".dc " ); wxString simCmd = wxString( ".dc " );
if( m_dcEnable1->IsChecked() ) if( m_dcEnable1->IsChecked() )
{ {
if( m_dcSource1->GetValue().IsEmpty() || m_dcStart1->IsEmpty() || if( m_dcSource1->GetValue().IsEmpty() )
m_dcStop1->IsEmpty() || m_dcIncr1->IsEmpty() ) {
DisplayError( this, wxT( "You need to select DC source (sweep 1)" ) );
return false;
}
/// @todo for some reason it does not trigger the assigned SPICE_VALIDATOR,
// hence try..catch below
if( !m_dcStart1->Validate() || !m_dcStop1->Validate() || !m_dcIncr1->Validate() )
return false; return false;
simCmd += wxString::Format( "v%s %s %s %s", try
m_dcSource1->GetValue(), m_dcStart1->GetValue(), {
m_dcStop1->GetValue(), m_dcIncr1->GetValue() ); simCmd += wxString::Format( "v%s %s %s %s",
m_dcSource1->GetValue(),
SPICE_VALUE( m_dcStart1->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_dcStop1->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_dcIncr1->GetValue() ).ToSpiceString() );
}
catch( std::exception& e )
{
DisplayError( this, e.what() );
return false;
}
} }
if( m_dcEnable2->IsChecked() ) if( m_dcEnable2->IsChecked() )
{ {
if( m_dcSource2->GetValue().IsEmpty() || m_dcStart2->IsEmpty() || if( m_dcSource2->GetValue().IsEmpty() )
m_dcStop2->IsEmpty() || m_dcIncr2->IsEmpty() ) {
DisplayError( this, wxT( "You need to select DC source (sweep 2)" ) );
return false;
}
/// @todo for some reason it does not trigger the assigned SPICE_VALIDATOR,
// hence try..catch below
if( !m_dcStart2->Validate() || !m_dcStop2->Validate() || !m_dcIncr2->Validate() )
return false; return false;
simCmd += wxString::Format( "v%s %s %s %s", try
m_dcSource2->GetValue(), m_dcStart2->GetValue(), {
m_dcStop2->GetValue(), m_dcIncr2->GetValue() ); simCmd += wxString::Format( "v%s %s %s %s",
m_dcSource2->GetValue(),
SPICE_VALUE( m_dcStart2->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_dcStop2->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_dcIncr2->GetValue() ).ToSpiceString() );
}
catch( std::exception& e )
{
DisplayError( this, e.what() );
return false;
}
} }
m_simCommand = simCmd; m_simCommand = simCmd;
@ -130,7 +168,9 @@ bool DIALOG_SIM_SETTINGS::TransferDataFromWindow()
m_simCommand = wxString::Format( ".noise v(%d%s) v%s %s %s %s %s", m_simCommand = wxString::Format( ".noise v(%d%s) v%s %s %s %s %s",
netMap.at( m_noiseMeas->GetValue() ), ref, netMap.at( m_noiseMeas->GetValue() ), ref,
m_noiseSrc->GetValue(), scaleToString( m_noiseScale->GetSelection() ), m_noiseSrc->GetValue(), scaleToString( m_noiseScale->GetSelection() ),
m_noisePointsNumber->GetValue(), m_noiseFreqStart->GetValue(), m_noiseFreqStop->GetValue() ); m_noisePointsNumber->GetValue(),
SPICE_VALUE( m_noiseFreqStart->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_noiseFreqStop->GetValue() ).ToSpiceString() );
} }
@ -144,11 +184,16 @@ bool DIALOG_SIM_SETTINGS::TransferDataFromWindow()
// Transient analysis // Transient analysis
else if( page == m_pgTransient ) else if( page == m_pgTransient )
{ {
if( m_transStep->IsEmpty() || m_transFinal->IsEmpty() ) if( !m_pgTransient->Validate() )
return false; return false;
wxString initial =
m_transInitial->IsEmpty() ? "" : SPICE_VALUE( m_transInitial->GetValue() ).ToSpiceString();
m_simCommand = wxString::Format( ".tran %s %s %s", m_simCommand = wxString::Format( ".tran %s %s %s",
m_transStep->GetValue(), m_transFinal->GetValue(), m_transInitial->GetValue() ); SPICE_VALUE( m_transStep->GetValue() ).ToSpiceString(),
SPICE_VALUE( m_transFinal->GetValue() ).ToSpiceString(),
initial );
} }

View File

@ -26,6 +26,8 @@
#define DIALOG_SIM_SETTINGS_BASE_H #define DIALOG_SIM_SETTINGS_BASE_H
#include "dialog_sim_settings_base.h" #include "dialog_sim_settings_base.h"
#include <sim/spice_value.h>
#include <wx/valnum.h> #include <wx/valnum.h>
class NETLIST_EXPORTER_PSPICE_SIM; class NETLIST_EXPORTER_PSPICE_SIM;
@ -94,7 +96,8 @@ private:
int m_netlistOpts; int m_netlistOpts;
NETLIST_EXPORTER_PSPICE_SIM* m_exporter; NETLIST_EXPORTER_PSPICE_SIM* m_exporter;
wxFloatingPointValidator<double> m_posFloatValidator; SPICE_VALIDATOR m_spiceValidator;
SPICE_VALIDATOR m_spiceEmptyValidator;
wxIntegerValidator<int> m_posIntValidator; wxIntegerValidator<int> m_posIntValidator;
}; };

View File

@ -27,10 +27,16 @@
#include <stdexcept> #include <stdexcept>
#include <cmath> #include <cmath>
#include <wx/textentry.h>
#include <confirm.h>
SPICE_VALUE::SPICE_VALUE( const wxString& aString ) SPICE_VALUE::SPICE_VALUE( const wxString& aString )
{ {
char buf[8] = { 0, }; char buf[8] = { 0, };
if( aString.IsEmpty() )
throw std::invalid_argument( "Spice value cannot be empty" );
if( sscanf( (const char*) aString.c_str(), "%lf%7s", &m_base, buf ) == 0 ) if( sscanf( (const char*) aString.c_str(), "%lf%7s", &m_base, buf ) == 0 )
throw std::invalid_argument( "Invalid Spice value string" ); throw std::invalid_argument( "Invalid Spice value string" );
@ -221,3 +227,36 @@ void SPICE_VALUE::stripZeros( wxString& aString )
if( aString.EndsWith( '.' ) || aString.EndsWith( ',' ) ) if( aString.EndsWith( '.' ) || aString.EndsWith( ',' ) )
aString.RemoveLast(); aString.RemoveLast();
} }
bool SPICE_VALIDATOR::Validate( wxWindow* aParent )
{
wxTextEntry* const text = GetTextEntry();
if( !text )
return false;
if( text->IsEmpty() )
{
if( m_emptyAllowed )
return true;
DisplayError( aParent, wxString::Format( wxT( "Fill required fields" ) ) );
return false;
}
try
{
// If SPICE_VALUE can be constructed, then it is a valid Spice value
SPICE_VALUE val( text->GetValue() );
}
catch( ... )
{
DisplayError( aParent,
wxString::Format( wxT( "'%s' is not a valid Spice value" ), text->GetValue() ) );
return false;
}
return true;
}

View File

@ -26,6 +26,7 @@
#define SPICE_VALUE_H #define SPICE_VALUE_H
#include <wx/string.h> #include <wx/string.h>
#include <wx/valtext.h>
///> Helper class to handle Spice way of expressing values (e.g. 10.5 Meg) ///> Helper class to handle Spice way of expressing values (e.g. 10.5 Meg)
class SPICE_VALUE class SPICE_VALUE
@ -120,4 +121,25 @@ private:
static void stripZeros( wxString& aString ); static void stripZeros( wxString& aString );
}; };
///> Helper class to recognize Spice formatted values
class SPICE_VALIDATOR : public wxTextValidator
{
public:
SPICE_VALIDATOR( bool aEmptyAllowed = false )
: m_emptyAllowed( aEmptyAllowed )
{
}
wxObject* Clone() const override
{
return new SPICE_VALIDATOR( *this );
}
bool Validate( wxWindow* aParent ) override;
private:
bool m_emptyAllowed;
};
#endif /* SPICE_VALUE_H */ #endif /* SPICE_VALUE_H */