Source file src/compress/lzw/reader_test.go

     1  // Copyright 2011 The Go Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package lzw
     6  
     7  import (
     8  	"bytes"
     9  	"fmt"
    10  	"io"
    11  	"math"
    12  	"os"
    13  	"runtime"
    14  	"strconv"
    15  	"strings"
    16  	"testing"
    17  )
    18  
    19  type lzwTest struct {
    20  	desc       string
    21  	raw        string
    22  	compressed string
    23  	err        error
    24  }
    25  
    26  var lzwTests = []lzwTest{
    27  	{
    28  		"empty;LSB;8",
    29  		"",
    30  		"\x01\x01",
    31  		nil,
    32  	},
    33  	{
    34  		"empty;MSB;8",
    35  		"",
    36  		"\x80\x80",
    37  		nil,
    38  	},
    39  	{
    40  		"tobe;LSB;7",
    41  		"TOBEORNOTTOBEORTOBEORNOT",
    42  		"\x54\x4f\x42\x45\x4f\x52\x4e\x4f\x54\x82\x84\x86\x8b\x85\x87\x89\x81",
    43  		nil,
    44  	},
    45  	{
    46  		"tobe;LSB;8",
    47  		"TOBEORNOTTOBEORTOBEORNOT",
    48  		"\x54\x9e\x08\x29\xf2\x44\x8a\x93\x27\x54\x04\x12\x34\xb8\xb0\xe0\xc1\x84\x01\x01",
    49  		nil,
    50  	},
    51  	{
    52  		"tobe;MSB;7",
    53  		"TOBEORNOTTOBEORTOBEORNOT",
    54  		"\x54\x4f\x42\x45\x4f\x52\x4e\x4f\x54\x82\x84\x86\x8b\x85\x87\x89\x81",
    55  		nil,
    56  	},
    57  	{
    58  		"tobe;MSB;8",
    59  		"TOBEORNOTTOBEORTOBEORNOT",
    60  		"\x2a\x13\xc8\x44\x52\x79\x48\x9c\x4f\x2a\x40\xa0\x90\x68\x5c\x16\x0f\x09\x80\x80",
    61  		nil,
    62  	},
    63  	{
    64  		"tobe-truncated;LSB;8",
    65  		"TOBEORNOTTOBEORTOBEORNOT",
    66  		"\x54\x9e\x08\x29\xf2\x44\x8a\x93\x27\x54\x04",
    67  		io.ErrUnexpectedEOF,
    68  	},
    69  	// This example comes from https://en.wikipedia.org/wiki/Graphics_Interchange_Format.
    70  	{
    71  		"gif;LSB;8",
    72  		"\x28\xff\xff\xff\x28\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff",
    73  		"\x00\x51\xfc\x1b\x28\x70\xa0\xc1\x83\x01\x01",
    74  		nil,
    75  	},
    76  	// This example comes from http://compgroups.net/comp.lang.ruby/Decompressing-LZW-compression-from-PDF-file
    77  	{
    78  		"pdf;MSB;8",
    79  		"-----A---B",
    80  		"\x80\x0b\x60\x50\x22\x0c\x0c\x85\x01",
    81  		nil,
    82  	},
    83  }
    84  
    85  func TestReader(t *testing.T) {
    86  	var b bytes.Buffer
    87  	for _, tt := range lzwTests {
    88  		d := strings.Split(tt.desc, ";")
    89  		var order Order
    90  		switch d[1] {
    91  		case "LSB":
    92  			order = LSB
    93  		case "MSB":
    94  			order = MSB
    95  		default:
    96  			t.Errorf("%s: bad order %q", tt.desc, d[1])
    97  		}
    98  		litWidth, _ := strconv.Atoi(d[2])
    99  		rc := NewReader(strings.NewReader(tt.compressed), order, litWidth)
   100  		defer rc.Close()
   101  		b.Reset()
   102  		n, err := io.Copy(&b, rc)
   103  		s := b.String()
   104  		if err != nil {
   105  			if err != tt.err {
   106  				t.Errorf("%s: io.Copy: %v want %v", tt.desc, err, tt.err)
   107  			}
   108  			if err == io.ErrUnexpectedEOF {
   109  				// Even if the input is truncated, we should still return the
   110  				// partial decoded result.
   111  				if n == 0 || !strings.HasPrefix(tt.raw, s) {
   112  					t.Errorf("got %d bytes (%q), want a non-empty prefix of %q", n, s, tt.raw)
   113  				}
   114  			}
   115  			continue
   116  		}
   117  		if s != tt.raw {
   118  			t.Errorf("%s: got %d-byte %q want %d-byte %q", tt.desc, n, s, len(tt.raw), tt.raw)
   119  		}
   120  	}
   121  }
   122  
   123  func TestReaderReset(t *testing.T) {
   124  	var b bytes.Buffer
   125  	for _, tt := range lzwTests {
   126  		d := strings.Split(tt.desc, ";")
   127  		var order Order
   128  		switch d[1] {
   129  		case "LSB":
   130  			order = LSB
   131  		case "MSB":
   132  			order = MSB
   133  		default:
   134  			t.Errorf("%s: bad order %q", tt.desc, d[1])
   135  		}
   136  		litWidth, _ := strconv.Atoi(d[2])
   137  		rc := NewReader(strings.NewReader(tt.compressed), order, litWidth)
   138  		defer rc.Close()
   139  		b.Reset()
   140  		n, err := io.Copy(&b, rc)
   141  		b1 := b.Bytes()
   142  		if err != nil {
   143  			if err != tt.err {
   144  				t.Errorf("%s: io.Copy: %v want %v", tt.desc, err, tt.err)
   145  			}
   146  			if err == io.ErrUnexpectedEOF {
   147  				// Even if the input is truncated, we should still return the
   148  				// partial decoded result.
   149  				if n == 0 || !strings.HasPrefix(tt.raw, b.String()) {
   150  					t.Errorf("got %d bytes (%q), want a non-empty prefix of %q", n, b.String(), tt.raw)
   151  				}
   152  			}
   153  			continue
   154  		}
   155  
   156  		b.Reset()
   157  		rc.(*Reader).Reset(strings.NewReader(tt.compressed), order, litWidth)
   158  		n, err = io.Copy(&b, rc)
   159  		b2 := b.Bytes()
   160  		if err != nil {
   161  			t.Errorf("%s: io.Copy: %v want %v", tt.desc, err, nil)
   162  			continue
   163  		}
   164  		if !bytes.Equal(b1, b2) {
   165  			t.Errorf("bytes read were not the same")
   166  		}
   167  	}
   168  }
   169  
   170  type devZero struct{}
   171  
   172  func (devZero) Read(p []byte) (int, error) {
   173  	for i := range p {
   174  		p[i] = 0
   175  	}
   176  	return len(p), nil
   177  }
   178  
   179  func TestHiCodeDoesNotOverflow(t *testing.T) {
   180  	r := NewReader(devZero{}, LSB, 8)
   181  	d := r.(*Reader)
   182  	buf := make([]byte, 1024)
   183  	oldHi := uint16(0)
   184  	for i := 0; i < 100; i++ {
   185  		if _, err := io.ReadFull(r, buf); err != nil {
   186  			t.Fatalf("i=%d: %v", i, err)
   187  		}
   188  		// The hi code should never decrease.
   189  		if d.hi < oldHi {
   190  			t.Fatalf("i=%d: hi=%d decreased from previous value %d", i, d.hi, oldHi)
   191  		}
   192  		oldHi = d.hi
   193  	}
   194  }
   195  
   196  // TestNoLongerSavingPriorExpansions tests the decoder state when codes other
   197  // than clear codes continue to be seen after decoder.hi and decoder.width
   198  // reach their maximum values (4095 and 12), i.e. after we no longer save prior
   199  // expansions. In particular, it tests seeing the highest possible code, 4095.
   200  func TestNoLongerSavingPriorExpansions(t *testing.T) {
   201  	// Iterations is used to calculate how many input bits are needed to get
   202  	// the decoder.hi and decoder.width values up to their maximum.
   203  	iterations := []struct {
   204  		width, n int
   205  	}{
   206  		// The final term is 257, not 256, as NewReader initializes d.hi to
   207  		// d.clear+1 and the clear code is 256.
   208  		{9, 512 - 257},
   209  		{10, 1024 - 512},
   210  		{11, 2048 - 1024},
   211  		{12, 4096 - 2048},
   212  	}
   213  	nCodes, nBits := 0, 0
   214  	for _, e := range iterations {
   215  		nCodes += e.n
   216  		nBits += e.n * e.width
   217  	}
   218  	if nCodes != 3839 {
   219  		t.Fatalf("nCodes: got %v, want %v", nCodes, 3839)
   220  	}
   221  	if nBits != 43255 {
   222  		t.Fatalf("nBits: got %v, want %v", nBits, 43255)
   223  	}
   224  
   225  	// Construct our input of 43255 zero bits (which gets d.hi and d.width up
   226  	// to 4095 and 12), followed by 0xfff (4095) as 12 bits, followed by 0x101
   227  	// (EOF) as 12 bits.
   228  	//
   229  	// 43255 = 5406*8 + 7, and codes are read in LSB order. The final bytes are
   230  	// therefore:
   231  	//
   232  	// xwwwwwww xxxxxxxx yyyyyxxx zyyyyyyy
   233  	// 10000000 11111111 00001111 00001000
   234  	//
   235  	// or split out:
   236  	//
   237  	// .0000000 ........ ........ ........   w = 0x000
   238  	// 1....... 11111111 .....111 ........   x = 0xfff
   239  	// ........ ........ 00001... .0001000   y = 0x101
   240  	//
   241  	// The 12 'w' bits (not all are shown) form the 3839'th code, with value
   242  	// 0x000. Just after decoder.read returns that code, d.hi == 4095 and
   243  	// d.last == 0.
   244  	//
   245  	// The 12 'x' bits form the 3840'th code, with value 0xfff or 4095. Just
   246  	// after decoder.read returns that code, d.hi == 4095 and d.last ==
   247  	// decoderInvalidCode.
   248  	//
   249  	// The 12 'y' bits form the 3841'st code, with value 0x101, the EOF code.
   250  	//
   251  	// The 'z' bit is unused.
   252  	in := make([]byte, 5406)
   253  	in = append(in, 0x80, 0xff, 0x0f, 0x08)
   254  
   255  	r := NewReader(bytes.NewReader(in), LSB, 8)
   256  	nDecoded, err := io.Copy(io.Discard, r)
   257  	if err != nil {
   258  		t.Fatalf("Copy: %v", err)
   259  	}
   260  	// nDecoded should be 3841: 3839 literal codes and then 2 decoded bytes
   261  	// from 1 non-literal code. The EOF code contributes 0 decoded bytes.
   262  	if nDecoded != int64(nCodes+2) {
   263  		t.Fatalf("nDecoded: got %v, want %v", nDecoded, nCodes+2)
   264  	}
   265  }
   266  
   267  func BenchmarkDecoder(b *testing.B) {
   268  	buf, err := os.ReadFile("../testdata/e.txt")
   269  	if err != nil {
   270  		b.Fatal(err)
   271  	}
   272  	if len(buf) == 0 {
   273  		b.Fatalf("test file has no data")
   274  	}
   275  
   276  	getInputBuf := func(buf []byte, n int) []byte {
   277  		compressed := new(bytes.Buffer)
   278  		w := NewWriter(compressed, LSB, 8)
   279  		for i := 0; i < n; i += len(buf) {
   280  			if len(buf) > n-i {
   281  				buf = buf[:n-i]
   282  			}
   283  			w.Write(buf)
   284  		}
   285  		w.Close()
   286  		return compressed.Bytes()
   287  	}
   288  
   289  	for e := 4; e <= 6; e++ {
   290  		n := int(math.Pow10(e))
   291  		b.Run(fmt.Sprint("1e", e), func(b *testing.B) {
   292  			b.StopTimer()
   293  			b.SetBytes(int64(n))
   294  			buf1 := getInputBuf(buf, n)
   295  			runtime.GC()
   296  			b.StartTimer()
   297  			for i := 0; i < b.N; i++ {
   298  				io.Copy(io.Discard, NewReader(bytes.NewReader(buf1), LSB, 8))
   299  			}
   300  		})
   301  		b.Run(fmt.Sprint("1e-Reuse", e), func(b *testing.B) {
   302  			b.StopTimer()
   303  			b.SetBytes(int64(n))
   304  			buf1 := getInputBuf(buf, n)
   305  			runtime.GC()
   306  			b.StartTimer()
   307  			r := NewReader(bytes.NewReader(buf1), LSB, 8)
   308  			for i := 0; i < b.N; i++ {
   309  				io.Copy(io.Discard, r)
   310  				r.Close()
   311  				r.(*Reader).Reset(bytes.NewReader(buf1), LSB, 8)
   312  			}
   313  		})
   314  	}
   315  }
   316  

View as plain text