/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2007-2011 SoftPLC Corporation, Dick Hollenbeck * Copyright (C) 2017-2021 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 #include // HAVE_FGETC_NOLOCK #include #include #include #include #include #include #include // Fall back to getc() when getc_unlocked() is not available on the target platform. #if !defined( HAVE_FGETC_NOLOCK ) #ifdef _MSC_VER //getc is not a macro on windows and adds a tiny overhead for the indirection to eventually calling fgetc #define getc_unlocked _fgetc_nolock #else #define getc_unlocked getc #endif #endif static int vprint( std::string* result, const char* format, va_list ap ) { va_list tmp; va_copy( tmp, ap ); size_t len = vsnprintf( nullptr, 0, format, tmp ); va_end( tmp ); // Resize the output to hold the required data size_t size = result->size(); result->resize( size + len ); // Now do the actual printing len = vsnprintf( result->data() + size, len + 1, format, ap ); return len; } int StrPrintf( std::string* result, const char* format, ... ) { va_list args; va_start( args, format ); int ret = vprint( result, format, args ); va_end( args ); return ret; } std::string StrPrintf( const char* format, ... ) { std::string ret; va_list args; va_start( args, format ); ignore_unused( vprint( &ret, format, args ) ); va_end( args ); return ret; } wxString SafeReadFile( const wxString& aFilePath, const wxString& aReadType ) { auto From_UTF8_WINE = []( const char* cstring ) { wxString line = wxString::FromUTF8( cstring ); if( line.IsEmpty() ) // happens when cstring is not a valid UTF8 sequence line = wxConvCurrent->cMB2WC( cstring ); // try to use locale conversion // We have trouble here *probably* because Wine-hosted LTSpice writes out MSW // encoded text on a macOS, where it isn't expected. In any case, wxWidgets' // wxSafeConvert() appears to get around it. if( line.IsEmpty() ) line = wxSafeConvertMB2WX( cstring ); // I'm not sure what the source of this style of line-endings is, but it can be // found in some Fairchild Semiconductor SPICE files. line.Replace( wxS( "\r\r\n" ), wxS( "\n" ) ); return line; }; // Open file FILE* fp = wxFopen( aFilePath, aReadType ); if( !fp ) THROW_IO_ERROR( wxString::Format( _( "Cannot open file '%s'." ), aFilePath ) ); FILE_LINE_READER fileReader( fp, aFilePath ); wxString contents; while( fileReader.ReadLine() ) contents += From_UTF8_WINE( fileReader.Line() ); return contents; } //----------------------------------------------------------- LINE_READER::LINE_READER( unsigned aMaxLineLength ) : m_length( 0 ), m_lineNum( 0 ), m_line( nullptr ), m_capacity( 0 ), m_maxLineLength( aMaxLineLength ) { if( aMaxLineLength != 0 ) { // start at the INITIAL size, expand as needed up to the MAX size in maxLineLength m_capacity = LINE_READER_LINE_INITIAL_SIZE; // but never go above user's aMaxLineLength, and leave space for trailing nul if( m_capacity > aMaxLineLength+1 ) m_capacity = aMaxLineLength+1; // Be sure there is room for a null EOL char, so reserve at least capacity+1 bytes // to ensure capacity line length and avoid corner cases // Use capacity+5 to cover and corner case m_line = new char[m_capacity+5]; m_line[0] = '\0'; } } LINE_READER::~LINE_READER() { delete[] m_line; } void LINE_READER::expandCapacity( unsigned aNewsize ) { // m_length can equal maxLineLength and nothing breaks, there's room for // the terminating nul. cannot go over this. if( aNewsize > m_maxLineLength+1 ) aNewsize = m_maxLineLength+1; if( aNewsize > m_capacity ) { m_capacity = aNewsize; // resize the buffer, and copy the original data // Be sure there is room for the null EOL char, so reserve capacity+1 bytes // to ensure capacity line length. Use capacity+5 to cover and corner case char* bigger = new char[m_capacity+5]; wxASSERT( m_capacity >= m_length+1 ); memcpy( bigger, m_line, m_length ); bigger[m_length] = 0; delete[] m_line; m_line = bigger; } } FILE_LINE_READER::FILE_LINE_READER( const wxString& aFileName, unsigned aStartingLineNumber, unsigned aMaxLineLength ): LINE_READER( aMaxLineLength ), m_iOwn( true ) { m_fp = KIPLATFORM::IO::SeqFOpen( aFileName, wxT( "rt" ) ); if( !m_fp ) { wxString msg = wxString::Format( _( "Unable to open %s for reading." ), aFileName.GetData() ); THROW_IO_ERROR( msg ); } m_source = aFileName; m_lineNum = aStartingLineNumber; } FILE_LINE_READER::FILE_LINE_READER( FILE* aFile, const wxString& aFileName, bool doOwn, unsigned aStartingLineNumber, unsigned aMaxLineLength ) : LINE_READER( aMaxLineLength ), m_iOwn( doOwn ), m_fp( aFile ) { m_source = aFileName; m_lineNum = aStartingLineNumber; } FILE_LINE_READER::~FILE_LINE_READER() { if( m_iOwn && m_fp ) fclose( m_fp ); } long int FILE_LINE_READER::FileLength() { fseek( m_fp, 0, SEEK_END ); long int fileLength = ftell( m_fp ); rewind( m_fp ); return fileLength; } long int FILE_LINE_READER::CurPos() { return ftell( m_fp ); } char* FILE_LINE_READER::ReadLine() { m_length = 0; for( ;; ) { if( m_length >= m_maxLineLength ) THROW_IO_ERROR( _( "Maximum line length exceeded" ) ); if( m_length >= m_capacity ) expandCapacity( m_capacity * 2 ); // faster, POSIX compatible fgetc(), no locking. int cc = getc_unlocked( m_fp ); if( cc == EOF ) break; m_line[ m_length++ ] = (char) cc; if( cc == '\n' ) break; } m_line[ m_length ] = 0; // m_lineNum is incremented even if there was no line read, because this // leads to better error reporting when we hit an end of file. ++m_lineNum; return m_length ? m_line : nullptr; } STRING_LINE_READER::STRING_LINE_READER( const std::string& aString, const wxString& aSource ): LINE_READER( LINE_READER_LINE_DEFAULT_MAX ), m_lines( aString ), m_ndx( 0 ) { // Clipboard text should be nice and _use multiple lines_ so that // we can report _line number_ oriented error messages when parsing. m_source = aSource; } STRING_LINE_READER::STRING_LINE_READER( const STRING_LINE_READER& aStartingPoint ): LINE_READER( LINE_READER_LINE_DEFAULT_MAX ), m_lines( aStartingPoint.m_lines ), m_ndx( aStartingPoint.m_ndx ) { // since we are keeping the same "source" name, for error reporting purposes // we need to have the same notion of line number and offset. m_source = aStartingPoint.m_source; m_lineNum = aStartingPoint.m_lineNum; } char* STRING_LINE_READER::ReadLine() { size_t nlOffset = m_lines.find( '\n', m_ndx ); unsigned new_length; if( nlOffset == std::string::npos ) new_length = m_lines.length() - m_ndx; else new_length = nlOffset - m_ndx + 1; // include the newline, so +1 if( new_length ) { if( new_length >= m_maxLineLength ) THROW_IO_ERROR( _("Line length exceeded") ); if( new_length+1 > m_capacity ) // +1 for terminating nul expandCapacity( new_length+1 ); wxASSERT( m_ndx + new_length <= m_lines.length() ); memcpy( m_line, &m_lines[m_ndx], new_length ); m_ndx += new_length; } m_length = new_length; ++m_lineNum; // this gets incremented even if no bytes were read m_line[m_length] = 0; return m_length ? m_line : nullptr; } INPUTSTREAM_LINE_READER::INPUTSTREAM_LINE_READER( wxInputStream* aStream, const wxString& aSource ) : LINE_READER( LINE_READER_LINE_DEFAULT_MAX ), m_stream( aStream ) { m_source = aSource; } char* INPUTSTREAM_LINE_READER::ReadLine() { m_length = 0; for( ;; ) { if( m_length >= m_maxLineLength ) THROW_IO_ERROR( _( "Maximum line length exceeded" ) ); if( m_length + 1 > m_capacity ) expandCapacity( m_capacity * 2 ); // this read may fail, docs say to test LastRead() before trusting cc. char cc = m_stream->GetC(); if( !m_stream->LastRead() ) break; m_line[ m_length++ ] = cc; if( cc == '\n' ) break; } m_line[ m_length ] = 0; // m_lineNum is incremented even if there was no line read, because this // leads to better error reporting when we hit an end of file. ++m_lineNum; return m_length ? m_line : nullptr; } //--------------------------------------------------------- // factor out a common GetQuoteChar const char* OUTPUTFORMATTER::GetQuoteChar( const char* wrapee, const char* quote_char ) { // Include '#' so a symbol is not confused with a comment. We intend // to wrap any symbol starting with a '#'. // Our LEXER class handles comments, and comments appear to be an extension // to the SPECCTRA DSN specification. if( *wrapee == '#' ) return quote_char; if( strlen( wrapee ) == 0 ) return quote_char; bool isFirst = true; for( ; *wrapee; ++wrapee, isFirst = false ) { static const char quoteThese[] = "\t ()" "%" // per Alfons of freerouting.net, he does not like this unquoted as of 1-Feb-2008 "{}" // guessing that these are problems too ; // if the string to be wrapped (wrapee) has a delimiter in it, // return the quote_char so caller wraps the wrapee. if( strchr( quoteThese, *wrapee ) ) return quote_char; if( !isFirst && '-' == *wrapee ) return quote_char; } return ""; // caller does not need to wrap, can use an unwrapped string. } const char* OUTPUTFORMATTER::GetQuoteChar( const char* wrapee ) const { return GetQuoteChar( wrapee, quoteChar ); } int OUTPUTFORMATTER::vprint( const char* fmt, va_list ap ) { // This function can call vsnprintf twice. // But internally, vsnprintf retrieves arguments from the va_list identified by arg as if // va_arg was used on it, and thus the state of the va_list is likely to be altered by the call. // see: www.cplusplus.com/reference/cstdio/vsnprintf // we make a copy of va_list ap for the second call, if happens va_list tmp; va_copy( tmp, ap ); int ret = vsnprintf( &m_buffer[0], m_buffer.size(), fmt, ap ); if( ret >= (int) m_buffer.size() ) { m_buffer.resize( ret + 1000 ); ret = vsnprintf( &m_buffer[0], m_buffer.size(), fmt, tmp ); } va_end( tmp ); // Release the temporary va_list, initialised from ap if( ret > 0 ) write( &m_buffer[0], ret ); return ret; } int OUTPUTFORMATTER::sprint( const char* fmt, ... ) { va_list args; va_start( args, fmt ); int ret = vprint( fmt, args); va_end( args ); return ret; } int OUTPUTFORMATTER::Print( int nestLevel, const char* fmt, ... ) { #define NESTWIDTH 2 ///< how many spaces per nestLevel va_list args; va_start( args, fmt ); int result = 0; int total = 0; for( int i = 0; i < nestLevel; ++i ) { // no error checking needed, an exception indicates an error. result = sprint( "%*c", NESTWIDTH, ' ' ); total += result; } // no error checking needed, an exception indicates an error. result = vprint( fmt, args ); va_end( args ); total += result; return total; } std::string OUTPUTFORMATTER::Quotes( const std::string& aWrapee ) const { std::string ret; ret.reserve( aWrapee.size() * 2 + 2 ); ret += '"'; for( std::string::const_iterator it = aWrapee.begin(); it != aWrapee.end(); ++it ) { switch( *it ) { case '\n': ret += '\\'; ret += 'n'; break; case '\r': ret += '\\'; ret += 'r'; break; case '\\': ret += '\\'; ret += '\\'; break; case '"': ret += '\\'; ret += '"'; break; default: ret += *it; } } ret += '"'; return ret; } std::string OUTPUTFORMATTER::Quotew( const wxString& aWrapee ) const { // wxStrings are always encoded as UTF-8 as we convert to a byte sequence. // The non-virtual function calls the virtual workhorse function, and if // a different quoting or escaping strategy is desired from the standard, // a derived class can overload Quotes() above, but // should never be a reason to overload this Quotew() here. return Quotes( (const char*) aWrapee.utf8_str() ); } //--------------------------------------------------------- void STRING_FORMATTER::write( const char* aOutBuf, int aCount ) { m_mystring.append( aOutBuf, aCount ); } void STRING_FORMATTER::StripUseless() { std::string copy = m_mystring; m_mystring.clear(); for( std::string::iterator i = copy.begin(); i != copy.end(); ++i ) { if( !isspace( *i ) && *i != ')' && *i != '(' && *i != '"' ) { m_mystring += *i; } } } FILE_OUTPUTFORMATTER::FILE_OUTPUTFORMATTER( const wxString& aFileName, const wxChar* aMode, char aQuoteChar ): OUTPUTFORMATTER( OUTPUTFMTBUFZ, aQuoteChar ), m_filename( aFileName ) { m_fp = wxFopen( aFileName, aMode ); if( !m_fp ) THROW_IO_ERROR( strerror( errno ) ); } FILE_OUTPUTFORMATTER::~FILE_OUTPUTFORMATTER() { if( m_fp ) fclose( m_fp ); } void FILE_OUTPUTFORMATTER::write( const char* aOutBuf, int aCount ) { if( fwrite( aOutBuf, (unsigned) aCount, 1, m_fp ) != 1 ) THROW_IO_ERROR( strerror( errno ) ); } PRETTIFIED_FILE_OUTPUTFORMATTER::PRETTIFIED_FILE_OUTPUTFORMATTER( const wxString& aFileName, const wxChar* aMode, char aQuoteChar ) : OUTPUTFORMATTER( OUTPUTFMTBUFZ, aQuoteChar ) { m_fp = wxFopen( aFileName, aMode ); if( !m_fp ) THROW_IO_ERROR( strerror( errno ) ); } PRETTIFIED_FILE_OUTPUTFORMATTER::~PRETTIFIED_FILE_OUTPUTFORMATTER() { try { PRETTIFIED_FILE_OUTPUTFORMATTER::Finish(); } catch( ... ) {} } bool PRETTIFIED_FILE_OUTPUTFORMATTER::Finish() { if( !m_fp ) return false; KICAD_FORMAT::Prettify( m_buf ); if( fwrite( m_buf.c_str(), m_buf.length(), 1, m_fp ) != 1 ) THROW_IO_ERROR( strerror( errno ) ); fclose( m_fp ); m_fp = nullptr; return true; } void PRETTIFIED_FILE_OUTPUTFORMATTER::write( const char* aOutBuf, int aCount ) { m_buf.append( aOutBuf, aCount ); }