2022-08-31 20:10:49 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
package snp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
|
2022-09-21 13:47:57 +02:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/attestation/vtpm"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/oid"
|
2022-08-31 20:10:49 +02:00
|
|
|
tpmclient "github.com/google/go-tpm-tools/client"
|
|
|
|
"github.com/google/go-tpm/tpm2"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
lenHclHeader = 0x20
|
|
|
|
lenSnpReport = 0x4a0
|
|
|
|
lenSnpReportRuntimeDataPadding = 0x14
|
|
|
|
tpmReportIdx = 0x01400001
|
2022-10-04 16:44:44 +02:00
|
|
|
tpmAkIdx = 0x81000003
|
2022-08-31 20:10:49 +02:00
|
|
|
)
|
|
|
|
|
2022-10-05 15:02:46 +02:00
|
|
|
// GetIDKeyDigest reads the idkeydigest from the snp report saved in the TPM's non-volatile memory.
|
|
|
|
func GetIDKeyDigest(open vtpm.TPMOpenFunc) ([]byte, error) {
|
2022-08-31 20:10:49 +02:00
|
|
|
tpm, err := open()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer tpm.Close()
|
|
|
|
|
|
|
|
reportRaw, err := tpm2.NVReadEx(tpm, tpmReportIdx, tpm2.HandleOwner, "", 0)
|
|
|
|
if err != nil {
|
2022-10-04 16:44:44 +02:00
|
|
|
return nil, fmt.Errorf("reading idx %x from TPM: %w", tpmReportIdx, err)
|
2022-08-31 20:10:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
report, err := newSNPReportFromBytes(reportRaw[lenHclHeader:])
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("creating snp report: %w", err)
|
|
|
|
}
|
|
|
|
|
2022-09-07 10:39:38 +02:00
|
|
|
return report.IDKeyDigest[:], nil
|
2022-08-31 20:10:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Issuer for Azure TPM attestation.
|
|
|
|
type Issuer struct {
|
|
|
|
oid.AzureSNP
|
|
|
|
*vtpm.Issuer
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewIssuer initializes a new Azure Issuer.
|
|
|
|
func NewIssuer() *Issuer {
|
|
|
|
imdsAPI := imdsClient{
|
|
|
|
client: &http.Client{Transport: &http.Transport{Proxy: nil}},
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Issuer{
|
|
|
|
Issuer: vtpm.NewIssuer(
|
|
|
|
vtpm.OpenVTPM,
|
|
|
|
getAttestationKey,
|
|
|
|
getInstanceInfo(&tpmReport{}, imdsAPI),
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// getInstanceInfo loads and returns the SEV-SNP attestation report [1] and the
|
|
|
|
// AMD VCEK certificate chain.
|
|
|
|
// The attestation report is loaded from the TPM, the certificate chain is queried
|
|
|
|
// from the cloud metadata API.
|
|
|
|
// [1] https://github.com/AMDESE/sev-guest/blob/main/include/attestation.h
|
2022-10-05 15:02:46 +02:00
|
|
|
func getInstanceInfo(reportGetter tpmReportGetter, imdsapi imdsAPI) func(tpm io.ReadWriteCloser) ([]byte, error) {
|
2022-08-31 20:10:49 +02:00
|
|
|
return func(tpm io.ReadWriteCloser) ([]byte, error) {
|
|
|
|
hclReport, err := reportGetter.get(tpm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("reading report from TPM: %w", err)
|
|
|
|
}
|
|
|
|
if len(hclReport) < lenHclHeader+lenSnpReport+lenSnpReportRuntimeDataPadding {
|
|
|
|
return nil, fmt.Errorf("report read from TPM is shorter then expected: %x", hclReport)
|
|
|
|
}
|
|
|
|
hclReport = hclReport[lenHclHeader:]
|
|
|
|
|
|
|
|
runtimeData, _, _ := bytes.Cut(hclReport[lenSnpReport+lenSnpReportRuntimeDataPadding:], []byte{0})
|
|
|
|
|
2022-10-05 15:02:46 +02:00
|
|
|
vcekResponse, err := imdsapi.getVcek(context.TODO())
|
2022-08-31 20:10:49 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("getVcekFromIMDS: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
instanceInfo := azureInstanceInfo{
|
|
|
|
Vcek: []byte(vcekResponse.VcekCert),
|
|
|
|
CertChain: []byte(vcekResponse.CertificateChain),
|
|
|
|
AttestationReport: hclReport[:0x4a0],
|
|
|
|
RuntimeData: runtimeData,
|
|
|
|
}
|
|
|
|
statement, err := json.Marshal(instanceInfo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("marshalling AzureInstanceInfo: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return statement, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-27 11:04:23 +02:00
|
|
|
// getAttestationKey reads the attestation key put into the TPM during early boot.
|
2022-08-31 20:10:49 +02:00
|
|
|
func getAttestationKey(tpm io.ReadWriter) (*tpmclient.Key, error) {
|
2022-10-04 16:44:44 +02:00
|
|
|
ak, err := tpmclient.LoadCachedKey(tpm, tpmAkIdx)
|
2022-08-31 20:10:49 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("reading HCL attestation key from TPM: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ak, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type tpmReport struct{}
|
|
|
|
|
|
|
|
func (s *tpmReport) get(tpm io.ReadWriteCloser) ([]byte, error) {
|
|
|
|
return tpm2.NVReadEx(tpm, tpmReportIdx, tpm2.HandleOwner, "", 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
type tpmReportGetter interface {
|
|
|
|
get(tpm io.ReadWriteCloser) ([]byte, error)
|
|
|
|
}
|
|
|
|
|
2022-10-05 15:02:46 +02:00
|
|
|
type imdsAPI interface {
|
2022-08-31 20:10:49 +02:00
|
|
|
getVcek(ctx context.Context) (vcekResponse, error)
|
|
|
|
}
|