• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2025 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "common_components/heap/barrier/barrier.h"
17 
18 #include "common_components/heap/collector/collector.h"
19 #include "common_components/heap/heap.h"
20 #if defined(COMMON_TSAN_SUPPORT)
21 #include "common_components/sanitizer/sanitizer_interface.h"
22 #endif
23 #include "common_components/log/log.h"
24 #include "securec.h"
25 
26 namespace common {
27 
WriteRefField(BaseObject * obj,RefField<false> & field,BaseObject * ref) const28 void Barrier::WriteRefField(BaseObject* obj, RefField<false>& field, BaseObject* ref) const
29 {
30     DLOG(BARRIER, "write obj %p ref-field@%p: %p => %p", obj, &field, field.GetTargetObject(), ref);
31     field.SetTargetObject(ref);
32 }
33 
WriteBarrier(BaseObject * obj,RefField<false> & field,BaseObject * ref) const34 void Barrier::WriteBarrier(BaseObject* obj, RefField<false>& field, BaseObject* ref) const
35 {
36     DLOG(BARRIER, "write obj %p ref-field@%p: %p => %p", obj, &field, field.GetTargetObject(), ref);
37 }
38 
WriteRoot(BaseObject * obj) const39 void Barrier::WriteRoot(BaseObject *obj) const
40 {
41     DLOG(BARRIER, "write root obj %p", obj);
42 }
43 
WriteStruct(BaseObject * obj,HeapAddress dst,size_t dstLen,HeapAddress src,size_t srcLen) const44 void Barrier::WriteStruct(BaseObject* obj, HeapAddress dst, size_t dstLen, HeapAddress src, size_t srcLen) const
45 {
46     LOGF_CHECK(memcpy_s(reinterpret_cast<void*>(dst), dstLen, reinterpret_cast<void*>(src), srcLen) == EOK) <<
47         "memcpy_s failed";
48 #if defined(COMMON_TSAN_SUPPORT)
49     CHECK_EQ(srcLen, dstLen);
50     Sanitizer::TsanWriteMemoryRange(reinterpret_cast<void*>(dst), dstLen);
51     Sanitizer::TsanReadMemoryRange(reinterpret_cast<void*>(src), srcLen);
52 #endif
53 }
54 
WriteStaticRef(RefField<false> & field,BaseObject * ref) const55 void Barrier::WriteStaticRef(RefField<false>& field, BaseObject* ref) const
56 {
57     DLOG(BARRIER, "write (barrier) static ref@%p: %p", &field, ref);
58     field.SetTargetObject(ref);
59 }
60 
ReadRefField(BaseObject * obj,RefField<false> & field) const61 BaseObject* Barrier::ReadRefField(BaseObject* obj, RefField<false>& field) const
62 {
63     HeapAddress target = field.GetFieldValue();
64     return reinterpret_cast<BaseObject*>(target);
65 }
66 
ReadStaticRef(RefField<false> & field) const67 BaseObject* Barrier::ReadStaticRef(RefField<false>& field) const
68 {
69     HeapAddress target = field.GetFieldValue();
70     return reinterpret_cast<BaseObject*>(target);
71 }
72 
ReadStringTableStaticRef(RefField<false> & field) const73 BaseObject* Barrier::ReadStringTableStaticRef(RefField<false>& field) const
74 {
75     HeapAddress target = field.GetFieldValue();
76     return reinterpret_cast<BaseObject*>(target);
77 }
78 
79 // barrier for atomic operation.
AtomicWriteRefField(BaseObject * obj,RefField<true> & field,BaseObject * ref,MemoryOrder order) const80 void Barrier::AtomicWriteRefField(BaseObject* obj, RefField<true>& field, BaseObject* ref, MemoryOrder order) const
81 {
82     if (obj != nullptr) {
83         DLOG(BARRIER, "atomic write obj %p<%p>(%zu) ref@%p: %#zx -> %p", obj, obj->GetTypeInfo(), obj->GetSize(),
84              &field, field.GetFieldValue(), ref);
85     } else {
86         DLOG(BARRIER, "atomic write static ref@%p: %#zx -> %p", &field, field.GetFieldValue(), ref);
87     }
88 
89     field.SetTargetObject(ref, order);
90 }
91 
AtomicSwapRefField(BaseObject * obj,RefField<true> & field,BaseObject * newRef,MemoryOrder order) const92 BaseObject* Barrier::AtomicSwapRefField(BaseObject* obj, RefField<true>& field, BaseObject* newRef,
93                                         MemoryOrder order) const
94 {
95     HeapAddress oldValue = field.Exchange(newRef, order);
96     RefField<> oldField(oldValue);
97     BaseObject* oldRef = ReadRefField(nullptr, oldField);
98     DLOG(BARRIER, "atomic swap obj %p<%p>(%zu) ref-field@%p: old %#zx(%p), new %#zx(%p)", obj, obj->GetTypeInfo(),
99          obj->GetSize(), &field, oldValue, oldRef, field.GetFieldValue(), newRef);
100     return oldRef;
101 }
102 
AtomicReadRefField(BaseObject * obj,RefField<true> & field,MemoryOrder order) const103 BaseObject* Barrier::AtomicReadRefField(BaseObject* obj, RefField<true>& field, MemoryOrder order) const
104 {
105     RefField<false> tmpField(field.GetFieldValue(order));
106     HeapAddress target = tmpField.GetFieldValue();
107     DLOG(BARRIER, "atomic read obj %p ref@%p: %#zx -> %p", obj, &field, tmpField.GetFieldValue(), target);
108     return (BaseObject*)target;
109 }
110 
CompareAndSwapRefField(BaseObject * obj,RefField<true> & field,BaseObject * oldRef,BaseObject * newRef,MemoryOrder succOrder,MemoryOrder failOrder) const111 bool Barrier::CompareAndSwapRefField(BaseObject* obj, RefField<true>& field, BaseObject* oldRef, BaseObject* newRef,
112                                      MemoryOrder succOrder, MemoryOrder failOrder) const
113 {
114     HeapAddress oldFieldValue = field.GetFieldValue(std::memory_order_seq_cst);
115     RefField<false> oldField(oldFieldValue);
116     BaseObject* oldVersion = ReadRefField(nullptr, oldField);
117     (void)oldVersion;
118     bool res = field.CompareExchange(oldRef, newRef, succOrder, failOrder);
119     DLOG(BARRIER, "cas %u for obj %p reffield@%p: old %#zx->%p, expect %p, new %p", res, obj, &field, oldFieldValue,
120          oldVersion, oldRef, newRef);
121     return res;
122 }
123 
CopyStructArray(BaseObject * dstObj,HeapAddress dstField,MIndex dstSize,BaseObject * srcObj,HeapAddress srcField,MIndex srcSize) const124 void Barrier::CopyStructArray(BaseObject* dstObj, HeapAddress dstField, MIndex dstSize, BaseObject* srcObj,
125                               HeapAddress srcField, MIndex srcSize) const
126 {
127     LOGF_CHECK(memmove_s(reinterpret_cast<void*>(dstField), dstSize, reinterpret_cast<void*>(srcField), srcSize) ==
128         EOK) << "memmove_s failed";
129 #if defined(COMMON_TSAN_SUPPORT)
130     size_t copyLen = (dstSize < srcSize ? dstSize : srcSize);
131     Sanitizer::TsanWriteMemoryRange(reinterpret_cast<void*>(dstField), copyLen);
132     Sanitizer::TsanReadMemoryRange(reinterpret_cast<void*>(srcField), copyLen);
133 #endif
134 }
135 
ReadStruct(HeapAddress dst,BaseObject * obj,HeapAddress src,size_t size) const136 void Barrier::ReadStruct(HeapAddress dst, BaseObject* obj, HeapAddress src, size_t size) const
137 {
138     size_t dstSize = size;
139     size_t srcSize = size;
140     if (obj != nullptr) {
141         obj->ForEachRefInStruct(
142             [this, obj](RefField<false>& field) {
143                 // HeapAddress bias = reinterpret_cast<HeapAddress>(&field) - reinterpret_cast<HeapAddress>(src);
144                 // RefField<false>* dstField = reinterpret_cast<RefField<false>*>(dst + bias);
145                 BaseObject* fromVersion = field.GetTargetObject();
146                 (void)fromVersion;
147                 BaseObject* toVersion = nullptr;
148                 theCollector.TryUpdateRefField(obj, field, toVersion);
149             },
150             src, src + size);
151     }
152 
153     LOGF_CHECK(memcpy_s(reinterpret_cast<void*>(dst), dstSize, reinterpret_cast<void*>(src), srcSize) == EOK) <<
154         "read struct memcpy_s failed";
155 }
156 
157 } // namespace common
158