123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- package io
- import "core:strconv"
- write_u64 :: proc(w: Writer, i: u64, base: int = 10) -> (n: int, err: Error) {
- buf: [32]byte;
- s := strconv.append_bits(buf[:], i, base, false, 64, strconv.digits, nil);
- return write_string(w, s);
- }
- write_i64 :: proc(w: Writer, i: i64, base: int = 10) -> (n: int, err: Error) {
- buf: [32]byte;
- s := strconv.append_bits(buf[:], u64(i), base, true, 64, strconv.digits, nil);
- return write_string(w, s);
- }
- write_uint :: proc(w: Writer, i: uint, base: int = 10) -> (n: int, err: Error) {
- return write_u64(w, u64(i), base);
- }
- write_int :: proc(w: Writer, i: int, base: int = 10) -> (n: int, err: Error) {
- return write_i64(w, i64(i), base);
- }
- Tee_Reader :: struct {
- r: Reader,
- w: Writer,
- }
- @(private)
- _tee_reader_vtable := &Stream_VTable{
- impl_read = proc(s: Stream, p: []byte) -> (n: int, err: Error) {
- t := (^Tee_Reader)(s.stream_data);
- n, err = read(t.r, p);
- if n > 0 {
- if wn, werr := write(t.w, p[:n]); werr != nil {
- return wn, werr;
- }
- }
- return;
- },
- };
- // tee_reader_init returns a Reader that writes to 'w' what it reads from 'r'
- // All reads from 'r' performed through it are matched with a corresponding write to 'w'
- // There is no internal buffering done
- // The write must complete before th read completes
- // Any error encountered whilst writing is reported as a 'read' error
- // tee_reader_init must call io.destroy when done with
- tee_reader_init :: proc(t: ^Tee_Reader, r: Reader, w: Writer, allocator := context.allocator) -> Reader {
- t.r, t.w = r, w;
- return tee_reader_to_reader(t);
- }
- tee_reader_to_reader :: proc(t: ^Tee_Reader) -> (r: Reader) {
- r.stream_data = t;
- r.stream_vtable = _tee_reader_vtable;
- return;
- }
- // A Limited_Reader reads from r but limits the amount of data returned to just n bytes.
- // Each call to read updates n to reflect the new amount remaining.
- // read returns EOF when n <= 0 or when the underlying r returns EOF.
- Limited_Reader :: struct {
- r: Reader, // underlying reader
- n: i64, // max_bytes
- }
- @(private)
- _limited_reader_vtable := &Stream_VTable{
- impl_read = proc(s: Stream, p: []byte) -> (n: int, err: Error) {
- l := (^Limited_Reader)(s.stream_data);
- if l.n <= 0 {
- return 0, .EOF;
- }
- p := p;
- if i64(len(p)) > l.n {
- p = p[0:l.n];
- }
- n, err = read(l.r, p);
- l.n -= i64(n);
- return;
- },
- };
- limited_reader_init :: proc(l: ^Limited_Reader, r: Reader, n: i64) -> Reader {
- l.r = r;
- l.n = n;
- return limited_reader_to_reader(l);
- }
- limited_reader_to_reader :: proc(l: ^Limited_Reader) -> (r: Reader) {
- r.stream_vtable = _limited_reader_vtable;
- r.stream_data = l;
- return;
- }
- // Section_Reader implements read, seek, and read_at on a section of an underlying Reader_At
- Section_Reader :: struct {
- r: Reader_At,
- base: i64,
- off: i64,
- limit: i64,
- }
- section_reader_init :: proc(s: ^Section_Reader, r: Reader_At, off: i64, n: i64) {
- s.r = r;
- s.off = off;
- s.limit = off + n;
- return;
- }
- section_reader_to_stream :: proc(s: ^Section_Reader) -> (out: Stream) {
- out.stream_data = s;
- out.stream_vtable = _section_reader_vtable;
- return;
- }
- @(private)
- _section_reader_vtable := &Stream_VTable{
- impl_read = proc(stream: Stream, p: []byte) -> (n: int, err: Error) {
- s := (^Section_Reader)(stream.stream_data);
- if s.off >= s.limit {
- return 0, .EOF;
- }
- p := p;
- if max := s.limit - s.off; i64(len(p)) > max {
- p = p[0:max];
- }
- n, err = read_at(s.r, p, s.off);
- s.off += i64(n);
- return;
- },
- impl_read_at = proc(stream: Stream, p: []byte, off: i64) -> (n: int, err: Error) {
- s := (^Section_Reader)(stream.stream_data);
- p, off := p, off;
- if off < 0 || off >= s.limit - s.base {
- return 0, .EOF;
- }
- off += s.base;
- if max := s.limit - off; i64(len(p)) > max {
- p = p[0:max];
- n, err = read_at(s.r, p, off);
- if err == nil {
- err = .EOF;
- }
- return;
- }
- return read_at(s.r, p, off);
- },
- impl_seek = proc(stream: Stream, offset: i64, whence: Seek_From) -> (n: i64, err: Error) {
- s := (^Section_Reader)(stream.stream_data);
- offset := offset;
- switch whence {
- case:
- return 0, .Invalid_Whence;
- case .Start:
- offset += s.base;
- case .Current:
- offset += s.off;
- case .End:
- offset += s.limit;
- }
- if offset < s.base {
- return 0, .Invalid_Offset;
- }
- s.off = offset;
- n = offset - s.base;
- return;
- },
- impl_size = proc(stream: Stream) -> i64 {
- s := (^Section_Reader)(stream.stream_data);
- return s.limit - s.base;
- },
- };
|