mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-11-04 18:13:42 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			183 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package wclayer
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"errors"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"syscall"
 | 
						|
 | 
						|
	"github.com/Microsoft/go-winio"
 | 
						|
	"github.com/Microsoft/hcsshim/internal/hcserror"
 | 
						|
	"github.com/Microsoft/hcsshim/internal/oc"
 | 
						|
	"github.com/Microsoft/hcsshim/internal/safefile"
 | 
						|
	"github.com/Microsoft/hcsshim/internal/winapi"
 | 
						|
	"go.opencensus.io/trace"
 | 
						|
)
 | 
						|
 | 
						|
type baseLayerWriter struct {
 | 
						|
	ctx context.Context
 | 
						|
	s   *trace.Span
 | 
						|
 | 
						|
	root         *os.File
 | 
						|
	f            *os.File
 | 
						|
	bw           *winio.BackupFileWriter
 | 
						|
	err          error
 | 
						|
	hasUtilityVM bool
 | 
						|
	dirInfo      []dirInfo
 | 
						|
}
 | 
						|
 | 
						|
type dirInfo struct {
 | 
						|
	path     string
 | 
						|
	fileInfo winio.FileBasicInfo
 | 
						|
}
 | 
						|
 | 
						|
// reapplyDirectoryTimes reapplies directory modification, creation, etc. times
 | 
						|
// after processing of the directory tree has completed. The times are expected
 | 
						|
// to be ordered such that parent directories come before child directories.
 | 
						|
func reapplyDirectoryTimes(root *os.File, dis []dirInfo) error {
 | 
						|
	for i := range dis {
 | 
						|
		di := &dis[len(dis)-i-1] // reverse order: process child directories first
 | 
						|
		f, err := safefile.OpenRelative(di.path, root, syscall.GENERIC_READ|syscall.GENERIC_WRITE, syscall.FILE_SHARE_READ, winapi.FILE_OPEN, winapi.FILE_DIRECTORY_FILE|syscall.FILE_FLAG_OPEN_REPARSE_POINT)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		err = winio.SetFileBasicInfo(f, &di.fileInfo)
 | 
						|
		f.Close()
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (w *baseLayerWriter) closeCurrentFile() error {
 | 
						|
	if w.f != nil {
 | 
						|
		err := w.bw.Close()
 | 
						|
		err2 := w.f.Close()
 | 
						|
		w.f = nil
 | 
						|
		w.bw = nil
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if err2 != nil {
 | 
						|
			return err2
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (w *baseLayerWriter) Add(name string, fileInfo *winio.FileBasicInfo) (err error) {
 | 
						|
	defer func() {
 | 
						|
		if err != nil {
 | 
						|
			w.err = err
 | 
						|
		}
 | 
						|
	}()
 | 
						|
 | 
						|
	err = w.closeCurrentFile()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	if filepath.ToSlash(name) == `UtilityVM/Files` {
 | 
						|
		w.hasUtilityVM = true
 | 
						|
	}
 | 
						|
 | 
						|
	var f *os.File
 | 
						|
	defer func() {
 | 
						|
		if f != nil {
 | 
						|
			f.Close()
 | 
						|
		}
 | 
						|
	}()
 | 
						|
 | 
						|
	extraFlags := uint32(0)
 | 
						|
	if fileInfo.FileAttributes&syscall.FILE_ATTRIBUTE_DIRECTORY != 0 {
 | 
						|
		extraFlags |= winapi.FILE_DIRECTORY_FILE
 | 
						|
		w.dirInfo = append(w.dirInfo, dirInfo{name, *fileInfo})
 | 
						|
	}
 | 
						|
 | 
						|
	mode := uint32(syscall.GENERIC_READ | syscall.GENERIC_WRITE | winio.WRITE_DAC | winio.WRITE_OWNER | winio.ACCESS_SYSTEM_SECURITY)
 | 
						|
	f, err = safefile.OpenRelative(name, w.root, mode, syscall.FILE_SHARE_READ, winapi.FILE_CREATE, extraFlags)
 | 
						|
	if err != nil {
 | 
						|
		return hcserror.New(err, "Failed to safefile.OpenRelative", name)
 | 
						|
	}
 | 
						|
 | 
						|
	err = winio.SetFileBasicInfo(f, fileInfo)
 | 
						|
	if err != nil {
 | 
						|
		return hcserror.New(err, "Failed to SetFileBasicInfo", name)
 | 
						|
	}
 | 
						|
 | 
						|
	w.f = f
 | 
						|
	w.bw = winio.NewBackupFileWriter(f, true)
 | 
						|
	f = nil
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (w *baseLayerWriter) AddLink(name string, target string) (err error) {
 | 
						|
	defer func() {
 | 
						|
		if err != nil {
 | 
						|
			w.err = err
 | 
						|
		}
 | 
						|
	}()
 | 
						|
 | 
						|
	err = w.closeCurrentFile()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	return safefile.LinkRelative(target, w.root, name, w.root)
 | 
						|
}
 | 
						|
 | 
						|
func (w *baseLayerWriter) Remove(name string) error {
 | 
						|
	return errors.New("base layer cannot have tombstones")
 | 
						|
}
 | 
						|
 | 
						|
func (w *baseLayerWriter) Write(b []byte) (int, error) {
 | 
						|
	n, err := w.bw.Write(b)
 | 
						|
	if err != nil {
 | 
						|
		w.err = err
 | 
						|
	}
 | 
						|
	return n, err
 | 
						|
}
 | 
						|
 | 
						|
func (w *baseLayerWriter) Close() (err error) {
 | 
						|
	defer w.s.End()
 | 
						|
	defer func() { oc.SetSpanStatus(w.s, err) }()
 | 
						|
	defer func() {
 | 
						|
		w.root.Close()
 | 
						|
		w.root = nil
 | 
						|
	}()
 | 
						|
 | 
						|
	err = w.closeCurrentFile()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if w.err == nil {
 | 
						|
		// Restore the file times of all the directories, since they may have
 | 
						|
		// been modified by creating child directories.
 | 
						|
		err = reapplyDirectoryTimes(w.root, w.dirInfo)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		err = ProcessBaseLayer(w.ctx, w.root.Name())
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if w.hasUtilityVM {
 | 
						|
			err := safefile.EnsureNotReparsePointRelative("UtilityVM", w.root)
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
			err = ProcessUtilityVMImage(w.ctx, filepath.Join(w.root.Name(), "UtilityVM"))
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return w.err
 | 
						|
}
 |