| // Copyright 2024 The BoringSSL Authors |
| // |
| // 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 |
| // |
| // https://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. |
| |
| package subprocess |
| |
| import ( |
| "encoding/binary" |
| "encoding/hex" |
| "encoding/json" |
| "fmt" |
| ) |
| |
| // The following structures reflect the JSON of ACVP shake tests. See |
| // https://pages.nist.gov/ACVP/draft-celi-acvp-sha3.html#name-test-vectors |
| |
| type shakeTestVectorSet struct { |
| Groups []shakeTestGroup `json:"testGroups"` |
| } |
| |
| type shakeTestGroup struct { |
| ID uint64 `json:"tgId"` |
| Type string `json:"testType"` |
| MaxOutLenBits uint32 `json:"maxOutLen"` |
| MinOutLenBits uint32 `json:"minOutLen"` |
| Tests []struct { |
| ID uint64 `json:"tcId"` |
| BitLength uint64 `json:"len"` |
| BitOutLength uint32 `json:"outLen"` |
| MsgHex string `json:"msg"` |
| } `json:"tests"` |
| } |
| |
| type shakeTestGroupResponse struct { |
| ID uint64 `json:"tgId"` |
| Tests []shakeTestResponse `json:"tests"` |
| } |
| |
| type shakeTestResponse struct { |
| ID uint64 `json:"tcId"` |
| DigestHex string `json:"md,omitempty"` |
| MCTResults []shakeMCTResult `json:"resultsArray,omitempty"` |
| } |
| |
| type shakeMCTResult struct { |
| DigestHex string `json:"md"` |
| OutputLen uint32 `json:"outLen,omitempty"` |
| } |
| |
| // shake implements an ACVP algorithm by making requests to the |
| // subprocess to hash strings. |
| type shake struct { |
| // algo is the ACVP name for this algorithm and also the command name |
| // given to the subprocess to hash with this hash function. |
| algo string |
| // size is the number of bytes of digest that the hash produces for AFT tests. |
| size int |
| } |
| |
| func (h *shake) Process(vectorSet []byte, m Transactable) (any, error) { |
| var parsed shakeTestVectorSet |
| if err := json.Unmarshal(vectorSet, &parsed); err != nil { |
| return nil, err |
| } |
| |
| var ret []shakeTestGroupResponse |
| // See |
| // https://pages.nist.gov/ACVP/draft-celi-acvp-sha3.html#name-test-types |
| // for details about the tests. |
| for _, group := range parsed.Groups { |
| group := group |
| response := shakeTestGroupResponse{ |
| ID: group.ID, |
| } |
| |
| for _, test := range group.Tests { |
| test := test |
| |
| if uint64(len(test.MsgHex))*4 != test.BitLength { |
| return nil, fmt.Errorf("test case %d/%d contains hex message of length %d but specifies a bit length of %d", group.ID, test.ID, len(test.MsgHex), test.BitLength) |
| } |
| msg, err := hex.DecodeString(test.MsgHex) |
| if err != nil { |
| return nil, fmt.Errorf("failed to decode hex in test case %d/%d: %s", group.ID, test.ID, err) |
| } |
| |
| if test.BitOutLength%8 != 0 { |
| return nil, fmt.Errorf("test case %d/%d has bit length %d - fractional bytes not supported", group.ID, test.ID, test.BitOutLength) |
| } |
| |
| switch group.Type { |
| case "AFT": |
| // "AFTs all produce a single digest size, matching the security strength of the extendable output function." |
| if test.BitOutLength != uint32(h.size*8) { |
| return nil, fmt.Errorf("AFT test case %d/%d has bit length %d but expected %d", group.ID, test.ID, test.BitOutLength, h.size*8) |
| } |
| |
| m.TransactAsync(h.algo, 1, [][]byte{msg, uint32le(test.BitOutLength / 8)}, func(result [][]byte) error { |
| response.Tests = append(response.Tests, shakeTestResponse{ |
| ID: test.ID, |
| DigestHex: hex.EncodeToString(result[0]), |
| }) |
| return nil |
| }) |
| case "VOT": |
| // "The VOTs SHALL produce varying digest sizes based on the capabilities of the IUT" |
| m.TransactAsync(h.algo+"/VOT", 1, [][]byte{msg, uint32le(test.BitOutLength / 8)}, func(result [][]byte) error { |
| response.Tests = append(response.Tests, shakeTestResponse{ |
| ID: test.ID, |
| DigestHex: hex.EncodeToString(result[0]), |
| }) |
| return nil |
| }) |
| case "MCT": |
| // https://pages.nist.gov/ACVP/draft-celi-acvp-sha3.html#name-shake-monte-carlo-test |
| testResponse := shakeTestResponse{ID: test.ID} |
| |
| if group.MinOutLenBits%8 != 0 { |
| return nil, fmt.Errorf("MCT test group %d has min output length %d - fractional bytes not supported", group.ID, group.MinOutLenBits) |
| } |
| if group.MaxOutLenBits%8 != 0 { |
| return nil, fmt.Errorf("MCT test group %d has max output length %d - fractional bytes not supported", group.ID, group.MaxOutLenBits) |
| } |
| |
| digest := msg |
| minOutLenBytes := uint32le(group.MinOutLenBits / 8) |
| maxOutLenBytes := uint32le(group.MaxOutLenBits / 8) |
| outputLenBytes := uint32le(group.MaxOutLenBits / 8) |
| |
| for i := 0; i < 100; i++ { |
| args := [][]byte{digest, minOutLenBytes, maxOutLenBytes, outputLenBytes} |
| result, err := m.Transact(h.algo+"/MCT", 2, args...) |
| if err != nil { |
| panic(h.algo + " mct operation failed: " + err.Error()) |
| } |
| |
| digest = result[0] |
| outputLenBytes = uint32le(binary.LittleEndian.Uint32(result[1])) |
| mctResult := shakeMCTResult{DigestHex: hex.EncodeToString(digest), OutputLen: uint32(len(digest) * 8)} |
| testResponse.MCTResults = append(testResponse.MCTResults, mctResult) |
| } |
| |
| response.Tests = append(response.Tests, testResponse) |
| default: |
| return nil, fmt.Errorf("test group %d has unknown type %q", group.ID, group.Type) |
| } |
| } |
| |
| m.Barrier(func() { |
| ret = append(ret, response) |
| }) |
| } |
| |
| if err := m.Flush(); err != nil { |
| return nil, err |
| } |
| |
| return ret, nil |
| } |