-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathluks.go
75 lines (72 loc) · 2.73 KB
/
luks.go
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
package luksy
import (
"bytes"
"encoding/json"
"fmt"
"io"
)
// ReadHeaderOptions can control some of what ReadHeaders() does.
type ReadHeaderOptions struct{}
// ReadHeaders reads LUKS headers from the specified file, returning either a
// LUKSv1 header, or two LUKSv2 headers and a LUKSv2 JSON block, depending on
// which format is detected.
func ReadHeaders(f io.ReaderAt, options ReadHeaderOptions) (*V1Header, *V2Header, *V2Header, *V2JSON, error) {
var v1 V1Header
var v2a, v2b V2Header
n, err := f.ReadAt(v2a[:], 0)
if err != nil {
return nil, nil, nil, nil, err
}
if n != len(v2a) {
return nil, nil, nil, nil, fmt.Errorf("only able to read %d bytes - file truncated?", n)
}
if n, err = f.ReadAt(v1[:], 0); err != nil {
return nil, nil, nil, nil, err
}
if n != len(v1) {
return nil, nil, nil, nil, fmt.Errorf("only able to read %d bytes - file truncated?", n)
}
if v2a.Magic() != V2Magic1 {
return nil, nil, nil, nil, fmt.Errorf("internal error: magic mismatch in LUKS header (%q)", v2a.Magic())
}
switch v2a.Version() { // is it a v1 header, or the first v2 header?
case 1:
return &v1, nil, nil, nil, nil
case 2:
size := v2a.HeaderSize()
if size > 0x7fffffffffffffff {
return nil, nil, nil, nil, fmt.Errorf("unsupported header size while looking for second header")
}
if size < 4096 {
return nil, nil, nil, nil, fmt.Errorf("unsupported header size while looking for JSON data")
}
if n, err = f.ReadAt(v2b[:], int64(size)); err != nil || n != len(v2b) {
if err == nil && n != len(v2b) {
err = fmt.Errorf("short read: read only %d bytes, should have read %d", n, len(v2b))
}
return nil, nil, nil, nil, err
}
if v2b.Magic() != V2Magic2 {
return nil, nil, nil, nil, fmt.Errorf("internal error: magic mismatch in second LUKS header (%q)", v2b.Magic())
}
jsonSize := size - 4096
buf := make([]byte, jsonSize)
n, err = f.ReadAt(buf[:], 4096)
if err != nil {
return nil, nil, nil, nil, fmt.Errorf("internal error: while reading JSON data: %w", err)
}
if n < 0 || uint64(n) != jsonSize {
return nil, nil, nil, nil, fmt.Errorf("internal error: short read while reading JSON data (wanted %d, got %d)", jsonSize, n)
}
var jsonData V2JSON
buf = bytes.TrimRightFunc(buf, func(r rune) bool { return r == 0 })
if err = json.Unmarshal(buf, &jsonData); err != nil {
return nil, nil, nil, nil, fmt.Errorf("internal error: decoding JSON data: %w", err)
}
if uint64(jsonData.Config.JsonSize) != jsonSize {
return nil, nil, nil, nil, fmt.Errorf("internal error: JSON data size mismatch: (expected %d, used %d)", jsonData.Config.JsonSize, jsonSize)
}
return nil, &v2a, &v2b, &jsonData, nil
}
return nil, nil, nil, nil, fmt.Errorf("error reading LUKS header - magic identifier not found")
}