crc32_field.go 1.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. package sarama
  2. import (
  3. "encoding/binary"
  4. "fmt"
  5. "hash/crc32"
  6. )
  7. const (
  8. crcIEEE = iota
  9. crcCastagnoli
  10. )
  11. var castagnoliTable = crc32.MakeTable(crc32.Castagnoli)
  12. // crc32Field implements the pushEncoder and pushDecoder interfaces for calculating CRC32s.
  13. type crc32Field struct {
  14. startOffset int
  15. polynomial int
  16. }
  17. func (c *crc32Field) saveOffset(in int) {
  18. c.startOffset = in
  19. }
  20. func (c *crc32Field) reserveLength() int {
  21. return 4
  22. }
  23. func newCRC32Field(polynomial int) *crc32Field {
  24. return &crc32Field{polynomial: polynomial}
  25. }
  26. func (c *crc32Field) run(curOffset int, buf []byte) error {
  27. crc, err := c.crc(curOffset, buf)
  28. if err != nil {
  29. return err
  30. }
  31. binary.BigEndian.PutUint32(buf[c.startOffset:], crc)
  32. return nil
  33. }
  34. func (c *crc32Field) check(curOffset int, buf []byte) error {
  35. crc, err := c.crc(curOffset, buf)
  36. if err != nil {
  37. return err
  38. }
  39. expected := binary.BigEndian.Uint32(buf[c.startOffset:])
  40. if crc != expected {
  41. return PacketDecodingError{fmt.Sprintf("CRC didn't match expected %#x got %#x", expected, crc)}
  42. }
  43. return nil
  44. }
  45. func (c *crc32Field) crc(curOffset int, buf []byte) (uint32, error) {
  46. var tab *crc32.Table
  47. switch c.polynomial {
  48. case crcIEEE:
  49. tab = crc32.IEEETable
  50. case crcCastagnoli:
  51. tab = castagnoliTable
  52. default:
  53. return 0, PacketDecodingError{"invalid CRC type"}
  54. }
  55. return crc32.Checksum(buf[c.startOffset+4:curOffset], tab), nil
  56. }