Unnecessary call to secp256k1_sha256_initialize #1179

issue Coding-Enthusiast openend this issue on December 16, 2022
  1. Coding-Enthusiast commented at 5:19 pm on December 16, 2022: none

    When computing tagged-hashes for Schnorr sigs the 3 methods (challenge, aux, nonce) first call secp256k1_sha256_initialize that sets the hashstate (ie. s[0] to s[7] and bytes) to their default SHA256 values then they each immediately change all those values to the precomputed “midstate” values. The first call to secp256k1_sha256_initialize seems wasteful.

    https://github.com/bitcoin-core/secp256k1/blob/9a8d65f07f171b07bd7a33828dce073d819fbdef/src/modules/schnorrsig/main_impl.h#L16-L28

    https://github.com/bitcoin-core/secp256k1/blob/9a8d65f07f171b07bd7a33828dce073d819fbdef/src/hash_impl.h#L31-L41

    Cross post: https://github.com/bitcoin/bitcoin/issues/26712

  2. real-or-random commented at 9:41 am on December 19, 2022: contributor

    I think the call is correct at this level of abstraction. The caller in the schnorrsig module is not supposed to know what secp256k1_sha256_initialize() does, e.g., it could initialize further struct members in the future. Moreover, it’s not a performance problem because any sane compiler will hopefully drop the call.

    However, that means that one could say that the code in the schnorrsig module shouldn’t set the struct members directly. This was also discussed in #731 but I ended up not doing it. I think my reasoning was that a midstate is really just a triple of the state of “internal” arrays, buffer of unwritten bytes and bytes written so far. This matches the struct exactly. So one way to look at this is that the struct is “public” to other modules, and those modules can simply write it to.

    If we really want to fix this code smell, here are three ways:

    1. We could still introduce a setter function but the API would be somewhat annoying due to the 8 integers plus the array (which means we need a pointer). https://github.com/bitcoin-core/secp256k1/pull/731/files#diff-c2d5f1f7616875ab71cd41b053cfb428696988ff89642b931a0963d50f34f7e8R557-R558.
    2. Alternatively, we could restrict the API to midstates with no unwritten bytes (i.e., where the byte counter is divisible by 64). This is all the callers currently need and then we could at least drop the pointer argument. This would mean we introduce a function secp256k1_sha256_initialize_midstate(uint64_t bytes, uint32_t s1, ..., uint32_t s2). (Or alternatively make the 8 arguments an array if 8 arguments feel too many. But I think 8 args is still fine and they provide more type-safety than an array.)
    3. We could make the hash module offer specific initialization functions, e.g., secp256k1_sha256_initialize_bip340_nonce. This works but it feels a little bit wrong to have the bip340-specific code in the hash module.

    I prefer solution 2. @Coding-Enthusiast, would be by any chance be interested in working on this?

  3. Coding-Enthusiast commented at 10:20 am on December 19, 2022: none

    My C knowledge is readonly!

    How about letting the hash module handle the tagged hash computation in general. Something like what bitcoin core project does in the interpreter.cpp file using HashWriter which I believe is reusable and also doesn’t need hard-coded midstate values.

  4. real-or-random commented at 10:28 am on December 19, 2022: contributor

    and also doesn’t need hard-coded midstate values.

    We want hard-coded midstates for performance.

  5. Coding-Enthusiast commented at 10:35 am on December 19, 2022: none
    I see. My understanding is that HashWriter precomputes the midstates and is stored in memory to be reused so the performance should be the same.
  6. sipa commented at 2:24 pm on December 19, 2022: contributor
    @Coding-Enthusiast That’s not easily doable in C, as it has no runtime-computed constants. We’d need to store it in the context or so, and compute it on initialization. I think that’s overkill for what we’re doing here.

github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin-core/secp256k1. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2025-01-24 08:15 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me