Skip to content

Commit

Permalink
Take advantage of JIT optimizations in the serialization hot path
Browse files Browse the repository at this point in the history
  • Loading branch information
eiriktsarpalis committed Jun 18, 2021
1 parent 4ba2497 commit cb59430
Show file tree
Hide file tree
Showing 2 changed files with 67 additions and 66 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ protected internal JsonConverter()
// In the future, this will be check for !IsSealed (and excluding value types).
CanBePolymorphic = TypeToConvert == JsonTypeInfo.ObjectType;
IsValueType = TypeToConvert.IsValueType;
CanBeNull = !IsValueType || TypeToConvert.IsNullableOfT();
CanBeNull = default(T) is null;
IsInternalConverter = GetType().Assembly == typeof(JsonConverter).Assembly;

if (HandleNull)
Expand Down Expand Up @@ -321,7 +321,7 @@ internal bool TryWrite(Utf8JsonWriter writer, in T value, JsonSerializerOptions
ThrowHelper.ThrowJsonException_SerializerCycleDetected(options.EffectiveMaxDepth);
}

if (CanBeNull && !HandleNullOnWrite && IsNull(value))
if (default(T) is null && !HandleNullOnWrite && IsNull(value))
{
// We do not pass null values to converters unless HandleNullOnWrite is true. Null values for properties were
// already handled in GetMemberAndWriteJson() so we don't need to check for IgnoreNullValues here.
Expand All @@ -331,81 +331,76 @@ internal bool TryWrite(Utf8JsonWriter writer, in T value, JsonSerializerOptions

bool ignoreCyclesPopReference = false;

if (options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles &&
// .NET types that are serialized as JSON primitive values don't need to be tracked for cycle detection e.g: string.
ConverterStrategy != ConverterStrategy.Value &&
!IsValueType && !IsNull(value))
if (
#if NET6_0_OR_GREATER
!typeof(T).IsValueType && // treated as a constant by recent versions of the JIT.
#else
!IsValueType &&
#endif
value is not null)
{
// Custom (user) converters shall not track references
// it is responsibility of the user to break cycles in case there's any
// if we compare against Preserve, objects don't get preserved when a custom converter exists
// given that the custom converter executes prior to the preserve logic.
Debug.Assert(IsInternalConverter);
Debug.Assert(value != null);

ReferenceResolver resolver = state.ReferenceResolver;

// Write null to break reference cycles.
if (resolver.ContainsReferenceForCycleDetection(value))
if (options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles &&
// .NET types that are serialized as JSON primitive values don't need to be tracked for cycle detection e.g: string.
ConverterStrategy != ConverterStrategy.Value)
{
writer.WriteNullValue();
return true;
}
// Custom (user) converters shall not track references
// it is responsibility of the user to break cycles in case there's any
// if we compare against Preserve, objects don't get preserved when a custom converter exists
// given that the custom converter executes prior to the preserve logic.
Debug.Assert(IsInternalConverter);

// For boxed reference types: do not push when boxed in order to avoid false positives
// when we run the ContainsReferenceForCycleDetection check for the converter of the unboxed value.
Debug.Assert(!CanBePolymorphic);
resolver.PushReferenceForCycleDetection(value);
ignoreCyclesPopReference = true;
}

if (CanBePolymorphic)
{
if (value == null)
{
Debug.Assert(ConverterStrategy == ConverterStrategy.Value);
Debug.Assert(!state.IsContinuation);
Debug.Assert(HandleNullOnWrite);
ReferenceResolver resolver = state.ReferenceResolver;

int originalPropertyDepth = writer.CurrentDepth;
Write(writer, value, options);
VerifyWrite(originalPropertyDepth, writer);
// Write null to break reference cycles.
if (resolver.ContainsReferenceForCycleDetection(value))
{
writer.WriteNullValue();
return true;
}

return true;
// For boxed reference types: do not push when boxed in order to avoid false positives
// when we run the ContainsReferenceForCycleDetection check for the converter of the unboxed value.
Debug.Assert(!CanBePolymorphic);
resolver.PushReferenceForCycleDetection(value);
ignoreCyclesPopReference = true;
}

Type type = value.GetType();
if (type == JsonTypeInfo.ObjectType)
if (CanBePolymorphic)
{
writer.WriteStartObject();
writer.WriteEndObject();
return true;
}

if (type != TypeToConvert && IsInternalConverter)
{
// For internal converter only: Handle polymorphic case and get the new converter.
// Custom converter, even though polymorphic converter, get called for reading AND writing.
JsonConverter jsonConverter = state.Current.InitializeReEntry(type, options);
Debug.Assert(jsonConverter != this);

if (options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles &&
jsonConverter.IsValueType)
Type type = value.GetType();
if (type == JsonTypeInfo.ObjectType)
{
// For boxed value types: push the value before it gets unboxed on TryWriteAsObject.
state.ReferenceResolver.PushReferenceForCycleDetection(value);
ignoreCyclesPopReference = true;
writer.WriteStartObject();
writer.WriteEndObject();
return true;
}

// We found a different converter; forward to that.
bool success2 = jsonConverter.TryWriteAsObject(writer, value, options, ref state);

if (ignoreCyclesPopReference)
if (type != TypeToConvert && IsInternalConverter)
{
state.ReferenceResolver.PopReferenceForCycleDetection();
}
// For internal converter only: Handle polymorphic case and get the new converter.
// Custom converter, even though polymorphic converter, get called for reading AND writing.
JsonConverter jsonConverter = state.Current.InitializeReEntry(type, options);
Debug.Assert(jsonConverter != this);

if (options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles &&
jsonConverter.IsValueType)
{
// For boxed value types: push the value before it gets unboxed on TryWriteAsObject.
state.ReferenceResolver.PushReferenceForCycleDetection(value);
ignoreCyclesPopReference = true;
}

return success2;
// We found a different converter; forward to that.
bool success2 = jsonConverter.TryWriteAsObject(writer, value, options, ref state);

if (ignoreCyclesPopReference)
{
state.ReferenceResolver.PopReferenceForCycleDetection();
}

return success2;
}
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -234,11 +234,17 @@ internal override bool GetMemberAndWriteJson(object obj, ref WriteStack state, U
{
T value = Get!(obj);

if (Options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles &&
value != null &&
if (
#if NET6_0_OR_GREATER
!typeof(T).IsValueType && // treated as a constant by recent versions of the JIT.
#else
!Converter.IsValueType &&
#endif
Options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles &&
value is not null &&
// .NET types that are serialized as JSON primitive values don't need to be tracked for cycle detection e.g: string.
// However JsonConverter<object> that uses ConverterStrategy == Value is an exception.
(Converter.CanBePolymorphic || (!Converter.IsValueType && ConverterStrategy != ConverterStrategy.Value)) &&
(Converter.CanBePolymorphic || ConverterStrategy != ConverterStrategy.Value) &&
state.ReferenceResolver.ContainsReferenceForCycleDetection(value))
{
// If a reference cycle is detected, treat value as null.
Expand Down

0 comments on commit cb59430

Please sign in to comment.