aboutsummaryrefslogtreecommitdiff
path: root/libjava/java/io/PushbackReader.java
blob: 0cfd63cdb2bcd4e2d2e23c42bb14fed39ee28426 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
/* Copyright (C) 1998, 1999  Cygnus Solutions

   This file is part of libgcj.

This software is copyrighted work licensed under the terms of the
Libgcj License.  Please consult the file "LIBGCJ_LICENSE" for
details.  */
 
package java.io;

/**
 * @author Warren Levy <warrenl@cygnus.com>
 * @date October 16, 1998.  
 */
/* Written using "Java Class Libraries", 2nd edition, ISBN 0-201-31002-3
 * "The Java Language Specification", ISBN 0-201-63451-1
 * plus online API docs for JDK 1.2 beta from http://www.javasoft.com.
 * Status:  Believed complete and correct.
 */
 
public class PushbackReader extends FilterReader
{
  /* Internal buffer array for data. */
  private char[] buf;

  /* The current position in the buffer. */
  private int pos;

  public PushbackReader(Reader in)
  {
    this(in, 1);
  }

  public PushbackReader(Reader in, int size)
  {
    super(in);
    if (size < 0)
      throw new IllegalArgumentException();
    buf = new char[size];
    pos = buf.length;
  }

  public void close() throws IOException
  {
    synchronized (lock)
    {
      buf = null;
      super.close();
    }
  }

  public boolean markSupported()
  {
    return false;
  }

  public int read() throws IOException
  {
    synchronized (lock)
    {
      if (buf == null)
        throw new IOException();

      if (pos < buf.length)
        return ((int) buf[pos++]) & 0xFFFF;

      return super.read();
    }
  }

  public int read(char[] b, int off, int len) throws IOException
  {
    synchronized (lock)
    {
      if (buf == null)
        throw new IOException();

      if (off < 0 || len < 0 || off + len > b.length)
        throw new ArrayIndexOutOfBoundsException();

      int numBytes = Math.min(buf.length - pos, len);
      for (int i = 0; i < numBytes; i++)
        b[off++] = buf[pos++];

      return numBytes + super.read(b, off, len - numBytes);
    }
  }

  public boolean ready() throws IOException
  {
    synchronized (lock)
    {
      if (buf == null)
        throw new IOException();

      if (buf.length - pos > 0)
        return true;

      return super.ready();
    }
  }

  public void unread(int b) throws IOException
  {
    synchronized (lock)
    {
      if (buf == null || pos <= 0)
        throw new IOException();

      buf[--pos] = (char) b;
    }
  }

  public void unread(char[] b) throws IOException
  {
    unread(b, 0, b.length);
  }

  public void unread(char[] b, int off, int len) throws IOException
  {
    synchronized (lock)
    {
      if (buf == null || pos < len)
        throw new IOException();

      // Note the order that these chars are being added is the opposite
      // of what would be done if they were added to the buffer one at a time.
      // See the Java Class Libraries book p. 1397.
      System.arraycopy(b, off, buf, pos - len, len);

      // Don't put this into the arraycopy above, an exception might be thrown
      // and in that case we don't want to modify pos.
      pos -= len;
    }
  }
}