You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Darwinism/XDRStream/Reading/ReadBuilder.EmitDynReadMapp...

100 lines
4.7 KiB

#Region "Microsoft.VisualBasic::ea446aac1e03ab1cea5e67ee0c8247e6, XDRStream\Reading\ReadBuilder.EmitDynReadMapper.vb"
' Author:
'
' asuka (amethyst.asuka@gcmodeller.org)
' xie (genetics@smrucc.org)
' xieguigang (xie.guigang@live.com)
'
' Copyright (c) 2018 GPL3 Licensed
'
'
' GNU GENERAL PUBLIC LICENSE (GPL3)
'
'
' This program is free software: you can redistribute it and/or modify
' it under the terms of the GNU General Public License as published by
' the Free Software Foundation, either version 3 of the License, or
' (at your option) any later version.
'
' This program is distributed in the hope that it will be useful,
' but WITHOUT ANY WARRANTY; without even the implied warranty of
' MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
' GNU General Public License for more details.
'
' You should have received a copy of the GNU General Public License
' along with this program. If not, see <http://www.gnu.org/licenses/>.
' /********************************************************************************/
' Summaries:
' Class ReadBuilder
'
' Function: DefineCacheField, EmitDynReadMapper
'
' Sub: EmitInitField, EmitOverride_GetCacheType
'
'
' /********************************************************************************/
#End Region
Imports System
Imports System.Reflection
Imports System.Reflection.Emit
Namespace Reading
Partial Public NotInheritable Class ReadBuilder
Private Function EmitDynReadMapper() As Type
Dim typeBuilder = _modBuilder.DefineType("DynReadMapper", TypeAttributes.NotPublic Or TypeAttributes.Class Or TypeAttributes.Sealed, GetType(ReadMapper))
Dim fb_oneCacheType = DefineCacheField(typeBuilder, "_oneCacheType")
Dim fb_fixCacheType = DefineCacheField(typeBuilder, "_fixCacheType")
Dim fb_varCacheType = DefineCacheField(typeBuilder, "_varCacheType")
Dim ctor = typeBuilder.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, New Type(-1) {})
Dim ilCtor As ILGenerator = ctor.GetILGenerator()
ilCtor.Emit(OpCodes.Ldarg_0)
ilCtor.Emit(OpCodes.Call, GetType(ReadMapper).GetConstructor(BindingFlags.Instance Or BindingFlags.NonPublic, Nothing, New Type(-1) {}, Nothing))
ilCtor.Emit(OpCodes.Ldarg_0)
ilCtor.Emit(OpCodes.Ldftn, GetType(ReadMapper).GetMethod("AppendBuildRequest", BindingFlags.NonPublic Or BindingFlags.Instance))
ilCtor.Emit(OpCodes.Newobj, GetType(Action(Of Type, OpaqueType)).GetConstructor(New Type() {GetType(Object), GetType(IntPtr)}))
ilCtor.Emit(OpCodes.Stsfld, _buildBinderDescription.BuildRequest)
EmitInitField(ilCtor, fb_oneCacheType, _oneCacheDescription.Result)
EmitInitField(ilCtor, fb_fixCacheType, _fixCacheDescription.Result)
EmitInitField(ilCtor, fb_varCacheType, _varCacheDescription.Result)
' run init
ilCtor.Emit(OpCodes.Ldarg_0)
ilCtor.Emit(OpCodes.Call, GetType(ReadMapper).GetMethod("Init", BindingFlags.Instance Or BindingFlags.NonPublic))
ilCtor.Emit(OpCodes.Ret)
EmitOverride_GetCacheType(typeBuilder, "GetOneCacheType", fb_oneCacheType)
EmitOverride_GetCacheType(typeBuilder, "GetFixCacheType", fb_fixCacheType)
EmitOverride_GetCacheType(typeBuilder, "GetVarCacheType", fb_varCacheType)
Return typeBuilder.CreateType()
End Function
Private Shared Function DefineCacheField(typeBuilder As TypeBuilder, name As String) As FieldBuilder
Return typeBuilder.DefineField(name, GetType(Type), FieldAttributes.Private Or FieldAttributes.InitOnly)
End Function
Private Shared Sub EmitInitField(il As ILGenerator, fb As FieldBuilder, type As Type)
il.Emit(OpCodes.Ldarg_0)
il.Emit(OpCodes.Ldtoken, type)
il.Emit(OpCodes.Call, GetType(Type).GetMethod("GetTypeFromHandle"))
il.Emit(OpCodes.Stfld, fb)
End Sub
Private Sub EmitOverride_GetCacheType(typeBuilder As TypeBuilder, overrideName As String, fb_cacheType As FieldBuilder)
Dim mb = typeBuilder.DefineMethod(overrideName, MethodAttributes.Family Or MethodAttributes.Virtual)
mb.SetReturnType(GetType(Type))
typeBuilder.DefineMethodOverride(mb, GetType(ReadMapper).GetMethod(overrideName, BindingFlags.NonPublic Or BindingFlags.Instance))
Dim il As ILGenerator = mb.GetILGenerator()
il.Emit(OpCodes.Ldarg_0)
il.Emit(OpCodes.Ldfld, fb_cacheType)
il.Emit(OpCodes.Ret)
End Sub
End Class
End Namespace