1 /*
2  * Copyright (C) 2008 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.internal.os;
18 
19 import java.io.IOException;
20 import java.io.OutputStream;
21 import java.io.PrintStream;
22 import java.nio.ByteBuffer;
23 import java.nio.CharBuffer;
24 import java.nio.charset.Charset;
25 import java.nio.charset.CharsetDecoder;
26 import java.nio.charset.CoderResult;
27 import java.nio.charset.CodingErrorAction;
28 import java.util.Formatter;
29 import java.util.Locale;
30 
31 /**
32  * A print stream which logs output line by line.
33  *
34  * {@hide}
35  */
36 abstract class LoggingPrintStream extends PrintStream {
37 
38     private final StringBuilder builder = new StringBuilder();
39 
40     /**
41      * A buffer that is initialized when raw bytes are first written to this
42      * stream. It may contain the leading bytes of multi-byte characters.
43      * Between writes this buffer is always ready to receive data; ie. the
44      * position is at the first unassigned byte and the limit is the capacity.
45      */
46     private ByteBuffer encodedBytes;
47 
48     /**
49      * A buffer that is initialized when raw bytes are first written to this
50      * stream. Between writes this buffer is always clear; ie. the position is
51      * zero and the limit is the capacity.
52      */
53     private CharBuffer decodedChars;
54 
55     /**
56      * Decodes bytes to characters using the system default charset. Initialized
57      * when raw bytes are first written to this stream.
58      */
59     private CharsetDecoder decoder;
60 
LoggingPrintStream()61     protected LoggingPrintStream() {
62         super(new OutputStream() {
63             public void write(int oneByte) throws IOException {
64                 throw new AssertionError();
65             }
66         });
67     }
68 
69     /**
70      * Logs the given line.
71      */
log(String line)72     protected abstract void log(String line);
73 
74     @Override
flush()75     public synchronized void flush() {
76         flush(true);
77     }
78 
79     /**
80      * Searches buffer for line breaks and logs a message for each one.
81      *
82      * @param completely true if the ending chars should be treated as a line
83      *  even though they don't end in a line break
84      */
flush(boolean completely)85     private void flush(boolean completely) {
86         int length = builder.length();
87 
88         int start = 0;
89         int nextBreak;
90 
91         // Log one line for each line break.
92         while (start < length
93                 && (nextBreak = builder.indexOf("\n", start)) != -1) {
94             log(builder.substring(start, nextBreak));
95             start = nextBreak + 1;
96         }
97 
98         if (completely) {
99             // Log the remainder of the buffer.
100             if (start < length) {
101                 log(builder.substring(start));
102             }
103             builder.setLength(0);
104         } else {
105             // Delete characters leading up to the next starting point.
106             builder.delete(0, start);
107         }
108     }
109 
write(int oneByte)110     public void write(int oneByte) {
111         write(new byte[] { (byte) oneByte }, 0, 1);
112     }
113 
114     @Override
write(byte[] buffer)115     public void write(byte[] buffer) {
116         write(buffer, 0, buffer.length);
117     }
118 
119     @Override
write(byte bytes[], int start, int count)120     public synchronized void write(byte bytes[], int start, int count) {
121         if (decoder == null) {
122             encodedBytes = ByteBuffer.allocate(80);
123             decodedChars = CharBuffer.allocate(80);
124             decoder = Charset.defaultCharset().newDecoder()
125                     .onMalformedInput(CodingErrorAction.REPLACE)
126                     .onUnmappableCharacter(CodingErrorAction.REPLACE);
127         }
128 
129         int end = start + count;
130         while (start < end) {
131             // copy some bytes from the array to the long-lived buffer. This
132             // way, if we end with a partial character we don't lose it.
133             int numBytes = Math.min(encodedBytes.remaining(), end - start);
134             encodedBytes.put(bytes, start, numBytes);
135             start += numBytes;
136 
137             encodedBytes.flip();
138             CoderResult coderResult;
139             do {
140                 // decode bytes from the byte buffer into the char buffer
141                 coderResult = decoder.decode(encodedBytes, decodedChars, false);
142 
143                 // copy chars from the char buffer into our string builder
144                 decodedChars.flip();
145                 builder.append(decodedChars);
146                 decodedChars.clear();
147             } while (coderResult.isOverflow());
148             encodedBytes.compact();
149         }
150         flush(false);
151     }
152 
153     /** Always returns false. */
154     @Override
checkError()155     public boolean checkError() {
156         return false;
157     }
158 
159     /** Ignored. */
160     @Override
setError()161     protected void setError() { /* ignored */ }
162 
163     /** Ignored. */
164     @Override
close()165     public void close() { /* ignored */ }
166 
167     @Override
format(String format, Object... args)168     public PrintStream format(String format, Object... args) {
169         return format(Locale.getDefault(), format, args);
170     }
171 
172     @Override
printf(String format, Object... args)173     public PrintStream printf(String format, Object... args) {
174         return format(format, args);
175     }
176 
177     @Override
printf(Locale l, String format, Object... args)178     public PrintStream printf(Locale l, String format, Object... args) {
179         return format(l, format, args);
180     }
181 
182     private final Formatter formatter = new Formatter(builder, null);
183 
184     @Override
format( Locale l, String format, Object... args)185     public synchronized PrintStream format(
186             Locale l, String format, Object... args) {
187         if (format == null) {
188             throw new NullPointerException("format");
189         }
190 
191         formatter.format(l, format, args);
192         flush(false);
193         return this;
194     }
195 
196     @Override
print(char[] charArray)197     public synchronized void print(char[] charArray) {
198         builder.append(charArray);
199         flush(false);
200     }
201 
202     @Override
print(char ch)203     public synchronized void print(char ch) {
204         builder.append(ch);
205         if (ch == '\n') {
206             flush(false);
207         }
208     }
209 
210     @Override
print(double dnum)211     public synchronized void print(double dnum) {
212         builder.append(dnum);
213     }
214 
215     @Override
print(float fnum)216     public synchronized void print(float fnum) {
217         builder.append(fnum);
218     }
219 
220     @Override
print(int inum)221     public synchronized void print(int inum) {
222         builder.append(inum);
223     }
224 
225     @Override
print(long lnum)226     public synchronized void print(long lnum) {
227         builder.append(lnum);
228     }
229 
230     @Override
print(Object obj)231     public synchronized void print(Object obj) {
232         builder.append(obj);
233         flush(false);
234     }
235 
236     @Override
print(String str)237     public synchronized void print(String str) {
238         builder.append(str);
239         flush(false);
240     }
241 
242     @Override
print(boolean bool)243     public synchronized void print(boolean bool) {
244         builder.append(bool);
245     }
246 
247     @Override
println()248     public synchronized void println() {
249         flush(true);
250     }
251 
252     @Override
println(char[] charArray)253     public synchronized void println(char[] charArray) {
254         builder.append(charArray);
255         flush(true);
256     }
257 
258     @Override
println(char ch)259     public synchronized void println(char ch) {
260         builder.append(ch);
261         flush(true);
262     }
263 
264     @Override
println(double dnum)265     public synchronized void println(double dnum) {
266         builder.append(dnum);
267         flush(true);
268     }
269 
270     @Override
println(float fnum)271     public synchronized void println(float fnum) {
272         builder.append(fnum);
273         flush(true);
274     }
275 
276     @Override
println(int inum)277     public synchronized void println(int inum) {
278         builder.append(inum);
279         flush(true);
280     }
281 
282     @Override
println(long lnum)283     public synchronized void println(long lnum) {
284         builder.append(lnum);
285         flush(true);
286     }
287 
288     @Override
println(Object obj)289     public synchronized void println(Object obj) {
290         builder.append(obj);
291         flush(true);
292     }
293 
294     @Override
println(String s)295     public synchronized void println(String s) {
296         if (builder.length() == 0 && s != null) {
297             // Optimization for a simple println.
298             int length = s.length();
299 
300             int start = 0;
301             int nextBreak;
302 
303             // Log one line for each line break.
304             while (start < length
305                     && (nextBreak = s.indexOf('\n', start)) != -1) {
306                 log(s.substring(start, nextBreak));
307                 start = nextBreak + 1;
308             }
309 
310             if (start < length) {
311                 log(s.substring(start));
312             }
313         } else {
314             builder.append(s);
315             flush(true);
316         }
317     }
318 
319     @Override
println(boolean bool)320     public synchronized void println(boolean bool) {
321         builder.append(bool);
322         flush(true);
323     }
324 
325     @Override
append(char c)326     public synchronized PrintStream append(char c) {
327         print(c);
328         return this;
329     }
330 
331     @Override
append(CharSequence csq)332     public synchronized PrintStream append(CharSequence csq) {
333         builder.append(csq);
334         flush(false);
335         return this;
336     }
337 
338     @Override
append( CharSequence csq, int start, int end)339     public synchronized PrintStream append(
340             CharSequence csq, int start, int end) {
341         builder.append(csq, start, end);
342         flush(false);
343         return this;
344     }
345 }
346