blob: 7a16abe2cb2ad0649a0585a1177076d502eb65e2 [file] [log] [blame]
Nigel Tao981bf192018-05-23 12:17:55 +10001// Copyright 2018 The Wuffs Authors.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// https://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15// +build ignore
16
17package main
18
19// make-artificial.go makes test data in various formats.
20//
21// See test/data/artificial/*.make.txt for examples.
22//
23// Usage: go run make-artificial.go < foo.make.txt
24
25import (
26 "bytes"
27 "compress/lzw"
28 "fmt"
29 "io/ioutil"
30 "os"
31 "strconv"
32 "strings"
33)
34
35type stateFunc func(line string) (stateFunc, error)
36
37type repeat struct {
38 count uint32
39 remaining string
40}
41
42var (
43 state stateFunc
44 repeats []repeat
45 out []byte
46 formats = map[string]stateFunc{}
47)
48
49func main() {
50 if err := main1(); err != nil {
51 os.Stderr.WriteString(err.Error() + "\n")
52 os.Exit(1)
53 }
54}
55
56func main1() error {
57 stdin, err := ioutil.ReadAll(os.Stdin)
58 if err != nil {
59 return err
60 }
61 s := string(stdin)
62 for remaining := ""; len(s) > 0; s, remaining = remaining, "" {
63 if i := strings.IndexByte(s, '\n'); i >= 0 {
64 s, remaining = s[:i], s[i+1:]
65 }
66 s = strings.TrimSpace(s)
67 if s == "" || s[0] == '#' {
68 continue
69 }
70
71 if state == nil {
72 const m = "make "
73 if !strings.HasPrefix(s, m) {
74 return fmt.Errorf(`input must start with "make foo"`)
75 }
76 s = s[len(m):]
77 state = formats[s]
78 if state == nil {
79 return fmt.Errorf("unsupported format %q", s)
80 }
81 continue
82 }
83
84 const rep = "repeat "
85 if strings.HasPrefix(s, rep) {
86 args := s[len(rep):]
87 count, args, ok := parseNum(args)
88 if !ok || count <= 0 || args != "[" {
89 return fmt.Errorf("bad repeat command: %q", s)
90 }
91 repeats = append(repeats, repeat{
92 count: count,
93 remaining: remaining,
94 })
95 continue
96 }
97
98 if s == "]" {
99 if len(repeats) <= 0 {
100 return fmt.Errorf(`unbalanced close-repeat command: "]"`)
101 }
102 i := len(repeats) - 1
103 repeats[i].count--
104 if repeats[i].count == 0 {
105 repeats = repeats[:i]
106 } else {
107 remaining = repeats[i].remaining
108 }
109 continue
110 }
111
112 state, err = state(s)
113 if err != nil {
114 return err
115 }
116 if state == nil {
117 return fmt.Errorf("bad state transition")
118 }
119 }
120
Nigel Tao12347602019-06-16 23:15:05 +1000121 if state == nil {
122 return fmt.Errorf("no 'make' line")
123 } else {
124 state, err = state("")
125 if err != nil {
126 return err
127 }
128 }
129
Nigel Tao981bf192018-05-23 12:17:55 +1000130 _, err = os.Stdout.Write(out)
131 return err
132}
133
134// ----
135
136func appendU16LE(b []byte, u uint16) []byte {
137 return append(b, uint8(u), uint8(u>>8))
138}
139
140func log2(u uint32) (i int32) {
141 for i, pow := uint32(0), uint32(1); i < 32; i, pow = i+1, pow<<1 {
142 if u == pow {
143 return int32(i)
144 }
145 if u < pow {
146 break
147 }
148 }
149 return -1
150}
151
152func parseHex(s string) (num uint32, remaining string, ok bool) {
153 if i := strings.IndexByte(s, ' '); i >= 0 {
154 s, remaining = s[:i], s[i+1:]
155 for len(remaining) > 0 && remaining[0] == ' ' {
156 remaining = remaining[1:]
157 }
158 }
159
160 if len(s) < 2 || s[0] != '0' || s[1] != 'x' {
161 return 0, "", false
162 }
163 s = s[2:]
164
165 u, err := strconv.ParseUint(s, 16, 32)
166 if err != nil {
167 return 0, "", false
168 }
169 return uint32(u), remaining, true
170}
171
172func parseNum(s string) (num uint32, remaining string, ok bool) {
173 if i := strings.IndexByte(s, ' '); i >= 0 {
174 s, remaining = s[:i], s[i+1:]
175 for len(remaining) > 0 && remaining[0] == ' ' {
176 remaining = remaining[1:]
177 }
178 }
179
180 u, err := strconv.ParseUint(s, 10, 32)
181 if err != nil {
182 return 0, "", false
183 }
184 return uint32(u), remaining, true
185}
186
187func reverse(x uint32, n uint32) uint32 {
188 x = uint32(reverse8[0xFF&(x>>0)])<<24 |
189 uint32(reverse8[0xFF&(x>>8)])<<16 |
190 uint32(reverse8[0xFF&(x>>16)])<<8 |
191 uint32(reverse8[0xFF&(x>>24)])<<0
192 return x >> (32 - n)
193}
194
195var reverse8 = [256]byte{
196 0x00, 0x80, 0x40, 0xC0, 0x20, 0xA0, 0x60, 0xE0, // 0x00 - 0x07
197 0x10, 0x90, 0x50, 0xD0, 0x30, 0xB0, 0x70, 0xF0, // 0x08 - 0x0F
198 0x08, 0x88, 0x48, 0xC8, 0x28, 0xA8, 0x68, 0xE8, // 0x10 - 0x17
199 0x18, 0x98, 0x58, 0xD8, 0x38, 0xB8, 0x78, 0xF8, // 0x18 - 0x1F
200 0x04, 0x84, 0x44, 0xC4, 0x24, 0xA4, 0x64, 0xE4, // 0x20 - 0x27
201 0x14, 0x94, 0x54, 0xD4, 0x34, 0xB4, 0x74, 0xF4, // 0x28 - 0x2F
202 0x0C, 0x8C, 0x4C, 0xCC, 0x2C, 0xAC, 0x6C, 0xEC, // 0x30 - 0x37
203 0x1C, 0x9C, 0x5C, 0xDC, 0x3C, 0xBC, 0x7C, 0xFC, // 0x38 - 0x3F
204 0x02, 0x82, 0x42, 0xC2, 0x22, 0xA2, 0x62, 0xE2, // 0x40 - 0x47
205 0x12, 0x92, 0x52, 0xD2, 0x32, 0xB2, 0x72, 0xF2, // 0x48 - 0x4F
206 0x0A, 0x8A, 0x4A, 0xCA, 0x2A, 0xAA, 0x6A, 0xEA, // 0x50 - 0x57
207 0x1A, 0x9A, 0x5A, 0xDA, 0x3A, 0xBA, 0x7A, 0xFA, // 0x58 - 0x5F
208 0x06, 0x86, 0x46, 0xC6, 0x26, 0xA6, 0x66, 0xE6, // 0x60 - 0x67
209 0x16, 0x96, 0x56, 0xD6, 0x36, 0xB6, 0x76, 0xF6, // 0x68 - 0x6F
210 0x0E, 0x8E, 0x4E, 0xCE, 0x2E, 0xAE, 0x6E, 0xEE, // 0x70 - 0x77
211 0x1E, 0x9E, 0x5E, 0xDE, 0x3E, 0xBE, 0x7E, 0xFE, // 0x78 - 0x7F
212 0x01, 0x81, 0x41, 0xC1, 0x21, 0xA1, 0x61, 0xE1, // 0x80 - 0x87
213 0x11, 0x91, 0x51, 0xD1, 0x31, 0xB1, 0x71, 0xF1, // 0x88 - 0x8F
214 0x09, 0x89, 0x49, 0xC9, 0x29, 0xA9, 0x69, 0xE9, // 0x90 - 0x97
215 0x19, 0x99, 0x59, 0xD9, 0x39, 0xB9, 0x79, 0xF9, // 0x98 - 0x9F
216 0x05, 0x85, 0x45, 0xC5, 0x25, 0xA5, 0x65, 0xE5, // 0xA0 - 0xA7
217 0x15, 0x95, 0x55, 0xD5, 0x35, 0xB5, 0x75, 0xF5, // 0xA8 - 0xAF
218 0x0D, 0x8D, 0x4D, 0xCD, 0x2D, 0xAD, 0x6D, 0xED, // 0xB0 - 0xB7
219 0x1D, 0x9D, 0x5D, 0xDD, 0x3D, 0xBD, 0x7D, 0xFD, // 0xB8 - 0xBF
220 0x03, 0x83, 0x43, 0xC3, 0x23, 0xA3, 0x63, 0xE3, // 0xC0 - 0xC7
221 0x13, 0x93, 0x53, 0xD3, 0x33, 0xB3, 0x73, 0xF3, // 0xC8 - 0xCF
222 0x0B, 0x8B, 0x4B, 0xCB, 0x2B, 0xAB, 0x6B, 0xEB, // 0xD0 - 0xD7
223 0x1B, 0x9B, 0x5B, 0xDB, 0x3B, 0xBB, 0x7B, 0xFB, // 0xD8 - 0xDF
224 0x07, 0x87, 0x47, 0xC7, 0x27, 0xA7, 0x67, 0xE7, // 0xE0 - 0xE7
225 0x17, 0x97, 0x57, 0xD7, 0x37, 0xB7, 0x77, 0xF7, // 0xE8 - 0xEF
226 0x0F, 0x8F, 0x4F, 0xCF, 0x2F, 0xAF, 0x6F, 0xEF, // 0xF0 - 0xF7
227 0x1F, 0x9F, 0x5F, 0xDF, 0x3F, 0xBF, 0x7F, 0xFF, // 0xF8 - 0xFF
228}
229
230// ----
231
232func init() {
233 formats["deflate"] = stateDeflate
234}
235
Nigel Tao12347602019-06-16 23:15:05 +1000236var deflateCodeOrder = [19]uint32{
237 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15,
238}
239
240type deflateHuffmanTable map[uint32]string
241
Nigel Tao981bf192018-05-23 12:17:55 +1000242var deflateGlobals struct {
243 bncData []byte
244 stream deflateBitStream
Nigel Tao12347602019-06-16 23:15:05 +1000245
246 // Dynamic Huffman state.
Nigel Taoff7bffc2019-10-13 23:23:48 +1100247 whichHuffman uint32
Nigel Tao12347602019-06-16 23:15:05 +1000248 // 0=Unused, 1=CodeLength, 2=Length/Literal, 3=Distance.
249 huffmans [4]deflateHuffmanTable
Nigel Taoff7bffc2019-10-13 23:23:48 +1100250
251 // DHH (Dynamic Huffman, inside a Huffman table) state.
252 prevLine string
253 etcetera bool
Nigel Tao12347602019-06-16 23:15:05 +1000254}
255
256func deflateGlobalsClearDynamicHuffmanState() {
Nigel Tao12347602019-06-16 23:15:05 +1000257 deflateGlobals.whichHuffman = 0
258 deflateGlobals.huffmans = [4]deflateHuffmanTable{}
Nigel Taoff7bffc2019-10-13 23:23:48 +1100259 deflateGlobals.prevLine = ""
260 deflateGlobals.etcetera = false
261}
262
263func deflateGlobalsCountCodes() (numLCodes uint32, numDCodes uint32, numCLCodeLengths uint32, retErr error) {
264 for k := range deflateGlobals.huffmans[2] {
265 if numLCodes < (k + 1) {
266 numLCodes = (k + 1)
267 }
268 }
269
270 for k := range deflateGlobals.huffmans[3] {
271 if numDCodes < (k + 1) {
272 numDCodes = (k + 1)
273 }
274 }
275
276 for k := range deflateGlobals.huffmans[1] {
277 if (k < 0) || (18 < k) {
278 return 0, 0, 0, fmt.Errorf("bad CodeLength: %d", k)
279 }
280 }
281 for i := len(deflateCodeOrder) - 1; i >= 0; i-- {
282 cl := deflateCodeOrder[i]
283 if _, ok := deflateGlobals.huffmans[1][cl]; ok {
284 numCLCodeLengths = uint32(i + 1)
285 break
286 }
287 }
288 if numCLCodeLengths < 4 {
289 numCLCodeLengths = 4
290 }
291
292 return numLCodes, numDCodes, numCLCodeLengths, nil
Nigel Tao12347602019-06-16 23:15:05 +1000293}
294
295func deflateGlobalsWriteDynamicHuffmanTables() error {
296 g := &deflateGlobals
Nigel Taoff7bffc2019-10-13 23:23:48 +1100297 numLCodes, numDCodes, numCLCodeLengths, err := deflateGlobalsCountCodes()
298 if err != nil {
299 return err
Nigel Tao12347602019-06-16 23:15:05 +1000300 }
Nigel Taoff7bffc2019-10-13 23:23:48 +1100301 if (numLCodes < 257) || (257+31 < numLCodes) {
302 return fmt.Errorf("bad numLCodes: %d", numLCodes)
Nigel Tao12347602019-06-16 23:15:05 +1000303 }
Nigel Taoff7bffc2019-10-13 23:23:48 +1100304 g.stream.writeBits(numLCodes-257, 5)
305 if (numDCodes < 1) || (1+31 < numDCodes) {
306 return fmt.Errorf("bad numDCodes: %d", numDCodes)
Nigel Tao12347602019-06-16 23:15:05 +1000307 }
Nigel Taoff7bffc2019-10-13 23:23:48 +1100308 g.stream.writeBits(numDCodes-1, 5)
309 if (numCLCodeLengths < 4) || (4+15 < numCLCodeLengths) {
310 return fmt.Errorf("bad numCLCodeLengths: %d", numCLCodeLengths)
311 }
312 g.stream.writeBits(numCLCodeLengths-4, 4)
Nigel Tao12347602019-06-16 23:15:05 +1000313
314 // Write the Huffman table for CodeLength.
315 {
Nigel Taoff7bffc2019-10-13 23:23:48 +1100316 for i := uint32(0); i < numCLCodeLengths; i++ {
Nigel Tao12347602019-06-16 23:15:05 +1000317 n := len(g.huffmans[1][deflateCodeOrder[i]])
318 g.stream.writeBits(uint32(n), 3)
319 }
Nigel Taoff7bffc2019-10-13 23:23:48 +1100320 for i := numCLCodeLengths; i < uint32(len(deflateCodeOrder)); i++ {
Nigel Tao12347602019-06-16 23:15:05 +1000321 n := len(g.huffmans[1][deflateCodeOrder[i]])
322 if n > 0 {
Nigel Taoff7bffc2019-10-13 23:23:48 +1100323 return fmt.Errorf("short numCLCodeLengths: %d", numCLCodeLengths)
Nigel Tao12347602019-06-16 23:15:05 +1000324 }
325 }
326 }
327
328 // Write the Huffman tables for Length/Literal and Distance.
329 {
330 numZeroes := uint32(0)
Nigel Taoff7bffc2019-10-13 23:23:48 +1100331 for i := uint32(0); i < numLCodes+numDCodes; i++ {
332 codeLen := uint32(0)
333 if i < numLCodes {
334 codeLen = uint32(len(g.huffmans[2][i]))
Nigel Tao12347602019-06-16 23:15:05 +1000335 } else {
Nigel Taoff7bffc2019-10-13 23:23:48 +1100336 codeLen = uint32(len(g.huffmans[3][i-numLCodes]))
Nigel Tao12347602019-06-16 23:15:05 +1000337 }
Nigel Taoff7bffc2019-10-13 23:23:48 +1100338 if codeLen == 0 {
Nigel Tao12347602019-06-16 23:15:05 +1000339 numZeroes++
340 continue
341 }
342
343 if err := deflateGlobalsWriteDynamicHuffmanZeroes(numZeroes); err != nil {
344 return err
345 }
346 numZeroes = 0
347
Nigel Taoff7bffc2019-10-13 23:23:48 +1100348 codeLenCode := g.huffmans[1][codeLen]
349 if codeLenCode == "" {
350 return fmt.Errorf("no code for code-length %d", codeLen)
351 }
352 deflateGlobalsWriteDynamicHuffmanBits(g.huffmans[1][codeLen])
Nigel Tao12347602019-06-16 23:15:05 +1000353 }
354 if err := deflateGlobalsWriteDynamicHuffmanZeroes(numZeroes); err != nil {
355 return err
356 }
357 }
358
359 return nil
360}
361
362func deflateGlobalsWriteDynamicHuffmanZeroes(numZeroes uint32) error {
363 g := &deflateGlobals
364 if numZeroes == 0 {
365 return nil
366 }
367
368 if s := g.huffmans[1][18]; s != "" {
369 for numZeroes >= 11 {
370 extra := numZeroes - 11
371 if extra > 127 {
372 extra = 127
373 }
374 deflateGlobalsWriteDynamicHuffmanBits(s)
375 g.stream.writeBits(extra, 7)
376 numZeroes -= 11 + extra
377 }
378 }
379
380 if s := g.huffmans[1][17]; s != "" {
381 for numZeroes >= 3 {
382 extra := numZeroes - 3
383 if extra > 7 {
384 extra = 7
385 }
386 deflateGlobalsWriteDynamicHuffmanBits(s)
387 g.stream.writeBits(extra, 3)
388 numZeroes -= 3 + extra
389 }
390 }
391
392 if s := g.huffmans[1][0]; s != "" {
393 for ; numZeroes > 0; numZeroes-- {
394 deflateGlobalsWriteDynamicHuffmanBits(s)
395 }
396 }
397
398 if numZeroes > 0 {
399 return fmt.Errorf("could not write a run of zero-valued code lengths")
400 }
401 return nil
402}
403
404func deflateGlobalsWriteDynamicHuffmanBits(s string) {
405 g := &deflateGlobals
406 for i := 0; i < len(s); i++ {
407 g.stream.writeBits(uint32(s[i]&1), 1)
408 }
409}
410
411func parseDeflateWhichHuffman(s string) (num uint32, remaining string, ok bool) {
412 if i := strings.IndexByte(s, ' '); i >= 0 {
413 s, remaining = s[:i], s[i+1:]
414 for len(remaining) > 0 && remaining[0] == ' ' {
415 remaining = remaining[1:]
416 }
417 }
418
419 switch s {
420 case "CodeLength":
421 return 1, remaining, true
422 case "Length/Literal":
423 return 2, remaining, true
424 case "Distance":
425 return 3, remaining, true
426 }
427 return 0, "", false
Nigel Tao981bf192018-05-23 12:17:55 +1000428}
429
430func stateDeflate(line string) (stateFunc, error) {
431 g := &deflateGlobals
432 const (
Nigel Tao0cc17982019-05-19 23:43:20 +1000433 cmdB = "bytes "
Nigel Tao12347602019-06-16 23:15:05 +1000434 cmdBDH = "blockDynamicHuffman "
Nigel Tao981bf192018-05-23 12:17:55 +1000435 cmdBFH = "blockFixedHuffman "
Nigel Tao12347602019-06-16 23:15:05 +1000436 cmdBNC = "blockNoCompression "
Nigel Tao981bf192018-05-23 12:17:55 +1000437 )
438 bits := uint32(0)
439 s := ""
440
441 retState := stateFunc(nil)
442 switch {
Nigel Tao12347602019-06-16 23:15:05 +1000443 case line == "":
444 g.stream.flush()
445 return stateDeflate, nil
446
Nigel Tao0cc17982019-05-19 23:43:20 +1000447 case strings.HasPrefix(line, cmdB):
448 s := line[len(cmdB):]
449 for s != "" {
450 x, ok := uint32(0), false
451 x, s, ok = parseHex(s)
452 if !ok {
453 return nil, fmt.Errorf("bad stateDeflate command: %q", line)
454 }
455 out = append(out, uint8(x))
456 }
457 return stateDeflate, nil
458
Nigel Tao981bf192018-05-23 12:17:55 +1000459 case strings.HasPrefix(line, cmdBNC):
460 s = line[len(cmdBNC):]
461 retState = stateDeflateNoCompression
462 bits |= 0 << 1
463 case strings.HasPrefix(line, cmdBFH):
464 s = line[len(cmdBFH):]
465 retState = stateDeflateFixedHuffman
466 bits |= 1 << 1
Nigel Tao12347602019-06-16 23:15:05 +1000467 case strings.HasPrefix(line, cmdBDH):
468 s = line[len(cmdBDH):]
469 retState = stateDeflateDynamicHuffman
470 bits |= 2 << 1
Nigel Tao981bf192018-05-23 12:17:55 +1000471 default:
472 return nil, fmt.Errorf("bad stateDeflate command: %q", line)
473 }
474
475 switch s {
476 case "(final) {":
477 bits |= 1
478 case "(nonFinal) {":
479 // No-op.
480 default:
481 return nil, fmt.Errorf("bad stateDeflate command: %q", line)
482 }
483
484 g.stream.writeBits(bits, 3)
485 return retState, nil
486}
487
488func stateDeflateNoCompression(line string) (stateFunc, error) {
489 g := &deflateGlobals
490 if line == "}" {
491 if len(g.bncData) > 0xFFFF {
492 return nil, fmt.Errorf("bncData is too long")
493 }
494 n := uint32(len(g.bncData))
495 g.stream.flush()
496 g.stream.writeBits(n, 16)
497 g.stream.writeBits(0xFFFF-n, 16)
498 g.stream.writeBytes(g.bncData)
499 g.bncData = g.bncData[:0]
500 return stateDeflate, nil
501 }
502
503 if lit, ok := deflateParseLiteral(line); ok {
504 g.bncData = append(g.bncData, lit...)
505 return stateDeflateNoCompression, nil
506 }
507
508 return nil, fmt.Errorf("bad blockNoCompression command: %q", line)
509}
510
511func stateDeflateFixedHuffman(line string) (stateFunc, error) {
512 g := &deflateGlobals
513 if line == "}" {
Nigel Tao981bf192018-05-23 12:17:55 +1000514 return stateDeflate, nil
515 }
516
517 if line == "endOfBlock" {
518 g.stream.writeBits(0, 7)
519 return stateDeflateFixedHuffman, nil
520 }
521
522 if lit, ok := deflateParseLiteral(line); ok {
Nigel Tao981bf192018-05-23 12:17:55 +1000523 for i := 0; i < len(lit); i++ {
Nigel Tao12347602019-06-16 23:15:05 +1000524 g.stream.writeFixedHuffmanLCode(uint32(lit[i]))
Nigel Tao981bf192018-05-23 12:17:55 +1000525 }
526 return stateDeflateFixedHuffman, nil
527 }
528
Nigel Tao11a68192019-02-02 13:04:09 +1100529 if line == "len 3 distCode 31" {
530 lCode, lExtra, lNExtra := deflateEncodeLength(3)
Nigel Tao12347602019-06-16 23:15:05 +1000531 g.stream.writeFixedHuffmanLCode(lCode)
Nigel Tao11a68192019-02-02 13:04:09 +1100532 g.stream.writeBits(lExtra, lNExtra)
533 dCode, dExtra, dNExtra := uint32(31), uint32(0), uint32(0)
534 g.stream.writeBits(reverse(dCode, 5), 5)
535 g.stream.writeBits(dExtra, dNExtra)
536 return stateDeflateFixedHuffman, nil
537 }
538
Nigel Tao981bf192018-05-23 12:17:55 +1000539 if l, d, ok := deflateParseLenDist(line); ok {
540 lCode, lExtra, lNExtra := deflateEncodeLength(l)
Nigel Tao12347602019-06-16 23:15:05 +1000541 g.stream.writeFixedHuffmanLCode(lCode)
Nigel Tao981bf192018-05-23 12:17:55 +1000542 g.stream.writeBits(lExtra, lNExtra)
543 dCode, dExtra, dNExtra := deflateEncodeDistance(d)
544 g.stream.writeBits(reverse(dCode, 5), 5)
545 g.stream.writeBits(dExtra, dNExtra)
546 return stateDeflateFixedHuffman, nil
547 }
548
549 return nil, fmt.Errorf("bad stateDeflateFixedHuffman command: %q", line)
550}
551
Nigel Tao12347602019-06-16 23:15:05 +1000552func stateDeflateDynamicHuffman(line string) (stateFunc, error) {
553 g := &deflateGlobals
554 const (
Nigel Taoff7bffc2019-10-13 23:23:48 +1100555 cmdH = "huffman "
Nigel Tao12347602019-06-16 23:15:05 +1000556 )
557 switch {
558 case line == "}":
559 deflateGlobalsClearDynamicHuffmanState()
560 return stateDeflate, nil
561
562 case strings.HasPrefix(line, cmdH):
563 s := line[len(cmdH):]
564 n, s, ok := parseDeflateWhichHuffman(s)
565 if !ok {
566 break
567 }
568 g.whichHuffman = n
569 return stateDeflateDynamicHuffmanHuffman, nil
Nigel Tao12347602019-06-16 23:15:05 +1000570 }
571
572 if lit, ok := deflateParseLiteral(line); ok {
573 for i := 0; i < len(lit); i++ {
574 s := g.huffmans[2][uint32(lit[i])]
575 if s == "" {
576 return nil, fmt.Errorf("no code for literal %q (%d)", lit[i:i+1], lit[i])
577 }
578 deflateGlobalsWriteDynamicHuffmanBits(s)
579 }
580 return stateDeflateDynamicHuffman, nil
Nigel Taoff7bffc2019-10-13 23:23:48 +1100581
582 } else if l, d, ok := deflateParseLenDist(line); ok {
583 if (l != 3) || (d != 2) {
584 return nil, fmt.Errorf("TODO: support len/dist pairs for dynamic Huffman blocks")
585 }
586 // len 3 is code 257, with no extra bits.
587 if s := g.huffmans[2][257]; s != "" {
588 deflateGlobalsWriteDynamicHuffmanBits(s)
589 } else {
590 return nil, fmt.Errorf("no code for literal/length symbol 257")
591 }
592 // dist 2 is code 1, with no extra bits.
593 if s := g.huffmans[3][1]; s != "" {
594 deflateGlobalsWriteDynamicHuffmanBits(s)
595 } else {
596 return nil, fmt.Errorf("no code for distance symbol 2")
597 }
598 return stateDeflateDynamicHuffman, nil
599
Nigel Tao12347602019-06-16 23:15:05 +1000600 } else if line == "endOfBlock" {
601 s := g.huffmans[2][256]
602 if s == "" {
603 return nil, fmt.Errorf("no code for end-of-block (256)")
604 }
605 deflateGlobalsWriteDynamicHuffmanBits(s)
606 return stateDeflateDynamicHuffman, nil
607 }
608
609 return nil, fmt.Errorf("bad stateDeflateDynamicHuffman command: %q", line)
610}
611
612func stateDeflateDynamicHuffmanHuffman(line string) (stateFunc, error) {
613 g := &deflateGlobals
614outer:
615 switch {
616 case line == "}":
617 g.whichHuffman = 0
Nigel Taoff7bffc2019-10-13 23:23:48 +1100618 g.prevLine = ""
619 g.etcetera = false
Nigel Tao12347602019-06-16 23:15:05 +1000620
621 // If we have all three Huffman tables, write them.
622 for i := 1; ; i++ {
623 if i == 4 {
624 if err := deflateGlobalsWriteDynamicHuffmanTables(); err != nil {
625 return nil, err
626 }
627 break
628 }
629 if g.huffmans[i] == nil {
630 break
631 }
632 }
633
634 return stateDeflateDynamicHuffman, nil
635
Nigel Taoff7bffc2019-10-13 23:23:48 +1100636 case line == "etcetera":
637 g.etcetera = true
638 return stateDeflateDynamicHuffmanHuffman, nil
639
Nigel Tao12347602019-06-16 23:15:05 +1000640 default:
641 s := line
642 n, s, ok := parseNum(s)
643 if !ok || s == "" {
644 break
645 }
646 for i := 0; i < len(s); i++ {
647 if c := s[i]; c != '0' && c != '1' {
648 break outer
649 }
650 }
Nigel Taoff7bffc2019-10-13 23:23:48 +1100651 if (len(s) < 1) || (15 < len(s)) {
652 return nil, fmt.Errorf("%q code length, %d, is out of range", s, len(s))
653 }
654
655 if g.etcetera {
656 g.etcetera = false
657 n0, s0, ok := parseNum(g.prevLine)
658 if !ok {
659 return nil, fmt.Errorf("bad etcetera command")
660 }
661 if err := stateDeflateDHHEtcetera(n0, s0, n, s); err != nil {
662 return nil, err
663 }
664 }
665
Nigel Tao12347602019-06-16 23:15:05 +1000666 if g.huffmans[g.whichHuffman] == nil {
667 g.huffmans[g.whichHuffman] = deflateHuffmanTable{}
668 }
669 g.huffmans[g.whichHuffman][n] = s
Nigel Taoff7bffc2019-10-13 23:23:48 +1100670 g.prevLine = line
Nigel Tao12347602019-06-16 23:15:05 +1000671 return stateDeflateDynamicHuffmanHuffman, nil
672 }
673
674 return nil, fmt.Errorf("bad stateDeflateDynamicHuffmanHuffman command: %q", line)
675}
676
Nigel Taoff7bffc2019-10-13 23:23:48 +1100677// stateDeflateDHHEtcetera expands the "etcetera" line in:
678//
679// 0 0000
680// 1 0001
681// etcetera
682// 7 0111
683//
684// to produce the implicit lines:
685//
686// 0 0000
687// 1 0001
688// 2 0010
689// 3 0011
690// 4 0100
691// 5 0101
692// 6 0110
693// 7 0111
694func stateDeflateDHHEtcetera(n0 uint32, s0 string, n1 uint32, s1 string) error {
695 b := []byte(s0)
696 if !incrementBitstring(b) {
697 return fmt.Errorf("etcetera: could not increment bitstring")
698 }
699 for n := n0 + 1; n < n1; n++ {
700 line := fmt.Sprintf("%d %s", n, b)
701 if _, err := stateDeflateDynamicHuffmanHuffman(line); err != nil {
702 return err
703 }
704 if !incrementBitstring(b) {
705 return fmt.Errorf("etcetera: could not increment bitstring")
706 }
707 }
708 if string(b) != s1 {
709 return fmt.Errorf("etcetera: final bitstring: got %q, want %q", b, s1)
710 }
711 return nil
712}
713
714func incrementBitstring(b []byte) (ok bool) {
715 for i := len(b) - 1; i >= 0; i-- {
716 switch b[i] {
717 case '0':
718 b[i] = '1'
719 return true
720 case '1':
721 b[i] = '0'
722 default:
723 return false
724 }
725 }
726 return false
727}
728
Nigel Tao981bf192018-05-23 12:17:55 +1000729type deflateBitStream struct {
730 bits uint32
731 nBits uint32 // Always within [0, 7].
732}
733
734// writeBits writes the low n bits of b to z.
735func (z *deflateBitStream) writeBits(b uint32, n uint32) {
736 if n > 24 {
737 panic("writeBits: n is too large")
738 }
739 z.bits |= b << z.nBits
740 z.nBits += n
741 for z.nBits >= 8 {
742 out = append(out, uint8(z.bits))
743 z.bits >>= 8
744 z.nBits -= 8
745 }
746}
747
748func (z *deflateBitStream) writeBytes(b []byte) {
749 z.flush()
750 out = append(out, b...)
751}
752
Nigel Tao12347602019-06-16 23:15:05 +1000753func (z *deflateBitStream) writeFixedHuffmanLCode(lCode uint32) {
Nigel Tao981bf192018-05-23 12:17:55 +1000754 switch {
755 case lCode < 144: // 0b._0011_0000 through 0b._1011_1111
756 lCode += 0x030
757 z.writeBits(reverse(lCode, 8), 8)
758 case lCode < 256: // 0b1_1001_0000 through 0b1_1111_1111
759 lCode += 0x190 - 144
760 z.writeBits(reverse(lCode, 9), 9)
761 case lCode < 280: // 0b._.000_0000 through 0b._.001_0111
762 lCode -= 256 - 0x000
763 z.writeBits(reverse(lCode, 7), 7)
764 default: // 0b._1100_0000 through 0b._1100_0111
765 lCode -= 280 - 0x0C0
766 z.writeBits(reverse(lCode, 8), 8)
767 }
768}
769
770func (z *deflateBitStream) flush() {
771 if z.nBits > 0 {
772 out = append(out, uint8(z.bits))
773 z.bits = 0
774 z.nBits = 0
775 }
776}
777
778func deflateEncodeLength(l uint32) (code uint32, extra uint32, nExtra uint32) {
779 switch {
780 case l < 3:
781 // No-op.
782 case l < 11:
783 l -= 3
784 return (l >> 0) + 257, l & 0x0000, 0
785 case l < 19:
786 l -= 11
787 return (l >> 1) + 265, l & 0x0001, 1
788 case l < 35:
789 l -= 19
790 return (l >> 2) + 269, l & 0x0003, 2
791 case l < 67:
792 l -= 35
793 return (l >> 3) + 273, l & 0x0007, 3
794 case l < 131:
795 l -= 67
796 return (l >> 4) + 277, l & 0x000F, 4
797 case l < 258:
798 l -= 131
799 return (l >> 5) + 281, l & 0x001F, 5
800 case l == 258:
801 return 285, 0, 0
802 }
803 panic(fmt.Sprintf("deflateEncodeLength: l=%d", l))
804}
805
806func deflateEncodeDistance(d uint32) (code uint32, extra uint32, nExtra uint32) {
807 switch {
808 case d < 1:
809 // No-op.
810 case d < 5:
811 d -= 1
812 return (d >> 0) + 0, d & 0x0000, 0
813 case d < 9:
814 d -= 5
815 return (d >> 1) + 4, d & 0x0001, 1
816 case d < 17:
817 d -= 9
818 return (d >> 2) + 6, d & 0x0003, 2
819 case d < 33:
820 d -= 17
821 return (d >> 3) + 8, d & 0x0007, 3
822 case d < 65:
823 d -= 33
824 return (d >> 4) + 10, d & 0x000F, 4
825 case d < 129:
826 d -= 65
827 return (d >> 5) + 12, d & 0x001F, 5
828 case d < 257:
829 d -= 129
830 return (d >> 6) + 14, d & 0x003F, 6
831 case d < 513:
832 d -= 257
833 return (d >> 7) + 16, d & 0x007F, 7
834 case d < 1025:
835 d -= 513
836 return (d >> 8) + 18, d & 0x00FF, 8
837 case d < 2049:
838 d -= 1025
839 return (d >> 9) + 20, d & 0x01FF, 9
840 case d < 4097:
841 d -= 2049
842 return (d >> 10) + 22, d & 0x03FF, 10
843 case d < 8193:
844 d -= 4097
845 return (d >> 11) + 24, d & 0x07FF, 11
846 case d < 16385:
847 d -= 8193
848 return (d >> 12) + 26, d & 0x0FFF, 12
849 case d < 32769:
850 d -= 16385
851 return (d >> 13) + 28, d & 0x1FFF, 13
852 }
853 panic(fmt.Sprintf("deflateEncodeDistance: d=%d", d))
854}
855
856func deflateParseLiteral(s string) (lit string, ok bool) {
Nigel Tao12347602019-06-16 23:15:05 +1000857 // TODO: support "\xAB" escape codes in the script?
Nigel Tao981bf192018-05-23 12:17:55 +1000858 const (
859 prefix = `literal "`
860 suffix = `"`
861 )
862 if strings.HasPrefix(s, prefix) {
863 s = s[len(prefix):]
864 if strings.HasSuffix(s, suffix) {
865 return s[:len(s)-len(suffix)], true
866 }
867 }
868 return "", false
869}
870
871func deflateParseLenDist(line string) (l uint32, d uint32, ok bool) {
872 const (
873 lStr = "len "
874 dStr = "dist "
875 )
876 s := line
877 if strings.HasPrefix(s, lStr) {
878 s = s[len(lStr):]
879 if l, s, ok := parseNum(s); ok && 3 <= l && l <= 258 {
880 if strings.HasPrefix(s, dStr) {
881 s = s[len(dStr):]
882 if d, s, ok := parseNum(s); ok && 1 <= d && d <= 32768 && s == "" {
883 return l, d, true
884 }
885 }
886 }
887 }
888 return 0, 0, false
889}
890
891// ----
892
893func init() {
894 formats["gif"] = stateGif
895}
896
897var gifGlobals struct {
Nigel Tao31a2bc92019-05-18 17:32:24 +1000898 imageWidth uint32
899 imageHeight uint32
900 imageBackgroundColorIndex uint32
Nigel Tao981bf192018-05-23 12:17:55 +1000901
Nigel Taoefb18032019-10-05 11:40:24 +1000902 frameInterlaced bool
903 frameLeft uint32
904 frameTop uint32
905 frameWidth uint32
906 frameHeight uint32
Nigel Tao981bf192018-05-23 12:17:55 +1000907
908 globalPalette [][4]uint8
Nigel Tao746681e2019-05-04 13:49:22 +1000909 localPalette [][4]uint8
Nigel Tao981bf192018-05-23 12:17:55 +1000910}
911
912func stateGif(line string) (stateFunc, error) {
913 const (
Nigel Tao6700c722019-05-26 09:41:55 +1000914 cmdB = "bytes "
915 cmdGC = "graphicControl "
916 cmdL = "lzw "
917 cmdLC = "loopCount "
Nigel Tao981bf192018-05-23 12:17:55 +1000918 )
919outer:
920 switch {
Nigel Tao12347602019-06-16 23:15:05 +1000921 case line == "":
922 return stateGif, nil
923
Nigel Tao981bf192018-05-23 12:17:55 +1000924 case line == "frame {":
Nigel Taoefb18032019-10-05 11:40:24 +1000925 gifGlobals.frameInterlaced = false
926 gifGlobals.frameLeft = 0
927 gifGlobals.frameTop = 0
928 gifGlobals.frameWidth = 0
929 gifGlobals.frameHeight = 0
Nigel Tao746681e2019-05-04 13:49:22 +1000930 gifGlobals.localPalette = nil
Nigel Tao981bf192018-05-23 12:17:55 +1000931 out = append(out, 0x2C)
932 return stateGifFrame, nil
933
934 case line == "header":
935 out = append(out, "GIF89a"...)
936 return stateGif, nil
937
938 case line == "image {":
939 return stateGifImage, nil
940
941 case line == "trailer":
942 out = append(out, 0x3B)
943 return stateGif, nil
944
Nigel Tao21f28902019-04-20 16:33:11 +1000945 case strings.HasPrefix(line, cmdB):
946 s := line[len(cmdB):]
947 for s != "" {
948 x, ok := uint32(0), false
949 x, s, ok = parseHex(s)
950 if !ok {
951 break outer
952 }
953 out = append(out, uint8(x))
954 }
955 return stateGif, nil
956
Nigel Tao6700c722019-05-26 09:41:55 +1000957 case strings.HasPrefix(line, cmdGC):
958 s := line[len(cmdGC):]
959
960 flags := uint8(0)
961 if i := strings.IndexByte(s, ' '); i < 0 {
962 break
963 } else {
964 switch s[:i] {
965 case "animationDisposalNone":
966 flags |= 0x00
967 case "animationDisposalRestoreBackground":
968 flags |= 0x08
969 case "animationDisposalRestorePrevious":
970 flags |= 0x0C
971 default:
972 break outer
973 }
974 s = s[i+1:]
975 }
976
Nigel Taob1ff27f2019-05-12 22:00:44 +1000977 if !strings.HasSuffix(s, "ms") {
978 break
979 }
980 s = s[:len(s)-2]
981 duration, s, ok := parseNum(s)
982 if !ok || s != "" {
983 break
984 }
985 duration /= 10 // GIF's unit of time is 10ms.
Nigel Tao6700c722019-05-26 09:41:55 +1000986
987 transparentIndex := uint8(0)
988
Nigel Taob1ff27f2019-05-12 22:00:44 +1000989 out = append(out,
Nigel Tao6700c722019-05-26 09:41:55 +1000990 0x21, 0xF9, 0x04,
991 flags,
Nigel Taob1ff27f2019-05-12 22:00:44 +1000992 uint8(duration>>0),
993 uint8(duration>>8),
Nigel Tao6700c722019-05-26 09:41:55 +1000994 transparentIndex,
995 0x00,
Nigel Taob1ff27f2019-05-12 22:00:44 +1000996 )
997 return stateGif, nil
998
Nigel Tao981bf192018-05-23 12:17:55 +1000999 case strings.HasPrefix(line, cmdL):
1000 s := line[len(cmdL):]
1001 litWidth, s, ok := parseNum(s)
1002 if !ok || litWidth < 2 || 8 < litWidth {
1003 break
1004 }
1005 out = append(out, uint8(litWidth))
1006
1007 uncompressed := []byte(nil)
1008 for s != "" {
1009 x := uint32(0)
1010 x, s, ok = parseHex(s)
1011 if !ok {
1012 break outer
1013 }
1014 uncompressed = append(uncompressed, uint8(x))
1015 }
1016
1017 buf := bytes.NewBuffer(nil)
1018 w := lzw.NewWriter(buf, lzw.LSB, int(litWidth))
1019 if _, err := w.Write(uncompressed); err != nil {
1020 return nil, err
1021 }
1022 if err := w.Close(); err != nil {
1023 return nil, err
1024 }
1025 compressed := buf.Bytes()
1026
1027 for len(compressed) > 0 {
1028 if len(compressed) <= 0xFF {
1029 out = append(out, uint8(len(compressed)))
1030 out = append(out, compressed...)
1031 compressed = nil
1032 } else {
1033 out = append(out, 0xFF)
1034 out = append(out, compressed[:0xFF]...)
1035 compressed = compressed[0xFF:]
1036 }
1037 }
1038 out = append(out, 0x00)
1039 return stateGif, nil
Nigel Tao7be9c392018-10-13 17:00:45 +11001040
1041 case strings.HasPrefix(line, cmdLC):
1042 s := line[len(cmdLC):]
1043 loopCount, _, ok := parseNum(s)
1044 if !ok || 0xFFFF < loopCount {
1045 break
1046 }
1047 out = append(out,
1048 0x21, // Extension Introducer.
1049 0xFF, // Application Extension Label.
1050 0x0B, // Block Size.
1051 )
1052 out = append(out, "NETSCAPE2.0"...)
1053 out = append(out,
1054 0x03, // Block Size.
1055 0x01, // Magic Number.
1056 byte(loopCount),
1057 byte(loopCount>>8),
1058 0x00, // Block Terminator.
1059 )
1060 return stateGif, nil
Nigel Tao981bf192018-05-23 12:17:55 +10001061 }
1062
1063 return nil, fmt.Errorf("bad stateGif command: %q", line)
1064}
1065
1066func stateGifImage(line string) (stateFunc, error) {
1067 g := &gifGlobals
1068 if line == "}" {
1069 out = appendU16LE(out, uint16(g.imageWidth))
1070 out = appendU16LE(out, uint16(g.imageHeight))
1071 if g.globalPalette == nil {
1072 out = append(out, 0x00)
1073 } else if n := log2(uint32(len(g.globalPalette))); n < 2 || 8 < n {
1074 return nil, fmt.Errorf("bad len(g.globalPalette): %d", len(g.globalPalette))
1075 } else {
1076 out = append(out, 0x80|uint8(n-1))
1077 }
Nigel Tao31a2bc92019-05-18 17:32:24 +10001078 out = append(out, uint8(g.imageBackgroundColorIndex))
Nigel Tao981bf192018-05-23 12:17:55 +10001079 out = append(out, 0x00)
1080 for _, x := range g.globalPalette {
1081 out = append(out, x[0], x[1], x[2])
1082 }
1083 return stateGif, nil
1084 }
1085
1086 const (
Nigel Tao31a2bc92019-05-18 17:32:24 +10001087 cmdBCI = "backgroundColorIndex "
Nigel Tao981bf192018-05-23 12:17:55 +10001088 cmdIWH = "imageWidthHeight "
1089 cmdP = "palette {"
1090 )
1091 switch {
Nigel Tao31a2bc92019-05-18 17:32:24 +10001092 case strings.HasPrefix(line, cmdBCI):
1093 s := line[len(cmdBCI):]
1094 if i, _, ok := parseNum(s); ok {
1095 g.imageBackgroundColorIndex = i
1096 }
1097 return stateGifImage, nil
1098
Nigel Tao981bf192018-05-23 12:17:55 +10001099 case strings.HasPrefix(line, cmdIWH):
1100 s := line[len(cmdIWH):]
1101 if w, s, ok := parseNum(s); ok {
1102 if h, _, ok := parseNum(s); ok {
1103 g.imageWidth = w
1104 g.imageHeight = h
1105 return stateGifImage, nil
1106 }
1107 }
1108
1109 case strings.HasPrefix(line, cmdP):
1110 return stateGifImagePalette, nil
1111 }
1112
1113 return nil, fmt.Errorf("bad stateGifImage command: %q", line)
1114}
1115
1116func stateGifFrame(line string) (stateFunc, error) {
1117 g := &gifGlobals
1118 if line == "}" {
1119 out = appendU16LE(out, uint16(g.frameLeft))
1120 out = appendU16LE(out, uint16(g.frameTop))
1121 out = appendU16LE(out, uint16(g.frameWidth))
1122 out = appendU16LE(out, uint16(g.frameHeight))
Nigel Taoefb18032019-10-05 11:40:24 +10001123 flags := byte(0x00)
1124 if g.frameInterlaced {
1125 flags |= 0x40
1126 }
Nigel Tao746681e2019-05-04 13:49:22 +10001127 if g.localPalette == nil {
Nigel Taoefb18032019-10-05 11:40:24 +10001128 out = append(out, flags)
Nigel Tao746681e2019-05-04 13:49:22 +10001129 } else if n := log2(uint32(len(g.localPalette))); n < 2 || 8 < n {
1130 return nil, fmt.Errorf("bad len(g.localPalette): %d", len(g.localPalette))
1131 } else {
Nigel Taoefb18032019-10-05 11:40:24 +10001132 out = append(out, flags|0x80|uint8(n-1))
Nigel Tao746681e2019-05-04 13:49:22 +10001133 }
1134 for _, x := range g.localPalette {
1135 out = append(out, x[0], x[1], x[2])
1136 }
Nigel Tao981bf192018-05-23 12:17:55 +10001137 return stateGif, nil
1138 }
1139
1140 const (
1141 cmdFLTWH = "frameLeftTopWidthHeight "
Nigel Tao746681e2019-05-04 13:49:22 +10001142 cmdP = "palette {"
Nigel Tao981bf192018-05-23 12:17:55 +10001143 )
1144 switch {
Nigel Taoefb18032019-10-05 11:40:24 +10001145 case line == "interlaced":
1146 g.frameInterlaced = true
1147 return stateGifFrame, nil
1148
Nigel Tao981bf192018-05-23 12:17:55 +10001149 case strings.HasPrefix(line, cmdFLTWH):
1150 s := line[len(cmdFLTWH):]
1151 if l, s, ok := parseNum(s); ok {
1152 if t, s, ok := parseNum(s); ok {
1153 if w, s, ok := parseNum(s); ok {
1154 if h, _, ok := parseNum(s); ok {
1155 g.frameLeft = l
1156 g.frameTop = t
1157 g.frameWidth = w
1158 g.frameHeight = h
1159 return stateGifFrame, nil
1160 }
1161 }
1162 }
1163 }
Nigel Tao746681e2019-05-04 13:49:22 +10001164
1165 case strings.HasPrefix(line, cmdP):
1166 return stateGifFramePalette, nil
Nigel Tao981bf192018-05-23 12:17:55 +10001167 }
1168
1169 return nil, fmt.Errorf("bad stateGifFrame command: %q", line)
1170}
1171
1172func stateGifImagePalette(line string) (stateFunc, error) {
1173 g := &gifGlobals
1174 if line == "}" {
1175 return stateGifImage, nil
1176 }
1177
1178 s := line
1179 if rgb0, s, ok := parseHex(s); ok {
1180 if rgb1, s, ok := parseHex(s); ok {
1181 if rgb2, _, ok := parseHex(s); ok {
1182 g.globalPalette = append(g.globalPalette,
1183 [4]uint8{uint8(rgb0), uint8(rgb1), uint8(rgb2), 0xFF})
1184 return stateGifImagePalette, nil
1185 }
1186 }
1187 }
1188
1189 return nil, fmt.Errorf("bad stateGifImagePalette command: %q", line)
1190}
Nigel Tao746681e2019-05-04 13:49:22 +10001191
1192func stateGifFramePalette(line string) (stateFunc, error) {
1193 g := &gifGlobals
1194 if line == "}" {
1195 return stateGifFrame, nil
1196 }
1197
1198 s := line
1199 if rgb0, s, ok := parseHex(s); ok {
1200 if rgb1, s, ok := parseHex(s); ok {
1201 if rgb2, _, ok := parseHex(s); ok {
1202 g.localPalette = append(g.localPalette,
1203 [4]uint8{uint8(rgb0), uint8(rgb1), uint8(rgb2), 0xFF})
1204 return stateGifFramePalette, nil
1205 }
1206 }
1207 }
1208
1209 return nil, fmt.Errorf("bad stateGifFramePalette command: %q", line)
1210}