Avoiding performance pitfalls in incremental generators
by Andrew Lock
posted on: January 16, 2024
Creating a source generator - Part 9
by Andrew Lock
posted on: January 16, 2024
Creating a source generator - Part 9
by Steve Gordon
posted on: January 15, 2024
SERIES: A Guide to Developing GitHub Apps on .NET In this post, I cover the steps required to create and sign a JSON Web Token, herein abbreviated as JWT, to authenticate a GitHub App built using .NET. I want to state clearly up front that I’m learning as I go while experimenting with a hobby […]
by Oren Eini
posted on: January 15, 2024
I spoke with Jaime recently in the Modern .NET Podcast:In this episode of The Modern .NET Show podcast, Oren Eini, a seasoned developer with over 20 years of experience in the .NET field, discussed the evolution of the .NET framework and the complexities that come with it. Eini highlighted the rapid pace of change in the language, from the introduction of generics at version 2.0 to switch expressions and pattern matching in the latest versions. While these new features allow for more concise code, Eini acknowledged that they also increase the scope and complexity of learning C# from scratch.Would love to hear your feedback.
by Gérald Barré
posted on: January 15, 2024
When you want to get a CultureInfo object, you can use the static method CultureInfo.GetCultureInfo or the constructor of the CultureInfo class. In this post, I describe the difference between CultureInfo.GetCultureInfo and the constructor of the CultureInfo class.C#copyvar cultureInfo1 = CultureIn
by Burke Holland
posted on: January 12, 2024
Blazor Sortable is a new a open source community Blazor component for creating sortable lists of items using SortableJS.
by Oren Eini
posted on: January 11, 2024
One of the most frequent operations we make in RavenDB is getting a page pointer. That is the basis of pretty much everything that you can think of inside the storage engine. On the surface, that is pretty simple, here is the API we’ll call: public Page GetPage ( long pageNumber ) Easy, right? But internally, we need to ensure that we get the right page. And that is where some complexity enters the picture. Voron, our storage engine, implements a pattern called MVCC (multi-version concurrency control). In other words, two transactions loading the same page may see different versions of the page at the same time. What this means is that the call to GetPage () needs to check if the page: Has been modified in the current transaction Has been modified in a previously committed transaction and has not yet flushed to disk The on-disk version is the most up-to-date one Each one of those checks is cheap, but getting a page is a common operation. So we implemented a small cache in front of these operations, which resulted in a substantial performance improvement. Conceptually, here is what that cache looks like: public unsafe class PageLocator { private struct PageData { public long PageNumber ; public byte* Pointer ; public bool IsWritable ; } private PageData [] _cache = new PageData [ 512 ]; public byte* Get ( long page , out bool isWritable ) { var index = page & 511 ; ref var data = ref _cache [ index ]; if ( data.PageNumber == page ) { isWritable = data.IsWritable ; return data.Pointer ; } return LookupAndGetPage ( page , ref data , out isWritable ); } public void Reset () { for ( int i = 0 ; i < _cache.Length ; i++) _cache [ i ].PageNumber =-1 ; } } This is intended to be a fairly simple cache, and it is fine if certain access patterns aren’t going to produce optimal results. After all, the source it is using is already quite fast, we simply want to get even better performance when we can. This is important because caching is quite a complex topic on its own. The PageLocator itself is used in the context of a transaction and is pooled. Transactions in RavenDB tend to be pretty short, so that alleviates a lot of the complexity around cache management. This is actually a pretty solved problem for us, and has been for a long while. We have been using some variant of the code above for about 9 years. The reason for this post, however, is that we are trying to optimize things further. And this class showed up in our performance traces as problematic. Surprisingly enough, what is actually costly isn’t the lookup part, but making the PageLocator ready for the next transaction. We are talking about the Reset () method. The question is: how can we significantly optimize the process of making the instance ready for a new transaction? We don’t want to allocate, and resetting the page numbers is what is costing us. One option is to add an int Generation field to the PageData structure, which we’ll then check on getting from the cache. Resetting the cache can then be done by incrementing the locator’s generation with a single instruction. That is pretty awesome, right? It sadly exposes a problem, what happens when we use the locator enough to encounter an overflow? What happens if we have a sequence of events that brings us back to the same generation as a cached instance? We’ll be risking getting an old instance (from a previous transaction, which happened long ago). The chances for that are low, seriously low. But that is not an acceptable risk for us. So we need to consider going further. Here is what we ended up with: public unsafe class PageLocator { private struct PageData { public long PageNumber ; public byte* Pointer ; public ushort Generation ; public bool IsWritable ; } private PageData [] _cache = new PageData [ 512 ]; private int _generation = 1 ; public byte* Get ( long page , out bool isWritable ) { var index = page & 511 ; ref var data = ref _cache [ index ]; if ( data.PageNumber == page && data.Generation == _generation ) { isWritable = data.IsWritable ; return data.Pointer ; } return LookupAndGetPage ( page , ref data , out isWritable ); } public void Reset () { _generation++; if ( _generation >= 65535 ){ _generation = 1 ; MemoryMarshal.Cast < PageData , byte >( _cache ).Fill ( 0 ); } } } Once every 64K operations, we’ll pay the cost of resetting the entire buffer, but we do that in an instruction that is heavily optimized. If needed, we can take it further, but here are our results before the change: And afterward: The cost of the Renew ()call, which was composed mostly of the Reset () call, basically dropped off the radar,and the performance roughly doubled. That is a pretty cool benefit for a straightforward change.
by Salini Agarwal
posted on: January 09, 2024
January 2024 Security and Quality Rollup Updates for .NET Framework
by Rahul Bhandari (MSFT)
posted on: January 09, 2024
Check out latest Janaury 2024 updates for .NET 7.0 and .NET 6.0
by Andrew Lock
posted on: January 09, 2024
In this post I look at the StringValues type, where it's used in ASP.NET Core, why it's useful, how it's implemented, and why.…
by Gérald Barré
posted on: January 08, 2024
C# 12 introduced a new feature called Primary constructor. This feature allows us to define a constructor directly in the class declaration.C#copy// You can define a constructor directly in the class declaration. public readonly struct Distance(double dx, double dy) { public readonly double Mag