mirror of
https://review.haiku-os.org/haiku
synced 2025-01-18 20:48:48 +01:00
9a79e531ef
* Introduce TracingMetaData::IsInBuffer() to validate that a certain memory range is within the valid tracing buffer limits. * Use that when validating in tracing_is_entry_valid() before trying to access the entry, resolving a TODO. * Validate the candidate time against the handed in time (if specified) as an additional check. * Tiny unrelated text cleanup. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@43116 a95241bf-73f2-0310-859d-f6bbb57e9c96
301 lines
5.7 KiB
C++
301 lines
5.7 KiB
C++
/*
|
|
* Copyright 2008, Ingo Weinhold, ingo_weinhold@gmx.de.
|
|
* Copyright 2008, Axel Dörfler, axeld@pinc-software.de.
|
|
* Distributed under the terms of the MIT License.
|
|
*/
|
|
#ifndef KERNEL_TRACING_H
|
|
#define KERNEL_TRACING_H
|
|
|
|
|
|
#include <SupportDefs.h>
|
|
#include <KernelExport.h>
|
|
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
|
|
#include "tracing_config.h"
|
|
|
|
|
|
struct trace_entry {
|
|
uint32 size : 13; // actual size is *4
|
|
uint32 previous_size : 13; // actual size is *4
|
|
uint32 flags : 6;
|
|
};
|
|
|
|
struct tracing_stack_trace {
|
|
int32 depth;
|
|
addr_t return_addresses[0];
|
|
};
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
#include <new>
|
|
|
|
|
|
// trace output flags
|
|
#define TRACE_OUTPUT_TEAM_ID 0x01
|
|
// print the team ID
|
|
#define TRACE_OUTPUT_DIFF_TIME 0x02
|
|
// print the difference time to the previously printed entry instead of the
|
|
// absolute time
|
|
|
|
|
|
class TraceOutput {
|
|
public:
|
|
TraceOutput(char* buffer, size_t bufferSize, uint32 flags);
|
|
|
|
void Clear();
|
|
void Print(const char* format,...)
|
|
__attribute__ ((format (__printf__, 2, 3)));
|
|
void PrintArgs(const char* format, va_list args);
|
|
void PrintStackTrace(tracing_stack_trace* stackTrace);
|
|
bool IsFull() const { return fSize >= fCapacity; }
|
|
|
|
char* Buffer() const { return fBuffer; }
|
|
size_t Capacity() const { return fCapacity; }
|
|
size_t Size() const { return fSize; }
|
|
|
|
uint32 Flags() const { return fFlags; }
|
|
|
|
void SetLastEntryTime(bigtime_t time);
|
|
bigtime_t LastEntryTime() const;
|
|
|
|
private:
|
|
char* fBuffer;
|
|
size_t fCapacity;
|
|
size_t fSize;
|
|
uint32 fFlags;
|
|
bigtime_t fLastEntryTime;
|
|
};
|
|
|
|
|
|
class TraceEntry {
|
|
public:
|
|
TraceEntry();
|
|
virtual ~TraceEntry();
|
|
|
|
virtual void Dump(TraceOutput& out);
|
|
virtual void DumpStackTrace(TraceOutput& out);
|
|
|
|
size_t Size() const { return ToTraceEntry()->size; }
|
|
uint16 Flags() const { return ToTraceEntry()->flags; }
|
|
|
|
void Initialized();
|
|
|
|
void* operator new(size_t size, const std::nothrow_t&) throw();
|
|
|
|
trace_entry* ToTraceEntry() const
|
|
{
|
|
return (trace_entry*)this - 1;
|
|
}
|
|
|
|
static TraceEntry* FromTraceEntry(trace_entry* entry)
|
|
{
|
|
return (TraceEntry*)(entry + 1);
|
|
}
|
|
};
|
|
|
|
|
|
class AbstractTraceEntry : public TraceEntry {
|
|
public:
|
|
AbstractTraceEntry()
|
|
{
|
|
_Init();
|
|
}
|
|
|
|
// dummy, ignores all arguments
|
|
AbstractTraceEntry(size_t, size_t, bool)
|
|
{
|
|
_Init();
|
|
}
|
|
|
|
virtual ~AbstractTraceEntry();
|
|
|
|
virtual void Dump(TraceOutput& out);
|
|
|
|
virtual void AddDump(TraceOutput& out);
|
|
|
|
thread_id ThreadID() const { return fThread; }
|
|
thread_id TeamID() const { return fTeam; }
|
|
bigtime_t Time() const { return fTime; }
|
|
|
|
protected:
|
|
typedef AbstractTraceEntry TraceEntryBase;
|
|
|
|
private:
|
|
void _Init();
|
|
|
|
protected:
|
|
thread_id fThread;
|
|
team_id fTeam;
|
|
bigtime_t fTime;
|
|
};
|
|
|
|
|
|
class AbstractTraceEntryWithStackTrace : public AbstractTraceEntry {
|
|
public:
|
|
AbstractTraceEntryWithStackTrace(size_t stackTraceDepth,
|
|
size_t skipFrames, bool kernelOnly);
|
|
|
|
virtual void DumpStackTrace(TraceOutput& out);
|
|
|
|
tracing_stack_trace* StackTrace() const
|
|
{
|
|
return fStackTrace;
|
|
}
|
|
|
|
protected:
|
|
typedef AbstractTraceEntryWithStackTrace TraceEntryBase;
|
|
|
|
private:
|
|
tracing_stack_trace* fStackTrace;
|
|
};
|
|
|
|
|
|
template<bool stackTraceDepth>
|
|
struct AbstractTraceEntrySelector {
|
|
typedef AbstractTraceEntryWithStackTrace Type;
|
|
};
|
|
|
|
|
|
template<>
|
|
struct AbstractTraceEntrySelector<0> {
|
|
typedef AbstractTraceEntry Type;
|
|
};
|
|
|
|
|
|
#define TRACE_ENTRY_SELECTOR(stackTraceDepth) \
|
|
AbstractTraceEntrySelector<stackTraceDepth>::Type
|
|
|
|
|
|
class LazyTraceOutput : public TraceOutput {
|
|
public:
|
|
LazyTraceOutput(char* buffer, size_t bufferSize, uint32 flags)
|
|
: TraceOutput(buffer, bufferSize, flags)
|
|
{
|
|
}
|
|
|
|
const char* DumpEntry(const TraceEntry* entry)
|
|
{
|
|
if (Size() == 0) {
|
|
const_cast<TraceEntry*>(entry)->Dump(*this);
|
|
// Dump() should probably be const
|
|
}
|
|
|
|
return Buffer();
|
|
}
|
|
};
|
|
|
|
|
|
class TraceFilter {
|
|
public:
|
|
virtual ~TraceFilter();
|
|
|
|
virtual bool Filter(const TraceEntry* entry, LazyTraceOutput& out);
|
|
|
|
public:
|
|
union {
|
|
thread_id fThread;
|
|
team_id fTeam;
|
|
const char* fString;
|
|
uint64 fValue;
|
|
struct {
|
|
TraceFilter* first;
|
|
TraceFilter* second;
|
|
} fSubFilters;
|
|
};
|
|
};
|
|
|
|
|
|
class WrapperTraceFilter : public TraceFilter {
|
|
public:
|
|
virtual void Init(TraceFilter* filter, int direction, bool continued) = 0;
|
|
};
|
|
|
|
|
|
class TraceEntryIterator {
|
|
public:
|
|
TraceEntryIterator()
|
|
:
|
|
fEntry(NULL),
|
|
fIndex(0)
|
|
{
|
|
}
|
|
|
|
void Reset()
|
|
{
|
|
fEntry = NULL;
|
|
fIndex = 0;
|
|
}
|
|
|
|
int32 Index() const
|
|
{
|
|
return fIndex;
|
|
}
|
|
|
|
TraceEntry* Current() const
|
|
{
|
|
return fEntry != NULL ? TraceEntry::FromTraceEntry(fEntry) : NULL;
|
|
}
|
|
|
|
TraceEntry* Next();
|
|
TraceEntry* Previous();
|
|
TraceEntry* MoveTo(int32 index);
|
|
|
|
private:
|
|
trace_entry* _NextNonBufferEntry(trace_entry* entry);
|
|
trace_entry* _PreviousNonBufferEntry(trace_entry* entry);
|
|
|
|
private:
|
|
trace_entry* fEntry;
|
|
int32 fIndex;
|
|
};
|
|
|
|
|
|
inline void
|
|
TraceOutput::Print(const char* format,...)
|
|
{
|
|
va_list args;
|
|
va_start(args, format);
|
|
PrintArgs(format, args);
|
|
va_end(args);
|
|
}
|
|
|
|
|
|
int dump_tracing(int argc, char** argv, WrapperTraceFilter* wrapperFilter);
|
|
|
|
bool tracing_is_entry_valid(AbstractTraceEntry* entry,
|
|
bigtime_t entryTime = -1);
|
|
|
|
#endif // __cplusplus
|
|
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
uint8* alloc_tracing_buffer(size_t size);
|
|
uint8* alloc_tracing_buffer_memcpy(const void* source, size_t size, bool user);
|
|
char* alloc_tracing_buffer_strcpy(const char* source, size_t maxSize,
|
|
bool user);
|
|
|
|
struct tracing_stack_trace* capture_tracing_stack_trace(int32 maxCount,
|
|
int32 skipFrames, bool kernelOnly);
|
|
addr_t tracing_find_caller_in_stack_trace(
|
|
struct tracing_stack_trace* stackTrace, const addr_t excludeRanges[],
|
|
uint32 excludeRangeCount);
|
|
void tracing_print_stack_trace(struct tracing_stack_trace* stackTrace);
|
|
|
|
void lock_tracing_buffer();
|
|
void unlock_tracing_buffer();
|
|
status_t tracing_init(void);
|
|
|
|
void _user_ktrace_output(const char *message);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* KERNEL_TRACING_H */
|