123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417 |
- /* LineNumberReader.java -- A character input stream which counts line numbers
- Copyright (C) 1998, 1999, 2001, 2003, 2005 Free Software Foundation, Inc.
- This file is part of GNU Classpath.
- GNU Classpath is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 2, or (at your option)
- any later version.
- GNU Classpath is distributed in the hope that it will be useful, but
- WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with GNU Classpath; see the file COPYING. If not, write to the
- Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- 02110-1301 USA.
- Linking this library statically or dynamically with other modules is
- making a combined work based on this library. Thus, the terms and
- conditions of the GNU General Public License cover the whole
- combination.
- As a special exception, the copyright holders of this library give you
- permission to link this library with independent modules to produce an
- executable, regardless of the license terms of these independent
- modules, and to copy and distribute the resulting executable under
- terms of your choice, provided that you also meet, for each linked
- independent module, the terms and conditions of the license of that
- module. An independent module is a module which is not derived from
- or based on this library. If you modify this library, you may extend
- this exception to your version of the library, but you are not
- obligated to do so. If you do not wish to do so, delete this
- exception statement from your version. */
- package java.io;
- /**
- * This class functions like a standard <code>Reader</code> except that it
- * counts line numbers, and canonicalizes newline characters. As data
- * is read, whenever the char sequences "\r", "\n", or "\r\n" are encountered,
- * the running line count is incremeted by one. Additionally, the whatever
- * line termination sequence was encountered will be converted to a "\n"
- * char. Note that this class numbers lines from 0. When the first
- * line terminator is encountered, the line number is incremented to 1, and
- * so on. Also note that actual "\r" and "\n" characters are looked for.
- * The system dependent line separator sequence is ignored.
- * <p>
- * This class counts only line termination characters. If the last line
- * read from the stream does not end in a line termination sequence, it
- * will not be counted as a line.
- *
- * @author Per Bothner (bothner@cygnus.com)
- * @author Aaron M. Renn (arenn@urbanophile.com)
- * @author Guilhem Lavaux (guilhem@kaffe.org)
- * @date December 28, 2003.
- */
- /* Written using "Java Class Libraries", 2nd edition, plus online
- * API docs for JDK 1.2 beta from http://www.javasoft.com.
- * Status: Believed complete and correct.
- *
- * This implementation has the feature that if '\r' is read, it
- * does not look for a '\n', but immediately returns '\n'.
- * On the next read(), if a '\n' is read, it is skipped.
- * This has the advantage that we do not read (and hang) unnecessarily.
- *
- * This implementation is also minimal in the number of fields it uses.
- */
- public class LineNumberReader extends BufferedReader
- {
- /** The current line number. */
- private int lineNumber;
- /** Whether we already found a new line in the former call. */
- private boolean matchedNewLine;
- /** The saved line number when calling mark() */
- private int savedLineNumber;
- /**
- * Create a new <code>LineNumberReader</code> that reads from the
- * specified subordinate <code>Reader</code>. A default 8K char sized
- * buffer will be used for reads.
- *
- * @param in The subordinate <code>Reader</code> to read from
- */
- public LineNumberReader(Reader in)
- {
- super(in, DEFAULT_BUFFER_SIZE);
- }
- /**
- * This method initializes a new <code>LineNumberReader</code> to read
- * from the specified subordinate <code>Reader</code> using the specified
- * read buffer size.
- *
- * @param in The subordinate <code>Reader</code> to read from
- * @param size The buffer size to use for reading
- */
- public LineNumberReader(Reader in, int size)
- {
- super(in, size);
- }
- /**
- * This method returns the current line number
- *
- * @return The current line number
- */
- public int getLineNumber()
- {
- return lineNumber;
- }
- /**
- * This method sets the current line number to the specified value.
- *
- * @param lineNumber The new line number
- */
- public void setLineNumber(int lineNumber)
- {
- this.lineNumber = lineNumber;
- }
- /**
- * This method marks a position in the input to which the stream can be
- * "reset" char calling the <code>reset()</code> method. The parameter
- * <code>readlimit</code> is the number of chars that can be read from the
- * stream after setting the mark before the mark becomes invalid. For
- * example, if <code>mark()</code> is called with a read limit of 10,
- * then when
- * 11 chars of data are read from the stream before the <code>reset()</code>
- * method is called, then the mark is invalid and the stream object
- * instance is not required to remember the mark.
- * <p>
- * In this class, this method will remember the current line number as well
- * as the current position in the stream. When the <code>reset()</code>
- * method
- * is called, the line number will be restored to the saved line number in
- * addition to the stream position.
- *
- * @param readLimit The number of chars that can be read before the
- * mark becomes invalid
- *
- * @exception IOException If an error occurs
- */
- public void mark(int readLimit) throws IOException
- {
- if (readLimit < 0)
- throw new IllegalArgumentException("Read-ahead limit is negative");
- synchronized (lock)
- {
- // This is basically the same as BufferedReader.mark.
- // However, if the previous character was a '\r', we need to
- // save that 'r', in case the next character is a '\n'.
- if (pos + readLimit > limit)
- {
- int saveCR = matchedNewLine ? 1 : 0;
- char[] old_buffer = buffer;
- if (readLimit > limit)
- buffer = new char[saveCR + readLimit];
- int copy_start = pos - saveCR;
- savedLineNumber = lineNumber;
- limit -= copy_start;
- System.arraycopy(old_buffer, copy_start, buffer, 0, limit);
- pos = saveCR;
- }
- markPos = pos;
- }
- }
- /**
- * This method resets a stream to the point where the <code>mark()</code>
- * method
- * was called. Any chars that were read after the mark point was set will
- * be re-read during subsequent reads.
- * <p>
- * In this class, this method will also restore the line number that was
- * current when the <code>mark()</code> method was called.
- *
- * @exception IOException If an error occurs
- */
- public void reset() throws IOException
- {
- synchronized (lock)
- {
- if (markPos < 0)
- throw new IOException("mark never set or invalidated");
- lineNumber = savedLineNumber;
- pos = markPos;
- matchedNewLine = (markPos > 0 && buffer[markPos-1] == '\r');
- }
- }
- /**
- * This private method fills the input buffer whatever pos is.
- * Consequently pos should be checked before calling this method.
- *
- * @return the number of bytes actually read from the input stream or
- * -1 if end of stream.
- * @exception IOException If an error occurs.
- */
- private int fill() throws IOException
- {
- if (markPos >= 0 && limit == buffer.length)
- markPos = -1;
- if (markPos < 0)
- pos = limit = 0;
- int count = in.read(buffer, limit, buffer.length - limit);
- if (count <= 0)
- return -1;
- limit += count;
- return count;
- }
- /**
- * This method reads an unsigned char from the input stream and returns it
- * as an int in the range of 0-65535. This method will return -1 if the
- * end of the stream has been reached.
- * <p>
- * Note that if a line termination sequence is encountered (ie, "\r",
- * "\n", or "\r\n") then that line termination sequence is converted to
- * a single "\n" value which is returned from this method. This means
- * that it is possible this method reads two chars from the subordinate
- * stream instead of just one.
- * <p>
- * Note that this method will block until a char of data is available
- * to be read.
- *
- * @return The char read or -1 if end of stream
- *
- * @exception IOException If an error occurs
- */
- public int read() throws IOException
- {
- synchronized (lock)
- {
- skipRedundantLF();
- if (pos >= limit && fill() < 0)
- return -1;
- char ch = buffer[pos++];
- if ((matchedNewLine = (ch == '\r')) || ch == '\n')
- {
- lineNumber++;
- return '\n';
- }
- matchedNewLine = false;
- return (int) ch;
- }
- }
- /**
- * This method reads chars from a stream and stores them into a caller
- * supplied buffer. It starts storing data at index <code>offset</code> into
- * the buffer and attemps to read <code>len</code> chars. This method can
- * return before reading the number of chars requested. The actual number
- * of chars read is returned as an int. A -1 is returned to indicated the
- * end of the stream.
- * <p>
- * This method will block until some data can be read.
- * <p>
- * Note that if a line termination sequence is encountered (ie, "\r",
- * "\n", or "\r\n") then that line termination sequence is converted to
- * a single "\n" value which is stored in the buffer. Only a single
- * char is counted towards the number of chars read in this case.
- *
- * @param buf The array into which the chars read should be stored
- * @param offset The offset into the array to start storing chars
- * @param count The requested number of chars to read
- *
- * @return The actual number of chars read, or -1 if end of stream
- *
- * @exception IOException If an error occurs.
- * @exception NullPointerException If buf is null (in any case).
- * @exception IndexOutOfBoundsException If buffer parameters (offset and
- * count) lies outside of the buffer capacity.
- */
- public int read(char[] buf, int offset, int count) throws IOException
- {
- if (buf == null)
- throw new NullPointerException();
- if (offset + count > buf.length || offset < 0)
- throw new IndexOutOfBoundsException();
- if (count <= 0)
- {
- if (count < 0)
- throw new IndexOutOfBoundsException();
- return 0;
- }
- synchronized (lock)
- {
- if (pos >= limit && fill() < 0)
- return -1;
- int start_offset = offset;
- boolean matched = matchedNewLine;
- while (count-- > 0 && pos < limit)
- {
- char ch = buffer[pos++];
- if (ch == '\r')
- {
- lineNumber++;
- matched = true;
- }
- else if (ch == '\n' && !matched)
- lineNumber++;
- else
- matched = false;
- buf[offset++] = ch;
- }
- matchedNewLine = matched;
- return offset - start_offset;
- }
- }
- private void skipRedundantLF() throws IOException
- {
- if (pos > 0 && matchedNewLine)
- {
- if (pos < limit)
- { // fast case
- if (buffer[pos] == '\n')
- pos++;
- }
- else
- { // check whether the next buffer begins with '\n'.
- // in that case kill the '\n'.
- if (fill() <= 0)
- return;
- if (buffer[pos] == '\n')
- pos++;
- }
- matchedNewLine = true;
- }
- }
- /**
- * This method reads a line of text from the input stream and returns
- * it as a <code>String</code>. A line is considered to be terminated
- * by a "\r", "\n", or "\r\n" sequence, not by the system dependent line
- * separator.
- *
- * @return The line read as a <code>String</code> or <code>null</code>
- * if end of stream.
- *
- * @exception IOException If an error occurs
- */
- public String readLine() throws IOException
- {
- // BufferedReader.readLine already does this. Shouldn't need to keep
- // track of newlines (since the read method deals with this for us).
- // But if the buffer is large, we may not call the read method at all
- // and super.readLine can't increment lineNumber itself.
- // Though it may seem kludgy, the safest thing to do is to save off
- // lineNumber and increment it explicitly when we're done (iff we
- // ended with a '\n' or '\r' as opposed to EOF).
- //
- // Also, we need to undo the special casing done by BufferedReader.readLine
- // when a '\r' is the last char in the buffer. That situation is marked
- // by 'pos > limit'.
- int tmpLineNumber = lineNumber;
- skipRedundantLF();
- String str = super.readLine();
- if (pos > limit)
- --pos;
- // The only case where you mustn't increment the line number is you are
- // at the EOS.
- if (str != null)
- lineNumber = tmpLineNumber + 1;
- return str;
- }
- /**
- * This method skips over characters in the stream. This method will
- * skip the specified number of characters if possible, but is not required
- * to skip them all. The actual number of characters skipped is returned.
- * This method returns 0 if the specified number of chars is less than 1.
- *
- * @param count The specified number of chars to skip.
- *
- * @return The actual number of chars skipped.
- *
- * @exception IOException If an error occurs
- */
- public long skip (long count) throws IOException
- {
- if (count < 0)
- throw new IllegalArgumentException("skip() value is negative");
- if (count == 0)
- return 0;
- int skipped;
- char[] buf = new char[1];
- for (skipped = 0; skipped < count; skipped++)
- {
- int ch = read(buf, 0, 1);
- if (ch < 0)
- break;
- }
- return skipped;
- }
- }
|