| /* |
| * Copyright (c) 2016, The OpenThread Authors. |
| * All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of the copyright holder nor the |
| * names of its contributors may be used to endorse or promote products |
| * derived from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE |
| * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
| * POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| /** |
| * @file |
| * This file implements the logging related functions. |
| */ |
| |
| #include "logging.hpp" |
| |
| #include "common/code_utils.hpp" |
| #include "common/instance.hpp" |
| #include "common/string.hpp" |
| |
| /* |
| * Verify debug uart dependency. |
| * |
| * It is reasonable to only enable the debug uart and not enable logs to the DEBUG uart. |
| */ |
| #if (OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_DEBUG_UART) && (!OPENTHREAD_CONFIG_ENABLE_DEBUG_UART) |
| #error OPENTHREAD_CONFIG_ENABLE_DEBUG_UART_LOG requires OPENTHREAD_CONFIG_ENABLE_DEBUG_UART |
| #endif |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif |
| |
| #if !OPENTHREAD_CONFIG_LOG_DEFINE_AS_MACRO_ONLY |
| |
| static void Log(otLogLevel aLogLevel, |
| otLogRegion aLogRegion, |
| const char *aRegionPrefix, |
| const char *aFormat, |
| va_list aArgs) |
| { |
| ot::String<OPENTHREAD_CONFIG_LOG_MAX_SIZE> logString; |
| |
| #if OPENTHREAD_CONFIG_LOG_LEVEL_DYNAMIC_ENABLE |
| VerifyOrExit(otLoggingGetLevel() >= aLogLevel); |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_PREPEND_LEVEL |
| { |
| const char *levelStr = ""; |
| |
| switch (aLogLevel) |
| { |
| case OT_LOG_LEVEL_CRIT: |
| levelStr = _OT_LEVEL_CRIT_PREFIX; |
| break; |
| |
| case OT_LOG_LEVEL_WARN: |
| levelStr = _OT_LEVEL_WARN_PREFIX; |
| break; |
| |
| case OT_LOG_LEVEL_NOTE: |
| levelStr = _OT_LEVEL_NOTE_PREFIX; |
| break; |
| |
| case OT_LOG_LEVEL_INFO: |
| levelStr = _OT_LEVEL_INFO_PREFIX; |
| break; |
| |
| case OT_LOG_LEVEL_DEBG: |
| levelStr = _OT_LEVEL_DEBG_PREFIX; |
| break; |
| |
| case OT_LOG_LEVEL_NONE: |
| default: |
| levelStr = _OT_LEVEL_NONE_PREFIX; |
| break; |
| } |
| |
| #if OPENTHREAD_CONFIG_LOG_PREPEND_REGION |
| IgnoreError(logString.Append("[%s]", levelStr)); |
| #endif |
| } |
| #endif // OPENTHREAD_CONFIG_LOG_PREPEND_LEVEL |
| |
| IgnoreError(logString.Append("%s", aRegionPrefix)); |
| VerifyOrExit(logString.AppendVarArgs(aFormat, aArgs) != OT_ERROR_INVALID_ARGS); |
| otPlatLog(aLogLevel, aLogRegion, "%s" OPENTHREAD_CONFIG_LOG_SUFFIX, logString.AsCString()); |
| |
| exit: |
| return; |
| } |
| |
| #if OPENTHREAD_CONFIG_LOG_LEVEL >= OT_LOG_LEVEL_CRIT |
| void otLogCrit(otLogRegion aRegion, const char *aRegionPrefix, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_CRIT, aRegion, aRegionPrefix, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_LEVEL >= OT_LOG_LEVEL_WARN |
| void otLogWarn(otLogRegion aRegion, const char *aRegionPrefix, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_WARN, aRegion, aRegionPrefix, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_LEVEL >= OT_LOG_LEVEL_NOTE |
| void otLogNote(otLogRegion aRegion, const char *aRegionPrefix, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_NOTE, aRegion, aRegionPrefix, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_LEVEL >= OT_LOG_LEVEL_INFO |
| void otLogInfo(otLogRegion aRegion, const char *aRegionPrefix, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_INFO, aRegion, aRegionPrefix, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_LEVEL >= OT_LOG_LEVEL_DEBG |
| void otLogDebg(otLogRegion aRegion, const char *aRegionPrefix, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_DEBG, aRegion, aRegionPrefix, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_MAC |
| void otLogMac(otLogLevel aLogLevel, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| VerifyOrExit(otLoggingGetLevel() >= aLogLevel); |
| |
| va_start(args, aFormat); |
| Log(aLogLevel, OT_LOG_REGION_MAC, _OT_REGION_MAC_PREFIX, aFormat, args); |
| va_end(args); |
| |
| exit: |
| return; |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_REFERENCE_DEVICE_ENABLE |
| void otLogCertMeshCoP(const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_NONE, OT_LOG_REGION_MESH_COP, _OT_REGION_MESH_COP_PREFIX, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_OTNS_ENABLE |
| void otLogOtns(const char *aFormat, ...) |
| { |
| va_list args; |
| |
| va_start(args, aFormat); |
| Log(OT_LOG_LEVEL_NONE, OT_LOG_REGION_CORE, _OT_REGION_CORE_PREFIX, aFormat, args); |
| va_end(args); |
| } |
| #endif |
| |
| #endif // #if !OPENTHREAD_CONFIG_LOG_DEFINE_AS_MACRO_ONLY |
| |
| #if OPENTHREAD_CONFIG_LOG_PKT_DUMP |
| |
| #if OPENTHREAD_CONFIG_LOG_DEFINE_AS_MACRO_ONLY |
| #define otLogDump(aLogLevel, aLogRegion, aFormat, ...) \ |
| _otDynamicLog(aLogLevel, aLogRegion, aFormat OPENTHREAD_CONFIG_LOG_SUFFIX, __VA_ARGS__) |
| #else |
| static void otLogDump(otLogLevel aLogLevel, otLogRegion aRegion, const char *aFormat, ...) |
| { |
| va_list args; |
| |
| VerifyOrExit(otLoggingGetLevel() >= aLogLevel); |
| |
| va_start(args, aFormat); |
| Log(aLogLevel, aRegion, "", aFormat, args); |
| va_end(args); |
| |
| exit: |
| return; |
| } |
| #endif |
| |
| enum : uint8_t |
| { |
| kStringLineLength = 80, |
| kDumpBytesPerLine = 16, |
| }; |
| |
| static void DumpLine(otLogLevel aLogLevel, otLogRegion aLogRegion, const uint8_t *aBytes, const size_t aLength) |
| { |
| ot::String<kStringLineLength> string("|"); |
| |
| for (uint8_t i = 0; i < kDumpBytesPerLine; i++) |
| { |
| if (i < aLength) |
| { |
| IgnoreError(string.Append(" %02X", aBytes[i])); |
| } |
| else |
| { |
| IgnoreError(string.Append(" ..")); |
| } |
| |
| if (!((i + 1) % 8)) |
| { |
| IgnoreError(string.Append(" |")); |
| } |
| } |
| |
| IgnoreError(string.Append(" ")); |
| |
| for (uint8_t i = 0; i < kDumpBytesPerLine; i++) |
| { |
| char c = '.'; |
| |
| if (i < aLength) |
| { |
| char byteAsChar = static_cast<char>(0x7f & aBytes[i]); |
| |
| if (isprint(byteAsChar)) |
| { |
| c = byteAsChar; |
| } |
| } |
| |
| IgnoreError(string.Append("%c", c)); |
| } |
| |
| otLogDump(aLogLevel, aLogRegion, "%s", string.AsCString()); |
| } |
| |
| void otDump(otLogLevel aLogLevel, otLogRegion aLogRegion, const char *aId, const void *aBuf, const size_t aLength) |
| { |
| enum : uint8_t |
| { |
| kWidth = 72, |
| }; |
| |
| size_t idLen = strlen(aId); |
| ot::String<kStringLineLength> string; |
| |
| for (size_t i = 0; i < (kWidth - idLen) / 2 - 5; i++) |
| { |
| IgnoreError(string.Append("=")); |
| } |
| |
| IgnoreError(string.Append("[%s len=%03u]", aId, static_cast<unsigned>(aLength))); |
| |
| for (size_t i = 0; i < (kWidth - idLen) / 2 - 4; i++) |
| { |
| IgnoreError(string.Append("=")); |
| } |
| |
| otLogDump(aLogLevel, aLogRegion, "%s", string.AsCString()); |
| |
| for (size_t i = 0; i < aLength; i += kDumpBytesPerLine) |
| { |
| DumpLine(aLogLevel, aLogRegion, static_cast<const uint8_t *>(aBuf) + i, |
| OT_MIN((aLength - i), static_cast<size_t>(kDumpBytesPerLine))); |
| } |
| |
| string.Clear(); |
| |
| for (size_t i = 0; i < kWidth; i++) |
| { |
| IgnoreError(string.Append("-")); |
| } |
| |
| otLogDump(aLogLevel, aLogRegion, "%s", string.AsCString()); |
| } |
| #else // OPENTHREAD_CONFIG_LOG_PKT_DUMP |
| void otDump(otLogLevel, otLogRegion, const char *, const void *, const size_t) |
| { |
| } |
| #endif // OPENTHREAD_CONFIG_LOG_PKT_DUMP |
| |
| static const char *const sThreadErrorStrings[OT_NUM_ERRORS] = { |
| "OK", // OT_ERROR_NONE = 0 |
| "Failed", // OT_ERROR_FAILED = 1 |
| "Drop", // OT_ERROR_DROP = 2 |
| "NoBufs", // OT_ERROR_NO_BUFS = 3 |
| "NoRoute", // OT_ERROR_NO_ROUTE = 4 |
| "Busy", // OT_ERROR_BUSY = 5 |
| "Parse", // OT_ERROR_PARSE = 6 |
| "InvalidArgs", // OT_ERROR_INVALID_ARGS = 7 |
| "Security", // OT_ERROR_SECURITY = 8 |
| "AddressQuery", // OT_ERROR_ADDRESS_QUERY = 9 |
| "NoAddress", // OT_ERROR_NO_ADDRESS = 10 |
| "Abort", // OT_ERROR_ABORT = 11 |
| "NotImplemented", // OT_ERROR_NOT_IMPLEMENTED = 12 |
| "InvalidState", // OT_ERROR_INVALID_STATE = 13 |
| "NoAck", // OT_ERROR_NO_ACK = 14 |
| "ChannelAccessFailure", // OT_ERROR_CHANNEL_ACCESS_FAILURE = 15 |
| "Detached", // OT_ERROR_DETACHED = 16 |
| "FcsErr", // OT_ERROR_FCS = 17 |
| "NoFrameReceived", // OT_ERROR_NO_FRAME_RECEIVED = 18 |
| "UnknownNeighbor", // OT_ERROR_UNKNOWN_NEIGHBOR = 19 |
| "InvalidSourceAddress", // OT_ERROR_INVALID_SOURCE_ADDRESS = 20 |
| "AddressFiltered", // OT_ERROR_ADDRESS_FILTERED = 21 |
| "DestinationAddressFiltered", // OT_ERROR_DESTINATION_ADDRESS_FILTERED = 22 |
| "NotFound", // OT_ERROR_NOT_FOUND = 23 |
| "Already", // OT_ERROR_ALREADY = 24 |
| "ReservedError25", // otError 25 is reserved |
| "Ipv6AddressCreationFailure", // OT_ERROR_IP6_ADDRESS_CREATION_FAILURE = 26 |
| "NotCapable", // OT_ERROR_NOT_CAPABLE = 27 |
| "ResponseTimeout", // OT_ERROR_RESPONSE_TIMEOUT = 28 |
| "Duplicated", // OT_ERROR_DUPLICATED = 29 |
| "ReassemblyTimeout", // OT_ERROR_REASSEMBLY_TIMEOUT = 30 |
| "NotTmf", // OT_ERROR_NOT_TMF = 31 |
| "NonLowpanDataFrame", // OT_ERROR_NOT_LOWPAN_DATA_FRAME = 32 |
| "ReservedError33", // otError 33 is reserved |
| "LinkMarginLow", // OT_ERROR_LINK_MARGIN_LOW = 34 |
| "InvalidCommand", // OT_ERROR_INVALID_COMMAND = 35 |
| "Pending", // OT_ERROR_PENDING = 36 |
| }; |
| |
| const char *otThreadErrorToString(otError aError) |
| { |
| return aError < OT_ARRAY_LENGTH(sThreadErrorStrings) ? sThreadErrorStrings[aError] : "UnknownErrorType"; |
| } |
| |
| #if OPENTHREAD_CONFIG_LOG_DEFINE_AS_MACRO_ONLY |
| |
| const char *otLogLevelToPrefixString(otLogLevel aLogLevel) |
| { |
| static const char *const kLevelStrings[] = { |
| _OT_LEVEL_NONE_PREFIX, _OT_LEVEL_CRIT_PREFIX, _OT_LEVEL_WARN_PREFIX, |
| _OT_LEVEL_NOTE_PREFIX, _OT_LEVEL_INFO_PREFIX, _OT_LEVEL_DEBG_PREFIX, |
| }; |
| |
| return ((aLogLevel >= 0) && (aLogLevel < static_cast<int>(OT_ARRAY_LENGTH(kLevelStrings)))) |
| ? kLevelStrings[aLogLevel] |
| : ""; |
| } |
| #endif |
| |
| #if OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_NONE |
| /* this provides a stub, in case something uses the function */ |
| void otPlatLog(otLogLevel aLogLevel, otLogRegion aLogRegion, const char *aFormat, ...) |
| { |
| OT_UNUSED_VARIABLE(aLogLevel); |
| OT_UNUSED_VARIABLE(aLogRegion); |
| OT_UNUSED_VARIABLE(aFormat); |
| } |
| #endif |
| |
| OT_TOOL_WEAK void otPlatLogLine(otLogLevel aLogLevel, otLogRegion aLogRegion, const char *aLogLine) |
| { |
| otPlatLog(aLogLevel, aLogRegion, "%s", aLogLine); |
| } |
| |
| #ifdef __cplusplus |
| } |
| #endif |