github.com/google/netstack@v0.0.0-20191123085552-55fcc16cd0eb/tcpip/link/sharedmem/pipe/pipe_test.go (about)

     1  // Copyright 2018 The gVisor 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  //     http://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  package pipe
    16  
    17  import (
    18  	"math/rand"
    19  	"reflect"
    20  	"runtime"
    21  	"sync"
    22  	"testing"
    23  )
    24  
    25  func TestSimpleReadWrite(t *testing.T) {
    26  	// Check that a simple write can be properly read from the rx side.
    27  	tr := rand.New(rand.NewSource(99))
    28  	rr := rand.New(rand.NewSource(99))
    29  
    30  	b := make([]byte, 100)
    31  	var tx Tx
    32  	tx.Init(b)
    33  
    34  	wb := tx.Push(10)
    35  	if wb == nil {
    36  		t.Fatalf("Push failed on empty pipe")
    37  	}
    38  	for i := range wb {
    39  		wb[i] = byte(tr.Intn(256))
    40  	}
    41  	tx.Flush()
    42  
    43  	var rx Rx
    44  	rx.Init(b)
    45  	rb := rx.Pull()
    46  	if len(rb) != 10 {
    47  		t.Fatalf("Bad buffer size returned: got %v, want %v", len(rb), 10)
    48  	}
    49  
    50  	for i := range rb {
    51  		if v := byte(rr.Intn(256)); v != rb[i] {
    52  			t.Fatalf("Bad read buffer at index %v: got %v, want %v", i, rb[i], v)
    53  		}
    54  	}
    55  	rx.Flush()
    56  }
    57  
    58  func TestEmptyRead(t *testing.T) {
    59  	// Check that pulling from an empty pipe fails.
    60  	b := make([]byte, 100)
    61  	var tx Tx
    62  	tx.Init(b)
    63  
    64  	var rx Rx
    65  	rx.Init(b)
    66  	if rb := rx.Pull(); rb != nil {
    67  		t.Fatalf("Pull succeeded on empty pipe")
    68  	}
    69  }
    70  
    71  func TestTooLargeWrite(t *testing.T) {
    72  	// Check that writes that are too large are properly rejected.
    73  	b := make([]byte, 96)
    74  	var tx Tx
    75  	tx.Init(b)
    76  
    77  	if wb := tx.Push(96); wb != nil {
    78  		t.Fatalf("Write of 96 bytes succeeded on 96-byte pipe")
    79  	}
    80  
    81  	if wb := tx.Push(88); wb != nil {
    82  		t.Fatalf("Write of 88 bytes succeeded on 96-byte pipe")
    83  	}
    84  
    85  	if wb := tx.Push(80); wb == nil {
    86  		t.Fatalf("Write of 80 bytes failed on 96-byte pipe")
    87  	}
    88  }
    89  
    90  func TestFullWrite(t *testing.T) {
    91  	// Check that writes fail when the pipe is full.
    92  	b := make([]byte, 100)
    93  	var tx Tx
    94  	tx.Init(b)
    95  
    96  	if wb := tx.Push(80); wb == nil {
    97  		t.Fatalf("Write of 80 bytes failed on 96-byte pipe")
    98  	}
    99  
   100  	if wb := tx.Push(1); wb != nil {
   101  		t.Fatalf("Write succeeded on full pipe")
   102  	}
   103  }
   104  
   105  func TestFullAndFlushedWrite(t *testing.T) {
   106  	// Check that writes fail when the pipe is full and has already been
   107  	// flushed.
   108  	b := make([]byte, 100)
   109  	var tx Tx
   110  	tx.Init(b)
   111  
   112  	if wb := tx.Push(80); wb == nil {
   113  		t.Fatalf("Write of 80 bytes failed on 96-byte pipe")
   114  	}
   115  
   116  	tx.Flush()
   117  
   118  	if wb := tx.Push(1); wb != nil {
   119  		t.Fatalf("Write succeeded on full pipe")
   120  	}
   121  }
   122  
   123  func TestTxFlushTwice(t *testing.T) {
   124  	// Checks that a second consecutive tx flush is a no-op.
   125  	b := make([]byte, 100)
   126  	var tx Tx
   127  	tx.Init(b)
   128  
   129  	if wb := tx.Push(50); wb == nil {
   130  		t.Fatalf("Push failed on empty pipe")
   131  	}
   132  	tx.Flush()
   133  
   134  	// Make copy of original tx queue, flush it, then check that it didn't
   135  	// change.
   136  	orig := tx
   137  	tx.Flush()
   138  
   139  	if !reflect.DeepEqual(orig, tx) {
   140  		t.Fatalf("Flush mutated tx pipe: got %v, want %v", tx, orig)
   141  	}
   142  }
   143  
   144  func TestRxFlushTwice(t *testing.T) {
   145  	// Checks that a second consecutive rx flush is a no-op.
   146  	b := make([]byte, 100)
   147  	var tx Tx
   148  	tx.Init(b)
   149  
   150  	if wb := tx.Push(50); wb == nil {
   151  		t.Fatalf("Push failed on empty pipe")
   152  	}
   153  	tx.Flush()
   154  
   155  	var rx Rx
   156  	rx.Init(b)
   157  	if rb := rx.Pull(); rb == nil {
   158  		t.Fatalf("Pull failed on non-empty pipe")
   159  	}
   160  	rx.Flush()
   161  
   162  	// Make copy of original rx queue, flush it, then check that it didn't
   163  	// change.
   164  	orig := rx
   165  	rx.Flush()
   166  
   167  	if !reflect.DeepEqual(orig, rx) {
   168  		t.Fatalf("Flush mutated rx pipe: got %v, want %v", rx, orig)
   169  	}
   170  }
   171  
   172  func TestWrapInMiddleOfTransaction(t *testing.T) {
   173  	// Check that writes are not flushed when we need to wrap the buffer
   174  	// around.
   175  	b := make([]byte, 100)
   176  	var tx Tx
   177  	tx.Init(b)
   178  
   179  	if wb := tx.Push(50); wb == nil {
   180  		t.Fatalf("Push failed on empty pipe")
   181  	}
   182  	tx.Flush()
   183  
   184  	var rx Rx
   185  	rx.Init(b)
   186  	if rb := rx.Pull(); rb == nil {
   187  		t.Fatalf("Pull failed on non-empty pipe")
   188  	}
   189  	rx.Flush()
   190  
   191  	// At this point the ring buffer is empty, but the write is at offset
   192  	// 64 (50 + sizeOfSlotHeader + padding-for-8-byte-alignment).
   193  	if wb := tx.Push(10); wb == nil {
   194  		t.Fatalf("Push failed on empty pipe")
   195  	}
   196  
   197  	if wb := tx.Push(50); wb == nil {
   198  		t.Fatalf("Push failed on non-full pipe")
   199  	}
   200  
   201  	// We haven't flushed yet, so pull must return nil.
   202  	if rb := rx.Pull(); rb != nil {
   203  		t.Fatalf("Pull succeeded on non-flushed pipe")
   204  	}
   205  
   206  	tx.Flush()
   207  
   208  	// The two buffers must be available now.
   209  	if rb := rx.Pull(); rb == nil {
   210  		t.Fatalf("Pull failed on non-empty pipe")
   211  	}
   212  
   213  	if rb := rx.Pull(); rb == nil {
   214  		t.Fatalf("Pull failed on non-empty pipe")
   215  	}
   216  }
   217  
   218  func TestWriteAbort(t *testing.T) {
   219  	// Check that a read fails on a pipe that has had data pushed to it but
   220  	// has aborted the push.
   221  	b := make([]byte, 100)
   222  	var tx Tx
   223  	tx.Init(b)
   224  
   225  	if wb := tx.Push(10); wb == nil {
   226  		t.Fatalf("Write failed on empty pipe")
   227  	}
   228  
   229  	var rx Rx
   230  	rx.Init(b)
   231  	if rb := rx.Pull(); rb != nil {
   232  		t.Fatalf("Pull succeeded on empty pipe")
   233  	}
   234  
   235  	tx.Abort()
   236  	if rb := rx.Pull(); rb != nil {
   237  		t.Fatalf("Pull succeeded on empty pipe")
   238  	}
   239  }
   240  
   241  func TestWrappedWriteAbort(t *testing.T) {
   242  	// Check that writes are properly aborted even if the writes wrap
   243  	// around.
   244  	b := make([]byte, 100)
   245  	var tx Tx
   246  	tx.Init(b)
   247  
   248  	if wb := tx.Push(50); wb == nil {
   249  		t.Fatalf("Push failed on empty pipe")
   250  	}
   251  	tx.Flush()
   252  
   253  	var rx Rx
   254  	rx.Init(b)
   255  	if rb := rx.Pull(); rb == nil {
   256  		t.Fatalf("Pull failed on non-empty pipe")
   257  	}
   258  	rx.Flush()
   259  
   260  	// At this point the ring buffer is empty, but the write is at offset
   261  	// 64 (50 + sizeOfSlotHeader + padding-for-8-byte-alignment).
   262  	if wb := tx.Push(10); wb == nil {
   263  		t.Fatalf("Push failed on empty pipe")
   264  	}
   265  
   266  	if wb := tx.Push(50); wb == nil {
   267  		t.Fatalf("Push failed on non-full pipe")
   268  	}
   269  
   270  	// We haven't flushed yet, so pull must return nil.
   271  	if rb := rx.Pull(); rb != nil {
   272  		t.Fatalf("Pull succeeded on non-flushed pipe")
   273  	}
   274  
   275  	tx.Abort()
   276  
   277  	// The pushes were aborted, so no data should be readable.
   278  	if rb := rx.Pull(); rb != nil {
   279  		t.Fatalf("Pull succeeded on non-flushed pipe")
   280  	}
   281  
   282  	// Try the same transactions again, but flush this time.
   283  	if wb := tx.Push(10); wb == nil {
   284  		t.Fatalf("Push failed on empty pipe")
   285  	}
   286  
   287  	if wb := tx.Push(50); wb == nil {
   288  		t.Fatalf("Push failed on non-full pipe")
   289  	}
   290  
   291  	tx.Flush()
   292  
   293  	// The two buffers must be available now.
   294  	if rb := rx.Pull(); rb == nil {
   295  		t.Fatalf("Pull failed on non-empty pipe")
   296  	}
   297  
   298  	if rb := rx.Pull(); rb == nil {
   299  		t.Fatalf("Pull failed on non-empty pipe")
   300  	}
   301  }
   302  
   303  func TestEmptyReadOnNonFlushedWrite(t *testing.T) {
   304  	// Check that a read fails on a pipe that has had data pushed to it
   305  	// but not yet flushed.
   306  	b := make([]byte, 100)
   307  	var tx Tx
   308  	tx.Init(b)
   309  
   310  	if wb := tx.Push(10); wb == nil {
   311  		t.Fatalf("Write failed on empty pipe")
   312  	}
   313  
   314  	var rx Rx
   315  	rx.Init(b)
   316  	if rb := rx.Pull(); rb != nil {
   317  		t.Fatalf("Pull succeeded on empty pipe")
   318  	}
   319  
   320  	tx.Flush()
   321  	if rb := rx.Pull(); rb == nil {
   322  		t.Fatalf("Pull on failed on non-empty pipe")
   323  	}
   324  }
   325  
   326  func TestPullAfterPullingEntirePipe(t *testing.T) {
   327  	// Check that Pull fails when the pipe is full, but all of it has
   328  	// already been pulled but not yet flushed.
   329  	b := make([]byte, 100)
   330  	var tx Tx
   331  	tx.Init(b)
   332  
   333  	if wb := tx.Push(50); wb == nil {
   334  		t.Fatalf("Push failed on empty pipe")
   335  	}
   336  	tx.Flush()
   337  
   338  	var rx Rx
   339  	rx.Init(b)
   340  	if rb := rx.Pull(); rb == nil {
   341  		t.Fatalf("Pull failed on non-empty pipe")
   342  	}
   343  	rx.Flush()
   344  
   345  	// At this point the ring buffer is empty, but the write is at offset
   346  	// 64 (50 + sizeOfSlotHeader + padding-for-8-byte-alignment). Write 3
   347  	// buffers that will fill the pipe.
   348  	if wb := tx.Push(10); wb == nil {
   349  		t.Fatalf("Push failed on empty pipe")
   350  	}
   351  
   352  	if wb := tx.Push(20); wb == nil {
   353  		t.Fatalf("Push failed on non-full pipe")
   354  	}
   355  
   356  	if wb := tx.Push(24); wb == nil {
   357  		t.Fatalf("Push failed on non-full pipe")
   358  	}
   359  
   360  	tx.Flush()
   361  
   362  	// The three buffers must be available now.
   363  	if rb := rx.Pull(); rb == nil {
   364  		t.Fatalf("Pull failed on non-empty pipe")
   365  	}
   366  
   367  	if rb := rx.Pull(); rb == nil {
   368  		t.Fatalf("Pull failed on non-empty pipe")
   369  	}
   370  
   371  	if rb := rx.Pull(); rb == nil {
   372  		t.Fatalf("Pull failed on non-empty pipe")
   373  	}
   374  
   375  	// Fourth pull must fail.
   376  	if rb := rx.Pull(); rb != nil {
   377  		t.Fatalf("Pull succeeded on empty pipe")
   378  	}
   379  }
   380  
   381  func TestNoRoomToWrapOnPush(t *testing.T) {
   382  	// Check that Push fails when it tries to allocate room to add a wrap
   383  	// message.
   384  	b := make([]byte, 100)
   385  	var tx Tx
   386  	tx.Init(b)
   387  
   388  	if wb := tx.Push(50); wb == nil {
   389  		t.Fatalf("Push failed on empty pipe")
   390  	}
   391  	tx.Flush()
   392  
   393  	var rx Rx
   394  	rx.Init(b)
   395  	if rb := rx.Pull(); rb == nil {
   396  		t.Fatalf("Pull failed on non-empty pipe")
   397  	}
   398  	rx.Flush()
   399  
   400  	// At this point the ring buffer is empty, but the write is at offset
   401  	// 64 (50 + sizeOfSlotHeader + padding-for-8-byte-alignment). Write 20,
   402  	// which won't fit (64+20+8+padding = 96, which wouldn't leave room for
   403  	// the padding), so it wraps around.
   404  	if wb := tx.Push(20); wb == nil {
   405  		t.Fatalf("Push failed on empty pipe")
   406  	}
   407  
   408  	tx.Flush()
   409  
   410  	// Buffer offset is at 28. Try to write 70, which would require a wrap
   411  	// slot which cannot be created now.
   412  	if wb := tx.Push(70); wb != nil {
   413  		t.Fatalf("Push succeeded on pipe with no room for wrap message")
   414  	}
   415  }
   416  
   417  func TestRxImplicitFlushOfWrapMessage(t *testing.T) {
   418  	// Check if the first read is that of a wrapping message, that it gets
   419  	// immediately flushed.
   420  	b := make([]byte, 100)
   421  	var tx Tx
   422  	tx.Init(b)
   423  
   424  	if wb := tx.Push(50); wb == nil {
   425  		t.Fatalf("Push failed on empty pipe")
   426  	}
   427  	tx.Flush()
   428  
   429  	// This will cause a wrapping message to written.
   430  	if wb := tx.Push(60); wb != nil {
   431  		t.Fatalf("Push succeeded when there is no room in pipe")
   432  	}
   433  
   434  	var rx Rx
   435  	rx.Init(b)
   436  
   437  	// Read the first message.
   438  	if rb := rx.Pull(); rb == nil {
   439  		t.Fatalf("Pull failed on non-empty pipe")
   440  	}
   441  	rx.Flush()
   442  
   443  	// This should fail because of the wrapping message is taking up space.
   444  	if wb := tx.Push(60); wb != nil {
   445  		t.Fatalf("Push succeeded when there is no room in pipe")
   446  	}
   447  
   448  	// Try to read the next one. This should consume the wrapping message.
   449  	rx.Pull()
   450  
   451  	// This must now succeed.
   452  	if wb := tx.Push(60); wb == nil {
   453  		t.Fatalf("Push failed on empty pipe")
   454  	}
   455  }
   456  
   457  func TestConcurrentReaderWriter(t *testing.T) {
   458  	// Push a million buffers of random sizes and random contents. Check
   459  	// that buffers read match what was written.
   460  	tr := rand.New(rand.NewSource(99))
   461  	rr := rand.New(rand.NewSource(99))
   462  
   463  	b := make([]byte, 100)
   464  	var tx Tx
   465  	tx.Init(b)
   466  
   467  	var rx Rx
   468  	rx.Init(b)
   469  
   470  	const count = 1000000
   471  	var wg sync.WaitGroup
   472  	wg.Add(1)
   473  	go func() {
   474  		defer wg.Done()
   475  		runtime.Gosched()
   476  		for i := 0; i < count; i++ {
   477  			n := 1 + tr.Intn(80)
   478  			wb := tx.Push(uint64(n))
   479  			for wb == nil {
   480  				wb = tx.Push(uint64(n))
   481  			}
   482  
   483  			for j := range wb {
   484  				wb[j] = byte(tr.Intn(256))
   485  			}
   486  
   487  			tx.Flush()
   488  		}
   489  	}()
   490  
   491  	wg.Add(1)
   492  	go func() {
   493  		defer wg.Done()
   494  		runtime.Gosched()
   495  		for i := 0; i < count; i++ {
   496  			n := 1 + rr.Intn(80)
   497  			rb := rx.Pull()
   498  			for rb == nil {
   499  				rb = rx.Pull()
   500  			}
   501  
   502  			if n != len(rb) {
   503  				t.Fatalf("Bad %v-th buffer length: got %v, want %v", i, len(rb), n)
   504  			}
   505  
   506  			for j := range rb {
   507  				if v := byte(rr.Intn(256)); v != rb[j] {
   508  					t.Fatalf("Bad %v-th read buffer at index %v: got %v, want %v", i, j, rb[j], v)
   509  				}
   510  			}
   511  
   512  			rx.Flush()
   513  		}
   514  	}()
   515  
   516  	wg.Wait()
   517  }