_Z14ImportCallbackPvPKcS1_PPcS3_Pm:
   26|     20|                     char** found_here, char **buf, size_t *buflen) {
   27|       |  // Don't load file and mark it as failure.
   28|       |  *buf = NULL;
   29|     20|  *buflen = 0;
   30|     20|  return 1;
   31|     20|}
_Z20ConvertJsonnetToJsonRKNSt3__112basic_stringIcNS_11char_traitsIcEENS_9allocatorIcEEEE:
   33|  6.45k|std::string ConvertJsonnetToJson(const std::string& jsonnet) {
   34|  6.45k|  JsonnetVm* jvm = jsonnet_make();
   35|  6.45k|  jsonnet_import_callback(jvm, ImportCallback, jvm);
   36|  6.45k|  int error = 0;
   37|  6.45k|  char* res =
   38|  6.45k|      jsonnet_evaluate_snippet(jvm, /*filename=*/"", jsonnet.c_str(), &error);
   39|       |
   40|  6.45k|  std::string json;
   41|  6.45k|  if (error == 0 && res != nullptr) {
  ------------------
  |  Branch (41:7): [True: 1.17k, False: 5.27k]
  |  Branch (41:21): [True: 1.17k, False: 0]
  ------------------
   42|  1.17k|    json = res;
   43|  1.17k|  }
   44|       |
   45|  6.45k|  if (res) {
  ------------------
  |  Branch (45:7): [True: 6.45k, False: 0]
  ------------------
   46|  6.45k|    jsonnet_realloc(jvm, res, 0);
   47|  6.45k|  }
   48|  6.45k|  jsonnet_destroy(jvm);
   49|  6.45k|  return json;
   50|  6.45k|}
LLVMFuzzerTestOneInput:
   52|  6.45k|extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
   53|  6.45k|  std::string fuzz_jsonnet(reinterpret_cast<const char*>(data), size);
   54|  6.45k|  ConvertJsonnetToJson(fuzz_jsonnet);
   55|  6.45k|  return 0;
   56|  6.45k|}

libjsonnet.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
libjsonnet.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
libjsonnet.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9AllocatorD2Ev:
 1020|  6.45k|    {
 1021|  85.3M|        for (auto x : allocated) {
  ------------------
  |  Branch (1021:21): [True: 85.3M, False: 6.45k]
  ------------------
 1022|  85.3M|            delete x;
 1023|  85.3M|        }
 1024|  6.45k|        allocated.clear();
 1025|  1.90M|        for (const auto &x : internedIdentifiers) {
  ------------------
  |  Branch (1025:28): [True: 1.90M, False: 6.45k]
  ------------------
 1026|  1.90M|            delete x.second;
 1027|  1.90M|        }
 1028|  6.45k|        internedIdentifiers.clear();
 1029|   145k|        for (const auto &x : internedBuiltins) {
  ------------------
  |  Branch (1029:28): [True: 145k, False: 6.45k]
  ------------------
 1030|   145k|            delete x.second;
 1031|   145k|        }
 1032|  6.45k|        internedBuiltins.clear();
 1033|  6.45k|    }
_ZN7jsonnet8internal10IdentifierC2ERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  113|  2.36M|    Identifier(const UString &name) : name(name) {}
_ZN7jsonnet8internal3ASTC2ERKNS0_13LocationRangeENS0_7ASTTypeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEE:
  132|  77.4M|        : location(location), type(type), openFodder(open_fodder)
  133|  77.4M|    {
  134|  77.4M|    }
_ZN7jsonnet8internal3ASTD2Ev:
  135|  85.4M|    virtual ~AST(void) {}
_ZN7jsonnet8internal8ArgParamC2ERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEEPKNS0_10IdentifierES9_:
  158|   506k|        : idFodder(id_fodder), id(id), expr(nullptr), commaFodder(comma_fodder)
  159|   506k|    {
  160|   506k|    }
_ZN7jsonnet8internal8ArgParamC2EPNS0_3ASTERKNSt3__16vectorINS0_13FodderElementENS4_9allocatorIS6_EEEE:
  163|  5.53M|        : id(nullptr), expr(expr), commaFodder(comma_fodder)
  164|  5.53M|    {
  165|  5.53M|    }
_ZN7jsonnet8internal8ArgParamC2ERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEEPKNS0_10IdentifierES9_PNS0_3ASTES9_:
  169|  11.3M|        : idFodder(id_fodder), id(id), eqFodder(eq_fodder), expr(expr), commaFodder(comma_fodder)
  170|  11.3M|    {
  171|  11.3M|    }
_ZN7jsonnet8internal17ComprehensionSpecC2ENS1_4KindERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEESA_PKNS0_10IdentifierESA_PNS0_3ASTE:
  187|   430k|        : kind(kind),
  188|   430k|          openFodder(open_fodder),
  189|   430k|          varFodder(var_fodder),
  190|   430k|          var(var),
  191|   430k|          inFodder(in_fodder),
  192|   430k|          expr(expr)
  193|   430k|    {
  194|   430k|    }
_ZN7jsonnet8internal5ApplyC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_RKNS6_INS0_8ArgParamENS8_ISF_EEEEbSC_SC_b:
  209|  8.06M|        : AST(lr, AST_APPLY, open_fodder),
  210|  8.06M|          target(target),
  211|  8.06M|          fodderL(fodder_l),
  212|  8.06M|          args(args),
  213|  8.06M|          trailingComma(trailing_comma),
  214|  8.06M|          fodderR(fodder_r),
  215|  8.06M|          tailstrictFodder(tailstrict_fodder),
  216|  8.06M|          tailstrict(tailstrict)
  217|  8.06M|    {
  218|  8.06M|    }
_ZN7jsonnet8internal10ApplyBraceC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESE_:
  226|  59.3k|        : AST(lr, AST_APPLY_BRACE, open_fodder), left(left), right(right)
  227|  59.3k|    {
  228|  59.3k|    }
_ZN7jsonnet8internal5Array7ElementC2EPNS0_3ASTERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  236|  1.92M|        Element(AST *expr, const Fodder &comma_fodder) : expr(expr), commaFodder(comma_fodder) {}
_ZN7jsonnet8internal5ArrayC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS1_7ElementENS8_ISD_EEEEbSC_:
  244|  1.17M|        : AST(lr, AST_ARRAY, open_fodder),
  245|  1.17M|          elements(elements),
  246|  1.17M|          trailingComma(trailing_comma),
  247|  1.17M|          closeFodder(close_fodder)
  248|  1.17M|    {
  249|  1.17M|    }
_ZN7jsonnet8internal18ArrayComprehensionC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_bRKNS6_INS0_17ComprehensionSpecENS8_ISF_EEEESC_:
  262|   231k|        : AST(lr, AST_ARRAY_COMPREHENSION, open_fodder),
  263|   231k|          body(body),
  264|   231k|          commaFodder(comma_fodder),
  265|   231k|          trailingComma(trailing_comma),
  266|   231k|          specs(specs),
  267|   231k|          closeFodder(close_fodder)
  268|   231k|    {
  269|       |        assert(specs.size() > 0);
  ------------------
  |  Branch (269:9): [True: 231k, False: 0]
  ------------------
  270|   231k|    }
_ZN7jsonnet8internal6AssertC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SE_SC_SE_:
  286|   260k|        : AST(lr, AST_ASSERT, open_fodder),
  287|   260k|          cond(cond),
  288|   260k|          colonFodder(colon_fodder),
  289|   260k|          message(message),
  290|   260k|          semicolonFodder(semicolon_fodder),
  291|   260k|          rest(rest)
  292|   260k|    {
  293|   260k|    }
_ZN7jsonnet8internal6BinaryC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_NS0_8BinaryOpESE_:
  367|  6.46M|        : AST(lr, AST_BINARY, open_fodder), left(left), opFodder(op_fodder), op(op), right(right)
  368|  6.46M|    {
  369|  6.46M|    }
_ZN7jsonnet8internal19BuiltinFunctionBodyC2ERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEERKNS5_6vectorIPKNS0_10IdentifierENS9_ISH_EEEE:
  382|   145k|        : AST(lr, AST_BUILTIN_FUNCTION_BODY, Fodder{}), name(name), params(params)
  383|   145k|    {
  384|   145k|    }
_ZN7jsonnet8internal15BuiltinFunctionC2ERKNS0_13LocationRangeEPKNS0_19BuiltinFunctionBodyE:
  395|   266k|        : AST(lr, AST_BUILTIN_FUNCTION, Fodder{}), body(body)
  396|   266k|    {
  397|   266k|    }
_ZN7jsonnet8internal11ConditionalC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SE_SC_SE_:
  414|  3.03M|        : AST(lr, AST_CONDITIONAL, open_fodder),
  415|  3.03M|          cond(cond),
  416|  3.03M|          thenFodder(then_fodder),
  417|  3.03M|          branchTrue(branch_true),
  418|  3.03M|          elseFodder(else_fodder),
  419|  3.03M|          branchFalse(branch_false)
  420|  3.03M|    {
  421|  3.03M|    }
_ZN7jsonnet8internal6DollarC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  426|  37.4k|    Dollar(const LocationRange &lr, const Fodder &open_fodder) : AST(lr, AST_DOLLAR, open_fodder) {}
_ZN7jsonnet8internal5ErrorC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTE:
  433|   957k|        : AST(lr, AST_ERROR, open_fodder), expr(expr)
  434|   957k|    {
  435|   957k|    }
_ZN7jsonnet8internal8FunctionC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEESC_RKNS6_INS0_8ArgParamENS8_ISD_EEEEbSC_PNS0_3ASTE:
  448|  1.64M|        : AST(lr, AST_FUNCTION, open_fodder),
  449|  1.64M|          parenLeftFodder(paren_left_fodder),
  450|  1.64M|          params(params),
  451|  1.64M|          trailingComma(trailing_comma),
  452|  1.64M|          parenRightFodder(paren_right_fodder),
  453|  1.64M|          body(body)
  454|  1.64M|    {
  455|  1.64M|    }
_ZN7jsonnet8internal6ImportC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_13LiteralStringE:
  464|    398|        : AST(lr, AST_IMPORT, open_fodder), file(file)
  465|    398|    {
  466|    398|    }
_ZN7jsonnet8internal9ImportstrC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_13LiteralStringE:
  473|    437|        : AST(lr, AST_IMPORTSTR, open_fodder), file(file)
  474|    437|    {
  475|    437|    }
_ZN7jsonnet8internal9ImportbinC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_13LiteralStringE:
  482|    274|        : AST(lr, AST_IMPORTBIN, open_fodder), file(file)
  483|    274|    {
  484|    274|    }
_ZN7jsonnet8internal5IndexC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SC_PKNS0_10IdentifierE:
  505|  4.56M|        : AST(lr, AST_INDEX, open_fodder),
  506|  4.56M|          target(target),
  507|  4.56M|          dotFodder(dot_fodder),
  508|  4.56M|          isSlice(false),
  509|  4.56M|          index(nullptr),
  510|  4.56M|          end(nullptr),
  511|  4.56M|          step(nullptr),
  512|  4.56M|          idFodder(id_fodder),
  513|  4.56M|          id(id)
  514|  4.56M|    {
  515|  4.56M|    }
_ZN7jsonnet8internal5IndexC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_bSE_SC_SE_SC_SE_SC_:
  520|  3.62M|        : AST(lr, AST_INDEX, open_fodder),
  521|  3.62M|          target(target),
  522|  3.62M|          dotFodder(dot_fodder),
  523|  3.62M|          isSlice(is_slice),
  524|  3.62M|          index(index),
  525|  3.62M|          endColonFodder(end_colon_fodder),
  526|  3.62M|          end(end),
  527|  3.62M|          stepColonFodder(step_colon_fodder),
  528|  3.62M|          step(step),
  529|  3.62M|          idFodder(id_fodder),
  530|  3.62M|          id(nullptr)
  531|  3.62M|    {
  532|  3.62M|    }
_ZN7jsonnet8internal5Local4BindC2ERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEEPKNS0_10IdentifierESA_PNS0_3ASTEbSA_RKNS4_INS0_8ArgParamENS6_ISG_EEEEbSA_SA_:
  551|  2.60M|            : varFodder(var_fodder),
  552|  2.60M|              var(var),
  553|  2.60M|              opFodder(op_fodder),
  554|  2.60M|              body(body),
  555|  2.60M|              functionSugar(function_sugar),
  556|  2.60M|              parenLeftFodder(paren_left_fodder),
  557|  2.60M|              params(params),
  558|  2.60M|              trailingComma(trailing_comma),
  559|  2.60M|              parenRightFodder(paren_right_fodder),
  560|  2.60M|              closeFodder(close_fodder)
  561|  2.60M|        {
  562|  2.60M|        }
_ZN7jsonnet8internal5LocalC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS1_4BindENS8_ISD_EEEEPNS0_3ASTE:
  568|  2.91M|        : AST(lr, AST_LOCAL, open_fodder), binds(binds), body(body)
  569|  2.91M|    {
  570|  2.91M|    }
_ZN7jsonnet8internal14LiteralBooleanC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEb:
  577|   568k|        : AST(lr, AST_LITERAL_BOOLEAN, open_fodder), value(value)
  578|   568k|    {
  579|   568k|    }
_ZN7jsonnet8internal11LiteralNullC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  585|   330k|        : AST(lr, AST_LITERAL_NULL, open_fodder)
  586|   330k|    {
  587|   330k|    }
_ZN7jsonnet8internal13LiteralNumberC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS5_12basic_stringIcNS5_11char_traitsIcEENS8_IcEEEE:
  595|  3.84M|        : AST(lr, AST_LITERAL_NUMBER, open_fodder),
  596|  3.84M|          value(strtod(str.c_str(), nullptr)),
  597|  3.84M|          originalString(str)
  598|  3.84M|    {
  599|  3.84M|    }
_ZN7jsonnet8internal13LiteralStringC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS5_12basic_stringIDiNS5_11char_traitsIDiEENS8_IDiEEEENS1_9TokenKindERKNSD_IcNSE_IcEENS8_IcEEEESP_:
  612|  12.1M|        : AST(lr, AST_LITERAL_STRING, open_fodder),
  613|  12.1M|          value(value),
  614|  12.1M|          tokenKind(token_kind),
  615|  12.1M|          blockIndent(block_indent),
  616|  12.1M|          blockTermIndent(block_term_indent)
  617|  12.1M|    {
  618|  12.1M|    }
_ZN7jsonnet8internal11ObjectFieldC2ENS1_4KindERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEESA_SA_SA_NS1_4HideEbbPNS0_3ASTEPKNS0_10IdentifierERKNS0_13LocationRangeERKNS4_INS0_8ArgParamENS6_ISK_EEEEbSA_SD_SD_SA_:
  682|  1.62M|        : kind(kind),
  683|  1.62M|          fodder1(fodder1),
  684|  1.62M|          fodder2(fodder2),
  685|  1.62M|          fodderL(fodder_l),
  686|  1.62M|          fodderR(fodder_r),
  687|  1.62M|          hide(hide),
  688|  1.62M|          superSugar(super_sugar),
  689|  1.62M|          methodSugar(method_sugar),
  690|  1.62M|          expr1(expr1),
  691|  1.62M|          id(id),
  692|  1.62M|          idLocation(id_lr),
  693|  1.62M|          params(params),
  694|  1.62M|          trailingComma(trailing_comma),
  695|  1.62M|          opFodder(op_fodder),
  696|  1.62M|          expr2(expr2),
  697|  1.62M|          expr3(expr3),
  698|  1.62M|          commaFodder(comma_fodder)
  699|  1.62M|    {
  700|       |        // Enforce what is written in comments above.
  701|  1.62M|        assert(kind != ASSERT || (hide == VISIBLE && !superSugar && !methodSugar));
  ------------------
  |  Branch (701:9): [True: 29.3k, False: 0]
  |  Branch (701:9): [True: 29.3k, False: 0]
  |  Branch (701:9): [True: 29.3k, False: 0]
  |  Branch (701:9): [True: 1.59M, False: 29.3k]
  |  Branch (701:9): [True: 1.62M, False: 0]
  ------------------
  702|  1.62M|        assert(kind != LOCAL || (hide == VISIBLE && !superSugar));
  ------------------
  |  Branch (702:9): [True: 66.3k, False: 0]
  |  Branch (702:9): [True: 66.3k, False: 0]
  |  Branch (702:9): [True: 1.55M, False: 66.3k]
  |  Branch (702:9): [True: 1.62M, False: 0]
  ------------------
  703|  1.62M|        assert(kind != FIELD_ID || (id != nullptr && expr1 == nullptr));
  ------------------
  |  Branch (703:9): [True: 1.43M, False: 0]
  |  Branch (703:9): [True: 1.43M, False: 0]
  |  Branch (703:9): [True: 185k, False: 1.43M]
  |  Branch (703:9): [True: 1.62M, False: 0]
  ------------------
  704|  1.62M|        assert(kind == FIELD_ID || kind == LOCAL || id == nullptr);
  ------------------
  |  Branch (704:9): [True: 1.43M, False: 185k]
  |  Branch (704:9): [True: 66.3k, False: 118k]
  |  Branch (704:9): [True: 118k, False: 0]
  |  Branch (704:9): [True: 1.62M, False: 0]
  ------------------
  705|  1.62M|        assert(methodSugar || (params.size() == 0 && !trailingComma));
  ------------------
  |  Branch (705:9): [True: 826k, False: 0]
  |  Branch (705:9): [True: 826k, False: 0]
  |  Branch (705:9): [True: 797k, False: 826k]
  |  Branch (705:9): [True: 1.62M, False: 0]
  ------------------
  706|  1.62M|        assert(kind == ASSERT || expr3 == nullptr);
  ------------------
  |  Branch (706:9): [True: 29.3k, False: 1.59M]
  |  Branch (706:9): [True: 1.59M, False: 0]
  |  Branch (706:9): [True: 1.62M, False: 0]
  ------------------
  707|  1.62M|    }
_ZN7jsonnet8internal11ObjectField5LocalERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEES9_S9_S9_bPKNS0_10IdentifierERKNS3_INS0_8ArgParamENS5_ISD_EEEEbS9_PNS0_3ASTES9_:
  713|  44.6k|    {
  714|  44.6k|        return ObjectField(LOCAL,
  715|  44.6k|                           fodder1,
  716|  44.6k|                           fodder2,
  717|  44.6k|                           fodder_l,
  718|  44.6k|                           fodder_r,
  719|  44.6k|                           VISIBLE,
  720|  44.6k|                           false,
  721|  44.6k|                           method_sugar,
  722|  44.6k|                           nullptr,
  723|  44.6k|                           id,
  724|  44.6k|                           LocationRange(),
  725|  44.6k|                           params,
  726|  44.6k|                           trailing_comma,
  727|  44.6k|                           op_fodder,
  728|  44.6k|                           body,
  729|  44.6k|                           nullptr,
  730|  44.6k|                           comma_fodder);
  731|  44.6k|    }
_ZN7jsonnet8internal11ObjectField5LocalERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEES9_PKNS0_10IdentifierES9_PNS0_3ASTES9_:
  734|  21.6k|    {
  735|  21.6k|        return ObjectField(LOCAL,
  736|  21.6k|                           fodder1,
  737|  21.6k|                           fodder2,
  738|  21.6k|                           Fodder{},
  739|  21.6k|                           Fodder{},
  740|  21.6k|                           VISIBLE,
  741|  21.6k|                           false,
  742|  21.6k|                           false,
  743|  21.6k|                           nullptr,
  744|  21.6k|                           id,
  745|  21.6k|                           LocationRange(),
  746|  21.6k|                           ArgParams{},
  747|  21.6k|                           false,
  748|  21.6k|                           op_fodder,
  749|  21.6k|                           body,
  750|  21.6k|                           nullptr,
  751|  21.6k|                           comma_fodder);
  752|  21.6k|    }
_ZN7jsonnet8internal11ObjectField6AssertERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEEPNS0_3ASTES9_SB_S9_:
  779|  29.3k|    {
  780|  29.3k|        return ObjectField(ASSERT,
  781|  29.3k|                           fodder1,
  782|  29.3k|                           Fodder{},
  783|  29.3k|                           Fodder{},
  784|  29.3k|                           Fodder{},
  785|  29.3k|                           VISIBLE,
  786|  29.3k|                           false,
  787|  29.3k|                           false,
  788|  29.3k|                           nullptr,
  789|  29.3k|                           nullptr,
  790|  29.3k|                           LocationRange(),
  791|  29.3k|                           ArgParams{},
  792|  29.3k|                           false,
  793|  29.3k|                           op_fodder,
  794|  29.3k|                           body,
  795|  29.3k|                           msg,
  796|  29.3k|                           comma_fodder);
  797|  29.3k|    }
_ZN7jsonnet8internal6ObjectC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS0_11ObjectFieldENS8_ISD_EEEEbSC_:
  812|   346k|        : AST(lr, AST_OBJECT, open_fodder),
  813|   346k|          fields(fields),
  814|   346k|          trailingComma(trailing_comma),
  815|   346k|          closeFodder(close_fodder)
  816|   346k|    {
  817|   346k|        assert(fields.size() > 0 || !trailing_comma);
  ------------------
  |  Branch (817:9): [True: 307k, False: 39.0k]
  |  Branch (817:9): [True: 39.0k, False: 0]
  |  Branch (817:9): [True: 346k, False: 0]
  ------------------
  818|   346k|        if (fields.size() > 0)
  ------------------
  |  Branch (818:13): [True: 307k, False: 39.0k]
  ------------------
  819|       |            assert(trailing_comma || fields[fields.size() - 1].commaFodder.size() == 0);
  ------------------
  |  Branch (819:13): [True: 89.0k, False: 218k]
  |  Branch (819:13): [True: 218k, False: 0]
  |  Branch (819:13): [True: 307k, False: 0]
  ------------------
  820|   346k|    }
_ZN7jsonnet8internal15DesugaredObject5FieldC2ENS0_11ObjectField4HideEPNS0_3ASTES6_:
  833|  1.71M|            : hide(hide), name(name), body(body)
  834|  1.71M|        {
  835|  1.71M|        }
_ZN7jsonnet8internal15DesugaredObjectC2ERKNS0_13LocationRangeERKNSt3__14listIPNS0_3ASTENS5_9allocatorIS8_EEEERKNS5_6vectorINS1_5FieldENS9_ISF_EEEE:
  841|   329k|        : AST(lr, AST_DESUGARED_OBJECT, Fodder{}), asserts(asserts), fields(fields)
  842|   329k|    {
  843|   329k|    }
_ZN7jsonnet8internal19ObjectComprehensionC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS0_11ObjectFieldENS8_ISD_EEEEbRKNS6_INS0_17ComprehensionSpecENS8_ISI_EEEESC_:
  856|  27.3k|        : AST(lr, AST_OBJECT_COMPREHENSION, open_fodder),
  857|  27.3k|          fields(fields),
  858|  27.3k|          trailingComma(trailing_comma),
  859|  27.3k|          specs(specs),
  860|  27.3k|          closeFodder(close_fodder)
  861|  27.3k|    {
  862|  27.3k|    }
_ZN7jsonnet8internal25ObjectComprehensionSimpleC2ERKNS0_13LocationRangeEPNS0_3ASTES6_PKNS0_10IdentifierES6_:
  873|  27.3k|        : AST(lr, AST_OBJECT_COMPREHENSION_SIMPLE, Fodder{}),
  874|  27.3k|          field(field),
  875|  27.3k|          value(value),
  876|  27.3k|          id(id),
  877|  27.3k|          array(array)
  878|  27.3k|    {
  879|  27.3k|    }
_ZN7jsonnet8internal6ParensC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_:
  888|   317k|        : AST(lr, AST_PARENS, open_fodder), expr(expr), closeFodder(close_fodder)
  889|   317k|    {
  890|   317k|    }
_ZN7jsonnet8internal4SelfC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  895|  66.6k|    Self(const LocationRange &lr, const Fodder &open_fodder) : AST(lr, AST_SELF, open_fodder) {}
_ZN7jsonnet8internal10SuperIndexC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEESC_PNS0_3ASTESC_PKNS0_10IdentifierE:
  910|  12.1k|        : AST(lr, AST_SUPER_INDEX, open_fodder),
  911|  12.1k|          dotFodder(dot_fodder),
  912|  12.1k|          index(index),
  913|  12.1k|          idFodder(id_fodder),
  914|  12.1k|          id(id)
  915|  12.1k|    {
  916|  12.1k|    }
_ZN7jsonnet8internal7InSuperC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SC_:
  927|  11.0k|        : AST(lr, AST_IN_SUPER, open_fodder),
  928|  11.0k|          element(element),
  929|  11.0k|          inFodder(in_fodder),
  930|  11.0k|          superFodder(super_fodder)
  931|  11.0k|    {
  932|  11.0k|    }
_ZN7jsonnet8internal5UnaryC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS0_7UnaryOpEPNS0_3ASTE:
  956|   530k|        : AST(lr, AST_UNARY, open_fodder), op(op), expr(expr)
  957|   530k|    {
  958|   530k|    }
_ZN7jsonnet8internal3VarC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPKNS0_10IdentifierE:
  965|  25.3M|        : AST(lr, AST_VAR, open_fodder), id(id)
  966|  25.3M|    {
  967|  25.3M|    }
_ZN7jsonnet8internal9Allocator14makeIdentifierERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  998|  48.9M|    {
  999|  48.9M|        auto it = internedIdentifiers.find(name);
 1000|  48.9M|        if (it != internedIdentifiers.end()) {
  ------------------
  |  Branch (1000:13): [True: 47.0M, False: 1.90M]
  ------------------
 1001|  47.0M|            return it->second;
 1002|  47.0M|        }
 1003|  1.90M|        auto r = new Identifier(name);
 1004|  1.90M|        internedIdentifiers[name] = r;
 1005|  1.90M|        return r;
 1006|  48.9M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator4makeINS0_6AssertEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SF_SG_SF_EEEPT_DpOT0_:
  980|   260k|    {
  981|   260k|        auto r = new T(std::forward<Args>(args)...);
  982|   260k|        allocated.push_back(r);
  983|   260k|        return r;
  984|   260k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ErrorEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   416k|    {
  981|   416k|        auto r = new T(std::forward<Args>(args)...);
  982|   416k|        allocated.push_back(r);
  983|   416k|        return r;
  984|   416k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SF_SG_SF_EEEPT_DpOT0_:
  980|  2.04M|    {
  981|  2.04M|        auto r = new T(std::forward<Args>(args)...);
  982|  2.04M|        allocated.push_back(r);
  983|  2.04M|        return r;
  984|  2.04M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SF_SA_DnEEEPT_DpOT0_:
  980|  25.5k|    {
  981|  25.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  25.5k|        allocated.push_back(r);
  983|  25.5k|        return r;
  984|  25.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERSA_RNS6_INS0_8ArgParamENS8_ISE_EEEERbSD_RPNS0_3ASTEEEEPT_DpOT0_:
  980|   127k|    {
  981|   127k|        auto r = new T(std::forward<Args>(args)...);
  982|   127k|        allocated.push_back(r);
  983|   127k|        return r;
  984|   127k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6ImportEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_13LiteralStringEEEEPT_DpOT0_:
  980|    398|    {
  981|    398|        auto r = new T(std::forward<Args>(args)...);
  982|    398|        allocated.push_back(r);
  983|    398|        return r;
  984|    398|    }
_ZN7jsonnet8internal9Allocator4makeINS0_9ImportstrEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_13LiteralStringEEEEPT_DpOT0_:
  980|    437|    {
  981|    437|        auto r = new T(std::forward<Args>(args)...);
  982|    437|        allocated.push_back(r);
  983|    437|        return r;
  984|    437|    }
_ZN7jsonnet8internal9Allocator4makeINS0_9ImportbinEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_13LiteralStringEEEEPT_DpOT0_:
  980|    274|    {
  981|    274|        auto r = new T(std::forward<Args>(args)...);
  982|    274|        allocated.push_back(r);
  983|    274|        return r;
  984|    274|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS3_4BindENS8_ISD_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|  1.53M|    {
  981|  1.53M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.53M|        allocated.push_back(r);
  983|  1.53M|        return r;
  984|  1.53M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5UnaryEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS0_7UnaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|   372k|    {
  981|   372k|        auto r = new T(std::forward<Args>(args)...);
  982|   372k|        allocated.push_back(r);
  983|   372k|        return r;
  984|   372k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6ObjectEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS0_11ObjectFieldENS8_ISD_EEEERbRSA_EEEPT_DpOT0_:
  980|   346k|    {
  981|   346k|        auto r = new T(std::forward<Args>(args)...);
  982|   346k|        allocated.push_back(r);
  983|   346k|        return r;
  984|   346k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_19ObjectComprehensionEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS0_11ObjectFieldENS8_ISD_EEEERbRNS6_INS0_17ComprehensionSpecENS8_ISI_EEEERSA_EEEPT_DpOT0_:
  980|  27.3k|    {
  981|  27.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.3k|        allocated.push_back(r);
  983|  27.3k|        return r;
  984|  27.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS6_12basic_stringIDiNS6_11char_traitsIDiEENS9_IDiEEEENS3_9TokenKindERA1_KcSL_EEEPT_DpOT0_:
  980|  62.3k|    {
  981|  62.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  62.3k|        allocated.push_back(r);
  983|  62.3k|        return r;
  984|  62.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS6_12basic_stringIDiNS6_11char_traitsIDiEENS9_IDiEEEENS3_9TokenKindERNSD_IcNSE_IcEENS9_IcEEEESM_EEEPT_DpOT0_:
  980|    626|    {
  981|    626|        auto r = new T(std::forward<Args>(args)...);
  982|    626|        allocated.push_back(r);
  983|    626|        return r;
  984|    626|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS6_INS3_7ElementENS8_ISC_EEEEbSB_EEEPT_DpOT0_:
  980|   150k|    {
  981|   150k|        auto r = new T(std::forward<Args>(args)...);
  982|   150k|        allocated.push_back(r);
  983|   150k|        return r;
  984|   150k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_18ArrayComprehensionEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTESB_RbRNS6_INS0_17ComprehensionSpecENS8_ISG_EEEESB_EEEPT_DpOT0_:
  980|   203k|    {
  981|   203k|        auto r = new T(std::forward<Args>(args)...);
  982|   203k|        allocated.push_back(r);
  983|   203k|        return r;
  984|   203k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS3_7ElementENS8_ISC_EEEERbSB_EEEPT_DpOT0_:
  980|   538k|    {
  981|   538k|        auto r = new T(std::forward<Args>(args)...);
  982|   538k|        allocated.push_back(r);
  983|   538k|        return r;
  984|   538k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6ParensEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTESB_EEEPT_DpOT0_:
  980|   317k|    {
  981|   317k|        auto r = new T(std::forward<Args>(args)...);
  982|   317k|        allocated.push_back(r);
  983|   317k|        return r;
  984|   317k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralNumberEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS5_12basic_stringIcNS5_11char_traitsIcEENS8_IcEEEEEEEPT_DpOT0_:
  980|  3.32M|    {
  981|  3.32M|        auto r = new T(std::forward<Args>(args)...);
  982|  3.32M|        allocated.push_back(r);
  983|  3.32M|        return r;
  984|  3.32M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS5_12basic_stringIDiNS5_11char_traitsIDiEENS8_IDiEEEENS3_9TokenKindERA1_KcSK_EEEPT_DpOT0_:
  980|  3.03M|    {
  981|  3.03M|        auto r = new T(std::forward<Args>(args)...);
  982|  3.03M|        allocated.push_back(r);
  983|  3.03M|        return r;
  984|  3.03M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS5_12basic_stringIDiNS5_11char_traitsIDiEENS8_IDiEEEENS3_9TokenKindERNSC_IcNSD_IcEENS8_IcEEEESL_EEEPT_DpOT0_:
  980|  2.61k|    {
  981|  2.61k|        auto r = new T(std::forward<Args>(args)...);
  982|  2.61k|        allocated.push_back(r);
  983|  2.61k|        return r;
  984|  2.61k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_14LiteralBooleanEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEbEEEPT_DpOT0_:
  980|   540k|    {
  981|   540k|        auto r = new T(std::forward<Args>(args)...);
  982|   540k|        allocated.push_back(r);
  983|   540k|        return r;
  984|   540k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11LiteralNullEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEEEEPT_DpOT0_:
  980|   114k|    {
  981|   114k|        auto r = new T(std::forward<Args>(args)...);
  982|   114k|        allocated.push_back(r);
  983|   114k|        return r;
  984|   114k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6DollarEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEEEEPT_DpOT0_:
  980|  37.4k|    {
  981|  37.4k|        auto r = new T(std::forward<Args>(args)...);
  982|  37.4k|        allocated.push_back(r);
  983|  37.4k|        return r;
  984|  37.4k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  19.8M|    {
  981|  19.8M|        auto r = new T(std::forward<Args>(args)...);
  982|  19.8M|        allocated.push_back(r);
  983|  19.8M|        return r;
  984|  19.8M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_4SelfEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEEEEPT_DpOT0_:
  980|  44.5k|    {
  981|  44.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  44.5k|        allocated.push_back(r);
  983|  44.5k|        return r;
  984|  44.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_10SuperIndexEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEESB_RPNS0_3ASTESB_RPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  1.63k|    {
  981|  1.63k|        auto r = new T(std::forward<Args>(args)...);
  982|  1.63k|        allocated.push_back(r);
  983|  1.63k|        return r;
  984|  1.63k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_RbSF_SG_SF_SG_SF_SG_EEEPT_DpOT0_:
  980|  1.07M|    {
  981|  1.07M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.07M|        allocated.push_back(r);
  983|  1.07M|        return r;
  984|  1.07M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SG_RPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  4.56M|    {
  981|  4.56M|        auto r = new T(std::forward<Args>(args)...);
  982|  4.56M|        allocated.push_back(r);
  983|  4.56M|        return r;
  984|  4.56M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_RNS6_INS0_8ArgParamENS8_ISH_EEEERbSG_SG_SL_EEEPT_DpOT0_:
  980|  5.15M|    {
  981|  5.15M|        auto r = new T(std::forward<Args>(args)...);
  982|  5.15M|        allocated.push_back(r);
  983|  5.15M|        return r;
  984|  5.15M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_10ApplyBraceEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTESF_EEEPT_DpOT0_:
  980|  59.3k|    {
  981|  59.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  59.3k|        allocated.push_back(r);
  983|  59.3k|        return r;
  984|  59.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_7InSuperEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SG_EEEPT_DpOT0_:
  980|    480|    {
  981|    480|        auto r = new T(std::forward<Args>(args)...);
  982|    480|        allocated.push_back(r);
  983|    480|        return r;
  984|    480|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_RNS0_8BinaryOpESF_EEEPT_DpOT0_:
  980|  5.51M|    {
  981|  5.51M|        auto r = new T(std::forward<Args>(args)...);
  982|  5.51M|        allocated.push_back(r);
  983|  5.51M|        return r;
  984|  5.51M|    }
pass.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
pass.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
pass.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator5cloneINS0_5ApplyEEEPT_S5_:
  988|   241k|    {
  989|   241k|        auto r = new T(*ast);
  990|   241k|        allocated.push_back(r);
  991|   241k|        return r;
  992|   241k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5ArrayEEEPT_S5_:
  988|   159k|    {
  989|   159k|        auto r = new T(*ast);
  990|   159k|        allocated.push_back(r);
  991|   159k|        return r;
  992|   159k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_6BinaryEEEPT_S5_:
  988|   858k|    {
  989|   858k|        auto r = new T(*ast);
  990|   858k|        allocated.push_back(r);
  991|   858k|        return r;
  992|   858k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_11ConditionalEEEPT_S5_:
  988|   492k|    {
  989|   492k|        auto r = new T(*ast);
  990|   492k|        allocated.push_back(r);
  991|   492k|        return r;
  992|   492k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_15DesugaredObjectEEEPT_S5_:
  988|   701k|    {
  989|   701k|        auto r = new T(*ast);
  990|   701k|        allocated.push_back(r);
  991|   701k|        return r;
  992|   701k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5ErrorEEEPT_S5_:
  988|  56.1k|    {
  989|  56.1k|        auto r = new T(*ast);
  990|  56.1k|        allocated.push_back(r);
  991|  56.1k|        return r;
  992|  56.1k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_8FunctionEEEPT_S5_:
  988|  41.5k|    {
  989|  41.5k|        auto r = new T(*ast);
  990|  41.5k|        allocated.push_back(r);
  991|  41.5k|        return r;
  992|  41.5k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_6ImportEEEPT_S5_:
  988|    457|    {
  989|    457|        auto r = new T(*ast);
  990|    457|        allocated.push_back(r);
  991|    457|        return r;
  992|    457|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_9ImportstrEEEPT_S5_:
  988|    337|    {
  989|    337|        auto r = new T(*ast);
  990|    337|        allocated.push_back(r);
  991|    337|        return r;
  992|    337|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_9ImportbinEEEPT_S5_:
  988|    268|    {
  989|    268|        auto r = new T(*ast);
  990|    268|        allocated.push_back(r);
  991|    268|        return r;
  992|    268|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5IndexEEEPT_S5_:
  988|   261k|    {
  989|   261k|        auto r = new T(*ast);
  990|   261k|        allocated.push_back(r);
  991|   261k|        return r;
  992|   261k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_7InSuperEEEPT_S5_:
  988|   293k|    {
  989|   293k|        auto r = new T(*ast);
  990|   293k|        allocated.push_back(r);
  991|   293k|        return r;
  992|   293k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_14LiteralBooleanEEEPT_S5_:
  988|  19.8k|    {
  989|  19.8k|        auto r = new T(*ast);
  990|  19.8k|        allocated.push_back(r);
  991|  19.8k|        return r;
  992|  19.8k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_11LiteralNullEEEPT_S5_:
  988|  14.5k|    {
  989|  14.5k|        auto r = new T(*ast);
  990|  14.5k|        allocated.push_back(r);
  991|  14.5k|        return r;
  992|  14.5k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_13LiteralNumberEEEPT_S5_:
  988|   386k|    {
  989|   386k|        auto r = new T(*ast);
  990|   386k|        allocated.push_back(r);
  991|   386k|        return r;
  992|   386k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_13LiteralStringEEEPT_S5_:
  988|  1.37M|    {
  989|  1.37M|        auto r = new T(*ast);
  990|  1.37M|        allocated.push_back(r);
  991|  1.37M|        return r;
  992|  1.37M|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5LocalEEEPT_S5_:
  988|   166k|    {
  989|   166k|        auto r = new T(*ast);
  990|   166k|        allocated.push_back(r);
  991|   166k|        return r;
  992|   166k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_25ObjectComprehensionSimpleEEEPT_S5_:
  988|  29.9k|    {
  989|  29.9k|        auto r = new T(*ast);
  990|  29.9k|        allocated.push_back(r);
  991|  29.9k|        return r;
  992|  29.9k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_4SelfEEEPT_S5_:
  988|  8.27k|    {
  989|  8.27k|        auto r = new T(*ast);
  990|  8.27k|        allocated.push_back(r);
  991|  8.27k|        return r;
  992|  8.27k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_10SuperIndexEEEPT_S5_:
  988|   310k|    {
  989|   310k|        auto r = new T(*ast);
  990|   310k|        allocated.push_back(r);
  991|   310k|        return r;
  992|   310k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5UnaryEEEPT_S5_:
  988|  1.03M|    {
  989|  1.03M|        auto r = new T(*ast);
  990|  1.03M|        allocated.push_back(r);
  991|  1.03M|        return r;
  992|  1.03M|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_3VarEEEPT_S5_:
  988|  1.58M|    {
  989|  1.58M|        auto r = new T(*ast);
  990|  1.58M|        allocated.push_back(r);
  991|  1.58M|        return r;
  992|  1.58M|    }
static_analysis.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
static_analysis.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
static_analysis.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJNS0_13LocationRangeENSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  2.88k|    {
  981|  2.88k|        auto r = new T(std::forward<Args>(args)...);
  982|  2.88k|        allocated.push_back(r);
  983|  2.88k|        return r;
  984|  2.88k|    }
vm.cpp:_ZN7jsonnet8internalL10bop_stringENS0_8BinaryOpE:
  325|    214|{
  326|    214|    switch (bop) {
  327|      8|        case BOP_MULT: return "*";
  ------------------
  |  Branch (327:9): [True: 8, False: 206]
  ------------------
  328|      7|        case BOP_DIV: return "/";
  ------------------
  |  Branch (328:9): [True: 7, False: 207]
  ------------------
  329|      0|        case BOP_PERCENT: return "%";
  ------------------
  |  Branch (329:9): [True: 0, False: 214]
  ------------------
  330|       |
  331|     98|        case BOP_PLUS: return "+";
  ------------------
  |  Branch (331:9): [True: 98, False: 116]
  ------------------
  332|     19|        case BOP_MINUS: return "-";
  ------------------
  |  Branch (332:9): [True: 19, False: 195]
  ------------------
  333|       |
  334|      2|        case BOP_SHIFT_L: return "<<";
  ------------------
  |  Branch (334:9): [True: 2, False: 212]
  ------------------
  335|      2|        case BOP_SHIFT_R: return ">>";
  ------------------
  |  Branch (335:9): [True: 2, False: 212]
  ------------------
  336|       |
  337|     18|        case BOP_GREATER: return ">";
  ------------------
  |  Branch (337:9): [True: 18, False: 196]
  ------------------
  338|      7|        case BOP_GREATER_EQ: return ">=";
  ------------------
  |  Branch (338:9): [True: 7, False: 207]
  ------------------
  339|     19|        case BOP_LESS: return "<";
  ------------------
  |  Branch (339:9): [True: 19, False: 195]
  ------------------
  340|      6|        case BOP_LESS_EQ: return "<=";
  ------------------
  |  Branch (340:9): [True: 6, False: 208]
  ------------------
  341|      0|        case BOP_IN: return "in";
  ------------------
  |  Branch (341:9): [True: 0, False: 214]
  ------------------
  342|       |
  343|      0|        case BOP_MANIFEST_EQUAL: return "==";
  ------------------
  |  Branch (343:9): [True: 0, False: 214]
  ------------------
  344|      0|        case BOP_MANIFEST_UNEQUAL: return "!=";
  ------------------
  |  Branch (344:9): [True: 0, False: 214]
  ------------------
  345|       |
  346|      9|        case BOP_BITWISE_AND: return "&";
  ------------------
  |  Branch (346:9): [True: 9, False: 205]
  ------------------
  347|      4|        case BOP_BITWISE_XOR: return "^";
  ------------------
  |  Branch (347:9): [True: 4, False: 210]
  ------------------
  348|      9|        case BOP_BITWISE_OR: return "|";
  ------------------
  |  Branch (348:9): [True: 9, False: 205]
  ------------------
  349|       |
  350|      1|        case BOP_AND: return "&&";
  ------------------
  |  Branch (350:9): [True: 1, False: 213]
  ------------------
  351|      5|        case BOP_OR: return "||";
  ------------------
  |  Branch (351:9): [True: 5, False: 209]
  ------------------
  352|       |
  353|      0|        default:
  ------------------
  |  Branch (353:9): [True: 0, False: 214]
  ------------------
  354|      0|            std::cerr << "INTERNAL ERROR: Unrecognised binary operator: " << bop << std::endl;
  355|      0|            std::abort();
  356|    214|    }
  357|    214|}
vm.cpp:_ZN7jsonnet8internalL10uop_stringENS0_7UnaryOpE:
  938|     76|{
  939|     76|    switch (uop) {
  940|     44|        case UOP_PLUS: return "+";
  ------------------
  |  Branch (940:9): [True: 44, False: 32]
  ------------------
  941|      7|        case UOP_MINUS: return "-";
  ------------------
  |  Branch (941:9): [True: 7, False: 69]
  ------------------
  942|      4|        case UOP_BITWISE_NOT: return "~";
  ------------------
  |  Branch (942:9): [True: 4, False: 72]
  ------------------
  943|     21|        case UOP_NOT: return "!";
  ------------------
  |  Branch (943:9): [True: 21, False: 55]
  ------------------
  944|       |
  945|      0|        default:
  ------------------
  |  Branch (945:9): [True: 0, False: 76]
  ------------------
  946|      0|            std::cerr << "INTERNAL ERROR: Unrecognised unary operator: " << uop << std::endl;
  947|      0|            std::abort();
  948|     76|    }
  949|     76|}
desugarer.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
desugarer.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
desugarer.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_NS0_8BinaryOpESF_EEEPT_DpOT0_:
  980|  51.1k|    {
  981|  51.1k|        auto r = new T(std::forward<Args>(args)...);
  982|  51.1k|        allocated.push_back(r);
  983|  51.1k|        return r;
  984|  51.1k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERA4_KcEEEPT_DpOT0_:
  980|   488k|    {
  981|   488k|        auto r = new T(std::forward<Args>(args)...);
  982|   488k|        allocated.push_back(r);
  983|   488k|        return r;
  984|   488k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  980|   230k|    {
  981|   230k|        auto r = new T(std::forward<Args>(args)...);
  982|   230k|        allocated.push_back(r);
  983|   230k|        return r;
  984|   230k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|   409k|    {
  981|   409k|        auto r = new T(std::forward<Args>(args)...);
  982|   409k|        allocated.push_back(r);
  983|   409k|        return r;
  984|   409k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ArrayEEEEPT_DpOT0_:
  980|   230k|    {
  981|   230k|        auto r = new T(std::forward<Args>(args)...);
  982|   230k|        allocated.push_back(r);
  983|   230k|        return r;
  984|   230k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  980|   230k|    {
  981|   230k|        auto r = new T(std::forward<Args>(args)...);
  982|   230k|        allocated.push_back(r);
  983|   230k|        return r;
  984|   230k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  980|   431k|    {
  981|   431k|        auto r = new T(std::forward<Args>(args)...);
  982|   431k|        allocated.push_back(r);
  983|   431k|        return r;
  984|   431k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERKPNS0_3ASTESD_RSF_SD_SI_EEEPT_DpOT0_:
  980|   156k|    {
  981|   156k|        auto r = new T(std::forward<Args>(args)...);
  982|   156k|        allocated.push_back(r);
  983|   156k|        return r;
  984|   156k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  980|   253k|    {
  981|   253k|        auto r = new T(std::forward<Args>(args)...);
  982|   253k|        allocated.push_back(r);
  983|   253k|        return r;
  984|   253k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_NS7_INS0_8ArgParamENS9_ISE_EEEEbSD_PNS0_11ConditionalEEEEPT_DpOT0_:
  980|   253k|    {
  981|   253k|        auto r = new T(std::forward<Args>(args)...);
  982|   253k|        allocated.push_back(r);
  983|   253k|        return r;
  984|   253k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_6BinaryESD_RPNS0_3ASTESD_PNS0_5LocalEEEEPT_DpOT0_:
  980|   253k|    {
  981|   253k|        auto r = new T(std::forward<Args>(args)...);
  982|   253k|        allocated.push_back(r);
  983|   253k|        return r;
  984|   253k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ApplyEEEEPT_DpOT0_:
  980|   253k|    {
  981|   253k|        auto r = new T(std::forward<Args>(args)...);
  982|   253k|        allocated.push_back(r);
  983|   253k|        return r;
  984|   253k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   260k|    {
  981|   260k|        auto r = new T(std::forward<Args>(args)...);
  982|   260k|        allocated.push_back(r);
  983|   260k|        return r;
  984|   260k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bSG_SE_DnSE_DnSE_EEEPT_DpOT0_:
  980|   253k|    {
  981|   253k|        auto r = new T(std::forward<Args>(args)...);
  982|   253k|        allocated.push_back(r);
  983|   253k|        return r;
  984|   253k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ApplyESD_SF_SD_PNS0_5ErrorEEEEPT_DpOT0_:
  980|   253k|    {
  981|   253k|        auto r = new T(std::forward<Args>(args)...);
  982|   253k|        allocated.push_back(r);
  983|   253k|        return r;
  984|   253k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS8_INS3_7ElementENSA_ISF_EEEEbSE_EEEPT_DpOT0_:
  980|   230k|    {
  981|   230k|        auto r = new T(std::forward<Args>(args)...);
  982|   230k|        allocated.push_back(r);
  983|   230k|        return r;
  984|   230k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERKNS7_12basic_stringIDiNS7_11char_traitsIDiEENSA_IDiEEEENS3_9TokenKindERA1_KcSP_EEEPT_DpOT0_:
  980|  9.08M|    {
  981|  9.08M|        auto r = new T(std::forward<Args>(args)...);
  982|  9.08M|        allocated.push_back(r);
  983|  9.08M|        return r;
  984|  9.08M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ErrorEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   541k|    {
  981|   541k|        auto r = new T(std::forward<Args>(args)...);
  982|   541k|        allocated.push_back(r);
  983|   541k|        return r;
  984|   541k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_SF_SH_SF_EEEPT_DpOT0_:
  980|   260k|    {
  981|   260k|        auto r = new T(std::forward<Args>(args)...);
  982|   260k|        allocated.push_back(r);
  983|   260k|        return r;
  984|   260k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralStringESE_DnSE_DnSE_EEEPT_DpOT0_:
  980|  2.24M|    {
  981|  2.24M|        auto r = new T(std::forward<Args>(args)...);
  982|  2.24M|        allocated.push_back(r);
  983|  2.24M|        return r;
  984|  2.24M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_RNS7_INS0_8ArgParamENS9_ISI_EEEEbSH_SH_bEEEPT_DpOT0_:
  980|   193k|    {
  981|   193k|        auto r = new T(std::forward<Args>(args)...);
  982|   193k|        allocated.push_back(r);
  983|   193k|        return r;
  984|   193k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5IndexESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  980|  1.42M|    {
  981|  1.42M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.42M|        allocated.push_back(r);
  983|  1.42M|        return r;
  984|  1.42M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5UnaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS0_7UnaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|   158k|    {
  981|   158k|        auto r = new T(std::forward<Args>(args)...);
  982|   158k|        allocated.push_back(r);
  983|   158k|        return r;
  984|   158k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11LiteralNullEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  980|   215k|    {
  981|   215k|        auto r = new T(std::forward<Args>(args)...);
  982|   215k|        allocated.push_back(r);
  983|   215k|        return r;
  984|   215k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  5.36M|    {
  981|  5.36M|        auto r = new T(std::forward<Args>(args)...);
  982|  5.36M|        allocated.push_back(r);
  983|  5.36M|        return r;
  984|  5.36M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5IndexESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  980|   626k|    {
  981|   626k|        auto r = new T(std::forward<Args>(args)...);
  982|   626k|        allocated.push_back(r);
  983|   626k|        return r;
  984|   626k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESC_RNS7_INS0_8ArgParamENS9_ISD_EEEEbSC_RPNS0_3ASTEEEEPT_DpOT0_:
  980|   470k|    {
  981|   470k|        auto r = new T(std::forward<Args>(args)...);
  982|   470k|        allocated.push_back(r);
  983|   470k|        return r;
  984|   470k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_4SelfEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  980|  22.0k|    {
  981|  22.0k|        auto r = new T(std::forward<Args>(args)...);
  982|  22.0k|        allocated.push_back(r);
  983|  22.0k|        return r;
  984|  22.0k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_PNS0_14LiteralBooleanESD_PNS0_5ErrorEEEEPT_DpOT0_:
  980|  27.8k|    {
  981|  27.8k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.8k|        allocated.push_back(r);
  983|  27.8k|        return r;
  984|  27.8k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_14LiteralBooleanEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEbEEEPT_DpOT0_:
  980|  27.8k|    {
  981|  27.8k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.8k|        allocated.push_back(r);
  983|  27.8k|        return r;
  984|  27.8k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERSB_RNS7_INS0_8ArgParamENS9_ISF_EEEERbSE_RPNS0_3ASTEEEEPT_DpOT0_:
  980|   796k|    {
  981|   796k|        auto r = new T(std::forward<Args>(args)...);
  982|   796k|        allocated.push_back(r);
  983|   796k|        return r;
  984|   796k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|   132k|    {
  981|   132k|        auto r = new T(std::forward<Args>(args)...);
  982|   132k|        allocated.push_back(r);
  983|   132k|        return r;
  984|   132k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_7InSuperESD_PNS0_6BinaryESD_PNS0_3ASTEEEEPT_DpOT0_:
  980|  10.5k|    {
  981|  10.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  10.5k|        allocated.push_back(r);
  983|  10.5k|        return r;
  984|  10.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_7InSuperEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_SD_EEEPT_DpOT0_:
  980|  10.5k|    {
  981|  10.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  10.5k|        allocated.push_back(r);
  983|  10.5k|        return r;
  984|  10.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_10SuperIndexESD_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|  10.5k|    {
  981|  10.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  10.5k|        allocated.push_back(r);
  983|  10.5k|        return r;
  984|  10.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_10SuperIndexEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_PNS0_3ASTESD_DnEEEPT_DpOT0_:
  980|  10.5k|    {
  981|  10.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  10.5k|        allocated.push_back(r);
  983|  10.5k|        return r;
  984|  10.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_15DesugaredObjectEJRNS0_13LocationRangeERNSt3__14listIPNS0_3ASTENS6_9allocatorIS9_EEEERNS6_6vectorINS3_5FieldENSA_ISF_EEEEEEEPT_DpOT0_:
  980|   329k|    {
  981|   329k|        auto r = new T(std::forward<Args>(args)...);
  982|   329k|        allocated.push_back(r);
  983|   329k|        return r;
  984|   329k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   862k|    {
  981|   862k|        auto r = new T(std::forward<Args>(args)...);
  982|   862k|        allocated.push_back(r);
  983|   862k|        return r;
  984|   862k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralNumberESE_DnSE_DnSE_EEEPT_DpOT0_:
  980|  28.8k|    {
  981|  28.8k|        auto r = new T(std::forward<Args>(args)...);
  982|  28.8k|        allocated.push_back(r);
  983|  28.8k|        return r;
  984|  28.8k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS7_12basic_stringIcNS7_11char_traitsIcEENSA_IcEEEEEEEPT_DpOT0_:
  980|  28.8k|    {
  981|  28.8k|        auto r = new T(std::forward<Args>(args)...);
  982|  28.8k|        allocated.push_back(r);
  983|  28.8k|        return r;
  984|  28.8k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_18ArrayComprehensionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ArrayESD_bRNS7_INS0_17ComprehensionSpecENS9_ISG_EEEESD_EEEPT_DpOT0_:
  980|  27.3k|    {
  981|  27.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.3k|        allocated.push_back(r);
  983|  27.3k|        return r;
  984|  27.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  980|  27.3k|    {
  981|  27.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.3k|        allocated.push_back(r);
  983|  27.3k|        return r;
  984|  27.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_25ObjectComprehensionSimpleEJRNS0_13LocationRangeEPNS0_5IndexEPNS0_5LocalERPKNS0_10IdentifierERPNS0_3ASTEEEEPT_DpOT0_:
  980|  27.3k|    {
  981|  27.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.3k|        allocated.push_back(r);
  983|  27.3k|        return r;
  984|  27.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bRPNS0_3ASTESE_DnSE_DnSE_EEEPT_DpOT0_:
  980|  27.3k|    {
  981|  27.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  27.3k|        allocated.push_back(r);
  983|  27.3k|        return r;
  984|  27.3k|    }
_ZN7jsonnet8internal9Allocator15makeBuiltinBodyIZNS0_9Desugarer9stdlibASTENSt3__112basic_stringIcNS4_11char_traitsIcEENS4_9allocatorIcEEEEEUlvE_EEPKNS0_19BuiltinFunctionBodyERKNS0_13LocationRangeERKSA_OT_:
 1009|   266k|    {
 1010|   266k|        auto it = internedBuiltins.find(name);
 1011|   266k|        if (it != internedBuiltins.end()) {
  ------------------
  |  Branch (1011:13): [True: 121k, False: 145k]
  ------------------
 1012|   121k|            return it->second;
 1013|   121k|        }
 1014|   145k|        const Identifiers params = make_params();
 1015|   145k|        auto r = new BuiltinFunctionBody(loc, name, params);
 1016|   145k|        internedBuiltins[name] = r;
 1017|   145k|        return r;
 1018|   266k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_15BuiltinFunctionEJRKNS0_13LocationRangeERPKNS0_19BuiltinFunctionBodyEEEEPT_DpOT0_:
  980|   266k|    {
  981|   266k|        auto r = new T(std::forward<Args>(args)...);
  982|   266k|        allocated.push_back(r);
  983|   266k|        return r;
  984|   266k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISD_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  980|  3.47k|    {
  981|  3.47k|        auto r = new T(std::forward<Args>(args)...);
  982|  3.47k|        allocated.push_back(r);
  983|  3.47k|        return r;
  984|  3.47k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5ApplyERKSC_SF_SD_PNS0_3VarEEEEPT_DpOT0_:
  980|  3.47k|    {
  981|  3.47k|        auto r = new T(std::forward<Args>(args)...);
  982|  3.47k|        allocated.push_back(r);
  983|  3.47k|        return r;
  984|  3.47k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_RNS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  980|  3.47k|    {
  981|  3.47k|        auto r = new T(std::forward<Args>(args)...);
  982|  3.47k|        allocated.push_back(r);
  983|  3.47k|        return r;
  984|  3.47k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  6.94k|    {
  981|  6.94k|        auto r = new T(std::forward<Args>(args)...);
  982|  6.94k|        allocated.push_back(r);
  983|  6.94k|        return r;
  984|  6.94k|    }
formatter.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
formatter.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
formatter.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}

_ZN7jsonnet8internal20jsonnet_builtin_declEm:
   41|   266k|{
   42|   266k|    switch (builtin) {
   43|  6.35k|        case 0: return {U"makeArray", {U"sz", U"func"}};
  ------------------
  |  Branch (43:9): [True: 6.35k, False: 260k]
  ------------------
   44|  6.35k|        case 1: return {U"pow", {U"x", U"n"}};
  ------------------
  |  Branch (44:9): [True: 6.35k, False: 260k]
  ------------------
   45|  6.35k|        case 2: return {U"floor", {U"x"}};
  ------------------
  |  Branch (45:9): [True: 6.35k, False: 260k]
  ------------------
   46|  6.35k|        case 3: return {U"ceil", {U"x"}};
  ------------------
  |  Branch (46:9): [True: 6.35k, False: 260k]
  ------------------
   47|  6.35k|        case 4: return {U"sqrt", {U"x"}};
  ------------------
  |  Branch (47:9): [True: 6.35k, False: 260k]
  ------------------
   48|  6.35k|        case 5: return {U"sin", {U"x"}};
  ------------------
  |  Branch (48:9): [True: 6.35k, False: 260k]
  ------------------
   49|  6.35k|        case 6: return {U"cos", {U"x"}};
  ------------------
  |  Branch (49:9): [True: 6.35k, False: 260k]
  ------------------
   50|  6.35k|        case 7: return {U"tan", {U"x"}};
  ------------------
  |  Branch (50:9): [True: 6.35k, False: 260k]
  ------------------
   51|  6.35k|        case 8: return {U"asin", {U"x"}};
  ------------------
  |  Branch (51:9): [True: 6.35k, False: 260k]
  ------------------
   52|  6.35k|        case 9: return {U"acos", {U"x"}};
  ------------------
  |  Branch (52:9): [True: 6.35k, False: 260k]
  ------------------
   53|  6.35k|        case 10: return {U"atan", {U"x"}};
  ------------------
  |  Branch (53:9): [True: 6.35k, False: 260k]
  ------------------
   54|  6.35k|        case 11: return {U"type", {U"x"}};
  ------------------
  |  Branch (54:9): [True: 6.35k, False: 260k]
  ------------------
   55|  6.35k|        case 12: return {U"filter", {U"func", U"arr"}};
  ------------------
  |  Branch (55:9): [True: 6.35k, False: 260k]
  ------------------
   56|  6.35k|        case 13: return {U"objectHasEx", {U"obj", U"f", U"inc_hidden"}};
  ------------------
  |  Branch (56:9): [True: 6.35k, False: 260k]
  ------------------
   57|  6.35k|        case 14: return {U"length", {U"x"}};
  ------------------
  |  Branch (57:9): [True: 6.35k, False: 260k]
  ------------------
   58|  6.35k|        case 15: return {U"objectFieldsEx", {U"obj", U"inc_hidden"}};
  ------------------
  |  Branch (58:9): [True: 6.35k, False: 260k]
  ------------------
   59|  6.35k|        case 16: return {U"codepoint", {U"str"}};
  ------------------
  |  Branch (59:9): [True: 6.35k, False: 260k]
  ------------------
   60|  6.35k|        case 17: return {U"char", {U"n"}};
  ------------------
  |  Branch (60:9): [True: 6.35k, False: 260k]
  ------------------
   61|  6.35k|        case 18: return {U"log", {U"n"}};
  ------------------
  |  Branch (61:9): [True: 6.35k, False: 260k]
  ------------------
   62|  6.35k|        case 19: return {U"exp", {U"n"}};
  ------------------
  |  Branch (62:9): [True: 6.35k, False: 260k]
  ------------------
   63|  6.35k|        case 20: return {U"mantissa", {U"n"}};
  ------------------
  |  Branch (63:9): [True: 6.35k, False: 260k]
  ------------------
   64|  6.35k|        case 21: return {U"exponent", {U"n"}};
  ------------------
  |  Branch (64:9): [True: 6.35k, False: 260k]
  ------------------
   65|  6.35k|        case 22: return {U"modulo", {U"a", U"b"}};
  ------------------
  |  Branch (65:9): [True: 6.35k, False: 260k]
  ------------------
   66|  6.35k|        case 23: return {U"extVar", {U"x"}};
  ------------------
  |  Branch (66:9): [True: 6.35k, False: 260k]
  ------------------
   67|  6.35k|        case 24: return {U"primitiveEquals", {U"a", U"b"}};
  ------------------
  |  Branch (67:9): [True: 6.35k, False: 260k]
  ------------------
   68|  6.35k|        case 25: return {U"native", {U"name"}};
  ------------------
  |  Branch (68:9): [True: 6.35k, False: 260k]
  ------------------
   69|  6.35k|        case 26: return {U"md5", {U"str"}};
  ------------------
  |  Branch (69:9): [True: 6.35k, False: 260k]
  ------------------
   70|  6.35k|        case 27: return {U"trace", {U"str", U"rest"}};
  ------------------
  |  Branch (70:9): [True: 6.35k, False: 260k]
  ------------------
   71|  6.35k|        case 28: return {U"splitLimit", {U"str", U"c", U"maxsplits"}};
  ------------------
  |  Branch (71:9): [True: 6.35k, False: 260k]
  ------------------
   72|  6.35k|        case 29: return {U"substr", {U"str", U"from", U"len"}};
  ------------------
  |  Branch (72:9): [True: 6.35k, False: 260k]
  ------------------
   73|  6.35k|        case 30: return {U"range", {U"from", U"to"}};
  ------------------
  |  Branch (73:9): [True: 6.35k, False: 260k]
  ------------------
   74|  6.35k|        case 31: return {U"strReplace", {U"str", U"from", U"to"}};
  ------------------
  |  Branch (74:9): [True: 6.35k, False: 260k]
  ------------------
   75|  6.35k|        case 32: return {U"asciiLower", {U"str"}};
  ------------------
  |  Branch (75:9): [True: 6.35k, False: 260k]
  ------------------
   76|  6.35k|        case 33: return {U"asciiUpper", {U"str"}};
  ------------------
  |  Branch (76:9): [True: 6.35k, False: 260k]
  ------------------
   77|  6.35k|        case 34: return {U"join", {U"sep", U"arr"}};
  ------------------
  |  Branch (77:9): [True: 6.35k, False: 260k]
  ------------------
   78|  6.35k|        case 35: return {U"parseJson", {U"str"}};
  ------------------
  |  Branch (78:9): [True: 6.35k, False: 260k]
  ------------------
   79|  6.35k|        case 36: return {U"parseYaml", {U"str"}};
  ------------------
  |  Branch (79:9): [True: 6.35k, False: 260k]
  ------------------
   80|  6.35k|        case 37: return {U"encodeUTF8", {U"str"}};
  ------------------
  |  Branch (80:9): [True: 6.35k, False: 260k]
  ------------------
   81|  6.35k|        case 38: return {U"decodeUTF8", {U"arr"}};
  ------------------
  |  Branch (81:9): [True: 6.35k, False: 260k]
  ------------------
   82|  6.35k|        case 39: return {U"atan2", {U"y", U"x"}};
  ------------------
  |  Branch (82:9): [True: 6.35k, False: 260k]
  ------------------
   83|  6.35k|        case 40: return {U"hypot", {U"a", U"b"}};
  ------------------
  |  Branch (83:9): [True: 6.35k, False: 260k]
  ------------------
   84|  6.35k|        case 41: return {U"objectRemoveKey", {U"obj", U"key"}};
  ------------------
  |  Branch (84:9): [True: 6.35k, False: 260k]
  ------------------
   85|      0|        default:
  ------------------
  |  Branch (85:9): [True: 0, False: 266k]
  ------------------
   86|      0|            std::cerr << "INTERNAL ERROR: Unrecognized builtin function: " << builtin << std::endl;
   87|      0|            std::abort();
   88|   266k|    }
   89|       |    // Quiet, compiler.
   90|      0|    return BuiltinDecl();
   91|   266k|}
_ZN7jsonnet8internal13makeStdlibASTEPNS0_9AllocatorENSt3__112basic_stringIcNS3_11char_traitsIcEENS3_9allocatorIcEEEE:
 1022|  2.88k|DesugaredObject *makeStdlibAST(Allocator *alloc, std::string filename) {
 1023|  2.88k|    Desugarer desugarer(alloc, true);
 1024|  2.88k|    return desugarer.stdlibAST(filename);
 1025|  2.88k|}
_ZN7jsonnet8internal15jsonnet_desugarEPNS0_9AllocatorERPNS0_3ASTEPNSt3__13mapINS6_12basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEENS0_5VmExtENS6_4lessISD_EENSB_INS6_4pairIKSD_SE_EEEEEE:
 1028|  3.88k|{
 1029|  3.88k|    Desugarer desugarer(alloc);
 1030|  3.88k|    desugarer.desugarFile(ast, tlas);
 1031|  3.88k|}
_ZN7jsonnet8internal9DesugarerC2EPNS0_9AllocatorEb:
  229|  6.77k|    Desugarer(Allocator *alloc, bool isStdlib = false) : alloc(alloc), isStdlib(isStdlib) {}
_ZN7jsonnet8internal9Desugarer9stdlibASTENSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEE:
  921|  6.35k|    DesugaredObject *stdlibAST(std::string filename) {
  922|       |        // Now, implement the std library by wrapping in a local construct.
  923|  6.35k|        Tokens tokens = jsonnet_lex("std.jsonnet", STD_CODE);
  924|  6.35k|        AST *std_ast = jsonnet_parse(alloc, tokens);
  925|  6.35k|        desugar(std_ast, 0);
  926|  6.35k|        auto *std_obj = dynamic_cast<DesugaredObject *>(std_ast);
  927|  6.35k|        if (std_obj == nullptr) {
  ------------------
  |  Branch (927:13): [True: 0, False: 6.35k]
  ------------------
  928|      0|            std::cerr << "INTERNAL ERROR: std.jsonnet not an object." << std::endl;
  929|      0|            std::abort();
  930|      0|        }
  931|       |
  932|       |        // Bind 'std' builtins that are implemented natively.
  933|  6.35k|        DesugaredObject::Fields &fields = std_obj->fields;
  934|   273k|        for (unsigned long c = 0; c <= max_builtin; ++c) {
  ------------------
  |  Branch (934:35): [True: 266k, False: 6.35k]
  ------------------
  935|   266k|            const auto &decl = jsonnet_builtin_decl(c);
  936|   266k|            auto name = str(decl.name);
  937|   266k|            auto name_utf8 = "std:" + encode_utf8(decl.name);
  938|   266k|            auto fnbody = alloc->makeBuiltinBody(E, name_utf8, [this, &decl]()->Identifiers {
  939|   266k|                Identifiers params;
  940|   266k|                for (const auto &p : decl.params)
  941|   266k|                    params.push_back(this->id(p));
  942|   266k|                return params;
  943|   266k|            });
  944|   266k|            auto field = std::find_if(fields.begin(), fields.end(),
  945|   266k|                [=](const DesugaredObject::Field& f) {
  946|   266k|                    return static_cast<LiteralString*>(f.name)->value == decl.name;
  947|   266k|                });
  948|   266k|            auto fn = make<BuiltinFunction>(E, fnbody);
  949|   266k|            if (field != fields.end()) {
  ------------------
  |  Branch (949:17): [True: 44.4k, False: 222k]
  ------------------
  950|  44.4k|                field->body = fn;
  951|   222k|            } else {
  952|   222k|                fields.emplace_back(ObjectField::HIDDEN, name, fn);
  953|   222k|            }
  954|   266k|        }
  955|  6.35k|        fields.emplace_back(
  956|  6.35k|            ObjectField::HIDDEN, str(U"thisFile"), str(decode_utf8(filename)));
  957|  6.35k|        return std_obj;
  958|  6.35k|    }
_ZN7jsonnet8internal9Desugarer7desugarERPNS0_3ASTEj:
  686|  53.2M|    {
  687|  53.2M|        if (auto *ast = dynamic_cast<Apply *>(ast_)) {
  ------------------
  |  Branch (687:19): [True: 5.23M, False: 48.0M]
  ------------------
  688|  5.23M|            desugar(ast->target, obj_level);
  689|  5.23M|            for (ArgParam &arg : ast->args)
  ------------------
  |  Branch (689:32): [True: 8.66M, False: 5.23M]
  ------------------
  690|  8.66M|                desugar(arg.expr, obj_level);
  691|       |
  692|  48.0M|        } else if (auto *ast = dynamic_cast<ApplyBrace *>(ast_)) {
  ------------------
  |  Branch (692:26): [True: 51.2k, False: 47.9M]
  ------------------
  693|  51.2k|            desugar(ast->left, obj_level);
  694|  51.2k|            desugar(ast->right, obj_level);
  695|  51.2k|            ast_ =
  696|  51.2k|                make<Binary>(ast->location, ast->openFodder, ast->left, EF, BOP_PLUS, ast->right);
  697|       |
  698|  47.9M|        } else if (auto *ast = dynamic_cast<Array *>(ast_)) {
  ------------------
  |  Branch (698:26): [True: 744k, False: 47.2M]
  ------------------
  699|   744k|            for (auto &el : ast->elements)
  ------------------
  |  Branch (699:27): [True: 1.70M, False: 744k]
  ------------------
  700|  1.70M|                desugar(el.expr, obj_level);
  701|       |
  702|  47.2M|        } else if (auto *ast = dynamic_cast<ArrayComprehension *>(ast_)) {
  ------------------
  |  Branch (702:26): [True: 230k, False: 47.0M]
  ------------------
  703|   230k|            for (ComprehensionSpec &spec : ast->specs)
  ------------------
  |  Branch (703:42): [True: 409k, False: 230k]
  ------------------
  704|   409k|                desugar(spec.expr, obj_level);
  705|   230k|            desugar(ast->body, obj_level + 1);
  706|       |
  707|   230k|            ast_ = makeArrayComprehension(ast);
  708|       |
  709|  47.0M|        } else if (auto *ast = dynamic_cast<Assert *>(ast_)) {
  ------------------
  |  Branch (709:26): [True: 260k, False: 46.7M]
  ------------------
  710|   260k|            desugar(ast->cond, obj_level);
  711|   260k|            if (ast->message == nullptr) {
  ------------------
  |  Branch (711:17): [True: 19.1k, False: 241k]
  ------------------
  712|  19.1k|                ast->message = str(U"Assertion failed.");
  713|  19.1k|            }
  714|   260k|            desugar(ast->message, obj_level);
  715|   260k|            desugar(ast->rest, obj_level);
  716|       |
  717|       |            // if cond then rest else error msg
  718|   260k|            AST *branch_false = make<Error>(ast->location, EF, ast->message);
  719|   260k|            ast_ = make<Conditional>(
  720|   260k|                ast->location, ast->openFodder, ast->cond, EF, ast->rest, EF, branch_false);
  721|       |
  722|  46.7M|        } else if (auto *ast = dynamic_cast<Binary *>(ast_)) {
  ------------------
  |  Branch (722:26): [True: 5.63M, False: 41.1M]
  ------------------
  723|  5.63M|            desugar(ast->left, obj_level);
  724|  5.63M|            desugar(ast->right, obj_level);
  725|       |
  726|  5.63M|            bool invert = false;
  727|       |
  728|  5.63M|            switch (ast->op) {
  729|   193k|                case BOP_PERCENT: {
  ------------------
  |  Branch (729:17): [True: 193k, False: 5.43M]
  ------------------
  730|   193k|                    AST *f_mod = make<Index>(
  731|   193k|                        E, EF, std(), EF, false, str(U"mod"), EF, nullptr, EF, nullptr, EF);
  732|   193k|                    ArgParams args = {{ast->left, EF}, {ast->right, EF}};
  733|   193k|                    ast_ = make<Apply>(
  734|   193k|                        ast->location, ast->openFodder, f_mod, EF, args, false, EF, EF, false);
  735|   193k|                } break;
  736|       |
  737|   158k|                case BOP_MANIFEST_UNEQUAL: invert = true;
  ------------------
  |  Branch (737:17): [True: 158k, False: 5.47M]
  ------------------
  738|   158k|                [[fallthrough]];
  739|  1.16M|                case BOP_MANIFEST_EQUAL: {
  ------------------
  |  Branch (739:17): [True: 1.00M, False: 4.62M]
  ------------------
  740|  1.16M|                    ast_ = equals(ast->location, ast->left, ast->right);
  741|  1.16M|                    if (invert)
  ------------------
  |  Branch (741:25): [True: 158k, False: 1.00M]
  ------------------
  742|   158k|                        ast_ = make<Unary>(ast->location, ast->openFodder, UOP_NOT, ast_);
  743|  1.16M|                } break;
  744|       |
  745|  4.27M|                default:;
  ------------------
  |  Branch (745:17): [True: 4.27M, False: 1.36M]
  ------------------
  746|       |                    // Otherwise don't change it.
  747|  5.63M|            }
  748|       |
  749|  41.1M|        } else if (dynamic_cast<const BuiltinFunction *>(ast_)) {
  ------------------
  |  Branch (749:20): [True: 0, False: 41.1M]
  ------------------
  750|       |            // Nothing to do.
  751|       |
  752|  41.1M|        } else if (auto *ast = dynamic_cast<Conditional *>(ast_)) {
  ------------------
  |  Branch (752:26): [True: 2.12M, False: 38.9M]
  ------------------
  753|  2.12M|            desugar(ast->cond, obj_level);
  754|  2.12M|            desugar(ast->branchTrue, obj_level);
  755|  2.12M|            if (ast->branchFalse == nullptr)
  ------------------
  |  Branch (755:17): [True: 25.4k, False: 2.09M]
  ------------------
  756|  25.4k|                ast->branchFalse = null();
  757|  2.12M|            desugar(ast->branchFalse, obj_level);
  758|       |
  759|  38.9M|        } else if (auto *ast = dynamic_cast<Dollar *>(ast_)) {
  ------------------
  |  Branch (759:26): [True: 28.0k, False: 38.9M]
  ------------------
  760|  28.0k|            if (obj_level == 0) {
  ------------------
  |  Branch (760:17): [True: 178, False: 27.8k]
  ------------------
  761|    178|                throw StaticError(ast->location, "No top-level object found.");
  762|    178|            }
  763|  27.8k|            ast_ = var(id(U"$"));
  764|       |
  765|  38.9M|        } else if (auto *ast = dynamic_cast<Error *>(ast_)) {
  ------------------
  |  Branch (765:26): [True: 432k, False: 38.5M]
  ------------------
  766|   432k|            desugar(ast->expr, obj_level);
  767|       |
  768|  38.5M|        } else if (auto *ast = dynamic_cast<Function *>(ast_)) {
  ------------------
  |  Branch (768:26): [True: 143k, False: 38.3M]
  ------------------
  769|   143k|            desugar(ast->body, obj_level);
  770|   143k|            desugarParams(ast->params, obj_level);
  771|       |
  772|  38.3M|        } else if (auto *ast = dynamic_cast<Import *>(ast_)) {
  ------------------
  |  Branch (772:26): [True: 335, False: 38.3M]
  ------------------
  773|       |            // TODO(dcunnin): Abstract this into a template function if it becomes more common.
  774|    335|            AST *file = ast->file;
  775|    335|            desugar(file, obj_level);
  776|    335|            ast->file = dynamic_cast<LiteralString *>(file);
  777|       |
  778|  38.3M|        } else if (auto *ast = dynamic_cast<Importstr *>(ast_)) {
  ------------------
  |  Branch (778:26): [True: 375, False: 38.3M]
  ------------------
  779|       |            // TODO(dcunnin): Abstract this into a template function if it becomes more common.
  780|    375|            AST *file = ast->file;
  781|    375|            desugar(file, obj_level);
  782|    375|            ast->file = dynamic_cast<LiteralString *>(file);
  783|       |
  784|  38.3M|        } else if (auto *ast = dynamic_cast<Importbin *>(ast_)) {
  ------------------
  |  Branch (784:26): [True: 268, False: 38.3M]
  ------------------
  785|       |            // TODO(dcunnin): Abstract this into a template function if it becomes more common.
  786|    268|            AST *file = ast->file;
  787|    268|            desugar(file, obj_level);
  788|    268|            ast->file = dynamic_cast<LiteralString *>(file);
  789|       |
  790|  38.3M|        } else if (auto *ast = dynamic_cast<InSuper *>(ast_)) {
  ------------------
  |  Branch (790:26): [True: 5.25k, False: 38.3M]
  ------------------
  791|  5.25k|            desugar(ast->element, obj_level);
  792|       |
  793|  38.3M|        } else if (auto *ast = dynamic_cast<Index *>(ast_)) {
  ------------------
  |  Branch (793:26): [True: 5.73M, False: 32.6M]
  ------------------
  794|  5.73M|            desugar(ast->target, obj_level);
  795|  5.73M|            if (ast->isSlice) {
  ------------------
  |  Branch (795:17): [True: 116k, False: 5.62M]
  ------------------
  796|   116k|                if (ast->index == nullptr)
  ------------------
  |  Branch (796:21): [True: 14.8k, False: 101k]
  ------------------
  797|  14.8k|                    ast->index = null();
  798|   116k|                desugar(ast->index, obj_level);
  799|       |
  800|   116k|                if (ast->end == nullptr)
  ------------------
  |  Branch (800:21): [True: 65.6k, False: 50.9k]
  ------------------
  801|  65.6k|                    ast->end = null();
  802|   116k|                desugar(ast->end, obj_level);
  803|       |
  804|   116k|                if (ast->step == nullptr)
  ------------------
  |  Branch (804:21): [True: 109k, False: 6.71k]
  ------------------
  805|   109k|                    ast->step = null();
  806|   116k|                desugar(ast->step, obj_level);
  807|       |
  808|   116k|                ast_ = make<Apply>(
  809|   116k|                    ast->location,
  810|   116k|                    EF,
  811|   116k|                    make<Index>(
  812|   116k|                        E, EF, std(), EF, false, str(U"slice"), EF, nullptr, EF, nullptr, EF),
  813|   116k|                    EF,
  814|   116k|                    ArgParams{
  815|   116k|                        {ast->target, EF},
  816|   116k|                        {ast->index, EF},
  817|   116k|                        {ast->end, EF},
  818|   116k|                        {ast->step, EF},
  819|   116k|                    },
  820|   116k|                    false,  // trailing comma
  821|   116k|                    EF,
  822|   116k|                    EF,
  823|   116k|                    false  // tailstrict
  824|   116k|                );
  825|  5.62M|            } else {
  826|  5.62M|                if (ast->id != nullptr) {
  ------------------
  |  Branch (826:21): [True: 4.56M, False: 1.05M]
  ------------------
  827|  4.56M|                    assert(ast->index == nullptr);
  ------------------
  |  Branch (827:21): [True: 4.56M, False: 0]
  ------------------
  828|  4.56M|                    ast->index = str(ast->id->name);
  829|  4.56M|                    ast->id = nullptr;
  830|  4.56M|                }
  831|  5.62M|                desugar(ast->index, obj_level);
  832|  5.62M|            }
  833|       |
  834|  32.6M|        } else if (auto *ast = dynamic_cast<Local *>(ast_)) {
  ------------------
  |  Branch (834:26): [True: 1.58M, False: 31.0M]
  ------------------
  835|  1.58M|            for (auto &bind : ast->binds)
  ------------------
  |  Branch (835:29): [True: 1.69M, False: 1.58M]
  ------------------
  836|  1.69M|                desugar(bind.body, obj_level);
  837|  1.58M|            desugar(ast->body, obj_level);
  838|       |
  839|  1.69M|            for (auto &bind : ast->binds) {
  ------------------
  |  Branch (839:29): [True: 1.69M, False: 1.58M]
  ------------------
  840|  1.69M|                if (bind.functionSugar) {
  ------------------
  |  Branch (840:21): [True: 470k, False: 1.21M]
  ------------------
  841|   470k|                    desugarParams(bind.params, obj_level);
  842|   470k|                    bind.body = make<Function>(ast->location,
  843|   470k|                                               ast->openFodder,
  844|   470k|                                               bind.parenLeftFodder,
  845|   470k|                                               bind.params,
  846|   470k|                                               false,
  847|   470k|                                               bind.parenRightFodder,
  848|   470k|                                               bind.body);
  849|   470k|                    bind.functionSugar = false;
  850|   470k|                    bind.params.clear();
  851|   470k|                }
  852|  1.69M|            }
  853|       |
  854|  31.0M|        } else if (dynamic_cast<const LiteralBoolean *>(ast_)) {
  ------------------
  |  Branch (854:20): [True: 540k, False: 30.5M]
  ------------------
  855|       |            // Nothing to do.
  856|       |
  857|  30.5M|        } else if (dynamic_cast<const LiteralNumber *>(ast_)) {
  ------------------
  |  Branch (857:20): [True: 3.32M, False: 27.1M]
  ------------------
  858|       |            // Nothing to do.
  859|       |
  860|  27.1M|        } else if (auto *ast = dynamic_cast<LiteralString *>(ast_)) {
  ------------------
  |  Branch (860:26): [True: 7.78M, False: 19.4M]
  ------------------
  861|  7.78M|            if ((ast->tokenKind != LiteralString::RAW_DESUGARED) &&
  ------------------
  |  Branch (861:17): [True: 3.07M, False: 4.71M]
  ------------------
  862|  3.07M|                (ast->tokenKind != LiteralString::BLOCK) &&
  ------------------
  |  Branch (862:17): [True: 3.07M, False: 664]
  ------------------
  863|  3.07M|                (ast->tokenKind != LiteralString::VERBATIM_DOUBLE) &&
  ------------------
  |  Branch (863:17): [True: 3.07M, False: 2.88k]
  ------------------
  864|  3.07M|                (ast->tokenKind != LiteralString::VERBATIM_SINGLE)) {
  ------------------
  |  Branch (864:17): [True: 3.07M, False: 434]
  ------------------
  865|  3.07M|                ast->value = jsonnet_string_unescape(ast->location, ast->value);
  866|  3.07M|            }
  867|  7.78M|            ast->tokenKind = LiteralString::RAW_DESUGARED;
  868|  7.78M|            ast->blockIndent.clear();
  869|       |
  870|  19.4M|        } else if (dynamic_cast<const LiteralNull *>(ast_)) {
  ------------------
  |  Branch (870:20): [True: 330k, False: 19.0M]
  ------------------
  871|       |            // Nothing to do.
  872|       |
  873|  19.0M|        } else if (auto *ast = dynamic_cast<DesugaredObject *>(ast_)) {
  ------------------
  |  Branch (873:26): [True: 25.3k, False: 19.0M]
  ------------------
  874|  25.3k|            for (auto &field : ast->fields) {
  ------------------
  |  Branch (874:30): [True: 14.6k, False: 25.3k]
  ------------------
  875|  14.6k|                desugar(field.name, obj_level);
  876|  14.6k|                desugar(field.body, obj_level + 1);
  877|  14.6k|            }
  878|  25.3k|            for (AST *assert : ast->asserts) {
  ------------------
  |  Branch (878:30): [True: 226, False: 25.3k]
  ------------------
  879|    226|                desugar(assert, obj_level + 1);
  880|    226|            }
  881|       |
  882|  19.0M|        } else if (auto *ast = dynamic_cast<Object *>(ast_)) {
  ------------------
  |  Branch (882:26): [True: 330k, False: 18.7M]
  ------------------
  883|   330k|            ast_ = makeObject(ast, obj_level);
  884|       |
  885|  18.7M|        } else if (auto *ast = dynamic_cast<ObjectComprehension *>(ast_)) {
  ------------------
  |  Branch (885:26): [True: 27.3k, False: 18.6M]
  ------------------
  886|  27.3k|            ast_ = makeObjectComprehension(ast, obj_level);
  887|       |
  888|  18.6M|        } else if (auto *ast = dynamic_cast<ObjectComprehensionSimple *>(ast_)) {
  ------------------
  |  Branch (888:26): [True: 15.7k, False: 18.6M]
  ------------------
  889|  15.7k|            desugar(ast->field, obj_level);
  890|  15.7k|            desugar(ast->value, obj_level + 1);
  891|  15.7k|            desugar(ast->array, obj_level);
  892|       |
  893|  18.6M|        } else if (auto *ast = dynamic_cast<Parens *>(ast_)) {
  ------------------
  |  Branch (893:26): [True: 317k, False: 18.3M]
  ------------------
  894|       |            // Strip parens.
  895|   317k|            desugar(ast->expr, obj_level);
  896|   317k|            ast_ = ast->expr;
  897|       |
  898|  18.3M|        } else if (dynamic_cast<const Self *>(ast_)) {
  ------------------
  |  Branch (898:20): [True: 66.2k, False: 18.2M]
  ------------------
  899|       |            // Nothing to do.
  900|       |
  901|  18.2M|        } else if (auto *ast = dynamic_cast<SuperIndex *>(ast_)) {
  ------------------
  |  Branch (901:26): [True: 5.62k, False: 18.2M]
  ------------------
  902|  5.62k|            if (ast->id != nullptr) {
  ------------------
  |  Branch (902:17): [True: 592, False: 5.03k]
  ------------------
  903|    592|                assert(ast->index == nullptr);
  ------------------
  |  Branch (903:17): [True: 592, False: 0]
  ------------------
  904|    592|                ast->index = str(ast->id->name);
  905|    592|                ast->id = nullptr;
  906|    592|            }
  907|  5.62k|            desugar(ast->index, obj_level);
  908|       |
  909|  18.2M|        } else if (auto *ast = dynamic_cast<Unary *>(ast_)) {
  ------------------
  |  Branch (909:26): [True: 677k, False: 17.6M]
  ------------------
  910|   677k|            desugar(ast->expr, obj_level);
  911|       |
  912|  17.6M|        } else if (dynamic_cast<const Var *>(ast_)) {
  ------------------
  |  Branch (912:20): [True: 17.6M, False: 0]
  ------------------
  913|       |            // Nothing to do.
  914|       |
  915|  17.6M|        } else {
  916|      0|            std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  917|      0|            std::abort();
  918|      0|        }
  919|  53.2M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_NS0_8BinaryOpESF_EEEPT_DpOT0_:
  110|  51.1k|    {
  111|  51.1k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  51.1k|    }
_ZN7jsonnet8internal9Desugarer22makeArrayComprehensionEPNS0_18ArrayComprehensionE:
  417|   230k|    AST* makeArrayComprehension(ArrayComprehension *ast) {
  418|   230k|        int n = ast->specs.size();
  419|   230k|        AST *zero = make<LiteralNumber>(E, EF, "0.0");
  420|   230k|        AST *one = make<LiteralNumber>(E, EF, "1.0");
  421|   230k|        auto *_r = id(U"$r");
  422|   230k|        auto *_l = id(U"$l");
  423|   230k|        std::vector<const Identifier *> _i(n);
  424|   640k|        for (int i = 0; i < n; ++i) {
  ------------------
  |  Branch (424:25): [True: 409k, False: 230k]
  ------------------
  425|   409k|            UStringStream ss;
  426|   409k|            ss << U"$i_" << i;
  427|   409k|            _i[i] = id(ss.str());
  428|   409k|        }
  429|   230k|        std::vector<const Identifier *> _aux(n);
  430|   640k|        for (int i = 0; i < n; ++i) {
  ------------------
  |  Branch (430:25): [True: 409k, False: 230k]
  ------------------
  431|   409k|            UStringStream ss;
  432|   409k|            ss << U"$aux_" << i;
  433|   409k|            _aux[i] = id(ss.str());
  434|   409k|        }
  435|       |
  436|       |        // Build it from the inside out.  We keep wrapping 'in' with more ASTs.
  437|   230k|        assert(ast->specs[0].kind == ComprehensionSpec::FOR);
  ------------------
  |  Branch (437:9): [True: 230k, False: 0]
  ------------------
  438|       |
  439|   230k|        int last_for = n - 1;
  440|   380k|        while (ast->specs[last_for].kind != ComprehensionSpec::FOR)
  ------------------
  |  Branch (440:16): [True: 149k, False: 230k]
  ------------------
  441|   149k|            last_for--;
  442|       |        // $aux_{last_for}($i_{last_for} + 1, $r + [body])
  443|   230k|        AST *in = make<Apply>(
  444|   230k|            ast->body->location,
  445|   230k|            EF,
  446|   230k|            var(_aux[last_for]),
  447|   230k|            EF,
  448|   230k|            ArgParams{{make<Binary>(E, EF, var(_i[last_for]), EF, BOP_PLUS, one), EF},
  449|   230k|                {make<Binary>(E, EF, var(_r), EF, BOP_PLUS, singleton(ast->body)), EF}},
  450|   230k|            false,  // trailingComma
  451|   230k|            EF,
  452|   230k|            EF,
  453|   230k|            true  // tailstrict
  454|   230k|        );
  455|   640k|        for (int i = n - 1; i >= 0; --i) {
  ------------------
  |  Branch (455:29): [True: 409k, False: 230k]
  ------------------
  456|   409k|            const ComprehensionSpec &spec = ast->specs[i];
  457|   409k|            AST *out;
  458|   409k|            if (i > 0) {
  ------------------
  |  Branch (458:17): [True: 178k, False: 230k]
  ------------------
  459|   178k|                int prev_for = i - 1;
  460|  1.21G|                while (ast->specs[prev_for].kind != ComprehensionSpec::FOR)
  ------------------
  |  Branch (460:24): [True: 1.21G, False: 178k]
  ------------------
  461|  1.21G|                    prev_for--;
  462|       |
  463|       |                // aux_{prev_for}($i_{prev_for} + 1, $r)
  464|   178k|                out = make<Apply>(  // False branch.
  465|   178k|                    E,
  466|   178k|                    EF,
  467|   178k|                    var(_aux[prev_for]),
  468|   178k|                    EF,
  469|   178k|                    ArgParams{{
  470|   178k|                        make<Binary>(E, EF, var(_i[prev_for]), EF, BOP_PLUS, one),
  471|   178k|                            EF,
  472|   178k|                            },
  473|   178k|                      {
  474|   178k|                        var(_r),
  475|   178k|                            EF,
  476|   178k|                            }},
  477|   178k|                    false,  // trailingComma
  478|   178k|                    EF,
  479|   178k|                    EF,
  480|   178k|                    true  // tailstrict
  481|   178k|                );
  482|   230k|            } else {
  483|   230k|                out = var(_r);
  484|   230k|            }
  485|   409k|            switch (spec.kind) {
  ------------------
  |  Branch (485:21): [True: 409k, False: 0]
  ------------------
  486|   156k|                case ComprehensionSpec::IF: {
  ------------------
  |  Branch (486:17): [True: 156k, False: 253k]
  ------------------
  487|       |                  /*
  488|       |                    if [[[...cond...]]] then
  489|       |                    [[[...in...]]]
  490|       |                    else
  491|       |                    [[[...out...]]]
  492|       |                  */
  493|   156k|                    in = make<Conditional>(ast->location,
  494|   156k|                                           EF,
  495|   156k|                                           spec.expr,
  496|   156k|                                           EF,
  497|   156k|                                           in,  // True branch.
  498|   156k|                                           EF,
  499|   156k|                                           out);  // False branch.
  500|   156k|                } break;
  501|   253k|                case ComprehensionSpec::FOR: {
  ------------------
  |  Branch (501:17): [True: 253k, False: 156k]
  ------------------
  502|       |                    /*
  503|       |                      local $l = [[[...array...]]]
  504|       |                      aux_{i}(i_{i}, r) =
  505|       |                      if i_{i} >= std.length($l) then
  506|       |                      [[[...out...]]]
  507|       |                      else
  508|       |                      local [[[...var...]]] = $l[i_{i}];
  509|       |                      [[[...in...]]];
  510|       |                      if std.type($l) == "array" then
  511|       |                      aux_{i}(0, $r) tailstrict
  512|       |                      else
  513|       |                      error "In comprehension, can only iterate over array..";
  514|       |                    */
  515|   253k|                    in = make<Local>(
  516|   253k|                        ast->location,
  517|   253k|                        EF,
  518|   253k|                        Local::Binds{
  519|   253k|                          bind(_l, spec.expr),  // Need to check expr is an array
  520|   253k|                              bind(_aux[i],
  521|   253k|                                   make<Function>(
  522|   253k|                                       ast->location,
  523|   253k|                                       EF,
  524|   253k|                                       EF,
  525|   253k|                                       ArgParams{{EF, _i[i], EF}, {EF, _r, EF}},
  526|   253k|                                       false,  // trailingComma
  527|   253k|                                       EF,
  528|   253k|                                       make<Conditional>(ast->location,
  529|   253k|                                                         EF,
  530|   253k|                                                         make<Binary>(E,
  531|   253k|                                                                      EF,
  532|   253k|                                                                      var(_i[i]),
  533|   253k|                                                                      EF,
  534|   253k|                                                                      BOP_GREATER_EQ,
  535|   253k|                                                                      length(var(_l))),
  536|   253k|                                                         EF,
  537|   253k|                                                         out,
  538|   253k|                                                         EF,
  539|   253k|                                                         make<Local>(
  540|   253k|                                                             ast->location,
  541|   253k|                                                             EF,
  542|   253k|                                                             singleBind(spec.var,
  543|   253k|                                                                        make<Index>(E,
  544|   253k|                                                                                    EF,
  545|   253k|                                                                                    var(_l),
  546|   253k|                                                                                    EF,
  547|   253k|                                                                                    false,
  548|   253k|                                                                                    var(_i[i]),
  549|   253k|                                                                                    EF,
  550|   253k|                                                                                    nullptr,
  551|   253k|                                                                                    EF,
  552|   253k|                                                                                    nullptr,
  553|   253k|                                                                                    EF)),
  554|   253k|                                                             in))))},
  555|   253k|                        make<Conditional>(
  556|   253k|                            ast->location,
  557|   253k|                            EF,
  558|   253k|                            equals(ast->location, type(var(_l)), str(U"array")),
  559|   253k|                            EF,
  560|   253k|                            make<Apply>(
  561|   253k|                                E,
  562|   253k|                                EF,
  563|   253k|                                var(_aux[i]),
  564|   253k|                                EF,
  565|   253k|                                ArgParams{{zero, EF},
  566|   253k|                                  {
  567|   253k|                                    i == 0 ? make<Array>(
  ------------------
  |  Branch (567:37): [True: 230k, False: 22.2k]
  ------------------
  568|   230k|                                        E, EF, Array::Elements{}, false, EF)
  569|   253k|                                        : static_cast<AST *>(var(_r)),
  570|   253k|                                        EF,
  571|   253k|                                        }},
  572|   253k|                                false,  // trailingComma
  573|   253k|                                EF,
  574|   253k|                                EF,
  575|   253k|                                true),  // tailstrict
  576|   253k|                            EF,
  577|   253k|                            error(ast->location,
  578|   253k|                                  U"In comprehension, can only iterate over array.")));
  579|   253k|                } break;
  580|   409k|            }
  581|   409k|        }
  582|       |
  583|   230k|        return in;
  584|   230k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERA4_KcEEEPT_DpOT0_:
  110|   488k|    {
  111|   488k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   488k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  110|   230k|    {
  111|   230k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   230k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  110|   409k|    {
  111|   409k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   409k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ArrayEEEEPT_DpOT0_:
  110|   230k|    {
  111|   230k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   230k|    }
_ZN7jsonnet8internal9Desugarer9singletonEPNS0_3ASTE:
  163|   230k|    {
  164|   230k|        return make<Array>(
  165|   230k|            body->location, EF, Array::Elements{Array::Element(body, EF)}, false, EF);
  166|   230k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  110|   230k|    {
  111|   230k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   230k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  110|   431k|    {
  111|   431k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   431k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERKPNS0_3ASTESD_RSF_SD_SI_EEEPT_DpOT0_:
  110|   156k|    {
  111|   156k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   156k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  110|   253k|    {
  111|   253k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   253k|    }
_ZN7jsonnet8internal9Desugarer4bindEPKNS0_10IdentifierEPNS0_3ASTE:
  153|   984k|    {
  154|   984k|        return Local::Bind(EF, id, EF, body, false, EF, ArgParams{}, false, EF, EF);
  155|   984k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_NS7_INS0_8ArgParamENS9_ISE_EEEEbSD_PNS0_11ConditionalEEEEPT_DpOT0_:
  110|   253k|    {
  111|   253k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   253k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_6BinaryESD_RPNS0_3ASTESD_PNS0_5LocalEEEEPT_DpOT0_:
  110|   253k|    {
  111|   253k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   253k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ApplyEEEEPT_DpOT0_:
  110|   253k|    {
  111|   253k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   253k|    }
_ZN7jsonnet8internal9Desugarer6lengthEPNS0_3ASTE:
  199|   253k|    {
  200|   253k|        return stdFunc(U"length", v);
  201|   253k|    }
_ZN7jsonnet8internal9Desugarer7stdFuncERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEEPNS0_3ASTE:
  169|   509k|    {
  170|   509k|        return make<Apply>(
  171|   509k|            v->location,
  172|   509k|            EF,
  173|   509k|            make<Index>(E, EF, std(), EF, false, str(name), EF, nullptr, EF, nullptr, EF),
  174|   509k|            EF,
  175|   509k|            ArgParams{{v, EF}},
  176|   509k|            false,  // trailingComma
  177|   509k|            EF,
  178|   509k|            EF,
  179|   509k|            true  // tailstrict
  180|   509k|        );
  181|   509k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  110|   260k|    {
  111|   260k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   260k|    }
_ZN7jsonnet8internal9Desugarer10singleBindEPKNS0_10IdentifierEPNS0_3ASTE:
  158|   263k|    {
  159|   263k|        return {bind(id, body)};
  160|   263k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bSG_SE_DnSE_DnSE_EEEPT_DpOT0_:
  110|   253k|    {
  111|   253k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   253k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ApplyESD_SF_SD_PNS0_5ErrorEEEEPT_DpOT0_:
  110|   253k|    {
  111|   253k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   253k|    }
_ZN7jsonnet8internal9Desugarer4typeEPNS0_3ASTE:
  204|   256k|    {
  205|   256k|        return stdFunc(U"type", v);
  206|   256k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ArrayEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS8_INS3_7ElementENSA_ISF_EEEEbSE_EEEPT_DpOT0_:
  110|   230k|    {
  111|   230k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   230k|    }
_ZN7jsonnet8internal9Desugarer5errorERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEE:
  224|   253k|    {
  225|   253k|        return error(str(loc, msg));
  226|   253k|    }
_ZN7jsonnet8internal9Desugarer5errorEPNS0_3ASTE:
  219|   280k|    {
  220|   280k|        return make<Error>(msg->location, EF, msg);
  221|   280k|    }
_ZN7jsonnet8internal9Desugarer3strERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEE:
  130|  1.71M|    {
  131|  1.71M|        return make<LiteralString>(loc, EF, s, LiteralString::RAW_DESUGARED, "", "");
  132|  1.71M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_13LiteralStringEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERKNS7_12basic_stringIDiNS7_11char_traitsIDiEENSA_IDiEEEENS3_9TokenKindERA1_KcSP_EEEPT_DpOT0_:
  110|  9.08M|    {
  111|  9.08M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  9.08M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ErrorEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  110|   541k|    {
  111|   541k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   541k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_SF_SH_SF_EEEPT_DpOT0_:
  110|   260k|    {
  111|   260k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   260k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralStringESE_DnSE_DnSE_EEEPT_DpOT0_:
  110|  2.24M|    {
  111|  2.24M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  2.24M|    }
_ZN7jsonnet8internal9Desugarer3stdEv:
  145|  2.24M|    {
  146|       |        // In most places, there is a "$std" variable inserted by
  147|       |        // the desugarer. On the standard library itself there isn't,
  148|       |        // so use "std" instead.
  149|  2.24M|        return var(id(isStdlib ? U"std" : U"$std"));
  ------------------
  |  Branch (149:23): [True: 997k, False: 1.24M]
  ------------------
  150|  2.24M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_RNS7_INS0_8ArgParamENS9_ISI_EEEEbSH_SH_bEEEPT_DpOT0_:
  110|   193k|    {
  111|   193k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   193k|    }
_ZN7jsonnet8internal9Desugarer6equalsERKNS0_13LocationRangeEPNS0_3ASTES6_:
  214|  1.41M|    {
  215|  1.41M|        return stdFunc(loc, U"equals", a, b);
  216|  1.41M|    }
_ZN7jsonnet8internal9Desugarer7stdFuncERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEEPNS0_3ASTESF_:
  184|  1.42M|    {
  185|  1.42M|        return make<Apply>(
  186|  1.42M|            loc,
  187|  1.42M|            EF,
  188|  1.42M|            make<Index>(E, EF, std(), EF, false, str(name), EF, nullptr, EF, nullptr, EF),
  189|  1.42M|            EF,
  190|  1.42M|            ArgParams{{a, EF}, {b, EF}},
  191|  1.42M|            false,  // trailingComma
  192|  1.42M|            EF,
  193|  1.42M|            EF,
  194|  1.42M|            true  // tailstrict
  195|  1.42M|        );
  196|  1.42M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5IndexESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  110|  1.42M|    {
  111|  1.42M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  1.42M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5UnaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS0_7UnaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  110|   158k|    {
  111|   158k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   158k|    }
_ZN7jsonnet8internal9Desugarer4nullEv:
  135|   215k|    {
  136|   215k|        return make<LiteralNull>(E, EF);
  137|   215k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11LiteralNullEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  110|   215k|    {
  111|   215k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   215k|    }
_ZN7jsonnet8internal9Desugarer3varEPKNS0_10IdentifierE:
  140|  5.36M|    {
  141|  5.36M|        return make<Var>(E, EF, ident);
  142|  5.36M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_3VarEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  110|  5.36M|    {
  111|  5.36M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  5.36M|    }
_ZN7jsonnet8internal9Desugarer2idERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  120|  3.95M|    {
  121|  3.95M|        return alloc->makeIdentifier(s);
  122|  3.95M|    }
_ZN7jsonnet8internal9Desugarer13desugarParamsERNSt3__16vectorINS0_8ArgParamENS2_9allocatorIS4_EEEEj:
  232|  2.21M|    {
  233|  2.76M|        for (auto &param : params) {
  ------------------
  |  Branch (233:26): [True: 2.76M, False: 2.21M]
  ------------------
  234|  2.76M|            if (param.expr) {
  ------------------
  |  Branch (234:17): [True: 136k, False: 2.62M]
  ------------------
  235|       |                // Default arg.
  236|   136k|                desugar(param.expr, obj_level);
  237|   136k|            }
  238|  2.76M|        }
  239|  2.21M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5IndexESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  110|   626k|    {
  111|   626k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   626k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_8FunctionEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESC_RNS7_INS0_8ArgParamENS9_ISD_EEEEbSC_RPNS0_3ASTEEEEPT_DpOT0_:
  110|   470k|    {
  111|   470k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   470k|    }
_ZN7jsonnet8internal9Desugarer10makeObjectEPNS0_6ObjectEj:
  586|   330k|    AST* makeObject(Object *ast, unsigned obj_level) {
  587|       |        // Hidden variable to allow outer/top binding.
  588|   330k|        if (obj_level == 0) {
  ------------------
  |  Branch (588:13): [True: 21.0k, False: 308k]
  ------------------
  589|  21.0k|            const Identifier *hidden_var = id(U"$");
  590|  21.0k|            auto *body = make<Self>(E, EF);
  591|  21.0k|            ast->fields.push_back(ObjectField::Local(EF, EF, hidden_var, EF, body, EF));
  592|  21.0k|        }
  593|       |
  594|   330k|        SuperVars svs = desugarFields(ast, ast->fields, obj_level);
  595|       |
  596|   330k|        DesugaredObject::Fields new_fields;
  597|   330k|        ASTs new_asserts;
  598|  1.51M|        for (const ObjectField &field : ast->fields) {
  ------------------
  |  Branch (598:39): [True: 1.51M, False: 330k]
  ------------------
  599|  1.51M|            if (field.kind == ObjectField::ASSERT) {
  ------------------
  |  Branch (599:17): [True: 27.8k, False: 1.48M]
  ------------------
  600|  27.8k|                new_asserts.push_back(field.expr2);
  601|  1.48M|            } else if (field.kind == ObjectField::FIELD_EXPR) {
  ------------------
  |  Branch (601:24): [True: 1.48M, False: 0]
  ------------------
  602|  1.48M|                new_fields.emplace_back(field.hide, field.expr1, field.expr2);
  603|  1.48M|            } else {
  604|      0|                std::cerr << "INTERNAL ERROR: field should have been desugared: " << field.kind
  605|      0|                          << std::endl;
  606|      0|            }
  607|  1.51M|        }
  608|       |
  609|   330k|        AST* retval = make<DesugaredObject>(ast->location, new_asserts, new_fields);
  610|   330k|        if (svs.size() > 0) {
  ------------------
  |  Branch (610:13): [True: 846, False: 329k]
  ------------------
  611|    846|            Local::Binds binds;
  612|   119k|            for (const auto &pair : svs) {
  ------------------
  |  Branch (612:35): [True: 119k, False: 846]
  ------------------
  613|   119k|                if (pair.second == nullptr) {
  ------------------
  |  Branch (613:21): [True: 415, False: 119k]
  ------------------
  614|       |                    // Self binding
  615|    415|                    binds.push_back(bind(pair.first, make<Self>(E, EF)));
  616|   119k|                } else {
  617|       |                    // Super binding
  618|   119k|                    binds.push_back(bind(pair.first, pair.second));
  619|   119k|                }
  620|   119k|            }
  621|    846|            retval = make<Local>(ast->location, EF, binds, retval);
  622|    846|        }
  623|       |
  624|   330k|        return retval;
  625|   330k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_4SelfEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  110|  22.0k|    {
  111|  22.0k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  22.0k|    }
_ZN7jsonnet8internal9Desugarer13desugarFieldsEPNS0_3ASTERNSt3__16vectorINS0_11ObjectFieldENS4_9allocatorIS6_EEEEj:
  246|   357k|    {
  247|       |        // Desugar children
  248|  1.60M|        for (auto &field : fields) {
  ------------------
  |  Branch (248:26): [True: 1.60M, False: 357k]
  ------------------
  249|  1.60M|            if (field.expr1 != nullptr)
  ------------------
  |  Branch (249:17): [True: 74.5k, False: 1.53M]
  ------------------
  250|  74.5k|                desugar(field.expr1, obj_level);
  251|  1.60M|            desugar(field.expr2, obj_level + 1);
  252|  1.60M|            if (field.expr3 != nullptr)
  ------------------
  |  Branch (252:17): [True: 121, False: 1.60M]
  ------------------
  253|    121|                desugar(field.expr3, obj_level + 1);
  254|  1.60M|            desugarParams(field.params, obj_level + 1);
  255|  1.60M|        }
  256|       |
  257|       |        // Simplify asserts
  258|  1.60M|        for (auto &field : fields) {
  ------------------
  |  Branch (258:26): [True: 1.60M, False: 357k]
  ------------------
  259|  1.60M|            if (field.kind != ObjectField::ASSERT)
  ------------------
  |  Branch (259:17): [True: 1.57M, False: 27.8k]
  ------------------
  260|  1.57M|                continue;
  261|  27.8k|            AST *msg = field.expr3;
  262|  27.8k|            field.expr3 = nullptr;
  263|  27.8k|            if (msg == nullptr) {
  ------------------
  |  Branch (263:17): [True: 27.7k, False: 121]
  ------------------
  264|       |                // The location is what appears in the stacktrace.
  265|  27.7k|                msg = str(field.expr2->location, U"Object assertion failed.");
  266|  27.7k|            }
  267|       |
  268|       |            // if expr2 then true else error msg
  269|  27.8k|            field.expr2 = make<Conditional>(field.expr2->location,
  270|  27.8k|                                            EF,
  271|  27.8k|                                            field.expr2,
  272|  27.8k|                                            EF,
  273|  27.8k|                                            make<LiteralBoolean>(E, EF, true),
  274|  27.8k|                                            EF,
  275|  27.8k|                                            error(msg));
  276|  27.8k|        }
  277|       |
  278|       |        // Remove methods
  279|  1.60M|        for (auto &field : fields) {
  ------------------
  |  Branch (279:26): [True: 1.60M, False: 357k]
  ------------------
  280|  1.60M|            if (!field.methodSugar)
  ------------------
  |  Branch (280:17): [True: 808k, False: 796k]
  ------------------
  281|   808k|                continue;
  282|   796k|            field.expr2 = make<Function>(field.expr2->location,
  283|   796k|                                         EF,
  284|   796k|                                         field.fodderL,
  285|   796k|                                         field.params,
  286|   796k|                                         field.trailingComma,
  287|   796k|                                         field.fodderR,
  288|   796k|                                         field.expr2);
  289|   796k|            field.methodSugar = false;
  290|   796k|            field.params.clear();
  291|   796k|        }
  292|       |
  293|       |        // Remove object-level locals
  294|   357k|        auto copy = fields;
  295|   357k|        fields.clear();
  296|   357k|        Local::Binds binds;
  297|  1.60M|        for (auto &local : copy) {
  ------------------
  |  Branch (297:26): [True: 1.60M, False: 357k]
  ------------------
  298|  1.60M|            if (local.kind != ObjectField::LOCAL)
  ------------------
  |  Branch (298:17): [True: 1.53M, False: 66.1k]
  ------------------
  299|  1.53M|                continue;
  300|  66.1k|            binds.push_back(bind(local.id, local.expr2));
  301|  66.1k|        }
  302|  1.60M|        for (auto &field : copy) {
  ------------------
  |  Branch (302:26): [True: 1.60M, False: 357k]
  ------------------
  303|  1.60M|            if (field.kind == ObjectField::LOCAL)
  ------------------
  |  Branch (303:17): [True: 66.1k, False: 1.53M]
  ------------------
  304|  66.1k|                continue;
  305|  1.53M|            if (!binds.empty())
  ------------------
  |  Branch (305:17): [True: 834k, False: 704k]
  ------------------
  306|   834k|                field.expr2 = make<Local>(field.expr2->location, EF, binds, field.expr2);
  307|  1.53M|            fields.push_back(field);
  308|  1.53M|        }
  309|       |
  310|       |        // Change all to FIELD_EXPR
  311|  1.53M|        for (auto &field : fields) {
  ------------------
  |  Branch (311:26): [True: 1.53M, False: 357k]
  ------------------
  312|  1.53M|            switch (field.kind) {
  ------------------
  |  Branch (312:21): [True: 1.53M, False: 0]
  ------------------
  313|  27.8k|                case ObjectField::ASSERT:
  ------------------
  |  Branch (313:17): [True: 27.8k, False: 1.51M]
  ------------------
  314|       |                    // Nothing to do.
  315|  27.8k|                    break;
  316|       |
  317|  1.43M|                case ObjectField::FIELD_ID:
  ------------------
  |  Branch (317:17): [True: 1.43M, False: 102k]
  ------------------
  318|  1.43M|                    field.expr1 = str(field.idLocation, field.id->name);
  319|  1.43M|                    field.kind = ObjectField::FIELD_EXPR;
  320|  1.43M|                    break;
  321|       |
  322|  28.5k|                case ObjectField::FIELD_EXPR:
  ------------------
  |  Branch (322:17): [True: 28.5k, False: 1.51M]
  ------------------
  323|       |                    // Nothing to do.
  324|  28.5k|                    break;
  325|       |
  326|  45.8k|                case ObjectField::FIELD_STR:
  ------------------
  |  Branch (326:17): [True: 45.8k, False: 1.49M]
  ------------------
  327|       |                    // Just set the flag.
  328|  45.8k|                    field.kind = ObjectField::FIELD_EXPR;
  329|  45.8k|                    break;
  330|       |
  331|      0|                case ObjectField::LOCAL:
  ------------------
  |  Branch (331:17): [True: 0, False: 1.53M]
  ------------------
  332|      0|                    std::cerr << "Locals should be removed by now." << std::endl;
  333|      0|                    abort();
  334|  1.53M|            }
  335|  1.53M|        }
  336|       |
  337|       |        /** Replaces all occurrences of self, super[f] and e in super with variables.
  338|       |         *
  339|       |         * Returns all variables and original expressions via super_vars.
  340|       |         */
  341|   357k|        class SubstituteSelfSuper : public CompilerPass {
  342|   357k|            Desugarer *desugarer;
  343|   357k|            SuperVars &superVars;
  344|   357k|            unsigned &counter;
  345|   357k|            const Identifier *newSelf;
  346|       |
  347|   357k|           public:
  348|   357k|            SubstituteSelfSuper(Desugarer *desugarer, SuperVars &super_vars, unsigned &counter)
  349|   357k|                : CompilerPass(*desugarer->alloc),
  350|   357k|                  desugarer(desugarer),
  351|   357k|                  superVars(super_vars),
  352|   357k|                  counter(counter),
  353|   357k|                  newSelf(nullptr)
  354|   357k|            {
  355|   357k|            }
  356|   357k|            void visitExpr(AST *&expr)
  357|   357k|            {
  358|   357k|                if (dynamic_cast<Self *>(expr)) {
  359|   357k|                    if (newSelf == nullptr) {
  360|   357k|                        newSelf = desugarer->id(U"$outer_self");
  361|   357k|                        superVars.emplace_back(newSelf, nullptr);
  362|   357k|                    }
  363|   357k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, newSelf);
  364|   357k|                } else if (auto *super_index = dynamic_cast<SuperIndex *>(expr)) {
  365|   357k|                    UStringStream ss;
  366|   357k|                    ss << U"$outer_super_index" << (counter++);
  367|   357k|                    const Identifier *super_var = desugarer->id(ss.str());
  368|       |                    // Desugaring of expr should already have occurred.
  369|   357k|                    assert(super_index->index != nullptr);
  370|       |                    // Re-use super_index since we're replacing it here.
  371|   357k|                    superVars.emplace_back(super_var, super_index);
  372|   357k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, super_var);
  373|   357k|                } else if (auto *in_super = dynamic_cast<InSuper *>(expr)) {
  374|   357k|                    UStringStream ss;
  375|   357k|                    ss << U"$outer_in_super" << (counter++);
  376|   357k|                    const Identifier *in_super_var = desugarer->id(ss.str());
  377|       |                    // Re-use in_super since we're replacing it here.
  378|   357k|                    superVars.emplace_back(in_super_var, in_super);
  379|   357k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, in_super_var);
  380|   357k|                }
  381|   357k|                CompilerPass::visitExpr(expr);
  382|   357k|            }
  383|   357k|        };
  384|       |
  385|   357k|        SuperVars super_vars;
  386|   357k|        unsigned counter = 0;
  387|       |
  388|       |        // Remove +:
  389|  1.53M|        for (auto &field : fields) {
  ------------------
  |  Branch (389:26): [True: 1.53M, False: 357k]
  ------------------
  390|  1.53M|            if (!field.superSugar)
  ------------------
  |  Branch (390:17): [True: 1.52M, False: 10.5k]
  ------------------
  391|  1.52M|                continue;
  392|       |            // We have to bind self/super from expr1 outside the class, as we copy the expression
  393|       |            // into the field body.
  394|       |            // Clone it so that we maintain the AST as a tree.
  395|  10.5k|            AST *index = clone(field.expr1);
  396|       |            // This will remove self/super.
  397|  10.5k|            SubstituteSelfSuper(this, super_vars, counter).expr(index);
  398|  10.5k|            field.expr2 = make<Conditional>(
  399|  10.5k|                ast->location,
  400|  10.5k|                EF,
  401|  10.5k|                make<InSuper>(ast->location, EF, index, EF, EF),
  402|  10.5k|                EF,
  403|  10.5k|                make<Binary>(ast->location,
  404|  10.5k|                             EF,
  405|  10.5k|                             make<SuperIndex>(ast->location, EF, EF, clone(index), EF, nullptr),
  406|  10.5k|                             EF,
  407|  10.5k|                             BOP_PLUS,
  408|  10.5k|                             field.expr2),
  409|  10.5k|                EF,
  410|  10.5k|                clone(field.expr2));
  411|  10.5k|            field.superSugar = false;
  412|  10.5k|        }
  413|       |
  414|   357k|        return super_vars;
  415|   357k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_PNS0_14LiteralBooleanESD_PNS0_5ErrorEEEEPT_DpOT0_:
  110|  27.8k|    {
  111|  27.8k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  27.8k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_14LiteralBooleanEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEbEEEPT_DpOT0_:
  110|  27.8k|    {
  111|  27.8k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  27.8k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERSB_RNS7_INS0_8ArgParamENS9_ISF_EEEERbSE_RPNS0_3ASTEEEEPT_DpOT0_:
  110|   796k|    {
  111|   796k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   796k|    }
_ZN7jsonnet8internal9Desugarer5cloneEPNS0_3ASTE:
  115|  31.6k|    {
  116|  31.6k|        return clone_ast(*alloc, ast);
  117|  31.6k|    }
_ZZN7jsonnet8internal9Desugarer13desugarFieldsEPNS0_3ASTERNSt3__16vectorINS0_11ObjectFieldENS4_9allocatorIS6_EEEEjEN19SubstituteSelfSuperC2EPS1_RNS5_INS4_4pairIPKNS0_10IdentifierES3_EENS7_ISH_EEEERj:
  349|  10.5k|                : CompilerPass(*desugarer->alloc),
  350|  10.5k|                  desugarer(desugarer),
  351|  10.5k|                  superVars(super_vars),
  352|  10.5k|                  counter(counter),
  353|  10.5k|                  newSelf(nullptr)
  354|  10.5k|            {
  355|  10.5k|            }
_ZZN7jsonnet8internal9Desugarer13desugarFieldsEPNS0_3ASTERNSt3__16vectorINS0_11ObjectFieldENS4_9allocatorIS6_EEEEjEN19SubstituteSelfSuper9visitExprERS3_:
  357|   791k|            {
  358|   791k|                if (dynamic_cast<Self *>(expr)) {
  ------------------
  |  Branch (358:21): [True: 3.04k, False: 788k]
  ------------------
  359|  3.04k|                    if (newSelf == nullptr) {
  ------------------
  |  Branch (359:25): [True: 425, False: 2.62k]
  ------------------
  360|    425|                        newSelf = desugarer->id(U"$outer_self");
  361|    425|                        superVars.emplace_back(newSelf, nullptr);
  362|    425|                    }
  363|  3.04k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, newSelf);
  364|   788k|                } else if (auto *super_index = dynamic_cast<SuperIndex *>(expr)) {
  ------------------
  |  Branch (364:34): [True: 73.6k, False: 715k]
  ------------------
  365|  73.6k|                    UStringStream ss;
  366|  73.6k|                    ss << U"$outer_super_index" << (counter++);
  367|  73.6k|                    const Identifier *super_var = desugarer->id(ss.str());
  368|       |                    // Desugaring of expr should already have occurred.
  369|  73.6k|                    assert(super_index->index != nullptr);
  ------------------
  |  Branch (369:21): [True: 73.6k, False: 0]
  ------------------
  370|       |                    // Re-use super_index since we're replacing it here.
  371|  73.6k|                    superVars.emplace_back(super_var, super_index);
  372|  73.6k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, super_var);
  373|   715k|                } else if (auto *in_super = dynamic_cast<InSuper *>(expr)) {
  ------------------
  |  Branch (373:34): [True: 55.8k, False: 659k]
  ------------------
  374|  55.8k|                    UStringStream ss;
  375|  55.8k|                    ss << U"$outer_in_super" << (counter++);
  376|  55.8k|                    const Identifier *in_super_var = desugarer->id(ss.str());
  377|       |                    // Re-use in_super since we're replacing it here.
  378|  55.8k|                    superVars.emplace_back(in_super_var, in_super);
  379|  55.8k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, in_super_var);
  380|  55.8k|                }
  381|   791k|                CompilerPass::visitExpr(expr);
  382|   791k|            }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_7InSuperESD_PNS0_6BinaryESD_PNS0_3ASTEEEEPT_DpOT0_:
  110|  10.5k|    {
  111|  10.5k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  10.5k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_7InSuperEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_SD_EEEPT_DpOT0_:
  110|  10.5k|    {
  111|  10.5k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  10.5k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_10SuperIndexESD_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  110|  10.5k|    {
  111|  10.5k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  10.5k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_10SuperIndexEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_PNS0_3ASTESD_DnEEEPT_DpOT0_:
  110|  10.5k|    {
  111|  10.5k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  10.5k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_15DesugaredObjectEJRNS0_13LocationRangeERNSt3__14listIPNS0_3ASTENS6_9allocatorIS9_EEEERNS6_6vectorINS3_5FieldENSA_ISF_EEEEEEEPT_DpOT0_:
  110|   329k|    {
  111|   329k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   329k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  110|   862k|    {
  111|   862k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   862k|    }
_ZN7jsonnet8internal9Desugarer23makeObjectComprehensionEPNS0_19ObjectComprehensionEj:
  627|  27.3k|    AST* makeObjectComprehension(ObjectComprehension *ast, unsigned obj_level) {
  628|       |        // Hidden variable to allow outer/top binding.
  629|  27.3k|        if (obj_level == 0) {
  ------------------
  |  Branch (629:13): [True: 569, False: 26.7k]
  ------------------
  630|    569|            const Identifier *hidden_var = id(U"$");
  631|    569|            auto *body = make<Self>(E, EF);
  632|    569|            ast->fields.push_back(ObjectField::Local(EF, EF, hidden_var, EF, body, EF));
  633|    569|        }
  634|       |
  635|  27.3k|        SuperVars svs = desugarFields(ast, ast->fields, obj_level);
  636|       |
  637|  27.3k|        AST *field = ast->fields.front().expr1;
  638|  27.3k|        AST *value = ast->fields.front().expr2;
  639|       |
  640|       |        /*  {
  641|       |            [arr[0]]: local x = arr[1], y = arr[2], z = arr[3]; val_expr
  642|       |            for arr in [ [key_expr, x, y, z] for ...  ]
  643|       |            }
  644|       |        */
  645|  27.3k|        auto *_arr = id(U"$arr");
  646|  27.3k|        AST *zero = make<LiteralNumber>(E, EF, "0.0");
  647|  27.3k|        int counter = 1;
  648|  27.3k|        Local::Binds binds;
  649|  27.3k|        Array::Elements arr_e{Array::Element(field, EF)};
  650|  39.8k|        for (ComprehensionSpec &spec : ast->specs) {
  ------------------
  |  Branch (650:38): [True: 39.8k, False: 27.3k]
  ------------------
  651|  39.8k|            if (spec.kind == ComprehensionSpec::FOR) {
  ------------------
  |  Branch (651:17): [True: 28.8k, False: 11.0k]
  ------------------
  652|  28.8k|                std::stringstream num;
  653|  28.8k|                num << counter++;
  654|  28.8k|                binds.push_back(bind(spec.var,
  655|  28.8k|                                     make<Index>(E,
  656|  28.8k|                                                 EF,
  657|  28.8k|                                                 var(_arr),
  658|  28.8k|                                                 EF,
  659|  28.8k|                                                 false,
  660|  28.8k|                                                 make<LiteralNumber>(E, EF, num.str()),
  661|  28.8k|                                                 EF,
  662|  28.8k|                                                 nullptr,
  663|  28.8k|                                                 EF,
  664|  28.8k|                                                 nullptr,
  665|  28.8k|                                                 EF)));
  666|  28.8k|                arr_e.emplace_back(var(spec.var), EF);
  667|  28.8k|            }
  668|  39.8k|        }
  669|  27.3k|        AST *arr = make<ArrayComprehension>(ast->location,
  670|  27.3k|                                            EF,
  671|  27.3k|                                            make<Array>(ast->location, EF, arr_e, false, EF),
  672|  27.3k|                                            EF,
  673|  27.3k|                                            false,
  674|  27.3k|                                            ast->specs,
  675|  27.3k|                                            EF);
  676|  27.3k|        desugar(arr, obj_level);
  677|  27.3k|        return make<ObjectComprehensionSimple>(
  678|  27.3k|            ast->location,
  679|  27.3k|            make<Index>(E, EF, var(_arr), EF, false, zero, EF, nullptr, EF, nullptr, EF),
  680|  27.3k|            make<Local>(ast->location, EF, binds, value),
  681|  27.3k|            _arr,
  682|  27.3k|            arr);
  683|  27.3k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralNumberESE_DnSE_DnSE_EEEPT_DpOT0_:
  110|  28.8k|    {
  111|  28.8k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  28.8k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS7_12basic_stringIcNS7_11char_traitsIcEENSA_IcEEEEEEEPT_DpOT0_:
  110|  28.8k|    {
  111|  28.8k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  28.8k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_18ArrayComprehensionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ArrayESD_bRNS7_INS0_17ComprehensionSpecENS9_ISG_EEEESD_EEEPT_DpOT0_:
  110|  27.3k|    {
  111|  27.3k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  27.3k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  110|  27.3k|    {
  111|  27.3k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  27.3k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_25ObjectComprehensionSimpleEJRNS0_13LocationRangeEPNS0_5IndexEPNS0_5LocalERPKNS0_10IdentifierERPNS0_3ASTEEEEPT_DpOT0_:
  110|  27.3k|    {
  111|  27.3k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  27.3k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bRPNS0_3ASTESE_DnSE_DnSE_EEEPT_DpOT0_:
  110|  27.3k|    {
  111|  27.3k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  27.3k|    }
_ZN7jsonnet8internal9Desugarer3strERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  125|  7.36M|    {
  126|  7.36M|        return make<LiteralString>(E, EF, s, LiteralString::RAW_DESUGARED, "", "");
  127|  7.36M|    }
_ZZN7jsonnet8internal9Desugarer9stdlibASTENSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEEENKUlvE_clEv:
  938|   145k|            auto fnbody = alloc->makeBuiltinBody(E, name_utf8, [this, &decl]()->Identifiers {
  939|   145k|                Identifiers params;
  940|   145k|                for (const auto &p : decl.params)
  ------------------
  |  Branch (940:36): [True: 215k, False: 145k]
  ------------------
  941|   215k|                    params.push_back(this->id(p));
  942|   145k|                return params;
  943|   145k|            });
_ZZN7jsonnet8internal9Desugarer9stdlibASTENSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEEENKUlRKNS0_15DesugaredObject5FieldEE_clESC_:
  945|  32.6M|                [=](const DesugaredObject::Field& f) {
  946|  32.6M|                    return static_cast<LiteralString*>(f.name)->value == decl.name;
  947|  32.6M|                });
_ZN7jsonnet8internal9Desugarer4makeINS0_15BuiltinFunctionEJRKNS0_13LocationRangeERPKNS0_19BuiltinFunctionBodyEEEEPT_DpOT0_:
  110|   266k|    {
  111|   266k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   266k|    }
_ZN7jsonnet8internal9Desugarer11desugarFileERPNS0_3ASTEPNSt3__13mapINS5_12basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEENS0_5VmExtENS5_4lessISC_EENSA_INS5_4pairIKSC_SD_EEEEEE:
  961|  3.88k|    {
  962|  3.88k|        desugar(ast, 0);
  963|       |
  964|  3.88k|        DesugaredObject *std_obj = stdlibAST(ast->location.file);
  965|       |
  966|  3.88k|        std::vector<std::string> empty;
  967|  3.88k|        auto line_end_blank = Fodder{{FodderElement::LINE_END, 1, 0, empty}};
  968|  3.88k|        auto line_end = Fodder{{FodderElement::LINE_END, 0, 0, empty}};
  969|       |
  970|       |        // local body = ast;
  971|       |        // if std.type(body) == "function") then
  972|       |        //     body(tlas...)
  973|       |        // else
  974|       |        //     body
  975|  3.88k|        if (tlas != nullptr) {
  ------------------
  |  Branch (975:13): [True: 3.47k, False: 415]
  ------------------
  976|  3.47k|            LocationRange tla_loc("Top-level function");
  977|  3.47k|            ArgParams args;
  978|  3.47k|            for (const auto &pair : *tlas) {
  ------------------
  |  Branch (978:35): [True: 0, False: 3.47k]
  ------------------
  979|      0|                AST *expr;
  980|      0|                if (pair.second.isCode) {
  ------------------
  |  Branch (980:21): [True: 0, False: 0]
  ------------------
  981|      0|                    Tokens tokens = jsonnet_lex("tla:" + pair.first, pair.second.data.c_str());
  982|      0|                    expr = jsonnet_parse(alloc, tokens);
  983|      0|                    desugar(expr, 0);
  984|      0|                } else {
  985|      0|                    expr = str(decode_utf8(pair.second.data));
  986|      0|                }
  987|       |                // Add them as named arguments, so order does not matter.
  988|      0|                args.emplace_back(EF, id(decode_utf8(pair.first)), EF, expr, EF);
  989|      0|            }
  990|  3.47k|            const Identifier *body = id(U"top_level");
  991|  3.47k|            ast =
  992|  3.47k|                make<Local>(ast->location,
  993|  3.47k|                            line_end_blank,
  994|  3.47k|                            singleBind(body, ast),
  995|  3.47k|                            make<Conditional>(E,
  996|  3.47k|                                              line_end,
  997|  3.47k|                                              primitiveEquals(E, type(var(body)), str(U"function")),
  998|  3.47k|                                              EF,
  999|  3.47k|                                              make<Apply>(tla_loc,
 1000|  3.47k|                                                          EF,
 1001|  3.47k|                                                          make<Var>(E, line_end, body),
 1002|  3.47k|                                                          EF,
 1003|  3.47k|                                                          args,
 1004|  3.47k|                                                          false,  // trailing comma
 1005|  3.47k|                                                          EF,
 1006|  3.47k|                                                          EF,
 1007|  3.47k|                                                          false  // tailstrict
 1008|  3.47k|                                                          ),
 1009|  3.47k|                                              line_end,
 1010|  3.47k|                                              make<Var>(E, line_end, body)));
 1011|  3.47k|        }
 1012|       |
 1013|       |        // local $std = (std.jsonnet stuff); std = $std; ast
 1014|       |        // The standard library is bound to $std, which cannot be overriden,
 1015|       |        // so redefining std won't break expressions that desugar to calls
 1016|       |        // to standard library functions.
 1017|  3.88k|        ast = make<Local>(ast->location, EF, singleBind(id(U"std"), std()), ast);
 1018|  3.88k|        ast = make<Local>(ast->location, EF, singleBind(id(U"$std"), std_obj), ast);
 1019|  3.88k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISD_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  110|  3.47k|    {
  111|  3.47k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  3.47k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5ApplyERKSC_SF_SD_PNS0_3VarEEEEPT_DpOT0_:
  110|  3.47k|    {
  111|  3.47k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  3.47k|    }
_ZN7jsonnet8internal9Desugarer15primitiveEqualsERKNS0_13LocationRangeEPNS0_3ASTES6_:
  209|  3.47k|    {
  210|  3.47k|        return stdFunc(loc, U"primitiveEquals", a, b);
  211|  3.47k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_RNS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  110|  3.47k|    {
  111|  3.47k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  3.47k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_3VarEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  110|  6.94k|    {
  111|  6.94k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  6.94k|    }

_ZN7jsonnet8internal7FmtOptsC2Ev:
   37|  6.45k|        : stringStyle('s'),
   38|  6.45k|          commentStyle('s'),
   39|  6.45k|          indent(2),
   40|  6.45k|          maxBlankLines(2),
   41|  6.45k|          padArrays(false),
   42|  6.45k|          padObjects(true),
   43|  6.45k|          stripComments(false),
   44|  6.45k|          stripAllButComments(false),
   45|  6.45k|          stripEverything(false),
   46|  6.45k|          prettyFieldNames(true),
   47|  6.45k|          sortImports(true)
   48|  6.45k|    {
   49|  6.45k|    }

_ZN7jsonnet8internal26allowed_at_end_of_operatorEc:
  174|  3.80M|bool allowed_at_end_of_operator(char c) {
  175|  3.80M|    switch (c) {
  ------------------
  |  Branch (175:13): [True: 1.08M, False: 2.72M]
  ------------------
  176|   601k|        case '+':
  ------------------
  |  Branch (176:9): [True: 601k, False: 3.20M]
  ------------------
  177|   767k|        case '-':
  ------------------
  |  Branch (177:9): [True: 166k, False: 3.64M]
  ------------------
  178|  1.00M|        case '~':
  ------------------
  |  Branch (178:9): [True: 235k, False: 3.57M]
  ------------------
  179|  1.03M|        case '!':
  ------------------
  |  Branch (179:9): [True: 32.1k, False: 3.77M]
  ------------------
  180|  1.08M|        case '$': return false;
  ------------------
  |  Branch (180:9): [True: 46.4k, False: 3.76M]
  ------------------
  181|  3.80M|    }
  182|  2.72M|    return true;
  183|  3.80M|}
_ZN7jsonnet8internal20lex_get_keyword_kindERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  207|  38.2M|{
  208|  38.2M|    auto it = keywords.find(identifier);
  209|  38.2M|    if (it == keywords.end())
  ------------------
  |  Branch (209:9): [True: 27.9M, False: 10.2M]
  ------------------
  210|  27.9M|        return Token::IDENTIFIER;
  211|  10.2M|    return it->second;
  212|  38.2M|}
_ZN7jsonnet8internal10lex_numberERPKcRKNSt3__112basic_stringIcNS4_11char_traitsIcEENS4_9allocatorIcEEEERKNS0_8LocationE:
  215|  3.33M|{
  216|       |    // This function should be understood with reference to the linked image:
  217|       |    // https://www.json.org/img/number.png
  218|       |
  219|       |    // Note, we deviate from the json.org documentation as follows:
  220|       |    // * There is no reason to lex negative numbers as atomic tokens, it is better to parse them
  221|       |    //   as a unary operator combined with a numeric literal.  This avoids x-1 being tokenized as
  222|       |    //   <identifier> <number> instead of the intended <identifier> <binop> <number>.
  223|       |    // * We support digit separators using the _ character for readability in
  224|       |    //   large numeric literals.
  225|       |
  226|  3.33M|    enum State {
  227|  3.33M|        BEGIN,
  228|  3.33M|        AFTER_ZERO,
  229|  3.33M|        AFTER_ONE_TO_NINE,
  230|  3.33M|        AFTER_INT_UNDERSCORE,
  231|  3.33M|        AFTER_DOT,
  232|  3.33M|        AFTER_DIGIT,
  233|  3.33M|        AFTER_FRAC_UNDERSCORE,
  234|  3.33M|        AFTER_E,
  235|  3.33M|        AFTER_EXP_SIGN,
  236|  3.33M|        AFTER_EXP_DIGIT,
  237|  3.33M|        AFTER_EXP_UNDERSCORE
  238|  3.33M|    } state;
  239|       |
  240|  3.33M|    std::string r;
  241|       |
  242|  3.33M|    state = BEGIN;
  243|  8.73M|    while (true) {
  ------------------
  |  Branch (243:12): [True: 8.73M, Folded]
  ------------------
  244|  8.73M|        switch (state) {
  ------------------
  |  Branch (244:17): [True: 8.73M, False: 0]
  ------------------
  245|  3.33M|            case BEGIN:
  ------------------
  |  Branch (245:13): [True: 3.33M, False: 5.40M]
  ------------------
  246|  3.33M|                switch (*c) {
  247|   852k|                    case '0': state = AFTER_ZERO; break;
  ------------------
  |  Branch (247:21): [True: 852k, False: 2.48M]
  ------------------
  248|       |
  249|  1.20M|                    case '1':
  ------------------
  |  Branch (249:21): [True: 1.20M, False: 2.13M]
  ------------------
  250|  1.43M|                    case '2':
  ------------------
  |  Branch (250:21): [True: 232k, False: 3.10M]
  ------------------
  251|  1.60M|                    case '3':
  ------------------
  |  Branch (251:21): [True: 171k, False: 3.16M]
  ------------------
  252|  1.97M|                    case '4':
  ------------------
  |  Branch (252:21): [True: 371k, False: 2.96M]
  ------------------
  253|  2.04M|                    case '5':
  ------------------
  |  Branch (253:21): [True: 66.5k, False: 3.26M]
  ------------------
  254|  2.09M|                    case '6':
  ------------------
  |  Branch (254:21): [True: 55.4k, False: 3.28M]
  ------------------
  255|  2.14M|                    case '7':
  ------------------
  |  Branch (255:21): [True: 51.4k, False: 3.28M]
  ------------------
  256|  2.41M|                    case '8':
  ------------------
  |  Branch (256:21): [True: 263k, False: 3.07M]
  ------------------
  257|  2.48M|                    case '9': state = AFTER_ONE_TO_NINE; break;
  ------------------
  |  Branch (257:21): [True: 71.2k, False: 3.26M]
  ------------------
  258|       |
  259|      0|                    default: throw StaticError(filename, begin, "couldn't lex number");
  ------------------
  |  Branch (259:21): [True: 0, False: 3.33M]
  ------------------
  260|  3.33M|                }
  261|  3.33M|                break;
  262|       |
  263|  3.33M|            case AFTER_ZERO:
  ------------------
  |  Branch (263:13): [True: 852k, False: 7.88M]
  ------------------
  264|   852k|                switch (*c) {
  265|  13.3k|                    case '.': state = AFTER_DOT; break;
  ------------------
  |  Branch (265:21): [True: 13.3k, False: 839k]
  ------------------
  266|       |
  267|  1.10k|                    case 'e':
  ------------------
  |  Branch (267:21): [True: 1.10k, False: 851k]
  ------------------
  268|  1.52k|                    case 'E': state = AFTER_E; break;
  ------------------
  |  Branch (268:21): [True: 418, False: 852k]
  ------------------
  269|       |
  270|      2|                    case '_': {
  ------------------
  |  Branch (270:21): [True: 2, False: 852k]
  ------------------
  271|      2|                        std::stringstream ss;
  272|      2|                        ss << "couldn't lex number, _ not allowed after leading 0";
  273|      2|                        throw StaticError(filename, begin, ss.str());
  274|  1.10k|                    }
  275|       |
  276|   838k|                    default: goto end;
  ------------------
  |  Branch (276:21): [True: 838k, False: 14.8k]
  ------------------
  277|   852k|                }
  278|  14.8k|                break;
  279|       |
  280|  4.13M|            case AFTER_ONE_TO_NINE:
  ------------------
  |  Branch (280:13): [True: 4.13M, False: 4.60M]
  ------------------
  281|  4.13M|                switch (*c) {
  282|  8.08k|                    case '.': state = AFTER_DOT; break;
  ------------------
  |  Branch (282:21): [True: 8.08k, False: 4.12M]
  ------------------
  283|       |
  284|  1.56k|                    case 'e':
  ------------------
  |  Branch (284:21): [True: 1.56k, False: 4.13M]
  ------------------
  285|  2.24k|                    case 'E': state = AFTER_E; break;
  ------------------
  |  Branch (285:21): [True: 671, False: 4.13M]
  ------------------
  286|       |
  287|   631k|                    case '0':
  ------------------
  |  Branch (287:21): [True: 631k, False: 3.50M]
  ------------------
  288|   642k|                    case '1':
  ------------------
  |  Branch (288:21): [True: 10.9k, False: 4.12M]
  ------------------
  289|   718k|                    case '2':
  ------------------
  |  Branch (289:21): [True: 76.1k, False: 4.05M]
  ------------------
  290|   741k|                    case '3':
  ------------------
  |  Branch (290:21): [True: 23.0k, False: 4.11M]
  ------------------
  291|   781k|                    case '4':
  ------------------
  |  Branch (291:21): [True: 40.4k, False: 4.09M]
  ------------------
  292|   846k|                    case '5':
  ------------------
  |  Branch (292:21): [True: 64.8k, False: 4.06M]
  ------------------
  293|   882k|                    case '6':
  ------------------
  |  Branch (293:21): [True: 35.6k, False: 4.09M]
  ------------------
  294|   899k|                    case '7':
  ------------------
  |  Branch (294:21): [True: 17.4k, False: 4.11M]
  ------------------
  295|   918k|                    case '8':
  ------------------
  |  Branch (295:21): [True: 18.9k, False: 4.11M]
  ------------------
  296|  1.65M|                    case '9': state = AFTER_ONE_TO_NINE; break;
  ------------------
  |  Branch (296:21): [True: 731k, False: 3.40M]
  ------------------
  297|       |
  298|    272|                    case '_': state = AFTER_INT_UNDERSCORE; goto skip_char;
  ------------------
  |  Branch (298:21): [True: 272, False: 4.13M]
  ------------------
  299|       |
  300|  2.47M|                    default: goto end;
  ------------------
  |  Branch (300:21): [True: 2.47M, False: 1.66M]
  ------------------
  301|  4.13M|                }
  302|  1.66M|                break;
  303|       |
  304|  1.66M|            case AFTER_INT_UNDERSCORE:
  ------------------
  |  Branch (304:13): [True: 272, False: 8.73M]
  ------------------
  305|    272|                switch (*c) {
  306|       |                    // The only valid transition from _ is to a digit.
  307|     85|                    case '0':
  ------------------
  |  Branch (307:21): [True: 85, False: 187]
  ------------------
  308|     88|                    case '1':
  ------------------
  |  Branch (308:21): [True: 3, False: 269]
  ------------------
  309|    109|                    case '2':
  ------------------
  |  Branch (309:21): [True: 21, False: 251]
  ------------------
  310|    109|                    case '3':
  ------------------
  |  Branch (310:21): [True: 0, False: 272]
  ------------------
  311|    109|                    case '4':
  ------------------
  |  Branch (311:21): [True: 0, False: 272]
  ------------------
  312|    264|                    case '5':
  ------------------
  |  Branch (312:21): [True: 155, False: 117]
  ------------------
  313|    264|                    case '6':
  ------------------
  |  Branch (313:21): [True: 0, False: 272]
  ------------------
  314|    268|                    case '7':
  ------------------
  |  Branch (314:21): [True: 4, False: 268]
  ------------------
  315|    268|                    case '8':
  ------------------
  |  Branch (315:21): [True: 0, False: 272]
  ------------------
  316|    268|                    case '9': state = AFTER_ONE_TO_NINE; break;
  ------------------
  |  Branch (316:21): [True: 0, False: 272]
  ------------------
  317|       |
  318|      4|                    default: {
  ------------------
  |  Branch (318:21): [True: 4, False: 268]
  ------------------
  319|      4|                        std::stringstream ss;
  320|      4|                        ss << "couldn't lex number, junk after _: " << *c;
  321|      4|                        throw StaticError(filename, begin, ss.str());
  322|    268|                    }
  323|    272|                }
  324|    268|                break;
  325|       |
  326|  21.4k|            case AFTER_DOT:
  ------------------
  |  Branch (326:13): [True: 21.4k, False: 8.71M]
  ------------------
  327|  21.4k|                switch (*c) {
  328|    536|                    case '0':
  ------------------
  |  Branch (328:21): [True: 536, False: 20.9k]
  ------------------
  329|  7.95k|                    case '1':
  ------------------
  |  Branch (329:21): [True: 7.41k, False: 14.0k]
  ------------------
  330|  8.09k|                    case '2':
  ------------------
  |  Branch (330:21): [True: 140, False: 21.3k]
  ------------------
  331|  8.15k|                    case '3':
  ------------------
  |  Branch (331:21): [True: 63, False: 21.4k]
  ------------------
  332|  8.24k|                    case '4':
  ------------------
  |  Branch (332:21): [True: 95, False: 21.3k]
  ------------------
  333|  21.1k|                    case '5':
  ------------------
  |  Branch (333:21): [True: 12.8k, False: 8.60k]
  ------------------
  334|  21.3k|                    case '6':
  ------------------
  |  Branch (334:21): [True: 224, False: 21.2k]
  ------------------
  335|  21.3k|                    case '7':
  ------------------
  |  Branch (335:21): [True: 42, False: 21.4k]
  ------------------
  336|  21.4k|                    case '8':
  ------------------
  |  Branch (336:21): [True: 34, False: 21.4k]
  ------------------
  337|  21.4k|                    case '9': state = AFTER_DIGIT; break;
  ------------------
  |  Branch (337:21): [True: 53, False: 21.4k]
  ------------------
  338|       |
  339|      4|                    default: {
  ------------------
  |  Branch (339:21): [True: 4, False: 21.4k]
  ------------------
  340|      4|                        std::stringstream ss;
  341|      4|                        ss << "couldn't lex number, junk after decimal point: " << *c;
  342|      4|                        throw StaticError(filename, begin, ss.str());
  343|  21.4k|                    }
  344|  21.4k|                }
  345|  21.4k|                break;
  346|       |
  347|   215k|            case AFTER_DIGIT:
  ------------------
  |  Branch (347:13): [True: 215k, False: 8.52M]
  ------------------
  348|   215k|                switch (*c) {
  349|    315|                    case 'e':
  ------------------
  |  Branch (349:21): [True: 315, False: 215k]
  ------------------
  350|    638|                    case 'E': state = AFTER_E; break;
  ------------------
  |  Branch (350:21): [True: 323, False: 215k]
  ------------------
  351|       |
  352|  70.3k|                    case '0':
  ------------------
  |  Branch (352:21): [True: 70.3k, False: 145k]
  ------------------
  353|  92.9k|                    case '1':
  ------------------
  |  Branch (353:21): [True: 22.5k, False: 193k]
  ------------------
  354|   100k|                    case '2':
  ------------------
  |  Branch (354:21): [True: 7.89k, False: 208k]
  ------------------
  355|   115k|                    case '3':
  ------------------
  |  Branch (355:21): [True: 14.4k, False: 201k]
  ------------------
  356|   123k|                    case '4':
  ------------------
  |  Branch (356:21): [True: 8.38k, False: 207k]
  ------------------
  357|   144k|                    case '5':
  ------------------
  |  Branch (357:21): [True: 20.7k, False: 195k]
  ------------------
  358|   158k|                    case '6':
  ------------------
  |  Branch (358:21): [True: 13.6k, False: 202k]
  ------------------
  359|   166k|                    case '7':
  ------------------
  |  Branch (359:21): [True: 7.93k, False: 208k]
  ------------------
  360|   173k|                    case '8':
  ------------------
  |  Branch (360:21): [True: 7.67k, False: 208k]
  ------------------
  361|   193k|                    case '9': state = AFTER_DIGIT; break;
  ------------------
  |  Branch (361:21): [True: 19.8k, False: 196k]
  ------------------
  362|       |
  363|    887|                    case '_': state = AFTER_FRAC_UNDERSCORE; goto skip_char;
  ------------------
  |  Branch (363:21): [True: 887, False: 215k]
  ------------------
  364|       |
  365|  20.8k|                    default: goto end;
  ------------------
  |  Branch (365:21): [True: 20.8k, False: 195k]
  ------------------
  366|   215k|                }
  367|   194k|                break;
  368|       |
  369|   194k|            case AFTER_FRAC_UNDERSCORE:
  ------------------
  |  Branch (369:13): [True: 887, False: 8.73M]
  ------------------
  370|    887|                switch (*c) {
  371|       |                    // The only valid transition from _ is to a digit.
  372|    225|                    case '0':
  ------------------
  |  Branch (372:21): [True: 225, False: 662]
  ------------------
  373|    225|                    case '1':
  ------------------
  |  Branch (373:21): [True: 0, False: 887]
  ------------------
  374|    646|                    case '2':
  ------------------
  |  Branch (374:21): [True: 421, False: 466]
  ------------------
  375|    646|                    case '3':
  ------------------
  |  Branch (375:21): [True: 0, False: 887]
  ------------------
  376|    705|                    case '4':
  ------------------
  |  Branch (376:21): [True: 59, False: 828]
  ------------------
  377|    877|                    case '5':
  ------------------
  |  Branch (377:21): [True: 172, False: 715]
  ------------------
  378|    877|                    case '6':
  ------------------
  |  Branch (378:21): [True: 0, False: 887]
  ------------------
  379|    877|                    case '7':
  ------------------
  |  Branch (379:21): [True: 0, False: 887]
  ------------------
  380|    878|                    case '8':
  ------------------
  |  Branch (380:21): [True: 1, False: 886]
  ------------------
  381|    878|                    case '9': state = AFTER_DIGIT; break;
  ------------------
  |  Branch (381:21): [True: 0, False: 887]
  ------------------
  382|       |
  383|      9|                    default: {
  ------------------
  |  Branch (383:21): [True: 9, False: 878]
  ------------------
  384|      9|                        std::stringstream ss;
  385|      9|                        ss << "couldn't lex number, junk after _: " << *c;
  386|      9|                        throw StaticError(filename, begin, ss.str());
  387|    878|                    }
  388|    887|                }
  389|    878|                break;
  390|       |
  391|  4.39k|            case AFTER_E:
  ------------------
  |  Branch (391:13): [True: 4.39k, False: 8.73M]
  ------------------
  392|  4.39k|                switch (*c) {
  393|    404|                    case '+':
  ------------------
  |  Branch (393:21): [True: 404, False: 3.99k]
  ------------------
  394|  1.05k|                    case '-': state = AFTER_EXP_SIGN; break;
  ------------------
  |  Branch (394:21): [True: 647, False: 3.75k]
  ------------------
  395|       |
  396|    874|                    case '0':
  ------------------
  |  Branch (396:21): [True: 874, False: 3.52k]
  ------------------
  397|    949|                    case '1':
  ------------------
  |  Branch (397:21): [True: 75, False: 4.32k]
  ------------------
  398|  1.95k|                    case '2':
  ------------------
  |  Branch (398:21): [True: 1.00k, False: 3.39k]
  ------------------
  399|  2.75k|                    case '3':
  ------------------
  |  Branch (399:21): [True: 808, False: 3.59k]
  ------------------
  400|  2.76k|                    case '4':
  ------------------
  |  Branch (400:21): [True: 11, False: 4.38k]
  ------------------
  401|  3.06k|                    case '5':
  ------------------
  |  Branch (401:21): [True: 298, False: 4.10k]
  ------------------
  402|  3.25k|                    case '6':
  ------------------
  |  Branch (402:21): [True: 190, False: 4.20k]
  ------------------
  403|  3.27k|                    case '7':
  ------------------
  |  Branch (403:21): [True: 22, False: 4.37k]
  ------------------
  404|  3.31k|                    case '8':
  ------------------
  |  Branch (404:21): [True: 34, False: 4.36k]
  ------------------
  405|  3.32k|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (405:21): [True: 16, False: 4.38k]
  ------------------
  406|       |
  407|     18|                    default: {
  ------------------
  |  Branch (407:21): [True: 18, False: 4.38k]
  ------------------
  408|     18|                        std::stringstream ss;
  409|     18|                        ss << "couldn't lex number, junk after 'E': " << *c;
  410|     18|                        throw StaticError(filename, begin, ss.str());
  411|  3.31k|                    }
  412|  4.39k|                }
  413|  4.38k|                break;
  414|       |
  415|  4.38k|            case AFTER_EXP_SIGN:
  ------------------
  |  Branch (415:13): [True: 1.05k, False: 8.73M]
  ------------------
  416|  1.05k|                switch (*c) {
  417|     18|                    case '0':
  ------------------
  |  Branch (417:21): [True: 18, False: 1.03k]
  ------------------
  418|     23|                    case '1':
  ------------------
  |  Branch (418:21): [True: 5, False: 1.04k]
  ------------------
  419|     79|                    case '2':
  ------------------
  |  Branch (419:21): [True: 56, False: 995]
  ------------------
  420|    772|                    case '3':
  ------------------
  |  Branch (420:21): [True: 693, False: 358]
  ------------------
  421|    869|                    case '4':
  ------------------
  |  Branch (421:21): [True: 97, False: 954]
  ------------------
  422|    870|                    case '5':
  ------------------
  |  Branch (422:21): [True: 1, False: 1.05k]
  ------------------
  423|    942|                    case '6':
  ------------------
  |  Branch (423:21): [True: 72, False: 979]
  ------------------
  424|    972|                    case '7':
  ------------------
  |  Branch (424:21): [True: 30, False: 1.02k]
  ------------------
  425|  1.03k|                    case '8':
  ------------------
  |  Branch (425:21): [True: 62, False: 989]
  ------------------
  426|  1.04k|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (426:21): [True: 11, False: 1.04k]
  ------------------
  427|       |
  428|      6|                    default: {
  ------------------
  |  Branch (428:21): [True: 6, False: 1.04k]
  ------------------
  429|      6|                        std::stringstream ss;
  430|      6|                        ss << "couldn't lex number, junk after exponent sign: " << *c;
  431|      6|                        throw StaticError(filename, begin, ss.str());
  432|  1.03k|                    }
  433|  1.05k|                }
  434|  1.04k|                break;
  435|       |
  436|   171k|            case AFTER_EXP_DIGIT:
  ------------------
  |  Branch (436:13): [True: 171k, False: 8.56M]
  ------------------
  437|   171k|                switch (*c) {
  438|   159k|                    case '0':
  ------------------
  |  Branch (438:21): [True: 159k, False: 11.9k]
  ------------------
  439|   160k|                    case '1':
  ------------------
  |  Branch (439:21): [True: 635, False: 170k]
  ------------------
  440|   160k|                    case '2':
  ------------------
  |  Branch (440:21): [True: 822, False: 170k]
  ------------------
  441|   161k|                    case '3':
  ------------------
  |  Branch (441:21): [True: 584, False: 170k]
  ------------------
  442|   162k|                    case '4':
  ------------------
  |  Branch (442:21): [True: 1.35k, False: 169k]
  ------------------
  443|   163k|                    case '5':
  ------------------
  |  Branch (443:21): [True: 278, False: 171k]
  ------------------
  444|   163k|                    case '6':
  ------------------
  |  Branch (444:21): [True: 339, False: 170k]
  ------------------
  445|   164k|                    case '7':
  ------------------
  |  Branch (445:21): [True: 1.05k, False: 170k]
  ------------------
  446|   165k|                    case '8':
  ------------------
  |  Branch (446:21): [True: 1.42k, False: 169k]
  ------------------
  447|   166k|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (447:21): [True: 865, False: 170k]
  ------------------
  448|       |
  449|    197|                    case '_': state = AFTER_EXP_UNDERSCORE; goto skip_char;
  ------------------
  |  Branch (449:21): [True: 197, False: 171k]
  ------------------
  450|       |
  451|  4.37k|                    default: goto end;
  ------------------
  |  Branch (451:21): [True: 4.37k, False: 166k]
  ------------------
  452|   171k|                }
  453|   166k|                break;
  454|       |
  455|   166k|            case AFTER_EXP_UNDERSCORE:
  ------------------
  |  Branch (455:13): [True: 197, False: 8.73M]
  ------------------
  456|    197|                switch (*c) {
  457|       |                    // The only valid transition from _ is to a digit.
  458|    191|                    case '0':
  ------------------
  |  Branch (458:21): [True: 191, False: 6]
  ------------------
  459|    191|                    case '1':
  ------------------
  |  Branch (459:21): [True: 0, False: 197]
  ------------------
  460|    191|                    case '2':
  ------------------
  |  Branch (460:21): [True: 0, False: 197]
  ------------------
  461|    191|                    case '3':
  ------------------
  |  Branch (461:21): [True: 0, False: 197]
  ------------------
  462|    191|                    case '4':
  ------------------
  |  Branch (462:21): [True: 0, False: 197]
  ------------------
  463|    194|                    case '5':
  ------------------
  |  Branch (463:21): [True: 3, False: 194]
  ------------------
  464|    194|                    case '6':
  ------------------
  |  Branch (464:21): [True: 0, False: 197]
  ------------------
  465|    194|                    case '7':
  ------------------
  |  Branch (465:21): [True: 0, False: 197]
  ------------------
  466|    195|                    case '8':
  ------------------
  |  Branch (466:21): [True: 1, False: 196]
  ------------------
  467|    195|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (467:21): [True: 0, False: 197]
  ------------------
  468|       |
  469|      2|                    default: {
  ------------------
  |  Branch (469:21): [True: 2, False: 195]
  ------------------
  470|      2|                        std::stringstream ss;
  471|      2|                        ss << "couldn't lex number, junk after _: " << *c;
  472|      2|                        throw StaticError(filename, begin, ss.str());
  473|    195|                    }
  474|    197|                }
  475|    195|                break;
  476|  8.73M|        }
  477|  5.40M|        r += *c;
  478|       |
  479|  5.40M|skip_char:
  480|  5.40M|        c++;
  481|  5.40M|    }
  482|  3.33M|end:
  483|  3.33M|    return r;
  484|  3.33M|}
_ZN7jsonnet8internal11jsonnet_lexERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEEPKc:
  521|  12.8k|{
  522|  12.8k|    unsigned long line_number = 1;
  523|  12.8k|    const char *line_start = input;
  524|       |
  525|  12.8k|    Tokens r;
  526|       |
  527|  12.8k|    const char *c = input;
  528|       |
  529|  12.8k|    Fodder fodder;
  530|  12.8k|    bool fresh_line = true;  // Are we tokenizing from the beginning of a new line?
  531|       |
  532|  89.2M|    while (*c != '\0') {
  ------------------
  |  Branch (532:12): [True: 89.2M, False: 5.77k]
  ------------------
  533|       |        // Used to ensure we have actually advanced the pointer by the end of the iteration.
  534|  89.2M|        const char *original_c = c;
  535|       |
  536|  89.2M|        Token::Kind kind;
  537|  89.2M|        std::string data;
  538|  89.2M|        std::string string_block_indent;
  539|  89.2M|        std::string string_block_term_indent;
  540|       |
  541|  89.2M|        unsigned new_lines, indent;
  542|  89.2M|        lex_ws(c, new_lines, indent, line_start, line_number);
  543|       |
  544|       |        // If it's the end of the file, discard final whitespace.
  545|  89.2M|        if (*c == '\0')
  ------------------
  |  Branch (545:13): [True: 6.52k, False: 89.2M]
  ------------------
  546|  6.52k|            break;
  547|       |
  548|  89.2M|        if (new_lines > 0) {
  ------------------
  |  Branch (548:13): [True: 9.60M, False: 79.6M]
  ------------------
  549|       |            // Otherwise store whitespace in fodder.
  550|  9.60M|            unsigned blanks = new_lines - 1;
  551|  9.60M|            fodder.emplace_back(FodderElement::LINE_END, blanks, indent, EMPTY);
  552|  9.60M|            fresh_line = true;
  553|  9.60M|        }
  554|       |
  555|  89.2M|        Location begin(line_number, c - line_start + 1);
  556|       |
  557|  89.2M|        switch (*c) {
  558|       |            // The following operators should never be combined with subsequent symbols.
  559|   382k|            case '{':
  ------------------
  |  Branch (559:13): [True: 382k, False: 88.8M]
  ------------------
  560|   382k|                kind = Token::BRACE_L;
  561|   382k|                c++;
  562|   382k|                break;
  563|       |
  564|   375k|            case '}':
  ------------------
  |  Branch (564:13): [True: 375k, False: 88.8M]
  ------------------
  565|   375k|                kind = Token::BRACE_R;
  566|   375k|                c++;
  567|   375k|                break;
  568|       |
  569|  2.00M|            case '[':
  ------------------
  |  Branch (569:13): [True: 2.00M, False: 87.2M]
  ------------------
  570|  2.00M|                kind = Token::BRACKET_L;
  571|  2.00M|                c++;
  572|  2.00M|                break;
  573|       |
  574|  1.99M|            case ']':
  ------------------
  |  Branch (574:13): [True: 1.99M, False: 87.2M]
  ------------------
  575|  1.99M|                kind = Token::BRACKET_R;
  576|  1.99M|                c++;
  577|  1.99M|                break;
  578|       |
  579|  7.36M|            case ',':
  ------------------
  |  Branch (579:13): [True: 7.36M, False: 81.8M]
  ------------------
  580|  7.36M|                kind = Token::COMMA;
  581|  7.36M|                c++;
  582|  7.36M|                break;
  583|       |
  584|  4.56M|            case '.':
  ------------------
  |  Branch (584:13): [True: 4.56M, False: 84.6M]
  ------------------
  585|  4.56M|                kind = Token::DOT;
  586|  4.56M|                c++;
  587|  4.56M|                break;
  588|       |
  589|  6.88M|            case '(':
  ------------------
  |  Branch (589:13): [True: 6.88M, False: 82.3M]
  ------------------
  590|  6.88M|                kind = Token::PAREN_L;
  591|  6.88M|                c++;
  592|  6.88M|                break;
  593|       |
  594|  6.87M|            case ')':
  ------------------
  |  Branch (594:13): [True: 6.87M, False: 82.3M]
  ------------------
  595|  6.87M|                kind = Token::PAREN_R;
  596|  6.87M|                c++;
  597|  6.87M|                break;
  598|       |
  599|  1.80M|            case ';':
  ------------------
  |  Branch (599:13): [True: 1.80M, False: 87.4M]
  ------------------
  600|  1.80M|                kind = Token::SEMICOLON;
  601|  1.80M|                c++;
  602|  1.80M|                break;
  603|       |
  604|       |            // Numeric literals.
  605|   852k|            case '0':
  ------------------
  |  Branch (605:13): [True: 852k, False: 88.3M]
  ------------------
  606|  2.05M|            case '1':
  ------------------
  |  Branch (606:13): [True: 1.20M, False: 88.0M]
  ------------------
  607|  2.28M|            case '2':
  ------------------
  |  Branch (607:13): [True: 232k, False: 89.0M]
  ------------------
  608|  2.45M|            case '3':
  ------------------
  |  Branch (608:13): [True: 171k, False: 89.0M]
  ------------------
  609|  2.82M|            case '4':
  ------------------
  |  Branch (609:13): [True: 371k, False: 88.8M]
  ------------------
  610|  2.89M|            case '5':
  ------------------
  |  Branch (610:13): [True: 66.5k, False: 89.1M]
  ------------------
  611|  2.94M|            case '6':
  ------------------
  |  Branch (611:13): [True: 55.4k, False: 89.1M]
  ------------------
  612|  3.00M|            case '7':
  ------------------
  |  Branch (612:13): [True: 51.4k, False: 89.1M]
  ------------------
  613|  3.26M|            case '8':
  ------------------
  |  Branch (613:13): [True: 263k, False: 88.9M]
  ------------------
  614|  3.33M|            case '9':
  ------------------
  |  Branch (614:13): [True: 71.2k, False: 89.1M]
  ------------------
  615|  3.33M|                kind = Token::NUMBER;
  616|  3.33M|                data = lex_number(c, filename, begin);
  617|  3.33M|                break;
  618|       |
  619|       |            // UString literals.
  620|  77.8k|            case '"': {
  ------------------
  |  Branch (620:13): [True: 77.8k, False: 89.1M]
  ------------------
  621|  77.8k|                c++;
  622|  17.6M|                for (;; ++c) {
  623|  17.6M|                    if (*c == '\0') {
  ------------------
  |  Branch (623:25): [True: 27, False: 17.6M]
  ------------------
  624|     27|                        throw StaticError(filename, begin, "unterminated string");
  625|     27|                    }
  626|  17.6M|                    if (*c == '"') {
  ------------------
  |  Branch (626:25): [True: 77.8k, False: 17.5M]
  ------------------
  627|  77.8k|                        break;
  628|  77.8k|                    }
  629|  17.5M|                    if (*c == '\\' && *(c + 1) != '\0') {
  ------------------
  |  Branch (629:25): [True: 75.7k, False: 17.4M]
  |  Branch (629:39): [True: 75.7k, False: 2]
  ------------------
  630|  75.7k|                        data += *c;
  631|  75.7k|                        ++c;
  632|  75.7k|                    }
  633|  17.5M|                    if (*c == '\n') {
  ------------------
  |  Branch (633:25): [True: 2.33M, False: 15.2M]
  ------------------
  634|       |                        // Maintain line/column counters.
  635|  2.33M|                        line_number++;
  636|  2.33M|                        line_start = c + 1;
  637|  2.33M|                    }
  638|  17.5M|                    data += *c;
  639|  17.5M|                }
  640|  77.8k|                c++;  // Advance beyond the ".
  641|  77.8k|                kind = Token::STRING_DOUBLE;
  642|  77.8k|            } break;
  643|       |
  644|       |            // UString literals.
  645|  3.01M|            case '\'': {
  ------------------
  |  Branch (645:13): [True: 3.01M, False: 86.2M]
  ------------------
  646|  3.01M|                c++;
  647|  39.3M|                for (;; ++c) {
  648|  39.3M|                    if (*c == '\0') {
  ------------------
  |  Branch (648:25): [True: 23, False: 39.3M]
  ------------------
  649|     23|                        throw StaticError(filename, begin, "unterminated string");
  650|     23|                    }
  651|  39.3M|                    if (*c == '\'') {
  ------------------
  |  Branch (651:25): [True: 3.01M, False: 36.3M]
  ------------------
  652|  3.01M|                        break;
  653|  3.01M|                    }
  654|  36.3M|                    if (*c == '\\' && *(c + 1) != '\0') {
  ------------------
  |  Branch (654:25): [True: 306k, False: 36.0M]
  |  Branch (654:39): [True: 306k, False: 1]
  ------------------
  655|   306k|                        data += *c;
  656|   306k|                        ++c;
  657|   306k|                    }
  658|  36.3M|                    if (*c == '\n') {
  ------------------
  |  Branch (658:25): [True: 273k, False: 36.0M]
  ------------------
  659|       |                        // Maintain line/column counters.
  660|   273k|                        line_number++;
  661|   273k|                        line_start = c + 1;
  662|   273k|                    }
  663|  36.3M|                    data += *c;
  664|  36.3M|                }
  665|  3.01M|                c++;  // Advance beyond the '.
  666|  3.01M|                kind = Token::STRING_SINGLE;
  667|  3.01M|            } break;
  668|       |
  669|       |            // Verbatim string literals.
  670|       |            // ' and " quoting is interpreted here, unlike non-verbatim strings
  671|       |            // where it is done later by jsonnet_string_unescape.  This is OK
  672|       |            // in this case because no information is lost by resoving the
  673|       |            // repeated quote into a single quote, so we can go back to the
  674|       |            // original form in the formatter.
  675|  6.01k|            case '@': {
  ------------------
  |  Branch (675:13): [True: 6.01k, False: 89.2M]
  ------------------
  676|  6.01k|                c++;
  677|  6.01k|                if (*c != '"' && *c != '\'') {
  ------------------
  |  Branch (677:21): [True: 1.77k, False: 4.24k]
  |  Branch (677:34): [True: 16, False: 1.75k]
  ------------------
  678|     16|                    std::stringstream ss;
  679|     16|                    ss << "couldn't lex verbatim string, junk after '@': " << *c;
  680|     16|                    throw StaticError(filename, begin, ss.str());
  681|     16|                }
  682|  5.99k|                const char quot = *c;
  683|  5.99k|                c++;  // Advance beyond the opening quote.
  684|   144k|                for (;; ++c) {
  685|   144k|                    if (*c == '\0') {
  ------------------
  |  Branch (685:25): [True: 20, False: 144k]
  ------------------
  686|     20|                        throw StaticError(filename, begin, "unterminated verbatim string");
  687|     20|                    }
  688|   144k|                    if (*c == quot) {
  ------------------
  |  Branch (688:25): [True: 7.08k, False: 137k]
  ------------------
  689|  7.08k|                        if (*(c + 1) == quot) {
  ------------------
  |  Branch (689:29): [True: 1.11k, False: 5.97k]
  ------------------
  690|  1.11k|                            c++;
  691|  5.97k|                        } else {
  692|  5.97k|                            break;
  693|  5.97k|                        }
  694|  7.08k|                    }
  695|   138k|                    data += *c;
  696|   138k|                }
  697|  5.97k|                c++;  // Advance beyond the closing quote.
  698|  5.97k|                if (quot == '"') {
  ------------------
  |  Branch (698:21): [True: 4.23k, False: 1.74k]
  ------------------
  699|  4.23k|                    kind = Token::VERBATIM_STRING_DOUBLE;
  700|  4.23k|                } else {
  701|  1.74k|                    kind = Token::VERBATIM_STRING_SINGLE;
  702|  1.74k|                }
  703|  5.97k|            } break;
  704|       |
  705|       |            // Keywords
  706|  50.5M|            default:
  ------------------
  |  Branch (706:13): [True: 50.5M, False: 38.6M]
  ------------------
  707|  50.5M|                if (is_identifier_first(*c)) {
  ------------------
  |  Branch (707:21): [True: 38.2M, False: 12.3M]
  ------------------
  708|  38.2M|                    std::string id;
  709|   204M|                    for (; is_identifier(*c); ++c)
  ------------------
  |  Branch (709:28): [True: 166M, False: 38.2M]
  ------------------
  710|   166M|                        id += *c;
  711|  38.2M|                    kind = lex_get_keyword_kind(id);
  712|  38.2M|                    data = id;
  713|       |
  714|  38.2M|                } else if (is_symbol(*c) || *c == '#') {
  ------------------
  |  Branch (714:28): [True: 10.6M, False: 1.72M]
  |  Branch (714:45): [True: 1.72M, False: 103]
  ------------------
  715|       |                    // Single line C++ and Python style comments.
  716|  12.3M|                    if (*c == '#' || (*c == '/' && *(c + 1) == '/')) {
  ------------------
  |  Branch (716:25): [True: 1.72M, False: 10.6M]
  |  Branch (716:39): [True: 1.04M, False: 9.58M]
  |  Branch (716:52): [True: 592k, False: 447k]
  ------------------
  717|  2.32M|                        std::vector<std::string> comment(1);
  718|  2.32M|                        unsigned blanks;
  719|  2.32M|                        unsigned indent;
  720|  2.32M|                        lex_until_newline(c, comment[0], blanks, indent, line_start, line_number);
  721|  2.32M|                        auto kind = fresh_line ? FodderElement::PARAGRAPH : FodderElement::LINE_END;
  ------------------
  |  Branch (721:37): [True: 2.29M, False: 25.5k]
  ------------------
  722|  2.32M|                        fodder.emplace_back(kind, blanks, indent, comment);
  723|  2.32M|                        fresh_line = true;
  724|  2.32M|                        continue;  // We've not got a token, just fodder, so keep scanning.
  725|  2.32M|                    }
  726|       |
  727|       |                    // Multi-line C style comment.
  728|  10.0M|                    if (*c == '/' && *(c + 1) == '*') {
  ------------------
  |  Branch (728:25): [True: 447k, False: 9.58M]
  |  Branch (728:38): [True: 363k, False: 83.9k]
  ------------------
  729|   363k|                        unsigned margin = c - line_start;
  730|       |
  731|   363k|                        const char *initial_c = c;
  732|   363k|                        c += 2;  // Avoid matching /*/: skip the /* before starting the search for
  733|       |                                 // */.
  734|       |
  735|  34.3M|                        while (!(*c == '*' && *(c + 1) == '/')) {
  ------------------
  |  Branch (735:34): [True: 762k, False: 33.6M]
  |  Branch (735:47): [True: 363k, False: 398k]
  ------------------
  736|  34.0M|                            if (*c == '\0') {
  ------------------
  |  Branch (736:33): [True: 77, False: 34.0M]
  ------------------
  737|     77|                                auto msg = "multi-line comment has no terminating */.";
  738|     77|                                throw StaticError(filename, begin, msg);
  739|     77|                            }
  740|  34.0M|                            if (*c == '\n') {
  ------------------
  |  Branch (740:33): [True: 4.09M, False: 29.9M]
  ------------------
  741|       |                                // Just keep track of the line / column counters.
  742|  4.09M|                                line_number++;
  743|  4.09M|                                line_start = c + 1;
  744|  4.09M|                            }
  745|  34.0M|                            ++c;
  746|  34.0M|                        }
  747|   363k|                        c += 2;  // Move the pointer to the char after the closing '/'.
  748|       |
  749|   363k|                        std::string comment(initial_c,
  750|   363k|                                            c - initial_c);  // Includes the "/*" and "*/".
  751|       |
  752|       |                        // Lex whitespace after comment
  753|   363k|                        unsigned new_lines_after, indent_after;
  754|   363k|                        lex_ws(c, new_lines_after, indent_after, line_start, line_number);
  755|   363k|                        std::vector<std::string> lines;
  756|   363k|                        if (comment.find('\n') >= comment.length()) {
  ------------------
  |  Branch (756:29): [True: 267k, False: 96.5k]
  ------------------
  757|       |                            // Comment looks like /* foo */
  758|   267k|                            lines.push_back(comment);
  759|   267k|                            fodder.emplace_back(FodderElement::INTERSTITIAL, 0, 0, lines);
  760|   267k|                            if (new_lines_after > 0) {
  ------------------
  |  Branch (760:33): [True: 258k, False: 8.66k]
  ------------------
  761|   258k|                                fodder.emplace_back(FodderElement::LINE_END,
  762|   258k|                                                    new_lines_after - 1,
  763|   258k|                                                    indent_after,
  764|   258k|                                                    EMPTY);
  765|   258k|                                fresh_line = true;
  766|   258k|                            }
  767|   267k|                        } else {
  768|  96.5k|                            lines = line_split(comment, margin);
  769|  96.5k|                            assert(lines[0][0] == '/');
  ------------------
  |  Branch (769:29): [True: 96.5k, False: 0]
  ------------------
  770|       |                            // Little hack to support PARAGRAPHs with * down the LHS:
  771|       |                            // Add a space to lines that start with a '*'
  772|  96.5k|                            bool all_star = true;
  773|  4.19M|                            for (auto &l : lines) {
  ------------------
  |  Branch (773:42): [True: 4.19M, False: 96.5k]
  ------------------
  774|  4.19M|                                if (l[0] != '*')
  ------------------
  |  Branch (774:37): [True: 4.13M, False: 55.3k]
  ------------------
  775|  4.13M|                                    all_star = false;
  776|  4.19M|                            }
  777|  96.5k|                            if (all_star) {
  ------------------
  |  Branch (777:33): [True: 0, False: 96.5k]
  ------------------
  778|      0|                                for (auto &l : lines) {
  ------------------
  |  Branch (778:46): [True: 0, False: 0]
  ------------------
  779|      0|                                    if (l[0] == '*')
  ------------------
  |  Branch (779:41): [True: 0, False: 0]
  ------------------
  780|      0|                                        l = " " + l;
  781|      0|                                }
  782|      0|                            }
  783|  96.5k|                            if (new_lines_after == 0) {
  ------------------
  |  Branch (783:33): [True: 11.2k, False: 85.2k]
  ------------------
  784|       |                                // Ensure a line end after the paragraph.
  785|  11.2k|                                new_lines_after = 1;
  786|  11.2k|                                indent_after = 0;
  787|  11.2k|                            }
  788|  96.5k|                            fodder_push_back(fodder,
  789|  96.5k|                                             FodderElement(FodderElement::PARAGRAPH,
  790|  96.5k|                                                           new_lines_after - 1,
  791|  96.5k|                                                           indent_after,
  792|  96.5k|                                                           lines));
  793|  96.5k|                            fresh_line = true;
  794|  96.5k|                        }
  795|   363k|                        continue;  // We've not got a token, just fodder, so keep scanning.
  796|   363k|                    }
  797|       |
  798|       |                    // Text block
  799|  9.67M|                    if (*c == '|' && *(c + 1) == '|' && *(c + 2) == '|') {
  ------------------
  |  Branch (799:25): [True: 178k, False: 9.49M]
  |  Branch (799:38): [True: 138k, False: 40.1k]
  |  Branch (799:57): [True: 5.00k, False: 133k]
  ------------------
  800|  5.00k|                        c += 3;  // Skip the "|||".
  801|       |
  802|  5.00k|                        bool chomp_trailing_nl = false;
  803|  5.00k|                        if (*c == '-') {
  ------------------
  |  Branch (803:29): [True: 497, False: 4.51k]
  ------------------
  804|    497|                            chomp_trailing_nl = true;
  805|    497|                            c++;
  806|    497|                        }
  807|       |
  808|  6.44k|                        while (is_horz_ws(*c)) ++c;  // Chomp whitespace at end of line.
  ------------------
  |  Branch (808:32): [True: 1.44k, False: 5.00k]
  ------------------
  809|  5.00k|                        if (*c != '\n') {
  ------------------
  |  Branch (809:29): [True: 31, False: 4.97k]
  ------------------
  810|     31|                            auto msg = "text block syntax requires new line after |||.";
  811|     31|                            throw StaticError(filename, begin, msg);
  812|     31|                        }
  813|  4.97k|                        std::stringstream block;
  814|  4.97k|                        c++;  // Skip the "\n"
  815|  4.97k|                        line_number++;
  816|       |                        // Skip any blank lines at the beginning of the block.
  817|  6.18k|                        while (*c == '\n') {
  ------------------
  |  Branch (817:32): [True: 1.20k, False: 4.97k]
  ------------------
  818|  1.20k|                            line_number++;
  819|  1.20k|                            ++c;
  820|  1.20k|                            block << '\n';
  821|  1.20k|                        }
  822|  4.97k|                        line_start = c;
  823|  4.97k|                        const char *first_line = c;
  824|  4.97k|                        int ws_chars = whitespace_check(first_line, c);
  825|  4.97k|                        string_block_indent = std::string(first_line, ws_chars);
  826|  4.97k|                        if (ws_chars == 0) {
  ------------------
  |  Branch (826:29): [True: 20, False: 4.95k]
  ------------------
  827|     20|                            auto msg = "text block's first line must start with whitespace.";
  828|     20|                            throw StaticError(filename, begin, msg);
  829|     20|                        }
  830|  7.04k|                        while (true) {
  ------------------
  |  Branch (830:32): [True: 7.04k, Folded]
  ------------------
  831|  7.04k|                            assert(ws_chars > 0);
  ------------------
  |  Branch (831:29): [True: 7.04k, False: 0]
  ------------------
  832|       |                            // Read up to the \n
  833|  8.96M|                            for (c = &c[ws_chars]; *c != '\n'; ++c) {
  ------------------
  |  Branch (833:52): [True: 8.95M, False: 6.97k]
  ------------------
  834|  8.95M|                                if (*c == '\0')
  ------------------
  |  Branch (834:37): [True: 69, False: 8.95M]
  ------------------
  835|     69|                                    throw StaticError(filename, begin, "unexpected EOF");
  836|  8.95M|                                block << *c;
  837|  8.95M|                            }
  838|       |                            // Add the \n
  839|  6.97k|                            block << '\n';
  840|  6.97k|                            ++c;
  841|  6.97k|                            line_number++;
  842|  6.97k|                            line_start = c;
  843|       |                            // Skip any blank lines
  844|  8.44k|                            while (*c == '\n') {
  ------------------
  |  Branch (844:36): [True: 1.47k, False: 6.97k]
  ------------------
  845|  1.47k|                                line_number++;
  846|  1.47k|                                ++c;
  847|  1.47k|                                block << '\n';
  848|  1.47k|                            }
  849|       |                            // Examine next line
  850|  6.97k|                            ws_chars = whitespace_check(first_line, c);
  851|  6.97k|                            if (ws_chars == 0) {
  ------------------
  |  Branch (851:33): [True: 4.88k, False: 2.08k]
  ------------------
  852|       |                                // End of text block (or indentation error).
  853|       |                                // Count actual whitespace on this line.
  854|  4.88k|                                int actual_ws = 0;
  855|   117k|                                while (c[actual_ws] == ' ' ||
  ------------------
  |  Branch (855:40): [True: 10.0k, False: 107k]
  ------------------
  856|   113k|                                       c[actual_ws] == '\t') {
  ------------------
  |  Branch (856:40): [True: 102k, False: 4.88k]
  ------------------
  857|   113k|                                    actual_ws++;
  858|   113k|                                }
  859|       |
  860|       |                                // Check if this is the terminator |||
  861|  4.88k|                                bool is_terminator = (
  862|  4.88k|                                    c[actual_ws] == '|' &&
  ------------------
  |  Branch (862:37): [True: 4.83k, False: 58]
  ------------------
  863|  4.83k|                                    c[actual_ws + 1] == '|' &&
  ------------------
  |  Branch (863:37): [True: 4.82k, False: 8]
  ------------------
  864|  4.82k|                                    c[actual_ws + 2] == '|');
  ------------------
  |  Branch (864:37): [True: 4.81k, False: 4]
  ------------------
  865|       |
  866|  4.88k|                                if (!is_terminator) {
  ------------------
  |  Branch (866:37): [True: 70, False: 4.81k]
  ------------------
  867|       |                                    // Not a terminator - check if it's an
  868|       |                                    // indentation issue.
  869|     70|                                    if (actual_ws > 0) {
  ------------------
  |  Branch (869:41): [True: 38, False: 32]
  ------------------
  870|       |                                        // Has whitespace but doesn't match expected
  871|       |                                        // indentation.
  872|     38|                                        std::stringstream msg;
  873|     38|                                        msg << "text block indentation mismatch: "
  874|     38|                                                "expected at least ";
  875|     38|                                        describe_whitespace(msg, string_block_indent);
  876|     38|                                        msg << ", found ";
  877|     38|                                        describe_whitespace(msg, std::string(c, actual_ws));
  878|     38|                                        throw StaticError(filename, begin, msg.str());
  879|     38|                                    } else {
  880|       |                                        // No whitespace and no ||| - missing
  881|       |                                        // terminator.
  882|     32|                                        auto msg =
  883|     32|                                            "text block not terminated with |||";
  884|     32|                                        throw StaticError(filename, begin, msg);
  885|     32|                                    }
  886|     70|                                }
  887|       |
  888|       |                                // Valid termination - skip over any whitespace.
  889|  27.4k|                                while (*c == ' ' || *c == '\t') {
  ------------------
  |  Branch (889:40): [True: 1.28k, False: 26.1k]
  |  Branch (889:53): [True: 21.3k, False: 4.81k]
  ------------------
  890|  22.6k|                                    string_block_term_indent += *c;
  891|  22.6k|                                    ++c;
  892|  22.6k|                                }
  893|       |                                // Skip the |||
  894|  4.81k|                                c += 3;  // Leave after the last |
  895|  4.81k|                                data = block.str();
  896|  4.81k|                                kind = Token::STRING_BLOCK;
  897|  4.81k|                                if (chomp_trailing_nl) {
  ------------------
  |  Branch (897:37): [True: 489, False: 4.32k]
  ------------------
  898|    489|                                    assert(data.back() == '\n');
  ------------------
  |  Branch (898:37): [True: 489, False: 0]
  ------------------
  899|    489|                                    data.pop_back();
  900|    489|                                }
  901|  4.81k|                                break;  // Out of the while loop.
  902|  4.81k|                            }
  903|  6.97k|                        }
  904|       |
  905|  4.81k|                        break;  // Out of the switch.
  906|  4.95k|                    }
  907|       |
  908|  9.66M|                    const char *operator_begin = c;
  909|  28.4M|                    for (; is_symbol(*c); ++c) {
  ------------------
  |  Branch (909:28): [True: 18.7M, False: 9.66M]
  ------------------
  910|       |                        // Not allowed // in operators
  911|  18.7M|                        if (*c == '/' && *(c + 1) == '/')
  ------------------
  |  Branch (911:29): [True: 85.5k, False: 18.6M]
  |  Branch (911:42): [True: 352, False: 85.1k]
  ------------------
  912|    352|                            break;
  913|       |                        // Not allowed /* in operators
  914|  18.7M|                        if (*c == '/' && *(c + 1) == '*')
  ------------------
  |  Branch (914:29): [True: 85.1k, False: 18.6M]
  |  Branch (914:42): [True: 723, False: 84.4k]
  ------------------
  915|    723|                            break;
  916|       |                        // Not allowed ||| in operators
  917|  18.7M|                        if (*c == '|' && *(c + 1) == '|' && *(c + 2) == '|')
  ------------------
  |  Branch (917:29): [True: 308k, False: 18.4M]
  |  Branch (917:42): [True: 134k, False: 173k]
  |  Branch (917:61): [True: 353, False: 133k]
  ------------------
  918|    353|                            break;
  919|  18.7M|                    }
  920|       |                    // Not allowed to end with a + - ~ ! unless a single char.
  921|       |                    // So, wind it back if we need to (but not too far).
  922|  10.7M|                    while (c > operator_begin + 1 && !allowed_at_end_of_operator(*(c - 1))) {
  ------------------
  |  Branch (922:28): [True: 3.80M, False: 6.93M]
  |  Branch (922:54): [True: 1.08M, False: 2.72M]
  ------------------
  923|  1.08M|                        c--;
  924|  1.08M|                    }
  925|  9.66M|                    data += std::string(operator_begin, c);
  926|  9.66M|                    if (data == "$") {
  ------------------
  |  Branch (926:25): [True: 38.4k, False: 9.62M]
  ------------------
  927|  38.4k|                        kind = Token::DOLLAR;
  928|  38.4k|                        data = "";
  929|  9.62M|                    } else {
  930|  9.62M|                        kind = Token::OPERATOR;
  931|  9.62M|                    }
  932|  9.66M|                } else {
  933|    103|                    std::stringstream ss;
  934|    103|                    ss << "Could not lex the character ";
  935|    103|                    auto uc = (unsigned char)(*c);
  936|    103|                    if (*c < 32)
  ------------------
  |  Branch (936:25): [True: 97, False: 6]
  ------------------
  937|     97|                        ss << "code " << unsigned(uc);
  938|      6|                    else
  939|      6|                        ss << "'" << *c << "'";
  940|    103|                    throw StaticError(filename, begin, ss.str());
  941|    103|                }
  942|  89.2M|        }
  943|       |
  944|       |        // Ensure that a bug in the above code does not cause an infinite memory consuming loop due
  945|       |        // to pushing empty tokens.
  946|  86.5M|        if (c == original_c) {
  ------------------
  |  Branch (946:13): [True: 0, False: 86.5M]
  ------------------
  947|      0|            throw StaticError(filename, begin, "internal lexing error:  pointer did not advance");
  948|      0|        }
  949|       |
  950|  86.5M|        Location end(line_number, (c + 1) - line_start);
  951|  86.5M|        r.emplace_back(kind,
  952|  86.5M|                       fodder,
  953|  86.5M|                       data,
  954|  86.5M|                       string_block_indent,
  955|  86.5M|                       string_block_term_indent,
  956|  86.5M|                       LocationRange(filename, begin, end));
  957|  86.5M|        fodder.clear();
  958|  86.5M|        fresh_line = false;
  959|  86.5M|    }
  960|       |
  961|  12.3k|    Location begin(line_number, c - line_start + 1);
  962|  12.3k|    Location end(line_number, (c + 1) - line_start + 1);
  963|  12.3k|    r.emplace_back(Token::END_OF_FILE, fodder, "", "", "", LocationRange(filename, begin, end));
  964|  12.3k|    return r;
  965|  12.8k|}
lexer.cpp:_ZN7jsonnet8internalL6lex_wsERPKcRjS4_S3_Rm:
   81|  91.9M|{
   82|  91.9M|    indent = 0;
   83|  91.9M|    new_lines = 0;
   84|   214M|    for (; *c != '\0' && is_ws(*c); c++) {
  ------------------
  |  Branch (84:12): [True: 214M, False: 6.92k]
  |  Branch (84:26): [True: 122M, False: 91.9M]
  ------------------
   85|   122M|        switch (*c) {
  ------------------
  |  Branch (85:17): [True: 122M, False: 0]
  ------------------
   86|   467k|            case '\r':
  ------------------
  |  Branch (86:13): [True: 467k, False: 121M]
  ------------------
   87|       |                // Ignore.
   88|   467k|                break;
   89|       |
   90|  13.3M|            case '\n':
  ------------------
  |  Branch (90:13): [True: 13.3M, False: 108M]
  ------------------
   91|  13.3M|                indent = 0;
   92|  13.3M|                new_lines++;
   93|  13.3M|                line_number++;
   94|  13.3M|                line_start = c + 1;
   95|  13.3M|                break;
   96|       |
   97|   108M|            case ' ': indent += 1; break;
  ------------------
  |  Branch (97:13): [True: 108M, False: 13.7M]
  ------------------
   98|       |
   99|       |            // This only works for \t at the beginning of lines, but we strip it everywhere else
  100|       |            // anyway.  The only case where this will cause a problem is spaces followed by \t
  101|       |            // at the beginning of a line.  However that is rare, ill-advised, and if re-indentation
  102|       |            // is enabled it will be fixed later.
  103|  7.08k|            case '\t': indent += 8; break;
  ------------------
  |  Branch (103:13): [True: 7.08k, False: 122M]
  ------------------
  104|   122M|        }
  105|   122M|    }
  106|  91.9M|}
lexer.cpp:_ZN7jsonnet8internalL5is_wsEc:
   39|   214M|{
   40|   214M|    return c == '\n' || is_horz_ws(c);
  ------------------
  |  Branch (40:12): [True: 13.3M, False: 200M]
  |  Branch (40:25): [True: 108M, False: 91.9M]
  ------------------
   41|   214M|}
lexer.cpp:_ZN7jsonnet8internalL19is_identifier_firstEc:
  143|   255M|{
  144|   255M|    return is_upper(c) || is_lower(c) || c == '_';
  ------------------
  |  Branch (144:12): [True: 3.62M, False: 251M]
  |  Branch (144:27): [True: 194M, False: 56.8M]
  |  Branch (144:42): [True: 2.77M, False: 54.0M]
  ------------------
  145|   255M|}
lexer.cpp:_ZN7jsonnet8internalL8is_upperEc:
  128|   255M|{
  129|   255M|    return c >= 'A' && c <= 'Z';
  ------------------
  |  Branch (129:12): [True: 203M, False: 51.7M]
  |  Branch (129:24): [True: 3.62M, False: 199M]
  ------------------
  130|   255M|}
lexer.cpp:_ZN7jsonnet8internalL8is_lowerEc:
  133|   251M|{
  134|   251M|    return c >= 'a' && c <= 'z';
  ------------------
  |  Branch (134:12): [True: 194M, False: 56.6M]
  |  Branch (134:24): [True: 194M, False: 201k]
  ------------------
  135|   251M|}
lexer.cpp:_ZN7jsonnet8internalL13is_identifierEc:
  148|   204M|{
  149|   204M|    return is_identifier_first(c) || is_number(c);
  ------------------
  |  Branch (149:12): [True: 162M, False: 41.7M]
  |  Branch (149:38): [True: 3.49M, False: 38.2M]
  ------------------
  150|   204M|}
lexer.cpp:_ZN7jsonnet8internalL9is_numberEc:
  138|  41.7M|{
  139|  41.7M|    return c >= '0' && c <= '9';
  ------------------
  |  Branch (139:12): [True: 6.84M, False: 34.8M]
  |  Branch (139:24): [True: 3.49M, False: 3.34M]
  ------------------
  140|  41.7M|}
lexer.cpp:_ZN7jsonnet8internalL9is_symbolEc:
  153|  40.7M|{
  154|  40.7M|    switch (c) {
  ------------------
  |  Branch (154:13): [True: 29.3M, False: 11.3M]
  ------------------
  155|   798k|        case '!':
  ------------------
  |  Branch (155:9): [True: 798k, False: 39.9M]
  ------------------
  156|   921k|        case '$':
  ------------------
  |  Branch (156:9): [True: 123k, False: 40.6M]
  ------------------
  157|  5.51M|        case ':':
  ------------------
  |  Branch (157:9): [True: 4.59M, False: 36.1M]
  ------------------
  158|  5.77M|        case '~':
  ------------------
  |  Branch (158:9): [True: 260k, False: 40.5M]
  ------------------
  159|  11.2M|        case '+':
  ------------------
  |  Branch (159:9): [True: 5.51M, False: 35.2M]
  ------------------
  160|  12.1M|        case '-':
  ------------------
  |  Branch (160:9): [True: 839k, False: 39.9M]
  ------------------
  161|  12.9M|        case '&':
  ------------------
  |  Branch (161:9): [True: 789k, False: 40.0M]
  ------------------
  162|  13.4M|        case '|':
  ------------------
  |  Branch (162:9): [True: 487k, False: 40.3M]
  ------------------
  163|  13.4M|        case '^':
  ------------------
  |  Branch (163:9): [True: 1.36k, False: 40.7M]
  ------------------
  164|  23.0M|        case '=':
  ------------------
  |  Branch (164:9): [True: 9.64M, False: 31.1M]
  ------------------
  165|  23.8M|        case '<':
  ------------------
  |  Branch (165:9): [True: 806k, False: 39.9M]
  ------------------
  166|  24.8M|        case '>':
  ------------------
  |  Branch (166:9): [True: 1.03M, False: 39.7M]
  ------------------
  167|  27.8M|        case '*':
  ------------------
  |  Branch (167:9): [True: 2.98M, False: 37.8M]
  ------------------
  168|  29.0M|        case '/':
  ------------------
  |  Branch (168:9): [True: 1.12M, False: 39.6M]
  ------------------
  169|  29.3M|        case '%': return true;
  ------------------
  |  Branch (169:9): [True: 391k, False: 40.3M]
  ------------------
  170|  40.7M|    }
  171|  11.3M|    return false;
  172|  40.7M|}
lexer.cpp:_ZN7jsonnet8internalL17lex_until_newlineERPKcRNSt3__112basic_stringIcNS4_11char_traitsIcEENS4_9allocatorIcEEEERjSC_S3_Rm:
  113|  2.32M|{
  114|  2.32M|    const char *original_c = c;
  115|  2.32M|    const char *last_non_space = c;
  116|  35.2M|    for (; *c != '\0' && *c != '\n'; c++) {
  ------------------
  |  Branch (116:12): [True: 35.2M, False: 182]
  |  Branch (116:26): [True: 32.9M, False: 2.31M]
  ------------------
  117|  32.9M|        if (!is_horz_ws(*c))
  ------------------
  |  Branch (117:13): [True: 28.2M, False: 4.73M]
  ------------------
  118|  28.2M|            last_non_space = c;
  119|  32.9M|    }
  120|  2.32M|    text = std::string(original_c, last_non_space - original_c + 1);
  121|       |    // Consume subsequent whitespace including the '\n'.
  122|  2.32M|    unsigned new_lines;
  123|  2.32M|    lex_ws(c, new_lines, indent, line_start, line_number);
  124|  2.32M|    blanks = new_lines == 0 ? 0 : new_lines - 1;
  ------------------
  |  Branch (124:14): [True: 182, False: 2.31M]
  ------------------
  125|  2.32M|}
lexer.cpp:_ZN7jsonnet8internalL10line_splitERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEEj:
   60|  96.5k|{
   61|  96.5k|    std::vector<std::string> ret;
   62|  96.5k|    std::stringstream ss;
   63|  31.8M|    for (size_t i = 0; i < s.length(); ++i) {
  ------------------
  |  Branch (63:24): [True: 31.7M, False: 96.5k]
  ------------------
   64|  31.7M|        if (s[i] == '\n') {
  ------------------
  |  Branch (64:13): [True: 4.09M, False: 27.6M]
  ------------------
   65|  4.09M|            ret.emplace_back(strip_ws(ss.str(), margin));
   66|  4.09M|            ss.str("");
   67|  27.6M|        } else {
   68|  27.6M|            ss << s[i];
   69|  27.6M|        }
   70|  31.7M|    }
   71|  96.5k|    ret.emplace_back(strip_ws(ss.str(), margin));
   72|  96.5k|    return ret;
   73|  96.5k|}
lexer.cpp:_ZN7jsonnet8internalL8strip_wsERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEEj:
   45|  4.19M|{
   46|  4.19M|    if (s.size() == 0)
  ------------------
  |  Branch (46:9): [True: 3.65M, False: 535k]
  ------------------
   47|  3.65M|        return s;  // Avoid underflow below.
   48|   535k|    size_t i = 0;
   49|  2.06M|    while (i < s.length() && is_horz_ws(s[i]) && i < margin)
  ------------------
  |  Branch (49:12): [True: 2.06M, False: 509]
  |  Branch (49:30): [True: 1.76M, False: 304k]
  |  Branch (49:50): [True: 1.53M, False: 230k]
  ------------------
   50|  1.53M|        i++;
   51|   535k|    size_t j = s.size();
   52|   554k|    while (j > i && is_horz_ws(s[j - 1])) {
  ------------------
  |  Branch (52:12): [True: 553k, False: 1.45k]
  |  Branch (52:21): [True: 19.1k, False: 534k]
  ------------------
   53|  19.1k|        j--;
   54|  19.1k|    }
   55|   535k|    return std::string(&s[i], &s[j]);
   56|  4.19M|}
lexer.cpp:_ZN7jsonnet8internalL10is_horz_wsEc:
   33|   236M|{
   34|   236M|    return c == ' ' || c == '\t' || c == '\r';
  ------------------
  |  Branch (34:12): [True: 114M, False: 121M]
  |  Branch (34:24): [True: 16.3k, False: 121M]
  |  Branch (34:37): [True: 476k, False: 120M]
  ------------------
   35|   236M|}
lexer.cpp:_ZN7jsonnet8internalL16whitespace_checkEPKcS2_:
  489|  11.9k|{
  490|  11.9k|    int i = 0;
  491|   786k|    while (a[i] == ' ' || a[i] == '\t') {
  ------------------
  |  Branch (491:12): [True: 64.2k, False: 722k]
  |  Branch (491:27): [True: 715k, False: 7.06k]
  ------------------
  492|   779k|        if (b[i] != a[i])
  ------------------
  |  Branch (492:13): [True: 4.88k, False: 774k]
  ------------------
  493|  4.88k|            return 0;
  494|   774k|        i++;
  495|   774k|    }
  496|  7.06k|    return i;
  497|  11.9k|}
lexer.cpp:_ZN7jsonnet8internalL19describe_whitespaceERNSt3__118basic_stringstreamIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKNS1_12basic_stringIcS4_S6_EE:
  499|     76|static void describe_whitespace(std::stringstream& msg, const std::string& ws) {
  500|     76|    int spaces = 0;
  501|     76|    int tabs = 0;
  502|   437k|    for (char c : ws) {
  ------------------
  |  Branch (502:17): [True: 437k, False: 76]
  ------------------
  503|   437k|        if (c == ' ')
  ------------------
  |  Branch (503:13): [True: 33.8k, False: 403k]
  ------------------
  504|  33.8k|            spaces++;
  505|   403k|        else if (c == '\t')
  ------------------
  |  Branch (505:18): [True: 403k, False: 0]
  ------------------
  506|   403k|            tabs++;
  507|   437k|    }
  508|     76|    if (spaces > 0 && tabs > 0) {
  ------------------
  |  Branch (508:9): [True: 55, False: 21]
  |  Branch (508:23): [True: 35, False: 20]
  ------------------
  509|     35|        msg << spaces << (spaces == 1 ? " space" : " spaces") << " and " << tabs
  ------------------
  |  Branch (509:27): [True: 10, False: 25]
  ------------------
  510|     35|            << (tabs == 1 ? " tab" : " tabs");
  ------------------
  |  Branch (510:17): [True: 2, False: 33]
  ------------------
  511|     41|    } else if (spaces > 0) {
  ------------------
  |  Branch (511:16): [True: 20, False: 21]
  ------------------
  512|     20|        msg << spaces << (spaces == 1 ? " space" : " spaces");
  ------------------
  |  Branch (512:27): [True: 9, False: 11]
  ------------------
  513|     21|    } else if (tabs > 0) {
  ------------------
  |  Branch (513:16): [True: 21, False: 0]
  ------------------
  514|     21|        msg << tabs << (tabs == 1 ? " tab" : " tabs");
  ------------------
  |  Branch (514:25): [True: 5, False: 16]
  ------------------
  515|     21|    } else {
  516|      0|        msg << "no indentation";
  517|      0|    }
  518|     76|}

_ZN7jsonnet8internal13FodderElementC2ENS1_4KindEjjRKNSt3__16vectorINS3_12basic_stringIcNS3_11char_traitsIcEENS3_9allocatorIcEEEENS8_ISA_EEEE:
   92|  12.5M|        : kind(kind), blanks(blanks), indent(indent), comment(comment)
   93|  12.5M|    {
   94|  12.5M|        assert(kind != LINE_END || comment.size() <= 1);
  ------------------
  |  Branch (94:9): [True: 2.65M, False: 9.91M]
  |  Branch (94:9): [True: 9.91M, False: 0]
  |  Branch (94:9): [True: 12.5M, False: 0]
  ------------------
   95|  12.5M|        assert(kind != INTERSTITIAL || (blanks == 0 && indent == 0 && comment.size() == 1));
  ------------------
  |  Branch (95:9): [True: 267k, False: 0]
  |  Branch (95:9): [True: 267k, False: 0]
  |  Branch (95:9): [True: 267k, False: 0]
  |  Branch (95:9): [True: 12.3M, False: 267k]
  |  Branch (95:9): [True: 12.5M, False: 0]
  ------------------
   96|  12.5M|        assert(kind != PARAGRAPH || comment.size() >= 1);
  ------------------
  |  Branch (96:9): [True: 10.1M, False: 2.39M]
  |  Branch (96:9): [True: 2.39M, False: 0]
  |  Branch (96:9): [True: 12.5M, False: 0]
  ------------------
   97|  12.5M|    }
_ZNK7jsonnet8internal5Token6data32Ev:
  299|  31.0M|    {
  300|  31.0M|        return decode_utf8(data);
  301|  31.0M|    }
_ZN7jsonnet8internal5TokenC2ENS1_4KindERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEERKNS3_12basic_stringIcNS3_11char_traitsIcEENS6_IcEEEESH_SH_RKNS0_13LocationRangeE:
  308|  86.5M|        : kind(kind),
  309|  86.5M|          fodder(fodder),
  310|  86.5M|          data(data),
  311|  86.5M|          stringBlockIndent(string_block_indent),
  312|  86.5M|          stringBlockTermIndent(string_block_term_indent),
  313|  86.5M|          location(location)
  314|  86.5M|    {
  315|  86.5M|    }
_ZN7jsonnet8internal5Token8toStringENS1_4KindE:
  320|    948|    {
  321|    948|        switch (v) {
  322|     16|            case BRACE_L: return "\"{\"";
  ------------------
  |  Branch (322:13): [True: 16, False: 932]
  ------------------
  323|     22|            case BRACE_R: return "\"}\"";
  ------------------
  |  Branch (323:13): [True: 22, False: 926]
  ------------------
  324|      1|            case BRACKET_L: return "\"[\"";
  ------------------
  |  Branch (324:13): [True: 1, False: 947]
  ------------------
  325|     62|            case BRACKET_R: return "\"]\"";
  ------------------
  |  Branch (325:13): [True: 62, False: 886]
  ------------------
  326|     39|            case COMMA: return "\",\"";
  ------------------
  |  Branch (326:13): [True: 39, False: 909]
  ------------------
  327|     27|            case DOLLAR: return "\"$\"";
  ------------------
  |  Branch (327:13): [True: 27, False: 921]
  ------------------
  328|     13|            case DOT: return "\".\"";
  ------------------
  |  Branch (328:13): [True: 13, False: 935]
  ------------------
  329|       |
  330|      0|            case PAREN_L: return "\"(\"";
  ------------------
  |  Branch (330:13): [True: 0, False: 948]
  ------------------
  331|     46|            case PAREN_R: return "\")\"";
  ------------------
  |  Branch (331:13): [True: 46, False: 902]
  ------------------
  332|     24|            case SEMICOLON: return "\";\"";
  ------------------
  |  Branch (332:13): [True: 24, False: 924]
  ------------------
  333|       |
  334|    134|            case IDENTIFIER: return "IDENTIFIER";
  ------------------
  |  Branch (334:13): [True: 134, False: 814]
  ------------------
  335|     43|            case NUMBER: return "NUMBER";
  ------------------
  |  Branch (335:13): [True: 43, False: 905]
  ------------------
  336|    135|            case OPERATOR: return "OPERATOR";
  ------------------
  |  Branch (336:13): [True: 135, False: 813]
  ------------------
  337|     11|            case STRING_SINGLE: return "STRING_SINGLE";
  ------------------
  |  Branch (337:13): [True: 11, False: 937]
  ------------------
  338|     23|            case STRING_DOUBLE: return "STRING_DOUBLE";
  ------------------
  |  Branch (338:13): [True: 23, False: 925]
  ------------------
  339|      1|            case VERBATIM_STRING_SINGLE: return "VERBATIM_STRING_SINGLE";
  ------------------
  |  Branch (339:13): [True: 1, False: 947]
  ------------------
  340|      3|            case VERBATIM_STRING_DOUBLE: return "VERBATIM_STRING_DOUBLE";
  ------------------
  |  Branch (340:13): [True: 3, False: 945]
  ------------------
  341|      3|            case STRING_BLOCK: return "STRING_BLOCK";
  ------------------
  |  Branch (341:13): [True: 3, False: 945]
  ------------------
  342|       |
  343|      0|            case ASSERT: return "assert";
  ------------------
  |  Branch (343:13): [True: 0, False: 948]
  ------------------
  344|      1|            case ELSE: return "else";
  ------------------
  |  Branch (344:13): [True: 1, False: 947]
  ------------------
  345|      0|            case ERROR: return "error";
  ------------------
  |  Branch (345:13): [True: 0, False: 948]
  ------------------
  346|      0|            case FALSE: return "false";
  ------------------
  |  Branch (346:13): [True: 0, False: 948]
  ------------------
  347|      1|            case FOR: return "for";
  ------------------
  |  Branch (347:13): [True: 1, False: 947]
  ------------------
  348|      0|            case FUNCTION: return "function";
  ------------------
  |  Branch (348:13): [True: 0, False: 948]
  ------------------
  349|      3|            case IF: return "if";
  ------------------
  |  Branch (349:13): [True: 3, False: 945]
  ------------------
  350|      0|            case IMPORT: return "import";
  ------------------
  |  Branch (350:13): [True: 0, False: 948]
  ------------------
  351|      1|            case IMPORTSTR: return "importstr";
  ------------------
  |  Branch (351:13): [True: 1, False: 947]
  ------------------
  352|      0|            case IMPORTBIN: return "importbin";
  ------------------
  |  Branch (352:13): [True: 0, False: 948]
  ------------------
  353|     17|            case IN: return "in";
  ------------------
  |  Branch (353:13): [True: 17, False: 931]
  ------------------
  354|      1|            case LOCAL: return "local";
  ------------------
  |  Branch (354:13): [True: 1, False: 947]
  ------------------
  355|      1|            case NULL_LIT: return "null";
  ------------------
  |  Branch (355:13): [True: 1, False: 947]
  ------------------
  356|      0|            case SELF: return "self";
  ------------------
  |  Branch (356:13): [True: 0, False: 948]
  ------------------
  357|      0|            case SUPER: return "super";
  ------------------
  |  Branch (357:13): [True: 0, False: 948]
  ------------------
  358|      2|            case TAILSTRICT: return "tailstrict";
  ------------------
  |  Branch (358:13): [True: 2, False: 946]
  ------------------
  359|      9|            case THEN: return "then";
  ------------------
  |  Branch (359:13): [True: 9, False: 939]
  ------------------
  360|      0|            case TRUE: return "true";
  ------------------
  |  Branch (360:13): [True: 0, False: 948]
  ------------------
  361|       |
  362|    309|            case END_OF_FILE: return "end of file";
  ------------------
  |  Branch (362:13): [True: 309, False: 639]
  ------------------
  363|      0|            default:
  ------------------
  |  Branch (363:13): [True: 0, False: 948]
  ------------------
  364|      0|                std::cerr << "INTERNAL ERROR: Unknown token kind: " << v << std::endl;
  365|      0|                std::abort();
  366|    948|        }
  367|    948|    }
parser.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEENS0_5Token4KindE:
  387|    528|{
  388|    528|    o << Token::toString(v);
  389|    528|    return o;
  390|    528|}
parser.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_5TokenE:
  393|    439|{
  394|    439|    if (v.data == "") {
  ------------------
  |  Branch (394:9): [True: 267, False: 172]
  ------------------
  395|    267|        o << Token::toString(v.kind);
  396|    267|    } else if (v.kind == Token::OPERATOR) {
  ------------------
  |  Branch (396:16): [True: 19, False: 153]
  ------------------
  397|     19|        o << "\"" << v.data << "\"";
  398|    153|    } else {
  399|    153|        o << "(" << Token::toString(v.kind) << ", \"" << v.data << "\")";
  400|    153|    }
  401|    439|    return o;
  402|    439|}
lexer.cpp:_ZN7jsonnet8internalL16fodder_push_backERNSt3__16vectorINS0_13FodderElementENS1_9allocatorIS3_EEEERKS3_:
  143|  96.5k|{
  144|  96.5k|    if (fodder_has_clean_endline(a) && elem.kind == FodderElement::LINE_END) {
  ------------------
  |  Branch (144:9): [True: 79.9k, False: 16.5k]
  |  Branch (144:40): [True: 0, False: 79.9k]
  ------------------
  145|      0|        if (elem.comment.size() > 0) {
  ------------------
  |  Branch (145:13): [True: 0, False: 0]
  ------------------
  146|       |            // The line end had a comment, so create a single line paragraph for it.
  147|      0|            a.emplace_back(FodderElement::PARAGRAPH, elem.blanks, elem.indent, elem.comment);
  148|      0|        } else {
  149|       |            // Merge it into the previous line end.
  150|      0|            a.back().indent = elem.indent;
  151|      0|            a.back().blanks += elem.blanks;
  152|      0|        }
  153|  96.5k|    } else {
  154|  96.5k|        if (!fodder_has_clean_endline(a) && elem.kind == FodderElement::PARAGRAPH) {
  ------------------
  |  Branch (154:13): [True: 16.5k, False: 79.9k]
  |  Branch (154:45): [True: 16.5k, False: 0]
  ------------------
  155|  16.5k|            a.emplace_back(FodderElement::LINE_END, 0, elem.indent, std::vector<std::string>());
  156|  16.5k|        }
  157|  96.5k|        a.push_back(elem);
  158|  96.5k|    }
  159|  96.5k|}
lexer.cpp:_ZN7jsonnet8internalL24fodder_has_clean_endlineERKNSt3__16vectorINS0_13FodderElementENS1_9allocatorIS3_EEEE:
  134|   193k|{
  135|   193k|    return !fodder.empty() && fodder.back().kind != FodderElement::INTERSTITIAL;
  ------------------
  |  Branch (135:12): [True: 174k, False: 18.3k]
  |  Branch (135:31): [True: 159k, False: 14.7k]
  ------------------
  136|   193k|}

jsonnet_version:
  334|  12.9k|{
  335|  12.9k|    return LIB_JSONNET_VERSION;
  ------------------
  |  |   34|  12.9k|#define LIB_JSONNET_VERSION "v0.22.0"
  ------------------
  336|  12.9k|}
jsonnet_make:
  339|  6.45k|{
  340|  6.45k|    TRY
  ------------------
  |  |  319|  6.45k|#define TRY try {
  ------------------
  341|  6.45k|        return new JsonnetVm();
  342|  6.45k|    CATCH("jsonnet_make")
  ------------------
  |  |  321|  6.45k|    }                                                                                         \
  |  |  322|  6.45k|    catch (const std::bad_alloc &)                                                            \
  |  |  323|  6.45k|    {                                                                                         \
  |  |  324|      0|        memory_panic();                                                                       \
  |  |  325|      0|    }                                                                                         \
  |  |  326|  6.45k|    catch (const std::exception &e)                                                           \
  |  |  327|  6.45k|    {                                                                                         \
  |  |  328|      0|        std::cerr << "Something went wrong during " func ", please report this: " << e.what() \
  |  |  329|      0|                  << std::endl;                                                               \
  |  |  330|      0|        abort();                                                                              \
  |  |  331|      0|    }
  ------------------
  343|      0|    return nullptr;
  344|  6.45k|}
jsonnet_destroy:
  347|  6.45k|{
  348|  6.45k|    TRY
  ------------------
  |  |  319|  6.45k|#define TRY try {
  ------------------
  349|  6.45k|        delete vm;
  350|  6.45k|    CATCH("jsonnet_destroy")
  ------------------
  |  |  321|  6.45k|    }                                                                                         \
  |  |  322|  6.45k|    catch (const std::bad_alloc &)                                                            \
  |  |  323|  6.45k|    {                                                                                         \
  |  |  324|      0|        memory_panic();                                                                       \
  |  |  325|      0|    }                                                                                         \
  |  |  326|  6.45k|    catch (const std::exception &e)                                                           \
  |  |  327|  6.45k|    {                                                                                         \
  |  |  328|      0|        std::cerr << "Something went wrong during " func ", please report this: " << e.what() \
  |  |  329|      0|                  << std::endl;                                                               \
  |  |  330|      0|        abort();                                                                              \
  |  |  331|      0|    }
  ------------------
  351|  6.45k|}
jsonnet_import_callback:
  379|  6.45k|{
  380|  6.45k|    vm->importCallback = cb;
  381|  6.45k|    vm->importCallbackContext = ctx;
  382|  6.45k|}
jsonnet_evaluate_snippet:
  729|  6.45k|{
  730|  6.45k|    TRY
  ------------------
  |  |  319|  6.45k|#define TRY try {
  ------------------
  731|  6.45k|        return jsonnet_evaluate_snippet_aux(vm, filename, snippet, error, REGULAR);
  732|  6.45k|    CATCH("jsonnet_evaluate_snippet")
  ------------------
  |  |  321|  6.45k|    }                                                                                         \
  |  |  322|  6.45k|    catch (const std::bad_alloc &)                                                            \
  |  |  323|  6.45k|    {                                                                                         \
  |  |  324|      0|        memory_panic();                                                                       \
  |  |  325|      0|    }                                                                                         \
  |  |  326|  6.45k|    catch (const std::exception &e)                                                           \
  |  |  327|  6.45k|    {                                                                                         \
  |  |  328|      0|        std::cerr << "Something went wrong during " func ", please report this: " << e.what() \
  |  |  329|      0|                  << std::endl;                                                               \
  |  |  330|      0|        abort();                                                                              \
  |  |  331|      0|    }
  ------------------
  733|      0|    return nullptr;  // Never happens.
  734|  6.45k|}
jsonnet_realloc:
  755|  12.9k|{
  756|  12.9k|    (void)vm;
  757|  12.9k|    if (str == nullptr) {
  ------------------
  |  Branch (757:9): [True: 6.45k, False: 6.45k]
  ------------------
  758|  6.45k|        if (sz == 0)
  ------------------
  |  Branch (758:13): [True: 0, False: 6.45k]
  ------------------
  759|      0|            return nullptr;
  760|  6.45k|        auto *r = static_cast<char *>(::malloc(sz));
  761|  6.45k|        if (r == nullptr)
  ------------------
  |  Branch (761:13): [True: 0, False: 6.45k]
  ------------------
  762|      0|            memory_panic();
  763|  6.45k|        return r;
  764|  6.45k|    } else {
  765|  6.45k|        if (sz == 0) {
  ------------------
  |  Branch (765:13): [True: 6.45k, False: 0]
  ------------------
  766|  6.45k|            ::free(str);
  767|  6.45k|            return nullptr;
  768|  6.45k|        } else {
  769|      0|            auto *r = static_cast<char *>(::realloc(str, sz));
  770|      0|            if (r == nullptr)
  ------------------
  |  Branch (770:17): [True: 0, False: 0]
  ------------------
  771|      0|                memory_panic();
  772|      0|            return r;
  773|      0|        }
  774|  6.45k|    }
  775|  12.9k|}
_ZN9JsonnetVmC2Ev:
  228|  6.45k|        : gcGrowthTrigger(2.0),
  229|  6.45k|          maxStack(500),
  230|  6.45k|          gcMinObjects(1000),
  231|  6.45k|          maxTrace(20),
  232|  6.45k|          importCallback(default_import_callback),
  233|  6.45k|          importCallbackContext(this),
  234|  6.45k|          stringOutput(false),
  235|  6.45k|          trailingNewline(true),
  236|  6.45k|          fmtDebugDesugaring(false)
  237|  6.45k|    {
  238|  6.45k|        jpaths.emplace_back("/usr/share/jsonnet-" + std::string(jsonnet_version()) + "/");
  239|  6.45k|        jpaths.emplace_back("/usr/local/share/jsonnet-" + std::string(jsonnet_version()) + "/");
  240|  6.45k|    }
libjsonnet.cpp:_ZL11from_stringP9JsonnetVmRKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   90|  6.45k|{
   91|  6.45k|    char *r = jsonnet_realloc(vm, nullptr, v.length() + 1);
   92|  6.45k|    std::memcpy(r, v.c_str(), v.length() + 1);
   93|  6.45k|    return r;
   94|  6.45k|}
libjsonnet.cpp:_ZL28jsonnet_evaluate_snippet_auxP9JsonnetVmPKcS2_PiN12_GLOBAL__N_18EvalKindE:
  541|  6.45k|{
  542|  6.45k|    try {
  543|  6.45k|        Allocator alloc;
  544|  6.45k|        AST *expr;
  545|  6.45k|        Tokens tokens = jsonnet_lex(filename, snippet);
  546|       |
  547|  6.45k|        expr = jsonnet_parse(&alloc, tokens);
  548|       |
  549|  6.45k|        jsonnet_desugar(&alloc, expr, &vm->tla);
  550|       |
  551|  6.45k|        unsigned max_stack = vm->maxStack;
  552|       |
  553|       |        // For the stdlib desugaring.
  554|  6.45k|        max_stack++;
  555|       |
  556|       |        // For the TLA desugaring.
  557|  6.45k|        max_stack++;
  558|       |
  559|  6.45k|        jsonnet_static_analysis(expr);
  560|  6.45k|        switch (kind) {
  561|  2.88k|            case REGULAR: {
  ------------------
  |  Branch (561:13): [True: 2.88k, False: 3.56k]
  ------------------
  562|  2.88k|                std::string json_str = jsonnet_vm_execute(&alloc,
  563|  2.88k|                                                          expr,
  564|  2.88k|                                                          vm->ext,
  565|  2.88k|                                                          max_stack,
  566|  2.88k|                                                          vm->gcMinObjects,
  567|  2.88k|                                                          vm->gcGrowthTrigger,
  568|  2.88k|                                                          vm->nativeCallbacks,
  569|  2.88k|                                                          vm->importCallback,
  570|  2.88k|                                                          vm->importCallbackContext,
  571|  2.88k|                                                          vm->stringOutput);
  572|  2.88k|                if (vm->trailingNewline) {
  ------------------
  |  Branch (572:21): [True: 1.17k, False: 1.71k]
  ------------------
  573|  1.17k|                    json_str += "\n";
  574|  1.17k|                }
  575|  2.88k|                *error = false;
  576|  2.88k|                return from_string(vm, json_str);
  577|      0|            } break;
  578|       |
  579|      0|            case MULTI: {
  ------------------
  |  Branch (579:13): [True: 0, False: 6.45k]
  ------------------
  580|      0|                std::map<std::string, std::string> files =
  581|      0|                    jsonnet_vm_execute_multi(&alloc,
  582|      0|                                             expr,
  583|      0|                                             vm->ext,
  584|      0|                                             max_stack,
  585|      0|                                             vm->gcMinObjects,
  586|      0|                                             vm->gcGrowthTrigger,
  587|      0|                                             vm->nativeCallbacks,
  588|      0|                                             vm->importCallback,
  589|      0|                                             vm->importCallbackContext,
  590|      0|                                             vm->stringOutput);
  591|      0|                size_t sz = 1;  // final sentinel
  592|      0|                for (const auto &pair : files) {
  ------------------
  |  Branch (592:39): [True: 0, False: 0]
  ------------------
  593|      0|                    sz += pair.first.length() + 1;   // include sentinel
  594|      0|                    sz += pair.second.length() + 2;  // Add a '\n' as well as sentinel
  595|      0|                }
  596|      0|                char *buf = (char *)::malloc(sz);
  597|      0|                if (buf == nullptr)
  ------------------
  |  Branch (597:21): [True: 0, False: 0]
  ------------------
  598|      0|                    memory_panic();
  599|      0|                std::ptrdiff_t i = 0;
  600|      0|                for (const auto &pair : files) {
  ------------------
  |  Branch (600:39): [True: 0, False: 0]
  ------------------
  601|      0|                    memcpy(&buf[i], pair.first.c_str(), pair.first.length() + 1);
  602|      0|                    i += pair.first.length() + 1;
  603|      0|                    memcpy(&buf[i], pair.second.c_str(), pair.second.length());
  604|      0|                    i += pair.second.length();
  605|      0|                    if (vm->trailingNewline) {
  ------------------
  |  Branch (605:25): [True: 0, False: 0]
  ------------------
  606|      0|                        buf[i] = '\n';
  607|      0|                        i++;
  608|      0|                    }
  609|      0|                    buf[i] = '\0';
  610|      0|                    i++;
  611|      0|                }
  612|      0|                buf[i] = '\0';  // final sentinel
  613|      0|                *error = false;
  614|      0|                return buf;
  615|      0|            } break;
  616|       |
  617|      0|            case STREAM: {
  ------------------
  |  Branch (617:13): [True: 0, False: 6.45k]
  ------------------
  618|      0|                if (!vm->trailingNewline) {
  ------------------
  |  Branch (618:21): [True: 0, False: 0]
  ------------------
  619|      0|                    *error = true;
  620|      0|                    return from_string(vm, "INTERNAL: trailing-newline is required for streamed output");
  621|      0|                }
  622|      0|                std::vector<std::string> documents =
  623|      0|                    jsonnet_vm_execute_stream(&alloc,
  624|      0|                                              expr,
  625|      0|                                              vm->ext,
  626|      0|                                              max_stack,
  627|      0|                                              vm->gcMinObjects,
  628|      0|                                              vm->gcGrowthTrigger,
  629|      0|                                              vm->nativeCallbacks,
  630|      0|                                              vm->importCallback,
  631|      0|                                              vm->importCallbackContext,
  632|      0|                                              vm->stringOutput);
  633|      0|                size_t sz = 1;  // final sentinel
  634|      0|                for (const auto &doc : documents) {
  ------------------
  |  Branch (634:38): [True: 0, False: 0]
  ------------------
  635|      0|                    sz += doc.length() + 2;  // Add a '\n' as well as sentinel
  636|      0|                }
  637|      0|                char *buf = (char *)::malloc(sz);
  638|      0|                if (buf == nullptr)
  ------------------
  |  Branch (638:21): [True: 0, False: 0]
  ------------------
  639|      0|                    memory_panic();
  640|      0|                std::ptrdiff_t i = 0;
  641|      0|                for (const auto &doc : documents) {
  ------------------
  |  Branch (641:38): [True: 0, False: 0]
  ------------------
  642|      0|                    memcpy(&buf[i], doc.c_str(), doc.length());
  643|      0|                    i += doc.length();
  644|      0|                    buf[i] = '\n';
  645|      0|                    i++;
  646|      0|                    buf[i] = '\0';
  647|      0|                    i++;
  648|      0|                }
  649|      0|                buf[i] = '\0';  // final sentinel
  650|      0|                *error = false;
  651|      0|                return buf;
  652|      0|            } break;
  653|       |
  654|      0|            default:
  ------------------
  |  Branch (654:13): [True: 0, False: 6.45k]
  ------------------
  655|      0|                fputs("INTERNAL ERROR: bad value of 'kind', probably memory corruption.\n", stderr);
  656|      0|                abort();
  657|  6.45k|        }
  658|       |
  659|  6.45k|    } catch (StaticError &e) {
  660|  3.56k|        std::stringstream ss;
  661|  3.56k|        ss << "STATIC ERROR: " << e << std::endl;
  662|  3.56k|        *error = true;
  663|  3.56k|        return from_string(vm, ss.str());
  664|       |
  665|  3.56k|    } catch (RuntimeError &e) {
  666|  1.71k|        std::stringstream ss;
  667|  1.71k|        ss << "RUNTIME ERROR: " << e.msg << std::endl;
  668|  1.71k|        const long max_above = vm->maxTrace / 2;
  669|  1.71k|        const long max_below = vm->maxTrace - max_above;
  670|  1.71k|        const long sz = e.stackTrace.size();
  671|   243k|        for (long i = 0; i < sz; ++i) {
  ------------------
  |  Branch (671:26): [True: 242k, False: 1.71k]
  ------------------
  672|   242k|            const auto &f = e.stackTrace[i];
  673|   242k|            if (vm->maxTrace > 0 && i >= max_above && i < sz - max_below) {
  ------------------
  |  Branch (673:17): [True: 242k, False: 0]
  |  Branch (673:37): [True: 232k, False: 10.0k]
  |  Branch (673:55): [True: 225k, False: 6.88k]
  ------------------
  674|   225k|                if (i == max_above)
  ------------------
  |  Branch (674:21): [True: 609, False: 224k]
  ------------------
  675|    609|                    ss << "\t..." << std::endl;
  676|   225k|            } else {
  677|  16.9k|                ss << "\t" << f.location << "\t" << f.name << std::endl;
  678|  16.9k|            }
  679|   242k|        }
  680|  1.71k|        *error = true;
  681|  1.71k|        return from_string(vm, ss.str());
  682|  1.71k|    }
  683|       |
  684|      0|    return nullptr;  // Quiet, compiler.
  685|  6.45k|}

_ZN7jsonnet8internal22jsonnet_unparse_numberEd:
   38|  1.85M|{
   39|  1.85M|    std::stringstream ss;
   40|       |    // Make sure we output the same thing, even if the user
   41|       |    // of the library changed the global locale
   42|  1.85M|    ss.imbue(std::locale::classic());
   43|  1.85M|    if (v == floor(v)) {
  ------------------
  |  Branch (43:9): [True: 1.83M, False: 16.0k]
  ------------------
   44|  1.83M|        ss << std::fixed << std::setprecision(0) << v;
   45|  1.83M|    } else {
   46|       |        // See "What Every Computer Scientist Should Know About Floating-Point Arithmetic"
   47|       |        // Theorem 15
   48|       |        // https://docs.oracle.com/cd/E19957-01/806-3568/ncg_goldberg.html
   49|  16.0k|        ss << std::setprecision(17);
   50|  16.0k|        ss << v;
   51|  16.0k|    }
   52|  1.85M|    return ss.str();
   53|  1.85M|}
_ZN7jsonnet8internal13jsonnet_parseEPNS0_9AllocatorERNSt3__14listINS0_5TokenENS3_9allocatorIS5_EEEE:
 1210|  12.3k|{
 1211|  12.3k|    Parser parser(tokens, alloc);
 1212|  12.3k|    unsigned parse_depth = 0;
 1213|  12.3k|    AST *expr = parser.parse(MAX_PRECEDENCE, parse_depth);
 1214|  12.3k|    if (tokens.front().kind != Token::END_OF_FILE) {
  ------------------
  |  Branch (1214:9): [True: 176, False: 12.1k]
  ------------------
 1215|    176|        std::stringstream ss;
 1216|    176|        ss << "did not expect: " << tokens.front();
 1217|    176|        throw StaticError(tokens.front().location, ss.str());
 1218|    176|    }
 1219|       |
 1220|  12.1k|    return expr;
 1221|  12.3k|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16ParserC2ERNSt3__14listINS0_5TokenENS3_9allocatorIS5_EEEEPNS0_9AllocatorE:
  155|  12.3k|    Parser(Tokens &tokens, Allocator *alloc) : tokens(tokens), alloc(alloc) {}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser5parseEjj:
  980|  33.2M|    {
  981|  33.2M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (981:13): [True: 4, False: 33.2M]
  ------------------
  982|      4|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  983|      4|        }
  984|       |
  985|  33.2M|        AST *ast = maybeParseGreedy(current_depth + 1);
  986|       |        // There cannot be an operator after a greedy parse.
  987|  33.2M|        if (ast != nullptr) return ast;
  ------------------
  |  Branch (987:13): [True: 4.41M, False: 28.8M]
  ------------------
  988|       |
  989|       |        // If we get here, we could be parsing an infix construct.
  990|       |
  991|       |        // Allocate this on the heap to control stack growth.
  992|  28.8M|        std::unique_ptr<Token> begin_(new Token(peek()));
  993|  28.8M|        const Token &begin = *begin_;
  994|       |
  995|  28.8M|        AST *lhs = parseTerminalBracketsOrUnary(current_depth + 1);
  996|       |
  997|  28.8M|        return parseInfix(lhs, begin, max_precedence, current_depth + 1);
  998|  33.2M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser4peekEv:
  123|   275M|    {
  124|   275M|        return tokens.front();
  125|   275M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser16maybeParseGreedyEj:
  811|  33.2M|    {
  812|  33.2M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (812:13): [True: 3, False: 33.2M]
  ------------------
  813|      3|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  814|      3|        }
  815|       |
  816|       |        // Allocate this on the heap to control stack growth.
  817|  33.2M|        std::unique_ptr<Token> begin_(new Token(peek()));
  818|  33.2M|        const Token &begin = *begin_;
  819|       |
  820|  33.2M|        switch (begin.kind) {
  821|       |            // These cases have effectively MAX_PRECEDENCE as the first
  822|       |            // call to parse will parse them.
  823|   261k|            case Token::ASSERT: {
  ------------------
  |  Branch (823:13): [True: 261k, False: 33.0M]
  ------------------
  824|   261k|                pop();
  825|   261k|                AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  826|   261k|                Fodder colonFodder;
  827|   261k|                AST *msg = nullptr;
  828|   261k|                if (peek().kind == Token::OPERATOR && peek().data == ":") {
  ------------------
  |  Branch (828:21): [True: 241k, False: 19.4k]
  |  Branch (828:55): [True: 241k, False: 1]
  ------------------
  829|   241k|                    Token colon = pop();
  830|   241k|                    colonFodder = colon.fodder;
  831|   241k|                    msg = parse(MAX_PRECEDENCE, current_depth + 1);
  832|   241k|                }
  833|   261k|                Token semicolon = popExpect(Token::SEMICOLON);
  834|   261k|                AST *rest = parse(MAX_PRECEDENCE, current_depth + 1);
  835|   261k|                return alloc->make<Assert>(span(begin, rest),
  836|   261k|                                           begin.fodder,
  837|   261k|                                           cond,
  838|   261k|                                           colonFodder,
  839|   261k|                                           msg,
  840|   261k|                                           semicolon.fodder,
  841|   261k|                                           rest);
  842|      0|            }
  843|       |
  844|   416k|            case Token::ERROR: {
  ------------------
  |  Branch (844:13): [True: 416k, False: 32.8M]
  ------------------
  845|   416k|                pop();
  846|   416k|                AST *expr = parse(MAX_PRECEDENCE, current_depth + 1);
  847|   416k|                return alloc->make<Error>(span(begin, expr), begin.fodder, expr);
  848|      0|            }
  849|       |
  850|  2.07M|            case Token::IF: {
  ------------------
  |  Branch (850:13): [True: 2.07M, False: 31.2M]
  ------------------
  851|  2.07M|                pop();
  852|  2.07M|                AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  853|  2.07M|                Token then = popExpect(Token::THEN);
  854|  2.07M|                AST *branch_true = parse(MAX_PRECEDENCE, current_depth + 1);
  855|  2.07M|                if (peek().kind == Token::ELSE) {
  ------------------
  |  Branch (855:21): [True: 2.04M, False: 25.9k]
  ------------------
  856|  2.04M|                    Token else_ = pop();
  857|  2.04M|                    AST *branch_false = parse(MAX_PRECEDENCE, current_depth + 1);
  858|  2.04M|                    return alloc->make<Conditional>(span(begin, branch_false),
  859|  2.04M|                                                    begin.fodder,
  860|  2.04M|                                                    cond,
  861|  2.04M|                                                    then.fodder,
  862|  2.04M|                                                    branch_true,
  863|  2.04M|                                                    else_.fodder,
  864|  2.04M|                                                    branch_false);
  865|  2.04M|                }
  866|  25.9k|                return alloc->make<Conditional>(span(begin, branch_true),
  867|  25.9k|                                                begin.fodder,
  868|  25.9k|                                                cond,
  869|  25.9k|                                                then.fodder,
  870|  25.9k|                                                branch_true,
  871|  25.9k|                                                Fodder{},
  872|  25.9k|                                                nullptr);
  873|  2.07M|            }
  874|       |
  875|   128k|            case Token::FUNCTION: {
  ------------------
  |  Branch (875:13): [True: 128k, False: 33.1M]
  ------------------
  876|   128k|                pop();  // Still available in 'begin'.
  877|   128k|                Token paren_l = pop();
  878|   128k|                if (paren_l.kind == Token::PAREN_L) {
  ------------------
  |  Branch (878:21): [True: 128k, False: 6]
  ------------------
  879|   128k|                    std::vector<AST *> params_asts;
  880|   128k|                    bool got_comma;
  881|   128k|                    Fodder paren_r_fodder;
  882|   128k|                    ArgParams params = parseParams("function parameter", got_comma, paren_r_fodder, current_depth);
  883|   128k|                    AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  884|   128k|                    return alloc->make<Function>(span(begin, body),
  885|   128k|                                                 begin.fodder,
  886|   128k|                                                 paren_l.fodder,
  887|   128k|                                                 params,
  888|   128k|                                                 got_comma,
  889|   128k|                                                 paren_r_fodder,
  890|   128k|                                                 body);
  891|   128k|                } else {
  892|      6|                    std::stringstream ss;
  893|      6|                    ss << "expected ( but got " << paren_l;
  894|      6|                    throw StaticError(paren_l.location, ss.str());
  895|      6|                }
  896|   128k|            }
  897|       |
  898|    656|            case Token::IMPORT: {
  ------------------
  |  Branch (898:13): [True: 656, False: 33.2M]
  ------------------
  899|    656|                pop();
  900|    656|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  901|    656|                if (body->type == AST_LITERAL_STRING) {
  ------------------
  |  Branch (901:21): [True: 399, False: 257]
  ------------------
  902|    399|                    auto *lit = static_cast<LiteralString *>(body);
  903|    399|                    if (lit->tokenKind == LiteralString::BLOCK) {
  ------------------
  |  Branch (903:25): [True: 1, False: 398]
  ------------------
  904|      1|                        throw StaticError(lit->location,
  905|      1|                                          "Cannot use text blocks in import statements.");
  906|      1|                    }
  907|    398|                    return alloc->make<Import>(span(begin, body), begin.fodder, lit);
  908|    399|                } else {
  909|    257|                    std::stringstream ss;
  910|    257|                    ss << "computed imports are not allowed.";
  911|    257|                    throw StaticError(body->location, ss.str());
  912|    257|                }
  913|    656|            }
  914|       |
  915|    509|            case Token::IMPORTSTR: {
  ------------------
  |  Branch (915:13): [True: 509, False: 33.2M]
  ------------------
  916|    509|                pop();
  917|    509|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  918|    509|                if (body->type == AST_LITERAL_STRING) {
  ------------------
  |  Branch (918:21): [True: 438, False: 71]
  ------------------
  919|    438|                    auto *lit = static_cast<LiteralString *>(body);
  920|    438|                    if (lit->tokenKind == LiteralString::BLOCK) {
  ------------------
  |  Branch (920:25): [True: 1, False: 437]
  ------------------
  921|      1|                        throw StaticError(lit->location,
  922|      1|                                          "Cannot use text blocks in import statements.");
  923|      1|                    }
  924|    437|                    return alloc->make<Importstr>(span(begin, body), begin.fodder, lit);
  925|    438|                } else {
  926|     71|                    std::stringstream ss;
  927|     71|                    ss << "computed imports are not allowed.";
  928|     71|                    throw StaticError(body->location, ss.str());
  929|     71|                }
  930|    509|            }
  931|       |
  932|    351|            case Token::IMPORTBIN: {
  ------------------
  |  Branch (932:13): [True: 351, False: 33.2M]
  ------------------
  933|    351|                pop();
  934|    351|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  935|    351|                if (body->type == AST_LITERAL_STRING) {
  ------------------
  |  Branch (935:21): [True: 274, False: 77]
  ------------------
  936|    274|                    auto *lit = static_cast<LiteralString *>(body);
  937|    274|                    if (lit->tokenKind == LiteralString::BLOCK) {
  ------------------
  |  Branch (937:25): [True: 0, False: 274]
  ------------------
  938|      0|                        throw StaticError(lit->location,
  939|      0|                                          "Cannot use text blocks in import statements.");
  940|      0|                    }
  941|    274|                    return alloc->make<Importbin>(span(begin, body), begin.fodder, lit);
  942|    274|                } else {
  943|     77|                    std::stringstream ss;
  944|     77|                    ss << "computed imports are not allowed.";
  945|     77|                    throw StaticError(body->location, ss.str());
  946|     77|                }
  947|    351|            }
  948|       |
  949|  1.53M|            case Token::LOCAL: {
  ------------------
  |  Branch (949:13): [True: 1.53M, False: 31.7M]
  ------------------
  950|  1.53M|                pop();
  951|  1.53M|                Local::Binds binds;
  952|  1.62M|                do {
  953|  1.62M|                    Token delim = parseBind(binds, current_depth + 1);
  954|  1.62M|                    if (delim.kind != Token::SEMICOLON && delim.kind != Token::COMMA) {
  ------------------
  |  Branch (954:25): [True: 83.1k, False: 1.53M]
  |  Branch (954:59): [True: 16, False: 83.0k]
  ------------------
  955|     16|                        std::stringstream ss;
  956|     16|                        ss << "expected , or ; but got " << delim;
  957|     16|                        throw StaticError(delim.location, ss.str());
  958|     16|                    }
  959|  1.62M|                    if (delim.kind == Token::SEMICOLON)
  ------------------
  |  Branch (959:25): [True: 1.53M, False: 83.4k]
  ------------------
  960|  1.53M|                        break;
  961|  1.62M|                } while (true);
  ------------------
  |  Branch (961:26): [True: 83.0k, Folded]
  ------------------
  962|  1.53M|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  963|  1.53M|                return alloc->make<Local>(span(begin, body), begin.fodder, binds, body);
  964|  1.53M|            }
  965|       |
  966|  28.8M|            default:
  ------------------
  |  Branch (966:13): [True: 28.8M, False: 4.41M]
  ------------------
  967|  28.8M|            return nullptr;
  968|  33.2M|        }
  969|  33.2M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser3popEv:
  111|  86.5M|    {
  112|  86.5M|        Token tok = peek();
  113|  86.5M|        tokens.pop_front();
  114|  86.5M|        return tok;
  115|  86.5M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser9popExpectENS0_5Token4KindEPKc:
  136|  13.6M|    {
  137|  13.6M|        Token tok = pop();
  138|  13.6M|        if (tok.kind != k) {
  ------------------
  |  Branch (138:13): [True: 217, False: 13.6M]
  ------------------
  139|    217|            std::stringstream ss;
  140|    217|            ss << "expected token " << k << " but got " << tok;
  141|    217|            throw StaticError(tok.location, ss.str());
  142|    217|        }
  143|  13.6M|        if (data != nullptr && tok.data != data) {
  ------------------
  |  Branch (143:13): [True: 1.66M, False: 12.0M]
  |  Branch (143:32): [True: 11, False: 1.66M]
  ------------------
  144|     11|            std::stringstream ss;
  145|     11|            ss << "expected operator " << data << " but got " << tok.data;
  146|     11|            throw StaticError(tok.location, ss.str());
  147|     11|        }
  148|  13.6M|        return tok;
  149|  13.6M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14spanERKNS0_5TokenEPNS0_3ASTE:
   94|  10.3M|{
   95|  10.3M|    return LocationRange(begin.location.file, begin.location.begin, end->location.end);
   96|  10.3M|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser11parseParamsERKNSt3__112basic_stringIcNS3_11char_traitsIcEENS3_9allocatorIcEEEERbRNS3_6vectorINS0_13FodderElementENS7_ISE_EEEEj:
  218|  1.39M|    {
  219|  1.39M|        ArgParams params;
  220|  1.39M|        Token paren_r = parseArgs(params, element_kind, got_comma, current_depth);
  221|       |
  222|       |        // Check they're all identifiers
  223|       |        // parseArgs returns f(x) with x as an expression.  Convert it here.
  224|  2.73M|        for (auto &p : params) {
  ------------------
  |  Branch (224:22): [True: 2.73M, False: 1.39M]
  ------------------
  225|  2.73M|            if (p.id == nullptr) {
  ------------------
  |  Branch (225:17): [True: 2.59M, False: 138k]
  ------------------
  226|  2.59M|                if (p.expr->type != AST_VAR) {
  ------------------
  |  Branch (226:21): [True: 5, False: 2.59M]
  ------------------
  227|      5|                    throw StaticError(p.expr->location, "could not parse parameter here.");
  228|      5|                }
  229|  2.59M|                auto *pv = static_cast<Var *>(p.expr);
  230|  2.59M|                p.id = pv->id;
  231|  2.59M|                p.idFodder = pv->openFodder;
  232|  2.59M|                p.expr = nullptr;
  233|  2.59M|            }
  234|  2.73M|        }
  235|       |
  236|  1.39M|        close_fodder = paren_r.fodder;
  237|       |
  238|  1.39M|        return params;
  239|  1.39M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser9parseArgsERNSt3__16vectorINS0_8ArgParamENS3_9allocatorIS5_EEEERKNS3_12basic_stringIcNS3_11char_traitsIcEENS6_IcEEEERbj:
  167|  6.55M|    {
  168|  6.55M|        got_comma = false;
  169|  6.55M|        bool first = true;
  170|  17.9M|        do {
  171|  17.9M|            Token next = peek();
  172|  17.9M|            if (next.kind == Token::PAREN_R) {
  ------------------
  |  Branch (172:17): [True: 6.55M, False: 11.3M]
  ------------------
  173|       |                // got_comma can be true or false here.
  174|  6.55M|                return pop();
  175|  6.55M|            }
  176|  11.3M|            if (!first && !got_comma) {
  ------------------
  |  Branch (176:17): [True: 4.80M, False: 6.55M]
  |  Branch (176:27): [True: 118, False: 4.80M]
  ------------------
  177|    118|                std::stringstream ss;
  178|    118|                ss << "expected a comma before next " << element_kind << ".";
  179|    118|                throw StaticError(next.location, ss.str());
  180|    118|            }
  181|       |            // Either id=expr or id or expr, but note that expr could be id==1 so this needs
  182|       |            // look-ahead.
  183|  11.3M|            Fodder id_fodder;
  184|  11.3M|            const Identifier *id = nullptr;
  185|  11.3M|            Fodder eq_fodder;
  186|  11.3M|            if (peek().kind == Token::IDENTIFIER) {
  ------------------
  |  Branch (186:17): [True: 9.74M, False: 1.61M]
  ------------------
  187|  9.74M|                Token maybe_eq = doublePeek();
  188|  9.74M|                if (maybe_eq.kind == Token::OPERATOR && maybe_eq.data == "=") {
  ------------------
  |  Branch (188:21): [True: 1.01M, False: 8.72M]
  |  Branch (188:57): [True: 158k, False: 858k]
  ------------------
  189|   158k|                    id_fodder = peek().fodder;
  190|   158k|                    id = alloc->makeIdentifier(peek().data32());
  191|   158k|                    eq_fodder = maybe_eq.fodder;
  192|   158k|                    pop();  // id
  193|   158k|                    pop();  // eq
  194|   158k|                }
  195|  9.74M|            }
  196|  11.3M|            AST *expr = parse(MAX_PRECEDENCE, current_depth + 1);
  197|  11.3M|            got_comma = false;
  198|  11.3M|            first = false;
  199|  11.3M|            Fodder comma_fodder;
  200|  11.3M|            if (peek().kind == Token::COMMA) {
  ------------------
  |  Branch (200:17): [True: 4.80M, False: 6.55M]
  ------------------
  201|  4.80M|                Token comma = pop();
  202|  4.80M|                comma_fodder = comma.fodder;
  203|  4.80M|                got_comma = true;
  204|  4.80M|            }
  205|  11.3M|            args.emplace_back(id_fodder, id, eq_fodder, expr, comma_fodder);
  206|  11.3M|        } while (true);
  ------------------
  |  Branch (206:18): [True: 11.3M, Folded]
  ------------------
  207|  6.55M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser10doublePeekEv:
  129|  9.74M|    {
  130|  9.74M|        Tokens::iterator it = tokens.begin();  // First one.
  131|  9.74M|        it++;                                  // Now pointing at the second one.
  132|  9.74M|        return *(it);
  133|  9.74M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser9parseBindERNSt3__16vectorINS0_5Local4BindENS3_9allocatorIS6_EEEEj:
  248|  1.62M|    {
  249|  1.62M|        Token var_id = popExpect(Token::IDENTIFIER);
  250|  1.62M|        auto *id = alloc->makeIdentifier(var_id.data32());
  251|  1.62M|        for (const auto &bind : binds) {
  ------------------
  |  Branch (251:31): [True: 224k, False: 1.62M]
  ------------------
  252|   224k|            if (bind.var == id)
  ------------------
  |  Branch (252:17): [True: 5, False: 224k]
  ------------------
  253|      5|                throw StaticError(var_id.location, "duplicate local var: " + var_id.data);
  254|   224k|        }
  255|  1.62M|        bool is_function = false;
  256|  1.62M|        ArgParams params;
  257|  1.62M|        bool trailing_comma = false;
  258|  1.62M|        Fodder fodder_l, fodder_r;
  259|  1.62M|        if (peek().kind == Token::PAREN_L) {
  ------------------
  |  Branch (259:13): [True: 470k, False: 1.15M]
  ------------------
  260|   470k|            Token paren_l = pop();
  261|   470k|            fodder_l = paren_l.fodder;
  262|   470k|            params = parseParams("function parameter", trailing_comma, fodder_r, current_depth);
  263|   470k|            is_function = true;
  264|   470k|        }
  265|  1.62M|        Token eq = popExpect(Token::OPERATOR, "=");
  266|  1.62M|        AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  267|  1.62M|        Token delim = pop();
  268|  1.62M|        binds.emplace_back(var_id.fodder,
  269|  1.62M|                           id,
  270|  1.62M|                           eq.fodder,
  271|  1.62M|                           body,
  272|  1.62M|                           is_function,
  273|  1.62M|                           fodder_l,
  274|  1.62M|                           params,
  275|  1.62M|                           trailing_comma,
  276|  1.62M|                           fodder_r,
  277|  1.62M|                           delim.fodder);
  278|  1.62M|        return delim;
  279|  1.62M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser28parseTerminalBracketsOrUnaryEj:
  627|  28.8M|    {
  628|  28.8M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (628:13): [True: 0, False: 28.8M]
  ------------------
  629|      0|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  630|      0|        }
  631|       |
  632|  28.8M|        Token tok = pop();
  633|  28.8M|        switch (tok.kind) {
  ------------------
  |  Branch (633:17): [True: 28.8M, False: 0]
  ------------------
  634|      0|            case Token::ASSERT:
  ------------------
  |  Branch (634:13): [True: 0, False: 28.8M]
  ------------------
  635|      7|            case Token::BRACE_R:
  ------------------
  |  Branch (635:13): [True: 7, False: 28.8M]
  ------------------
  636|     16|            case Token::BRACKET_R:
  ------------------
  |  Branch (636:13): [True: 9, False: 28.8M]
  ------------------
  637|     40|            case Token::COMMA:
  ------------------
  |  Branch (637:13): [True: 24, False: 28.8M]
  ------------------
  638|     52|            case Token::DOT:
  ------------------
  |  Branch (638:13): [True: 12, False: 28.8M]
  ------------------
  639|     53|            case Token::ELSE:
  ------------------
  |  Branch (639:13): [True: 1, False: 28.8M]
  ------------------
  640|     53|            case Token::ERROR:
  ------------------
  |  Branch (640:13): [True: 0, False: 28.8M]
  ------------------
  641|     54|            case Token::FOR:
  ------------------
  |  Branch (641:13): [True: 1, False: 28.8M]
  ------------------
  642|     54|            case Token::FUNCTION:
  ------------------
  |  Branch (642:13): [True: 0, False: 28.8M]
  ------------------
  643|     54|            case Token::IF:
  ------------------
  |  Branch (643:13): [True: 0, False: 28.8M]
  ------------------
  644|     60|            case Token::IN:
  ------------------
  |  Branch (644:13): [True: 6, False: 28.8M]
  ------------------
  645|     60|            case Token::IMPORT:
  ------------------
  |  Branch (645:13): [True: 0, False: 28.8M]
  ------------------
  646|     60|            case Token::IMPORTSTR:
  ------------------
  |  Branch (646:13): [True: 0, False: 28.8M]
  ------------------
  647|     60|            case Token::IMPORTBIN:
  ------------------
  |  Branch (647:13): [True: 0, False: 28.8M]
  ------------------
  648|     60|            case Token::LOCAL:
  ------------------
  |  Branch (648:13): [True: 0, False: 28.8M]
  ------------------
  649|     85|            case Token::PAREN_R:
  ------------------
  |  Branch (649:13): [True: 25, False: 28.8M]
  ------------------
  650|     94|            case Token::SEMICOLON:
  ------------------
  |  Branch (650:13): [True: 9, False: 28.8M]
  ------------------
  651|     95|            case Token::TAILSTRICT:
  ------------------
  |  Branch (651:13): [True: 1, False: 28.8M]
  ------------------
  652|     96|            case Token::THEN: throw unexpected(tok, "parsing terminal");
  ------------------
  |  Branch (652:13): [True: 1, False: 28.8M]
  ------------------
  653|       |
  654|    487|            case Token::END_OF_FILE: throw StaticError(tok.location, "unexpected end of file.");
  ------------------
  |  Branch (654:13): [True: 487, False: 28.8M]
  ------------------
  655|       |
  656|   376k|            case Token::OPERATOR: {
  ------------------
  |  Branch (656:13): [True: 376k, False: 28.4M]
  ------------------
  657|   376k|                UnaryOp uop;
  658|   376k|                if (!op_is_unary(tok.data, uop)) {
  ------------------
  |  Branch (658:21): [True: 166, False: 375k]
  ------------------
  659|    166|                    std::stringstream ss;
  660|    166|                    ss << "not a unary operator: " << tok.data;
  661|    166|                    throw StaticError(tok.location, ss.str());
  662|    166|                }
  663|   375k|                AST *expr = parse(UNARY_PRECEDENCE, current_depth + 1);
  664|   375k|                return alloc->make<Unary>(span(tok, expr), tok.fodder, uop, expr);
  665|   376k|            }
  666|   318k|            case Token::BRACE_L: {
  ------------------
  |  Branch (666:13): [True: 318k, False: 28.5M]
  ------------------
  667|   318k|                AST *obj;
  668|   318k|                parseObjectRemainder(obj, tok, current_depth + 1);
  669|   318k|                return obj;
  670|   376k|            }
  671|       |
  672|   898k|            case Token::BRACKET_L: {
  ------------------
  |  Branch (672:13): [True: 898k, False: 27.9M]
  ------------------
  673|   898k|                Token next = peek();
  674|   898k|                if (next.kind == Token::BRACKET_R) {
  ------------------
  |  Branch (674:21): [True: 150k, False: 747k]
  ------------------
  675|   150k|                    Token bracket_r = pop();
  676|   150k|                    return alloc->make<Array>(
  677|   150k|                        span(tok, next), tok.fodder, Array::Elements{}, false, bracket_r.fodder);
  678|   150k|                }
  679|   747k|                AST *first = parse(MAX_PRECEDENCE, current_depth + 1);
  680|   747k|                bool got_comma = false;
  681|   747k|                Fodder comma_fodder;
  682|   747k|                next = peek();
  683|   747k|                if (!got_comma && next.kind == Token::COMMA) {
  ------------------
  |  Branch (683:21): [True: 744k, False: 3.72k]
  |  Branch (683:35): [True: 112k, False: 631k]
  ------------------
  684|   112k|                    Token comma = pop();
  685|   112k|                    comma_fodder = comma.fodder;
  686|   112k|                    next = peek();
  687|   112k|                    got_comma = true;
  688|   112k|                }
  689|       |
  690|   747k|                if (next.kind == Token::FOR) {
  ------------------
  |  Branch (690:21): [True: 203k, False: 544k]
  ------------------
  691|       |                    // It's a comprehension
  692|   203k|                    Token for_token = pop();
  693|   203k|                    std::vector<ComprehensionSpec> specs;
  694|   203k|                    Token last = parseComprehensionSpecs(Token::BRACKET_R, for_token.fodder, specs, current_depth + 1);
  695|   203k|                    return alloc->make<ArrayComprehension>(span(tok, last),
  696|   203k|                                                           tok.fodder,
  697|   203k|                                                           first,
  698|   203k|                                                           comma_fodder,
  699|   203k|                                                           got_comma,
  700|   203k|                                                           specs,
  701|   203k|                                                           last.fodder);
  702|   203k|                }
  703|       |
  704|       |                // Not a comprehension: It can have more elements.
  705|   544k|                Array::Elements elements;
  706|   544k|                elements.emplace_back(first, comma_fodder);
  707|  1.64M|                do {
  708|  1.64M|                    if (next.kind == Token::BRACKET_R) {
  ------------------
  |  Branch (708:25): [True: 538k, False: 1.10M]
  ------------------
  709|   538k|                        Token bracket_r = pop();
  710|   538k|                        return alloc->make<Array>(
  711|   538k|                            span(tok, next), tok.fodder, elements, got_comma, bracket_r.fodder);
  712|   538k|                    }
  713|  1.10M|                    if (!got_comma) {
  ------------------
  |  Branch (713:25): [True: 266, False: 1.10M]
  ------------------
  714|    266|                        std::stringstream ss;
  715|    266|                        ss << "expected a comma before next array element.";
  716|    266|                        throw StaticError(next.location, ss.str());
  717|    266|                    }
  718|  1.10M|                    AST *expr = parse(MAX_PRECEDENCE, current_depth + 1);
  719|  1.10M|                    comma_fodder.clear();
  720|  1.10M|                    got_comma = false;
  721|  1.10M|                    next = peek();
  722|  1.10M|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (722:25): [True: 1.00M, False: 103k]
  ------------------
  723|  1.00M|                        Token comma = pop();
  724|  1.00M|                        comma_fodder = comma.fodder;
  725|  1.00M|                        next = peek();
  726|  1.00M|                        got_comma = true;
  727|  1.00M|                    }
  728|  1.10M|                    elements.emplace_back(expr, comma_fodder);
  729|  1.10M|                } while (true);
  ------------------
  |  Branch (729:26): [True: 1.09M, Folded]
  ------------------
  730|   544k|            }
  731|       |
  732|   324k|            case Token::PAREN_L: {
  ------------------
  |  Branch (732:13): [True: 320k, False: 28.5M]
  ------------------
  733|   324k|                auto *inner = parse(MAX_PRECEDENCE, current_depth + 1);
  734|   324k|                Token close = popExpect(Token::PAREN_R);
  735|   324k|                return alloc->make<Parens>(span(tok, close), tok.fodder, inner, close.fodder);
  736|   544k|            }
  737|       |
  738|       |            // Literals
  739|  3.32M|            case Token::NUMBER: return alloc->make<LiteralNumber>(span(tok), tok.fodder, tok.data);
  ------------------
  |  Branch (739:13): [True: 3.32M, False: 25.5M]
  ------------------
  740|       |
  741|  2.98M|            case Token::STRING_SINGLE:
  ------------------
  |  Branch (741:13): [True: 2.98M, False: 25.8M]
  ------------------
  742|  2.98M|                return alloc->make<LiteralString>(
  743|  2.98M|                    span(tok), tok.fodder, tok.data32(), LiteralString::SINGLE, "", "");
  744|  42.4k|            case Token::STRING_DOUBLE:
  ------------------
  |  Branch (744:13): [True: 42.4k, False: 28.8M]
  ------------------
  745|  42.4k|                return alloc->make<LiteralString>(
  746|  42.4k|                    span(tok), tok.fodder, tok.data32(), LiteralString::DOUBLE, "", "");
  747|  2.61k|            case Token::STRING_BLOCK:
  ------------------
  |  Branch (747:13): [True: 2.61k, False: 28.8M]
  ------------------
  748|  2.61k|                return alloc->make<LiteralString>(span(tok),
  749|  2.61k|                                                  tok.fodder,
  750|  2.61k|                                                  tok.data32(),
  751|  2.61k|                                                  LiteralString::BLOCK,
  752|  2.61k|                                                  tok.stringBlockIndent,
  753|  2.61k|                                                  tok.stringBlockTermIndent);
  754|    979|            case Token::VERBATIM_STRING_SINGLE:
  ------------------
  |  Branch (754:13): [True: 979, False: 28.8M]
  ------------------
  755|    979|                return alloc->make<LiteralString>(
  756|    979|                    span(tok), tok.fodder, tok.data32(), LiteralString::VERBATIM_SINGLE, "", "");
  757|  3.12k|            case Token::VERBATIM_STRING_DOUBLE:
  ------------------
  |  Branch (757:13): [True: 3.12k, False: 28.8M]
  ------------------
  758|  3.12k|                return alloc->make<LiteralString>(
  759|  3.12k|                    span(tok), tok.fodder, tok.data32(), LiteralString::VERBATIM_DOUBLE, "", "");
  760|       |
  761|   311k|            case Token::FALSE: return alloc->make<LiteralBoolean>(span(tok), tok.fodder, false);
  ------------------
  |  Branch (761:13): [True: 311k, False: 28.5M]
  ------------------
  762|       |
  763|   228k|            case Token::TRUE: return alloc->make<LiteralBoolean>(span(tok), tok.fodder, true);
  ------------------
  |  Branch (763:13): [True: 228k, False: 28.6M]
  ------------------
  764|       |
  765|   114k|            case Token::NULL_LIT: return alloc->make<LiteralNull>(span(tok), tok.fodder);
  ------------------
  |  Branch (765:13): [True: 114k, False: 28.7M]
  ------------------
  766|       |
  767|       |            // Variables
  768|  37.4k|            case Token::DOLLAR: return alloc->make<Dollar>(span(tok), tok.fodder);
  ------------------
  |  Branch (768:13): [True: 37.4k, False: 28.8M]
  ------------------
  769|       |
  770|  19.8M|            case Token::IDENTIFIER:
  ------------------
  |  Branch (770:13): [True: 19.8M, False: 9.01M]
  ------------------
  771|  19.8M|                return alloc->make<Var>(span(tok), tok.fodder, alloc->makeIdentifier(tok.data32()));
  772|       |
  773|  44.5k|            case Token::SELF: return alloc->make<Self>(span(tok), tok.fodder);
  ------------------
  |  Branch (773:13): [True: 44.5k, False: 28.8M]
  ------------------
  774|       |
  775|  1.72k|            case Token::SUPER: {
  ------------------
  |  Branch (775:13): [True: 1.72k, False: 28.8M]
  ------------------
  776|  1.72k|                Token next = pop();
  777|  1.72k|                AST *index = nullptr;
  778|  1.72k|                const Identifier *id = nullptr;
  779|  1.72k|                Fodder id_fodder;
  780|  1.72k|                switch (next.kind) {
  781|  1.41k|                    case Token::DOT: {
  ------------------
  |  Branch (781:21): [True: 1.41k, False: 312]
  ------------------
  782|  1.41k|                        Token field_id = popExpect(Token::IDENTIFIER);
  783|  1.41k|                        id_fodder = field_id.fodder;
  784|  1.41k|                        id = alloc->makeIdentifier(field_id.data32());
  785|  1.41k|                    } break;
  786|    307|                    case Token::BRACKET_L: {
  ------------------
  |  Branch (786:21): [True: 307, False: 1.41k]
  ------------------
  787|    307|                        index = parse(MAX_PRECEDENCE, current_depth + 1);
  788|    307|                        Token bracket_r = popExpect(Token::BRACKET_R);
  789|    307|                        id_fodder = bracket_r.fodder;  // Not id_fodder, but use the same var.
  790|    307|                    } break;
  791|      5|                    default: throw StaticError(tok.location, "expected . or [ after super.");
  ------------------
  |  Branch (791:21): [True: 5, False: 1.71k]
  ------------------
  792|  1.72k|                }
  793|  1.63k|                return alloc->make<SuperIndex>(
  794|  1.63k|                    span(tok), tok.fodder, next.fodder, index, id_fodder, id);
  795|  1.72k|            }
  796|  28.8M|        }
  797|       |
  798|      0|        std::cerr << "INTERNAL ERROR: Unknown tok kind: " << tok.kind << std::endl;
  799|      0|        std::abort();
  800|      0|        return nullptr;  // Quiet, compiler.
  801|  28.8M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser10unexpectedERKNS0_5TokenERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEE:
  104|    287|    {
  105|    287|        std::stringstream ss;
  106|    287|        ss << "unexpected: " << tok.kind << " while " << while_;
  107|    287|        return StaticError(tok.location, ss.str());
  108|    287|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111op_is_unaryERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERNS0_7UnaryOpE:
   66|   376k|{
   67|   376k|    auto it = unary_map.find(op);
   68|   376k|    if (it == unary_map.end())
  ------------------
  |  Branch (68:9): [True: 166, False: 375k]
  ------------------
   69|    166|        return false;
   70|   375k|    uop = it->second;
   71|   375k|    return true;
   72|   376k|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser20parseObjectRemainderERPNS0_3ASTERKNS0_5TokenEj:
  289|   379k|    {
  290|   379k|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (290:13): [True: 1, False: 379k]
  ------------------
  291|      1|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  292|      1|        }
  293|       |
  294|   379k|        ObjectFields fields;
  295|   379k|        std::set<std::string> literal_fields;  // For duplicate fields detection.
  296|   379k|        std::set<const Identifier *> binds;    // For duplicate locals detection.
  297|       |
  298|   379k|        bool got_comma = false;
  299|   379k|        bool first = true;
  300|   379k|        Token next = pop();
  301|       |
  302|  1.98M|        do {
  303|  1.98M|            if (next.kind == Token::BRACE_R) {
  ------------------
  |  Branch (303:17): [True: 346k, False: 1.63M]
  ------------------
  304|   346k|                obj = alloc->make<Object>(
  305|   346k|                    span(tok, next), tok.fodder, fields, got_comma, next.fodder);
  306|   346k|                return next;
  307|       |
  308|  1.63M|            } else if (next.kind == Token::FOR) {
  ------------------
  |  Branch (308:24): [True: 27.5k, False: 1.60M]
  ------------------
  309|       |                // It's a comprehension
  310|  27.5k|                unsigned num_fields = 0;
  311|  27.5k|                unsigned num_asserts = 0;
  312|  27.5k|                const ObjectField *field_ptr = nullptr;
  313|  27.5k|                for (const auto &field : fields) {
  ------------------
  |  Branch (313:40): [True: 27.5k, False: 27.5k]
  ------------------
  314|  27.5k|                    if (field.kind == ObjectField::LOCAL)
  ------------------
  |  Branch (314:25): [True: 1, False: 27.5k]
  ------------------
  315|      1|                        continue;
  316|  27.5k|                    if (field.kind == ObjectField::ASSERT) {
  ------------------
  |  Branch (316:25): [True: 89, False: 27.5k]
  ------------------
  317|     89|                        num_asserts++;
  318|     89|                        continue;
  319|     89|                    }
  320|  27.5k|                    field_ptr = &field;
  321|  27.5k|                    num_fields++;
  322|  27.5k|                }
  323|  27.5k|                if (num_asserts > 0) {
  ------------------
  |  Branch (323:21): [True: 10, False: 27.5k]
  ------------------
  324|     10|                    auto msg = "object comprehension cannot have asserts.";
  325|     10|                    throw StaticError(next.location, msg);
  326|     10|                }
  327|  27.5k|                if (num_fields != 1) {
  ------------------
  |  Branch (327:21): [True: 3, False: 27.5k]
  ------------------
  328|      3|                    auto msg = "object comprehension can only have one field.";
  329|      3|                    throw StaticError(next.location, msg);
  330|      3|                }
  331|  27.5k|                const ObjectField &field = *field_ptr;
  332|       |
  333|  27.5k|                if (field.hide != ObjectField::INHERIT) {
  ------------------
  |  Branch (333:21): [True: 1, False: 27.5k]
  ------------------
  334|      1|                    auto msg = "object comprehensions cannot have hidden fields.";
  335|      1|                    throw StaticError(next.location, msg);
  336|      1|                }
  337|       |
  338|  27.5k|                if (field.kind != ObjectField::FIELD_EXPR) {
  ------------------
  |  Branch (338:21): [True: 1, False: 27.5k]
  ------------------
  339|      1|                    auto msg = "object comprehensions can only have [e] fields.";
  340|      1|                    throw StaticError(next.location, msg);
  341|      1|                }
  342|       |
  343|  27.5k|                std::vector<ComprehensionSpec> specs;
  344|  27.5k|                Token last = parseComprehensionSpecs(Token::BRACE_R, next.fodder, specs, current_depth + 1);
  345|  27.5k|                obj = alloc->make<ObjectComprehension>(
  346|  27.5k|                    span(tok, last), tok.fodder, fields, got_comma, specs, last.fodder);
  347|       |
  348|  27.5k|                return last;
  349|  27.5k|            }
  350|       |
  351|  1.60M|            if (!got_comma && !first)
  ------------------
  |  Branch (351:17): [True: 340k, False: 1.26M]
  |  Branch (351:31): [True: 80, False: 340k]
  ------------------
  352|     80|                throw StaticError(next.location, "expected a comma before next field.");
  353|       |
  354|  1.60M|            first = false;
  355|  1.60M|            got_comma = false;
  356|       |
  357|  1.60M|            switch (next.kind) {
  358|  29.6k|                case Token::BRACKET_L:
  ------------------
  |  Branch (358:17): [True: 29.6k, False: 1.57M]
  ------------------
  359|  1.46M|                case Token::IDENTIFIER:
  ------------------
  |  Branch (359:17): [True: 1.43M, False: 167k]
  ------------------
  360|  1.50M|                case Token::STRING_DOUBLE:
  ------------------
  |  Branch (360:17): [True: 34.9k, False: 1.57M]
  ------------------
  361|  1.52M|                case Token::STRING_SINGLE:
  ------------------
  |  Branch (361:17): [True: 25.9k, False: 1.58M]
  ------------------
  362|  1.52M|                case Token::STRING_BLOCK:
  ------------------
  |  Branch (362:17): [True: 626, False: 1.60M]
  ------------------
  363|  1.53M|                case Token::VERBATIM_STRING_DOUBLE:
  ------------------
  |  Branch (363:17): [True: 1.05k, False: 1.60M]
  ------------------
  364|  1.53M|                case Token::VERBATIM_STRING_SINGLE: {
  ------------------
  |  Branch (364:17): [True: 346, False: 1.60M]
  ------------------
  365|  1.53M|                    ObjectField::Kind kind;
  366|  1.53M|                    AST *expr1 = nullptr;
  367|  1.53M|                    const Identifier *id = nullptr;
  368|  1.53M|                    Fodder fodder1, fodder2;
  369|  1.53M|                    LocationRange idLocation;
  370|  1.53M|                    if (next.kind == Token::IDENTIFIER) {
  ------------------
  |  Branch (370:25): [True: 1.43M, False: 92.5k]
  ------------------
  371|  1.43M|                        fodder1 = next.fodder;
  372|  1.43M|                        kind = ObjectField::FIELD_ID;
  373|  1.43M|                        id = alloc->makeIdentifier(next.data32());
  374|  1.43M|                        idLocation = next.location;
  375|  1.43M|                    } else if (next.kind == Token::STRING_DOUBLE) {
  ------------------
  |  Branch (375:32): [True: 34.9k, False: 57.6k]
  ------------------
  376|  34.9k|                        kind = ObjectField::FIELD_STR;
  377|  34.9k|                        expr1 = alloc->make<LiteralString>(next.location,
  378|  34.9k|                                                           next.fodder,
  379|  34.9k|                                                           next.data32(),
  380|  34.9k|                                                           LiteralString::DOUBLE,
  381|  34.9k|                                                           "",
  382|  34.9k|                                                           "");
  383|  57.6k|                    } else if (next.kind == Token::STRING_SINGLE) {
  ------------------
  |  Branch (383:32): [True: 25.9k, False: 31.6k]
  ------------------
  384|  25.9k|                        kind = ObjectField::FIELD_STR;
  385|  25.9k|                        expr1 = alloc->make<LiteralString>(next.location,
  386|  25.9k|                                                           next.fodder,
  387|  25.9k|                                                           next.data32(),
  388|  25.9k|                                                           LiteralString::SINGLE,
  389|  25.9k|                                                           "",
  390|  25.9k|                                                           "");
  391|  31.6k|                    } else if (next.kind == Token::STRING_BLOCK) {
  ------------------
  |  Branch (391:32): [True: 626, False: 31.0k]
  ------------------
  392|    626|                        kind = ObjectField::FIELD_STR;
  393|    626|                        expr1 = alloc->make<LiteralString>(next.location,
  394|    626|                                                           next.fodder,
  395|    626|                                                           next.data32(),
  396|    626|                                                           LiteralString::BLOCK,
  397|    626|                                                           next.stringBlockIndent,
  398|    626|                                                           next.stringBlockTermIndent);
  399|  31.0k|                    } else if (next.kind == Token::VERBATIM_STRING_SINGLE) {
  ------------------
  |  Branch (399:32): [True: 346, False: 30.6k]
  ------------------
  400|    346|                        kind = ObjectField::FIELD_STR;
  401|    346|                        expr1 = alloc->make<LiteralString>(next.location,
  402|    346|                                                           next.fodder,
  403|    346|                                                           next.data32(),
  404|    346|                                                           LiteralString::VERBATIM_SINGLE,
  405|    346|                                                           "",
  406|    346|                                                           "");
  407|  30.6k|                    } else if (next.kind == Token::VERBATIM_STRING_DOUBLE) {
  ------------------
  |  Branch (407:32): [True: 1.05k, False: 29.6k]
  ------------------
  408|  1.05k|                        kind = ObjectField::FIELD_STR;
  409|  1.05k|                        expr1 = alloc->make<LiteralString>(next.location,
  410|  1.05k|                                                           next.fodder,
  411|  1.05k|                                                           next.data32(),
  412|  1.05k|                                                           LiteralString::VERBATIM_DOUBLE,
  413|  1.05k|                                                           "",
  414|  1.05k|                                                           "");
  415|  29.6k|                    } else {
  416|  29.6k|                        kind = ObjectField::FIELD_EXPR;
  417|  29.6k|                        fodder1 = next.fodder;
  418|  29.6k|                        expr1 = parse(MAX_PRECEDENCE, current_depth + 1);
  419|  29.6k|                        Token bracket_r = popExpect(Token::BRACKET_R);
  420|  29.6k|                        fodder2 = bracket_r.fodder;
  421|  29.6k|                    }
  422|       |
  423|  1.53M|                    bool is_method = false;
  424|  1.53M|                    bool meth_comma = false;
  425|  1.53M|                    ArgParams params;
  426|  1.53M|                    Fodder fodder_l;
  427|  1.53M|                    Fodder fodder_r;
  428|  1.53M|                    if (peek().kind == Token::PAREN_L) {
  ------------------
  |  Branch (428:25): [True: 791k, False: 739k]
  ------------------
  429|   791k|                        Token paren_l = pop();
  430|   791k|                        fodder_l = paren_l.fodder;
  431|   791k|                        params = parseParams("method parameter", meth_comma, fodder_r, current_depth);
  432|   791k|                        is_method = true;
  433|   791k|                    }
  434|       |
  435|  1.53M|                    bool plus_sugar = false;
  436|       |
  437|  1.53M|                    Token op = popExpect(Token::OPERATOR);
  438|  1.53M|                    const char *od = op.data.c_str();
  439|  1.53M|                    if (*od == '+') {
  ------------------
  |  Branch (439:25): [True: 11.7k, False: 1.51M]
  ------------------
  440|  11.7k|                        plus_sugar = true;
  441|  11.7k|                        od++;
  442|  11.7k|                    }
  443|  1.53M|                    unsigned colons = 0;
  444|  3.85M|                    for (; *od != '\0'; ++od) {
  ------------------
  |  Branch (444:28): [True: 2.32M, False: 1.53M]
  ------------------
  445|  2.32M|                        if (*od != ':') {
  ------------------
  |  Branch (445:29): [True: 16, False: 2.32M]
  ------------------
  446|     16|                            throw StaticError(
  447|     16|                                next.location,
  448|     16|                                "expected one of :, ::, :::, +:, +::, +:::, got: " + op.data);
  449|     16|                        }
  450|  2.32M|                        ++colons;
  451|  2.32M|                    }
  452|  1.53M|                    ObjectField::Hide field_hide;
  453|  1.53M|                    switch (colons) {
  454|   735k|                        case 1: field_hide = ObjectField::INHERIT; break;
  ------------------
  |  Branch (454:25): [True: 735k, False: 796k]
  ------------------
  455|       |
  456|   794k|                        case 2: field_hide = ObjectField::HIDDEN; break;
  ------------------
  |  Branch (456:25): [True: 794k, False: 736k]
  ------------------
  457|       |
  458|      9|                        case 3: field_hide = ObjectField::VISIBLE; break;
  ------------------
  |  Branch (458:25): [True: 9, False: 1.53M]
  ------------------
  459|       |
  460|     11|                        default:
  ------------------
  |  Branch (460:25): [True: 11, False: 1.53M]
  ------------------
  461|     11|                            throw StaticError(
  462|     11|                                next.location,
  463|     11|                                "expected one of :, ::, :::, +:, +::, +:::, got: " + op.data);
  464|  1.53M|                    }
  465|       |
  466|       |                    // Basic checks for invalid Jsonnet code.
  467|  1.52M|                    if (is_method && plus_sugar) {
  ------------------
  |  Branch (467:25): [True: 790k, False: 739k]
  |  Branch (467:38): [True: 1, False: 790k]
  ------------------
  468|      1|                        throw StaticError(next.location,
  469|      1|                                          "cannot use +: syntax sugar in a method: " + next.data);
  470|      1|                    }
  471|  1.52M|                    if (kind != ObjectField::FIELD_EXPR) {
  ------------------
  |  Branch (471:25): [True: 1.50M, False: 29.0k]
  ------------------
  472|  1.50M|                        if (!literal_fields.insert(next.data).second) {
  ------------------
  |  Branch (472:29): [True: 8, False: 1.50M]
  ------------------
  473|      8|                            throw StaticError(next.location, "duplicate field: " + next.data);
  474|      8|                        }
  475|  1.50M|                    }
  476|       |
  477|  1.52M|                    AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  478|       |
  479|  1.52M|                    Fodder comma_fodder;
  480|  1.52M|                    next = pop();
  481|  1.52M|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (481:25): [True: 1.28M, False: 247k]
  ------------------
  482|  1.28M|                        comma_fodder = next.fodder;
  483|  1.28M|                        next = pop();
  484|  1.28M|                        got_comma = true;
  485|  1.28M|                    }
  486|  1.52M|                    fields.emplace_back(kind,
  487|  1.52M|                                        fodder1,
  488|  1.52M|                                        fodder2,
  489|  1.52M|                                        fodder_l,
  490|  1.52M|                                        fodder_r,
  491|  1.52M|                                        field_hide,
  492|  1.52M|                                        plus_sugar,
  493|  1.52M|                                        is_method,
  494|  1.52M|                                        expr1,
  495|  1.52M|                                        id,
  496|  1.52M|                                        idLocation,
  497|  1.52M|                                        params,
  498|  1.52M|                                        meth_comma,
  499|  1.52M|                                        op.fodder,
  500|  1.52M|                                        body,
  501|  1.52M|                                        nullptr,
  502|  1.52M|                                        comma_fodder);
  503|  1.52M|                } break;
  504|       |
  505|  44.8k|                case Token::LOCAL: {
  ------------------
  |  Branch (505:17): [True: 44.8k, False: 1.56M]
  ------------------
  506|  44.8k|                    Fodder local_fodder = next.fodder;
  507|  44.8k|                    Token var_id = popExpect(Token::IDENTIFIER);
  508|  44.8k|                    auto *id = alloc->makeIdentifier(var_id.data32());
  509|       |
  510|  44.8k|                    if (binds.find(id) != binds.end()) {
  ------------------
  |  Branch (510:25): [True: 2, False: 44.8k]
  ------------------
  511|      2|                        throw StaticError(var_id.location, "duplicate local var: " + var_id.data);
  512|      2|                    }
  513|  44.8k|                    bool is_method = false;
  514|  44.8k|                    bool func_comma = false;
  515|  44.8k|                    ArgParams params;
  516|  44.8k|                    Fodder paren_l_fodder;
  517|  44.8k|                    Fodder paren_r_fodder;
  518|  44.8k|                    if (peek().kind == Token::PAREN_L) {
  ------------------
  |  Branch (518:25): [True: 6.63k, False: 38.1k]
  ------------------
  519|  6.63k|                        Token paren_l = pop();
  520|  6.63k|                        paren_l_fodder = paren_l.fodder;
  521|  6.63k|                        is_method = true;
  522|  6.63k|                        params = parseParams("function parameter", func_comma, paren_r_fodder, current_depth);
  523|  6.63k|                    }
  524|  44.8k|                    Token eq = popExpect(Token::OPERATOR, "=");
  525|  44.8k|                    AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  526|  44.8k|                    binds.insert(id);
  527|       |
  528|  44.8k|                    Fodder comma_fodder;
  529|  44.8k|                    next = pop();
  530|  44.8k|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (530:25): [True: 44.5k, False: 313]
  ------------------
  531|  44.5k|                        comma_fodder = next.fodder;
  532|  44.5k|                        next = pop();
  533|  44.5k|                        got_comma = true;
  534|  44.5k|                    }
  535|  44.8k|                    fields.push_back(ObjectField::Local(local_fodder,
  536|  44.8k|                                                        var_id.fodder,
  537|  44.8k|                                                        paren_l_fodder,
  538|  44.8k|                                                        paren_r_fodder,
  539|  44.8k|                                                        is_method,
  540|  44.8k|                                                        id,
  541|  44.8k|                                                        params,
  542|  44.8k|                                                        func_comma,
  543|  44.8k|                                                        eq.fodder,
  544|  44.8k|                                                        body,
  545|  44.8k|                                                        comma_fodder));
  546|       |
  547|  44.8k|                } break;
  548|       |
  549|  30.1k|                case Token::ASSERT: {
  ------------------
  |  Branch (549:17): [True: 30.1k, False: 1.57M]
  ------------------
  550|  30.1k|                    Fodder assert_fodder = next.fodder;
  551|  30.1k|                    AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  552|  30.1k|                    AST *msg = nullptr;
  553|  30.1k|                    Fodder colon_fodder;
  554|  30.1k|                    if (peek().kind == Token::OPERATOR && peek().data == ":") {
  ------------------
  |  Branch (554:25): [True: 775, False: 29.4k]
  |  Branch (554:59): [True: 774, False: 1]
  ------------------
  555|    774|                        Token colon = pop();
  556|    774|                        colon_fodder = colon.fodder;
  557|    774|                        msg = parse(MAX_PRECEDENCE, current_depth + 1);
  558|    774|                    }
  559|       |
  560|  30.1k|                    Fodder comma_fodder;
  561|  30.1k|                    next = pop();
  562|  30.1k|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (562:25): [True: 28.1k, False: 1.98k]
  ------------------
  563|  28.1k|                        comma_fodder = next.fodder;
  564|  28.1k|                        next = pop();
  565|  28.1k|                        got_comma = true;
  566|  28.1k|                    }
  567|  30.1k|                    fields.push_back(
  568|  30.1k|                        ObjectField::Assert(assert_fodder, cond, colon_fodder, msg, comma_fodder));
  569|  30.1k|                } break;
  570|       |
  571|     70|                default: throw unexpected(next, "parsing field definition");
  ------------------
  |  Branch (571:17): [True: 70, False: 1.60M]
  ------------------
  572|  1.60M|            }
  573|       |
  574|  1.60M|        } while (true);
  ------------------
  |  Branch (574:18): [True: 1.60M, Folded]
  ------------------
  575|   379k|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14spanERKNS0_5TokenES4_:
   89|  12.4M|{
   90|  12.4M|    return LocationRange(begin.location.file, begin.location.begin, end.location.end);
   91|  12.4M|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser23parseComprehensionSpecsENS0_5Token4KindENSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS0_17ComprehensionSpecENS8_ISB_EEEEj:
  588|   231k|    {
  589|   231k|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (589:13): [True: 0, False: 231k]
  ------------------
  590|      0|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  591|      0|        }
  592|       |
  593|   254k|        while (true) {
  ------------------
  |  Branch (593:16): [True: 254k, Folded]
  ------------------
  594|   254k|            LocationRange l;
  595|   254k|            Token id_token = popExpect(Token::IDENTIFIER);
  596|   254k|            const Identifier *id = alloc->makeIdentifier(id_token.data32());
  597|   254k|            Token in_token = popExpect(Token::IN);
  598|   254k|            AST *arr = parse(MAX_PRECEDENCE, current_depth + 1);
  599|   254k|            specs.emplace_back(
  600|   254k|                ComprehensionSpec::FOR, for_fodder, id_token.fodder, id, in_token.fodder, arr);
  601|       |
  602|   254k|            Token maybe_if = pop();
  603|   430k|            for (; maybe_if.kind == Token::IF; maybe_if = pop()) {
  ------------------
  |  Branch (603:20): [True: 176k, False: 254k]
  ------------------
  604|   176k|                AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  605|   176k|                specs.emplace_back(
  606|   176k|                    ComprehensionSpec::IF, maybe_if.fodder, Fodder{}, nullptr, Fodder{}, cond);
  607|   176k|            }
  608|   254k|            if (maybe_if.kind == end) {
  ------------------
  |  Branch (608:17): [True: 231k, False: 22.9k]
  ------------------
  609|   231k|                return maybe_if;
  610|   231k|            }
  611|  22.9k|            if (maybe_if.kind != Token::FOR) {
  ------------------
  |  Branch (611:17): [True: 24, False: 22.9k]
  ------------------
  612|     24|                std::stringstream ss;
  613|     24|                ss << "expected for, if or " << end << " after for clause, got: " << maybe_if;
  614|     24|                throw StaticError(maybe_if.location, ss.str());
  615|     24|            }
  616|  22.9k|            for_fodder = maybe_if.fodder;
  617|  22.9k|        }
  618|   231k|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14spanERKNS0_5TokenE:
   84|  26.9M|{
   85|  26.9M|    return LocationRange(begin.location.file, begin.location.begin, begin.location.end);
   86|  26.9M|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser10parseInfixEPNS0_3ASTERKNS0_5TokenEjj:
 1009|  28.8M|    {
 1010|  28.8M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (1010:13): [True: 0, False: 28.8M]
  ------------------
 1011|      0|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
 1012|      0|        }
 1013|       |
 1014|  45.2M|        while (true) {
  ------------------
  |  Branch (1014:16): [True: 45.2M, Folded]
  ------------------
 1015|       |
 1016|  45.2M|            BinaryOp bop = BOP_PLUS;
 1017|  45.2M|            unsigned op_precedence = 0;
 1018|       |
 1019|  45.2M|            switch (peek().kind) {
 1020|       |                // Logical / arithmetic binary operator.
 1021|  3.29k|                case Token::IN:
  ------------------
  |  Branch (1021:17): [True: 3.29k, False: 45.2M]
  ------------------
 1022|  7.00M|                case Token::OPERATOR:
  ------------------
  |  Branch (1022:17): [True: 7.00M, False: 38.2M]
  ------------------
 1023|       |                    // These occur if the outer statement was an assert or array slice.
 1024|       |                    // Either way, we terminate the parsing here.
 1025|  7.00M|                    if (peek().data == ":" || peek().data == "::") {
  ------------------
  |  Branch (1025:25): [True: 505k, False: 6.49M]
  |  Branch (1025:47): [True: 210, False: 6.49M]
  ------------------
 1026|   506k|                        return lhs;
 1027|   506k|                    }
 1028|  6.49M|                    if (!op_is_binary(peek().data, bop)) {
  ------------------
  |  Branch (1028:25): [True: 119, False: 6.49M]
  ------------------
 1029|    119|                        std::stringstream ss;
 1030|    119|                        ss << "not a binary operator: " << peek().data;
 1031|    119|                        throw StaticError(peek().location, ss.str());
 1032|    119|                    }
 1033|  6.49M|                    op_precedence = precedence_map[bop];
 1034|  6.49M|                    break;
 1035|       |
 1036|       |                // Index, Apply
 1037|  4.56M|                case Token::DOT:
  ------------------
  |  Branch (1037:17): [True: 4.56M, False: 40.6M]
  ------------------
 1038|  5.64M|                case Token::BRACKET_L:
  ------------------
  |  Branch (1038:17): [True: 1.07M, False: 44.1M]
  ------------------
 1039|  10.8M|                case Token::PAREN_L:
  ------------------
  |  Branch (1039:17): [True: 5.16M, False: 40.0M]
  ------------------
 1040|  10.8M|                case Token::BRACE_L:
  ------------------
  |  Branch (1040:17): [True: 60.4k, False: 45.1M]
  ------------------
 1041|  10.8M|                    op_precedence = APPLY_PRECEDENCE;
 1042|  10.8M|                    break;
 1043|       |
 1044|  27.3M|                default:
  ------------------
  |  Branch (1044:17): [True: 27.3M, False: 17.8M]
  ------------------
 1045|       |                    // This happens when we reach EOF or the terminating token of an outer context.
 1046|  27.3M|                    return lhs;
 1047|  45.2M|            }
 1048|       |
 1049|       |            // If higher precedence than the outer recursive call, let that handle it.
 1050|  17.3M|            if (op_precedence >= max_precedence)
  ------------------
  |  Branch (1050:17): [True: 983k, False: 16.3M]
  ------------------
 1051|   983k|                return lhs;
 1052|       |
 1053|  16.3M|            Token op = pop();
 1054|       |
 1055|  16.3M|            switch (op.kind) {
 1056|  1.07M|                case Token::BRACKET_L: {
  ------------------
  |  Branch (1056:17): [True: 1.07M, False: 15.3M]
  ------------------
 1057|  1.07M|                    bool is_slice;
 1058|  1.07M|                    AST *first = nullptr;
 1059|  1.07M|                    Fodder second_fodder;
 1060|  1.07M|                    AST *second = nullptr;
 1061|  1.07M|                    Fodder third_fodder;
 1062|  1.07M|                    AST *third = nullptr;
 1063|       |
 1064|  1.07M|                    if (peek().kind == Token::BRACKET_R)
  ------------------
  |  Branch (1064:25): [True: 3, False: 1.07M]
  ------------------
 1065|      3|                        throw unexpected(pop(), "parsing index");
 1066|       |
 1067|  1.07M|                    if (peek().data != ":" && peek().data != "::") {
  ------------------
  |  Branch (1067:25): [True: 1.05M, False: 18.3k]
  |  Branch (1067:47): [True: 1.05M, False: 669]
  ------------------
 1068|  1.05M|                        first = parse(MAX_PRECEDENCE, current_depth + 1);
 1069|  1.05M|                    }
 1070|       |
 1071|  1.07M|                    if (peek().kind == Token::OPERATOR && peek().data == "::") {
  ------------------
  |  Branch (1071:25): [True: 120k, False: 955k]
  |  Branch (1071:59): [True: 688, False: 120k]
  ------------------
 1072|       |                        // Handle ::
 1073|    688|                        is_slice = true;
 1074|    688|                        Token joined = pop();
 1075|    688|                        second_fodder = joined.fodder;
 1076|       |
 1077|    688|                        if (peek().kind != Token::BRACKET_R)
  ------------------
  |  Branch (1077:29): [True: 528, False: 160]
  ------------------
 1078|    528|                            third = parse(MAX_PRECEDENCE, current_depth + 1);
 1079|       |
 1080|  1.07M|                    } else if (peek().kind != Token::BRACKET_R) {
  ------------------
  |  Branch (1080:32): [True: 120k, False: 955k]
  ------------------
 1081|   120k|                        is_slice = true;
 1082|   120k|                        Token delim = pop();
 1083|   120k|                        if (delim.data != ":")
  ------------------
  |  Branch (1083:29): [True: 84, False: 120k]
  ------------------
 1084|     84|                            throw unexpected(delim, "parsing slice");
 1085|       |
 1086|   120k|                        second_fodder = delim.fodder;
 1087|       |
 1088|   120k|                        if (peek().data != ":" && peek().kind != Token::BRACKET_R)
  ------------------
  |  Branch (1088:29): [True: 120k, False: 136]
  |  Branch (1088:51): [True: 54.5k, False: 65.5k]
  ------------------
 1089|  54.5k|                            second = parse(MAX_PRECEDENCE, current_depth + 1);
 1090|       |
 1091|   120k|                        if (peek().kind != Token::BRACKET_R) {
  ------------------
  |  Branch (1091:29): [True: 8.24k, False: 112k]
  ------------------
 1092|  8.24k|                            Token delim = pop();
 1093|  8.24k|                            if (delim.data != ":")
  ------------------
  |  Branch (1093:33): [True: 34, False: 8.21k]
  ------------------
 1094|     34|                                throw unexpected(delim, "parsing slice");
 1095|       |
 1096|  8.21k|                            third_fodder = delim.fodder;
 1097|       |
 1098|  8.21k|                            if (peek().kind != Token::BRACKET_R)
  ------------------
  |  Branch (1098:33): [True: 7.44k, False: 771]
  ------------------
 1099|  7.44k|                                third = parse(MAX_PRECEDENCE, current_depth + 1);
 1100|  8.21k|                        }
 1101|   955k|                    } else {
 1102|   955k|                        is_slice = false;
 1103|   955k|                    }
 1104|  1.07M|                    Token end = popExpect(Token::BRACKET_R);
 1105|  1.07M|                    lhs = alloc->make<Index>(span(begin, end),
 1106|  1.07M|                                             EMPTY_FODDER,
 1107|  1.07M|                                             lhs,
 1108|  1.07M|                                             op.fodder,
 1109|  1.07M|                                             is_slice,
 1110|  1.07M|                                             first,
 1111|  1.07M|                                             second_fodder,
 1112|  1.07M|                                             second,
 1113|  1.07M|                                             third_fodder,
 1114|  1.07M|                                             third,
 1115|  1.07M|                                             end.fodder);
 1116|  1.07M|                    break;
 1117|  1.07M|                }
 1118|  4.56M|                case Token::DOT: {
  ------------------
  |  Branch (1118:17): [True: 4.56M, False: 11.8M]
  ------------------
 1119|  4.56M|                    Token field_id = popExpect(Token::IDENTIFIER);
 1120|  4.56M|                    const Identifier *id = alloc->makeIdentifier(field_id.data32());
 1121|  4.56M|                    lhs = alloc->make<Index>(span(begin, field_id),
 1122|  4.56M|                                             EMPTY_FODDER,
 1123|  4.56M|                                             lhs,
 1124|  4.56M|                                             op.fodder,
 1125|  4.56M|                                             field_id.fodder,
 1126|  4.56M|                                             id);
 1127|  4.56M|                    break;
 1128|  1.07M|                }
 1129|  5.16M|                case Token::PAREN_L: {
  ------------------
  |  Branch (1129:17): [True: 5.16M, False: 11.2M]
  ------------------
 1130|  5.16M|                    ArgParams args;
 1131|  5.16M|                    bool got_comma;
 1132|  5.16M|                    Token end = parseArgs(args, "function argument", got_comma, current_depth);
 1133|  5.16M|                    bool got_named = false;
 1134|  8.60M|                    for (const auto& arg : args) {
  ------------------
  |  Branch (1134:42): [True: 8.60M, False: 5.16M]
  ------------------
 1135|  8.60M|                        if (arg.id != nullptr) {
  ------------------
  |  Branch (1135:29): [True: 19.4k, False: 8.58M]
  ------------------
 1136|  19.4k|                            got_named = true;
 1137|  8.58M|                        } else {
 1138|  8.58M|                            if (got_named) {
  ------------------
  |  Branch (1138:33): [True: 2, False: 8.58M]
  ------------------
 1139|      2|                                throw StaticError(arg.expr->location, "Positional argument after a named argument is not allowed");
 1140|      2|                            }
 1141|  8.58M|                        }
 1142|  8.60M|                    }
 1143|  5.16M|                    bool tailstrict = false;
 1144|  5.16M|                    Fodder tailstrict_fodder;
 1145|  5.16M|                    if (peek().kind == Token::TAILSTRICT) {
  ------------------
  |  Branch (1145:25): [True: 254k, False: 4.90M]
  ------------------
 1146|   254k|                        Token tailstrict_token = pop();
 1147|   254k|                        tailstrict_fodder = tailstrict_token.fodder;
 1148|   254k|                        tailstrict = true;
 1149|   254k|                    }
 1150|  5.16M|                    lhs = alloc->make<Apply>(span(begin, end),
 1151|  5.16M|                                             EMPTY_FODDER,
 1152|  5.16M|                                             lhs,
 1153|  5.16M|                                             op.fodder,
 1154|  5.16M|                                             args,
 1155|  5.16M|                                             got_comma,
 1156|  5.16M|                                             end.fodder,
 1157|  5.16M|                                             tailstrict_fodder,
 1158|  5.16M|                                             tailstrict);
 1159|  5.16M|                    break;
 1160|  5.16M|                }
 1161|  60.4k|                case Token::BRACE_L: {
  ------------------
  |  Branch (1161:17): [True: 60.4k, False: 16.3M]
  ------------------
 1162|  60.4k|                    AST *obj;
 1163|  60.4k|                    Token end = parseObjectRemainder(obj, op, current_depth + 1);
 1164|  60.4k|                    lhs = alloc->make<ApplyBrace>(span(begin, end), EMPTY_FODDER, lhs, obj);
 1165|  60.4k|                    break;
 1166|  5.16M|                }
 1167|       |
 1168|  2.70k|                case Token::IN: {
  ------------------
  |  Branch (1168:17): [True: 2.70k, False: 16.3M]
  ------------------
 1169|  2.70k|                    if (peek().kind == Token::SUPER) {
  ------------------
  |  Branch (1169:25): [True: 480, False: 2.22k]
  ------------------
 1170|    480|                        Token super = pop();
 1171|    480|                        lhs = alloc->make<InSuper>(
 1172|    480|                            span(begin, super), EMPTY_FODDER, lhs, op.fodder, super.fodder);
 1173|  2.22k|                    } else {
 1174|  2.22k|                        AST *rhs = parse(op_precedence, current_depth + 1);
 1175|  2.22k|                        lhs = alloc->make<Binary>(
 1176|  2.22k|                            span(begin, rhs), EMPTY_FODDER, lhs, op.fodder, bop, rhs);
 1177|  2.22k|                    }
 1178|  2.70k|                    break;
 1179|  5.16M|                }
 1180|       |
 1181|  5.51M|                case Token::OPERATOR: {
  ------------------
  |  Branch (1181:17): [True: 5.51M, False: 10.8M]
  ------------------
 1182|  5.51M|                    AST *rhs = parse(op_precedence, current_depth + 1);
 1183|  5.51M|                    lhs = alloc->make<Binary>(
 1184|  5.51M|                        span(begin, rhs), EMPTY_FODDER, lhs, op.fodder, bop, rhs);
 1185|  5.51M|                    break;
 1186|  5.16M|                }
 1187|       |
 1188|      0|                default: {
  ------------------
  |  Branch (1188:17): [True: 0, False: 16.3M]
  ------------------
 1189|      0|                    std::cerr << "Should not be here." << std::endl;
 1190|      0|                    abort();
 1191|  5.16M|                }
 1192|  16.3M|            }
 1193|  16.3M|        }
 1194|       |
 1195|       |        // (1 & ((1 + (1 * 1)) + 1)) & 1
 1196|       |        //
 1197|       |        //
 1198|       |
 1199|       |/*
 1200|       |        // Allocate this on the heap to control stack growth.
 1201|       |        std::unique_ptr<Token> begin_(new Token(peek()));
 1202|       |        const Token &begin = *begin_;
 1203|       |*/
 1204|  28.8M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_112op_is_binaryERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERNS0_8BinaryOpE:
   75|  6.49M|{
   76|  6.49M|    auto it = binary_map.find(op);
   77|  6.49M|    if (it == binary_map.end())
  ------------------
  |  Branch (77:9): [True: 119, False: 6.49M]
  ------------------
   78|    119|        return false;
   79|  6.49M|    bop = it->second;
   80|  6.49M|    return true;
   81|  6.49M|}

_ZN7jsonnet8internal12CompilerPass6fodderERNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEE:
   22|  15.3M|{
   23|  15.3M|    for (auto &f : fodder)
  ------------------
  |  Branch (23:18): [True: 205k, False: 15.3M]
  ------------------
   24|   205k|        fodderElement(f);
   25|  15.3M|}
_ZN7jsonnet8internal12CompilerPass6paramsERNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEERNS3_INS0_8ArgParamENS5_IS9_EEEES8_:
   43|   313k|{
   44|   313k|    fodder(fodder_l);
   45|   507k|    for (auto &param : params) {
  ------------------
  |  Branch (45:22): [True: 507k, False: 313k]
  ------------------
   46|   507k|        fodder(param.idFodder);
   47|   507k|        if (param.expr) {
  ------------------
  |  Branch (47:13): [True: 423k, False: 83.7k]
  ------------------
   48|   423k|            fodder(param.eqFodder);
   49|   423k|            expr(param.expr);
   50|   423k|        }
   51|   507k|        fodder(param.commaFodder);
   52|   507k|    }
   53|   313k|    fodder(fodder_r);
   54|   313k|}
_ZN7jsonnet8internal12CompilerPass4exprERPNS0_3ASTE:
  110|  8.82M|{
  111|  8.82M|    fodder(ast_->openFodder);
  112|  8.82M|    visitExpr(ast_);
  113|  8.82M|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5ApplyE:
  116|   267k|{
  117|   267k|    expr(ast->target);
  118|   267k|    params(ast->fodderL, ast->args, ast->fodderR);
  119|   267k|    if (ast->tailstrict) {
  ------------------
  |  Branch (119:9): [True: 206k, False: 61.3k]
  ------------------
  120|   206k|        fodder(ast->tailstrictFodder);
  121|   206k|    }
  122|   267k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5ArrayE:
  131|   175k|{
  132|   181k|    for (auto &element : ast->elements) {
  ------------------
  |  Branch (132:24): [True: 181k, False: 175k]
  ------------------
  133|   181k|        expr(element.expr);
  134|   181k|        fodder(element.commaFodder);
  135|   181k|    }
  136|   175k|    fodder(ast->closeFodder);
  137|   175k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_6BinaryE:
  159|   943k|{
  160|   943k|    expr(ast->left);
  161|   943k|    fodder(ast->opFodder);
  162|   943k|    expr(ast->right);
  163|   943k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_11ConditionalE:
  166|   542k|{
  167|   542k|    expr(ast->cond);
  168|   542k|    fodder(ast->thenFodder);
  169|   542k|    if (ast->branchFalse != nullptr) {
  ------------------
  |  Branch (169:9): [True: 542k, False: 0]
  ------------------
  170|   542k|        expr(ast->branchTrue);
  171|   542k|        fodder(ast->elseFodder);
  172|   542k|        expr(ast->branchFalse);
  173|   542k|    } else {
  174|      0|        expr(ast->branchTrue);
  175|      0|    }
  176|   542k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5ErrorE:
  179|  62.0k|{
  180|  62.0k|    expr(ast->expr);
  181|  62.0k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_8FunctionE:
  184|  46.2k|{
  185|  46.2k|    params(ast->parenLeftFodder, ast->params, ast->parenRightFodder);
  186|  46.2k|    expr(ast->body);
  187|  46.2k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_6ImportE:
  190|    485|{
  191|    485|    visit(ast->file);
  192|    485|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_9ImportstrE:
  195|    337|{
  196|    337|    visit(ast->file);
  197|    337|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_9ImportbinE:
  200|    268|{
  201|    268|    visit(ast->file);
  202|    268|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_7InSuperE:
  205|   293k|{
  206|   293k|    expr(ast->element);
  207|   293k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5IndexE:
  210|   293k|{
  211|   293k|    expr(ast->target);
  212|   293k|    if (ast->id != nullptr) {
  ------------------
  |  Branch (212:9): [True: 0, False: 293k]
  ------------------
  213|   293k|    } else {
  214|   293k|        if (ast->isSlice) {
  ------------------
  |  Branch (214:13): [True: 0, False: 293k]
  ------------------
  215|      0|            if (ast->index != nullptr)
  ------------------
  |  Branch (215:17): [True: 0, False: 0]
  ------------------
  216|      0|                expr(ast->index);
  217|      0|            if (ast->end != nullptr)
  ------------------
  |  Branch (217:17): [True: 0, False: 0]
  ------------------
  218|      0|                expr(ast->end);
  219|      0|            if (ast->step != nullptr)
  ------------------
  |  Branch (219:17): [True: 0, False: 0]
  ------------------
  220|      0|                expr(ast->step);
  221|   293k|        } else {
  222|   293k|            expr(ast->index);
  223|   293k|        }
  224|   293k|    }
  225|   293k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5LocalE:
  228|   183k|{
  229|   183k|    assert(ast->binds.size() > 0);
  ------------------
  |  Branch (229:5): [True: 183k, False: 0]
  ------------------
  230|   634k|    for (auto &bind : ast->binds) {
  ------------------
  |  Branch (230:21): [True: 634k, False: 183k]
  ------------------
  231|   634k|        fodder(bind.varFodder);
  232|   634k|        if (bind.functionSugar) {
  ------------------
  |  Branch (232:13): [True: 0, False: 634k]
  ------------------
  233|      0|            params(bind.parenLeftFodder, bind.params, bind.parenRightFodder);
  234|      0|        }
  235|   634k|        fodder(bind.opFodder);
  236|   634k|        expr(bind.body);
  237|   634k|        fodder(bind.closeFodder);
  238|   634k|    }
  239|   183k|    expr(ast->body);
  240|   183k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_15DesugaredObjectE:
  249|   781k|{
  250|   781k|    for (AST *assert : ast->asserts) {
  ------------------
  |  Branch (250:22): [True: 19.6k, False: 781k]
  ------------------
  251|  19.6k|        expr(assert);
  252|  19.6k|    }
  253|   781k|    for (auto &field : ast->fields) {
  ------------------
  |  Branch (253:22): [True: 502k, False: 781k]
  ------------------
  254|   502k|        expr(field.name);
  255|   502k|        expr(field.body);
  256|   502k|    }
  257|   781k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_25ObjectComprehensionSimpleE:
  267|  34.5k|{
  268|  34.5k|    expr(ast->field);
  269|  34.5k|    expr(ast->value);
  270|  34.5k|    expr(ast->array);
  271|  34.5k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_10SuperIndexE:
  280|   310k|{
  281|   310k|    if (ast->id != nullptr) {
  ------------------
  |  Branch (281:9): [True: 0, False: 310k]
  ------------------
  282|   310k|    } else {
  283|   310k|        expr(ast->index);
  284|   310k|    }
  285|   310k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5UnaryE:
  288|  1.15M|{
  289|  1.15M|    expr(ast->expr);
  290|  1.15M|}
_ZN7jsonnet8internal12CompilerPass9visitExprERPNS0_3ASTE:
  300|  8.82M|{
  301|  8.82M|    switch(ast_->type) {
  302|   267k|        VISIT(ast_, AST_APPLY, Apply);
  ------------------
  |  |  293|   267k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 267k, False: 8.56M]
  |  |  ------------------
  |  |  294|   267k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   267k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   267k|     visit(ast); \
  |  |  297|   267k|   } break
  ------------------
  |  Branch (302:9): [True: 267k, False: 0]
  ------------------
  303|      0|        VISIT(ast_, AST_APPLY_BRACE, ApplyBrace);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (303:9): [True: 0, False: 0]
  ------------------
  304|   175k|        VISIT(ast_, AST_ARRAY, Array);
  ------------------
  |  |  293|   175k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 175k, False: 8.65M]
  |  |  ------------------
  |  |  294|   175k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   175k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   175k|     visit(ast); \
  |  |  297|   175k|   } break
  ------------------
  |  Branch (304:9): [True: 175k, False: 0]
  ------------------
  305|      0|        VISIT(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehension);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (305:9): [True: 0, False: 0]
  ------------------
  306|       |        // VISIT(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehensionSimple);
  307|      0|        VISIT(ast_, AST_ASSERT, Assert);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (307:9): [True: 0, False: 0]
  ------------------
  308|   943k|        VISIT(ast_, AST_BINARY, Binary);
  ------------------
  |  |  293|   943k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 943k, False: 7.88M]
  |  |  ------------------
  |  |  294|   943k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   943k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   943k|     visit(ast); \
  |  |  297|   943k|   } break
  ------------------
  |  Branch (308:9): [True: 943k, False: 0]
  ------------------
  309|      0|        VISIT(ast_, AST_BUILTIN_FUNCTION, BuiltinFunction);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (309:9): [True: 0, False: 0]
  ------------------
  310|      0|        VISIT(ast_, AST_BUILTIN_FUNCTION_BODY, BuiltinFunctionBody);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (310:9): [True: 0, False: 0]
  ------------------
  311|   542k|        VISIT(ast_, AST_CONDITIONAL, Conditional);
  ------------------
  |  |  293|   542k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 542k, False: 8.28M]
  |  |  ------------------
  |  |  294|   542k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   542k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   542k|     visit(ast); \
  |  |  297|   542k|   } break
  ------------------
  |  Branch (311:9): [True: 542k, False: 0]
  ------------------
  312|   781k|        VISIT(ast_, AST_DESUGARED_OBJECT, DesugaredObject);
  ------------------
  |  |  293|   781k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 781k, False: 8.04M]
  |  |  ------------------
  |  |  294|   781k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   781k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   781k|     visit(ast); \
  |  |  297|   781k|   } break
  ------------------
  |  Branch (312:9): [True: 781k, False: 0]
  ------------------
  313|      0|        VISIT(ast_, AST_DOLLAR, Dollar);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (313:9): [True: 0, False: 0]
  ------------------
  314|  62.0k|        VISIT(ast_, AST_ERROR, Error);
  ------------------
  |  |  293|  62.0k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 62.0k, False: 8.76M]
  |  |  ------------------
  |  |  294|  62.0k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  62.0k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  62.0k|     visit(ast); \
  |  |  297|  62.0k|   } break
  ------------------
  |  Branch (314:9): [True: 62.0k, False: 0]
  ------------------
  315|  46.2k|        VISIT(ast_, AST_FUNCTION, Function);
  ------------------
  |  |  293|  46.2k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 46.2k, False: 8.78M]
  |  |  ------------------
  |  |  294|  46.2k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  46.2k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  46.2k|     visit(ast); \
  |  |  297|  46.2k|   } break
  ------------------
  |  Branch (315:9): [True: 46.2k, False: 0]
  ------------------
  316|    485|        VISIT(ast_, AST_IMPORT, Import);
  ------------------
  |  |  293|    485|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 485, False: 8.82M]
  |  |  ------------------
  |  |  294|    485|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|    485|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|    485|     visit(ast); \
  |  |  297|    485|   } break
  ------------------
  |  Branch (316:9): [True: 485, False: 0]
  ------------------
  317|    337|        VISIT(ast_, AST_IMPORTSTR, Importstr);
  ------------------
  |  |  293|    337|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 337, False: 8.82M]
  |  |  ------------------
  |  |  294|    337|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|    337|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|    337|     visit(ast); \
  |  |  297|    337|   } break
  ------------------
  |  Branch (317:9): [True: 337, False: 0]
  ------------------
  318|    268|        VISIT(ast_, AST_IMPORTBIN, Importbin);
  ------------------
  |  |  293|    268|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 268, False: 8.82M]
  |  |  ------------------
  |  |  294|    268|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|    268|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|    268|     visit(ast); \
  |  |  297|    268|   } break
  ------------------
  |  Branch (318:9): [True: 268, False: 0]
  ------------------
  319|   293k|        VISIT(ast_, AST_INDEX, Index);
  ------------------
  |  |  293|   293k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 293k, False: 8.53M]
  |  |  ------------------
  |  |  294|   293k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   293k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   293k|     visit(ast); \
  |  |  297|   293k|   } break
  ------------------
  |  Branch (319:9): [True: 293k, False: 0]
  ------------------
  320|   293k|        VISIT(ast_, AST_IN_SUPER, InSuper);
  ------------------
  |  |  293|   293k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 293k, False: 8.53M]
  |  |  ------------------
  |  |  294|   293k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   293k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   293k|     visit(ast); \
  |  |  297|   293k|   } break
  ------------------
  |  Branch (320:9): [True: 293k, False: 0]
  ------------------
  321|  20.1k|        VISIT(ast_, AST_LITERAL_BOOLEAN, LiteralBoolean);
  ------------------
  |  |  293|  20.1k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 20.1k, False: 8.80M]
  |  |  ------------------
  |  |  294|  20.1k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  20.1k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  20.1k|     visit(ast); \
  |  |  297|  20.1k|   } break
  ------------------
  |  Branch (321:9): [True: 20.1k, False: 0]
  ------------------
  322|  14.5k|        VISIT(ast_, AST_LITERAL_NULL, LiteralNull);
  ------------------
  |  |  293|  14.5k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 14.5k, False: 8.81M]
  |  |  ------------------
  |  |  294|  14.5k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  14.5k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  14.5k|     visit(ast); \
  |  |  297|  14.5k|   } break
  ------------------
  |  Branch (322:9): [True: 14.5k, False: 0]
  ------------------
  323|   425k|        VISIT(ast_, AST_LITERAL_NUMBER, LiteralNumber);
  ------------------
  |  |  293|   425k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 425k, False: 8.40M]
  |  |  ------------------
  |  |  294|   425k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   425k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   425k|     visit(ast); \
  |  |  297|   425k|   } break
  ------------------
  |  Branch (323:9): [True: 425k, False: 0]
  ------------------
  324|  1.45M|        VISIT(ast_, AST_LITERAL_STRING, LiteralString);
  ------------------
  |  |  293|  1.45M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.45M, False: 7.37M]
  |  |  ------------------
  |  |  294|  1.45M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.45M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.45M|     visit(ast); \
  |  |  297|  1.45M|   } break
  ------------------
  |  Branch (324:9): [True: 1.45M, False: 0]
  ------------------
  325|   183k|        VISIT(ast_, AST_LOCAL, Local);
  ------------------
  |  |  293|   183k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 183k, False: 8.64M]
  |  |  ------------------
  |  |  294|   183k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   183k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   183k|     visit(ast); \
  |  |  297|   183k|   } break
  ------------------
  |  Branch (325:9): [True: 183k, False: 0]
  ------------------
  326|      0|        VISIT(ast_, AST_OBJECT, Object);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (326:9): [True: 0, False: 0]
  ------------------
  327|      0|        VISIT(ast_, AST_OBJECT_COMPREHENSION, ObjectComprehension);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (327:9): [True: 0, False: 0]
  ------------------
  328|  34.5k|        VISIT(ast_, AST_OBJECT_COMPREHENSION_SIMPLE, ObjectComprehensionSimple);
  ------------------
  |  |  293|  34.5k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 34.5k, False: 8.79M]
  |  |  ------------------
  |  |  294|  34.5k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  34.5k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  34.5k|     visit(ast); \
  |  |  297|  34.5k|   } break
  ------------------
  |  Branch (328:9): [True: 34.5k, False: 0]
  ------------------
  329|      0|        VISIT(ast_, AST_PARENS, Parens);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 8.82M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (329:9): [True: 0, False: 0]
  ------------------
  330|  8.27k|        VISIT(ast_, AST_SELF, Self);
  ------------------
  |  |  293|  8.27k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 8.27k, False: 8.82M]
  |  |  ------------------
  |  |  294|  8.27k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  8.27k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  8.27k|     visit(ast); \
  |  |  297|  8.27k|   } break
  ------------------
  |  Branch (330:9): [True: 8.27k, False: 0]
  ------------------
  331|   310k|        VISIT(ast_, AST_SUPER_INDEX, SuperIndex);
  ------------------
  |  |  293|   310k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 310k, False: 8.51M]
  |  |  ------------------
  |  |  294|   310k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   310k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   310k|     visit(ast); \
  |  |  297|   310k|   } break
  ------------------
  |  Branch (331:9): [True: 310k, False: 0]
  ------------------
  332|  1.15M|        VISIT(ast_, AST_UNARY, Unary);
  ------------------
  |  |  293|  1.15M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.15M, False: 7.67M]
  |  |  ------------------
  |  |  294|  1.15M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.15M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.15M|     visit(ast); \
  |  |  297|  1.15M|   } break
  ------------------
  |  Branch (332:9): [True: 1.15M, False: 0]
  ------------------
  333|  1.81M|        VISIT(ast_, AST_VAR, Var);
  ------------------
  |  |  293|  1.81M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.81M, False: 7.01M]
  |  |  ------------------
  |  |  294|  1.81M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.81M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.81M|     visit(ast); \
  |  |  297|  1.81M|   } break
  ------------------
  |  Branch (333:9): [True: 1.81M, False: 0]
  ------------------
  334|      0|        default:
  ------------------
  |  Branch (334:9): [True: 0, False: 8.82M]
  ------------------
  335|      0|            std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  336|      0|            std::abort();
  337|      0|            break;
  338|  8.82M|    }
  339|  8.82M|}
_ZN7jsonnet8internal9ClonePass4exprERPNS0_3ASTE:
  362|  8.03M|{
  363|  8.03M|    switch(ast_->type) {
  364|   241k|        CLONE(ast_, AST_APPLY, Apply);
  ------------------
  |  |  355|   241k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 241k, False: 7.79M]
  |  |  ------------------
  |  |  356|   241k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   241k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   241k|     var = alloc.clone(ast); \
  |  |  359|   241k|   } break
  ------------------
  |  Branch (364:9): [True: 241k, False: 0]
  ------------------
  365|      0|        CLONE(ast_, AST_APPLY_BRACE, ApplyBrace);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (365:9): [True: 0, False: 0]
  ------------------
  366|   159k|        CLONE(ast_, AST_ARRAY, Array);
  ------------------
  |  |  355|   159k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 159k, False: 7.87M]
  |  |  ------------------
  |  |  356|   159k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   159k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   159k|     var = alloc.clone(ast); \
  |  |  359|   159k|   } break
  ------------------
  |  Branch (366:9): [True: 159k, False: 0]
  ------------------
  367|      0|        CLONE(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehension);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (367:9): [True: 0, False: 0]
  ------------------
  368|       |        // CLONE(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehensionSimple);
  369|      0|        CLONE(ast_, AST_ASSERT, Assert);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (369:9): [True: 0, False: 0]
  ------------------
  370|   858k|        CLONE(ast_, AST_BINARY, Binary);
  ------------------
  |  |  355|   858k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 858k, False: 7.17M]
  |  |  ------------------
  |  |  356|   858k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   858k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   858k|     var = alloc.clone(ast); \
  |  |  359|   858k|   } break
  ------------------
  |  Branch (370:9): [True: 858k, False: 0]
  ------------------
  371|      0|        CLONE(ast_, AST_BUILTIN_FUNCTION, BuiltinFunction);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (371:9): [True: 0, False: 0]
  ------------------
  372|   492k|        CLONE(ast_, AST_CONDITIONAL, Conditional);
  ------------------
  |  |  355|   492k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 492k, False: 7.54M]
  |  |  ------------------
  |  |  356|   492k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   492k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   492k|     var = alloc.clone(ast); \
  |  |  359|   492k|   } break
  ------------------
  |  Branch (372:9): [True: 492k, False: 0]
  ------------------
  373|   701k|        CLONE(ast_, AST_DESUGARED_OBJECT, DesugaredObject);
  ------------------
  |  |  355|   701k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 701k, False: 7.33M]
  |  |  ------------------
  |  |  356|   701k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   701k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   701k|     var = alloc.clone(ast); \
  |  |  359|   701k|   } break
  ------------------
  |  Branch (373:9): [True: 701k, False: 0]
  ------------------
  374|      0|        CLONE(ast_, AST_DOLLAR, Dollar);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (374:9): [True: 0, False: 0]
  ------------------
  375|  56.1k|        CLONE(ast_, AST_ERROR, Error);
  ------------------
  |  |  355|  56.1k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 56.1k, False: 7.98M]
  |  |  ------------------
  |  |  356|  56.1k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  56.1k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  56.1k|     var = alloc.clone(ast); \
  |  |  359|  56.1k|   } break
  ------------------
  |  Branch (375:9): [True: 56.1k, False: 0]
  ------------------
  376|  41.5k|        CLONE(ast_, AST_FUNCTION, Function);
  ------------------
  |  |  355|  41.5k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 41.5k, False: 7.99M]
  |  |  ------------------
  |  |  356|  41.5k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  41.5k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  41.5k|     var = alloc.clone(ast); \
  |  |  359|  41.5k|   } break
  ------------------
  |  Branch (376:9): [True: 41.5k, False: 0]
  ------------------
  377|    457|        CLONE(ast_, AST_IMPORT, Import);
  ------------------
  |  |  355|    457|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 457, False: 8.03M]
  |  |  ------------------
  |  |  356|    457|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|    457|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|    457|     var = alloc.clone(ast); \
  |  |  359|    457|   } break
  ------------------
  |  Branch (377:9): [True: 457, False: 0]
  ------------------
  378|    337|        CLONE(ast_, AST_IMPORTSTR, Importstr);
  ------------------
  |  |  355|    337|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 337, False: 8.03M]
  |  |  ------------------
  |  |  356|    337|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|    337|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|    337|     var = alloc.clone(ast); \
  |  |  359|    337|   } break
  ------------------
  |  Branch (378:9): [True: 337, False: 0]
  ------------------
  379|    268|        CLONE(ast_, AST_IMPORTBIN, Importbin);
  ------------------
  |  |  355|    268|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 268, False: 8.03M]
  |  |  ------------------
  |  |  356|    268|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|    268|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|    268|     var = alloc.clone(ast); \
  |  |  359|    268|   } break
  ------------------
  |  Branch (379:9): [True: 268, False: 0]
  ------------------
  380|   261k|        CLONE(ast_, AST_INDEX, Index);
  ------------------
  |  |  355|   261k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 261k, False: 7.77M]
  |  |  ------------------
  |  |  356|   261k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   261k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   261k|     var = alloc.clone(ast); \
  |  |  359|   261k|   } break
  ------------------
  |  Branch (380:9): [True: 261k, False: 0]
  ------------------
  381|   293k|        CLONE(ast_, AST_IN_SUPER, InSuper);
  ------------------
  |  |  355|   293k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 293k, False: 7.74M]
  |  |  ------------------
  |  |  356|   293k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   293k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   293k|     var = alloc.clone(ast); \
  |  |  359|   293k|   } break
  ------------------
  |  Branch (381:9): [True: 293k, False: 0]
  ------------------
  382|  19.8k|        CLONE(ast_, AST_LITERAL_BOOLEAN, LiteralBoolean);
  ------------------
  |  |  355|  19.8k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 19.8k, False: 8.01M]
  |  |  ------------------
  |  |  356|  19.8k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  19.8k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  19.8k|     var = alloc.clone(ast); \
  |  |  359|  19.8k|   } break
  ------------------
  |  Branch (382:9): [True: 19.8k, False: 0]
  ------------------
  383|  14.5k|        CLONE(ast_, AST_LITERAL_NULL, LiteralNull);
  ------------------
  |  |  355|  14.5k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 14.5k, False: 8.02M]
  |  |  ------------------
  |  |  356|  14.5k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  14.5k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  14.5k|     var = alloc.clone(ast); \
  |  |  359|  14.5k|   } break
  ------------------
  |  Branch (383:9): [True: 14.5k, False: 0]
  ------------------
  384|   386k|        CLONE(ast_, AST_LITERAL_NUMBER, LiteralNumber);
  ------------------
  |  |  355|   386k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 386k, False: 7.65M]
  |  |  ------------------
  |  |  356|   386k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   386k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   386k|     var = alloc.clone(ast); \
  |  |  359|   386k|   } break
  ------------------
  |  Branch (384:9): [True: 386k, False: 0]
  ------------------
  385|  1.37M|        CLONE(ast_, AST_LITERAL_STRING, LiteralString);
  ------------------
  |  |  355|  1.37M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 1.37M, False: 6.66M]
  |  |  ------------------
  |  |  356|  1.37M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  1.37M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  1.37M|     var = alloc.clone(ast); \
  |  |  359|  1.37M|   } break
  ------------------
  |  Branch (385:9): [True: 1.37M, False: 0]
  ------------------
  386|   166k|        CLONE(ast_, AST_LOCAL, Local);
  ------------------
  |  |  355|   166k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 166k, False: 7.87M]
  |  |  ------------------
  |  |  356|   166k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   166k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   166k|     var = alloc.clone(ast); \
  |  |  359|   166k|   } break
  ------------------
  |  Branch (386:9): [True: 166k, False: 0]
  ------------------
  387|      0|        CLONE(ast_, AST_OBJECT, Object);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (387:9): [True: 0, False: 0]
  ------------------
  388|      0|        CLONE(ast_, AST_OBJECT_COMPREHENSION, ObjectComprehension);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (388:9): [True: 0, False: 0]
  ------------------
  389|  29.9k|        CLONE(ast_, AST_OBJECT_COMPREHENSION_SIMPLE, ObjectComprehensionSimple);
  ------------------
  |  |  355|  29.9k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 29.9k, False: 8.00M]
  |  |  ------------------
  |  |  356|  29.9k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  29.9k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  29.9k|     var = alloc.clone(ast); \
  |  |  359|  29.9k|   } break
  ------------------
  |  Branch (389:9): [True: 29.9k, False: 0]
  ------------------
  390|      0|        CLONE(ast_, AST_PARENS, Parens);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 8.03M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (390:9): [True: 0, False: 0]
  ------------------
  391|  8.27k|        CLONE(ast_, AST_SELF, Self);
  ------------------
  |  |  355|  8.27k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 8.27k, False: 8.02M]
  |  |  ------------------
  |  |  356|  8.27k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  8.27k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  8.27k|     var = alloc.clone(ast); \
  |  |  359|  8.27k|   } break
  ------------------
  |  Branch (391:9): [True: 8.27k, False: 0]
  ------------------
  392|   310k|        CLONE(ast_, AST_SUPER_INDEX, SuperIndex);
  ------------------
  |  |  355|   310k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 310k, False: 7.72M]
  |  |  ------------------
  |  |  356|   310k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   310k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   310k|     var = alloc.clone(ast); \
  |  |  359|   310k|   } break
  ------------------
  |  Branch (392:9): [True: 310k, False: 0]
  ------------------
  393|  1.03M|        CLONE(ast_, AST_UNARY, Unary);
  ------------------
  |  |  355|  1.03M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 1.03M, False: 7.00M]
  |  |  ------------------
  |  |  356|  1.03M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  1.03M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  1.03M|     var = alloc.clone(ast); \
  |  |  359|  1.03M|   } break
  ------------------
  |  Branch (393:9): [True: 1.03M, False: 0]
  ------------------
  394|  1.58M|        CLONE(ast_, AST_VAR, Var);
  ------------------
  |  |  355|  1.58M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 1.58M, False: 6.45M]
  |  |  ------------------
  |  |  356|  1.58M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  1.58M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  1.58M|     var = alloc.clone(ast); \
  |  |  359|  1.58M|   } break
  ------------------
  |  Branch (394:9): [True: 1.58M, False: 0]
  ------------------
  395|      0|        default:
  ------------------
  |  Branch (395:9): [True: 0, False: 8.03M]
  ------------------
  396|      0|            std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  397|      0|            std::abort();
  398|      0|            break;
  399|  8.03M|    }
  400|       |
  401|  8.03M|    CompilerPass::expr(ast_);
  402|  8.03M|}
_ZN7jsonnet8internal9clone_astERNS0_9AllocatorEPNS0_3ASTE:
  405|  31.6k|{
  406|  31.6k|    AST *r = ast;
  407|  31.6k|    ClonePass(alloc).expr(r);
  408|  31.6k|    return r;
  409|  31.6k|}
_ZN7jsonnet8internal9ClonePassC2ERNS0_9AllocatorE:
  350|  31.6k|    ClonePass(Allocator &alloc) : CompilerPass(alloc) {}

_ZN7jsonnet8internal12CompilerPassC2ERNS0_9AllocatorE:
   32|  42.2k|    CompilerPass(Allocator &alloc) : alloc(alloc) {}
_ZN7jsonnet8internal12CompilerPass13fodderElementERNS0_13FodderElementE:
   34|   205k|    virtual void fodderElement(FodderElement &) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_14LiteralBooleanE:
   84|  20.1k|    virtual void visit(LiteralBoolean *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_13LiteralNumberE:
   86|   425k|    virtual void visit(LiteralNumber *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_11LiteralNullE:
   90|  14.5k|    virtual void visit(LiteralNull *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_4SelfE:
  102|  8.27k|    virtual void visit(Self *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_3VarE:
  108|  1.81M|    virtual void visit(Var *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_13LiteralStringE:
   88|  1.45M|    virtual void visit(LiteralString *) {}

_ZN7jsonnet8internal32path_dir_with_trailing_separatorERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   31|     20|std::string path_dir_with_trailing_separator(const std::string &path) {
   32|     20|    size_t last_slash = path.find_last_of(DIR_SEPARATORS);
   33|     20|    if (last_slash != std::string::npos) {
  ------------------
  |  Branch (33:9): [True: 0, False: 20]
  ------------------
   34|      0|        return path.substr(0, last_slash + 1);
   35|      0|    }
   36|     20|    return "";
   37|     20|}

vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14HeapC2Ejd:
  346|  2.88k|        : gcTuneMinObjects(gc_tune_min_objects),
  347|  2.88k|          gcTuneGrowthTrigger(gc_tune_growth_trigger),
  348|  2.88k|          lastMark(0),
  349|  2.88k|          lastNumEntities(0),
  350|  2.88k|          numEntities(0)
  351|  2.88k|    {
  352|  2.88k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_18type_strENS1_5Value4TypeE:
   76|    328|{
   77|    328|    switch (t) {
   78|      2|        case Value::NULL_TYPE: return "null";
  ------------------
  |  Branch (78:9): [True: 2, False: 326]
  ------------------
   79|     54|        case Value::BOOLEAN: return "boolean";
  ------------------
  |  Branch (79:9): [True: 54, False: 274]
  ------------------
   80|     85|        case Value::NUMBER: return "number";
  ------------------
  |  Branch (80:9): [True: 85, False: 243]
  ------------------
   81|     17|        case Value::ARRAY: return "array";
  ------------------
  |  Branch (81:9): [True: 17, False: 311]
  ------------------
   82|      1|        case Value::FUNCTION: return "function";
  ------------------
  |  Branch (82:9): [True: 1, False: 327]
  ------------------
   83|     91|        case Value::OBJECT: return "object";
  ------------------
  |  Branch (83:9): [True: 91, False: 237]
  ------------------
   84|     78|        case Value::STRING: return "string";
  ------------------
  |  Branch (84:9): [True: 78, False: 250]
  ------------------
   85|      0|        default:
  ------------------
  |  Branch (85:9): [True: 0, False: 328]
  ------------------
   86|      0|            std::cerr << "INTERNAL ERROR: Unknown type: " << t << std::endl;
   87|      0|            std::abort();
   88|      0|            return "";  // Quiet, compiler.
   89|    328|    }
   90|    328|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_18type_strERKNS1_5ValueE:
   94|    328|{
   95|    328|    return type_str(v.t);
   96|    328|}
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_15Value6isHeapEv:
   69|  58.7M|    {
   70|  58.7M|        return t & 0x10;
   71|  58.7M|    }
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_111HeapClosure9isBuiltinEv:
  293|  37.1M|    bool isBuiltin() const { return !this->body || this->body->type == AST_BUILTIN_FUNCTION_BODY; }
  ------------------
  |  Branch (293:37): [True: 0, False: 37.1M]
  |  Branch (293:52): [True: 25.5M, False: 11.5M]
  ------------------
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_19HeapThunkC2EPKNS0_10IdentifierEPNS1_10HeapObjectEjPKNS0_3ASTE:
  140|  49.4M|        : HeapEntity(THUNK), filled(false), name(name), self(self), offset(offset), body(body)
  141|  49.4M|    {
  142|  49.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapEntityC2ENS2_4TypeE:
   43|   105M|    HeapEntity(Type type_) : type(type_) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapEntityD2Ev:
   44|   105M|    virtual ~HeapEntity() {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap9checkHeapEv:
  489|   105M|    {
  490|   105M|        return numEntities > gcTuneMinObjects &&
  ------------------
  |  Branch (490:16): [True: 31.1M, False: 74.3M]
  ------------------
  491|  31.1M|               numEntities > gcTuneGrowthTrigger * lastNumEntities;
  ------------------
  |  Branch (491:16): [True: 126k, False: 31.0M]
  ------------------
  492|   105M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap8markFromEPNS1_10HeapEntityE:
  369|  91.0M|    {
  370|  91.0M|        assert(from != nullptr);
  ------------------
  |  Branch (370:9): [True: 91.0M, False: 0]
  ------------------
  371|  91.0M|        const GarbageCollectionMark thisMark = lastMark + 1;
  372|  91.0M|        struct State {
  373|  91.0M|            HeapEntity *ent;
  374|  91.0M|            std::vector<HeapEntity *> children;
  375|  91.0M|            State(HeapEntity *ent) : ent(ent) {}
  376|  91.0M|        };
  377|       |
  378|  91.0M|        std::vector<State> stack;
  379|  91.0M|        stack.emplace_back(from);
  380|       |
  381|   352M|        while (stack.size() > 0) {
  ------------------
  |  Branch (381:16): [True: 261M, False: 91.0M]
  ------------------
  382|   261M|            size_t curr_index = stack.size() - 1;
  383|   261M|            State &s = stack[curr_index];
  384|   261M|            HeapEntity *curr = s.ent;
  385|   261M|            if (curr->mark != thisMark) {
  ------------------
  |  Branch (385:17): [True: 65.9M, False: 195M]
  ------------------
  386|  65.9M|                curr->mark = thisMark;
  387|       |
  388|  65.9M|                switch(curr->type) {
  389|  8.00M|                    case HeapEntity::SIMPLE_OBJECT: {
  ------------------
  |  Branch (389:21): [True: 8.00M, False: 57.9M]
  ------------------
  390|  8.00M|                        assert(dynamic_cast<HeapSimpleObject *>(curr));
  ------------------
  |  Branch (390:25): [True: 8.00M, False: 0]
  ------------------
  391|  8.00M|                        auto *obj = static_cast<HeapSimpleObject *>(curr);
  392|  8.00M|                        for (auto upv : obj->upValues)
  ------------------
  |  Branch (392:39): [True: 6.51M, False: 8.00M]
  ------------------
  393|  6.51M|                            addIfHeapEntity(upv.second, s.children);
  394|  8.00M|                        break;
  395|  8.00M|                    }
  396|  5.50M|                    case HeapEntity::EXTENDED_OBJECT: {
  ------------------
  |  Branch (396:21): [True: 5.50M, False: 60.4M]
  ------------------
  397|  5.50M|                        assert(dynamic_cast<HeapExtendedObject *>(curr));
  ------------------
  |  Branch (397:25): [True: 5.50M, False: 0]
  ------------------
  398|  5.50M|                        auto *obj = static_cast<HeapExtendedObject *>(curr);
  399|  5.50M|                        addIfHeapEntity(obj->left, s.children);
  400|  5.50M|                        addIfHeapEntity(obj->right, s.children);
  401|  5.50M|                        break;
  402|  5.50M|                    }
  403|      0|                    case HeapEntity::RESTRICTED_OBJECT: {
  ------------------
  |  Branch (403:21): [True: 0, False: 65.9M]
  ------------------
  404|      0|                        assert(dynamic_cast<HeapRestrictedObject *>(curr));
  ------------------
  |  Branch (404:25): [True: 0, False: 0]
  ------------------
  405|      0|                        auto *obj = static_cast<HeapRestrictedObject *>(curr);
  406|      0|                        addIfHeapEntity(obj->obj, s.children);
  407|      0|                        break;
  408|      0|                    }
  409|      0|                    case HeapEntity::COMPREHENSION_OBJECT: {
  ------------------
  |  Branch (409:21): [True: 0, False: 65.9M]
  ------------------
  410|      0|                        assert(dynamic_cast<HeapComprehensionObject *>(curr));
  ------------------
  |  Branch (410:25): [True: 0, False: 0]
  ------------------
  411|      0|                        auto *obj = static_cast<HeapComprehensionObject *>(curr);
  412|      0|                        for (auto upv : obj->upValues)
  ------------------
  |  Branch (412:39): [True: 0, False: 0]
  ------------------
  413|      0|                            addIfHeapEntity(upv.second, s.children);
  414|      0|                        for (auto upv : obj->compValues)
  ------------------
  |  Branch (414:39): [True: 0, False: 0]
  ------------------
  415|      0|                            addIfHeapEntity(upv.second, s.children);
  416|      0|                        break;
  417|      0|                    }
  418|   918k|                    case HeapEntity::ARRAY: {
  ------------------
  |  Branch (418:21): [True: 918k, False: 64.9M]
  ------------------
  419|   918k|                        assert(dynamic_cast<HeapArray *>(curr));
  ------------------
  |  Branch (419:25): [True: 918k, False: 0]
  ------------------
  420|   918k|                        auto *arr = static_cast<HeapArray *>(curr);
  421|   918k|                        for (auto el : arr->elements)
  ------------------
  |  Branch (421:38): [True: 9.69M, False: 918k]
  ------------------
  422|  9.69M|                            addIfHeapEntity(el, s.children);
  423|   918k|                        break;
  424|   918k|                    }
  425|  2.18M|                    case HeapEntity::CLOSURE: {
  ------------------
  |  Branch (425:21): [True: 2.18M, False: 63.7M]
  ------------------
  426|  2.18M|                        assert(dynamic_cast<HeapClosure *>(curr));
  ------------------
  |  Branch (426:25): [True: 2.18M, False: 0]
  ------------------
  427|  2.18M|                        auto *func = static_cast<HeapClosure *>(curr);
  428|  2.18M|                        for (auto upv : func->upValues)
  ------------------
  |  Branch (428:39): [True: 5.49M, False: 2.18M]
  ------------------
  429|  5.49M|                            addIfHeapEntity(upv.second, s.children);
  430|  2.18M|                        if (func->self)
  ------------------
  |  Branch (430:29): [True: 1.97M, False: 208k]
  ------------------
  431|  1.97M|                            addIfHeapEntity(func->self, s.children);
  432|  2.18M|                        break;
  433|  2.18M|                    }
  434|  45.8M|                    case HeapEntity::THUNK: {
  ------------------
  |  Branch (434:21): [True: 45.8M, False: 20.0M]
  ------------------
  435|  45.8M|                        assert(dynamic_cast<HeapThunk *>(curr));
  ------------------
  |  Branch (435:25): [True: 45.8M, False: 0]
  ------------------
  436|  45.8M|                        auto *thunk = static_cast<HeapThunk *>(curr);
  437|  45.8M|                        if (thunk->filled) {
  ------------------
  |  Branch (437:29): [True: 12.9M, False: 32.8M]
  ------------------
  438|  12.9M|                            if (thunk->content.isHeap())
  ------------------
  |  Branch (438:33): [True: 9.42M, False: 3.55M]
  ------------------
  439|  9.42M|                                addIfHeapEntity(thunk->content.v.h, s.children);
  440|  32.8M|                        } else {
  441|  32.8M|                            for (auto upv : thunk->upValues)
  ------------------
  |  Branch (441:43): [True: 8.89M, False: 32.8M]
  ------------------
  442|  8.89M|                                addIfHeapEntity(upv.second, s.children);
  443|  32.8M|                            if (thunk->self)
  ------------------
  |  Branch (443:33): [True: 32.3M, False: 467k]
  ------------------
  444|  32.3M|                                addIfHeapEntity(thunk->self, s.children);
  445|  32.8M|                        }
  446|  45.8M|                        break;
  447|  45.8M|                    }
  448|  3.46M|                    case HeapEntity::STRING:
  ------------------
  |  Branch (448:21): [True: 3.46M, False: 62.4M]
  ------------------
  449|  3.46M|                        assert(dynamic_cast<HeapString *>(curr));
  ------------------
  |  Branch (449:25): [True: 3.46M, False: 0]
  ------------------
  450|  3.46M|                        break;
  451|  3.46M|                    default:
  ------------------
  |  Branch (451:21): [True: 0, False: 65.9M]
  ------------------
  452|      0|                        assert(false);
  ------------------
  |  Branch (452:25): [Folded, False: 0]
  ------------------
  453|      0|                        break;
  454|  65.9M|                }
  455|  65.9M|            }
  456|       |
  457|   261M|            if (s.children.size() > 0) {
  ------------------
  |  Branch (457:17): [True: 85.3M, False: 176M]
  ------------------
  458|  85.3M|                HeapEntity *next = s.children[s.children.size() - 1];
  459|  85.3M|                s.children.pop_back();
  460|  85.3M|                stack.emplace_back(next);  // CAUTION: s invalidated here
  461|   176M|            } else {
  462|   176M|                stack.pop_back();  // CAUTION: s invalidated here
  463|   176M|            }
  464|   261M|        }
  465|  91.0M|    }
vm.cpp:_ZZN7jsonnet8internal12_GLOBAL__N_14Heap8markFromEPNS1_10HeapEntityEEN5StateC2ES4_:
  375|   176M|            State(HeapEntity *ent) : ent(ent) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap15addIfHeapEntityEPNS1_10HeapEntityERNSt3__16vectorIS4_NS5_9allocatorIS4_EEEE:
  340|  85.3M|    {
  341|  85.3M|        vec.push_back(v);
  342|  85.3M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap8markFromENS1_5ValueE:
  362|  45.4M|    {
  363|  45.4M|        if (v.isHeap())
  ------------------
  |  Branch (363:13): [True: 11.5M, False: 33.8M]
  ------------------
  364|  11.5M|            markFrom(v.v.h);
  365|  45.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap5sweepEv:
  469|   129k|    {
  470|   129k|        lastMark++;
  471|       |        // Heap shrinks during this loop.  Do not cache entities.size().
  472|   171M|        for (unsigned long i = 0; i < entities.size(); ++i) {
  ------------------
  |  Branch (472:35): [True: 171M, False: 129k]
  ------------------
  473|   171M|            HeapEntity *x = entities[i];
  474|   171M|            if (x->mark != lastMark) {
  ------------------
  |  Branch (474:17): [True: 105M, False: 65.9M]
  ------------------
  475|   105M|                delete x;
  476|   105M|                if (i != entities.size() - 1) {
  ------------------
  |  Branch (476:21): [True: 105M, False: 106k]
  ------------------
  477|       |                    // Swap it with the back.
  478|   105M|                    entities[i] = entities[entities.size() - 1];
  479|   105M|                }
  480|   105M|                entities.pop_back();
  481|   105M|                --i;
  482|   105M|            }
  483|   171M|        }
  484|   129k|        lastNumEntities = numEntities = entities.size();
  485|   129k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_19HeapThunk4fillERKNS1_5ValueE:
  145|  30.4M|    {
  146|  30.4M|        content = v;
  147|  30.4M|        filled = true;
  148|  30.4M|        self = nullptr;
  149|  30.4M|        upValues.clear();
  150|  30.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapArrayEJRKNSt3__16vectorIPNS1_9HeapThunkENS5_9allocatorIS8_EEEEEEEPT_DpOT0_:
  501|   639k|    {
  502|   639k|        T *r = new T(std::forward<Args>(args)...);
  503|   639k|        entities.push_back(r);
  504|   639k|        r->mark = lastMark;
  505|   639k|        numEntities = entities.size();
  506|   639k|        return r;
  507|   639k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_19HeapArrayC2ERKNSt3__16vectorIPNS1_9HeapThunkENS3_9allocatorIS6_EEEE:
  159|   639k|        : HeapEntity(ARRAY), elements(elements)
  160|   639k|    {
  161|   639k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_10HeapStringEJRKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEEEEEPT_DpOT0_:
  501|  40.5M|    {
  502|  40.5M|        T *r = new T(std::forward<Args>(args)...);
  503|  40.5M|        entities.push_back(r);
  504|  40.5M|        r->mark = lastMark;
  505|  40.5M|        numEntities = entities.size();
  506|  40.5M|        return r;
  507|  40.5M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapStringC2ERKNSt3__112basic_stringIDiNS3_11char_traitsIDiEENS3_9allocatorIDiEEEE:
  299|  40.5M|    HeapString(const UString &value) : HeapEntity(STRING), value(value) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierEDniDnEEEPT_DpOT0_:
  501|   498k|    {
  502|   498k|        T *r = new T(std::forward<Args>(args)...);
  503|   498k|        entities.push_back(r);
  504|   498k|        r->mark = lastMark;
  505|   498k|        numEntities = entities.size();
  506|   498k|        return r;
  507|   498k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_114HeapLeafObjectC2ENS1_10HeapEntity4TypeE:
  166|  1.56M|    HeapLeafObject(Type type) : HeapObject(type) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapObjectC2ENS1_10HeapEntity4TypeE:
  109|  2.67M|    HeapObject(Type type) : HeapEntity(type) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111HeapClosure5ParamC2EPKNS0_10IdentifierEPKNS0_3ASTE:
  275|  20.0M|        Param(const Identifier *id, const AST *def) : id(id), def(def) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_11HeapClosureEJNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEEDniRNS5_6vectorINS4_5ParamENSE_ISL_EEEERPKNS0_19BuiltinFunctionBodyERKNS5_12basic_stringIcNS5_11char_traitsIcEENSE_IcEEEEEEEPT_DpOT0_:
  501|  9.92M|    {
  502|  9.92M|        T *r = new T(std::forward<Args>(args)...);
  503|  9.92M|        entities.push_back(r);
  504|  9.92M|        r->mark = lastMark;
  505|  9.92M|        numEntities = entities.size();
  506|  9.92M|        return r;
  507|  9.92M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111HeapClosureC2ERKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS3_4lessIS7_EENS3_9allocatorINS3_4pairIKS7_S9_EEEEEEPNS1_10HeapObjectEjRKNS3_6vectorINS2_5ParamENSC_ISN_EEEEPKNS0_3ASTERKNS3_12basic_stringIcNS3_11char_traitsIcEENSC_IcEEEE:
  283|  12.2M|        : HeapEntity(CLOSURE),
  284|  12.2M|          upValues(up_values),
  285|  12.2M|          self(self),
  286|  12.2M|          offset(offset),
  287|  12.2M|          params(params),
  288|  12.2M|          body(body),
  289|  12.2M|          builtinName(builtin_name)
  290|  12.2M|    {
  291|  12.2M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJDnDniPKNS0_3ASTEEEEPT_DpOT0_:
  501|  2.88k|    {
  502|  2.88k|        T *r = new T(std::forward<Args>(args)...);
  503|  2.88k|        entities.push_back(r);
  504|  2.88k|        r->mark = lastMark;
  505|  2.88k|        numEntities = entities.size();
  506|  2.88k|        return r;
  507|  2.88k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJPNS0_10IdentifierERPNS1_10HeapObjectEiRKPNS0_3ASTEEEEPT_DpOT0_:
  501|   464k|    {
  502|   464k|        T *r = new T(std::forward<Args>(args)...);
  503|   464k|        entities.push_back(r);
  504|   464k|        r->mark = lastMark;
  505|   464k|        numEntities = entities.size();
  506|   464k|        return r;
  507|   464k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14HeapD2Ev:
  355|  2.88k|    {
  356|       |        // Nothing is marked, everything will be collected.
  357|  2.88k|        sweep();
  358|  2.88k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  501|  26.6M|    {
  502|  26.6M|        T *r = new T(std::forward<Args>(args)...);
  503|  26.6M|        entities.push_back(r);
  504|  26.6M|        r->mark = lastMark;
  505|  26.6M|        numEntities = entities.size();
  506|  26.6M|        return r;
  507|  26.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_11HeapClosureEJRKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERPNS1_10HeapObjectERjRKNS5_6vectorINS4_5ParamENSE_ISR_EEEERPNS0_3ASTERA1_KcEEEPT_DpOT0_:
  501|  2.32M|    {
  502|  2.32M|        T *r = new T(std::forward<Args>(args)...);
  503|  2.32M|        entities.push_back(r);
  504|  2.32M|        r->mark = lastMark;
  505|  2.32M|        numEntities = entities.size();
  506|  2.32M|        return r;
  507|  2.32M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRKPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  501|  21.6M|    {
  502|  21.6M|        T *r = new T(std::forward<Args>(args)...);
  503|  21.6M|        entities.push_back(r);
  504|  21.6M|        r->mark = lastMark;
  505|  21.6M|        numEntities = entities.size();
  506|  21.6M|        return r;
  507|  21.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_16HeapSimpleObjectEJRNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERNS6_IS9_NS4_5FieldESD_NSE_INSF_ISG_SL_EEEEEERNS5_4listIPNS0_3ASTENSE_ISS_EEEEEEEPT_DpOT0_:
  501|  1.56M|    {
  502|  1.56M|        T *r = new T(std::forward<Args>(args)...);
  503|  1.56M|        entities.push_back(r);
  504|  1.56M|        r->mark = lastMark;
  505|  1.56M|        numEntities = entities.size();
  506|  1.56M|        return r;
  507|  1.56M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116HeapSimpleObjectC2ERKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS3_4lessIS7_EENS3_9allocatorINS3_4pairIKS7_S9_EEEEEENS4_IS7_NS2_5FieldESB_NSC_INSD_ISE_SK_EEEEEENS3_4listIPNS0_3ASTENSC_ISQ_EEEE:
  196|  1.56M|        : HeapLeafObject(SIMPLE_OBJECT), upValues(up_values), fields(fields), asserts(asserts)
  197|  1.56M|    {
  198|  1.56M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPKNS0_3ASTEEEEPT_DpOT0_:
  501|    243|    {
  502|    243|        T *r = new T(std::forward<Args>(args)...);
  503|    243|        entities.push_back(r);
  504|    243|        r->mark = lastMark;
  505|    243|        numEntities = entities.size();
  506|    243|        return r;
  507|    243|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_18HeapExtendedObjectEJRPNS1_10HeapObjectES7_EEEPT_DpOT0_:
  501|  1.10M|    {
  502|  1.10M|        T *r = new T(std::forward<Args>(args)...);
  503|  1.10M|        entities.push_back(r);
  504|  1.10M|        r->mark = lastMark;
  505|  1.10M|        numEntities = entities.size();
  506|  1.10M|        return r;
  507|  1.10M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_118HeapExtendedObjectC2EPNS1_10HeapObjectES4_:
  210|  1.10M|        : HeapObject(EXTENDED_OBJECT), left(left), right(right)
  211|  1.10M|    {
  212|  1.10M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRPNS0_3ASTEEEEPT_DpOT0_:
  501|   125k|    {
  502|   125k|        T *r = new T(std::forward<Args>(args)...);
  503|   125k|        entities.push_back(r);
  504|   125k|        r->mark = lastMark;
  505|   125k|        numEntities = entities.size();
  506|   125k|        return r;
  507|   125k|    }

_ZN7jsonnet8internal23jsonnet_static_analysisEPNS0_3ASTE:
  228|  6.35k|{
  229|  6.35k|    static_analysis(ast, false, IdSet{});
  230|  6.35k|}
static_analysis.cpp:_ZN7jsonnet8internalL15static_analysisEPNS0_3ASTEbRKNSt3__13setIPKNS0_10IdentifierENS3_4lessIS7_EENS3_9allocatorIS7_EEEE:
   41|   199M|{
   42|   199M|    IdSet r;
   43|       |
   44|   199M|    switch (ast_->type) {
   45|  17.9M|    case AST_APPLY: {
  ------------------
  |  Branch (45:5): [True: 17.9M, False: 181M]
  ------------------
   46|  17.9M|        assert(dynamic_cast<Apply *>(ast_));
  ------------------
  |  Branch (46:9): [True: 17.9M, False: 0]
  ------------------
   47|  17.9M|        auto* ast = static_cast<Apply *>(ast_);
   48|  17.9M|        append(r, static_analysis(ast->target, in_object, vars));
   49|  17.9M|        for (const auto &arg : ast->args)
  ------------------
  |  Branch (49:30): [True: 29.4M, False: 17.9M]
  ------------------
   50|  29.4M|            append(r, static_analysis(arg.expr, in_object, vars));
   51|  17.9M|    } break;
   52|      0|    case AST_APPLY_BRACE: {
  ------------------
  |  Branch (52:5): [True: 0, False: 199M]
  ------------------
   53|      0|        assert(dynamic_cast<ApplyBrace *>(ast_));
  ------------------
  |  Branch (53:9): [True: 0, False: 0]
  ------------------
   54|       |        // Nothing to do.
   55|      0|    } break;
   56|  4.12M|    case AST_ARRAY: {
  ------------------
  |  Branch (56:5): [True: 4.12M, False: 195M]
  ------------------
   57|  4.12M|        assert(dynamic_cast<Array *>(ast_));
  ------------------
  |  Branch (57:9): [True: 4.12M, False: 0]
  ------------------
   58|  4.12M|        auto* ast = static_cast<Array *>(ast_);
   59|  4.12M|        for (auto &el : ast->elements)
  ------------------
  |  Branch (59:23): [True: 5.48M, False: 4.12M]
  ------------------
   60|  5.48M|            append(r, static_analysis(el.expr, in_object, vars));
   61|  4.12M|    } break;
   62|  18.2M|    case AST_BINARY: {
  ------------------
  |  Branch (62:5): [True: 18.2M, False: 181M]
  ------------------
   63|  18.2M|        assert(dynamic_cast<Binary *>(ast_));
  ------------------
  |  Branch (63:9): [True: 18.2M, False: 0]
  ------------------
   64|  18.2M|        auto* ast = static_cast<Binary *>(ast_);
   65|  18.2M|        append(r, static_analysis(ast->left, in_object, vars));
   66|  18.2M|        append(r, static_analysis(ast->right, in_object, vars));
   67|  18.2M|    } break;
   68|   266k|    case AST_BUILTIN_FUNCTION: {
  ------------------
  |  Branch (68:5): [True: 266k, False: 199M]
  ------------------
   69|   266k|        assert(dynamic_cast<BuiltinFunction *>(ast_));
  ------------------
  |  Branch (69:9): [True: 266k, False: 0]
  ------------------
   70|       |        // Nothing to do.
   71|   266k|    } break;
   72|  7.38M|    case AST_CONDITIONAL: {
  ------------------
  |  Branch (72:5): [True: 7.38M, False: 191M]
  ------------------
   73|  7.38M|        assert(dynamic_cast<Conditional *>(ast_));
  ------------------
  |  Branch (73:9): [True: 7.38M, False: 0]
  ------------------
   74|  7.38M|        auto* ast = static_cast<Conditional *>(ast_);
   75|  7.38M|        append(r, static_analysis(ast->cond, in_object, vars));
   76|  7.38M|        append(r, static_analysis(ast->branchTrue, in_object, vars));
   77|  7.38M|        append(r, static_analysis(ast->branchFalse, in_object, vars));
   78|  7.38M|    } break;
   79|  3.86M|    case AST_ERROR: {
  ------------------
  |  Branch (79:5): [True: 3.86M, False: 195M]
  ------------------
   80|  3.86M|        assert(dynamic_cast<Error *>(ast_));
  ------------------
  |  Branch (80:9): [True: 3.86M, False: 0]
  ------------------
   81|  3.86M|        auto* ast = static_cast<Error *>(ast_);
   82|  3.86M|        append(r, static_analysis(ast->expr, in_object, vars));
   83|  3.86M|    } break;
   84|  4.54M|    case AST_FUNCTION: {
  ------------------
  |  Branch (84:5): [True: 4.54M, False: 194M]
  ------------------
   85|  4.54M|        assert(dynamic_cast<Function *>(ast_));
  ------------------
  |  Branch (85:9): [True: 4.54M, False: 0]
  ------------------
   86|  4.54M|        auto* ast = static_cast<Function *>(ast_);
   87|  4.54M|        auto new_vars = vars;
   88|  4.54M|        IdSet params;
   89|  8.27M|        for (const auto &p : ast->params) {
  ------------------
  |  Branch (89:28): [True: 8.27M, False: 4.54M]
  ------------------
   90|  8.27M|            if (params.find(p.id) != params.end()) {
  ------------------
  |  Branch (90:17): [True: 6, False: 8.27M]
  ------------------
   91|      6|                std::string msg = "Duplicate function parameter: " + encode_utf8(p.id->name);
   92|      6|                throw StaticError(ast_->location, msg);
   93|      6|            }
   94|  8.27M|            params.insert(p.id);
   95|  8.27M|            new_vars.insert(p.id);
   96|  8.27M|        }
   97|       |
   98|  4.54M|        auto fv = static_analysis(ast->body, in_object, new_vars);
   99|  8.27M|        for (const auto &p : ast->params) {
  ------------------
  |  Branch (99:28): [True: 8.27M, False: 4.54M]
  ------------------
  100|  8.27M|            if (p.expr != nullptr)
  ------------------
  |  Branch (100:17): [True: 135k, False: 8.13M]
  ------------------
  101|   135k|                append(fv, static_analysis(p.expr, in_object, new_vars));
  102|  8.27M|        }
  103|  4.54M|        for (const auto &p : ast->params)
  ------------------
  |  Branch (103:28): [True: 8.27M, False: 4.54M]
  ------------------
  104|  8.27M|            fv.erase(p.id);
  105|  4.54M|        append(r, fv);
  106|  4.54M|    } break;
  107|    267|    case AST_IMPORT: {
  ------------------
  |  Branch (107:5): [True: 267, False: 199M]
  ------------------
  108|    267|        assert(dynamic_cast<Import *>(ast_));
  ------------------
  |  Branch (108:9): [True: 267, False: 0]
  ------------------
  109|       |        // Nothing to do.
  110|    267|    } break;
  111|    426|    case AST_IMPORTSTR: {
  ------------------
  |  Branch (111:5): [True: 426, False: 199M]
  ------------------
  112|    426|        assert(dynamic_cast<Importstr *>(ast_));
  ------------------
  |  Branch (112:9): [True: 426, False: 0]
  ------------------
  113|       |        // Nothing to do.
  114|    426|    } break;
  115|    530|    case AST_IMPORTBIN: {
  ------------------
  |  Branch (115:5): [True: 530, False: 199M]
  ------------------
  116|    530|        assert(dynamic_cast<Importbin *>(ast_));
  ------------------
  |  Branch (116:9): [True: 530, False: 0]
  ------------------
  117|       |        // Nothing to do.
  118|    530|    } break;
  119|   103k|    case AST_IN_SUPER: {
  ------------------
  |  Branch (119:5): [True: 103k, False: 199M]
  ------------------
  120|   103k|        assert(dynamic_cast<const InSuper *>(ast_));
  ------------------
  |  Branch (120:9): [True: 103k, False: 0]
  ------------------
  121|   103k|        auto* ast = static_cast<const InSuper *>(ast_);
  122|   103k|        if (!in_object)
  ------------------
  |  Branch (122:13): [True: 19, False: 103k]
  ------------------
  123|     19|            throw StaticError(ast_->location, "Can't use super outside of an object.");
  124|   103k|        append(r, static_analysis(ast->element, in_object, vars));
  125|   103k|    } break;
  126|  19.6M|    case AST_INDEX: {
  ------------------
  |  Branch (126:5): [True: 19.6M, False: 179M]
  ------------------
  127|  19.6M|        assert(dynamic_cast<const Index *>(ast_));
  ------------------
  |  Branch (127:9): [True: 19.6M, False: 0]
  ------------------
  128|  19.6M|        auto* ast = static_cast<const Index *>(ast_);
  129|  19.6M|        append(r, static_analysis(ast->target, in_object, vars));
  130|  19.6M|        append(r, static_analysis(ast->index, in_object, vars));
  131|  19.6M|    } break;
  132|  9.51M|    case AST_LOCAL: {
  ------------------
  |  Branch (132:5): [True: 9.51M, False: 189M]
  ------------------
  133|  9.51M|        assert(dynamic_cast<const Local *>(ast_));
  ------------------
  |  Branch (133:9): [True: 9.51M, False: 0]
  ------------------
  134|  9.51M|        auto* ast = static_cast<const Local *>(ast_);
  135|  9.51M|        IdSet ast_vars;
  136|  16.0M|        for (const auto &bind : ast->binds) {
  ------------------
  |  Branch (136:31): [True: 16.0M, False: 9.51M]
  ------------------
  137|  16.0M|            ast_vars.insert(bind.var);
  138|  16.0M|        }
  139|  9.51M|        auto new_vars = vars;
  140|  9.51M|        append(new_vars, ast_vars);
  141|  9.51M|        IdSet fvs;
  142|  16.0M|        for (const auto &bind : ast->binds) {
  ------------------
  |  Branch (142:31): [True: 16.0M, False: 9.51M]
  ------------------
  143|  16.0M|            append(fvs, static_analysis(bind.body, in_object, new_vars));
  144|  16.0M|        }
  145|       |
  146|  9.51M|        append(fvs, static_analysis(ast->body, in_object, new_vars));
  147|       |
  148|  9.51M|        for (const auto &bind : ast->binds)
  ------------------
  |  Branch (148:31): [True: 15.9M, False: 9.51M]
  ------------------
  149|  15.9M|            fvs.erase(bind.var);
  150|       |
  151|  9.51M|        append(r, fvs);
  152|  9.51M|    } break;
  153|   560k|    case AST_LITERAL_BOOLEAN: {
  ------------------
  |  Branch (153:5): [True: 560k, False: 198M]
  ------------------
  154|   560k|        assert(dynamic_cast<const LiteralBoolean *>(ast_));
  ------------------
  |  Branch (154:9): [True: 560k, False: 0]
  ------------------
  155|       |        // Nothing to do.
  156|   560k|    } break;
  157|  12.5M|    case AST_LITERAL_NUMBER: {
  ------------------
  |  Branch (157:5): [True: 12.5M, False: 186M]
  ------------------
  158|  12.5M|        assert(dynamic_cast<const LiteralNumber *>(ast_));
  ------------------
  |  Branch (158:9): [True: 12.5M, False: 0]
  ------------------
  159|       |        // Nothing to do.
  160|  12.5M|    } break;
  161|  35.1M|    case AST_LITERAL_STRING: {
  ------------------
  |  Branch (161:5): [True: 35.1M, False: 164M]
  ------------------
  162|  35.1M|        assert(dynamic_cast<const LiteralString *>(ast_));
  ------------------
  |  Branch (162:9): [True: 35.1M, False: 0]
  ------------------
  163|       |        // Nothing to do.
  164|  35.1M|    } break;
  165|  35.1M|    case AST_LITERAL_NULL: {
  ------------------
  |  Branch (165:5): [True: 314k, False: 199M]
  ------------------
  166|   314k|        assert(dynamic_cast<const LiteralNull *>(ast_));
  ------------------
  |  Branch (166:9): [True: 314k, False: 0]
  ------------------
  167|       |        // Nothing to do.
  168|   314k|    } break;
  169|  1.36M|    case AST_DESUGARED_OBJECT: {
  ------------------
  |  Branch (169:5): [True: 1.36M, False: 198M]
  ------------------
  170|  1.36M|        assert(dynamic_cast<DesugaredObject *>(ast_));
  ------------------
  |  Branch (170:9): [True: 1.36M, False: 0]
  ------------------
  171|  1.36M|        auto* ast = static_cast<DesugaredObject *>(ast_);
  172|  5.58M|        for (auto &field : ast->fields) {
  ------------------
  |  Branch (172:26): [True: 5.58M, False: 1.36M]
  ------------------
  173|  5.58M|            append(r, static_analysis(field.name, in_object, vars));
  174|  5.58M|            append(r, static_analysis(field.body, true, vars));
  175|  5.58M|        }
  176|  1.36M|        for (AST *assert : ast->asserts) {
  ------------------
  |  Branch (176:26): [True: 46.2k, False: 1.36M]
  ------------------
  177|  46.2k|            append(r, static_analysis(assert, true, vars));
  178|  46.2k|        }
  179|  1.36M|    } break;
  180|   779k|    case AST_OBJECT_COMPREHENSION_SIMPLE: {
  ------------------
  |  Branch (180:5): [True: 779k, False: 198M]
  ------------------
  181|   779k|        assert(dynamic_cast<ObjectComprehensionSimple *>(ast_));
  ------------------
  |  Branch (181:9): [True: 779k, False: 0]
  ------------------
  182|   779k|        auto* ast = static_cast<ObjectComprehensionSimple *>(ast_);
  183|   779k|        auto new_vars = vars;
  184|   779k|        new_vars.insert(ast->id);
  185|   779k|        append(r, static_analysis(ast->field, false, new_vars));
  186|   779k|        append(r, static_analysis(ast->value, true, new_vars));
  187|   779k|        r.erase(ast->id);
  188|   779k|        append(r, static_analysis(ast->array, in_object, vars));
  189|   779k|    } break;
  190|  1.57M|    case AST_SELF: {
  ------------------
  |  Branch (190:5): [True: 1.57M, False: 197M]
  ------------------
  191|  1.57M|        assert(dynamic_cast<const Self *>(ast_));
  ------------------
  |  Branch (191:9): [True: 1.57M, False: 0]
  ------------------
  192|  1.57M|        if (!in_object)
  ------------------
  |  Branch (192:13): [True: 6, False: 1.57M]
  ------------------
  193|      6|            throw StaticError(ast_->location, "Can't use self outside of an object.");
  194|  1.57M|    } break;
  195|  1.57M|    case AST_SUPER_INDEX: {
  ------------------
  |  Branch (195:5): [True: 103k, False: 199M]
  ------------------
  196|   103k|        assert(dynamic_cast<const SuperIndex *>(ast_));
  ------------------
  |  Branch (196:9): [True: 103k, False: 0]
  ------------------
  197|   103k|        auto* ast = static_cast<const SuperIndex *>(ast_);
  198|   103k|        if (!in_object)
  ------------------
  |  Branch (198:13): [True: 24, False: 103k]
  ------------------
  199|     24|            throw StaticError(ast_->location, "Can't use super outside of an object.");
  200|   103k|        append(r, static_analysis(ast->index, in_object, vars));
  201|   103k|    } break;
  202|   686k|    case AST_UNARY: {
  ------------------
  |  Branch (202:5): [True: 686k, False: 198M]
  ------------------
  203|   686k|        assert(dynamic_cast<const Unary *>(ast_));
  ------------------
  |  Branch (203:9): [True: 686k, False: 0]
  ------------------
  204|   686k|        auto* ast = static_cast<const Unary *>(ast_);
  205|   686k|        append(r, static_analysis(ast->expr, in_object, vars));
  206|   686k|    } break;
  207|  60.6M|    case AST_VAR: {
  ------------------
  |  Branch (207:5): [True: 60.6M, False: 138M]
  ------------------
  208|  60.6M|        assert(dynamic_cast<const Var *>(ast_));
  ------------------
  |  Branch (208:9): [True: 60.6M, False: 0]
  ------------------
  209|  60.6M|        auto* ast = static_cast<const Var *>(ast_);
  210|  60.6M|        if (vars.find(ast->id) == vars.end()) {
  ------------------
  |  Branch (210:13): [True: 537, False: 60.6M]
  ------------------
  211|    537|            throw StaticError(ast->location, "Unknown variable: " + encode_utf8(ast->id->name));
  212|    537|        }
  213|  60.6M|        r.insert(ast->id);
  214|  60.6M|    } break;
  215|      0|    default:
  ------------------
  |  Branch (215:5): [True: 0, False: 199M]
  ------------------
  216|      0|        std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  217|      0|        std::abort();
  218|      0|        break;
  219|   199M|    }
  220|       |
  221|   199M|    for (auto *id : r)
  ------------------
  |  Branch (221:19): [True: 271M, False: 199M]
  ------------------
  222|   271M|        ast_->freeVariables.push_back(id);
  223|       |
  224|   199M|    return r;
  225|   199M|}
static_analysis.cpp:_ZN7jsonnet8internalL6appendERNSt3__13setIPKNS0_10IdentifierENS1_4lessIS5_EENS1_9allocatorIS5_EEEERKSA_:
   29|   218M|{
   30|   218M|    r.insert(s.begin(), s.end());
   31|   218M|}

libjsonnet.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_11StaticErrorE:
  108|  3.56k|{
  109|  3.56k|    o << err.toString();
  110|  3.56k|    return o;
  111|  3.56k|}
_ZNK7jsonnet8internal11StaticError8toStringEv:
   97|  3.56k|    {
   98|  3.56k|        std::stringstream ss;
   99|  3.56k|        if (location.isSet()) {
  ------------------
  |  Branch (99:13): [True: 3.56k, False: 5]
  ------------------
  100|  3.56k|            ss << location << ":";
  101|  3.56k|        }
  102|  3.56k|        ss << " " << msg;
  103|  3.56k|        return ss.str();
  104|  3.56k|    }
_ZNK7jsonnet8internal13LocationRange5isSetEv:
   58|   323k|    {
   59|   323k|        return begin.isSet();
   60|   323k|    }
_ZNK7jsonnet8internal8Location5isSetEv:
   31|   323k|    {
   32|   323k|        return line != 0;
   33|   323k|    }
libjsonnet.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_13LocationRangeE:
   64|  20.5k|{
   65|  20.5k|    if (loc.file.length() > 0)
  ------------------
  |  Branch (65:9): [True: 9.86k, False: 10.6k]
  ------------------
   66|  9.86k|        o << loc.file;
   67|  20.5k|    if (loc.isSet()) {
  ------------------
  |  Branch (67:9): [True: 19.6k, False: 918]
  ------------------
   68|  19.6k|        if (loc.file.length() > 0)
  ------------------
  |  Branch (68:13): [True: 9.00k, False: 10.6k]
  ------------------
   69|  9.00k|            o << ":";
   70|  19.6k|        if (loc.begin.line == loc.end.line) {
  ------------------
  |  Branch (70:13): [True: 17.4k, False: 2.12k]
  ------------------
   71|  17.4k|            if (loc.begin.column == loc.end.column - 1) {
  ------------------
  |  Branch (71:17): [True: 5.19k, False: 12.2k]
  ------------------
   72|  5.19k|                o << loc.begin;
   73|  12.2k|            } else {
   74|  12.2k|                o << loc.begin << "-" << loc.end.column;
   75|  12.2k|            }
   76|  17.4k|        } else {
   77|  2.12k|            o << "(" << loc.begin << ")-(" << loc.end << ")";
   78|  2.12k|        }
   79|  19.6k|    }
   80|  20.5k|    return o;
   81|  20.5k|}
libjsonnet.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_8LocationE:
   41|  21.7k|{
   42|  21.7k|    o << loc.line << ":" << loc.column;
   43|  21.7k|    return o;
   44|  21.7k|}
_ZN7jsonnet8internal8LocationC2Ev:
   28|  3.77M|    Location(void) : line(0), column(0) {}
_ZN7jsonnet8internal8LocationC2Emm:
   29|   175M|    Location(unsigned long line, unsigned long column) : line(line), column(column) {}
_ZNK7jsonnet8internal8Location9successorEv:
   35|    501|    {
   36|    501|        return Location(this->line, this->column + 1);
   37|    501|    }
_ZN7jsonnet8internal13LocationRangeC2Ev:
   50|  1.88M|    LocationRange(void) {}
_ZN7jsonnet8internal13LocationRangeC2ERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEE:
   52|  5.47k|    LocationRange(const std::string &msg) : file(msg) {}
_ZN7jsonnet8internal13LocationRangeC2ERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERKNS0_8LocationESD_:
   54|   136M|        : file(file), begin(begin), end(end)
   55|   136M|    {
   56|   136M|    }
_ZN7jsonnet8internal11StaticErrorC2ERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERKNS0_8LocationESA_:
   88|    501|        : location(filename, location, location.successor()), msg(msg)
   89|    501|    {
   90|    501|    }
_ZN7jsonnet8internal11StaticErrorC2ERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEE:
   92|  3.06k|        : location(location), msg(msg)
   93|  3.06k|    {
   94|  3.06k|    }

_ZN7jsonnet8internal22jsonnet_string_unparseERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEEb:
   25|   255k|{
   26|   255k|    UStringStream ss;
   27|   255k|    ss << (single ? U'\'' : U'\"');
  ------------------
  |  Branch (27:12): [True: 0, False: 255k]
  ------------------
   28|   255k|    ss << jsonnet_string_escape(str, single);
   29|   255k|    ss << (single ? U'\'' : U'\"');
  ------------------
  |  Branch (29:12): [True: 0, False: 255k]
  ------------------
   30|   255k|    return ss.str();
   31|   255k|}
_ZN7jsonnet8internal21jsonnet_string_escapeERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEEb:
   34|   255k|{
   35|   255k|    UStringStream ss;
   36|   596M|    for (std::size_t i = 0; i < str.length(); ++i) {
  ------------------
  |  Branch (36:29): [True: 596M, False: 255k]
  ------------------
   37|   596M|        char32_t c = str[i];
   38|   596M|        switch (c) {
   39|   361k|            case U'\"': ss << (single ? U"\"" : U"\\\""); break;
  ------------------
  |  Branch (39:13): [True: 361k, False: 596M]
  |  Branch (39:32): [True: 0, False: 361k]
  ------------------
   40|  6.69k|            case U'\'': ss << (single ? U"\\\'" : U"\'"); break;
  ------------------
  |  Branch (40:13): [True: 6.69k, False: 596M]
  |  Branch (40:32): [True: 0, False: 6.69k]
  ------------------
   41|   547M|            case U'\\': ss << U"\\\\"; break;
  ------------------
  |  Branch (41:13): [True: 547M, False: 49.4M]
  ------------------
   42|  93.4k|            case U'\b': ss << U"\\b"; break;
  ------------------
  |  Branch (42:13): [True: 93.4k, False: 596M]
  ------------------
   43|  21.0k|            case U'\f': ss << U"\\f"; break;
  ------------------
  |  Branch (43:13): [True: 21.0k, False: 596M]
  ------------------
   44|  1.67M|            case U'\n': ss << U"\\n"; break;
  ------------------
  |  Branch (44:13): [True: 1.67M, False: 594M]
  ------------------
   45|   214k|            case U'\r': ss << U"\\r"; break;
  ------------------
  |  Branch (45:13): [True: 214k, False: 596M]
  ------------------
   46|   465k|            case U'\t': ss << U"\\t"; break;
  ------------------
  |  Branch (46:13): [True: 465k, False: 596M]
  ------------------
   47|   128k|            case U'\0': ss << U"\\u0000"; break;
  ------------------
  |  Branch (47:13): [True: 128k, False: 596M]
  ------------------
   48|  46.4M|            default: {
  ------------------
  |  Branch (48:13): [True: 46.4M, False: 550M]
  ------------------
   49|  46.4M|                if (c < 0x20 || (c >= 0x7f && c <= 0x9f)) {
  ------------------
  |  Branch (49:21): [True: 10.8M, False: 35.6M]
  |  Branch (49:34): [True: 3.81M, False: 31.7M]
  |  Branch (49:47): [True: 421k, False: 3.39M]
  ------------------
   50|       |                    // Unprintable, use \u
   51|  11.2M|                    std::stringstream ss8;
   52|  11.2M|                    ss8 << "\\u" << std::hex << std::setfill('0') << std::setw(4)
   53|  11.2M|                        << (unsigned long)(c);
   54|  11.2M|                    ss << decode_utf8(ss8.str());
   55|  35.1M|                } else {
   56|       |                    // Printable, write verbatim
   57|  35.1M|                    ss << c;
   58|  35.1M|                }
   59|  46.4M|            }
   60|   596M|        }
   61|   596M|    }
   62|   255k|    return ss.str();
   63|   255k|}
_ZN7jsonnet8internal28jsonnet_string_parse_unicodeERKNS0_13LocationRangeEPKDi:
   66|  22.3k|{
   67|  22.3k|    unsigned long codepoint = 0;
   68|       |    // Expect 4 hex digits.
   69|   111k|    for (unsigned i = 0; i < 4; ++i) {
  ------------------
  |  Branch (69:26): [True: 89.2k, False: 22.2k]
  ------------------
   70|  89.2k|        auto x = (unsigned char)(c[i]);
   71|  89.2k|        unsigned digit;
   72|  89.2k|        if (x == '\0') {
  ------------------
  |  Branch (72:13): [True: 17, False: 89.2k]
  ------------------
   73|     17|            auto msg = "Truncated unicode escape sequence in string literal.";
   74|     17|            throw StaticError(loc, msg);
   75|  89.2k|        } else if (x >= '0' && x <= '9') {
  ------------------
  |  Branch (75:20): [True: 89.1k, False: 11]
  |  Branch (75:32): [True: 62.8k, False: 26.3k]
  ------------------
   76|  62.8k|            digit = x - '0';
   77|  62.8k|        } else if (x >= 'a' && x <= 'f') {
  ------------------
  |  Branch (77:20): [True: 15.4k, False: 10.9k]
  |  Branch (77:32): [True: 15.4k, False: 17]
  ------------------
   78|  15.4k|            digit = x - 'a' + 10;
   79|  15.4k|        } else if (x >= 'A' && x <= 'F') {
  ------------------
  |  Branch (79:20): [True: 10.9k, False: 20]
  |  Branch (79:32): [True: 10.8k, False: 28]
  ------------------
   80|  10.8k|            digit = x - 'A' + 10;
   81|  10.8k|        } else {
   82|     48|            std::stringstream ss;
   83|     48|            ss << "Malformed unicode escape character, "
   84|     48|               << "should be hex: '" << x << "'";
   85|     48|            throw StaticError(loc, ss.str());
   86|     48|        }
   87|  89.1k|        codepoint *= 16;
   88|  89.1k|        codepoint += digit;
   89|  89.1k|    }
   90|  22.2k|    return codepoint;
   91|  22.3k|}
_ZN7jsonnet8internal16is_bmp_codepointEm:
   94|  21.8k|{
   95|  21.8k|    return codepoint < 0xd800 || (codepoint >= 0xe000 && codepoint < 0x10000);
  ------------------
  |  Branch (95:12): [True: 19.1k, False: 2.64k]
  |  Branch (95:35): [True: 2.17k, False: 468]
  |  Branch (95:58): [True: 2.17k, False: 0]
  ------------------
   96|  21.8k|}
_ZN7jsonnet8internal23decode_utf16_surrogatesERKNS0_13LocationRangeEmm:
   99|    446|{
  100|    446|    if (high >= 0xd800 && high < 0xdc00 && low >= 0xdc00 && low < 0xe000) {
  ------------------
  |  Branch (100:9): [True: 446, False: 0]
  |  Branch (100:27): [True: 438, False: 8]
  |  Branch (100:44): [True: 430, False: 8]
  |  Branch (100:61): [True: 428, False: 2]
  ------------------
  101|    428|        return 0x10000 + ((high & 0x03ff) << 10) + (low & 0x03ff);
  102|    428|    } else {
  103|     18|        std::stringstream ss;
  104|     18|        ss << "Invalid UTF-16 bytes";
  105|     18|        throw StaticError(loc, ss.str());
  106|     18|    }
  107|    446|}
_ZN7jsonnet8internal23jsonnet_string_unescapeERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS4_11char_traitsIDiEENS4_9allocatorIDiEEEE:
  110|  3.07M|{
  111|  3.07M|    UString r;
  112|  3.07M|    const char32_t *s_ptr = s.c_str();
  113|  51.3M|    for (const char32_t *c = s_ptr; *c != U'\0'; ++c) {
  ------------------
  |  Branch (113:37): [True: 48.3M, False: 3.07M]
  ------------------
  114|  48.3M|        switch (*c) {
  115|   369k|            case '\\':
  ------------------
  |  Branch (115:13): [True: 369k, False: 47.9M]
  ------------------
  116|   369k|                switch (*(++c)) {
  117|  14.4k|                    case '"':
  ------------------
  |  Branch (117:21): [True: 14.4k, False: 355k]
  ------------------
  118|  15.3k|                    case '\'': r += *c; break;
  ------------------
  |  Branch (118:21): [True: 852, False: 369k]
  ------------------
  119|       |
  120|  91.6k|                    case '\\': r += *c; break;
  ------------------
  |  Branch (120:21): [True: 91.6k, False: 278k]
  ------------------
  121|       |
  122|    474|                    case '/': r += *c; break;
  ------------------
  |  Branch (122:21): [True: 474, False: 369k]
  ------------------
  123|       |
  124|  13.4k|                    case 'b': r += '\b'; break;
  ------------------
  |  Branch (124:21): [True: 13.4k, False: 356k]
  ------------------
  125|       |
  126|  17.3k|                    case 'f': r += '\f'; break;
  ------------------
  |  Branch (126:21): [True: 17.3k, False: 352k]
  ------------------
  127|       |
  128|   182k|                    case 'n': r += '\n'; break;
  ------------------
  |  Branch (128:21): [True: 182k, False: 187k]
  ------------------
  129|       |
  130|  13.0k|                    case 'r': r += '\r'; break;
  ------------------
  |  Branch (130:21): [True: 13.0k, False: 356k]
  ------------------
  131|       |
  132|  14.2k|                    case 't': r += '\t'; break;
  ------------------
  |  Branch (132:21): [True: 14.2k, False: 355k]
  ------------------
  133|       |
  134|  21.8k|                    case 'u': {
  ------------------
  |  Branch (134:21): [True: 21.8k, False: 347k]
  ------------------
  135|  21.8k|                        ++c;  // Consume the 'u'.
  136|  21.8k|                        unsigned long codepoint = jsonnet_string_parse_unicode(loc, c);
  137|       |
  138|       |                        // Leave us on the last char, ready for the ++c at
  139|       |                        // the outer for loop.
  140|  21.8k|                        c += 3;
  141|  21.8k|                        if (!is_bmp_codepoint(codepoint)) {
  ------------------
  |  Branch (141:29): [True: 468, False: 21.4k]
  ------------------
  142|    468|                           if (*(++c) != '\\') {
  ------------------
  |  Branch (142:32): [True: 15, False: 453]
  ------------------
  143|     15|                                std::stringstream ss;
  144|     15|                                ss << "Invalid non-BMP Unicode escape in string literal";
  145|     15|                                throw StaticError(loc, ss.str());
  146|     15|                           }
  147|    453|                           if (*(++c) != 'u') {
  ------------------
  |  Branch (147:32): [True: 5, False: 448]
  ------------------
  148|      5|                                std::stringstream ss;
  149|      5|                                ss << "Invalid non-BMP Unicode escape in string literal";
  150|      5|                                throw StaticError(loc, ss.str());
  151|      5|                           }
  152|    448|                           ++c;
  153|    448|                           unsigned long codepoint2 = jsonnet_string_parse_unicode(loc, c);
  154|    448|                           c += 3;
  155|    448|                           codepoint = decode_utf16_surrogates(loc, codepoint, codepoint2);
  156|    448|                       }
  157|  21.8k|                       r += codepoint;
  158|  21.8k|                    } break;
  159|       |
  160|      2|                    case '\0': {
  ------------------
  |  Branch (160:21): [True: 2, False: 369k]
  ------------------
  161|      2|                        auto msg = "Truncated escape sequence in string literal.";
  162|      2|                        throw StaticError(loc, msg);
  163|  21.8k|                    }
  164|       |
  165|    132|                    default: {
  ------------------
  |  Branch (165:21): [True: 132, False: 369k]
  ------------------
  166|    132|                        std::stringstream ss;
  167|    132|                        std::string utf8;
  168|    132|                        encode_utf8(*c, utf8);
  169|    132|                        ss << "Unknown escape sequence in string literal: '" << utf8 << "'";
  170|    132|                        throw StaticError(loc, ss.str());
  171|  21.8k|                    }
  172|   369k|                }
  173|   369k|                break;
  174|       |
  175|  47.9M|            default:
  ------------------
  |  Branch (175:13): [True: 47.9M, False: 369k]
  ------------------
  176|       |                // Just a regular letter.
  177|  47.9M|                r += *c;
  178|  48.3M|        }
  179|  48.3M|    }
  180|  3.07M|    return r;
  181|  3.07M|}

_ZN7jsonnet8internal13UStringStreamlsERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  155|  17.9M|    {
  156|  17.9M|        buf.append(s);
  157|  17.9M|        return *this;
  158|  17.9M|    }
_ZN7jsonnet8internal13UStringStreamlsEPKDi:
  160|   553M|    {
  161|   553M|        buf.append(s);
  162|   553M|        return *this;
  163|   553M|    }
_ZN7jsonnet8internal13UStringStreamlsEDi:
  165|  35.6M|    {
  166|  35.6M|        buf.push_back(c);
  167|  35.6M|        return *this;
  168|  35.6M|    }
_ZN7jsonnet8internal13UStringStream3strEv:
  179|  3.63M|    {
  180|  3.63M|        return buf;
  181|  3.63M|    }
parser.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  31.0M|{
  141|  31.0M|    UString r;
  142|   212M|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 181M, False: 31.0M]
  ------------------
  143|   181M|        r.push_back(decode_utf8(s, i));
  144|  31.0M|    return r;
  145|  31.0M|}
parser.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|   181M|{
   76|   181M|    char c0 = str[i];
   77|   181M|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 180M, False: 1.19M]
  ------------------
   78|   180M|        return c0;
   79|   180M|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 121k, False: 1.07M]
  ------------------
   80|   121k|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 3.63k, False: 118k]
  ------------------
   81|  3.63k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  3.63k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|  3.63k|        }
   83|   118k|        char c1 = str[++i];
   84|   118k|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 110k, False: 7.53k]
  ------------------
   85|   110k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|   110k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|   110k|        }
   87|  7.53k|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|  1.07M|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 202k, False: 871k]
  ------------------
   89|   202k|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 1.84k, False: 200k]
  ------------------
   90|  1.84k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  1.84k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|  1.84k|        }
   92|   200k|        char c1 = str[++i];
   93|   200k|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 190k, False: 9.79k]
  ------------------
   94|   190k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|   190k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|   190k|        }
   96|  9.79k|        char c2 = str[++i];
   97|  9.79k|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 6.41k, False: 3.38k]
  ------------------
   98|  6.41k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  6.41k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|  6.41k|        }
  100|  3.38k|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|   871k|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 102k, False: 769k]
  ------------------
  102|   102k|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 599, False: 101k]
  ------------------
  103|    599|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|    599|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|    599|        }
  105|   101k|        char c1 = str[++i];
  106|   101k|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 38.8k, False: 62.9k]
  ------------------
  107|  38.8k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  38.8k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|  38.8k|        }
  109|  62.9k|        char c2 = str[++i];
  110|  62.9k|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 2.32k, False: 60.6k]
  ------------------
  111|  2.32k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  2.32k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|  2.32k|        }
  113|  60.6k|        char c3 = str[++i];
  114|  60.6k|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 1.48k, False: 59.1k]
  ------------------
  115|  1.48k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  1.48k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|  1.48k|        }
  117|  59.1k|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|   769k|    } else {
  119|   769k|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|   769k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|   769k|    }
  121|   181M|}
static_analysis.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEE:
  133|    543|{
  134|    543|    std::string r;
  135|    543|    encode_utf8(s, r);
  136|    543|    return r;
  137|    543|}
static_analysis.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEERNS2_IcNS3_IcEENS5_IcEEEE:
  127|    543|{
  128|    543|    for (char32_t cp : s)
  ------------------
  |  Branch (128:22): [True: 4.17M, False: 543]
  ------------------
  129|  4.17M|        encode_utf8(cp, r);
  130|    543|}
static_analysis.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|  4.17M|{
   34|  4.17M|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|  4.17M|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 0, False: 4.17M]
  ------------------
   35|      0|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|  4.17M|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|  4.17M|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 4.17M, False: 0]
  ------------------
   41|  4.17M|        s.push_back((char)x);
   42|  4.17M|        return 1;
   43|  4.17M|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 0, False: 0]
  ------------------
   44|      0|        bytes |= 0xC080;
   45|      0|        s.push_back((bytes >> 8) & 0xFF);
   46|      0|        s.push_back((bytes >> 0) & 0xFF);
   47|      0|        return 2;
   48|      0|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 0, False: 0]
  ------------------
   49|      0|        bytes |= 0xE08080;
   50|      0|        s.push_back((bytes >> 16) & 0xFF);
   51|      0|        s.push_back((bytes >> 8) & 0xFF);
   52|      0|        s.push_back((bytes >> 0) & 0xFF);
   53|      0|        return 3;
   54|      0|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 0, False: 0]
  ------------------
   55|      0|        bytes |= 0xF0808080;
   56|      0|        s.push_back((bytes >> 24) & 0xFF);
   57|      0|        s.push_back((bytes >> 16) & 0xFF);
   58|      0|        s.push_back((bytes >> 8) & 0xFF);
   59|      0|        s.push_back((bytes >> 0) & 0xFF);
   60|      0|        return 4;
   61|      0|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|  4.17M|}
string_utils.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  11.2M|{
  141|  11.2M|    UString r;
  142|  78.8M|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 67.5M, False: 11.2M]
  ------------------
  143|  67.5M|        r.push_back(decode_utf8(s, i));
  144|  11.2M|    return r;
  145|  11.2M|}
string_utils.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|  67.5M|{
   76|  67.5M|    char c0 = str[i];
   77|  67.5M|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 67.5M, False: 0]
  ------------------
   78|  67.5M|        return c0;
   79|  67.5M|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 0, False: 0]
  ------------------
   80|      0|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 0, False: 0]
  ------------------
   81|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|      0|        }
   83|      0|        char c1 = str[++i];
   84|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 0, False: 0]
  ------------------
   85|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|      0|        }
   87|      0|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|      0|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 0, False: 0]
  ------------------
   89|      0|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 0, False: 0]
  ------------------
   90|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|      0|        }
   92|      0|        char c1 = str[++i];
   93|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 0, False: 0]
  ------------------
   94|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|      0|        }
   96|      0|        char c2 = str[++i];
   97|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 0, False: 0]
  ------------------
   98|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|      0|        }
  100|      0|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|      0|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 0, False: 0]
  ------------------
  102|      0|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 0, False: 0]
  ------------------
  103|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|      0|        }
  105|      0|        char c1 = str[++i];
  106|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 0, False: 0]
  ------------------
  107|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|      0|        }
  109|      0|        char c2 = str[++i];
  110|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 0, False: 0]
  ------------------
  111|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|      0|        }
  113|      0|        char c3 = str[++i];
  114|      0|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 0, False: 0]
  ------------------
  115|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|      0|        }
  117|      0|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|      0|    } else {
  119|      0|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|      0|    }
  121|  67.5M|}
string_utils.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|    132|{
   34|    132|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|    132|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 3, False: 129]
  ------------------
   35|      3|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      3|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|    132|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|    132|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 69, False: 63]
  ------------------
   41|     69|        s.push_back((char)x);
   42|     69|        return 1;
   43|     69|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 8, False: 55]
  ------------------
   44|      8|        bytes |= 0xC080;
   45|      8|        s.push_back((bytes >> 8) & 0xFF);
   46|      8|        s.push_back((bytes >> 0) & 0xFF);
   47|      8|        return 2;
   48|     55|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 47, False: 8]
  ------------------
   49|     47|        bytes |= 0xE08080;
   50|     47|        s.push_back((bytes >> 16) & 0xFF);
   51|     47|        s.push_back((bytes >> 8) & 0xFF);
   52|     47|        s.push_back((bytes >> 0) & 0xFF);
   53|     47|        return 3;
   54|     47|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 8, False: 0]
  ------------------
   55|      8|        bytes |= 0xF0808080;
   56|      8|        s.push_back((bytes >> 24) & 0xFF);
   57|      8|        s.push_back((bytes >> 16) & 0xFF);
   58|      8|        s.push_back((bytes >> 8) & 0xFF);
   59|      8|        s.push_back((bytes >> 0) & 0xFF);
   60|      8|        return 4;
   61|      8|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|    132|}
vm.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  1.85M|{
  141|  1.85M|    UString r;
  142|  5.17M|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 3.31M, False: 1.85M]
  ------------------
  143|  3.31M|        r.push_back(decode_utf8(s, i));
  144|  1.85M|    return r;
  145|  1.85M|}
vm.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|  3.31M|{
   76|  3.31M|    char c0 = str[i];
   77|  3.31M|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 3.31M, False: 0]
  ------------------
   78|  3.31M|        return c0;
   79|  3.31M|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 0, False: 0]
  ------------------
   80|      0|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 0, False: 0]
  ------------------
   81|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|      0|        }
   83|      0|        char c1 = str[++i];
   84|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 0, False: 0]
  ------------------
   85|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|      0|        }
   87|      0|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|      0|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 0, False: 0]
  ------------------
   89|      0|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 0, False: 0]
  ------------------
   90|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|      0|        }
   92|      0|        char c1 = str[++i];
   93|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 0, False: 0]
  ------------------
   94|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|      0|        }
   96|      0|        char c2 = str[++i];
   97|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 0, False: 0]
  ------------------
   98|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|      0|        }
  100|      0|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|      0|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 0, False: 0]
  ------------------
  102|      0|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 0, False: 0]
  ------------------
  103|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|      0|        }
  105|      0|        char c1 = str[++i];
  106|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 0, False: 0]
  ------------------
  107|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|      0|        }
  109|      0|        char c2 = str[++i];
  110|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 0, False: 0]
  ------------------
  111|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|      0|        }
  113|      0|        char c3 = str[++i];
  114|      0|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 0, False: 0]
  ------------------
  115|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|      0|        }
  117|      0|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|      0|    } else {
  119|      0|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|      0|    }
  121|  3.31M|}
vm.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEE:
  133|   589k|{
  134|   589k|    std::string r;
  135|   589k|    encode_utf8(s, r);
  136|   589k|    return r;
  137|   589k|}
vm.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEERNS2_IcNS3_IcEENS5_IcEEEE:
  127|   589k|{
  128|   589k|    for (char32_t cp : s)
  ------------------
  |  Branch (128:22): [True: 530M, False: 589k]
  ------------------
  129|   530M|        encode_utf8(cp, r);
  130|   589k|}
vm.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|   530M|{
   34|   530M|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|   530M|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 997, False: 530M]
  ------------------
   35|    997|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|    997|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|   530M|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|   530M|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 530M, False: 182k]
  ------------------
   41|   530M|        s.push_back((char)x);
   42|   530M|        return 1;
   43|   530M|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 2.00k, False: 180k]
  ------------------
   44|  2.00k|        bytes |= 0xC080;
   45|  2.00k|        s.push_back((bytes >> 8) & 0xFF);
   46|  2.00k|        s.push_back((bytes >> 0) & 0xFF);
   47|  2.00k|        return 2;
   48|   180k|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 170k, False: 10.2k]
  ------------------
   49|   170k|        bytes |= 0xE08080;
   50|   170k|        s.push_back((bytes >> 16) & 0xFF);
   51|   170k|        s.push_back((bytes >> 8) & 0xFF);
   52|   170k|        s.push_back((bytes >> 0) & 0xFF);
   53|   170k|        return 3;
   54|   170k|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 10.2k, False: 0]
  ------------------
   55|  10.2k|        bytes |= 0xF0808080;
   56|  10.2k|        s.push_back((bytes >> 24) & 0xFF);
   57|  10.2k|        s.push_back((bytes >> 16) & 0xFF);
   58|  10.2k|        s.push_back((bytes >> 8) & 0xFF);
   59|  10.2k|        s.push_back((bytes >> 0) & 0xFF);
   60|  10.2k|        return 4;
   61|  10.2k|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|   530M|}
_ZN7jsonnet8internal13UStringStreamlsIiEERS1_T_:
  171|   819k|    {
  172|   819k|        std::stringstream ss;
  173|   819k|        ss << c;
  174|   819k|        for (char c : ss.str())
  ------------------
  |  Branch (174:21): [True: 1.57M, False: 819k]
  ------------------
  175|  1.57M|            buf.push_back(char32_t(c));
  176|   819k|        return *this;
  177|   819k|    }
_ZN7jsonnet8internal13UStringStreamlsIjEERS1_T_:
  171|   129k|    {
  172|   129k|        std::stringstream ss;
  173|   129k|        ss << c;
  174|   129k|        for (char c : ss.str())
  ------------------
  |  Branch (174:21): [True: 454k, False: 129k]
  ------------------
  175|   454k|            buf.push_back(char32_t(c));
  176|   129k|        return *this;
  177|   129k|    }
desugarer.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEE:
  133|   266k|{
  134|   266k|    std::string r;
  135|   266k|    encode_utf8(s, r);
  136|   266k|    return r;
  137|   266k|}
desugarer.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEERNS2_IcNS3_IcEENS5_IcEEEE:
  127|   266k|{
  128|   266k|    for (char32_t cp : s)
  ------------------
  |  Branch (128:22): [True: 1.78M, False: 266k]
  ------------------
  129|  1.78M|        encode_utf8(cp, r);
  130|   266k|}
desugarer.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|  1.78M|{
   34|  1.78M|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|  1.78M|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 0, False: 1.78M]
  ------------------
   35|      0|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|  1.78M|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|  1.78M|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 1.78M, False: 0]
  ------------------
   41|  1.78M|        s.push_back((char)x);
   42|  1.78M|        return 1;
   43|  1.78M|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 0, False: 0]
  ------------------
   44|      0|        bytes |= 0xC080;
   45|      0|        s.push_back((bytes >> 8) & 0xFF);
   46|      0|        s.push_back((bytes >> 0) & 0xFF);
   47|      0|        return 2;
   48|      0|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 0, False: 0]
  ------------------
   49|      0|        bytes |= 0xE08080;
   50|      0|        s.push_back((bytes >> 16) & 0xFF);
   51|      0|        s.push_back((bytes >> 8) & 0xFF);
   52|      0|        s.push_back((bytes >> 0) & 0xFF);
   53|      0|        return 3;
   54|      0|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 0, False: 0]
  ------------------
   55|      0|        bytes |= 0xF0808080;
   56|      0|        s.push_back((bytes >> 24) & 0xFF);
   57|      0|        s.push_back((bytes >> 16) & 0xFF);
   58|      0|        s.push_back((bytes >> 8) & 0xFF);
   59|      0|        s.push_back((bytes >> 0) & 0xFF);
   60|      0|        return 4;
   61|      0|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|  1.78M|}
desugarer.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  6.35k|{
  141|  6.35k|    UString r;
  142|  40.9k|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 34.5k, False: 6.35k]
  ------------------
  143|  34.5k|        r.push_back(decode_utf8(s, i));
  144|  6.35k|    return r;
  145|  6.35k|}
desugarer.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|  34.5k|{
   76|  34.5k|    char c0 = str[i];
   77|  34.5k|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 34.5k, False: 0]
  ------------------
   78|  34.5k|        return c0;
   79|  34.5k|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 0, False: 0]
  ------------------
   80|      0|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 0, False: 0]
  ------------------
   81|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|      0|        }
   83|      0|        char c1 = str[++i];
   84|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 0, False: 0]
  ------------------
   85|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|      0|        }
   87|      0|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|      0|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 0, False: 0]
  ------------------
   89|      0|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 0, False: 0]
  ------------------
   90|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|      0|        }
   92|      0|        char c1 = str[++i];
   93|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 0, False: 0]
  ------------------
   94|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|      0|        }
   96|      0|        char c2 = str[++i];
   97|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 0, False: 0]
  ------------------
   98|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|      0|        }
  100|      0|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|      0|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 0, False: 0]
  ------------------
  102|      0|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 0, False: 0]
  ------------------
  103|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|      0|        }
  105|      0|        char c1 = str[++i];
  106|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 0, False: 0]
  ------------------
  107|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|      0|        }
  109|      0|        char c2 = str[++i];
  110|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 0, False: 0]
  ------------------
  111|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|      0|        }
  113|      0|        char c3 = str[++i];
  114|      0|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 0, False: 0]
  ------------------
  115|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|      0|        }
  117|      0|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|      0|    } else {
  119|      0|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|      0|    }
  121|  34.5k|}

_ZN7jsonnet8internal18jsonnet_vm_executeEPNS0_9AllocatorEPKNS0_3ASTERKNSt3__13mapINS6_12basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEENS0_5VmExtENS6_4lessISD_EENSB_INS6_4pairIKSD_SE_EEEEEEjddRKNS7_ISD_NS0_16VmNativeCallbackESG_NSB_INSH_ISI_SO_EEEEEEPFiPvPKcSW_PPcSY_PmESU_b:
 3498|  2.88k|{
 3499|  2.88k|    Interpreter vm(alloc,
 3500|  2.88k|                   ext_vars,
 3501|  2.88k|                   max_stack,
 3502|  2.88k|                   gc_min_objects,
 3503|  2.88k|                   gc_growth_trigger,
 3504|  2.88k|                   natives,
 3505|  2.88k|                   import_callback,
 3506|  2.88k|                   ctx);
 3507|  2.88k|    vm.evaluate(ast, 0);
 3508|  2.88k|    if (string_output) {
  ------------------
  |  Branch (3508:9): [True: 0, False: 2.88k]
  ------------------
 3509|      0|        return encode_utf8(vm.manifestString(LocationRange("During manifestation")));
 3510|  2.88k|    } else {
 3511|  2.88k|        return encode_utf8(vm.manifestJson(LocationRange("During manifestation"), true, U""));
 3512|  2.88k|    }
 3513|  2.88k|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111InterpreterC2EPNS0_9AllocatorERKNSt3__13mapINS5_12basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEENS0_5VmExtENS5_4lessISC_EENSA_INS5_4pairIKSC_SD_EEEEEEjddRKNS6_ISC_NS0_16VmNativeCallbackESF_NSA_INSG_ISH_SN_EEEEEEPFiPvPKcSV_PPcSX_PmEST_:
  945|  2.88k|        : heap(gc_min_objects, gc_growth_trigger),
  946|  2.88k|          stack(max_stack),
  947|  2.88k|          alloc(alloc),
  948|  2.88k|          idImport(alloc->makeIdentifier(U"import")),
  949|  2.88k|          idArrayElement(alloc->makeIdentifier(U"array_element")),
  950|  2.88k|          idInvariant(alloc->makeIdentifier(U"object_assert")),
  951|  2.88k|          idInternal(alloc->makeIdentifier(U"__internal__")),
  952|  2.88k|          idJsonObjVar(alloc->makeIdentifier(U"_")),
  953|  2.88k|          idEmpty(alloc->makeIdentifier(U"")),
  954|  2.88k|          jsonObjVar(alloc->make<Var>(LocationRange(), Fodder{}, idJsonObjVar)),
  955|  2.88k|          externalVars(ext_vars),
  956|  2.88k|          importCallback(import_callback),
  957|  2.88k|          importCallbackContext(import_callback_context)
  958|  2.88k|    {
  959|  2.88k|        scratch = makeNull();
  960|       |        // Add a prefix to avoid conflicting with names from `native_callbacks`.
  961|  2.88k|        builtins["std:makeArray"] = &Interpreter::builtinMakeArray;
  962|  2.88k|        builtins["std:pow"] = &Interpreter::builtinPow;
  963|  2.88k|        builtins["std:floor"] = &Interpreter::builtinFloor;
  964|  2.88k|        builtins["std:ceil"] = &Interpreter::builtinCeil;
  965|  2.88k|        builtins["std:sqrt"] = &Interpreter::builtinSqrt;
  966|  2.88k|        builtins["std:sin"] = &Interpreter::builtinSin;
  967|  2.88k|        builtins["std:cos"] = &Interpreter::builtinCos;
  968|  2.88k|        builtins["std:tan"] = &Interpreter::builtinTan;
  969|  2.88k|        builtins["std:asin"] = &Interpreter::builtinAsin;
  970|  2.88k|        builtins["std:acos"] = &Interpreter::builtinAcos;
  971|  2.88k|        builtins["std:atan"] = &Interpreter::builtinAtan;
  972|  2.88k|        builtins["std:type"] = &Interpreter::builtinType;
  973|  2.88k|        builtins["std:filter"] = &Interpreter::builtinFilter;
  974|  2.88k|        builtins["std:objectHasEx"] = &Interpreter::builtinObjectHasEx;
  975|  2.88k|        builtins["std:length"] = &Interpreter::builtinLength;
  976|  2.88k|        builtins["std:objectFieldsEx"] = &Interpreter::builtinObjectFieldsEx;
  977|  2.88k|        builtins["std:objectRemoveKey"] = &Interpreter::builtinObjectRemoveKey;
  978|  2.88k|        builtins["std:codepoint"] = &Interpreter::builtinCodepoint;
  979|  2.88k|        builtins["std:char"] = &Interpreter::builtinChar;
  980|  2.88k|        builtins["std:log"] = &Interpreter::builtinLog;
  981|  2.88k|        builtins["std:exp"] = &Interpreter::builtinExp;
  982|  2.88k|        builtins["std:mantissa"] = &Interpreter::builtinMantissa;
  983|  2.88k|        builtins["std:exponent"] = &Interpreter::builtinExponent;
  984|  2.88k|        builtins["std:modulo"] = &Interpreter::builtinModulo;
  985|  2.88k|        builtins["std:extVar"] = &Interpreter::builtinExtVar;
  986|  2.88k|        builtins["std:primitiveEquals"] = &Interpreter::builtinPrimitiveEquals;
  987|  2.88k|        builtins["std:native"] = &Interpreter::builtinNative;
  988|  2.88k|        builtins["std:md5"] = &Interpreter::builtinMd5;
  989|  2.88k|        builtins["std:trace"] = &Interpreter::builtinTrace;
  990|  2.88k|        builtins["std:splitLimit"] = &Interpreter::builtinSplitLimit;
  991|  2.88k|        builtins["std:substr"] = &Interpreter::builtinSubstr;
  992|  2.88k|        builtins["std:range"] = &Interpreter::builtinRange;
  993|  2.88k|        builtins["std:strReplace"] = &Interpreter::builtinStrReplace;
  994|  2.88k|        builtins["std:asciiLower"] = &Interpreter::builtinAsciiLower;
  995|  2.88k|        builtins["std:asciiUpper"] = &Interpreter::builtinAsciiUpper;
  996|  2.88k|        builtins["std:join"] = &Interpreter::builtinJoin;
  997|  2.88k|        builtins["std:parseJson"] = &Interpreter::builtinParseJson;
  998|  2.88k|        builtins["std:parseYaml"] = &Interpreter::builtinParseYaml;
  999|  2.88k|        builtins["std:encodeUTF8"] = &Interpreter::builtinEncodeUTF8;
 1000|  2.88k|        builtins["std:decodeUTF8"] = &Interpreter::builtinDecodeUTF8;
 1001|  2.88k|        builtins["std:atan2"] = &Interpreter::builtinAtan2;
 1002|  2.88k|        builtins["std:hypot"] = &Interpreter::builtinHypot;
 1003|       |
 1004|       |        // Add a prefix `native:` to names of provided callbacks to ensure they can't clash with the builtins above.
 1005|       |        // Although we have separate lookup tables for them, the HeapClosure and BuiltinFunctionBody types just hold
 1006|       |        // function "name" as a std::string to identify the function.
 1007|  2.88k|        for (const auto& [name, cb] : native_callbacks) {
  ------------------
  |  Branch (1007:37): [True: 0, False: 2.88k]
  ------------------
 1008|      0|            nativeCallbacks.emplace("native:" + name, cb);
 1009|      0|        }
 1010|       |
 1011|  2.88k|        DesugaredObject *stdlib = makeStdlibAST(alloc, "__internal__");
 1012|  2.88k|        jsonnet_static_analysis(stdlib);
 1013|  2.88k|        stdlibAST = stdlib; // stdlibAST is const, so we need to do analysis before this assignment
 1014|  2.88k|        auto stdThunk = makeHeap<HeapThunk>(nullptr, nullptr, 0, static_cast<const AST*>(stdlibAST));
 1015|  2.88k|        stack.newCall(stdThunk->body->location, stdThunk, stdThunk->self, stdThunk->offset, stdThunk->upValues);
 1016|  2.88k|        evaluate(stdThunk->body, 0);
 1017|  2.88k|        stdObject = dynamic_cast<HeapObject*>(scratch.v.h);
 1018|  2.88k|        prepareSourceValThunks();
 1019|  2.88k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15StackC2Ej:
  258|  2.88k|    Stack(unsigned limit) : calls(0), limit(limit) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeNullEv:
  634|   108k|    {
  635|   108k|        Value r;
  636|   108k|        r.t = Value::NULL_TYPE;
  637|   108k|        return r;
  638|   108k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack3topEv:
  291|   664M|    {
  292|   664M|        return stack.back();
  293|   664M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter19validateBuiltinArgsERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEERKNS6_6vectorINS1_5ValueENSA_ISG_EEEENSF_INSG_4TypeENSA_ISL_EEEE:
 1043|   241k|    {
 1044|   241k|        if (args.size() == params.size()) {
  ------------------
  |  Branch (1044:13): [True: 241k, False: 0]
  ------------------
 1045|   692k|            for (std::size_t i = 0; i < args.size(); ++i) {
  ------------------
  |  Branch (1045:37): [True: 450k, False: 241k]
  ------------------
 1046|   450k|                if (args[i].t != params[i])
  ------------------
  |  Branch (1046:21): [True: 0, False: 450k]
  ------------------
 1047|      0|                    goto bad;
 1048|   450k|            }
 1049|   241k|            return;
 1050|   241k|        }
 1051|      0|    bad:;
 1052|      0|        std::stringstream ss;
 1053|      0|        ss << "Builtin function " + name + " expected (";
 1054|      0|        const char *prefix = "";
 1055|      0|        for (auto p : params) {
  ------------------
  |  Branch (1055:21): [True: 0, False: 0]
  ------------------
 1056|      0|            ss << prefix << type_str(p);
 1057|      0|            prefix = ", ";
 1058|      0|        }
 1059|      0|        ss << ") but got (";
 1060|      0|        prefix = "";
 1061|      0|        for (auto a : args) {
  ------------------
  |  Branch (1061:21): [True: 0, False: 0]
  ------------------
 1062|      0|            ss << prefix << type_str(a);
 1063|      0|            prefix = ", ";
 1064|      0|        }
 1065|      0|        ss << ")";
 1066|      0|        throw makeError(loc, ss.str());
 1067|   241k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter9makeErrorERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEE:
  567|  1.14k|    {
  568|  1.14k|        return stack.makeError(loc, msg);
  569|  1.14k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack9makeErrorERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEE:
  368|  1.71k|    {
  369|  1.71k|        std::vector<TraceFrame> stack_trace;
  370|  1.71k|        stack_trace.push_back(TraceFrame(loc));
  371|   549k|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (371:40): [True: 548k, False: 1.71k]
  ------------------
  372|   548k|            const auto &f = stack[i];
  373|   548k|            if (f.isCall()) {
  ------------------
  |  Branch (373:17): [True: 299k, False: 248k]
  ------------------
  374|   299k|                if (f.context != nullptr) {
  ------------------
  |  Branch (374:21): [True: 299k, False: 0]
  ------------------
  375|       |                    // Give the last line a name.
  376|   299k|                    stack_trace[stack_trace.size() - 1].name = getName(i, f.context);
  377|   299k|                }
  378|   299k|                if (f.location.isSet() || f.location.file.length() > 0)
  ------------------
  |  Branch (378:21): [True: 239k, False: 60.1k]
  |  Branch (378:43): [True: 829, False: 59.3k]
  ------------------
  379|   240k|                    stack_trace.push_back(TraceFrame(f.location));
  380|   299k|            }
  381|   548k|        }
  382|  1.71k|        return RuntimeError(stack_trace, msg);
  383|  1.71k|    }
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_15Frame6isCallEv:
  241|   818M|    {
  242|   818M|        return kind == FRAME_CALL;
  243|   818M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack7getNameEjPKNS1_10HeapEntityE:
  313|   299k|    {
  314|   299k|        std::string name;
  315|   538k|        for (int i = from_here - 1; i >= 0; --i) {
  ------------------
  |  Branch (315:37): [True: 536k, False: 1.67k]
  ------------------
  316|   536k|            const auto &f = stack[i];
  317|   536k|            for (const auto &pair : f.bindings) {
  ------------------
  |  Branch (317:35): [True: 526k, False: 536k]
  ------------------
  318|   526k|                HeapThunk *thunk = pair.second;
  319|   526k|                if (!thunk->filled)
  ------------------
  |  Branch (319:21): [True: 219k, False: 307k]
  ------------------
  320|   219k|                    continue;
  321|   307k|                if (!thunk->content.isHeap())
  ------------------
  |  Branch (321:21): [True: 60.2k, False: 246k]
  ------------------
  322|  60.2k|                    continue;
  323|   246k|                if (e != thunk->content.v.h)
  ------------------
  |  Branch (323:21): [True: 187k, False: 59.4k]
  ------------------
  324|   187k|                    continue;
  325|  59.4k|                name = encode_utf8(pair.first->name);
  326|  59.4k|            }
  327|       |            // Do not go into the next call frame, keep local reasoning.
  328|   536k|            if (f.isCall())
  ------------------
  |  Branch (328:17): [True: 298k, False: 238k]
  ------------------
  329|   298k|                break;
  330|   536k|        }
  331|       |
  332|   299k|        if (name == "")
  ------------------
  |  Branch (332:13): [True: 240k, False: 59.4k]
  ------------------
  333|   240k|            name = "anonymous";
  334|   299k|        if (dynamic_cast<const HeapObject *>(e)) {
  ------------------
  |  Branch (334:13): [True: 166k, False: 132k]
  ------------------
  335|   166k|            return "object <" + name + ">";
  336|   166k|        } else if (auto *thunk = dynamic_cast<const HeapThunk *>(e)) {
  ------------------
  |  Branch (336:26): [True: 69.0k, False: 63.8k]
  ------------------
  337|  69.0k|            if (thunk->name == nullptr) {
  ------------------
  |  Branch (337:17): [True: 5.13k, False: 63.9k]
  ------------------
  338|  5.13k|                return "";  // Argument of builtin, or root (since top level functions).
  339|  63.9k|            } else {
  340|  63.9k|                return "thunk <" + encode_utf8(thunk->name->name) + ">";
  341|  63.9k|            }
  342|  69.0k|        } else {
  343|  63.8k|            const auto *func = static_cast<const HeapClosure *>(e);
  344|  63.8k|            if (func->isBuiltin()) {
  ------------------
  |  Branch (344:17): [True: 0, False: 63.8k]
  ------------------
  345|      0|                return "builtin function <" + func->builtinName + ">";
  346|      0|            }
  347|  63.8k|            return "function <" + name + ">";
  348|  63.8k|        }
  349|   299k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack4markERNS1_4HeapE:
  284|   126k|    {
  285|  22.6M|        for (const auto &f : stack) {
  ------------------
  |  Branch (285:28): [True: 22.6M, False: 126k]
  ------------------
  286|  22.6M|            f.mark(heap);
  287|  22.6M|        }
  288|   126k|    }
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_15Frame4markERNS1_4HeapE:
  225|  22.6M|    {
  226|  22.6M|        heap.markFrom(val);
  227|  22.6M|        heap.markFrom(val2);
  228|  22.6M|        if (context)
  ------------------
  |  Branch (228:13): [True: 15.5M, False: 7.10M]
  ------------------
  229|  15.5M|            heap.markFrom(context);
  230|  22.6M|        if (self)
  ------------------
  |  Branch (230:13): [True: 15.5M, False: 7.08M]
  ------------------
  231|  15.5M|            heap.markFrom(self);
  232|  22.6M|        for (const auto &bind : bindings)
  ------------------
  |  Branch (232:31): [True: 26.3M, False: 22.6M]
  ------------------
  233|  26.3M|            heap.markFrom(bind.second);
  234|  22.6M|        for (const auto &el : elements)
  ------------------
  |  Branch (234:29): [True: 0, False: 22.6M]
  ------------------
  235|      0|            heap.markFrom(el.second);
  236|  22.6M|        for (const auto &th : thunks)
  ------------------
  |  Branch (236:29): [True: 1.62M, False: 22.6M]
  ------------------
  237|  1.62M|            heap.markFrom(th);
  238|  22.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeNumberEd:
  615|  7.13M|    {
  616|  7.13M|        Value r;
  617|  7.13M|        r.t = Value::NUMBER;
  618|  7.13M|        r.v.d = v;
  619|  7.13M|        return r;
  620|  7.13M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter9makeArrayERKNSt3__16vectorIPNS1_9HeapThunkENS3_9allocatorIS6_EEEE:
  641|   639k|    {
  642|   639k|        Value r;
  643|   639k|        r.t = Value::ARRAY;
  644|   639k|        r.v.h = makeHeap<HeapArray>(v);
  645|   639k|        return r;
  646|   639k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapArrayEJRKNSt3__16vectorIPNS1_9HeapThunkENS5_9allocatorIS8_EEEEEEEPT_DpOT0_:
  577|   639k|    {
  578|   639k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   639k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 1.19k, False: 637k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  1.19k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  1.19k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  1.19k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  1.19k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 1.19k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   192k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 192k, False: 1.19k]
  ------------------
  597|   192k|                heap.markFrom(sourceVal.second);
  598|   192k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  1.19k|            heap.sweep();
  602|  1.19k|        }
  603|   639k|        return r;
  604|   639k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10builtinPowERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1103|  4.39k|    {
 1104|  4.39k|        validateBuiltinArgs(loc, "pow", args, {Value::NUMBER, Value::NUMBER});
 1105|  4.39k|        scratch = makeNumberCheck(loc, std::pow(args[0].v.d, args[1].v.d));
 1106|  4.39k|        return nullptr;
 1107|  4.39k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter15makeNumberCheckERKNS0_13LocationRangeEd:
  623|  6.35M|    {
  624|  6.35M|        if (std::isnan(v)) {
  ------------------
  |  Branch (624:13): [True: 0, False: 6.35M]
  ------------------
  625|      0|            throw makeError(loc, "not a number");
  626|      0|        }
  627|  6.35M|        if (std::isinf(v)) {
  ------------------
  |  Branch (627:13): [True: 32, False: 6.35M]
  ------------------
  628|     32|            throw makeError(loc, "overflow");
  629|     32|        }
  630|  6.35M|        return makeNumber(v);
  631|  6.35M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter12builtinFloorERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1110|  29.2k|    {
 1111|  29.2k|        validateBuiltinArgs(loc, "floor", args, {Value::NUMBER});
 1112|  29.2k|        scratch = makeNumberCheck(loc, std::floor(args[0].v.d));
 1113|  29.2k|        return nullptr;
 1114|  29.2k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11builtinTypeERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1187|  3.52M|    {
 1188|  3.52M|        switch (args[0].t) {
  ------------------
  |  Branch (1188:17): [True: 3.52M, False: 0]
  ------------------
 1189|   142k|            case Value::NULL_TYPE: scratch = makeString(U"null"); return nullptr;
  ------------------
  |  Branch (1189:13): [True: 142k, False: 3.38M]
  ------------------
 1190|       |
 1191|   114k|            case Value::BOOLEAN: scratch = makeString(U"boolean"); return nullptr;
  ------------------
  |  Branch (1191:13): [True: 114k, False: 3.40M]
  ------------------
 1192|       |
 1193|   386k|            case Value::NUMBER: scratch = makeString(U"number"); return nullptr;
  ------------------
  |  Branch (1193:13): [True: 386k, False: 3.13M]
  ------------------
 1194|       |
 1195|   167k|            case Value::ARRAY: scratch = makeString(U"array"); return nullptr;
  ------------------
  |  Branch (1195:13): [True: 167k, False: 3.35M]
  ------------------
 1196|       |
 1197|  3.18k|            case Value::FUNCTION: scratch = makeString(U"function"); return nullptr;
  ------------------
  |  Branch (1197:13): [True: 3.18k, False: 3.52M]
  ------------------
 1198|       |
 1199|   324k|            case Value::OBJECT: scratch = makeString(U"object"); return nullptr;
  ------------------
  |  Branch (1199:13): [True: 324k, False: 3.19M]
  ------------------
 1200|       |
 1201|  2.38M|            case Value::STRING: scratch = makeString(U"string"); return nullptr;
  ------------------
  |  Branch (1201:13): [True: 2.38M, False: 1.13M]
  ------------------
 1202|  3.52M|        }
 1203|      0|        return nullptr;  // Quiet, compiler.
 1204|  3.52M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeStringERKNSt3__112basic_stringIDiNS3_11char_traitsIDiEENS3_9allocatorIDiEEEE:
  679|  40.5M|    {
  680|  40.5M|        Value r;
  681|  40.5M|        r.t = Value::STRING;
  682|  40.5M|        r.v.h = makeHeap<HeapString>(v);
  683|  40.5M|        return r;
  684|  40.5M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_10HeapStringEJRKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEEEEEPT_DpOT0_:
  577|  40.5M|    {
  578|  40.5M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  40.5M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 51.6k, False: 40.4M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  51.6k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  51.6k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  51.6k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  51.6k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 51.6k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  8.32M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 8.32M, False: 51.6k]
  ------------------
  597|  8.32M|                heap.markFrom(sourceVal.second);
  598|  8.32M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  51.6k|            heap.sweep();
  602|  51.6k|        }
  603|  40.5M|        return r;
  604|  40.5M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter18builtinObjectHasExERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1234|    842|    {
 1235|    842|        validateBuiltinArgs(
 1236|    842|            loc, "objectHasEx", args, {Value::OBJECT, Value::STRING, Value::BOOLEAN});
 1237|    842|        const auto *obj = static_cast<const HeapObject *>(args[0].v.h);
 1238|    842|        const auto *str = static_cast<const HeapString *>(args[1].v.h);
 1239|    842|        bool include_hidden = args[2].v.b;
 1240|    842|        bool found = false;
 1241|  2.28k|        for (const auto &field : objectFields(obj, !include_hidden)) {
  ------------------
  |  Branch (1241:32): [True: 2.28k, False: 32]
  ------------------
 1242|  2.28k|            if (field->name == str->value) {
  ------------------
  |  Branch (1242:17): [True: 810, False: 1.47k]
  ------------------
 1243|    810|                found = true;
 1244|    810|                break;
 1245|    810|            }
 1246|  2.28k|        }
 1247|    842|        scratch = makeBoolean(found);
 1248|    842|        return nullptr;
 1249|    842|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter12objectFieldsEPKNS1_10HeapObjectEb:
  773|   371k|    {
  774|   371k|        std::set<const Identifier *> r;
  775|  1.03M|        for (const auto &pair : objectFieldsAux(obj_)) {
  ------------------
  |  Branch (775:31): [True: 1.03M, False: 371k]
  ------------------
  776|  1.03M|            if (!manifesting || pair.second != ObjectField::HIDDEN)
  ------------------
  |  Branch (776:17): [True: 3.77k, False: 1.02M]
  |  Branch (776:33): [True: 892k, False: 136k]
  ------------------
  777|   896k|                r.insert(pair.first);
  778|  1.03M|        }
  779|   371k|        return r;
  780|   371k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter15objectFieldsAuxEPKNS1_10HeapObjectE:
  740|  17.6M|    {
  741|  17.6M|        IdHideMap r;
  742|  17.6M|        if (auto *obj = dynamic_cast<const HeapSimpleObject *>(obj_)) {
  ------------------
  |  Branch (742:19): [True: 9.03M, False: 8.66M]
  ------------------
  743|  9.03M|            for (const auto &f : obj->fields) {
  ------------------
  |  Branch (743:32): [True: 7.04M, False: 9.03M]
  ------------------
  744|  7.04M|                r[f.first] = f.second.hide;
  745|  7.04M|            }
  746|       |
  747|  9.03M|        } else if (auto *obj = dynamic_cast<const HeapRestrictedObject *>(obj_)) {
  ------------------
  |  Branch (747:26): [True: 0, False: 8.66M]
  ------------------
  748|      0|            return obj->retainedKeys;
  749|       |
  750|  8.66M|        } else if (auto *obj = dynamic_cast<const HeapExtendedObject *>(obj_)) {
  ------------------
  |  Branch (750:26): [True: 8.66M, False: 0]
  ------------------
  751|  8.66M|            r = objectFieldsAux(obj->right);
  752|  20.3M|            for (const auto &pair : objectFieldsAux(obj->left)) {
  ------------------
  |  Branch (752:35): [True: 20.3M, False: 8.66M]
  ------------------
  753|  20.3M|                auto it = r.find(pair.first);
  754|  20.3M|                if (it == r.end()) {
  ------------------
  |  Branch (754:21): [True: 14.3M, False: 6.01M]
  ------------------
  755|       |                    // First time it is seen
  756|  14.3M|                    r[pair.first] = pair.second;
  757|  14.3M|                } else if (it->second == ObjectField::INHERIT) {
  ------------------
  |  Branch (757:28): [True: 5.67M, False: 338k]
  ------------------
  758|       |                    // Seen before, but with inherited visibility so use new visibility
  759|  5.67M|                    r[pair.first] = pair.second;
  760|  5.67M|                }
  761|  20.3M|            }
  762|       |
  763|  8.66M|        } else if (auto *obj = dynamic_cast<const HeapComprehensionObject *>(obj_)) {
  ------------------
  |  Branch (763:26): [True: 0, False: 0]
  ------------------
  764|      0|            for (const auto &f : obj->compValues)
  ------------------
  |  Branch (764:32): [True: 0, False: 0]
  ------------------
  765|      0|                r[f.first] = ObjectField::INHERIT;
  766|      0|        }
  767|  17.6M|        return r;
  768|  17.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11makeBooleanEb:
  607|  11.5M|    {
  608|  11.5M|        Value r;
  609|  11.5M|        r.t = Value::BOOLEAN;
  610|  11.5M|        r.v.b = v;
  611|  11.5M|        return r;
  612|  11.5M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13builtinLengthERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1252|   654k|    {
 1253|   654k|        if (args.size() != 1) {
  ------------------
  |  Branch (1253:13): [True: 0, False: 654k]
  ------------------
 1254|      0|            throw makeError(loc, "length takes 1 parameter.");
 1255|      0|        }
 1256|   654k|        HeapEntity *e = args[0].v.h;
 1257|   654k|        switch (args[0].t) {
 1258|      0|            case Value::OBJECT: {
  ------------------
  |  Branch (1258:13): [True: 0, False: 654k]
  ------------------
 1259|      0|                auto fields = objectFields(static_cast<HeapObject *>(e), true);
 1260|      0|                scratch = makeNumber(fields.size());
 1261|      0|            } break;
 1262|       |
 1263|   274k|            case Value::ARRAY:
  ------------------
  |  Branch (1263:13): [True: 274k, False: 380k]
  ------------------
 1264|   274k|                scratch = makeNumber(static_cast<HeapArray *>(e)->elements.size());
 1265|   274k|                break;
 1266|       |
 1267|   380k|            case Value::STRING:
  ------------------
  |  Branch (1267:13): [True: 380k, False: 274k]
  ------------------
 1268|   380k|                scratch = makeNumber(static_cast<HeapString *>(e)->value.length());
 1269|   380k|                break;
 1270|       |
 1271|      0|            case Value::FUNCTION:
  ------------------
  |  Branch (1271:13): [True: 0, False: 654k]
  ------------------
 1272|      0|                scratch = makeNumber(static_cast<HeapClosure *>(e)->params.size());
 1273|      0|                break;
 1274|       |
 1275|      0|            default:
  ------------------
  |  Branch (1275:13): [True: 0, False: 654k]
  ------------------
 1276|      0|                throw makeError(loc,
 1277|      0|                                "length operates on strings, objects, "
 1278|      0|                                "and arrays, got " +
 1279|      0|                                    type_str(args[0]));
 1280|   654k|        }
 1281|   654k|        return nullptr;
 1282|   654k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter21builtinObjectFieldsExERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1285|   159k|    {
 1286|   159k|        validateBuiltinArgs(loc, "objectFieldsEx", args, {Value::OBJECT, Value::BOOLEAN});
 1287|   159k|        const auto *obj = static_cast<HeapObject *>(args[0].v.h);
 1288|   159k|        bool include_hidden = args[1].v.b;
 1289|       |        // Stash in a set first to sort them.
 1290|   159k|        std::set<UString> fields;
 1291|   498k|        for (const auto &field : objectFields(obj, !include_hidden)) {
  ------------------
  |  Branch (1291:32): [True: 498k, False: 159k]
  ------------------
 1292|   498k|            fields.insert(field->name);
 1293|   498k|        }
 1294|   159k|        scratch = makeArray({});
 1295|   159k|        auto &elements = static_cast<HeapArray *>(scratch.v.h)->elements;
 1296|   498k|        for (const auto &field : fields) {
  ------------------
  |  Branch (1296:32): [True: 498k, False: 159k]
  ------------------
 1297|   498k|            auto *th = makeHeap<HeapThunk>(idArrayElement, nullptr, 0, nullptr);
 1298|   498k|            elements.push_back(th);
 1299|   498k|            th->fill(makeString(field));
 1300|   498k|        }
 1301|   159k|        return nullptr;
 1302|   159k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierEDniDnEEEPT_DpOT0_:
  577|   498k|    {
  578|   498k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   498k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 785, False: 497k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|    785|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|    785|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|    785|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|    785|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 785]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   126k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 126k, False: 785]
  ------------------
  597|   126k|                heap.markFrom(sourceVal.second);
  598|   126k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|    785|            heap.sweep();
  602|    785|        }
  603|   498k|        return r;
  604|   498k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11builtinCharERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1335|  1.49k|    {
 1336|  1.49k|        validateBuiltinArgs(loc, "char", args, {Value::NUMBER});
 1337|  1.49k|        long l = long(args[0].v.d);
 1338|  1.49k|        if (l < 0) {
  ------------------
  |  Branch (1338:13): [True: 46, False: 1.44k]
  ------------------
 1339|     46|            std::stringstream ss;
 1340|     46|            ss << "codepoints must be >= 0, got " << l;
 1341|     46|            throw makeError(loc, ss.str());
 1342|     46|        }
 1343|  1.44k|        if (l >= JSONNET_CODEPOINT_MAX) {
  ------------------
  |  |   22|  1.44k|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (1343:13): [True: 27, False: 1.41k]
  ------------------
 1344|     27|            std::stringstream ss;
 1345|     27|            ss << "invalid unicode codepoint, got " << l;
 1346|     27|            throw makeError(loc, ss.str());
 1347|     27|        }
 1348|  1.41k|        char32_t c = l;
 1349|  1.41k|        scratch = makeString(UString(&c, 1));
 1350|  1.41k|        return nullptr;
 1351|  1.44k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10builtinLogERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1354|  4.06k|    {
 1355|  4.06k|        validateBuiltinArgs(loc, "log", args, {Value::NUMBER});
 1356|  4.06k|        scratch = makeNumberCheck(loc, std::log(args[0].v.d));
 1357|  4.06k|        return nullptr;
 1358|  4.06k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13builtinModuloERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1386|  41.3k|    {
 1387|  41.3k|        validateBuiltinArgs(loc, "modulo", args, {Value::NUMBER, Value::NUMBER});
 1388|  41.3k|        double a = args[0].v.d;
 1389|  41.3k|        double b = args[1].v.d;
 1390|  41.3k|        if (b == 0)
  ------------------
  |  Branch (1390:13): [True: 20, False: 41.2k]
  ------------------
 1391|     20|            throw makeError(loc, "division by zero.");
 1392|  41.2k|        scratch = makeNumberCheck(loc, std::fmod(a, b));
 1393|  41.2k|        return nullptr;
 1394|  41.3k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack3popEv:
  301|   125M|    {
  302|   125M|        if (top().isCall())
  ------------------
  |  Branch (302:13): [True: 49.3M, False: 76.4M]
  ------------------
  303|  49.3M|            calls--;
  304|   125M|        stack.pop_back();
  305|   125M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter22builtinPrimitiveEqualsERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1422|  5.49M|    {
 1423|  5.49M|        if (args.size() != 2) {
  ------------------
  |  Branch (1423:13): [True: 0, False: 5.49M]
  ------------------
 1424|      0|            std::stringstream ss;
 1425|      0|            ss << "primitiveEquals takes 2 parameters, got " << args.size();
 1426|      0|            throw makeError(loc, ss.str());
 1427|      0|        }
 1428|  5.49M|        if (args[0].t != args[1].t) {
  ------------------
  |  Branch (1428:13): [True: 0, False: 5.49M]
  ------------------
 1429|      0|            scratch = makeBoolean(false);
 1430|      0|            return nullptr;
 1431|      0|        }
 1432|  5.49M|        bool r;
 1433|  5.49M|        switch (args[0].t) {
 1434|  9.54k|            case Value::BOOLEAN: r = args[0].v.b == args[1].v.b; break;
  ------------------
  |  Branch (1434:13): [True: 9.54k, False: 5.48M]
  ------------------
 1435|       |
 1436|   134k|            case Value::NUMBER: r = args[0].v.d == args[1].v.d; break;
  ------------------
  |  Branch (1436:13): [True: 134k, False: 5.36M]
  ------------------
 1437|       |
 1438|  5.30M|            case Value::STRING:
  ------------------
  |  Branch (1438:13): [True: 5.30M, False: 186k]
  ------------------
 1439|  5.30M|                r = static_cast<HeapString *>(args[0].v.h)->value ==
 1440|  5.30M|                    static_cast<HeapString *>(args[1].v.h)->value;
 1441|  5.30M|                break;
 1442|       |
 1443|  43.0k|            case Value::NULL_TYPE: r = true; break;
  ------------------
  |  Branch (1443:13): [True: 43.0k, False: 5.45M]
  ------------------
 1444|       |
 1445|      0|            case Value::FUNCTION: throw makeError(loc, "cannot test equality of functions"); break;
  ------------------
  |  Branch (1445:13): [True: 0, False: 5.49M]
  ------------------
 1446|       |
 1447|      0|            default:
  ------------------
  |  Branch (1447:13): [True: 0, False: 5.49M]
  ------------------
 1448|      0|                throw makeError(loc,
 1449|      0|                                "primitiveEquals operates on primitive "
 1450|      0|                                "types, got " +
 1451|      0|                                    type_str(args[0]));
 1452|  5.49M|        }
 1453|  5.49M|        scratch = makeBoolean(r);
 1454|  5.49M|        return nullptr;
 1455|  5.49M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter18makeBuiltinFromASTEPKNS0_19BuiltinFunctionBodyE:
  658|  9.92M|    {
  659|  9.92M|        HeapClosure::Params hc_params;
  660|  15.6M|        for (const auto p : body->params) {
  ------------------
  |  Branch (660:27): [True: 15.6M, False: 9.92M]
  ------------------
  661|  15.6M|            hc_params.emplace_back(p, nullptr);
  662|  15.6M|        }
  663|  9.92M|        Value r;
  664|  9.92M|        r.t = Value::FUNCTION;
  665|  9.92M|        r.v.h = makeHeap<HeapClosure>(BindingFrame(), nullptr, 0, hc_params, body, body->name);
  666|  9.92M|        return r;
  667|  9.92M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_11HeapClosureEJNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEEDniRNS5_6vectorINS4_5ParamENSE_ISL_EEEERPKNS0_19BuiltinFunctionBodyERKNS5_12basic_stringIcNS5_11char_traitsIcEENSE_IcEEEEEEEPT_DpOT0_:
  577|  9.92M|    {
  578|  9.92M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  9.92M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 10.7k, False: 9.91M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  10.7k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  10.7k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  10.7k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  10.7k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 10.7k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  1.72M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 1.72M, False: 10.7k]
  ------------------
  597|  1.72M|                heap.markFrom(sourceVal.second);
  598|  1.72M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  10.7k|            heap.sweep();
  602|  10.7k|        }
  603|  9.92M|        return r;
  604|  9.92M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13builtinSubstrERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1599|    564|    {
 1600|    564|        validateBuiltinArgs(loc, "substr", args, {Value::STRING, Value::NUMBER, Value::NUMBER});
 1601|    564|        const auto *str = static_cast<const HeapString *>(args[0].v.h);
 1602|    564|        long from = long(args[1].v.d);
 1603|    564|        long len = long(args[2].v.d);
 1604|    564|        if (from < 0) {
  ------------------
  |  Branch (1604:13): [True: 0, False: 564]
  ------------------
 1605|      0|            std::stringstream ss;
 1606|      0|            ss << "substr second parameter should be greater than zero, got " << from;
 1607|      0|            throw makeError(loc, ss.str());
 1608|      0|        }
 1609|    564|        if (len < 0) {
  ------------------
  |  Branch (1609:13): [True: 0, False: 564]
  ------------------
 1610|      0|            std::stringstream ss;
 1611|      0|            ss << "substr third parameter should be greater than zero, got " << len;
 1612|      0|            throw makeError(loc, ss.str());
 1613|      0|        }
 1614|    564|        if (static_cast<unsigned long>(from) > str->value.size()) {
  ------------------
  |  Branch (1614:13): [True: 0, False: 564]
  ------------------
 1615|      0|            scratch = makeString(UString());
 1616|      0|            return nullptr;
 1617|      0|        }
 1618|    564|        if (size_t(len + from) > str->value.size()) {
  ------------------
  |  Branch (1618:13): [True: 0, False: 564]
  ------------------
 1619|      0|          len = str->value.size() - from;
 1620|      0|        }
 1621|    564|        scratch = makeString(str->value.substr(from, len));
 1622|    564|        return nullptr;
 1623|    564|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJDnDniPKNS0_3ASTEEEEPT_DpOT0_:
  577|  2.88k|    {
  578|  2.88k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  2.88k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 0, False: 2.88k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|      0|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|      0|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|      0|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|      0|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 0]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|      0|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 0, False: 0]
  ------------------
  597|      0|                heap.markFrom(sourceVal.second);
  598|      0|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|      0|            heap.sweep();
  602|      0|        }
  603|  2.88k|        return r;
  604|  2.88k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack7newCallERKNS0_13LocationRangeEPNS1_10HeapEntityEPNS1_10HeapObjectEjRKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENSA_4lessISE_EENSA_9allocatorINSA_4pairIKSE_SG_EEEEEE:
  418|  49.9M|    {
  419|  49.9M|        tailCallTrimStack();
  420|  49.9M|        if (calls >= limit) {
  ------------------
  |  Branch (420:13): [True: 570, False: 49.9M]
  ------------------
  421|    570|            throw makeError(loc, "max stack frames exceeded.");
  422|    570|        }
  423|  49.9M|        stack.emplace_back(FRAME_CALL, loc);
  424|  49.9M|        calls++;
  425|  49.9M|        top().context = context;
  426|  49.9M|        top().self = self;
  427|  49.9M|        top().offset = offset;
  428|  49.9M|        top().bindings = up_values;
  429|  49.9M|        top().tailCall = false;
  430|       |
  431|  49.9M|#ifndef NDEBUG
  432|  59.4M|        for (const auto &bind : up_values) {
  ------------------
  |  Branch (432:31): [True: 59.4M, False: 49.9M]
  ------------------
  433|  59.4M|            if (bind.second == nullptr) {
  ------------------
  |  Branch (433:17): [True: 0, False: 59.4M]
  ------------------
  434|      0|                std::cerr << "INTERNAL ERROR: No binding for variable "
  435|      0|                          << encode_utf8(bind.first->name) << std::endl;
  436|      0|                std::abort();
  437|      0|            }
  438|  59.4M|        }
  439|  49.9M|#endif
  440|  49.9M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack17tailCallTrimStackEv:
  394|  49.9M|    {
  395|  50.8M|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (395:40): [True: 50.6M, False: 198k]
  ------------------
  396|  50.6M|            switch (stack[i].kind) {
  397|  13.8M|                case FRAME_CALL: {
  ------------------
  |  Branch (397:17): [True: 13.8M, False: 36.7M]
  ------------------
  398|  13.8M|                    if (!stack[i].tailCall || stack[i].thunks.size() > 0) {
  ------------------
  |  Branch (398:25): [True: 9.49M, False: 4.35M]
  |  Branch (398:47): [True: 4.11M, False: 239k]
  ------------------
  399|  13.6M|                        return;
  400|  13.6M|                    }
  401|       |                    // Remove all stack frames including this one.
  402|   830k|                    while (stack.size() > unsigned(i))
  ------------------
  |  Branch (402:28): [True: 590k, False: 239k]
  ------------------
  403|   590k|                        stack.pop_back();
  404|   239k|                    calls--;
  405|   239k|                    return;
  406|  13.8M|                } break;
  407|       |
  408|   887k|                case FRAME_LOCAL: break;
  ------------------
  |  Branch (408:17): [True: 887k, False: 49.7M]
  ------------------
  409|       |
  410|  35.8M|                default: return;
  ------------------
  |  Branch (410:17): [True: 35.8M, False: 14.7M]
  ------------------
  411|  50.6M|            }
  412|  50.6M|        }
  413|  49.9M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15FrameC2ERKNS1_9FrameKindERKNS0_13LocationRangeE:
  209|  62.8M|        : kind(kind),
  210|  62.8M|          ast(nullptr),
  211|  62.8M|          location(location),
  212|  62.8M|          tailCall(false),
  213|  62.8M|          elementId(0),
  214|  62.8M|          first(false),
  215|  62.8M|          context(NULL),
  216|  62.8M|          self(NULL),
  217|  62.8M|          offset(0)
  218|  62.8M|    {
  219|  62.8M|        val.t = Value::NULL_TYPE;
  220|  62.8M|        val2.t = Value::NULL_TYPE;
  221|  62.8M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter22prepareSourceValThunksEv:
  887|  2.88k|    void prepareSourceValThunks() {
  888|   464k|        for (const auto &field : stdlibAST->fields) {
  ------------------
  |  Branch (888:32): [True: 464k, False: 2.88k]
  ------------------
  889|   464k|            AST *nameAST = field.name;
  890|   464k|            if (nameAST->type != AST_LITERAL_STRING) {
  ------------------
  |  Branch (890:17): [True: 0, False: 464k]
  ------------------
  891|       |                // Skip any fields without a known name.
  892|      0|                continue;
  893|      0|            }
  894|   464k|            UString name = dynamic_cast<LiteralString *>(nameAST)->value;
  895|       |
  896|   464k|            sourceFuncIds.emplace_back(new Identifier(name));
  897|   464k|            auto *th = makeHeap<HeapThunk>(sourceFuncIds.back().get(), stdObject, 0, field.body);
  898|   464k|            sourceVals[encode_utf8(name)] = th;
  899|   464k|        }
  900|  2.88k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJPNS0_10IdentifierERPNS1_10HeapObjectEiRKPNS0_3ASTEEEEPT_DpOT0_:
  577|   464k|    {
  578|   464k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   464k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 0, False: 464k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|      0|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|      0|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|      0|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|      0|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 0]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|      0|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 0, False: 0]
  ------------------
  597|      0|                heap.markFrom(sourceVal.second);
  598|      0|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|      0|            heap.sweep();
  602|      0|        }
  603|   464k|        return r;
  604|   464k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15StackD2Ev:
  260|  2.88k|    ~Stack(void) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8evaluateEPKNS0_3ASTEj:
 2096|  2.26M|    {
 2097|   146M|    recurse:
 2098|       |
 2099|   146M|        switch (ast_->type) {
 2100|  13.6M|            case AST_APPLY: {
  ------------------
  |  Branch (2100:13): [True: 13.6M, False: 132M]
  ------------------
 2101|  13.6M|                const auto &ast = *static_cast<const Apply *>(ast_);
 2102|  13.6M|                stack.newFrame(FRAME_APPLY_TARGET, ast_);
 2103|  13.6M|                ast_ = ast.target;
 2104|  13.6M|                goto recurse;
 2105|      0|            } break;
 2106|       |
 2107|   290k|            case AST_ARRAY: {
  ------------------
  |  Branch (2107:13): [True: 290k, False: 146M]
  ------------------
 2108|   290k|                const auto &ast = *static_cast<const Array *>(ast_);
 2109|   290k|                HeapObject *self;
 2110|   290k|                unsigned offset;
 2111|   290k|                stack.getSelfBinding(self, offset);
 2112|   290k|                scratch = makeArray({});
 2113|   290k|                auto &elements = static_cast<HeapArray *>(scratch.v.h)->elements;
 2114|  3.20M|                for (const auto &el : ast.elements) {
  ------------------
  |  Branch (2114:37): [True: 3.20M, False: 290k]
  ------------------
 2115|  3.20M|                    auto *el_th = makeHeap<HeapThunk>(idArrayElement, self, offset, el.expr);
 2116|  3.20M|                    el_th->upValues = capture(el.expr->freeVariables);
 2117|  3.20M|                    elements.push_back(el_th);
 2118|  3.20M|                }
 2119|   290k|            } break;
 2120|       |
 2121|  6.86M|            case AST_BINARY: {
  ------------------
  |  Branch (2121:13): [True: 6.86M, False: 139M]
  ------------------
 2122|  6.86M|                const auto &ast = *static_cast<const Binary *>(ast_);
 2123|  6.86M|                stack.newFrame(FRAME_BINARY_LEFT, ast_);
 2124|  6.86M|                ast_ = ast.left;
 2125|  6.86M|                goto recurse;
 2126|      0|            } break;
 2127|       |
 2128|  9.92M|            case AST_BUILTIN_FUNCTION: {
  ------------------
  |  Branch (2128:13): [True: 9.92M, False: 136M]
  ------------------
 2129|  9.92M|                const auto &ast = *static_cast<const BuiltinFunction *>(ast_);
 2130|  9.92M|                scratch = makeBuiltinFromAST(ast.body);
 2131|  9.92M|            } break;
 2132|       |
 2133|      0|            case AST_BUILTIN_FUNCTION_BODY: {
  ------------------
  |  Branch (2133:13): [True: 0, False: 146M]
  ------------------
 2134|       |                // We can reach here if we end up evaluating a HeapThunk that was constructed
 2135|       |                // from a built-in function. Synthesize a FRAME_BUILTIN_FORCE_THUNKS to call it.
 2136|      0|                assert(stack.top().isCall());
  ------------------
  |  Branch (2136:17): [True: 0, False: 0]
  ------------------
 2137|      0|                const auto fbody = static_cast<const BuiltinFunctionBody*>(ast_);
 2138|       |
 2139|      0|                const BindingFrame up_values = stack.top().bindings;
 2140|      0|                std::vector<HeapThunk*> arg_thunks;
 2141|      0|                for (const auto &p : fbody->params) {
  ------------------
  |  Branch (2141:36): [True: 0, False: 0]
  ------------------
 2142|      0|                    const auto it = up_values.find(p);
 2143|      0|                    if (it != up_values.end()) {
  ------------------
  |  Branch (2143:25): [True: 0, False: 0]
  ------------------
 2144|      0|                        arg_thunks.push_back(it->second);
 2145|      0|                    } else {
 2146|      0|                        JSONNET_UNREACHABLE();
  ------------------
  |  |   53|      0|  do {                                                             \
  |  |   54|      0|    std::cerr << __FILE__ << ":" << __LINE__                       \
  |  |   55|      0|              << ": INTERNAL ERROR: reached unreachable code path" \
  |  |   56|      0|              << std::endl;                                        \
  |  |   57|      0|    abort();                                                       \
  |  |   58|      0|  } while (0)
  |  |  ------------------
  |  |  |  Branch (58:12): [Folded, False: 0]
  |  |  ------------------
  ------------------
 2147|      0|                    }
 2148|      0|                }
 2149|      0|                const LocationRange &loc = stack.top().location;
 2150|       |
 2151|      0|                stack.newFrame(FRAME_BUILTIN_FORCE_THUNKS, loc);
 2152|      0|                stack.top().bindings = up_values;
 2153|      0|                stack.top().thunks = arg_thunks;
 2154|      0|                stack.top().val = makeBuiltinFromAST(fbody);
 2155|      0|            } break;
 2156|       |
 2157|  8.57M|            case AST_CONDITIONAL: {
  ------------------
  |  Branch (2157:13): [True: 8.57M, False: 138M]
  ------------------
 2158|  8.57M|                const auto &ast = *static_cast<const Conditional *>(ast_);
 2159|  8.57M|                stack.newFrame(FRAME_IF, ast_);
 2160|  8.57M|                ast_ = ast.cond;
 2161|  8.57M|                goto recurse;
 2162|      0|            } break;
 2163|       |
 2164|  6.85k|            case AST_ERROR: {
  ------------------
  |  Branch (2164:13): [True: 6.85k, False: 146M]
  ------------------
 2165|  6.85k|                const auto &ast = *static_cast<const Error *>(ast_);
 2166|  6.85k|                stack.newFrame(FRAME_ERROR, ast_);
 2167|  6.85k|                ast_ = ast.expr;
 2168|  6.85k|                goto recurse;
 2169|      0|            } break;
 2170|       |
 2171|  2.32M|            case AST_FUNCTION: {
  ------------------
  |  Branch (2171:13): [True: 2.32M, False: 144M]
  ------------------
 2172|  2.32M|                const auto &ast = *static_cast<const Function *>(ast_);
 2173|  2.32M|                auto env = capture(ast.freeVariables);
 2174|  2.32M|                HeapObject *self;
 2175|  2.32M|                unsigned offset;
 2176|  2.32M|                stack.getSelfBinding(self, offset);
 2177|  2.32M|                HeapClosure::Params params;
 2178|  2.32M|                params.reserve(ast.params.size());
 2179|  4.42M|                for (const auto &p : ast.params) {
  ------------------
  |  Branch (2179:36): [True: 4.42M, False: 2.32M]
  ------------------
 2180|  4.42M|                    params.emplace_back(p.id, p.expr);
 2181|  4.42M|                }
 2182|  2.32M|                scratch = makeClosure(env, self, offset, params, ast.body);
 2183|  2.32M|            } break;
 2184|       |
 2185|      7|            case AST_IMPORT: {
  ------------------
  |  Branch (2185:13): [True: 7, False: 146M]
  ------------------
 2186|      7|                const auto &ast = *static_cast<const Import *>(ast_);
 2187|      7|                HeapThunk *thunk = import(ast.location, ast.file);
 2188|      7|                if (thunk->filled) {
  ------------------
  |  Branch (2188:21): [True: 0, False: 7]
  ------------------
 2189|      0|                    scratch = thunk->content;
 2190|      7|                } else {
 2191|      7|                    stack.newCall(ast.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 2192|      7|                    ast_ = thunk->body;
 2193|      7|                    goto recurse;
 2194|      7|                }
 2195|      7|            } break;
 2196|       |
 2197|      5|            case AST_IMPORTSTR: {
  ------------------
  |  Branch (2197:13): [True: 5, False: 146M]
  ------------------
 2198|      5|                const auto &ast = *static_cast<const Importstr *>(ast_);
 2199|      5|                const ImportCacheValue *value = importData(ast.location, ast.file);
 2200|      5|                scratch = makeString(decode_utf8(value->content));
 2201|      5|            } break;
 2202|       |
 2203|      8|            case AST_IMPORTBIN: {
  ------------------
  |  Branch (2203:13): [True: 8, False: 146M]
  ------------------
 2204|      8|                const auto &ast = *static_cast<const Importbin *>(ast_);
 2205|      8|                const ImportCacheValue *value = importData(ast.location, ast.file);
 2206|      8|                scratch = makeArray({});
 2207|      8|                auto &elements = static_cast<HeapArray *>(scratch.v.h)->elements;
 2208|      8|                elements.reserve(value->content.size());
 2209|      8|                for (const auto c : value->content) {
  ------------------
  |  Branch (2209:35): [True: 0, False: 8]
  ------------------
 2210|      0|                    auto *th = makeHeap<HeapThunk>(idArrayElement, nullptr, 0, nullptr);
 2211|      0|                    elements.push_back(th);
 2212|      0|                    th->fill(makeNumber(uint8_t(c)));
 2213|      0|                }
 2214|      8|            } break;
 2215|       |
 2216|  1.28M|            case AST_IN_SUPER: {
  ------------------
  |  Branch (2216:13): [True: 1.28M, False: 145M]
  ------------------
 2217|  1.28M|                const auto &ast = *static_cast<const InSuper *>(ast_);
 2218|  1.28M|                stack.newFrame(FRAME_IN_SUPER_ELEMENT, ast_);
 2219|  1.28M|                ast_ = ast.element;
 2220|  1.28M|                goto recurse;
 2221|      7|            } break;
 2222|       |
 2223|  13.0M|            case AST_INDEX: {
  ------------------
  |  Branch (2223:13): [True: 13.0M, False: 133M]
  ------------------
 2224|  13.0M|                const auto &ast = *static_cast<const Index *>(ast_);
 2225|  13.0M|                stack.newFrame(FRAME_INDEX_TARGET, ast_);
 2226|  13.0M|                ast_ = ast.target;
 2227|  13.0M|                goto recurse;
 2228|      7|            } break;
 2229|       |
 2230|  6.75M|            case AST_LOCAL: {
  ------------------
  |  Branch (2230:13): [True: 6.75M, False: 139M]
  ------------------
 2231|  6.75M|                const auto &ast = *static_cast<const Local *>(ast_);
 2232|  6.75M|                stack.newFrame(FRAME_LOCAL, ast_);
 2233|  6.75M|                Frame &f = stack.top();
 2234|       |                // First build all the thunks and bind them.
 2235|  6.75M|                HeapObject *self;
 2236|  6.75M|                unsigned offset;
 2237|  6.75M|                stack.getSelfBinding(self, offset);
 2238|  21.6M|                for (const auto &bind : ast.binds) {
  ------------------
  |  Branch (2238:39): [True: 21.6M, False: 6.75M]
  ------------------
 2239|       |                    // Note that these 2 lines must remain separate to avoid the GC running
 2240|       |                    // when bindings has a nullptr for key bind.first.
 2241|  21.6M|                    auto *th = makeHeap<HeapThunk>(bind.var, self, offset, bind.body);
 2242|  21.6M|                    f.bindings[bind.var] = th;
 2243|  21.6M|                }
 2244|       |                // Now capture the environment (including the new thunks, to make cycles).
 2245|  21.6M|                for (const auto &bind : ast.binds) {
  ------------------
  |  Branch (2245:39): [True: 21.6M, False: 6.75M]
  ------------------
 2246|  21.6M|                    auto *thunk = f.bindings[bind.var];
 2247|  21.6M|                    thunk->upValues = capture(bind.body->freeVariables);
 2248|  21.6M|                }
 2249|  6.75M|                ast_ = ast.body;
 2250|  6.75M|                goto recurse;
 2251|      7|            } break;
 2252|       |
 2253|   706k|            case AST_LITERAL_BOOLEAN: {
  ------------------
  |  Branch (2253:13): [True: 706k, False: 145M]
  ------------------
 2254|   706k|                const auto &ast = *static_cast<const LiteralBoolean *>(ast_);
 2255|   706k|                scratch = makeBoolean(ast.value);
 2256|   706k|            } break;
 2257|       |
 2258|  4.76M|            case AST_LITERAL_NUMBER: {
  ------------------
  |  Branch (2258:13): [True: 4.76M, False: 141M]
  ------------------
 2259|  4.76M|                const auto &ast = *static_cast<const LiteralNumber *>(ast_);
 2260|  4.76M|                scratch = makeNumberCheck(ast_->location, ast.value);
 2261|  4.76M|            } break;
 2262|       |
 2263|  34.3M|            case AST_LITERAL_STRING: {
  ------------------
  |  Branch (2263:13): [True: 34.3M, False: 112M]
  ------------------
 2264|  34.3M|                const auto &ast = *static_cast<const LiteralString *>(ast_);
 2265|  34.3M|                scratch = makeString(ast.value);
 2266|  34.3M|            } break;
 2267|       |
 2268|   105k|            case AST_LITERAL_NULL: {
  ------------------
  |  Branch (2268:13): [True: 105k, False: 146M]
  ------------------
 2269|   105k|                scratch = makeNull();
 2270|   105k|            } break;
 2271|       |
 2272|  1.56M|            case AST_DESUGARED_OBJECT: {
  ------------------
  |  Branch (2272:13): [True: 1.56M, False: 145M]
  ------------------
 2273|  1.56M|                const auto &ast = *static_cast<const DesugaredObject *>(ast_);
 2274|  1.56M|                if (ast.fields.empty()) {
  ------------------
  |  Branch (2274:21): [True: 1.07M, False: 491k]
  ------------------
 2275|  1.07M|                    auto env = capture(ast.freeVariables);
 2276|  1.07M|                    std::map<const Identifier *, HeapSimpleObject::Field> fields;
 2277|  1.07M|                    scratch = makeObject<HeapSimpleObject>(env, fields, ast.asserts);
 2278|  1.07M|                } else {
 2279|   491k|                    auto env = capture(ast.freeVariables);
 2280|   491k|                    stack.newFrame(FRAME_OBJECT, ast_);
 2281|   491k|                    auto fit = ast.fields.begin();
 2282|   491k|                    stack.top().fit = fit;
 2283|   491k|                    ast_ = fit->name;
 2284|   491k|                    goto recurse;
 2285|   491k|                }
 2286|  1.56M|            } break;
 2287|       |
 2288|  1.07M|            case AST_OBJECT_COMPREHENSION_SIMPLE: {
  ------------------
  |  Branch (2288:13): [True: 28, False: 146M]
  ------------------
 2289|     28|                const auto &ast = *static_cast<const ObjectComprehensionSimple *>(ast_);
 2290|     28|                stack.newFrame(FRAME_OBJECT_COMP_ARRAY, ast_);
 2291|     28|                ast_ = ast.array;
 2292|     28|                goto recurse;
 2293|  1.56M|            } break;
 2294|       |
 2295|  2.32M|            case AST_SELF: {
  ------------------
  |  Branch (2295:13): [True: 2.32M, False: 144M]
  ------------------
 2296|  2.32M|                scratch.t = Value::OBJECT;
 2297|  2.32M|                HeapObject *self;
 2298|  2.32M|                unsigned offset;
 2299|  2.32M|                stack.getSelfBinding(self, offset);
 2300|  2.32M|                scratch.v.h = self;
 2301|  2.32M|            } break;
 2302|       |
 2303|  1.21M|            case AST_SUPER_INDEX: {
  ------------------
  |  Branch (2303:13): [True: 1.21M, False: 145M]
  ------------------
 2304|  1.21M|                const auto &ast = *static_cast<const SuperIndex *>(ast_);
 2305|  1.21M|                stack.newFrame(FRAME_SUPER_INDEX, ast_);
 2306|  1.21M|                ast_ = ast.index;
 2307|  1.21M|                goto recurse;
 2308|  1.56M|            } break;
 2309|       |
 2310|  2.28M|            case AST_UNARY: {
  ------------------
  |  Branch (2310:13): [True: 2.28M, False: 144M]
  ------------------
 2311|  2.28M|                const auto &ast = *static_cast<const Unary *>(ast_);
 2312|  2.28M|                stack.newFrame(FRAME_UNARY, ast_);
 2313|  2.28M|                ast_ = ast.expr;
 2314|  2.28M|                goto recurse;
 2315|  1.56M|            } break;
 2316|       |
 2317|  36.5M|            case AST_VAR: {
  ------------------
  |  Branch (2317:13): [True: 36.5M, False: 110M]
  ------------------
 2318|  36.5M|                const auto &ast = *static_cast<const Var *>(ast_);
 2319|  36.5M|                auto *thunk = stack.lookUpVar(ast.id);
 2320|  36.5M|                if (thunk == nullptr) {
  ------------------
  |  Branch (2320:21): [True: 0, False: 36.5M]
  ------------------
 2321|      0|                    std::cerr << "INTERNAL ERROR: Could not bind variable: "
 2322|      0|                              << encode_utf8(ast.id->name) << " at "
 2323|      0|                              << ast.location << std::endl;
 2324|      0|                    std::abort();
 2325|      0|                }
 2326|  36.5M|                if (thunk->filled) {
  ------------------
  |  Branch (2326:21): [True: 26.4M, False: 10.0M]
  ------------------
 2327|  26.4M|                    scratch = thunk->content;
 2328|  26.4M|                } else {
 2329|  10.0M|                    stack.newCall(ast.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 2330|  10.0M|                    ast_ = thunk->body;
 2331|  10.0M|                    goto recurse;
 2332|  10.0M|                }
 2333|  36.5M|            } break;
 2334|       |
 2335|  26.4M|            default:
  ------------------
  |  Branch (2335:13): [True: 0, False: 146M]
  ------------------
 2336|      0|                std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_->type << std::endl;
 2337|      0|                std::abort();
 2338|   146M|        }
 2339|       |
 2340|       |        // To evaluate another AST, set ast to it, then goto recurse.
 2341|       |        // To pop, exit the switch or goto popframe
 2342|       |        // To change the frame and re-enter the switch, goto replaceframe
 2343|   170M|        while (stack.size() > initial_stack_size) {
  ------------------
  |  Branch (2343:16): [True: 168M, False: 2.25M]
  ------------------
 2344|   168M|            Frame &f = stack.top();
 2345|   168M|            switch (f.kind) {
 2346|  13.6M|                case FRAME_APPLY_TARGET: {
  ------------------
  |  Branch (2346:17): [True: 13.6M, False: 154M]
  ------------------
 2347|  13.6M|                    const auto &ast = *static_cast<const Apply *>(f.ast);
 2348|  13.6M|                    if (scratch.t != Value::FUNCTION) {
  ------------------
  |  Branch (2348:25): [True: 11, False: 13.6M]
  ------------------
 2349|     11|                        throw makeError(ast.location,
 2350|     11|                                        "only functions can be called, got " + type_str(scratch));
 2351|     11|                    }
 2352|  13.6M|                    auto *func = static_cast<HeapClosure *>(scratch.v.h);
 2353|       |
 2354|  13.6M|                    std::set<const Identifier *> params_needed;
 2355|  23.3M|                    for (const auto &param : func->params) {
  ------------------
  |  Branch (2355:44): [True: 23.3M, False: 13.6M]
  ------------------
 2356|  23.3M|                        params_needed.insert(param.id);
 2357|  23.3M|                    }
 2358|       |
 2359|       |                    // Create thunks for arguments.
 2360|  13.6M|                    std::vector<HeapThunk *> positional_args;
 2361|  13.6M|                    BindingFrame args;
 2362|  13.6M|                    bool got_named = false;
 2363|  37.0M|                    for (std::size_t i = 0; i < ast.args.size(); ++i) {
  ------------------
  |  Branch (2363:45): [True: 23.3M, False: 13.6M]
  ------------------
 2364|  23.3M|                        const auto &arg = ast.args[i];
 2365|       |
 2366|  23.3M|                        const Identifier *name;
 2367|  23.3M|                        if (arg.id != nullptr) {
  ------------------
  |  Branch (2367:29): [True: 0, False: 23.3M]
  ------------------
 2368|      0|                            got_named = true;
 2369|      0|                            name = arg.id;
 2370|  23.3M|                        } else {
 2371|  23.3M|                            if (got_named) {
  ------------------
  |  Branch (2371:33): [True: 0, False: 23.3M]
  ------------------
 2372|      0|                                std::stringstream ss;
 2373|      0|                                ss << "internal error: got positional param after named at index "
 2374|      0|                                   << i;
 2375|      0|                                throw makeError(ast.location, ss.str());
 2376|      0|                            }
 2377|  23.3M|                            if (i >= func->params.size()) {
  ------------------
  |  Branch (2377:33): [True: 0, False: 23.3M]
  ------------------
 2378|      0|                                std::stringstream ss;
 2379|      0|                                ss << "too many args, function has " << func->params.size()
 2380|      0|                                   << " parameter(s)";
 2381|      0|                                throw makeError(ast.location, ss.str());
 2382|      0|                            }
 2383|  23.3M|                            name = func->params[i].id;
 2384|  23.3M|                        }
 2385|       |                        // Special case for builtin functions -- leave identifier blank for
 2386|       |                        // them in the thunk.  This removes the thunk frame from the stacktrace.
 2387|  23.3M|                        const Identifier *name_ = func->isBuiltin() ? nullptr : name;
  ------------------
  |  Branch (2387:51): [True: 15.6M, False: 7.75M]
  ------------------
 2388|  23.3M|                        HeapObject *self;
 2389|  23.3M|                        unsigned offset;
 2390|  23.3M|                        stack.getSelfBinding(self, offset);
 2391|  23.3M|                        auto *thunk = makeHeap<HeapThunk>(name_, self, offset, arg.expr);
 2392|  23.3M|                        thunk->upValues = capture(arg.expr->freeVariables);
 2393|       |                        // While making the thunks, keep them in a frame to avoid premature garbage
 2394|       |                        // collection.
 2395|  23.3M|                        f.thunks.push_back(thunk);
 2396|  23.3M|                        if (args.find(name) != args.end()) {
  ------------------
  |  Branch (2396:29): [True: 0, False: 23.3M]
  ------------------
 2397|      0|                            std::stringstream ss;
 2398|      0|                            ss << "binding parameter a second time: " << encode_utf8(name->name);
 2399|      0|                            throw makeError(ast.location, ss.str());
 2400|      0|                        }
 2401|  23.3M|                        args[name] = thunk;
 2402|  23.3M|                        if (params_needed.find(name) == params_needed.end()) {
  ------------------
  |  Branch (2402:29): [True: 0, False: 23.3M]
  ------------------
 2403|      0|                            std::stringstream ss;
 2404|      0|                            ss << "function has no parameter " << encode_utf8(name->name);
 2405|      0|                            throw makeError(ast.location, ss.str());
 2406|      0|                        }
 2407|  23.3M|                    }
 2408|       |
 2409|       |                    // For any func params for which there was no arg, create a thunk for those and
 2410|       |                    // bind the default argument.  Allow default thunks to see other params.  If no
 2411|       |                    // default argument than raise an error.
 2412|       |
 2413|       |                    // Raise errors for unbound params, create thunks (but don't fill in upvalues).
 2414|       |                    // This is a subset of f.thunks, so will not get garbage collected.
 2415|  13.6M|                    std::vector<HeapThunk *> def_arg_thunks;
 2416|  23.3M|                    for (const auto &param : func->params) {
  ------------------
  |  Branch (2416:44): [True: 23.3M, False: 13.6M]
  ------------------
 2417|  23.3M|                        if (args.find(param.id) != args.end())
  ------------------
  |  Branch (2417:29): [True: 23.3M, False: 274]
  ------------------
 2418|  23.3M|                            continue;
 2419|    274|                        if (param.def == nullptr) {
  ------------------
  |  Branch (2419:29): [True: 31, False: 243]
  ------------------
 2420|     31|                            std::stringstream ss;
 2421|     31|                            ss << "function parameter " << encode_utf8(param.id->name)
 2422|     31|                               << " not bound in call.";
 2423|     31|                            throw makeError(ast.location, ss.str());
 2424|     31|                        }
 2425|       |
 2426|       |                        // Special case for builtin functions -- leave identifier blank for
 2427|       |                        // them in the thunk.  This removes the thunk frame from the stacktrace.
 2428|    243|                        const Identifier *name_ = func->isBuiltin() ? nullptr : param.id;
  ------------------
  |  Branch (2428:51): [True: 0, False: 243]
  ------------------
 2429|    243|                        auto *thunk =
 2430|    243|                            makeHeap<HeapThunk>(name_, func->self, func->offset, param.def);
 2431|    243|                        f.thunks.push_back(thunk);
 2432|    243|                        def_arg_thunks.push_back(thunk);
 2433|    243|                        args[param.id] = thunk;
 2434|    243|                    }
 2435|       |
 2436|  13.6M|                    BindingFrame up_values = func->upValues;
 2437|  13.6M|                    up_values.insert(args.begin(), args.end());
 2438|       |
 2439|       |                    // Fill in upvalues
 2440|  13.6M|                    for (HeapThunk *thunk : def_arg_thunks) {
  ------------------
  |  Branch (2440:43): [True: 10, False: 13.6M]
  ------------------
 2441|     10|                        thunk->upValues = up_values;
 2442|     10|                    }
 2443|       |
 2444|       |                    // Cache these, because pop will invalidate them.
 2445|  13.6M|                    std::vector<HeapThunk *> thunks_copy = f.thunks;
 2446|       |
 2447|  13.6M|                    const AST *f_ast = f.ast;
 2448|  13.6M|                    stack.pop();
 2449|       |
 2450|  13.6M|                    if (func->isBuiltin()) {
  ------------------
  |  Branch (2450:25): [True: 9.92M, False: 3.76M]
  ------------------
 2451|       |                        // Built-in function.
 2452|       |                        // Give nullptr for self because no one looking at this frame will
 2453|       |                        // attempt to bind to self (it's native code).
 2454|  9.92M|                        stack.newFrame(FRAME_BUILTIN_FORCE_THUNKS, f_ast);
 2455|  9.92M|                        assert(func->upValues.empty());
  ------------------
  |  Branch (2455:25): [True: 9.92M, False: 0]
  ------------------
 2456|  9.92M|                        stack.top().bindings = up_values;
 2457|  9.92M|                        stack.top().thunks = thunks_copy;
 2458|  9.92M|                        stack.top().val = scratch;
 2459|  9.92M|                        goto replaceframe;
 2460|  9.92M|                    } else {
 2461|       |                        // User defined function.
 2462|  3.76M|                        stack.newCall(ast.location, func, func->self, func->offset, up_values);
 2463|  3.76M|                        if (ast.tailstrict) {
  ------------------
  |  Branch (2463:29): [True: 1.85M, False: 1.90M]
  ------------------
 2464|  1.85M|                            stack.top().tailCall = true;
 2465|  1.85M|                            if (thunks_copy.size() == 0) {
  ------------------
  |  Branch (2465:33): [True: 0, False: 1.85M]
  ------------------
 2466|       |                                // No need to force thunks, proceed straight to body.
 2467|      0|                                ast_ = func->body;
 2468|      0|                                goto recurse;
 2469|  1.85M|                            } else {
 2470|       |                                // The check for args.size() > 0
 2471|  1.85M|                                stack.top().thunks = thunks_copy;
 2472|  1.85M|                                stack.top().val = scratch;
 2473|  1.85M|                                goto replaceframe;
 2474|  1.85M|                            }
 2475|  1.90M|                        } else {
 2476|  1.90M|                            ast_ = func->body;
 2477|  1.90M|                            goto recurse;
 2478|  1.90M|                        }
 2479|  3.76M|                    }
 2480|  13.6M|                } break;
 2481|       |
 2482|  6.77M|                case FRAME_BINARY_LEFT: {
  ------------------
  |  Branch (2482:17): [True: 6.77M, False: 161M]
  ------------------
 2483|  6.77M|                    const auto &ast = *static_cast<const Binary *>(f.ast);
 2484|  6.77M|                    const Value &lhs = scratch;
 2485|  6.77M|                    if (lhs.t == Value::BOOLEAN) {
  ------------------
  |  Branch (2485:25): [True: 225k, False: 6.55M]
  ------------------
 2486|       |                        // Handle short-cut semantics
 2487|   225k|                        switch (ast.op) {
 2488|  72.9k|                            case BOP_AND: {
  ------------------
  |  Branch (2488:29): [True: 72.9k, False: 152k]
  ------------------
 2489|  72.9k|                                if (!lhs.v.b) {
  ------------------
  |  Branch (2489:37): [True: 19.6k, False: 53.2k]
  ------------------
 2490|  19.6k|                                    scratch = makeBoolean(false);
 2491|  19.6k|                                    goto popframe;
 2492|  19.6k|                                }
 2493|  72.9k|                            } break;
 2494|       |
 2495|   151k|                            case BOP_OR: {
  ------------------
  |  Branch (2495:29): [True: 151k, False: 73.6k]
  ------------------
 2496|   151k|                                if (lhs.v.b) {
  ------------------
  |  Branch (2496:37): [True: 2.48k, False: 149k]
  ------------------
 2497|  2.48k|                                    scratch = makeBoolean(true);
 2498|  2.48k|                                    goto popframe;
 2499|  2.48k|                                }
 2500|   151k|                            } break;
 2501|       |
 2502|   149k|                            default:;
  ------------------
  |  Branch (2502:29): [True: 739, False: 224k]
  ------------------
 2503|   225k|                        }
 2504|   225k|                    }
 2505|  6.75M|                    stack.top().kind = FRAME_BINARY_RIGHT;
 2506|  6.75M|                    stack.top().val = lhs;
 2507|  6.75M|                    ast_ = ast.right;
 2508|  6.75M|                    goto recurse;
 2509|  6.77M|                } break;
 2510|       |
 2511|  6.75M|                case FRAME_BINARY_RIGHT: {
  ------------------
  |  Branch (2511:17): [True: 6.75M, False: 161M]
  ------------------
 2512|  6.75M|                    stack.top().val2 = scratch;
 2513|  6.75M|                    stack.top().kind = FRAME_BINARY_OP;
 2514|  6.75M|                }
 2515|  6.75M|                [[fallthrough]];
 2516|  6.75M|                case FRAME_BINARY_OP: {
  ------------------
  |  Branch (2516:17): [True: 206, False: 168M]
  ------------------
 2517|  6.75M|                    const auto &ast = *static_cast<const Binary *>(f.ast);
 2518|  6.75M|                    const Value &lhs = stack.top().val;
 2519|  6.75M|                    const Value &rhs = stack.top().val2;
 2520|       |
 2521|       |                    // Handle cases where the LHS and RHS are not the same type.
 2522|  6.75M|                    if (lhs.t == Value::STRING || rhs.t == Value::STRING) {
  ------------------
  |  Branch (2522:25): [True: 1.89M, False: 4.85M]
  |  Branch (2522:51): [True: 35.0k, False: 4.81M]
  ------------------
 2523|  1.93M|                        if (ast.op == BOP_PLUS) {
  ------------------
  |  Branch (2523:29): [True: 1.89M, False: 36.0k]
  ------------------
 2524|       |                            // Handle co-ercions for string processing.
 2525|  1.89M|                            stack.top().kind = FRAME_STRING_CONCAT;
 2526|  1.89M|                            stack.top().val2 = rhs;
 2527|  1.89M|                            goto replaceframe;
 2528|  1.89M|                        }
 2529|  1.93M|                    }
 2530|  4.85M|                    switch (ast.op) {
 2531|       |                        // Equality can be used when the types don't match.
 2532|      0|                        case BOP_MANIFEST_EQUAL:
  ------------------
  |  Branch (2532:25): [True: 0, False: 4.85M]
  ------------------
 2533|      0|                            std::cerr << "INTERNAL ERROR: Equals not desugared" << std::endl;
 2534|      0|                            abort();
 2535|       |
 2536|       |                        // Equality can be used when the types don't match.
 2537|      0|                        case BOP_MANIFEST_UNEQUAL:
  ------------------
  |  Branch (2537:25): [True: 0, False: 4.85M]
  ------------------
 2538|      0|                            std::cerr << "INTERNAL ERROR: Notequals not desugared" << std::endl;
 2539|      0|                            abort();
 2540|       |
 2541|       |                        // e in e
 2542|    598|                        case BOP_IN: {
  ------------------
  |  Branch (2542:25): [True: 598, False: 4.85M]
  ------------------
 2543|    598|                            if (lhs.t != Value::STRING) {
  ------------------
  |  Branch (2543:33): [True: 20, False: 578]
  ------------------
 2544|     20|                                throw makeError(ast.location,
 2545|     20|                                                "the left hand side of the 'in' operator should be "
 2546|     20|                                                "a string,  got " +
 2547|     20|                                                    type_str(lhs));
 2548|     20|                            }
 2549|    578|                            auto *field = static_cast<HeapString *>(lhs.v.h);
 2550|    578|                            switch (rhs.t) {
 2551|    575|                                case Value::OBJECT: {
  ------------------
  |  Branch (2551:33): [True: 575, False: 3]
  ------------------
 2552|    575|                                    auto *obj = static_cast<HeapObject *>(rhs.v.h);
 2553|    575|                                    auto *fid = alloc->makeIdentifier(field->value);
 2554|    575|                                    unsigned unused_found_at = 0;
 2555|    575|                                    bool in = findObject(fid, obj, 0, unused_found_at);
 2556|    575|                                    scratch = makeBoolean(in);
 2557|    575|                                } break;
 2558|       |
 2559|      3|                                default:
  ------------------
  |  Branch (2559:33): [True: 3, False: 575]
  ------------------
 2560|      3|                                    throw makeError(
 2561|      3|                                        ast.location,
 2562|      3|                                        "the right hand side of the 'in' operator should be"
 2563|      3|                                        " an object, got " +
 2564|      3|                                            type_str(rhs));
 2565|    578|                            }
 2566|    575|                            goto popframe;
 2567|    578|                        }
 2568|       |
 2569|  4.85M|                        default:;
  ------------------
  |  Branch (2569:25): [True: 4.85M, False: 598]
  ------------------
 2570|  4.85M|                    }
 2571|       |                    // Everything else requires matching types.
 2572|  4.85M|                    if (lhs.t != rhs.t) {
  ------------------
  |  Branch (2572:25): [True: 99, False: 4.85M]
  ------------------
 2573|     99|                        throw makeError(ast.location,
 2574|     99|                                        "binary operator " + bop_string(ast.op) +
 2575|     99|                                            " requires "
 2576|     99|                                            "matching types, got " +
 2577|     99|                                            type_str(lhs) + " and " + type_str(rhs) + ".");
 2578|     99|                    }
 2579|  4.85M|                    switch (lhs.t) {
  ------------------
  |  Branch (2579:29): [True: 4.85M, False: 0]
  ------------------
 2580|   210k|                        case Value::ARRAY:
  ------------------
  |  Branch (2580:25): [True: 210k, False: 4.64M]
  ------------------
 2581|   210k|                            if (ast.op == BOP_PLUS) {
  ------------------
  |  Branch (2581:33): [True: 188k, False: 22.3k]
  ------------------
 2582|   188k|                                auto *arr_l = static_cast<HeapArray *>(lhs.v.h);
 2583|   188k|                                auto *arr_r = static_cast<HeapArray *>(rhs.v.h);
 2584|   188k|                                std::vector<HeapThunk *> elements;
 2585|   188k|                                for (auto *el : arr_l->elements)
  ------------------
  |  Branch (2585:47): [True: 24.5M, False: 188k]
  ------------------
 2586|  24.5M|                                    elements.push_back(el);
 2587|   188k|                                for (auto *el : arr_r->elements)
  ------------------
  |  Branch (2587:47): [True: 974k, False: 188k]
  ------------------
 2588|   974k|                                    elements.push_back(el);
 2589|   188k|                                scratch = makeArray(elements);
 2590|   188k|                            } else if (ast.op == BOP_LESS || ast.op == BOP_LESS_EQ || ast.op == BOP_GREATER || ast.op == BOP_GREATER_EQ) {
  ------------------
  |  Branch (2590:40): [True: 7.51k, False: 14.8k]
  |  Branch (2590:62): [True: 2.49k, False: 12.3k]
  |  Branch (2590:87): [True: 138, False: 12.1k]
  |  Branch (2590:112): [True: 12.1k, False: 7]
  ------------------
 2591|  22.3k|                                HeapThunk *func;
 2592|  22.3k|                                switch(ast.op) {
 2593|  7.51k|                                case BOP_LESS:
  ------------------
  |  Branch (2593:33): [True: 7.51k, False: 14.7k]
  ------------------
 2594|  7.51k|                                    func = sourceVals["__array_less"];
 2595|  7.51k|                                    break;
 2596|  2.49k|                                case BOP_LESS_EQ:
  ------------------
  |  Branch (2596:33): [True: 2.49k, False: 19.8k]
  ------------------
 2597|  2.49k|                                    func = sourceVals["__array_less_or_equal"];
 2598|  2.49k|                                    break;
 2599|    138|                                case BOP_GREATER:
  ------------------
  |  Branch (2599:33): [True: 138, False: 22.1k]
  ------------------
 2600|    138|                                    func = sourceVals["__array_greater"];
 2601|    138|                                    break;
 2602|  12.1k|                                case BOP_GREATER_EQ:
  ------------------
  |  Branch (2602:33): [True: 12.1k, False: 10.1k]
  ------------------
 2603|  12.1k|                                    func = sourceVals["__array_greater_or_equal"];
 2604|  12.1k|                                    break;
 2605|      0|                                default:
  ------------------
  |  Branch (2605:33): [True: 0, False: 22.3k]
  ------------------
 2606|      0|                                    JSONNET_UNREACHABLE();
  ------------------
  |  |   53|      0|  do {                                                             \
  |  |   54|      0|    std::cerr << __FILE__ << ":" << __LINE__                       \
  |  |   55|      0|              << ": INTERNAL ERROR: reached unreachable code path" \
  |  |   56|      0|              << std::endl;                                        \
  |  |   57|      0|    abort();                                                       \
  |  |   58|      0|  } while (0)
  |  |  ------------------
  |  |  |  Branch (58:12): [Folded, False: 0]
  |  |  ------------------
  ------------------
 2607|  22.3k|                                }
 2608|  22.3k|                                if (!func->filled) {
  ------------------
  |  Branch (2608:37): [True: 206, False: 22.1k]
  ------------------
 2609|    206|                                    stack.newCall(ast.location, func, func->self, func->offset, func->upValues);
 2610|    206|                                    ast_ = func->body;
 2611|    206|                                    goto recurse;
 2612|    206|                                }
 2613|  22.1k|                                auto *lhs_th = makeHeap<HeapThunk>(idInternal, f.self, f.offset, ast.left);
 2614|  22.1k|                                lhs_th->fill(lhs);
 2615|  22.1k|                                f.thunks.push_back(lhs_th);
 2616|  22.1k|                                auto *rhs_th = makeHeap<HeapThunk>(idInternal, f.self, f.offset, ast.right);
 2617|  22.1k|                                rhs_th->fill(rhs);
 2618|  22.1k|                                f.thunks.push_back(rhs_th);
 2619|  22.1k|                                const AST *orig_ast = ast_;
 2620|  22.1k|                                stack.pop();
 2621|  22.1k|                                ast_ = callSourceVal(orig_ast, func, {lhs_th, rhs_th});
 2622|  22.1k|                                goto recurse;
 2623|  22.3k|                            } else {
 2624|      7|                                throw makeError(ast.location,
 2625|      7|                                                "binary operator " + bop_string(ast.op) +
 2626|      7|                                                    " does not operate on arrays.");
 2627|      7|                            }
 2628|   188k|                            break;
 2629|       |
 2630|   202k|                        case Value::BOOLEAN:
  ------------------
  |  Branch (2630:25): [True: 202k, False: 4.64M]
  ------------------
 2631|   202k|                            switch (ast.op) {
 2632|  53.2k|                                case BOP_AND: scratch = makeBoolean(lhs.v.b && rhs.v.b); break;
  ------------------
  |  Branch (2632:33): [True: 53.2k, False: 149k]
  |  Branch (2632:69): [True: 53.2k, False: 0]
  |  Branch (2632:80): [True: 41.9k, False: 11.3k]
  ------------------
 2633|       |
 2634|   149k|                                case BOP_OR: scratch = makeBoolean(lhs.v.b || rhs.v.b); break;
  ------------------
  |  Branch (2634:33): [True: 149k, False: 53.3k]
  |  Branch (2634:68): [True: 0, False: 149k]
  |  Branch (2634:79): [True: 2.32k, False: 146k]
  ------------------
 2635|       |
 2636|     69|                                default:
  ------------------
  |  Branch (2636:33): [True: 69, False: 202k]
  ------------------
 2637|     69|                                    throw makeError(ast.location,
 2638|     69|                                                    "binary operator " + bop_string(ast.op) +
 2639|     69|                                                        " does not operate on booleans.");
 2640|   202k|                            }
 2641|   202k|                            break;
 2642|       |
 2643|  3.29M|                        case Value::NUMBER:
  ------------------
  |  Branch (2643:25): [True: 3.29M, False: 1.55M]
  ------------------
 2644|  3.29M|                            switch (ast.op) {
 2645|   340k|                                case BOP_PLUS:
  ------------------
  |  Branch (2645:33): [True: 340k, False: 2.95M]
  ------------------
 2646|   340k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d + rhs.v.d);
 2647|   340k|                                    break;
 2648|       |
 2649|  1.12M|                                case BOP_MINUS:
  ------------------
  |  Branch (2649:33): [True: 1.12M, False: 2.17M]
  ------------------
 2650|  1.12M|                                    scratch = makeNumberCheck(ast.location, lhs.v.d - rhs.v.d);
 2651|  1.12M|                                    break;
 2652|       |
 2653|  28.6k|                                case BOP_MULT:
  ------------------
  |  Branch (2653:33): [True: 28.6k, False: 3.26M]
  ------------------
 2654|  28.6k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d * rhs.v.d);
 2655|  28.6k|                                    break;
 2656|       |
 2657|  25.5k|                                case BOP_DIV:
  ------------------
  |  Branch (2657:33): [True: 25.5k, False: 3.26M]
  ------------------
 2658|  25.5k|                                    if (rhs.v.d == 0)
  ------------------
  |  Branch (2658:41): [True: 2, False: 25.5k]
  ------------------
 2659|      2|                                        throw makeError(ast.location, "division by zero.");
 2660|  25.5k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d / rhs.v.d);
 2661|  25.5k|                                    break;
 2662|       |
 2663|       |                                    // No need to check doubles made from longs
 2664|       |
 2665|  2.81k|                                case BOP_SHIFT_L: {
  ------------------
  |  Branch (2665:33): [True: 2.81k, False: 3.29M]
  ------------------
 2666|  2.81k|                                    if (rhs.v.d < 0)
  ------------------
  |  Branch (2666:41): [True: 1, False: 2.80k]
  ------------------
 2667|      1|                                        throw makeError(ast.location, "shift by negative exponent.");
 2668|       |
 2669|  2.80k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2670|  2.80k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2671|  2.80k|                                    long_r = long_r % 64;
 2672|       |
 2673|       |                                    // Additional safety check for left shifts to prevent undefined behavior.
 2674|       |                                    // Left-shift that would move the highest set bit into the sign bit position is undefined.
 2675|  2.80k|                                    if (long_r >= 1 && abs(long_l) >= (1LL << (63 - long_r))) {
  ------------------
  |  Branch (2675:41): [True: 2.72k, False: 83]
  |  Branch (2675:56): [True: 25, False: 2.70k]
  ------------------
 2676|     25|                                        throw makeError(ast.location,
 2677|     25|                                            "numeric value outside safe integer range for bitwise operation.");
 2678|     25|                                    }
 2679|       |
 2680|       |                                    // Left-shift of a negative number is undefined until C++20.
 2681|       |                                    // Perform the shift on unsigned int to avoid that.
 2682|  2.78k|                                    scratch = makeNumber(static_cast<int64_t>(static_cast<uint64_t>(long_l) << long_r));
 2683|  2.78k|                                } break;
 2684|       |
 2685|     20|                                case BOP_SHIFT_R: {
  ------------------
  |  Branch (2685:33): [True: 20, False: 3.29M]
  ------------------
 2686|     20|                                    if (rhs.v.d < 0)
  ------------------
  |  Branch (2686:41): [True: 2, False: 18]
  ------------------
 2687|      2|                                        throw makeError(ast.location, "shift by negative exponent.");
 2688|       |
 2689|     18|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2690|     18|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2691|     18|                                    long_r = long_r % 64;
 2692|     18|                                    scratch = makeNumber(long_l >> long_r);
 2693|     18|                                } break;
 2694|       |
 2695|  6.39k|                                case BOP_BITWISE_AND: {
  ------------------
  |  Branch (2695:33): [True: 6.39k, False: 3.28M]
  ------------------
 2696|  6.39k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2697|  6.39k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2698|  6.39k|                                    scratch = makeNumber(long_l & long_r);
 2699|  6.39k|                                } break;
 2700|       |
 2701|  5.18k|                                case BOP_BITWISE_XOR: {
  ------------------
  |  Branch (2701:33): [True: 5.18k, False: 3.28M]
  ------------------
 2702|  5.18k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2703|  5.18k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2704|  5.18k|                                    scratch = makeNumber(long_l ^ long_r);
 2705|  5.18k|                                } break;
 2706|       |
 2707|  1.79k|                                case BOP_BITWISE_OR: {
  ------------------
  |  Branch (2707:33): [True: 1.79k, False: 3.29M]
  ------------------
 2708|  1.79k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2709|  1.79k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2710|  1.79k|                                    scratch = makeNumber(long_l | long_r);
 2711|  1.79k|                                } break;
 2712|       |
 2713|  1.10M|                                case BOP_LESS_EQ: scratch = makeBoolean(lhs.v.d <= rhs.v.d); break;
  ------------------
  |  Branch (2713:33): [True: 1.10M, False: 2.19M]
  ------------------
 2714|       |
 2715|   368k|                                case BOP_GREATER_EQ:
  ------------------
  |  Branch (2715:33): [True: 368k, False: 2.92M]
  ------------------
 2716|   368k|                                    scratch = makeBoolean(lhs.v.d >= rhs.v.d);
 2717|   368k|                                    break;
 2718|       |
 2719|   259k|                                case BOP_LESS: scratch = makeBoolean(lhs.v.d < rhs.v.d); break;
  ------------------
  |  Branch (2719:33): [True: 259k, False: 3.03M]
  ------------------
 2720|       |
 2721|  29.5k|                                case BOP_GREATER: scratch = makeBoolean(lhs.v.d > rhs.v.d); break;
  ------------------
  |  Branch (2721:33): [True: 29.5k, False: 3.26M]
  ------------------
 2722|       |
 2723|      6|                                default:
  ------------------
  |  Branch (2723:33): [True: 6, False: 3.29M]
  ------------------
 2724|      6|                                    throw makeError(ast.location,
 2725|      6|                                                    "binary operator " + bop_string(ast.op) +
 2726|      6|                                                        " does not operate on numbers.");
 2727|  3.29M|                            }
 2728|  3.29M|                            break;
 2729|       |
 2730|  3.29M|                        case Value::FUNCTION:
  ------------------
  |  Branch (2730:25): [True: 0, False: 4.85M]
  ------------------
 2731|      0|                            throw makeError(ast.location,
 2732|      0|                                            "binary operator " + bop_string(ast.op) +
 2733|      0|                                                " does not operate on functions.");
 2734|       |
 2735|      5|                        case Value::NULL_TYPE:
  ------------------
  |  Branch (2735:25): [True: 5, False: 4.85M]
  ------------------
 2736|      5|                            throw makeError(ast.location,
 2737|      5|                                            "binary operator " + bop_string(ast.op) +
 2738|      5|                                                " does not operate on null.");
 2739|       |
 2740|  1.10M|                        case Value::OBJECT: {
  ------------------
  |  Branch (2740:25): [True: 1.10M, False: 3.74M]
  ------------------
 2741|  1.10M|                            if (ast.op != BOP_PLUS) {
  ------------------
  |  Branch (2741:33): [True: 9, False: 1.10M]
  ------------------
 2742|      9|                                throw makeError(ast.location,
 2743|      9|                                                "binary operator " + bop_string(ast.op) +
 2744|      9|                                                    " does not operate on objects.");
 2745|      9|                            }
 2746|  1.10M|                            auto *lhs_obj = static_cast<HeapObject *>(lhs.v.h);
 2747|  1.10M|                            auto *rhs_obj = static_cast<HeapObject *>(rhs.v.h);
 2748|  1.10M|                            scratch = makeObject<HeapExtendedObject>(lhs_obj, rhs_obj);
 2749|  1.10M|                        } break;
 2750|       |
 2751|  35.4k|                        case Value::STRING: {
  ------------------
  |  Branch (2751:25): [True: 35.4k, False: 4.81M]
  ------------------
 2752|  35.4k|                            const UString &lhs_str = static_cast<HeapString *>(lhs.v.h)->value;
 2753|  35.4k|                            const UString &rhs_str = static_cast<HeapString *>(rhs.v.h)->value;
 2754|  35.4k|                            switch (ast.op) {
 2755|      0|                                case BOP_PLUS: scratch = makeString(lhs_str + rhs_str); break;
  ------------------
  |  Branch (2755:33): [True: 0, False: 35.4k]
  ------------------
 2756|       |
 2757|  4.91k|                                case BOP_LESS_EQ: scratch = makeBoolean(lhs_str <= rhs_str); break;
  ------------------
  |  Branch (2757:33): [True: 4.91k, False: 30.5k]
  ------------------
 2758|       |
 2759|  20.7k|                                case BOP_GREATER_EQ:
  ------------------
  |  Branch (2759:33): [True: 20.7k, False: 14.6k]
  ------------------
 2760|  20.7k|                                    scratch = makeBoolean(lhs_str >= rhs_str);
 2761|  20.7k|                                    break;
 2762|       |
 2763|  5.08k|                                case BOP_LESS: scratch = makeBoolean(lhs_str < rhs_str); break;
  ------------------
  |  Branch (2763:33): [True: 5.08k, False: 30.3k]
  ------------------
 2764|       |
 2765|  4.64k|                                case BOP_GREATER: scratch = makeBoolean(lhs_str > rhs_str); break;
  ------------------
  |  Branch (2765:33): [True: 4.64k, False: 30.7k]
  ------------------
 2766|       |
 2767|     19|                                default:
  ------------------
  |  Branch (2767:33): [True: 19, False: 35.3k]
  ------------------
 2768|     19|                                    throw makeError(ast.location,
 2769|     19|                                                    "binary operator " + bop_string(ast.op) +
 2770|     19|                                                        " does not operate on strings.");
 2771|  35.4k|                            }
 2772|  35.4k|                        } break;
 2773|  4.85M|                    }
 2774|  4.85M|                } break;
 2775|       |
 2776|  4.83M|                case FRAME_BUILTIN_FILTER: {
  ------------------
  |  Branch (2776:17): [True: 0, False: 168M]
  ------------------
 2777|      0|                    const auto &ast = *static_cast<const Apply *>(f.ast);
 2778|      0|                    auto *func = static_cast<HeapClosure *>(f.val.v.h);
 2779|      0|                    auto *arr = static_cast<HeapArray *>(f.val2.v.h);
 2780|      0|                    if (scratch.t != Value::BOOLEAN) {
  ------------------
  |  Branch (2780:25): [True: 0, False: 0]
  ------------------
 2781|      0|                        throw makeError(
 2782|      0|                            ast.location,
 2783|      0|                            "filter function must return boolean, got: " + type_str(scratch));
 2784|      0|                    }
 2785|      0|                    if (scratch.v.b)
  ------------------
  |  Branch (2785:25): [True: 0, False: 0]
  ------------------
 2786|      0|                        f.thunks.push_back(arr->elements[f.elementId]);
 2787|      0|                    f.elementId++;
 2788|       |                    // Iterate through arr, calling the function on each.
 2789|      0|                    if (f.elementId == arr->elements.size()) {
  ------------------
  |  Branch (2789:25): [True: 0, False: 0]
  ------------------
 2790|      0|                        scratch = makeArray(f.thunks);
 2791|      0|                    } else {
 2792|      0|                        auto *thunk = arr->elements[f.elementId];
 2793|      0|                        BindingFrame bindings = func->upValues;
 2794|      0|                        bindings[func->params[0].id] = thunk;
 2795|      0|                        stack.newCall(ast.location, func, func->self, func->offset, bindings);
 2796|      0|                        ast_ = func->body;
 2797|      0|                        goto recurse;
 2798|      0|                    }
 2799|      0|                } break;
 2800|       |
 2801|  25.5M|                case FRAME_BUILTIN_FORCE_THUNKS: {
  ------------------
  |  Branch (2801:17): [True: 25.5M, False: 142M]
  ------------------
 2802|  25.5M|                    const auto &location = f.location;
 2803|  25.5M|                    auto *func = static_cast<HeapClosure *>(f.val.v.h);
 2804|  25.5M|                    if (f.elementId == f.thunks.size()) {
  ------------------
  |  Branch (2804:25): [True: 9.91M, False: 15.6M]
  ------------------
 2805|       |                        // All thunks forced, now the builtin implementations.
 2806|  9.91M|                        const std::string &builtin_name = func->builtinName;
 2807|  9.91M|                        std::vector<Value> args;
 2808|  15.6M|                        for (const auto &p : func->params) {
  ------------------
  |  Branch (2808:44): [True: 15.6M, False: 9.91M]
  ------------------
 2809|  15.6M|                            const auto it = f.bindings.find(p.id);
 2810|  15.6M|                            if (it != f.bindings.end()) {
  ------------------
  |  Branch (2810:33): [True: 15.6M, False: 0]
  ------------------
 2811|  15.6M|                                assert(it->second->filled);
  ------------------
  |  Branch (2811:33): [True: 15.6M, False: 0]
  ------------------
 2812|  15.6M|                                args.push_back(it->second->content);
 2813|  15.6M|                            } else {
 2814|      0|                                std::stringstream ss;
 2815|      0|                                ss << "function parameter " << encode_utf8(p.id->name)
 2816|      0|                                    << " not bound in call.";
 2817|      0|                                throw makeError(location, ss.str());
 2818|      0|                            }
 2819|  15.6M|                        }
 2820|       |
 2821|  9.91M|                        BuiltinMap::const_iterator bit = builtins.find(builtin_name);
 2822|  9.91M|                        if (bit != builtins.end()) {
  ------------------
  |  Branch (2822:29): [True: 9.91M, False: 0]
  ------------------
 2823|  9.91M|                            const AST *new_ast = (this->*bit->second)(location, args);
 2824|  9.91M|                            if (new_ast != nullptr) {
  ------------------
  |  Branch (2824:33): [True: 0, False: 9.91M]
  ------------------
 2825|      0|                                ast_ = new_ast;
 2826|      0|                                goto recurse;
 2827|      0|                            }
 2828|  9.91M|                            break;
 2829|  9.91M|                        }
 2830|      0|                        VmNativeCallbackMap::const_iterator nit =
 2831|      0|                            nativeCallbacks.find(builtin_name);
 2832|       |                        // TODO(dcunnin): Support arrays.
 2833|       |                        // TODO(dcunnin): Support objects.
 2834|      0|                        std::vector<JsonnetJsonValue> args2;
 2835|      0|                        for (const Value &arg : args) {
  ------------------
  |  Branch (2835:47): [True: 0, False: 0]
  ------------------
 2836|      0|                            switch (arg.t) {
 2837|      0|                                case Value::STRING:
  ------------------
  |  Branch (2837:33): [True: 0, False: 0]
  ------------------
 2838|      0|                                    args2.emplace_back(
 2839|      0|                                        JsonnetJsonValue::STRING,
 2840|      0|                                        encode_utf8(static_cast<HeapString *>(arg.v.h)->value),
 2841|      0|                                        0);
 2842|      0|                                    break;
 2843|       |
 2844|      0|                                case Value::BOOLEAN:
  ------------------
  |  Branch (2844:33): [True: 0, False: 0]
  ------------------
 2845|      0|                                    args2.emplace_back(
 2846|      0|                                        JsonnetJsonValue::BOOL, "", arg.v.b ? 1.0 : 0.0);
  ------------------
  |  Branch (2846:69): [True: 0, False: 0]
  ------------------
 2847|      0|                                    break;
 2848|       |
 2849|      0|                                case Value::NUMBER:
  ------------------
  |  Branch (2849:33): [True: 0, False: 0]
  ------------------
 2850|      0|                                    args2.emplace_back(JsonnetJsonValue::NUMBER, "", arg.v.d);
 2851|      0|                                    break;
 2852|       |
 2853|      0|                                case Value::NULL_TYPE:
  ------------------
  |  Branch (2853:33): [True: 0, False: 0]
  ------------------
 2854|      0|                                    args2.emplace_back(JsonnetJsonValue::NULL_KIND, "", 0);
 2855|      0|                                    break;
 2856|       |
 2857|      0|                                default:
  ------------------
  |  Branch (2857:33): [True: 0, False: 0]
  ------------------
 2858|      0|                                    throw makeError(location,
 2859|      0|                                                    "native extensions can only take primitives.");
 2860|      0|                            }
 2861|      0|                        }
 2862|      0|                        std::vector<const JsonnetJsonValue *> args3;
 2863|      0|                        for (size_t i = 0; i < args2.size(); ++i) {
  ------------------
  |  Branch (2863:44): [True: 0, False: 0]
  ------------------
 2864|      0|                            args3.push_back(&args2[i]);
 2865|      0|                        }
 2866|      0|                        if (nit == nativeCallbacks.end()) {
  ------------------
  |  Branch (2866:29): [True: 0, False: 0]
  ------------------
 2867|      0|                            throw makeError(location,
 2868|      0|                                            "unrecognized builtin name: " + builtin_name);
 2869|      0|                        }
 2870|      0|                        const VmNativeCallback &cb = nit->second;
 2871|       |
 2872|      0|                        int succ;
 2873|      0|                        std::unique_ptr<JsonnetJsonValue> r(cb.cb(cb.ctx, args3.data(), &succ));
 2874|       |
 2875|      0|                        if (succ) {
  ------------------
  |  Branch (2875:29): [True: 0, False: 0]
  ------------------
 2876|      0|                            bool unused;
 2877|      0|                            jsonToHeap(r, unused, scratch);
 2878|      0|                        } else {
 2879|      0|                            if (r->kind != JsonnetJsonValue::STRING) {
  ------------------
  |  Branch (2879:33): [True: 0, False: 0]
  ------------------
 2880|      0|                                throw makeError(
 2881|      0|                                    location,
 2882|      0|                                    "native extension returned an error that was not a string.");
 2883|      0|                            }
 2884|      0|                            std::string rs = r->string;
 2885|      0|                            throw makeError(location, rs);
 2886|      0|                        }
 2887|       |
 2888|  15.6M|                    } else {
 2889|       |                        // Not all arguments forced yet.
 2890|  15.6M|                        HeapThunk *th = f.thunks[f.elementId++];
 2891|  15.6M|                        if (!th->filled) {
  ------------------
  |  Branch (2891:29): [True: 15.6M, False: 0]
  ------------------
 2892|  15.6M|                            stack.newCall(location, th, th->self, th->offset, th->upValues);
 2893|  15.6M|                            ast_ = th->body;
 2894|  15.6M|                            goto recurse;
 2895|  15.6M|                        } else {
 2896|       |                            // Otherwise loop back around to force the next thunk.
 2897|      0|                            goto replaceframe;
 2898|      0|                        }
 2899|  15.6M|                    }
 2900|  25.5M|                } break;
 2901|       |
 2902|  53.2M|                case FRAME_CALL: {
  ------------------
  |  Branch (2902:17): [True: 53.2M, False: 115M]
  ------------------
 2903|  53.2M|                    if (auto *thunk = dynamic_cast<HeapThunk *>(f.context)) {
  ------------------
  |  Branch (2903:31): [True: 29.9M, False: 23.3M]
  ------------------
 2904|       |                        // If we called a thunk, cache result.
 2905|  29.9M|                        thunk->fill(scratch);
 2906|  29.9M|                    } else if (auto *closure = dynamic_cast<HeapClosure *>(f.context)) {
  ------------------
  |  Branch (2906:38): [True: 9.44M, False: 13.8M]
  ------------------
 2907|  9.44M|                        if (f.elementId < f.thunks.size()) {
  ------------------
  |  Branch (2907:29): [True: 4.11M, False: 5.32M]
  ------------------
 2908|       |                            // If tailstrict, force thunks
 2909|  4.11M|                            HeapThunk *th = f.thunks[f.elementId++];
 2910|  4.11M|                            if (!th->filled) {
  ------------------
  |  Branch (2910:33): [True: 4.11M, False: 0]
  ------------------
 2911|  4.11M|                                stack.newCall(f.location, th, th->self, th->offset, th->upValues);
 2912|  4.11M|                                ast_ = th->body;
 2913|  4.11M|                                goto recurse;
 2914|  4.11M|                            }
 2915|  5.32M|                        } else if (f.thunks.size() == 0) {
  ------------------
  |  Branch (2915:36): [True: 3.48M, False: 1.84M]
  ------------------
 2916|       |                            // Body has now been executed
 2917|  3.48M|                        } else {
 2918|       |                            // Execute the body
 2919|  1.84M|                            f.thunks.clear();
 2920|  1.84M|                            f.elementId = 0;
 2921|  1.84M|                            ast_ = closure->body;
 2922|  1.84M|                            goto recurse;
 2923|  1.84M|                        }
 2924|  9.44M|                    }
 2925|       |                    // Result of call is in scratch, just pop.
 2926|  53.2M|                } break;
 2927|       |
 2928|  47.2M|                case FRAME_ERROR: {
  ------------------
  |  Branch (2928:17): [True: 3.20k, False: 168M]
  ------------------
 2929|  3.20k|                    const auto &ast = *static_cast<const Error *>(f.ast);
 2930|  3.20k|                    UString msg;
 2931|  3.20k|                    if (scratch.t == Value::STRING) {
  ------------------
  |  Branch (2931:25): [True: 327, False: 2.87k]
  ------------------
 2932|    327|                        msg = static_cast<HeapString *>(scratch.v.h)->value;
 2933|  2.87k|                    } else {
 2934|  2.87k|                        msg = toString(ast.location);
 2935|  2.87k|                    }
 2936|  3.20k|                    throw makeError(ast.location, encode_utf8(msg));
 2937|  53.2M|                } break;
 2938|       |
 2939|  8.56M|                case FRAME_IF: {
  ------------------
  |  Branch (2939:17): [True: 8.56M, False: 159M]
  ------------------
 2940|  8.56M|                    const auto &ast = *static_cast<const Conditional *>(f.ast);
 2941|  8.56M|                    if (scratch.t != Value::BOOLEAN) {
  ------------------
  |  Branch (2941:25): [True: 21, False: 8.56M]
  ------------------
 2942|     21|                        throw makeError(
 2943|     21|                            ast.location,
 2944|     21|                            "condition must be boolean, got " + type_str(scratch) + ".");
 2945|     21|                    }
 2946|  8.56M|                    ast_ = scratch.v.b ? ast.branchTrue : ast.branchFalse;
  ------------------
  |  Branch (2946:28): [True: 2.33M, False: 6.23M]
  ------------------
 2947|  8.56M|                    stack.pop();
 2948|  8.56M|                    goto recurse;
 2949|  8.56M|                } break;
 2950|       |
 2951|  1.21M|                case FRAME_SUPER_INDEX: {
  ------------------
  |  Branch (2951:17): [True: 1.21M, False: 167M]
  ------------------
 2952|  1.21M|                    const auto &ast = *static_cast<const SuperIndex *>(f.ast);
 2953|  1.21M|                    HeapObject *self;
 2954|  1.21M|                    unsigned offset;
 2955|  1.21M|                    stack.getSelfBinding(self, offset);
 2956|  1.21M|                    offset++;
 2957|  1.21M|                    if (offset >= countLeaves(self)) {
  ------------------
  |  Branch (2957:25): [True: 1, False: 1.21M]
  ------------------
 2958|      1|                        throw makeError(ast.location,
 2959|      1|                                        "attempt to use super when there is no super class.");
 2960|      1|                    }
 2961|  1.21M|                    if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (2961:25): [True: 0, False: 1.21M]
  ------------------
 2962|      0|                        throw makeError(
 2963|      0|                            ast.location,
 2964|      0|                            "super index must be string, got " + type_str(scratch) + ".");
 2965|      0|                    }
 2966|       |
 2967|  1.21M|                    const UString &index_name = static_cast<HeapString *>(scratch.v.h)->value;
 2968|  1.21M|                    auto *fid = alloc->makeIdentifier(index_name);
 2969|  1.21M|                    stack.pop();
 2970|  1.21M|                    ast_ = objectIndex(ast.location, self, fid, offset);
 2971|  1.21M|                    goto recurse;
 2972|  1.21M|                } break;
 2973|       |
 2974|  1.28M|                case FRAME_IN_SUPER_ELEMENT: {
  ------------------
  |  Branch (2974:17): [True: 1.28M, False: 167M]
  ------------------
 2975|  1.28M|                    const auto &ast = *static_cast<const InSuper *>(f.ast);
 2976|  1.28M|                    HeapObject *self;
 2977|  1.28M|                    unsigned offset;
 2978|  1.28M|                    stack.getSelfBinding(self, offset);
 2979|  1.28M|                    offset++;
 2980|  1.28M|                    if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (2980:25): [True: 1, False: 1.28M]
  ------------------
 2981|      1|                        throw makeError(ast.location,
 2982|      1|                                        "left hand side of e in super must be string, got " +
 2983|      1|                                            type_str(scratch) + ".");
 2984|      1|                    }
 2985|  1.28M|                    if (offset >= countLeaves(self)) {
  ------------------
  |  Branch (2985:25): [True: 47.3k, False: 1.24M]
  ------------------
 2986|       |                        // There is no super object.
 2987|  47.3k|                        scratch = makeBoolean(false);
 2988|  1.24M|                    } else {
 2989|  1.24M|                        const UString &element_name = static_cast<HeapString *>(scratch.v.h)->value;
 2990|  1.24M|                        auto *fid = alloc->makeIdentifier(element_name);
 2991|  1.24M|                        unsigned unused_found_at = 0;
 2992|  1.24M|                        bool in = findObject(fid, self, offset, unused_found_at);
 2993|  1.24M|                        scratch = makeBoolean(in);
 2994|  1.24M|                    }
 2995|  1.28M|                } break;
 2996|       |
 2997|  13.0M|                case FRAME_INDEX_INDEX: {
  ------------------
  |  Branch (2997:17): [True: 13.0M, False: 155M]
  ------------------
 2998|  13.0M|                    const auto &ast = *static_cast<const Index *>(f.ast);
 2999|  13.0M|                    const Value &target = f.val;
 3000|  13.0M|                    if (target.t == Value::ARRAY) {
  ------------------
  |  Branch (3000:25): [True: 96.3k, False: 13.0M]
  ------------------
 3001|  96.3k|                        const auto *array = static_cast<HeapArray *>(target.v.h);
 3002|  96.3k|                        if (scratch.t == Value::STRING) {
  ------------------
  |  Branch (3002:29): [True: 9, False: 96.3k]
  ------------------
 3003|      9|                            const UString &str = static_cast<HeapString *>(scratch.v.h)->value;
 3004|      9|                            throw makeError(
 3005|      9|                                ast.location,
 3006|      9|                                "attempted index an array with string \""
 3007|      9|                                + encode_utf8(jsonnet_string_escape(str, false)) + "\".");
 3008|      9|                        }
 3009|  96.3k|                        if (scratch.t != Value::NUMBER) {
  ------------------
  |  Branch (3009:29): [True: 4, False: 96.3k]
  ------------------
 3010|      4|                            throw makeError(
 3011|      4|                                ast.location,
 3012|      4|                                "array index must be number, got " + type_str(scratch) + ".");
 3013|      4|                        }
 3014|  96.3k|                        double index = ::floor(scratch.v.d);
 3015|  96.3k|                        long sz = array->elements.size();
 3016|  96.3k|                        if (index < 0 || index >= sz) {
  ------------------
  |  Branch (3016:29): [True: 1, False: 96.3k]
  |  Branch (3016:42): [True: 3, False: 96.3k]
  ------------------
 3017|      4|                            std::stringstream ss;
 3018|      4|                            ss << "array bounds error: " << index << " not within [0, " << sz
 3019|      4|                               << ")";
 3020|      4|                            throw makeError(ast.location, ss.str());
 3021|      4|                        }
 3022|  96.3k|                        if (scratch.v.d != index) {
  ------------------
  |  Branch (3022:29): [True: 1, False: 96.3k]
  ------------------
 3023|      1|                            std::stringstream ss;
 3024|      1|                            ss << "array index was not integer: " << scratch.v.d;
 3025|      1|                            throw makeError(ast.location, ss.str());
 3026|      1|                        }
 3027|       |                        // index < sz <= SIZE_T_MAX
 3028|  96.3k|                        auto *thunk = array->elements[size_t(index)];
 3029|  96.3k|                        if (thunk->filled) {
  ------------------
  |  Branch (3029:29): [True: 51.3k, False: 44.9k]
  ------------------
 3030|  51.3k|                            scratch = thunk->content;
 3031|  51.3k|                        } else {
 3032|  44.9k|                            stack.pop();
 3033|  44.9k|                            stack.newCall(
 3034|  44.9k|                                ast.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 3035|  44.9k|                            ast_ = thunk->body;
 3036|  44.9k|                            goto recurse;
 3037|  44.9k|                        }
 3038|  13.0M|                    } else if (target.t == Value::OBJECT) {
  ------------------
  |  Branch (3038:32): [True: 12.6M, False: 305k]
  ------------------
 3039|  12.6M|                        auto *obj = static_cast<HeapObject *>(target.v.h);
 3040|  12.6M|                        assert(obj != nullptr);
  ------------------
  |  Branch (3040:25): [True: 12.6M, False: 0]
  ------------------
 3041|  12.6M|                        if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (3041:29): [True: 5, False: 12.6M]
  ------------------
 3042|      5|                            throw makeError(
 3043|      5|                                ast.location,
 3044|      5|                                "object index must be string, got " + type_str(scratch) + ".");
 3045|      5|                        }
 3046|  12.6M|                        const UString &index_name = static_cast<HeapString *>(scratch.v.h)->value;
 3047|  12.6M|                        auto *fid = alloc->makeIdentifier(index_name);
 3048|  12.6M|                        stack.pop();
 3049|  12.6M|                        ast_ = objectIndex(ast.location, obj, fid, 0);
 3050|  12.6M|                        goto recurse;
 3051|  12.6M|                    } else if (target.t == Value::STRING) {
  ------------------
  |  Branch (3051:32): [True: 305k, False: 0]
  ------------------
 3052|   305k|                        auto *obj = static_cast<HeapString *>(target.v.h);
 3053|   305k|                        assert(obj != nullptr);
  ------------------
  |  Branch (3053:25): [True: 305k, False: 0]
  ------------------
 3054|   305k|                        if (scratch.t != Value::NUMBER) {
  ------------------
  |  Branch (3054:29): [True: 2, False: 305k]
  ------------------
 3055|      2|                            throw makeError(
 3056|      2|                                ast.location,
 3057|      2|                                "string index must be a number, got " + type_str(scratch) + ".");
 3058|      2|                        }
 3059|   305k|                        long sz = obj->value.length();
 3060|   305k|                        long i = (long)scratch.v.d;
 3061|   305k|                        if (i < 0 || i >= sz) {
  ------------------
  |  Branch (3061:29): [True: 47, False: 304k]
  |  Branch (3061:38): [True: 46, False: 304k]
  ------------------
 3062|     93|                            std::stringstream ss;
 3063|     93|                            ss << "string bounds error: " << i << " not within [0, " << sz << ")";
 3064|     93|                            throw makeError(ast.location, ss.str());
 3065|     93|                        }
 3066|   304k|                        char32_t ch[] = {obj->value[i], U'\0'};
 3067|   304k|                        scratch = makeString(ch);
 3068|   304k|                    } else {
 3069|      0|                        std::cerr << "INTERNAL ERROR: not object / array / string." << std::endl;
 3070|      0|                        abort();
 3071|      0|                    }
 3072|  13.0M|                } break;
 3073|       |
 3074|  13.0M|                case FRAME_INDEX_TARGET: {
  ------------------
  |  Branch (3074:17): [True: 13.0M, False: 155M]
  ------------------
 3075|  13.0M|                    const auto &ast = *static_cast<const Index *>(f.ast);
 3076|  13.0M|                    if (scratch.t != Value::ARRAY && scratch.t != Value::OBJECT &&
  ------------------
  |  Branch (3076:25): [True: 13.0M, False: 96.3k]
  |  Branch (3076:54): [True: 305k, False: 12.6M]
  ------------------
 3077|   305k|                        scratch.t != Value::STRING) {
  ------------------
  |  Branch (3077:25): [True: 6, False: 305k]
  ------------------
 3078|      6|                        throw makeError(ast.location,
 3079|      6|                                        "can only index objects, strings, and arrays, got " +
 3080|      6|                                            type_str(scratch) + ".");
 3081|      6|                    }
 3082|  13.0M|                    f.val = scratch;
 3083|  13.0M|                    f.kind = FRAME_INDEX_INDEX;
 3084|  13.0M|                    if (scratch.t == Value::OBJECT) {
  ------------------
  |  Branch (3084:25): [True: 12.6M, False: 401k]
  ------------------
 3085|  12.6M|                        auto *self = static_cast<HeapObject *>(scratch.v.h);
 3086|  12.6M|                        if (!stack.alreadyExecutingInvariants(self)) {
  ------------------
  |  Branch (3086:29): [True: 12.6M, False: 304]
  ------------------
 3087|  12.6M|                            stack.newFrame(FRAME_INVARIANTS, ast.location);
 3088|  12.6M|                            Frame &f2 = stack.top();
 3089|  12.6M|                            f2.self = self;
 3090|  12.6M|                            unsigned counter = 0;
 3091|  12.6M|                            objectInvariants(self, self, counter, f2.thunks);
 3092|  12.6M|                            if (f2.thunks.size() > 0) {
  ------------------
  |  Branch (3092:33): [True: 301, False: 12.6M]
  ------------------
 3093|    301|                                auto *thunk = f2.thunks[0];
 3094|    301|                                f2.elementId = 1;
 3095|    301|                                stack.newCall(ast.location,
 3096|    301|                                              thunk,
 3097|    301|                                              thunk->self,
 3098|    301|                                              thunk->offset,
 3099|    301|                                              thunk->upValues);
 3100|    301|                                ast_ = thunk->body;
 3101|    301|                                goto recurse;
 3102|    301|                            }
 3103|  12.6M|                        }
 3104|  12.6M|                    }
 3105|  13.0M|                    ast_ = ast.index;
 3106|  13.0M|                    goto recurse;
 3107|  13.0M|                } break;
 3108|       |
 3109|  12.7M|                case FRAME_INVARIANTS: {
  ------------------
  |  Branch (3109:17): [True: 12.7M, False: 155M]
  ------------------
 3110|  12.7M|                    if (f.elementId >= f.thunks.size()) {
  ------------------
  |  Branch (3110:25): [True: 12.7M, False: 85.8k]
  ------------------
 3111|  12.7M|                        if (stack.size() == initial_stack_size + 1) {
  ------------------
  |  Branch (3111:29): [True: 13.8k, False: 12.6M]
  ------------------
 3112|       |                            // Just pop, evaluate was invoked by runInvariants.
 3113|  13.8k|                            break;
 3114|  13.8k|                        }
 3115|  12.6M|                        stack.pop();
 3116|  12.6M|                        Frame &f2 = stack.top();
 3117|  12.6M|                        const auto &ast = *static_cast<const Index *>(f2.ast);
 3118|  12.6M|                        ast_ = ast.index;
 3119|  12.6M|                        goto recurse;
 3120|  12.7M|                    }
 3121|  85.8k|                    auto *thunk = f.thunks[f.elementId++];
 3122|  85.8k|                    stack.newCall(f.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 3123|  85.8k|                    ast_ = thunk->body;
 3124|  85.8k|                    goto recurse;
 3125|  12.7M|                } break;
 3126|       |
 3127|  6.36M|                case FRAME_LOCAL: {
  ------------------
  |  Branch (3127:17): [True: 6.36M, False: 161M]
  ------------------
 3128|       |                    // Result of execution is in scratch already.
 3129|  6.36M|                } break;
 3130|       |
 3131|  1.85M|                case FRAME_OBJECT: {
  ------------------
  |  Branch (3131:17): [True: 1.85M, False: 166M]
  ------------------
 3132|  1.85M|                    const auto &ast = *static_cast<const DesugaredObject *>(f.ast);
 3133|  1.85M|                    if (scratch.t != Value::NULL_TYPE) {
  ------------------
  |  Branch (3133:25): [True: 1.85M, False: 0]
  ------------------
 3134|  1.85M|                        if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (3134:29): [True: 2, False: 1.85M]
  ------------------
 3135|      2|                            throw makeError(ast.location, "field name was not a string.");
 3136|      2|                        }
 3137|  1.85M|                        const auto &fname = static_cast<const HeapString *>(scratch.v.h)->value;
 3138|  1.85M|                        const Identifier *fid = alloc->makeIdentifier(fname);
 3139|  1.85M|                        if (f.objectFields.find(fid) != f.objectFields.end()) {
  ------------------
  |  Branch (3139:29): [True: 4, False: 1.85M]
  ------------------
 3140|      4|                            std::string msg =
 3141|      4|                                "duplicate field name: \"" + encode_utf8(fname) + "\"";
 3142|      4|                            throw makeError(ast.location, msg);
 3143|      4|                        }
 3144|  1.85M|                        f.objectFields[fid].hide = f.fit->hide;
 3145|  1.85M|                        f.objectFields[fid].body = f.fit->body;
 3146|  1.85M|                    }
 3147|  1.85M|                    f.fit++;
 3148|  1.85M|                    if (f.fit != ast.fields.end()) {
  ------------------
  |  Branch (3148:25): [True: 1.36M, False: 491k]
  ------------------
 3149|  1.36M|                        ast_ = f.fit->name;
 3150|  1.36M|                        goto recurse;
 3151|  1.36M|                    } else {
 3152|   491k|                        auto env = capture(ast.freeVariables);
 3153|   491k|                        scratch = makeObject<HeapSimpleObject>(env, f.objectFields, ast.asserts);
 3154|   491k|                    }
 3155|  1.85M|                } break;
 3156|       |
 3157|   491k|                case FRAME_OBJECT_COMP_ARRAY: {
  ------------------
  |  Branch (3157:17): [True: 0, False: 168M]
  ------------------
 3158|      0|                    const auto &ast = *static_cast<const ObjectComprehensionSimple *>(f.ast);
 3159|      0|                    const Value &arr_v = scratch;
 3160|      0|                    if (scratch.t != Value::ARRAY) {
  ------------------
  |  Branch (3160:25): [True: 0, False: 0]
  ------------------
 3161|      0|                        throw makeError(ast.location,
 3162|      0|                                        "object comprehension needs array, got " + type_str(arr_v));
 3163|      0|                    }
 3164|      0|                    const auto *arr = static_cast<const HeapArray *>(arr_v.v.h);
 3165|      0|                    if (arr->elements.size() == 0) {
  ------------------
  |  Branch (3165:25): [True: 0, False: 0]
  ------------------
 3166|       |                        // Degenerate case.  Just create the object now.
 3167|      0|                        scratch = makeObject<HeapComprehensionObject>(
 3168|      0|                            BindingFrame{}, ast.value, ast.id, BindingFrame{});
 3169|      0|                    } else {
 3170|      0|                        f.kind = FRAME_OBJECT_COMP_ELEMENT;
 3171|      0|                        f.val = scratch;
 3172|      0|                        f.bindings[ast.id] = arr->elements[0];
 3173|      0|                        f.elementId = 0;
 3174|      0|                        ast_ = ast.field;
 3175|      0|                        goto recurse;
 3176|      0|                    }
 3177|      0|                } break;
 3178|       |
 3179|      0|                case FRAME_OBJECT_COMP_ELEMENT: {
  ------------------
  |  Branch (3179:17): [True: 0, False: 168M]
  ------------------
 3180|      0|                    const auto &ast = *static_cast<const ObjectComprehensionSimple *>(f.ast);
 3181|      0|                    const auto *arr = static_cast<const HeapArray *>(f.val.v.h);
 3182|      0|                    if (scratch.t != Value::NULL_TYPE) {
  ------------------
  |  Branch (3182:25): [True: 0, False: 0]
  ------------------
 3183|      0|                        if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (3183:29): [True: 0, False: 0]
  ------------------
 3184|      0|                            std::stringstream ss;
 3185|      0|                            ss << "field must be string, got: " << type_str(scratch);
 3186|      0|                            throw makeError(ast.location, ss.str());
 3187|      0|                        }
 3188|      0|                        const auto &fname = static_cast<const HeapString *>(scratch.v.h)->value;
 3189|      0|                        const Identifier *fid = alloc->makeIdentifier(fname);
 3190|      0|                        if (f.elements.find(fid) != f.elements.end()) {
  ------------------
  |  Branch (3190:29): [True: 0, False: 0]
  ------------------
 3191|      0|                            throw makeError(ast.location,
 3192|      0|                                            "duplicate field name: \"" + encode_utf8(fname) + "\"");
 3193|      0|                        }
 3194|      0|                        f.elements[fid] = arr->elements[f.elementId];
 3195|      0|                    }
 3196|      0|                    f.elementId++;
 3197|       |
 3198|      0|                    if (f.elementId == arr->elements.size()) {
  ------------------
  |  Branch (3198:25): [True: 0, False: 0]
  ------------------
 3199|      0|                        auto env = capture(ast.freeVariables);
 3200|      0|                        scratch =
 3201|      0|                            makeObject<HeapComprehensionObject>(env, ast.value, ast.id, f.elements);
 3202|      0|                    } else {
 3203|      0|                        f.bindings[ast.id] = arr->elements[f.elementId];
 3204|      0|                        ast_ = ast.field;
 3205|      0|                        goto recurse;
 3206|      0|                    }
 3207|      0|                } break;
 3208|       |
 3209|  1.89M|                case FRAME_STRING_CONCAT: {
  ------------------
  |  Branch (3209:17): [True: 1.89M, False: 166M]
  ------------------
 3210|  1.89M|                    const auto &ast = *static_cast<const Binary *>(f.ast);
 3211|  1.89M|                    const Value &lhs = stack.top().val;
 3212|  1.89M|                    UString output;
 3213|  1.89M|                    if (lhs.t == Value::STRING) {
  ------------------
  |  Branch (3213:25): [True: 1.86M, False: 35.0k]
  ------------------
 3214|  1.86M|                        output.append(static_cast<const HeapString *>(lhs.v.h)->value);
 3215|  1.86M|                    } else {
 3216|  35.0k|                        scratch = lhs;
 3217|  35.0k|                        output.append(toString(ast.left->location));
 3218|  35.0k|                    }
 3219|  1.89M|                    const Value &rhs = stack.top().val2;
 3220|  1.89M|                    if (rhs.t == Value::STRING) {
  ------------------
  |  Branch (3220:25): [True: 1.85M, False: 37.5k]
  ------------------
 3221|  1.85M|                        output.append(static_cast<const HeapString *>(rhs.v.h)->value);
 3222|  1.85M|                    } else {
 3223|  37.5k|                        scratch = rhs;
 3224|  37.5k|                        output.append(toString(ast.right->location));
 3225|  37.5k|                    }
 3226|  1.89M|                    scratch = makeString(output);
 3227|  1.89M|                } break;
 3228|       |
 3229|  2.18M|                case FRAME_UNARY: {
  ------------------
  |  Branch (3229:17): [True: 2.18M, False: 166M]
  ------------------
 3230|  2.18M|                    const auto &ast = *static_cast<const Unary *>(f.ast);
 3231|  2.18M|                    switch (scratch.t) {
 3232|  2.08M|                        case Value::BOOLEAN:
  ------------------
  |  Branch (3232:25): [True: 2.08M, False: 103k]
  ------------------
 3233|  2.08M|                            if (ast.op == UOP_NOT) {
  ------------------
  |  Branch (3233:33): [True: 2.08M, False: 1]
  ------------------
 3234|  2.08M|                                scratch = makeBoolean(!scratch.v.b);
 3235|  2.08M|                            } else {
 3236|      1|                                throw makeError(ast.location,
 3237|      1|                                                "unary operator " + uop_string(ast.op) +
 3238|      1|                                                    " does not operate on booleans.");
 3239|      1|                            }
 3240|  2.08M|                            break;
 3241|       |
 3242|  2.08M|                        case Value::NUMBER:
  ------------------
  |  Branch (3242:25): [True: 103k, False: 2.08M]
  ------------------
 3243|   103k|                            switch (ast.op) {
 3244|  3.95k|                                case UOP_PLUS: break;
  ------------------
  |  Branch (3244:33): [True: 3.95k, False: 99.4k]
  ------------------
 3245|       |
 3246|  90.0k|                                case UOP_MINUS: scratch = makeNumber(-scratch.v.d); break;
  ------------------
  |  Branch (3246:33): [True: 90.0k, False: 13.2k]
  ------------------
 3247|       |
 3248|  9.30k|                                case UOP_BITWISE_NOT:
  ------------------
  |  Branch (3248:33): [True: 9.30k, False: 94.0k]
  ------------------
 3249|  9.30k|                                    scratch = makeNumber(~(long)(scratch.v.d));
 3250|  9.30k|                                    break;
 3251|       |
 3252|     18|                                default:
  ------------------
  |  Branch (3252:33): [True: 18, False: 103k]
  ------------------
 3253|     18|                                    throw makeError(ast.location,
 3254|     18|                                                    "unary operator " + uop_string(ast.op) +
 3255|     18|                                                        " does not operate on numbers.");
 3256|   103k|                            }
 3257|   103k|                            break;
 3258|       |
 3259|   103k|                        default:
  ------------------
  |  Branch (3259:25): [True: 57, False: 2.18M]
  ------------------
 3260|     57|                            throw makeError(ast.location,
 3261|     57|                                            "unary operator " + uop_string(ast.op) +
 3262|     57|                                                " does not operate on type " + type_str(scratch));
 3263|  2.18M|                    }
 3264|  2.18M|                } break;
 3265|       |
 3266|  2.18M|                case FRAME_BUILTIN_JOIN_STRINGS: {
  ------------------
  |  Branch (3266:17): [True: 0, False: 168M]
  ------------------
 3267|      0|                    joinString(f.first, f.str, f.val, f.elementId, scratch);
 3268|      0|                    f.elementId++;
 3269|      0|                    auto *ast = joinStrings();
 3270|      0|                    if (ast != nullptr) {
  ------------------
  |  Branch (3270:25): [True: 0, False: 0]
  ------------------
 3271|      0|                        ast_ = ast;
 3272|      0|                        goto recurse;
 3273|      0|                    }
 3274|      0|                } break;
 3275|       |
 3276|      0|                case FRAME_BUILTIN_JOIN_ARRAYS: {
  ------------------
  |  Branch (3276:17): [True: 0, False: 168M]
  ------------------
 3277|      0|                    joinArray(f.first, f.thunks, f.val, f.elementId, scratch);
 3278|      0|                    f.elementId++;
 3279|      0|                    auto *ast = joinArrays();
 3280|      0|                    if (ast != nullptr) {
  ------------------
  |  Branch (3280:25): [True: 0, False: 0]
  ------------------
 3281|      0|                        ast_ = ast;
 3282|      0|                        goto recurse;
 3283|      0|                    }
 3284|      0|                } break;
 3285|       |
 3286|      0|                 case FRAME_BUILTIN_DECODE_UTF8: {
  ------------------
  |  Branch (3286:18): [True: 0, False: 168M]
  ------------------
 3287|      0|                    auto *ast = decodeUTF8();
 3288|      0|                    if (ast != nullptr) {
  ------------------
  |  Branch (3288:25): [True: 0, False: 0]
  ------------------
 3289|      0|                        ast_ = ast;
 3290|      0|                        goto recurse;
 3291|      0|                    }
 3292|      0|                } break;
 3293|       |
 3294|      0|                default:
  ------------------
  |  Branch (3294:17): [True: 0, False: 168M]
  ------------------
 3295|      0|                    std::cerr << "INTERNAL ERROR: Unknown FrameKind:  " << f.kind << std::endl;
 3296|      0|                    std::abort();
 3297|   168M|            }
 3298|       |
 3299|  74.6M|        popframe:;
 3300|       |
 3301|  74.6M|            stack.pop();
 3302|       |
 3303|  88.3M|        replaceframe:;
 3304|  88.3M|        }
 3305|  82.3M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack8newFrameIJNS1_9FrameKindEPKNS0_3ASTEEEEvDpT_:
  388|  64.1M|    {
  389|  64.1M|        stack.emplace_back(args...);
  390|  64.1M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15FrameC2ERKNS1_9FrameKindEPKNS0_3ASTE:
  194|  64.1M|        : kind(kind),
  195|  64.1M|          ast(ast),
  196|  64.1M|          location(ast->location),
  197|  64.1M|          tailCall(false),
  198|  64.1M|          elementId(0),
  199|  64.1M|          first(false),
  200|  64.1M|          context(NULL),
  201|  64.1M|          self(NULL),
  202|  64.1M|          offset(0)
  203|  64.1M|    {
  204|  64.1M|        val.t = Value::NULL_TYPE;
  205|  64.1M|        val2.t = Value::NULL_TYPE;
  206|  64.1M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack14getSelfBindingERPNS1_10HeapObjectERj:
  444|  37.5M|    {
  445|  37.5M|        self = nullptr;
  446|  37.5M|        offset = 0;
  447|   125M|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (447:40): [True: 125M, False: 14.4k]
  ------------------
  448|   125M|            if (stack[i].isCall()) {
  ------------------
  |  Branch (448:17): [True: 37.5M, False: 87.5M]
  ------------------
  449|  37.5M|                self = stack[i].self;
  450|  37.5M|                offset = stack[i].offset;
  451|  37.5M|                return;
  452|  37.5M|            }
  453|   125M|        }
  454|  37.5M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  577|  26.6M|    {
  578|  26.6M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  26.6M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 29.1k, False: 26.6M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  29.1k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  29.1k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  29.1k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  29.1k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 29.1k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  4.69M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 4.69M, False: 29.1k]
  ------------------
  597|  4.69M|                heap.markFrom(sourceVal.second);
  598|  4.69M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  29.1k|            heap.sweep();
  602|  29.1k|        }
  603|  26.6M|        return r;
  604|  26.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter7captureERKNSt3__16vectorIPKNS0_10IdentifierENS3_9allocatorIS7_EEEE:
  861|  52.6M|    {
  862|  52.6M|        BindingFrame env;
  863|  1.22G|        for (auto fv : free_vars) {
  ------------------
  |  Branch (863:22): [True: 1.22G, False: 52.6M]
  ------------------
  864|  1.22G|            auto *th = stack.lookUpVar(fv);
  865|  1.22G|            env[fv] = th;
  866|  1.22G|        }
  867|  52.6M|        return env;
  868|  52.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack8newFrameIJNS1_9FrameKindENS0_13LocationRangeEEEEvDpT_:
  388|  12.9M|    {
  389|  12.9M|        stack.emplace_back(args...);
  390|  12.9M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11makeClosureERKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS3_4lessIS7_EENS3_9allocatorINS3_4pairIKS7_S9_EEEEEEPNS1_10HeapObjectEjRKNS3_6vectorINS1_11HeapClosure5ParamENSC_ISO_EEEEPNS0_3ASTE:
  650|  2.32M|    {
  651|  2.32M|        Value r;
  652|  2.32M|        r.t = Value::FUNCTION;
  653|  2.32M|        r.v.h = makeHeap<HeapClosure>(env, self, offset, params, body, "");
  654|  2.32M|        return r;
  655|  2.32M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_11HeapClosureEJRKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERPNS1_10HeapObjectERjRKNS5_6vectorINS4_5ParamENSE_ISR_EEEERPNS0_3ASTERA1_KcEEEPT_DpOT0_:
  577|  2.32M|    {
  578|  2.32M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  2.32M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 2.66k, False: 2.32M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  2.66k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  2.66k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  2.66k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  2.66k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 2.66k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   429k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 429k, False: 2.66k]
  ------------------
  597|   429k|                heap.markFrom(sourceVal.second);
  598|   429k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  2.66k|            heap.sweep();
  602|  2.66k|        }
  603|  2.32M|        return r;
  604|  2.32M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter6importERKNS0_13LocationRangeEPKNS0_13LiteralStringE:
  792|      7|    {
  793|      7|        ImportCacheValue *input = importData(loc, file);
  794|      7|        if (input->thunk == nullptr) {
  ------------------
  |  Branch (794:13): [True: 0, False: 7]
  ------------------
  795|      0|            Tokens tokens = jsonnet_lex(input->foundHere, input->content.c_str());
  796|      0|            AST *expr = jsonnet_parse(alloc, tokens);
  797|      0|            jsonnet_desugar(alloc, expr, nullptr);
  798|      0|            jsonnet_static_analysis(expr);
  799|       |            // If no errors then populate cache.
  800|      0|            auto *thunk = makeHeap<HeapThunk>(idImport, nullptr, 0, expr);
  801|      0|            input->thunk = thunk;
  802|      0|        }
  803|      7|        return input->thunk;
  804|      7|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10importDataERKNS0_13LocationRangeEPKNS0_13LiteralStringE:
  816|     20|    {
  817|       |        // `dir` is passed to the importCallback, which may be externally defined.
  818|       |        // For backwards compatibility, we need to keep the trailing directory separator.
  819|       |        // For example, the default callback in libjsonnet.cpp joins paths with simple
  820|       |        // string concatenation. Other (external) implementations might do the same.
  821|     20|        std::string dir = path_dir_with_trailing_separator(loc.file);
  822|       |
  823|     20|        const UString &path = file->value;
  824|       |
  825|     20|        std::pair<std::string, UString> key(dir, path);
  826|     20|        ImportCacheValue *cached_value = cachedImports[key];
  827|     20|        if (cached_value != nullptr)
  ------------------
  |  Branch (827:13): [True: 0, False: 20]
  ------------------
  828|      0|            return cached_value;
  829|       |
  830|     20|        char *found_here_cptr;
  831|     20|        char *buf = NULL;
  832|     20|        size_t buflen = 0;
  833|     20|        int result = importCallback(importCallbackContext,
  834|     20|                                    dir.c_str(),
  835|     20|                                    encode_utf8(path).c_str(),
  836|     20|                                    &found_here_cptr,
  837|     20|                                    &buf,
  838|     20|                                    &buflen);
  839|       |
  840|     20|        std::string input(buf, buflen);
  841|     20|        ::free(buf);
  842|       |
  843|     20|        if (result == 1) {  // failure
  ------------------
  |  Branch (843:13): [True: 20, False: 0]
  ------------------
  844|     20|            std::string epath = encode_utf8(jsonnet_string_escape(path, false));
  845|     20|            std::string msg = "couldn't open import \"" + epath + "\": ";
  846|     20|            msg += input;
  847|     20|            throw makeError(loc, msg);
  848|     20|        }
  849|       |
  850|      0|        auto *input_ptr = new ImportCacheValue();
  851|      0|        input_ptr->foundHere = found_here_cptr;
  852|      0|        input_ptr->content = input;
  853|      0|        input_ptr->thunk = nullptr;  // May be filled in later by import().
  854|      0|        ::free(found_here_cptr);
  855|      0|        cachedImports[key] = input_ptr;
  856|      0|        return input_ptr;
  857|     20|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRKPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  577|  21.6M|    {
  578|  21.6M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  21.6M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 26.6k, False: 21.6M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  26.6k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  26.6k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  26.6k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  26.6k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 26.6k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  4.29M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 4.29M, False: 26.6k]
  ------------------
  597|  4.29M|                heap.markFrom(sourceVal.second);
  598|  4.29M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  26.6k|            heap.sweep();
  602|  26.6k|        }
  603|  21.6M|        return r;
  604|  21.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeObjectINS1_16HeapSimpleObjectEJNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEENS6_IS9_NS4_5FieldESD_NSE_INSF_ISG_SK_EEEEEENS5_4listIPNS0_3ASTENSE_ISQ_EEEEEEENS1_5ValueEDpT0_:
  671|  1.56M|    {
  672|  1.56M|        Value r;
  673|  1.56M|        r.t = Value::OBJECT;
  674|  1.56M|        r.v.h = makeHeap<T>(args...);
  675|  1.56M|        return r;
  676|  1.56M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_16HeapSimpleObjectEJRNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERNS6_IS9_NS4_5FieldESD_NSE_INSF_ISG_SL_EEEEEERNS5_4listIPNS0_3ASTENSE_ISS_EEEEEEEPT_DpOT0_:
  577|  1.56M|    {
  578|  1.56M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  1.56M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 2.03k, False: 1.56M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  2.03k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  2.03k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  2.03k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  2.03k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 2.03k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   328k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 328k, False: 2.03k]
  ------------------
  597|   328k|                heap.markFrom(sourceVal.second);
  598|   328k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  2.03k|            heap.sweep();
  602|  2.03k|        }
  603|  1.56M|        return r;
  604|  1.56M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack9lookUpVarEPKNS0_10IdentifierE:
  269|  1.26G|    {
  270|  1.82G|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (270:40): [True: 1.82G, False: 0]
  ------------------
  271|  1.82G|            const auto &binds = stack[i].bindings;
  272|  1.82G|            auto it = binds.find(id);
  273|  1.82G|            if (it != binds.end()) {
  ------------------
  |  Branch (273:17): [True: 1.26G, False: 566M]
  ------------------
  274|  1.26G|                return it->second;
  275|  1.26G|            }
  276|   566M|            if (stack[i].isCall())
  ------------------
  |  Branch (276:17): [True: 0, False: 566M]
  ------------------
  277|      0|                break;
  278|   566M|        }
  279|      0|        return nullptr;
  280|  1.26G|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack4sizeEv:
  263|   185M|    {
  264|   185M|        return stack.size();
  265|   185M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPKNS0_3ASTEEEEPT_DpOT0_:
  577|    243|    {
  578|    243|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|    243|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 0, False: 243]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|      0|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|      0|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|      0|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|      0|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 0]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|      0|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 0, False: 0]
  ------------------
  597|      0|                heap.markFrom(sourceVal.second);
  598|      0|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|      0|            heap.sweep();
  602|      0|        }
  603|    243|        return r;
  604|    243|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10findObjectEPKNS0_10IdentifierEPNS1_10HeapObjectEjRj:
  698|   470M|    {
  699|   470M|        if (auto *ext = dynamic_cast<HeapExtendedObject *>(curr)) {
  ------------------
  |  Branch (699:19): [True: 228M, False: 241M]
  ------------------
  700|   228M|            auto *r = findObject(f, ext->right, start_from, counter);
  701|   228M|            if (r)
  ------------------
  |  Branch (701:17): [True: 2.71M, False: 226M]
  ------------------
  702|  2.71M|                return r;
  703|   226M|            auto *l = findObject(f, ext->left, start_from, counter);
  704|   226M|            if (l)
  ------------------
  |  Branch (704:17): [True: 183M, False: 42.7M]
  ------------------
  705|   183M|                return l;
  706|   241M|        } else if (auto *ext = dynamic_cast<HeapRestrictedObject *>(curr)) {
  ------------------
  |  Branch (706:26): [True: 0, False: 241M]
  ------------------
  707|       |            // The RestrictedObject is on the 'right', that is it is checked/counted first.
  708|      0|            if (counter >= start_from) {
  ------------------
  |  Branch (708:17): [True: 0, False: 0]
  ------------------
  709|      0|                if (ext->retainedKeys.find(f) == ext->retainedKeys.end()) {
  ------------------
  |  Branch (709:21): [True: 0, False: 0]
  ------------------
  710|      0|                    counter += 1 + countLeaves(ext->obj);
  711|      0|                    return nullptr;
  712|      0|                }
  713|      0|            }
  714|      0|            ++counter;
  715|      0|            return findObject(f, ext->obj, start_from, counter);
  716|   241M|        } else {
  717|   241M|            if (counter >= start_from) {
  ------------------
  |  Branch (717:17): [True: 23.0M, False: 218M]
  ------------------
  718|  23.0M|                if (auto *simp = dynamic_cast<HeapSimpleObject *>(curr)) {
  ------------------
  |  Branch (718:27): [True: 23.0M, False: 0]
  ------------------
  719|  23.0M|                    auto it = simp->fields.find(f);
  720|  23.0M|                    if (it != simp->fields.end()) {
  ------------------
  |  Branch (720:25): [True: 15.4M, False: 7.59M]
  ------------------
  721|  15.4M|                        return simp;
  722|  15.4M|                    }
  723|  23.0M|                } else if (auto *comp = dynamic_cast<HeapComprehensionObject *>(curr)) {
  ------------------
  |  Branch (723:34): [True: 0, False: 0]
  ------------------
  724|      0|                    auto it = comp->compValues.find(f);
  725|      0|                    if (it != comp->compValues.end()) {
  ------------------
  |  Branch (725:25): [True: 0, False: 0]
  ------------------
  726|      0|                        return comp;
  727|      0|                    }
  728|      0|                }
  729|  23.0M|            }
  730|   226M|            counter++;
  731|   226M|        }
  732|   268M|        return nullptr;
  733|   470M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13callSourceValEPKNS0_3ASTEPNS1_9HeapThunkENSt3__16vectorIS7_NS8_9allocatorIS7_EEEE:
 2070|  22.1k|    const AST *callSourceVal(const AST *ast, HeapThunk *sourceVal, std::vector<HeapThunk*> args) {
 2071|  22.1k|        assert(sourceVal != nullptr);
  ------------------
  |  Branch (2071:9): [True: 22.1k, False: 0]
  ------------------
 2072|  22.1k|        assert(sourceVal->filled);
  ------------------
  |  Branch (2072:9): [True: 22.1k, False: 0]
  ------------------
 2073|  22.1k|        assert(sourceVal->content.t == Value::FUNCTION);
  ------------------
  |  Branch (2073:9): [True: 22.1k, False: 0]
  ------------------
 2074|  22.1k|        auto *func = static_cast<HeapClosure *>(sourceVal->content.v.h);
 2075|  22.1k|        BindingFrame up_values = func->upValues;
 2076|  66.3k|        for (size_t i = 0; i < args.size(); ++i) {
  ------------------
  |  Branch (2076:28): [True: 44.2k, False: 22.1k]
  ------------------
 2077|  44.2k|            up_values.insert({func->params[i].id, args[i]});
 2078|  44.2k|        }
 2079|  22.1k|        stack.newCall(ast->location, func, func->self, func->offset, up_values);
 2080|  22.1k|        return func->body;
 2081|  22.1k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter17safeDoubleToInt64EdRKNS0_13LocationRangeE:
  919|  32.4k|    int64_t safeDoubleToInt64(double value, const internal::LocationRange& loc) {
  920|  32.4k|        if (std::isnan(value) || std::isinf(value)) {
  ------------------
  |  Branch (920:13): [True: 0, False: 32.4k]
  |  Branch (920:34): [True: 0, False: 32.4k]
  ------------------
  921|      0|            throw internal::StaticError(loc, "numeric value is not finite");
  922|      0|        }
  923|       |
  924|       |        // Constants for safe double-to-int conversion
  925|       |        // Jsonnet uses IEEE 754 doubles, which precisely represent integers in the range [-2^53 + 1, 2^53 - 1].
  926|  32.4k|        constexpr int64_t DOUBLE_MAX_SAFE_INTEGER = (1LL << 53) - 1;
  927|  32.4k|        constexpr int64_t DOUBLE_MIN_SAFE_INTEGER = -((1LL << 53) - 1);
  928|       |
  929|       |        // Check if the value is within the safe integer range
  930|  32.4k|        if (value < DOUBLE_MIN_SAFE_INTEGER || value > DOUBLE_MAX_SAFE_INTEGER) {
  ------------------
  |  Branch (930:13): [True: 1, False: 32.4k]
  |  Branch (930:48): [True: 7, False: 32.3k]
  ------------------
  931|      8|            throw makeError(loc, "numeric value outside safe integer range for bitwise operation.");
  932|      8|        }
  933|  32.3k|        return static_cast<int64_t>(value);
  934|  32.4k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeObjectINS1_18HeapExtendedObjectEJPNS1_10HeapObjectES6_EEENS1_5ValueEDpT0_:
  671|  1.10M|    {
  672|  1.10M|        Value r;
  673|  1.10M|        r.t = Value::OBJECT;
  674|  1.10M|        r.v.h = makeHeap<T>(args...);
  675|  1.10M|        return r;
  676|  1.10M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_18HeapExtendedObjectEJRPNS1_10HeapObjectES7_EEEPT_DpOT0_:
  577|  1.10M|    {
  578|  1.10M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  1.10M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 993, False: 1.10M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|    993|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|    993|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|    993|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|    993|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 993]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   159k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 159k, False: 993]
  ------------------
  597|   159k|                heap.markFrom(sourceVal.second);
  598|   159k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|    993|            heap.sweep();
  602|    993|        }
  603|  1.10M|        return r;
  604|  1.10M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8toStringERKNS0_13LocationRangeE:
 1979|  73.8k|    {
 1980|  73.8k|        return manifestJson(loc, false, U"");
 1981|  73.8k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11countLeavesEPNS1_10HeapObjectE:
  876|  1.29G|    {
  877|  1.29G|        if (auto *ext = dynamic_cast<HeapExtendedObject *>(obj)) {
  ------------------
  |  Branch (877:19): [True: 646M, False: 648M]
  ------------------
  878|   646M|            return countLeaves(ext->left) + countLeaves(ext->right);
  879|   648M|        } else if (auto *ext = dynamic_cast<HeapRestrictedObject *>(obj)) {
  ------------------
  |  Branch (879:26): [True: 0, False: 648M]
  ------------------
  880|      0|            return countLeaves(ext->obj) + 1;
  881|   648M|        } else {
  882|       |            // Must be a HeapLeafObject.
  883|   648M|            return 1;
  884|   648M|        }
  885|  1.29G|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11objectIndexERKNS0_13LocationRangeEPNS1_10HeapObjectEPKNS0_10IdentifierEj:
 2018|  14.2M|    {
 2019|  14.2M|        unsigned found_at = 0;
 2020|  14.2M|        HeapObject *self = obj;
 2021|  14.2M|        HeapLeafObject *found = findObject(f, obj, offset, found_at);
 2022|  14.2M|        if (found == nullptr) {
  ------------------
  |  Branch (2022:13): [True: 23, False: 14.2M]
  ------------------
 2023|     23|            throw makeError(loc, "field does not exist: " + encode_utf8(f->name));
 2024|     23|        }
 2025|  14.2M|        if (auto *simp = dynamic_cast<HeapSimpleObject *>(found)) {
  ------------------
  |  Branch (2025:19): [True: 14.2M, False: 0]
  ------------------
 2026|  14.2M|            auto it = simp->fields.find(f);
 2027|  14.2M|            const AST *body = it->second.body;
 2028|       |
 2029|  14.2M|            stack.newCall(loc, simp, self, found_at, simp->upValues);
 2030|  14.2M|            return body;
 2031|  14.2M|        } else {
 2032|       |            // If a HeapLeafObject is not HeapSimpleObject, it must be HeapComprehensionObject.
 2033|      0|            auto *comp = static_cast<HeapComprehensionObject *>(found);
 2034|      0|            auto it = comp->compValues.find(f);
 2035|      0|            auto *th = it->second;
 2036|      0|            BindingFrame binds = comp->upValues;
 2037|      0|            binds[comp->id] = th;
 2038|      0|            stack.newCall(loc, comp, self, found_at, binds);
 2039|      0|            return comp->value;
 2040|      0|        }
 2041|  14.2M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack26alreadyExecutingInvariantsEPNS1_10HeapObjectE:
  458|  12.9M|    {
  459|  2.52G|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (459:40): [True: 2.51G, False: 12.9M]
  ------------------
  460|  2.51G|            if (stack[i].kind == FRAME_INVARIANTS) {
  ------------------
  |  Branch (460:17): [True: 11.6M, False: 2.49G]
  ------------------
  461|  11.6M|                if (stack[i].self == self)
  ------------------
  |  Branch (461:21): [True: 610, False: 11.6M]
  ------------------
  462|    610|                    return true;
  463|  11.6M|            }
  464|  2.51G|        }
  465|  12.9M|        return false;
  466|  12.9M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter16objectInvariantsEPNS1_10HeapObjectES4_RjRNSt3__16vectorIPNS1_9HeapThunkENS6_9allocatorIS9_EEEE:
 1992|  29.2M|    {
 1993|  29.2M|        if (auto *ext = dynamic_cast<HeapExtendedObject *>(curr)) {
  ------------------
  |  Branch (1993:19): [True: 8.17M, False: 21.0M]
  ------------------
 1994|  8.17M|            objectInvariants(ext->right, self, counter, thunks);
 1995|  8.17M|            objectInvariants(ext->left, self, counter, thunks);
 1996|  21.0M|        } else if (auto *ext = dynamic_cast<HeapRestrictedObject *>(curr)) {
  ------------------
  |  Branch (1996:26): [True: 0, False: 21.0M]
  ------------------
 1997|      0|            objectInvariants(ext->obj, self, counter, thunks);
 1998|  21.0M|        } else {
 1999|  21.0M|            if (auto *simp = dynamic_cast<HeapSimpleObject *>(curr)) {
  ------------------
  |  Branch (1999:23): [True: 21.0M, False: 0]
  ------------------
 2000|  21.0M|                for (AST *assert : simp->asserts) {
  ------------------
  |  Branch (2000:34): [True: 125k, False: 21.0M]
  ------------------
 2001|   125k|                    auto *el_th = makeHeap<HeapThunk>(idInvariant, self, counter, assert);
 2002|   125k|                    el_th->upValues = simp->upValues;
 2003|   125k|                    thunks.push_back(el_th);
 2004|   125k|                }
 2005|  21.0M|            }
 2006|  21.0M|            counter++;
 2007|  21.0M|        }
 2008|  29.2M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRPNS0_3ASTEEEEPT_DpOT0_:
  577|   125k|    {
  578|   125k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   125k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 293, False: 125k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|    293|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|    293|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|    293|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|    293|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 293]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  47.1k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 47.1k, False: 293]
  ------------------
  597|  47.1k|                heap.markFrom(sourceVal.second);
  598|  47.1k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|    293|            heap.sweep();
  602|    293|        }
  603|   125k|        return r;
  604|   125k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter12manifestJsonERKNS0_13LocationRangeEbRKNSt3__112basic_stringIDiNS6_11char_traitsIDiEENS6_9allocatorIDiEEEE:
 3315|  2.31M|    {
 3316|       |        // Printing fields means evaluating and binding them, which can trigger
 3317|       |        // garbage collection.
 3318|       |
 3319|  2.31M|        UStringStream ss;
 3320|  2.31M|        switch (scratch.t) {
  ------------------
  |  Branch (3320:17): [True: 2.31M, False: 0]
  ------------------
 3321|  33.9k|            case Value::ARRAY: {
  ------------------
  |  Branch (3321:13): [True: 33.9k, False: 2.27M]
  ------------------
 3322|  33.9k|                HeapArray *arr = static_cast<HeapArray *>(scratch.v.h);
 3323|  33.9k|                if (arr->elements.size() == 0) {
  ------------------
  |  Branch (3323:21): [True: 2.46k, False: 31.4k]
  ------------------
 3324|  2.46k|                    ss << U"[ ]";
 3325|  31.4k|                } else {
 3326|  31.4k|                    const char32_t *prefix = multiline ? U"[\n" : U"[";
  ------------------
  |  Branch (3326:46): [True: 18.2k, False: 13.2k]
  ------------------
 3327|  31.4k|                    UString indent2 = multiline ? indent + U"   " : indent;
  ------------------
  |  Branch (3327:39): [True: 18.2k, False: 13.2k]
  ------------------
 3328|  1.90M|                    for (auto *thunk : arr->elements) {
  ------------------
  |  Branch (3328:38): [True: 1.90M, False: 31.4k]
  ------------------
 3329|  1.90M|                        LocationRange tloc = thunk->body == nullptr ? loc : thunk->body->location;
  ------------------
  |  Branch (3329:46): [True: 0, False: 1.90M]
  ------------------
 3330|  1.90M|                        if (thunk->filled) {
  ------------------
  |  Branch (3330:29): [True: 0, False: 1.90M]
  ------------------
 3331|      0|                            stack.newCall(loc, thunk, nullptr, 0, BindingFrame{});
 3332|       |                            // Keep arr alive when scratch is overwritten
 3333|      0|                            stack.top().val = scratch;
 3334|      0|                            scratch = thunk->content;
 3335|  1.90M|                        } else {
 3336|  1.90M|                            stack.newCall(loc, thunk, thunk->self, thunk->offset, thunk->upValues);
 3337|       |                            // Keep arr alive when scratch is overwritten
 3338|  1.90M|                            stack.top().val = scratch;
 3339|  1.90M|                            evaluate(thunk->body, stack.size());
 3340|  1.90M|                        }
 3341|  1.90M|                        auto element = manifestJson(tloc, multiline, indent2);
 3342|       |                        // Restore scratch
 3343|  1.90M|                        scratch = stack.top().val;
 3344|  1.90M|                        stack.pop();
 3345|  1.90M|                        ss << prefix << indent2 << element;
 3346|  1.90M|                        prefix = multiline ? U",\n" : U", ";
  ------------------
  |  Branch (3346:34): [True: 1.88M, False: 28.8k]
  ------------------
 3347|  1.90M|                    }
 3348|  31.4k|                    ss << (multiline ? U"\n" : U"") << indent << U"]";
  ------------------
  |  Branch (3348:28): [True: 11.1k, False: 20.3k]
  ------------------
 3349|  31.4k|                }
 3350|  33.9k|            } break;
 3351|       |
 3352|   171k|            case Value::BOOLEAN: ss << (scratch.v.b ? U"true" : U"false"); break;
  ------------------
  |  Branch (3352:13): [True: 171k, False: 2.14M]
  |  Branch (3352:41): [True: 45.1k, False: 126k]
  ------------------
 3353|       |
 3354|  1.85M|            case Value::NUMBER: ss << decode_utf8(jsonnet_unparse_number(scratch.v.d)); break;
  ------------------
  |  Branch (3354:13): [True: 1.85M, False: 460k]
  ------------------
 3355|       |
 3356|     13|            case Value::FUNCTION:
  ------------------
  |  Branch (3356:13): [True: 13, False: 2.31M]
  ------------------
 3357|     13|                throw makeError(loc, "couldn't manifest function in JSON output.");
 3358|       |
 3359|    552|            case Value::NULL_TYPE: ss << U"null"; break;
  ------------------
  |  Branch (3359:13): [True: 552, False: 2.31M]
  ------------------
 3360|       |
 3361|   211k|            case Value::OBJECT: {
  ------------------
  |  Branch (3361:13): [True: 211k, False: 2.10M]
  ------------------
 3362|   211k|                auto *obj = static_cast<HeapObject *>(scratch.v.h);
 3363|   211k|                runInvariants(loc, obj);
 3364|       |                // Using std::map has the useful side-effect of ordering the fields
 3365|       |                // alphabetically.
 3366|   211k|                std::map<UString, const Identifier *> fields;
 3367|   394k|                for (const auto &f : objectFields(obj, true)) {
  ------------------
  |  Branch (3367:36): [True: 394k, False: 211k]
  ------------------
 3368|   394k|                    fields[f->name] = f;
 3369|   394k|                }
 3370|   211k|                if (fields.size() == 0) {
  ------------------
  |  Branch (3370:21): [True: 36.8k, False: 174k]
  ------------------
 3371|  36.8k|                    ss << U"{ }";
 3372|   174k|                } else {
 3373|   174k|                    UString indent2 = multiline ? indent + U"   " : indent;
  ------------------
  |  Branch (3373:39): [True: 135k, False: 39.1k]
  ------------------
 3374|   174k|                    const char32_t *prefix = multiline ? U"{\n" : U"{";
  ------------------
  |  Branch (3374:46): [True: 135k, False: 39.1k]
  ------------------
 3375|   337k|                    for (const auto &f : fields) {
  ------------------
  |  Branch (3375:40): [True: 337k, False: 174k]
  ------------------
 3376|       |                        // pushes FRAME_CALL
 3377|   337k|                        const AST *body = objectIndex(loc, obj, f.second, 0);
 3378|   337k|                        stack.top().val = scratch;
 3379|   337k|                        evaluate(body, stack.size());
 3380|   337k|                        auto vstr = manifestJson(body->location, multiline, indent2);
 3381|       |                        // Reset scratch so that the object we're manifesting doesn't
 3382|       |                        // get GC'd.
 3383|   337k|                        scratch = stack.top().val;
 3384|   337k|                        stack.pop();
 3385|   337k|                        ss << prefix << indent2 << jsonnet_string_unparse(f.first, false) << U": "
 3386|   337k|                           << vstr;
 3387|   337k|                        prefix = multiline ? U",\n" : U", ";
  ------------------
  |  Branch (3387:34): [True: 188k, False: 149k]
  ------------------
 3388|   337k|                    }
 3389|   174k|                    ss << (multiline ? U"\n" : U"") << indent << U"}";
  ------------------
  |  Branch (3389:28): [True: 34.5k, False: 140k]
  ------------------
 3390|   174k|                }
 3391|   211k|            } break;
 3392|       |
 3393|  42.8k|            case Value::STRING: {
  ------------------
  |  Branch (3393:13): [True: 42.8k, False: 2.27M]
  ------------------
 3394|  42.8k|                const UString &str = static_cast<HeapString *>(scratch.v.h)->value;
 3395|  42.8k|                ss << jsonnet_string_unparse(str, false);
 3396|  42.8k|            } break;
 3397|  2.31M|        }
 3398|  2.17M|        return ss.str();
 3399|  2.31M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13runInvariantsERKNS0_13LocationRangeEPNS1_10HeapObjectE:
 2044|   211k|    {
 2045|   211k|        if (stack.alreadyExecutingInvariants(self))
  ------------------
  |  Branch (2045:13): [True: 306, False: 211k]
  ------------------
 2046|    306|            return;
 2047|       |
 2048|   211k|        unsigned counter = 0;
 2049|   211k|        unsigned initial_stack_size = stack.size();
 2050|   211k|        stack.newFrame(FRAME_INVARIANTS, loc);
 2051|   211k|        std::vector<HeapThunk *> &thunks = stack.top().thunks;
 2052|   211k|        objectInvariants(self, self, counter, thunks);
 2053|   211k|        if (thunks.size() == 0) {
  ------------------
  |  Branch (2053:13): [True: 196k, False: 14.9k]
  ------------------
 2054|   196k|            stack.pop();
 2055|   196k|            return;
 2056|   196k|        }
 2057|  14.9k|        HeapThunk *thunk = thunks[0];
 2058|  14.9k|        stack.top().elementId = 1;
 2059|  14.9k|        stack.top().self = self;
 2060|  14.9k|        stack.newCall(loc, thunk, thunk->self, thunk->offset, thunk->upValues);
 2061|  14.9k|        evaluate(thunk->body, initial_stack_size);
 2062|  14.9k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111InterpreterD2Ev:
 1024|  2.88k|    {
 1025|  2.88k|        for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (1025:31): [True: 20, False: 2.88k]
  ------------------
 1026|     20|            delete pair.second;
 1027|     20|        }
 1028|  2.88k|    }

_ZN7jsonnet8internal10TraceFrameC2ERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEE:
   33|   242k|        : location(location), name(name)
   34|   242k|    {
   35|   242k|    }
_ZN7jsonnet8internal12RuntimeErrorC2ENSt3__16vectorINS0_10TraceFrameENS2_9allocatorIS4_EEEERKNS2_12basic_stringIcNS2_11char_traitsIcEENS5_IcEEEE:
   45|  1.71k|        : stackTrace(stack_trace), msg(msg)
   46|  1.71k|    {
   47|  1.71k|    }

_ZN2c43yml9CallbacksC2Ev:
31979|      2|    m_user_data(nullptr),
31980|       |    #ifndef RYML_NO_DEFAULT_CALLBACKS
31981|      2|    m_allocate(allocate_impl),
31982|      2|    m_free(free_impl),
31983|      2|    m_error(error_impl)
31984|       |    #else
31985|       |    m_allocate(nullptr),
31986|       |    m_free(nullptr),
31987|       |    m_error(nullptr)
31988|       |    #endif
31989|      2|{
31990|      2|}

