Merge pull request #3444 from nspcc-dev/fix-native-init

native: perform initialisation for the set of hardforks
This commit is contained in:
Roman Khimov 2024-05-17 16:34:00 +03:00 committed by GitHub
commit 6be757af3e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -601,6 +601,7 @@ func (m *Management) OnPersist(ic *interop.Context) error {
isDeploy bool isDeploy bool
isUpdate bool isUpdate bool
latestHF config.Hardfork latestHF config.Hardfork
currentActiveHFs []config.Hardfork
) )
activeHFs := native.Metadata().ActiveHFs activeHFs := native.Metadata().ActiveHFs
isDeploy = activeIn == nil && ic.Block.Index == 0 || isDeploy = activeIn == nil && ic.Block.Index == 0 ||
@ -611,16 +612,18 @@ func (m *Management) OnPersist(ic *interop.Context) error {
isUpdate = true isUpdate = true
activation := hf // avoid loop variable pointer exporting. activation := hf // avoid loop variable pointer exporting.
activeIn = &activation // reuse ActiveIn variable for the initialization hardfork. activeIn = &activation // reuse ActiveIn variable for the initialization hardfork.
// Break immediately since native Initialize should be called only for the first hardfork in a raw // Break immediately since native Initialize should be called starting from the first hardfork in a raw
// (if there are multiple hardforks with the same enabling height). // (if there are multiple hardforks with the same enabling height).
break break
} }
} }
} }
// Search for the latest active hardfork to properly construct manifest. // Search for the latest active hardfork to properly construct manifest and
// initialize natives for the range of active hardforks.
for _, hf := range config.Hardforks { for _, hf := range config.Hardforks {
if _, ok := activeHFs[hf]; ok && ic.IsHardforkActivation(hf) { if _, ok := activeHFs[hf]; ok && ic.IsHardforkActivation(hf) {
latestHF = hf latestHF = hf
currentActiveHFs = append(currentActiveHFs, hf)
} }
} }
if !(isDeploy || isUpdate) { if !(isDeploy || isUpdate) {
@ -654,9 +657,21 @@ func (m *Management) OnPersist(ic *interop.Context) error {
if err != nil { if err != nil {
return fmt.Errorf("failed to put contract state: %w", err) return fmt.Errorf("failed to put contract state: %w", err)
} }
// Deploy hardfork (contract's ActiveIn) is not a part of contract's active hardforks and
// allowed to be nil, this, a special initialization call for it.
if isDeploy {
if err := native.Initialize(ic, activeIn, hfSpecificMD); err != nil { if err := native.Initialize(ic, activeIn, hfSpecificMD); err != nil {
return fmt.Errorf("initializing %s native contract at HF %v: %w", md.Name, activeIn, err)
}
}
// The rest of activating hardforks also require initialization.
for _, hf := range currentActiveHFs {
if err := native.Initialize(ic, &hf, hfSpecificMD); err != nil {
return fmt.Errorf("initializing %s native contract at HF %d: %w", md.Name, activeIn, err) return fmt.Errorf("initializing %s native contract at HF %d: %w", md.Name, activeIn, err)
} }
}
if cache == nil { if cache == nil {
cache = ic.DAO.GetRWCache(m.ID).(*ManagementCache) cache = ic.DAO.GetRWCache(m.ID).(*ManagementCache)
} }