blob: 8df0775645db29099e90416d803ae25bcc91bf3d [file] [log] [blame]
// Copyright 2021 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package hwsec
import (
"context"
"sort"
"strings"
"github.com/google/go-cmp/cmp"
"chromiumos/tast/common/hwsec"
hwsecremote "chromiumos/tast/remote/hwsec"
"chromiumos/tast/testing"
)
func init() {
testing.AddTest(&testing.Test{
Func: PinWeaver,
Desc: "Checks that LE credentials work",
Contacts: []string{
"kerrnel@chromium.org", // Test author
"cros-cryptohome-dev@google.com",
},
Attr: []string{"informational", "group:mainline"},
SoftwareDeps: []string{"pinweaver", "reboot"},
})
}
const (
keyLabel1 = "lecred1"
keyLabel2 = "lecred2"
goodPin = "123456"
badPin = "000000"
testPassword = "~"
user1 = "user1@example.com"
user2 = "user2@example.com"
)
func PinWeaver(ctx context.Context, s *testing.State) {
r := hwsecremote.NewCmdRunner(s.DUT())
helper, err := hwsecremote.NewHelper(r, s.DUT())
if err != nil {
s.Fatal("Helper creation error: ", err)
}
// Start with clean state. The vault may not even exist, so ignore the error returned.
helper.CryptohomeClient().RemoveVault(ctx, user1)
helper.CryptohomeClient().RemoveVault(ctx, user2)
// Be sure to cleanup when the test is done.
defer func() {
helper.CryptohomeClient().RemoveVault(ctx, user1)
helper.CryptohomeClient().RemoveVault(ctx, user2)
}()
testPINs(ctx, user1, user2, true, r, helper, s)
// Because Cr50 stores state in the firmware, that persists across reboots, this test
// needs to run before and after a reboot.
helper.Reboot(ctx)
testPINs(ctx, user1, user2, false, r, helper, s)
}
func leCredsFromDisk(ctx context.Context, r *hwsecremote.CmdRunnerRemote) ([]string, error) {
output, err := r.Run(ctx, "/bin/ls", "/home/.shadow/low_entropy_creds")
if err != nil {
return nil, err
}
labels := strings.Split(string(output), "\n")
sort.Strings(labels)
return labels, nil
}
// testPINs requires user1 and user2 because their le credential state is shored in the same place
// and should not conflict.
func testPINs(ctx context.Context, user1, user2 string, resetUsers bool, r *hwsecremote.CmdRunnerRemote, helper *hwsecremote.CmdHelperRemote, s *testing.State) {
cryptohomeHelper := helper.CryptohomeClient()
supportsLE, err := cryptohomeHelper.SupportsLECredentials(ctx)
if err != nil {
s.Fatal("Failed to get supported policies: ", err)
} else if !supportsLE {
s.Fatal("Device does not support PinWeaver")
}
if resetUsers {
if err := helper.DaemonController().Stop(ctx, hwsec.CryptohomeDaemon); err != nil {
s.Fatal("Failed to stop cryptohomeHelper")
}
// These are to ensure the machine is in a proper state.
// Error is not check from these calls because the machine could have no users or le creds yet.
r.Run(ctx, "rm -rf /home/.shadow/low_entropy_creds")
cryptohomeHelper.RemoveVault(ctx, user1)
cryptohomeHelper.RemoveVault(ctx, user2)
if err := helper.DaemonController().Start(ctx, hwsec.CryptohomeDaemon); err != nil {
s.Fatal("Failed to start cryptohomeHelper")
}
if err := cryptohomeHelper.UnmountAll(ctx); err != nil {
s.Fatal("Failed to unmountAll: ", err)
}
if err := cryptohomeHelper.MountVault(ctx, "default", hwsec.NewPassAuthConfig(user1, testPassword), true, hwsec.NewVaultConfig()); err != nil {
s.Fatal("Failed to create initial user: ", err)
}
if err := cryptohomeHelper.AddVaultKey(ctx, user1, testPassword, "default", goodPin, keyLabel1, true); err != nil {
s.Fatal("Failed to add le credential: ", err)
}
output, err := cryptohomeHelper.GetKeyData(ctx, user1, keyLabel1)
if err != nil {
s.Fatal("Failed to get key data: ", err)
}
if strings.Contains(output, "auth_locked: true") {
s.Fatal("Newly created credential is auth locked")
}
if err := cryptohomeHelper.UnmountAll(ctx); err != nil {
s.Fatal("Failed to unmountAll: ", err)
}
}
if err := cryptohomeHelper.MountVault(ctx, keyLabel1, hwsec.NewPassAuthConfig(user1, goodPin), false, hwsec.NewVaultConfig()); err != nil {
s.Fatal("Failed to mount with PIN: ", err)
}
if err := cryptohomeHelper.UnmountAll(ctx); err != nil {
s.Fatal("Failed to unmountAll: ", err)
}
// Supply invalid credentials five times to trigger firmware lockout of the credential.
for i := 0; i < 5; i++ {
if err := cryptohomeHelper.MountVault(ctx, keyLabel1, hwsec.NewPassAuthConfig(user1, badPin), false, hwsec.NewVaultConfig()); err == nil {
s.Fatal("Mount succeeded but should have failed")
}
}
if err := cryptohomeHelper.MountVault(ctx, "default", hwsec.NewPassAuthConfig(user1, testPassword), false, hwsec.NewVaultConfig()); err != nil {
s.Fatal("Failed to mount user: ", err)
}
output, err := cryptohomeHelper.GetKeyData(ctx, user1, keyLabel1)
if err != nil {
s.Fatal("Failed to get key data: ", err)
}
if !strings.Contains(output, "auth_locked: false") {
s.Fatal("Reset PIN credential is auth locked")
}
if err := cryptohomeHelper.UnmountAll(ctx); err != nil {
s.Fatal("Failed to unmountAll: ", err)
}
if err := cryptohomeHelper.MountVault(ctx, keyLabel1, hwsec.NewPassAuthConfig(user1, goodPin), false, hwsec.NewVaultConfig()); err != nil {
s.Fatal("Failed to mount with PIN: ", err)
}
if err := cryptohomeHelper.UnmountAll(ctx); err != nil {
s.Fatal("Failed to unmountAll: ", err)
}
// Create a new user to test removing.
if err := cryptohomeHelper.MountVault(ctx, "default", hwsec.NewPassAuthConfig(user2, testPassword), true, hwsec.NewVaultConfig()); err != nil {
s.Fatal("Failed to create user2: ", err)
}
leCredsBeforeAdd, err := leCredsFromDisk(ctx, r)
if err != nil {
s.Fatal("Failed to get le creds from disk: ", err)
}
if err := cryptohomeHelper.AddVaultKey(ctx, user2, testPassword, "default", goodPin, keyLabel1, true); err != nil {
s.Fatalf("Failed to add le credential %s: %v", keyLabel1, err)
}
if err := cryptohomeHelper.AddVaultKey(ctx, user2, testPassword, "default", goodPin, keyLabel2, true); err != nil {
s.Fatalf("Failed to add le credential %s: %v", keyLabel2, err)
}
if err := cryptohomeHelper.UnmountAll(ctx); err != nil {
s.Fatal("Failed to unmountAll: ", err)
}
leCredsAfterAdd, err := leCredsFromDisk(ctx, r)
if err != nil {
s.Fatal("Failed to get le creds from disk: ", err)
}
if _, err := cryptohomeHelper.RemoveVault(ctx, user2); err != nil {
s.Fatal("Failed to remove vault: ", err)
}
leCredsAfterRemove, err := leCredsFromDisk(ctx, r)
if err != nil {
s.Fatal("Failed to get le creds from disk: ", err)
}
if diff := cmp.Diff(leCredsAfterAdd, leCredsBeforeAdd); diff == "" {
s.Fatal("LE cred not added successfully")
}
if diff := cmp.Diff(leCredsAfterRemove, leCredsBeforeAdd); diff != "" {
s.Fatal("LE cred not cleaned up successfully (-got +want): ", diff)
}
}