KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
kigit_pcb_merge.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.TXT for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 3
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/gpl-3.0.html
19 * or you may search the http://www.gnu.org website for the version 3 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include "kigit_pcb_merge.h"
28
29#include <board.h>
32#include <richio.h>
33#include <trace_helpers.h>
34
35#include <wx/log.h>
36
37void KIGIT_PCB_MERGE::findSetDifferences( const BOARD_ITEM_SET& aAncestorSet, const BOARD_ITEM_SET& aOtherSet,
38 std::vector<BOARD_ITEM*>& aAdded, std::vector<BOARD_ITEM*>& aRemoved,
39 std::vector<BOARD_ITEM*>& aChanged )
40{
41 auto it1 = aAncestorSet.begin();
42 auto it2 = aOtherSet.begin();
43
44 while( it1 != aAncestorSet.end() && it2 != aOtherSet.end() )
45 {
46 BOARD_ITEM* item1 = *it1;
47 BOARD_ITEM* item2 = *it2;
48
49 if( item1->m_Uuid < item2->m_Uuid )
50 {
51 aRemoved.push_back( item1 );
52 ++it1;
53 }
54 else if( item1->m_Uuid > item2->m_Uuid )
55 {
56 aAdded.push_back( item2 );
57 ++it2;
58 }
59 else
60 {
61 if( !( *item1 == *item2 ) )
62 aChanged.push_back( item1 );
63
64 ++it1;
65 ++it2;
66 }
67 }
68}
69
70
72{
74
75 const auto ancestor_set = aAncestor->GetItemSet();
76 const auto other_set = aOther->GetItemSet();
77
78 findSetDifferences( ancestor_set, other_set, differences.m_added, differences.m_removed, differences.m_changed );
79
80 return differences;
81}
82
83
85{
86 auto repo = const_cast<git_repository*>( git_merge_driver_source_repo( m_mergeDriver ) );
87 const git_index_entry* ancestor = git_merge_driver_source_ancestor( m_mergeDriver );
88 const git_index_entry* ours = git_merge_driver_source_ours( m_mergeDriver );
89 const git_index_entry* theirs = git_merge_driver_source_theirs( m_mergeDriver );
90
91 // Read ancestor index entry into a buffer
92
93 git_blob* ancestor_blob;
94 git_blob* ours_blob;
95 git_blob* theirs_blob;
96
97 if( git_blob_lookup( &ancestor_blob, repo, &ancestor->id ) != 0 )
98 {
99 wxLogTrace( traceGit, "Could not find ancestor blob: %s", KIGIT_COMMON::GetLastGitError() );
100 return GIT_ENOTFOUND;
101 }
102
103 KIGIT::GitBlobPtr ancestor_blob_ptr( ancestor_blob );
104
105 if( git_blob_lookup( &ours_blob, repo, &ours->id ) != 0 )
106 {
107 wxLogTrace( traceGit, "Could not find ours blob: %s", KIGIT_COMMON::GetLastGitError() );
108 return GIT_ENOTFOUND;
109 }
110
111 KIGIT::GitBlobPtr ours_blob_ptr( ours_blob );
112
113 if( git_blob_lookup( &theirs_blob, repo, &theirs->id ) != 0 )
114 {
115 wxLogTrace( traceGit, "Could not find theirs blob: %s", KIGIT_COMMON::GetLastGitError() );
116 return GIT_ENOTFOUND;
117 }
118
119 KIGIT::GitBlobPtr theirs_blob_ptr( theirs_blob );
120
121 // Get the raw data from the blobs
122 BLOB_READER ancestor_reader( ancestor_blob );
123 PCB_IO_KICAD_SEXPR_PARSER ancestor_parser( &ancestor_reader, nullptr, nullptr );
124 BLOB_READER ours_reader( ours_blob );
125 PCB_IO_KICAD_SEXPR_PARSER ours_parser( &ours_reader, nullptr, nullptr );
126 BLOB_READER theirs_reader( theirs_blob );
127 PCB_IO_KICAD_SEXPR_PARSER theirs_parser( &theirs_reader, nullptr, nullptr );
128
129 std::unique_ptr<BOARD> ancestor_board;
130 std::unique_ptr<BOARD> ours_board;
131 std::unique_ptr<BOARD> theirs_board;
132
133 try
134 {
135 ancestor_board.reset( static_cast<BOARD*>( ancestor_parser.Parse() ) );
136 ours_board.reset( static_cast<BOARD*>( ours_parser.Parse() ) );
137 theirs_board.reset( static_cast<BOARD*>( theirs_parser.Parse() ) );
138 }
139 catch(const IO_ERROR& e)
140 {
141 wxLogTrace( traceGit, "Could not parse board: %s", e.What() );
142 return GIT_EUSER;
143 }
144 catch( ... )
145 {
146 wxLogTrace( traceGit, "Could not parse board: unknown error" );
147 return GIT_EUSER;
148 }
149
150 // Parse the differences between the ancestor and ours
151 KIGIT_PCB_MERGE_DIFFERENCES ancestor_ours_differences = compareBoards( ancestor_board.get(), ours_board.get() );
152 KIGIT_PCB_MERGE_DIFFERENCES ancestor_theirs_differences = compareBoards( ancestor_board.get(), theirs_board.get() );
153
154 // Find the items that we modified and they deleted
155 std::set_intersection( ancestor_ours_differences.m_changed.begin(), ancestor_ours_differences.m_changed.end(),
156 ancestor_theirs_differences.m_removed.begin(), ancestor_theirs_differences.m_removed.end(),
157 std::inserter( we_modified_they_deleted, we_modified_they_deleted.begin() ) );
158
159 // Find the items that they modified and we deleted
160 std::set_intersection( ancestor_theirs_differences.m_changed.begin(), ancestor_theirs_differences.m_changed.end(),
161 ancestor_ours_differences.m_removed.begin(), ancestor_ours_differences.m_removed.end(),
162 std::inserter( they_modified_we_deleted, they_modified_we_deleted.begin() ) );
163
164 // Find the items that both we and they modified
165 std::set_intersection( ancestor_ours_differences.m_changed.begin(), ancestor_ours_differences.m_changed.end(),
166 ancestor_theirs_differences.m_changed.begin(), ancestor_theirs_differences.m_changed.end(),
167 std::inserter( both_modified, both_modified.begin() ) );
168
169 return 0;
170}
171
172
173std::unique_ptr<BOARD> readBoard( wxString& aFilename )
174{
175 // Take input from stdin
176 FILE_LINE_READER reader( aFilename );
177
178 PCB_IO_KICAD_SEXPR_PARSER parser( &reader, nullptr, nullptr );
179 std::unique_ptr<BOARD> board;
180
181 try
182 {
183 board.reset( static_cast<BOARD*>( parser.Parse() ) );
184 }
185 catch( const IO_ERROR& )
186 {
187 }
188
189 return board;
190}
std::set< BOARD_ITEM *, CompareByUuid > BOARD_ITEM_SET
Set of BOARD_ITEMs ordered by UUID.
Definition: board.h:281
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
const BOARD_ITEM_SET GetItemSet()
Definition: board.cpp:3068
const KIID m_Uuid
Definition: eda_item.h:498
A LINE_READER that reads from an open file.
Definition: richio.h:185
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
virtual const wxString What() const
A composite of Problem() and Where()
Definition: exceptions.cpp:30
static wxString GetLastGitError()
std::set< BOARD_ITEM * > both_modified
void 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)
git_merge_driver_source * m_mergeDriver
KIGIT_PCB_MERGE_DIFFERENCES compareBoards(BOARD *aAncestor, BOARD *aOther)
std::set< BOARD_ITEM * > they_modified_we_deleted
std::set< BOARD_ITEM * > we_modified_they_deleted
Read a Pcbnew s-expression formatted LINE_READER object and returns the appropriate BOARD_ITEM object...
const wxChar *const traceGit
Flag to enable Git debugging output.
std::unique_ptr< BOARD > readBoard(wxString &aFilename)
std::unique_ptr< git_blob, decltype([](git_blob *aBlob) { git_blob_free(aBlob) GitBlobPtr
A unique pointer for git_blob objects with automatic cleanup.
Pcbnew s-expression file format parser definition.
std::vector< BOARD_ITEM * > m_removed
std::vector< BOARD_ITEM * > m_changed
std::vector< BOARD_ITEM * > m_added
wxLogTrace helper definitions.