zeek/src/Scope.cc
2021-09-16 15:35:39 -07:00

202 lines
4.1 KiB
C++

// See the file "COPYING" in the main distribution directory for copyright.
#include "zeek/Scope.h"
#include "zeek/Desc.h"
#include "zeek/ID.h"
#include "zeek/IntrusivePtr.h"
#include "zeek/Reporter.h"
#include "zeek/Val.h"
#include "zeek/module_util.h"
#include "zeek/zeek-config.h"
namespace zeek::detail
{
static std::vector<ScopePtr> scopes;
static ScopePtr top_scope;
Scope::Scope(IDPtr id, std::unique_ptr<std::vector<AttrPtr>> al)
: scope_id(std::move(id)), attrs(std::move(al))
{
return_type = nullptr;
if ( id )
{
const auto& id_type = scope_id->GetType();
if ( id_type->Tag() == TYPE_ERROR )
return;
else if ( id_type->Tag() != TYPE_FUNC )
reporter->InternalError("bad scope id");
FuncType* ft = id->GetType()->AsFuncType();
return_type = ft->Yield();
}
}
const IDPtr& Scope::Find(std::string_view name) const
{
auto entry = local.find(name);
if ( entry != local.end() )
return entry->second;
return ID::nil;
}
IDPtr Scope::GenerateTemporary(const char* name)
{
return make_intrusive<ID>(name, SCOPE_FUNCTION, false);
}
std::vector<IDPtr> Scope::GetInits()
{
auto rval = std::move(inits);
inits = {};
return rval;
}
void Scope::Describe(ODesc* d) const
{
if ( d->IsReadable() )
d->AddSP("scope");
else
{
d->Add(scope_id != nullptr);
d->SP();
d->Add(return_type != nullptr);
d->SP();
d->Add(static_cast<uint64_t>(local.size()));
d->SP();
}
if ( scope_id )
{
scope_id->Describe(d);
d->NL();
}
if ( return_type )
{
return_type->Describe(d);
d->NL();
}
for ( const auto& entry : local )
{
ID* id = entry.second.get();
id->Describe(d);
d->NL();
}
}
TraversalCode Scope::Traverse(TraversalCallback* cb) const
{
for ( const auto& entry : local )
{
ID* id = entry.second.get();
TraversalCode tc = id->Traverse(cb);
HANDLE_TC_STMT_PRE(tc);
}
return TC_CONTINUE;
}
const IDPtr& lookup_ID(const char* name, const char* curr_module, bool no_global,
bool same_module_only, bool check_export)
{
std::string fullname = make_full_var_name(curr_module, name);
std::string ID_module = extract_module_name(fullname.c_str());
bool need_export =
check_export && (ID_module != GLOBAL_MODULE_NAME && ID_module != curr_module);
for ( auto s_i = scopes.rbegin(); s_i != scopes.rend(); ++s_i )
{
const auto& id = (*s_i)->Find(fullname);
if ( id )
{
if ( need_export && ! id->IsExport() && ! in_debug )
reporter->Error("identifier is not exported: %s", fullname.c_str());
return id;
}
}
if ( ! no_global && (strcmp(GLOBAL_MODULE_NAME, curr_module) == 0 || ! same_module_only) )
{
std::string globalname = make_full_var_name(GLOBAL_MODULE_NAME, name);
return global_scope()->Find(globalname);
}
return ID::nil;
}
IDPtr install_ID(const char* name, const char* module_name, bool is_global, bool is_export)
{
if ( scopes.empty() && ! is_global )
reporter->InternalError("local identifier in global scope");
IDScope scope;
if ( is_export || ! module_name ||
(is_global && normalized_module_name(module_name) == GLOBAL_MODULE_NAME) )
scope = SCOPE_GLOBAL;
else if ( is_global )
scope = SCOPE_MODULE;
else
scope = SCOPE_FUNCTION;
std::string full_name = make_full_var_name(module_name, name);
auto id = make_intrusive<ID>(full_name.data(), scope, is_export);
if ( SCOPE_FUNCTION != scope )
global_scope()->Insert(std::move(full_name), id);
else
{
id->SetOffset(top_scope->Length());
top_scope->Insert(std::move(full_name), id);
}
return id;
}
void push_existing_scope(ScopePtr scope)
{
top_scope = scope;
scopes.push_back(scope);
}
void push_scope(IDPtr id, std::unique_ptr<std::vector<AttrPtr>> attrs)
{
top_scope = make_intrusive<Scope>(std::move(id), std::move(attrs));
scopes.push_back(top_scope);
}
ScopePtr pop_scope()
{
if ( scopes.empty() )
reporter->InternalError("scope underflow");
scopes.pop_back();
auto old_top = top_scope;
top_scope = scopes.empty() ? nullptr : scopes.back();
return old_top;
}
ScopePtr current_scope()
{
return top_scope;
}
ScopePtr global_scope()
{
return scopes.empty() ? 0 : scopes.front();
}
} // namespace zeek::detail