-
Notifications
You must be signed in to change notification settings - Fork 89
/
keys.go
156 lines (136 loc) · 3.66 KB
/
keys.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
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
package sbctl
import (
"bytes"
"crypto"
"errors"
"fmt"
"os"
"path/filepath"
"github.com/foxboron/go-uefi/authenticode"
"github.com/foxboron/go-uefi/efi"
"github.com/foxboron/sbctl/backend"
"github.com/foxboron/sbctl/certs"
"github.com/foxboron/sbctl/config"
"github.com/foxboron/sbctl/fs"
"github.com/foxboron/sbctl/hierarchy"
"github.com/spf13/afero"
)
func EnrollCustom(customBytes []byte, efivar string) error {
return efi.WriteEFIVariable(efivar, customBytes)
}
func VerifyFile(state *config.State, kh *backend.KeyHierarchy, ev hierarchy.Hierarchy, file string) (bool, error) {
peFile, err := state.Fs.Open(file)
if err != nil {
return false, err
}
defer peFile.Close()
return kh.VerifyFile(ev, peFile)
}
var ErrAlreadySigned = errors.New("already signed file")
func SignFile(state *config.State, kh *backend.KeyHierarchy, ev hierarchy.Hierarchy, file, output string) error {
// Check to see if input and output binary is the same
var same bool
// Make sure that output is always populated by atleast the file path
if output == "" {
output = file
}
// Check file exists before we do anything
if _, err := state.Fs.Stat(file); errors.Is(err, os.ErrNotExist) {
return fmt.Errorf("%s does not exist", file)
}
// We want to write the file back with correct permissions
si, err := state.Fs.Stat(file)
if err != nil {
return fmt.Errorf("failed stat of file: %w", err)
}
peFile, err := state.Fs.Open(file)
if err != nil {
return err
}
defer peFile.Close()
inputBinary, err := authenticode.Parse(peFile)
if err != nil {
return err
}
// Check if the files are identical
if file != output {
if outputFile, err := state.Fs.Open(output); err == nil {
defer outputFile.Close()
outputBinary, err := authenticode.Parse(outputFile)
if err != nil {
return err
}
b := outputBinary.Hash(crypto.SHA256)
bb := inputBinary.Hash(crypto.SHA256)
if bytes.Equal(b, bb) {
same = true
}
}
}
if file == output {
same = true
}
// Let's check if we have signed it already AND the original file hasn't changed
// TODO: This will run authenticode.Parse again, *and* open the file
// this should be refactored to be nicer
ok, err := VerifyFile(state, kh, ev, output)
if errors.Is(err, authenticode.ErrNoValidSignatures) {
// If we tried to verify the file, but it has signatures but nothing signed
// by our key, we catch the error and continue.
} else if errors.Is(err, os.ErrNotExist) {
// Ignore the error if the file doesn't exist
} else if ok && same {
// If already signed, and the input/output binaries are identical,
// we can just assume everything is fine.
return ErrAlreadySigned
} else if err != nil {
return err
}
b, err := kh.SignFile(ev, inputBinary)
if err != nil {
return err
}
if err = fs.WriteFile(state.Fs, output, b, si.Mode()); err != nil {
return err
}
return nil
}
// Map up our default keys in a struct
var SecureBootKeys = []struct {
Key string
Description string
}{
{
Key: "PK",
Description: "Platform Key",
},
{
Key: "KEK",
Description: "Key Exchange Key",
},
{
Key: "db",
Description: "Database Key",
},
// {
// Key: "dbx",
// Description: "Forbidden Database Key",
// },
}
// Check if we have already intialized keys in the given output directory
func CheckIfKeysInitialized(vfs afero.Fs, output string) bool {
for _, key := range SecureBootKeys {
path := filepath.Join(output, key.Key)
if _, err := vfs.Stat(path); errors.Is(err, os.ErrNotExist) {
return false
}
}
return true
}
func GetEnrolledVendorCerts() []string {
db, err := efi.Getdb()
if err != nil {
return []string{}
}
return certs.DetectVendorCerts(db)
}