ring.go 1.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. package server
  2. import (
  3. "go-common/app/interface/main/broadcast/conf"
  4. "go-common/app/service/main/broadcast/model"
  5. "go-common/library/log"
  6. )
  7. // Ring .
  8. type Ring struct {
  9. // read
  10. rp uint64
  11. num uint64
  12. mask uint64
  13. // TODO split cacheline, many cpu cache line size is 64
  14. // pad [40]byte
  15. // write
  16. wp uint64
  17. data []model.Proto
  18. }
  19. // NewRing .
  20. func NewRing(num int) *Ring {
  21. r := new(Ring)
  22. r.init(uint64(num))
  23. return r
  24. }
  25. // Init .
  26. func (r *Ring) Init(num int) {
  27. r.init(uint64(num))
  28. }
  29. func (r *Ring) init(num uint64) {
  30. // 2^N
  31. if num&(num-1) != 0 {
  32. for num&(num-1) != 0 {
  33. num &= (num - 1)
  34. }
  35. num = num << 1
  36. }
  37. r.data = make([]model.Proto, num)
  38. r.num = num
  39. r.mask = r.num - 1
  40. }
  41. // Get .
  42. func (r *Ring) Get() (proto *model.Proto, err error) {
  43. if r.rp == r.wp {
  44. return nil, ErrRingEmpty
  45. }
  46. proto = &r.data[r.rp&r.mask]
  47. return
  48. }
  49. // GetAdv .
  50. func (r *Ring) GetAdv() {
  51. r.rp++
  52. if conf.Conf.Broadcast.Debug {
  53. log.Info("ring rp: %d, idx: %d", r.rp, r.rp&r.mask)
  54. }
  55. }
  56. // Set .
  57. func (r *Ring) Set() (proto *model.Proto, err error) {
  58. if r.wp-r.rp >= r.num {
  59. return nil, ErrRingFull
  60. }
  61. proto = &r.data[r.wp&r.mask]
  62. return
  63. }
  64. // SetAdv .
  65. func (r *Ring) SetAdv() {
  66. r.wp++
  67. if conf.Conf.Broadcast.Debug {
  68. log.Info("ring wp: %d, idx: %d", r.wp, r.wp&r.mask)
  69. }
  70. }
  71. // Reset .
  72. func (r *Ring) Reset() {
  73. r.rp = 0
  74. r.wp = 0
  75. // prevent pad compiler optimization
  76. // r.pad = [40]byte{}
  77. }