blob: b1d72c599a79a7aea9f44c2169087d97bbbeeacb [file] [log] [blame]
// Copyright 2024 The BoringSSL Authors
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice appear in all copies.
//
// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
// SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
// OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
// CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
package subprocess
import (
"encoding/hex"
"encoding/json"
"fmt"
)
// The following structures reflect the JSON of ACVP PBKDF tests. See
// https://pages.nist.gov/ACVP/draft-celi-acvp-pbkdf.html#name-test-vectors
type pbkdfTestVectorSet struct {
Groups []pbkdfTestGroup `json:"testGroups"`
Mode string `json:"mode"`
}
type pbkdfTestGroup struct {
ID uint64 `json:"tgId"`
Type string `json:"testType"`
HmacAlgo string `json:"hmacAlg"`
Tests []struct {
ID uint64 `json:"tcId"`
KeyLen uint32 `json:"keyLen,omitempty"`
Salt string `json:"salt,omitempty"`
Password string `json:"password,omitempty"`
IterationCount uint32 `json:"iterationCount,omitempty"`
} `json:"tests"`
}
type pbkdfTestGroupResponse struct {
ID uint64 `json:"tgId"`
Tests []pbkdfTestResponse `json:"tests"`
}
type pbkdfTestResponse struct {
ID uint64 `json:"tcId"`
DerivedKey string `json:"derivedKey,omitempty"`
}
// pbkdf implements an ACVP algorithm by making requests to the
// subprocess to generate PBKDF2 keys.
type pbkdf struct{}
func (p *pbkdf) Process(vectorSet []byte, m Transactable) (any, error) {
var parsed pbkdfTestVectorSet
if err := json.Unmarshal(vectorSet, &parsed); err != nil {
return nil, err
}
var ret []pbkdfTestGroupResponse
// See
// https://pages.nist.gov/ACVP/draft-celi-acvp-pbkdf.html#name-test-vectors
// for details about the tests.
for _, group := range parsed.Groups {
group := group
// "There is only one test type: functional tests."
// https://pages.nist.gov/ACVP/draft-celi-acvp-pbkdf.html#section-6.1
if group.Type != "AFT" {
return nil, fmt.Errorf("test type %q in test group %d not supported", group.Type, group.ID)
}
response := pbkdfTestGroupResponse{
ID: group.ID,
}
for _, test := range group.Tests {
test := test
if test.KeyLen < 8 {
return nil, fmt.Errorf("key length must be at least 8 bits in test case %d/%d", group.ID, test.ID)
}
keyLen := uint32le(test.KeyLen)
salt, err := hex.DecodeString(test.Salt)
if err != nil {
return nil, fmt.Errorf("failed to decode hex salt in test case %d/%d: %s", group.ID, test.ID, err)
}
if test.IterationCount < 1 {
return nil, fmt.Errorf("iteration count must be at least 1 in test case %d/%d", group.ID, test.ID)
}
iterationCount := uint32le(test.IterationCount)
msg := [][]byte{[]byte(group.HmacAlgo), keyLen, salt, []byte(test.Password), iterationCount}
m.TransactAsync("PBKDF", 1, msg, func(result [][]byte) error {
response.Tests = append(response.Tests, pbkdfTestResponse{
ID: test.ID,
DerivedKey: hex.EncodeToString(result[0]),
})
return nil
})
}
m.Barrier(func() {
ret = append(ret, response)
})
}
if err := m.Flush(); err != nil {
return nil, err
}
return ret, nil
}