Cryptographic primitive implementation is notoriously difficult, as any error in the implementation could allow an attacker to recover private information. To reduce the possibility of such error and obtain an independent evaluation of the robustness of our work, Western Digital engaged the security research firm Trail of Bits to review Sweet B prior to its public release. During the review, Trail of Bits assessed the conformance, test coverage, API design, and timing behavior of the library, and developed an harness based on qemu to experimentally validate that instruction traces captured from library routines are independent of secret inputs to the routines.
The full report detailed six main findings, as well as a set of informational and long term recommendations. The status of these findings and recommendations are detailed below.
Finding | Severity | Summary | Status |
---|---|---|---|
TOB-SB-004 | Medium | Debug definitions may violate layout invariants relied on by assembly | Remediated |
TOB-SB-003 | Low | C library routines may violate timing guarantees. | See below |
TOB-SB-001 | Low | Debug asserts violate timing guarantees | Remediated |
TOB-SB-006 | Low | HMAC_DRBG does not provide backtracking resistance without additional input | Remediated |
TOB-SB-002 | Informational | SDL mandates use of Annex K bounds-checking interfaces | See below |
TOB-SB-005 | Informational | APIs for ECDSA signing and verification do not enforce secure hashing | Remediated |
Non-security findings | N/A | Several typos and code quality suggestions | Remediated |
Long-term recommendations | N/A | Several API, documentation, and build structure suggestions | Not yet incorporated |
Additional unit testing | N/A | Improvement of unit test coverage | Partially incorporated |
This finding has been remediated in commit
2180aa8
by ensuring that debug defines cannot be enabled if assembly support is
compiled in, and as per TOB-SB-001, that debug definitions cannot be used
outside the unit testing harness.
Cryptographic routines must be timing-invariant with respect to secret inputs in order to avoid information leakage that may be leveraged by an adversary. The C language provides no guarantees about timing, and C library routines might violate timing constraints through unexpected data-dependence. Trail of Bits has recommended avoiding the use of C library routines in timing-sensitive operations. Unfortunately, the naïve approach of replacing calls to library routine with hand-written code has a significant pitfall in that the compiler is also permitted to replace any section of code with a call to an equivalent C library routine. In particular, hand-written loops for memory copying or initialization and structure assignments are both candidates for replacement with calls to library routines.
The GNU C Compiler supports an option called -ffreestanding
which purports to
prevent the compiler from assuming that standard C library routines exist,
but due to a longstanding bug,
this option does not prevent the compiler from inserting calls to memcpy
in
all circumstances.
In the short term, we believe the risk of data-dependent timing in C library routines is low, as these routines have little reason to inspect the data being operated on. In the long term, the intended mitigation is to provide assembly routines for memory copying and initialization with defined timing behavior.
This finding has been remediated in commit e8ccfb3
by ensuring that debug asserts can only be enabled when compiling the unit
test harness. Attempting to enable debug asserts in other configurations
will result in a preprocessor error.
This finding has been remediated in commit 75c5f58
by providing a new API sb_hmac_drbg_generate_additional_dummy
which
provides static additional input to the HMAC_DRBG generate routine, and by
using this API in all contexts where additional input is not otherwise
available, with the exception of RFC6979 signing. The existing
sb_hmac_drbg_generate
API has been retained for compliance with
NIST SP 800-90A Rev. 1,
but the new API is recommended for use instead.
In order to facilitate integration with as many embedded and low-level host environments as possible, Sweet B is written in C. Improper use of memory-related routines in C is a longstanding cause of exploitable undefined behavior, and memory-related undefined behavior has historically been the cause of roughly 70% of Microsoft's disclosed CVEs. As a result, Microsoft designed a new set of "bounds-checking" C routines for addressing memory unsafety in C, and these routines were added as Annex K to ISO C11. Additionally, Microsoft added the non-bounds-checking versions of these routines to a list of banned functions as part of their Security Design Lifecycle. However, these bounds-checking routines remain problematic and poorly adopted for several reasons:
- Calculation of the correct bounds for these routines must still be done manually, and in many uses, the use of the "bounds-checking" interface would involve simply passing the same static parameter to the interface twice, which provides no additional safety in practice.
- Even when using the Annex K routines, memory unsafety and other forms of undefined behavior are still frighteningly easy to trigger in C, and other tooling must still be used to detect these errors (such as static analyzers, compiler-based sanitizers, and interpreter-based undefined behavior detectors, all of which have been applied to Sweet B).
- Bounds-checking interfaces are not implemented in most standard C libraries, and the use of a third-party library complicates the software supply chain by adding a dependency where none would otherwise exist.
Annex K has been proposed for removal from the next revision of the C standard, and while others have proposed revising and retaining its functionality, it seems clear that these interfaces will not be widely supported in the near future.
As such, while bounds-checking interfaces could be optionally adopted in Sweet B, it seems unlikely that they would provide any benefit to most users unless the use of an additional third-party library is mandated. Given our explicit goal of facilitating wide adoption in embedded and low-level projects, this would be counterproductive. The authors believe that concerns with memory unsafety in C are best addressed by eschewing the use of C entirely except in specific contexts, and when C is used, through the use of sanitizers and other mitigations in conjunction with extensive testing.
This finding has been remediated in commit
e2e5de6
by providing additional APIs as recommended which more completely encapsulate
the needed message digest step:
- a complete
sb_sha256_message
API, which produces a digest for an entire message; - a complete
sb_hmac_sha256
API to parallel thesb_sha256_message
API; - a complete
sb_sw_sign_message_sha256
API, which takes an entire message as input and provides its signature and digest as output; - a complete
sb_sw_verify_signature_sha256
API, which takes an entire message as input and verifies the signature while providing its digest; - a
sb_sw_sign_message_sha256_start API
, which takes asb_sha256_state_t
and internally callssb_sha256_finish
to produce the message digest before starting incremental signing of the digest; and - a corresponding
sb_sw_verify_signature_sha256_start
API.
The existing sb_sw_sign_message_digest
signature has been retained for use
cases where the message is not available in one contiguous block of
memory and the message digest is needed in multiple contexts, but the other
signatures are be recommended for use over sb_sw_sign_message_digest
.
These findings have been remediated in commit aae5c71
.
- Trail of Bits has recommended that we produce a user's guide detailing how to incorporate Sweet B into a project. This is an excellent idea and will be pursued after other findings have been remediated.
- Scanning for external dependencies (undefined symbols) in objects containing code that is intended it be constant time is a good idea, and could be straightforwardly automated.
- Elimination of the hardcoded offset that is relied upon by assembly is not a high priority if the correct offset is asserted at compile time. When debugging asserts are not enabled, this offset should be stable on all reasonable platforms, and alternative solutions would complicate cross-compilation.
- Additional hash functions are not planned in the near-term future, but more unit tests would certainly be added if new hash functions were incorporated.
In Appendix C of the report, Trail of Bits provided an analysis of unit test coverage and made several suggestions:
- One routine (
sb_fe_rshift
) insb_fe_tests.c.h
was uncovered. This was remediated by removing the routine in commite2c4d98
. - Error handling code in the HMAC_DRBG implementation was uncovered. This was
remediated by adding an additional unit test in commit
a6561da
. - Two edge cases in
sb_sw_lib.c
were uncovered. This was remediated by adding additional unit tests in commit0cc50e6
. - A number of field-element routines in
sb_fe.c
are not covered directly; rather, they are covered by higher-level tests of curve operations. This will be remediated by adding additional unit tests.