32 #ifndef _QORE_LOCALVAR_H 34 #define _QORE_LOCALVAR_H 36 #include "qore/intern/qore_thread_intern.h" 37 #include "qore/intern/QoreLValue.h" 38 #include "qore/intern/RSection.h" 39 #include "qore/intern/RSet.h" 40 #include "qore/ReferenceNode.h" 41 #include "qore/intern/WeakReferenceNode.h" 46 class LocalRefHelper :
public RuntimeReferenceHelper {
53 : RuntimeReferenceHelper(ref, xsink),
57 DLLLOCAL
operator bool()
const {
63 class LValueRefHelper :
public LocalRefHelper<T> {
68 DLLLOCAL LValueRefHelper(T* val,
ExceptionSink* xsink) : LocalRefHelper<T>(val, xsink), valp(this->valid ? new LValueHelper(*((
ReferenceNode*)val->v.n), xsink) : nullptr) {
71 DLLLOCAL ~LValueRefHelper() {
75 DLLLOCAL
operator bool()
const {
79 DLLLOCAL LValueHelper* operator->() {
88 xsink->
raiseException(
"DESTRUCTOR-ERROR",
"illegal variable assignment after second phase of variable destruction");
95 QoreLValueGeneric val;
98 bool frame_boundary : 1;
100 DLLLOCAL VarValueBase(
const char* n_id, valtype_t t =
QV_Node) : val(t), id(n_id), finalized(false), frame_boundary(false) {
103 DLLLOCAL VarValueBase(
const char* n_id,
const QoreTypeInfo* varTypeInfo) : val(varTypeInfo), id(n_id), finalized(false), frame_boundary(false) {
106 DLLLOCAL VarValueBase() : val(
QV_Bool), id(nullptr), finalized(false), frame_boundary(false) {
109 DLLLOCAL
void setFrameBoundary() {
110 assert(!frame_boundary);
111 frame_boundary =
true;
115 val.removeValue(
true).discard(xsink);
118 DLLLOCAL
bool isRef()
const {
128 return val.removeValue(
true);
132 class LocalVarValue :
public VarValueBase {
134 DLLLOCAL
void set(
const char* n_id,
const QoreTypeInfo* varTypeInfo,
QoreValue nval,
bool assign,
bool static_assignment) {
141 val.set(varTypeInfo);
146 discard(val.assignAssumeInitial(nval, static_assignment),
nullptr);
148 assert(!val.assigned);
157 DLLLOCAL
void uninstantiateSelf() {
158 val.unassignIgnore();
161 DLLLOCAL
int getLValue(LValueHelper& lvh,
bool for_remove,
const QoreTypeInfo* typeInfo,
const QoreTypeInfo* refTypeInfo)
const;
162 DLLLOCAL
void remove(LValueRemoveHelper& lvrh,
const QoreTypeInfo* typeInfo);
168 LocalRefHelper<LocalVarValue> helper(
this, *ref, xsink);
173 return erh.takeValue(needs_deref);
178 return val.get<WeakReferenceNode>()->
get();
181 return val.getReferencedValue(needs_deref);
187 LocalRefHelper<LocalVarValue> helper(
this, *ref, xsink);
192 return *xsink ?
QoreValue() : erh.takeReferencedValue();
196 return val.get<WeakReferenceNode>()->
get()->refSelf();
199 return val.getReferencedValue();
203 hashdecl ClosureVarValue :
public VarValueBase,
public RObject {
205 const QoreTypeInfo* typeInfo =
nullptr;
206 const QoreTypeInfo* refTypeInfo;
208 mutable std::atomic_int references;
210 DLLLOCAL ClosureVarValue(
const char* n_id,
const QoreTypeInfo* varTypeInfo,
QoreValue& nval,
bool assign) : VarValueBase(n_id, varTypeInfo), RObject(references), typeInfo(varTypeInfo), refTypeInfo(QoreTypeInfo::getReferenceTarget(varTypeInfo)), references(1) {
215 val.set(varTypeInfo);
220 discard(val.assignAssumeInitial(nval),
nullptr);
223 assert(!val.assigned);
227 DLLLOCAL
virtual ~ClosureVarValue() {
231 DLLLOCAL
void ref()
const;
235 DLLLOCAL
const void* getLValueId()
const;
238 DLLLOCAL
virtual bool needsScan(
bool scan_now) {
239 return QoreTypeInfo::needsScan(typeInfo);
242 DLLLOCAL
virtual bool scanMembers(RSetHelper& rsh);
244 DLLLOCAL
int getLValue(LValueHelper& lvh,
bool for_remove)
const;
245 DLLLOCAL
void remove(LValueRemoveHelper& lvrh);
247 DLLLOCAL ClosureVarValue* refSelf()
const {
249 return const_cast<ClosureVarValue*
>(
this);
254 QoreSafeVarRWWriteLocker sl(rml);
255 return VarValueBase::finalize();
259 QoreSafeVarRWReadLocker sl(rml);
263 LocalRefHelper<ClosureVarValue> helper(
this, **ref, xsink);
264 return helper ? lvalue_ref::get(*ref)->vexp.eval(needs_deref, xsink) :
QoreValue();
269 return val.get<WeakReferenceNode>()->
get();
272 return val.getReferencedValue();
276 QoreSafeVarRWReadLocker sl(rml);
280 LocalRefHelper<ClosureVarValue> helper(
this, **ref, xsink);
281 return helper ? lvalue_ref::get(*ref)->vexp.eval(xsink) :
QoreValue();
285 return val.get<WeakReferenceNode>()->
get()->refSelf();
288 return val.getReferencedValue();
291 DLLLOCAL
AbstractQoreNode* getReference(
const QoreProgramLocation* loc,
const char* name,
const void*& lvalue_id);
294 DLLLOCAL
virtual void deleteObject() {
299 DLLLOCAL
virtual const char* getName()
const {
308 bool closure_use =
false,
309 parse_assigned =
false;
310 const QoreTypeInfo* typeInfo;
311 const QoreTypeInfo* refTypeInfo;
313 DLLLOCAL LocalVarValue* get_var()
const {
314 return thread_find_lvar(name.c_str());
318 DLLLOCAL LocalVar(
const char* n_name,
const QoreTypeInfo* ti) : name(n_name), typeInfo(ti), refTypeInfo(QoreTypeInfo::getReferenceTarget(ti)) {
321 DLLLOCAL LocalVar(
const LocalVar& old) : name(old.name), closure_use(old.closure_use), parse_assigned(old.parse_assigned), typeInfo(old.typeInfo), refTypeInfo(old.refTypeInfo) {
324 DLLLOCAL ~LocalVar() {
327 DLLLOCAL
void parseAssigned() {
329 parse_assigned =
true;
332 DLLLOCAL
void parseUnassigned() {
334 parse_assigned =
false;
337 DLLLOCAL
void instantiate() {
340 instantiateIntern(QoreTypeInfo::getDefaultQoreValue(typeInfo),
true);
347 DLLLOCAL
void instantiate(
QoreValue nval) {
348 instantiateIntern(nval,
true);
351 DLLLOCAL
void instantiateIntern(
QoreValue nval,
bool assign) {
355 LocalVarValue* val = thread_instantiate_lvar();
356 val->set(name.c_str(), typeInfo, nval, assign,
false);
358 thread_instantiate_closure_var(name.c_str(), typeInfo, nval, assign);
361 DLLLOCAL
void instantiateSelf(
QoreObject* value)
const {
362 printd(5,
"LocalVar::instantiateSelf(%p) this: %p '%s'\n", value,
this, name.c_str());
364 LocalVarValue* val = thread_instantiate_lvar();
365 val->set(name.c_str(), typeInfo, value,
true,
true);
368 thread_instantiate_closure_var(name.c_str(), typeInfo, val,
true);
376 thread_uninstantiate_lvar(xsink);
378 thread_uninstantiate_closure_var(xsink);
381 DLLLOCAL
void uninstantiateSelf()
const {
383 thread_uninstantiate_self();
385 thread_uninstantiate_closure_var(0);
390 LocalVarValue* val = get_var();
392 return val->eval(needs_deref, xsink);
395 ClosureVarValue* val = thread_find_closure_var(name.c_str());
396 return val->eval(needs_deref, xsink);
400 DLLLOCAL
bool needsScan()
const {
401 return QoreTypeInfo::needsScan(typeInfo);
404 DLLLOCAL
const char* getName()
const {
408 DLLLOCAL
const std::string& getNameStr()
const {
412 DLLLOCAL
void setClosureUse() {
416 DLLLOCAL
bool closureUse()
const {
420 DLLLOCAL
bool isRef()
const {
421 return !closure_use ? get_var()->isRef() : thread_find_closure_var(name.c_str())->isRef();
424 DLLLOCAL
int getLValue(LValueHelper& lvh,
bool for_remove,
bool initial_assignment)
const {
427 return get_var()->getLValue(lvh, for_remove, typeInfo, refTypeInfo);
430 return thread_find_closure_var(name.c_str())->getLValue(lvh, for_remove);
433 DLLLOCAL
void remove(LValueRemoveHelper& lvrh) {
435 return get_var()->remove(lvrh, typeInfo);
437 return thread_find_closure_var(name.c_str())->
remove(lvrh);
440 DLLLOCAL
const QoreTypeInfo* getTypeInfo()
const {
444 DLLLOCAL
const QoreTypeInfo* parseGetTypeInfo()
const {
445 return parse_assigned && refTypeInfo ? refTypeInfo : typeInfo;
448 DLLLOCAL
const QoreTypeInfo* parseGetTypeInfoForInitialAssignment()
const {
453 return !closure_use ? get_var()->val.getType() : thread_find_closure_var(name.c_str())->val.getType();
456 DLLLOCAL
const char* getValueTypeName()
const {
457 return !closure_use ? get_var()->val.getTypeName() : thread_find_closure_var(name.c_str())->val.getTypeName();
461 typedef LocalVar* lvar_ptr_t;
const qore_type_t NT_WEAKREF
type value for WeakReferenceNode
Definition: node_types.h:87
evaluates an AbstractQoreNode and dereferences the stored value in the destructor ...
Definition: QoreValue.h:593
The base class for all value and parse types in Qore expression trees.
Definition: AbstractQoreNode.h:54
static void discard(AbstractQoreNode *n, ExceptionSink *xsink)
to deref an AbstractQoreNode (when the pointer may be 0)
Definition: QoreLib.h:323
DLLEXPORT AbstractQoreNode * raiseException(const char *err, const char *fmt,...)
appends a Qore-language exception to the list
DLLEXPORT AbstractQoreNode * refSelf() const
returns "this" with an incremented reference count
DLLEXPORT QoreProgram * getProgram()
returns the current QoreProgram
#define QV_Node
for heap-allocated values
Definition: QoreValue.h:46
#define QV_Bool
for boolean values
Definition: QoreValue.h:43
parse type: reference to a lvalue expression
Definition: ReferenceNode.h:45
DLLEXPORT ReferenceNode * refRefSelf() const
returns a reference to itself
#define PO_STRICT_TYPES
enforce strict type checking and setting default values
Definition: Restrictions.h:98
const qore_type_t NT_REFERENCE
type value for ReferenceNode
Definition: node_types.h:64
The main value class in Qore, designed to be passed by value.
Definition: QoreValue.h:262
the implementation of Qore's object data type, reference counted, dynamically-allocated only ...
Definition: QoreObject.h:61
container for holding Qore-language exception information and also for registering a "thread_exit" ca...
Definition: ExceptionSink.h:46
int16_t qore_type_t
used to identify unique Qore data and parse types (descendents of AbstractQoreNode) ...
Definition: common.h:70
a templated class to manage a reference count of an object that can throw a Qore-language exception w...
Definition: ReferenceHolder.h:52