From 667bafc8bbcaa85a2803ca6b36bc97e1fd6eed3b Mon Sep 17 00:00:00 2001
From: Jon Evans <jon@craftyjon.com>
Date: Sat, 15 Feb 2025 12:52:43 -0500
Subject: [PATCH] Restore accidentally-removed schematic parity code from CLI

This will need a better fix later

Fixes https://gitlab.com/kicad/code/kicad/-/issues/19929
---
 eeschema/eeschema.cpp          | 68 ++++++++++++++++++++++++++++++++++
 include/kiface_ids.h           |  1 +
 pcbnew/pcbnew_jobs_handler.cpp | 32 +++++++++++++++-
 3 files changed, 100 insertions(+), 1 deletion(-)

diff --git a/eeschema/eeschema.cpp b/eeschema/eeschema.cpp
index 8602a948e4..67c7dc2fec 100644
--- a/eeschema/eeschema.cpp
+++ b/eeschema/eeschema.cpp
@@ -76,6 +76,68 @@ SCH_SHEET*  g_RootSheet = nullptr;
 namespace SCH {
 
 
+// TODO: This should move out of this file
+static std::unique_ptr<SCHEMATIC> readSchematicFromFile( const std::string& aFilename )
+{
+    SCH_IO* pi = SCH_IO_MGR::FindPlugin( SCH_IO_MGR::SCH_KICAD );
+    std::unique_ptr<SCHEMATIC> schematic = std::make_unique<SCHEMATIC>( nullptr );
+
+    SETTINGS_MANAGER& manager = Pgm().GetSettingsManager();
+
+    manager.LoadProject( "" );
+    schematic->Reset();
+    schematic->SetProject( &manager.Prj() );
+    schematic->SetRoot( pi->LoadSchematicFile( aFilename, schematic.get() ) );
+    schematic->CurrentSheet().push_back( &schematic->Root() );
+
+    SCH_SCREENS screens( schematic->Root() );
+
+    for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )
+        screen->UpdateLocalLibSymbolLinks();
+
+    SCH_SHEET_LIST sheets = schematic->Hierarchy();
+
+    // Restore all of the loaded symbol instances from the root sheet screen.
+    sheets.UpdateSymbolInstanceData( schematic->RootScreen()->GetSymbolInstances() );
+
+    if( schematic->RootScreen()->GetFileFormatVersionAtLoad() < 20230221 )
+    {
+        for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )
+            screen->FixLegacyPowerSymbolMismatches();
+    }
+
+    for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )
+        screen->MigrateSimModels();
+
+    sheets.AnnotatePowerSymbols();
+
+    // NOTE: This is required for multi-unit symbols to be correct
+    for( SCH_SHEET_PATH& sheet : sheets )
+        sheet.UpdateAllScreenReferences();
+
+    // NOTE: SchematicCleanUp is not called; QA schematics must already be clean or else
+    // SchematicCleanUp must be freed from its UI dependencies.
+
+    schematic->ConnectionGraph()->Recalculate( sheets, true );
+
+    return schematic;
+}
+
+
+// TODO: This should move out of this file
+bool generateSchematicNetlist( const wxString& aFilename, std::string& aNetlist )
+{
+    std::unique_ptr<SCHEMATIC> schematic = readSchematicFromFile( aFilename.ToStdString() );
+    NETLIST_EXPORTER_KICAD exporter( schematic.get() );
+    STRING_FORMATTER formatter;
+
+    exporter.Format( &formatter, GNL_ALL | GNL_OPT_KICAD );
+    aNetlist = formatter.GetString();
+
+    return true;
+}
+
+
 static struct IFACE : public KIFACE_BASE, public UNITS_PROVIDER
 {
     // Of course all are virtual overloads, implementations of the KIFACE.
@@ -265,6 +327,12 @@ static struct IFACE : public KIFACE_BASE, public UNITS_PROVIDER
      */
     void* IfaceOrAddress( int aDataId ) override
     {
+        switch( aDataId )
+        {
+            case KIFACE_NETLIST_SCHEMATIC:
+                return (void*) generateSchematicNetlist;
+        }
+
         return nullptr;
     }
 
diff --git a/include/kiface_ids.h b/include/kiface_ids.h
index 3361433e15..754ffd4e45 100644
--- a/include/kiface_ids.h
+++ b/include/kiface_ids.h
@@ -53,6 +53,7 @@ enum KIFACE_ADDR_ID : int
     KIFACE_GLOBAL_FOOTPRINT_TABLE,
 
     KIFACE_LOAD_SCHEMATIC,
+    KIFACE_NETLIST_SCHEMATIC,
     KIFACE_SCRIPTING_LEGACY,
     KIFACE_SCRIPTING,
 
diff --git a/pcbnew/pcbnew_jobs_handler.cpp b/pcbnew/pcbnew_jobs_handler.cpp
index 08cf31467b..d0d0a4d391 100644
--- a/pcbnew/pcbnew_jobs_handler.cpp
+++ b/pcbnew/pcbnew_jobs_handler.cpp
@@ -1820,7 +1820,37 @@ int PCBNEW_JOBS_HANDLER::JobExportDrc( JOB* aJob )
         wxString annotateMsg = _( "Schematic parity tests require a fully annotated schematic." );
         netlist_str = annotateMsg;
 
-        m_kiway->ExpressMail( FRAME_SCH, MAIL_SCH_GET_NETLIST, netlist_str );
+        // The KIFACE_NETLIST_SCHEMATIC function has some broken-ness that the schematic
+        // frame's version does not, but it is the only one that works in CLI, so we use it
+        // if we don't have the sch frame open.
+        // TODO: clean this up, see https://gitlab.com/kicad/code/kicad/-/issues/19929
+        if( m_kiway->Player( FRAME_SCH, false ) )
+        {
+            m_kiway->ExpressMail( FRAME_SCH, MAIL_SCH_GET_NETLIST, netlist_str );
+        }
+        else
+        {
+            wxFileName schematicPath( drcJob->m_filename );
+            schematicPath.SetExt( FILEEXT::KiCadSchematicFileExtension );
+
+            if( !schematicPath.Exists() )
+                schematicPath.SetExt( FILEEXT::LegacySchematicFileExtension );
+
+            if( !schematicPath.Exists() )
+            {
+                m_reporter->Report( _( "Failed to fetch schematic netlist for parity tests.\n" ),
+                                    RPT_SEVERITY_ERROR );
+                checkParity = false;
+            }
+            else
+            {
+                typedef bool ( *NETLIST_FN_PTR )( const wxString&, std::string& );
+                KIFACE* eeschema = m_kiway->KiFACE( KIWAY::FACE_SCH );
+                NETLIST_FN_PTR netlister =
+                        (NETLIST_FN_PTR) eeschema->IfaceOrAddress( KIFACE_NETLIST_SCHEMATIC );
+                ( *netlister )( schematicPath.GetFullPath(), netlist_str );
+            }
+        }
 
         if( netlist_str == annotateMsg )
         {