File: System\Reflection\Metadata\Ecma335\MetadataBuilder.Tables.cs
Web Access
Project: src\src\libraries\System.Reflection.Metadata\src\System.Reflection.Metadata.csproj (System.Reflection.Metadata)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Diagnostics;
using System.Reflection.Internal;
 
namespace System.Reflection.Metadata.Ecma335
{
    public partial class MetadataBuilder
    {
        private const byte MetadataFormatMajorVersion = 2;
        private const byte MetadataFormatMinorVersion = 0;
 
        // type system table rows:
        private struct AssemblyRefTableRow { public Version Version; public BlobHandle PublicKeyToken; public StringHandle Name; public StringHandle Culture; public uint Flags; public BlobHandle HashValue; }
        private struct ModuleRow { public ushort Generation; public StringHandle Name; public GuidHandle ModuleVersionId; public GuidHandle EncId; public GuidHandle EncBaseId; }
        private struct AssemblyRow { public uint HashAlgorithm; public Version Version; public ushort Flags; public BlobHandle AssemblyKey; public StringHandle AssemblyName; public StringHandle AssemblyCulture; }
        private struct ClassLayoutRow { public ushort PackingSize; public uint ClassSize; public int Parent; }
        private struct ConstantRow { public byte Type; public int Parent; public BlobHandle Value; }
        private struct CustomAttributeRow { public int Parent; public int Type; public BlobHandle Value; }
        private struct DeclSecurityRow { public ushort Action; public int Parent; public BlobHandle PermissionSet; }
        private struct EncLogRow { public int Token; public byte FuncCode; }
        private struct EncMapRow { public int Token; }
        private struct EventRow { public ushort EventFlags; public StringHandle Name; public int EventType; }
        private struct EventMapRow { public int Parent; public int EventList; }
        private struct ExportedTypeRow { public uint Flags; public int TypeDefId; public StringHandle TypeName; public StringHandle TypeNamespace; public int Implementation; }
        private struct FieldLayoutRow { public int Offset; public int Field; }
        private struct FieldMarshalRow { public int Parent; public BlobHandle NativeType; }
        private struct FieldRvaRow { public int Offset; public int Field; }
        private struct FieldDefRow { public ushort Flags; public StringHandle Name; public BlobHandle Signature; }
        private struct FileTableRow { public uint Flags; public StringHandle FileName; public BlobHandle HashValue; }
        private struct GenericParamConstraintRow { public int Owner; public int Constraint; }
        private struct GenericParamRow { public ushort Number; public ushort Flags; public int Owner; public StringHandle Name; }
        private struct ImplMapRow { public ushort MappingFlags; public int MemberForwarded; public StringHandle ImportName; public int ImportScope; }
        private struct InterfaceImplRow { public int Class; public int Interface; }
        private struct ManifestResourceRow { public uint Offset; public uint Flags; public StringHandle Name; public int Implementation; }
        private struct MemberRefRow { public int Class; public StringHandle Name; public BlobHandle Signature; }
        private struct MethodImplRow { public int Class; public int MethodBody; public int MethodDecl; }
        private struct MethodSemanticsRow { public ushort Semantic; public int Method; public int Association; }
        private struct MethodSpecRow { public int Method; public BlobHandle Instantiation; }
        private struct MethodRow { public int BodyOffset; public ushort ImplFlags; public ushort Flags; public StringHandle Name; public BlobHandle Signature; public int ParamList; }
        private struct ModuleRefRow { public StringHandle Name; }
        private struct NestedClassRow { public int NestedClass; public int EnclosingClass; }
        private struct ParamRow { public ushort Flags; public ushort Sequence; public StringHandle Name; }
        private struct PropertyMapRow { public int Parent; public int PropertyList; }
        private struct PropertyRow { public ushort PropFlags; public StringHandle Name; public BlobHandle Type; }
        private struct TypeDefRow { public uint Flags; public StringHandle Name; public StringHandle Namespace; public int Extends; public int FieldList; public int MethodList; }
        private struct TypeRefRow { public int ResolutionScope; public StringHandle Name; public StringHandle Namespace; }
        private struct TypeSpecRow { public BlobHandle Signature; }
        private struct StandaloneSigRow { public BlobHandle Signature; }
 
        // debug table rows:
        private struct DocumentRow { public BlobHandle Name; public GuidHandle HashAlgorithm; public BlobHandle Hash; public GuidHandle Language; }
        private struct MethodDebugInformationRow { public int Document; public BlobHandle SequencePoints; }
        private struct LocalScopeRow { public int Method; public int ImportScope; public int VariableList; public int ConstantList; public int StartOffset; public int Length; }
        private struct LocalVariableRow { public ushort Attributes; public ushort Index; public StringHandle Name; }
        private struct LocalConstantRow { public StringHandle Name; public BlobHandle Signature; }
        private struct ImportScopeRow { public int Parent; public BlobHandle Imports; }
        private struct StateMachineMethodRow { public int MoveNextMethod; public int KickoffMethod; }
        private struct CustomDebugInformationRow { public int Parent; public GuidHandle Kind; public BlobHandle Value; }
 
        // type system tables:
        private ModuleRow? _moduleRow;
        private AssemblyRow? _assemblyRow;
        private readonly List<ClassLayoutRow> _classLayoutTable = new List<ClassLayoutRow>();
 
        private readonly List<ConstantRow> _constantTable = new List<ConstantRow>();
        private int _constantTableLastParent;
        private bool _constantTableNeedsSorting;
 
        private readonly List<CustomAttributeRow> _customAttributeTable = new List<CustomAttributeRow>();
        private int _customAttributeTableLastParent;
        private bool _customAttributeTableNeedsSorting;
 
        private readonly List<DeclSecurityRow> _declSecurityTable = new List<DeclSecurityRow>();
        private int _declSecurityTableLastParent;
        private bool _declSecurityTableNeedsSorting;
 
        private readonly List<EncLogRow> _encLogTable = new List<EncLogRow>();
        private readonly List<EncMapRow> _encMapTable = new List<EncMapRow>();
        private readonly List<EventRow> _eventTable = new List<EventRow>();
        private readonly List<EventMapRow> _eventMapTable = new List<EventMapRow>();
        private readonly List<ExportedTypeRow> _exportedTypeTable = new List<ExportedTypeRow>();
        private readonly List<FieldLayoutRow> _fieldLayoutTable = new List<FieldLayoutRow>();
 
        private readonly List<FieldMarshalRow> _fieldMarshalTable = new List<FieldMarshalRow>();
        private int _fieldMarshalTableLastParent;
        private bool _fieldMarshalTableNeedsSorting;
 
        private readonly List<FieldRvaRow> _fieldRvaTable = new List<FieldRvaRow>();
        private readonly List<FieldDefRow> _fieldTable = new List<FieldDefRow>();
        private readonly List<FileTableRow> _fileTable = new List<FileTableRow>();
        private readonly List<GenericParamConstraintRow> _genericParamConstraintTable = new List<GenericParamConstraintRow>();
        private readonly List<GenericParamRow> _genericParamTable = new List<GenericParamRow>();
        private readonly List<ImplMapRow> _implMapTable = new List<ImplMapRow>();
        private readonly List<InterfaceImplRow> _interfaceImplTable = new List<InterfaceImplRow>();
        private readonly List<ManifestResourceRow> _manifestResourceTable = new List<ManifestResourceRow>();
        private readonly List<MemberRefRow> _memberRefTable = new List<MemberRefRow>();
        private readonly List<MethodImplRow> _methodImplTable = new List<MethodImplRow>();
 
        private readonly List<MethodSemanticsRow> _methodSemanticsTable = new List<MethodSemanticsRow>();
        private int _methodSemanticsTableLastAssociation;
        private bool _methodSemanticsTableNeedsSorting;
 
        private readonly List<MethodSpecRow> _methodSpecTable = new List<MethodSpecRow>();
        private readonly List<MethodRow> _methodDefTable = new List<MethodRow>();
        private readonly List<ModuleRefRow> _moduleRefTable = new List<ModuleRefRow>();
        private readonly List<NestedClassRow> _nestedClassTable = new List<NestedClassRow>();
        private readonly List<ParamRow> _paramTable = new List<ParamRow>();
        private readonly List<PropertyMapRow> _propertyMapTable = new List<PropertyMapRow>();
        private readonly List<PropertyRow> _propertyTable = new List<PropertyRow>();
        private readonly List<TypeDefRow> _typeDefTable = new List<TypeDefRow>();
        private readonly List<TypeRefRow> _typeRefTable = new List<TypeRefRow>();
        private readonly List<TypeSpecRow> _typeSpecTable = new List<TypeSpecRow>();
        private readonly List<AssemblyRefTableRow> _assemblyRefTable = new List<AssemblyRefTableRow>();
        private readonly List<StandaloneSigRow> _standAloneSigTable = new List<StandaloneSigRow>();
 
        // debug tables:
        private readonly List<DocumentRow> _documentTable = new List<DocumentRow>();
        private readonly List<MethodDebugInformationRow> _methodDebugInformationTable = new List<MethodDebugInformationRow>();
        private readonly List<LocalScopeRow> _localScopeTable = new List<LocalScopeRow>();
        private readonly List<LocalVariableRow> _localVariableTable = new List<LocalVariableRow>();
        private readonly List<LocalConstantRow> _localConstantTable = new List<LocalConstantRow>();
        private readonly List<ImportScopeRow> _importScopeTable = new List<ImportScopeRow>();
        private readonly List<StateMachineMethodRow> _stateMachineMethodTable = new List<StateMachineMethodRow>();
        private readonly List<CustomDebugInformationRow> _customDebugInformationTable = new List<CustomDebugInformationRow>();
 
        /// <summary>
        /// Sets the capacity of the specified table.
        /// </summary>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="table"/> is not a valid table index.</exception>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="rowCount"/> is negative.</exception>
        /// <remarks>
        /// Use to reduce allocations if the approximate number of rows is known ahead of time.
        /// </remarks>
        public void SetCapacity(TableIndex table, int rowCount)
        {
            if (rowCount < 0)
            {
                Throw.ArgumentOutOfRange(nameof(rowCount));
            }
 
            switch (table)
            {
                case TableIndex.Module:                 break; // no-op, max row count is 1
                case TableIndex.TypeRef:                SetTableCapacity(_typeRefTable, rowCount); break;
                case TableIndex.TypeDef:                SetTableCapacity(_typeDefTable, rowCount); break;
                case TableIndex.Field:                  SetTableCapacity(_fieldTable, rowCount); break;
                case TableIndex.MethodDef:              SetTableCapacity(_methodDefTable, rowCount); break;
                case TableIndex.Param:                  SetTableCapacity(_paramTable, rowCount); break;
                case TableIndex.InterfaceImpl:          SetTableCapacity(_interfaceImplTable, rowCount); break;
                case TableIndex.MemberRef:              SetTableCapacity(_memberRefTable, rowCount); break;
                case TableIndex.Constant:               SetTableCapacity(_constantTable, rowCount); break;
                case TableIndex.CustomAttribute:        SetTableCapacity(_customAttributeTable, rowCount); break;
                case TableIndex.FieldMarshal:           SetTableCapacity(_fieldMarshalTable, rowCount); break;
                case TableIndex.DeclSecurity:           SetTableCapacity(_declSecurityTable, rowCount); break;
                case TableIndex.ClassLayout:            SetTableCapacity(_classLayoutTable, rowCount); break;
                case TableIndex.FieldLayout:            SetTableCapacity(_fieldLayoutTable, rowCount); break;
                case TableIndex.StandAloneSig:          SetTableCapacity(_standAloneSigTable, rowCount); break;
                case TableIndex.EventMap:               SetTableCapacity(_eventMapTable, rowCount); break;
                case TableIndex.Event:                  SetTableCapacity(_eventTable, rowCount); break;
                case TableIndex.PropertyMap:            SetTableCapacity(_propertyMapTable, rowCount); break;
                case TableIndex.Property:               SetTableCapacity(_propertyTable, rowCount); break;
                case TableIndex.MethodSemantics:        SetTableCapacity(_methodSemanticsTable, rowCount); break;
                case TableIndex.MethodImpl:             SetTableCapacity(_methodImplTable, rowCount); break;
                case TableIndex.ModuleRef:              SetTableCapacity(_moduleRefTable, rowCount); break;
                case TableIndex.TypeSpec:               SetTableCapacity(_typeSpecTable, rowCount); break;
                case TableIndex.ImplMap:                SetTableCapacity(_implMapTable, rowCount); break;
                case TableIndex.FieldRva:               SetTableCapacity(_fieldRvaTable, rowCount); break;
                case TableIndex.EncLog:                 SetTableCapacity(_encLogTable, rowCount); break;
                case TableIndex.EncMap:                 SetTableCapacity(_encMapTable, rowCount); break;
                case TableIndex.Assembly:               break; // no-op, max row count is 1
                case TableIndex.AssemblyRef:            SetTableCapacity(_assemblyRefTable, rowCount); break;
                case TableIndex.File:                   SetTableCapacity(_fileTable, rowCount); break;
                case TableIndex.ExportedType:           SetTableCapacity(_exportedTypeTable, rowCount); break;
                case TableIndex.ManifestResource:       SetTableCapacity(_manifestResourceTable, rowCount); break;
                case TableIndex.NestedClass:            SetTableCapacity(_nestedClassTable, rowCount); break;
                case TableIndex.GenericParam:           SetTableCapacity(_genericParamTable, rowCount); break;
                case TableIndex.MethodSpec:             SetTableCapacity(_methodSpecTable, rowCount); break;
                case TableIndex.GenericParamConstraint: SetTableCapacity(_genericParamConstraintTable, rowCount); break;
                case TableIndex.Document:               SetTableCapacity(_documentTable, rowCount); break;
                case TableIndex.MethodDebugInformation: SetTableCapacity(_methodDebugInformationTable, rowCount); break;
                case TableIndex.LocalScope:             SetTableCapacity(_localScopeTable, rowCount); break;
                case TableIndex.LocalVariable:          SetTableCapacity(_localVariableTable, rowCount); break;
                case TableIndex.LocalConstant:          SetTableCapacity(_localConstantTable, rowCount); break;
                case TableIndex.ImportScope:            SetTableCapacity(_importScopeTable, rowCount); break;
                case TableIndex.StateMachineMethod:     SetTableCapacity(_stateMachineMethodTable, rowCount); break;
                case TableIndex.CustomDebugInformation: SetTableCapacity(_customDebugInformationTable, rowCount); break;
 
                case TableIndex.AssemblyOS:
                case TableIndex.AssemblyProcessor:
                case TableIndex.AssemblyRefOS:
                case TableIndex.AssemblyRefProcessor:
                case TableIndex.EventPtr:
                case TableIndex.FieldPtr:
                case TableIndex.MethodPtr:
                case TableIndex.ParamPtr:
                case TableIndex.PropertyPtr:
                    // these tables are currently not serialized
                    break;
 
                default:
                    throw new ArgumentOutOfRangeException(nameof(table));
            }
        }
 
        private static void SetTableCapacity<T>(List<T> table, int rowCount)
        {
            if (rowCount > table.Count)
            {
                table.Capacity = rowCount;
            }
        }
 
        /// <summary>
        /// Returns the current number of entries in the specified table.
        /// </summary>
        /// <param name="table">Table index.</param>
        /// <returns>The number of entries in the table.</returns>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="table"/> is not a valid table index.</exception>
        public int GetRowCount(TableIndex table)
        {
            switch (table)
            {
                case TableIndex.Assembly                : return _assemblyRow.HasValue ? 1 : 0;
                case TableIndex.AssemblyRef             : return _assemblyRefTable.Count;
                case TableIndex.ClassLayout             : return _classLayoutTable.Count;
                case TableIndex.Constant                : return _constantTable.Count;
                case TableIndex.CustomAttribute         : return _customAttributeTable.Count;
                case TableIndex.DeclSecurity            : return _declSecurityTable.Count;
                case TableIndex.EncLog                  : return _encLogTable.Count;
                case TableIndex.EncMap                  : return _encMapTable.Count;
                case TableIndex.EventMap                : return _eventMapTable.Count;
                case TableIndex.Event                   : return _eventTable.Count;
                case TableIndex.ExportedType            : return _exportedTypeTable.Count;
                case TableIndex.FieldLayout             : return _fieldLayoutTable.Count;
                case TableIndex.FieldMarshal            : return _fieldMarshalTable.Count;
                case TableIndex.FieldRva                : return _fieldRvaTable.Count;
                case TableIndex.Field                   : return _fieldTable.Count;
                case TableIndex.File                    : return _fileTable.Count;
                case TableIndex.GenericParamConstraint  : return _genericParamConstraintTable.Count;
                case TableIndex.GenericParam            : return _genericParamTable.Count;
                case TableIndex.ImplMap                 : return _implMapTable.Count;
                case TableIndex.InterfaceImpl           : return _interfaceImplTable.Count;
                case TableIndex.ManifestResource        : return _manifestResourceTable.Count;
                case TableIndex.MemberRef               : return _memberRefTable.Count;
                case TableIndex.MethodImpl              : return _methodImplTable.Count;
                case TableIndex.MethodSemantics         : return _methodSemanticsTable.Count;
                case TableIndex.MethodSpec              : return _methodSpecTable.Count;
                case TableIndex.MethodDef               : return _methodDefTable.Count;
                case TableIndex.ModuleRef               : return _moduleRefTable.Count;
                case TableIndex.Module                  : return _moduleRow.HasValue ? 1 : 0;
                case TableIndex.NestedClass             : return _nestedClassTable.Count;
                case TableIndex.Param                   : return _paramTable.Count;
                case TableIndex.PropertyMap             : return _propertyMapTable.Count;
                case TableIndex.Property                : return _propertyTable.Count;
                case TableIndex.StandAloneSig           : return _standAloneSigTable.Count;
                case TableIndex.TypeDef                 : return _typeDefTable.Count;
                case TableIndex.TypeRef                 : return _typeRefTable.Count;
                case TableIndex.TypeSpec                : return _typeSpecTable.Count;
                case TableIndex.Document                : return _documentTable.Count;
                case TableIndex.MethodDebugInformation  : return _methodDebugInformationTable.Count;
                case TableIndex.LocalScope              : return _localScopeTable.Count;
                case TableIndex.LocalVariable           : return _localVariableTable.Count;
                case TableIndex.LocalConstant           : return _localConstantTable.Count;
                case TableIndex.StateMachineMethod      : return _stateMachineMethodTable.Count;
                case TableIndex.ImportScope             : return _importScopeTable.Count;
                case TableIndex.CustomDebugInformation  : return _customDebugInformationTable.Count;
 
                case TableIndex.AssemblyOS:
                case TableIndex.AssemblyProcessor:
                case TableIndex.AssemblyRefOS:
                case TableIndex.AssemblyRefProcessor:
                case TableIndex.EventPtr:
                case TableIndex.FieldPtr:
                case TableIndex.MethodPtr:
                case TableIndex.ParamPtr:
                case TableIndex.PropertyPtr:
                    return 0;
 
                default:
                    throw new ArgumentOutOfRangeException(nameof(table));
            }
        }
 
        /// <summary>
        /// Returns the current number of entries in each table.
        /// </summary>
        /// <returns>
        /// An array of size <see cref="MetadataTokens.TableCount"/> with each item filled with the current row count of the corresponding table.
        /// </returns>
        public ImmutableArray<int> GetRowCounts()
        {
            var rowCounts = ImmutableArray.CreateBuilder<int>(MetadataTokens.TableCount);
            rowCounts.Count = MetadataTokens.TableCount;
 
            rowCounts[(int)TableIndex.Assembly] = _assemblyRow.HasValue ? 1 : 0;
            rowCounts[(int)TableIndex.AssemblyRef] = _assemblyRefTable.Count;
            rowCounts[(int)TableIndex.ClassLayout] = _classLayoutTable.Count;
            rowCounts[(int)TableIndex.Constant] = _constantTable.Count;
            rowCounts[(int)TableIndex.CustomAttribute] = _customAttributeTable.Count;
            rowCounts[(int)TableIndex.DeclSecurity] = _declSecurityTable.Count;
            rowCounts[(int)TableIndex.EncLog] = _encLogTable.Count;
            rowCounts[(int)TableIndex.EncMap] = _encMapTable.Count;
            rowCounts[(int)TableIndex.EventMap] = _eventMapTable.Count;
            rowCounts[(int)TableIndex.Event] = _eventTable.Count;
            rowCounts[(int)TableIndex.ExportedType] = _exportedTypeTable.Count;
            rowCounts[(int)TableIndex.FieldLayout] = _fieldLayoutTable.Count;
            rowCounts[(int)TableIndex.FieldMarshal] = _fieldMarshalTable.Count;
            rowCounts[(int)TableIndex.FieldRva] = _fieldRvaTable.Count;
            rowCounts[(int)TableIndex.Field] = _fieldTable.Count;
            rowCounts[(int)TableIndex.File] = _fileTable.Count;
            rowCounts[(int)TableIndex.GenericParamConstraint] = _genericParamConstraintTable.Count;
            rowCounts[(int)TableIndex.GenericParam] = _genericParamTable.Count;
            rowCounts[(int)TableIndex.ImplMap] = _implMapTable.Count;
            rowCounts[(int)TableIndex.InterfaceImpl] = _interfaceImplTable.Count;
            rowCounts[(int)TableIndex.ManifestResource] = _manifestResourceTable.Count;
            rowCounts[(int)TableIndex.MemberRef] = _memberRefTable.Count;
            rowCounts[(int)TableIndex.MethodImpl] = _methodImplTable.Count;
            rowCounts[(int)TableIndex.MethodSemantics] = _methodSemanticsTable.Count;
            rowCounts[(int)TableIndex.MethodSpec] = _methodSpecTable.Count;
            rowCounts[(int)TableIndex.MethodDef] = _methodDefTable.Count;
            rowCounts[(int)TableIndex.ModuleRef] = _moduleRefTable.Count;
            rowCounts[(int)TableIndex.Module] = _moduleRow.HasValue ? 1 : 0;
            rowCounts[(int)TableIndex.NestedClass] = _nestedClassTable.Count;
            rowCounts[(int)TableIndex.Param] = _paramTable.Count;
            rowCounts[(int)TableIndex.PropertyMap] = _propertyMapTable.Count;
            rowCounts[(int)TableIndex.Property] = _propertyTable.Count;
            rowCounts[(int)TableIndex.StandAloneSig] = _standAloneSigTable.Count;
            rowCounts[(int)TableIndex.TypeDef] = _typeDefTable.Count;
            rowCounts[(int)TableIndex.TypeRef] = _typeRefTable.Count;
            rowCounts[(int)TableIndex.TypeSpec] = _typeSpecTable.Count;
 
            rowCounts[(int)TableIndex.Document] = _documentTable.Count;
            rowCounts[(int)TableIndex.MethodDebugInformation] = _methodDebugInformationTable.Count;
            rowCounts[(int)TableIndex.LocalScope] = _localScopeTable.Count;
            rowCounts[(int)TableIndex.LocalVariable] = _localVariableTable.Count;
            rowCounts[(int)TableIndex.LocalConstant] = _localConstantTable.Count;
            rowCounts[(int)TableIndex.StateMachineMethod] = _stateMachineMethodTable.Count;
            rowCounts[(int)TableIndex.ImportScope] = _importScopeTable.Count;
            rowCounts[(int)TableIndex.CustomDebugInformation] = _customDebugInformationTable.Count;
 
            return rowCounts.MoveToImmutable();
        }
 
        #region Building
 
        // Note on argument value checking:
        //
        // To avoid perf overhead we do minimal argument checking.
        // We only check integer parameter bounds if we perform a narrowing conversion,
        // so that we don't mangle values that match the Add method signature but are too big to represent in the metadata.
        // We do not validate ranges of enum values, we just ignore high bits when converting to smaller integral type.
        // Adding such checks would unlikely catch bugs in reasonably written code.
 
        public ModuleDefinitionHandle AddModule(
            int generation,
            StringHandle moduleName,
            GuidHandle mvid,
            GuidHandle encId,
            GuidHandle encBaseId)
        {
            if (unchecked((uint)generation) > ushort.MaxValue)
            {
                Throw.ArgumentOutOfRange(nameof(generation));
            }
 
            if (_moduleRow.HasValue)
            {
                Throw.InvalidOperation(SR.ModuleAlreadyAdded);
            }
 
            _moduleRow = new ModuleRow
            {
                Generation = (ushort)generation,
                Name = moduleName,
                ModuleVersionId = mvid,
                EncId = encId,
                EncBaseId = encBaseId,
            };
 
            return EntityHandle.ModuleDefinition;
        }
 
        public AssemblyDefinitionHandle AddAssembly(
            StringHandle name,
            Version version,
            StringHandle culture,
            BlobHandle publicKey,
            AssemblyFlags flags,
            AssemblyHashAlgorithm hashAlgorithm)
        {
            if (version is null)
            {
                Throw.ArgumentNull(nameof(version));
            }
 
            if (_assemblyRow.HasValue)
            {
                Throw.InvalidOperation(SR.AssemblyAlreadyAdded);
            }
 
            _assemblyRow = new AssemblyRow
            {
                Flags = unchecked((ushort)flags),
                HashAlgorithm = unchecked((uint)hashAlgorithm),
                Version = version,
                AssemblyKey = publicKey,
                AssemblyName = name,
                AssemblyCulture = culture
            };
 
            return EntityHandle.AssemblyDefinition;
        }
 
        public AssemblyReferenceHandle AddAssemblyReference(
            StringHandle name,
            Version version,
            StringHandle culture,
            BlobHandle publicKeyOrToken,
            AssemblyFlags flags,
            BlobHandle hashValue)
        {
            if (version is null)
            {
                Throw.ArgumentNull(nameof(version));
            }
 
            _assemblyRefTable.Add(new AssemblyRefTableRow
            {
                Name = name,
                Version = version,
                Culture = culture,
                PublicKeyToken = publicKeyOrToken,
                Flags = unchecked((uint)flags),
                HashValue = hashValue
            });
 
            return AssemblyReferenceHandle.FromRowId(_assemblyRefTable.Count);
        }
 
        /// <summary>
        /// Adds a type definition.
        /// </summary>
        /// <param name="attributes">Attributes</param>
        /// <param name="namespace">Namespace</param>
        /// <param name="name">Type name</param>
        /// <param name="baseType"><see cref="TypeDefinitionHandle"/>, <see cref="TypeReferenceHandle"/>, <see cref="TypeSpecificationHandle"/> or nil.</param>
        /// <param name="fieldList">
        /// If the type declares fields the handle of the first one, otherwise the handle of the first field declared by the next type definition.
        /// If no type defines any fields in the module, <see cref="MetadataTokens.FieldDefinitionHandle(int)"/>(1).
        /// </param>
        /// <param name="methodList">
        /// If the type declares methods the handle of the first one, otherwise the handle of the first method declared by the next type definition.
        /// If no type defines any methods in the module, <see cref="MetadataTokens.MethodDefinitionHandle(int)"/>(1).
        /// </param>
        /// <exception cref="ArgumentException"><paramref name="baseType"/> doesn't have the expected handle kind.</exception>
        public TypeDefinitionHandle AddTypeDefinition(
            TypeAttributes attributes,
            StringHandle @namespace,
            StringHandle name,
            EntityHandle baseType,
            FieldDefinitionHandle fieldList,
            MethodDefinitionHandle methodList)
        {
            _typeDefTable.Add(new TypeDefRow
            {
                Flags = unchecked((uint)attributes),
                Name = name,
                Namespace = @namespace,
                Extends = baseType.IsNil ? 0 : CodedIndex.TypeDefOrRefOrSpec(baseType),
                FieldList = fieldList.RowId,
                MethodList = methodList.RowId
            });
 
            return TypeDefinitionHandle.FromRowId(_typeDefTable.Count);
        }
 
        /// <summary>
        /// Defines a type layout of a type definition.
        /// </summary>
        /// <param name="type">Type definition.</param>
        /// <param name="packingSize">
        /// Specifies that fields should be placed within the type instance at byte addresses which are a multiple of the value,
        /// or at natural alignment for that field type, whichever is smaller. Shall be one of the following: 0, 1, 2, 4, 8, 16, 32, 64, or 128.
        /// A value of zero indicates that the packing size used should match the default for the current platform.
        /// </param>
        /// <param name="size">
        /// Indicates a minimum size of the type instance, and is intended to allow for padding.
        /// The amount of memory allocated is the maximum of the size calculated from the layout and <paramref name="size"/>.
        /// Note that if this directive applies to a value type, then the size shall be less than 1 MB.
        /// </param>
        /// <remarks>
        /// Entries must be added in the same order as the corresponding type definitions.
        /// </remarks>
        public void AddTypeLayout(
            TypeDefinitionHandle type,
            ushort packingSize,
            uint size)
        {
            _classLayoutTable.Add(new ClassLayoutRow
            {
                Parent = type.RowId,
                PackingSize = packingSize,
                ClassSize = size
            });
        }
 
        /// <summary>
        /// Adds an interface implementation to a type.
        /// </summary>
        /// <param name="type">The type implementing the interface.</param>
        /// <param name="implementedInterface">
        /// The interface being implemented:
        /// <see cref="TypeDefinitionHandle"/>, <see cref="TypeReferenceHandle"/> or <see cref="TypeSpecificationHandle"/>.
        /// </param>
        /// <remarks>
        /// Interface implementations must be added in the same order as the corresponding type definitions implementing the interface.
        /// If a type implements multiple interfaces the corresponding entries must be added in the order determined by their coded indices (<see cref="CodedIndex.TypeDefOrRefOrSpec"/>).
        /// </remarks>
        /// <exception cref="ArgumentException"><paramref name="implementedInterface"/> doesn't have the expected handle kind.</exception>
        public InterfaceImplementationHandle AddInterfaceImplementation(
            TypeDefinitionHandle type,
            EntityHandle implementedInterface)
        {
            _interfaceImplTable.Add(new InterfaceImplRow
            {
                Class = type.RowId,
                Interface = CodedIndex.TypeDefOrRefOrSpec(implementedInterface)
            });
 
            return InterfaceImplementationHandle.FromRowId(_interfaceImplTable.Count);
        }
 
        /// <summary>
        /// Defines a nesting relationship to specified type definitions.
        /// </summary>
        /// <param name="type">The nested type definition handle.</param>
        /// <param name="enclosingType">The enclosing type definition handle.</param>
        /// <remarks>
        /// Entries must be added in the same order as the corresponding nested type definitions.
        /// </remarks>
        public void AddNestedType(
            TypeDefinitionHandle type,
            TypeDefinitionHandle enclosingType)
        {
            _nestedClassTable.Add(new NestedClassRow
            {
                NestedClass = type.RowId,
                EnclosingClass = enclosingType.RowId
            });
        }
 
        /// <summary>
        /// Add a type reference.
        /// </summary>
        /// <param name="resolutionScope">
        /// The entity declaring the target type:
        /// <see cref="ModuleDefinitionHandle"/>, <see cref="ModuleReferenceHandle"/>, <see cref="AssemblyReferenceHandle"/>, <see cref="TypeReferenceHandle"/>, or nil.
        /// </param>
        /// <param name="namespace">Namespace.</param>
        /// <param name="name">Type name.</param>
        /// <exception cref="ArgumentException"><paramref name="resolutionScope"/> doesn't have the expected handle kind.</exception>
        public TypeReferenceHandle AddTypeReference(
            EntityHandle resolutionScope,
            StringHandle @namespace,
            StringHandle name)
        {
            _typeRefTable.Add(new TypeRefRow
            {
                ResolutionScope = resolutionScope.IsNil ? 0 : CodedIndex.ResolutionScope(resolutionScope),
                Name = name,
                Namespace = @namespace
            });
 
            return TypeReferenceHandle.FromRowId(_typeRefTable.Count);
        }
 
        public TypeSpecificationHandle AddTypeSpecification(BlobHandle signature)
        {
            _typeSpecTable.Add(new TypeSpecRow
            {
                Signature = signature
            });
 
            return TypeSpecificationHandle.FromRowId(_typeSpecTable.Count);
        }
 
        public StandaloneSignatureHandle AddStandaloneSignature(BlobHandle signature)
        {
            _standAloneSigTable.Add(new StandaloneSigRow
            {
                Signature = signature
            });
 
            return StandaloneSignatureHandle.FromRowId(_standAloneSigTable.Count);
        }
 
        /// <summary>
        /// Adds a property definition.
        /// </summary>
        /// <param name="attributes">Attributes</param>
        /// <param name="name">Name</param>
        /// <param name="signature">Signature of the property.</param>
        public PropertyDefinitionHandle AddProperty(PropertyAttributes attributes, StringHandle name, BlobHandle signature)
        {
            _propertyTable.Add(new PropertyRow
            {
                PropFlags = unchecked((ushort)attributes),
                Name = name,
                Type = signature
            });
 
            return PropertyDefinitionHandle.FromRowId(_propertyTable.Count);
        }
 
        public void AddPropertyMap(TypeDefinitionHandle declaringType, PropertyDefinitionHandle propertyList)
        {
            _propertyMapTable.Add(new PropertyMapRow
            {
                Parent = declaringType.RowId,
                PropertyList = propertyList.RowId
            });
        }
 
        /// <summary>
        /// Adds an event definition.
        /// </summary>
        /// <param name="attributes">Attributes</param>
        /// <param name="name">Name</param>
        /// <param name="type">Type of the event: <see cref="TypeDefinitionHandle"/>, <see cref="TypeReferenceHandle"/>, or <see cref="TypeSpecificationHandle"/></param>
        /// <exception cref="ArgumentException"><paramref name="type"/> doesn't have the expected handle kind.</exception>
        public EventDefinitionHandle AddEvent(EventAttributes attributes, StringHandle name, EntityHandle type)
        {
            _eventTable.Add(new EventRow
            {
                EventFlags = unchecked((ushort)attributes),
                Name = name,
                EventType = CodedIndex.TypeDefOrRefOrSpec(type)
            });
 
            return EventDefinitionHandle.FromRowId(_eventTable.Count);
        }
 
        public void AddEventMap(TypeDefinitionHandle declaringType, EventDefinitionHandle eventList)
        {
            _eventMapTable.Add(new EventMapRow
            {
                Parent = declaringType.RowId,
                EventList = eventList.RowId
            });
        }
 
        /// <summary>
        /// Adds a default value for a parameter, field or property.
        /// </summary>
        /// <param name="parent"><see cref="ParameterHandle"/>, <see cref="FieldDefinitionHandle"/>, or <see cref="PropertyDefinitionHandle"/></param>
        /// <param name="value">The constant value.</param>
        /// <remarks>
        /// Entries may be added in any order. The table is automatically sorted when serialized.
        /// </remarks>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        public ConstantHandle AddConstant(EntityHandle parent, object? value)
        {
            int parentCodedIndex = CodedIndex.HasConstant(parent);
 
            // the table is required to be sorted by Parent:
            _constantTableNeedsSorting |= parentCodedIndex < _constantTableLastParent;
            _constantTableLastParent = parentCodedIndex;
 
            _constantTable.Add(new ConstantRow
            {
                Type = (byte)MetadataWriterUtilities.GetConstantTypeCode(value),
                Parent = parentCodedIndex,
                Value = GetOrAddConstantBlob(value)
            });
 
            return ConstantHandle.FromRowId(_constantTable.Count);
        }
 
        /// <summary>
        /// Associates a method (a getter, a setter, an adder, etc.) with a property or an event.
        /// </summary>
        /// <param name="association"><see cref="EventDefinitionHandle"/> or <see cref="PropertyDefinitionHandle"/>.</param>
        /// <param name="semantics">Semantics.</param>
        /// <param name="methodDefinition">Method definition.</param>
        /// <exception cref="ArgumentException"><paramref name="association"/> doesn't have the expected handle kind.</exception>
        /// <remarks>
        /// Entries may be added in any order. The table is automatically sorted when serialized.
        /// </remarks>
        public void AddMethodSemantics(EntityHandle association, MethodSemanticsAttributes semantics, MethodDefinitionHandle methodDefinition)
        {
            int associationCodedIndex = CodedIndex.HasSemantics(association);
 
            // the table is required to be sorted by Association:
            _methodSemanticsTableNeedsSorting |= associationCodedIndex < _methodSemanticsTableLastAssociation;
            _methodSemanticsTableLastAssociation = associationCodedIndex;
 
            _methodSemanticsTable.Add(new MethodSemanticsRow
            {
                Association = associationCodedIndex,
                Method = methodDefinition.RowId,
                Semantic = unchecked((ushort)semantics)
            });
        }
 
        /// <summary>
        /// Add a custom attribute.
        /// </summary>
        /// <param name="parent">
        /// An entity to attach the custom attribute to:
        /// <see cref="MethodDefinitionHandle"/>,
        /// <see cref="FieldDefinitionHandle"/>,
        /// <see cref="TypeReferenceHandle"/>,
        /// <see cref="TypeDefinitionHandle"/>,
        /// <see cref="ParameterHandle"/>,
        /// <see cref="InterfaceImplementationHandle"/>,
        /// <see cref="MemberReferenceHandle"/>,
        /// <see cref="ModuleDefinitionHandle"/>,
        /// <see cref="DeclarativeSecurityAttributeHandle"/>,
        /// <see cref="PropertyDefinitionHandle"/>,
        /// <see cref="EventDefinitionHandle"/>,
        /// <see cref="StandaloneSignatureHandle"/>,
        /// <see cref="ModuleReferenceHandle"/>,
        /// <see cref="TypeSpecificationHandle"/>,
        /// <see cref="AssemblyDefinitionHandle"/>,
        /// <see cref="AssemblyReferenceHandle"/>,
        /// <see cref="AssemblyFileHandle"/>,
        /// <see cref="ExportedTypeHandle"/>,
        /// <see cref="ManifestResourceHandle"/>,
        /// <see cref="GenericParameterHandle"/>,
        /// <see cref="GenericParameterConstraintHandle"/> or
        /// <see cref="MethodSpecificationHandle"/>.
        /// </param>
        /// <param name="constructor">
        /// Custom attribute constructor: <see cref="MethodDefinitionHandle"/> or <see cref="MemberReferenceHandle"/>
        /// </param>
        /// <param name="value">
        /// Custom attribute value blob.
        /// </param>
        /// <remarks>
        /// Entries may be added in any order. The table is automatically sorted when serialized.
        /// </remarks>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        public CustomAttributeHandle AddCustomAttribute(EntityHandle parent, EntityHandle constructor, BlobHandle value)
        {
            int parentCodedIndex = CodedIndex.HasCustomAttribute(parent);
 
            // the table is required to be sorted by Parent:
            _customAttributeTableNeedsSorting |= parentCodedIndex < _customAttributeTableLastParent;
            _customAttributeTableLastParent = parentCodedIndex;
 
            _customAttributeTable.Add(new CustomAttributeRow
            {
                Parent = parentCodedIndex,
                Type = CodedIndex.CustomAttributeType(constructor),
                Value = value
            });
 
            return CustomAttributeHandle.FromRowId(_customAttributeTable.Count);
        }
 
        /// <summary>
        /// Adds a method specification (instantiation).
        /// </summary>
        /// <param name="method">Generic method: <see cref="MethodDefinitionHandle"/> or <see cref="MemberReferenceHandle"/></param>
        /// <param name="instantiation">Instantiation blob encoding the generic arguments of the method.</param>
        /// <exception cref="ArgumentException"><paramref name="method"/> doesn't have the expected handle kind.</exception>
        public MethodSpecificationHandle AddMethodSpecification(EntityHandle method, BlobHandle instantiation)
        {
            _methodSpecTable.Add(new MethodSpecRow
            {
                Method = CodedIndex.MethodDefOrRef(method),
                Instantiation = instantiation
            });
 
            return MethodSpecificationHandle.FromRowId(_methodSpecTable.Count);
        }
 
        public ModuleReferenceHandle AddModuleReference(StringHandle moduleName)
        {
            _moduleRefTable.Add(new ModuleRefRow
            {
                Name = moduleName
            });
 
            return ModuleReferenceHandle.FromRowId(_moduleRefTable.Count);
        }
 
        /// <summary>
        /// Adds a parameter definition.
        /// </summary>
        /// <param name="attributes"><see cref="ParameterAttributes"/></param>
        /// <param name="name">Parameter name (optional).</param>
        /// <param name="sequenceNumber">Sequence number of the parameter. Value of 0 refers to the owner method's return type; its parameters are then numbered from 1 onwards.</param>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="sequenceNumber"/> is greater than <see cref="ushort.MaxValue"/>.</exception>
        public ParameterHandle AddParameter(ParameterAttributes attributes, StringHandle name, int sequenceNumber)
        {
            if (unchecked((uint)sequenceNumber) > ushort.MaxValue)
            {
                Throw.ArgumentOutOfRange(nameof(sequenceNumber));
            }
 
            _paramTable.Add(new ParamRow
            {
                Flags = unchecked((ushort)attributes),
                Name = name,
                Sequence = (ushort)sequenceNumber
            });
 
            return ParameterHandle.FromRowId(_paramTable.Count);
        }
 
        /// <summary>
        /// Adds a generic parameter definition.
        /// </summary>
        /// <param name="parent">Parent entity handle: <see cref="TypeDefinitionHandle"/> or <see cref="MethodDefinitionHandle"/></param>
        /// <param name="attributes">Attributes.</param>
        /// <param name="name">Parameter name.</param>
        /// <param name="index">Zero-based parameter index.</param>
        /// <remarks>
        /// Generic parameters must be added in an order determined by the coded index of their parent entity (<see cref="CodedIndex.TypeOrMethodDef"/>).
        /// Generic parameters with the same parent must be ordered by their <paramref name="index"/>.
        /// </remarks>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="index"/> is greater than <see cref="ushort.MaxValue"/>.</exception>
        public GenericParameterHandle AddGenericParameter(
            EntityHandle parent,
            GenericParameterAttributes attributes,
            StringHandle name,
            int index)
        {
            if (unchecked((uint)index) > ushort.MaxValue)
            {
                Throw.ArgumentOutOfRange(nameof(index));
            }
 
            _genericParamTable.Add(new GenericParamRow
            {
                Flags = unchecked((ushort)attributes),
                Name = name,
                Number = (ushort)index,
                Owner = CodedIndex.TypeOrMethodDef(parent)
            });
 
            return GenericParameterHandle.FromRowId(_genericParamTable.Count);
        }
 
        /// <summary>
        /// Adds a type constraint to a generic parameter.
        /// </summary>
        /// <param name="genericParameter">Generic parameter to constrain.</param>
        /// <param name="constraint">Type constraint: <see cref="TypeDefinitionHandle"/>, <see cref="TypeReferenceHandle"/> or <see cref="TypeSpecificationHandle"/></param>
        /// <exception cref="ArgumentException"><paramref name="genericParameter"/> doesn't have the expected handle kind.</exception>
        /// <remarks>
        /// Constraints must be added in the same order as the corresponding generic parameters.
        /// </remarks>
        public GenericParameterConstraintHandle AddGenericParameterConstraint(
            GenericParameterHandle genericParameter,
            EntityHandle constraint)
        {
            _genericParamConstraintTable.Add(new GenericParamConstraintRow
            {
                Owner = genericParameter.RowId,
                Constraint = CodedIndex.TypeDefOrRefOrSpec(constraint),
            });
 
            return GenericParameterConstraintHandle.FromRowId(_genericParamConstraintTable.Count);
        }
 
        /// <summary>
        /// Adds a field definition.
        /// </summary>
        /// <param name="attributes">Field attributes.</param>
        /// <param name="name">Field name.</param>
        /// <param name="signature">Field signature. Use <see cref="BlobEncoder.FieldSignature"/> to construct the blob.</param>
        public FieldDefinitionHandle AddFieldDefinition(
            FieldAttributes attributes,
            StringHandle name,
            BlobHandle signature)
        {
            _fieldTable.Add(new FieldDefRow
            {
                Flags = unchecked((ushort)attributes),
                Name = name,
                Signature = signature
            });
 
            return FieldDefinitionHandle.FromRowId(_fieldTable.Count);
        }
 
        /// <summary>
        /// Defines a field layout of a field definition.
        /// </summary>
        /// <param name="field">Field definition.</param>
        /// <param name="offset">The byte offset of the field within the declaring type instance.</param>
        /// <remarks>
        /// Entries must be added in the same order as the corresponding field definitions.
        /// </remarks>
        public void AddFieldLayout(
            FieldDefinitionHandle field,
            int offset)
        {
            _fieldLayoutTable.Add(new FieldLayoutRow
            {
                Field = field.RowId,
                Offset = offset
            });
        }
 
        /// <summary>
        /// Add marshalling information to a field or a parameter.
        /// </summary>
        /// <param name="parent"><see cref="ParameterHandle"/> or <see cref="FieldDefinitionHandle"/>.</param>
        /// <param name="descriptor">Descriptor blob.</param>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        /// <remarks>
        /// Entries may be added in any order. The table is automatically sorted when serialized.
        /// </remarks>
        public void AddMarshallingDescriptor(
            EntityHandle parent,
            BlobHandle descriptor)
        {
            int codedIndex = CodedIndex.HasFieldMarshal(parent);
 
            // the table is required to be sorted by Parent:
            _fieldMarshalTableNeedsSorting |= codedIndex < _fieldMarshalTableLastParent;
            _fieldMarshalTableLastParent = codedIndex;
 
            _fieldMarshalTable.Add(new FieldMarshalRow
            {
                Parent = codedIndex,
                NativeType = descriptor
            });
        }
 
        /// <summary>
        /// Adds a mapping from a field to its initial value stored in the PE image.
        /// </summary>
        /// <param name="field">Field definition handle.</param>
        /// <param name="offset">
        /// Offset within the block in the PE image that stores initial values of mapped fields (usually in .text section).
        /// The final relative virtual address stored in the metadata is calculated when the metadata is serialized
        /// by adding the offset to the virtual address of the block start.
        /// </param>
        /// <remarks>
        /// Entries must be added in the same order as the corresponding field definitions.
        /// </remarks>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="offset"/> is negative.</exception>
        public void AddFieldRelativeVirtualAddress(FieldDefinitionHandle field, int offset)
        {
            if (offset < 0)
            {
                Throw.ArgumentOutOfRange(nameof(offset));
            }
 
            _fieldRvaTable.Add(new FieldRvaRow
            {
                Field = field.RowId,
                Offset = offset
            });
        }
 
        /// <summary>
        /// Adds a method definition.
        /// </summary>
        /// <param name="attributes"><see cref="MethodAttributes"/></param>
        /// <param name="implAttributes"><see cref="MethodImplAttributes"/></param>
        /// <param name="name">Method name/</param>
        /// <param name="signature">Method signature.</param>
        /// <param name="bodyOffset">
        /// Offset within the block in the PE image that stores method bodies (IL stream),
        /// or -1 if the method doesn't have a body.
        ///
        /// The final relative virtual address stored in the metadata is calculated when the metadata is serialized
        /// by adding the offset to the virtual address of the beginning of the block.
        /// </param>
        /// <param name="parameterList">
        /// If the method declares parameters in Params table the handle of the first one, otherwise the handle of the first parameter declared by the next method definition.
        /// If no parameters are declared in the module, <see cref="MetadataTokens.ParameterHandle(int)"/>(1).
        /// </param>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="bodyOffset"/> is less than -1.</exception>
        public MethodDefinitionHandle AddMethodDefinition(
            MethodAttributes attributes,
            MethodImplAttributes implAttributes,
            StringHandle name,
            BlobHandle signature,
            int bodyOffset,
            ParameterHandle parameterList)
        {
            if (bodyOffset < -1)
            {
                Throw.ArgumentOutOfRange(nameof(bodyOffset));
            }
 
            _methodDefTable.Add(new MethodRow
            {
                Flags = unchecked((ushort)attributes),
                ImplFlags = unchecked((ushort)implAttributes),
                Name = name,
                Signature = signature,
                BodyOffset = bodyOffset,
                ParamList = parameterList.RowId
            });
 
            return MethodDefinitionHandle.FromRowId(_methodDefTable.Count);
        }
 
        /// <summary>
        /// Adds import information to a method definition (P/Invoke).
        /// </summary>
        /// <param name="method">Method definition handle.</param>
        /// <param name="attributes">Attributes.</param>
        /// <param name="name">Unmanaged method name.</param>
        /// <param name="module">Module containing the unmanaged method.</param>
        /// <remarks>
        /// Method imports must be added in the same order as the corresponding method definitions.
        /// </remarks>
        public void AddMethodImport(
            MethodDefinitionHandle method,
            MethodImportAttributes attributes,
            StringHandle name,
            ModuleReferenceHandle module)
        {
            _implMapTable.Add(new ImplMapRow
            {
                MemberForwarded = CodedIndex.MemberForwarded(method),
                ImportName = name,
                ImportScope = module.RowId,
                MappingFlags = unchecked((ushort)attributes),
            });
        }
 
        /// <summary>
        /// Defines an implementation for a method declaration within a type.
        /// </summary>
        /// <param name="type">Type</param>
        /// <param name="methodBody"><see cref="MethodDefinitionHandle"/> or <see cref="MemberReferenceHandle"/> which provides the implementation.</param>
        /// <param name="methodDeclaration"><see cref="MethodDefinitionHandle"/> or <see cref="MemberReferenceHandle"/> the method being implemented.</param>
        /// <remarks>
        /// Method implementations must be added in the same order as the corresponding type definitions.
        /// </remarks>
        /// <exception cref="ArgumentException"><paramref name="methodBody"/> or <paramref name="methodDeclaration"/> doesn't have the expected handle kind.</exception>
        public MethodImplementationHandle AddMethodImplementation(
            TypeDefinitionHandle type,
            EntityHandle methodBody,
            EntityHandle methodDeclaration)
        {
            _methodImplTable.Add(new MethodImplRow
            {
                Class = type.RowId,
                MethodBody = CodedIndex.MethodDefOrRef(methodBody),
                MethodDecl = CodedIndex.MethodDefOrRef(methodDeclaration)
            });
 
            return MethodImplementationHandle.FromRowId(_methodImplTable.Count);
        }
 
        /// <summary>
        /// Adds a MemberRef table row.
        /// </summary>
        /// <param name="parent">Containing entity:
        /// <see cref="TypeDefinitionHandle"/>,
        /// <see cref="TypeReferenceHandle"/>,
        /// <see cref="ModuleReferenceHandle"/>,
        /// <see cref="MethodDefinitionHandle"/>, or
        /// <see cref="TypeSpecificationHandle"/>.
        /// </param>
        /// <param name="name">Member name.</param>
        /// <param name="signature">Member signature.</param>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        public MemberReferenceHandle AddMemberReference(
            EntityHandle parent,
            StringHandle name,
            BlobHandle signature)
        {
            _memberRefTable.Add(new MemberRefRow
            {
                Class = CodedIndex.MemberRefParent(parent),
                Name = name,
                Signature = signature
            });
 
            return MemberReferenceHandle.FromRowId(_memberRefTable.Count);
        }
 
        /// <summary>
        /// Adds a manifest resource.
        /// </summary>
        /// <param name="attributes">Attributes</param>
        /// <param name="name">Resource name</param>
        /// <param name="implementation"><see cref="AssemblyFileHandle"/>, <see cref="AssemblyReferenceHandle"/>, or nil</param>
        /// <param name="offset">Specifies the byte offset within the referenced file at which this resource record begins.</param>
        /// <exception cref="ArgumentException"><paramref name="implementation"/> doesn't have the expected handle kind.</exception>
        public ManifestResourceHandle AddManifestResource(
            ManifestResourceAttributes attributes,
            StringHandle name,
            EntityHandle implementation,
            uint offset)
        {
            _manifestResourceTable.Add(new ManifestResourceRow
            {
                Flags = unchecked((uint)attributes),
                Name = name,
                Implementation = implementation.IsNil ? 0 : CodedIndex.Implementation(implementation),
                Offset = offset
            });
 
            return ManifestResourceHandle.FromRowId(_manifestResourceTable.Count);
        }
 
        public AssemblyFileHandle AddAssemblyFile(
            StringHandle name,
            BlobHandle hashValue,
            bool containsMetadata)
        {
            _fileTable.Add(new FileTableRow
            {
                FileName = name,
                Flags = containsMetadata ? 0u : 1u,
                HashValue = hashValue
            });
 
            return AssemblyFileHandle.FromRowId(_fileTable.Count);
        }
 
        /// <summary>
        /// Adds an exported type.
        /// </summary>
        /// <param name="attributes">Attributes</param>
        /// <param name="namespace">Namespace</param>
        /// <param name="name">Type name</param>
        /// <param name="implementation"><see cref="AssemblyFileHandle"/>, <see cref="ExportedTypeHandle"/> or <see cref="AssemblyReferenceHandle"/></param>
        /// <param name="typeDefinitionId">Type definition id</param>
        /// <exception cref="ArgumentException"><paramref name="implementation"/> doesn't have the expected handle kind.</exception>
        public ExportedTypeHandle AddExportedType(
            TypeAttributes attributes,
            StringHandle @namespace,
            StringHandle name,
            EntityHandle implementation,
            int typeDefinitionId)
        {
            _exportedTypeTable.Add(new ExportedTypeRow
            {
                Flags = unchecked((uint)attributes),
                Implementation = CodedIndex.Implementation(implementation),
                TypeNamespace = @namespace,
                TypeName = name,
                TypeDefId = typeDefinitionId
            });
 
            return ExportedTypeHandle.FromRowId(_exportedTypeTable.Count);
        }
 
        /// <summary>
        /// Adds declarative security attribute to a type, method or an assembly.
        /// </summary>
        /// <param name="parent"><see cref="TypeDefinitionHandle"/>, <see cref="MethodDefinitionHandle"/>, or <see cref="AssemblyDefinitionHandle"/></param>
        /// <param name="action">Security action</param>
        /// <param name="permissionSet">Permission set blob.</param>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        /// <remarks>
        /// Entries may be added in any order. The table is automatically sorted when serialized.
        /// </remarks>
        public DeclarativeSecurityAttributeHandle AddDeclarativeSecurityAttribute(
            EntityHandle parent,
            DeclarativeSecurityAction action,
            BlobHandle permissionSet)
        {
            int parentCodedIndex = CodedIndex.HasDeclSecurity(parent);
 
            // the table is required to be sorted by Parent:
            _declSecurityTableNeedsSorting |= parentCodedIndex < _declSecurityTableLastParent;
            _declSecurityTableLastParent = parentCodedIndex;
 
            _declSecurityTable.Add(new DeclSecurityRow
            {
                Parent = parentCodedIndex,
                Action = unchecked((ushort)action),
                PermissionSet = permissionSet
            });
 
            return DeclarativeSecurityAttributeHandle.FromRowId(_declSecurityTable.Count);
        }
 
        public void AddEncLogEntry(EntityHandle entity, EditAndContinueOperation code)
        {
            _encLogTable.Add(new EncLogRow
            {
                Token = entity.Token,
                FuncCode = unchecked((byte)code)
            });
        }
 
        public void AddEncMapEntry(EntityHandle entity)
        {
            _encMapTable.Add(new EncMapRow
            {
                Token = entity.Token
            });
        }
 
        /// <summary>
        /// Add document debug information.
        /// </summary>
        /// <param name="name">
        /// Document Name blob.
        /// </param>
        /// <param name="hashAlgorithm">
        /// GUID of the hash algorithm used to calculate the value of <paramref name="hash"/>.
        /// </param>
        /// <param name="hash">
        /// The hash of the document content.
        /// </param>
        /// <param name="language">
        /// GUID of the language.
        /// </param>
        /// See https://github.com/dotnet/runtime/blob/main/src/libraries/System.Reflection.Metadata/specs/PortablePdb-Metadata.md
        public DocumentHandle AddDocument(BlobHandle name, GuidHandle hashAlgorithm, BlobHandle hash, GuidHandle language)
        {
            _documentTable.Add(new DocumentRow
            {
                Name = name,
                HashAlgorithm = hashAlgorithm,
                Hash = hash,
                Language = language
            });
 
            return DocumentHandle.FromRowId(_documentTable.Count);
        }
 
        /// <summary>
        /// Add method debug information.
        /// </summary>
        /// <param name="document">
        /// The handle of a single document containing all sequence points of the method, or nil if the method doesn't have sequence points or spans multiple documents.
        /// </param>
        /// <param name="sequencePoints">
        /// Sequence Points blob, or nil if the method doesn't have sequence points.
        /// See https://github.com/dotnet/runtime/blob/main/src/libraries/System.Reflection.Metadata/specs/PortablePdb-Metadata.md#sequence-points-blob.
        /// </param>
        public MethodDebugInformationHandle AddMethodDebugInformation(DocumentHandle document, BlobHandle sequencePoints)
        {
            _methodDebugInformationTable.Add(new MethodDebugInformationRow
            {
                Document = document.RowId,
                SequencePoints = sequencePoints
            });
 
            return MethodDebugInformationHandle.FromRowId(_methodDebugInformationTable.Count);
        }
 
        /// <summary>
        /// Add local scope debug information.
        /// </summary>
        /// <param name="method">The containing method.</param>
        /// <param name="importScope">Handle of the associated import scope.</param>
        /// <param name="variableList">
        /// If the scope declares variables the handle of the first one, otherwise the handle of the first variable declared by the next scope definition.
        /// If no scope defines any variables, <see cref="MetadataTokens.LocalVariableHandle(int)"/>(1).
        /// </param>
        /// <param name="constantList">
        /// If the scope declares constants the handle of the first one, otherwise the handle of the first constant declared by the next scope definition.
        /// If no scope defines any constants, <see cref="MetadataTokens.LocalConstantHandle(int)"/>(1).
        /// </param>
        /// <param name="startOffset">Offset of the first instruction covered by the scope.</param>
        /// <param name="length">The length (in bytes) of the scope.</param>
        /// <remarks>
        /// Local scopes should be added in the same order as the corresponding method definition.
        /// Within a method they should be ordered by ascending <paramref name="startOffset"/> and then by descending <paramref name="length"/>.
        /// </remarks>
        public LocalScopeHandle AddLocalScope(MethodDefinitionHandle method, ImportScopeHandle importScope, LocalVariableHandle variableList, LocalConstantHandle constantList, int startOffset, int length)
        {
            _localScopeTable.Add(new LocalScopeRow
            {
                Method = method.RowId,
                ImportScope = importScope.RowId,
                VariableList = variableList.RowId,
                ConstantList = constantList.RowId,
                StartOffset = startOffset,
                Length = length
            });
 
            return LocalScopeHandle.FromRowId(_localScopeTable.Count);
        }
 
        /// <summary>
        /// Add local variable debug information.
        /// </summary>
        /// <param name="attributes"><see cref="LocalVariableAttributes"/></param>
        /// <param name="index">Local variable index in the local signature (zero-based).</param>
        /// <param name="name">Name of the variable.</param>
        /// <exception cref="ArgumentOutOfRangeException"><paramref name="index"/> is greater than <see cref="ushort.MaxValue"/>.</exception>
        public LocalVariableHandle AddLocalVariable(LocalVariableAttributes attributes, int index, StringHandle name)
        {
            if (unchecked((uint)index) > ushort.MaxValue)
            {
                Throw.ArgumentOutOfRange(nameof(index));
            }
 
            _localVariableTable.Add(new LocalVariableRow
            {
                Attributes = unchecked((ushort)attributes),
                Index = (ushort)index,
                Name = name
            });
 
            return LocalVariableHandle.FromRowId(_localVariableTable.Count);
        }
 
        /// <summary>
        /// Add local constant debug information.
        /// </summary>
        /// <param name="name">Name of the variable.</param>
        /// <param name="signature">
        /// LocalConstantSig blob, see https://github.com/dotnet/runtime/blob/main/src/libraries/System.Reflection.Metadata/specs/PortablePdb-Metadata.md#localconstantsig-blob.
        /// </param>
        public LocalConstantHandle AddLocalConstant(StringHandle name, BlobHandle signature)
        {
            _localConstantTable.Add(new LocalConstantRow
            {
                Name = name,
                Signature = signature
            });
 
            return LocalConstantHandle.FromRowId(_localConstantTable.Count);
        }
 
        /// <summary>
        /// Add local scope debug information.
        /// </summary>
        /// <param name="parentScope">Parent scope handle.</param>
        /// <param name="imports">
        /// Imports blob, see https://github.com/dotnet/runtime/blob/main/src/libraries/System.Reflection.Metadata/specs/PortablePdb-Metadata.md#imports-blob.
        /// </param>
        public ImportScopeHandle AddImportScope(ImportScopeHandle parentScope, BlobHandle imports)
        {
            _importScopeTable.Add(new ImportScopeRow
            {
                Parent = parentScope.RowId,
                Imports = imports
            });
 
            return ImportScopeHandle.FromRowId(_importScopeTable.Count);
        }
 
        /// <summary>
        /// Add state machine method debug information.
        /// </summary>
        /// <param name="moveNextMethod">Handle of the MoveNext method of the state machine (the compiler-generated method).</param>
        /// <param name="kickoffMethod">Handle of the kickoff method (the user defined iterator/async method)</param>
        /// <remarks>
        /// Entries should be added in the same order as the corresponding MoveNext method definitions.
        /// </remarks>
        public void AddStateMachineMethod(MethodDefinitionHandle moveNextMethod, MethodDefinitionHandle kickoffMethod)
        {
            _stateMachineMethodTable.Add(new StateMachineMethodRow
            {
                MoveNextMethod = moveNextMethod.RowId,
                KickoffMethod = kickoffMethod.RowId
            });
        }
 
        /// <summary>
        /// Add custom debug information.
        /// </summary>
        /// <param name="parent">
        /// An entity to attach the debug information to:
        /// <see cref="MethodDefinitionHandle"/>,
        /// <see cref="FieldDefinitionHandle"/>,
        /// <see cref="TypeReferenceHandle"/>,
        /// <see cref="TypeDefinitionHandle"/>,
        /// <see cref="ParameterHandle"/>,
        /// <see cref="InterfaceImplementationHandle"/>,
        /// <see cref="MemberReferenceHandle"/>,
        /// <see cref="ModuleDefinitionHandle"/>,
        /// <see cref="DeclarativeSecurityAttributeHandle"/>,
        /// <see cref="PropertyDefinitionHandle"/>,
        /// <see cref="EventDefinitionHandle"/>,
        /// <see cref="StandaloneSignatureHandle"/>,
        /// <see cref="ModuleReferenceHandle"/>,
        /// <see cref="TypeSpecificationHandle"/>,
        /// <see cref="AssemblyDefinitionHandle"/>,
        /// <see cref="AssemblyReferenceHandle"/>,
        /// <see cref="AssemblyFileHandle"/>,
        /// <see cref="ExportedTypeHandle"/>,
        /// <see cref="ManifestResourceHandle"/>,
        /// <see cref="GenericParameterHandle"/>,
        /// <see cref="GenericParameterConstraintHandle"/>,
        /// <see cref="MethodSpecificationHandle"/>,
        /// <see cref="DocumentHandle"/>,
        /// <see cref="LocalScopeHandle"/>,
        /// <see cref="LocalVariableHandle"/>,
        /// <see cref="LocalConstantHandle"/> or
        /// <see cref="ImportScopeHandle"/>.
        /// </param>
        /// <param name="kind">Information kind. Determines the structure of the <paramref name="value"/> blob.</param>
        /// <param name="value">Custom debug information blob.</param>
        /// <exception cref="ArgumentException"><paramref name="parent"/> doesn't have the expected handle kind.</exception>
        /// <remarks>
        /// Entries may be added in any order. The table is automatically sorted when serialized.
        /// </remarks>
        public CustomDebugInformationHandle AddCustomDebugInformation(EntityHandle parent, GuidHandle kind, BlobHandle value)
        {
            _customDebugInformationTable.Add(new CustomDebugInformationRow
            {
                Parent = CodedIndex.HasCustomDebugInformation(parent),
                Kind = kind,
                Value = value
            });
 
            return CustomDebugInformationHandle.FromRowId(_customDebugInformationTable.Count);
        }
 
        #endregion
 
        #region Validation
 
        internal void ValidateOrder()
        {
            // Certain tables are required to be sorted by a primary key, as follows:
            //
            // Table                    Keys                                Auto-ordered
            // --------------------------------------------------------------------------
            // ClassLayout              Parent                              No*
            // Constant                 Parent                              Yes
            // CustomAttribute          Parent                              Yes
            // DeclSecurity             Parent                              Yes
            // FieldLayout              Field                               No*
            // FieldMarshal             Parent                              Yes
            // FieldRVA                 Field                               No*
            // GenericParam             Owner, Number                       No**
            // GenericParamConstraint   Owner                               No**
            // ImplMap                  MemberForwarded                     No*
            // InterfaceImpl            Class                               No**
            // MethodImpl               Class                               No*
            // MethodSemantics          Association                         Yes
            // NestedClass              NestedClass                         No*
            // LocalScope               Method, StartOffset, Length (desc)  No**
            // StateMachineMethod       MoveNextMethod                      No*
            // CustomDebugInformation   Parent                              Yes
            //
            // Tables of entities that can't be referenced from other tables or blobs
            // are automatically ordered during serialization and thus don't require validation.
            //
            // * We could potentially auto-order these. These tables are adding extra (optional)
            // information to a primary entity (TypeDef, FiledDef, etc.) and are thus easily emitted
            // in the same order as the parent entity. Hence they would usually be ordered already and
            // it would be extra overhead to order them. Let's just required them ordered.
            //
            // ** We can't easily automatically order these since they represent entities that can be referenced
            // by other tables/blobs (e.g. CustomAttribute and CustomDebugInformation). Ordering these tables
            // would require updating all references.
 
            ValidateClassLayoutTable();
            ValidateFieldLayoutTable();
            ValidateFieldRvaTable();
            ValidateGenericParamTable();
            ValidateGenericParamConstaintTable();
            ValidateImplMapTable();
            ValidateInterfaceImplTable();
            ValidateMethodImplTable();
            ValidateNestedClassTable();
            ValidateLocalScopeTable();
            ValidateStateMachineMethodTable();
        }
 
        private void ValidateClassLayoutTable()
        {
            for (int i = 1; i < _classLayoutTable.Count; i++)
            {
                if (_classLayoutTable[i - 1].Parent >= _classLayoutTable[i].Parent)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.ClassLayout);
                }
            }
        }
 
        private void ValidateFieldLayoutTable()
        {
            for (int i = 1; i < _fieldLayoutTable.Count; i++)
            {
                if (_fieldLayoutTable[i - 1].Field >= _fieldLayoutTable[i].Field)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.FieldLayout);
                }
            }
        }
 
        private void ValidateFieldRvaTable()
        {
            for (int i = 1; i < _fieldRvaTable.Count; i++)
            {
                // Spec: each row in the FieldRVA table is an extension to exactly one row in the Field table
                if (_fieldRvaTable[i - 1].Field >= _fieldRvaTable[i].Field)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.FieldRva);
                }
            }
        }
 
        private void ValidateGenericParamTable()
        {
            if (_genericParamTable.Count == 0)
            {
                return;
            }
 
            GenericParamRow current, previous = _genericParamTable[0];
            for (int i = 1; i < _genericParamTable.Count; i++, previous = current)
            {
                current = _genericParamTable[i];
 
                if (current.Owner > previous.Owner)
                {
                    continue;
                }
 
                if (previous.Owner == current.Owner && current.Number > previous.Number)
                {
                    continue;
                }
 
                Throw.InvalidOperation_TableNotSorted(TableIndex.GenericParam);
            }
        }
 
        private void ValidateGenericParamConstaintTable()
        {
            for (int i = 1; i < _genericParamConstraintTable.Count; i++)
            {
                if (_genericParamConstraintTable[i - 1].Owner > _genericParamConstraintTable[i].Owner)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.GenericParamConstraint);
                }
            }
        }
 
        private void ValidateImplMapTable()
        {
            for (int i = 1; i < _implMapTable.Count; i++)
            {
                if (_implMapTable[i - 1].MemberForwarded >= _implMapTable[i].MemberForwarded)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.ImplMap);
                }
            }
        }
 
        private void ValidateInterfaceImplTable()
        {
            for (int i = 1; i < _interfaceImplTable.Count; i++)
            {
                if (_interfaceImplTable[i - 1].Class > _interfaceImplTable[i].Class)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.InterfaceImpl);
                }
            }
        }
 
        private void ValidateMethodImplTable()
        {
            for (int i = 1; i < _methodImplTable.Count; i++)
            {
                if (_methodImplTable[i - 1].Class > _methodImplTable[i].Class)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.MethodImpl);
                }
            }
        }
 
        private void ValidateNestedClassTable()
        {
            for (int i = 1; i < _nestedClassTable.Count; i++)
            {
                if (_nestedClassTable[i - 1].NestedClass >= _nestedClassTable[i].NestedClass)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.NestedClass);
                }
            }
        }
 
        private void ValidateLocalScopeTable()
        {
            if (_localScopeTable.Count == 0)
            {
                return;
            }
 
            // Spec: The table is required to be sorted first by Method in ascending order,
            // then by StartOffset in ascending order, then by Length in descending order.
            LocalScopeRow current, previous = _localScopeTable[0];
            for (int i = 1; i < _localScopeTable.Count; i++, previous = current)
            {
                current = _localScopeTable[i];
 
                if (current.Method > previous.Method)
                {
                    continue;
                }
 
                if (current.Method == previous.Method)
                {
                    if (current.StartOffset > previous.StartOffset)
                    {
                        continue;
                    }
 
                    if (current.StartOffset == previous.StartOffset && previous.Length >= current.Length)
                    {
                        continue;
                    }
                }
 
                Throw.InvalidOperation_TableNotSorted(TableIndex.LocalScope);
            }
        }
 
        private void ValidateStateMachineMethodTable()
        {
            for (int i = 1; i < _stateMachineMethodTable.Count; i++)
            {
                if (_stateMachineMethodTable[i - 1].MoveNextMethod >= _stateMachineMethodTable[i].MoveNextMethod)
                {
                    Throw.InvalidOperation_TableNotSorted(TableIndex.StateMachineMethod);
                }
            }
        }
 
        #endregion
 
        #region Serialization
 
        internal void SerializeMetadataTables(
            BlobBuilder writer,
            MetadataSizes metadataSizes,
            ImmutableArray<int> stringMap,
            int methodBodyStreamRva,
            int mappedFieldDataStreamRva)
        {
            int startPosition = writer.Count;
 
            SerializeTablesHeader(writer, metadataSizes);
 
            if (metadataSizes.IsPresent(TableIndex.Module))
            {
                SerializeModuleTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.TypeRef))
            {
                SerializeTypeRefTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.TypeDef))
            {
                SerializeTypeDefTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.Field))
            {
                SerializeFieldTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.MethodDef))
            {
                SerializeMethodDefTable(writer, stringMap, metadataSizes, methodBodyStreamRva);
            }
 
            if (metadataSizes.IsPresent(TableIndex.Param))
            {
                SerializeParamTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.InterfaceImpl))
            {
                SerializeInterfaceImplTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.MemberRef))
            {
                SerializeMemberRefTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.Constant))
            {
                SerializeConstantTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.CustomAttribute))
            {
                SerializeCustomAttributeTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.FieldMarshal))
            {
                SerializeFieldMarshalTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.DeclSecurity))
            {
                SerializeDeclSecurityTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.ClassLayout))
            {
                SerializeClassLayoutTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.FieldLayout))
            {
                SerializeFieldLayoutTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.StandAloneSig))
            {
                SerializeStandAloneSigTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.EventMap))
            {
                SerializeEventMapTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.Event))
            {
                SerializeEventTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.PropertyMap))
            {
                SerializePropertyMapTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.Property))
            {
                SerializePropertyTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.MethodSemantics))
            {
                SerializeMethodSemanticsTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.MethodImpl))
            {
                SerializeMethodImplTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.ModuleRef))
            {
                SerializeModuleRefTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.TypeSpec))
            {
                SerializeTypeSpecTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.ImplMap))
            {
                SerializeImplMapTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.FieldRva))
            {
                SerializeFieldRvaTable(writer, metadataSizes, mappedFieldDataStreamRva);
            }
 
            if (metadataSizes.IsPresent(TableIndex.EncLog))
            {
                SerializeEncLogTable(writer);
            }
 
            if (metadataSizes.IsPresent(TableIndex.EncMap))
            {
                SerializeEncMapTable(writer);
            }
 
            if (metadataSizes.IsPresent(TableIndex.Assembly))
            {
                SerializeAssemblyTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.AssemblyRef))
            {
                SerializeAssemblyRefTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.File))
            {
                SerializeFileTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.ExportedType))
            {
                SerializeExportedTypeTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.ManifestResource))
            {
                SerializeManifestResourceTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.NestedClass))
            {
                SerializeNestedClassTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.GenericParam))
            {
                SerializeGenericParamTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.MethodSpec))
            {
                SerializeMethodSpecTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.GenericParamConstraint))
            {
                SerializeGenericParamConstraintTable(writer, metadataSizes);
            }
 
            // debug tables
            if (metadataSizes.IsPresent(TableIndex.Document))
            {
                SerializeDocumentTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.MethodDebugInformation))
            {
                SerializeMethodDebugInformationTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.LocalScope))
            {
                SerializeLocalScopeTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.LocalVariable))
            {
                SerializeLocalVariableTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.LocalConstant))
            {
                SerializeLocalConstantTable(writer, stringMap, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.ImportScope))
            {
                SerializeImportScopeTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.StateMachineMethod))
            {
                SerializeStateMachineMethodTable(writer, metadataSizes);
            }
 
            if (metadataSizes.IsPresent(TableIndex.CustomDebugInformation))
            {
                SerializeCustomDebugInformationTable(writer, metadataSizes);
            }
 
            writer.WriteByte(0);
            writer.Align(4);
 
            int endPosition = writer.Count;
            Debug.Assert(metadataSizes.MetadataTableStreamSize == endPosition - startPosition);
        }
 
        private static void SerializeTablesHeader(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            int startPosition = writer.Count;
 
            HeapSizeFlag heapSizes = 0;
            if (!metadataSizes.StringReferenceIsSmall)
            {
                heapSizes |= HeapSizeFlag.StringHeapLarge;
            }
 
            if (!metadataSizes.GuidReferenceIsSmall)
            {
                heapSizes |= HeapSizeFlag.GuidHeapLarge;
            }
 
            if (!metadataSizes.BlobReferenceIsSmall)
            {
                heapSizes |= HeapSizeFlag.BlobHeapLarge;
            }
 
            if (metadataSizes.IsEncDelta)
            {
                heapSizes |= HeapSizeFlag.EncDeltas | HeapSizeFlag.DeletedMarks;
            }
 
            // Custom Attribute table is not sorted in delta metadata:
            ulong sortedTables =
                metadataSizes.IsEncDelta ? MetadataSizes.SortedTypeSystemTables & ~(1UL << (int)TableIndex.CustomAttribute) :
                metadataSizes.IsStandaloneDebugMetadata ? 0 :
                MetadataSizes.SortedTypeSystemTables;
 
            // Consider filtering out type system tables that are not present:
            sortedTables |= metadataSizes.PresentTablesMask & MetadataSizes.SortedDebugTables;
 
            writer.WriteUInt32(0); // reserved
            writer.WriteByte(MetadataFormatMajorVersion);
            writer.WriteByte(MetadataFormatMinorVersion);
            writer.WriteByte((byte)heapSizes);
            writer.WriteByte(1); // reserved
            writer.WriteUInt64(metadataSizes.PresentTablesMask);
            writer.WriteUInt64(sortedTables);
            MetadataWriterUtilities.SerializeRowCounts(writer, metadataSizes.RowCounts);
 
            int endPosition = writer.Count;
            Debug.Assert(metadataSizes.CalculateTableStreamHeaderSize() == endPosition - startPosition);
        }
 
        // internal for testing
        internal void SerializeModuleTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            if (_moduleRow.HasValue)
            {
                writer.WriteUInt16(_moduleRow.Value.Generation);
                writer.WriteReference(SerializeHandle(stringMap, _moduleRow.Value.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(_moduleRow.Value.ModuleVersionId), metadataSizes.GuidReferenceIsSmall);
                writer.WriteReference(SerializeHandle(_moduleRow.Value.EncId), metadataSizes.GuidReferenceIsSmall);
                writer.WriteReference(SerializeHandle(_moduleRow.Value.EncBaseId), metadataSizes.GuidReferenceIsSmall);
            }
        }
 
        private void SerializeEncLogTable(BlobBuilder writer)
        {
            foreach (EncLogRow encLog in _encLogTable)
            {
                writer.WriteInt32(encLog.Token);
                writer.WriteUInt32(encLog.FuncCode);
            }
        }
 
        private void SerializeEncMapTable(BlobBuilder writer)
        {
            foreach (EncMapRow encMap in _encMapTable)
            {
                writer.WriteInt32(encMap.Token);
            }
        }
 
        private void SerializeTypeRefTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (TypeRefRow typeRef in _typeRefTable)
            {
                writer.WriteReference(typeRef.ResolutionScope, metadataSizes.ResolutionScopeCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, typeRef.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, typeRef.Namespace), metadataSizes.StringReferenceIsSmall);
            }
        }
 
        private void SerializeTypeDefTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (TypeDefRow typeDef in _typeDefTable)
            {
                writer.WriteUInt32(typeDef.Flags);
                writer.WriteReference(SerializeHandle(stringMap, typeDef.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, typeDef.Namespace), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(typeDef.Extends, metadataSizes.TypeDefOrRefCodedIndexIsSmall);
                writer.WriteReference(typeDef.FieldList, metadataSizes.FieldDefReferenceIsSmall);
                writer.WriteReference(typeDef.MethodList, metadataSizes.MethodDefReferenceIsSmall);
            }
        }
 
        private void SerializeFieldTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (FieldDefRow fieldDef in _fieldTable)
            {
                writer.WriteUInt16(fieldDef.Flags);
                writer.WriteReference(SerializeHandle(stringMap, fieldDef.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(fieldDef.Signature), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeMethodDefTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes, int methodBodyStreamRva)
        {
            foreach (MethodRow method in _methodDefTable)
            {
                if (method.BodyOffset == -1)
                {
                    writer.WriteUInt32(0);
                }
                else
                {
                    writer.WriteInt32(methodBodyStreamRva + method.BodyOffset);
                }
 
                writer.WriteUInt16(method.ImplFlags);
                writer.WriteUInt16(method.Flags);
                writer.WriteReference(SerializeHandle(stringMap, method.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(method.Signature), metadataSizes.BlobReferenceIsSmall);
                writer.WriteReference(method.ParamList, metadataSizes.ParameterReferenceIsSmall);
            }
        }
 
        private void SerializeParamTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (ParamRow param in _paramTable)
            {
                writer.WriteUInt16(param.Flags);
                writer.WriteUInt16(param.Sequence);
                writer.WriteReference(SerializeHandle(stringMap, param.Name), metadataSizes.StringReferenceIsSmall);
            }
        }
 
        private void SerializeInterfaceImplTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // TODO (bug https://github.com/dotnet/roslyn/issues/3905):
            // We should sort the table by Class and then by Interface.
            foreach (InterfaceImplRow interfaceImpl in _interfaceImplTable)
            {
                writer.WriteReference(interfaceImpl.Class, metadataSizes.TypeDefReferenceIsSmall);
                writer.WriteReference(interfaceImpl.Interface, metadataSizes.TypeDefOrRefCodedIndexIsSmall);
            }
        }
 
        private void SerializeMemberRefTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (MemberRefRow memberRef in _memberRefTable)
            {
                writer.WriteReference(memberRef.Class, metadataSizes.MemberRefParentCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, memberRef.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(memberRef.Signature), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeConstantTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // Note: we can sort the table at this point since no other table can reference its rows via RowId or CodedIndex (which would need updating otherwise).
            var ordered = _constantTableNeedsSorting ? _constantTable.OrderBy((x, y) => x.Parent - y.Parent) : _constantTable;
 
            foreach (ConstantRow constant in ordered)
            {
                writer.WriteByte(constant.Type);
                writer.WriteByte(0);
                writer.WriteReference(constant.Parent, metadataSizes.HasConstantCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(constant.Value), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeCustomAttributeTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // Note: we can sort the table at this point since no other table can reference its rows via RowId or CodedIndex (which would need updating otherwise).
            // OrderBy performs a stable sort, so multiple attributes with the same parent will be sorted in the order they were added to the table.
            // Avoid sorting the table when emitting EnC delta. Deleted attributes are represented in the table as rows with nil Parent field.
            // Sorting the table would move them to the beginning of the table and break mapping specified in EncMap table.
            var ordered = _customAttributeTableNeedsSorting && !metadataSizes.IsEncDelta ? _customAttributeTable.OrderBy((x, y) => x.Parent - y.Parent) : _customAttributeTable;
 
            foreach (CustomAttributeRow customAttribute in ordered)
            {
                writer.WriteReference(customAttribute.Parent, metadataSizes.HasCustomAttributeCodedIndexIsSmall);
                writer.WriteReference(customAttribute.Type, metadataSizes.CustomAttributeTypeCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(customAttribute.Value), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeFieldMarshalTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // Note: we can sort the table at this point since no other table can reference its rows via RowId or CodedIndex (which would need updating otherwise).
            var ordered = _fieldMarshalTableNeedsSorting ? _fieldMarshalTable.OrderBy((x, y) => x.Parent - y.Parent) : _fieldMarshalTable;
 
            foreach (FieldMarshalRow fieldMarshal in ordered)
            {
                writer.WriteReference(fieldMarshal.Parent, metadataSizes.HasFieldMarshalCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(fieldMarshal.NativeType), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeDeclSecurityTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // Note: we can sort the table at this point since no other table can reference its rows via RowId or CodedIndex (which would need updating otherwise).
            // OrderBy performs a stable sort, so multiple attributes with the same parent will be sorted in the order they were added to the table.
            var ordered = _declSecurityTableNeedsSorting ? _declSecurityTable.OrderBy((x, y) => x.Parent - y.Parent) : _declSecurityTable;
 
            foreach (DeclSecurityRow declSecurity in ordered)
            {
                writer.WriteUInt16(declSecurity.Action);
                writer.WriteReference(declSecurity.Parent, metadataSizes.DeclSecurityCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(declSecurity.PermissionSet), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeClassLayoutTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (ClassLayoutRow classLayout in _classLayoutTable)
            {
                writer.WriteUInt16(classLayout.PackingSize);
                writer.WriteUInt32(classLayout.ClassSize);
                writer.WriteReference(classLayout.Parent, metadataSizes.TypeDefReferenceIsSmall);
            }
        }
 
        private void SerializeFieldLayoutTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (FieldLayoutRow fieldLayout in _fieldLayoutTable)
            {
                writer.WriteInt32(fieldLayout.Offset);
                writer.WriteReference(fieldLayout.Field, metadataSizes.FieldDefReferenceIsSmall);
            }
        }
 
        private void SerializeStandAloneSigTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (StandaloneSigRow row in _standAloneSigTable)
            {
                writer.WriteReference(SerializeHandle(row.Signature), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeEventMapTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (EventMapRow eventMap in _eventMapTable)
            {
                writer.WriteReference(eventMap.Parent, metadataSizes.TypeDefReferenceIsSmall);
                writer.WriteReference(eventMap.EventList, metadataSizes.EventDefReferenceIsSmall);
            }
        }
 
        private void SerializeEventTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (EventRow eventRow in _eventTable)
            {
                writer.WriteUInt16(eventRow.EventFlags);
                writer.WriteReference(SerializeHandle(stringMap, eventRow.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(eventRow.EventType, metadataSizes.TypeDefOrRefCodedIndexIsSmall);
            }
        }
 
        private void SerializePropertyMapTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (PropertyMapRow propertyMap in _propertyMapTable)
            {
                writer.WriteReference(propertyMap.Parent, metadataSizes.TypeDefReferenceIsSmall);
                writer.WriteReference(propertyMap.PropertyList, metadataSizes.PropertyDefReferenceIsSmall);
            }
        }
 
        private void SerializePropertyTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (PropertyRow property in _propertyTable)
            {
                writer.WriteUInt16(property.PropFlags);
                writer.WriteReference(SerializeHandle(stringMap, property.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(property.Type), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeMethodSemanticsTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // Note: we can sort the table at this point since no other table can reference its rows via RowId or CodedIndex (which would need updating otherwise).
            // OrderBy performs a stable sort, so multiple attributes with the same parent will be sorted in the order they were added to the table.
            var ordered = _methodSemanticsTableNeedsSorting ? _methodSemanticsTable.OrderBy((x, y) => (int)x.Association - (int)y.Association) : _methodSemanticsTable;
 
            foreach (MethodSemanticsRow methodSemantic in ordered)
            {
                writer.WriteUInt16(methodSemantic.Semantic);
                writer.WriteReference(methodSemantic.Method, metadataSizes.MethodDefReferenceIsSmall);
                writer.WriteReference(methodSemantic.Association, metadataSizes.HasSemanticsCodedIndexIsSmall);
            }
        }
 
        private void SerializeMethodImplTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (MethodImplRow methodImpl in _methodImplTable)
            {
                writer.WriteReference(methodImpl.Class, metadataSizes.TypeDefReferenceIsSmall);
                writer.WriteReference(methodImpl.MethodBody, metadataSizes.MethodDefOrRefCodedIndexIsSmall);
                writer.WriteReference(methodImpl.MethodDecl, metadataSizes.MethodDefOrRefCodedIndexIsSmall);
            }
        }
 
        private void SerializeModuleRefTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (ModuleRefRow moduleRef in _moduleRefTable)
            {
                writer.WriteReference(SerializeHandle(stringMap, moduleRef.Name), metadataSizes.StringReferenceIsSmall);
            }
        }
 
        private void SerializeTypeSpecTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (TypeSpecRow typeSpec in _typeSpecTable)
            {
                writer.WriteReference(SerializeHandle(typeSpec.Signature), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeImplMapTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (ImplMapRow implMap in _implMapTable)
            {
                writer.WriteUInt16(implMap.MappingFlags);
                writer.WriteReference(implMap.MemberForwarded, metadataSizes.MemberForwardedCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, implMap.ImportName), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(implMap.ImportScope, metadataSizes.ModuleRefReferenceIsSmall);
            }
        }
 
        private void SerializeFieldRvaTable(BlobBuilder writer, MetadataSizes metadataSizes, int mappedFieldDataStreamRva)
        {
            foreach (FieldRvaRow fieldRva in _fieldRvaTable)
            {
                writer.WriteInt32(mappedFieldDataStreamRva + fieldRva.Offset);
                writer.WriteReference(fieldRva.Field, metadataSizes.FieldDefReferenceIsSmall);
            }
        }
 
        private void SerializeAssemblyTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            if (_assemblyRow.HasValue)
            {
                var version = _assemblyRow.Value.Version;
                writer.WriteUInt32(_assemblyRow.Value.HashAlgorithm);
                writer.WriteUInt16((ushort)version.Major);
                writer.WriteUInt16((ushort)version.Minor);
                writer.WriteUInt16((ushort)version.Build);
                writer.WriteUInt16((ushort)version.Revision);
                writer.WriteUInt32(_assemblyRow.Value.Flags);
                writer.WriteReference(SerializeHandle(_assemblyRow.Value.AssemblyKey), metadataSizes.BlobReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, _assemblyRow.Value.AssemblyName), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, _assemblyRow.Value.AssemblyCulture), metadataSizes.StringReferenceIsSmall);
            }
        }
 
        private void SerializeAssemblyRefTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (AssemblyRefTableRow row in _assemblyRefTable)
            {
                writer.WriteUInt16((ushort)row.Version.Major);
                writer.WriteUInt16((ushort)row.Version.Minor);
                writer.WriteUInt16((ushort)row.Version.Build);
                writer.WriteUInt16((ushort)row.Version.Revision);
                writer.WriteUInt32(row.Flags);
                writer.WriteReference(SerializeHandle(row.PublicKeyToken), metadataSizes.BlobReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, row.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, row.Culture), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.HashValue), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeFileTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (FileTableRow fileReference in _fileTable)
            {
                writer.WriteUInt32(fileReference.Flags);
                writer.WriteReference(SerializeHandle(stringMap, fileReference.FileName), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(fileReference.HashValue), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeExportedTypeTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (ExportedTypeRow exportedType in _exportedTypeTable)
            {
                writer.WriteUInt32(exportedType.Flags);
                writer.WriteInt32(exportedType.TypeDefId);
                writer.WriteReference(SerializeHandle(stringMap, exportedType.TypeName), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, exportedType.TypeNamespace), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(exportedType.Implementation, metadataSizes.ImplementationCodedIndexIsSmall);
            }
        }
 
        private void SerializeManifestResourceTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (ManifestResourceRow manifestResource in _manifestResourceTable)
            {
                writer.WriteUInt32(manifestResource.Offset);
                writer.WriteUInt32(manifestResource.Flags);
                writer.WriteReference(SerializeHandle(stringMap, manifestResource.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(manifestResource.Implementation, metadataSizes.ImplementationCodedIndexIsSmall);
            }
        }
 
        private void SerializeNestedClassTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (NestedClassRow nestedClass in _nestedClassTable)
            {
                writer.WriteReference(nestedClass.NestedClass, metadataSizes.TypeDefReferenceIsSmall);
                writer.WriteReference(nestedClass.EnclosingClass, metadataSizes.TypeDefReferenceIsSmall);
            }
        }
 
        private void SerializeGenericParamTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (GenericParamRow genericParam in _genericParamTable)
            {
                writer.WriteUInt16(genericParam.Number);
                writer.WriteUInt16(genericParam.Flags);
                writer.WriteReference(genericParam.Owner, metadataSizes.TypeOrMethodDefCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(stringMap, genericParam.Name), metadataSizes.StringReferenceIsSmall);
            }
        }
 
        private void SerializeGenericParamConstraintTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (GenericParamConstraintRow genericParamConstraint in _genericParamConstraintTable)
            {
                writer.WriteReference(genericParamConstraint.Owner, metadataSizes.GenericParamReferenceIsSmall);
                writer.WriteReference(genericParamConstraint.Constraint, metadataSizes.TypeDefOrRefCodedIndexIsSmall);
            }
        }
 
        private void SerializeMethodSpecTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (MethodSpecRow methodSpec in _methodSpecTable)
            {
                writer.WriteReference(methodSpec.Method, metadataSizes.MethodDefOrRefCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(methodSpec.Instantiation), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeDocumentTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (var row in _documentTable)
            {
                writer.WriteReference(SerializeHandle(row.Name), metadataSizes.BlobReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.HashAlgorithm), metadataSizes.GuidReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.Hash), metadataSizes.BlobReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.Language), metadataSizes.GuidReferenceIsSmall);
            }
        }
 
        private void SerializeMethodDebugInformationTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (var row in _methodDebugInformationTable)
            {
                writer.WriteReference(row.Document, metadataSizes.DocumentReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.SequencePoints), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeLocalScopeTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (var row in _localScopeTable)
            {
                writer.WriteReference(row.Method, metadataSizes.MethodDefReferenceIsSmall);
                writer.WriteReference(row.ImportScope, metadataSizes.ImportScopeReferenceIsSmall);
                writer.WriteReference(row.VariableList, metadataSizes.LocalVariableReferenceIsSmall);
                writer.WriteReference(row.ConstantList, metadataSizes.LocalConstantReferenceIsSmall);
                writer.WriteInt32(row.StartOffset);
                writer.WriteInt32(row.Length);
            }
        }
 
        private void SerializeLocalVariableTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (var row in _localVariableTable)
            {
                writer.WriteUInt16(row.Attributes);
                writer.WriteUInt16(row.Index);
                writer.WriteReference(SerializeHandle(stringMap, row.Name), metadataSizes.StringReferenceIsSmall);
            }
        }
 
        private void SerializeLocalConstantTable(BlobBuilder writer, ImmutableArray<int> stringMap, MetadataSizes metadataSizes)
        {
            foreach (var row in _localConstantTable)
            {
                writer.WriteReference(SerializeHandle(stringMap, row.Name), metadataSizes.StringReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.Signature), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeImportScopeTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (var row in _importScopeTable)
            {
                writer.WriteReference(row.Parent, metadataSizes.ImportScopeReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.Imports), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        private void SerializeStateMachineMethodTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            foreach (var row in _stateMachineMethodTable)
            {
                writer.WriteReference(row.MoveNextMethod, metadataSizes.MethodDefReferenceIsSmall);
                writer.WriteReference(row.KickoffMethod, metadataSizes.MethodDefReferenceIsSmall);
            }
        }
 
        private void SerializeCustomDebugInformationTable(BlobBuilder writer, MetadataSizes metadataSizes)
        {
            // Note: we can sort the table at this point since no other table can reference its rows via RowId or CodedIndex (which would need updating otherwise).
            // OrderBy performs a stable sort, so multiple attributes with the same parent and kind will be sorted in the order they were added to the table.
            foreach (CustomDebugInformationRow row in _customDebugInformationTable.OrderBy((x, y) =>
            {
                int result = x.Parent - y.Parent;
                return (result != 0) ? result : x.Kind.Index - y.Kind.Index;
            }))
            {
                writer.WriteReference(row.Parent, metadataSizes.HasCustomDebugInformationCodedIndexIsSmall);
                writer.WriteReference(SerializeHandle(row.Kind), metadataSizes.GuidReferenceIsSmall);
                writer.WriteReference(SerializeHandle(row.Value), metadataSizes.BlobReferenceIsSmall);
            }
        }
 
        #endregion
    }
}