2019-03-28 20:21:16 +00:00
|
|
|
// 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.
|
2019-03-28 20:29:24 +00:00
|
|
|
|
2019-03-28 20:21:16 +00:00
|
|
|
package attest
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto"
|
|
|
|
"crypto/rsa"
|
2019-04-05 21:49:36 +00:00
|
|
|
"flag"
|
2019-03-28 20:21:16 +00:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/certificate-transparency-go/x509"
|
2019-04-05 21:49:36 +00:00
|
|
|
)
|
2019-03-28 20:21:16 +00:00
|
|
|
|
2019-04-05 21:49:36 +00:00
|
|
|
var (
|
|
|
|
testLocal = flag.Bool("testLocal", false, "run tests against local hardware")
|
2019-03-28 20:21:16 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestOpen(t *testing.T) {
|
2019-04-05 21:49:36 +00:00
|
|
|
if !*testLocal {
|
|
|
|
t.SkipNow()
|
|
|
|
}
|
2019-03-28 20:21:16 +00:00
|
|
|
tpm, err := OpenTPM(nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("OpenTPM() failed: %v", err)
|
|
|
|
}
|
|
|
|
if tpm == nil {
|
|
|
|
t.Fatalf("Expected non-nil tpm struct")
|
|
|
|
}
|
|
|
|
defer tpm.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInfo(t *testing.T) {
|
2019-04-05 21:49:36 +00:00
|
|
|
if !*testLocal {
|
|
|
|
t.SkipNow()
|
|
|
|
}
|
2019-03-28 20:21:16 +00:00
|
|
|
tpm, err := OpenTPM(nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("OpenTPM() failed: %v", err)
|
|
|
|
}
|
|
|
|
defer tpm.Close()
|
|
|
|
|
|
|
|
info, err := tpm.Info()
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("tpm.Info() failed: %v", err)
|
|
|
|
}
|
|
|
|
if info.Manufacturer.String() == "" {
|
|
|
|
t.Error("Expected info.Manufacturer.String() != ''")
|
|
|
|
}
|
|
|
|
t.Logf("TPM Info = %+v", info)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestEKs(t *testing.T) {
|
2019-04-05 21:49:36 +00:00
|
|
|
if !*testLocal {
|
|
|
|
t.SkipNow()
|
|
|
|
}
|
2019-03-28 20:21:16 +00:00
|
|
|
tpm, err := OpenTPM(nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("OpenTPM() failed: %v", err)
|
|
|
|
}
|
|
|
|
defer tpm.Close()
|
|
|
|
|
|
|
|
eks, err := tpm.EKs()
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("EKs() failed: %v", err)
|
|
|
|
}
|
|
|
|
if len(eks) == 0 {
|
|
|
|
t.Log("EKs() did not return anything. This could be an issue if an EK is present.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAIKCreateAndLoad(t *testing.T) {
|
2019-04-05 21:49:36 +00:00
|
|
|
if !*testLocal {
|
|
|
|
t.SkipNow()
|
|
|
|
}
|
2019-03-28 20:21:16 +00:00
|
|
|
tpm, err := OpenTPM(nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("OpenTPM() failed: %v", err)
|
|
|
|
}
|
|
|
|
defer tpm.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)
|
|
|
|
}
|
|
|
|
|
2019-07-19 20:05:18 +00:00
|
|
|
loaded, err := tpm.LoadAIK(enc)
|
2019-03-28 20:21:16 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("LoadKey() failed: %v", err)
|
|
|
|
}
|
|
|
|
defer loaded.Close(tpm)
|
|
|
|
|
2019-07-19 20:05:18 +00:00
|
|
|
k1, k2 := aik.aik.(*key20), loaded.aik.(*key20)
|
|
|
|
|
|
|
|
if !bytes.Equal(k1.public, k2.public) {
|
2019-03-28 20:21:16 +00:00
|
|
|
t.Error("Original & loaded AIK public blobs did not match.")
|
2019-07-19 20:05:18 +00:00
|
|
|
t.Logf("Original = %v", k1.public)
|
|
|
|
t.Logf("Loaded = %v", k2.public)
|
2019-03-28 20:21:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// chooseEK selects the EK public which will be activated against.
|
|
|
|
func chooseEK(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
|
|
|
|
}
|
|
|
|
|
2019-04-05 21:49:36 +00:00
|
|
|
func TestPCRs(t *testing.T) {
|
|
|
|
if !*testLocal {
|
|
|
|
t.SkipNow()
|
2019-03-28 20:21:16 +00:00
|
|
|
}
|
|
|
|
tpm, err := OpenTPM(nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("OpenTPM() failed: %v", err)
|
|
|
|
}
|
|
|
|
defer tpm.Close()
|
|
|
|
|
|
|
|
PCRs, _, 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)
|
|
|
|
}
|
|
|
|
for i, pcr := range PCRs {
|
|
|
|
if pcr.Index != i {
|
|
|
|
t.Errorf("PCR index %d does not match map index %d", pcr.Index, i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|