MySQL 8.4.3
Source Code Documentation
|
This file contains. More...
#include "log_sink_perfschema.h"
#include <mysql/components/services/log_shared.h>
#include <string.h>
#include "log_builtins_internal.h"
#include "log_sink_perfschema_imp.h"
#include "log_sink_trad.h"
#include "my_dir.h"
#include "my_systime.h"
#include "mysql/components/services/log_builtins.h"
#include "mysql/my_loglevel.h"
#include "mysqld_error.h"
#include "mysys_err.h"
#include "sql/log.h"
Macros | |
#define | LOG_ERR_READ_LINE_SIZE (LOG_BUFF_MAX * 2) |
Functions | |
static size_t | log_sink_pfs_event_size (log_sink_pfs_event *e) |
Calculate the size of the given event (header + blob + '\0' + alignment). More... | |
static bool | log_sink_pfs_event_header_fits (char *p) |
Test whether we're so close to the end of the ring-buffer that another event header would not fit. More... | |
void | log_sink_pfs_read_start () |
Acquire a read-lock on the ring-buffer. More... | |
void | log_sink_pfs_read_end () |
Release read-lock on ring-buffer. More... | |
size_t | log_sink_pfs_event_count () |
Get number of events currently in ring-buffer. More... | |
log_sink_pfs_event * | log_sink_pfs_event_first () |
Get oldest event still in ring-buffer. More... | |
log_sink_pfs_event * | log_sink_pfs_event_next (log_sink_pfs_event *e) |
Get event following the supplied one. More... | |
log_sink_pfs_event * | log_sink_pfs_event_valid (log_sink_pfs_event *e, ulonglong logged) |
Use timestamp to check whether a given event-pointer still points to a valid event in the ring-buffer. More... | |
static void | log_sink_pfs_event_expire (void) |
static int | log_sink_pfs_write_wrap (size_t s) |
If the current event can fit in the ring-buffer, but the write position is so close to the physical end of the ring-buffer that the event won't fit there, wrap to the beginning of the ring-buffer. More... | |
static void | log_sink_pfs_sanitize_timestamp (log_sink_pfs_event *e) |
log_service_error | log_sink_pfs_event_add (log_sink_pfs_event *e, const char *blob_src) |
Add a log-event to the ring buffer. More... | |
static log_service_error | log_error_read_loop (const char *log_file, size_t size) |
Add all rows from a log file to the error-log ring-buffer. More... | |
log_service_error | log_error_read_log (const char *log_name) |
Restore error log messages from previous shutdown. More... | |
int | log_error_read_log_exit () |
Release error log ring-buffer. More... | |
int | log_error_read_log_init () |
Set up ring-buffer for error-log. More... | |
int | log_sink_perfschema (void *instance, log_line *ll) |
services: log sinks: logging to performance_schema ring-buffer More... | |
Variables | |
static const size_t | ring_buffer_size = 5 * 1024 * 1024 |
In the interest of not adding more settings to confuse the using, the error-log ring-buffer is of a static size for now. More... | |
static char * | ring_buffer_start = nullptr |
buffer start More... | |
static char * | ring_buffer_end = nullptr |
buffer end (for convenience) More... | |
static char * | ring_buffer_write = nullptr |
write position ("head") More... | |
static char * | ring_buffer_read = nullptr |
read pos (oldest entry, "tail") More... | |
ulong | log_sink_pfs_buffered_bytes = 0 |
bytes in use (now) More... | |
ulong | log_sink_pfs_buffered_events = 0 |
events in buffer (now) More... | |
ulong | log_sink_pfs_expired_events = 0 |
number of expired entries (ever) More... | |
ulong | log_sink_pfs_longest_event = 0 |
longest event seen (ever) More... | |
ulonglong | log_sink_pfs_latest_timestamp |
timestamp of most recent write More... | |
PSI_memory_key | key_memory_log_sink_pfs |
memory instrumentation More... | |
static mysql_rwlock_t | THR_LOCK_log_perfschema |
ring-buffer rwlock Local to the sink that feeds the table performance_schema.error_log. More... | |
This file contains.
a) the ring-buffer that stores a backlog of error-messages so they can be exposed to the SQL layer via performance_schema.error_log;
b) the log-sink that adds errors logged at run-time to the ring-buffer.
c) the error log reader that reads error log file at start-up (These functions will in turn use a parse-function defined in a log-sink. Whichever log-sink that has a parse-function is listed first in @global.log_error_services will be used; that service will decide what log-file to read (i.e. its name) and how to parse it. We initially support the reading of JSON- formatted error log files and of the traditional MySQL error log files.) This lets us restore error log information from previous runs when the server starts. These functions are called from mysqld.cc at start-up.
#define LOG_ERR_READ_LINE_SIZE (LOG_BUFF_MAX * 2) |
log_service_error log_error_read_log | ( | const char * | log_name | ) |
Restore error log messages from previous shutdown.
We try restoring from the first (leftmost) of those services listed in @global.log_error_services that have the LOG_SERVICE_LOG_PARSER characteristic.
It is assumed that the last run's log file name is the same as the current one's. That is to say, we check whether the file supplied to –log-error already exists.
Once we have determined what file to read from, we'll call
It should be noted that at the point this function is normally called, buffered error logging will not have flushed yet.
a) If we are using the built-in "trad" sink/reader, the start-up messages are usually not buffered, and have already been written to the error log. In this case, they will be restored from the log (and flushing won't add another event to the ring-buffer).
b) If we are using a reader in a loadable log-service,
log_name | The log file to read (log_error_dest). |
LOG_SERVICE_SUCCESS | Success (log read and parsed) |
LOG_SERVICE_UNABLE_TO_READ | Could not read/access() file |
LOG_SERVICE_INVALID_ARGUMENT | Invalid file-name (no '.') |
LOG_SERVICE_NOT_AVAILABLE | No log_component active that can parse log-files |
LOG_SERVICE_ARGUMENT_TOO_LONG | File-name too long |
LOG_SERVICE_COULD_NOT_MAKE_LOG_NAME | Could not determine file extension |
otherwise | Return value from reader |
int log_error_read_log_exit | ( | ) |
Release error log ring-buffer.
0 | Success - buffer was released, or did not exist in the first place. |
int log_error_read_log_init | ( | ) |
Set up ring-buffer for error-log.
0 | Success - buffer was allocated. |
!=0 | Failure - buffer was not allocated. |
|
static |
Add all rows from a log file to the error-log ring-buffer.
We have to guesstimate where to start reading in the log:
log_file | The file's name |
size | length of the input file (in bytes) |
LOG_SERVICE_SUCCESS | success |
LOG_SERVICE_OPEN_FAILED | failed to open file |
LOG_SERVICE_SEEK_FAILED | seek failed |
LOG_SERVICE_UNABLE_TO_READ | read failed |
LOG_SERVICE_PARSE_ERROR | could not find delimiter (' ') |
int log_sink_perfschema | ( | void * | instance, |
log_line * | ll | ||
) |
services: log sinks: logging to performance_schema ring-buffer
Will write timestamp, label, thread-ID, and message to stderr/file. If you should not be able to specify a label, one will be generated for you from the line's priority field.
instance | instance handle |
ll | the log line to write |
int | number of added fields, if any |
log_service_error log_sink_pfs_event_add | ( | log_sink_pfs_event * | e, |
const char * | blob_src | ||
) |
Add a log-event to the ring buffer.
In the ring-buffer, each event exists as a header and a blob. The header is a log_sink_pfs_event struct containing the traditional error-log columns. It is followed by a variable-length blob that contains just the message string in traditional log mode, and the complete event as JSON in JSON log format. The length of the event will be align to the correct boundary.
If writing the event would go past the end of the ring-buffer, we wrap around to the beginning of the buffer.
After the function success, ring_buffer_read
will be set to a valid, non-zero value.
e | filled-in event struct to copy into the ring-buffer |
blob_src | variable-length part of the event to add to the ring-buffer |
LOG_SERVICE_SUCCESS | event was added |
LOG_SERVICE_NOT_AVAILABLE | ring-buffer not available |
LOG_SERVICE_INVALID_ARGUMENT | event has no message |
LOG_SERVICE_ARGUMENT_TOO_LONG | event is larger than buffer(!) |
size_t log_sink_pfs_event_count | ( | ) |
Get number of events currently in ring-buffer.
Caller should hold THR_LOCK_log_perschema when reading this.
|
inlinestatic |
log_sink_pfs_event * log_sink_pfs_event_first | ( | ) |
Get oldest event still in ring-buffer.
Caller should hold read-lock on THR_LOCK_log_perfschema when calling this.
nullptr | No events in buffer |
otherwise | Address of oldest event in ring-buffer |
|
inlinestatic |
Test whether we're so close to the end of the ring-buffer that another event header would not fit.
p | The address where we'd like to write (e.g. ring_buffer_write) |
true | There is enough space to write a header. |
false | Insufficient space, must wrap around to buffer-start to write. |
log_sink_pfs_event * log_sink_pfs_event_next | ( | log_sink_pfs_event * | e | ) |
Get event following the supplied one.
Caller should hold read-lock on THR_LOCK_log_perfschema when calling this.
If advancing the read position puts the read-pointer beyond the highest-address event in the ring-buffer (which isn't necessarily the latest event, which is defined as the last event before catching up with the write-pointer), i.e. at a position where either a wrap- around marker exists, or there is not enough space for a header, we wrap around to the start of the ring-buffer.
e | Last event the caller was processing. This event should be valid, non-NULL, and should not be a wrap-around marker (m_messages_length == 0). |
nullptr | No more events in ring-buffer (caught up with writer) |
otherwise | Address of the next event in the ring-buffer |
|
inlinestatic |
Calculate the size of the given event (header + blob + '\0' + alignment).
The header is followed by a blob (error message or JSON representation of the complete event) and a '\0' terminator (for safety); it is then aligned to the correct address boundary if needed.
e | The event we're interested in. Must be non-NULL. |
The | total size (header + message + '\0' + padding) in bytes. |
log_sink_pfs_event * log_sink_pfs_event_valid | ( | log_sink_pfs_event * | e, |
ulonglong | logged | ||
) |
Use timestamp to check whether a given event-pointer still points to a valid event in the ring-buffer.
Caller should hold read-lock on THR_LOCK_log_perfschema when calling this.
e | Address of event |
logged | unique timestamp of event |
nullptr | Event no longer exists in ring-buffer |
otherwise | Address of the event in the ring-buffer |
void log_sink_pfs_read_end | ( | ) |
Release read-lock on ring-buffer.
void log_sink_pfs_read_start | ( | ) |
Acquire a read-lock on the ring-buffer.
|
inlinestatic |
|
inlinestatic |
If the current event can fit in the ring-buffer, but the write position is so close to the physical end of the ring-buffer that the event won't fit there, wrap to the beginning of the ring-buffer.
Write a wrap-marker if possible. Adjust the pointers as needed:
If there isn't enough space to write the current event (of size s
): a) if the read-pointer > write-pointer (which should always be the case after the first wrap), we wrap the read-pointer to the physical start of the ring-buffer b) write a wrap-marker if space permits c) wrap the write-pointer to the (physical) start of the ring-buffer
Caller must guarantee that event size <= ring-buffer-size.
0 | didn't need to wrap |
1 | write-pointer wrapped |
2 | write-pointer and read-pointer wrapped |
PSI_memory_key key_memory_log_sink_pfs |
memory instrumentation
ulong log_sink_pfs_buffered_bytes = 0 |
bytes in use (now)
ulong log_sink_pfs_buffered_events = 0 |
events in buffer (now)
ulong log_sink_pfs_expired_events = 0 |
number of expired entries (ever)
ulonglong log_sink_pfs_latest_timestamp |
timestamp of most recent write
ulong log_sink_pfs_longest_event = 0 |
longest event seen (ever)
|
static |
buffer end (for convenience)
|
static |
read pos (oldest entry, "tail")
|
static |
In the interest of not adding more settings to confuse the using, the error-log ring-buffer is of a static size for now.
This will be easy enough to change later if needs or policy change.
While a log-event can currently be up to 8 KB in size (and with minor changes be of practically arbitrary size), a majority of common events seem to be in the 150 - 200 bytes range (in trad mode, perhaps 100 more each in JSON mode) at the time of this writing. That lead us to expect a yield of 4-6 events per KB, and thus about 25,000 for a buffer of 5 MB.
|
static |
buffer start
|
static |
write position ("head")
|
static |
ring-buffer rwlock Local to the sink that feeds the table performance_schema.error_log.
Code outside of the sink can acquire / release this lock using log_sink_pfs_read_start() / log_sink_pfs_read_start().