github.com/u-root/u-root@v7.0.1-0.20200915234505-ad7babab0a8e+incompatible/pkg/cpio/fs_plan9.go (about) 1 // Copyright 2013-2017 the u-root Authors. All rights reserved 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 package cpio 6 7 import ( 8 "fmt" 9 "io" 10 "os" 11 "path/filepath" 12 "time" 13 14 "github.com/u-root/u-root/pkg/ls" 15 "github.com/u-root/u-root/pkg/uio" 16 ) 17 18 // A Recorder is a structure that contains variables used to calculate 19 // file parameters such as inode numbers for a CPIO file. The life-time 20 // of a Record structure is meant to be the same as the construction of a 21 // single CPIO archive. Do not reuse between CPIOs if you don't know what 22 // you're doing. 23 type Recorder struct { 24 inumber uint64 25 } 26 27 var modeMap = map[uint64]os.FileMode{ 28 modeFile: 0, 29 modeDir: os.ModeDir, 30 } 31 32 func unixModeToFileType(m uint64) (os.FileMode, error) { 33 if t, ok := modeMap[m&modeTypeMask]; ok { 34 return t, nil 35 } 36 return 0, fmt.Errorf("invalid file type %#o", m&modeTypeMask) 37 } 38 39 func toFileMode(r Record) os.FileMode { 40 return os.FileMode(perm(r)) 41 } 42 43 // setModes sets the modes. 44 func setModes(r Record) error { 45 if err := os.Chmod(r.Name, toFileMode(r)&os.ModePerm); err != nil { 46 return err 47 } 48 return nil 49 } 50 51 func perm(r Record) uint32 { 52 return uint32(r.Mode) & modePermissions 53 } 54 55 func dev(r Record) int { 56 return int(r.Rmajor<<8 | r.Rminor) 57 } 58 59 // CreateFile creates a local file for f relative to the current working 60 // directory. 61 // 62 // CreateFile will attempt to set all metadata for the file, including 63 // ownership, times, and permissions. 64 func CreateFile(f Record) error { 65 return CreateFileInRoot(f, ".", true) 66 } 67 68 // CreateFileInRoot creates a local file for f relative to rootDir. 69 func CreateFileInRoot(f Record, rootDir string, forcePriv bool) error { 70 m, err := unixModeToFileType(f.Mode) 71 if err != nil { 72 return err 73 } 74 75 f.Name = filepath.Clean(filepath.Join(rootDir, f.Name)) 76 dir := filepath.Dir(f.Name) 77 // The problem: many cpio archives do not specify the directories and 78 // hence the permissions. They just specify the whole path. In order 79 // to create files in these directories, we have to make them at least 80 // mode 755. 81 if _, err := os.Stat(dir); os.IsNotExist(err) && len(dir) > 0 { 82 if err := os.MkdirAll(dir, 0755); err != nil { 83 return fmt.Errorf("CreateFileInRoot %q: %v", f.Name, err) 84 } 85 } 86 87 switch m { 88 case os.FileMode(0): 89 nf, err := os.Create(f.Name) 90 if err != nil { 91 return err 92 } 93 defer nf.Close() 94 if _, err := io.Copy(nf, uio.Reader(f)); err != nil { 95 return err 96 } 97 98 case os.ModeDir: 99 if err := os.MkdirAll(f.Name, toFileMode(f)); err != nil { 100 return err 101 } 102 103 default: 104 return fmt.Errorf("%v: Unknown type %#o", f.Name, m) 105 } 106 107 if err := setModes(f); err != nil && forcePriv { 108 return err 109 } 110 return nil 111 } 112 113 func (r *Recorder) inode(i Info) Info { 114 i.Ino = r.inumber 115 r.inumber++ 116 return i 117 } 118 119 // GetRecord returns a cpio Record for the given path on the local file system. 120 // 121 // GetRecord does not follow symlinks. If path is a symlink, the record 122 // returned will reflect that symlink. 123 func (r *Recorder) GetRecord(path string) (Record, error) { 124 fi, err := os.Stat(path) 125 if err != nil { 126 return Record{}, err 127 } 128 info := r.inode(sysInfo(path, fi)) 129 switch fi.Mode() & os.ModeType { 130 case 0: // Regular file. 131 return Record{Info: info, ReaderAt: uio.NewLazyFile(path)}, nil 132 default: 133 return StaticRecord(nil, info), nil 134 } 135 } 136 137 // NewRecorder creates a new Recorder. 138 // 139 // A recorder is a structure that contains variables used to calculate 140 // file parameters such as inode numbers for a CPIO file. The life-time 141 // of a Record structure is meant to be the same as the construction of a 142 // single CPIO archive. Do not reuse between CPIOs if you don't know what 143 // you're doing. 144 func NewRecorder() *Recorder { 145 return &Recorder{} 146 } 147 148 // LSInfoFromRecord converts a Record to be usable with the ls package for 149 // listing files. 150 func LSInfoFromRecord(rec Record) ls.FileInfo { 151 mode := modeFromLinux(rec.Mode) 152 return ls.FileInfo{ 153 Name: rec.Name, 154 Mode: mode, 155 UID: fmt.Sprint("%d", rec.UID), 156 Size: int64(rec.FileSize), 157 MTime: time.Unix(int64(rec.MTime), 0).UTC(), 158 } 159 }