• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2021 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include <cstdlib>
17 #include <cstring>
18 #include <cinttypes>
19 #include <iostream>
20 #include <string>
21 #include <thread>
22 #include <string_view>
23 
24 #include "log_kmsg.h"
25 #include "flow_control.h"
26 #include "log_domains.h"
27 #include "log_collector.h"
28 
29 namespace OHOS {
30 namespace HiviewDFX {
31 using namespace std;
32 
InsertDropInfo(const HilogMsg & msg,int droppedCount)33 void LogCollector::InsertDropInfo(const HilogMsg &msg, int droppedCount)
34 {
35     string dropLog = to_string(droppedCount) + " line(s) dropped!";
36     constexpr auto tag = "LOGLIMITD"sv;
37     std::vector<char> buffer(sizeof(HilogMsg) + tag.size() + dropLog.size() + 1, '\0');
38     HilogMsg *dropMsg = reinterpret_cast<HilogMsg *>(buffer.data());
39     if (dropMsg != nullptr) {
40         dropMsg->len     = buffer.size();
41         dropMsg->version = msg.version;
42         dropMsg->type    = msg.type;
43         dropMsg->level   = msg.level;
44         dropMsg->tag_len = tag.size();
45         dropMsg->tv_sec  = msg.tv_sec;
46         dropMsg->tv_nsec = msg.tv_nsec;
47         dropMsg->mono_sec = msg.mono_sec;
48         dropMsg->pid     = msg.pid;
49         dropMsg->tid     = msg.tid;
50         dropMsg->domain  = msg.domain;
51 
52         auto remainSize = buffer.size() - sizeof(HilogMsg);
53         if (memcpy_s(dropMsg->tag, remainSize, tag.data(), tag.size())) {
54             std::cerr << "Can't copy tag info of Drop Info message\n";
55         }
56 
57         remainSize -= tag.size();
58         auto logTextPtr = dropMsg->tag + tag.size(); // log text is behind tag data
59         if (memcpy_s(logTextPtr, remainSize, dropLog.data(), dropLog.size() + 1)) {
60             std::cerr << "Can't copy log text info of Drop Info message\n";
61         }
62 
63         InsertLogToBuffer(*dropMsg);
64     }
65 }
66 #ifndef __RECV_MSG_WITH_UCRED_
onDataRecv(std::vector<char> & data)67 void LogCollector::onDataRecv(std::vector<char>& data)
68 #else
69 void LogCollector::onDataRecv(const ucred& cred, std::vector<char>& data)
70 #endif
71 {
72     if (data.size() < sizeof(HilogMsg)) {
73         std::cerr << "Internal error - received data less than HilogMsg size";
74         return;
75     }
76 
77     HilogMsg& msg = *(reinterpret_cast<HilogMsg *>(data.data()));
78     // check domain id
79     if (!debug && IsValidDomain(static_cast<LogType>(msg.type), msg.domain) == false) {
80         std::cout << "Invalid domain id: 0x" << std::hex << msg.domain << std::dec << ", type:" << msg.type << endl;
81         return;
82     }
83 #ifdef __RECV_MSG_WITH_UCRED_
84     msg.pid = cred.pid;
85 #endif
86     bool dropped = false;
87     // Domain flow control
88     do {
89         int ret = 0;
90         if (msg.type != static_cast<uint16_t>(LOG_APP) && flowControl && !debug) {
91             ret = FlowCtrlDomain(msg);
92         }
93         if (ret < 0) {
94             // dropping message
95             dropped = true;
96             break;
97         } else if (ret > 0) {
98             // >0 means a new statistic period start, "ret" is the number of dropping lines in last period
99             InsertDropInfo(msg, ret);
100         }
101         // If insert msg fail, return 0
102         if (InsertLogToBuffer(msg) == 0) {
103             return;
104         }
105     } while (0);
106 
107     // Log statistics
108     if (countEnable) {
109         if (*(msg.tag + msg.tag_len - 1) != '\0') {
110             return;
111         }
112         StatsInfo info = {
113             .level = msg.level,
114             .type = msg.type,
115             .len = (msg.len - sizeof(HilogMsg) - 1 - 1), // don't count '\0' of tag and content
116             .dropped = dropped ? 1 : 0,
117             .domain = msg.domain,
118             .pid = msg.pid,
119             .tv_sec = msg.tv_sec,
120             .tv_nsec = msg.tv_nsec,
121             .mono_sec = msg.mono_sec,
122             .tag = msg.tag
123         };
124         m_hilogBuffer.CountLog(info);
125     }
126 }
127 
InsertLogToBuffer(const HilogMsg & msg)128 size_t LogCollector::InsertLogToBuffer(const HilogMsg& msg)
129 {
130     return m_hilogBuffer.Insert(msg);
131 }
132 
SetLogFlowControl(bool on)133 void LogCollector::SetLogFlowControl(bool on)
134 {
135     flowControl = on;
136 }
137 
SetDebuggable(bool on)138 void LogCollector::SetDebuggable(bool on)
139 {
140     debug = on;
141 }
142 } // namespace HiviewDFX
143 } // namespace OHOS
144