diff --git a/pkg/storage/donut/v1/donut.go b/pkg/storage/donut/v1/donut.go index 65581f167..a6f93ae0a 100644 --- a/pkg/storage/donut/v1/donut.go +++ b/pkg/storage/donut/v1/donut.go @@ -25,6 +25,7 @@ import ( "sync" "github.com/minio-io/minio/pkg/storage/erasure" + "github.com/minio-io/minio/pkg/utils/checksum/crc32c" ) /* @@ -61,8 +62,10 @@ type DonutFormat struct { Reserved uint64 GobHeaderLen uint32 GobHeader []byte + HeaderCrc32c uint32 BlockData uint32 // Magic="DATA"=1096040772 Data io.Reader + FooterCrc uint32 BlockLen uint64 BlockEnd uint32 } @@ -107,10 +110,21 @@ func (donut *Donut) WriteGob(gobHeader GobHeader) (bytes.Buffer, error) { } func (donut *Donut) WriteEnd(target io.Writer, donutFormat DonutFormat) error { - if err := binary.Write(target, binary.LittleEndian, donutFormat.BlockLen); err != nil { + var tempBuffer bytes.Buffer + if err := binary.Write(&tempBuffer, binary.LittleEndian, donutFormat.BlockLen); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.BlockEnd); err != nil { + if err := binary.Write(&tempBuffer, binary.LittleEndian, donutFormat.BlockEnd); err != nil { + return err + } + crc, err := crc32c.Crc32c(tempBuffer.Bytes()) + if err != nil { + return err + } + if err := binary.Write(target, binary.LittleEndian, crc); err != nil { + return err + } + if _, err := io.Copy(target, &tempBuffer); err != nil { return err } return nil @@ -128,33 +142,42 @@ func (donut *Donut) WriteData(target io.Writer, donutFormat DonutFormat) error { } func (donut *Donut) WriteBegin(target io.Writer, donutFormat DonutFormat) error { - if err := binary.Write(target, binary.LittleEndian, donutFormat.BlockStart); err != nil { + var headerBytes bytes.Buffer + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.BlockStart); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.VersionMajor); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.VersionMajor); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.VersionMinor); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.VersionMinor); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.VersionPatch); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.VersionPatch); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.VersionReserved); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.VersionReserved); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.Reserved); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.Reserved); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.GobHeaderLen); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.GobHeaderLen); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.GobHeader); err != nil { + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.GobHeader); err != nil { return err } - if err := binary.Write(target, binary.LittleEndian, donutFormat.BlockData); err != nil { + crc, err := crc32c.Crc32c(headerBytes.Bytes()) + if err != nil { return err } + if err := binary.Write(&headerBytes, binary.LittleEndian, crc); err != nil { + return err + } + if err := binary.Write(&headerBytes, binary.LittleEndian, donutFormat.BlockData); err != nil { + return err + } + io.Copy(target, &headerBytes) return nil } @@ -183,17 +206,26 @@ func (donut *Donut) Write(gobHeader GobHeader, object io.Reader) error { BlockEnd: MagicINIM, } - if err := donut.WriteBegin(donut.file, donutFormat); err != nil { + var tempBuffer bytes.Buffer + + // write header + if err := donut.WriteBegin(&tempBuffer, donutFormat); err != nil { return err } - if err := donut.WriteData(donut.file, donutFormat); err != nil { + // write data + if err := donut.WriteData(&tempBuffer, donutFormat); err != nil { return err } - if err := donut.WriteEnd(donut.file, donutFormat); err != nil { + // write footer crc + if err := donut.WriteEnd(&tempBuffer, donutFormat); err != nil { return err } + // write footer + donut.file.Seek(0, 2) + io.Copy(donut.file, &tempBuffer) + return nil }