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"
46class LocalRefHelper :
public RuntimeReferenceHelper {
53 : RuntimeReferenceHelper(ref, xsink),
57 DLLLOCAL
operator bool()
const {
63class 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);
132class 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();
203hashdecl 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 {
307 DLLLOCAL LocalVar(
const char* n_name,
const QoreTypeInfo* ti)
308 : name(n_name), typeInfo(ti), refTypeInfo(QoreTypeInfo::getReferenceTarget(ti)) {
311 DLLLOCAL LocalVar(
const LocalVar& old) : name(old.name), closure_use(old.closure_use),
312 parse_assigned(old.parse_assigned), is_self(old.is_self), typeInfo(old.typeInfo),
313 refTypeInfo(old.refTypeInfo) {
316 DLLLOCAL ~LocalVar() {
319 DLLLOCAL
void parseAssigned() {
320 if (!parse_assigned) {
321 parse_assigned =
true;
325 DLLLOCAL
void parseUnassigned() {
326 if (parse_assigned) {
327 parse_assigned =
false;
331 DLLLOCAL
bool isAssigned()
const {
332 return parse_assigned;
335 DLLLOCAL
void instantiate() {
339 instantiateIntern(QoreTypeInfo::getDefaultQoreValue(typeInfo),
true);
347 DLLLOCAL
void instantiate(
QoreValue nval) {
348 instantiateIntern(nval,
true);
351 DLLLOCAL
void instantiateIntern(
QoreValue nval,
bool assign) {
358 LocalVarValue* val = thread_instantiate_lvar();
359 val->set(name.c_str(), typeInfo, nval, assign,
false);
361 thread_instantiate_closure_var(name.c_str(), typeInfo, nval, assign);
365 DLLLOCAL
void instantiateSelf(
QoreObject* value)
const {
366 printd(5,
"LocalVar::instantiateSelf(%p) this: %p '%s'\n", value,
this, name.c_str());
368 LocalVarValue* val = thread_instantiate_lvar();
369 val->set(name.c_str(), typeInfo, value,
true,
true);
372 thread_instantiate_closure_var(name.c_str(), typeInfo, val,
true);
381 thread_uninstantiate_lvar(xsink);
383 thread_uninstantiate_closure_var(xsink);
387 DLLLOCAL
void uninstantiateSelf()
const {
389 thread_uninstantiate_self();
391 thread_uninstantiate_closure_var(
nullptr);
397 LocalVarValue* val = get_var();
400 return val->eval(needs_deref, xsink);
403 ClosureVarValue* val = thread_find_closure_var(name.c_str());
404 return val->eval(needs_deref, xsink);
408 DLLLOCAL
bool needsScan()
const {
409 return QoreTypeInfo::needsScan(typeInfo);
412 DLLLOCAL
const char* getName()
const {
416 DLLLOCAL
const std::string& getNameStr()
const {
420 DLLLOCAL
void setClosureUse() {
424 DLLLOCAL
bool closureUse()
const {
428 DLLLOCAL
bool isRef()
const {
429 return !closure_use ? get_var()->isRef() : thread_find_closure_var(name.c_str())->isRef();
432 DLLLOCAL
int getLValue(LValueHelper& lvh,
bool for_remove,
bool initial_assignment)
const {
436 return get_var()->getLValue(lvh, for_remove, typeInfo, refTypeInfo);
439 return thread_find_closure_var(name.c_str())->getLValue(lvh, for_remove);
442 DLLLOCAL
void remove(LValueRemoveHelper& lvrh) {
444 return get_var()->remove(lvrh, typeInfo);
447 return thread_find_closure_var(name.c_str())->remove(lvrh);
450 DLLLOCAL
const QoreTypeInfo* getTypeInfo()
const {
454 DLLLOCAL
const QoreTypeInfo* parseGetTypeInfo()
const {
455 return parse_assigned && refTypeInfo ? refTypeInfo : typeInfo;
458 DLLLOCAL
const QoreTypeInfo* parseGetTypeInfoForInitialAssignment()
const {
463 return !closure_use ? get_var()->val.getType() : thread_find_closure_var(name.c_str())->val.getType();
466 DLLLOCAL
const char* getValueTypeName()
const {
467 return !closure_use ? get_var()->val.getTypeName() : thread_find_closure_var(name.c_str())->val.getTypeName();
470 DLLLOCAL
bool isSelf()
const {
474 DLLLOCAL
void setSelf() {
476 assert(name ==
"self");
482 bool closure_use =
false,
483 parse_assigned =
false,
485 const QoreTypeInfo* typeInfo =
nullptr;
486 const QoreTypeInfo* refTypeInfo =
nullptr;
488 DLLLOCAL LocalVarValue* get_var()
const {
489 return thread_find_lvar(name.c_str());
493typedef LocalVar* lvar_ptr_t;
static void discard(AbstractQoreNode *n, ExceptionSink *xsink)
to deref an AbstractQoreNode (when the pointer may be 0)
Definition: QoreLib.h:327
#define PO_STRICT_TYPES
enforce strict type checking and setting default values
Definition: Restrictions.h:98
The base class for all value and parse types in Qore expression trees.
Definition: AbstractQoreNode.h:57
DLLEXPORT AbstractQoreNode * refSelf() const
returns "this" with an incremented reference count
container for holding Qore-language exception information and also for registering a "thread_exit" ca...
Definition: ExceptionSink.h:50
DLLEXPORT AbstractQoreNode * raiseException(const char *err, const char *fmt,...)
appends a Qore-language exception to the list
the implementation of Qore's object data type, reference counted, dynamically-allocated only
Definition: QoreObject.h:60
a templated class to manage a reference count of an object that can throw a Qore-language exception w...
Definition: ReferenceHolder.h:52
parse type: reference to a lvalue expression
Definition: ReferenceNode.h:45
DLLEXPORT ReferenceNode * refRefSelf() const
returns a reference to itself
evaluates an AbstractQoreNode and dereferences the stored value in the destructor
Definition: QoreValue.h:624
int16_t qore_type_t
used to identify unique Qore data and parse types (descendents of AbstractQoreNode)
Definition: common.h:70
#define QV_Node
for heap-allocated values
Definition: QoreValue.h:46
#define QV_Bool
for boolean values
Definition: QoreValue.h:43
const qore_type_t NT_WEAKREF
type value for WeakReferenceNode
Definition: node_types.h:87
const qore_type_t NT_REFERENCE
type value for ReferenceNode
Definition: node_types.h:64
DLLEXPORT QoreProgram * getProgram()
returns the current QoreProgram
The main value class in Qore, designed to be passed by value.
Definition: QoreValue.h:276