queue.odin 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. package container
  2. Queue :: struct(T: typeid) {
  3. data: Array(T),
  4. len: int,
  5. offset: int,
  6. }
  7. /*
  8. queue_init :: proc{
  9. queue_init_none,
  10. queue_init_len,
  11. queue_init_len_cap,
  12. }
  13. queue_delete
  14. queue_clear
  15. queue_len
  16. queue_cap
  17. queue_space
  18. queue_get
  19. queue_set
  20. queue_reserve
  21. queue_resize
  22. queue_push :: proc{
  23. queue_push_back,
  24. queue_push_elems,
  25. };
  26. queue_push_front
  27. queue_pop_front
  28. queue_pop_back
  29. queue_consume
  30. */
  31. queue_init_none :: proc(q: ^$Q/Queue($T), allocator := context.allocator) {
  32. queue_init_len(q, 0, allocator);
  33. }
  34. queue_init_len :: proc(q: ^$Q/Queue($T), len: int, allocator := context.allocator) {
  35. queue_init_len_cap(q, 0, 16, allocator);
  36. }
  37. queue_init_len_cap :: proc(q: ^$Q/Queue($T), len: int, cap: int, allocator := context.allocator) {
  38. array_init(&q.data, len, cap, allocator);
  39. q.len = len;
  40. q.offset = 0;
  41. }
  42. queue_init :: proc{queue_init_none, queue_init_len, queue_init_len_cap};
  43. queue_delete :: proc(q: $Q/Queue($T)) {
  44. array_delete(q.data);
  45. }
  46. queue_clear :: proc(q: ^$Q/Queue($T)) {
  47. q.len = 0;
  48. }
  49. queue_len :: proc(q: $Q/Queue($T)) -> int {
  50. return q.len;
  51. }
  52. queue_cap :: proc(q: $Q/Queue($T)) -> int {
  53. return array_cap(q.data);
  54. }
  55. queue_space :: proc(q: $Q/Queue($T)) -> int {
  56. return array_len(q.data) - q.len;
  57. }
  58. queue_get :: proc(q: $Q/Queue($T), index: int) -> T {
  59. i := (index + q.offset) % array_len(q.data);
  60. data := array_slice(q.data);
  61. return data[i];
  62. }
  63. queue_set :: proc(q: ^$Q/Queue($T), index: int, item: T) {
  64. i := (index + q.offset) % array_len(q.data);
  65. data := array_slice(q.data);
  66. data[i] = item;
  67. }
  68. queue_reserve :: proc(q: ^$Q/Queue($T), capacity: int) {
  69. if capacity > q.len {
  70. _queue_increase_capacity(q, capacity);
  71. }
  72. }
  73. queue_resize :: proc(q: ^$Q/Queue($T), length: int) {
  74. if length > q.len {
  75. _queue_increase_capacity(q, length);
  76. }
  77. q.len = length;
  78. }
  79. queue_push_back :: proc(q: ^$Q/Queue($T), item: T) {
  80. if queue_space(q^) == 0 {
  81. _queue_grow(q);
  82. }
  83. queue_set(q, q.len, item);
  84. q.len += 1;
  85. }
  86. queue_push_front :: proc(q: ^$Q/Queue($T), item: T) {
  87. if queue_space(q^) == 0 {
  88. _queue_grow(q);
  89. }
  90. q.offset = (q.offset - 1 + array_len(q.data)) % array_len(q.data);
  91. q.len += 1;
  92. queue_set(q, 0, item);
  93. }
  94. queue_pop_front :: proc(q: ^$Q/Queue($T)) -> T {
  95. assert(q.len > 0);
  96. item := queue_get(q^, 0);
  97. q.offset = (q.offset + 1) % array_len(q.data);
  98. q.len -= 1;
  99. if q.len == 0 {
  100. q.offset = 0;
  101. }
  102. return item;
  103. }
  104. queue_pop_back :: proc(q: ^$Q/Queue($T)) -> T {
  105. assert(q.len > 0);
  106. item := queue_get(q^, q.len-1);
  107. q.len -= 1;
  108. return item;
  109. }
  110. queue_consume :: proc(q: ^$Q/Queue($T), count: int) {
  111. q.offset = (q.offset + count) & array_len(q.data);
  112. q.len -= count;
  113. }
  114. queue_push_elems :: proc(q: ^$Q/Queue($T), items: ..T) {
  115. if queue_space(q^) < len(items) {
  116. _queue_grow(q, q.len + len(items));
  117. }
  118. size := array_len(q.data);
  119. insert := (q.offset + q.len) % size;
  120. to_insert := len(items);
  121. if insert + to_insert > size {
  122. to_insert = size - insert;
  123. }
  124. the_items := items[:];
  125. data := array_slice(q.data);
  126. q.len += copy(data[insert:][:to_insert], the_items);
  127. the_items = the_items[to_insert:];
  128. q.len += copy(data[:], the_items);
  129. }
  130. queue_push :: proc{queue_push_back, queue_push_elems};
  131. _queue_increase_capacity :: proc(q: ^$Q/Queue($T), new_capacity: int) {
  132. end := array_len(q.data);
  133. array_resize(&q.data, new_capacity);
  134. if q.offset + q.len > end {
  135. end_items := q.len + end;
  136. data := array_slice(q.data);
  137. copy(data[new_capacity-end_items:][:end_items], data[q.offset:][:end_items]);
  138. q.offset += new_capacity - end;
  139. }
  140. }
  141. _queue_grow :: proc(q: ^$Q/Queue($T), min_capacity: int = 0) {
  142. new_capacity := max(array_len(q.data)*2 + 8, min_capacity);
  143. _queue_increase_capacity(q, new_capacity);
  144. }