168 lines
4.5 KiB
C++
168 lines
4.5 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2019 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
|
|
*/
|
|
|
|
#include <qa_utils/utility_registry.h>
|
|
|
|
#include <cstdio>
|
|
#include <string>
|
|
|
|
#include <common.h>
|
|
#include <profile.h>
|
|
|
|
#include <wx/cmdline.h>
|
|
|
|
#include <board_item.h>
|
|
#include <plugins/kicad/pcb_plugin.h>
|
|
#include <plugins/kicad/pcb_parser.h>
|
|
#include <richio.h>
|
|
|
|
#include <wx/cmdline.h>
|
|
|
|
#include <qa_utils/stdstream_line_reader.h>
|
|
#include <qa_utils/utility_registry.h>
|
|
|
|
|
|
using PARSE_DURATION = std::chrono::microseconds;
|
|
|
|
|
|
/**
|
|
* Parse a PCB or footprint file from the given input stream
|
|
*
|
|
* @param aStream the input stream to read from
|
|
* @return success, duration (in us)
|
|
*/
|
|
bool parse( std::istream& aStream, bool aVerbose )
|
|
{
|
|
// Take input from stdin
|
|
STDISTREAM_LINE_READER reader;
|
|
reader.SetStream( aStream );
|
|
|
|
PCB_PARSER parser;
|
|
|
|
parser.SetLineReader( &reader );
|
|
|
|
BOARD_ITEM* board = nullptr;
|
|
|
|
PARSE_DURATION duration{};
|
|
|
|
try
|
|
{
|
|
PROF_COUNTER timer;
|
|
board = parser.Parse();
|
|
|
|
duration = timer.SinceStart<PARSE_DURATION>();
|
|
}
|
|
catch( const IO_ERROR& )
|
|
{
|
|
}
|
|
|
|
if( aVerbose )
|
|
{
|
|
std::cout << "Took: " << duration.count() << "us" << std::endl;
|
|
}
|
|
|
|
return board != nullptr;
|
|
}
|
|
|
|
|
|
static const wxCmdLineEntryDesc g_cmdLineDesc[] = {
|
|
{ wxCMD_LINE_SWITCH, "h", "help", _( "displays help on the command line parameters" ).mb_str(),
|
|
wxCMD_LINE_VAL_NONE, wxCMD_LINE_OPTION_HELP },
|
|
{ wxCMD_LINE_SWITCH, "v", "verbose", _( "print parsing information" ).mb_str() },
|
|
{ wxCMD_LINE_PARAM, nullptr, nullptr, _( "input file" ).mb_str(), wxCMD_LINE_VAL_STRING,
|
|
wxCMD_LINE_PARAM_OPTIONAL | wxCMD_LINE_PARAM_MULTIPLE },
|
|
{ wxCMD_LINE_NONE }
|
|
};
|
|
|
|
|
|
enum PARSER_RET_CODES
|
|
{
|
|
PARSE_FAILED = KI_TEST::RET_CODES::TOOL_SPECIFIC,
|
|
};
|
|
|
|
|
|
int pcb_parser_main_func( int argc, char** argv )
|
|
{
|
|
#ifdef __AFL_COMPILER
|
|
__AFL_INIT();
|
|
#endif
|
|
|
|
wxMessageOutput::Set( new wxMessageOutputStderr );
|
|
wxCmdLineParser cl_parser( argc, argv );
|
|
cl_parser.SetDesc( g_cmdLineDesc );
|
|
cl_parser.AddUsageText(
|
|
_( "This program parses PCB files, either from the "
|
|
"stdin stream or from the given filenames. This can be used either for "
|
|
"standalone testing of the parser or for fuzz testing." ) );
|
|
|
|
int cmd_parsed_ok = cl_parser.Parse();
|
|
if( cmd_parsed_ok != 0 )
|
|
{
|
|
// Help and invalid input both stop here
|
|
return ( cmd_parsed_ok == -1 ) ? KI_TEST::RET_CODES::OK : KI_TEST::RET_CODES::BAD_CMDLINE;
|
|
}
|
|
|
|
const bool verbose = cl_parser.Found( "verbose" );
|
|
|
|
bool ok = true;
|
|
|
|
const auto file_count = cl_parser.GetParamCount();
|
|
|
|
if( file_count == 0 )
|
|
{
|
|
// Parse the file provided on stdin - used by AFL to drive the
|
|
// program
|
|
// while (__AFL_LOOP(2))
|
|
{
|
|
ok = parse( std::cin, verbose );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Parse 'n' files given on the command line
|
|
// (this is useful for input minimisation (e.g. afl-tmin) as
|
|
// well as manual testing
|
|
for( unsigned i = 0; i < file_count; i++ )
|
|
{
|
|
const auto filename = cl_parser.GetParam( i ).ToStdString();
|
|
|
|
if( verbose )
|
|
std::cout << "Parsing: " << filename << std::endl;
|
|
|
|
std::ifstream fin;
|
|
fin.open( filename );
|
|
|
|
ok = ok && parse( fin, verbose );
|
|
}
|
|
}
|
|
|
|
if( !ok )
|
|
return PARSER_RET_CODES::PARSE_FAILED;
|
|
|
|
return KI_TEST::RET_CODES::OK;
|
|
}
|
|
|
|
|
|
static bool registered = UTILITY_REGISTRY::Register(
|
|
{ "pcb_parser", "Parse a KiCad PCB file", pcb_parser_main_func } );
|