summaryrefslogtreecommitdiffstats
path: root/debian/uncrustify-trinity/uncrustify-trinity-0.78.1/src/output.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'debian/uncrustify-trinity/uncrustify-trinity-0.78.1/src/output.cpp')
-rw-r--r--debian/uncrustify-trinity/uncrustify-trinity-0.78.1/src/output.cpp3580
1 files changed, 0 insertions, 3580 deletions
diff --git a/debian/uncrustify-trinity/uncrustify-trinity-0.78.1/src/output.cpp b/debian/uncrustify-trinity/uncrustify-trinity-0.78.1/src/output.cpp
deleted file mode 100644
index 6de21597..00000000
--- a/debian/uncrustify-trinity/uncrustify-trinity-0.78.1/src/output.cpp
+++ /dev/null
@@ -1,3580 +0,0 @@
-/**
- * @file output.cpp
- * Does all the output & comment formatting.
- *
- * @author Ben Gardner
- * @author Guy Maurel October 2015, 2021
- * @license GPL v2+
- */
-
-#include "output.h"
-
-#include "align_tab_column.h"
-#include "braces.h"
-#include "indent.h"
-#include "prototypes.h"
-#include "tokenize.h"
-#include "unc_ctype.h"
-#include "unicode.h"
-
-#include <ctime>
-#include <map>
-#include <regex>
-#include <set>
-
-// if you need more logs, commented out the next define line
-#define EXTRA_LOG
-
-constexpr static auto LCURRENT = LOUTPUT;
-
-using namespace uncrustify;
-
-
-struct cmt_reflow
-{
- Chunk *pc = Chunk::NullChunkPtr;
- size_t column = 0; //! Column of the comment start
- size_t brace_col = 0; //! Brace column (for indenting with tabs)
- size_t base_col = 0; //! Base column (for indenting with tabs)
- size_t word_count = 0; //! number of words on this line
- size_t xtra_indent = 0; //! extra indent of non-first lines (0 or 1)
- UncText cont_text; //! fixed text to output at the start of a line (0 to 3 chars)
- bool reflow = false; //! reflow the current line
-};
-
-
-// for tracking line numbering
-bool numbering_status = false;
-int line_number;
-char char_number[16] = { 0 };
-
-
-void set_numbering(bool status)
-{
- if (options::set_numbering_for_html_output())
- {
- numbering_status = status;
- }
-}
-
-
-bool get_numbering()
-{
- return(numbering_status);
-}
-
-
-void set_line_number()
-{
- line_number = 0;
-}
-
-
-void print_numbering()
-{
- if (get_numbering())
- {
- line_number++;
- sprintf(char_number, "%d ", line_number);
- write_string(char_number);
- }
-}
-
-
-/**
- * A multiline comment
- * The only trick here is that we have to trim out whitespace characters
- * to get the comment to line up.
- */
-static void output_comment_multi(Chunk *pc);
-
-
-static bool kw_fcn_filename(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_class(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_message(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_category(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_scope(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_function(Chunk *cmt, UncText &out_txt);
-
-
-/**
- * Adds the javadoc-style @param and @return stuff, based on the params and
- * return value for pc.
- * If the arg list is '()' or '(void)', then no @params are added.
- * Likewise, if the return value is 'void', then no @return is added.
- */
-static bool kw_fcn_javaparam(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_fclass(Chunk *cmt, UncText &out_txt);
-
-
-static bool kw_fcn_year(Chunk *cmt, UncText &out_txt);
-
-
-/**
- * Output a multiline comment without any reformatting other than shifting
- * it left or right to get the column right.
- *
- * Trims trailing whitespaces.
- */
-static void output_comment_multi_simple(Chunk *pc);
-
-
-/**
- * This renders the #if condition to a string buffer.
- *
- * @param[out] dst UncText buffer to be filled
- * @param[in] ifdef if conditional as chunk list
- */
-static void generate_if_conditional_as_text(UncText &dst, Chunk *ifdef);
-
-
-/**
- * Do keyword substitution on a comment.
- * NOTE: it is assumed that a comment will contain at most one of each type
- * of keyword.
- */
-static void do_kw_subst(Chunk *pc);
-
-
-//! All output text is sent here, one char at a time.
-static void add_char(UINT32 ch, bool is_literal = false);
-
-
-static void add_text(const char *ascii_text);
-
-
-static void add_text(const UncText &text, bool is_ignored, bool is_literal);
-
-
-/**
- * Count the number of characters to the end of the next chunk of text.
- * If it exceeds the limit, return true.
- */
-static bool next_word_exceeds_limit(const UncText &text, size_t idx);
-
-
-/**
- * Output a comment to the column using indent_with_tabs and
- * indent_cmt_with_tabs as the rules.
- * base_col is the indent of the first line of the comment.
- * On the first line, column == base_col.
- * On subsequent lines, column >= base_col.
- *
- * @param brace_col the brace-level indent of the comment
- * @param base_col the indent of the start of the comment (multiline)
- * @param column the column that we should end up in
- */
-static void cmt_output_indent(size_t brace_col, size_t base_col, size_t column);
-
-
-/**
- * Checks for and updates the lead chars.
- *
- * @param line the comment line
- *
- * @return 0: not present, >0: number of chars that are part of the lead
- */
-static size_t cmt_parse_lead(const UncText &line, bool is_last);
-
-
-/**
- * Scans a multiline comment to determine the following:
- * - the extra indent of the non-first line (0 or 1)
- * - the continuation text ('' or '* ')
- *
- * The decision is based on:
- * - cmt_indent_multi
- * - cmt_star_cont
- * - cmt_multi_first_len_minimum
- * - the first line length
- * - the second line leader length
- * - the last line length (without leading space/tab)
- *
- * If the first and last line are the same length and don't contain any alnum
- * chars and (the first line len > 2 or the second leader is the same as the
- * first line length), then the indent is 0.
- *
- * If the leader on the second line is 1 wide or missing, then the indent is 1.
- *
- * Otherwise, the indent is 0.
- *
- * @param str The comment string
- * @param len Length of the comment
- * @param start_col Starting column
- *
- * @return cmt.xtra_indent is set to 0 or 1
- */
-static void calculate_comment_body_indent(cmt_reflow &cmt, const UncText &str);
-
-
-static int next_up(const UncText &text, size_t idx, const UncText &tag);
-
-
-/**
- * Outputs the C comment at pc.
- * C comment combining is done here
- *
- * @return the last chunk output'd
- */
-static Chunk *output_comment_c(Chunk *pc);
-
-
-/**
- * Outputs the CPP comment at pc.
- * CPP comment combining is done here
- *
- * @return the last chunk output'd
- */
-static Chunk *output_comment_cpp(Chunk *pc);
-
-
-static void cmt_trim_whitespace(UncText &line, bool in_preproc);
-
-
-/**
- * Outputs a comment. The initial opening '//' may be included in the text.
- * Subsequent openings (if combining comments), should not be included.
- * The closing (for C/D comments) should not be included.
- *
- * TODO:
- * If reflowing text, the comment should be added one word (or line) at a time.
- * A newline should only be sent if a blank line is encountered or if the next
- * line is indented beyond the current line (optional?).
- * If the last char on a line is a ':' or '.', then the next line won't be
- * combined.
- */
-static void add_comment_text(const UncText &text, cmt_reflow &cmt, bool esc_close, size_t continuation_indent = 0);
-
-
-static void output_cmt_start(cmt_reflow &cmt, Chunk *pc);
-
-
-/**
- * Checks to see if the current comment can be combined with the next comment.
- * The two can be combined if:
- * 1. They are the same type
- * 2. There is exactly one newline between then
- * 3. They are indented to the same level
- */
-static bool can_combine_comment(Chunk *pc, cmt_reflow &cmt);
-
-
-#define LOG_CONTTEXT() \
- LOG_FMT(LCONTTEXT, "%s(%d): set cont_text to '%s'\n", __func__, __LINE__, cmt.cont_text.c_str())
-
-
-static void add_spaces()
-{
- while (cpd.spaces > 0)
- {
- write_char(' ');
- cpd.spaces--;
- }
-}
-
-
-static void add_char(UINT32 ch, bool is_literal)
-{
- // If we did a '\r' and it isn't followed by a '\n', then output a newline
- if ( (cpd.last_char == '\r')
- && (ch != '\n'))
- {
- write_string(cpd.newline);
- cpd.column = 1;
- cpd.did_newline = true;
- cpd.spaces = 0;
- }
-
- // convert a newline into the LF/CRLF/CR sequence
- if (ch == '\n')
- {
- add_spaces();
- write_string(cpd.newline);
- cpd.column = 1;
- cpd.did_newline = true;
- cpd.spaces = 0;
- print_numbering();
- }
- else if (ch == '\r') // do not output the CARRIAGERETURN
- {
- // do not output '\r'
- cpd.column = 1;
- cpd.did_newline = true;
- cpd.spaces = 0;
- }
- else if ( (ch == '\t')
- && cpd.output_tab_as_space)
- {
- size_t endcol = next_tab_column(cpd.column);
-
- while (cpd.column < endcol)
- {
- add_char(' ');
- }
- return;
- }
- else
- {
- // explicitly disallow a tab after a space
- if ( !is_literal
- && ch == '\t'
- && cpd.last_char == ' ')
- {
- log_rule_B("indent_with_tabs");
-
- int indent_with_tabs = options::pp_indent_with_tabs();
-
- if ( cpd.in_preproc != CT_PREPROC
- || indent_with_tabs == -1)
- {
- indent_with_tabs = options::indent_with_tabs();
- }
-
- if (indent_with_tabs == 0)
- {
- size_t endcol = next_tab_column(cpd.column);
-
- while (cpd.column < endcol)
- {
- add_char(' ');
- }
- return;
- }
- }
-
- if ( (ch == ' ')
- && !cpd.output_trailspace)
- {
- cpd.spaces++;
- cpd.column++;
- }
- else
- {
- add_spaces();
- write_char(ch);
-
- if (ch == '\t')
- {
- cpd.column = next_tab_column(cpd.column);
- }
- else
- {
- cpd.column++;
- }
- }
- }
- cpd.last_char = ch;
-} // add_char
-
-
-static void add_text(const char *ascii_text)
-{
- char ch;
-
- while ((ch = *ascii_text) != 0)
- {
- ascii_text++;
- add_char(ch);
- }
-}
-
-
-static void add_text(const UncText &text, bool is_ignored = false, bool is_literal = false)
-{
- for (size_t idx = 0; idx < text.size(); idx++)
- {
- int ch = text[idx];
-
- if (is_ignored)
- {
- write_char(ch);
- }
- else
- {
- add_char(ch, is_literal);
- }
- }
-}
-
-
-static bool next_word_exceeds_limit(const UncText &text, size_t idx)
-{
- size_t length = 0;
-
- // Count any whitespace
- while ( (idx < text.size())
- && unc_isspace(text[idx]))
- {
- idx++;
- length++;
- }
-
- // Count non-whitespace
- while ( (idx < text.size())
- && !unc_isspace(text[idx]))
- {
- idx++;
- length++;
- }
- bool exceed_limit = (cpd.column + length - 1) > options::cmt_width();
- LOG_FMT(LCONTTEXT, "%s(%d): idx is %zu%s\n",
- __func__, __LINE__, idx, (exceed_limit ? " exceeds limit" : ""));
- return(exceed_limit);
-}
-
-
-/**
- * Advance to a specific column
- * cpd.column is the current column
- *
- * @param column The column to advance to
- */
-static void output_to_column(size_t column, bool allow_tabs)
-{
- cpd.did_newline = false;
-
- if (allow_tabs)
- {
- // tab out as far as possible and then use spaces
- size_t next_column = next_tab_column(cpd.column);
-
- while (next_column <= column)
- {
- add_text("\t");
- next_column = next_tab_column(cpd.column);
- }
- }
-
- // space out the final bit
- while (cpd.column < column)
- {
- add_text(" ");
- }
-}
-
-
-static void cmt_output_indent(size_t brace_col, size_t base_col, size_t column)
-{
- log_rule_B("indent_cmt_with_tabs");
- size_t iwt = options::indent_cmt_with_tabs() ? 2 :
- (options::indent_with_tabs() ? 1 : 0);
-
- size_t tab_col = (iwt == 0) ? 0 : ((iwt == 1) ? brace_col : base_col);
-
- // LOG_FMT(LSYS, "%s(brace=%zd base=%zd col=%zd iwt=%zd) tab=%zd cur=%zd\n",
- // __func__, brace_col, base_col, column, iwt, tab_col, cpd.column);
-
- cpd.did_newline = false;
-
- if ( iwt == 2
- || ( cpd.column == 1
- && iwt == 1))
- {
- // tab out as far as possible and then use spaces
- while (next_tab_column(cpd.column) <= tab_col)
- {
- add_text("\t");
- }
- }
-
- // space out the rest
- while (cpd.column < column)
- {
- add_text(" ");
- }
-} // cmt_output_indent
-
-
-void output_parsed(FILE *pfile, bool withOptions)
-{
- const char *eol_marker = get_eol_marker();
-
- if (withOptions)
- {
- save_option_file(pfile, false, true);
- }
- fprintf(pfile, "# -=====-%s", eol_marker);
- fprintf(pfile, "# number of loops = %d\n", cpd.changes);
- fprintf(pfile, "# -=====-%s", eol_marker);
- fprintf(pfile, "# language = %s\n", language_name_from_flags(cpd.lang_flags));
- fprintf(pfile, "# -=====-%s", eol_marker);
- // MAXLENGTHOFTHENAME must be consider at the format line at the file
- // output.cpp, line 427: fprintf(pfile, "# Line Tag Parent...
- // and 430: ... fprintf(pfile, "%s# %3zu>%19.19s[%19.19s] ...
- // here xx xx xx xx
-#ifdef WIN32
- fprintf(pfile, "# Line Tag Parent_type Type of the parent Columns Br/Lvl/pp Nl Text");
-#else // not WIN32
- fprintf(pfile, "# Line Tag Parent_type Type of the parent Columns Br/Lvl/pp Flags Nl Text");
-#endif // ifdef WIN32
-
- for (Chunk *pc = Chunk::GetHead(); pc->IsNotNullChunk(); pc = pc->GetNext())
- {
-#ifdef WIN32
- fprintf(pfile, "%s# %3d>%19.19s|%19.19s|%19.19s[%3d/%3d/%3d/%3d][%d/%d/%d][%d-%d]",
- eol_marker, (int)pc->GetOrigLine(), get_token_name(pc->GetType()),
- get_token_name(pc->GetParentType()), get_token_name(pc->GetTypeOfParent()),
- (int)pc->GetColumn(), (int)pc->GetOrigCol(), (int)pc->GetOrigColEnd(), (int)pc->GetOrigPrevSp(),
- (int)pc->GetBraceLevel(), (int)pc->GetLevel(), (int)pc->GetPpLevel(), (int)pc->GetNlCount(), pc->GetAfterTab());
-#else // not WIN32
- fprintf(pfile, "%s# %3zu>%19.19s|%19.19s|%19.19s[%3zu/%3zu/%3zu/%3zu][%zu/%zu/%zu]",
- eol_marker, pc->GetOrigLine(), get_token_name(pc->GetType()),
- get_token_name(pc->GetParentType()), get_token_name(pc->GetTypeOfParent()),
- pc->GetColumn(), pc->GetOrigCol(), pc->GetOrigColEnd(), pc->GetOrigPrevSp(),
- pc->GetBraceLevel(), pc->GetLevel(), pc->GetPpLevel());
- // Print pc flags in groups of 4 hex characters
- char flag_string[24];
- sprintf(flag_string, "%16llx", static_cast<PcfFlags::int_t>(pc->GetFlags()));
- fprintf(pfile, "[%.4s %.4s %.4s %.4s]", flag_string, flag_string + 4, flag_string + 8, flag_string + 12);
- fprintf(pfile, "[%zu-%d]",
- pc->GetNlCount(), pc->GetAfterTab());
-#endif // ifdef WIN32
-
- if ( pc->IsNot(CT_NEWLINE)
- && (pc->Len() != 0))
- {
- for (size_t cnt = 0; cnt < pc->GetColumn(); cnt++)
- {
- fprintf(pfile, " ");
- }
-
- if (pc->IsNot(CT_NL_CONT))
- {
- fprintf(pfile, "%s", pc->Text());
- }
- else
- {
- fprintf(pfile, "\\");
- }
- }
-
- if (options::debug_decode_the_flags())
- {
- // such as:
- // The flags are: [0xc0400:IN_CLASS,STMT_START,EXPR_START]
- fprintf(pfile, "%s The flags are: ", eol_marker);
- fprintf(pfile, "%s", pcf_flags_str(pc->GetFlags()).c_str());
- }
- }
-
- fprintf(pfile, "%s# -=====-%s", eol_marker, eol_marker);
- fflush(pfile);
-} // output_parsed
-
-
-void output_parsed_csv(FILE *pfile)
-{
- const char *eol_marker = get_eol_marker();
-
- fprintf(pfile, "number of loops,%d,\n", cpd.changes);
- fprintf(pfile, "language,%s,\n", language_name_from_flags(cpd.lang_flags));
- fprintf(pfile, "Line,Tag,Parent_type,Type of the parent,Column,Orig Col Strt,"
- "Orig Col End,Orig Sp Before,Br,Lvl,pp,Flags,Nl Before,Nl After,Text,");
-
- for (Chunk *pc = Chunk::GetHead(); pc->IsNotNullChunk(); pc = pc->GetNext())
- {
- fprintf(pfile, "%s%zu,%s,%s,%s,%zu,%zu,%zu,%zu,%zu,%zu,%zu,",
- eol_marker, pc->GetOrigLine(), get_token_name(pc->GetType()),
- get_token_name(pc->GetParentType()), get_token_name(pc->GetTypeOfParent()),
- pc->GetColumn(), pc->GetOrigCol(), pc->GetOrigColEnd(), pc->GetOrigPrevSp(),
- pc->GetBraceLevel(), pc->GetLevel(), pc->GetPpLevel());
-
- auto pcf_flag_str = pcf_flags_str(E_PcfFlag(pc->GetFlags()));
-#ifdef WIN32
- auto pcf_flag_str_start = pcf_flag_str.find("[") + 1;
-#else // not WIN32
- auto pcf_flag_str_start = pcf_flag_str.find(":") + 1;
-#endif // ifdef WIN32
- auto pcf_flag_str_end = pcf_flag_str.find("]");
- auto pcf_names = pcf_flag_str.substr(pcf_flag_str_start,
- pcf_flag_str_end - pcf_flag_str_start);
- fprintf(pfile, "\"%s\",", pcf_names.c_str());
- fprintf(pfile, "%zu,%d,",
- pc->GetNlCount(), pc->GetAfterTab());
-
- if ( pc->IsNot(CT_NEWLINE)
- && (pc->Len() != 0))
- {
- fprintf(pfile, "\"");
-
- for (size_t cnt = 0; cnt < pc->GetColumn(); cnt++)
- {
- fprintf(pfile, " ");
- }
-
- if (pc->IsNot(CT_NL_CONT))
- {
- for (auto *ch = pc->Text(); *ch != '\0'; ++ch)
- {
- fprintf(pfile, "%c", *ch);
-
- if (*ch == '"')
- {
- // need to escape the double-quote for csv-format
- fprintf(pfile, "\"");
- }
- }
- }
- else
- {
- fprintf(pfile, "\\");
- }
- fprintf(pfile, "\"");
- }
- }
-
- fflush(pfile);
-} // output_parsed_csv
-
-
-// Compares two tracks according to second in descending order.
-bool compareTrack(TrackNumber t1, TrackNumber t2)
-{
- char *t1s = t1.second;
- char *t2s = t2.second;
- int vergleich = strcmp(t2s, t1s);
- bool int_vergleich = vergleich > 0;
-
- return(int_vergleich);
-}
-
-
-void output_text(FILE *pfile)
-{
- bool tracking = cpd.html_type != tracking_type_e::TT_NONE; // special for debugging
-
- cpd.fout = pfile;
- cpd.did_newline = true;
- cpd.column = 1;
-
- if (cpd.bom)
- {
- write_bom();
- }
- Chunk *pc;
-
- if (cpd.frag_cols > 0)
- {
- size_t indent = cpd.frag_cols - 1;
-
- // loop over the whole chunk list
- for (pc = Chunk::GetHead(); pc->IsNotNullChunk(); pc = pc->GetNext())
- {
- pc->SetColumn(pc->GetColumn() + indent);
- pc->SetColumnIndent(pc->GetColumnIndent() + indent);
- }
-
- cpd.frag_cols = 0;
- }
-
- if (tracking)
- {
- set_numbering(false);
- add_text("<html>\n");
- add_text("<head>\n");
- add_text(" <meta http-equiv=\"content-type\" content=\"text/html; charset=utf-8\"/>\n");
- add_text(" <title>Uncrustify: where do the Spaces options work</title>\n");
- add_text("</head>\n");
- add_text("<body lang=\"en-US\">\n");
- add_text("<p>\n");
- add_text("</p>\n");
- add_text("<pre>\n");
- set_numbering(true);
- set_line_number();
- print_numbering();
- }
- bool write_in_tracking = false;
- int pp_indent_with_tabs = options::pp_indent_with_tabs();
-
- if (pp_indent_with_tabs == -1)
- {
- pp_indent_with_tabs = options::indent_with_tabs();
- }
-
- // loop over the whole chunk list
- for (pc = Chunk::GetHead(); pc->IsNotNullChunk(); pc = pc->GetNext())
- {
- char copy[1000];
- LOG_FMT(LCONTTEXT, "%s(%d): Text() is '%s', type is %s, orig line is %zu, column is %zu, nl is %zu\n",
- __func__, __LINE__, pc->ElidedText(copy), get_token_name(pc->GetType()), pc->GetOrigLine(), pc->GetColumn(), pc->GetNlCount());
- cpd.output_tab_as_space = false;
-
- if (pc->Is(CT_NEWLINE))
- {
- for (size_t cnt = 0; cnt < pc->GetNlCount(); cnt++)
- {
- if ( cnt > 0
- && pc->GetNlColumn() > 1)
- {
- log_rule_B("indent_with_tabs - newline");
-
- if (pc->IsPreproc())
- {
- output_to_column(pc->GetNlColumn(), (pp_indent_with_tabs == 2));
- }
- else
- {
- output_to_column(pc->GetNlColumn(), (options::indent_with_tabs() == 2));
- }
- }
- add_char('\n');
- }
-
- cpd.did_newline = true;
- cpd.column = 1;
- }
- else if (pc->Is(CT_NL_CONT))
- {
- // FIXME: this really shouldn't be done here!
- if (!pc->TestFlags(PCF_WAS_ALIGNED))
- {
- // Add or remove space before a backslash-newline at the end of a line.
- log_rule_B("sp_before_nl_cont");
-
- if (options::sp_before_nl_cont() & IARF_REMOVE)
- {
- log_rule_B("sp_before_nl_cont");
- pc->SetColumn(cpd.column + (options::sp_before_nl_cont() == IARF_FORCE));
- }
- else
- {
- // Try to keep the same relative spacing
- Chunk *prev = pc->GetPrev();
-
- if (prev->Is(CT_PP_IGNORE))
- {
- /*
- * Want to completely leave alone PP_IGNORE'd blocks because
- * they likely have special column aligned newline
- * continuations (common in multiline macros)
- */
- pc->SetColumn(pc->GetOrigCol());
- }
- else
- {
- // Try to keep the same relative spacing
- while ( prev->IsNotNullChunk()
- && prev->GetOrigCol() == 0
- && prev->GetNlCount() == 0)
- {
- prev = prev->GetPrev();
- }
-
- if ( prev->IsNotNullChunk()
- && prev->GetNlCount() == 0)
- {
- int orig_sp = pc->GetOrigPrevSp();
-
- if ((int)(cpd.column + orig_sp) < 0)
- {
-#ifdef WIN32
- fprintf(stderr, "FATAL: negative value.\n pc->GetOrigCol() is %d, prev->GetOrigColEnd() is %d\n",
- (int)pc->GetOrigCol(), (int)prev->GetOrigColEnd());
-#else // not WIN32
- fprintf(stderr, "FATAL: negative value.\n pc->GetOrigCol() is %zu, prev->GetOrigColEnd() is %zu\n",
- pc->GetOrigCol(), prev->GetOrigColEnd());
-#endif // ifdef WIN32
- log_flush(true);
- exit(EX_SOFTWARE);
- }
- pc->SetColumn(cpd.column + orig_sp);
-
- // Add or remove space before a backslash-newline at the end of a line.
- log_rule_B("sp_before_nl_cont");
-
- if ( (options::sp_before_nl_cont() != IARF_IGNORE)
- && (pc->GetColumn() < (cpd.column + 1)))
- {
- pc->SetColumn(cpd.column + 1);
- }
- }
- }
- }
- output_to_column(pc->GetColumn(), false);
- }
- else
- {
- log_rule_B("indent_with_tabs - newline cont");
-
- if (pc->IsPreproc())
- {
- output_to_column(pc->GetColumn(), (pp_indent_with_tabs == 2));
- }
- else
- {
- output_to_column(pc->GetColumn(), (options::indent_with_tabs() == 2));
- }
- }
- add_char('\\');
- add_char('\n');
- cpd.did_newline = true;
- cpd.column = 1;
- }
- else if (pc->Is(CT_COMMENT_MULTI))
- {
- log_rule_B("cmt_indent_multi");
- log_rule_B("cmt_convert_tab_to_spaces - multi");
- cpd.output_tab_as_space = options::cmt_convert_tab_to_spaces();
-
- if (options::cmt_indent_multi())
- {
- output_comment_multi(pc);
- }
- else
- {
- output_comment_multi_simple(pc);
- }
- }
- else if ( pc->Is(CT_COMMENT_CPP)
- || pc->Is(CT_COMMENT_CPP_ENDIF))
- {
- log_rule_B("cmt_comment_cpp");
- log_rule_B("cmt_convert_tab_to_spaces - comment_cpp");
- cpd.output_tab_as_space = options::cmt_convert_tab_to_spaces();
-
- bool tmp = cpd.output_trailspace;
- /*
- * keep trailing spaces if they are still present in a chunk;
- * note that tokenize() already strips spaces in comments,
- * so if they made it up to here, they are to stay
- */
- cpd.output_trailspace = true;
- pc = output_comment_cpp(pc);
- cpd.output_trailspace = tmp;
- }
- else if ( pc->Is(CT_COMMENT)
- || pc->Is(CT_COMMENT_ENDIF))
- {
- log_rule_B("cmt_comment");
- log_rule_B("cmt_convert_tab_to_spaces - comment");
- cpd.output_tab_as_space = options::cmt_convert_tab_to_spaces();
-
- pc = output_comment_c(pc);
- }
- else if ( pc->Is(CT_JUNK)
- || pc->Is(CT_IGNORED))
- {
- LOG_FMT(LOUTIND, "%s(%d): orig line is %zu, orig col is %zu,\npc->Text() >%s<, pc->str.size() is %zu\n",
- __func__, __LINE__, pc->GetOrigLine(), pc->GetOrigCol(), pc->Text(), pc->GetStr().size());
- // do not adjust the column for junk
- add_text(pc->GetStr(), true);
- }
- else if (pc->Len() == 0)
- {
- // don't do anything for non-visible stuff
- LOG_FMT(LOUTIND, "%s(%d): orig line is %zu, column is %zu, non-visible stuff: type is %s\n",
- __func__, __LINE__, pc->GetOrigLine(), pc->GetColumn(), get_token_name(pc->GetType()));
- }
- else
- {
- bool allow_tabs;
- cpd.output_trailspace = (pc->Is(CT_STRING_MULTI));
-
- // indent to the 'level' first
- if (cpd.did_newline)
- {
- if ( ( pc->IsPreproc()
- && pp_indent_with_tabs == 1)
- || ( !pc->IsPreproc()
- && options::indent_with_tabs() == 1))
- {
- size_t lvlcol;
-
- /*
- * FIXME: it would be better to properly set m_columnIndent in
- * indent_text(), but this hack for '}' and '#' seems to work.
- */
- if ( pc->Is(CT_BRACE_CLOSE)
- || pc->Is(CT_CASE_COLON)
- || pc->IsPreproc())
- {
- lvlcol = pc->GetColumn();
- }
- else
- {
- lvlcol = pc->GetColumnIndent();
-
- if (lvlcol > pc->GetColumn())
- {
- lvlcol = pc->GetColumn();
- }
- }
-
- if (lvlcol > 1)
- {
- log_rule_B("indent_with_tabs - hack");
- output_to_column(lvlcol, true);
- }
- }
- log_rule_B("indent_with_tabs");
- allow_tabs = ( pc->IsPreproc()
- && pp_indent_with_tabs == 2)
- || ( !pc->IsPreproc()
- && options::indent_with_tabs() == 2)
- || ( pc->IsComment()
- && options::indent_with_tabs() != 0);
-
- LOG_FMT(LOUTIND, "%s(%d): orig line is %zu, column is %zu, column indent is %zu, cpd.column is %zu\n",
- __func__, __LINE__, pc->GetOrigLine(), pc->GetColumn(), pc->GetColumnIndent(), cpd.column);
- }
- else
- {
- /*
- * Reformatting multi-line comments can screw up the column.
- * Make sure we don't mess up the spacing on this line.
- * This has to be done here because comments are not formatted
- * until the output phase.
- */
- if (pc->GetColumn() < cpd.column)
- {
- reindent_line(pc, cpd.column);
- }
- // not the first item on a line
- Chunk *prev = pc->GetPrev();
- log_rule_B("align_with_tabs");
- allow_tabs = ( options::align_with_tabs()
- && pc->TestFlags(PCF_WAS_ALIGNED)
- && ((prev->GetColumn() + prev->Len() + 1) != pc->GetColumn()));
-
- log_rule_B("align_keep_tabs");
-
- if (options::align_keep_tabs())
- {
- allow_tabs |= pc->GetAfterTab();
- }
- LOG_FMT(LOUTIND, "%s(%d): at column %zu(%s)\n",
- __func__, __LINE__, pc->GetColumn(), (allow_tabs ? "true" : "FALSE"));
- }
- output_to_column(pc->GetColumn(), allow_tabs);
-
- if (write_in_tracking)
- {
- if (pc->Is(CT_ANGLE_OPEN))
- {
- add_text("&lt;", false, false);
- }
- else if (pc->Is(CT_ANGLE_CLOSE))
- {
- add_text("&gt;", false, false);
- }
- else
- {
- if (tracking)
- {
- if (pc->GetStr()[0] == '<')
- {
- add_text("&lt;", false, false);
- size_t lang = pc->GetStr().size();
-
- for (size_t idx = 1; idx < lang - 1; idx++)
- {
- int ch = pc->GetStr()[idx];
- add_char(ch);
- }
-
- add_text("&gt;", false, false);
- }
- }
- add_text(pc->GetStr(), false, pc->Is(CT_STRING));
- }
- write_in_tracking = false;
- }
- else
- {
- add_text(pc->GetStr(), false, pc->Is(CT_STRING));
- }
-
- if (pc->Is(CT_PP_DEFINE)) // Issue #876
- {
- // If true, a <TAB> is inserted after #define.
- log_rule_B("force_tab_after_define");
-
- if (options::force_tab_after_define())
- {
- add_char('\t');
- }
- }
- cpd.did_newline = pc->IsNewline();
- cpd.output_trailspace = false;
- }
-
- if (pc->GetTrackingData() != nullptr)
- {
- LOG_FMT(LGUY, " Tracking info are: \n");
- LOG_FMT(LGUY, " number of track(s) %zu\n", pc->GetTrackingData()->size());
- add_text("<a title=\"");
- char tempText[80];
-
- // is sorting necessary?
- size_t many = pc->GetTrackingData()->size();
-
- if (many > 1)
- {
-#ifdef EXTRA_LOG
- // protocol before sort
- for (size_t track = 0; track < pc->GetTrackingData()->size(); track++)
- {
- const TrackList *A = pc->GetTrackingData();
- const TrackNumber B = A->at(track);
- size_t Bfirst = B.first;
- char *Bsecond = B.second;
-
- LOG_FMT(LGUY, " %zu, tracking number is %zu\n", track, Bfirst);
- LOG_FMT(LGUY, " %zu, rule is %s\n", track, Bsecond);
- }
-#endif
-
- if (options::debug_sort_the_tracks())
- {
- TrackList *A1 = pc->TrackingData();
- sort(A1->begin(), A1->end(), compareTrack);
- }
-#ifdef EXTRA_LOG
- // protocol after sort
- for (size_t track = 0; track < pc->GetTrackingData()->size(); track++)
- {
- const TrackList *A = pc->GetTrackingData();
- const TrackNumber B = A->at(track);
- size_t Bfirst = B.first;
- char *Bsecond = B.second;
-
- LOG_FMT(LGUY, " %zu, tracking number is %zu\n", track, Bfirst);
- LOG_FMT(LGUY, " %zu, rule is %s\n", track, Bsecond);
- }
-#endif
- }
- char *old_one = nullptr;
-
- for (size_t track = 0; track < pc->GetTrackingData()->size(); track++)
- {
- const TrackList *A = pc->GetTrackingData();
- const TrackNumber B = A->at(track);
- size_t Bfirst = B.first;
- char *Bsecond = B.second;
- bool first_text = true;
-
- if ( old_one == nullptr
- || strcmp(old_one, Bsecond) != 0)
- {
- // first time this option
- if (old_one != nullptr)
- {
- add_text("&#010;");
- }
- old_one = Bsecond;
-
- if (first_text)
- {
- sprintf(tempText, "%s", Bsecond);
- add_text(tempText);
- add_text(": ");
- first_text = false;
- }
- }
- else
- {
- add_text(", ");
- }
- sprintf(tempText, "%zu", Bfirst);
- add_text(tempText);
- } // for (size_t track = 0; track < pc->GetTrackingData()->size(); track++)
-
- add_text("\"><font color=\"red\">M</font></a>");
- write_in_tracking = true;
- }
- } // loop over the whole chunk list
-
- if (tracking)
- {
- set_numbering(false);
- add_text("</pre>\n");
- add_text("</body>\n");
- add_text("</html>\n");
- }
-} // output_text
-
-
-static size_t cmt_parse_lead(const UncText &line, bool is_last)
-{
- size_t len = 0;
-
- while ( len < 32
- && len < line.size()) // TODO what is the meaning of 32?
- {
- if ( len > 0
- && line[len] == '/')
- {
- // ignore combined comments
- size_t tmp = len + 1;
-
- while ( tmp < line.size()
- && unc_isspace(line[tmp]))
- {
- tmp++;
- }
-
- if ( tmp < line.size()
- && line[tmp] == '/')
- {
- return(1);
- }
- break;
- }
- else if (strchr("*|\\#+", line[len]) == nullptr)
- {
- break; // none of the characters '*|\#+' found in line
- }
- len++;
- }
-
- if (len > 30) // TODO: what is the meaning of 30?
- {
- return(1);
- }
-
- if ( len > 0
- && ( len >= line.size()
- || unc_isspace(line[len])))
- {
- return(len);
- }
-
- if ( len == 1
- && line[0] == '*')
- {
- return(len);
- }
-
- if ( is_last
- && len > 0)
- {
- return(len);
- }
- return(0);
-} // cmt_parse_lead
-
-
-/**
- * Eat whitespace characters starting at the specified index in the forward or reverse direction
- * within a single line
- * @param str the input string containing the comment text
- * @param idx the starting index
- * @param forward if true, searches in the forward direction;
- * if false, searches in the reverse direction
- * @return the first index at which a non-whitespace character is encountered, including
- * a newline character
- */
-template<typename String>
-static int eat_line_whitespace(const String &str,
- int idx, bool
- forward = true)
-{
- auto advance_index = [&](int i)
- {
- return(forward ? i + 1 : i - 1);
- };
-
- auto index_in_range = [&](int i)
- {
- // TODO: the following BREAKS with source code formatting; uncrustify seems to
- // think that the following is a template. This will NEED to be fixed!!!
- // For now, reformulate the statement
- //return(forward ? i<int(str.size()) : i> = 0);
- return(forward ? (i < int(str.size())) : (i >= 0));
- };
-
- while ( index_in_range(idx)
- && str[idx] != '\n'
- && str[idx] != '\r'
- && unc_isspace(str[idx]))
- {
- idx = advance_index(idx);
- }
- return(idx);
-} // eat_line_whitespace
-
-
-/**
- * Returns whether or not a javaparam tag is the leading
- * text in a comment line, with only a sequence of whitespace
- * and/or '*' characters preceding it
- * @param str the input string containing the comment text
- * @param idx the starting index
- * @return true/false
- */
-template<typename String>
-static bool javaparam_tag_is_start_of_line(const String &str, int idx)
-{
- idx = eat_line_whitespace(str,
- str[idx] == '@' ? idx - 1 : idx,
- false);
-
- while (true)
- {
- if ( idx < 0
- || str[idx] == '\n'
- || str[idx] == '\r')
- {
- return(true);
- }
-
- if (str[idx] == '*')
- {
- idx = eat_line_whitespace(str,
- idx - 1,
- false);
- }
- else
- {
- return(false);
- }
- }
-} // javaparam_tag_is_start_of_line
-
-
-/**
- * Attempts to match a doxygen/javadoc-style comment tag
- * @param str the input string containing the comment text
- * @param idx the starting index
- * @return the index of the character immediately following the matched tag,
- * or -1 if no match is found
- */
-static int match_doxygen_javadoc_tag(const std::wstring &str, size_t idx)
-{
- std::wsmatch match;
-
- if (str[idx] == L'@')
- {
- // Issue #3357
- std::wregex criteria(L"(@(?:author|"
- L"deprecated|"
- L"exception|"
- L"param(?:\\s*?\\[\\s*(?:in\\s*,\\s*out|in|out)\\s*?\\])?|"
- L"return|"
- L"see|"
- L"since|"
- L"throws|"
- L"version)(?=\\s))");
-
- if ( std::regex_search(str.cbegin() + idx, str.cend(), match, criteria)
- && match[1].matched
- && match.position(1) == std::wsmatch::difference_type(0))
- {
- std::set<std::wstring> block_tags =
- {
- L"@author",
- L"@deprecated",
- L"@exception",
- L"@param",
- L"@param[in]",
- L"@param[in,out]",
- L"@param[out]",
- L"@return",
- L"@see",
- L"@since",
- L"@throws",
- L"@version"
- };
- std::wstring result(match[1]);
- result.erase(std::remove_if(result.begin(), result.end(), ::isspace), result.end());
- auto &&it_block_tag = block_tags.find(result);
-
- if ( it_block_tag != block_tags.end()
- && javaparam_tag_is_start_of_line(str, idx))
- {
- return(int(idx + match[1].length()));
- }
- }
- }
- return(-1);
-} // match_javadoc_block_tag
-
-
-static void calculate_doxygen_javadoc_indent_alignment(const std::wstring &str,
- size_t &doxygen_javadoc_param_name_indent,
- size_t &doxygen_javadoc_continuation_indent)
-{
- log_rule_B("cmt_align_doxygen_javadoc_tags");
-
- doxygen_javadoc_continuation_indent = 0;
- doxygen_javadoc_param_name_indent = 0;
-
- if (!options::cmt_align_doxygen_javadoc_tags())
- {
- return;
- }
-
- for (size_t idx = 0; idx < str.size(); ++idx)
- {
- int start_idx = idx;
- int end_idx = match_doxygen_javadoc_tag(str, start_idx);
-
- if (end_idx > start_idx)
- {
- size_t block_tag_width = 1 + std::count_if(str.begin() + start_idx,
- str.begin() + end_idx,
- [](wchar_t ch) {
- return(!unc_isspace(ch));
- });
-
- if (block_tag_width > doxygen_javadoc_param_name_indent)
- {
- doxygen_javadoc_param_name_indent = block_tag_width;
- }
- idx = eat_line_whitespace(str, end_idx);
-
- size_t param_name_width = 0;
-
- if (str.find(L"@param", start_idx) == size_t(start_idx))
- {
- param_name_width = 1;
-
- while (true)
- {
- while ( !unc_isspace(str[idx])
- && str[idx] != ',')
- {
- ++param_name_width;
- ++idx;
- }
- idx = eat_line_whitespace(str, idx);
-
- if (str[idx] != ',')
- {
- break;
- }
- param_name_width += 2;
- idx = eat_line_whitespace(str, idx + 1);
- }
- }
-
- if (param_name_width > doxygen_javadoc_continuation_indent)
- {
- doxygen_javadoc_continuation_indent = param_name_width;
- }
- }
- }
-
- if (doxygen_javadoc_param_name_indent > 0)
- {
- log_rule_B("cmt_sp_before_doxygen_javadoc_tags");
-
- doxygen_javadoc_param_name_indent += options::cmt_sp_before_doxygen_javadoc_tags();
- doxygen_javadoc_continuation_indent += doxygen_javadoc_param_name_indent;
- }
-} // calculate_doxygen_javadoc_indent_alignment
-
-
-static void calculate_comment_body_indent(cmt_reflow &cmt, const UncText &str)
-{
- cmt.xtra_indent = 0;
-
- log_rule_B("cmt_indent_multi");
-
- if (!options::cmt_indent_multi())
- {
- return;
- }
- size_t idx = 0;
- size_t len = str.size();
- size_t last_len = 0;
-
- log_rule_B("cmt_multi_check_last");
-
- if (options::cmt_multi_check_last())
- {
- // find the last line length
- for (idx = len - 1; idx > 0; idx--)
- {
- if ( str[idx] == '\n'
- || str[idx] == '\r')
- {
- idx++;
-
- while ( idx < len
- && ( str[idx] == ' '
- || str[idx] == '\t'))
- {
- idx++;
- }
- last_len = len - idx;
- break;
- }
- }
- }
- // find the first line length
- size_t first_len = 0;
-
- for (idx = 0; idx < len; idx++)
- {
- if ( str[idx] == '\n'
- || str[idx] == '\r')
- {
- first_len = idx;
-
- while ( str[first_len - 1] == ' '
- || str[first_len - 1] == '\t')
- {
- if (first_len == 0)
- {
- fprintf(stderr, "%s(%d): first_len is ZERO, cannot be decremented.\n",
- __func__, __LINE__);
- log_flush(true);
- exit(EX_SOFTWARE);
- }
- first_len--;
- }
-
- // handle DOS endings
- if ( str[idx] == '\r'
- && str[idx + 1] == '\n')
- {
- idx++;
- }
- idx++;
- break;
- }
- }
-
- // Scan the second line
- size_t width = 0;
-
- for ( ; idx < len - 1; idx++)
- {
- if ( str[idx] == ' '
- || str[idx] == '\t')
- {
- if (width > 0)
- {
- break;
- }
- continue;
- }
-
- if ( str[idx] == '\n'
- || str[idx] == '\r')
- {
- break; // Done with second line
- }
-
- // Count the leading chars
- if ( str[idx] == '*'
- || str[idx] == '|'
- || str[idx] == '\\'
- || str[idx] == '#'
- || str[idx] == '+')
- {
- width++;
- }
- else
- {
- if ( width != 1
- || str[idx - 1] != '*')
- {
- width = 0;
- }
- break;
- }
- }
-
- // LOG_FMT(LSYS, "%s: first=%d last=%d width=%d\n", __func__, first_len, last_len, width);
-
- /*
- * If the first and last line are the same length and don't contain any
- * alphanumeric chars and (the first line len > cmt_multi_first_len_minimum
- * or the second leader is the same as the first line length), then the
- * indent is 0.
- */
- log_rule_B("cmt_multi_first_len_minimum");
-
- if ( first_len == last_len
- && ( first_len > options::cmt_multi_first_len_minimum()
- || first_len == width))
- {
- return;
- }
- cmt.xtra_indent = (width == 2) ? 0 : 1;
-} // calculate_comment_body_indent
-
-
-// TODO: can we use search_next_chunk here?
-static Chunk *get_next_function(Chunk *pc)
-{
- while ((pc = pc->GetNext())->IsNotNullChunk())
- {
- if ( pc->Is(CT_FUNC_DEF)
- || pc->Is(CT_FUNC_PROTO)
- || pc->Is(CT_FUNC_CLASS_DEF)
- || pc->Is(CT_FUNC_CLASS_PROTO)
- || pc->Is(CT_OC_MSG_DECL))
- {
- return(pc);
- }
- }
- return(Chunk::NullChunkPtr);
-}
-
-
-static Chunk *get_next_class(Chunk *pc)
-{
- return(pc->GetNextType(CT_CLASS)->GetNext());
-}
-
-
-static Chunk *get_prev_category(Chunk *pc)
-{
- return(pc->GetPrevType(CT_OC_CATEGORY));
-}
-
-
-static Chunk *get_next_scope(Chunk *pc)
-{
- return(pc->GetNextType(CT_OC_SCOPE));
-}
-
-
-static Chunk *get_prev_oc_class(Chunk *pc)
-{
- return(pc->GetPrevType(CT_OC_CLASS));
-}
-
-
-static int next_up(const UncText &text, size_t idx, const UncText &tag)
-{
- size_t offs = 0;
-
- while ( idx < text.size()
- && unc_isspace(text[idx]))
- {
- idx++;
- offs++;
- }
-
- if (text.startswith(tag, idx))
- {
- return(offs);
- }
- return(-1);
-}
-
-
-static void add_comment_text(const UncText &text,
- cmt_reflow &cmt,
- bool esc_close,
- size_t continuation_indent)
-{
- bool was_star = false;
- bool was_slash = false;
- bool in_word = false;
- size_t len = text.size();
- size_t ch_cnt = 0; // chars since newline
-
- // If the '//' is included write it first else we may wrap an empty line
- size_t idx = 0;
-
- if (text.startswith("//"))
- {
- add_text("//");
- idx += 2;
-
- while (unc_isspace(text[idx]))
- {
- add_char(text[idx++]);
- }
- }
-
- for ( ; idx < len; idx++) // TODO: avoid modifying idx in loop
- {
- // Split the comment
- if (text[idx] == '\n')
- {
- in_word = false;
- add_char('\n');
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
-
- if (cmt.xtra_indent > 0)
- {
- add_char(' ');
- }
- // hack to get escaped newlines to align and not duplicate the leading '//'
- int tmp = next_up(text, idx + 1, "//");
-
- if (tmp < 0)
- {
- add_text(cmt.cont_text);
- }
- else
- {
- idx += tmp;
- }
- ch_cnt = 0;
- }
- else if ( cmt.reflow
- && text[idx] == ' '
- && options::cmt_width() > 0
- && ( cpd.column > options::cmt_width()
- || ( ch_cnt > 1
- && next_word_exceeds_limit(text, idx))))
- {
- log_rule_B("cmt_width");
- in_word = false;
- add_char('\n');
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
-
- if (cmt.xtra_indent > 0)
- {
- add_char(' ');
- }
- // The number of spaces to insert after the star on subsequent comment lines.
- log_rule_B("cmt_sp_after_star_cont");
-
- /**
- * calculate the output column
- */
- size_t column = options::cmt_sp_after_star_cont();
-
- if ( text[idx + 1] == 42 // this is star *
- && text[idx + 2] == 47) // this is /
- {
- LOG_FMT(LCONTTEXT, "%s(%d): we have a comment end\n",
- __func__, __LINE__);
-
- column += cmt.column;
- }
- else
- {
- add_text(cmt.cont_text);
-
- if (continuation_indent > 0)
- {
- if (options::cmt_align_doxygen_javadoc_tags())
- {
- log_rule_B("cmt_align_doxygen_javadoc_tags");
- }
- else if (options::cmt_reflow_indent_to_paragraph_start())
- {
- log_rule_B("cmt_reflow_indent_to_paragraph_start");
- }
- column += continuation_indent;
-
- log_rule_B("cmt_sp_after_star_cont");
-
- if (column >= options::cmt_sp_after_star_cont())
- {
- column -= options::cmt_sp_after_star_cont();
- }
- }
- /**
- * count the number trailing spaces in the comment continuation text
- */
- size_t num_trailing_sp = 0;
-
- while ( num_trailing_sp < cmt.cont_text.size()
- && unc_isspace(cmt.cont_text[cmt.cont_text.size() - 1 - num_trailing_sp]))
- {
- ++num_trailing_sp;
- }
- column += cpd.column;
-
- if (column >= num_trailing_sp)
- {
- column -= num_trailing_sp;
- }
- }
- output_to_column(column,
- false);
- ch_cnt = 0;
- }
- else
- {
- // Escape a C closure in a CPP comment
- if ( esc_close
- && ( ( was_star
- && text[idx] == '/')
- || ( was_slash
- && text[idx] == '*')))
- {
- add_char(' ');
- }
-
- if ( !in_word
- && !unc_isspace(text[idx]))
- {
- cmt.word_count++;
- }
- in_word = !unc_isspace(text[idx]);
-
- add_char(text[idx]);
- was_star = (text[idx] == '*');
- was_slash = (text[idx] == '/');
- ch_cnt++;
- }
- }
-} // add_comment_text
-
-
-static void output_cmt_start(cmt_reflow &cmt, Chunk *pc)
-{
- cmt.pc = pc;
- cmt.column = pc->GetColumn();
- cmt.brace_col = pc->GetColumnIndent();
- cmt.base_col = pc->GetColumnIndent();
- cmt.word_count = 0;
- cmt.xtra_indent = 0;
- cmt.cont_text.clear();
- cmt.reflow = false;
-
- // Issue #2752
- log_rule_B("cmt_insert_file_header");
- log_rule_B("cmt_insert_file_footer");
- log_rule_B("cmt_insert_func_header");
- log_rule_B("cmt_insert_class_header");
- log_rule_B("cmt_insert_oc_msg_header");
-
- if ( options::cmt_insert_file_header().size() > 0
- || options::cmt_insert_file_footer().size() > 0
- || options::cmt_insert_func_header().size() > 0
- || options::cmt_insert_class_header().size() > 0
- || options::cmt_insert_oc_msg_header().size() > 0)
- {
- LOG_FMT(LCONTTEXT, "%s(%d): cmt_insert_file\n", __func__, __LINE__);
- do_kw_subst(pc);
- }
- else
- {
- LOG_FMT(LCONTTEXT, "%s(%d): no cmt_insert_file\n", __func__, __LINE__);
- }
-
- if (cmt.brace_col == 0)
- {
- log_rule_B("output_tab_size");
- cmt.brace_col = 1 + (pc->GetBraceLevel() * options::output_tab_size());
- }
- // LOG_FMT(LSYS, "%s: line %zd, brace=%zd base=%zd col=%zd orig=%zd aligned=%x\n",
- // __func__, pc->GetOrigLine(), cmt.brace_col, cmt.base_col, cmt.column, pc->GetOrigCol(),
- // pc->GetFlags() & (PCF_WAS_ALIGNED | PCF_RIGHT_COMMENT));
-
- if ( pc->GetParentType() == CT_COMMENT_START
- || pc->GetParentType() == CT_COMMENT_WHOLE)
- {
- log_rule_B("indent_col1_comment");
-
- if ( !options::indent_col1_comment()
- && pc->GetOrigCol() == 1
- && !pc->TestFlags(PCF_INSERTED))
- {
- cmt.column = 1;
- cmt.base_col = 1;
- cmt.brace_col = 1;
- }
- }
- // tab aligning code
- log_rule_B("indent_cmt_with_tabs");
-
- if ( options::indent_cmt_with_tabs()
- && ( pc->GetParentType() == CT_COMMENT_END
- || pc->GetParentType() == CT_COMMENT_WHOLE))
- {
- cmt.column = align_tab_column(cmt.column - 1);
- // LOG_FMT(LSYS, "%s: line %d, orig:%d new:%d\n",
- // __func__, pc->GetOrigLine(), pc->GetColumn(), cmt.column);
- pc->SetColumn(cmt.column);
- }
- cmt.base_col = cmt.column;
-
- // LOG_FMT(LSYS, "%s: -- brace=%d base=%d col=%d\n",
- // __func__, cmt.brace_col, cmt.base_col, cmt.column);
-
- // Bump out to the column
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
-} // output_cmt_start
-
-
-static bool can_combine_comment(Chunk *pc, cmt_reflow &cmt)
-{
- // We can't combine if there is something other than a newline next
- if (pc->GetParentType() == CT_COMMENT_START)
- {
- return(false);
- }
-
- // next is a newline for sure, make sure it is a single newline
- if (pc == nullptr)
- {
- pc = Chunk::NullChunkPtr;
- }
- Chunk *next = pc->GetNext();
-
- if ( next->IsNotNullChunk()
- && next->GetNlCount() == 1)
- {
- // Make sure the comment is the same type at the same column
- next = next->GetNext();
-
- if ( next->Is(pc->GetType())
- && ( ( next->GetColumn() == 1
- && pc->GetColumn() == 1)
- || ( next->GetColumn() == cmt.base_col
- && pc->GetColumn() == cmt.base_col)
- || ( next->GetColumn() > cmt.base_col
- && pc->GetParentType() == CT_COMMENT_END)))
- {
- return(true);
- }
- }
- return(false);
-} // can_combine_comment
-
-
-static Chunk *output_comment_c(Chunk *first)
-{
- cmt_reflow cmt;
-
- output_cmt_start(cmt, first);
- log_rule_B("cmt_reflow_mode");
- cmt.reflow = (options::cmt_reflow_mode() != 1);
-
- // See if we can combine this comment with the next comment
- log_rule_B("cmt_c_group");
-
- if ( !options::cmt_c_group()
- || !can_combine_comment(first, cmt))
- {
- // Just add the single comment
- log_rule_B("cmt_star_cont");
- cmt.cont_text = options::cmt_star_cont() ? " * " : " ";
- LOG_CONTTEXT();
-
- bool replace_comment = ( options::cmt_trailing_single_line_c_to_cpp()
- && first->IsLastChunkOnLine()
- && first->Str().at(2) != '*');
-
- if ( replace_comment
- && first->TestFlags(PCF_IN_PREPROC))
- {
- // Do not replace a single line comment if we are inside a #define line
- if (first->GetPpStart()->GetParentType() == CT_PP_DEFINE)
- {
- replace_comment = false;
- }
- }
-
- if (replace_comment)
- {
- // Transform the comment to CPP and reuse the same logic (issue #4121)
- log_rule_B("cmt_trailing_single_line_c_to_cpp");
-
- UncText tmp(first->GetStr(), 0, first->Len() - 2);
- tmp.at(1) = '/'; // Change '/*' to '//'
- cmt_trim_whitespace(tmp, false);
- first->Str() = tmp;
-
- output_comment_cpp(first);
- }
- else
- {
- add_comment_text(first->GetStr(), cmt, false);
- }
- return(first);
- }
- log_rule_B("cmt_star_cont");
- cmt.cont_text = options::cmt_star_cont() ? " *" : " ";
- LOG_CONTTEXT();
-
- add_text("/*");
-
- log_rule_B("cmt_c_nl_start");
-
- if (options::cmt_c_nl_start())
- {
- add_comment_text("\n", cmt, false);
- }
- Chunk *pc = first;
- UncText tmp;
-
- while (can_combine_comment(pc, cmt))
- {
- LOG_FMT(LCONTTEXT, "%s(%d): Text() is '%s'\n",
- __func__, __LINE__, pc->Text());
- tmp.set(pc->GetStr(), 2, pc->Len() - 4);
-
- if ( cpd.last_char == '*'
- && tmp[0] != ' ') // Issue #1908
- {
- LOG_FMT(LCONTTEXT, "%s(%d): add_text a " "\n", __func__, __LINE__);
- add_text(" ");
- }
- // In case of reflow, original comment could contain trailing spaces before closing the comment, we don't need them after reflow
- LOG_FMT(LCONTTEXT, "%s(%d): trim\n", __func__, __LINE__);
- cmt_trim_whitespace(tmp, false);
- LOG_FMT(LCONTTEXT, "%s(%d): add_comment_text(tmp is '%s')\n",
- __func__, __LINE__, tmp.c_str());
- add_comment_text(tmp, cmt, false);
- LOG_FMT(LCONTTEXT, "%s(%d): add_comment_text(newline)\n",
- __func__, __LINE__);
- add_comment_text("\n", cmt, false);
- pc = pc->GetNext();
- pc = pc->GetNext();
- }
- tmp.set(pc->GetStr(), 2, pc->Len() - 4);
-
- if ( cpd.last_char == '*'
- && tmp[0] == '/')
- {
- add_text(" ");
- }
- // In case of reflow, original comment could contain trailing spaces before closing the comment, we don't need them after reflow
- cmt_trim_whitespace(tmp, false);
- add_comment_text(tmp, cmt, false);
-
- log_rule_B("cmt_c_nl_end");
-
- if (options::cmt_c_nl_end())
- {
- cmt.cont_text = " ";
- LOG_CONTTEXT();
- add_comment_text("\n", cmt, false);
- }
- add_comment_text("*/", cmt, false);
- return(pc);
-} // output_comment_c
-
-
-static Chunk *output_comment_cpp(Chunk *first)
-{
- cmt_reflow cmt;
-
- output_cmt_start(cmt, first);
- log_rule_B("cmt_reflow_mode");
- cmt.reflow = (options::cmt_reflow_mode() != 1);
-
- UncText leadin = "//"; // default setting to keep previous behaviour
-
- // If true, space is added with sp_cmt_cpp_start will be added after doxygen
- // sequences like '///', '///<', '//!' and '//!<'.
- log_rule_B("sp_cmt_cpp_doxygen");
-
- if (options::sp_cmt_cpp_doxygen()) // special treatment for doxygen style comments (treat as unity)
- {
- const char *sComment = first->Text();
- bool grouping = (sComment[2] == '@');
- size_t brace = 3;
-
- if ( sComment[2] == '/'
- || sComment[2] == '!') // doxygen style found!
- {
- leadin += sComment[2]; // at least one additional char (either "///" or "//!")
-
- if (sComment[3] == '<') // and a further one (either "///<" or "//!<")
- {
- leadin += '<';
- }
- else
- {
- grouping = (sComment[3] == '@'); // or a further one (grouping)
- brace = 4;
- }
- }
-
- if ( grouping
- && ( sComment[brace] == '{'
- || sComment[brace] == '}'))
- {
- leadin += '@';
- leadin += sComment[brace];
- }
- }
- // Special treatment for Qt translator or meta-data comments (treat as unity)
- // If true, space is added with sp_cmt_cpp_start will be added after Qt
- // translator or meta-data comments like '//:', '//=', and '//~'.
- log_rule_B("sp_cmt_cpp_qttr");
-
- if (options::sp_cmt_cpp_qttr())
- {
- const int c = first->GetStr()[2];
-
- if ( c == ':'
- || c == '='
- || c == '~')
- {
- leadin += c;
- }
- }
- // CPP comments can't be grouped unless they are converted to C comments
- log_rule_B("cmt_cpp_to_c");
-
- if (!options::cmt_cpp_to_c())
- {
- auto const *cmt_text = first->GetStr().c_str() + 2;
- // Add or remove space after the opening of a C++ comment,
- // i.e. '// A' vs. '//A'.
- auto *sp_cmt = &options::sp_cmt_cpp_start;
-
- cmt.cont_text = leadin;
-
- // Get start of comment text
- while ( *cmt_text != '\0'
- && unc_isspace(*cmt_text))
- {
- ++cmt_text;
- }
-
- // Determine if we are dealing with a region marker
- if ( ( first->GetPrev()->IsNullChunk()
- || first->GetPrev()->GetOrigLine() != first->GetOrigLine())
- && ( strncmp(cmt_text, "BEGIN", 5) == 0
- || strncmp(cmt_text, "END", 3) == 0))
- {
- // If sp_cmt_cpp_region is not ignore, use that instead of
- // sp_cmt_cpp_start
- if (options::sp_cmt_cpp_region() != IARF_IGNORE)
- {
- sp_cmt = &options::sp_cmt_cpp_region;
- }
- }
- // Add or remove space after the opening of a C++ comment,
- // i.e. '// A' vs. '//A'.
- log_rule_B(sp_cmt->name());
-
- if ((*sp_cmt)() != IARF_REMOVE)
- {
- cmt.cont_text += ' ';
- }
- LOG_CONTTEXT();
-
- // Add or remove space after the opening of a C++ comment,
- // i.e. '// A' vs. '//A'.
- log_rule_B(sp_cmt->name());
-
- if ((*sp_cmt)() == IARF_IGNORE)
- {
- add_comment_text(first->GetStr(), cmt, false);
- }
- else
- {
- size_t iLISz = leadin.size();
- UncText tmp(first->GetStr(), 0, iLISz);
- add_comment_text(tmp, cmt, false);
-
- tmp.set(first->GetStr(), iLISz, first->Len() - iLISz);
-
- // Add or remove space after the opening of a C++ comment,
- // i.e. '// A' vs. '//A'.
- log_rule_B("sp_cmt_cpp_start");
-
- if ((*sp_cmt)() & IARF_REMOVE)
- {
- while ( (tmp.size() > 0)
- && unc_isspace(tmp[0]))
- {
- tmp.pop_front();
- }
- }
-
- if (tmp.size() > 0)
- {
- // Add or remove space after the opening of a C++ comment,
- // i.e. '// A' vs. '//A'.
- log_rule_B("sp_cmt_cpp_start");
-
- if ((*sp_cmt)() & IARF_ADD)
- {
- if ( !unc_isspace(tmp[0])
- && (tmp[0] != '/'))
- {
- // only with sp_cmt_cpp_start set to 'add' or 'force'
- bool sp_cmt_pvs = options::sp_cmt_cpp_pvs(); // Issue #3919
- bool sp_cmt_lint = options::sp_cmt_cpp_lint(); // Issue #3614
- UncText temp = first->GetStr();
- int PVS = temp.find("//-V");
- int LINT = temp.find("//lint");
-
- // @return == -1 if not found
- // @return >= 0 the position
- if ( ( PVS == 0
- && sp_cmt_pvs)
- || ( LINT == 0
- && sp_cmt_lint))
- {
- // do not include a space
- }
- else
- {
- add_comment_text(" ", cmt, false);
- }
- }
- }
- add_comment_text(tmp, cmt, false);
- }
- }
- return(first);
- }
- // We are going to convert the CPP comments to C comments
- log_rule_B("cmt_star_cont");
- cmt.cont_text = options::cmt_star_cont() ? " * " : " ";
- LOG_CONTTEXT();
-
- UncText tmp;
-
- // See if we can combine this comment with the next comment
- log_rule_B("cmt_cpp_group");
-
- if ( !options::cmt_cpp_group()
- || !can_combine_comment(first, cmt))
- {
- // nothing to group: just output a single line
- add_text("/*");
-
- // patch # 32, 2012-03-23
- // Add or remove space after the opening of a C++ comment,
- // i.e. '// A' vs. '//A'.
- log_rule_B("sp_cmt_cpp_start");
-
- if ( !unc_isspace(first->GetStr()[2])
- && (options::sp_cmt_cpp_start() & IARF_ADD))
- {
- add_char(' ');
- }
- tmp.set(first->GetStr(), 2, first->Len() - 2);
- add_comment_text(tmp, cmt, true);
- add_text(" */");
- return(first);
- }
- add_text("/*");
-
- log_rule_B("cmt_cpp_nl_start");
-
- if (options::cmt_cpp_nl_start())
- {
- add_comment_text("\n", cmt, false);
- }
- else
- {
- add_text(" ");
- }
- Chunk *pc = first;
- int offs;
-
- while (can_combine_comment(pc, cmt))
- {
- offs = unc_isspace(pc->GetStr()[2]) ? 1 : 0;
- tmp.set(pc->GetStr(), 2 + offs, pc->Len() - (2 + offs));
-
- if ( cpd.last_char == '*'
- && tmp[0] == '/')
- {
- add_text(" ");
- }
- add_comment_text(tmp, cmt, true);
- add_comment_text("\n", cmt, false);
- pc = pc->GetNext()->GetNext();
- }
- offs = unc_isspace(pc->GetStr()[2]) ? 1 : 0;
- tmp.set(pc->GetStr(), 2 + offs, pc->Len() - (2 + offs));
- add_comment_text(tmp, cmt, true);
-
- log_rule_B("cmt_cpp_nl_end");
-
- if (options::cmt_cpp_nl_end())
- {
- cmt.cont_text = "";
- LOG_CONTTEXT();
- add_comment_text("\n", cmt, false);
- }
- add_comment_text(" */", cmt, false);
- return(pc);
-} // output_comment_cpp
-
-
-static void cmt_trim_whitespace(UncText &line, bool in_preproc)
-{
- // Remove trailing whitespace on the line
- while ( line.size() > 0
- && ( line.back() == ' '
- || line.back() == '\t'))
- {
- line.pop_back();
- }
-
- // Shift back to the comment text, ...
- if ( in_preproc // if in a preproc ...
- && line.size() > 1 // with a line that holds ...
- && line.back() == '\\') // a backslash-newline ...
- {
- bool do_space = false;
-
- // If there was any space before the backslash, change it to 1 space
- line.pop_back();
-
- while ( line.size() > 0
- && ( line.back() == ' '
- || line.back() == '\t'))
- {
- do_space = true;
- line.pop_back();
- }
-
- if (do_space)
- {
- line.append(' ');
- }
- line.append('\\');
- }
-} // cmt_trim_whitespace
-
-
-/**
- * Return an indexed-map of reflow fold end of line/beginning of line regex pairs read
- * from file
- */
-static std::map<std::size_t, std::pair<std::wregex, std::wregex> > get_reflow_fold_regex_map()
-{
- /**
- * TODO: should the following be static to prevent initializing it multiple times?
- */
- static std::map<std::size_t, std::pair<std::wregex, std::wregex> > regex_map;
-
- if (regex_map.empty())
- {
- if (!options::cmt_reflow_fold_regex_file().empty())
- {
- std::wstring raw_wstring(cpd.reflow_fold_regex.raw.begin(),
- cpd.reflow_fold_regex.raw.end());
-
- std::wregex criteria(L"\\s*(?:(?:(beg_of_next)|(end_of_prev))_line_regex)"
- "\\s*\\[\\s*([0-9]+)\\s*\\]\\s*=\\s*\"(.*)\"\\s*"
- "(?=\\r\\n|\\r|\\n|$)");
- std::wsregex_iterator it_regex(raw_wstring.cbegin(), raw_wstring.cend(), criteria);
- std::wsregex_iterator it_regex_end = std::wsregex_iterator();
-
- while (it_regex != it_regex_end)
- {
- std::wsmatch match = *it_regex;
-
- if ( (( match[1].matched
- || match[2].matched))
- && match[3].matched
- && match[4].matched)
- {
- auto &&index = std::stoi(match[3].str());
- std::wregex *p_wregex = match[1].matched ? &regex_map[index].second
- : &regex_map[index].first;
- *p_wregex = match[4].str();
- }
- ++it_regex;
- }
- }
- else
- {
- regex_map.emplace(0L, std::make_pair((std::wregex)L"[\\w,\\]\\)]$", (std::wregex)L"^[\\w,\\[\\(]"));
- regex_map.emplace(1L, std::make_pair((std::wregex)L"\\.$", (std::wregex)L"^[A-Z]"));
- }
- }
- return(regex_map);
-} // get_reflow_fold_regex_map
-
-
-static void output_comment_multi(Chunk *pc)
-{
- if (pc == nullptr)
- {
- return;
- }
- cmt_reflow cmt;
-
- char copy[1000];
-
- LOG_FMT(LCONTTEXT, "%s(%d): Text() is '%s', type is %s, orig col is %zu, column is %zu\n",
- __func__, __LINE__, pc->ElidedText(copy), get_token_name(pc->GetType()), pc->GetOrigCol(), pc->GetColumn());
-
- output_cmt_start(cmt, pc);
- log_rule_B("cmt_reflow_mode");
- cmt.reflow = (options::cmt_reflow_mode() != 1);
-
- size_t cmt_col = cmt.base_col;
- int col_diff = pc->GetOrigCol() - cmt.base_col;
-
- calculate_comment_body_indent(cmt, pc->GetStr());
-
- log_rule_B("cmt_indent_multi");
- log_rule_B("cmt_star_cont");
- cmt.cont_text = !options::cmt_indent_multi() ? "" :
- (options::cmt_star_cont() ? "* " : " ");
- LOG_CONTTEXT();
-
- std::wstring pc_wstring(pc->GetStr().get().cbegin(),
- pc->GetStr().get().cend());
-
- size_t doxygen_javadoc_param_name_indent = 0;
- size_t doxygen_javadoc_continuation_indent = 0;
- size_t reflow_paragraph_continuation_indent = 0;
-
- calculate_doxygen_javadoc_indent_alignment(pc_wstring,
- doxygen_javadoc_param_name_indent,
- doxygen_javadoc_continuation_indent);
-
- size_t line_count = 0;
- size_t ccol = pc->GetColumn(); // the col of subsequent comment lines
- size_t cmt_idx = 0;
- bool nl_end = false;
- bool doxygen_javadoc_indent_align = false;
- UncText line;
-
- /*
- * Get a map of regex pairs that define expressions to match at both the end
- * of the previous line and the beginning of the next line
- */
- auto &&cmt_reflow_regex_map = get_reflow_fold_regex_map();
-
- line.clear();
- LOG_FMT(LCONTTEXT, "%s(%d): pc->Len() is %zu\n",
- __func__, __LINE__, pc->Len());
- //LOG_FMT(LCONTTEXT, "%s(%d): pc->str is %s\n",
- // __func__, __LINE__, pc->str.c_str());
-
- /**
- * check for enable/disable processing comment strings that may
- * both be embedded within the same multi-line comment
- */
- auto disable_processing_cmt_idx = find_disable_processing_comment_marker(pc->GetStr());
- auto enable_processing_cmt_idx = find_enable_processing_comment_marker(pc->GetStr());
-
- while (cmt_idx < pc->Len())
- {
- int ch = pc->GetStr()[cmt_idx];
- cmt_idx++;
-
- if ( cmt_idx > std::size_t(disable_processing_cmt_idx)
- && enable_processing_cmt_idx > disable_processing_cmt_idx)
- {
- auto length = enable_processing_cmt_idx - disable_processing_cmt_idx;
- UncText verbatim_text(pc->GetStr(),
- disable_processing_cmt_idx,
- length);
-
- add_text(verbatim_text);
-
- cmt_idx = enable_processing_cmt_idx;
-
- /**
- * check for additional enable/disable processing comment strings that may
- * both be embedded within the same multi-line comment
- */
- disable_processing_cmt_idx = find_disable_processing_comment_marker(pc->GetStr(),
- enable_processing_cmt_idx);
- enable_processing_cmt_idx = find_enable_processing_comment_marker(pc->GetStr(),
- enable_processing_cmt_idx);
-
- /**
- * it's probably necessary to reset the line count to prevent line
- * continuation characters from being added to the end of the current line
- */
- line_count = 0;
- }
-
- // handle the CRLF and CR endings. convert both to LF
- if (ch == '\r')
- {
- ch = '\n';
-
- if ( cmt_idx < pc->Len()
- && pc->GetStr()[cmt_idx] == '\n')
- {
- cmt_idx++;
- }
- }
-
- // Find the start column
- if (line.size() == 0)
- {
- nl_end = false;
-
- if (ch == ' ')
- {
- ccol++;
- continue;
- }
- else if (ch == '\t')
- {
- log_rule_B("input_tab_size");
- ccol = calc_next_tab_column(ccol, options::input_tab_size());
- continue;
- }
- else
- {
- LOG_FMT(LCONTTEXT, "%s(%d):ch is %d, %c\n", __func__, __LINE__, ch, char(ch));
- }
- }
-
- if ( ch == '@'
- && options::cmt_align_doxygen_javadoc_tags())
- {
- int start_idx = cmt_idx - 1;
- int end_idx = match_doxygen_javadoc_tag(pc_wstring, start_idx);
-
- if (end_idx > start_idx)
- {
- doxygen_javadoc_indent_align = true;
-
- std::string match(pc->GetStr().get().cbegin() + start_idx,
- pc->GetStr().get().cbegin() + end_idx);
-
- match.erase(std::remove_if(match.begin(),
- match.end(),
- ::isspace),
- match.end());
-
- /**
- * remove whitespace before the '@'
- */
- int line_size_before_indent = line.size();
-
- while ( line_size_before_indent > 0
- && unc_isspace(line.back()))
- {
- line.pop_back();
- --line_size_before_indent;
- }
- log_rule_B("cmt_sp_before_doxygen_javadoc_tags");
-
- int indent = options::cmt_sp_before_doxygen_javadoc_tags();
-
- while (indent-- > 0)
- {
- line.append(' ');
- }
- cmt_idx += (end_idx - start_idx);
- line.append(match.c_str());
-
- bool is_exception_tag = match.find("@exception") != std::string::npos;
- bool is_param_tag = match.find("@param") != std::string::npos;
- bool is_throws_tag = match.find("@throws") != std::string::npos;
-
- if ( is_exception_tag
- || is_param_tag
- || is_throws_tag)
- {
- indent = int(doxygen_javadoc_param_name_indent) - int(line.size());
-
- while (indent-- > -line_size_before_indent)
- {
- line.append(' ');
- }
-
- while (true)
- {
- cmt_idx = eat_line_whitespace(pc->GetStr(),
- cmt_idx);
-
- while ( cmt_idx < pc->Len()
- && !unc_isspace(pc->GetStr()[cmt_idx])
- && pc->GetStr()[cmt_idx] != ',')
- {
- line.append(pc->Str()[cmt_idx++]);
- }
-
- if (!is_param_tag)
- {
- break;
- }
- /**
- * check for the possibility that comma-separated parameter names are present
- */
- cmt_idx = eat_line_whitespace(pc->GetStr(),
- cmt_idx);
-
- if (pc->GetStr()[cmt_idx] != ',')
- {
- break;
- }
- ++cmt_idx;
- line.append(", ");
- }
- }
- cmt_idx = eat_line_whitespace(pc->GetStr(),
- cmt_idx);
- indent = int(doxygen_javadoc_continuation_indent) - int(line.size());
-
- while (indent-- > -line_size_before_indent)
- {
- line.append(' ');
- }
-
- while ( cmt_idx < pc->Len()
- && !unc_isspace(pc->GetStr()[cmt_idx]))
- {
- line.append(pc->Str()[cmt_idx++]);
- }
- continue;
- }
- }
- /*
- * Now see if we need/must fold the next line with the current to enable
- * full reflow
- */
- log_rule_B("cmt_reflow_mode");
-
- if ( options::cmt_reflow_mode() == 2
- && ch == '\n'
- && cmt_idx < pc->Len())
- {
- int next_nonempty_line = -1;
- int prev_nonempty_line = -1;
- size_t nwidx = line.size();
-
- // strip trailing whitespace from the line collected so far
- while (nwidx > 0)
- {
- nwidx--;
-
- if ( prev_nonempty_line < 0
- && !unc_isspace(line[nwidx])
- && line[nwidx] != '*' // block comment: skip '*' at end of line
- && (pc->TestFlags(PCF_IN_PREPROC)
- ? ( line[nwidx] != '\\'
- || ( line[nwidx + 1] != '\r'
- && line[nwidx + 1] != '\n'))
- : true))
- {
- prev_nonempty_line = nwidx; // last non-whitespace char in the previous line
- }
- }
-
- for (size_t nxt_idx = cmt_idx;
- ( nxt_idx < pc->Len()
- && pc->GetStr()[nxt_idx] != '\r'
- && pc->GetStr()[nxt_idx] != '\n');
- nxt_idx++)
- {
- if ( next_nonempty_line < 0
- && !unc_isspace(pc->GetStr()[nxt_idx])
- && pc->GetStr()[nxt_idx] != '*'
- && (pc->TestFlags(PCF_IN_PREPROC)
- ? ( pc->GetStr()[nxt_idx] != '\\'
- || ( pc->GetStr()[nxt_idx + 1] != '\r'
- && pc->GetStr()[nxt_idx + 1] != '\n'))
- : true))
- {
- next_nonempty_line = nxt_idx; // first non-whitespace char in the next line
- }
- }
-
- if ( options::cmt_reflow_indent_to_paragraph_start()
- && next_nonempty_line >= 0
- && ( prev_nonempty_line <= 0
- || doxygen_javadoc_indent_align))
- {
- log_rule_B("cmt_reflow_indent_to_paragraph_start");
-
- int cmt_star_indent = 0;
-
- while ( next_nonempty_line > cmt_star_indent
- && pc->GetStr()[next_nonempty_line - cmt_star_indent - 1] != '*')
- {
- ++cmt_star_indent;
- }
- reflow_paragraph_continuation_indent = size_t(cmt_star_indent);
- }
-
- /*
- * see if we should fold up; usually that'd be a YES, but there are a few
- * situations where folding/reflowing by merging lines is frowned upon:
- *
- * - ASCII art in the comments (most often, these are drawings done in +-\/|.,*)
- *
- * - Doxygen/JavaDoc/etc. parameters: these often start with \ or @, at least
- * something clearly non-alphanumeric (you see where we're going with this?)
- *
- * - bullet lists that are closely spaced: bullets are always non-alphanumeric
- * characters, such as '-' or '+' (or, oh horror, '*' - that's bloody ambiguous
- * to parse :-( ... with or without '*' comment start prefix, that's the
- * question, then.)
- *
- * - semi-HTML formatted code, e.g. <pre>...</pre> comment sections (NDoc, etc.)
- *
- * - New lines which form a new paragraph without there having been added an
- * extra empty line between the last sentence and the new one.
- * A bit like this, really; so it is opportune to check if the last line ended
- * in a terminal (that would be the set '.:;!?') and the new line starts with
- * a capital.
- * Though new lines starting with comment delimiters, such as '(', should be
- * pulled up.
- *
- * So it bores down to this: the only folding (& reflowing) that's going to happen
- * is when the next line starts with an alphanumeric character AND the last
- * line didn't end with an non-alphanumeric character, except: ',' AND the next
- * line didn't start with a '*' all of a sudden while the previous one didn't
- * (the ambiguous '*'-for-bullet case!)
- */
- if ( prev_nonempty_line >= 0
- && next_nonempty_line >= int(cmt_idx))
- {
- std::wstring prev_line(line.get().cbegin(),
- line.get().cend());
- std::wstring next_line(pc->GetStr().get().cbegin() + next_nonempty_line,
- pc->GetStr().get().cend());
-
- for (auto &&cmt_reflow_regex_map_entry : cmt_reflow_regex_map)
- {
- auto &&cmt_reflow_regex_pair = cmt_reflow_regex_map_entry.second;
- auto &&end_of_prev_line_regex = cmt_reflow_regex_pair.first;
- auto &&beg_of_next_line_regex = cmt_reflow_regex_pair.second;
- std::wsmatch match[2];
-
- if ( std::regex_search(prev_line, match[0], end_of_prev_line_regex)
- && match[0].position(0) + match[0].length(0) == std::wsmatch::difference_type(line.size())
- && std::regex_search(next_line, match[1], beg_of_next_line_regex)
- && match[1].position(0) == 0)
- {
- // rewind the line to the last non-alpha:
- line.resize(prev_nonempty_line + 1);
-
- // roll the current line forward to the first non-alpha:
- cmt_idx = next_nonempty_line;
- // override the NL and make it a single whitespace:
- ch = ' ';
-
- break;
- }
- }
- }
- }
-
- if (ch == '\n')
- {
- LOG_FMT(LCONTTEXT, "%s(%d):ch is newline\n", __func__, __LINE__);
- }
- else
- {
- LOG_FMT(LCONTTEXT, "%s(%d):ch is %d, %c\n", __func__, __LINE__, ch, char(ch));
- }
- line.append(ch);
-
- // If we just hit an end of line OR we just hit end-of-comment...
- if ( ch == '\n'
- || cmt_idx == pc->Len())
- {
- if (ch == '\n')
- {
- LOG_FMT(LCONTTEXT, "%s(%d):ch is newline\n", __func__, __LINE__);
- }
- else
- {
- LOG_FMT(LCONTTEXT, "%s(%d):ch is %d, %c\n", __func__, __LINE__, ch, char(ch));
- }
- line_count++;
- LOG_FMT(LCONTTEXT, "%s(%d):line_count is %zu\n", __func__, __LINE__, line_count);
-
- // strip trailing tabs and spaces before the newline
- if (ch == '\n')
- {
- nl_end = true;
- line.pop_back();
- cmt_trim_whitespace(line, pc->TestFlags(PCF_IN_PREPROC));
- }
-
- if (line_count == 1)
- {
- // this is the first line - add unchanged
- add_comment_text(line, cmt, false);
-
- if (nl_end)
- {
- add_char('\n');
- }
- }
- else
- {
- /*
- * This is not the first line, so we need to indent to the
- * correct column. Each line is indented 0 or more spaces.
- */
- // Ensure ccol is not negative
- if (static_cast<int>(ccol) >= col_diff)
- {
- ccol -= col_diff;
- }
-
- if (ccol < (cmt_col + 3))
- {
- ccol = cmt_col + 3;
- }
-
- if (line.size() == 0)
- {
- // Empty line - just a '\n'
- log_rule_B("cmt_star_cont");
-
- if (options::cmt_star_cont())
- {
- // The number of spaces to insert at the start of subsequent comment lines.
- log_rule_B("cmt_sp_before_star_cont");
- cmt.column = cmt_col + options::cmt_sp_before_star_cont();
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
-
- if (cmt.xtra_indent > 0)
- {
- add_char(' ');
- }
- // multiline comments can have empty lines with some spaces in them for alignment
- // while adding * symbol and aligning them we don't want to keep these trailing spaces
- UncText tmp = UncText(cmt.cont_text);
- cmt_trim_whitespace(tmp, false);
- add_text(tmp);
- }
- add_char('\n');
- }
- else
- {
- /*
- * If this doesn't start with a '*' or '|'.
- * '\name' is a common parameter documentation thing.
- */
- log_rule_B("cmt_indent_multi");
-
- if ( options::cmt_indent_multi()
- && line[0] != '*'
- && line[0] != '|'
- && line[0] != '#'
- && ( line[0] != '\\'
- || unc_isalpha(line[1]))
- && line[0] != '+')
- {
- // The number of spaces to insert at the start of subsequent comment lines.
- log_rule_B("cmt_sp_before_star_cont");
- size_t start_col = cmt_col + options::cmt_sp_before_star_cont();
-
- log_rule_B("cmt_star_cont");
-
- if (options::cmt_star_cont())
- {
- cmt.column = start_col;
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
-
- if (cmt.xtra_indent > 0)
- {
- add_char(' ');
- }
- add_text(cmt.cont_text);
- // The number of spaces to insert after the star on subsequent comment lines.
- log_rule_B("cmt_sp_after_star_cont");
- output_to_column(ccol + options::cmt_sp_after_star_cont(), false);
- }
- else
- {
- cmt.column = ccol;
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
- }
- }
- else
- {
- // The number of spaces to insert at the start of subsequent comment lines.
- log_rule_B("cmt_sp_before_star_cont");
- cmt.column = cmt_col + options::cmt_sp_before_star_cont();
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
-
- if (cmt.xtra_indent > 0)
- {
- add_char(' ');
- }
- size_t idx;
-
- // Checks for and updates the lead chars.
- // @return 0=not present, >0=number of chars that are part of the lead
- idx = cmt_parse_lead(line, (cmt_idx == pc->Len()));
-
- if (idx > 0)
- {
- // >0=number of chars that are part of the lead
- cmt.cont_text.set(line, 0, idx);
- LOG_CONTTEXT();
-
- if ( (line.size() >= 2)
- && (line[0] == '*')
- && unc_isalnum(line[1]))
- {
- line.insert(1, ' ');
- }
- }
- else
- {
- // bug #653
- if (language_is_set(LANG_D))
- {
- // 0=no lead char present
- add_text(cmt.cont_text);
- }
- }
- }
- size_t continuation_indent = 0;
-
- if (doxygen_javadoc_indent_align)
- {
- continuation_indent = doxygen_javadoc_continuation_indent;
- }
- else if (reflow_paragraph_continuation_indent > 0)
- {
- continuation_indent = reflow_paragraph_continuation_indent;
- }
- add_comment_text(line,
- cmt,
- false,
- continuation_indent);
-
- if (nl_end)
- {
- add_text("\n");
- }
- }
- }
- line.clear();
- doxygen_javadoc_indent_align = false;
- ccol = 1;
- }
- }
-} // output_comment_multi
-
-
-static bool kw_fcn_filename(Chunk *cmt, UncText &out_txt)
-{
- UNUSED(cmt);
- out_txt.append(path_basename(cpd.filename.c_str()));
- return(true);
-}
-
-
-static bool kw_fcn_class(Chunk *cmt, UncText &out_txt)
-{
- Chunk *tmp = Chunk::NullChunkPtr;
-
- if (language_is_set(LANG_CPP | LANG_OC))
- {
- Chunk *fcn = get_next_function(cmt);
-
- if (fcn->Is(CT_OC_MSG_DECL))
- {
- tmp = get_prev_oc_class(cmt);
- }
- else
- {
- tmp = get_next_class(cmt);
- }
- }
- else if (language_is_set(LANG_OC))
- {
- tmp = get_prev_oc_class(cmt);
- }
-
- if (tmp->IsNullChunk())
- {
- tmp = get_next_class(cmt);
- }
-
- if (tmp->IsNotNullChunk())
- {
- out_txt.append(tmp->GetStr());
-
- while ((tmp = tmp->GetNext())->IsNotNullChunk())
- {
- if (tmp->IsNot(CT_DC_MEMBER))
- {
- break;
- }
- tmp = tmp->GetNext();
-
- if (tmp->IsNotNullChunk())
- {
- out_txt.append("::");
- out_txt.append(tmp->GetStr());
- }
- }
- return(true);
- }
- return(false);
-} // kw_fcn_class
-
-
-static bool kw_fcn_message(Chunk *cmt, UncText &out_txt)
-{
- Chunk *fcn = get_next_function(cmt);
-
- if (fcn->IsNullChunk())
- {
- return(false);
- }
- out_txt.append(fcn->GetStr());
-
- Chunk *tmp = fcn->GetNextNcNnl();
- Chunk *word = Chunk::NullChunkPtr;
-
- while (tmp->IsNotNullChunk())
- {
- if ( tmp->Is(CT_BRACE_OPEN)
- || tmp->Is(CT_SEMICOLON))
- {
- break;
- }
-
- if (tmp->Is(CT_OC_COLON))
- {
- if (word->IsNotNullChunk())
- {
- out_txt.append(word->GetStr());
- word = Chunk::NullChunkPtr;
- }
- out_txt.append(":");
- }
-
- if (tmp->Is(CT_WORD))
- {
- word = tmp;
- }
- tmp = tmp->GetNextNcNnl();
- }
- return(true);
-} // kw_fcn_message
-
-
-static bool kw_fcn_category(Chunk *cmt, UncText &out_txt)
-{
- Chunk *category = get_prev_category(cmt);
-
- if (category->IsNotNullChunk())
- {
- out_txt.append('(');
- out_txt.append(category->GetStr());
- out_txt.append(')');
- }
- return(true);
-} // kw_fcn_category
-
-
-static bool kw_fcn_scope(Chunk *cmt, UncText &out_txt)
-{
- Chunk *scope = get_next_scope(cmt);
-
- if (scope->IsNotNullChunk())
- {
- out_txt.append(scope->GetStr());
- return(true);
- }
- return(false);
-} // kw_fcn_scope
-
-
-static bool kw_fcn_function(Chunk *cmt, UncText &out_txt)
-{
- Chunk *fcn = get_next_function(cmt);
-
- if (fcn->IsNotNullChunk())
- {
- if (fcn->GetParentType() == CT_OPERATOR)
- {
- out_txt.append("operator ");
- }
-
- if (fcn->GetPrev()->GetType() == CT_DESTRUCTOR)
- {
- out_txt.append('~');
- }
- out_txt.append(fcn->GetStr());
- return(true);
- }
- return(false);
-}
-
-
-static bool kw_fcn_javaparam(Chunk *cmt, UncText &out_txt)
-{
- Chunk *fcn = get_next_function(cmt);
-
- if (fcn->IsNullChunk())
- {
- return(false);
- }
- Chunk *fpo;
- Chunk *fpc;
- bool has_param = true;
- bool need_nl = false;
-
- if (fcn->Is(CT_OC_MSG_DECL))
- {
- Chunk *tmp = fcn->GetNextNcNnl();
- has_param = false;
-
- while (tmp->IsNotNullChunk())
- {
- if ( tmp->Is(CT_BRACE_OPEN)
- || tmp->Is(CT_SEMICOLON))
- {
- break;
- }
-
- if (has_param)
- {
- if (need_nl)
- {
- out_txt.append("\n");
- }
- need_nl = true;
- out_txt.append("@param");
- out_txt.append(" ");
- out_txt.append(tmp->GetStr());
- out_txt.append(" TODO");
- }
- has_param = false;
-
- if (tmp->Is(CT_PAREN_CLOSE))
- {
- has_param = true;
- }
- tmp = tmp->GetNextNcNnl();
- }
- fpo = fpc = Chunk::NullChunkPtr;
- }
- else
- {
- fpo = fcn->GetNextType(CT_FPAREN_OPEN, fcn->GetLevel());
-
- if (fpo->IsNullChunk())
- {
- return(true);
- }
- fpc = fpo->GetNextType(CT_FPAREN_CLOSE, fcn->GetLevel());
-
- if (fpc->IsNullChunk())
- {
- return(true);
- }
- }
- Chunk *tmp;
-
- // Check for 'foo()' and 'foo(void)'
- if (fpo->IsNotNullChunk())
- {
- if (fpo->GetNextNcNnl() == fpc)
- {
- has_param = false;
- }
- else
- {
- tmp = fpo->GetNextNcNnl();
-
- if ( (tmp == fpc->GetPrevNcNnl())
- && tmp->IsString("void"))
- {
- has_param = false;
- }
- }
- }
-
- if (has_param)
- {
- Chunk *prev = Chunk::NullChunkPtr;
- tmp = fpo;
-
- while ((tmp = tmp->GetNext())->IsNotNullChunk())
- {
- if ( tmp->Is(CT_COMMA)
- || tmp == fpc)
- {
- if (need_nl)
- {
- out_txt.append("\n");
- }
- need_nl = true;
- out_txt.append("@param");
-
- if (prev->IsNotNullChunk())
- {
- out_txt.append(" ");
- out_txt.append(prev->GetStr());
- out_txt.append(" TODO");
- }
- prev = Chunk::NullChunkPtr;
-
- if (tmp == fpc)
- {
- break;
- }
- }
-
- if (tmp->Is(CT_WORD))
- {
- prev = tmp;
- }
- }
- }
- // Do the return stuff
- tmp = fcn->GetPrevNcNnl();
-
- // For Objective-C we need to go to the previous chunk
- if ( tmp->IsNotNullChunk()
- && tmp->GetParentType() == CT_OC_MSG_DECL
- && tmp->Is(CT_PAREN_CLOSE))
- {
- tmp = tmp->GetPrevNcNnl();
- }
-
- if ( tmp->IsNotNullChunk()
- && !tmp->IsString("void"))
- {
- if (need_nl)
- {
- out_txt.append("\n");
- }
- out_txt.append("@return TODO");
- }
- return(true);
-} // kw_fcn_javaparam
-
-
-static bool kw_fcn_fclass(Chunk *cmt, UncText &out_txt)
-{
- Chunk *fcn = get_next_function(cmt);
-
- if (!fcn)
- {
- return(false);
- }
-
- if (fcn->TestFlags(PCF_IN_CLASS))
- {
- // if inside a class, we need to find to the class name
- Chunk *tmp = fcn->GetPrevType(CT_BRACE_OPEN, fcn->GetLevel() - 1);
- tmp = tmp->GetPrevType(CT_CLASS, tmp->GetLevel());
-
- if (tmp->IsNullChunk())
- {
- tmp = Chunk::NullChunkPtr;
- }
- else
- {
- tmp = tmp->GetNextNcNnl();
- }
-
- while ( tmp->IsNotNullChunk()
- && tmp->GetNextNcNnl()->Is(CT_DC_MEMBER))
- {
- tmp = tmp->GetNextNcNnl();
- tmp = tmp->GetNextNcNnl();
- }
-
- if (tmp->IsNotNullChunk())
- {
- out_txt.append(tmp->GetStr());
- return(true);
- }
- }
- else
- {
- // if outside a class, we expect "CLASS::METHOD(...)"
- Chunk *tmp = fcn->GetPrevNcNnl();
-
- if (tmp->Is(CT_OPERATOR))
- {
- tmp = tmp->GetPrevNcNnl();
- }
-
- if ( tmp->IsNotNullChunk()
- && ( tmp->Is(CT_DC_MEMBER)
- || tmp->Is(CT_MEMBER)))
- {
- tmp = tmp->GetPrevNcNnl();
- out_txt.append(tmp->GetStr());
- return(true);
- }
- }
- return(false);
-} // kw_fcn_fclass
-
-
-static bool kw_fcn_year(Chunk *cmt, UncText &out_txt)
-{
- UNUSED(cmt);
- time_t now = time(nullptr);
-
- out_txt.append(std::to_string(1900 + localtime(&now)->tm_year));
- return(true);
-}
-
-
-struct kw_subst_t
-{
- const char *tag;
- bool (*func)(Chunk *cmt, UncText &out_txt);
-};
-
-
-static const kw_subst_t kw_subst_table[] =
-{
- { "$(filename)", kw_fcn_filename },
- { "$(class)", kw_fcn_class },
- { "$(message)", kw_fcn_message },
- { "$(category)", kw_fcn_category },
- { "$(scope)", kw_fcn_scope },
- { "$(function)", kw_fcn_function },
- { "$(javaparam)", kw_fcn_javaparam },
- { "$(fclass)", kw_fcn_fclass },
- { "$(year)", kw_fcn_year },
-};
-
-
-static void do_kw_subst(Chunk *pc)
-{
- for (const auto &kw : kw_subst_table)
- {
- int idx = pc->GetStr().find(kw.tag);
-
- if (idx < 0)
- {
- continue;
- }
- UncText tmp_txt;
- tmp_txt.clear();
-
- if (kw.func(pc, tmp_txt))
- {
- // if the replacement contains '\n' we need to fix the lead
- if (tmp_txt.find("\n") >= 0)
- {
- size_t nl_idx = pc->GetStr().rfind("\n", idx);
-
- if (nl_idx > 0)
- {
- // idx and nl_idx are both positive
- UncText nl_txt;
- nl_txt.append("\n");
- nl_idx++;
-
- while ( (nl_idx < static_cast<size_t>(idx))
- && !unc_isalnum(pc->GetStr()[nl_idx]))
- {
- nl_txt.append(pc->Str()[nl_idx++]);
- }
- tmp_txt.replace("\n", nl_txt);
- }
- }
- pc->Str().replace(kw.tag, tmp_txt);
- }
- }
-} // do_kw_subst
-
-
-static void output_comment_multi_simple(Chunk *pc)
-{
- if ( pc == nullptr
- && pc->IsNotNullChunk())
- {
- return;
- }
- cmt_reflow cmt;
-
- LOG_FMT(LCONTTEXT, "%s(%d): Text() is '%s', type is %s, orig col is %zu, column is %zu\n",
- __func__, __LINE__, pc->Text(), get_token_name(pc->GetType()), pc->GetOrigCol(), pc->GetColumn());
-
- output_cmt_start(cmt, pc);
-
- // The multiline comment is saved inside one chunk. If the comment is
- // shifted all lines of the comment need to be shifted by the same amount.
- // Save the difference of initial and current position to apply it on every
- // line_column
- const int col_diff = [pc]()
- {
- int diff = 0;
-
- if (pc->GetPrev()->IsNewline())
- {
- // The comment should be indented correctly
- diff = pc->GetColumn() - pc->GetOrigCol();
- }
- return(diff);
- }();
-
- /**
- * check for enable/disable processing comment strings that may
- * both be embedded within the same multi-line comment
- */
- auto disable_processing_cmt_idx = find_disable_processing_comment_marker(pc->GetStr());
- auto enable_processing_cmt_idx = find_enable_processing_comment_marker(pc->GetStr());
-
- UncText line;
- size_t line_count = 0;
- size_t line_column = pc->GetColumn();
- size_t cmt_idx = 0;
-
- while (cmt_idx < pc->Len())
- {
- int ch = pc->GetStr()[cmt_idx];
- cmt_idx++;
-
- if ( cmt_idx > std::size_t(disable_processing_cmt_idx)
- && enable_processing_cmt_idx > disable_processing_cmt_idx)
- {
- auto length = enable_processing_cmt_idx - disable_processing_cmt_idx;
- UncText verbatim_text(pc->GetStr(),
- disable_processing_cmt_idx,
- length);
-
- add_text(verbatim_text);
-
- cmt_idx = enable_processing_cmt_idx;
-
- /**
- * check for additional enable/disable processing comment strings that may
- * both be embedded within the same multi-line comment
- */
- disable_processing_cmt_idx = find_disable_processing_comment_marker(pc->GetStr(),
- enable_processing_cmt_idx);
- enable_processing_cmt_idx = find_enable_processing_comment_marker(pc->GetStr(),
- enable_processing_cmt_idx);
-
- line.clear();
-
- continue;
- }
- // 1: step through leading tabs and spaces to find the start column
- log_rule_B("cmt_convert_tab_to_spaces");
-
- if ( line.size() == 0
- && ( line_column < cmt.base_col
- || options::cmt_convert_tab_to_spaces()))
- {
- if (ch == ' ')
- {
- line_column++;
- continue;
- }
- else if (ch == '\t')
- {
- log_rule_B("input_tab_size");
- line_column = calc_next_tab_column(line_column, options::input_tab_size());
- continue;
- }
- else
- {
- LOG_FMT(LCONTTEXT, "%s(%d):ch is %d, %c\n", __func__, __LINE__, ch, char(ch));
- }
- }
-
- // 2: add chars to line, handle the CRLF and CR endings (convert both to LF)
- if (ch == '\r')
- {
- ch = '\n';
-
- if ( (cmt_idx < pc->Len())
- && (pc->GetStr()[cmt_idx] == '\n'))
- {
- cmt_idx++;
- }
- }
- LOG_FMT(LCONTTEXT, "%s(%d):Line is %s\n", __func__, __LINE__, line.c_str());
- line.append(ch);
- LOG_FMT(LCONTTEXT, "%s(%d):Line is %s\n", __func__, __LINE__, line.c_str());
-
- // If we just hit an end of line OR we just hit end-of-comment...
- if ( ch == '\n'
- || cmt_idx == pc->Len())
- {
- line_count++;
- LOG_FMT(LCONTTEXT, "%s(%d):line_count is %zu\n", __func__, __LINE__, line_count);
-
- // strip trailing tabs and spaces before the newline
- if (ch == '\n')
- {
- line.pop_back();
-
- // Say we aren't in a preproc to prevent changing any bs-nl
- cmt_trim_whitespace(line, false);
-
- line.append('\n');
- }
-
- if (line.size() > 0)
- {
- // unless line contains only a single newline char, indent if the
- // line consists of either:
- if ( line.size() > 1 // more than a single newline char or
- || ch != '\n') // (end-of-comment) a single non newline char
- {
- if (line_count > 1)
- {
- // apply comment column shift without underflowing
- line_column = (col_diff<0
- && (size_t)(abs(col_diff))> line_column)
- ? 0 : line_column + col_diff;
- }
- cmt.column = line_column;
- cmt_output_indent(cmt.brace_col, cmt.base_col, cmt.column);
- }
- add_text(line);
-
- line.clear();
- }
- line_column = 1;
- }
- }
-} // output_comment_multi_simple
-
-
-static void generate_if_conditional_as_text(UncText &dst, Chunk *ifdef)
-{
- int column = -1;
-
- dst.clear();
-
- for (Chunk *pc = ifdef; pc != nullptr && pc->IsNotNullChunk(); pc = pc->GetNext())
- {
- if (column == -1)
- {
- column = pc->GetColumn();
- }
-
- if ( pc->Is(CT_NEWLINE)
- || pc->Is(CT_COMMENT_MULTI)
- || pc->Is(CT_COMMENT_CPP))
- {
- break;
- }
- else if (pc->Is(CT_NL_CONT))
- {
- dst += ' ';
- column = -1;
- }
- else if ( pc->Is(CT_COMMENT)
- || pc->Is(CT_COMMENT_EMBED))
- {
- }
- else // if (pc->Is(CT_JUNK)) || else
- {
- for (int spacing = pc->GetColumn() - column; spacing > 0; spacing--)
- {
- dst += ' ';
- column++;
- }
-
- dst.append(pc->GetStr());
- column += pc->Len();
- }
- }
-} // generate_if_conditional_as_text
-
-
-void add_long_preprocessor_conditional_block_comment()
-{
- Chunk *pp_start = nullptr;
- Chunk *pp_end = nullptr;
-
- for (Chunk *pc = Chunk::GetHead(); pc->IsNotNullChunk(); pc = pc->GetNextNcNnl())
- {
- // just track the preproc level:
- if (pc->Is(CT_PREPROC))
- {
- pp_end = pp_start = pc;
- }
-
- if ( pc->IsNot(CT_PP_IF)
- || !pp_start)
- {
- continue;
- }
-#if 0
- if (pc->TestFlags(PCF_IN_PREPROC))
- {
- continue;
- }
-#endif
-
- Chunk *br_close;
- Chunk *br_open = pc;
- size_t nl_count = 0;
-
- Chunk *tmp = pc;
-
- while ((tmp = tmp->GetNext())->IsNotNullChunk())
- {
- // just track the preproc level:
- if (tmp->Is(CT_PREPROC))
- {
- pp_end = tmp;
- }
-
- if (tmp->IsNewline())
- {
- nl_count += tmp->GetNlCount();
- }
- else if ( pp_end->GetPpLevel() == pp_start->GetPpLevel()
- && ( tmp->Is(CT_PP_ENDIF)
- || ((br_open->Is(CT_PP_IF)) ? (tmp->Is(CT_PP_ELSE)) : 0)))
- {
- br_close = tmp;
-
- LOG_FMT(LPPIF, "found #if / %s section on lines %zu and %zu, new line count=%zu\n",
- (tmp->Is(CT_PP_ENDIF) ? "#endif" : "#else"),
- br_open->GetOrigLine(), br_close->GetOrigLine(), nl_count);
-
- // Found the matching #else or #endif - make sure a newline is next
- tmp = tmp->GetNext();
-
- LOG_FMT(LPPIF, "next item type %d (is %s)\n",
- (tmp ? tmp->GetType() : -1), (tmp ? tmp->IsNewline() ? "newline"
- : tmp->IsComment() ? "comment" : "other" : "---"));
-
- if ( tmp->IsNullChunk()
- || tmp->Is(CT_NEWLINE)) // tmp->IsNewline())
- {
- size_t nl_min;
-
- if (br_close->Is(CT_PP_ENDIF))
- {
- log_rule_B("mod_add_long_ifdef_endif_comment");
- nl_min = options::mod_add_long_ifdef_endif_comment();
- }
- else
- {
- log_rule_B("mod_add_long_ifdef_else_comment");
- nl_min = options::mod_add_long_ifdef_else_comment();
- }
- const char *txt = !tmp ? "EOF" : ((tmp->Is(CT_PP_ENDIF)) ? "#endif" : "#else");
- LOG_FMT(LPPIF, "#if / %s section candidate for augmenting when over NL threshold %zu != 0 (new line count=%zu)\n",
- txt, nl_min, nl_count);
-
- if ( nl_min > 0
- && nl_count > nl_min) // nl_count is 1 too large at all times as #if line was counted too
- {
- // determine the added comment style
- E_Token style = (language_is_set(LANG_CPP)) ?
- CT_COMMENT_CPP : CT_COMMENT;
-
- UncText str;
- generate_if_conditional_as_text(str, br_open);
-
- LOG_FMT(LPPIF, "#if / %s section over threshold %zu (new line count=%zu) --> insert comment after the %s: %s\n",
- txt, nl_min, nl_count, txt, str.c_str());
-
- // Add a comment after the close brace
- insert_comment_after(br_close, style, str);
- }
- }
-
- // checks both the #else and #endif for a given level, only then look further in the main loop
- if (br_close->Is(CT_PP_ENDIF))
- {
- break;
- }
- }
- }
- }
-} // add_long_preprocessor_conditional_block_comment