package io Multi_Reader :: struct { readers: [dynamic]Reader, } @(private) _multi_reader_proc :: proc(stream_data: rawptr, mode: Stream_Mode, p: []byte, offset: i64, whence: Seek_From) -> (n: i64, err: Error) { if mode == .Query { return query_utility({.Read, .Query}) } else if mode != .Read { return 0, .Empty } mr := (^Multi_Reader)(stream_data) for len(mr.readers) > 0 { r := mr.readers[0] n, err = _i64_err(read(r, p)) if err == .EOF { ordered_remove(&mr.readers, 0) } if n > 0 || err != .EOF { if err == .EOF && len(mr.readers) > 0 { // Don't return EOF yet, more readers remain err = nil } return } } return 0, .EOF } multi_reader_init :: proc(mr: ^Multi_Reader, readers: ..Reader, allocator := context.allocator) -> (r: Reader) { all_readers := make([dynamic]Reader, 0, len(readers), allocator) for w in readers { if w.procedure == _multi_reader_proc { other := (^Multi_Reader)(w.data) append(&all_readers, ..other.readers[:]) } else { append(&all_readers, w) } } mr.readers = all_readers r.procedure = _multi_reader_proc r.data = mr return } multi_reader_destroy :: proc(mr: ^Multi_Reader) { delete(mr.readers) } Multi_Writer :: struct { writers: [dynamic]Writer, } @(private) _multi_writer_proc :: proc(stream_data: rawptr, mode: Stream_Mode, p: []byte, offset: i64, whence: Seek_From) -> (n: i64, err: Error) { if mode == .Query { return query_utility({.Write, .Query}) } else if mode != .Write { return 0, .Empty } mw := (^Multi_Writer)(stream_data) for w in mw.writers { n, err = _i64_err(write(w, p)) if err != nil { return } if n != i64(len(p)) { err = .Short_Write return } } return i64(len(p)), nil } multi_writer_init :: proc(mw: ^Multi_Writer, writers: ..Writer, allocator := context.allocator) -> (out: Writer) { mw.writers = make([dynamic]Writer, 0, len(writers), allocator) for w in writers { if w.procedure == _multi_writer_proc { other := (^Multi_Writer)(w.data) append(&mw.writers, ..other.writers[:]) } else { append(&mw.writers, w) } } out.procedure = _multi_writer_proc out.data = mw return } multi_writer_destroy :: proc(mw: ^Multi_Writer) { delete(mw.writers) }