file.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. package hercules
  2. import "fmt"
  3. type File struct {
  4. tree *RBTree
  5. status map[int]int64
  6. }
  7. const TreeEnd int = -1
  8. // An ugly side of Go.
  9. // template <typename T> please!
  10. func min(a int, b int) int {
  11. if a < b {
  12. return a
  13. }
  14. return b
  15. }
  16. func min64(a int64, b int64) int64 {
  17. if a < b {
  18. return a
  19. }
  20. return b
  21. }
  22. func max(a int, b int) int {
  23. if a < b {
  24. return b
  25. }
  26. return a
  27. }
  28. func abs64(v int64) int64 {
  29. if v <= 0 {
  30. return -v
  31. }
  32. return v
  33. }
  34. func NewFile(time int, length int, status map[int]int64) *File {
  35. file := new(File)
  36. file.status = status
  37. file.tree = new(RBTree)
  38. if length > 0 {
  39. status[time] += int64(length)
  40. file.tree.Insert(Item{key: 0, value: time})
  41. }
  42. file.tree.Insert(Item{key: length, value: TreeEnd})
  43. return file
  44. }
  45. func (file *File) Len() int {
  46. return file.tree.Max().Item().key
  47. }
  48. func (file *File) Update(time int, pos int, ins_length int, del_length int) {
  49. if time < 0 {
  50. panic("time may not be negative")
  51. }
  52. if pos < 0 {
  53. panic("attempt to insert/delete at a negative position")
  54. }
  55. if ins_length < 0 || del_length < 0 {
  56. panic("ins_length and del_length must be nonnegative")
  57. }
  58. if ins_length|del_length == 0 {
  59. return
  60. }
  61. tree := file.tree
  62. if pos > tree.Max().Item().key {
  63. panic(fmt.Sprintf("attempt to insert after the end of the file: %d < %d",
  64. tree.Max().Item().key, pos))
  65. }
  66. if tree.Len() < 2 && tree.Min().Item().key != 0 {
  67. panic("invalid tree state")
  68. }
  69. status := file.status
  70. iter := tree.FindLE(pos)
  71. origin := *iter.Item()
  72. status[time] += int64(ins_length)
  73. if del_length == 0 {
  74. // simple case with insertions only
  75. if origin.key < pos || (origin.value == time && pos == 0) {
  76. iter = iter.Next()
  77. }
  78. for ; !iter.Limit(); iter = iter.Next() {
  79. iter.Item().key += ins_length
  80. }
  81. if origin.value != time {
  82. tree.Insert(Item{key: pos, value: time})
  83. if origin.key < pos {
  84. tree.Insert(Item{key: pos + ins_length, value: origin.value})
  85. }
  86. }
  87. return
  88. }
  89. // delete nodes
  90. for true {
  91. node := iter.Item()
  92. next_iter := iter.Next()
  93. if next_iter.Limit() {
  94. if pos+del_length > node.key {
  95. panic("attempt to delete after the end of the file")
  96. }
  97. break
  98. }
  99. delta := min(next_iter.Item().key, pos+del_length) - max(node.key, pos)
  100. if delta <= 0 {
  101. break
  102. }
  103. status[node.value] -= int64(delta)
  104. if node.key >= pos {
  105. origin = *node
  106. tree.DeleteWithIterator(iter)
  107. }
  108. iter = next_iter
  109. }
  110. // prepare for the keys update
  111. var previous *Item
  112. if ins_length > 0 && (origin.value != time || origin.key == pos) {
  113. // insert our new interval
  114. if iter.Item().value == time {
  115. prev := iter.Prev()
  116. if prev.Item().value != time {
  117. iter.Item().key = pos
  118. } else {
  119. tree.DeleteWithIterator(iter)
  120. iter = prev
  121. }
  122. origin.value = time // cancels the insertion after applying the delta
  123. } else {
  124. _, iter = tree.Insert(Item{key: pos, value: time})
  125. }
  126. } else {
  127. // rollback 1 position back, see "for true" deletion cycle ^
  128. iter = iter.Prev()
  129. previous = iter.Item()
  130. }
  131. // update the keys of all subsequent nodes
  132. delta := ins_length - del_length
  133. if delta != 0 {
  134. for iter = iter.Next(); !iter.Limit(); iter = iter.Next() {
  135. // we do not need to re-balance the tree
  136. iter.Item().key += delta
  137. }
  138. // have to adjust origin in case ins_length == 0
  139. if origin.key > pos {
  140. origin.key += delta
  141. }
  142. }
  143. if ins_length > 0 {
  144. if origin.value != time {
  145. tree.Insert(Item{pos + ins_length, origin.value})
  146. }
  147. } else if (pos > origin.key && previous.value != origin.value) || pos == origin.key {
  148. // continue the original interval
  149. tree.Insert(Item{pos, origin.value})
  150. }
  151. }
  152. func (file *File) Dump() string {
  153. buffer := ""
  154. for iter := file.tree.Min(); !iter.Limit(); iter = iter.Next() {
  155. node := iter.Item()
  156. buffer += fmt.Sprintf("%d %d\n", node.key, node.value)
  157. }
  158. return buffer
  159. }