Skip to content

Commit b062fe4

Browse files
authored
Merge pull request #85 from deeglaze/stepping
Fix "modelstepping" handling in verification
2 parents 307fceb + 93c77e9 commit b062fe4

File tree

8 files changed

+141
-117
lines changed

8 files changed

+141
-117
lines changed

abi/abi.go

+6-5
Original file line numberDiff line numberDiff line change
@@ -818,11 +818,12 @@ func SevProduct() *pb.SevProduct {
818818
// 15:14 reserved
819819
// 11:8 Family ID
820820
family := (eax >> 8) & 0xf
821-
// 7:4 Model, 3:0 Stepping
822-
modelStepping := eax & 0xff
821+
// 3:0 Stepping
822+
stepping := eax & 0xf
823823
// Ah, Fh, {0h,1h} values from the KDS specification,
824824
// section "Determining the Product Name".
825825
var productName pb.SevProduct_SevProductName
826+
// Product information specified by processor programming reference publications.
826827
if extendedFamily == 0xA && family == 0xF {
827828
switch extendedModel {
828829
case 0:
@@ -834,12 +835,12 @@ func SevProduct() *pb.SevProduct {
834835
}
835836
}
836837
return &pb.SevProduct{
837-
Name: productName,
838-
ModelStepping: modelStepping,
838+
Name: productName,
839+
Stepping: stepping,
839840
}
840841
}
841842

842843
// DefaultSevProduct returns the initial product version for a commercially available AMD SEV-SNP chip.
843844
func DefaultSevProduct() *pb.SevProduct {
844-
return &pb.SevProduct{Name: pb.SevProduct_SEV_PRODUCT_MILAN, ModelStepping: 0xB0}
845+
return &pb.SevProduct{Name: pb.SevProduct_SEV_PRODUCT_MILAN, Stepping: 1}
845846
}

kds/kds.go

+50-26
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,20 @@ var (
8484
kdsBaseURL = "https://" + kdsHostname
8585
kdsVcekPath = "/vcek/v1/"
8686
kdsVlekPath = "/vlek/v1/"
87+
88+
// Chip manufacturers assign stepping versions strings that are <letter><number>
89+
// to describe a stepping number for a particular model chip. There is no way
90+
// other than documentation to map a stepping number to a stepping version and
91+
// vice versa.
92+
steppingDecoder = map[string]*pb.SevProduct{
93+
"Milan-B0": {Name: pb.SevProduct_SEV_PRODUCT_MILAN, Stepping: 0},
94+
"Milan-B1": {Name: pb.SevProduct_SEV_PRODUCT_MILAN, Stepping: 1},
95+
"Genoa-B0": {Name: pb.SevProduct_SEV_PRODUCT_GENOA, Stepping: 0},
96+
"Genoa-B1": {Name: pb.SevProduct_SEV_PRODUCT_GENOA, Stepping: 1},
97+
"Genoa-B2": {Name: pb.SevProduct_SEV_PRODUCT_GENOA, Stepping: 2},
98+
}
99+
milanSteppingVersions = []string{"B0", "B1"}
100+
genoaSteppingVersions = []string{"B0", "B1", "B2"}
87101
)
88102

89103
// TCBVersion is a 64-bit bitfield of different security patch levels of AMD firmware and microcode.
@@ -692,45 +706,55 @@ func ProductName(product *pb.SevProduct) string {
692706
if product == nil {
693707
product = abi.DefaultSevProduct()
694708
}
695-
return fmt.Sprintf("%s-%02X", ProductString(product), product.ModelStepping)
709+
if product.Stepping > 15 {
710+
return "badstepping"
711+
}
712+
switch product.Name {
713+
case pb.SevProduct_SEV_PRODUCT_MILAN:
714+
if int(product.Stepping) >= len(milanSteppingVersions) {
715+
return "unmappedMilanStepping"
716+
}
717+
return fmt.Sprintf("Milan-%s", milanSteppingVersions[product.Stepping])
718+
case pb.SevProduct_SEV_PRODUCT_GENOA:
719+
if int(product.Stepping) >= len(genoaSteppingVersions) {
720+
return "unmappedGenoaStepping"
721+
}
722+
return fmt.Sprintf("Milan-%s", genoaSteppingVersions[product.Stepping])
723+
default:
724+
return "Unknown"
725+
}
726+
}
727+
728+
func parseProduct(product string) (pb.SevProduct_SevProductName, error) {
729+
switch product {
730+
case "Milan":
731+
return pb.SevProduct_SEV_PRODUCT_MILAN, nil
732+
case "Genoa":
733+
return pb.SevProduct_SEV_PRODUCT_GENOA, nil
734+
default:
735+
return pb.SevProduct_SEV_PRODUCT_UNKNOWN, fmt.Errorf("unknown AMD SEV product: %q", product)
736+
}
696737
}
697738

698739
// ParseProductName returns the KDS project input value, and the model, stepping numbers represented
699740
// by a given V[CL]EK productName extension value, or an error.
700741
func ParseProductName(productName string, key abi.ReportSigner) (*pb.SevProduct, error) {
701-
var product, stepping string
702-
var needStepping bool
703742
switch key {
704743
case abi.VcekReportSigner:
705-
subs := strings.SplitN(productName, "-", 2)
706-
if len(subs) != 2 {
707-
return nil, fmt.Errorf("productName value %q does not match the VCEK expected Name-ModelStepping format", productName)
744+
product, ok := steppingDecoder[productName]
745+
if !ok {
746+
return nil, fmt.Errorf("unknown product name (new stepping published?): %q", productName)
708747
}
709-
product = subs[0]
710-
stepping = subs[1]
711-
needStepping = true
748+
return product, nil
712749
case abi.VlekReportSigner:
713750
// VLEK certificates don't carry the stepping value in productName.
714-
product = productName
715-
}
716-
var name pb.SevProduct_SevProductName
717-
switch product {
718-
case "Milan":
719-
name = pb.SevProduct_SEV_PRODUCT_MILAN
720-
case "Genoa":
721-
name = pb.SevProduct_SEV_PRODUCT_GENOA
722-
default:
723-
return nil, fmt.Errorf("unknown AMD SEV product: %q", product)
724-
}
725-
var modelStepping uint64
726-
if needStepping {
727-
var err error
728-
modelStepping, err = strconv.ParseUint(stepping, 16, 8)
751+
name, err := parseProduct(productName)
729752
if err != nil {
730-
return nil, fmt.Errorf("model stepping in productName is not a hexadecimal byte: %q", stepping)
753+
return nil, err
731754
}
755+
return &pb.SevProduct{Name: name}, nil
732756
}
733-
return &pb.SevProduct{Name: name, ModelStepping: uint32(modelStepping)}, nil
757+
return nil, fmt.Errorf("internal: unhandled reportSigner %v", key)
734758
}
735759

736760
// CrlLinkByKey returns the CRL distribution point for the given key type's

kds/kds_test.go

+19-29
Original file line numberDiff line numberDiff line change
@@ -185,29 +185,29 @@ func TestProductName(t *testing.T) {
185185
}{
186186
{
187187
name: "nil",
188-
want: "Milan-B0",
188+
want: "Milan-B1",
189189
},
190190
{
191191
name: "unknown",
192192
input: &pb.SevProduct{
193-
ModelStepping: 0x1A,
193+
Stepping: 0x1A,
194194
},
195-
want: "Unknown-1A",
195+
want: "badstepping",
196196
},
197197
{
198-
name: "Milan-00",
198+
name: "Milan-B0",
199199
input: &pb.SevProduct{
200200
Name: pb.SevProduct_SEV_PRODUCT_MILAN,
201201
},
202-
want: "Milan-00",
202+
want: "Milan-B0",
203203
},
204204
{
205205
name: "Genoa-FF",
206206
input: &pb.SevProduct{
207-
Name: pb.SevProduct_SEV_PRODUCT_GENOA,
208-
ModelStepping: 0xFF,
207+
Name: pb.SevProduct_SEV_PRODUCT_GENOA,
208+
Stepping: 0xFF,
209209
},
210-
want: "Genoa-FF",
210+
want: "badstepping",
211211
},
212212
}
213213
for _, tc := range tcs {
@@ -229,36 +229,26 @@ func TestParseProductName(t *testing.T) {
229229
}{
230230
{
231231
name: "empty",
232-
wantErr: "does not match",
233-
},
234-
{
235-
name: "Too much",
236-
input: "Milan-B0-and some extra",
237-
wantErr: "not a hexadecimal byte: \"B0-and some extra\"",
238-
},
239-
{
240-
name: "start-",
241-
input: "-00",
242-
wantErr: "unknown AMD SEV product: \"\"",
243-
},
244-
{
245-
name: "end-",
246-
input: "Milan-",
247-
wantErr: "model stepping in productName is not a hexadecimal byte: \"\"",
232+
wantErr: "unknown product name",
248233
},
249234
{
250235
name: "Too big",
251236
input: "Milan-100",
252-
wantErr: "model stepping in productName is not a hexadecimal byte: \"100\"",
237+
wantErr: "unknown product name",
253238
},
254239
{
255-
name: "happy path",
256-
input: "Genoa-9C",
240+
name: "happy path Genoa",
241+
input: "Genoa-B1",
257242
want: &pb.SevProduct{
258-
Name: pb.SevProduct_SEV_PRODUCT_GENOA,
259-
ModelStepping: 0x9C,
243+
Name: pb.SevProduct_SEV_PRODUCT_GENOA,
244+
Stepping: 1,
260245
},
261246
},
247+
{
248+
name: "bad revision Milan",
249+
input: "Milan-A1",
250+
wantErr: "unknown product name",
251+
},
262252
{
263253
name: "vlek products have no stepping",
264254
input: "Genoa",

proto/sevsnp.proto

+1-1
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,7 @@ message SevProduct {
9090
}
9191

9292
SevProductName name = 1;
93-
uint32 model_stepping = 2; // Must be a byte
93+
uint32 stepping = 2; // Must be a 4-bit number
9494
}
9595

9696
message Attestation {

proto/sevsnp/sevsnp.pb.go

+28-29
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

testing/mocks.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -144,8 +144,8 @@ func (d *Device) Ioctl(command uintptr, req any) (uintptr, error) {
144144
func (d *Device) Product() *spb.SevProduct {
145145
if d.SevProduct == nil {
146146
return &spb.SevProduct{
147-
Name: spb.SevProduct_SEV_PRODUCT_MILAN,
148-
ModelStepping: 0xB0,
147+
Name: spb.SevProduct_SEV_PRODUCT_MILAN,
148+
Stepping: 0, // Milan-B0 is the product name we fake.
149149
}
150150
}
151151
return d.SevProduct

verify/verify.go

+24-10
Original file line numberDiff line numberDiff line change
@@ -451,9 +451,9 @@ func checkProductName(got, want *spb.SevProduct, key abi.ReportSigner) error {
451451
return fmt.Errorf("%v cert product name %v is not %v", key, got, want)
452452
}
453453
// The model stepping number is only part of the VLEK product name, not VLEK's.
454-
if key == abi.VcekReportSigner && got.ModelStepping != want.ModelStepping {
455-
return fmt.Errorf("%v cert product model-stepping number %02X is not %02X",
456-
key, got.ModelStepping, want.ModelStepping)
454+
if key == abi.VcekReportSigner && got.Stepping != want.Stepping {
455+
return fmt.Errorf("%v cert product stepping number %02X is not %02X",
456+
key, got.Stepping, want.Stepping)
457457
}
458458
return nil
459459
}
@@ -483,6 +483,7 @@ func decodeCerts(chain *spb.CertificateChain, key abi.ReportSigner, options *Opt
483483
if err != nil {
484484
return nil, nil, err
485485
}
486+
fmt.Println("product", exts.ProductName, product)
486487
productName := kds.ProductString(product)
487488
// Ensure the extension product info matches expectations.
488489
if err := checkProductName(product, options.Product, key); err != nil {
@@ -650,15 +651,13 @@ func SnpAttestation(attestation *spb.Attestation, options *Options) error {
650651
// fillInAttestation uses AMD's KDS to populate any empty certificate field in the attestation's
651652
// certificate chain.
652653
func fillInAttestation(attestation *spb.Attestation, options *Options) error {
654+
var productOverridden bool
653655
if options.Product != nil {
654656
attestation.Product = options.Product
655-
}
656-
if attestation.Product == nil {
657-
// The default product is the first launched SEV-SNP product value.
658-
attestation.Product = &spb.SevProduct{
659-
Name: spb.SevProduct_SEV_PRODUCT_MILAN,
660-
ModelStepping: 0xB0,
661-
}
657+
productOverridden = true
658+
} else if attestation.Product == nil {
659+
attestation.Product = abi.DefaultSevProduct()
660+
productOverridden = true
662661
}
663662
if options.DisableCertFetching {
664663
return nil
@@ -702,6 +701,21 @@ func fillInAttestation(attestation *spb.Attestation, options *Options) error {
702701
}
703702
}
704703
chain.VcekCert = vcek
704+
if productOverridden {
705+
cert, err := x509.ParseCertificate(vcek)
706+
if err != nil {
707+
return err
708+
}
709+
exts, err := kds.VcekCertificateExtensions(cert)
710+
if err != nil {
711+
return err
712+
}
713+
attestation.Product, err = kds.ParseProductName(exts.ProductName, abi.VcekReportSigner)
714+
fmt.Printf("filled in product with %v\n", attestation.Product)
715+
if err != nil {
716+
return err
717+
}
718+
}
705719
}
706720
case abi.VlekReportSigner:
707721
// We can't lazily ask KDS for the certificate as a user. The CSP must cache their provisioned

0 commit comments

Comments
 (0)