real_positions.cpp
来自「Boost provides free peer-reviewed portab」· C++ 代码 · 共 184 行
CPP
184 行
/*============================================================================= Boost.Wave: A Standard compliant C++ preprocessor library Sample showing how to correct the positions inside the returned tokens in a way that these appear to be consecutive (ignoring positions from macro definitions). http://www.boost.org/ Copyright (c) 2001-2008 Hartmut Kaiser. Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)=============================================================================*/#include <iostream>#include <fstream>#include <iomanip>#include <string>#include <vector>///////////////////////////////////////////////////////////////////////////////// Include Wave itself#include <boost/wave.hpp>///////////////////////////////////////////////////////////////////////////////// Include the lexer stuff#include "real_position_token.hpp" // token class#include <boost/wave/cpplexer/cpp_lex_iterator.hpp> // lexer type#include "correct_token_positions.hpp"/////////////////////////////////////////////////////////////////////////////////// Special output operator for a lex_token.//// Note: this doesn't compile if BOOST_SPIRIT_DEBUG is defined./////////////////////////////////////////////////////////////////////////////////template <typename PositionT>inline std::ostream &operator<< (std::ostream &stream, lex_token<PositionT> const &t){ using namespace std; using namespace boost::wave; token_id id = token_id(t); stream << setw(16) << left << boost::wave::get_token_name(id) << " (" << "#" << setw(3) << BASEID_FROM_TOKEN(id); if (ExtTokenTypeMask & id) { // this is an extended token id if (AltTokenType == (id & ExtTokenOnlyMask)) { stream << ", AltTokenType"; } else if (TriGraphTokenType == (id & ExtTokenOnlyMask)) { stream << ", TriGraphTokenType"; } else if (AltExtTokenType == (id & ExtTokenOnlyMask)){ stream << ", AltExtTokenType"; } } stream << "): >"; typedef typename lex_token<PositionT>::string_type string_type; string_type const& value = t.get_value(); for (std::size_t i = 0; i < value.size(); ++i) { switch (value[i]) { case '\r': stream << "\\r"; break; case '\n': stream << "\\n"; break; case '\t': stream << "\\t"; break; default: stream << value[i]; break; } } stream << "<" << std::endl; stream << " at: " << t.get_position().get_file() << " (" << setw(3) << right << t.get_position().get_line() << "/" << setw(2) << right << t.get_position().get_column() << ")" << std::endl; stream << " and: " << t.get_corrected_position().get_file() << " (" << setw(3) << right << t.get_corrected_position().get_line() << "/" << setw(2) << right << t.get_corrected_position().get_column() << ")"; return stream;}///////////////////////////////////////////////////////////////////////////////// main entry pointint main(int argc, char *argv[]){ if (2 != argc) { std::cerr << "Usage: real_positions infile" << std::endl; return -1; } // current file position is saved for exception handlingboost::wave::util::file_position_type current_position; try { // Open and read in the specified input file. std::ifstream instream(argv[1]); std::string instring; if (!instream.is_open()) { std::cerr << "Could not open input file: " << argv[1] << std::endl; return -2; } instream.unsetf(std::ios::skipws); instring = std::string(std::istreambuf_iterator<char>(instream.rdbuf()), std::istreambuf_iterator<char>()); // The template real_positions::lex_token<> is the token type to be // used by the Wave library. typedef lex_token<> token_type; // The template boost::wave::cpplexer::lex_iterator<> is the lexer type to // be used by the Wave library. typedef boost::wave::cpplexer::lex_iterator<token_type> lex_iterator_type; // This is the resulting context type to use. The first template parameter // should match the iterator type to be used during construction of the // corresponding context object (see below). typedef boost::wave::context< std::string::iterator, lex_iterator_type, boost::wave::iteration_context_policies::load_file_to_string, correct_token_position<token_type> > context_type; // This preprocessor hooks are used to correct the file positions inside // the tokens returned from the library correct_token_position<token_type> hooks(argv[1]); // The preprocessor iterator shouldn't be constructed directly. It is // to be generated through a wave::context<> object. This wave:context<> // object is to be used additionally to initialize and define different // parameters of the actual preprocessing (not done here). // // The preprocessing of the input stream is done on the fly behind the // scenes during iteration over the context_type::iterator_type stream. context_type ctx (instring.begin(), instring.end(), argv[1], hooks); // analyze the input file context_type::iterator_type first = ctx.begin(); context_type::iterator_type last = ctx.end(); while (first != last) { current_position = (*first).get_position(); std::cout << *first << std::endl; ++first; } } catch (boost::wave::cpp_exception const& e) { // some preprocessing error std::cerr << e.file_name() << "(" << e.line_no() << "): " << e.description() << std::endl; return 2; } catch (std::exception const& e) { // use last recognized token to retrieve the error position std::cerr << current_position.get_file() << "(" << current_position.get_line() << "): " << "exception caught: " << e.what() << std::endl; return 3; } catch (...) { // use last recognized token to retrieve the error position std::cerr << current_position.get_file() << "(" << current_position.get_line() << "): " << "unexpected exception caught." << std::endl; return 4; } return 0;}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?