mirror of
https://github.com/zeek/zeek.git
synced 2025-10-05 08:08:19 +00:00
616 lines
19 KiB
C++
616 lines
19 KiB
C++
// See the file "COPYING" in the main distribution directory for copyright.
|
|
|
|
// ZAM methods associated with instructions that replace calls to
|
|
// built-in functions.
|
|
|
|
#include "zeek/script_opt/ZAM/BuiltIn.h"
|
|
|
|
#include "zeek/Func.h"
|
|
#include "zeek/Reporter.h"
|
|
#include "zeek/script_opt/ZAM/Compile.h"
|
|
|
|
namespace zeek::detail {
|
|
|
|
// Maps BiF names to their associated ZBI class.
|
|
std::unordered_map<std::string, const ZAMBuiltIn*> builtins;
|
|
|
|
ZAMBuiltIn::ZAMBuiltIn(std::string name, bool _ret_val_matters) : ret_val_matters(_ret_val_matters) {
|
|
builtins[name] = this;
|
|
}
|
|
|
|
SimpleZBI::SimpleZBI(std::string name, ZOp _op, int _nargs, bool _ret_val_matters)
|
|
: ZAMBuiltIn(std::move(name), _ret_val_matters), op(_op), nargs(_nargs) {}
|
|
|
|
SimpleZBI::SimpleZBI(std::string name, ZOp _const_op, ZOp _op, bool _ret_val_matters)
|
|
: ZAMBuiltIn(std::move(name), _ret_val_matters), op(_op), const_op(_const_op), nargs(1) {}
|
|
|
|
bool SimpleZBI::Build(ZAMCompiler* zam, const NameExpr* n, const ExprPList& args) const {
|
|
ZInstI z;
|
|
if ( nargs == 0 ) {
|
|
if ( n ) {
|
|
z = ZInstI(op, zam->Frame1Slot(n, OP1_WRITE));
|
|
z.is_managed = ZVal::IsManagedType(n->GetType());
|
|
}
|
|
else
|
|
z = ZInstI(op);
|
|
}
|
|
else {
|
|
ASSERT(nargs == 1);
|
|
auto& t = args[0]->GetType();
|
|
|
|
if ( args[0]->Tag() == EXPR_NAME ) {
|
|
auto a0 = zam->FrameSlot(args[0]->AsNameExpr());
|
|
if ( n )
|
|
z = ZInstI(op, zam->Frame1Slot(n, OP1_WRITE), a0);
|
|
else
|
|
z = ZInstI(op, a0);
|
|
}
|
|
|
|
else {
|
|
if ( const_op == OP_NOP )
|
|
// This can happen for BiFs that aren't foldable, and for
|
|
// which it's implausible they'll be called with a constant
|
|
// argument.
|
|
return false;
|
|
|
|
if ( n )
|
|
z = ZInstI(const_op, zam->Frame1Slot(n, OP1_WRITE));
|
|
else
|
|
z = ZInstI(const_op);
|
|
|
|
z.c = ZVal(args[0]->AsConstExpr()->ValuePtr(), t);
|
|
}
|
|
|
|
z.SetType(t);
|
|
}
|
|
|
|
zam->AddInst(z);
|
|
|
|
return true;
|
|
}
|
|
|
|
CondZBI::CondZBI(std::string name, ZOp _op, ZOp _cond_op, int _nargs)
|
|
: SimpleZBI(std::move(name), _op, _nargs, true), cond_op(_cond_op) {}
|
|
|
|
bool CondZBI::BuildCond(ZAMCompiler* zam, const ExprPList& args, int& branch_v) const {
|
|
if ( cond_op == OP_NOP )
|
|
return false;
|
|
|
|
if ( nargs == 1 && args[0]->Tag() != EXPR_NAME )
|
|
// ZBI-worthy predicates called with constant arguments will generally
|
|
// have been folded. If not, for simplicity we don't support the
|
|
// flavor where they're called with a constant.
|
|
return false;
|
|
|
|
// If we get here, then the ZBI is good-to-go.
|
|
|
|
if ( ! zam )
|
|
// This was just a check, not an actual build.
|
|
return true;
|
|
|
|
ZInstI z;
|
|
|
|
if ( nargs == 0 ) {
|
|
z = ZInstI(cond_op, 0);
|
|
z.op_type = OP_V_I1;
|
|
branch_v = 1;
|
|
}
|
|
|
|
else {
|
|
ASSERT(nargs == 1);
|
|
|
|
auto a0 = args[0];
|
|
auto a0_slot = zam->FrameSlot(a0->AsNameExpr());
|
|
z = ZInstI(cond_op, a0_slot, 0);
|
|
z.op_type = OP_VV_I2;
|
|
z.SetType(a0->GetType());
|
|
branch_v = 2;
|
|
}
|
|
|
|
zam->AddInst(z);
|
|
|
|
return true;
|
|
}
|
|
|
|
OptAssignZBI::OptAssignZBI(std::string name, ZOp _op, ZOp _op2, int _nargs)
|
|
: SimpleZBI(std::move(name), _op, _nargs, false), op2(_op2) {
|
|
have_both = true;
|
|
}
|
|
|
|
bool OptAssignZBI::Build(ZAMCompiler* zam, const NameExpr* n, const ExprPList& args) const {
|
|
if ( n )
|
|
return SimpleZBI::Build(zam, n, args);
|
|
|
|
ZInstI z;
|
|
if ( nargs == 0 )
|
|
z = ZInstI(op2);
|
|
else {
|
|
ASSERT(nargs == 1);
|
|
auto a0 = zam->FrameSlot(args[0]->AsNameExpr());
|
|
z = ZInstI(op2, a0);
|
|
z.SetType(args[0]->GetType());
|
|
}
|
|
|
|
zam->AddInst(z);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CatZBI::Build(ZAMCompiler* zam, const NameExpr* n, const ExprPList& args) const {
|
|
auto nslot = zam->Frame1Slot(n, OP1_WRITE);
|
|
auto& a0 = args[0];
|
|
ZInstI z;
|
|
|
|
if ( args.empty() ) {
|
|
// Weird, but easy enough to support.
|
|
z = ZInstI(OP_CAT1_VC, nslot);
|
|
z.SetType(n->GetType());
|
|
z.c = ZVal(val_mgr->EmptyString());
|
|
}
|
|
|
|
else if ( args.size() > 1 ) {
|
|
switch ( args.size() ) {
|
|
case 2: z = zam->GenInst(OP_CAT2_V, n); break;
|
|
case 3: z = zam->GenInst(OP_CAT3_V, n); break;
|
|
case 4: z = zam->GenInst(OP_CAT4_V, n); break;
|
|
case 5: z = zam->GenInst(OP_CAT5_V, n); break;
|
|
case 6: z = zam->GenInst(OP_CAT6_V, n); break;
|
|
case 7: z = zam->GenInst(OP_CAT7_V, n); break;
|
|
case 8: z = zam->GenInst(OP_CAT8_V, n); break;
|
|
|
|
default: z = zam->GenInst(OP_CATN_V, n); break;
|
|
}
|
|
|
|
z.aux = BuildCatAux(zam, args);
|
|
}
|
|
|
|
else if ( a0->GetType()->Tag() != TYPE_STRING ) {
|
|
if ( a0->Tag() == EXPR_NAME ) {
|
|
z = zam->GenInst(OP_CAT1FULL_VV, n, a0->AsNameExpr());
|
|
z.SetType(a0->GetType());
|
|
}
|
|
else {
|
|
z = ZInstI(OP_CAT1_VC, nslot);
|
|
z.SetType(n->GetType());
|
|
z.c = ZVal(ZAM_val_cat(a0->AsConstExpr()->ValuePtr()));
|
|
}
|
|
}
|
|
|
|
else if ( a0->Tag() == EXPR_CONST ) {
|
|
z = zam->GenInst(OP_CAT1_VC, n, a0->AsConstExpr());
|
|
z.SetType(n->GetType());
|
|
}
|
|
|
|
else
|
|
z = zam->GenInst(OP_CAT1_VV, n, a0->AsNameExpr());
|
|
|
|
zam->AddInst(z);
|
|
|
|
return true;
|
|
}
|
|
|
|
ZInstAux* CatZBI::BuildCatAux(ZAMCompiler* zam, const ExprPList& args) const {
|
|
auto n = args.size();
|
|
auto aux = new ZInstAux(n);
|
|
aux->cat_args = new std::unique_ptr<CatArg>[n];
|
|
|
|
for ( size_t i = 0; i < n; ++i ) {
|
|
auto& a_i = args[i];
|
|
auto& t = a_i->GetType();
|
|
|
|
std::unique_ptr<CatArg> ca;
|
|
|
|
if ( a_i->Tag() == EXPR_CONST ) {
|
|
auto c = a_i->AsConstExpr()->ValuePtr();
|
|
aux->Add(i, c); // we add it to consume a slot, but it'll be ignored
|
|
|
|
// Convert it up front and transform into a fixed string.
|
|
auto sv = ZAM_val_cat(c);
|
|
auto s = sv->AsString();
|
|
auto b = reinterpret_cast<char*>(s->Bytes());
|
|
ca = std::make_unique<CatArg>(std::string(b, s->Len()));
|
|
}
|
|
|
|
else {
|
|
auto slot = zam->FrameSlot(a_i->AsNameExpr());
|
|
aux->Add(i, slot, t);
|
|
|
|
switch ( t->Tag() ) {
|
|
case TYPE_BOOL:
|
|
case TYPE_INT:
|
|
case TYPE_COUNT:
|
|
case TYPE_DOUBLE:
|
|
case TYPE_TIME:
|
|
case TYPE_ENUM:
|
|
case TYPE_PORT:
|
|
case TYPE_ADDR:
|
|
case TYPE_SUBNET: ca = std::make_unique<FixedCatArg>(t); break;
|
|
|
|
case TYPE_STRING: ca = std::make_unique<StringCatArg>(); break;
|
|
|
|
case TYPE_PATTERN: ca = std::make_unique<PatternCatArg>(); break;
|
|
|
|
default: ca = std::make_unique<DescCatArg>(t); break;
|
|
}
|
|
}
|
|
|
|
aux->cat_args[i] = std::move(ca);
|
|
}
|
|
|
|
return aux;
|
|
}
|
|
|
|
bool SortZBI::Build(ZAMCompiler* zam, const NameExpr* n, const ExprPList& args) const {
|
|
// The checks the sort() BiF does can all be computed statically.
|
|
if ( args.size() > 2 )
|
|
return false;
|
|
|
|
auto v = args[0]->AsNameExpr();
|
|
if ( v->GetType()->Tag() != TYPE_VECTOR )
|
|
return false;
|
|
|
|
const auto& elt_type = v->GetType()->Yield();
|
|
|
|
if ( args.size() == 1 ) {
|
|
if ( ! IsIntegral(elt_type->Tag()) && elt_type->InternalType() != TYPE_INTERNAL_DOUBLE )
|
|
return false;
|
|
|
|
return OptAssignZBI::Build(zam, n, args);
|
|
}
|
|
|
|
// If we get here, then there's a comparison function.
|
|
const auto& comp_val = args[1];
|
|
if ( ! IsFunc(comp_val->GetType()->Tag()) )
|
|
return false;
|
|
|
|
if ( comp_val->Tag() != EXPR_NAME )
|
|
return false;
|
|
|
|
auto comp_func = comp_val->AsNameExpr();
|
|
auto comp_type = comp_func->GetType()->AsFuncType();
|
|
|
|
if ( comp_type->Yield()->Tag() != TYPE_INT || ! comp_type->ParamList()->AllMatch(elt_type, false) ||
|
|
comp_type->ParamList()->GetTypes().size() != 2 )
|
|
return false;
|
|
|
|
ZInstI z;
|
|
|
|
if ( n )
|
|
z = ZInstI(OP_SORT_WITH_CMP_VVV, zam->Frame1Slot(n, OP1_WRITE), zam->FrameSlot(v), zam->FrameSlot(comp_func));
|
|
else
|
|
z = ZInstI(OP_SORT_WITH_CMP_VV, zam->FrameSlot(v), zam->FrameSlot(comp_func));
|
|
|
|
zam->AddInst(z);
|
|
|
|
return true;
|
|
}
|
|
|
|
MultiZBI::MultiZBI(std::string name, bool _ret_val_matters, BiFArgsInfo _args_info, int _type_arg)
|
|
: ZAMBuiltIn(std::move(name), _ret_val_matters), args_info(std::move(_args_info)), type_arg(_type_arg) {}
|
|
|
|
MultiZBI::MultiZBI(std::string name, BiFArgsInfo _args_info, BiFArgsInfo _assign_args_info, int _type_arg)
|
|
: MultiZBI(std::move(name), false, std::move(_args_info), _type_arg) {
|
|
assign_args_info = std::move(_assign_args_info);
|
|
have_both = true;
|
|
}
|
|
|
|
bool MultiZBI::Build(ZAMCompiler* zam, const NameExpr* n, const ExprPList& args) const {
|
|
auto ai = &args_info;
|
|
if ( n && have_both ) {
|
|
ai = &assign_args_info;
|
|
ASSERT(! ai->empty());
|
|
}
|
|
|
|
auto bif_arg_info = ai->find(ComputeArgsType(args));
|
|
if ( bif_arg_info == ai->end() )
|
|
// Not a Constant/Variable combination this ZBI supports.
|
|
return false;
|
|
|
|
const auto& bi = bif_arg_info->second;
|
|
auto op = bi.op;
|
|
|
|
std::vector<ValPtr> consts;
|
|
std::vector<int> v;
|
|
|
|
for ( const auto& a : args ) {
|
|
if ( a->Tag() == EXPR_NAME )
|
|
v.push_back(zam->FrameSlot(a->AsNameExpr()));
|
|
else
|
|
consts.push_back(a->AsConstExpr()->ValuePtr());
|
|
}
|
|
|
|
auto nslot = n ? zam->Frame1Slot(n, OP1_WRITE) : -1;
|
|
|
|
ZInstI z;
|
|
|
|
if ( args.size() == 2 ) {
|
|
if ( consts.empty() ) {
|
|
if ( n )
|
|
z = ZInstI(op, nslot, v[0], v[1]);
|
|
else
|
|
z = ZInstI(op, v[0], v[1]);
|
|
}
|
|
else {
|
|
ASSERT(consts.size() == 1);
|
|
if ( n )
|
|
z = ZInstI(op, nslot, v[0]);
|
|
else
|
|
z = ZInstI(op, v[0]);
|
|
}
|
|
}
|
|
|
|
else if ( args.size() == 3 ) {
|
|
switch ( consts.size() ) {
|
|
case 0:
|
|
if ( n )
|
|
z = ZInstI(op, nslot, v[0], v[1], v[2]);
|
|
else
|
|
z = ZInstI(op, v[0], v[1], v[2]);
|
|
break;
|
|
|
|
case 1:
|
|
if ( n )
|
|
z = ZInstI(op, nslot, v[0], v[1]);
|
|
else
|
|
z = ZInstI(op, v[0], v[1]);
|
|
break;
|
|
|
|
case 2: {
|
|
const auto& c2 = consts[1];
|
|
auto c2_t = c2->GetType()->Tag();
|
|
|
|
ASSERT(c2_t == TYPE_BOOL || c2_t == TYPE_INT || c2_t == TYPE_COUNT);
|
|
int slot_val;
|
|
if ( c2_t == TYPE_COUNT )
|
|
slot_val = static_cast<int>(c2->AsCount());
|
|
else
|
|
slot_val = c2->AsInt();
|
|
|
|
if ( n )
|
|
z = ZInstI(op, nslot, v[0], slot_val);
|
|
else
|
|
z = ZInstI(op, v[0], slot_val);
|
|
break;
|
|
}
|
|
|
|
default: reporter->InternalError("inconsistency in MultiZBI::Build");
|
|
}
|
|
}
|
|
|
|
else
|
|
reporter->InternalError("inconsistency in MultiZBI::Build");
|
|
|
|
z.op_type = bi.op_type;
|
|
|
|
if ( n )
|
|
z.is_managed = ZVal::IsManagedType(n->GetType());
|
|
|
|
if ( ! consts.empty() ) {
|
|
z.SetType(consts[0]->GetType());
|
|
z.c = ZVal(consts[0], z.GetType());
|
|
}
|
|
|
|
if ( type_arg >= 0 && ! z.GetType() )
|
|
z.SetType(args[type_arg]->GetType());
|
|
|
|
zam->AddInst(z);
|
|
|
|
return true;
|
|
}
|
|
|
|
BiFArgsType MultiZBI::ComputeArgsType(const ExprPList& args) const {
|
|
zeek_uint_t mask = 0;
|
|
|
|
for ( const auto& a : args ) {
|
|
mask <<= 1;
|
|
if ( a->Tag() == EXPR_CONST )
|
|
mask |= 1;
|
|
}
|
|
|
|
return BiFArgsType(mask);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// To create a new built-in, add it to the following collection. We chose
|
|
// this style with an aim to making the entries both easy to update & readable.
|
|
// The names of the variables don't matter, so we keep them short to aid
|
|
// readability.
|
|
|
|
SimpleZBI an_ZBI{"Analyzer::__name", OP_ANALYZER_NAME_VC, OP_ANALYZER_NAME_VV};
|
|
SimpleZBI ae_ZBI{"Files::__analyzer_enabled", OP_ANALYZER_ENABLED_VC, OP_ANALYZER_ENABLED_VV};
|
|
SimpleZBI fan_ZBI{"Files::__analyzer_name", OP_FILE_ANALYZER_NAME_VC, OP_FILE_ANALYZER_NAME_VV};
|
|
SimpleZBI fer_ZBI{"Files::__enable_reassembly", OP_FILES_ENABLE_REASSEMBLY_V, 1, false};
|
|
SimpleZBI ct_ZBI{"clear_table", OP_CLEAR_TABLE_V, 1, false};
|
|
SimpleZBI currt_ZBI{"current_time", OP_CURRENT_TIME_V, 0};
|
|
SimpleZBI gptp_ZBI{"get_port_transport_proto", OP_GET_PORT_TRANSPORT_PROTO_VV, 1};
|
|
SimpleZBI ipa_ZBI{"is_protocol_analyzer", OP_IS_PROTOCOL_ANALYZER_VC, OP_IS_PROTOCOL_ANALYZER_VV, true};
|
|
SimpleZBI lc_ZBI{"lookup_connection", OP_LOOKUP_CONN_VV, 1};
|
|
SimpleZBI nt_ZBI{"network_time", OP_NETWORK_TIME_V, 0};
|
|
SimpleZBI sfh_ZBI{"set_file_handle", OP_SET_FILE_HANDLE_V, 1, false};
|
|
SimpleZBI sta_ZBI{"subnet_to_addr", OP_SUBNET_TO_ADDR_VV, 1};
|
|
SimpleZBI ttd_ZBI{"time_to_double", OP_TIME_TO_DOUBLE_VV, 1};
|
|
SimpleZBI tl_ZBI{"to_lower", OP_TO_LOWER_VV, 1};
|
|
|
|
CondZBI ce_ZBI{"connection_exists", OP_CONN_EXISTS_VV, OP_CONN_EXISTS_COND_Vb, 1};
|
|
CondZBI iip_ZBI{"is_icmp_port", OP_IS_ICMP_PORT_VV, OP_IS_ICMP_PORT_COND_Vb, 1};
|
|
CondZBI itp_ZBI{"is_tcp_port", OP_IS_TCP_PORT_VV, OP_IS_TCP_PORT_COND_Vb, 1};
|
|
CondZBI iup_ZBI{"is_udp_port", OP_IS_UDP_PORT_VV, OP_IS_UDP_PORT_COND_Vb, 1};
|
|
CondZBI iv4_ZBI{"is_v4_addr", OP_IS_V4_ADDR_VV, OP_IS_V4_ADDR_COND_Vb, 1};
|
|
CondZBI iv6_ZBI{"is_v6_addr", OP_IS_V6_ADDR_VV, OP_IS_V6_ADDR_COND_Vb, 1};
|
|
CondZBI rlt_ZBI{"reading_live_traffic", OP_READING_LIVE_TRAFFIC_V, OP_READING_LIVE_TRAFFIC_COND_b, 0};
|
|
CondZBI rt_ZBI{"reading_traces", OP_READING_TRACES_V, OP_READING_TRACES_COND_b, 0};
|
|
|
|
// These have a different form to avoid invoking copy constructors.
|
|
auto cat_ZBI = CatZBI();
|
|
auto sort_ZBI = SortZBI();
|
|
|
|
// For the following, clang-format makes them hard to follow compared to
|
|
// a manual layout.
|
|
//
|
|
// clang-format off
|
|
|
|
OptAssignZBI bfl_ZBI{ "Broker::__flush_logs",
|
|
OP_BROKER_FLUSH_LOGS_V, OP_BROKER_FLUSH_LOGS_X,
|
|
0
|
|
};
|
|
|
|
MultiZBI fra_ZBI{ "Files::__remove_analyzer",
|
|
{{{VVV}, {OP_FILES_REMOVE_ANALYZER_VVV, OP_VVV}},
|
|
{{VCV}, {OP_FILES_REMOVE_ANALYZER_VCV, OP_VVC}}},
|
|
{{{VVV}, {OP_FILES_REMOVE_ANALYZER_VVVV, OP_VVVV}},
|
|
{{VCV}, {OP_FILES_REMOVE_ANALYZER_VVCV, OP_VVVC}}},
|
|
1
|
|
};
|
|
|
|
MultiZBI fsrb_ZBI{ "Files::__set_reassembly_buffer",
|
|
{{{VV}, {OP_FILES_SET_REASSEMBLY_BUFFER_VV, OP_VV}},
|
|
{{VC}, {OP_FILES_SET_REASSEMBLY_BUFFER_Vi, OP_VV_I2}}},
|
|
{{{VV}, {OP_FILES_SET_REASSEMBLY_BUFFER_VVV, OP_VVV}},
|
|
{{VC}, {OP_FILES_SET_REASSEMBLY_BUFFER_VVi, OP_VVV_I3}}}
|
|
};
|
|
|
|
MultiZBI lw_ZBI{ "Log::__write",
|
|
{{{VV}, {OP_LOG_WRITE_VV, OP_VV}},
|
|
{{CV}, {OP_LOG_WRITE_CV, OP_V}}},
|
|
{{{VV}, {OP_LOG_WRITE_VVV, OP_VVV}},
|
|
{{CV}, {OP_LOG_WRITEC_VCV, OP_VV}}}
|
|
};
|
|
|
|
MultiZBI gccbt_ZBI{ "get_current_conn_bytes_threshold", true,
|
|
{{{VV}, {OP_GET_BYTES_THRESH_VVV, OP_VVV}},
|
|
{{VC}, {OP_GET_BYTES_THRESH_VVC, OP_VVC}}}
|
|
};
|
|
|
|
MultiZBI sccbt_ZBI{ "set_current_conn_bytes_threshold",
|
|
{{{VVV}, {OP_SET_BYTES_THRESH_VVV, OP_VVV}},
|
|
{{VVC}, {OP_SET_BYTES_THRESH_VVC, OP_VVC}},
|
|
{{VCV}, {OP_SET_BYTES_THRESH_VCV, OP_VVC}},
|
|
{{VCC}, {OP_SET_BYTES_THRESH_VCi, OP_VVC_I2}}},
|
|
{{{VVV}, {OP_SET_BYTES_THRESH_VVVV, OP_VVVV}},
|
|
{{VVC}, {OP_SET_BYTES_THRESH_VVVC, OP_VVVC}},
|
|
{{VCV}, {OP_SET_BYTES_THRESH_VVCV, OP_VVVC}},
|
|
{{VCC}, {OP_SET_BYTES_THRESH_VVCi, OP_VVVC_I3}}}
|
|
};
|
|
|
|
MultiZBI sw_ZBI{ "starts_with", true,
|
|
{{{VV}, {OP_STARTS_WITH_VVV, OP_VVV}},
|
|
{{VC}, {OP_STARTS_WITH_VVC, OP_VVC}},
|
|
{{CV}, {OP_STARTS_WITH_VCV, OP_VVC}}}
|
|
};
|
|
|
|
MultiZBI strcmp_ZBI{ "strcmp", true,
|
|
{{{VV}, {OP_STRCMP_VVV, OP_VVV}},
|
|
{{VC}, {OP_STRCMP_VVC, OP_VVC}},
|
|
{{CV}, {OP_STRCMP_VCV, OP_VVC}}}
|
|
};
|
|
|
|
MultiZBI strstr_ZBI{ "strstr", true,
|
|
{{{VV}, {OP_STRSTR_VVV, OP_VVV}},
|
|
{{VC}, {OP_STRSTR_VVC, OP_VVC}},
|
|
{{CV}, {OP_STRSTR_VCV, OP_VVC}}}
|
|
};
|
|
|
|
MultiZBI sb_ZBI{ "sub_bytes", true,
|
|
{{{VVV}, {OP_SUB_BYTES_VVVV, OP_VVVV}},
|
|
{{VVC}, {OP_SUB_BYTES_VVVC, OP_VVVC}},
|
|
{{VCV}, {OP_SUB_BYTES_VVCV, OP_VVVC}},
|
|
{{VCC}, {OP_SUB_BYTES_VVCi, OP_VVVC_I3}},
|
|
{{CVV}, {OP_SUB_BYTES_VCVV, OP_VVVC}},
|
|
{{CVC}, {OP_SUB_BYTES_VCVi, OP_VVVC_I3}},
|
|
{{CCV}, {OP_SUB_BYTES2_VCVi, OP_VVVC_I3}}}
|
|
};
|
|
|
|
// clang-format on
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Helper function that extracts the underlying Func* from a CallExpr
|
|
// node. Returns nil if it's not accessible.
|
|
static const Func* get_func(const CallExpr* c) {
|
|
auto func_expr = c->Func();
|
|
if ( func_expr->Tag() != EXPR_NAME )
|
|
// An indirect call.
|
|
return nullptr;
|
|
|
|
auto func_val = func_expr->AsNameExpr()->Id()->GetVal();
|
|
if ( ! func_val )
|
|
// A call to a function that hasn't been defined.
|
|
return nullptr;
|
|
|
|
return func_val->AsFunc();
|
|
}
|
|
|
|
bool IsZAM_BuiltIn(ZAMCompiler* zam, const Expr* e) {
|
|
// The expression e is either directly a call (in which case there's
|
|
// no return value), or an assignment to a call.
|
|
const CallExpr* c;
|
|
|
|
if ( e->Tag() == EXPR_CALL )
|
|
c = e->AsCallExpr();
|
|
else
|
|
c = e->GetOp2()->AsCallExpr();
|
|
|
|
auto func = get_func(c);
|
|
if ( ! func )
|
|
return false;
|
|
|
|
auto fn = func->GetName();
|
|
|
|
// It's useful to intercept any lingering calls to the script-level
|
|
// Log::write as well as the Log::__write BiF. When inlining there can
|
|
// still be script-level calls if the calling function got too big to
|
|
// inline them. We could do this for other script-level functions that
|
|
// are simply direct wrappers for BiFs, but this is only one that has
|
|
// turned up as significant in profiling.
|
|
if ( fn == "Log::write" )
|
|
fn = "Log::__write";
|
|
|
|
auto b = builtins.find(fn);
|
|
if ( b == builtins.end() )
|
|
return false;
|
|
|
|
const auto& bi = b->second;
|
|
|
|
const NameExpr* n = nullptr; // name to assign to, if any
|
|
if ( e->Tag() != EXPR_CALL )
|
|
n = e->GetOp1()->AsRefExpr()->GetOp1()->AsNameExpr();
|
|
|
|
if ( bi->ReturnValMatters() ) {
|
|
if ( ! n ) {
|
|
reporter->Warning("return value from built-in function ignored");
|
|
|
|
// The call is a no-op. We could return false here and have it
|
|
// execute (for no purpose). We can also return true, which will
|
|
// have the effect of just ignoring the statement.
|
|
return true;
|
|
}
|
|
}
|
|
else if ( n && ! bi->HaveBothReturnValAndNon() )
|
|
// Because the return value "doesn't matter", we've built the
|
|
// corresponding ZIB assuming we don't need a version that does
|
|
// the assignment. If we *do* have an assignment, let the usual
|
|
// call take place.
|
|
return false;
|
|
|
|
return bi->Build(zam, n, c->Args()->Exprs());
|
|
}
|
|
|
|
bool IsZAM_BuiltInCond(ZAMCompiler* zam, const CallExpr* c, int& branch_v) {
|
|
auto func = get_func(c);
|
|
if ( ! func )
|
|
return false;
|
|
|
|
auto b = builtins.find(func->GetName());
|
|
if ( b == builtins.end() )
|
|
return false;
|
|
|
|
return b->second->BuildCond(zam, c->Args()->Exprs(), branch_v);
|
|
}
|
|
|
|
bool IsZAM_BuiltInCond(const CallExpr* c) {
|
|
int branch_v; // ignored
|
|
return IsZAM_BuiltInCond(nullptr, c, branch_v);
|
|
}
|
|
|
|
} // namespace zeek::detail
|