2023-09-14 21:39:42 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2023 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:
|
|
|
|
* http://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 "kigit_pcb_merge.h"
|
|
|
|
|
2023-12-24 01:21:58 +00:00
|
|
|
#include <pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.h>
|
|
|
|
#include <pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.h>
|
2023-09-14 21:39:42 +00:00
|
|
|
#include <richio.h>
|
|
|
|
|
|
|
|
#include <board.h>
|
|
|
|
|
|
|
|
#include <git/kicad_git_blob_reader.h>
|
|
|
|
|
|
|
|
void KIGIT_PCB_MERGE::findSetDifferences( const BOARD_ITEM_SET& aAncestorSet, const BOARD_ITEM_SET& aOtherSet,
|
|
|
|
std::vector<BOARD_ITEM*>& aAdded, std::vector<BOARD_ITEM*>& aRemoved,
|
|
|
|
std::vector<BOARD_ITEM*>& aChanged )
|
|
|
|
{
|
|
|
|
auto it1 = aAncestorSet.begin();
|
|
|
|
auto it2 = aOtherSet.begin();
|
|
|
|
|
|
|
|
while( it1 != aAncestorSet.end() && it2 != aOtherSet.end() )
|
|
|
|
{
|
|
|
|
BOARD_ITEM* item1 = *it1;
|
|
|
|
BOARD_ITEM* item2 = *it2;
|
|
|
|
|
|
|
|
if( item1->m_Uuid < item2->m_Uuid )
|
|
|
|
{
|
|
|
|
aRemoved.push_back( item1 );
|
|
|
|
++it1;
|
|
|
|
}
|
|
|
|
else if( item1->m_Uuid > item2->m_Uuid )
|
|
|
|
{
|
|
|
|
aAdded.push_back( item2 );
|
|
|
|
++it2;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( !( *item1 == *item2 ) )
|
|
|
|
aChanged.push_back( item1 );
|
|
|
|
|
|
|
|
++it1;
|
|
|
|
++it2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-12-11 21:21:11 +00:00
|
|
|
|
|
|
|
|
2023-09-14 21:39:42 +00:00
|
|
|
KIGIT_PCB_MERGE_DIFFERENCES KIGIT_PCB_MERGE::compareBoards( BOARD* aAncestor, BOARD* aOther )
|
|
|
|
{
|
|
|
|
KIGIT_PCB_MERGE_DIFFERENCES differences;
|
|
|
|
|
|
|
|
const auto ancestor_set = aAncestor->GetItemSet();
|
|
|
|
const auto other_set = aOther->GetItemSet();
|
|
|
|
|
|
|
|
findSetDifferences( ancestor_set, other_set, differences.m_added, differences.m_removed, differences.m_changed );
|
|
|
|
|
|
|
|
return differences;
|
|
|
|
}
|
|
|
|
|
2023-12-11 21:21:11 +00:00
|
|
|
|
2023-09-14 21:39:42 +00:00
|
|
|
int KIGIT_PCB_MERGE::Merge()
|
|
|
|
{
|
2023-12-11 21:21:11 +00:00
|
|
|
auto repo = const_cast<git_repository*>( git_merge_driver_source_repo( m_mergeDriver ) );
|
2023-09-14 21:39:42 +00:00
|
|
|
const git_index_entry* ancestor = git_merge_driver_source_ancestor( m_mergeDriver );
|
|
|
|
const git_index_entry* ours = git_merge_driver_source_ours( m_mergeDriver );
|
|
|
|
const git_index_entry* theirs = git_merge_driver_source_theirs( m_mergeDriver );
|
|
|
|
|
|
|
|
// Read ancestor index entry into a buffer
|
|
|
|
|
|
|
|
git_blob* ancestor_blob;
|
|
|
|
git_blob* ours_blob;
|
|
|
|
git_blob* theirs_blob;
|
|
|
|
|
|
|
|
if( git_blob_lookup( &ancestor_blob, repo, &ancestor->id ) != 0 )
|
|
|
|
{
|
|
|
|
return GIT_ENOTFOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( git_blob_lookup( &ours_blob, repo, &ours->id ) != 0 )
|
|
|
|
{
|
|
|
|
git_blob_free( ancestor_blob );
|
|
|
|
return GIT_ENOTFOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( git_blob_lookup( &theirs_blob, repo, &theirs->id ) != 0 )
|
|
|
|
{
|
|
|
|
git_blob_free( ancestor_blob );
|
|
|
|
git_blob_free( ours_blob );
|
|
|
|
return GIT_ENOTFOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the raw data from the blobs
|
|
|
|
BLOB_READER ancestor_reader( ancestor_blob );
|
2023-12-24 01:21:58 +00:00
|
|
|
PCB_IO_KICAD_SEXPR_PARSER ancestor_parser( &ancestor_reader, nullptr, nullptr );
|
2023-09-14 21:39:42 +00:00
|
|
|
BLOB_READER ours_reader( ours_blob );
|
2023-12-24 01:21:58 +00:00
|
|
|
PCB_IO_KICAD_SEXPR_PARSER ours_parser( &ours_reader, nullptr, nullptr );
|
2023-09-14 21:39:42 +00:00
|
|
|
BLOB_READER theirs_reader( theirs_blob );
|
2023-12-24 01:21:58 +00:00
|
|
|
PCB_IO_KICAD_SEXPR_PARSER theirs_parser( &theirs_reader, nullptr, nullptr );
|
2023-09-14 21:39:42 +00:00
|
|
|
|
|
|
|
std::unique_ptr<BOARD> ancestor_board;
|
|
|
|
std::unique_ptr<BOARD> ours_board;
|
|
|
|
std::unique_ptr<BOARD> theirs_board;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
ancestor_board.reset( static_cast<BOARD*>( ancestor_parser.Parse() ) );
|
|
|
|
ours_board.reset( static_cast<BOARD*>( ours_parser.Parse() ) );
|
|
|
|
theirs_board.reset( static_cast<BOARD*>( theirs_parser.Parse() ) );
|
|
|
|
}
|
|
|
|
catch(const IO_ERROR&)
|
|
|
|
{
|
|
|
|
git_blob_free( ancestor_blob );
|
|
|
|
git_blob_free( ours_blob );
|
|
|
|
git_blob_free( theirs_blob );
|
|
|
|
return GIT_EUSER;
|
|
|
|
}
|
|
|
|
|
|
|
|
git_blob_free( ancestor_blob );
|
|
|
|
git_blob_free( ours_blob );
|
|
|
|
git_blob_free( theirs_blob );
|
|
|
|
|
|
|
|
// Parse the differences between the ancestor and ours
|
|
|
|
KIGIT_PCB_MERGE_DIFFERENCES ancestor_ours_differences = compareBoards( ancestor_board.get(), ours_board.get() );
|
|
|
|
KIGIT_PCB_MERGE_DIFFERENCES ancestor_theirs_differences = compareBoards( ancestor_board.get(), theirs_board.get() );
|
|
|
|
|
|
|
|
// Find the items that we modified and they deleted
|
|
|
|
std::set_intersection( ancestor_ours_differences.m_changed.begin(), ancestor_ours_differences.m_changed.end(),
|
|
|
|
ancestor_theirs_differences.m_removed.begin(), ancestor_theirs_differences.m_removed.end(),
|
|
|
|
std::inserter( we_modified_they_deleted, we_modified_they_deleted.begin() ) );
|
|
|
|
|
|
|
|
// Find the items that they modified and we deleted
|
|
|
|
std::set_intersection( ancestor_theirs_differences.m_changed.begin(), ancestor_theirs_differences.m_changed.end(),
|
|
|
|
ancestor_ours_differences.m_removed.begin(), ancestor_ours_differences.m_removed.end(),
|
|
|
|
std::inserter( they_modified_we_deleted, they_modified_we_deleted.begin() ) );
|
|
|
|
|
|
|
|
// Find the items that both we and they modified
|
|
|
|
std::set_intersection( ancestor_ours_differences.m_changed.begin(), ancestor_ours_differences.m_changed.end(),
|
|
|
|
ancestor_theirs_differences.m_changed.begin(), ancestor_theirs_differences.m_changed.end(),
|
|
|
|
std::inserter( both_modified, both_modified.begin() ) );
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<BOARD> readBoard( wxString& aFilename )
|
|
|
|
{
|
|
|
|
// Take input from stdin
|
|
|
|
FILE_LINE_READER reader( aFilename );
|
|
|
|
|
2023-12-24 01:21:58 +00:00
|
|
|
PCB_IO_KICAD_SEXPR_PARSER parser( &reader, nullptr, nullptr );
|
2023-09-14 21:39:42 +00:00
|
|
|
std::unique_ptr<BOARD> board;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
board.reset( static_cast<BOARD*>( parser.Parse() ) );
|
|
|
|
}
|
|
|
|
catch( const IO_ERROR& )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
return board;
|
|
|
|
}
|