File: header.go

package info (click to toggle)
golang-github-sassoftware-go-rpmutils 0.2.0-1.1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 360 kB
  • sloc: makefile: 2
file content (200 lines) | stat: -rw-r--r-- 5,500 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
/*
 * Copyright (c) SAS Institute, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package cpio

import (
	"fmt"
	"io"
	"strconv"
)

// reference http://people.freebsd.org/~kientzle/libarchive/man/cpio.5.txt

const (
	newcHeaderLength = 110
	newcMagic        = "070701"
	strippedMagic    = "07070X"
)

// Cpio_newc_header is the raw header of a newc-style cpio archive
type Cpio_newc_header struct {
	magic     string
	ino       int
	mode      int
	uid       int
	gid       int
	nlink     int
	mtime     int
	filesize  int
	devmajor  int
	devminor  int
	rdevmajor int
	rdevminor int
	namesize  int
	check     int

	stripped bool
	filename string
	index    int
	size64   int64
}

type binaryReader struct {
	r   io.Reader
	buf [8]byte
}

func (br *binaryReader) Read16(buf *int) error {
	bb := br.buf[:8]
	if _, err := io.ReadFull(br.r, bb); err != nil {
		return err
	}
	i, err := strconv.ParseInt(string(bb), 16, 0)
	if err != nil {
		return err
	}
	*buf = int(i)
	return nil
}

func readHeader(r io.Reader) (*Cpio_newc_header, error) {
	hdr := Cpio_newc_header{}
	br := binaryReader{r: r}

	magic := make([]byte, 6)
	if _, err := io.ReadFull(r, magic); err != nil {
		return nil, err
	}
	if string(magic) == strippedMagic {
		return readStrippedHeader(br)
	} else if string(magic) != newcMagic {
		return nil, fmt.Errorf("bad magic")
	}
	hdr.magic = newcMagic

	if err := br.Read16(&hdr.ino); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.mode); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.uid); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.gid); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.nlink); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.mtime); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.filesize); err != nil {
		return nil, err
	}
	hdr.size64 = int64(hdr.filesize)
	if err := br.Read16(&hdr.devmajor); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.devminor); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.rdevmajor); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.rdevminor); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.namesize); err != nil {
		return nil, err
	}
	if err := br.Read16(&hdr.check); err != nil {
		return nil, err
	}
	return &hdr, nil
}

func readStrippedHeader(br binaryReader) (*Cpio_newc_header, error) {
	hdr := &Cpio_newc_header{
		magic:    strippedMagic,
		stripped: true,
	}
	if err := br.Read16(&hdr.index); err != nil {
		return nil, err
	}
	return hdr, nil
}

// Magic returns the magic number preceding the file entry
func (hdr *Cpio_newc_header) Magic() string { return hdr.magic }

// Ino returns the inode number of the file
func (hdr *Cpio_newc_header) Ino() int { return hdr.ino }

// Mode returns the file's permissions and file type
func (hdr *Cpio_newc_header) Mode() int { return hdr.mode }

// Uid returns the file's owner user ID
func (hdr *Cpio_newc_header) Uid() int { return hdr.uid }

// Gid returns the file's owner group ID
func (hdr *Cpio_newc_header) Gid() int { return hdr.gid }

// Nlink returns the number of hardlinks to the file
func (hdr *Cpio_newc_header) Nlink() int { return hdr.nlink }

// Mtime returns the file's modification time in seconds since the UNIX epoch
func (hdr *Cpio_newc_header) Mtime() int { return hdr.mtime }

// Filesize returns the size of the file in bytes
func (hdr *Cpio_newc_header) Filesize() int { return hdr.filesize }

// Devmajor returns the major device number of a character or block device
func (hdr *Cpio_newc_header) Devmajor() int { return hdr.devmajor }

// Devminor returns the minor device number of a character or block device
func (hdr *Cpio_newc_header) Devminor() int { return hdr.devminor }

// Rdevmajor returns the major device number of a character or block device
func (hdr *Cpio_newc_header) Rdevmajor() int { return hdr.rdevmajor }

// Rdevminor returns the minor device number of a character or block device
func (hdr *Cpio_newc_header) Rdevminor() int { return hdr.rdevminor }

// Namesize returns the length of the filename
func (hdr *Cpio_newc_header) Namesize() int { return hdr.namesize }

// Check returns the checksum of the entry, if present
func (hdr *Cpio_newc_header) Check() int { return hdr.check }

// Filename returns the name of the file entry
func (hdr *Cpio_newc_header) Filename() string { return hdr.filename }

// stripped header functions

// IsStripped returns true if the file header is missing info that must come
// from the preceding RPM header
func (hdr *Cpio_newc_header) IsStripped() bool { return hdr.stripped }

// Index returns the position in the RPM header file info array corresponding to
// this file
func (hdr *Cpio_newc_header) Index() int { return hdr.index }

// Filesize64 contains the file's size as a 64-bit integer, coming from either
// SetFileSizes() if used or from the regular cpio file entry
func (hdr *Cpio_newc_header) Filesize64() int64 { return hdr.size64 }