xref: /aosp_15_r20/external/pigweed/pw_assert_tokenized/log_handler.cc (revision 61c4878ac05f98d0ceed94b57d316916de578985)
1*61c4878aSAndroid Build Coastguard Worker // Copyright 2022 The Pigweed Authors
2*61c4878aSAndroid Build Coastguard Worker //
3*61c4878aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License"); you may not
4*61c4878aSAndroid Build Coastguard Worker // use this file except in compliance with the License. You may obtain a copy of
5*61c4878aSAndroid Build Coastguard Worker // the License at
6*61c4878aSAndroid Build Coastguard Worker //
7*61c4878aSAndroid Build Coastguard Worker //     https://www.apache.org/licenses/LICENSE-2.0
8*61c4878aSAndroid Build Coastguard Worker //
9*61c4878aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*61c4878aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11*61c4878aSAndroid Build Coastguard Worker // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12*61c4878aSAndroid Build Coastguard Worker // License for the specific language governing permissions and limitations under
13*61c4878aSAndroid Build Coastguard Worker // the License.
14*61c4878aSAndroid Build Coastguard Worker 
15*61c4878aSAndroid Build Coastguard Worker #include <cstdint>
16*61c4878aSAndroid Build Coastguard Worker #include <cstring>
17*61c4878aSAndroid Build Coastguard Worker #include <memory>
18*61c4878aSAndroid Build Coastguard Worker 
19*61c4878aSAndroid Build Coastguard Worker #include "pw_assert/config.h"
20*61c4878aSAndroid Build Coastguard Worker #include "pw_assert_tokenized/handler.h"
21*61c4878aSAndroid Build Coastguard Worker #include "pw_base64/base64.h"
22*61c4878aSAndroid Build Coastguard Worker #include "pw_bytes/endian.h"
23*61c4878aSAndroid Build Coastguard Worker #include "pw_log/log.h"
24*61c4878aSAndroid Build Coastguard Worker #include "pw_log_tokenized/config.h"
25*61c4878aSAndroid Build Coastguard Worker #include "pw_log_tokenized/handler.h"
26*61c4878aSAndroid Build Coastguard Worker #include "pw_log_tokenized/log_tokenized.h"
27*61c4878aSAndroid Build Coastguard Worker #include "pw_log_tokenized/metadata.h"
28*61c4878aSAndroid Build Coastguard Worker #include "pw_span/span.h"
29*61c4878aSAndroid Build Coastguard Worker 
pw_assert_HandleFailure(void)30*61c4878aSAndroid Build Coastguard Worker extern "C" void pw_assert_HandleFailure(void) {
31*61c4878aSAndroid Build Coastguard Worker   PW_HANDLE_LOG(PW_LOG_LEVEL_FATAL,
32*61c4878aSAndroid Build Coastguard Worker                 "pw_assert_tokenized",
33*61c4878aSAndroid Build Coastguard Worker                 PW_LOG_FLAGS,
34*61c4878aSAndroid Build Coastguard Worker #if PW_ASSERT_ENABLE_DEBUG
35*61c4878aSAndroid Build Coastguard Worker                 "PW_ASSERT() or PW_DASSERT() failure");
36*61c4878aSAndroid Build Coastguard Worker #else
37*61c4878aSAndroid Build Coastguard Worker                 "PW_ASSERT() failure. Note: PW_DASSERT disabled");
38*61c4878aSAndroid Build Coastguard Worker #endif  // PW_ASSERT_ENABLE_DEBUG
39*61c4878aSAndroid Build Coastguard Worker   PW_UNREACHABLE;
40*61c4878aSAndroid Build Coastguard Worker }
41*61c4878aSAndroid Build Coastguard Worker 
pw_assert_tokenized_HandleAssertFailure(uint32_t tokenized_file_name,int line_number)42*61c4878aSAndroid Build Coastguard Worker extern "C" void pw_assert_tokenized_HandleAssertFailure(
43*61c4878aSAndroid Build Coastguard Worker     uint32_t tokenized_file_name, int line_number) {
44*61c4878aSAndroid Build Coastguard Worker   // Buffer size for binary->base64 conversion with a null terminator.
45*61c4878aSAndroid Build Coastguard Worker   constexpr size_t kBufferSize =
46*61c4878aSAndroid Build Coastguard Worker       pw::base64::EncodedSize(sizeof(tokenized_file_name)) + 1;
47*61c4878aSAndroid Build Coastguard Worker   std::byte* hash_buffer = reinterpret_cast<std::byte*>(&tokenized_file_name);
48*61c4878aSAndroid Build Coastguard Worker   char base64_buffer[kBufferSize];
49*61c4878aSAndroid Build Coastguard Worker 
50*61c4878aSAndroid Build Coastguard Worker   size_t len =
51*61c4878aSAndroid Build Coastguard Worker       pw::base64::Encode(pw::span(hash_buffer, sizeof(tokenized_file_name)),
52*61c4878aSAndroid Build Coastguard Worker                          pw::span(base64_buffer));
53*61c4878aSAndroid Build Coastguard Worker   base64_buffer[len] = '\0';
54*61c4878aSAndroid Build Coastguard Worker #if PW_ASSERT_ENABLE_DEBUG
55*61c4878aSAndroid Build Coastguard Worker   PW_HANDLE_LOG(PW_LOG_LEVEL_FATAL,
56*61c4878aSAndroid Build Coastguard Worker                 "pw_assert_tokenized",
57*61c4878aSAndroid Build Coastguard Worker                 PW_LOG_FLAGS,
58*61c4878aSAndroid Build Coastguard Worker                 "PW_ASSERT() or PW_DASSERT() failure at $%s:%d",
59*61c4878aSAndroid Build Coastguard Worker                 base64_buffer,
60*61c4878aSAndroid Build Coastguard Worker                 line_number);
61*61c4878aSAndroid Build Coastguard Worker #else
62*61c4878aSAndroid Build Coastguard Worker   PW_HANDLE_LOG(PW_LOG_LEVEL_FATAL,
63*61c4878aSAndroid Build Coastguard Worker                 "pw_assert_tokenized",
64*61c4878aSAndroid Build Coastguard Worker                 PW_LOG_FLAGS,
65*61c4878aSAndroid Build Coastguard Worker                 "PW_ASSERT() failure. Note: PW_DASSERT disabled $%s:%d",
66*61c4878aSAndroid Build Coastguard Worker                 base64_buffer,
67*61c4878aSAndroid Build Coastguard Worker                 line_number);
68*61c4878aSAndroid Build Coastguard Worker #endif  // PW_ASSERT_ENABLE_DEBUG
69*61c4878aSAndroid Build Coastguard Worker   PW_UNREACHABLE;
70*61c4878aSAndroid Build Coastguard Worker }
71*61c4878aSAndroid Build Coastguard Worker 
pw_assert_tokenized_HandleCheckFailure(uint32_t tokenized_message,int line_number)72*61c4878aSAndroid Build Coastguard Worker extern "C" void pw_assert_tokenized_HandleCheckFailure(
73*61c4878aSAndroid Build Coastguard Worker     uint32_t tokenized_message, int line_number) {
74*61c4878aSAndroid Build Coastguard Worker   // If line_number is too large to fit in the packed payload, the Metadata
75*61c4878aSAndroid Build Coastguard Worker   // class will properly set it to 0, which is the expected value for line
76*61c4878aSAndroid Build Coastguard Worker   // number values that would cause the bit field to overflow.
77*61c4878aSAndroid Build Coastguard Worker   // See https://pigweed.dev/pw_log_tokenized/#c.PW_LOG_TOKENIZED_LINE_BITS for
78*61c4878aSAndroid Build Coastguard Worker   // more info.
79*61c4878aSAndroid Build Coastguard Worker   const uint32_t payload = pw::log_tokenized::Metadata(
80*61c4878aSAndroid Build Coastguard Worker                                PW_LOG_LEVEL_FATAL, 0, PW_LOG_FLAGS, line_number)
81*61c4878aSAndroid Build Coastguard Worker                                .value();
82*61c4878aSAndroid Build Coastguard Worker   std::array<std::byte, sizeof(tokenized_message)> token_buffer =
83*61c4878aSAndroid Build Coastguard Worker       pw::bytes::CopyInOrder(pw::endian::little, tokenized_message);
84*61c4878aSAndroid Build Coastguard Worker 
85*61c4878aSAndroid Build Coastguard Worker   pw_log_tokenized_HandleLog(
86*61c4878aSAndroid Build Coastguard Worker       payload,
87*61c4878aSAndroid Build Coastguard Worker       reinterpret_cast<const uint8_t*>(token_buffer.data()),
88*61c4878aSAndroid Build Coastguard Worker       token_buffer.size());
89*61c4878aSAndroid Build Coastguard Worker   PW_UNREACHABLE;
90*61c4878aSAndroid Build Coastguard Worker }
91