Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Exporter.Geneva] Avoid allocation for serializing scopes #818

Merged
merged 12 commits into from
Dec 19, 2022
3 changes: 3 additions & 0 deletions src/OpenTelemetry.Exporter.Geneva/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,9 @@
`System.Diagnostics.DiagnosticSource` v7.0.
[#838](https://github.com/open-telemetry/opentelemetry-dotnet-contrib/pull/838)

* Avoid allocation when serializing scopes.
([#818](https://github.com/open-telemetry/opentelemetry-dotnet-contrib/pull/818))

## 1.4.0-beta.6

Released 2022-Dec-09
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,9 @@ internal sealed class MsgPackLogExporter : MsgPackExporter, IDisposable
private readonly List<string> m_prepopulatedFieldKeys;
private readonly byte[] m_bufferEpilogue;
private readonly IDataTransport m_dataTransport;

// This is used for Scopes
private readonly ThreadLocal<SerializationDataForScopes> m_serializationData = new(() => null);
private bool isDisposed;

public MsgPackLogExporter(GenevaExporterOptions options)
Expand Down Expand Up @@ -279,33 +282,29 @@ internal int SerializeLogRecord(LogRecord logRecord)
cntFields += 1;
}

logRecord.ForEachScope(ProcessScopeForIndividualColumns, (object)null);
void ProcessScopeForIndividualColumns(LogRecordScope scope, object state)
// Prepare state for scopes
var dataForScopes = this.m_serializationData.Value;
if (dataForScopes == null)
{
foreach (KeyValuePair<string, object> scopeItem in scope)
dataForScopes = new SerializationDataForScopes
{
if (string.IsNullOrEmpty(scopeItem.Key) || scopeItem.Key == "{OriginalFormat}")
{
continue;
}
Buffer = buffer,
};

if (this.m_customFields == null || this.m_customFields.ContainsKey(scopeItem.Key))
{
if (scopeItem.Value != null)
{
// null is not supported.
cursor = MessagePackSerializer.SerializeUnicodeString(buffer, cursor, scopeItem.Key);
cursor = MessagePackSerializer.Serialize(buffer, cursor, scopeItem.Value);
cntFields += 1;
}
}
else
{
hasEnvProperties = true;
}
}
this.m_serializationData.Value = dataForScopes;
}

dataForScopes.Cursor = cursor;
dataForScopes.FieldsCount = cntFields;
dataForScopes.HasEnvProperties = hasEnvProperties;

logRecord.ForEachScope(ProcessScopeForIndividualColumns, this);

// Update the variables that could have been modified in ProcessScopeForIndividualColumns
hasEnvProperties = dataForScopes.HasEnvProperties;
cursor = dataForScopes.Cursor;
cntFields = dataForScopes.FieldsCount;

if (hasEnvProperties)
{
// Iteration #2 - Get all "other" fields and collapse them into single field
Expand All @@ -329,24 +328,15 @@ void ProcessScopeForIndividualColumns(LogRecordScope scope, object state)
}
}

logRecord.ForEachScope(ProcessScopeForEnvProperties, (object)null);
void ProcessScopeForEnvProperties(LogRecordScope scope, object state)
{
foreach (KeyValuePair<string, object> scopeItem in scope)
{
if (string.IsNullOrEmpty(scopeItem.Key) || scopeItem.Key == "{OriginalFormat}")
{
continue;
}

if (!this.m_customFields.ContainsKey(scopeItem.Key))
{
cursor = MessagePackSerializer.SerializeUnicodeString(buffer, cursor, scopeItem.Key);
cursor = MessagePackSerializer.Serialize(buffer, cursor, scopeItem.Value);
envPropertiesCount += 1;
}
}
}
// Prepare state for scopes
dataForScopes.Cursor = cursor;
dataForScopes.EnvPropertiesCount = envPropertiesCount;
utpilla marked this conversation as resolved.
Show resolved Hide resolved

logRecord.ForEachScope(ProcessScopeForEnvProperties, this);

// Update the variables that could have been modified in ProcessScopeForEnvProperties
cursor = dataForScopes.Cursor;
envPropertiesCount = dataForScopes.EnvPropertiesCount;

cntFields += 1;
MessagePackSerializer.WriteUInt16(buffer, idxMapSizeEnvPropertiesPatch, envPropertiesCount);
Expand Down Expand Up @@ -449,6 +439,7 @@ public void Dispose()
try
{
(this.m_dataTransport as IDisposable)?.Dispose();
this.m_serializationData.Dispose();
this.m_prepopulatedFieldKeys.Clear();
}
catch (Exception ex)
Expand All @@ -458,4 +449,64 @@ public void Dispose()

this.isDisposed = true;
}

private static readonly Action<LogRecordScope, MsgPackLogExporter> ProcessScopeForIndividualColumns = (scope, state) =>
{
var stateData = state.m_serializationData.Value;
var customFields = state.m_customFields;

foreach (KeyValuePair<string, object> scopeItem in scope)
{
if (string.IsNullOrEmpty(scopeItem.Key) || scopeItem.Key == "{OriginalFormat}")
{
continue;
}

if (customFields == null || customFields.ContainsKey(scopeItem.Key))
{
if (scopeItem.Value != null)
{
// null is not supported.
stateData.Cursor = MessagePackSerializer.SerializeUnicodeString(stateData.Buffer, stateData.Cursor, scopeItem.Key);
stateData.Cursor = MessagePackSerializer.Serialize(stateData.Buffer, stateData.Cursor, scopeItem.Value);
stateData.FieldsCount += 1;
}
}
else
{
stateData.HasEnvProperties = true;
}
}
};

private static readonly Action<LogRecordScope, MsgPackLogExporter> ProcessScopeForEnvProperties = (scope, state) =>
{
var stateData = state.m_serializationData.Value;
var customFields = state.m_customFields;

foreach (KeyValuePair<string, object> scopeItem in scope)
{
if (string.IsNullOrEmpty(scopeItem.Key) || scopeItem.Key == "{OriginalFormat}")
{
continue;
}

if (!customFields.ContainsKey(scopeItem.Key))
{
stateData.Cursor = MessagePackSerializer.SerializeUnicodeString(stateData.Buffer, stateData.Cursor, scopeItem.Key);
stateData.Cursor = MessagePackSerializer.Serialize(stateData.Buffer, stateData.Cursor, scopeItem.Value);
stateData.EnvPropertiesCount += 1;
}
}
};

private class SerializationDataForScopes
{
public bool HasEnvProperties;
public ushort EnvPropertiesCount;

public int Cursor;
public byte[] Buffer;
public ushort FieldsCount;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -400,7 +400,7 @@ public void Dispose()

private const int BUFFER_SIZE = 65360; // the maximum ETW payload (inclusive)

private readonly ThreadLocal<byte[]> m_buffer = new ThreadLocal<byte[]>(() => null);
private readonly ThreadLocal<byte[]> m_buffer = new(() => null);

private readonly byte[] m_bufferPrologue;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,42 +20,25 @@
using OpenTelemetry.Logs;

/*
BenchmarkDotNet=v0.13.1, OS=Windows 10.0.22621
BenchmarkDotNet=v0.13.2, OS=Windows 11 (10.0.22621.819)
Intel Core i7-9700 CPU 3.00GHz, 1 CPU, 8 logical and 8 physical cores
.NET SDK=7.0.100-preview.6.22352.1
[Host] : .NET 6.0.9 (6.0.922.41905), X64 RyuJIT
DefaultJob : .NET 6.0.9 (6.0.922.41905), X64 RyuJIT

Without Scopes

| Method | IncludeFormattedMessage | Mean | Error | StdDev | Gen 0 | Allocated |
|-------------------------- |------------------------ |-----------:|--------:|--------:|-------:|----------:|
| LoggerWithMessageTemplate | False | 1,273.1 ns | 6.09 ns | 5.39 ns | 0.0362 | 232 B |
| LoggerWithDirectLoggerAPI | False | 1,213.0 ns | 9.71 ns | 8.61 ns | 0.0572 | 368 B |
| LoggerWithSourceGenerator | False | 1,243.5 ns | 6.13 ns | 5.44 ns | 0.0305 | 192 B |
| SerializeLogRecord | False | 587.7 ns | 2.71 ns | 2.54 ns | - | - |
| Export | False | 955.0 ns | 5.46 ns | 5.11 ns | - | - |
| LoggerWithMessageTemplate | True | 1,261.1 ns | 6.59 ns | 5.84 ns | 0.0362 | 232 B |
| LoggerWithDirectLoggerAPI | True | 1,214.4 ns | 4.56 ns | 4.27 ns | 0.0572 | 368 B |
| LoggerWithSourceGenerator | True | 1,229.6 ns | 6.84 ns | 6.40 ns | 0.0305 | 192 B |
| SerializeLogRecord | True | 581.6 ns | 2.38 ns | 2.11 ns | - | - |
| Export | True | 958.4 ns | 3.02 ns | 2.52 ns | - | - |


With Scopes (https://github.com/open-telemetry/opentelemetry-dotnet-contrib/pull/545)

| Method | IncludeFormattedMessage | Mean | Error | StdDev | Gen 0 | Allocated |
|-------------------------- |------------------------ |-----------:|--------:|--------:|-------:|----------:|
| LoggerWithMessageTemplate | False | 1,280.8 ns | 7.45 ns | 6.61 ns | 0.0534 | 336 B |
| LoggerWithDirectLoggerAPI | False | 1,261.5 ns | 6.38 ns | 5.96 ns | 0.0744 | 472 B |
| LoggerWithSourceGenerator | False | 1,309.3 ns | 4.83 ns | 4.52 ns | 0.0458 | 296 B |
| SerializeLogRecord | False | 611.3 ns | 4.63 ns | 4.11 ns | 0.0162 | 104 B |
| Export | False | 1,012.2 ns | 7.56 ns | 7.07 ns | 0.0153 | 104 B |
| LoggerWithMessageTemplate | True | 1,278.3 ns | 6.63 ns | 5.88 ns | 0.0534 | 336 B |
| LoggerWithDirectLoggerAPI | True | 1,263.8 ns | 8.26 ns | 7.73 ns | 0.0744 | 472 B |
| LoggerWithSourceGenerator | True | 1,273.4 ns | 5.57 ns | 5.21 ns | 0.0458 | 296 B |
| SerializeLogRecord | True | 604.3 ns | 2.83 ns | 2.65 ns | 0.0162 | 104 B |
| Export | True | 1,003.6 ns | 9.29 ns | 8.69 ns | 0.0153 | 104 B |
.NET SDK=7.0.100
[Host] : .NET 6.0.11 (6.0.1122.52304), X64 RyuJIT AVX2
DefaultJob : .NET 6.0.11 (6.0.1122.52304), X64 RyuJIT AVX2


| Method | IncludeFormattedMessage | Mean | Error | StdDev | Gen0 | Allocated |
|-------------------------- |------------------------ |---------:|--------:|--------:|-------:|----------:|
| LoggerWithMessageTemplate | False | 832.5 ns | 5.99 ns | 5.00 ns | 0.0162 | 104 B |
| LoggerWithDirectLoggerAPI | False | 766.2 ns | 3.85 ns | 3.60 ns | 0.0381 | 240 B |
| LoggerWithSourceGenerator | False | 815.3 ns | 2.89 ns | 2.41 ns | 0.0095 | 64 B |
| SerializeLogRecord | False | 582.3 ns | 0.81 ns | 0.72 ns | - | - |
| Export | False | 646.0 ns | 1.10 ns | 0.86 ns | - | - |
| LoggerWithMessageTemplate | True | 847.7 ns | 5.56 ns | 5.20 ns | 0.0162 | 104 B |
| LoggerWithDirectLoggerAPI | True | 762.5 ns | 2.72 ns | 2.41 ns | 0.0381 | 240 B |
| LoggerWithSourceGenerator | True | 816.6 ns | 2.79 ns | 2.47 ns | 0.0095 | 64 B |
| SerializeLogRecord | True | 586.3 ns | 1.80 ns | 1.69 ns | - | - |
| Export | True | 659.5 ns | 6.00 ns | 5.61 ns | - | - |
*/

namespace OpenTelemetry.Exporter.Geneva.Benchmark;
Expand Down