20
20
#include " clang/Format/Format.h"
21
21
#include " llvm/Support/Debug.h"
22
22
23
- #define DEBUG_TYPE " format-formatter "
23
+ #define DEBUG_TYPE " format-indenter "
24
24
25
25
namespace clang {
26
26
namespace format {
@@ -1154,7 +1154,11 @@ unsigned ContinuationIndenter::breakProtrudingToken(const FormatToken &Current,
1154
1154
}
1155
1155
} else if (Current.is (TT_BlockComment)) {
1156
1156
if (!Current.isTrailingComment () || !Style .ReflowComments ||
1157
- CommentPragmasRegex.match (Current.TokenText .substr (2 )))
1157
+ CommentPragmasRegex.match (Current.TokenText .substr (2 )) ||
1158
+ // If a comment token switches formatting, like
1159
+ // /* clang-format on */, we don't want to break it further,
1160
+ // but we may still want to adjust its indentation.
1161
+ switchesFormatting (Current))
1158
1162
return addMultilineToken (Current, State);
1159
1163
Token.reset (new BreakableBlockComment (
1160
1164
Current, State.Line ->Level , StartColumn, Current.OriginalColumn ,
@@ -1163,11 +1167,13 @@ unsigned ContinuationIndenter::breakProtrudingToken(const FormatToken &Current,
1163
1167
(Current.Previous == nullptr ||
1164
1168
Current.Previous ->isNot (TT_ImplicitStringLiteral))) {
1165
1169
if (!Style .ReflowComments ||
1166
- CommentPragmasRegex.match (Current.TokenText .substr (2 )))
1170
+ CommentPragmasRegex.match (Current.TokenText .substr (2 )) ||
1171
+ switchesFormatting (Current))
1167
1172
return 0 ;
1168
- Token.reset (new BreakableLineComment (Current, State.Line ->Level ,
1169
- StartColumn, /* InPPDirective=*/ false ,
1170
- Encoding, Style ));
1173
+ Token.reset (new BreakableLineCommentSection (
1174
+ Current, State.Line ->Level , StartColumn, Current.OriginalColumn ,
1175
+ !Current.Previous ,
1176
+ /* InPPDirective=*/ false , Encoding, Style ));
1171
1177
// We don't insert backslashes when breaking line comments.
1172
1178
ColumnLimit = Style .ColumnLimit ;
1173
1179
} else {
@@ -1178,15 +1184,27 @@ unsigned ContinuationIndenter::breakProtrudingToken(const FormatToken &Current,
1178
1184
1179
1185
unsigned RemainingSpace = ColumnLimit - Current.UnbreakableTailLength ;
1180
1186
bool BreakInserted = false ;
1187
+ // We use a conservative reflowing strategy. Reflow starts after a line is
1188
+ // broken or the corresponding whitespace compressed. Reflow ends as soon as a
1189
+ // line that doesn't get reflown with the previous line is reached.
1190
+ bool ReflowInProgress = false ;
1181
1191
unsigned Penalty = 0 ;
1182
1192
unsigned RemainingTokenColumns = 0 ;
1183
1193
for (unsigned LineIndex = 0 , EndIndex = Token->getLineCount ();
1184
1194
LineIndex != EndIndex; ++LineIndex) {
1195
+ BreakableToken::Split SplitBefore (StringRef::npos, 0 );
1196
+ if (ReflowInProgress) {
1197
+ SplitBefore = Token->getSplitBefore (LineIndex, RemainingTokenColumns,
1198
+ RemainingSpace);
1199
+ }
1200
+ ReflowInProgress = SplitBefore.first != StringRef::npos;
1201
+ unsigned TailOffset =
1202
+ ReflowInProgress ? (SplitBefore.first + SplitBefore.second ) : 0 ;
1185
1203
if (!DryRun)
1186
- Token->replaceWhitespaceBefore (LineIndex, Whitespaces);
1187
- unsigned TailOffset = 0 ;
1188
- RemainingTokenColumns =
1189
- Token-> getLineLengthAfterSplit ( LineIndex, TailOffset, StringRef::npos );
1204
+ Token->replaceWhitespaceBefore (LineIndex, RemainingTokenColumns,
1205
+ RemainingSpace, SplitBefore, Whitespaces) ;
1206
+ RemainingTokenColumns = Token-> getLineLengthAfterSplitBefore (
1207
+ LineIndex, TailOffset, RemainingTokenColumns, ColumnLimit, SplitBefore );
1190
1208
while (RemainingTokenColumns > RemainingSpace) {
1191
1209
BreakableToken::Split Split =
1192
1210
Token->getSplit (LineIndex, TailOffset, ColumnLimit);
@@ -1198,17 +1216,23 @@ unsigned ContinuationIndenter::breakProtrudingToken(const FormatToken &Current,
1198
1216
break ;
1199
1217
}
1200
1218
assert (Split.first != 0 );
1201
- unsigned NewRemainingTokenColumns = Token->getLineLengthAfterSplit (
1202
- LineIndex, TailOffset + Split.first + Split.second , StringRef::npos);
1203
1219
1204
- // We can remove extra whitespace instead of breaking the line.
1205
- if (RemainingTokenColumns + 1 - Split.second <= RemainingSpace) {
1206
- RemainingTokenColumns = 0 ;
1220
+ // Check if compressing the whitespace range will bring the line length
1221
+ // under the limit. If that is the case, we perform whitespace compression
1222
+ // instead of inserting a line break.
1223
+ unsigned RemainingTokenColumnsAfterCompression =
1224
+ Token->getLineLengthAfterCompression (RemainingTokenColumns, Split);
1225
+ if (RemainingTokenColumnsAfterCompression <= RemainingSpace) {
1226
+ RemainingTokenColumns = RemainingTokenColumnsAfterCompression;
1227
+ ReflowInProgress = true ;
1207
1228
if (!DryRun)
1208
- Token->replaceWhitespace (LineIndex, TailOffset, Split, Whitespaces);
1229
+ Token->compressWhitespace (LineIndex, TailOffset, Split, Whitespaces);
1209
1230
break ;
1210
1231
}
1211
1232
1233
+ unsigned NewRemainingTokenColumns = Token->getLineLengthAfterSplit (
1234
+ LineIndex, TailOffset + Split.first + Split.second , StringRef::npos);
1235
+
1212
1236
// When breaking before a tab character, it may be moved by a few columns,
1213
1237
// but will still be expanded to the next tab stop, so we don't save any
1214
1238
// columns.
@@ -1226,6 +1250,7 @@ unsigned ContinuationIndenter::breakProtrudingToken(const FormatToken &Current,
1226
1250
}
1227
1251
TailOffset += Split.first + Split.second ;
1228
1252
RemainingTokenColumns = NewRemainingTokenColumns;
1253
+ ReflowInProgress = true ;
1229
1254
BreakInserted = true ;
1230
1255
}
1231
1256
}
@@ -1246,6 +1271,9 @@ unsigned ContinuationIndenter::breakProtrudingToken(const FormatToken &Current,
1246
1271
1247
1272
State.Stack .back ().LastSpace = StartColumn;
1248
1273
}
1274
+
1275
+ Token->updateNextToken (State);
1276
+
1249
1277
return Penalty;
1250
1278
}
1251
1279
0 commit comments