kicad/eeschema/load_one_schematic_file.cpp

427 lines
12 KiB
C++
Raw Normal View History

/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2004 Jean-Pierre Charras, jaen-pierre.charras@gipsa-lab.inpg.com
* Copyright (C) 2008 Wayne Stambaugh <stambaughw@verizon.net>
* Copyright (C) 1992-2011 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 2
* 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/old-licenses/gpl-2.0.html
* or you may search the http://www.gnu.org website for the version 2 license,
* or you may write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*/
/**
* @file load_one_schematic_file.cpp
* @brief Code to load and save Eeschema files.
*/
#include "fctsys.h"
#include "confirm.h"
#include "kicad_string.h"
#include "wxEeschemaStruct.h"
#include "richio.h"
#include "general.h"
#include "protos.h"
#include "sch_bus_entry.h"
#include "sch_marker.h"
#include "sch_junction.h"
#include "sch_line.h"
#include "sch_no_connect.h"
#include "sch_component.h"
#include "sch_polyline.h"
#include "sch_text.h"
#include "sch_sheet.h"
#include "sch_bitmap.h"
bool ReadSchemaDescr( LINE_READER* aLine, wxString& aMsgDiag, SCH_SCREEN* Window );
2010-10-21 00:18:42 +00:00
static void LoadLayers( LINE_READER* aLine );
bool SCH_EDIT_FRAME::LoadOneEEFile( SCH_SCREEN* aScreen, const wxString& aFullFileName )
{
char Name1[256];
bool itemLoaded = false;
SCH_ITEM* Phead;
SCH_ITEM* Pnext;
SCH_ITEM* item;
2010-08-31 10:44:38 +00:00
wxString MsgDiag; // Error and log messages
2011-02-27 06:22:34 +00:00
char* line;
wxFileName fn;
if( aScreen == NULL )
return false;
2010-08-31 10:44:38 +00:00
if( aFullFileName.IsEmpty() )
return false;
fn = aFullFileName;
CheckForAutoSaveFile( fn, g_SchematicBackupFileExtension );
wxLogTrace( traceAutoSave, wxT( "Loading schematic file " ) + aFullFileName );
aScreen->SetCurItem( NULL );
aScreen->SetFileName( aFullFileName );
FILE* f;
2010-08-31 10:44:38 +00:00
if( ( f = wxFopen( aFullFileName, wxT( "rt" ) ) ) == NULL )
{
MsgDiag = _( "Failed to open " ) + aFullFileName;
DisplayError( this, MsgDiag );
return false;
}
2010-10-21 00:18:42 +00:00
// reader now owns the open FILE.
FILE_LINE_READER reader( f, aFullFileName );
2010-10-21 00:18:42 +00:00
MsgDiag = _( "Loading " ) + aScreen->GetFileName();
PrintMsg( MsgDiag );
2010-10-21 00:18:42 +00:00
if( !reader.ReadLine()
|| strncmp( (char*)reader + 9, SCHEMATIC_HEAD_STRING,
sizeof( SCHEMATIC_HEAD_STRING ) - 1 ) != 0 )
{
MsgDiag = aFullFileName + _( " is NOT an Eeschema file!" );
DisplayError( this, MsgDiag );
return false;
}
2011-02-27 06:22:34 +00:00
line = reader.Line();
// get the file version here.
char *strversion = line + 9 + sizeof( SCHEMATIC_HEAD_STRING );
2011-02-27 06:22:34 +00:00
// Skip blanks
while( *strversion && *strversion < '0' )
strversion++;
2011-02-27 06:22:34 +00:00
int version = atoi( strversion );
if( version > EESCHEMA_VERSION )
{
MsgDiag = aFullFileName + _( " was created by a more recent \
version of Eeschema and may not load correctly. Please consider updating!" );
DisplayInfoMessage( this, MsgDiag );
}
2008-04-15 19:38:19 +00:00
#if 0
// Compile it if the new version is unreadable by previous Eeschema versions
else if( version < EESCHEMA_VERSION )
{
MsgDiag = aFullFileName + _( " was created by an older version of \
Eeschema. It will be stored in the new file format when you save this file \
again." );
DisplayInfoMessage( this, MsgDiag );
}
2008-04-15 19:38:19 +00:00
#endif
2011-02-27 06:22:34 +00:00
if( !reader.ReadLine() || strncmp( reader, "LIBS:", 5 ) != 0 )
{
MsgDiag = aFullFileName + _( " is NOT an Eeschema file!" );
DisplayError( this, MsgDiag );
return false;
}
2010-10-21 00:18:42 +00:00
LoadLayers( &reader );
2010-10-21 00:18:42 +00:00
while( reader.ReadLine() )
{
2011-02-27 06:22:34 +00:00
line = reader.Line();
item = NULL;
2010-10-21 00:18:42 +00:00
char* sline = line;
2010-10-21 00:18:42 +00:00
while( (*sline != ' ' ) && *sline )
sline++;
2010-10-21 00:18:42 +00:00
switch( line[0] )
{
2010-08-31 10:44:38 +00:00
case '$': // identification block
2010-10-21 00:18:42 +00:00
if( line[1] == 'C' )
item = new SCH_COMPONENT();
2010-10-21 00:18:42 +00:00
else if( line[1] == 'S' )
item = new SCH_SHEET();
2010-10-21 00:18:42 +00:00
else if( line[1] == 'D' )
itemLoaded = ReadSchemaDescr( &reader, MsgDiag, aScreen );
else if( line[1] == 'B' )
item = new SCH_BITMAP();
break;
2010-08-31 10:44:38 +00:00
case 'L': // Its a library item.
item = new SCH_COMPONENT();
break;
2010-08-31 10:44:38 +00:00
case 'W': // Its a Segment (WIRE or BUS) item.
item = new SCH_LINE();
break;
2010-08-31 10:44:38 +00:00
case 'E': // Its a WIRE or BUS item.
item = new SCH_BUS_ENTRY();
break;
2010-08-31 10:44:38 +00:00
case 'P': // Its a polyline item.
item = new SCH_POLYLINE();
break;
case 'C': // It is a connection item.
item = new SCH_JUNCTION();
break;
2010-08-31 10:44:38 +00:00
case 'K': // It is a Marker item.
// Markers are no more read from file. they are only created on
// demand in schematic
break;
case 'N': // It is a NoConnect item.
item = new SCH_NO_CONNECT();
break;
2010-08-31 10:44:38 +00:00
case 'T': // It is a text item.
if( sscanf( sline, "%s", Name1 ) != 1 )
{
MsgDiag.Printf( wxT( "Eeschema file text load error at line %d" ),
reader.LineNumber() );
itemLoaded = false;
}
else if( Name1[0] == 'L' )
item = new SCH_LABEL();
else if( Name1[0] == 'G' && version > 1 )
item = new SCH_GLOBALLABEL();
else if( (Name1[0] == 'H') || (Name1[0] == 'G' && version == 1) )
item = new SCH_HIERLABEL();
else
item = new SCH_TEXT();
break;
default:
itemLoaded = false;
MsgDiag.Printf( wxT( "Eeschema file undefined object at line %d, aborted" ),
2010-10-21 00:18:42 +00:00
reader.LineNumber() );
MsgDiag << wxT( "\n" ) << FROM_UTF8( line );
}
if( item )
{
itemLoaded = item->Load( reader, MsgDiag );
if( !itemLoaded )
{
SAFE_DELETE( item );
}
else
{
item->SetNext( aScreen->GetDrawItems() );
aScreen->SetDrawItems( item );
}
}
if( !itemLoaded )
{
DisplayError( this, MsgDiag );
break;
}
}
/* GetDrawItems() was constructed in reverse order - reverse it back: */
Phead = NULL;
while( aScreen->GetDrawItems() )
{
Pnext = aScreen->GetDrawItems();
aScreen->SetDrawItems( aScreen->GetDrawItems()->Next() );
Pnext->SetNext( Phead );
Phead = Pnext;
}
aScreen->SetDrawItems( Phead );
#if 0 && defined (DEBUG)
aScreen->Show( 0, std::cout );
2007-09-20 21:06:49 +00:00
#endif
2008-03-20 01:50:21 +00:00
aScreen->TestDanglingEnds();
MsgDiag = _( "Done Loading " ) + aScreen->GetFileName();
2008-03-20 01:50:21 +00:00
PrintMsg( MsgDiag );
2010-08-31 10:44:38 +00:00
return true; // Although it may be that file is only partially loaded.
}
2010-10-21 00:18:42 +00:00
static void LoadLayers( LINE_READER* aLine )
{
2009-03-15 21:13:56 +00:00
int Number;
2009-03-15 21:13:56 +00:00
//int Mode,Color,Layer;
char Name[256];
2010-10-21 00:18:42 +00:00
aLine->ReadLine();
2009-03-15 21:13:56 +00:00
2010-10-21 00:18:42 +00:00
sscanf( *aLine, "%s %d %d", Name, &Number, &g_LayerDescr.CurrentLayer );
if( strcmp( Name, "EELAYER" ) !=0 )
{
/* error : init par default */
Number = MAX_LAYER;
}
2009-03-15 21:13:56 +00:00
if( Number <= 0 )
Number = MAX_LAYER;
if( Number > MAX_LAYER )
Number = MAX_LAYER;
g_LayerDescr.NumberOfLayers = Number;
2010-10-21 00:18:42 +00:00
while( aLine->ReadLine() )
{
2010-10-21 00:18:42 +00:00
if( strnicmp( *aLine, "EELAYER END", 11 ) == 0 )
break;
}
}
/// Get the length of a string constant, at compile time
#define SZ( x ) (sizeof(x)-1)
static const char delims[] = " \t\r\n";
/* Read the schematic header. */
bool ReadSchemaDescr( LINE_READER* aLine, wxString& aMsgDiag, SCH_SCREEN* aScreen )
{
char* line = aLine->Line();
2011-02-27 06:22:34 +00:00
char* pageType = strtok( line + SZ( "$Descr" ), delims );
char* width = strtok( NULL, delims );
char* height = strtok( NULL, delims );
char* orient = strtok( NULL, delims );
wxString pagename = FROM_UTF8( pageType );
PAGE_INFO pageInfo;
TITLE_BLOCK tb;
if( !pageInfo.SetType( pagename ) )
{
aMsgDiag.Printf( wxT( "Eeschema file dimension definition error \
line %d, \aAbort reading file.\n" ),
aLine->LineNumber() );
aMsgDiag << FROM_UTF8( line );
}
if( pagename == PAGE_INFO::Custom )
{
if( width && height )
{
int w = atoi( width );
int h = atoi( height );
pageInfo.SetWidthMils( w );
pageInfo.SetHeightMils( h );
}
}
// portrait only supported in non custom sizes
else if( orient && !strcmp( orient, "portrait" ) )
{
pageInfo.SetPortrait( true );
}
aScreen->SetPageSettings( pageInfo );
for(;;)
{
char buf[1024];
if( !aLine->ReadLine() )
return true;
2011-02-27 06:22:34 +00:00
line = aLine->Line();
if( strnicmp( line, "$End", 4 ) == 0 )
{
aScreen->SetTitleBlock( tb );
break;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Sheet", 2 ) == 0 )
sscanf( line + 5, " %d %d",
&aScreen->m_ScreenNumber, &aScreen->m_NumberOfScreen );
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Title", 2 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetTitle( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Date", 2 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetDate( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Rev", 2 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetRevision( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Comp", 4 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetCompany( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Comment1", 8 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetComment1( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Comment2", 8 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetComment2( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Comment3", 8 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetComment3( FROM_UTF8( buf ) );
continue;
}
2011-02-27 06:22:34 +00:00
if( strnicmp( line, "Comment4", 8 ) == 0 )
{
2011-02-27 06:22:34 +00:00
ReadDelimitedText( buf, line, 256 );
tb.SetComment4( FROM_UTF8( buf ) );
continue;
}
}
return true;
}