• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2006-2009 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef V8_LOGGING_LOG_UTILS_H_
6 #define V8_LOGGING_LOG_UTILS_H_
7 
8 #include <stdio.h>
9 
10 #include <atomic>
11 #include <cstdarg>
12 #include <memory>
13 
14 #include "src/base/compiler-specific.h"
15 #include "src/base/optional.h"
16 #include "src/base/platform/mutex.h"
17 #include "src/flags/flags.h"
18 #include "src/utils/allocation.h"
19 #include "src/utils/ostreams.h"
20 
21 namespace v8 {
22 namespace internal {
23 
24 class Logger;
25 template <typename T>
26 class Vector;
27 
28 enum class LogSeparator { kSeparator };
29 
30 // Functions and data for performing output of log messages.
31 class Log {
32  public:
33   explicit Log(Logger* logger, std::string log_file_name);
34 
InitLogAtStart()35   static bool InitLogAtStart() {
36     return FLAG_log || FLAG_log_all || FLAG_log_api || FLAG_log_code ||
37            FLAG_log_handles || FLAG_log_suspect || FLAG_ll_prof ||
38            FLAG_perf_basic_prof || FLAG_perf_prof || FLAG_log_source_code ||
39            FLAG_gdbjit || FLAG_log_internal_timer_events || FLAG_prof_cpp ||
40            FLAG_trace_ic || FLAG_log_function_events || FLAG_trace_zone_stats ||
41            FLAG_turbo_profiling_log_builtins;
42   }
43 
44   V8_EXPORT_PRIVATE static bool IsLoggingToConsole(std::string file_name);
45   V8_EXPORT_PRIVATE static bool IsLoggingToTemporaryFile(std::string file_name);
46 
47   // Frees all resources acquired in Initialize and Open... functions.
48   // When a temporary file is used for the log, returns its stream descriptor,
49   // leaving the file open.
50   FILE* Close();
51 
52   std::string file_name() const;
53 
54   // Size of buffer used for formatting log messages.
55   static const int kMessageBufferSize = 2048;
56 
57   // This mode is only used in tests, as temporary files are automatically
58   // deleted on close and thus can't be accessed afterwards.
59   V8_EXPORT_PRIVATE static const char* const kLogToTemporaryFile;
60   static const char* const kLogToConsole;
61 
62   // Utility class for formatting log messages. It escapes the given messages
63   // and then appends them to the static buffer in Log.
64   class MessageBuilder {
65    public:
66     ~MessageBuilder() = default;
67 
68     void AppendString(String str,
69                       base::Optional<int> length_limit = base::nullopt);
70     void AppendString(Vector<const char> str);
71     void AppendString(const char* str);
72     void AppendString(const char* str, size_t length, bool is_one_byte = true);
73     void PRINTF_FORMAT(2, 3) AppendFormatString(const char* format, ...);
74     void AppendCharacter(char c);
75     void AppendTwoByteCharacter(char c1, char c2);
76     void AppendSymbolName(Symbol symbol);
77 
78     // Delegate insertion to the underlying {log_}.
79     // All appended strings are escaped to maintain one-line log entries.
80     template <typename T>
81     MessageBuilder& operator<<(T value) {
82       log_->os_ << value;
83       return *this;
84     }
85 
86     // Finish the current log line an flush the it to the log file.
87     void WriteToLogFile();
88 
89    private:
90     // Create a message builder starting from position 0.
91     // This acquires the mutex in the log as well.
92     explicit MessageBuilder(Log* log);
93 
94     // Prints the format string into |log_->format_buffer_|. Returns the length
95     // of the result, or kMessageBufferSize if it was truncated.
96     int PRINTF_FORMAT(2, 0)
97         FormatStringIntoBuffer(const char* format, va_list args);
98 
99     void AppendSymbolNameDetails(String str, bool show_impl_info);
100 
101     void PRINTF_FORMAT(2, 3) AppendRawFormatString(const char* format, ...);
102     void AppendRawCharacter(const char character);
103 
104     Log* log_;
105     base::MutexGuard lock_guard_;
106 
107     friend class Log;
108   };
109 
110   // Use this method to create an instance of Log::MessageBuilder. This method
111   // will return null if logging is disabled.
112   std::unique_ptr<Log::MessageBuilder> NewMessageBuilder();
113 
114  private:
115   static FILE* CreateOutputHandle(std::string file_name);
mutex()116   base::Mutex* mutex() { return &mutex_; }
117 
118   void WriteLogHeader();
119 
120   Logger* logger_;
121 
122   std::string file_name_;
123 
124   // When logging is active output_handle_ is used to store a pointer to log
125   // destination.  mutex_ should be acquired before using output_handle_.
126   FILE* output_handle_;
127 
128   OFStream os_;
129 
130   // mutex_ is a Mutex used for enforcing exclusive
131   // access to the formatting buffer and the log file or log memory buffer.
132   base::Mutex mutex_;
133 
134   // Buffer used for formatting log messages. This is a singleton buffer and
135   // mutex_ should be acquired before using it.
136   std::unique_ptr<char[]> format_buffer_;
137 
138   friend class Logger;
139 };
140 
141 template <>
142 Log::MessageBuilder& Log::MessageBuilder::operator<<<LogSeparator>(
143     LogSeparator separator);
144 template <>
145 Log::MessageBuilder& Log::MessageBuilder::operator<<<void*>(void* pointer);
146 template <>
147 Log::MessageBuilder& Log::MessageBuilder::operator<<<const char*>(
148     const char* string);
149 template <>
150 Log::MessageBuilder& Log::MessageBuilder::operator<<<char>(char c);
151 template <>
152 Log::MessageBuilder& Log::MessageBuilder::operator<<<String>(String string);
153 template <>
154 Log::MessageBuilder& Log::MessageBuilder::operator<<<Symbol>(Symbol symbol);
155 template <>
156 Log::MessageBuilder& Log::MessageBuilder::operator<<<Name>(Name name);
157 
158 }  // namespace internal
159 }  // namespace v8
160 
161 #endif  // V8_LOGGING_LOG_UTILS_H_
162