From c938011c1bab98936b25a3502477cd046ee317ac Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Costin=20Caraba=C8=99?= Date: Thu, 23 Nov 2023 14:08:00 +0200 Subject: [PATCH 1/6] Add mandos test for paymaster --- test/paymaster/output/paymaster.wasm | Bin 0 -> 4286 bytes .../test_forward_call_wegld.scen.json | 179 ++++++++++++++++++ 2 files changed, 179 insertions(+) create mode 100755 test/paymaster/output/paymaster.wasm create mode 100644 test/paymaster/scenarios/test_forward_call_wegld.scen.json diff --git a/test/paymaster/output/paymaster.wasm b/test/paymaster/output/paymaster.wasm new file mode 100755 index 0000000000000000000000000000000000000000..2e8d56eec55427d940d533140fe965f88471e0ff GIT binary patch literal 4286 zcmai1O>7)V6@FFKJ<~Hiwktm*j^o`_&zfb-CL{~N2`hwLPMqvdl8s1MaX_*sGi}FX zd&cf*C$WS)4lz*<9NT|z)*>BZtiwmUz@vC@5V}&|1cUI5u&HZs$zyT8G%32 zc!tM7w*R~z@V6$(isdRzNir0U@B&q3MsEz9BOn?qH_|BWkA;;?An5{_4`eTrS@h7D zk>R0y+I|e-#54+rAObd!9_F`8;CP9wUarasmMZLMipTc9)w&JR1jFEtd*hiqNiroX z5HD?|Wj$Sl4y#)e6n=w+8VD^}bJ?5bP}-yV#VUI#nUCy`K~mZ9XRt=Qp$;g76{8v7 zixE3|VoG}K0wNIpNC=hN=TDHFhS#N8c);#|-gvqc6)Z(cVZwT?YS9$l8WP602w%l` zMq%hIeHoS-za}eH#SW^0gKF17%)mj2Ek>$BAsc4oz!r#dt`5OM_=6B1*HNAjD3ZEj z>2hc|;wP$5t!q%}>1e-{@OJ}`6e6yW&GQfOK*S%+)E@5eaQoBZgPBoB-QIpS@?a)- zxWf~{Bg`=Y<53=ikLfjxu^glGk9Kx;&Zz*A64#LjobuU&cW%5fx`Pjde|EPMw5dPdS>XHLKgTZ00J?D>BynhFJp)-k8|klIDFBk zA|EfF{!&zMY>p19LL9yi#J!*aDA=Gf4Es|uz+1s7u!d2*zP}hU6uELjRRYVz7$f~< zNM=+S2{J!C{S_mOD_Oi%A~ut;f&~B@4ZudbV8aZ+7XDmFvDL3Yhlk8n1dxh!EsVsboOO0iZDK1B!FTl6wFpcL4=a45+ZgPkd&Ct;)Ia^3_`GU!Pot~9tjo=Q>Z zA@S-G76_nnr~4^{9 z#w4#X9)lP`V6Es3nOHJ@Bw=HQ>HonUB4BDgLQ^#`P~w%{H445s_hjy#bX}wGo^bAE zQF~8Fe_xHm0U){R0l8J5$H7%^%OT&D(BATA*u%Wt`scl8Zv!Zx)Vbpq&;W{1ei$vz z^*MA0eTrSeHrFzq8=0bOC2}B-paYnw$d50gEU zpjbee?6TH&IYR0Y{VAEsY3yaM&T=m8Giu@GBupyc8{mufSCiY=?0_$XuqJneJ~kyk z0DL(+Pw;C>Q}U@m;)U*z?tnGKI2=L|>bS?%+@*^#POtPm*P#@wQ$#INTl|DSC zKZl@}7&Z!DsgL0GRUD;ocp98F1|x%QTc5z+IQ+A}MIny6!-&+c=%Xmj#lZm`hcFz7 zPS`~~I>rh3Nbt!-5i&-OYuL4-hZ2R%7nXmBdVaGs&A8EN#wO;QvZ+nJX3|c_EW~$B zpQFnJ+^;qVl)Fi%y=dAStGD9BbQT5^yZ1uem;1M?{QA82wLiX^U5ej{8)m89jPIGw zM#}PV@m`E?)teBU)HltFyBIfd6-?rl`evNW)p6;*c{@&Gv(z Date: Thu, 23 Nov 2023 14:14:41 +0200 Subject: [PATCH 2/6] Paymaster wasm now uses async call --- test/paymaster/output/paymaster.wasm | Bin 4286 -> 7509 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/test/paymaster/output/paymaster.wasm b/test/paymaster/output/paymaster.wasm index 2e8d56eec55427d940d533140fe965f88471e0ff..b339e8384f4482b8fde6e9245942f3a9ec366d15 100755 GIT binary patch literal 7509 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 Date: Thu, 23 Nov 2023 15:50:57 +0200 Subject: [PATCH 3/6] fixed back transfers on callBack. --- .../json/scenariosContracts_test.go | 7 ++++++ vmhost/contexts/managedType.go | 24 +++++++++++++++---- vmhost/hostCore/execution.go | 1 + vmhost/interface.go | 1 + 4 files changed, 28 insertions(+), 5 deletions(-) diff --git a/integrationTests/json/scenariosContracts_test.go b/integrationTests/json/scenariosContracts_test.go index d1b83a964..826e630f7 100644 --- a/integrationTests/json/scenariosContracts_test.go +++ b/integrationTests/json/scenariosContracts_test.go @@ -119,3 +119,10 @@ func TestCAttestation(t *testing.T) { Run(). CheckNoError() } + +func TestRustPaymaster(t *testing.T) { + ScenariosTest(t). + Folder("paymaster/scenarios"). + Run(). + CheckNoError() +} diff --git a/vmhost/contexts/managedType.go b/vmhost/contexts/managedType.go index 37baf4ad3..edeb84c8f 100644 --- a/vmhost/contexts/managedType.go +++ b/vmhost/contexts/managedType.go @@ -5,6 +5,7 @@ import ( "crypto/elliptic" "encoding/binary" "errors" + "github.com/multiversx/mx-chain-core-go/data/vm" "io" basicMath "math" "math/big" @@ -139,7 +140,7 @@ func (context *managedTypesContext) InitState() { // PushState appends the values map to the state stack func (context *managedTypesContext) PushState() { newBigIntState, newBigFloatState, newEcState, newmBufferState, newmMapState := context.clone() - newTransfers := context.cloneBackTransfers() + newTransfers := cloneBackTransfers(context.managedTypesValues.backTransfers) context.managedTypesStack = append(context.managedTypesStack, managedTypesState{ bigIntValues: newBigIntState, bigFloatValues: newBigFloatState, @@ -150,6 +151,21 @@ func (context *managedTypesContext) PushState() { }) } +// PopBackTransferIfAsyncCallBack copies the back transfer from the top of the stack in case of callbacks +func (context *managedTypesContext) PopBackTransferIfAsyncCallBack(vmInput *vmcommon.ContractCallInput) { + if vmInput.CallType != vm.AsynchronousCallBack { + return + } + + managedTypesStackLen := len(context.managedTypesStack) + if managedTypesStackLen == 0 { + return + } + + prevState := context.managedTypesStack[managedTypesStackLen-1] + context.managedTypesValues.backTransfers = cloneBackTransfers(prevState.backTransfers) +} + // PopSetActiveState removes the latest entry from the state stack and sets it as the current values map func (context *managedTypesContext) PopSetActiveState() { managedTypesStackLen := len(context.managedTypesStack) @@ -797,7 +813,7 @@ func (context *managedTypesContext) AddValueOnlyBackTransfer(value *big.Int) { // GetBackTransfers returns all ESDT transfers and accumulated value as well, will clean accumulated values func (context *managedTypesContext) GetBackTransfers() ([]*vmcommon.ESDTTransfer, *big.Int) { - clonedTransfers := context.cloneBackTransfers() + clonedTransfers := cloneBackTransfers(context.managedTypesValues.backTransfers) context.managedTypesValues.backTransfers = backTransfers{ ESDTTransfers: make([]*vmcommon.ESDTTransfer, 0), CallValue: big.NewInt(0), @@ -806,9 +822,7 @@ func (context *managedTypesContext) GetBackTransfers() ([]*vmcommon.ESDTTransfer return clonedTransfers.ESDTTransfers, clonedTransfers.CallValue } -func (context *managedTypesContext) cloneBackTransfers() backTransfers { - currentBackTransfers := context.managedTypesValues.backTransfers - +func cloneBackTransfers(currentBackTransfers backTransfers) backTransfers { newBackTransfers := backTransfers{ ESDTTransfers: make([]*vmcommon.ESDTTransfer, len(currentBackTransfers.ESDTTransfers)), CallValue: big.NewInt(0).Set(currentBackTransfers.CallValue), diff --git a/vmhost/hostCore/execution.go b/vmhost/hostCore/execution.go index 3e21ce743..1eb04379e 100644 --- a/vmhost/hostCore/execution.go +++ b/vmhost/hostCore/execution.go @@ -448,6 +448,7 @@ func (host *vmHost) executeOnDestContextNoBuiltinFunction(input *vmcommon.Contra managedTypes, _, metering, output, runtime, async, storage := host.GetContexts() managedTypes.PushState() managedTypes.InitState() + managedTypes.PopBackTransferIfAsyncCallBack(input) output.PushState() output.CensorVMOutput() diff --git a/vmhost/interface.go b/vmhost/interface.go index e7edee399..03655d904 100644 --- a/vmhost/interface.go +++ b/vmhost/interface.go @@ -218,6 +218,7 @@ type ManagedTypesContext interface { GetBackTransfers() ([]*vmcommon.ESDTTransfer, *big.Int) AddValueOnlyBackTransfer(value *big.Int) AddBackTransfers(transfers []*vmcommon.ESDTTransfer) + PopBackTransferIfAsyncCallBack(vmInput *vmcommon.ContractCallInput) } // OutputContext defines the functionality needed for interacting with the output context From 72fc32c135147631c743f3e0ed91cb1520998975 Mon Sep 17 00:00:00 2001 From: robertsasu Date: Fri, 24 Nov 2023 11:57:14 +0200 Subject: [PATCH 4/6] fixed back transfers on callBack. --- vmhost/contexts/instanceTracker.go | 23 +++++++----- vmhost/contexts/instanceTracker_test.go | 26 +++++++------- vmhost/contexts/runtime.go | 47 ++++++++++++++++--------- 3 files changed, 59 insertions(+), 37 deletions(-) diff --git a/vmhost/contexts/instanceTracker.go b/vmhost/contexts/instanceTracker.go index 042a5c766..a9277fdfa 100644 --- a/vmhost/contexts/instanceTracker.go +++ b/vmhost/contexts/instanceTracker.go @@ -2,6 +2,7 @@ package contexts import ( "bytes" + "errors" "fmt" "github.com/multiversx/mx-chain-core-go/core/check" @@ -15,6 +16,8 @@ var _ vmhost.InstanceTracker = (*instanceTracker)(nil) type instanceCacheLevel int +var errTooManyInstances = errors.New("too many instances") + const ( // Warm indicates that the instance to track is a warm instance Warm instanceCacheLevel = iota @@ -178,27 +181,27 @@ func (tracker *instanceTracker) TrackedInstances() map[string]executor.Instance // UseWarmInstance attempts to retrieve a warm instance for the given codeHash // and to set it as active; returns false if not possible -func (tracker *instanceTracker) UseWarmInstance(codeHash []byte, newCode bool) bool { +func (tracker *instanceTracker) UseWarmInstance(codeHash []byte, newCode bool) (bool, error) { instance, ok := tracker.GetWarmInstance(codeHash) if !ok { - return false + return false, nil } ok = instance.Reset() if !ok { tracker.warmInstanceCache.Remove(codeHash) - return false + return false, nil } if newCode { // A warm instance was found, but newCode == true, meaning this is an // upgrade; the old warm instance must be cleaned tracker.ForceCleanInstance(false) - return false + return false, nil } - tracker.SetNewInstance(instance, Warm) - return true + err := tracker.SetNewInstance(instance, Warm) + return true, err } // ForceCleanInstance cleans the active instance and evicts it from the @@ -237,7 +240,6 @@ func (tracker *instanceTracker) ForceCleanInstance(bypassWarmAndStackChecks bool // SaveAsWarmInstance saves the active instance into the internal warm instance cache func (tracker *instanceTracker) SaveAsWarmInstance() { lenCacheBeforeSaving := tracker.warmInstanceCache.Len() - codeHashInWarmCache := tracker.warmInstanceCache.Has(tracker.codeHash) if codeHashInWarmCache { @@ -289,13 +291,18 @@ func (tracker *instanceTracker) GetCodeSize() uint64 { } // SetNewInstance sets the given instance as active and tracks its creation -func (tracker *instanceTracker) SetNewInstance(instance executor.Instance, cacheLevel instanceCacheLevel) { +func (tracker *instanceTracker) SetNewInstance(instance executor.Instance, cacheLevel instanceCacheLevel) error { tracker.ReplaceInstance(instance) tracker.cacheLevel = cacheLevel if cacheLevel != Warm { tracker.updateNumRunningInstances(+1) } tracker.instances[instance.ID()] = instance + + if len(tracker.instances) >= tracker.warmInstanceCache.MaxSize()-1 { + return errTooManyInstances + } + return nil } // ReplaceInstance replaces the currently active instance with the given one diff --git a/vmhost/contexts/instanceTracker_test.go b/vmhost/contexts/instanceTracker_test.go index 373a7d439..0aaf4b152 100644 --- a/vmhost/contexts/instanceTracker_test.go +++ b/vmhost/contexts/instanceTracker_test.go @@ -17,7 +17,7 @@ func TestInstanceTracker_TrackInstance(t *testing.T) { AlreadyClean: false, } - iTracker.SetNewInstance(newInstance, Bytecode) + _ = iTracker.SetNewInstance(newInstance, Bytecode) iTracker.codeHash = []byte("testinst") require.False(t, iTracker.IsCodeHashOnTheStack(iTracker.codeHash)) @@ -38,7 +38,7 @@ func TestInstanceTracker_InitState(t *testing.T) { require.Equal(t, 0, iTracker.numRunningInstances) for i := 0; i < 5; i++ { - iTracker.SetNewInstance(mock.NewInstanceMock(nil), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock(nil), Bytecode) } require.Equal(t, 5, iTracker.numRunningInstances) @@ -63,7 +63,7 @@ func TestInstanceTracker_GetWarmInstance(t *testing.T) { testData := []string{"warm1", "bytecode1", "bytecode2", "warm2"} for _, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if strings.Contains(codeHash, "warm") { iTracker.SaveAsWarmInstance() @@ -95,7 +95,7 @@ func TestInstanceTracker_UseWarmInstance(t *testing.T) { testData := []string{"warm1", "bytecode1", "warm2", "bytecode2"} for _, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if strings.Contains(codeHash, "warm") { @@ -106,7 +106,7 @@ func TestInstanceTracker_UseWarmInstance(t *testing.T) { require.Equal(t, []byte("bytecode2"), iTracker.CodeHash()) for _, codeHash := range testData { - ok := iTracker.UseWarmInstance([]byte(codeHash), false) + ok, _ := iTracker.UseWarmInstance([]byte(codeHash), false) if strings.Contains(codeHash, "warm") { require.True(t, ok) @@ -124,7 +124,7 @@ func TestInstanceTracker_IsCodeHashOnStack_Ok(t *testing.T) { testData := []string{"alpha", "beta", "alpha", "active"} for i, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if i < 2 || codeHash == "active" { iTracker.SaveAsWarmInstance() @@ -157,7 +157,7 @@ func TestInstanceTracker_PopSetActiveSelfScenario(t *testing.T) { testData := []string{"alpha", "alpha", "alpha", "alpha", "active"} for i, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if i == 0 || codeHash == "active" { iTracker.SaveAsWarmInstance() @@ -187,7 +187,7 @@ func TestInstanceTracker_PopSetActiveSimpleScenario(t *testing.T) { testData := []string{"alpha", "beta", "alpha", "beta", "active"} for i, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if i < 2 || codeHash == "active" { iTracker.SaveAsWarmInstance() @@ -226,7 +226,7 @@ func TestInstanceTracker_PopSetActiveComplexScenario(t *testing.T) { testData := []string{"alpha", "beta", "gamma", "beta", "gamma", "delta", "alpha", "active"} for i, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if i < 3 || codeHash == "delta" || codeHash == "active" { iTracker.SaveAsWarmInstance() @@ -255,7 +255,7 @@ func TestInstanceTracker_PopSetActiveWarmOnlyScenario(t *testing.T) { testData := []string{"alpha", "beta", "gamma", "delta", "active"} for _, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) iTracker.SaveAsWarmInstance() @@ -283,7 +283,7 @@ func TestInstanceTracker_ForceCleanInstanceWithBypass(t *testing.T) { testData := []string{"warm1", "bytecode1"} for _, codeHash := range testData { - iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock([]byte(codeHash)), Bytecode) iTracker.codeHash = []byte(codeHash) if strings.Contains(codeHash, "warm") { @@ -298,7 +298,7 @@ func TestInstanceTracker_ForceCleanInstanceWithBypass(t *testing.T) { iTracker.ForceCleanInstance(true) require.Nil(t, iTracker.instance) - iTracker.UseWarmInstance([]byte("warm1"), false) + _, _ = iTracker.UseWarmInstance([]byte("warm1"), false) require.NotNil(t, iTracker.instance) iTracker.ForceCleanInstance(true) @@ -312,7 +312,7 @@ func TestInstanceTracker_DoubleForceClean(t *testing.T) { iTracker, err := NewInstanceTracker() require.Nil(t, err) - iTracker.SetNewInstance(mock.NewInstanceMock(nil), Bytecode) + _ = iTracker.SetNewInstance(mock.NewInstanceMock(nil), Bytecode) require.NotNil(t, iTracker.instance) require.Equal(t, 1, iTracker.numRunningInstances) diff --git a/vmhost/contexts/runtime.go b/vmhost/contexts/runtime.go index e2c0a04cd..f2da057d8 100644 --- a/vmhost/contexts/runtime.go +++ b/vmhost/contexts/runtime.go @@ -152,12 +152,18 @@ func (context *runtimeContext) StartWasmerInstance(contract []byte, gasLimit uin logRuntime.Trace("code was new", "new", newCode) }() - warmInstanceUsed := context.useWarmInstanceIfExists(gasLimit, newCode) + warmInstanceUsed, err := context.useWarmInstanceIfExists(gasLimit, newCode) + if err != nil { + return err + } if warmInstanceUsed { return nil } - compiledCodeUsed := context.makeInstanceFromCompiledCode(gasLimit, newCode) + compiledCodeUsed, err := context.makeInstanceFromCompiledCode(gasLimit, newCode) + if err != nil { + return err + } if compiledCodeUsed { return nil } @@ -165,17 +171,17 @@ func (context *runtimeContext) StartWasmerInstance(contract []byte, gasLimit uin return context.makeInstanceFromContractByteCode(contract, gasLimit, newCode) } -func (context *runtimeContext) makeInstanceFromCompiledCode(gasLimit uint64, newCode bool) bool { +func (context *runtimeContext) makeInstanceFromCompiledCode(gasLimit uint64, newCode bool) (bool, error) { codeHash := context.iTracker.CodeHash() if newCode || len(codeHash) == 0 { - return false + return false, nil } blockchain := context.host.Blockchain() found, compiledCode := blockchain.GetCompiledCode(codeHash) if !found { logRuntime.Trace("instance creation", "code", "cached compilation", "error", "compiled code was not found") - return false + return false, nil } gasSchedule := context.host.Metering().GasSchedule() @@ -191,10 +197,13 @@ func (context *runtimeContext) makeInstanceFromCompiledCode(gasLimit uint64, new newInstance, err := context.vmExecutor.NewInstanceFromCompiledCodeWithOptions(compiledCode, options) if err != nil { logRuntime.Error("instance creation", "from", "cached compilation", "error", err) - return false + return false, nil } - context.iTracker.SetNewInstance(newInstance, Precompiled) + err = context.iTracker.SetNewInstance(newInstance, Precompiled) + if err != nil { + return false, err + } context.verifyCode = false context.saveWarmInstance() @@ -202,7 +211,7 @@ func (context *runtimeContext) makeInstanceFromCompiledCode(gasLimit uint64, new "id", context.iTracker.Instance().ID(), "codeHash", context.iTracker.codeHash, ) - return true + return true, nil } func (context *runtimeContext) makeInstanceFromContractByteCode(contract []byte, gasLimit uint64, newCode bool) error { @@ -223,7 +232,10 @@ func (context *runtimeContext) makeInstanceFromContractByteCode(contract []byte, return err } - context.iTracker.SetNewInstance(newInstance, Bytecode) + err = context.iTracker.SetNewInstance(newInstance, Bytecode) + if err != nil { + return err + } if newCode || len(context.iTracker.CodeHash()) == 0 { codeHash := context.hasher.Compute(string(contract)) @@ -249,23 +261,26 @@ func (context *runtimeContext) makeInstanceFromContractByteCode(contract []byte, return nil } -func (context *runtimeContext) useWarmInstanceIfExists(gasLimit uint64, newCode bool) bool { +func (context *runtimeContext) useWarmInstanceIfExists(gasLimit uint64, newCode bool) (bool, error) { if !WarmInstancesEnabled { - return false + return false, nil } codeHash := context.iTracker.CodeHash() if newCode || len(codeHash) == 0 { - return false + return false, nil } if context.isContractOrCodeHashOnTheStack() { - return false + return false, nil } - ok := context.iTracker.UseWarmInstance(codeHash, newCode) + ok, err := context.iTracker.UseWarmInstance(codeHash, newCode) + if err != nil { + return false, err + } if !ok { - return false + return false, nil } context.SetPointsUsed(0) @@ -273,7 +288,7 @@ func (context *runtimeContext) useWarmInstanceIfExists(gasLimit uint64, newCode context.SetRuntimeBreakpointValue(vmhost.BreakpointNone) context.verifyCode = false logRuntime.Trace("start instance", "from", "warm", "id", context.iTracker.Instance().ID()) - return true + return true, nil } // GetSCCode returns the SC code of the current SC. From 78ecab4e42c960053424aee72c11e42230cf0d43 Mon Sep 17 00:00:00 2001 From: robertsasu Date: Fri, 24 Nov 2023 11:59:39 +0200 Subject: [PATCH 5/6] goimports --- vmhost/contexts/managedType.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vmhost/contexts/managedType.go b/vmhost/contexts/managedType.go index edeb84c8f..40e26aa0b 100644 --- a/vmhost/contexts/managedType.go +++ b/vmhost/contexts/managedType.go @@ -5,12 +5,12 @@ import ( "crypto/elliptic" "encoding/binary" "errors" - "github.com/multiversx/mx-chain-core-go/data/vm" "io" basicMath "math" "math/big" "github.com/multiversx/mx-chain-core-go/core/check" + "github.com/multiversx/mx-chain-core-go/data/vm" logger "github.com/multiversx/mx-chain-logger-go" vmcommon "github.com/multiversx/mx-chain-vm-common-go" "github.com/multiversx/mx-chain-vm-go/math" From 071ae47774a36cbad0a5853a1228780dcea7a57a Mon Sep 17 00:00:00 2001 From: robertsasu Date: Fri, 24 Nov 2023 15:53:28 +0200 Subject: [PATCH 6/6] use constant and safer code --- vmhost/contexts/instanceTracker.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vmhost/contexts/instanceTracker.go b/vmhost/contexts/instanceTracker.go index a9277fdfa..f0fef8238 100644 --- a/vmhost/contexts/instanceTracker.go +++ b/vmhost/contexts/instanceTracker.go @@ -299,7 +299,7 @@ func (tracker *instanceTracker) SetNewInstance(instance executor.Instance, cache } tracker.instances[instance.ID()] = instance - if len(tracker.instances) >= tracker.warmInstanceCache.MaxSize()-1 { + if len(tracker.instances) >= warmCacheSize-1 { return errTooManyInstances } return nil