mirror of
https://github.com/google/go-attestation.git
synced 2025-01-02 11:16:45 +00:00
235 lines
5.9 KiB
Go
235 lines
5.9 KiB
Go
// Copyright 2019 Google 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.
|
|
|
|
// +build !localtest !tpm12
|
|
|
|
package attest
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto"
|
|
"crypto/rsa"
|
|
"testing"
|
|
|
|
"github.com/google/certificate-transparency-go/x509"
|
|
|
|
"github.com/google/go-tpm-tools/simulator"
|
|
"github.com/google/go-tpm/tpm2"
|
|
"github.com/google/go-tpm/tpm2/credactivation"
|
|
)
|
|
|
|
func setupSimulatedTPM(t *testing.T) (*simulator.Simulator, *TPM) {
|
|
t.Helper()
|
|
tpm, err := simulator.Get()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return tpm, &TPM{
|
|
version: TPMVersion20,
|
|
interf: TPMInterfaceKernelManaged,
|
|
sysPath: "/dev/tpmrm0",
|
|
rwc: tpm,
|
|
}
|
|
}
|
|
|
|
func TestSimTPM20EK(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
eks, err := tpm.EKs()
|
|
if err != nil {
|
|
t.Errorf("EKs() failed: %v", err)
|
|
}
|
|
if len(eks) == 0 || (eks[0].Cert == nil && eks[0].Public == nil) {
|
|
t.Errorf("EKs() = %v, want at least 1 EK with populated fields", eks)
|
|
}
|
|
}
|
|
|
|
func TestSimTPM20Info(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
info, err := tpm.Info()
|
|
if err != nil {
|
|
t.Errorf("tpm.Info() failed: %v", err)
|
|
}
|
|
// We dont expect anything to be meaningfully populated as this is a simulator.
|
|
t.Logf("TPM Info = %+v", info)
|
|
}
|
|
|
|
func TestSimTPM20AIKCreateAndLoad(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
aik, err := tpm.MintAIK(nil)
|
|
if err != nil {
|
|
t.Fatalf("MintAIK() failed: %v", err)
|
|
}
|
|
|
|
enc, err := aik.Marshal()
|
|
if err != nil {
|
|
aik.Close(tpm)
|
|
t.Fatalf("aik.Marshal() failed: %v", err)
|
|
}
|
|
if err := aik.Close(tpm); err != nil {
|
|
t.Fatalf("aik.Close() failed: %v", err)
|
|
}
|
|
|
|
loaded, err := tpm.LoadKey(enc)
|
|
if err != nil {
|
|
t.Fatalf("LoadKey() failed: %v", err)
|
|
}
|
|
defer loaded.Close(tpm)
|
|
|
|
if !bytes.Equal(loaded.Public, aik.Public) {
|
|
t.Error("Original & loaded AIK public blobs did not match.")
|
|
t.Logf("Original = %v", aik.Public)
|
|
t.Logf("Loaded = %v", loaded.Public)
|
|
}
|
|
}
|
|
|
|
// chooseEKPub selects the EK public which will be activated against.
|
|
func chooseEKPub(t *testing.T, eks []PlatformEK) crypto.PublicKey {
|
|
t.Helper()
|
|
|
|
for _, ek := range eks {
|
|
if ek.Cert != nil && ek.Cert.PublicKeyAlgorithm == x509.RSA {
|
|
return ek.Cert.PublicKey.(*rsa.PublicKey)
|
|
} else if ek.Public != nil {
|
|
return ek.Public
|
|
}
|
|
}
|
|
|
|
t.Skip("No suitable RSA EK found")
|
|
return nil
|
|
}
|
|
|
|
func TestSimTPM20ActivateCredential(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
aik, err := tpm.MintAIK(nil)
|
|
if err != nil {
|
|
t.Fatalf("MintAIK() failed: %v", err)
|
|
}
|
|
defer aik.Close(tpm)
|
|
|
|
EKs, err := tpm.EKs()
|
|
if err != nil {
|
|
t.Fatalf("EKs() failed: %v", err)
|
|
}
|
|
ek := chooseEKPub(t, EKs)
|
|
|
|
att, err := tpm2.DecodeAttestationData(aik.CreateAttestation)
|
|
if err != nil {
|
|
t.Fatalf("tpm2.DecodeAttestationData() failed: %v", err)
|
|
}
|
|
secret := []byte{1, 2, 3, 4, 5, 6, 7, 8, 1, 2, 3, 4, 5, 6, 7, 8}
|
|
|
|
id, encSecret, err := credactivation.Generate(att.AttestedCreationInfo.Name.Digest, ek, 16, secret)
|
|
if err != nil {
|
|
t.Fatalf("credactivation.Generate() failed: %v", err)
|
|
}
|
|
|
|
decryptedSecret, err := aik.ActivateCredential(tpm, EncryptedCredential{
|
|
Credential: id,
|
|
Secret: encSecret,
|
|
})
|
|
if err != nil {
|
|
t.Errorf("aik.ActivateCredential() failed: %v", err)
|
|
}
|
|
if !bytes.Equal(secret, decryptedSecret) {
|
|
t.Error("secret does not match decrypted secret")
|
|
t.Logf("Secret = %v", secret)
|
|
t.Logf("Decrypted secret = %v", decryptedSecret)
|
|
}
|
|
}
|
|
|
|
func TestSimTPM20Quote(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
aik, err := tpm.MintAIK(nil)
|
|
if err != nil {
|
|
t.Fatalf("MintAIK() failed: %v", err)
|
|
}
|
|
defer aik.Close(tpm)
|
|
|
|
nonce := []byte{1, 2, 3, 4, 5, 6, 7, 8}
|
|
quote, err := aik.Quote(tpm, nonce, tpm2.AlgSHA256)
|
|
if err != nil {
|
|
t.Fatalf("aik.Quote() failed: %v", err)
|
|
}
|
|
// TODO(jsonp): Parse quote structure once gotpm/tpm2 supports it.
|
|
if quote == nil {
|
|
t.Error("quote was nil, want *Quote")
|
|
}
|
|
}
|
|
|
|
func TestSimTPM20PCRs(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
PCRs, alg, err := tpm.PCRs()
|
|
if err != nil {
|
|
t.Fatalf("PCRs() failed: %v", err)
|
|
}
|
|
if len(PCRs) != 24 {
|
|
t.Errorf("len(PCRs) = %d, want %d", len(PCRs), 24)
|
|
}
|
|
if got, want := tpm2.AlgSHA256, alg; got != want {
|
|
t.Errorf("alg = %v, want %v", got, want)
|
|
}
|
|
for i, pcr := range PCRs {
|
|
if len(pcr.Digest) != pcr.DigestAlg.Size() {
|
|
t.Errorf("PCR %d len(digest) = %d, expected match with digest algorithm size (%d)", pcr.Index, len(pcr.Digest), pcr.DigestAlg.Size())
|
|
}
|
|
if pcr.Index != i {
|
|
t.Errorf("PCR index %d does not match map index %d", pcr.Index, i)
|
|
}
|
|
if pcr.DigestAlg != crypto.SHA256 {
|
|
t.Errorf("pcr.DigestAlg = %v, expected crypto.SHA256", pcr.DigestAlg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestSimTPM20Persistence(t *testing.T) {
|
|
sim, tpm := setupSimulatedTPM(t)
|
|
defer sim.Close()
|
|
|
|
ekHnd, p, err := tpm.getPrimaryKeyHandle(commonEkEquivalentHandle)
|
|
if err != nil {
|
|
t.Fatalf("getPrimaryKeyHandle() failed: %v", err)
|
|
}
|
|
if ekHnd != commonEkEquivalentHandle {
|
|
t.Fatalf("bad EK-equivalent handle: got 0x%x, wanted 0x%x", ekHnd, commonEkEquivalentHandle)
|
|
}
|
|
if p {
|
|
t.Logf("generated a new key")
|
|
} else {
|
|
t.Logf("used existing key")
|
|
}
|
|
|
|
ekHnd, p, err = tpm.getPrimaryKeyHandle(commonEkEquivalentHandle)
|
|
if err != nil {
|
|
t.Fatalf("second getPrimaryKeyHandle() failed: %v", err)
|
|
}
|
|
if ekHnd != commonEkEquivalentHandle {
|
|
t.Fatalf("bad EK-equivalent handle: got 0x%x, wanted 0x%x", ekHnd, commonEkEquivalentHandle)
|
|
}
|
|
if p {
|
|
t.Fatalf("generated a new key the second time; that shouldn't happen")
|
|
}
|
|
}
|