aboutsummaryrefslogtreecommitdiff
path: root/lib/Support/FormattedStream.cpp
diff options
context:
space:
mode:
authorDan Gohman <gohman@apple.com>2009-08-15 02:02:59 +0000
committerDan Gohman <gohman@apple.com>2009-08-15 02:02:59 +0000
commitfbcb5b678f0fc0432072212a26a0507954c805d8 (patch)
treed067190f8fce637737f02a72dc5a91ea6a2e53f4 /lib/Support/FormattedStream.cpp
parenta4a68c1b439af1bacf8b9c3c06cdb97f56be4d94 (diff)
Add support for column computation on unbuffered streams.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@79065 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Support/FormattedStream.cpp')
-rw-r--r--lib/Support/FormattedStream.cpp53
1 files changed, 38 insertions, 15 deletions
diff --git a/lib/Support/FormattedStream.cpp b/lib/Support/FormattedStream.cpp
index 867e5de4d1..0db21f651a 100644
--- a/lib/Support/FormattedStream.cpp
+++ b/lib/Support/FormattedStream.cpp
@@ -15,27 +15,34 @@
using namespace llvm;
-/// ComputeColumn - Examine the current output and figure out which
+/// CountColumns - Examine the given char sequence and figure out which
/// column we end up in after output.
///
-void formatted_raw_ostream::ComputeColumn() {
+static unsigned CountColumns(unsigned Column, const char *Ptr, size_t Size) {
// Keep track of the current column by scanning the string for
// special characters
- // The buffer may have been allocated underneath us.
- if (Scanned == 0 && GetNumBytesInBuffer() != 0) {
- Scanned = begin();
- }
-
- while (Scanned != end()) {
- ++ColumnScanned;
- if (*Scanned == '\n' || *Scanned == '\r')
- ColumnScanned = 0;
- else if (*Scanned == '\t')
+ for (const char *End = Ptr + Size; Ptr != End; ++Ptr) {
+ ++Column;
+ if (*Ptr == '\n' || *Ptr == '\r')
+ Column = 0;
+ else if (*Ptr == '\t')
// Assumes tab stop = 8 characters.
- ColumnScanned += (8 - (ColumnScanned & 0x7)) & 0x7;
- ++Scanned;
+ Column += (8 - (Column & 0x7)) & 0x7;
}
+
+ return Column;
+}
+
+/// ComputeColumn - Examine the current output and figure out which
+/// column we end up in after output.
+void formatted_raw_ostream::ComputeColumn() {
+ // The buffer may have been allocated underneath us.
+ if (Scanned == 0) Scanned = begin();
+ // Scan all characters added since our last scan to determine the new column.
+ ColumnScanned = CountColumns(ColumnScanned, Scanned, end() - Scanned);
+ // We're now current with everything in the buffer.
+ Scanned = end();
}
/// PadToColumn - Align the output to some column number.
@@ -63,9 +70,25 @@ void formatted_raw_ostream::PadToColumn(unsigned NewCol, unsigned MinPad) {
}
void formatted_raw_ostream::write_impl(const char *Ptr, size_t Size) {
+ // Figure out what's in the buffer and add it to the column count.
ComputeColumn();
+
+ // Write the data to the underlying stream (which is unbuffered, so
+ // the data will be immediately written out).
TheStream->write(Ptr, Size);
- Scanned = begin();
+
+ // If this FormattedStream is unbuffered, scan the string that
+ // was just written to determine the new column.
+ if (Ptr == begin()) {
+ // Buffered mode. The buffer is being flushed; reset the scanning
+ // position to the beginning of the buffer.
+ assert(Ptr + Size == end() && "Buffer is not being fully flushed!");
+ Scanned = begin();
+ } else {
+ // Unbuffered mode. Immediately scan the string that was just
+ // written to determine the new column.
+ ColumnScanned = CountColumns(ColumnScanned, Ptr, Size);
+ }
}
/// fouts() - This returns a reference to a formatted_raw_ostream for