multi.odin 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. package io
  2. Multi_Reader :: struct {
  3. readers: [dynamic]Reader,
  4. }
  5. @(private)
  6. _multi_reader_proc :: proc(stream_data: rawptr, mode: Stream_Mode, p: []byte, offset: i64, whence: Seek_From) -> (n: i64, err: Error) {
  7. if mode == .Query {
  8. return query_utility({.Read, .Query})
  9. } else if mode != .Read {
  10. return 0, .Empty
  11. }
  12. mr := (^Multi_Reader)(stream_data)
  13. for len(mr.readers) > 0 {
  14. r := mr.readers[0]
  15. n, err = _i64_err(read(r, p))
  16. if err == .EOF {
  17. ordered_remove(&mr.readers, 0)
  18. }
  19. if n > 0 || err != .EOF {
  20. if err == .EOF && len(mr.readers) > 0 {
  21. // Don't return EOF yet, more readers remain
  22. err = nil
  23. }
  24. return
  25. }
  26. }
  27. return 0, .EOF
  28. }
  29. multi_reader_init :: proc(mr: ^Multi_Reader, readers: ..Reader, allocator := context.allocator) -> (r: Reader) {
  30. all_readers := make([dynamic]Reader, 0, len(readers), allocator)
  31. for w in readers {
  32. if w.procedure == _multi_reader_proc {
  33. other := (^Multi_Reader)(w.data)
  34. append(&all_readers, ..other.readers[:])
  35. } else {
  36. append(&all_readers, w)
  37. }
  38. }
  39. mr.readers = all_readers
  40. r.procedure = _multi_reader_proc
  41. r.data = mr
  42. return
  43. }
  44. multi_reader_destroy :: proc(mr: ^Multi_Reader) {
  45. delete(mr.readers)
  46. }
  47. Multi_Writer :: struct {
  48. writers: [dynamic]Writer,
  49. }
  50. @(private)
  51. _multi_writer_proc :: proc(stream_data: rawptr, mode: Stream_Mode, p: []byte, offset: i64, whence: Seek_From) -> (n: i64, err: Error) {
  52. if mode == .Query {
  53. return query_utility({.Write, .Query})
  54. } else if mode != .Write {
  55. return 0, .Empty
  56. }
  57. mw := (^Multi_Writer)(stream_data)
  58. for w in mw.writers {
  59. n, err = _i64_err(write(w, p))
  60. if err != nil {
  61. return
  62. }
  63. if n != i64(len(p)) {
  64. err = .Short_Write
  65. return
  66. }
  67. }
  68. return i64(len(p)), nil
  69. }
  70. multi_writer_init :: proc(mw: ^Multi_Writer, writers: ..Writer, allocator := context.allocator) -> (out: Writer) {
  71. mw.writers = make([dynamic]Writer, 0, len(writers), allocator)
  72. for w in writers {
  73. if w.procedure == _multi_writer_proc {
  74. other := (^Multi_Writer)(w.data)
  75. append(&mw.writers, ..other.writers[:])
  76. } else {
  77. append(&mw.writers, w)
  78. }
  79. }
  80. out.procedure = _multi_writer_proc
  81. out.data = mw
  82. return
  83. }
  84. multi_writer_destroy :: proc(mw: ^Multi_Writer) {
  85. delete(mw.writers)
  86. }