...
1
2
3
4
5
6
7 package ascii85
8
9 import (
10 "io"
11 "strconv"
12 )
13
14
17
18
19
20
21
22
23
24
25
26
27 func Encode(dst, src []byte) int {
28 if len(src) == 0 {
29 return 0
30 }
31
32 n := 0
33 for len(src) > 0 {
34 dst[0] = 0
35 dst[1] = 0
36 dst[2] = 0
37 dst[3] = 0
38 dst[4] = 0
39
40
41 var v uint32
42 switch len(src) {
43 default:
44 v |= uint32(src[3])
45 fallthrough
46 case 3:
47 v |= uint32(src[2]) << 8
48 fallthrough
49 case 2:
50 v |= uint32(src[1]) << 16
51 fallthrough
52 case 1:
53 v |= uint32(src[0]) << 24
54 }
55
56
57 if v == 0 && len(src) >= 4 {
58 dst[0] = 'z'
59 dst = dst[1:]
60 src = src[4:]
61 n++
62 continue
63 }
64
65
66 for i := 4; i >= 0; i-- {
67 dst[i] = '!' + byte(v%85)
68 v /= 85
69 }
70
71
72 m := 5
73 if len(src) < 4 {
74 m -= 4 - len(src)
75 src = nil
76 } else {
77 src = src[4:]
78 }
79 dst = dst[m:]
80 n += m
81 }
82 return n
83 }
84
85
86 func MaxEncodedLen(n int) int { return (n + 3) / 4 * 5 }
87
88
89
90
91
92
93 func NewEncoder(w io.Writer) io.WriteCloser { return &encoder{w: w} }
94
95 type encoder struct {
96 err error
97 w io.Writer
98 buf [4]byte
99 nbuf int
100 out [1024]byte
101 }
102
103 func (e *encoder) Write(p []byte) (n int, err error) {
104 if e.err != nil {
105 return 0, e.err
106 }
107
108
109 if e.nbuf > 0 {
110 var i int
111 for i = 0; i < len(p) && e.nbuf < 4; i++ {
112 e.buf[e.nbuf] = p[i]
113 e.nbuf++
114 }
115 n += i
116 p = p[i:]
117 if e.nbuf < 4 {
118 return
119 }
120 nout := Encode(e.out[0:], e.buf[0:])
121 if _, e.err = e.w.Write(e.out[0:nout]); e.err != nil {
122 return n, e.err
123 }
124 e.nbuf = 0
125 }
126
127
128 for len(p) >= 4 {
129 nn := len(e.out) / 5 * 4
130 if nn > len(p) {
131 nn = len(p)
132 }
133 nn -= nn % 4
134 if nn > 0 {
135 nout := Encode(e.out[0:], p[0:nn])
136 if _, e.err = e.w.Write(e.out[0:nout]); e.err != nil {
137 return n, e.err
138 }
139 }
140 n += nn
141 p = p[nn:]
142 }
143
144
145 copy(e.buf[:], p)
146 e.nbuf = len(p)
147 n += len(p)
148 return
149 }
150
151
152
153 func (e *encoder) Close() error {
154
155 if e.err == nil && e.nbuf > 0 {
156 nout := Encode(e.out[0:], e.buf[0:e.nbuf])
157 e.nbuf = 0
158 _, e.err = e.w.Write(e.out[0:nout])
159 }
160 return e.err
161 }
162
163
166
167 type CorruptInputError int64
168
169 func (e CorruptInputError) Error() string {
170 return "illegal ascii85 data at input byte " + strconv.FormatInt(int64(e), 10)
171 }
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186 func Decode(dst, src []byte, flush bool) (ndst, nsrc int, err error) {
187 var v uint32
188 var nb int
189 for i, b := range src {
190 if len(dst)-ndst < 4 {
191 return
192 }
193 switch {
194 case b <= ' ':
195 continue
196 case b == 'z' && nb == 0:
197 nb = 5
198 v = 0
199 case '!' <= b && b <= 'u':
200 v = v*85 + uint32(b-'!')
201 nb++
202 default:
203 return 0, 0, CorruptInputError(i)
204 }
205 if nb == 5 {
206 nsrc = i + 1
207 dst[ndst] = byte(v >> 24)
208 dst[ndst+1] = byte(v >> 16)
209 dst[ndst+2] = byte(v >> 8)
210 dst[ndst+3] = byte(v)
211 ndst += 4
212 nb = 0
213 v = 0
214 }
215 }
216 if flush {
217 nsrc = len(src)
218 if nb > 0 {
219
220
221
222
223 if nb == 1 {
224 return 0, 0, CorruptInputError(len(src))
225 }
226 for i := nb; i < 5; i++ {
227
228
229
230 v = v*85 + 84
231 }
232 for i := 0; i < nb-1; i++ {
233 dst[ndst] = byte(v >> 24)
234 v <<= 8
235 ndst++
236 }
237 }
238 }
239 return
240 }
241
242
243 func NewDecoder(r io.Reader) io.Reader { return &decoder{r: r} }
244
245 type decoder struct {
246 err error
247 readErr error
248 r io.Reader
249 buf [1024]byte
250 nbuf int
251 out []byte
252 outbuf [1024]byte
253 }
254
255 func (d *decoder) Read(p []byte) (n int, err error) {
256 if len(p) == 0 {
257 return 0, nil
258 }
259 if d.err != nil {
260 return 0, d.err
261 }
262
263 for {
264
265 if len(d.out) > 0 {
266 n = copy(p, d.out)
267 d.out = d.out[n:]
268 return
269 }
270
271
272 var nn, nsrc, ndst int
273 if d.nbuf > 0 {
274 ndst, nsrc, d.err = Decode(d.outbuf[0:], d.buf[0:d.nbuf], d.readErr != nil)
275 if ndst > 0 {
276 d.out = d.outbuf[0:ndst]
277 d.nbuf = copy(d.buf[0:], d.buf[nsrc:d.nbuf])
278 continue
279 }
280 if ndst == 0 && d.err == nil {
281
282
283 off := 0
284 for i := 0; i < d.nbuf; i++ {
285 if d.buf[i] > ' ' {
286 d.buf[off] = d.buf[i]
287 off++
288 }
289 }
290 d.nbuf = off
291 }
292 }
293
294
295 if d.err != nil {
296 return 0, d.err
297 }
298 if d.readErr != nil {
299 d.err = d.readErr
300 return 0, d.err
301 }
302
303
304 nn, d.readErr = d.r.Read(d.buf[d.nbuf:])
305 d.nbuf += nn
306 }
307 }
308
View as plain text