Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor: make s2n_array_len constant #4801

Merged
merged 4 commits into from
Oct 3, 2024
Merged

Conversation

lrstewart
Copy link
Contributor

@lrstewart lrstewart commented Sep 25, 2024

Description of changes:

Our grep_simple_mistakes script requires that we use s2n_array_len everywhere instead of (sizeof(array) / sizeof(array[0])), but s2n_array_len is no longer defined as (sizeof(array) / sizeof(array[0])), so it's no longer constant. That means that you can't do like:

struct s2n_blob array1[100] = { 0 };
struct s2n_blob array2[s2n_array_len(array1)] = { 0 };

You could do (sizeof(array) / sizeof(struct s2n_blob)) as a workaround, but that's unnecessary and more brittle. We do it in multiple places though, so I modified our grep_simple_mistakes script to catch it.

The null check is also unnecessary because arrays can't be null. Only pointers can be null, and s2n_array_len doesn't work on pointers anyway. Instead, we should check that what was passed to s2n_array_len is actually an array. Doing that is a little tricky: I got a hacky solution working using __builtin_types_compatible_p and a BUILD_BUG_ON_ZERO-style compiler assert... then I realized that modern compilers will actually just check for you. So that's nice.

Testing:

If you call s2n_array_len with a pointer on more recent compiler versions (see comment on s2n_array_len), you get this error:

/home/ubuntu/s2n-tls/tests/unit/s2n_safety_test.c: In function ‘main’:
/home/ubuntu/s2n-tls/./utils/s2n_safety.h:121:20: warning: division ‘sizeof (uint16_t * {aka short unsigned int *}) / sizeof (uint16_t {aka short unsigned int})’ does not compute the number of array elements [-Wsizeof-pointer-div]
  121 |     (sizeof(array) / sizeof(array[0]) + \

Here's an godbolt to play with: https://godbolt.org/z/9MooK8z43

By submitting this pull request, I confirm that my contribution is made under the terms of the Apache 2.0 license.

@github-actions github-actions bot added the s2n-core team label Sep 25, 2024
POSIX_ENSURE_LT(protocol_len, s2n_array_len(conn->application_protocol));

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This now leads to a compiler error:

   /home/runner/work/s2n-tls/s2n-tls/tls/extensions/s2n_server_alpn.c:69:5: error: result of comparison of constant 256 with expression of type 'uint8_t' (aka 'unsigned char') is always true [-Werror,-Wtautological-constant-out-of-range-compare]
     69 |     POSIX_ENSURE_LT(protocol_len, s2n_array_len(conn->application_protocol));
        |     ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  /home/runner/work/s2n-tls/s2n-tls/utils/s2n_safety_macros.h:261:80: note: expanded from macro 'POSIX_ENSURE_LT'
    261 | #define POSIX_ENSURE_LT(a, b)                                 __S2N_ENSURE((a) < (b), POSIX_BAIL(S2N_ERR_SAFETY))
        |                                                               ~~~~~~~~~~~~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  /home/runner/work/s2n-tls/s2n-tls/utils/s2n_ensure.h:35:15: note: expanded from macro '__S2N_ENSURE'
     35 |         if (!(cond)) {             \
        |               ^~~~
  1 error generated.

I added a test instead.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

hmm... i wonder how we can preserve this check, though. in theory the conn->application_protocol size could be reduced and then we'd have an overwrite. this check would at least avoid that issue.

Copy link
Contributor Author

@lrstewart lrstewart Sep 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The new unit test I added should actually enforce that. The test I wrote sends an alpn based on the size of conn->application_protocol:

DEFER_CLEANUP(struct s2n_connection *server = s2n_connection_new(S2N_SERVER),
s2n_connection_ptr_free);
EXPECT_NOT_NULL(server);
memset(server->application_protocol, 'a', sizeof(server->application_protocol) - 1);

But then it checks that the final result matches the known max size, independent of the size of conn->application_protocol:
const char *client_alpn = s2n_get_application_protocol(client);
EXPECT_NOT_NULL(client_alpn);
EXPECT_EQUAL(strlen(client_alpn), max_size);

So if conn->application_protocol shrinks, that test fails.

@lrstewart lrstewart marked this pull request as ready for review September 25, 2024 20:36
@lrstewart lrstewart requested a review from dougch as a code owner September 25, 2024 20:36
@lrstewart lrstewart requested a review from camshaft September 25, 2024 20:36
@lrstewart lrstewart enabled auto-merge (squash) October 1, 2024 19:46
@lrstewart lrstewart disabled auto-merge October 2, 2024 00:08
@lrstewart lrstewart requested a review from dougch October 2, 2024 00:26
@lrstewart lrstewart enabled auto-merge (squash) October 3, 2024 01:30
@lrstewart lrstewart merged commit ead40c5 into aws:main Oct 3, 2024
38 checks passed
@lrstewart lrstewart deleted the arraylen_1 branch October 3, 2024 06:25
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants