multi.odin 2.0 KB

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