From 0e1f1bf64dc672e06aa901b0aecd081168b40ae2 Mon Sep 17 00:00:00 2001 From: "allen.wq" Date: Mon, 27 Apr 2020 14:47:38 +0800 Subject: [PATCH] add cache for seed. Signed-off-by: allen.wq --- dfdaemon/seed/cache.go | 440 ++++++++++++++++++++++++++++++++++++ dfdaemon/seed/cache_test.go | 320 ++++++++++++++++++++++++++ pkg/bitmap/bitmap.go | 16 ++ 3 files changed, 776 insertions(+) create mode 100644 dfdaemon/seed/cache.go create mode 100644 dfdaemon/seed/cache_test.go diff --git a/dfdaemon/seed/cache.go b/dfdaemon/seed/cache.go new file mode 100644 index 000000000..a6874add6 --- /dev/null +++ b/dfdaemon/seed/cache.go @@ -0,0 +1,440 @@ +/* + * Copyright The Dragonfly Authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package seed + +import ( + "bytes" + "fmt" + "io" + "net/http" + "os" + "sort" + "sync" + + "github.com/dragonflyoss/Dragonfly/pkg/bitmap" + "github.com/dragonflyoss/Dragonfly/pkg/errortypes" + + lbm "github.com/openacid/low/bitmap" +) + +// cacheBuffer interface caches the seed file +type cacheBuffer interface { + io.WriterAt + // write close + io.Closer + Sync() error + + // ReadStream prepares io.ReadCloser from cacheBuffer. + ReadStream(off int64, size int64) (io.ReadCloser, error) + + // remove the cache + Remove() error + + // the cache full size + FullSize() int64 +} + +func newFileCacheBuffer(path string, fullSize int64, trunc bool, memoryCache bool, blockOrder uint32) (cb cacheBuffer, err error) { + var ( + fw *os.File + ) + + _, err = os.Stat(path) + if err != nil { + if !os.IsNotExist(err) { + return nil, err + } + } + + if trunc { + fw, err = os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0644) + } else { + fw, err = os.OpenFile(path, os.O_CREATE|os.O_WRONLY, 0644) + } + + if err != nil { + return nil, err + } + + defer func() { + if err != nil { + fw.Close() + } + }() + + fcb := &fileCacheBuffer{path: path, fw: fw, fullSize: fullSize, memoryCache: memoryCache} + if memoryCache { + fcb.blockOrder = blockOrder + fcb.blockSize = 1 << blockOrder + blocks := fullSize / int64(fcb.blockSize) + if (fullSize % int64(fcb.blockSize)) > 0 { + blocks++ + } + + fcb.blockMeta, err = bitmap.NewBitMapWithNumBits(uint32(blocks), false) + if err != nil { + return nil, err + } + + fcb.memCacheMap = make(map[int64]*bytes.Buffer) + fcb.maxBlockIndex = uint32(blocks - 1) + } + + return fcb, nil +} + +type fileCacheBuffer struct { + // the lock protects fields of 'remove', 'memCacheMap' + sync.RWMutex + + path string + fw *os.File + remove bool + fullSize int64 + + // memory cache mode, in the mode, it will store cache in temperate memory. + // if sync is called, the temperate memory will sync to local fs. + // in memory cache mode, WriteAt should transfer a block buffer. + memoryCache bool + blockMeta *bitmap.BitMap + blockSize int32 + blockOrder uint32 + maxBlockIndex uint32 + + // memCacheMap caches the buffer, and the buffer should not be changed if added to the map. + // the key is bytes start index of buffer cache in full cache. + memCacheMap map[int64]*bytes.Buffer +} + +// if in memory mode, the write data buffer will be reused, so don't change the buffer. +func (fcb *fileCacheBuffer) WriteAt(p []byte, off int64) (n int, err error) { + err = fcb.checkWriteAtValid(off, int64(len(p))) + if err != nil { + return + } + + if fcb.memoryCache { + fcb.storeMemoryCache(p, off) + return len(p), nil + } + + return fcb.fw.WriteAt(p, off) +} + +// Close closes the file writer. +func (fcb *fileCacheBuffer) Close() error { + if err := fcb.Sync(); err != nil { + return err + } + return fcb.fw.Close() +} + +func (fcb *fileCacheBuffer) Sync() error { + fcb.Lock() + remove := fcb.remove + fcb.Unlock() + + if remove { + return io.ErrClosedPipe + } + + if fcb.memoryCache { + if err := fcb.syncMemoryCache(); err != nil { + return err + } + } + + return fcb.fw.Sync() +} + +func (fcb *fileCacheBuffer) ReadStream(off int64, size int64) (io.ReadCloser, error) { + off, size, err := fcb.checkReadStreamParam(off, size) + if err != nil { + return nil, err + } + + return fcb.openReadCloser(off, size) +} + +func (fcb *fileCacheBuffer) Remove() error { + fcb.Lock() + defer fcb.Unlock() + + if fcb.remove { + return nil + } + + fcb.remove = true + return os.Remove(fcb.path) +} + +func (fcb *fileCacheBuffer) FullSize() int64 { + fcb.RLock() + defer fcb.RUnlock() + + return fcb.fullSize +} + +func (fcb *fileCacheBuffer) checkReadStreamParam(off int64, size int64) (int64, int64, error) { + fcb.RLock() + defer fcb.RUnlock() + + if fcb.remove { + return 0, 0, io.ErrClosedPipe + } + + if off < 0 { + off = 0 + } + + // Note: if file size if zero, it should be specially handled in the upper caller. + // In current progress, if size <= 0, it means to read to the end of file. + // if size <= 0, set range to [off, fullSize-1] + if size <= 0 { + size = fcb.fullSize - off + } + + if off+size > fcb.fullSize { + return 0, 0, errortypes.NewHTTPError(http.StatusRequestedRangeNotSatisfiable, "out of range") + } + + return off, size, nil +} + +func (fcb *fileCacheBuffer) storeMemoryCache(p []byte, off int64) { + fcb.Lock() + defer fcb.Unlock() + + if _, ok := fcb.memCacheMap[off]; ok { + return + } + + buf := bytes.NewBuffer(p) + fcb.memCacheMap[off] = buf + + startBlockIndex := uint32(off >> fcb.blockOrder) + // set bits in bit map + fcb.blockMeta.Set(startBlockIndex, startBlockIndex, true) +} + +// syncMemoryCache flushes the memory cache to local file. +func (fcb *fileCacheBuffer) syncMemoryCache() error { + var ( + err error + ) + + var arr []*struct { + off int64 + buf *bytes.Buffer + } + + fcb.RLock() + for off, buf := range fcb.memCacheMap { + arr = append(arr, &struct { + off int64 + buf *bytes.Buffer + }{off: off, buf: buf}) + } + fcb.RUnlock() + + sort.Slice(arr, func(i, j int) bool { + return arr[i].off < arr[j].off + }) + + for i := 0; i < len(arr); i++ { + err = fcb.syncBlockToFile(arr[i].buf.Bytes(), arr[i].off) + if err != nil { + return err + } + } + + return nil +} + +func (fcb *fileCacheBuffer) syncBlockToFile(p []byte, off int64) error { + n, err := fcb.fw.WriteAt(p, off) + if err != nil { + return err + } + + if n < len(p) { + return io.ErrShortWrite + } + + fcb.Lock() + defer fcb.Unlock() + + blockIndex := uint32(off >> fcb.blockOrder) + delete(fcb.memCacheMap, off) + fcb.blockMeta.Set(blockIndex, blockIndex, false) + return nil +} + +func (fcb *fileCacheBuffer) openReadCloser(off int64, size int64) (io.ReadCloser, error) { + if fcb.memoryCache { + return fcb.openReadCloserInMemoryCacheMode(off, size) + } + + fr, err := os.Open(fcb.path) + if err != nil { + return nil, err + } + + return newLimitReadCloser(fr, off, size) +} + +// if in memory cache mode, the reader is multi reader in which some data in memory and others in file. +func (fcb *fileCacheBuffer) openReadCloserInMemoryCacheMode(off, size int64) (io.ReadCloser, error) { + var ( + rds []io.Reader + useFr bool + ) + + fr, err := os.Open(fcb.path) + if err != nil { + return nil, err + } + + fcb.RLock() + defer fcb.RUnlock() + + if len(fcb.memCacheMap) == 0 { + return newLimitReadCloser(fr, off, size) + } + + currentOff := off + var currentBlock int32 + var currentBlockStartBytes, currentBlockEndBytes, useBlockBytes, currentBlockOff, currentBlockOffEnd int64 + for { + if size <= 0 { + break + } + + currentBlock = int32(currentOff >> fcb.blockOrder) + currentBlockStartBytes = int64(currentBlock) << fcb.blockOrder + currentBlockEndBytes = currentBlockStartBytes + int64(fcb.blockSize) - 1 + if currentBlockEndBytes >= fcb.fullSize { + currentBlockEndBytes = fcb.fullSize - 1 + } + + useBlockBytes = currentBlockEndBytes - currentOff + 1 + if useBlockBytes > size { + useBlockBytes = size + } + + currentBlockOff = currentOff - currentBlockStartBytes + currentBlockOffEnd = currentBlockOff + useBlockBytes - 1 + buf, ok := fcb.memCacheMap[currentBlockStartBytes] + if ok { + // currentBlock in memory + b := buf.Bytes() + rd := bytes.NewReader(b[currentBlockOff : currentBlockOffEnd+1]) + rds = append(rds, rd) + } else { + // else currentBlock in file + rd := io.NewSectionReader(fr, currentOff, useBlockBytes) + rds = append(rds, rd) + useFr = true + } + + size -= useBlockBytes + currentOff += useBlockBytes + } + + if !useFr { + fr.Close() + fr = nil + } + + return newMultiReadCloser(rds, fr), nil +} + +func (fcb *fileCacheBuffer) checkWriteAtValid(off, size int64) error { + if !fcb.memoryCache { + return nil + } + + if uint64(off)&(lbm.Mask[fcb.blockOrder]) > 0 { + return fmt.Errorf("memory cache mode, off %d should be align with blockSize %d", off, fcb.blockSize) + } + + maxIndex := off + size - 1 + + if maxIndex >= fcb.fullSize { + return fmt.Errorf("memory cache mode, max write index %d should be smaller than max block index %d", maxIndex, fcb.fullSize) + } + + // if last block, the size may smaller than block size + if uint32(off>>fcb.blockOrder) == fcb.maxBlockIndex { + return nil + } + + if size != int64(fcb.blockSize) { + return fmt.Errorf("memory cache mode, len of bytes %d should be equal to block size %d", size, fcb.blockSize) + } + + return nil +} + +// fileReadCloser provides a selection readCloser of file. +type fileReadCloser struct { + sr *io.SectionReader + fr *os.File +} + +func newLimitReadCloser(fr *os.File, off int64, size int64) (io.ReadCloser, error) { + sr := io.NewSectionReader(fr, off, size) + return &fileReadCloser{ + sr: sr, + fr: fr, + }, nil +} + +func (lr *fileReadCloser) Read(p []byte) (n int, err error) { + return lr.sr.Read(p) +} + +func (lr *fileReadCloser) Close() error { + return lr.fr.Close() +} + +// multiReadCloser provides multi ReadCloser. +type multiReadCloser struct { + rds []io.Reader + realRd io.Reader + fr *os.File +} + +func newMultiReadCloser(rds []io.Reader, fr *os.File) *multiReadCloser { + return &multiReadCloser{ + rds: rds, + realRd: io.MultiReader(rds...), + fr: fr, + } +} + +func (mr *multiReadCloser) Read(p []byte) (n int, err error) { + return mr.realRd.Read(p) +} + +func (mr *multiReadCloser) Close() error { + if mr.fr != nil { + return mr.fr.Close() + } + + return nil +} diff --git a/dfdaemon/seed/cache_test.go b/dfdaemon/seed/cache_test.go new file mode 100644 index 000000000..6bb317e63 --- /dev/null +++ b/dfdaemon/seed/cache_test.go @@ -0,0 +1,320 @@ +/* + * Copyright The Dragonfly Authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package seed + +import ( + "bytes" + "io" + "io/ioutil" + "net/http" + "path/filepath" + + "github.com/dragonflyoss/Dragonfly/pkg/errortypes" + + "github.com/go-check/check" +) + +func (suite *SeedTestSuite) TestFileCacheBufferWithNoFile(c *check.C) { + testDir := suite.tmpDir + + cb, err := newFileCacheBuffer(filepath.Join(testDir, "fileA"), 30, true, false, 0) + c.Assert(err, check.IsNil) + + data := []byte("0123456789") + // write data + n, err := cb.WriteAt(data, 0) + c.Assert(int(n), check.Equals, len(data)) + c.Assert(err, check.IsNil) + + // write data + n, err = cb.WriteAt(data, 10) + c.Assert(int(n), check.Equals, len(data)) + c.Assert(err, check.IsNil) + + // read stream + rc, err := cb.ReadStream(0, 10) + c.Assert(err, check.IsNil) + data0, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data0), check.Equals, 10) + expectAllData := []byte("0123456789") + c.Assert(string(data0), check.Equals, string(expectAllData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + rc, err = cb.ReadStream(0, 20) + c.Assert(err, check.IsNil) + data0, err = ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data0), check.Equals, 20) + expectAllData = []byte("01234567890123456789") + c.Assert(string(data0), check.Equals, string(expectAllData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // write data + n, err = cb.WriteAt(data, 20) + c.Assert(int(n), check.Equals, len(data)) + c.Assert(err, check.IsNil) + err = cb.Sync() + c.Assert(err, check.IsNil) + + // close + err = cb.Close() + c.Assert(err, check.IsNil) + + // read all + rc, err = cb.ReadStream(0, -1) + c.Assert(err, check.IsNil) + data1, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data1), check.Equals, len(data)*3) + expectAllData = []byte("012345678901234567890123456789") + c.Assert(string(data1), check.Equals, string(expectAllData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // read 10- + rc, err = cb.ReadStream(10, -1) + c.Assert(err, check.IsNil) + data2, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data2), check.Equals, 20) + expectData2 := []byte("01234567890123456789") + c.Assert(string(data2), check.Equals, string(expectData2)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // read 5-14 + rc, err = cb.ReadStream(5, 10) + c.Assert(err, check.IsNil) + data3, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data3), check.Equals, 10) + expectData3 := []byte("5678901234") + c.Assert(string(data3), check.Equals, string(expectData3)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // read 20-30, expect failed + _, err = cb.ReadStream(20, 11) + httpErr, ok := err.(*errortypes.HTTPError) + c.Assert(ok, check.Equals, true) + c.Assert(httpErr.HTTPCode(), check.Equals, http.StatusRequestedRangeNotSatisfiable) + + // remove cache + err = cb.Remove() + c.Assert(err, check.IsNil) + + // read again + _, err = cb.ReadStream(20, 5) + c.Assert(err, check.NotNil) +} + +func (suite *SeedTestSuite) TestFileCacheBufferWithExistFile(c *check.C) { + testDir := suite.tmpDir + + // create cb + cb, err := newFileCacheBuffer(filepath.Join(testDir, "fileB"), 35, true, false, 0) + c.Assert(err, check.IsNil) + + inputData1 := []byte("0123456789") + inputData2 := []byte("abcde") + + // write data inputData1 * 3 + n, err := cb.WriteAt(inputData1, 0) + c.Assert(int(n), check.Equals, len(inputData1)) + c.Assert(err, check.IsNil) + + n, err = cb.WriteAt(inputData1, 10) + c.Assert(int(n), check.Equals, len(inputData1)) + c.Assert(err, check.IsNil) + + n, err = cb.WriteAt(inputData1, 20) + c.Assert(int(n), check.Equals, len(inputData1)) + c.Assert(err, check.IsNil) + + err = cb.Close() + c.Assert(err, check.IsNil) + + // reopen again + cb, err = newFileCacheBuffer(filepath.Join(testDir, "fileB"), 35, false, false, 0) + c.Assert(err, check.IsNil) + + rc, err := cb.ReadStream(0, 30) + c.Assert(err, check.IsNil) + data0, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data0), check.Equals, 30) + expectAllData := []byte("012345678901234567890123456789") + c.Assert(string(data0), check.Equals, string(expectAllData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // write data inputData2 + n, err = cb.WriteAt(inputData2, 30) + c.Assert(int(n), check.Equals, len(inputData2)) + c.Assert(err, check.IsNil) + + // close + err = cb.Close() + c.Assert(err, check.IsNil) + + // read all + rc, err = cb.ReadStream(0, -1) + c.Assert(err, check.IsNil) + data1, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data1), check.Equals, 35) + expectAllData = []byte("012345678901234567890123456789abcde") + c.Assert(string(data1), check.Equals, string(expectAllData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // read 5-29 + rc, err = cb.ReadStream(5, 25) + c.Assert(err, check.IsNil) + data2, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data2), check.Equals, 25) + expectData2 := []byte("5678901234567890123456789") + c.Assert(string(data2), check.Equals, string(expectData2)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // read 10-34 + rc, err = cb.ReadStream(10, 25) + c.Assert(err, check.IsNil) + data3, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + c.Assert(len(data3), check.Equals, 25) + expectData3 := []byte("01234567890123456789abcde") + c.Assert(string(data3), check.Equals, string(expectData3)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // read 10-35. expect failed + _, err = cb.ReadStream(10, 26) + httpErr, ok := err.(*errortypes.HTTPError) + c.Assert(ok, check.Equals, true) + c.Assert(httpErr.HTTPCode(), check.Equals, http.StatusRequestedRangeNotSatisfiable) + + // remove cache + err = cb.Remove() + c.Assert(err, check.IsNil) + + // read again + _, err = cb.ReadStream(20, 5) + c.Assert(err, check.NotNil) +} + +func (suite *SeedTestSuite) TestCacheMemoryMode(c *check.C) { + testDir := suite.tmpDir + cb1, err := newFileCacheBuffer(filepath.Join(testDir, "TestCacheMemoryModeFileB"), 64, true, true, 4) + c.Assert(err, check.IsNil) + + inputData1 := []byte("0123456789abcdef") + inputData2 := []byte("fedcba9876543210") + + data1 := make([]byte, 16) + copy(data1, inputData1) + n, err := cb1.WriteAt(data1, 0) + c.Assert(err, check.IsNil) + c.Assert(n, check.Equals, 16) + + rc, err := cb1.ReadStream(0, 10) + c.Assert(err, check.IsNil) + rcData, err := ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + expectData := []byte("0123456789") + c.Assert(string(rcData), check.Equals, string(expectData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + data1 = make([]byte, 16) + copy(data1, inputData2) + n, err = cb1.WriteAt(data1, 32) + c.Assert(err, check.IsNil) + c.Assert(n, check.Equals, 16) + + rc, err = cb1.ReadStream(32, 10) + c.Assert(err, check.IsNil) + rcData, err = ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + expectData = []byte("fedcba9876") + c.Assert(string(rcData), check.Equals, string(expectData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // sync + err = cb1.Sync() + c.Check(err, check.IsNil) + + data1 = make([]byte, 16) + copy(data1, inputData2) + n, err = cb1.WriteAt(data1, 16) + c.Assert(err, check.IsNil) + c.Assert(n, check.Equals, 16) + + rc, err = cb1.ReadStream(10, 20) + c.Assert(err, check.IsNil) + rcData, err = ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + expectData = []byte("abcdeffedcba98765432") + c.Assert(string(rcData), check.Equals, string(expectData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + rc, err = cb1.ReadStream(10, 20) + c.Assert(err, check.IsNil) + rcData, err = ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + expectData = []byte("abcdeffedcba98765432") + c.Assert(string(rcData), check.Equals, string(expectData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + rc, err = cb1.ReadStream(15, 20) + c.Assert(err, check.IsNil) + mrc, ok := rc.(*multiReadCloser) + c.Assert(ok, check.Equals, true) + c.Assert(len(mrc.rds), check.Equals, 3) + _, ok = mrc.rds[0].(*io.SectionReader) + c.Assert(ok, check.Equals, true) + _, ok = mrc.rds[1].(*bytes.Reader) + c.Assert(ok, check.Equals, true) + _, ok = mrc.rds[2].(*io.SectionReader) + c.Assert(ok, check.Equals, true) + rcData, err = ioutil.ReadAll(rc) + c.Assert(err, check.IsNil) + expectData = []byte("ffedcba9876543210fed") + c.Assert(string(rcData), check.Equals, string(expectData)) + err = rc.Close() + c.Assert(err, check.IsNil) + + // sync + err = cb1.Sync() + c.Assert(err, check.IsNil) + rc, err = cb1.ReadStream(15, 20) + c.Assert(err, check.IsNil) + _, ok = rc.(*fileReadCloser) + c.Assert(ok, check.Equals, true) + expectData = []byte("ffedcba9876543210fed") + c.Assert(string(rcData), check.Equals, string(expectData)) + err = rc.Close() + c.Assert(err, check.IsNil) +} diff --git a/pkg/bitmap/bitmap.go b/pkg/bitmap/bitmap.go index 383a9206d..10cf048a3 100644 --- a/pkg/bitmap/bitmap.go +++ b/pkg/bitmap/bitmap.go @@ -57,6 +57,22 @@ func NewBitMap(sizeOf64Bits uint32, allSetBit bool) (*BitMap, error) { }, nil } +// NewBitMapWithNumBits generates a BitMap. +func NewBitMapWithNumBits(numberBits uint32, allSetBit bool) (*BitMap, error) { + sizeOf64Bits := uint32(numberBits / 64) + if (numberBits % 64) > 0 { + sizeOf64Bits++ + } + + bm, err := NewBitMap(sizeOf64Bits, allSetBit) + if err != nil { + return nil, err + } + + bm.maxBitIndex = numberBits - 1 + return bm, nil +} + // RestoreBitMap generate the BitMap by input bytes. func RestoreBitMap(data []byte) (*BitMap, error) { if uint64(len(data)) > (sizeOf64BitsLimit << 3) {