This should eliminate sporadic failures, like the known double close in "seccomp". Signed-off-by: Ophestra <cat@gensokyo.uk>
		
			
				
	
	
		
			111 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			111 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package proc
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"runtime"
 | |
| )
 | |
| 
 | |
| // NewWriterTo returns a [File] that receives content from wt on fulfillment.
 | |
| func NewWriterTo(wt io.WriterTo) File { return &writeToFile{wt: wt} }
 | |
| 
 | |
| // writeToFile exports the read end of a pipe with data written by an [io.WriterTo].
 | |
| type writeToFile struct {
 | |
| 	wt io.WriterTo
 | |
| 	BaseFile
 | |
| }
 | |
| 
 | |
| func (f *writeToFile) ErrCount() int { return 3 }
 | |
| func (f *writeToFile) Fulfill(ctx context.Context, dispatchErr func(error)) error {
 | |
| 	r, w, err := os.Pipe()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	f.Set(r)
 | |
| 
 | |
| 	done := make(chan struct{})
 | |
| 	go func() {
 | |
| 		_, err = f.wt.WriteTo(w)
 | |
| 		dispatchErr(err)
 | |
| 		dispatchErr(w.Close())
 | |
| 		close(done)
 | |
| 		runtime.KeepAlive(r)
 | |
| 	}()
 | |
| 	go func() {
 | |
| 		select {
 | |
| 		case <-done:
 | |
| 			dispatchErr(nil)
 | |
| 		case <-ctx.Done():
 | |
| 			dispatchErr(w.Close()) // this aborts WriteTo with file already closed
 | |
| 			runtime.KeepAlive(r)
 | |
| 		}
 | |
| 	}()
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // NewStat returns a [File] implementing the behaviour
 | |
| // of the receiving end of xdg-dbus-proxy stat fd.
 | |
| func NewStat(s *io.Closer) File { return &statFile{s: s} }
 | |
| 
 | |
| var (
 | |
| 	ErrStatFault = errors.New("generic stat fd fault")
 | |
| 	ErrStatRead  = errors.New("unexpected stat behaviour")
 | |
| )
 | |
| 
 | |
| // statFile implements xdg-dbus-proxy stat fd behaviour.
 | |
| type statFile struct {
 | |
| 	s *io.Closer
 | |
| 	BaseFile
 | |
| }
 | |
| 
 | |
| func (f *statFile) ErrCount() int { return 2 }
 | |
| func (f *statFile) Fulfill(ctx context.Context, dispatchErr func(error)) error {
 | |
| 	r, w, err := os.Pipe()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	f.Set(w)
 | |
| 
 | |
| 	done := make(chan struct{})
 | |
| 	go func() {
 | |
| 		defer close(done)
 | |
| 		var n int
 | |
| 
 | |
| 		n, err = r.Read(make([]byte, 1))
 | |
| 		switch n {
 | |
| 		case -1:
 | |
| 			if err == nil {
 | |
| 				err = ErrStatFault
 | |
| 			}
 | |
| 			dispatchErr(err)
 | |
| 		case 0:
 | |
| 			if err == nil {
 | |
| 				err = ErrStatRead
 | |
| 			}
 | |
| 			dispatchErr(err)
 | |
| 		case 1:
 | |
| 			dispatchErr(err)
 | |
| 		default:
 | |
| 			panic("unreachable")
 | |
| 		}
 | |
| 		runtime.KeepAlive(w)
 | |
| 	}()
 | |
| 
 | |
| 	go func() {
 | |
| 		select {
 | |
| 		case <-done:
 | |
| 			dispatchErr(nil)
 | |
| 		case <-ctx.Done():
 | |
| 			dispatchErr(r.Close()) // this aborts Read with file already closed
 | |
| 			runtime.KeepAlive(w)
 | |
| 		}
 | |
| 	}()
 | |
| 
 | |
| 	// this gets closed by the caller
 | |
| 	*f.s = r
 | |
| 	return nil
 | |
| }
 |