OLD | NEW |
| (Empty) |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | |
2 // for details. All rights reserved. Use of this source code is governed by a | |
3 // BSD-style license that can be found in the LICENSE file. | |
4 | |
5 part of dart.convert; | |
6 | |
7 /** | |
8 * The [ByteConversionSink] provides an interface for converters to | |
9 * efficiently transmit byte data. | |
10 * | |
11 * Instead of limiting the interface to one non-chunked list of bytes it | |
12 * accepts its input in chunks (themselves being lists of bytes). | |
13 * | |
14 * This abstract class will likely get more methods over time. Implementers are | |
15 * urged to extend or mix in [ByteConversionSinkBase] to ensure that their | |
16 * class covers the newly added methods. | |
17 */ | |
18 abstract class ByteConversionSink extends ChunkedConversionSink<List<int>> { | |
19 ByteConversionSink(); | |
20 factory ByteConversionSink.withCallback(void callback(List<int> accumulated)) | |
21 = _ByteCallbackSink; | |
22 factory ByteConversionSink.from(Sink<List<int>> sink) | |
23 = _ByteAdapterSink; | |
24 | |
25 /** | |
26 * Adds the next [chunk] to `this`. | |
27 * | |
28 * Adds the bytes defined by [start] and [end]-exclusive to `this`. | |
29 * | |
30 * If [isLast] is `true` closes `this`. | |
31 * | |
32 * Contrary to `add` the given [chunk] must not be held onto. Once the method | |
33 * returns, it is safe to overwrite the data in it. | |
34 */ | |
35 void addSlice(List<int> chunk, int start, int end, bool isLast); | |
36 | |
37 // TODO(floitsch): add more methods: | |
38 // - iterateBytes. | |
39 } | |
40 | |
41 /** | |
42 * This class provides a base-class for converters that need to accept byte | |
43 * inputs. | |
44 */ | |
45 abstract class ByteConversionSinkBase extends ByteConversionSink { | |
46 | |
47 void add(List<int> chunk); | |
48 void close(); | |
49 | |
50 void addSlice(List<int> chunk, int start, int end, bool isLast) { | |
51 add(chunk.sublist(start, end)); | |
52 if (isLast) close(); | |
53 } | |
54 } | |
55 | |
56 /** | |
57 * This class adapts a simple [Sink] to a [ByteConversionSink]. | |
58 * | |
59 * All additional methods of the [ByteConversionSink] (compared to the | |
60 * ChunkedConversionSink) are redirected to the `add` method. | |
61 */ | |
62 class _ByteAdapterSink extends ByteConversionSinkBase { | |
63 final Sink<List<int>> _sink; | |
64 | |
65 _ByteAdapterSink(this._sink); | |
66 | |
67 void add(List<int> chunk) { _sink.add(chunk); } | |
68 void close() { _sink.close(); } | |
69 } | |
70 | |
71 /** | |
72 * This class accumulates all chunks into one list of bytes | |
73 * and invokes a callback when the sink is closed. | |
74 * | |
75 * This class can be used to terminate a chunked conversion. | |
76 */ | |
77 class _ByteCallbackSink extends ByteConversionSinkBase { | |
78 static const _INITIAL_BUFFER_SIZE = 1024; | |
79 | |
80 final _ChunkedConversionCallback<List<int>> _callback; | |
81 List<int> _buffer = new Uint8List(_INITIAL_BUFFER_SIZE); | |
82 int _bufferIndex = 0; | |
83 | |
84 _ByteCallbackSink(void callback(List<int> accumulated)) | |
85 : this._callback = callback; | |
86 | |
87 void add(Iterable<int> chunk) { | |
88 int freeCount = _buffer.length - _bufferIndex; | |
89 if (chunk.length > freeCount) { | |
90 // Grow the buffer. | |
91 int oldLength = _buffer.length; | |
92 int newLength = _roundToPowerOf2(chunk.length + oldLength) * 2; | |
93 List<int> grown = new Uint8List(newLength); | |
94 grown.setRange(0, _buffer.length, _buffer); | |
95 _buffer = grown; | |
96 } | |
97 _buffer.setRange(_bufferIndex, _bufferIndex + chunk.length, chunk); | |
98 _bufferIndex += chunk.length; | |
99 } | |
100 | |
101 static int _roundToPowerOf2(int v) { | |
102 assert(v > 0); | |
103 v--; | |
104 v |= v >> 1; | |
105 v |= v >> 2; | |
106 v |= v >> 4; | |
107 v |= v >> 8; | |
108 v |= v >> 16; | |
109 v++; | |
110 return v; | |
111 } | |
112 | |
113 void close() { | |
114 _callback(_buffer.sublist(0, _bufferIndex)); | |
115 } | |
116 } | |
OLD | NEW |